ccxt-ir 4.3.46.0.1__py2.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.
Files changed (772) hide show
  1. ccxt/__init__.py +358 -0
  2. ccxt/abantether.py +316 -0
  3. ccxt/abstract/__init__.py +0 -0
  4. ccxt/abstract/abantether.py +5 -0
  5. ccxt/abstract/ace.py +15 -0
  6. ccxt/abstract/afratether.py +6 -0
  7. ccxt/abstract/alpaca.py +70 -0
  8. ccxt/abstract/arzinja.py +5 -0
  9. ccxt/abstract/arzplus.py +7 -0
  10. ccxt/abstract/ascendex.py +77 -0
  11. ccxt/abstract/bequant.py +115 -0
  12. ccxt/abstract/bigone.py +45 -0
  13. ccxt/abstract/binance.py +712 -0
  14. ccxt/abstract/binancecoinm.py +712 -0
  15. ccxt/abstract/binanceus.py +764 -0
  16. ccxt/abstract/binanceusdm.py +712 -0
  17. ccxt/abstract/bingx.py +113 -0
  18. ccxt/abstract/bit2c.py +27 -0
  19. ccxt/abstract/bitbank.py +27 -0
  20. ccxt/abstract/bitbay.py +53 -0
  21. ccxt/abstract/bitbns.py +40 -0
  22. ccxt/abstract/bitcoincom.py +115 -0
  23. ccxt/abstract/bitfinex.py +69 -0
  24. ccxt/abstract/bitfinex2.py +139 -0
  25. ccxt/abstract/bitflyer.py +38 -0
  26. ccxt/abstract/bitget.py +508 -0
  27. ccxt/abstract/bithumb.py +32 -0
  28. ccxt/abstract/bitimen.py +7 -0
  29. ccxt/abstract/bitir.py +7 -0
  30. ccxt/abstract/bitmart.py +99 -0
  31. ccxt/abstract/bitmex.py +97 -0
  32. ccxt/abstract/bitopro.py +29 -0
  33. ccxt/abstract/bitpanda.py +35 -0
  34. ccxt/abstract/bitpin.py +7 -0
  35. ccxt/abstract/bitrue.py +72 -0
  36. ccxt/abstract/bitso.py +43 -0
  37. ccxt/abstract/bitstamp.py +258 -0
  38. ccxt/abstract/bitteam.py +29 -0
  39. ccxt/abstract/bitvavo.py +27 -0
  40. ccxt/abstract/bl3p.py +19 -0
  41. ccxt/abstract/blockchaincom.py +28 -0
  42. ccxt/abstract/blofin.py +37 -0
  43. ccxt/abstract/btcalpha.py +18 -0
  44. ccxt/abstract/btcbox.py +13 -0
  45. ccxt/abstract/btcmarkets.py +39 -0
  46. ccxt/abstract/btcturk.py +20 -0
  47. ccxt/abstract/bybit.py +298 -0
  48. ccxt/abstract/cex.py +33 -0
  49. ccxt/abstract/coinbase.py +94 -0
  50. ccxt/abstract/coinbaseadvanced.py +94 -0
  51. ccxt/abstract/coinbaseexchange.py +67 -0
  52. ccxt/abstract/coinbaseinternational.py +39 -0
  53. ccxt/abstract/coincatch.py +94 -0
  54. ccxt/abstract/coincheck.py +33 -0
  55. ccxt/abstract/coinex.py +237 -0
  56. ccxt/abstract/coinlist.py +54 -0
  57. ccxt/abstract/coinmate.py +62 -0
  58. ccxt/abstract/coinmetro.py +34 -0
  59. ccxt/abstract/coinone.py +67 -0
  60. ccxt/abstract/coinsph.py +54 -0
  61. ccxt/abstract/coinspot.py +28 -0
  62. ccxt/abstract/cryptocom.py +107 -0
  63. ccxt/abstract/currencycom.py +68 -0
  64. ccxt/abstract/delta.py +50 -0
  65. ccxt/abstract/deribit.py +125 -0
  66. ccxt/abstract/digifinex.py +91 -0
  67. ccxt/abstract/eterex.py +5 -0
  68. ccxt/abstract/excoino.py +7 -0
  69. ccxt/abstract/exir.py +8 -0
  70. ccxt/abstract/exmo.py +55 -0
  71. ccxt/abstract/exnovin.py +6 -0
  72. ccxt/abstract/farhadexchange.py +5 -0
  73. ccxt/abstract/fmfwio.py +115 -0
  74. ccxt/abstract/gate.py +265 -0
  75. ccxt/abstract/gateio.py +265 -0
  76. ccxt/abstract/gemini.py +58 -0
  77. ccxt/abstract/hashkey.py +67 -0
  78. ccxt/abstract/hitbtc.py +115 -0
  79. ccxt/abstract/hitbtc3.py +115 -0
  80. ccxt/abstract/hitobit.py +8 -0
  81. ccxt/abstract/hollaex.py +33 -0
  82. ccxt/abstract/htx.py +548 -0
  83. ccxt/abstract/huobi.py +548 -0
  84. ccxt/abstract/huobijp.py +114 -0
  85. ccxt/abstract/hyperliquid.py +6 -0
  86. ccxt/abstract/idex.py +26 -0
  87. ccxt/abstract/independentreserve.py +37 -0
  88. ccxt/abstract/indodax.py +26 -0
  89. ccxt/abstract/jibitex.py +7 -0
  90. ccxt/abstract/kraken.py +57 -0
  91. ccxt/abstract/krakenfutures.py +38 -0
  92. ccxt/abstract/kucoin.py +214 -0
  93. ccxt/abstract/kucoinfutures.py +233 -0
  94. ccxt/abstract/kuna.py +182 -0
  95. ccxt/abstract/latoken.py +56 -0
  96. ccxt/abstract/lbank.py +61 -0
  97. ccxt/abstract/luno.py +37 -0
  98. ccxt/abstract/lykke.py +29 -0
  99. ccxt/abstract/mercado.py +25 -0
  100. ccxt/abstract/mexc.py +178 -0
  101. ccxt/abstract/ndax.py +97 -0
  102. ccxt/abstract/nobitex.py +7 -0
  103. ccxt/abstract/novadax.py +29 -0
  104. ccxt/abstract/oceanex.py +22 -0
  105. ccxt/abstract/okcoin.py +74 -0
  106. ccxt/abstract/okexchange.py +8 -0
  107. ccxt/abstract/okx.py +324 -0
  108. ccxt/abstract/ompfinex.py +7 -0
  109. ccxt/abstract/onetrading.py +35 -0
  110. ccxt/abstract/oxfun.py +34 -0
  111. ccxt/abstract/p2b.py +22 -0
  112. ccxt/abstract/paradex.py +40 -0
  113. ccxt/abstract/paymium.py +28 -0
  114. ccxt/abstract/phemex.py +115 -0
  115. ccxt/abstract/poloniex.py +69 -0
  116. ccxt/abstract/poloniexfutures.py +48 -0
  117. ccxt/abstract/probit.py +23 -0
  118. ccxt/abstract/ramzinex.py +7 -0
  119. ccxt/abstract/sarmayex.py +5 -0
  120. ccxt/abstract/sarrafex.py +7 -0
  121. ccxt/abstract/tabdeal.py +7 -0
  122. ccxt/abstract/tetherland.py +5 -0
  123. ccxt/abstract/timex.py +62 -0
  124. ccxt/abstract/tokocrypto.py +37 -0
  125. ccxt/abstract/tradeogre.py +16 -0
  126. ccxt/abstract/twox.py +5 -0
  127. ccxt/abstract/ubitex.py +7 -0
  128. ccxt/abstract/upbit.py +38 -0
  129. ccxt/abstract/vertex.py +19 -0
  130. ccxt/abstract/wallex.py +8 -0
  131. ccxt/abstract/wavesexchange.py +154 -0
  132. ccxt/abstract/wazirx.py +30 -0
  133. ccxt/abstract/whitebit.py +98 -0
  134. ccxt/abstract/woo.py +83 -0
  135. ccxt/abstract/woofipro.py +119 -0
  136. ccxt/abstract/xt.py +152 -0
  137. ccxt/abstract/yobit.py +16 -0
  138. ccxt/abstract/zaif.py +38 -0
  139. ccxt/abstract/zonda.py +53 -0
  140. ccxt/ace.py +1012 -0
  141. ccxt/afratether.py +293 -0
  142. ccxt/alpaca.py +1083 -0
  143. ccxt/arzinja.py +285 -0
  144. ccxt/arzplus.py +412 -0
  145. ccxt/ascendex.py +3330 -0
  146. ccxt/async_support/__init__.py +337 -0
  147. ccxt/async_support/abantether.py +316 -0
  148. ccxt/async_support/ace.py +1012 -0
  149. ccxt/async_support/afratether.py +293 -0
  150. ccxt/async_support/alpaca.py +1083 -0
  151. ccxt/async_support/arzinja.py +285 -0
  152. ccxt/async_support/arzplus.py +412 -0
  153. ccxt/async_support/ascendex.py +3330 -0
  154. ccxt/async_support/base/__init__.py +1 -0
  155. ccxt/async_support/base/exchange.py +1966 -0
  156. ccxt/async_support/base/throttler.py +50 -0
  157. ccxt/async_support/base/ws/__init__.py +38 -0
  158. ccxt/async_support/base/ws/aiohttp_client.py +125 -0
  159. ccxt/async_support/base/ws/cache.py +212 -0
  160. ccxt/async_support/base/ws/client.py +193 -0
  161. ccxt/async_support/base/ws/fast_client.py +96 -0
  162. ccxt/async_support/base/ws/functions.py +59 -0
  163. ccxt/async_support/base/ws/future.py +58 -0
  164. ccxt/async_support/base/ws/order_book.py +78 -0
  165. ccxt/async_support/base/ws/order_book_side.py +174 -0
  166. ccxt/async_support/bequant.py +33 -0
  167. ccxt/async_support/bigone.py +2113 -0
  168. ccxt/async_support/binance.py +12234 -0
  169. ccxt/async_support/binancecoinm.py +45 -0
  170. ccxt/async_support/binanceus.py +211 -0
  171. ccxt/async_support/binanceusdm.py +58 -0
  172. ccxt/async_support/bingx.py +4325 -0
  173. ccxt/async_support/bit2c.py +866 -0
  174. ccxt/async_support/bitbank.py +1001 -0
  175. ccxt/async_support/bitbay.py +17 -0
  176. ccxt/async_support/bitbns.py +1154 -0
  177. ccxt/async_support/bitcoincom.py +17 -0
  178. ccxt/async_support/bitfinex.py +1617 -0
  179. ccxt/async_support/bitfinex2.py +3552 -0
  180. ccxt/async_support/bitflyer.py +995 -0
  181. ccxt/async_support/bitget.py +8273 -0
  182. ccxt/async_support/bithumb.py +1061 -0
  183. ccxt/async_support/bitimen.py +401 -0
  184. ccxt/async_support/bitir.py +490 -0
  185. ccxt/async_support/bitmart.py +4415 -0
  186. ccxt/async_support/bitmex.py +2756 -0
  187. ccxt/async_support/bitopro.py +1630 -0
  188. ccxt/async_support/bitpanda.py +16 -0
  189. ccxt/async_support/bitpin.py +454 -0
  190. ccxt/async_support/bitrue.py +3027 -0
  191. ccxt/async_support/bitso.py +1670 -0
  192. ccxt/async_support/bitstamp.py +2203 -0
  193. ccxt/async_support/bitteam.py +2239 -0
  194. ccxt/async_support/bitvavo.py +1968 -0
  195. ccxt/async_support/bl3p.py +485 -0
  196. ccxt/async_support/blockchaincom.py +1104 -0
  197. ccxt/async_support/blofin.py +2066 -0
  198. ccxt/async_support/btcalpha.py +891 -0
  199. ccxt/async_support/btcbox.py +544 -0
  200. ccxt/async_support/btcmarkets.py +1221 -0
  201. ccxt/async_support/btcturk.py +911 -0
  202. ccxt/async_support/bybit.py +8159 -0
  203. ccxt/async_support/cex.py +1605 -0
  204. ccxt/async_support/coinbase.py +4475 -0
  205. ccxt/async_support/coinbaseadvanced.py +17 -0
  206. ccxt/async_support/coinbaseexchange.py +1734 -0
  207. ccxt/async_support/coinbaseinternational.py +1899 -0
  208. ccxt/async_support/coincatch.py +5069 -0
  209. ccxt/async_support/coincheck.py +815 -0
  210. ccxt/async_support/coinex.py +5526 -0
  211. ccxt/async_support/coinlist.py +2243 -0
  212. ccxt/async_support/coinmate.py +1067 -0
  213. ccxt/async_support/coinmetro.py +1797 -0
  214. ccxt/async_support/coinone.py +1127 -0
  215. ccxt/async_support/coinsph.py +1850 -0
  216. ccxt/async_support/coinspot.py +534 -0
  217. ccxt/async_support/cryptocom.py +2822 -0
  218. ccxt/async_support/currencycom.py +1950 -0
  219. ccxt/async_support/delta.py +3376 -0
  220. ccxt/async_support/deribit.py +3437 -0
  221. ccxt/async_support/digifinex.py +3960 -0
  222. ccxt/async_support/eterex.py +286 -0
  223. ccxt/async_support/excoino.py +399 -0
  224. ccxt/async_support/exir.py +375 -0
  225. ccxt/async_support/exmo.py +2462 -0
  226. ccxt/async_support/exnovin.py +360 -0
  227. ccxt/async_support/farhadexchange.py +266 -0
  228. ccxt/async_support/fmfwio.py +34 -0
  229. ccxt/async_support/gate.py +6976 -0
  230. ccxt/async_support/gateio.py +16 -0
  231. ccxt/async_support/gemini.py +1825 -0
  232. ccxt/async_support/hashkey.py +4150 -0
  233. ccxt/async_support/hitbtc.py +3423 -0
  234. ccxt/async_support/hitbtc3.py +16 -0
  235. ccxt/async_support/hitobit.py +391 -0
  236. ccxt/async_support/hollaex.py +1813 -0
  237. ccxt/async_support/htx.py +8506 -0
  238. ccxt/async_support/huobi.py +16 -0
  239. ccxt/async_support/huobijp.py +1801 -0
  240. ccxt/async_support/hyperliquid.py +2431 -0
  241. ccxt/async_support/idex.py +1766 -0
  242. ccxt/async_support/independentreserve.py +784 -0
  243. ccxt/async_support/indodax.py +1247 -0
  244. ccxt/async_support/jibitex.py +395 -0
  245. ccxt/async_support/kraken.py +2894 -0
  246. ccxt/async_support/krakenfutures.py +2601 -0
  247. ccxt/async_support/kucoin.py +4602 -0
  248. ccxt/async_support/kucoinfutures.py +2698 -0
  249. ccxt/async_support/kuna.py +1841 -0
  250. ccxt/async_support/latoken.py +1664 -0
  251. ccxt/async_support/lbank.py +2683 -0
  252. ccxt/async_support/luno.py +1067 -0
  253. ccxt/async_support/lykke.py +1270 -0
  254. ccxt/async_support/mercado.py +842 -0
  255. ccxt/async_support/mexc.py +5369 -0
  256. ccxt/async_support/ndax.py +2354 -0
  257. ccxt/async_support/nobitex.py +419 -0
  258. ccxt/async_support/novadax.py +1484 -0
  259. ccxt/async_support/oceanex.py +903 -0
  260. ccxt/async_support/okcoin.py +2936 -0
  261. ccxt/async_support/okexchange.py +349 -0
  262. ccxt/async_support/okx.py +7827 -0
  263. ccxt/async_support/ompfinex.py +472 -0
  264. ccxt/async_support/onetrading.py +1911 -0
  265. ccxt/async_support/oxfun.py +2773 -0
  266. ccxt/async_support/p2b.py +1194 -0
  267. ccxt/async_support/paradex.py +2015 -0
  268. ccxt/async_support/paymium.py +564 -0
  269. ccxt/async_support/phemex.py +4473 -0
  270. ccxt/async_support/poloniex.py +2232 -0
  271. ccxt/async_support/poloniexfutures.py +1717 -0
  272. ccxt/async_support/probit.py +1734 -0
  273. ccxt/async_support/ramzinex.py +476 -0
  274. ccxt/async_support/sarmayex.py +357 -0
  275. ccxt/async_support/sarrafex.py +478 -0
  276. ccxt/async_support/tabdeal.py +364 -0
  277. ccxt/async_support/tetherland.py +349 -0
  278. ccxt/async_support/timex.py +1593 -0
  279. ccxt/async_support/tokocrypto.py +2405 -0
  280. ccxt/async_support/tradeogre.py +608 -0
  281. ccxt/async_support/twox.py +326 -0
  282. ccxt/async_support/ubitex.py +409 -0
  283. ccxt/async_support/upbit.py +1833 -0
  284. ccxt/async_support/vertex.py +2922 -0
  285. ccxt/async_support/wallex.py +445 -0
  286. ccxt/async_support/wavesexchange.py +2473 -0
  287. ccxt/async_support/wazirx.py +1224 -0
  288. ccxt/async_support/whitebit.py +2469 -0
  289. ccxt/async_support/woo.py +3114 -0
  290. ccxt/async_support/woofipro.py +2533 -0
  291. ccxt/async_support/xt.py +4454 -0
  292. ccxt/async_support/yobit.py +1283 -0
  293. ccxt/async_support/zaif.py +725 -0
  294. ccxt/async_support/zonda.py +1828 -0
  295. ccxt/base/__init__.py +27 -0
  296. ccxt/base/decimal_to_precision.py +174 -0
  297. ccxt/base/errors.py +242 -0
  298. ccxt/base/exchange.py +5941 -0
  299. ccxt/base/precise.py +287 -0
  300. ccxt/base/types.py +502 -0
  301. ccxt/bequant.py +33 -0
  302. ccxt/bigone.py +2112 -0
  303. ccxt/binance.py +12233 -0
  304. ccxt/binancecoinm.py +45 -0
  305. ccxt/binanceus.py +211 -0
  306. ccxt/binanceusdm.py +58 -0
  307. ccxt/bingx.py +4324 -0
  308. ccxt/bit2c.py +866 -0
  309. ccxt/bitbank.py +1001 -0
  310. ccxt/bitbay.py +17 -0
  311. ccxt/bitbns.py +1154 -0
  312. ccxt/bitcoincom.py +17 -0
  313. ccxt/bitfinex.py +1617 -0
  314. ccxt/bitfinex2.py +3552 -0
  315. ccxt/bitflyer.py +995 -0
  316. ccxt/bitget.py +8272 -0
  317. ccxt/bithumb.py +1061 -0
  318. ccxt/bitimen.py +401 -0
  319. ccxt/bitir.py +490 -0
  320. ccxt/bitmart.py +4415 -0
  321. ccxt/bitmex.py +2756 -0
  322. ccxt/bitopro.py +1630 -0
  323. ccxt/bitpanda.py +16 -0
  324. ccxt/bitpin.py +454 -0
  325. ccxt/bitrue.py +3026 -0
  326. ccxt/bitso.py +1670 -0
  327. ccxt/bitstamp.py +2203 -0
  328. ccxt/bitteam.py +2239 -0
  329. ccxt/bitvavo.py +1968 -0
  330. ccxt/bl3p.py +485 -0
  331. ccxt/blockchaincom.py +1104 -0
  332. ccxt/blofin.py +2066 -0
  333. ccxt/btcalpha.py +891 -0
  334. ccxt/btcbox.py +544 -0
  335. ccxt/btcmarkets.py +1221 -0
  336. ccxt/btcturk.py +911 -0
  337. ccxt/bybit.py +8158 -0
  338. ccxt/cex.py +1605 -0
  339. ccxt/coinbase.py +4474 -0
  340. ccxt/coinbaseadvanced.py +17 -0
  341. ccxt/coinbaseexchange.py +1734 -0
  342. ccxt/coinbaseinternational.py +1899 -0
  343. ccxt/coincatch.py +5069 -0
  344. ccxt/coincheck.py +815 -0
  345. ccxt/coinex.py +5525 -0
  346. ccxt/coinlist.py +2243 -0
  347. ccxt/coinmate.py +1067 -0
  348. ccxt/coinmetro.py +1797 -0
  349. ccxt/coinone.py +1127 -0
  350. ccxt/coinsph.py +1850 -0
  351. ccxt/coinspot.py +534 -0
  352. ccxt/cryptocom.py +2822 -0
  353. ccxt/currencycom.py +1950 -0
  354. ccxt/delta.py +3376 -0
  355. ccxt/deribit.py +3437 -0
  356. ccxt/digifinex.py +3959 -0
  357. ccxt/eterex.py +286 -0
  358. ccxt/excoino.py +399 -0
  359. ccxt/exir.py +375 -0
  360. ccxt/exmo.py +2462 -0
  361. ccxt/exnovin.py +360 -0
  362. ccxt/farhadexchange.py +266 -0
  363. ccxt/fmfwio.py +34 -0
  364. ccxt/gate.py +6975 -0
  365. ccxt/gateio.py +16 -0
  366. ccxt/gemini.py +1824 -0
  367. ccxt/hashkey.py +4150 -0
  368. ccxt/hitbtc.py +3423 -0
  369. ccxt/hitbtc3.py +16 -0
  370. ccxt/hitobit.py +391 -0
  371. ccxt/hollaex.py +1813 -0
  372. ccxt/htx.py +8505 -0
  373. ccxt/huobi.py +16 -0
  374. ccxt/huobijp.py +1801 -0
  375. ccxt/hyperliquid.py +2430 -0
  376. ccxt/idex.py +1766 -0
  377. ccxt/independentreserve.py +784 -0
  378. ccxt/indodax.py +1247 -0
  379. ccxt/jibitex.py +395 -0
  380. ccxt/kraken.py +2894 -0
  381. ccxt/krakenfutures.py +2601 -0
  382. ccxt/kucoin.py +4601 -0
  383. ccxt/kucoinfutures.py +2698 -0
  384. ccxt/kuna.py +1841 -0
  385. ccxt/latoken.py +1664 -0
  386. ccxt/lbank.py +2682 -0
  387. ccxt/luno.py +1067 -0
  388. ccxt/lykke.py +1270 -0
  389. ccxt/mercado.py +842 -0
  390. ccxt/mexc.py +5369 -0
  391. ccxt/ndax.py +2354 -0
  392. ccxt/nobitex.py +419 -0
  393. ccxt/novadax.py +1484 -0
  394. ccxt/oceanex.py +903 -0
  395. ccxt/okcoin.py +2936 -0
  396. ccxt/okexchange.py +349 -0
  397. ccxt/okx.py +7826 -0
  398. ccxt/ompfinex.py +472 -0
  399. ccxt/onetrading.py +1911 -0
  400. ccxt/oxfun.py +2772 -0
  401. ccxt/p2b.py +1194 -0
  402. ccxt/paradex.py +2015 -0
  403. ccxt/paymium.py +564 -0
  404. ccxt/phemex.py +4473 -0
  405. ccxt/poloniex.py +2232 -0
  406. ccxt/poloniexfutures.py +1717 -0
  407. ccxt/pro/__init__.py +149 -0
  408. ccxt/pro/alpaca.py +685 -0
  409. ccxt/pro/ascendex.py +916 -0
  410. ccxt/pro/bequant.py +38 -0
  411. ccxt/pro/binance.py +3488 -0
  412. ccxt/pro/binancecoinm.py +28 -0
  413. ccxt/pro/binanceus.py +48 -0
  414. ccxt/pro/binanceusdm.py +31 -0
  415. ccxt/pro/bingx.py +1264 -0
  416. ccxt/pro/bitcoincom.py +34 -0
  417. ccxt/pro/bitfinex.py +621 -0
  418. ccxt/pro/bitfinex2.py +1083 -0
  419. ccxt/pro/bitget.py +1692 -0
  420. ccxt/pro/bithumb.py +368 -0
  421. ccxt/pro/bitmart.py +1449 -0
  422. ccxt/pro/bitmex.py +1656 -0
  423. ccxt/pro/bitopro.py +445 -0
  424. ccxt/pro/bitpanda.py +15 -0
  425. ccxt/pro/bitrue.py +447 -0
  426. ccxt/pro/bitstamp.py +522 -0
  427. ccxt/pro/bitvavo.py +1270 -0
  428. ccxt/pro/blockchaincom.py +738 -0
  429. ccxt/pro/blofin.py +692 -0
  430. ccxt/pro/bybit.py +2000 -0
  431. ccxt/pro/cex.py +1440 -0
  432. ccxt/pro/coinbase.py +678 -0
  433. ccxt/pro/coinbaseadvanced.py +16 -0
  434. ccxt/pro/coinbaseexchange.py +895 -0
  435. ccxt/pro/coinbaseinternational.py +620 -0
  436. ccxt/pro/coincatch.py +1464 -0
  437. ccxt/pro/coincheck.py +199 -0
  438. ccxt/pro/coinex.py +1061 -0
  439. ccxt/pro/coinone.py +395 -0
  440. ccxt/pro/cryptocom.py +947 -0
  441. ccxt/pro/currencycom.py +536 -0
  442. ccxt/pro/deribit.py +892 -0
  443. ccxt/pro/exmo.py +629 -0
  444. ccxt/pro/gate.py +1416 -0
  445. ccxt/pro/gateio.py +15 -0
  446. ccxt/pro/gemini.py +865 -0
  447. ccxt/pro/hashkey.py +802 -0
  448. ccxt/pro/hitbtc.py +1216 -0
  449. ccxt/pro/hollaex.py +563 -0
  450. ccxt/pro/htx.py +2215 -0
  451. ccxt/pro/huobi.py +15 -0
  452. ccxt/pro/huobijp.py +570 -0
  453. ccxt/pro/hyperliquid.py +525 -0
  454. ccxt/pro/idex.py +672 -0
  455. ccxt/pro/independentreserve.py +270 -0
  456. ccxt/pro/kraken.py +1356 -0
  457. ccxt/pro/krakenfutures.py +1492 -0
  458. ccxt/pro/kucoin.py +1133 -0
  459. ccxt/pro/kucoinfutures.py +1081 -0
  460. ccxt/pro/lbank.py +843 -0
  461. ccxt/pro/luno.py +303 -0
  462. ccxt/pro/mexc.py +1122 -0
  463. ccxt/pro/ndax.py +506 -0
  464. ccxt/pro/okcoin.py +698 -0
  465. ccxt/pro/okx.py +1851 -0
  466. ccxt/pro/onetrading.py +1275 -0
  467. ccxt/pro/oxfun.py +950 -0
  468. ccxt/pro/p2b.py +419 -0
  469. ccxt/pro/paradex.py +352 -0
  470. ccxt/pro/phemex.py +1441 -0
  471. ccxt/pro/poloniex.py +1166 -0
  472. ccxt/pro/poloniexfutures.py +990 -0
  473. ccxt/pro/probit.py +551 -0
  474. ccxt/pro/upbit.py +520 -0
  475. ccxt/pro/vertex.py +943 -0
  476. ccxt/pro/wazirx.py +749 -0
  477. ccxt/pro/whitebit.py +864 -0
  478. ccxt/pro/woo.py +1078 -0
  479. ccxt/pro/woofipro.py +1183 -0
  480. ccxt/pro/xt.py +1067 -0
  481. ccxt/probit.py +1734 -0
  482. ccxt/ramzinex.py +476 -0
  483. ccxt/sarmayex.py +357 -0
  484. ccxt/sarrafex.py +478 -0
  485. ccxt/static_dependencies/__init__.py +1 -0
  486. ccxt/static_dependencies/ecdsa/__init__.py +14 -0
  487. ccxt/static_dependencies/ecdsa/_version.py +520 -0
  488. ccxt/static_dependencies/ecdsa/curves.py +56 -0
  489. ccxt/static_dependencies/ecdsa/der.py +221 -0
  490. ccxt/static_dependencies/ecdsa/ecdsa.py +310 -0
  491. ccxt/static_dependencies/ecdsa/ellipticcurve.py +197 -0
  492. ccxt/static_dependencies/ecdsa/keys.py +332 -0
  493. ccxt/static_dependencies/ecdsa/numbertheory.py +531 -0
  494. ccxt/static_dependencies/ecdsa/rfc6979.py +100 -0
  495. ccxt/static_dependencies/ecdsa/util.py +266 -0
  496. ccxt/static_dependencies/ethereum/__init__.py +7 -0
  497. ccxt/static_dependencies/ethereum/abi/__init__.py +16 -0
  498. ccxt/static_dependencies/ethereum/abi/abi.py +19 -0
  499. ccxt/static_dependencies/ethereum/abi/base.py +152 -0
  500. ccxt/static_dependencies/ethereum/abi/codec.py +217 -0
  501. ccxt/static_dependencies/ethereum/abi/constants.py +3 -0
  502. ccxt/static_dependencies/ethereum/abi/decoding.py +565 -0
  503. ccxt/static_dependencies/ethereum/abi/encoding.py +720 -0
  504. ccxt/static_dependencies/ethereum/abi/exceptions.py +139 -0
  505. ccxt/static_dependencies/ethereum/abi/grammar.py +443 -0
  506. ccxt/static_dependencies/ethereum/abi/packed.py +13 -0
  507. ccxt/static_dependencies/ethereum/abi/py.typed +0 -0
  508. ccxt/static_dependencies/ethereum/abi/registry.py +643 -0
  509. ccxt/static_dependencies/ethereum/abi/tools/__init__.py +3 -0
  510. ccxt/static_dependencies/ethereum/abi/tools/_strategies.py +230 -0
  511. ccxt/static_dependencies/ethereum/abi/utils/__init__.py +0 -0
  512. ccxt/static_dependencies/ethereum/abi/utils/numeric.py +83 -0
  513. ccxt/static_dependencies/ethereum/abi/utils/padding.py +27 -0
  514. ccxt/static_dependencies/ethereum/abi/utils/string.py +19 -0
  515. ccxt/static_dependencies/ethereum/account/__init__.py +3 -0
  516. ccxt/static_dependencies/ethereum/account/encode_typed_data/__init__.py +4 -0
  517. ccxt/static_dependencies/ethereum/account/encode_typed_data/encoding_and_hashing.py +239 -0
  518. ccxt/static_dependencies/ethereum/account/encode_typed_data/helpers.py +40 -0
  519. ccxt/static_dependencies/ethereum/account/messages.py +263 -0
  520. ccxt/static_dependencies/ethereum/account/py.typed +0 -0
  521. ccxt/static_dependencies/ethereum/hexbytes/__init__.py +5 -0
  522. ccxt/static_dependencies/ethereum/hexbytes/_utils.py +54 -0
  523. ccxt/static_dependencies/ethereum/hexbytes/main.py +65 -0
  524. ccxt/static_dependencies/ethereum/hexbytes/py.typed +0 -0
  525. ccxt/static_dependencies/ethereum/typing/__init__.py +63 -0
  526. ccxt/static_dependencies/ethereum/typing/abi.py +6 -0
  527. ccxt/static_dependencies/ethereum/typing/bls.py +7 -0
  528. ccxt/static_dependencies/ethereum/typing/discovery.py +5 -0
  529. ccxt/static_dependencies/ethereum/typing/encoding.py +7 -0
  530. ccxt/static_dependencies/ethereum/typing/enums.py +17 -0
  531. ccxt/static_dependencies/ethereum/typing/ethpm.py +9 -0
  532. ccxt/static_dependencies/ethereum/typing/evm.py +20 -0
  533. ccxt/static_dependencies/ethereum/typing/networks.py +1122 -0
  534. ccxt/static_dependencies/ethereum/typing/py.typed +0 -0
  535. ccxt/static_dependencies/ethereum/utils/__init__.py +115 -0
  536. ccxt/static_dependencies/ethereum/utils/abi.py +72 -0
  537. ccxt/static_dependencies/ethereum/utils/address.py +171 -0
  538. ccxt/static_dependencies/ethereum/utils/applicators.py +151 -0
  539. ccxt/static_dependencies/ethereum/utils/conversions.py +190 -0
  540. ccxt/static_dependencies/ethereum/utils/currency.py +107 -0
  541. ccxt/static_dependencies/ethereum/utils/curried/__init__.py +269 -0
  542. ccxt/static_dependencies/ethereum/utils/debug.py +20 -0
  543. ccxt/static_dependencies/ethereum/utils/decorators.py +132 -0
  544. ccxt/static_dependencies/ethereum/utils/encoding.py +6 -0
  545. ccxt/static_dependencies/ethereum/utils/exceptions.py +4 -0
  546. ccxt/static_dependencies/ethereum/utils/functional.py +75 -0
  547. ccxt/static_dependencies/ethereum/utils/hexadecimal.py +74 -0
  548. ccxt/static_dependencies/ethereum/utils/humanize.py +188 -0
  549. ccxt/static_dependencies/ethereum/utils/logging.py +159 -0
  550. ccxt/static_dependencies/ethereum/utils/module_loading.py +31 -0
  551. ccxt/static_dependencies/ethereum/utils/numeric.py +43 -0
  552. ccxt/static_dependencies/ethereum/utils/py.typed +0 -0
  553. ccxt/static_dependencies/ethereum/utils/toolz.py +76 -0
  554. ccxt/static_dependencies/ethereum/utils/types.py +54 -0
  555. ccxt/static_dependencies/ethereum/utils/typing/__init__.py +18 -0
  556. ccxt/static_dependencies/ethereum/utils/typing/misc.py +14 -0
  557. ccxt/static_dependencies/ethereum/utils/units.py +31 -0
  558. ccxt/static_dependencies/keccak/__init__.py +3 -0
  559. ccxt/static_dependencies/keccak/keccak.py +197 -0
  560. ccxt/static_dependencies/lark/__init__.py +38 -0
  561. ccxt/static_dependencies/lark/__pyinstaller/__init__.py +6 -0
  562. ccxt/static_dependencies/lark/__pyinstaller/hook-lark.py +14 -0
  563. ccxt/static_dependencies/lark/ast_utils.py +59 -0
  564. ccxt/static_dependencies/lark/common.py +86 -0
  565. ccxt/static_dependencies/lark/exceptions.py +292 -0
  566. ccxt/static_dependencies/lark/grammar.py +130 -0
  567. ccxt/static_dependencies/lark/grammars/__init__.py +0 -0
  568. ccxt/static_dependencies/lark/indenter.py +143 -0
  569. ccxt/static_dependencies/lark/lark.py +658 -0
  570. ccxt/static_dependencies/lark/lexer.py +678 -0
  571. ccxt/static_dependencies/lark/load_grammar.py +1428 -0
  572. ccxt/static_dependencies/lark/parse_tree_builder.py +391 -0
  573. ccxt/static_dependencies/lark/parser_frontends.py +257 -0
  574. ccxt/static_dependencies/lark/parsers/__init__.py +0 -0
  575. ccxt/static_dependencies/lark/parsers/cyk.py +340 -0
  576. ccxt/static_dependencies/lark/parsers/earley.py +314 -0
  577. ccxt/static_dependencies/lark/parsers/earley_common.py +42 -0
  578. ccxt/static_dependencies/lark/parsers/earley_forest.py +801 -0
  579. ccxt/static_dependencies/lark/parsers/grammar_analysis.py +203 -0
  580. ccxt/static_dependencies/lark/parsers/lalr_analysis.py +332 -0
  581. ccxt/static_dependencies/lark/parsers/lalr_interactive_parser.py +158 -0
  582. ccxt/static_dependencies/lark/parsers/lalr_parser.py +122 -0
  583. ccxt/static_dependencies/lark/parsers/lalr_parser_state.py +110 -0
  584. ccxt/static_dependencies/lark/parsers/xearley.py +165 -0
  585. ccxt/static_dependencies/lark/reconstruct.py +107 -0
  586. ccxt/static_dependencies/lark/tools/__init__.py +70 -0
  587. ccxt/static_dependencies/lark/tools/nearley.py +202 -0
  588. ccxt/static_dependencies/lark/tools/serialize.py +32 -0
  589. ccxt/static_dependencies/lark/tools/standalone.py +196 -0
  590. ccxt/static_dependencies/lark/tree.py +267 -0
  591. ccxt/static_dependencies/lark/tree_matcher.py +186 -0
  592. ccxt/static_dependencies/lark/tree_templates.py +180 -0
  593. ccxt/static_dependencies/lark/utils.py +343 -0
  594. ccxt/static_dependencies/lark/visitors.py +596 -0
  595. ccxt/static_dependencies/marshmallow/__init__.py +81 -0
  596. ccxt/static_dependencies/marshmallow/base.py +65 -0
  597. ccxt/static_dependencies/marshmallow/class_registry.py +94 -0
  598. ccxt/static_dependencies/marshmallow/decorators.py +231 -0
  599. ccxt/static_dependencies/marshmallow/error_store.py +60 -0
  600. ccxt/static_dependencies/marshmallow/exceptions.py +71 -0
  601. ccxt/static_dependencies/marshmallow/fields.py +2114 -0
  602. ccxt/static_dependencies/marshmallow/orderedset.py +89 -0
  603. ccxt/static_dependencies/marshmallow/schema.py +1228 -0
  604. ccxt/static_dependencies/marshmallow/types.py +12 -0
  605. ccxt/static_dependencies/marshmallow/utils.py +378 -0
  606. ccxt/static_dependencies/marshmallow/validate.py +678 -0
  607. ccxt/static_dependencies/marshmallow/warnings.py +2 -0
  608. ccxt/static_dependencies/marshmallow_dataclass/__init__.py +1047 -0
  609. ccxt/static_dependencies/marshmallow_dataclass/collection_field.py +51 -0
  610. ccxt/static_dependencies/marshmallow_dataclass/lazy_class_attribute.py +45 -0
  611. ccxt/static_dependencies/marshmallow_dataclass/mypy.py +71 -0
  612. ccxt/static_dependencies/marshmallow_dataclass/typing.py +14 -0
  613. ccxt/static_dependencies/marshmallow_dataclass/union_field.py +82 -0
  614. ccxt/static_dependencies/marshmallow_oneofschema/__init__.py +1 -0
  615. ccxt/static_dependencies/marshmallow_oneofschema/one_of_schema.py +193 -0
  616. ccxt/static_dependencies/msgpack/__init__.py +55 -0
  617. ccxt/static_dependencies/msgpack/exceptions.py +48 -0
  618. ccxt/static_dependencies/msgpack/ext.py +168 -0
  619. ccxt/static_dependencies/msgpack/fallback.py +951 -0
  620. ccxt/static_dependencies/parsimonious/__init__.py +10 -0
  621. ccxt/static_dependencies/parsimonious/exceptions.py +105 -0
  622. ccxt/static_dependencies/parsimonious/expressions.py +479 -0
  623. ccxt/static_dependencies/parsimonious/grammar.py +487 -0
  624. ccxt/static_dependencies/parsimonious/nodes.py +325 -0
  625. ccxt/static_dependencies/parsimonious/utils.py +40 -0
  626. ccxt/static_dependencies/starknet/__init__.py +0 -0
  627. ccxt/static_dependencies/starknet/cairo/__init__.py +0 -0
  628. ccxt/static_dependencies/starknet/cairo/data_types.py +123 -0
  629. ccxt/static_dependencies/starknet/cairo/deprecated_parse/__init__.py +0 -0
  630. ccxt/static_dependencies/starknet/cairo/deprecated_parse/cairo_types.py +77 -0
  631. ccxt/static_dependencies/starknet/cairo/deprecated_parse/parser.py +46 -0
  632. ccxt/static_dependencies/starknet/cairo/deprecated_parse/parser_transformer.py +138 -0
  633. ccxt/static_dependencies/starknet/cairo/felt.py +64 -0
  634. ccxt/static_dependencies/starknet/cairo/type_parser.py +121 -0
  635. ccxt/static_dependencies/starknet/cairo/v1/__init__.py +0 -0
  636. ccxt/static_dependencies/starknet/cairo/v1/type_parser.py +59 -0
  637. ccxt/static_dependencies/starknet/cairo/v2/__init__.py +0 -0
  638. ccxt/static_dependencies/starknet/cairo/v2/type_parser.py +77 -0
  639. ccxt/static_dependencies/starknet/ccxt_utils.py +7 -0
  640. ccxt/static_dependencies/starknet/common.py +15 -0
  641. ccxt/static_dependencies/starknet/constants.py +39 -0
  642. ccxt/static_dependencies/starknet/hash/__init__.py +0 -0
  643. ccxt/static_dependencies/starknet/hash/address.py +79 -0
  644. ccxt/static_dependencies/starknet/hash/compiled_class_hash_objects.py +111 -0
  645. ccxt/static_dependencies/starknet/hash/selector.py +16 -0
  646. ccxt/static_dependencies/starknet/hash/storage.py +12 -0
  647. ccxt/static_dependencies/starknet/hash/utils.py +78 -0
  648. ccxt/static_dependencies/starknet/models/__init__.py +0 -0
  649. ccxt/static_dependencies/starknet/models/typed_data.py +45 -0
  650. ccxt/static_dependencies/starknet/serialization/__init__.py +24 -0
  651. ccxt/static_dependencies/starknet/serialization/_calldata_reader.py +40 -0
  652. ccxt/static_dependencies/starknet/serialization/_context.py +142 -0
  653. ccxt/static_dependencies/starknet/serialization/data_serializers/__init__.py +10 -0
  654. ccxt/static_dependencies/starknet/serialization/data_serializers/_common.py +82 -0
  655. ccxt/static_dependencies/starknet/serialization/data_serializers/array_serializer.py +43 -0
  656. ccxt/static_dependencies/starknet/serialization/data_serializers/bool_serializer.py +37 -0
  657. ccxt/static_dependencies/starknet/serialization/data_serializers/byte_array_serializer.py +66 -0
  658. ccxt/static_dependencies/starknet/serialization/data_serializers/cairo_data_serializer.py +71 -0
  659. ccxt/static_dependencies/starknet/serialization/data_serializers/enum_serializer.py +71 -0
  660. ccxt/static_dependencies/starknet/serialization/data_serializers/felt_serializer.py +50 -0
  661. ccxt/static_dependencies/starknet/serialization/data_serializers/named_tuple_serializer.py +58 -0
  662. ccxt/static_dependencies/starknet/serialization/data_serializers/option_serializer.py +43 -0
  663. ccxt/static_dependencies/starknet/serialization/data_serializers/output_serializer.py +40 -0
  664. ccxt/static_dependencies/starknet/serialization/data_serializers/payload_serializer.py +72 -0
  665. ccxt/static_dependencies/starknet/serialization/data_serializers/struct_serializer.py +36 -0
  666. ccxt/static_dependencies/starknet/serialization/data_serializers/tuple_serializer.py +36 -0
  667. ccxt/static_dependencies/starknet/serialization/data_serializers/uint256_serializer.py +76 -0
  668. ccxt/static_dependencies/starknet/serialization/data_serializers/uint_serializer.py +100 -0
  669. ccxt/static_dependencies/starknet/serialization/data_serializers/unit_serializer.py +32 -0
  670. ccxt/static_dependencies/starknet/serialization/errors.py +10 -0
  671. ccxt/static_dependencies/starknet/serialization/factory.py +229 -0
  672. ccxt/static_dependencies/starknet/serialization/function_serialization_adapter.py +110 -0
  673. ccxt/static_dependencies/starknet/serialization/tuple_dataclass.py +59 -0
  674. ccxt/static_dependencies/starknet/utils/__init__.py +0 -0
  675. ccxt/static_dependencies/starknet/utils/constructor_args_translator.py +86 -0
  676. ccxt/static_dependencies/starknet/utils/iterable.py +13 -0
  677. ccxt/static_dependencies/starknet/utils/schema.py +13 -0
  678. ccxt/static_dependencies/starknet/utils/typed_data.py +182 -0
  679. ccxt/static_dependencies/starkware/__init__.py +0 -0
  680. ccxt/static_dependencies/starkware/crypto/__init__.py +0 -0
  681. ccxt/static_dependencies/starkware/crypto/fast_pedersen_hash.py +50 -0
  682. ccxt/static_dependencies/starkware/crypto/math_utils.py +78 -0
  683. ccxt/static_dependencies/starkware/crypto/signature.py +2344 -0
  684. ccxt/static_dependencies/starkware/crypto/utils.py +63 -0
  685. ccxt/static_dependencies/sympy/__init__.py +0 -0
  686. ccxt/static_dependencies/sympy/core/__init__.py +0 -0
  687. ccxt/static_dependencies/sympy/core/intfunc.py +35 -0
  688. ccxt/static_dependencies/sympy/external/__init__.py +0 -0
  689. ccxt/static_dependencies/sympy/external/gmpy.py +345 -0
  690. ccxt/static_dependencies/sympy/external/importtools.py +187 -0
  691. ccxt/static_dependencies/sympy/external/ntheory.py +637 -0
  692. ccxt/static_dependencies/sympy/external/pythonmpq.py +341 -0
  693. ccxt/static_dependencies/toolz/__init__.py +26 -0
  694. ccxt/static_dependencies/toolz/_signatures.py +784 -0
  695. ccxt/static_dependencies/toolz/_version.py +520 -0
  696. ccxt/static_dependencies/toolz/compatibility.py +30 -0
  697. ccxt/static_dependencies/toolz/curried/__init__.py +101 -0
  698. ccxt/static_dependencies/toolz/curried/exceptions.py +22 -0
  699. ccxt/static_dependencies/toolz/curried/operator.py +22 -0
  700. ccxt/static_dependencies/toolz/dicttoolz.py +339 -0
  701. ccxt/static_dependencies/toolz/functoolz.py +1049 -0
  702. ccxt/static_dependencies/toolz/itertoolz.py +1057 -0
  703. ccxt/static_dependencies/toolz/recipes.py +46 -0
  704. ccxt/static_dependencies/toolz/utils.py +9 -0
  705. ccxt/static_dependencies/typing_inspect/__init__.py +0 -0
  706. ccxt/static_dependencies/typing_inspect/typing_inspect.py +851 -0
  707. ccxt/tabdeal.py +364 -0
  708. ccxt/test/__init__.py +3 -0
  709. ccxt/test/base/__init__.py +29 -0
  710. ccxt/test/base/test_account.py +26 -0
  711. ccxt/test/base/test_balance.py +56 -0
  712. ccxt/test/base/test_borrow_interest.py +35 -0
  713. ccxt/test/base/test_borrow_rate.py +32 -0
  714. ccxt/test/base/test_calculate_fee.py +51 -0
  715. ccxt/test/base/test_crypto.py +127 -0
  716. ccxt/test/base/test_currency.py +76 -0
  717. ccxt/test/base/test_datetime.py +109 -0
  718. ccxt/test/base/test_decimal_to_precision.py +392 -0
  719. ccxt/test/base/test_deep_extend.py +68 -0
  720. ccxt/test/base/test_deposit_withdrawal.py +50 -0
  721. ccxt/test/base/test_exchange_datetime_functions.py +76 -0
  722. ccxt/test/base/test_funding_rate_history.py +29 -0
  723. ccxt/test/base/test_last_price.py +31 -0
  724. ccxt/test/base/test_ledger_entry.py +45 -0
  725. ccxt/test/base/test_ledger_item.py +48 -0
  726. ccxt/test/base/test_leverage_tier.py +33 -0
  727. ccxt/test/base/test_liquidation.py +50 -0
  728. ccxt/test/base/test_margin_mode.py +24 -0
  729. ccxt/test/base/test_margin_modification.py +35 -0
  730. ccxt/test/base/test_market.py +193 -0
  731. ccxt/test/base/test_number.py +411 -0
  732. ccxt/test/base/test_ohlcv.py +33 -0
  733. ccxt/test/base/test_open_interest.py +32 -0
  734. ccxt/test/base/test_order.py +64 -0
  735. ccxt/test/base/test_order_book.py +69 -0
  736. ccxt/test/base/test_position.py +60 -0
  737. ccxt/test/base/test_shared_methods.py +353 -0
  738. ccxt/test/base/test_status.py +24 -0
  739. ccxt/test/base/test_throttle.py +126 -0
  740. ccxt/test/base/test_ticker.py +92 -0
  741. ccxt/test/base/test_trade.py +47 -0
  742. ccxt/test/base/test_trading_fee.py +26 -0
  743. ccxt/test/base/test_transaction.py +39 -0
  744. ccxt/test/test_async.py +1649 -0
  745. ccxt/test/test_sync.py +1648 -0
  746. ccxt/test/tests_async.py +1558 -0
  747. ccxt/test/tests_helpers.py +287 -0
  748. ccxt/test/tests_init.py +39 -0
  749. ccxt/test/tests_sync.py +1555 -0
  750. ccxt/tetherland.py +349 -0
  751. ccxt/timex.py +1593 -0
  752. ccxt/tokocrypto.py +2405 -0
  753. ccxt/tradeogre.py +608 -0
  754. ccxt/twox.py +326 -0
  755. ccxt/ubitex.py +409 -0
  756. ccxt/upbit.py +1833 -0
  757. ccxt/vertex.py +2922 -0
  758. ccxt/wallex.py +445 -0
  759. ccxt/wavesexchange.py +2472 -0
  760. ccxt/wazirx.py +1224 -0
  761. ccxt/whitebit.py +2469 -0
  762. ccxt/woo.py +3114 -0
  763. ccxt/woofipro.py +2533 -0
  764. ccxt/xt.py +4453 -0
  765. ccxt/yobit.py +1283 -0
  766. ccxt/zaif.py +725 -0
  767. ccxt/zonda.py +1828 -0
  768. ccxt_ir-4.3.46.0.1.dist-info/LICENSE.txt +21 -0
  769. ccxt_ir-4.3.46.0.1.dist-info/METADATA +655 -0
  770. ccxt_ir-4.3.46.0.1.dist-info/RECORD +772 -0
  771. ccxt_ir-4.3.46.0.1.dist-info/WHEEL +6 -0
  772. ccxt_ir-4.3.46.0.1.dist-info/top_level.txt +1 -0
@@ -0,0 +1,1492 @@
1
+ # -*- coding: utf-8 -*-
2
+
3
+ # PLEASE DO NOT EDIT THIS FILE, IT IS GENERATED AND WILL BE OVERWRITTEN:
4
+ # https://github.com/ccxt/ccxt/blob/master/CONTRIBUTING.md#how-to-contribute-code
5
+
6
+ import ccxt.async_support
7
+ from ccxt.async_support.base.ws.cache import ArrayCache, ArrayCacheBySymbolById
8
+ import hashlib
9
+ from ccxt.base.types import Balances, Int, Order, OrderBook, Position, Str, Strings, Ticker, Tickers, Trade
10
+ from ccxt.async_support.base.ws.client import Client
11
+ from typing import List
12
+ from ccxt.base.errors import ExchangeError
13
+ from ccxt.base.errors import AuthenticationError
14
+ from ccxt.base.errors import ArgumentsRequired
15
+ from ccxt.base.precise import Precise
16
+
17
+
18
+ class krakenfutures(ccxt.async_support.krakenfutures):
19
+
20
+ def describe(self):
21
+ return self.deep_extend(super(krakenfutures, self).describe(), {
22
+ 'has': {
23
+ 'ws': True,
24
+ 'cancelAllOrdersWs': False,
25
+ 'cancelOrdersWs': False,
26
+ 'cancelOrderWs': False,
27
+ 'createOrderWs': False,
28
+ 'editOrderWs': False,
29
+ 'fetchBalanceWs': False,
30
+ 'fetchOpenOrdersWs': False,
31
+ 'fetchOrderWs': False,
32
+ 'fetchTradesWs': False,
33
+ 'watchOHLCV': False,
34
+ 'watchOrderBook': True,
35
+ 'watchOrderBookForSymbols': True,
36
+ 'watchTicker': True,
37
+ 'watchTickers': True,
38
+ 'watchBidsAsks': True,
39
+ 'watchTrades': True,
40
+ 'watchTradesForSymbols': True,
41
+ 'watchBalance': True,
42
+ # 'watchStatus': True, # https://docs.futures.kraken.com/#websocket-api-public-feeds-heartbeat
43
+ 'watchOrders': True,
44
+ 'watchMyTrades': True,
45
+ 'watchPositions': True,
46
+ },
47
+ 'urls': {
48
+ 'api': {
49
+ 'ws': 'wss://futures.kraken.com/ws/v1',
50
+ },
51
+ 'test': {
52
+ 'ws': 'wss://demo-futures.kraken.com/ws/v1',
53
+ },
54
+ },
55
+ 'options': {
56
+ 'tradesLimit': 1000,
57
+ 'ordersLimit': 1000,
58
+ 'OHLCVLimit': 1000,
59
+ 'connectionLimit': 100, # https://docs.futures.kraken.com/#websocket-api-websocket-api-introduction-subscriptions-limits
60
+ 'requestLimit': 100, # per second
61
+ 'fetchBalance': {
62
+ 'type': None,
63
+ },
64
+ },
65
+ 'streaming': {
66
+ 'keepAlive': 30000,
67
+ },
68
+ })
69
+
70
+ async def authenticate(self, params={}):
71
+ """
72
+ * @ignore
73
+ authenticates the user to access private web socket channels
74
+ :see: https://docs.futures.kraken.com/#websocket-api-public-feeds-challenge
75
+ :returns dict: response from exchange
76
+ """
77
+ self.check_required_credentials()
78
+ # Hash the challenge with the SHA-256 algorithm
79
+ # Base64-decode your api_secret
80
+ # Use the result of step 2 to hash the result of step 1 with the HMAC-SHA-512 algorithm
81
+ # Base64-encode the result of step 3
82
+ url = self.urls['api']['ws']
83
+ messageHash = 'challenge'
84
+ client = self.client(url)
85
+ future = self.safe_value(client.subscriptions, messageHash)
86
+ if future is None:
87
+ request: dict = {
88
+ 'event': 'challenge',
89
+ 'api_key': self.apiKey,
90
+ }
91
+ message = self.extend(request, params)
92
+ future = await self.watch(url, messageHash, message, messageHash)
93
+ client.subscriptions[messageHash] = future
94
+ return future
95
+
96
+ async def watch_order_book_for_symbols(self, symbols: List[str], limit: Int = None, params={}) -> OrderBook:
97
+ """
98
+ watches information on open orders with bid(buy) and ask(sell) prices, volumes and other data
99
+ :see: https://docs.futures.kraken.com/#websocket-api-public-feeds-challenge
100
+ :param str[] symbols: unified array of symbols
101
+ :param int [limit]: the maximum amount of order book entries to return
102
+ :param dict [params]: extra parameters specific to the exchange API endpoint
103
+ :returns dict: A dictionary of `order book structures <https://docs.ccxt.com/#/?id=order-book-structure>` indexed by market symbols
104
+ """
105
+ orderbook = await self.watch_multi_helper('orderbook', 'book', symbols, {'limit': limit}, params)
106
+ return orderbook.limit()
107
+
108
+ async def subscribe_public(self, name: str, symbols: List[str], params={}):
109
+ """
110
+ * @ignore
111
+ Connects to a websocket channel
112
+ :param str name: name of the channel
113
+ :param str[] symbols: CCXT market symbols
114
+ :param dict [params]: extra parameters specific to the krakenfutures api
115
+ :returns dict: data from the websocket stream
116
+ """
117
+ await self.load_markets()
118
+ url = self.urls['api']['ws']
119
+ subscribe: dict = {
120
+ 'event': 'subscribe',
121
+ 'feed': name,
122
+ }
123
+ marketIds = []
124
+ messageHash = name
125
+ if symbols is None:
126
+ symbols = []
127
+ for i in range(0, len(symbols)):
128
+ symbol = symbols[i]
129
+ marketIds.append(self.market_id(symbol))
130
+ length = len(symbols)
131
+ if length == 1:
132
+ market = self.market(marketIds[0])
133
+ messageHash = messageHash + ':' + market['symbol']
134
+ subscribe['product_ids'] = marketIds
135
+ request = self.extend(subscribe, params)
136
+ return await self.watch(url, messageHash, request, messageHash)
137
+
138
+ async def subscribe_private(self, name: str, messageHash: str, params={}):
139
+ """
140
+ * @ignore
141
+ Connects to a websocket channel
142
+ :param str name: name of the channel
143
+ :param str[] symbols: CCXT market symbols
144
+ :param dict [params]: extra parameters specific to the krakenfutures api
145
+ :returns dict: data from the websocket stream
146
+ """
147
+ await self.load_markets()
148
+ await self.authenticate()
149
+ url = self.urls['api']['ws']
150
+ subscribe: dict = {
151
+ 'event': 'subscribe',
152
+ 'feed': name,
153
+ 'api_key': self.apiKey,
154
+ 'original_challenge': self.options['challenge'],
155
+ 'signed_challenge': self.options['signedChallenge'],
156
+ }
157
+ request = self.extend(subscribe, params)
158
+ return await self.watch(url, messageHash, request, messageHash)
159
+
160
+ async def watch_ticker(self, symbol: str, params={}) -> Ticker:
161
+ """
162
+ watches a price ticker, a statistical calculation with the information calculated over the past 24 hours for a specific market
163
+ :see: https://docs.futures.kraken.com/#websocket-api-public-feeds-ticker
164
+ :param str symbol: unified symbol of the market to fetch the ticker for
165
+ :param dict [params]: extra parameters specific to the exchange API endpoint
166
+ :returns dict: a `ticker structure <https://docs.ccxt.com/#/?id=ticker-structure>`
167
+ """
168
+ await self.load_markets()
169
+ symbol = self.symbol(symbol)
170
+ tickers = await self.watch_tickers([symbol], params)
171
+ return tickers[symbol]
172
+
173
+ async def watch_tickers(self, symbols: Strings = None, params={}) -> Tickers:
174
+ """
175
+ watches a price ticker, a statistical calculation with the information calculated over the past 24 hours for a specific market
176
+ :see: https://docs.futures.kraken.com/#websocket-api-public-feeds-ticker
177
+ :param str symbol: unified symbol of the market to fetch the ticker for
178
+ :param dict [params]: extra parameters specific to the exchange API endpoint
179
+ :returns dict: a `ticker structure <https://docs.ccxt.com/#/?id=ticker-structure>`
180
+ """
181
+ await self.load_markets()
182
+ symbols = self.market_symbols(symbols, None, False)
183
+ ticker = await self.watch_multi_helper('ticker', 'ticker', symbols, None, params)
184
+ if self.newUpdates:
185
+ result: dict = {}
186
+ result[ticker['symbol']] = ticker
187
+ return result
188
+ return self.filter_by_array(self.tickers, 'symbol', symbols)
189
+
190
+ async def watch_bids_asks(self, symbols: Strings = None, params={}) -> Tickers:
191
+ """
192
+ :see: https://docs.futures.kraken.com/#websocket-api-public-feeds-ticker-lite
193
+ watches best bid & ask for symbols
194
+ :param str[] symbols: unified symbol of the market to fetch the ticker for
195
+ :param dict [params]: extra parameters specific to the exchange API endpoint
196
+ :returns dict: a `ticker structure <https://docs.ccxt.com/#/?id=ticker-structure>`
197
+ """
198
+ ticker = await self.watch_multi_helper('bidask', 'ticker_lite', symbols, None, params)
199
+ if self.newUpdates:
200
+ result: dict = {}
201
+ result[ticker['symbol']] = ticker
202
+ return result
203
+ return self.filter_by_array(self.bidsasks, 'symbol', symbols)
204
+
205
+ async def watch_trades(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Trade]:
206
+ """
207
+ get the list of most recent trades for a particular symbol
208
+ :see: https://docs.futures.kraken.com/#websocket-api-public-feeds-trade
209
+ :param str symbol: unified symbol of the market to fetch trades for
210
+ :param int [since]: timestamp in ms of the earliest trade to fetch
211
+ :param int [limit]: the maximum amount of trades to fetch
212
+ :param dict [params]: extra parameters specific to the exchange API endpoint
213
+ :returns dict[]: a list of `trade structures <https://docs.ccxt.com/#/?id=public-trades>`
214
+ """
215
+ return await self.watch_trades_for_symbols([symbol], since, limit, params)
216
+
217
+ async def watch_trades_for_symbols(self, symbols: List[str], since: Int = None, limit: Int = None, params={}) -> List[Trade]:
218
+ """
219
+ :see: https://docs.futures.kraken.com/#websocket-api-public-feeds-trade
220
+ get the list of most recent trades for a list of symbols
221
+ :param str[] symbols: unified symbol of the market to fetch trades for
222
+ :param int [since]: timestamp in ms of the earliest trade to fetch
223
+ :param int [limit]: the maximum amount of trades to fetch
224
+ :param dict [params]: extra parameters specific to the exchange API endpoint
225
+ :returns dict[]: a list of `trade structures <https://docs.ccxt.com/#/?id=public-trades>`
226
+ """
227
+ trades = await self.watch_multi_helper('trade', 'trade', symbols, None, params)
228
+ if self.newUpdates:
229
+ first = self.safe_list(trades, 0)
230
+ tradeSymbol = self.safe_string(first, 'symbol')
231
+ limit = trades.getLimit(tradeSymbol, limit)
232
+ return self.filter_by_since_limit(trades, since, limit, 'timestamp', True)
233
+
234
+ async def watch_order_book(self, symbol: str, limit: Int = None, params={}) -> OrderBook:
235
+ """
236
+ watches information on open orders with bid(buy) and ask(sell) prices, volumes and other data
237
+ :see: https://docs.futures.kraken.com/#websocket-api-public-feeds-book
238
+ :param str symbol: unified symbol of the market to fetch the order book for
239
+ :param int [limit]: not used by krakenfutures watchOrderBook
240
+ :param dict [params]: extra parameters specific to the exchange API endpoint
241
+ :returns dict: A dictionary of `order book structures <https://docs.ccxt.com/#/?id=order-book-structure>` indexed by market symbols
242
+ """
243
+ return await self.watch_order_book_for_symbols([symbol], limit, params)
244
+
245
+ async def watch_positions(self, symbols: Strings = None, since: Int = None, limit: Int = None, params={}) -> List[Position]:
246
+ """
247
+ :see: https://docs.futures.kraken.com/#websocket-api-private-feeds-open-positions
248
+ watch all open positions
249
+ :param str[]|None symbols: list of unified market symbols
250
+ :param dict params: extra parameters specific to the exchange API endpoint
251
+ :returns dict[]: a list of `position structure <https://docs.ccxt.com/en/latest/manual.html#position-structure>`
252
+ """
253
+ await self.load_markets()
254
+ messageHash = ''
255
+ symbols = self.market_symbols(symbols)
256
+ if not self.is_empty(symbols):
257
+ messageHash = '::' + ','.join(symbols)
258
+ messageHash = 'positions' + messageHash
259
+ newPositions = await self.subscribe_private('open_positions', messageHash, params)
260
+ if self.newUpdates:
261
+ return newPositions
262
+ return self.filter_by_symbols_since_limit(self.positions, symbols, since, limit, True)
263
+
264
+ def handle_positions(self, client, message):
265
+ #
266
+ # {
267
+ # feed: 'open_positions',
268
+ # account: '3b111acc-4fcc-45be-a622-57e611fe9f7f',
269
+ # positions: [
270
+ # {
271
+ # instrument: 'PF_LTCUSD',
272
+ # balance: 0.5,
273
+ # pnl: -0.8628305877699987,
274
+ # entry_price: 70.53,
275
+ # mark_price: 68.80433882446,
276
+ # index_price: 68.8091,
277
+ # liquidation_threshold: 0,
278
+ # effective_leverage: 0.007028866753648637,
279
+ # return_on_equity: -1.2233525985679834,
280
+ # unrealized_funding: 0.0000690610530935388,
281
+ # initial_margin: 0.7053,
282
+ # initial_margin_with_orders: 0.7053,
283
+ # maintenance_margin: 0.35265,
284
+ # pnl_currency: 'USD'
285
+ # }
286
+ # ],
287
+ # seq: 0,
288
+ # timestamp: 1698608414910
289
+ # }
290
+ #
291
+ if self.positions is None:
292
+ self.positions = ArrayCacheBySymbolById()
293
+ cache = self.positions
294
+ rawPositions = self.safe_value(message, 'positions', [])
295
+ newPositions = []
296
+ for i in range(0, len(rawPositions)):
297
+ rawPosition = rawPositions[i]
298
+ position = self.parse_ws_position(rawPosition)
299
+ timestamp = self.safe_integer(message, 'timestamp')
300
+ position['timestamp'] = timestamp
301
+ position['datetime'] = self.iso8601(timestamp)
302
+ newPositions.append(position)
303
+ cache.append(position)
304
+ messageHashes = self.find_message_hashes(client, 'positions::')
305
+ for i in range(0, len(messageHashes)):
306
+ messageHash = messageHashes[i]
307
+ parts = messageHash.split('::')
308
+ symbolsString = parts[1]
309
+ symbols = symbolsString.split(',')
310
+ positions = self.filter_by_array(newPositions, 'symbol', symbols, False)
311
+ if not self.is_empty(positions):
312
+ client.resolve(positions, messageHash)
313
+ client.resolve(newPositions, 'positions')
314
+
315
+ def parse_ws_position(self, position, market=None):
316
+ #
317
+ # {
318
+ # instrument: 'PF_LTCUSD',
319
+ # balance: 0.5,
320
+ # pnl: -0.8628305877699987,
321
+ # entry_price: 70.53,
322
+ # mark_price: 68.80433882446,
323
+ # index_price: 68.8091,
324
+ # liquidation_threshold: 0,
325
+ # effective_leverage: 0.007028866753648637,
326
+ # return_on_equity: -1.2233525985679834,
327
+ # unrealized_funding: 0.0000690610530935388,
328
+ # initial_margin: 0.7053,
329
+ # initial_margin_with_orders: 0.7053,
330
+ # maintenance_margin: 0.35265,
331
+ # pnl_currency: 'USD'
332
+ # }
333
+ #
334
+ marketId = self.safe_string(position, 'instrument')
335
+ hedged = 'both'
336
+ balance = self.safe_number(position, 'balance')
337
+ side = 'long' if (balance > 0) else 'short'
338
+ return self.safe_position({
339
+ 'info': position,
340
+ 'id': None,
341
+ 'symbol': self.safe_symbol(marketId),
342
+ 'notional': None,
343
+ 'marginMode': None,
344
+ 'liquidationPrice': self.safe_number(position, 'liquidation_threshold'),
345
+ 'entryPrice': self.safe_number(position, 'entry_price'),
346
+ 'unrealizedPnl': self.safe_number(position, 'pnl'),
347
+ 'percentage': self.safe_number(position, 'return_on_equity'),
348
+ 'contracts': self.parse_number(Precise.string_abs(self.number_to_string(balance))),
349
+ 'contractSize': None,
350
+ 'markPrice': self.safe_number(position, 'mark_price'),
351
+ 'side': side,
352
+ 'hedged': hedged,
353
+ 'timestamp': None,
354
+ 'datetime': None,
355
+ 'maintenanceMargin': self.safe_number(position, 'maintenance_margin'),
356
+ 'maintenanceMarginPercentage': None,
357
+ 'collateral': None,
358
+ 'initialMargin': self.safe_number(position, 'initial_margin'),
359
+ 'initialMarginPercentage': None,
360
+ 'leverage': None,
361
+ 'marginRatio': None,
362
+ })
363
+
364
+ async def watch_orders(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Order]:
365
+ """
366
+ watches information on multiple orders made by the user
367
+ :see: https://docs.futures.kraken.com/#websocket-api-private-feeds-open-orders
368
+ :see: https://docs.futures.kraken.com/#websocket-api-private-feeds-open-orders-verbose
369
+ :param str symbol: not used by krakenfutures watchOrders
370
+ :param int [since]: not used by krakenfutures watchOrders
371
+ :param int [limit]: not used by krakenfutures watchOrders
372
+ :param dict [params]: extra parameters specific to the exchange API endpoint
373
+ :returns dict[]: a list of `order structures <https://docs.ccxt.com/#/?id=order-structure>`
374
+ """
375
+ await self.load_markets()
376
+ name = 'open_orders'
377
+ messageHash = 'orders'
378
+ if symbol is not None:
379
+ market = self.market(symbol)
380
+ messageHash += ':' + market['symbol']
381
+ orders = await self.subscribe_private(name, messageHash, params)
382
+ if self.newUpdates:
383
+ limit = orders.getLimit(symbol, limit)
384
+ return self.filter_by_since_limit(orders, since, limit, 'timestamp', True)
385
+
386
+ async def watch_my_trades(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Trade]:
387
+ """
388
+ watches information on multiple trades made by the user
389
+ :see: https://docs.futures.kraken.com/#websocket-api-private-feeds-fills
390
+ :param str symbol: unified market symbol of the market orders were made in
391
+ :param int [since]: the earliest time in ms to fetch orders for
392
+ :param int [limit]: the maximum number of order structures to retrieve
393
+ :param dict [params]: extra parameters specific to the exchange API endpoint
394
+ :returns dict[]: a list of `trade structures <https://docs.ccxt.com/#/?id=trade-structure>`
395
+ """
396
+ await self.load_markets()
397
+ name = 'fills'
398
+ messageHash = 'myTrades'
399
+ if symbol is not None:
400
+ market = self.market(symbol)
401
+ messageHash += ':' + market['symbol']
402
+ trades = await self.subscribe_private(name, messageHash, params)
403
+ if self.newUpdates:
404
+ limit = trades.getLimit(symbol, limit)
405
+ return self.filter_by_since_limit(trades, since, limit, 'timestamp', True)
406
+
407
+ async def watch_balance(self, params={}) -> Balances:
408
+ """
409
+ watches information on multiple orders made by the user
410
+ :see: https://docs.futures.kraken.com/#websocket-api-private-feeds-balances
411
+ :param str symbol: not used by krakenfutures watchBalance
412
+ :param int [since]: not used by krakenfutures watchBalance
413
+ :param int [limit]: not used by krakenfutures watchBalance
414
+ :param dict [params]: extra parameters specific to the exchange API endpoint
415
+ :param str [params.account]: can be either 'futures' or 'flex_futures'
416
+ :returns dict} a object of wallet types each with a balance structure {@link https://docs.ccxt.com/#/?id=balance-structure:
417
+ """
418
+ await self.load_markets()
419
+ name = 'balances'
420
+ messageHash = name
421
+ account = None
422
+ account, params = self.handle_option_and_params(params, 'watchBalance', 'account')
423
+ if account is not None:
424
+ if account != 'futures' and account != 'flex_futures':
425
+ raise ArgumentsRequired(self.id + ' watchBalance account must be either \'futures\' or \'flex_futures\'')
426
+ messageHash += ':' + account
427
+ return await self.subscribe_private(name, messageHash, params)
428
+
429
+ def handle_trade(self, client: Client, message):
430
+ #
431
+ # snapshot
432
+ #
433
+ # {
434
+ # "feed": "trade_snapshot",
435
+ # "product_id": "PI_XBTUSD",
436
+ # "trades": [
437
+ # {
438
+ # "feed": "trade",
439
+ # "product_id": "PI_XBTUSD",
440
+ # "uid": "caa9c653-420b-4c24-a9f1-462a054d86f1",
441
+ # "side": "sell",
442
+ # "type": "fill",
443
+ # "seq": 655508,
444
+ # "time": 1612269657781,
445
+ # "qty": 440,
446
+ # "price": 34893
447
+ # },
448
+ # ...
449
+ # ]
450
+ # }
451
+ #
452
+ # update
453
+ #
454
+ # {
455
+ # "feed": "trade",
456
+ # "product_id": "PI_XBTUSD",
457
+ # "uid": "05af78ac-a774-478c-a50c-8b9c234e071e",
458
+ # "side": "sell",
459
+ # "type": "fill",
460
+ # "seq": 653355,
461
+ # "time": 1612266317519,
462
+ # "qty": 15000,
463
+ # "price": 34969.5
464
+ # }
465
+ #
466
+ channel = self.safe_string(message, 'feed')
467
+ marketId = self.safe_string(message, 'product_id')
468
+ if marketId is not None:
469
+ market = self.market(marketId)
470
+ symbol = market['symbol']
471
+ messageHash = self.get_message_hash('trade', None, symbol)
472
+ if self.safe_list(self.trades, symbol) is None:
473
+ tradesLimit = self.safe_integer(self.options, 'tradesLimit', 1000)
474
+ self.trades[symbol] = ArrayCache(tradesLimit)
475
+ tradesArray = self.trades[symbol]
476
+ if channel == 'trade_snapshot':
477
+ trades = self.safe_list(message, 'trades', [])
478
+ length = len(trades)
479
+ for i in range(0, length):
480
+ index = length - 1 - i # need reverse to correct chronology
481
+ item = trades[index]
482
+ trade = self.parse_ws_trade(item)
483
+ tradesArray.append(trade)
484
+ else:
485
+ trade = self.parse_ws_trade(message)
486
+ tradesArray.append(trade)
487
+ client.resolve(tradesArray, messageHash)
488
+
489
+ def parse_ws_trade(self, trade, market=None):
490
+ #
491
+ # {
492
+ # "feed": "trade",
493
+ # "product_id": "PI_XBTUSD",
494
+ # "uid": "caa9c653-420b-4c24-a9f1-462a054d86f1",
495
+ # "side": "sell",
496
+ # "type": "fill",
497
+ # "seq": 655508,
498
+ # "time": 1612269657781,
499
+ # "qty": 440,
500
+ # "price": 34893
501
+ # }
502
+ #
503
+ marketId = self.safe_string(trade, 'product_id')
504
+ market = self.safe_market(marketId, market)
505
+ timestamp = self.safe_integer(trade, 'time')
506
+ return self.safe_trade({
507
+ 'info': trade,
508
+ 'id': self.safe_string(trade, 'uid'),
509
+ 'symbol': self.safe_string(market, 'symbol'),
510
+ 'timestamp': timestamp,
511
+ 'datetime': self.iso8601(timestamp),
512
+ 'order': None,
513
+ 'type': self.safe_string(trade, 'type'),
514
+ 'side': self.safe_string(trade, 'side'),
515
+ 'takerOrMaker': 'taker',
516
+ 'price': self.safe_string(trade, 'price'),
517
+ 'amount': self.safe_string(trade, 'qty'),
518
+ 'cost': None,
519
+ 'fee': {
520
+ 'rate': None,
521
+ 'cost': None,
522
+ 'currency': None,
523
+ },
524
+ }, market)
525
+
526
+ def parse_ws_order_trade(self, trade, market=None):
527
+ #
528
+ # {
529
+ # "symbol": "BTC_USDT",
530
+ # "type": "LIMIT",
531
+ # "quantity": "1",
532
+ # "orderId": "32471407854219264",
533
+ # "tradeFee": "0",
534
+ # "clientOrderId": "",
535
+ # "accountType": "SPOT",
536
+ # "feeCurrency": "",
537
+ # "eventType": "place",
538
+ # "source": "API",
539
+ # "side": "BUY",
540
+ # "filledQuantity": "0",
541
+ # "filledAmount": "0",
542
+ # "matchRole": "MAKER",
543
+ # "state": "NEW",
544
+ # "tradeTime": 0,
545
+ # "tradeAmount": "0",
546
+ # "orderAmount": "0",
547
+ # "createTime": 1648708186922,
548
+ # "price": "47112.1",
549
+ # "tradeQty": "0",
550
+ # "tradePrice": "0",
551
+ # "tradeId": "0",
552
+ # "ts": 1648708187469
553
+ # }
554
+ #
555
+ timestamp = self.safe_integer(trade, 'tradeTime')
556
+ marketId = self.safe_string(trade, 'symbol')
557
+ return self.safe_trade({
558
+ 'info': trade,
559
+ 'id': self.safe_string(trade, 'tradeId'),
560
+ 'symbol': self.safe_symbol(marketId, market),
561
+ 'timestamp': timestamp,
562
+ 'datetime': self.iso8601(timestamp),
563
+ 'order': self.safe_string(trade, 'orderId'),
564
+ 'type': self.safe_string_lower(trade, 'type'),
565
+ 'side': self.safe_string(trade, 'side'),
566
+ 'takerOrMaker': self.safe_string(trade, 'matchRole'),
567
+ 'price': self.safe_string(trade, 'price'),
568
+ 'amount': self.safe_string(trade, 'tradeAmount'), # ? tradeQty?
569
+ 'cost': None,
570
+ 'fee': {
571
+ 'rate': None,
572
+ 'cost': self.safe_string(trade, 'tradeFee'),
573
+ 'currency': self.safe_string(trade, 'feeCurrency'),
574
+ },
575
+ }, market)
576
+
577
+ def handle_order(self, client: Client, message):
578
+ #
579
+ # update(verbose)
580
+ #
581
+ # {
582
+ # "feed": "open_orders_verbose",
583
+ # "order": {
584
+ # "instrument": "PI_XBTUSD",
585
+ # "time": 1567597581495,
586
+ # "last_update_time": 1567597581495,
587
+ # "qty": 102.0,
588
+ # "filled": 0.0,
589
+ # "limit_price": 10601.0,
590
+ # "stop_price": 0.0,
591
+ # "type": "limit",
592
+ # "order_id": "fa9806c9-cba9-4661-9f31-8c5fd045a95d",
593
+ # "direction": 0,
594
+ # "reduce_only": False
595
+ # },
596
+ # "is_cancel": True,
597
+ # "reason": "post_order_failed_because_it_would_be_filled"
598
+ # }
599
+ #
600
+ # update
601
+ #
602
+ # {
603
+ # "feed": "open_orders",
604
+ # "order": {
605
+ # "instrument": "PI_XBTUSD",
606
+ # "time": 1567702877410,
607
+ # "last_update_time": 1567702877410,
608
+ # "qty": 304.0,
609
+ # "filled": 0.0,
610
+ # "limit_price": 10640.0,
611
+ # "stop_price": 0.0,
612
+ # "type": "limit",
613
+ # "order_id": "59302619-41d2-4f0b-941f-7e7914760ad3",
614
+ # "direction": 1,
615
+ # "reduce_only": True
616
+ # },
617
+ # "is_cancel": False,
618
+ # "reason": "new_placed_order_by_user"
619
+ # }
620
+ # {
621
+ # "feed": "open_orders",
622
+ # "order_id": "ea8a7144-37db-449b-bb4a-b53c814a0f43",
623
+ # "is_cancel": True,
624
+ # "reason": "cancelled_by_user"
625
+ # }
626
+ #
627
+ # {
628
+ # "feed": 'open_orders',
629
+ # "order": {
630
+ # "instrument": 'PF_XBTUSD',
631
+ # "time": 1698159920097,
632
+ # "last_update_time": 1699835622988,
633
+ # "qty": 1.1,
634
+ # "filled": 0,
635
+ # "limit_price": 20000,
636
+ # "stop_price": 0,
637
+ # "type": 'limit',
638
+ # "order_id": '0eaf02b0-855d-4451-a3b7-e2b3070c1fa4',
639
+ # "direction": 0,
640
+ # "reduce_only": False
641
+ # },
642
+ # "is_cancel": False,
643
+ # "reason": 'edited_by_user'
644
+ # }
645
+ #
646
+ orders = self.orders
647
+ if orders is None:
648
+ limit = self.safe_integer(self.options, 'ordersLimit')
649
+ orders = ArrayCacheBySymbolById(limit)
650
+ self.orders = orders
651
+ order = self.safe_value(message, 'order')
652
+ if order is not None:
653
+ marketId = self.safe_string(order, 'instrument')
654
+ messageHash = 'orders'
655
+ symbol = self.safe_symbol(marketId)
656
+ orderId = self.safe_string(order, 'order_id')
657
+ previousOrders = self.safe_value(orders.hashmap, symbol, {})
658
+ previousOrder = self.safe_value(previousOrders, orderId)
659
+ reason = self.safe_string(message, 'reason')
660
+ if (previousOrder is None) or (reason == 'edited_by_user'):
661
+ parsed = self.parse_ws_order(order)
662
+ orders.append(parsed)
663
+ client.resolve(orders, messageHash)
664
+ client.resolve(orders, messageHash + ':' + symbol)
665
+ else:
666
+ trade = self.parse_ws_trade(order)
667
+ if previousOrder['trades'] is None:
668
+ previousOrder['trades'] = []
669
+ previousOrder['trades'].append(trade)
670
+ previousOrder['lastTradeTimestamp'] = trade['timestamp']
671
+ totalCost = '0'
672
+ totalAmount = '0'
673
+ trades = previousOrder['trades']
674
+ for i in range(0, len(trades)):
675
+ currentTrade = trades[i]
676
+ totalCost = Precise.string_add(totalCost, self.number_to_string(currentTrade['cost']))
677
+ totalAmount = Precise.string_add(totalAmount, self.number_to_string(currentTrade['amount']))
678
+ if Precise.string_gt(totalAmount, '0'):
679
+ previousOrder['average'] = Precise.string_div(totalCost, totalAmount)
680
+ previousOrder['cost'] = totalCost
681
+ if previousOrder['filled'] is not None:
682
+ stringOrderFilled = self.number_to_string(previousOrder['filled'])
683
+ previousOrder['filled'] = Precise.string_add(stringOrderFilled, self.number_to_string(trade['amount']))
684
+ if previousOrder['amount'] is not None:
685
+ previousOrder['remaining'] = Precise.string_sub(self.number_to_string(previousOrder['amount']), stringOrderFilled)
686
+ if previousOrder['fee'] is None:
687
+ previousOrder['fee'] = {
688
+ 'rate': None,
689
+ 'cost': '0',
690
+ 'currency': self.number_to_string(trade['fee']['currency']),
691
+ }
692
+ if (previousOrder['fee']['cost'] is not None) and (trade['fee']['cost'] is not None):
693
+ stringOrderCost = self.number_to_string(previousOrder['fee']['cost'])
694
+ stringTradeCost = self.number_to_string(trade['fee']['cost'])
695
+ previousOrder['fee']['cost'] = Precise.string_add(stringOrderCost, stringTradeCost)
696
+ # update the newUpdates count
697
+ orders.append(self.safe_order(previousOrder))
698
+ client.resolve(orders, messageHash + ':' + symbol)
699
+ client.resolve(orders, messageHash)
700
+ else:
701
+ isCancel = self.safe_value(message, 'is_cancel')
702
+ if isCancel:
703
+ # get order without symbol
704
+ for i in range(0, len(orders)):
705
+ currentOrder = orders[i]
706
+ if currentOrder['id'] == message['order_id']:
707
+ orders[i] = self.extend(currentOrder, {
708
+ 'status': 'canceled',
709
+ })
710
+ client.resolve(orders, 'orders')
711
+ client.resolve(orders, 'orders:' + currentOrder['symbol'])
712
+ break
713
+ return message
714
+
715
+ def handle_order_snapshot(self, client: Client, message):
716
+ #
717
+ # verbose
718
+ #
719
+ # {
720
+ # "feed": "open_orders_verbose_snapshot",
721
+ # "account": "0f9c23b8-63e2-40e4-9592-6d5aa57c12ba",
722
+ # "orders": [
723
+ # {
724
+ # "instrument": "PI_XBTUSD",
725
+ # "time": 1567428848005,
726
+ # "last_update_time": 1567428848005,
727
+ # "qty": 100.0,
728
+ # "filled": 0.0,
729
+ # "limit_price": 8500.0,
730
+ # "stop_price": 0.0,
731
+ # "type": "limit",
732
+ # "order_id": "566942c8-a3b5-4184-a451-622b09493129",
733
+ # "direction": 0,
734
+ # "reduce_only": False
735
+ # },
736
+ # ...
737
+ # ]
738
+ # }
739
+ #
740
+ # regular
741
+ #
742
+ # {
743
+ # "feed": "open_orders_snapshot",
744
+ # "account": "e258dba9-4dd4-4da5-bfef-75beb91c098e",
745
+ # "orders": [
746
+ # {
747
+ # "instrument": "PI_XBTUSD",
748
+ # "time": 1612275024153,
749
+ # "last_update_time": 1612275024153,
750
+ # "qty": 1000,
751
+ # "filled": 0,
752
+ # "limit_price": 34900,
753
+ # "stop_price": 13789,
754
+ # "type": "stop",
755
+ # "order_id": "723ba95f-13b7-418b-8fcf-ab7ba6620555",
756
+ # "direction": 1,
757
+ # "reduce_only": False,
758
+ # "triggerSignal": "last"
759
+ # },
760
+ # ...
761
+ # ]
762
+ # }
763
+ orders = self.safe_value(message, 'orders', [])
764
+ limit = self.safe_integer(self.options, 'ordersLimit')
765
+ self.orders = ArrayCacheBySymbolById(limit)
766
+ symbols: dict = {}
767
+ cachedOrders = self.orders
768
+ for i in range(0, len(orders)):
769
+ order = orders[i]
770
+ parsed = self.parse_ws_order(order)
771
+ symbol = parsed['symbol']
772
+ symbols[symbol] = True
773
+ cachedOrders.append(parsed)
774
+ length = len(self.orders)
775
+ if length > 0:
776
+ client.resolve(self.orders, 'orders')
777
+ keys = list(symbols.keys())
778
+ for i in range(0, len(keys)):
779
+ symbol = keys[i]
780
+ messageHash = 'orders:' + symbol
781
+ client.resolve(self.orders, messageHash)
782
+
783
+ def parse_ws_order(self, order, market=None):
784
+ #
785
+ # update
786
+ #
787
+ # {
788
+ # "feed": "open_orders_verbose",
789
+ # "order": {
790
+ # "instrument": "PI_XBTUSD",
791
+ # "time": 1567597581495,
792
+ # "last_update_time": 1567597581495,
793
+ # "qty": 102.0,
794
+ # "filled": 0.0,
795
+ # "limit_price": 10601.0,
796
+ # "stop_price": 0.0,
797
+ # "type": "limit",
798
+ # "order_id": "fa9806c9-cba9-4661-9f31-8c5fd045a95d",
799
+ # "direction": 0,
800
+ # "reduce_only": False
801
+ # },
802
+ # "is_cancel": True,
803
+ # "reason": "post_order_failed_because_it_would_be_filled"
804
+ # }
805
+ #
806
+ # snapshot
807
+ #
808
+ # {
809
+ # "instrument": "PI_XBTUSD",
810
+ # "time": 1567597581495,
811
+ # "last_update_time": 1567597581495,
812
+ # "qty": 102.0,
813
+ # "filled": 0.0,
814
+ # "limit_price": 10601.0,
815
+ # "stop_price": 0.0,
816
+ # "type": "limit",
817
+ # "order_id": "fa9806c9-cba9-4661-9f31-8c5fd045a95d",
818
+ # "direction": 0,
819
+ # "reduce_only": False
820
+ # }
821
+ #
822
+ isCancelled = self.safe_value(order, 'is_cancel')
823
+ unparsedOrder = order
824
+ status = None
825
+ if isCancelled is not None:
826
+ unparsedOrder = self.safe_value(order, 'order')
827
+ if isCancelled is True:
828
+ status = 'cancelled'
829
+ marketId = self.safe_string(unparsedOrder, 'instrument')
830
+ timestamp = self.safe_string(unparsedOrder, 'time')
831
+ direction = self.safe_integer(unparsedOrder, 'direction')
832
+ return self.safe_order({
833
+ 'info': order,
834
+ 'symbol': self.safe_symbol(marketId, market),
835
+ 'id': self.safe_string(unparsedOrder, 'order_id'),
836
+ 'clientOrderId': None,
837
+ 'timestamp': timestamp,
838
+ 'datetime': self.iso8601(timestamp),
839
+ 'lastTradeTimestamp': None,
840
+ 'type': self.safe_string(unparsedOrder, 'type'),
841
+ 'timeInForce': None,
842
+ 'postOnly': None,
843
+ 'side': 'buy' if (direction == 0) else 'sell',
844
+ 'price': self.safe_string(unparsedOrder, 'limit_price'),
845
+ 'stopPrice': self.safe_string(unparsedOrder, 'stop_price'),
846
+ 'triggerPrice': self.safe_string(unparsedOrder, 'stop_price'),
847
+ 'amount': self.safe_string(unparsedOrder, 'qty'),
848
+ 'cost': None,
849
+ 'average': None,
850
+ 'filled': self.safe_string(unparsedOrder, 'filled'),
851
+ 'remaining': None,
852
+ 'status': status,
853
+ 'fee': {
854
+ 'rate': None,
855
+ 'cost': None,
856
+ 'currency': None,
857
+ },
858
+ 'trades': None,
859
+ })
860
+
861
+ def handle_ticker(self, client: Client, message):
862
+ #
863
+ # {
864
+ # "time": 1680811086487,
865
+ # "product_id": "PI_XBTUSD",
866
+ # "funding_rate": 7.792297e-12,
867
+ # "funding_rate_prediction": -4.2671095e-11,
868
+ # "relative_funding_rate": 2.18013888889e-7,
869
+ # "relative_funding_rate_prediction": -0.0000011974,
870
+ # "next_funding_rate_time": 1680811200000,
871
+ # "feed": "ticker",
872
+ # "bid": 28060,
873
+ # "ask": 28070,
874
+ # "bid_size": 2844,
875
+ # "ask_size": 1902,
876
+ # "volume": 19628180,
877
+ # "dtm": 0,
878
+ # "leverage": "50x",
879
+ # "index": 28062.14,
880
+ # "premium": 0,
881
+ # "last": 28053.5,
882
+ # "change": -0.7710945651981715,
883
+ # "suspended": False,
884
+ # "tag": "perpetual",
885
+ # "pair": "XBT:USD",
886
+ # "openInterest": 28875946,
887
+ # "markPrice": 28064.92082724592,
888
+ # "maturityTime": 0,
889
+ # "post_only": False,
890
+ # "volumeQuote": 19628180
891
+ # }
892
+ #
893
+ marketId = self.safe_string(message, 'product_id')
894
+ if marketId is not None:
895
+ ticker = self.parse_ws_ticker(message)
896
+ symbol = ticker['symbol']
897
+ self.tickers[symbol] = ticker
898
+ messageHash = self.get_message_hash('ticker', None, symbol)
899
+ client.resolve(ticker, messageHash)
900
+
901
+ def handle_bid_ask(self, client: Client, message):
902
+ #
903
+ # {
904
+ # "feed": "ticker_lite",
905
+ # "product_id": "FI_ETHUSD_210625",
906
+ # "bid": 1753.45,
907
+ # "ask": 1760.35,
908
+ # "change": 13.448175559936647,
909
+ # "premium": 9.1,
910
+ # "volume": 6899673.0,
911
+ # "tag": "semiannual",
912
+ # "pair": "ETH:USD",
913
+ # "dtm": 141,
914
+ # "maturityTime": 1624633200000,
915
+ # "volumeQuote": 6899673.0
916
+ # }
917
+ #
918
+ marketId = self.safe_string(message, 'product_id')
919
+ if marketId is not None:
920
+ ticker = self.parse_ws_ticker(message)
921
+ symbol = ticker['symbol']
922
+ self.bidsasks[symbol] = ticker
923
+ messageHash = self.get_message_hash('bidask', None, symbol)
924
+ client.resolve(ticker, messageHash)
925
+
926
+ def parse_ws_ticker(self, ticker, market=None):
927
+ #
928
+ # {
929
+ # "time": 1680811086487,
930
+ # "product_id": "PI_XBTUSD",
931
+ # "funding_rate": 7.792297e-12,
932
+ # "funding_rate_prediction": -4.2671095e-11,
933
+ # "relative_funding_rate": 2.18013888889e-7,
934
+ # "relative_funding_rate_prediction": -0.0000011974,
935
+ # "next_funding_rate_time": 1680811200000,
936
+ # "feed": "ticker",
937
+ # "bid": 28060,
938
+ # "ask": 28070,
939
+ # "bid_size": 2844,
940
+ # "ask_size": 1902,
941
+ # "volume": 19628180,
942
+ # "dtm": 0,
943
+ # "leverage": "50x",
944
+ # "index": 28062.14,
945
+ # "premium": 0,
946
+ # "last": 28053.5,
947
+ # "change": -0.7710945651981715,
948
+ # "suspended": False,
949
+ # "tag": "perpetual",
950
+ # "pair": "XBT:USD",
951
+ # "openInterest": 28875946,
952
+ # "markPrice": 28064.92082724592,
953
+ # "maturityTime": 0,
954
+ # "post_only": False,
955
+ # "volumeQuote": 19628180
956
+ # }
957
+ #
958
+ # ticker_lite
959
+ #
960
+ # {
961
+ # "feed": "ticker_lite",
962
+ # "product_id": "FI_ETHUSD_210625",
963
+ # "bid": 1753.45,
964
+ # "ask": 1760.35,
965
+ # "change": 13.448175559936647,
966
+ # "premium": 9.1,
967
+ # "volume": 6899673.0,
968
+ # "tag": "semiannual",
969
+ # "pair": "ETH:USD",
970
+ # "dtm": 141,
971
+ # "maturityTime": 1624633200000,
972
+ # "volumeQuote": 6899673.0
973
+ # }
974
+ #
975
+ marketId = self.safe_string(ticker, 'product_id')
976
+ market = self.safe_market(marketId, market)
977
+ symbol = market['symbol']
978
+ timestamp = self.parse8601(self.safe_string(ticker, 'lastTime'))
979
+ last = self.safe_string(ticker, 'last')
980
+ return self.safe_ticker({
981
+ 'info': ticker,
982
+ 'symbol': symbol,
983
+ 'timestamp': timestamp,
984
+ 'datetime': self.iso8601(timestamp),
985
+ 'high': None,
986
+ 'low': None,
987
+ 'bid': self.safe_string(ticker, 'bid'),
988
+ 'bidVolume': self.safe_string(ticker, 'bid_size'),
989
+ 'ask': self.safe_string(ticker, 'ask'),
990
+ 'askVolume': self.safe_string(ticker, 'ask_size'),
991
+ 'vwap': None,
992
+ 'open': None,
993
+ 'close': last,
994
+ 'last': last,
995
+ 'previousClose': None,
996
+ 'change': self.safe_string(ticker, 'change'),
997
+ 'percentage': None,
998
+ 'average': None,
999
+ 'baseVolume': self.safe_string(ticker, 'volume'),
1000
+ 'quoteVolume': self.safe_string(ticker, 'volumeQuote'),
1001
+ })
1002
+
1003
+ def handle_order_book_snapshot(self, client: Client, message):
1004
+ #
1005
+ # {
1006
+ # "feed": "book_snapshot",
1007
+ # "product_id": "PI_XBTUSD",
1008
+ # "timestamp": 1612269825817,
1009
+ # "seq": 326072249,
1010
+ # "tickSize": null,
1011
+ # "bids": [
1012
+ # {
1013
+ # "price": 34892.5,
1014
+ # "qty": 6385
1015
+ # },
1016
+ # {
1017
+ # "price": 34892,
1018
+ # "qty": 10924
1019
+ # },
1020
+ # ],
1021
+ # "asks": [
1022
+ # {
1023
+ # "price": 34911.5,
1024
+ # "qty": 20598
1025
+ # },
1026
+ # {
1027
+ # "price": 34912,
1028
+ # "qty": 2300
1029
+ # },
1030
+ # ]
1031
+ # }
1032
+ #
1033
+ marketId = self.safe_string(message, 'product_id')
1034
+ market = self.safe_market(marketId)
1035
+ symbol = market['symbol']
1036
+ messageHash = self.get_message_hash('orderbook', None, symbol)
1037
+ subscription = self.safe_dict(client.subscriptions, messageHash, {})
1038
+ limit = self.safe_integer(subscription, 'limit')
1039
+ timestamp = self.safe_integer(message, 'timestamp')
1040
+ self.orderbooks[symbol] = self.order_book({}, limit)
1041
+ orderbook = self.orderbooks[symbol]
1042
+ bids = self.safe_list(message, 'bids')
1043
+ asks = self.safe_list(message, 'asks')
1044
+ for i in range(0, len(bids)):
1045
+ bid = bids[i]
1046
+ price = self.safe_number(bid, 'price')
1047
+ qty = self.safe_number(bid, 'qty')
1048
+ bidsSide = orderbook['bids']
1049
+ bidsSide.store(price, qty)
1050
+ for i in range(0, len(asks)):
1051
+ ask = asks[i]
1052
+ price = self.safe_number(ask, 'price')
1053
+ qty = self.safe_number(ask, 'qty')
1054
+ asksSide = orderbook['asks']
1055
+ asksSide.store(price, qty)
1056
+ orderbook['timestamp'] = timestamp
1057
+ orderbook['datetime'] = self.iso8601(timestamp)
1058
+ orderbook['symbol'] = symbol
1059
+ client.resolve(orderbook, messageHash)
1060
+
1061
+ def handle_order_book(self, client: Client, message):
1062
+ #
1063
+ # {
1064
+ # "feed": "book",
1065
+ # "product_id": "PI_XBTUSD",
1066
+ # "side": "sell",
1067
+ # "seq": 326094134,
1068
+ # "price": 34981,
1069
+ # "qty": 0,
1070
+ # "timestamp": 1612269953629
1071
+ # }
1072
+ #
1073
+ marketId = self.safe_string(message, 'product_id')
1074
+ market = self.safe_market(marketId)
1075
+ symbol = market['symbol']
1076
+ messageHash = self.get_message_hash('orderbook', None, symbol)
1077
+ orderbook = self.orderbooks[symbol]
1078
+ side = self.safe_string(message, 'side')
1079
+ price = self.safe_number(message, 'price')
1080
+ qty = self.safe_number(message, 'qty')
1081
+ timestamp = self.safe_integer(message, 'timestamp')
1082
+ if side == 'sell':
1083
+ asks = orderbook['asks']
1084
+ asks.store(price, qty)
1085
+ else:
1086
+ bids = orderbook['bids']
1087
+ bids.store(price, qty)
1088
+ orderbook['timestamp'] = timestamp
1089
+ orderbook['datetime'] = self.iso8601(timestamp)
1090
+ client.resolve(orderbook, messageHash)
1091
+
1092
+ def handle_balance(self, client: Client, message):
1093
+ #
1094
+ # snapshot
1095
+ #
1096
+ # {
1097
+ # "feed": "balances_snapshot",
1098
+ # "account": "4a012c31-df95-484a-9473-d51e4a0c4ae7",
1099
+ # "holding": {
1100
+ # "USDT": 4997.5012493753,
1101
+ # "XBT": 0.1285407184,
1102
+ # ...
1103
+ # },
1104
+ # "futures": {
1105
+ # "F-ETH:EUR": {
1106
+ # "name": "F-ETH:EUR",
1107
+ # "pair": "ETH/EUR",
1108
+ # "unit": "EUR",
1109
+ # "portfolio_value": 0.0,
1110
+ # "balance": 0.0,
1111
+ # "maintenance_margin": 0.0,
1112
+ # "initial_margin": 0.0,
1113
+ # "available": 0.0,
1114
+ # "unrealized_funding": 0.0,
1115
+ # "pnl": 0.0
1116
+ # },
1117
+ # ...
1118
+ # },
1119
+ # "flex_futures": {
1120
+ # "currencies": {
1121
+ # "USDT": {
1122
+ # "quantity": 0.0,
1123
+ # "value": 0.0,
1124
+ # "collateral_value": 0.0,
1125
+ # "available": 0.0,
1126
+ # "haircut": 0.0,
1127
+ # "conversion_spread": 0.0
1128
+ # },
1129
+ # ...
1130
+ # },
1131
+ # "balance_value":0.0,
1132
+ # "portfolio_value":0.0,
1133
+ # "collateral_value":0.0,
1134
+ # "initial_margin":0.0,
1135
+ # "initial_margin_without_orders":0.0,
1136
+ # "maintenance_margin":0.0,
1137
+ # "pnl":0.0,
1138
+ # "unrealized_funding":0.0,
1139
+ # "total_unrealized":0.0,
1140
+ # "total_unrealized_as_margin":0.0,
1141
+ # "margin_equity":0.0,
1142
+ # "available_margin":0.0
1143
+ # "isolated":{
1144
+ # },
1145
+ # "cross":{
1146
+ # "balance_value":9963.66,
1147
+ # "portfolio_value":9963.66,
1148
+ # "collateral_value":9963.66,
1149
+ # "initial_margin":0.0,
1150
+ # "initial_margin_without_orders":0.0,
1151
+ # "maintenance_margin":0.0,
1152
+ # "pnl":0.0,
1153
+ # "unrealized_funding":0.0,
1154
+ # "total_unrealized":0.0,
1155
+ # "total_unrealized_as_margin":0.0,
1156
+ # "margin_equity":9963.66,
1157
+ # "available_margin":9963.66,
1158
+ # "effective_leverage":0.0
1159
+ # },
1160
+ # },
1161
+ # "timestamp":1640995200000,
1162
+ # "seq":0
1163
+ # }
1164
+ #
1165
+ # update
1166
+ #
1167
+ # Holding Wallet
1168
+ #
1169
+ # {
1170
+ # "feed": "balances",
1171
+ # "account": "7a641082-55c7-4411-a85f-930ec2e09617",
1172
+ # "holding": {
1173
+ # "USD": 5000.0
1174
+ # },
1175
+ # "futures": {},
1176
+ # "timestamp": 1640995200000,
1177
+ # "seq": 83
1178
+ # }
1179
+ #
1180
+ # Multi-Collateral
1181
+ #
1182
+ # {
1183
+ # "feed": "balances"
1184
+ # "account": "7a641082-55c7-4411-a85f-930ec2e09617"
1185
+ # "flex_futures": {
1186
+ # "currencies": {
1187
+ # "USDT": {
1188
+ # "quantity": 0.0,
1189
+ # "value": 0.0,
1190
+ # "collateral_value": 0.0,
1191
+ # "available": 0.0,
1192
+ # "haircut": 0.0,
1193
+ # "conversion_spread": 0.0
1194
+ # },
1195
+ # ...
1196
+ # },
1197
+ # "balance_value": 5000.0,
1198
+ # "portfolio_value": 5000.0,
1199
+ # "collateral_value": 5000.0,
1200
+ # "initial_margin": 0.0,
1201
+ # "initial_margin_without_orders": 0.0,
1202
+ # "maintenance_margin": 0.0,
1203
+ # "pnl": 0.0,
1204
+ # "unrealized_funding": 0.0,
1205
+ # "total_unrealized": 0.0,
1206
+ # "total_unrealized_as_margin": 0.0,
1207
+ # "margin_equity": 5000.0,
1208
+ # "available_margin": 5000.0
1209
+ # },
1210
+ # "timestamp": 1640995200000,
1211
+ # "seq": 1
1212
+ # }
1213
+ #
1214
+ # Sample Single-Collateral Balance Delta
1215
+ #
1216
+ # {
1217
+ # "feed": "balances",
1218
+ # "account": "7a641082-55c7-4411-a85f-930ec2e09617",
1219
+ # "holding": {},
1220
+ # "futures": {
1221
+ # "F-XBT:USD": {
1222
+ # "name": "F-XBT:USD",
1223
+ # "pair": "XBT/USD",
1224
+ # "unit": "XBT",
1225
+ # "portfolio_value": 0.1219368845,
1226
+ # "balance": 0.1219368845,
1227
+ # "maintenance_margin": 0.0,
1228
+ # "initial_margin": 0.0,
1229
+ # "available": 0.1219368845,
1230
+ # "unrealized_funding": 0.0,
1231
+ # "pnl": 0.0
1232
+ # }
1233
+ # },
1234
+ # "timestamp": 1640995200000,
1235
+ # "seq": 2
1236
+ # }
1237
+ #
1238
+ holding = self.safe_value(message, 'holding')
1239
+ futures = self.safe_value(message, 'futures')
1240
+ flexFutures = self.safe_value(message, 'flex_futures')
1241
+ messageHash = 'balances'
1242
+ timestamp = self.safe_integer(message, 'timestamp')
1243
+ if holding is not None:
1244
+ holdingKeys = list(holding.keys()) # cashAccount
1245
+ holdingResult: dict = {
1246
+ 'info': message,
1247
+ 'timestamp': timestamp,
1248
+ 'datetime': self.iso8601(timestamp),
1249
+ }
1250
+ for i in range(0, len(holdingKeys)):
1251
+ key = holdingKeys[i]
1252
+ code = self.safe_currency_code(key)
1253
+ newAccount = self.account()
1254
+ newAccount['total'] = self.safe_string(holding, key)
1255
+ holdingResult[code] = newAccount
1256
+ self.balance['cash'] = holdingResult
1257
+ self.balance['cash'] = self.safe_balance(self.balance['cash'])
1258
+ client.resolve(holdingResult, messageHash)
1259
+ if futures is not None:
1260
+ futuresKeys = list(futures.keys()) # marginAccount
1261
+ futuresResult: dict = {
1262
+ 'info': message,
1263
+ 'timestamp': timestamp,
1264
+ 'datetime': self.iso8601(timestamp),
1265
+ }
1266
+ for i in range(0, len(futuresKeys)):
1267
+ key = futuresKeys[i]
1268
+ symbol = self.safe_symbol(key)
1269
+ newAccount = self.account()
1270
+ future = self.safe_value(futures, key)
1271
+ currencyId = self.safe_string(future, 'unit')
1272
+ code = self.safe_currency_code(currencyId)
1273
+ newAccount['free'] = self.safe_string(future, 'available')
1274
+ newAccount['used'] = self.safe_string(future, 'initial_margin')
1275
+ newAccount['total'] = self.safe_string(future, 'balance')
1276
+ futuresResult[symbol] = {}
1277
+ futuresResult[symbol][code] = newAccount
1278
+ self.balance['margin'] = futuresResult
1279
+ self.balance['margin'] = self.safe_balance(self.balance['margin'])
1280
+ client.resolve(self.balance['margin'], messageHash + 'futures')
1281
+ if flexFutures is not None:
1282
+ flexFutureCurrencies = self.safe_value(flexFutures, 'currencies', {})
1283
+ flexFuturesKeys = list(flexFutureCurrencies.keys()) # multi-collateral margin account
1284
+ flexFuturesResult: dict = {
1285
+ 'info': message,
1286
+ 'timestamp': timestamp,
1287
+ 'datetime': self.iso8601(timestamp),
1288
+ }
1289
+ for i in range(0, len(flexFuturesKeys)):
1290
+ key = flexFuturesKeys[i]
1291
+ flexFuture = self.safe_value(flexFutureCurrencies, key)
1292
+ code = self.safe_currency_code(key)
1293
+ newAccount = self.account()
1294
+ newAccount['free'] = self.safe_string(flexFuture, 'available')
1295
+ newAccount['used'] = self.safe_string(flexFuture, 'collateral_value')
1296
+ newAccount['total'] = self.safe_string(flexFuture, 'quantity')
1297
+ flexFuturesResult[code] = newAccount
1298
+ self.balance['flex'] = flexFuturesResult
1299
+ self.balance['flex'] = self.safe_balance(self.balance['flex'])
1300
+ client.resolve(self.balance['flex'], messageHash + 'flex_futures')
1301
+ client.resolve(self.balance, messageHash)
1302
+
1303
+ def handle_my_trades(self, client: Client, message):
1304
+ #
1305
+ # {
1306
+ # "feed": "fills_snapshot",
1307
+ # "account": "DemoUser",
1308
+ # "fills": [
1309
+ # {
1310
+ # "instrument": "FI_XBTUSD_200925",
1311
+ # "time": 1600256910739,
1312
+ # "price": 10937.5,
1313
+ # "seq": 36,
1314
+ # "buy": True,
1315
+ # "qty": 5000.0,
1316
+ # "order_id": "9e30258b-5a98-4002-968a-5b0e149bcfbf",
1317
+ # "cli_ord_id": "8b58d9da-fcaf-4f60-91bc-9973a3eba48d", # only on update, not on snapshot
1318
+ # "fill_id": "cad76f07-814e-4dc6-8478-7867407b6bff",
1319
+ # "fill_type": "maker",
1320
+ # "fee_paid": -0.00009142857,
1321
+ # "fee_currency": "BTC",
1322
+ # "taker_order_type": "ioc",
1323
+ # "order_type": "limit"
1324
+ # },
1325
+ # ...
1326
+ # ]
1327
+ # }
1328
+ #
1329
+ trades = self.safe_value(message, 'fills', [])
1330
+ stored = self.myTrades
1331
+ if stored is None:
1332
+ limit = self.safe_integer(self.options, 'tradesLimit', 1000)
1333
+ stored = ArrayCacheBySymbolById(limit)
1334
+ self.myTrades = stored
1335
+ tradeSymbols: dict = {}
1336
+ for i in range(0, len(trades)):
1337
+ trade = trades[i]
1338
+ parsedTrade = self.parse_ws_my_trade(trade)
1339
+ tradeSymbols[parsedTrade['symbol']] = True
1340
+ stored.append(parsedTrade)
1341
+ tradeSymbolKeys = list(tradeSymbols.keys())
1342
+ for i in range(0, len(tradeSymbolKeys)):
1343
+ symbol = tradeSymbolKeys[i]
1344
+ messageHash = 'myTrades:' + symbol
1345
+ client.resolve(stored, messageHash)
1346
+ client.resolve(stored, 'myTrades')
1347
+
1348
+ def parse_ws_my_trade(self, trade, market=None):
1349
+ #
1350
+ # {
1351
+ # "instrument": "FI_XBTUSD_200925",
1352
+ # "time": 1600256910739,
1353
+ # "price": 10937.5,
1354
+ # "seq": 36,
1355
+ # "buy": True,
1356
+ # "qty": 5000.0,
1357
+ # "order_id": "9e30258b-5a98-4002-968a-5b0e149bcfbf",
1358
+ # "cli_ord_id": "8b58d9da-fcaf-4f60-91bc-9973a3eba48d", # only on update, not on snapshot
1359
+ # "fill_id": "cad76f07-814e-4dc6-8478-7867407b6bff",
1360
+ # "fill_type": "maker",
1361
+ # "fee_paid": -0.00009142857,
1362
+ # "fee_currency": "BTC",
1363
+ # "taker_order_type": "ioc",
1364
+ # "order_type": "limit"
1365
+ # }
1366
+ #
1367
+ timestamp = self.safe_integer(trade, 'time')
1368
+ marketId = self.safe_string(trade, 'instrument')
1369
+ market = self.safe_market(marketId, market)
1370
+ isBuy = self.safe_value(trade, 'buy')
1371
+ feeCurrencyId = self.safe_string(trade, 'fee_currency')
1372
+ return self.safe_trade({
1373
+ 'info': trade,
1374
+ 'id': self.safe_string(trade, 'fill_id'),
1375
+ 'timestamp': timestamp,
1376
+ 'datetime': self.iso8601(timestamp),
1377
+ 'symbol': self.safe_string(market, 'symbol'),
1378
+ 'order': self.safe_string(trade, 'order_id'),
1379
+ 'type': self.safe_string(trade, 'type'),
1380
+ 'side': 'buy' if isBuy else 'sell',
1381
+ 'takerOrMaker': self.safe_string(trade, 'fill_type'),
1382
+ 'price': self.safe_string(trade, 'price'),
1383
+ 'amount': self.safe_string(trade, 'qty'),
1384
+ 'cost': None,
1385
+ 'fee': {
1386
+ 'currency': self.safe_currency_code(feeCurrencyId),
1387
+ 'cost': self.safe_string(trade, 'fee_paid'),
1388
+ 'rate': None,
1389
+ },
1390
+ })
1391
+
1392
+ async def watch_multi_helper(self, unifiedName: str, channelName: str, symbols: Strings = None, subscriptionArgs=None, params={}):
1393
+ await self.load_markets()
1394
+ # symbols are required
1395
+ symbols = self.market_symbols(symbols, None, False, True, False)
1396
+ messageHashes = []
1397
+ for i in range(0, len(symbols)):
1398
+ messageHashes.append(self.get_message_hash(unifiedName, None, self.symbol(symbols[i])))
1399
+ marketIds = self.market_ids(symbols)
1400
+ request: dict = {
1401
+ 'event': 'subscribe',
1402
+ 'feed': channelName,
1403
+ 'product_ids': marketIds,
1404
+ }
1405
+ url = self.urls['api']['ws']
1406
+ return await self.watch_multiple(url, messageHashes, self.extend(request, params), messageHashes, subscriptionArgs)
1407
+
1408
+ def get_message_hash(self, unifiedElementName: str, subChannelName: Str = None, symbol: Str = None):
1409
+ # unifiedElementName can be : orderbook, trade, ticker, bidask ...
1410
+ # subChannelName only applies to channel that needs specific variation(i.e. depth_50, depth_100..) to be selected
1411
+ withSymbol = symbol is not None
1412
+ messageHash = unifiedElementName
1413
+ if not withSymbol:
1414
+ messageHash += 's'
1415
+ else:
1416
+ messageHash += ':' + symbol
1417
+ if subChannelName is not None:
1418
+ messageHash += '#' + subChannelName
1419
+ return messageHash
1420
+
1421
+ def handle_error_message(self, client: Client, message):
1422
+ #
1423
+ # {
1424
+ # event: 'alert',
1425
+ # message: 'Failed to subscribe to authenticated feed'
1426
+ # }
1427
+ #
1428
+ errMsg = self.safe_string(message, 'message')
1429
+ try:
1430
+ raise ExchangeError(self.id + ' ' + errMsg)
1431
+ except Exception as error:
1432
+ client.reject(error)
1433
+
1434
+ def handle_message(self, client, message):
1435
+ event = self.safe_string(message, 'event')
1436
+ if event == 'challenge':
1437
+ self.handle_authenticate(client, message)
1438
+ elif event == 'alert':
1439
+ self.handle_error_message(client, message)
1440
+ elif event == 'pong':
1441
+ client.lastPong = self.milliseconds()
1442
+ elif event is None:
1443
+ feed = self.safe_string(message, 'feed')
1444
+ methods: dict = {
1445
+ 'ticker': self.handle_ticker,
1446
+ 'ticker_lite': self.handle_bid_ask,
1447
+ 'trade': self.handle_trade,
1448
+ 'trade_snapshot': self.handle_trade,
1449
+ # 'heartbeat': self.handleStatus,
1450
+ 'book': self.handle_order_book,
1451
+ 'book_snapshot': self.handle_order_book_snapshot,
1452
+ 'open_orders_verbose': self.handle_order,
1453
+ 'open_orders_verbose_snapshot': self.handle_order_snapshot,
1454
+ 'fills': self.handle_my_trades,
1455
+ 'fills_snapshot': self.handle_my_trades,
1456
+ 'open_orders': self.handle_order,
1457
+ 'open_orders_snapshot': self.handle_order_snapshot,
1458
+ 'balances': self.handle_balance,
1459
+ 'balances_snapshot': self.handle_balance,
1460
+ 'open_positions': self.handle_positions,
1461
+ }
1462
+ method = self.safe_value(methods, feed)
1463
+ if method is not None:
1464
+ method(client, message)
1465
+
1466
+ def handle_authenticate(self, client: Client, message):
1467
+ """
1468
+ * @ignore
1469
+ :see: https://docs.futures.kraken.com/#websocket-api-websocket-api-introduction-sign-challenge-challenge
1470
+ """
1471
+ #
1472
+ # {
1473
+ # "event": "challenge",
1474
+ # "message": "226aee50-88fc-4618-a42a-34f7709570b2"
1475
+ # }
1476
+ #
1477
+ event = self.safe_value(message, 'event')
1478
+ messageHash = 'challenge'
1479
+ if event != 'error':
1480
+ challenge = self.safe_value(message, 'message')
1481
+ hashedChallenge = self.hash(self.encode(challenge), 'sha256', 'binary')
1482
+ base64Secret = self.base64_to_binary(self.secret)
1483
+ signature = self.hmac(hashedChallenge, base64Secret, hashlib.sha512, 'base64')
1484
+ self.options['challenge'] = challenge
1485
+ self.options['signedChallenge'] = signature
1486
+ client.resolve(message, messageHash)
1487
+ else:
1488
+ error = AuthenticationError(self.id + ' ' + self.json(message))
1489
+ client.reject(error, messageHash)
1490
+ if messageHash in client.subscriptions:
1491
+ del client.subscriptions[messageHash]
1492
+ return message