pyqqq 0.12.196__tar.gz → 0.12.198__tar.gz

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.
Files changed (60) hide show
  1. {pyqqq-0.12.196 → pyqqq-0.12.198}/PKG-INFO +1 -1
  2. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyproject.toml +1 -1
  3. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/backtest/broker.py +8 -3
  4. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/kis/domestic_stock.py +46 -27
  5. pyqqq-0.12.198/pyqqq/utils/backoff.py +32 -0
  6. pyqqq-0.12.198/pyqqq/utils/indicators.py +1011 -0
  7. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/market_schedule.py +1 -1
  8. {pyqqq-0.12.196 → pyqqq-0.12.198}/README.md +0 -0
  9. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/__init__.py +0 -0
  10. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/backtest/__init__.py +0 -0
  11. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/backtest/environment.py +0 -0
  12. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/backtest/logger.py +0 -0
  13. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/backtest/positionprovider.py +0 -0
  14. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/backtest/strategy.py +0 -0
  15. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/backtest/utils.py +0 -0
  16. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/backtest/wallclock.py +0 -0
  17. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/__init__.py +0 -0
  18. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/ebest/__init__.py +0 -0
  19. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/ebest/domestic_stock.py +0 -0
  20. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/ebest/oauth.py +0 -0
  21. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/ebest/simple.py +0 -0
  22. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/ebest/tr_client.py +0 -0
  23. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/helper.py +0 -0
  24. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/kis/__init__.py +0 -0
  25. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/kis/oauth.py +0 -0
  26. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/kis/overseas_stock.py +0 -0
  27. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/kis/simple.py +0 -0
  28. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/kis/simple_overseas.py +0 -0
  29. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/kis/tr_client.py +0 -0
  30. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/multiprocess_tracker.py +0 -0
  31. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/brokerage/tracker.py +0 -0
  32. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/config.py +0 -0
  33. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/data/__init__.py +0 -0
  34. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/data/daily.py +0 -0
  35. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/data/domestic.py +0 -0
  36. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/data/index.py +0 -0
  37. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/data/minutes.py +0 -0
  38. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/data/overseas.py +0 -0
  39. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/data/realtime.py +0 -0
  40. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/data/ticks.py +0 -0
  41. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/data/us_stocks.py +0 -0
  42. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/datatypes.py +0 -0
  43. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/executors/__init__.py +0 -0
  44. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/executors/hook.py +0 -0
  45. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/__init__.py +0 -0
  46. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/api_client.py +0 -0
  47. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/array.py +0 -0
  48. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/casting.py +0 -0
  49. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/compute.py +0 -0
  50. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/copycat.py +0 -0
  51. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/daily_tickers.py +0 -0
  52. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/display.py +0 -0
  53. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/kvstore.py +0 -0
  54. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/limiter.py +0 -0
  55. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/local_cache.py +0 -0
  56. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/logger.py +0 -0
  57. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/mock_api.py +0 -0
  58. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/position_classifier.py +0 -0
  59. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/retry.py +0 -0
  60. {pyqqq-0.12.196 → pyqqq-0.12.198}/pyqqq/utils/singleton.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: pyqqq
3
- Version: 0.12.196
3
+ Version: 0.12.198
4
4
  Summary: Package for quantitative strategy development on the PyQQQ platform
5
5
  License: MIT
6
6
  Author: PyQQQ team
@@ -1,6 +1,6 @@
1
1
  [tool.poetry]
2
2
  name = "pyqqq"
3
- version = "0.12.196"
3
+ version = "0.12.198"
4
4
  description = "Package for quantitative strategy development on the PyQQQ platform"
5
5
  authors = ["PyQQQ team <pyqqq.cs@gmail.com>"]
6
6
  readme = "README.md"
@@ -1,6 +1,6 @@
1
1
  import datetime as dtm
2
- import os
3
2
  import fcntl
3
+ import os
4
4
  from abc import ABC, abstractmethod
5
5
  from dataclasses import asdict
6
6
  from decimal import Decimal
@@ -9,6 +9,7 @@ from typing import List, Literal, Optional, Union
9
9
 
10
10
  import numpy as np
11
11
  import pandas as pd
12
+ from cachetools.func import ttl_cache
12
13
 
13
14
  from pyqqq.backtest.logger import Logger, get_logger
14
15
  from pyqqq.backtest.positionprovider import BasePositionProvider
@@ -20,11 +21,14 @@ from pyqqq.data.domestic import get_market_cap_by_codes
20
21
  from pyqqq.data.domestic import get_ticker_info as get_kr_ticker_info
21
22
  from pyqqq.data.minutes import get_all_day_data as get_kr_minute_data
22
23
  from pyqqq.data.us_stocks import get_all_day_data as get_us_minute_data
23
- from pyqqq.data.us_stocks import get_ohlcv_by_codes_for_period as get_us_daily_data
24
+ from pyqqq.data.us_stocks import \
25
+ get_ohlcv_by_codes_for_period as get_us_daily_data
24
26
  from pyqqq.data.us_stocks import get_ticker_info as get_us_ticker_info
25
27
  from pyqqq.datatypes import *
26
28
  from pyqqq.utils.casting import casting
27
- from pyqqq.utils.market_schedule import get_last_trading_day, get_market_schedule, get_trading_day_with_offset
29
+ from pyqqq.utils.market_schedule import (get_last_trading_day,
30
+ get_market_schedule,
31
+ get_trading_day_with_offset)
28
32
  from pyqqq.utils.position_classifier import PositionClassifier
29
33
 
30
34
  MarketType = Literal["kr_stock", "us_stock"]
@@ -781,6 +785,7 @@ class MockBroker(BaseBroker):
781
785
 
782
786
  return df
783
787
 
788
+ @ttl_cache(maxsize=100, ttl=300)
784
789
  def get_daily_price(self, code: str, from_date: Optional[dtm.date] = None, end_date: Optional[dtm.date] = None):
785
790
  """
786
791
  지정된 기간 동안의 정규장 시작부터 현재 시각 (이전) 까지의 일봉 데이터를 조회합니다.
@@ -805,33 +805,33 @@ class KISDomesticStock:
805
805
  - msg1 (str): 메시지1
806
806
  - output1 (object): 응답상세1
807
807
 
808
- - BSTP_NMIX_PRDY_VRSS: 업종 지수 전일 대비 - 업종 지수 전일 대비
809
- - PRDY_VRSS_SIGN: 전일 대비 부호 - 전일 대비 부호
810
- - BSTP_NMIX_PRDY_CTRT: 업종 지수 전일 대비율 - 업종 지수 전일 대비율
811
- - PRDY_NMIX: 전일 지수 - 전일 지수
812
- - ACML_VOL: 누적 거래량 - 누적 거래량
813
- - ACML_TR_PBMN: 누적 거래 대금 - 누적 거래 대금
814
- - HTS_KOR_ISNM: HTS 한글 종목명 - HTS 한글 종목명
815
- - BSTP_NMIX_PRPR: 업종 지수 현재가 - 업종 지수 현재가
816
- - BSTP_CLS_CODE: 업종 구분 코드 - 업종 구분 코드
817
- - PRDY_VOL: 전일 거래량 - 전일 거래량
818
- - BSTP_NMIX_OPRC: 업종 지수 시가 - 업종 지수 시가
819
- - BSTP_NMIX_HGPR: 업종 지수 최고가 - 업종 지수 최고가
820
- - BSTP_NMIX_LWPR: 업종 지수 최저가 - 업종 지수 최저가
821
- - FUTS_PRDY_OPRC: 업종 전일 시가 - 업종 전일 시가
822
- - FUTS_PRDY_HGPR: 업종 전일 최고가 - 업종 전일 최고가
823
- - FUTS_PRDY_LWPR: 업종 전일 최저가 - 업종 전일 최저가
808
+ - bstp_nmix_prdy_vrss: 업종 지수 전일 대비
809
+ - prdy_vrss_sign: 전일 대비 부호
810
+ - bstp_nmix_prdy_ctrt: 업종 지수 전일 대비율
811
+ - prdy_nmix: 전일 지수
812
+ - acml_vol: 누적 거래량
813
+ - acml_tr_pbmn: 누적 거래 대금
814
+ - hts_kor_isnm: hts 한글 종목명
815
+ - bstp_nmix_prpr: 업종 지수 현재가
816
+ - bstp_cls_code: 업종 구분 코드
817
+ - prdy_vol: 전일 거래량
818
+ - bstp_nmix_oprc: 업종 지수 시가
819
+ - bstp_nmix_hgpr: 업종 지수 최고가
820
+ - bstp_nmix_lwpr: 업종 지수 최저가
821
+ - futs_prdy_oprc: 업종 전일 시가
822
+ - futs_prdy_hgpr: 업종 전일 최고가
823
+ - futs_prdy_lwpr: 업종 전일 최저가
824
824
 
825
825
  - output2 (list): 응답상세2
826
826
 
827
- - STCK_BSOP_DATE: 영업 일자 - 영업 일자
828
- - BSTP_NMIX_PRPR: 업종 지수 현재가 - 업종 지수 현재가
829
- - BSTP_NMIX_OPRC: 업종 지수 시가 - 업종 지수 시가
830
- - BSTP_NMIX_HGPR: 업종 지수 최고가 - 업종 지수 최고가
831
- - BSTP_NMIX_LWPR: 업종 지수 최저가 - 업종 지수 최저가
832
- - ACML_VOL: 누적 거래량 - 누적 거래량
833
- - ACML_TR_PBMN: 누적 거래 대금 - 누적 거래 대금
834
- - MOD_YN: 변경 여부 - 변경 여부
827
+ - stck_bsop_date: 영업 일자
828
+ - bstp_nmix_prpr: 업종 지수 현재가
829
+ - bstp_nmix_oprc: 업종 지수 시가
830
+ - bstp_nmix_hgpr: 업종 지수 최고가
831
+ - bstp_nmix_lwpr: 업종 지수 최저가
832
+ - acml_vol: 누적 거래량
833
+ - acml_tr_pbmn: 누적 거래 대금
834
+ - mod_yn: 변경 여부
835
835
 
836
836
  Raise:
837
837
  ValueError: API 에러 발생시
@@ -1700,6 +1700,7 @@ class KISDomesticStock:
1700
1700
  fid_input_price_1: int = None,
1701
1701
  fid_input_price_2: int = None,
1702
1702
  fid_vol_cnt: int = None,
1703
+ fid_cond_mrkt_div_code: str = "J",
1703
1704
  ):
1704
1705
  """
1705
1706
  (국내주식시세) 거래량순위[v1_국내주식-047]
@@ -1745,6 +1746,8 @@ class KISDomesticStock:
1745
1746
  | ex) 100000
1746
1747
  | 전체 거래량 대상 조회 시 FID_VOL_CNT None
1747
1748
 
1749
+ fid_cond_mrkt_div_code (str): FID 조건시장분류코드 - J:KRX, NX:NXT
1750
+
1748
1751
  Returns:
1749
1752
  dict:
1750
1753
 
@@ -1786,11 +1789,12 @@ class KISDomesticStock:
1786
1789
  assert fid_input_price_1 is None or isinstance(fid_input_price_1, int), "fid_input_price_1 must be None or int"
1787
1790
  assert fid_input_price_2 is None or isinstance(fid_input_price_2, int), "fid_input_price_2 must be None or int"
1788
1791
  assert fid_vol_cnt is None or isinstance(fid_vol_cnt, int), "fid_vol_cnt must be None or int"
1792
+ assert fid_cond_mrkt_div_code in ["J", "NX"], "fid_cond_mrkt_div_code must be 'J' or 'NX'"
1789
1793
 
1790
1794
  url_path = "/uapi/domestic-stock/v1/quotations/volume-rank"
1791
1795
  tr_id = "FHPST01710000"
1792
1796
  params = {
1793
- "FID_COND_MRKT_DIV_CODE": "J",
1797
+ "FID_COND_MRKT_DIV_CODE": fid_cond_mrkt_div_code,
1794
1798
  "FID_COND_SCR_DIV_CODE": "20171",
1795
1799
  "FID_INPUT_ISCD": fid_input_iscd,
1796
1800
  "FID_DIV_CLS_CODE": fid_div_cls_code,
@@ -2194,7 +2198,14 @@ class KISDomesticStock:
2194
2198
 
2195
2199
  return result
2196
2200
 
2197
- def inquire_daily_trade_volume(self, fid_input_iscd: str, fid_input_date_1: dtm.date, fid_input_date_2: dtm.date, tr_cont: str = ""):
2201
+ def inquire_daily_trade_volume(
2202
+ self,
2203
+ fid_input_iscd: str,
2204
+ fid_input_date_1: dtm.date,
2205
+ fid_input_date_2: dtm.date,
2206
+ tr_cont: str = "",
2207
+ fid_cond_mrkt_div_code: str = "J",
2208
+ ):
2198
2209
  """
2199
2210
  (국내주식시세) 종목별일별매수매도체결량 [v1_국내주식-056]
2200
2211
 
@@ -2206,6 +2217,7 @@ class KISDomesticStock:
2206
2217
  fid_input_date_1 (dtm.date): 입력 일자1 - from
2207
2218
  fid_input_date_2 (dtm.date): 입력 일자2 - to
2208
2219
  tr_cont (str): 연속조회여부
2220
+ fid_cond_mrkt_div_code (str): FID 조건시장분류코드 - J:KRX, NX:NXT, UN:통합
2209
2221
 
2210
2222
  Returns:
2211
2223
  dict:
@@ -2230,10 +2242,17 @@ class KISDomesticStock:
2230
2242
  ValueError: API 에러 발생시
2231
2243
  """
2232
2244
  assert not self.auth.paper_trading, "실전계좌에서만 사용 가능한 API입니다."
2245
+ assert fid_cond_mrkt_div_code in ["J", "NX", "UN"], "fid_cond_mrkt_div_code must be 'J', 'NX' or 'UN'"
2233
2246
 
2234
2247
  url_path = "/uapi/domestic-stock/v1/quotations/inquire-daily-trade-volume"
2235
2248
  tr_id = "FHKST03010800"
2236
- params = {"FID_COND_MRKT_DIV_CODE": "J", "FID_INPUT_ISCD": fid_input_iscd, "FID_INPUT_DATE_1": fid_input_date_1.strftime("%Y%m%d"), "FID_INPUT_DATE_2": fid_input_date_2.strftime("%Y%m%d"), "FID_PERIOD_DIV_CODE": "D"}
2249
+ params = {
2250
+ "FID_COND_MRKT_DIV_CODE": fid_cond_mrkt_div_code,
2251
+ "FID_INPUT_ISCD": fid_input_iscd,
2252
+ "FID_INPUT_DATE_1": fid_input_date_1.strftime("%Y%m%d"),
2253
+ "FID_INPUT_DATE_2": fid_input_date_2.strftime("%Y%m%d"),
2254
+ "FID_PERIOD_DIV_CODE": "D",
2255
+ }
2237
2256
  res_body, res_headers = self._tr_request(url_path, tr_id, tr_cont, params=params)
2238
2257
 
2239
2258
  if res_body["rt_cd"] != "0":
@@ -0,0 +1,32 @@
1
+ import random
2
+
3
+
4
+ def exponential_backoff_with_jitter(
5
+ attempt: int,
6
+ base_wait_time: float = 1.0,
7
+ max_wait_time: float = 30.0,
8
+ jitter: bool = True,
9
+ ) -> float:
10
+ """
11
+ 지터와 최대 대기 시간을 포함한 지수 백오프 시간을 계산합니다.
12
+
13
+ Args:
14
+ attempt (int): 현재 재시도 횟수 (1부터 시작).
15
+ base_wait_time (float): 기본 대기 시간 (초).
16
+ max_wait_time (float): 최대 대기 시간 (초).
17
+ jitter (bool): 무작위 지터 추가 여부.
18
+
19
+ Returns:
20
+ float: 최종 백오프 대기 시간 (초).
21
+ """
22
+ # 기본 백오프 시간 계산
23
+ backoff_time = base_wait_time * (2 ** (attempt - 1))
24
+
25
+ # 지터 추가 (full jitter)
26
+ if jitter:
27
+ backoff_time = random.uniform(0, backoff_time)
28
+
29
+ # 최대 대기 시간 적용
30
+ final_wait_time = min(backoff_time, max_wait_time)
31
+
32
+ return final_wait_time