그 외

[주식] 키움 영웅문 통한 주식 자료다운로드

cj92 2019. 11. 11. 01:41

영웅문 가입은 https://wikidocs.net/2872를 참고하여 진행하였다.

 

목차

1. python 과 키움 API를 활용한 자료 다운로드

2. R을 활용한 자동화

 

아래는 python 을 통해 주식 분봉 자료를 다운로드 받는 코드이다.

 

import sys
from PyQt5.QtWidgets import *
from PyQt5.QAxContainer import *
from PyQt5.QtCore import *
import time
import os

import pandas as pd


class Kiwoom(QAxWidget):
    def __init__(self):
        super().__init__()
        self._create_kiwoom_instance()
        self._set_signal_slots()

    def _create_kiwoom_instance(self):
        self.setControl("KHOPENAPI.KHOpenAPICtrl.1")

    def _set_signal_slots(self):
        self.OnEventConnect.connect(self._event_connect)
        self.OnReceiveTrData.connect(self._receive_tr_data)

    def comm_connect(self):
        self.dynamicCall("CommConnect()")
        self.login_event_loop = QEventLoop()
        self.login_event_loop.exec_()

    def _event_connect(self, err_code):
        if err_code == 0:
            print("connected")
        else:
            print("disconnected")

        self.login_event_loop.exit()

    def get_code_list_by_market(self, market):
        code_list = self.dynamicCall("GetCodeListByMarket(QString)", market)
        code_list = code_list.split(';')
        return code_list[:-1]

    def get_master_code_name(self, code):
        code_name = self.dynamicCall("GetMasterCodeName(QString)", code)
        return code_name

    def set_input_value(self, id, value):
        self.dynamicCall("SetInputValue(QString, QString)", id, value)

    def comm_rq_data(self, rqname, trcode, next, screen_no):
        self.dynamicCall("CommRqData(QString, QString, int, QString", rqname, trcode, next, screen_no)
        self.tr_event_loop = QEventLoop()
        self.tr_event_loop.exec_()

    def _comm_get_data(self, code, real_type, field_name, index, item_name):
        ret = self.dynamicCall("CommGetData(QString, QString, QString, int, QString", code,
                               real_type, field_name, index, item_name)
        return ret.strip()

    def _get_repeat_cnt(self, trcode, rqname):
        ret = self.dynamicCall("GetRepeatCnt(QString, QString)", trcode, rqname)
        return ret

    def _receive_tr_data(self, screen_no, rqname, trcode, record_name, next, unused1, unused2, unused3, unused4):
        if next == '2':
            self.remained_data = True
        else:
            self.remained_data = False

        if rqname == "opt10081_req":
            self._opt10081(rqname, trcode)
        elif rqname =='opt10080_req':
            self._opt10080(rqname, trcode)
        try:
            self.tr_event_loop.exit()
        except AttributeError:
            pass

    def _opt10081(self, rqname, trcode):
        data_cnt = self._get_repeat_cnt(trcode, rqname)
        
        for i in range(data_cnt):
            code= self._comm_get_data(trcode, "", rqname, i, "종목코드")
            date = self._comm_get_data(trcode, "", rqname, i, "일자")
            open = self._comm_get_data(trcode, "", rqname, i, "시가")
            high = self._comm_get_data(trcode, "", rqname, i, "고가")
            low = self._comm_get_data(trcode, "", rqname, i, "저가")
            close = self._comm_get_data(trcode, "", rqname, i, "현재가")
            volume = self._comm_get_data(trcode, "", rqname, i, "거래량")
            
            self.ohlcv['code'].append(code)
            self.ohlcv['date'].append(date)
            self.ohlcv['open'].append(abs(int(open)))
            self.ohlcv['high'].append(abs(int(high)))
            self.ohlcv['low'].append(abs(int(low)))
            self.ohlcv['close'].append(abs(int(close)))
            self.ohlcv['volume'].append(abs(int(volume)))

    def _opt10080(self, rqname, trcode):
        data_cnt = self._get_repeat_cnt(trcode, rqname)

        for i in range(data_cnt):
            code= self._comm_get_data(trcode, "", rqname, i, "종목코드")
            date = self._comm_get_data(trcode, "", rqname, i, "체결시간")
            open = self._comm_get_data(trcode, "", rqname, i, "시가")
            high = self._comm_get_data(trcode, "", rqname, i, "고가")
            low = self._comm_get_data(trcode, "", rqname, i, "저가")
            close = self._comm_get_data(trcode, "", rqname, i, "현재가")
            volume = self._comm_get_data(trcode, "", rqname, i, "거래량")
            
            self.ohlcv['code'].append(code)
            self.ohlcv['date'].append(date)
            self.ohlcv['open'].append(abs(int(open)))
            self.ohlcv['high'].append(abs(int(high)))
            self.ohlcv['low'].append(abs(int(low)))
            self.ohlcv['close'].append(abs(int(close)))
            self.ohlcv['volume'].append(abs(int(volume)))

#################

for switch in [0,1]:
    if switch==0:
        app = QApplication(sys.argv)
        kiwoom = Kiwoom()
        kiwoom.comm_connect()
    code_list = kiwoom.get_code_list_by_market('10')
    dic=[]
    for x in code_list:
        dic.append(kiwoom.get_master_code_name(x))
    dic2=dict(zip(code_list,dic))
    dic2
    for x in code_list:
        try:
            if not(os.path.isdir('D:/주식/data/일봉/'+dic2[x])):
                os.makedirs(os.path.join('D:/주식/data/일봉/'+dic2[x]))
        except OSError as e:
            if e.errno != errno.EEXIST:
                print("Failed to create directory!!!!!")
                raise
    for x in code_list:
        try:
            if not(os.path.isdir('D:/주식/data/분봉/'+dic2[x])):
                os.makedirs(os.path.join('D:/주식/data/분봉/'+dic2[x]))
        except OSError as e:
            if e.errno != errno.EEXIST:
                print("Failed to create directory!!!!!")
                raise

###########
    if switch==1:
        kiwoom.ohlcv = {'code':[],'date': [], 'open': [], 'high': [], 'low': [], 'close': [], 'volume': []}
        a=0
        t=0
        aa=0
        TR_REQ_TIME_INTERVAL = 0.35
        for code in code_list:
            if aa<1000:
                t=t+1
                filename=time.strftime('%Y%m%d', time.localtime(time.time()))+'.csv'
                if (filename in os.listdir('D:/주식/data/분봉/'+dic2[code]))==False:
                    aa=aa+1
                    if aa==999:
                        break
                    time.sleep(1.5)
                    kiwoom.ohlcv = {'code':[],'date': [], 'open': [], 'high': [], 'low': [], 'close': [], 'volume': []}
                    kiwoom.set_input_value("종목코드", code)
                    kiwoom.set_input_value("기준일자", time.strftime('%Y%m%d', time.localtime(time.time())))
                    kiwoom.set_input_value("수정주가구분", 1)
                    kiwoom.comm_rq_data("opt10080_req", "opt10080", 0, "0101")
                    sys.stdout.write(str(dic2[code])+str(aa)+'\n')
                    sys.stdout.flush()
    
                    if aa<1000:
                        while kiwoom.remained_data == True:
                            aa=aa+1
                            if aa==999:
                                break
                            sys.stdout.write(str(dic2[code])+str(aa)+'\n')
                            sys.stdout.flush()
                            time.sleep(TR_REQ_TIME_INTERVAL)
                            kiwoom.set_input_value("종목코드", code)
                            kiwoom.set_input_value("기준일자", time.strftime('%Y%m%d', time.localtime(time.time())))
                            kiwoom.set_input_value("수정주가구분", 1)
                            kiwoom.comm_rq_data("opt10080_req", "opt10080", 2, "0101")
                        
                        df = pd.DataFrame(kiwoom.ohlcv, columns=['date','code','open', 'high', 'low',
                        'close', 'volume'])
                        if len(df)!=0:
                            
                            if a%5==0:
                                a=a+1
                                
                                time.sleep(3)
                        df.to_csv('D:/주식/data/분봉/'+dic2[code]+'/'+filename,mode='w',index=False)



위의 파이썬 분봉자료의 경우 횟수 재한이 있어 자동화 하려면 추가 코드가 필요하다.

따라서 R에서 반복문을 활용하여 다운로드 받게 설정하였다.

 

library(rstudioapi) 
myTerm <- rstudioapi::terminalCreate() 
rstudioapi::terminalSend(myTerm, 'conda activate py36_32\n') 
system('conda activate py36_32') 
getwd() 
while(T){ 
  rstudioapi::terminalSend(myTerm, 'python C:/Users/cj/Desktop/code/miniteCode.py\n') 
  rstudioapi::terminalSend(myTerm, 'python C:/Users/cj/Desktop/code/miniteCode1.py\n') 
  rstudioapi::terminalSend(myTerm, 'python C:/Users/cj/Desktop/code/miniteCode_today.py\n') 
}