-
Notifications
You must be signed in to change notification settings - Fork 0
/
trader.py
377 lines (319 loc) · 12.1 KB
/
trader.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
import json
import os
import requests
import dotenv
class Trader:
"""
한국투자증권의 트레이딩 API를 통해 매매하기 위해 필요한 method들이 포함된 클래스이다.
인증 / 매입 / 매각 을 목적으로 한다.
"""
def __init__(self, app_key: str, app_secret: str, url_base: str, mode: str):
"""
거래 API 초기화
Args:
app_key (str): APP KEY
app_secret (str): APP_SECRET
url_base (str): 한투 api base url 실제/모의 투자 포트는 별도로 설정해야함
mode (str): 실전투자 | 모의투자 여부 (PROD | DEV)
"""
print(f"this is {mode} mode trader ...\n")
self.port = os.getenv("PORT")
self.app_key = app_key
self.app_secret = app_secret
self.url_base = url_base
self.headers = {"content-type": "application/json"}
self.access_token = os.getenv("KIS_ACCESS_TOKEN", None)
def get_credential_access_token(self):
"""
get method로 호출하는 API 요청에 필요한 access token을 받아온다.
Returns:
status_code (int): 200
"""
api = "oauth2/tokenP"
body = {
"grant_type": "client_credentials",
"appkey": self.app_key,
"appsecret": self.app_secret,
}
try:
response = requests.post(
url=f"{self.url_base}:{self.port}/{api}",
headers=self.headers,
data=json.dumps(body),
)
if response.status_code != 200:
raise Exception(f"Failed to get access token it was already generated.")
else:
self.access_token = response.json()["access_token"]
except Exception:
return self.access_token
return self.access_token
def hashkey(self, datas: dict):
"""
post 요청을 안전하게 암호화하여 통신하기 위해 hashkey를 활용한다.
Args:
datas (dict): 전달하고자 하는 data
Returns:
hashkey (str)
"""
api = "uapi/hashkey"
headers = {
'content-Type': 'application/json',
'appKey': self.app_key,
'appSecret': self.app_secret,
}
port = self.port
response = requests.post(
url=f"{self.url_base}:{port}/{api}",
headers=headers,
data=json.dumps(datas)
)
hashkey = response.json()["HASH"]
return hashkey
# 이미 token을 발급받은 경우 토큰을 해당 토큰으로 설정한다.
def set_credential_access_token(self, access_token: str):
self.access_token = access_token
os.environ["KIS_ACCESS_TOKEN"] = access_token
def buy_stock(self, stock_code: str, ord_qty: int, ord_price: int):
"""
장전 시간외 매수를 통해 전날 종가 기준으로 예약 구매한다.
해당 stock의 accum asset이 감당할 수 있을만큼 구매한다.
status_code와 에러를 반환한다.
Args:
stock_code (str): 종목코드
ord_qty (int): 주문 개수
ord_price (int): 주문 가격
Returns:
status_code (str): 상태 코드, (200 | 4xx | 5xx)
output (dict): 주문 결과 KRX_FWDG_ORD_ORGNO(한국거래소 전송주문조직번호), ORNO(주문번호), ORD_TMD(주문시간)을 갖는다.
error (str): 에러
"""
tr_id = os.getenv("BUY_TR_ID")
api = "uapi/domestic-stock/v1/trading/order-resv"
data = {
# 계좌번호
"CANO": os.getenv("ACCOUNT_FRONT"),
"ACNT_PRDT_CD": os.getenv("ACCOUNT_REAR"),
# 종목번호
"PDNO": stock_code,
# 주문구분 - 지정가(00), 시장가(01), 장전 시간외(05)
"ORD_DVSN_CD": "05",
# 주문 수량
"ORD_QTY": str(ord_qty),
# 주문 단가
"ORD_UNPR": str(ord_price),
# 매도 매수 구분 코드 (01: 매도 / 02: 매수)
"SLL_BUY_DVSN_CD": "02",
# 주문대상 잔고 구분코드 10: 현금
"ORD_OBJT_CBLC_DVSN_CD": "10",
}
headers = {
"Content-Type": "application/json",
"authorization": f"Bearer {self.access_token}",
"appKey": self.app_key,
"appSecret": self.app_secret,
"tr_id": tr_id,
# 개인
"custtype": "P",
"hashkey": self.hashkey(data)
}
return self._request(api, data, headers)
def sell_stock(self, stock_code: str, ord_qty: int, ord_price: int):
"""
지정가 매도.
status_code와 에러를 반환한다.
Args:
stock_code (str): 종목코드
ord_qty (int): 주문 개수
ord_price (int): 주문 가격
Returns:
status_code (str): 상태 코드, (200 | 4xx | 5xx)
output (dict): 주문 결과 KRX_FWDG_ORD_ORGNO(한국거래소 전송주문조직번호), ORNO(주문번호), ORD_TMD(주문시간)을 갖는다.
error (str): 에러
"""
tr_id = os.getenv("SELL_TR_ID")
api = "uapi/domestic-stock/v1/trading/order-cash"
data = {
# 계좌번호
"CANO": os.getenv("ACCOUNT_FRONT"),
"ACNT_PRDT_CD": os.getenv("ACCOUNT_REAR"),
# 종목번호
"PDNO": stock_code,
# 주문구분 - 지정가(00), 시장가(01), 장전 시간외(05)
"ORD_DVSN": "00",
# 주문 수량
"ORD_QTY": str(ord_qty),
# 주문 단가
"ORD_UNPR": str(ord_price),
}
headers = {
"Content-Type": "application/json",
"authorization": f"Bearer {self.access_token}",
"appKey": self.app_key,
"appSecret": self.app_secret,
"tr_id": tr_id,
# 개인
"custtype": "P",
"hashkey": self.hashkey(data)
}
return self._request(api, data, headers)
def _request(self, api: str, data: dict, headers: dict, method: str = "POST"):
status_code, output, error = "500", {}, None
# 클라이언트 에러 핸들링 (파라미터 오기입, 장미운영날 주문 등)
try:
if method == "POST":
response = requests.post(
url=f"{self.url_base}:{self.port}/{api}",
headers=headers,
data=json.dumps(data)
)
# GET
else:
response = requests.get(
url=f"{self.url_base}:{self.port}/{api}",
headers=headers,
params=data
)
status_code = str(response.status_code)
response_json = response.json()
# 0 이외의 값은 실패이다.
if response_json["rt_cd"] != "0":
status_code = "4xx"
error = f"{response_json['msg_cd']}: {response_json['msg1']}"
else:
output = response_json.get("output", None)
if output is None:
output = response_json
# 서버 에러 핸들링 (토큰 만료 등)
except Exception as e:
status_code = "5xx"
error = str(e)
finally:
return {
"status_code": status_code,
"output": output,
"error": error,
}
def cancel_request(self, ord_orgno: str, orgn_odno: str):
"""
주문당시의 영업점코드와 주문번호를 활용해 요청한 매입/매도 주문을 취소한다.
status_code와 에러를 반환한다.
Args:
ord_orgno (str): 한국거래소전송주문조직번호
orgn_odno (str): 원주문번호
Returns:
status_code (str): 상태 코드, (200 | 4xx | 5xx)
error (str): 에러
"""
tr_id = os.getenv("CANCEL_TR_ID")
api = "uapi/domestic-stock/v1/trading/order-rvsecncl"
data = {
# 계좌번호
"CANO": os.getenv("ACCOUNT_FRONT"),
"ACNT_PRDT_CD": os.getenv("ACCOUNT_REAR"),
# 한국거래소전송주문조직번호
"KRX_FWDG_ORD_ORGNO": ord_orgno,
# 주문번호
"ORGN_ODNO": orgn_odno,
# 주문구분 - 지정가(00), 시장가(01)
"ORD_DVSN": "00",
# 정정취소 구분코드 (01: 정정, 02: 취소)
"RVSE_CNCL_DVSN_CD": "02",
# 주문 수량 (0은 전량)
"ORD_QTY": "0",
# 주문 단가
"ORD_UNPR": "0",
# 잔량전부주문여부
"QTY_ALL_ORD_YN": "Y",
}
headers = {
"Content-Type": "application/json",
"authorization": f"Bearer {self.access_token}",
"appKey": self.app_key,
"appSecret": self.app_secret,
"tr_id": tr_id,
# 개인
"custtype": "P",
"hashkey": self.hashkey(data)
}
return self._request(api, data, headers)
def inquire_balance(self):
"""
주식 잔고조회
예수금 잔액을 확인하거나 주식 수익을 확인할 수 있다.
Returns:
status_code (str): 상태 코드, (200 | 4xx | 5xx)
output (dict): 주문 결과 KRX_FWDG_ORD_ORGNO(한국거래소 전송주문조직번호), ORNO(주문번호), ORD_TMD(주문시간)을 갖는다.
error (str): 에러
"""
tr_id = os.getenv("INQ_BAL_TR_ID")
api = "uapi/domestic-stock/v1/trading/inquire-balance"
data = {
# 계좌번호
"CANO": os.getenv("ACCOUNT_FRONT"),
"ACNT_PRDT_CD": os.getenv("ACCOUNT_REAR"),
# 시간외단일가여부
"AFHR_FLPR_YN": "N",
# 오프라인여부
"OFL_YN": "",
# 조회 구분
"INQR_DVSN": "02",
# 단가 구분
"UNPR_DVSN": "01",
"FUND_STTL_ICLD_YN": "N",
"FNCG_AMT_AUTO_RDPT_YN": 'N',
# 전일 매매 포함 여부
"PRCS_DVSN": "00",
"CTX_AREA_FK100": "",
"CTX_AREA_NK100": "",
}
headers = {
"Content-Type": "application/json",
"authorization": f"Bearer {self.access_token}",
"appKey": self.app_key,
"appSecret": self.app_secret,
"tr_id": tr_id,
# 개인
"custtype": "P",
"hashkey": self.hashkey(data)
}
return self._request(api, data, headers, method="GET")
# inquire_balance는 API response지만 get_balance는 직접 남은 예수금을 가져온다.
def get_balance(self):
response_json = self.inquire_balance()
# 참조: https://apiportal.koreainvestment.com/apiservice/apiservice-domestic-stock-order#L_66c61080-674f-4c91-a0cc-db5e64e9a5e6
response_json["output"] = {
"dnca_tot_amt": response_json["output"]["output2"][0]["dnca_tot_amt"],
"prvs_rcdl_excc_amt": response_json["output"]["output2"][0]["prvs_rcdl_excc_amt"]
}
return response_json
if __name__ == "__main__":
import argparse
argparse = argparse.ArgumentParser()
argparse.add_argument("--mode", type=str, default="PROD")
args = argparse.parse_args()
mode = args.mode
if mode == "PROD":
file_name = "prod.env"
else:
file_name = "dev.env"
dotenv.load_dotenv(f"./config/{file_name}")
trader = Trader(
app_key=os.getenv("APP_KEY"),
app_secret=os.getenv("APP_SECRET"),
url_base=os.getenv("BASE_URL"),
mode=mode,
)
#
# res = trader.get_credential_access_token()
# print(f"res : {res}")
# trader.set_credential_access_token(token)
res = trader.inquire_balance()
#
# res = trader.buy_stock(
# stock_code="453810",
# ord_qty=1,
# ord_price=13600
# )
print(res)
print(res["output"]["output2"])