siglab-py 0.3.3__tar.gz → 0.3.6__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.

Potentially problematic release.


This version of siglab-py might be problematic. Click here for more details.

Files changed (43) hide show
  1. {siglab_py-0.3.3 → siglab_py-0.3.6}/PKG-INFO +1 -1
  2. {siglab_py-0.3.3 → siglab_py-0.3.6}/pyproject.toml +1 -1
  3. {siglab_py-0.3.3 → siglab_py-0.3.6}/setup.cfg +1 -1
  4. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/market_data_providers/ccxt_candles_ta_to_csv.py +4 -4
  5. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/util/analytic_util.py +62 -29
  6. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/util/market_data_util.py +49 -1
  7. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/util/test_market_data_analytic_util.py +25 -13
  8. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py.egg-info/PKG-INFO +1 -1
  9. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/__init__.py +0 -0
  10. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/constants.py +0 -0
  11. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/exchanges/__init__.py +0 -0
  12. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/exchanges/any_exchange.py +0 -0
  13. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/exchanges/futubull.py +0 -0
  14. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/market_data_providers/__init__.py +0 -0
  15. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/market_data_providers/aggregated_orderbook_provider.py +0 -0
  16. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/market_data_providers/candles_provider.py +0 -0
  17. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/market_data_providers/candles_ta_provider.py +0 -0
  18. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/market_data_providers/deribit_options_expiry_provider.py +0 -0
  19. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/market_data_providers/futu_candles_ta_to_csv.py +0 -0
  20. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/market_data_providers/orderbooks_provider.py +0 -0
  21. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/market_data_providers/test_provider.py +0 -0
  22. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/ordergateway/__init__.py +0 -0
  23. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/ordergateway/client.py +0 -0
  24. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/ordergateway/encrypt_keys_util.py +0 -0
  25. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/ordergateway/gateway.py +0 -0
  26. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/ordergateway/test_ordergateway.py +0 -0
  27. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/tests/__init__.py +0 -0
  28. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/tests/integration/__init__.py +0 -0
  29. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/tests/integration/market_data_util_tests.py +0 -0
  30. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/tests/unit/__init__.py +0 -0
  31. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/tests/unit/analytic_util_tests.py +0 -0
  32. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/tests/unit/market_data_util_tests.py +0 -0
  33. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/tests/unit/trading_util_tests.py +0 -0
  34. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/util/__init__.py +0 -0
  35. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/util/aws_util.py +0 -0
  36. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/util/notification_util.py +0 -0
  37. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/util/retry_util.py +0 -0
  38. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/util/slack_notification_util.py +0 -0
  39. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py/util/trading_util.py +0 -0
  40. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py.egg-info/SOURCES.txt +0 -0
  41. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py.egg-info/dependency_links.txt +0 -0
  42. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py.egg-info/requires.txt +0 -0
  43. {siglab_py-0.3.3 → siglab_py-0.3.6}/siglab_py.egg-info/top_level.txt +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: siglab_py
3
- Version: 0.3.3
3
+ Version: 0.3.6
4
4
  Summary: Market data fetches, TA calculations and generic order gateway.
5
5
  Author: r0bbarh00d
6
6
  Author-email: r0bbarh00d <r0bbarh00d@gmail.com>
@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
4
4
 
5
5
  [project]
6
6
  name = "siglab_py"
7
- version = "0.3.3"
7
+ version = "0.3.6"
8
8
  description = "Market data fetches, TA calculations and generic order gateway."
9
9
  authors = [{name = "r0bbarh00d", email = "r0bbarh00d@gmail.com"}]
10
10
  license = {text = "MIT"}
@@ -1,6 +1,6 @@
1
1
  [metadata]
2
2
  name = siglab_py
3
- version = 0.3.3
3
+ version = 0.3.6
4
4
  description = Market data fetches, TA calculations and generic order gateway.
5
5
  author = r0bbarh00d
6
6
  author_email = r0bbarh00d@gmail.com
@@ -26,7 +26,7 @@ from siglab_py.util.analytic_util import compute_candles_stats
26
26
  '''
27
27
  Usage:
28
28
  set PYTHONPATH=%PYTHONPATH%;D:\dev\siglab\siglab_py
29
- python ccxt_candles_ta_to_csv.py --exchange_name bybit --symbol BTC/USDT:USDT --end_date "2025-03-11 0:0:0" --start_date "2021-03-11 0:0:0" --default_type linear --compute_ta Y --pypy_compatible N
29
+ python ccxt_candles_ta_to_csv.py --exchange_name okx --symbol BTC/USDT:USDT --candle_size 1h --end_date "2025-04-22 0:0:0" --start_date "2024-01-01 0:0:0" --default_type linear --compute_ta Y --pypy_compatible N
30
30
 
31
31
  (Remember: python -mpip install siglab_py)
32
32
 
@@ -60,8 +60,8 @@ If debugging from VSCode, launch.json:
60
60
  "args" : [
61
61
  "--exchange_name", "bybit",
62
62
  "--symbol", "BTC/USDT:USDT",
63
- "--end_date", "2025-03-11 0:0:0",
64
- "--start_date", "2024-03-11 0:0:0",
63
+ "--end_date", "2025-04-22 0:0:0",
64
+ "--start_date", "2024-01-01 0:0:0",
65
65
  "--default_type", "linear",
66
66
  "--compute_ta", "Y",
67
67
  "--pypy_compatible", "N"
@@ -83,7 +83,7 @@ param : Dict = {
83
83
  'start_date' : start_date,
84
84
  'end_date' : end_date,
85
85
  'exchange_params' : {
86
- 'rateLimit' : 100, # in ms
86
+ 'rateLimit' : 300, # in ms
87
87
  'options' : {
88
88
  'defaultType' : "linear"
89
89
  }
@@ -32,6 +32,25 @@ def estimate_fib_retracement(
32
32
 
33
33
  return retracement_price
34
34
 
35
+ def calculate_slope(
36
+ pd_data : pd.DataFrame,
37
+ src_col_name : str,
38
+ slope_col_name : str,
39
+ sliding_window_how_many_candles : int
40
+ ):
41
+ import statsmodels.api as sm # in-compatible with pypy
42
+
43
+ X = sm.add_constant(range(len(pd_data[src_col_name])))
44
+ rolling_slope = pd_data[src_col_name].rolling(window=sliding_window_how_many_candles).apply(lambda x: sm.OLS(x, X[:len(x)]).fit().params[1], raw=False)
45
+ pd_data[slope_col_name] = rolling_slope
46
+ max_abs_slope = pd_data[slope_col_name].abs().rolling(window=sliding_window_how_many_candles).max()
47
+ pd_data[f"normalized_{slope_col_name}"] = pd_data[slope_col_name] / max_abs_slope
48
+ normalized_slope_rolling = pd_data[f"normalized_{slope_col_name}"].rolling(window=sliding_window_how_many_candles)
49
+ pd_data[f"normalized_{slope_col_name}_min"] = normalized_slope_rolling.min()
50
+ pd_data[f"normalized_{slope_col_name}_max"] = normalized_slope_rolling.max()
51
+ pd_data[f"normalized_{slope_col_name}_idmin"] = normalized_slope_rolling.apply(lambda x : x.idxmin())
52
+ pd_data[f"normalized_{slope_col_name}_idmax"] = normalized_slope_rolling.apply(lambda x : x.idxmax())
53
+
35
54
  '''
36
55
  compute_candles_stats will calculate typical/basic technical indicators using in many trading strategies:
37
56
  a. Basic SMA/EMAs (And slopes)
@@ -365,40 +384,54 @@ def compute_candles_stats(
365
384
  pd_candles['macd_minus_signal'] = pd_candles['macd'] - pd_candles['signal']
366
385
 
367
386
  if not pypy_compat:
368
- import statsmodels.api as sm # in-compatible with pypy
369
-
370
- # Slopes
371
- X = sm.add_constant(range(len(pd_candles['close'])))
372
- rolling_slope = pd_candles['close'].rolling(window=int(sliding_window_how_many_candles/slow_fast_interval_ratio)).apply(lambda x: sm.OLS(x, X[:len(x)]).fit().params[1], raw=False)
373
- pd_candles['close_short_slope'] = rolling_slope
374
- max_abs_slope = pd_candles['close_short_slope'].abs().rolling(window=int(sliding_window_how_many_candles/slow_fast_interval_ratio)).max()
375
- pd_candles['normalized_close_short_slope'] = pd_candles['close_short_slope'] / (2* max_abs_slope)
376
-
377
- X = sm.add_constant(range(len(pd_candles['close'])))
378
- rolling_slope = pd_candles['close'].rolling(window=sliding_window_how_many_candles).apply(lambda x: sm.OLS(x, X[:len(x)]).fit().params[1], raw=False)
379
- pd_candles['close_long_slope'] = rolling_slope
380
- max_abs_slope = pd_candles['close_long_slope'].abs().rolling(window=sliding_window_how_many_candles).max()
381
- pd_candles['normalized_close_short_slope'] = pd_candles['close_long_slope'] / (2 * max_abs_slope)
387
+ calculate_slope(
388
+ pd_data=pd_candles,
389
+ src_col_name='close',
390
+ slope_col_name='close_short_slope',
391
+ sliding_window_how_many_candles=int(sliding_window_how_many_candles/slow_fast_interval_ratio)
392
+ )
393
+
394
+ calculate_slope(
395
+ pd_data=pd_candles,
396
+ src_col_name='close',
397
+ slope_col_name='close_long_slope',
398
+ sliding_window_how_many_candles=int(sliding_window_how_many_candles)
399
+ )
382
400
 
383
- X = sm.add_constant(range(len(pd_candles['ema_short_periods'])))
384
- rolling_slope = pd_candles['ema_short_periods'].rolling(window=int(sliding_window_how_many_candles/slow_fast_interval_ratio)).apply(lambda x: sm.OLS(x, X[:len(x)]).fit().params[1], raw=False)
385
- pd_candles['ema_short_slope'] = rolling_slope
401
+ calculate_slope(
402
+ pd_data=pd_candles,
403
+ src_col_name='ema_short_periods',
404
+ slope_col_name='ema_short_slope',
405
+ sliding_window_how_many_candles=int(sliding_window_how_many_candles/slow_fast_interval_ratio)
406
+ )
386
407
 
387
- X = sm.add_constant(range(len(pd_candles['ema_long_periods'])))
388
- rolling_slope = pd_candles['ema_long_periods'].rolling(window=sliding_window_how_many_candles).apply(lambda x: sm.OLS(x, X[:len(x)]).fit().params[1], raw=False)
389
- pd_candles['ema_long_slope'] = rolling_slope
408
+ calculate_slope(
409
+ pd_data=pd_candles,
410
+ src_col_name='ema_long_periods',
411
+ slope_col_name='ema_long_slope',
412
+ sliding_window_how_many_candles=int(sliding_window_how_many_candles)
413
+ )
390
414
 
391
- X = sm.add_constant(range(len(pd_candles['boillenger_upper'])))
392
- rolling_slope = pd_candles['boillenger_upper'].rolling(window=int(sliding_window_how_many_candles/slow_fast_interval_ratio)).apply(lambda x: sm.OLS(x, X[:len(x)]).fit().params[1], raw=False)
393
- pd_candles['boillenger_upper_slope'] = rolling_slope
415
+ calculate_slope(
416
+ pd_data=pd_candles,
417
+ src_col_name='boillenger_upper',
418
+ slope_col_name='boillenger_upper_slope',
419
+ sliding_window_how_many_candles=int(sliding_window_how_many_candles)
420
+ )
394
421
 
395
- X = sm.add_constant(range(len(pd_candles['boillenger_lower'])))
396
- rolling_slope = pd_candles['boillenger_lower'].rolling(window=sliding_window_how_many_candles).apply(lambda x: sm.OLS(x, X[:len(x)]).fit().params[1], raw=False)
397
- pd_candles['boillenger_lower_slope'] = rolling_slope
422
+ calculate_slope(
423
+ pd_data=pd_candles,
424
+ src_col_name='boillenger_lower',
425
+ slope_col_name='boillenger_lower_slope',
426
+ sliding_window_how_many_candles=int(sliding_window_how_many_candles)
427
+ )
398
428
 
399
- X = sm.add_constant(range(len(pd_candles['ema_rsi'])))
400
- rolling_slope = pd_candles['ema_rsi'].rolling(window=sliding_window_how_many_candles).apply(lambda x: sm.OLS(x, X[:len(x)]).fit().params[1], raw=False)
401
- pd_candles['ema_rsi_slope'] = rolling_slope
429
+ calculate_slope(
430
+ pd_data=pd_candles,
431
+ src_col_name='ema_rsi',
432
+ slope_col_name='ema_rsi_slope',
433
+ sliding_window_how_many_candles=int(sliding_window_how_many_candles)
434
+ )
402
435
 
403
436
  pd_candles['regular_divergence'] = (
404
437
  (pd_candles['ema_long_slope'] > 0) & (pd_candles['ema_rsi_slope'] < 0) |
@@ -626,4 +626,52 @@ def fetch_deribit_btc_option_expiries(
626
626
  'index_price' : index_price,
627
627
  'by_expiry' : sorted_expiry_data, # type: ignore Otherwise, Error: Type "dict[str, list[tuple[str, float]] | dict[str, Dict[Unknown, Unknown]]]" is not assignable to return type "Dict[str, Dict[str, float] | Dict[str, Dict[str, str | float]]]"
628
628
  'by_expiry_and_strike' : expiry_data_breakdown_by_strike
629
- }
629
+ }
630
+
631
+ def build_pair_candles(
632
+ pd_candles1 : pd.DataFrame,
633
+ pd_candles2 : pd.DataFrame
634
+ ) -> pd.DataFrame:
635
+ min_timestamp_ms1 = int(pd_candles1.iloc[0]['timestamp_ms'])
636
+ max_timestamp_ms1 = int(pd_candles1.iloc[-1]['timestamp_ms'])
637
+ min_timestamp_ms2 = int(pd_candles2.iloc[0]['timestamp_ms'])
638
+ max_timestamp_ms2 = int(pd_candles2.iloc[-1]['timestamp_ms'])
639
+
640
+ pd_candles1 = pd_candles1[(pd_candles1.timestamp_ms>=min_timestamp_ms2) & (pd_candles1.timestamp_ms<=max_timestamp_ms2) & (~pd_candles1.timestamp_ms.isna()) ]
641
+ pd_candles2 = pd_candles2[(pd_candles2.timestamp_ms>=min_timestamp_ms1) & (pd_candles2.timestamp_ms<=max_timestamp_ms1) & (~pd_candles2.timestamp_ms.isna())]
642
+ assert(pd_candles1.shape[0]==pd_candles2.shape[0])
643
+
644
+ pd_candles1['timestamp_ms_gap'] = pd_candles1['timestamp_ms'] - pd_candles1['timestamp_ms'].shift(1)
645
+ timestamp_ms_gap = pd_candles1.iloc[-1]['timestamp_ms_gap']
646
+
647
+ assert(pd_candles1[~pd_candles1.timestamp_ms_gap.isna()][pd_candles1.timestamp_ms_gap!=timestamp_ms_gap].shape[0]==0)
648
+ pd_candles1.drop(columns=['timestamp_ms_gap'], inplace=True)
649
+
650
+ pd_candles2['timestamp_ms_gap'] = pd_candles2['timestamp_ms'] - pd_candles2['timestamp_ms'].shift(1)
651
+ timestamp_ms_gap = pd_candles2.iloc[-1]['timestamp_ms_gap']
652
+ assert(pd_candles2[~pd_candles2.timestamp_ms_gap.isna()][pd_candles2.timestamp_ms_gap!=timestamp_ms_gap].shape[0]==0)
653
+ pd_candles2.drop(columns=['timestamp_ms_gap'], inplace=True)
654
+
655
+ min_timestamp_ms1 = int(pd_candles1.iloc[0]['timestamp_ms'])
656
+ max_timestamp_ms1 = int(pd_candles1.iloc[-1]['timestamp_ms'])
657
+ min_timestamp_ms2 = int(pd_candles2.iloc[0]['timestamp_ms'])
658
+ max_timestamp_ms2 = int(pd_candles2.iloc[-1]['timestamp_ms'])
659
+ assert(min_timestamp_ms1==min_timestamp_ms2)
660
+ assert(max_timestamp_ms1==max_timestamp_ms2)
661
+ assert(pd_candles1.shape[0]==pd_candles2.shape[0])
662
+
663
+ if len([ col for col in pd_candles1.columns if col[-2:]=='_1' ]) == 0:
664
+ pd_candles1.columns = [str(col) + '_1' for col in pd_candles1.columns]
665
+
666
+ if len([ col for col in pd_candles2.columns if col[-2:]=='_2' ]) == 0:
667
+ pd_candles2.columns = [str(col) + '_2' for col in pd_candles2.columns]
668
+
669
+ pd_candles1.reset_index(drop=True, inplace=True)
670
+ pd_candles2.reset_index(drop=True, inplace=True)
671
+ pd_candles = pd.concat([pd_candles1, pd_candles2], axis=1)
672
+ pd_candles['timestamp_ms_gap'] = pd_candles['timestamp_ms_1'] - pd_candles['timestamp_ms_2']
673
+ assert(pd_candles[pd_candles.timestamp_ms_gap!=0].shape[0]==0)
674
+
675
+ return pd_candles
676
+
677
+
@@ -1,8 +1,7 @@
1
1
  from datetime import datetime
2
- from typing import Dict, Union
3
-
2
+ import time
3
+ from typing import Union
4
4
  import pandas as pd
5
- import matplotlib.pyplot as plt
6
5
 
7
6
  from ccxt.okx import okx
8
7
 
@@ -14,6 +13,10 @@ base_ccy : str = "BTC"
14
13
  # ticker = "OL/USDT:USDT"
15
14
  ticker = f"{base_ccy}/USDT:USDT"
16
15
 
16
+ reload_raw_candles : bool = True
17
+ raw_candles_file : str = f"{base_ccy}_raw_candles.csv"
18
+ candles_with_ta_file : str = f"{base_ccy}_candles_ta.csv"
19
+
17
20
  param = {
18
21
  'rateLimit' : 100, # In ms
19
22
  'options' : {
@@ -23,7 +26,7 @@ param = {
23
26
  exchange = okx(param) # type: ignore
24
27
 
25
28
  start_date : datetime = datetime(2024,1,1)
26
- end_date : datetime = datetime(2025,4,20)
29
+ end_date : datetime = datetime(2025,4,22)
27
30
  candle_size : str = '1h'
28
31
  ma_long_intervals : int = 24*30
29
32
  ma_short_intervals : int = 24
@@ -33,15 +36,20 @@ pypy_compatible : bool = False
33
36
  markets = exchange.load_markets()
34
37
  assert(ticker in markets)
35
38
 
36
- pd_candles: Union[pd.DataFrame, None] = fetch_candles(
37
- start_ts=int(start_date.timestamp()),
38
- end_ts=int(end_date.timestamp()),
39
- exchange=exchange,
40
- normalized_symbols=[ ticker ],
41
- candle_size=candle_size
42
- )[ ticker ]
39
+ start = time.time()
43
40
 
44
- pd_candles.to_csv(f"{base_ccy}_raw_candles.csv") # type: ignore
41
+ if reload_raw_candles:
42
+ pd_candles: Union[pd.DataFrame, None] = fetch_candles(
43
+ start_ts=int(start_date.timestamp()),
44
+ end_ts=int(end_date.timestamp()),
45
+ exchange=exchange,
46
+ normalized_symbols=[ ticker ],
47
+ candle_size=candle_size
48
+ )[ ticker ]
49
+
50
+ pd_candles.to_csv(raw_candles_file) # type: ignore
51
+ else:
52
+ pd_candles = pd.read_csv(raw_candles_file) # type: ignore
45
53
 
46
54
  compute_candles_stats(
47
55
  pd_candles=pd_candles, # type: ignore
@@ -49,4 +57,8 @@ compute_candles_stats(
49
57
  sliding_window_how_many_candles=ma_long_intervals,
50
58
  slow_fast_interval_ratio=(ma_long_intervals/ma_short_intervals),
51
59
  pypy_compat=pypy_compatible
52
- )
60
+ )
61
+ pd_candles.to_csv(candles_with_ta_file) # type: ignore
62
+
63
+ compute_candles_stats_elapsed_ms = int((time.time() - start) *1000)
64
+ print(f"elapsed (ms): {compute_candles_stats_elapsed_ms}")
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: siglab-py
3
- Version: 0.3.3
3
+ Version: 0.3.6
4
4
  Summary: Market data fetches, TA calculations and generic order gateway.
5
5
  Author: r0bbarh00d
6
6
  Author-email: r0bbarh00d <r0bbarh00d@gmail.com>