pyqqq 0.12.170__py3-none-any.whl → 0.12.172__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of pyqqq might be problematic. Click here for more details.

pyqqq/backtest/broker.py CHANGED
@@ -269,10 +269,12 @@ class TradingBroker(BaseBroker):
269
269
  def get_price(self, code: str, data_exchange: Optional[DataExchange] = None) -> Decimal:
270
270
  if data_exchange:
271
271
  price_data = self.data_api.get_price(code, data_exchange=data_exchange)
272
+ if data_exchange == DataExchange.NXT and price_data.get("open_price", 0) == 0: # current_price는 어제의 값을 반환하므로 open_price가 0인지로 NXT 시장 여부 확인
273
+ price_data = self.data_api.get_price(code, data_exchange=DataExchange.KRX)
272
274
  else:
273
- if self.market_nxt_on:
275
+ if self.market_nxt_on: # TODO: 정규장에서도 NXT 가격을 우선 조회하는 것이니 수정 필요
274
276
  price_data = self.data_api.get_price(code, data_exchange=DataExchange.NXT)
275
- if price_data.get("current_price") == 0:
277
+ if price_data.get("open_price", 0) == 0: # current_price는 어제의 값을 반환하므로 open_price가 0인지로 NXT 시장 여부 확인
276
278
  price_data = self.data_api.get_price(code, data_exchange=DataExchange.KRX)
277
279
  else:
278
280
  price_data = self.data_api.get_price(code, data_exchange=DataExchange.KRX)
@@ -285,10 +287,12 @@ class TradingBroker(BaseBroker):
285
287
  def get_minute_price(self, code: str, data_exchange: Optional[DataExchange] = None) -> pd.DataFrame:
286
288
  if data_exchange:
287
289
  result = self.data_api.get_today_minute_data(code, data_exchange=data_exchange)
290
+ if data_exchange == DataExchange.NXT and (result.empty or result.iloc[-1].close == 0):
291
+ result = self.data_api.get_today_minute_data(code, data_exchange=DataExchange.KRX)
288
292
  else:
289
293
  if self.market_nxt_on:
290
294
  result = self.data_api.get_today_minute_data(code, data_exchange=DataExchange.NXT)
291
- if result.iloc[-1].close == 0:
295
+ if result.empty or result.iloc[-1].close == 0:
292
296
  result = self.data_api.get_today_minute_data(code, data_exchange=DataExchange.KRX)
293
297
  else:
294
298
  result = self.data_api.get_today_minute_data(code, data_exchange=DataExchange.KRX)
@@ -297,6 +301,8 @@ class TradingBroker(BaseBroker):
297
301
  def get_daily_price(self, code: str, from_date: dtm.date, to_date: dtm.date, data_exchange: Optional[DataExchange] = None) -> pd.DataFrame:
298
302
  if data_exchange:
299
303
  result = self.data_api.get_historical_daily_data(code, from_date, to_date, adjusted_price=True, data_exchange=data_exchange)
304
+ if data_exchange == DataExchange.NXT and result.empty:
305
+ result = self.data_api.get_historical_daily_data(code, from_date, to_date, adjusted_price=True, data_exchange=DataExchange.KRX)
300
306
  else:
301
307
  if self.market_nxt_on:
302
308
  result = self.data_api.get_historical_daily_data(code, from_date, to_date, adjusted_price=True, data_exchange=DataExchange.NXT)
@@ -309,6 +315,8 @@ class TradingBroker(BaseBroker):
309
315
  def get_orderbook(self, code: str, data_exchange: Optional[DataExchange] = None):
310
316
  if data_exchange:
311
317
  result = self.data_api.get_orderbook(code, data_exchange=data_exchange)
318
+ if data_exchange == DataExchange.NXT and not result:
319
+ result = self.data_api.get_orderbook(code, data_exchange=DataExchange.KRX)
312
320
  else:
313
321
  if self.market_nxt_on:
314
322
  result = self.data_api.get_orderbook(code, data_exchange=DataExchange.NXT)
@@ -326,6 +334,9 @@ class TradingBroker(BaseBroker):
326
334
  return next((o for o in orders if o["order_no"] == order_no), None)
327
335
 
328
336
  def get_positions(self):
337
+ """
338
+ 포지션 조회 (단, NXT 프리마켓, 애프터마켓에서 현재 가격은 NXT 가격으로 조회되지 않음)
339
+ """
329
340
  return self.trading_api.get_positions()
330
341
 
331
342
  def create_order(self, asset_code: str, side: OrderSide, quantity: int, order_type: OrderType = OrderType.MARKET, price: int | Decimal = 0, exchange: OrderExchange = OrderExchange.KRX) -> str:
@@ -311,6 +311,10 @@ class TradingTracker:
311
311
  def _find_pending_order(self, order_no) -> StockOrder:
312
312
  return find(lambda x: x.order_no == order_no, self.pending_orders)
313
313
 
314
+ def _find_order_in_today_orders(self, order_no) -> StockOrder:
315
+ today_orders = self.simple_api.get_today_order_history(order_no=order_no, exchanges=[OrderExchange.KRX, OrderExchange.NXT, OrderExchange.SOR])
316
+ return find(lambda x: x.order_no == order_no, today_orders)
317
+
314
318
  def _find_position(self, asset_code) -> StockPosition:
315
319
  return find(lambda x: x.asset_code == asset_code, self.positions)
316
320
 
@@ -458,6 +462,11 @@ class TradingTracker:
458
462
  event.filled_time,
459
463
  partial,
460
464
  )
465
+
466
+ if order is None:
467
+ self.logger.warn(f"Order not found for order_no: {order_no}")
468
+ order = self._find_order_in_today_orders(order_no)
469
+ self.logger.warn(f"Order found in today's orders: {order}")
461
470
  self._notify_pending_order_update("partial" if partial else "completed", order)
462
471
  self._notify_position_update(position_event_type, position)
463
472
 
pyqqq/data/realtime.py CHANGED
@@ -1,4 +1,5 @@
1
- from typing import List
1
+ from typing import List, Union
2
+ from pyqqq.datatypes import DataExchange
2
3
  from pyqqq.utils.api_client import raise_for_status, send_request
3
4
  from pyqqq.utils.logger import get_logger
4
5
  from pyqqq.utils.singleton import singleton
@@ -14,12 +15,13 @@ import datetime as dtm
14
15
  logger = get_logger(__name__)
15
16
 
16
17
 
17
- def get_all_last_trades(codes: List[str] = None):
18
+ def get_all_last_trades(codes: List[str] = None, exchange: str = "KRX"):
18
19
  """
19
20
  모든 종목의 최근 체결 정보를 반환합니다.
20
21
 
21
22
  Args:
22
23
  codes (List[str], optional): 종목 코드 리스트. 모든 종목의 체결 정보를 반환하려면 None을 전달합니다.
24
+ exchange (str, optional): 거래소. 기본값은 "KRX"입니다. 지원하는 거래소는 "KRX", "UN" 입니다.
23
25
 
24
26
  Returns:
25
27
  list:
@@ -54,7 +56,7 @@ def get_all_last_trades(codes: List[str] = None):
54
56
 
55
57
  params = None
56
58
  if codes:
57
- params = {"codes": ",".join(codes) if isinstance(codes, list) else codes}
59
+ params = {"codes": ",".join(codes) if isinstance(codes, list) else codes, "exchange": exchange}
58
60
 
59
61
  r = send_request("GET", f"{c.PYQQQ_API_URL}/domestic-stock/trades", params=params)
60
62
  raise_for_status(r)
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: pyqqq
3
- Version: 0.12.170
3
+ Version: 0.12.172
4
4
  Summary: Package for quantitative strategy development on the PyQQQ platform
5
5
  License: MIT
6
6
  Author: PyQQQ team
@@ -5,7 +5,7 @@ pyqqq/ai/domestic.py,sha256=FiJNInRlhcnxG7Jxmz2hDvaLhS8_jn-JFpQMze8Ch9s,1888
5
5
  pyqqq/ai/market_schedule.py,sha256=8HiivwC-xI2EKr8lXS_g4mTj2LYpCQ2QfZsJmIq61O0,818
6
6
  pyqqq/ai/minute.py,sha256=C0sTVkBY4-Vuj8Q9VZ7d9kZYAv963FUX4k3vIvhetng,1754
7
7
  pyqqq/backtest/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
8
- pyqqq/backtest/broker.py,sha256=yliZlXGQe3YFFFdRM8teSqrNkxrQTCCTBDbZT-HPM3s,65351
8
+ pyqqq/backtest/broker.py,sha256=ynb0hxGk89BGg0xsNGcK6vw2tYc7hchiR5H8gA4GcVQ,66556
9
9
  pyqqq/backtest/environment.py,sha256=bkdlwG9MynuNL1u7f0cQ1L271cnPqJ7gX89EYwl8oOg,9313
10
10
  pyqqq/backtest/logger.py,sha256=BmoEMjUU76z8rZtMCYCwbspD3AVaHJrdbbT1EAFgrAE,3294
11
11
  pyqqq/backtest/positionprovider.py,sha256=wrR7Bntg28Q5_vGQV6XNzxe-SYoO9_GLcV9gDVEDAN4,4164
@@ -27,7 +27,7 @@ pyqqq/brokerage/kis/simple.py,sha256=t9vCnyeiVSHra2oYe8Gb3qFzxe2wWwrnJVCPNm6N-_0
27
27
  pyqqq/brokerage/kis/simple_overseas.py,sha256=1DuQBuJosg0mJQV7Ey2N3UOY8F3uOhzPDay4ncothuc,50360
28
28
  pyqqq/brokerage/kis/tr_client.py,sha256=9fTok0d8FmfXw4YxZSdn6T8UTHIG2aN1yMSkiMJUB3c,5530
29
29
  pyqqq/brokerage/multiprocess_tracker.py,sha256=Xx0hSpRZYITBGWjxclOEtNZdHV5agX94s34q1A8EE-Y,7283
30
- pyqqq/brokerage/tracker.py,sha256=z_bZ3O-G0oaEE7OBJawd2GbQwvPh-lKOLdWXOV1m1mw,23981
30
+ pyqqq/brokerage/tracker.py,sha256=p26APsjzjM_qO3zrsre3GBqUx4G4hBYcc5TLvzfoKg8,24504
31
31
  pyqqq/config.py,sha256=55Vqc_pGkdbrBdCV1aLgoH_n5IFxmMC59sbPHId3LoI,498
32
32
  pyqqq/data/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
33
33
  pyqqq/data/daily.py,sha256=k4dSiXNnT_J3mo0j_6VDjkGezBieVpIaZT3W_M7EU1Y,8019
@@ -35,7 +35,7 @@ pyqqq/data/domestic.py,sha256=2FOYxDGw2W7DGwY61p3uGFb4IWqWUKiNdR3RlewjkCU,30352
35
35
  pyqqq/data/index.py,sha256=d5b-8a7IXu7yNJWt1tIe1Mj83NW0ZnQq8nsj2Sl3Gx8,6988
36
36
  pyqqq/data/minutes.py,sha256=b-bqThP39t7V5io5dzI953bi1oWrUH3uie8GsFS7HUY,14727
37
37
  pyqqq/data/overseas.py,sha256=yx7tCZHW8AvjIbtrP4dqIeC6wseRSzbg5ag3dm6H0LY,1234
38
- pyqqq/data/realtime.py,sha256=y48JQSr54LkeCD9oIQcBqmK0wV1r2Hf42Va6PCkp1WU,14889
38
+ pyqqq/data/realtime.py,sha256=Kjb0fK2mtQgluS165quAdGcxtdxfiEmBLp2B0i-5f44,15105
39
39
  pyqqq/data/ticks.py,sha256=DXioiKBsGTzwXyvEH0lpm8t5g-1nHIOLKMXoSrE1Rko,4127
40
40
  pyqqq/data/us_stocks.py,sha256=jXR9dQEVigrwTLEpX1aX1_AQvOlBopW265gwx8Nq8OA,12959
41
41
  pyqqq/datatypes.py,sha256=KnanWzat6w5w0vNvKHsWt9VOBwf9gh0njdt36PJBgXc,8370
@@ -58,6 +58,6 @@ pyqqq/utils/mock_api.py,sha256=7EsaVQ9mOVZQAqtQW24isPnk9QTbJII7x3guhFyEMAE,10569
58
58
  pyqqq/utils/position_classifier.py,sha256=EaomByAWM2lVuYow5OFdJNrN64Fpukhj-lhFkjYpjeo,14908
59
59
  pyqqq/utils/retry.py,sha256=4mw9MQvgSBC8bTLvDauaCEI5N9tL8upHCk8rSfaVRG8,2066
60
60
  pyqqq/utils/singleton.py,sha256=m6NZ8fwVDpI6U-gUUihMPgVK_NkDh-Z1NSAtjisrpjY,810
61
- pyqqq-0.12.170.dist-info/METADATA,sha256=sSP-LU4CsxfOTURvSpFF8me_R0gxZab2kLcbPm1OpFs,1664
62
- pyqqq-0.12.170.dist-info/WHEEL,sha256=sP946D7jFCHeNz5Iq4fL4Lu-PrWrFsgfLXbbkciIZwg,88
63
- pyqqq-0.12.170.dist-info/RECORD,,
61
+ pyqqq-0.12.172.dist-info/METADATA,sha256=NdLxiJ-cz-aA6ycgkvgTRh0WCNVc7HgOFcAi5HG6jLA,1664
62
+ pyqqq-0.12.172.dist-info/WHEEL,sha256=sP946D7jFCHeNz5Iq4fL4Lu-PrWrFsgfLXbbkciIZwg,88
63
+ pyqqq-0.12.172.dist-info/RECORD,,