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
ccxt/pro/onetrading.py ADDED
@@ -0,0 +1,1275 @@
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 ArrayCacheBySymbolById, ArrayCacheByTimestamp
8
+ from ccxt.base.types import Balances, Int, Order, OrderBook, Str, Strings, Ticker, Tickers, Trade
9
+ from ccxt.async_support.base.ws.client import Client
10
+ from typing import List
11
+ from ccxt.base.errors import ExchangeError
12
+ from ccxt.base.errors import NotSupported
13
+
14
+
15
+ class onetrading(ccxt.async_support.onetrading):
16
+
17
+ def describe(self):
18
+ return self.deep_extend(super(onetrading, self).describe(), {
19
+ 'has': {
20
+ 'ws': True,
21
+ 'watchBalance': True,
22
+ 'watchTicker': True,
23
+ 'watchTickers': True,
24
+ 'watchTrades': False,
25
+ 'watchMyTrades': True,
26
+ 'watchOrders': True,
27
+ 'watchOrderBook': True,
28
+ 'watchOHLCV': True,
29
+ },
30
+ 'urls': {
31
+ 'api': {
32
+ 'ws': 'wss://streams.onetrading.com/',
33
+ },
34
+ },
35
+ 'options': {
36
+ 'bp_remaining_quota': 200,
37
+ 'timeframes': {
38
+ '1m': {
39
+ 'unit': 'MINUTES',
40
+ 'period': 1,
41
+ },
42
+ '5m': {
43
+ 'unit': 'MINUTES',
44
+ 'period': 5,
45
+ },
46
+ '15m': {
47
+ 'unit': 'MINUTES',
48
+ 'period': 15,
49
+ },
50
+ '30m': {
51
+ 'unit': 'MINUTES',
52
+ 'period': 30,
53
+ },
54
+ '1h': {
55
+ 'unit': 'HOURS',
56
+ 'period': 1,
57
+ },
58
+ '4h': {
59
+ 'unit': 'HOURS',
60
+ 'period': 4,
61
+ },
62
+ '1d': {
63
+ 'unit': 'DAYS',
64
+ 'period': 1,
65
+ },
66
+ '1w': {
67
+ 'unit': 'WEEKS',
68
+ 'period': 1,
69
+ },
70
+ '1M': {
71
+ 'unit': 'MONTHS',
72
+ 'period': 1,
73
+ },
74
+ },
75
+ },
76
+ 'streaming': {
77
+ },
78
+ 'exceptions': {
79
+ },
80
+ })
81
+
82
+ async def watch_balance(self, params={}) -> Balances:
83
+ """
84
+ :see: https://developers.bitpanda.com/exchange/#account-history-channel
85
+ watch balance and get the amount of funds available for trading or funds locked in orders
86
+ :param dict [params]: extra parameters specific to the exchange API endpoint
87
+ :returns dict: a `balance structure <https://docs.ccxt.com/#/?id=balance-structure>`
88
+ """
89
+ await self.authenticate(params)
90
+ url = self.urls['api']['ws']
91
+ messageHash = 'balance'
92
+ subscribeHash = 'ACCOUNT_HISTORY'
93
+ bpRemainingQuota = self.safe_integer(self.options, 'bp_remaining_quota', 200)
94
+ subscribe: dict = {
95
+ 'type': 'SUBSCRIBE',
96
+ 'bp_remaining_quota': bpRemainingQuota,
97
+ 'channels': [
98
+ {
99
+ 'name': 'ACCOUNT_HISTORY',
100
+ },
101
+ ],
102
+ }
103
+ request = self.deep_extend(subscribe, params)
104
+ return await self.watch(url, messageHash, request, subscribeHash, request)
105
+
106
+ def handle_balance_snapshot(self, client, message):
107
+ #
108
+ # snapshot
109
+ # {
110
+ # "account_id": "b355abb8-aaae-4fae-903c-c60ff74723c6",
111
+ # "type": "BALANCES_SNAPSHOT",
112
+ # "channel_name": "ACCOUNT_HISTORY",
113
+ # "time": "2019-04-01T13:39:17.155Z",
114
+ # "balances": [{
115
+ # "account_id": "b355abb8-aaae-4fae-903c-c60ff74723c6",
116
+ # "currency_code": "BTC",
117
+ # "change": "0.5",
118
+ # "available": "10.0",
119
+ # "locked": "1.1234567",
120
+ # "sequence": 1,
121
+ # "time": "2019-04-01T13:39:17.155Z"
122
+ # },
123
+ # {
124
+ # "account_id": "b355abb8-aaae-4fae-903c-c60ff74723c6",
125
+ # "currency_code": "ETH",
126
+ # "change": "0.5",
127
+ # "available": "10.0",
128
+ # "locked": "1.1234567",
129
+ # "sequence": 2,
130
+ # "time": "2019-04-01T13:39:17.155Z"
131
+ # }
132
+ # ]
133
+ # }
134
+ #
135
+ self.balance = self.parse_balance(message)
136
+ messageHash = 'balance'
137
+ client.resolve(self.balance, messageHash)
138
+
139
+ async def watch_ticker(self, symbol: str, params={}) -> Ticker:
140
+ """
141
+ :see: https://developers.bitpanda.com/exchange/#market-ticker-channel
142
+ watches a price ticker, a statistical calculation with the information calculated over the past 24 hours for a specific market
143
+ :param str symbol: unified symbol of the market to fetch the ticker for
144
+ :param dict [params]: extra parameters specific to the exchange API endpoint
145
+ :returns dict: a `ticker structure <https://docs.ccxt.com/#/?id=ticker-structure>`
146
+ """
147
+ await self.load_markets()
148
+ market = self.market(symbol)
149
+ symbol = market['symbol']
150
+ subscriptionHash = 'MARKET_TICKER'
151
+ messageHash = 'ticker.' + symbol
152
+ request: dict = {
153
+ 'type': 'SUBSCRIBE',
154
+ 'channels': [
155
+ {
156
+ 'name': 'MARKET_TICKER',
157
+ 'price_points_mode': 'INLINE',
158
+ },
159
+ ],
160
+ }
161
+ return await self.watch_many(messageHash, request, subscriptionHash, [symbol], params)
162
+
163
+ async def watch_tickers(self, symbols: Strings = None, params={}) -> Tickers:
164
+ """
165
+ :see: https://developers.bitpanda.com/exchange/#market-ticker-channel
166
+ watches price tickers, a statistical calculation with the information for all markets or those specified.
167
+ :param str symbols: unified symbols of the markets to fetch the ticker for
168
+ :param dict [params]: extra parameters specific to the exchange API endpoint
169
+ :returns dict: an array of `ticker structure <https://docs.ccxt.com/#/?id=ticker-structure>`
170
+ """
171
+ await self.load_markets()
172
+ symbols = self.market_symbols(symbols)
173
+ if symbols is None:
174
+ symbols = []
175
+ subscriptionHash = 'MARKET_TICKER'
176
+ messageHash = 'tickers'
177
+ request: dict = {
178
+ 'type': 'SUBSCRIBE',
179
+ 'channels': [
180
+ {
181
+ 'name': 'MARKET_TICKER',
182
+ 'price_points_mode': 'INLINE',
183
+ },
184
+ ],
185
+ }
186
+ tickers = await self.watch_many(messageHash, request, subscriptionHash, symbols, params)
187
+ return self.filter_by_array(tickers, 'symbol', symbols)
188
+
189
+ def handle_ticker(self, client: Client, message):
190
+ #
191
+ # {
192
+ # "ticker_updates": [{
193
+ # "instrument": "ETH_BTC",
194
+ # "last_price": "0.053752",
195
+ # "price_change": "0.000623",
196
+ # "price_change_percentage": "1.17",
197
+ # "high": "0.055",
198
+ # "low": "0.052662",
199
+ # "volume": "6.3821593247"
200
+ # }],
201
+ # "channel_name": "MARKET_TICKER",
202
+ # "type": "MARKET_TICKER_UPDATES",
203
+ # "time": "2022-06-23T16:41:00.004162Z"
204
+ # }
205
+ #
206
+ tickers = self.safe_value(message, 'ticker_updates', [])
207
+ datetime = self.safe_string(message, 'time')
208
+ for i in range(0, len(tickers)):
209
+ ticker = tickers[i]
210
+ marketId = self.safe_string(ticker, 'instrument')
211
+ symbol = self.safe_symbol(marketId)
212
+ self.tickers[symbol] = self.parse_ws_ticker(ticker)
213
+ timestamp = self.parse8601(datetime)
214
+ self.tickers[symbol]['timestamp'] = timestamp
215
+ self.tickers[symbol]['datetime'] = self.iso8601(timestamp)
216
+ client.resolve(self.tickers[symbol], 'ticker.' + symbol)
217
+ client.resolve(self.tickers, 'tickers')
218
+
219
+ def parse_ws_ticker(self, ticker, market=None):
220
+ #
221
+ # {
222
+ # "instrument": "ETH_BTC",
223
+ # "last_price": "0.053752",
224
+ # "price_change": "-0.000623",
225
+ # "price_change_percentage": "-1.17",
226
+ # "high": "0.055",
227
+ # "low": "0.052662",
228
+ # "volume": "6.3821593247"
229
+ # }
230
+ #
231
+ marketId = self.safe_string(ticker, 'instrument')
232
+ return self.safe_ticker({
233
+ 'symbol': self.safe_symbol(marketId, market),
234
+ 'timestamp': None,
235
+ 'datetime': None,
236
+ 'high': self.safe_string(ticker, 'high'),
237
+ 'low': self.safe_string(ticker, 'low'),
238
+ 'bid': None,
239
+ 'bidVolume': None,
240
+ 'ask': None,
241
+ 'askVolume': None,
242
+ 'vwap': None,
243
+ 'open': None,
244
+ 'close': self.safe_string(ticker, 'last_price'),
245
+ 'last': self.safe_string(ticker, 'last_price'),
246
+ 'previousClose': None,
247
+ 'change': self.safe_string(ticker, 'price_change'),
248
+ 'percentage': self.safe_string(ticker, 'price_change_percentage'),
249
+ 'average': None,
250
+ 'baseVolume': None,
251
+ 'quoteVolume': self.safe_number(ticker, 'volume'),
252
+ 'info': ticker,
253
+ }, market)
254
+
255
+ async def watch_my_trades(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Trade]:
256
+ """
257
+ :see: https://developers.bitpanda.com/exchange/#account-history-channel
258
+ get the list of trades associated with the user
259
+ :param str symbol: unified symbol of the market to fetch trades for. Use 'any' to watch all trades
260
+ :param int [since]: timestamp in ms of the earliest trade to fetch
261
+ :param int [limit]: the maximum amount of trades to fetch
262
+ :param dict [params]: extra parameters specific to the exchange API endpoint
263
+ :returns dict[]: a list of `trade structures <https://docs.ccxt.com/#/?id=public-trades>`
264
+ """
265
+ await self.load_markets()
266
+ messageHash = 'myTrades'
267
+ if symbol is not None:
268
+ market = self.market(symbol)
269
+ symbol = market['symbol']
270
+ messageHash += ':' + symbol
271
+ await self.authenticate(params)
272
+ url = self.urls['api']['ws']
273
+ subscribeHash = 'ACCOUNT_HISTORY'
274
+ bpRemainingQuota = self.safe_integer(self.options, 'bp_remaining_quota', 200)
275
+ subscribe: dict = {
276
+ 'type': 'SUBSCRIBE',
277
+ 'bp_remaining_quota': bpRemainingQuota,
278
+ 'channels': [
279
+ {
280
+ 'name': 'ACCOUNT_HISTORY',
281
+ },
282
+ ],
283
+ }
284
+ request = self.deep_extend(subscribe, params)
285
+ trades = await self.watch(url, messageHash, request, subscribeHash, request)
286
+ if self.newUpdates:
287
+ limit = trades.getLimit(symbol, limit)
288
+ trades = self.filter_by_symbol_since_limit(trades, symbol, since, limit)
289
+ numTrades = len(trades)
290
+ if numTrades == 0:
291
+ return await self.watch_my_trades(symbol, since, limit, params)
292
+ return trades
293
+
294
+ async def watch_order_book(self, symbol: str, limit: Int = None, params={}) -> OrderBook:
295
+ """
296
+ :see: https://developers.bitpanda.com/exchange/#market-ticker-channel
297
+ watches information on open orders with bid(buy) and ask(sell) prices, volumes and other data
298
+ :param str symbol: unified symbol of the market to fetch the order book for
299
+ :param int [limit]: the maximum amount of order book entries to return
300
+ :param dict [params]: extra parameters specific to the exchange API endpoint
301
+ :returns dict: A dictionary of `order book structures <https://docs.ccxt.com/#/?id=order-book-structure>` indexed by market symbols
302
+ """
303
+ await self.load_markets()
304
+ market = self.market(symbol)
305
+ symbol = market['symbol']
306
+ messageHash = 'book:' + symbol
307
+ subscriptionHash = 'ORDER_BOOK'
308
+ depth = 0
309
+ if limit is not None:
310
+ depth = limit
311
+ request: dict = {
312
+ 'type': 'SUBSCRIBE',
313
+ 'channels': [
314
+ {
315
+ 'name': 'ORDER_BOOK',
316
+ 'depth': depth,
317
+ },
318
+ ],
319
+ }
320
+ orderbook = await self.watch_many(messageHash, request, subscriptionHash, [symbol], params)
321
+ return orderbook.limit()
322
+
323
+ def handle_order_book(self, client: Client, message):
324
+ #
325
+ # snapshot
326
+ # {
327
+ # "instrument_code": "ETH_BTC",
328
+ # "bids": [
329
+ # ['0.053595', "4.5352"],
330
+ # ...
331
+ # ],
332
+ # "asks": [
333
+ # ['0.055455', "0.2821"],
334
+ # ...
335
+ # ],
336
+ # "channel_name": "ORDER_BOOK",
337
+ # "type": "ORDER_BOOK_SNAPSHOT",
338
+ # "time": "2022-06-23T15:38:02.196282Z"
339
+ # }
340
+ #
341
+ # update
342
+ # {
343
+ # "instrument_code": "ETH_BTC",
344
+ # "changes": [
345
+ # ["BUY", '0.053593', "8.0587"]
346
+ # ],
347
+ # "channel_name": "ORDER_BOOK",
348
+ # "type": "ORDER_BOOK_UPDATE",
349
+ # "time": "2022-06-23T15:38:02.751301Z"
350
+ # }
351
+ #
352
+ type = self.safe_string(message, 'type')
353
+ marketId = self.safe_string(message, 'instrument_code')
354
+ symbol = self.safe_symbol(marketId)
355
+ dateTime = self.safe_string(message, 'time')
356
+ timestamp = self.parse8601(dateTime)
357
+ channel = 'book:' + symbol
358
+ orderbook = self.safe_value(self.orderbooks, symbol)
359
+ if orderbook is None:
360
+ orderbook = self.order_book({})
361
+ if type == 'ORDER_BOOK_SNAPSHOT':
362
+ snapshot = self.parse_order_book(message, symbol, timestamp, 'bids', 'asks')
363
+ orderbook.reset(snapshot)
364
+ elif type == 'ORDER_BOOK_UPDATE':
365
+ changes = self.safe_value(message, 'changes', [])
366
+ self.handle_deltas(orderbook, changes)
367
+ else:
368
+ raise NotSupported(self.id + ' watchOrderBook() did not recognize message type ' + type)
369
+ orderbook['nonce'] = timestamp
370
+ orderbook['timestamp'] = timestamp
371
+ orderbook['datetime'] = self.iso8601(timestamp)
372
+ self.orderbooks[symbol] = orderbook
373
+ client.resolve(orderbook, channel)
374
+
375
+ def handle_delta(self, orderbook, delta):
376
+ #
377
+ # ['BUY', "0.053595", "0"]
378
+ #
379
+ bidAsk = self.parse_bid_ask(delta, 1, 2)
380
+ type = self.safe_string(delta, 0)
381
+ if type == 'BUY':
382
+ bids = orderbook['bids']
383
+ bids.storeArray(bidAsk)
384
+ elif type == 'SELL':
385
+ asks = orderbook['asks']
386
+ asks.storeArray(bidAsk)
387
+ else:
388
+ raise NotSupported(self.id + ' watchOrderBook() received unknown change type ' + self.json(delta))
389
+
390
+ def handle_deltas(self, orderbook, deltas):
391
+ #
392
+ # [
393
+ # ['BUY', "0.053593", "0"],
394
+ # ['SELL', "0.053698", "0"]
395
+ # ]
396
+ #
397
+ for i in range(0, len(deltas)):
398
+ self.handle_delta(orderbook, deltas[i])
399
+
400
+ async def watch_orders(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Order]:
401
+ """
402
+ :see: https://developers.bitpanda.com/exchange/#account-history-channel
403
+ watches information on multiple orders made by the user
404
+ :param str symbol: unified market symbol of the market orders were made in
405
+ :param int [since]: the earliest time in ms to fetch orders for
406
+ :param int [limit]: the maximum number of order structures to retrieve
407
+ :param dict [params]: extra parameters specific to the exchange API endpoint
408
+ :param str [params.channel]: can listen to orders using ACCOUNT_HISTORY or TRADING
409
+ :returns dict[]: a list of `order structures <https://docs.ccxt.com/#/?id=order-structure>`
410
+ """
411
+ await self.load_markets()
412
+ messageHash = 'orders'
413
+ if symbol is not None:
414
+ market = self.market(symbol)
415
+ symbol = market['symbol']
416
+ messageHash += ':' + symbol
417
+ await self.authenticate(params)
418
+ url = self.urls['api']['ws']
419
+ subscribeHash = self.safe_string(params, 'channel', 'ACCOUNT_HISTORY')
420
+ bpRemainingQuota = self.safe_integer(self.options, 'bp_remaining_quota', 200)
421
+ subscribe: dict = {
422
+ 'type': 'SUBSCRIBE',
423
+ 'bp_remaining_quota': bpRemainingQuota,
424
+ 'channels': [
425
+ {
426
+ 'name': subscribeHash,
427
+ },
428
+ ],
429
+ }
430
+ request = self.deep_extend(subscribe, params)
431
+ orders = await self.watch(url, messageHash, request, subscribeHash, request)
432
+ if self.newUpdates:
433
+ limit = orders.getLimit(symbol, limit)
434
+ orders = self.filter_by_symbol_since_limit(orders, symbol, since, limit)
435
+ numOrders = len(orders)
436
+ if numOrders == 0:
437
+ return await self.watch_orders(symbol, since, limit, params)
438
+ return orders
439
+
440
+ def handle_trading(self, client: Client, message):
441
+ #
442
+ # {
443
+ # "order_book_sequence": 892925263,
444
+ # "side": "BUY",
445
+ # "amount": "0.00046",
446
+ # "trade_id": "d67b9b69-ab76-480f-9ba3-b33582202836",
447
+ # "matched_as": "TAKER",
448
+ # "matched_amount": "0.00046",
449
+ # "matched_price": "22231.08",
450
+ # "instrument_code": "BTC_EUR",
451
+ # "order_id": "7b39f316-0a71-4bfd-adda-3062e6f0bd37",
452
+ # "remaining": "0.0",
453
+ # "channel_name": "TRADING",
454
+ # "type": "FILL",
455
+ # "time": "2022-07-21T12:41:22.883341Z"
456
+ # }
457
+ #
458
+ # {
459
+ # "status": "CANCELLED",
460
+ # "order_book_sequence": 892928424,
461
+ # "amount": "0.0003",
462
+ # "side": "SELL",
463
+ # "price": "50338.65",
464
+ # "instrument_code": "BTC_EUR",
465
+ # "order_id": "b3994a08-a9e8-4a79-a08b-33e3480382df",
466
+ # "remaining": "0.0003",
467
+ # "channel_name": "TRADING",
468
+ # "type": "DONE",
469
+ # "time": "2022-07-21T12:44:24.267000Z"
470
+ # }
471
+ #
472
+ # {
473
+ # "order_book_sequence": 892934476,
474
+ # "side": "SELL",
475
+ # "amount": "0.00051",
476
+ # "price": "22349.02",
477
+ # "instrument_code": "BTC_EUR",
478
+ # "order_id": "1c6c585c-ec3d-4b94-9292-6c3d04a31dc8",
479
+ # "remaining": "0.00051",
480
+ # "channel_name": "TRADING",
481
+ # "type": "BOOKED",
482
+ # "time": "2022-07-21T12:50:10.093000Z"
483
+ # }
484
+ #
485
+ if self.orders is None:
486
+ limit = self.safe_integer(self.options, 'ordersLimit', 1000)
487
+ self.orders = ArrayCacheBySymbolById(limit)
488
+ order = self.parse_trading_order(message)
489
+ orders = self.orders
490
+ orders.append(order)
491
+ client.resolve(self.orders, 'orders:' + order['symbol'])
492
+ client.resolve(self.orders, 'orders')
493
+
494
+ def parse_trading_order(self, order, market=None):
495
+ #
496
+ # {
497
+ # "order_book_sequence": 892925263,
498
+ # "side": "BUY",
499
+ # "amount": "0.00046",
500
+ # "trade_id": "d67b9b69-ab76-480f-9ba3-b33582202836",
501
+ # "matched_as": "TAKER",
502
+ # "matched_amount": "0.00046",
503
+ # "matched_price": "22231.08",
504
+ # "instrument_code": "BTC_EUR",
505
+ # "order_id": "7b39f316-0a71-4bfd-adda-3062e6f0bd37",
506
+ # "remaining": "0.0",
507
+ # "channel_name": "TRADING",
508
+ # "type": "FILL",
509
+ # "time": "2022-07-21T12:41:22.883341Z"
510
+ # }
511
+ #
512
+ # {
513
+ # "status": "CANCELLED",
514
+ # "order_book_sequence": 892928424,
515
+ # "amount": "0.0003",
516
+ # "side": "SELL",
517
+ # "price": "50338.65",
518
+ # "instrument_code": "BTC_EUR",
519
+ # "order_id": "b3994a08-a9e8-4a79-a08b-33e3480382df",
520
+ # "remaining": "0.0003",
521
+ # "channel_name": "TRADING",
522
+ # "type": "DONE",
523
+ # "time": "2022-07-21T12:44:24.267000Z"
524
+ # }
525
+ #
526
+ # {
527
+ # "order_book_sequence": 892934476,
528
+ # "side": "SELL",
529
+ # "amount": "0.00051",
530
+ # "price": "22349.02",
531
+ # "instrument_code": "BTC_EUR",
532
+ # "order_id": "1c6c585c-ec3d-4b94-9292-6c3d04a31dc8",
533
+ # "remaining": "0.00051",
534
+ # "channel_name": "TRADING",
535
+ # "type": "BOOKED",
536
+ # "time": "2022-07-21T12:50:10.093000Z"
537
+ # }
538
+ #
539
+ # {
540
+ # "type":"UPDATE",
541
+ # "channel_name": "TRADING",
542
+ # "instrument_code": "BTC_EUR",
543
+ # "order_id": "1e842f13-762a-4745-9f3b-07f1b43e7058",
544
+ # "client_id": "d75fb03b-b599-49e9-b926-3f0b6d103206",
545
+ # "time": "2020-01-11T01:01:01.999Z",
546
+ # "remaining": "1.23456",
547
+ # "order_book_sequence": 42,
548
+ # "status": "APPLIED",
549
+ # "amount": "1.35756",
550
+ # "amount_delta": "0.123",
551
+ # "modification_id": "cc0eed67-aecc-4fb4-a625-ff3890ceb4cc"
552
+ # }
553
+ # tracked
554
+ # {
555
+ # "type": "STOP_TRACKED",
556
+ # "channel_name": "TRADING",
557
+ # "instrument_code": "BTC_EUR",
558
+ # "order_id": "1e842f13-762a-4745-9f3b-07f1b43e7058",
559
+ # "client_id": "d75fb03b-b599-49e9-b926-3f0b6d103206",
560
+ # "time": "2020-01-11T01:01:01.999Z",
561
+ # "remaining": "1.23456",
562
+ # "order_book_sequence": 42,
563
+ # "trigger_price": "12345.67",
564
+ # "current_price": "11111.11"
565
+ # }
566
+ #
567
+ # {
568
+ # "type": "STOP_TRIGGERED",
569
+ # "channel_name": "TRADING",
570
+ # "instrument_code": "BTC_EUR",
571
+ # "order_id": "1e842f13-762a-4745-9f3b-07f1b43e7058",
572
+ # "client_id": "d75fb03b-b599-49e9-b926-3f0b6d103206",
573
+ # "time": "2020-01-11T01:01:01.999Z",
574
+ # "remaining": "1.23456",
575
+ # "order_book_sequence": 42,
576
+ # "price": "13333.33"
577
+ # }
578
+ #
579
+ datetime = self.safe_string(order, 'time')
580
+ marketId = self.safe_string(order, 'instrument_code')
581
+ symbol = self.safe_symbol(marketId, market, '_')
582
+ return self.safe_order({
583
+ 'id': self.safe_string(order, 'order_id'),
584
+ 'clientOrderId': self.safe_string(order, 'client_id'),
585
+ 'info': order,
586
+ 'timestamp': self.parse8601(datetime),
587
+ 'datetime': datetime,
588
+ 'lastTradeTimestamp': None,
589
+ 'symbol': symbol,
590
+ 'type': None,
591
+ 'timeInForce': None,
592
+ 'postOnly': None,
593
+ 'side': self.safe_string_lower(order, 'side'),
594
+ 'price': self.safe_number_2(order, 'price', 'matched_price'),
595
+ 'stopPrice': self.safe_number(order, 'trigger_price'),
596
+ 'amount': self.safe_number(order, 'amount'),
597
+ 'cost': None,
598
+ 'average': None,
599
+ 'filled': None,
600
+ 'remaining': self.safe_string(order, 'remaining'),
601
+ 'status': self.parse_trading_order_status(self.safe_string(order, 'status')),
602
+ 'fee': None,
603
+ 'trades': None,
604
+ }, market)
605
+
606
+ def parse_trading_order_status(self, status):
607
+ statuses: dict = {
608
+ 'CANCELLED': 'canceled',
609
+ 'SELF_TRADE': 'rejected',
610
+ 'FILLED_FULLY': 'closed',
611
+ 'INSUFFICIENT_FUNDS': 'rejected',
612
+ 'INSUFFICIENT_LIQUIDITY': 'rejected',
613
+ 'TIME_TO_MARKET_EXCEEDED': 'rejected',
614
+ 'LAST_PRICE_UNKNOWN': 'rejected',
615
+ }
616
+ return self.safe_string(statuses, status, status)
617
+
618
+ def handle_orders(self, client: Client, message):
619
+ #
620
+ # snapshot
621
+ # {
622
+ # "account_id": "4920221a-48dc-423e-b336-bb65baccc7bd",
623
+ # "orders": [{
624
+ # "order": {
625
+ # "order_id": "30e2de8f-9a34-472f-bcf8-3af4b7757626",
626
+ # "account_holder": "49202c1a-48dc-423e-b336-bb65baccc7bd",
627
+ # "account_id": "49202c1a-48dc-423e-b336-bb65baccc7bd",
628
+ # "instrument_code": "BTC_EUR",
629
+ # "time": "2022-06-28T06:10:02.587345Z",
630
+ # "side": "SELL",
631
+ # "price": "19645.48",
632
+ # "amount": "0.00052",
633
+ # "filled_amount": "0.00052",
634
+ # "type": "MARKET",
635
+ # "sequence": 7633339971,
636
+ # "status": "FILLED_FULLY",
637
+ # "average_price": "19645.48",
638
+ # "is_post_only": False,
639
+ # "order_book_sequence": 866885897,
640
+ # "time_last_updated": "2022-06-28T06:10:02.766983Z",
641
+ # "update_modification_sequence": 866885897
642
+ # },
643
+ # "trades": [{
644
+ # "fee": {
645
+ # "fee_amount": "0.01532347",
646
+ # "fee_currency": "EUR",
647
+ # "fee_percentage": "0.15",
648
+ # "fee_group_id": "default",
649
+ # "fee_type": "TAKER",
650
+ # "running_trading_volume": "0.0",
651
+ # "collection_type": "STANDARD"
652
+ # },
653
+ # "trade": {
654
+ # "trade_id": "d83e302e-0b3a-4269-aa7d-ecf007cbe577",
655
+ # "order_id": "30e2de8f-9a34-472f-bcf8-3af4b7757626",
656
+ # "account_holder": "49203c1a-48dc-423e-b336-bb65baccc7bd",
657
+ # "account_id": "4920221a-48dc-423e-b336-bb65baccc7bd",
658
+ # "amount": "0.00052",
659
+ # "side": "SELL",
660
+ # "instrument_code": "BTC_EUR",
661
+ # "price": "19645.48",
662
+ # "time": "2022-06-28T06:10:02.693246Z",
663
+ # "price_tick_sequence": 0,
664
+ # "sequence": 7633339971
665
+ # }
666
+ # }]
667
+ # }],
668
+ # "channel_name": "ACCOUNT_HISTORY",
669
+ # "type": "INACTIVE_ORDERS_SNAPSHOT",
670
+ # "time": "2022-06-28T06:11:52.469242Z"
671
+ # }
672
+ #
673
+ #
674
+ if self.orders is None:
675
+ limit = self.safe_integer(self.options, 'ordersLimit', 1000)
676
+ self.orders = ArrayCacheBySymbolById(limit)
677
+ if self.myTrades is None:
678
+ limit = self.safe_integer(self.options, 'tradesLimit', 1000)
679
+ self.myTrades = ArrayCacheBySymbolById(limit)
680
+ rawOrders = self.safe_value(message, 'orders', [])
681
+ rawOrdersLength = len(rawOrders)
682
+ if rawOrdersLength == 0:
683
+ return
684
+ orders = self.orders
685
+ for i in range(0, len(rawOrders)):
686
+ order = self.parse_order(rawOrders[i])
687
+ symbol = self.safe_string(order, 'symbol', '')
688
+ orders.append(order)
689
+ client.resolve(self.orders, 'orders:' + symbol)
690
+ rawTrades = self.safe_value(rawOrders[i], 'trades', [])
691
+ for ii in range(0, len(rawTrades)):
692
+ trade = self.parse_trade(rawTrades[ii])
693
+ symbol = self.safe_string(trade, 'symbol', symbol)
694
+ self.myTrades.append(trade)
695
+ client.resolve(self.myTrades, 'myTrades:' + symbol)
696
+ client.resolve(self.orders, 'orders')
697
+ client.resolve(self.myTrades, 'myTrades')
698
+
699
+ def handle_account_update(self, client: Client, message):
700
+ #
701
+ # order created
702
+ # {
703
+ # "account_id": "49302c1a-48dc-423e-b336-bb65baccc7bd",
704
+ # "sequence": 7658332018,
705
+ # "update": {
706
+ # "type": "ORDER_CREATED",
707
+ # "activity": "TRADING",
708
+ # "account_holder": "43202c1a-48dc-423e-b336-bb65baccc7bd",
709
+ # "account_id": "49202c1a-48dc-423e-b336-bb65baccc7bd",
710
+ # "order_id": "8893fd69-5ebd-496b-aaa4-269b4c18aa77",
711
+ # "time": "2022-06-29T04:33:29.661257Z",
712
+ # "order": {
713
+ # "time_in_force": "GOOD_TILL_CANCELLED",
714
+ # "is_post_only": False,
715
+ # "order_id": "8892fd69-5ebd-496b-aaa4-269b4c18aa77",
716
+ # "account_holder": "43202c1a-48dc-423e-b336-bb65baccc7bd",
717
+ # "account_id": "49302c1a-48dc-423e-b336-bb65baccc7bd",
718
+ # "instrument_code": "BTC_EUR",
719
+ # "time": "2022-06-29T04:33:29.656896Z",
720
+ # "side": "SELL",
721
+ # "price": "50338.65",
722
+ # "amount": "0.00021",
723
+ # "filled_amount": "0.0",
724
+ # "type": "LIMIT"
725
+ # },
726
+ # "locked": {
727
+ # "currency_code": "BTC",
728
+ # "amount": "0.00021",
729
+ # "new_available": "0.00017",
730
+ # "new_locked": "0.00021"
731
+ # },
732
+ # "id": "26e9c36a-b231-4bb0-a686-aa915a2fc9e6",
733
+ # "sequence": 7658332018
734
+ # },
735
+ # "channel_name": "ACCOUNT_HISTORY",
736
+ # "type": "ACCOUNT_UPDATE",
737
+ # "time": "2022-06-29T04:33:29.684517Z"
738
+ # }
739
+ #
740
+ # order rejected
741
+ # {
742
+ # "account_id": "49302c1a-48dc-423e-b336-bb65baccc7bd",
743
+ # "sequence": 7658332018,
744
+ # "update": {
745
+ # "id": "d3fe6025-5b27-4df6-a957-98b8d131cb9d",
746
+ # "type": "ORDER_REJECTED",
747
+ # "activity": "TRADING",
748
+ # "account_id": "b355abb8-aaae-4fae-903c-c60ff74723c6",
749
+ # "sequence": 0,
750
+ # "timestamp": "2018-08-01T13:39:15.590Z",
751
+ # "reason": "INSUFFICIENT_FUNDS",
752
+ # "order_id": "6f991342-da2c-45c6-8830-8bf519cfc8cc",
753
+ # "client_id": "fb497387-8223-4111-87dc-66a86f98a7cf",
754
+ # "unlocked": {
755
+ # "currency_code": "BTC",
756
+ # "amount": "1.5",
757
+ # "new_locked": "2.0",
758
+ # "new_available": "1.5"
759
+ # }
760
+ # }
761
+ # }
762
+ #
763
+ # order closed
764
+ # {
765
+ # "account_id": "49202c1a-48dc-423e-b336-bb65baccc7bd",
766
+ # "sequence": 7658471216,
767
+ # "update": {
768
+ # "type": "ORDER_CLOSED",
769
+ # "activity": "TRADING",
770
+ # "account_holder": "49202c1a-48dc-423e-b336-bb65baccc7bd",
771
+ # "account_id": "49202c1a-48dc-423e-b336-bb65baccc7bd",
772
+ # "time": "2022-06-29T04:43:57.169616Z",
773
+ # "order_id": "8892fd69-5ebd-496b-aaa4-269b4c18aa77",
774
+ # "unlocked": {
775
+ # "currency_code": "BTC",
776
+ # "amount": "0.00021",
777
+ # "new_available": "0.00038",
778
+ # "new_locked": "0.0"
779
+ # },
780
+ # "order_book_sequence": 867964191,
781
+ # "id": "26c5e1d7-65ba-4a11-a661-14c0130ff484",
782
+ # "sequence": 7658471216
783
+ # },
784
+ # "channel_name": "ACCOUNT_HISTORY",
785
+ # "type": "ACCOUNT_UPDATE",
786
+ # "time": "2022-06-29T04:43:57.182153Z"
787
+ # }
788
+ #
789
+ # trade settled
790
+ # {
791
+ # "account_id": "49202c1a-48dc-423e-b336-bb65baccc7bd",
792
+ # "sequence": 7658502878,
793
+ # "update": {
794
+ # "type": "TRADE_SETTLED",
795
+ # "activity": "TRADING",
796
+ # "account_holder": "49202c1a-48dc-423e-b336-bb65baccc7bd",
797
+ # "account_id": "49202c1a-48dc-423e-b336-bb65baccc7bd",
798
+ # "time": "2022-06-29T04:46:12.933091Z",
799
+ # "order_id": "ad19951a-b616-401d-a062-8d0609f038a4",
800
+ # "order_book_sequence": 867965579,
801
+ # "filled_amount": "0.00052",
802
+ # "order": {
803
+ # "amount": "0.00052",
804
+ # "filled_amount": "0.00052"
805
+ # },
806
+ # "trade": {
807
+ # "trade_id": "21039eb9-2df0-4227-be2d-0ea9b691ac66",
808
+ # "order_id": "ad19951a-b616-401d-a062-8d0609f038a4",
809
+ # "account_holder": "49202c1a-48dc-423e-b336-bb65baccc7bd",
810
+ # "account_id": "49202c1a-48dc-423e-b336-bb65baccc7bd",
811
+ # "amount": "0.00052",
812
+ # "side": "BUY",
813
+ # "instrument_code": "BTC_EUR",
814
+ # "price": "19309.29",
815
+ # "time": "2022-06-29T04:46:12.870581Z",
816
+ # "price_tick_sequence": 0
817
+ # },
818
+ # "fee": {
819
+ # "fee_amount": "0.00000078",
820
+ # "fee_currency": "BTC",
821
+ # "fee_percentage": "0.15",
822
+ # "fee_group_id": "default",
823
+ # "fee_type": "TAKER",
824
+ # "running_trading_volume": "0.00052",
825
+ # "collection_type": "STANDARD"
826
+ # },
827
+ # "spent": {
828
+ # "currency_code": "EUR",
829
+ # "amount": "10.0408308",
830
+ # "new_available": "0.0",
831
+ # "new_locked": "0.15949533"
832
+ # },
833
+ # "credited": {
834
+ # "currency_code": "BTC",
835
+ # "amount": "0.00051922",
836
+ # "new_available": "0.00089922",
837
+ # "new_locked": "0.0"
838
+ # },
839
+ # "unlocked": {
840
+ # "currency_code": "EUR",
841
+ # "amount": "0.0",
842
+ # "new_available": "0.0",
843
+ # "new_locked": "0.15949533"
844
+ # },
845
+ # "id": "22b40199-2508-4176-8a14-d4785c933444",
846
+ # "sequence": 7658502878
847
+ # },
848
+ # "channel_name": "ACCOUNT_HISTORY",
849
+ # "type": "ACCOUNT_UPDATE",
850
+ # "time": "2022-06-29T04:46:12.941837Z"
851
+ # }
852
+ #
853
+ # Trade Settled with BEST fee collection enabled
854
+ # {
855
+ # "account_id": "49302c1a-48dc-423e-b336-bb65baccc7bd",
856
+ # "sequence": 7658951984,
857
+ # "update": {
858
+ # "id": "70e00504-d892-456f-9aae-4da7acb36aac",
859
+ # "sequence": 361792,
860
+ # "order_book_sequence": 123456,
861
+ # "type": "TRADE_SETTLED",
862
+ # "activity": "TRADING",
863
+ # "account_id": "379a12c0-4560-11e9-82fe-2b25c6f7d123",
864
+ # "time": "2019-10-22T12:09:55.731Z",
865
+ # "order_id": "9fcdd91c-7f6e-45f4-9956-61cddba55de5",
866
+ # "client_id": "fb497387-8223-4111-87dc-66a86f98a7cf",
867
+ # "order": {
868
+ # "amount": "0.5",
869
+ # "filled_amount": "0.5"
870
+ # },
871
+ # "trade": {
872
+ # "trade_id": "a828b63e-b2cb-48f0-8d99-8fc22cf98e08",
873
+ # "order_id": "9fcdd91c-7f6e-45f4-9956-61cddba55de5",
874
+ # "account_id": "379a12c0-4560-11e9-82fe-2b25c6f7d123",
875
+ # "amount": "0.5",
876
+ # "side": "BUY",
877
+ # "instrument_code": "BTC_EUR",
878
+ # "price": "7451.6",
879
+ # "time": "2019-10-22T12:09:55.667Z"
880
+ # },
881
+ # "fee": {
882
+ # "fee_amount": "23.28625",
883
+ # "fee_currency": "BEST",
884
+ # "fee_percentage": "0.075",
885
+ # "fee_group_id": "default",
886
+ # "fee_type": "TAKER",
887
+ # "running_trading_volume": "0.10058",
888
+ # "collection_type": "BEST",
889
+ # "applied_best_eur_rate": "1.04402"
890
+ # },
891
+ # "spent": {
892
+ # "currency_code": "EUR",
893
+ # "amount": "3725.8",
894
+ # "new_available": "14517885.0675703028781",
895
+ # "new_locked": "2354.882"
896
+ # },
897
+ # "spent_on_fees": {
898
+ # "currency_code": "BEST",
899
+ # "amount": "23.28625",
900
+ # "new_available": "9157.31375",
901
+ # "new_locked": "0.0"
902
+ # },
903
+ # "credited": {
904
+ # "currency_code": "BTC",
905
+ # "amount": "0.5",
906
+ # "new_available": "5839.89633700481",
907
+ # "new_locked": "0.0"
908
+ # },
909
+ # "unlocked": {
910
+ # "currency_code": "EUR",
911
+ # "amount": "0.15",
912
+ # "new_available": "14517885.0675703028781",
913
+ # "new_locked": "2354.882"
914
+ # }
915
+ # }
916
+ # "channel_name": "ACCOUNT_HISTORY",
917
+ # "type": "ACCOUNT_UPDATE",
918
+ # "time": "2022-06-29T05:18:51.760338Z"
919
+ # }
920
+ #
921
+ if self.orders is None:
922
+ limit = self.safe_integer(self.options, 'ordersLimit', 1000)
923
+ self.orders = ArrayCacheBySymbolById(limit)
924
+ if self.myTrades is None:
925
+ limit = self.safe_integer(self.options, 'tradesLimit', 1000)
926
+ self.myTrades = ArrayCacheBySymbolById(limit)
927
+ symbol = None
928
+ orders = self.orders
929
+ update = self.safe_value(message, 'update', {})
930
+ updateType = self.safe_string(update, 'type')
931
+ if updateType == 'ORDER_REJECTED' or updateType == 'ORDER_CLOSED' or updateType == 'STOP_ORDER_TRIGGERED':
932
+ orderId = self.safe_string(update, 'order_id')
933
+ datetime = self.safe_string_2(update, 'time', 'timestamp')
934
+ previousOrderArray = self.filter_by_array(self.orders, 'id', orderId, False)
935
+ previousOrder = self.safe_value(previousOrderArray, 0, {})
936
+ symbol = previousOrder['symbol']
937
+ filled = self.safe_number(update, 'filled_amount')
938
+ status = self.parse_ws_order_status(updateType)
939
+ if updateType == 'ORDER_CLOSED' and filled == 0:
940
+ status = 'canceled'
941
+ orderObject: dict = {
942
+ 'id': orderId,
943
+ 'symbol': symbol,
944
+ 'status': status,
945
+ 'timestamp': self.parse8601(datetime),
946
+ 'datetime': datetime,
947
+ }
948
+ orders.append(orderObject)
949
+ else:
950
+ parsed = self.parse_order(update)
951
+ symbol = self.safe_string(parsed, 'symbol', '')
952
+ orders.append(parsed)
953
+ client.resolve(self.orders, 'orders:' + symbol)
954
+ client.resolve(self.orders, 'orders')
955
+ # update balance
956
+ balanceKeys = ['locked', 'unlocked', 'spent', 'spent_on_fees', 'credited', 'deducted']
957
+ for i in range(0, len(balanceKeys)):
958
+ newBalance = self.safe_value(update, balanceKeys[i])
959
+ if newBalance is not None:
960
+ self.update_balance(newBalance)
961
+ client.resolve(self.balance, 'balance')
962
+ # update trades
963
+ if updateType == 'TRADE_SETTLED':
964
+ parsed = self.parse_trade(update)
965
+ symbol = self.safe_string(parsed, 'symbol', '')
966
+ self.myTrades.append(parsed)
967
+ client.resolve(self.myTrades, 'myTrades:' + symbol)
968
+ client.resolve(self.myTrades, 'myTrades')
969
+
970
+ def parse_ws_order_status(self, status):
971
+ statuses: dict = {
972
+ 'ORDER_REJECTED': 'rejected',
973
+ 'ORDER_CLOSED': 'closed',
974
+ 'STOP_ORDER_TRIGGERED': 'triggered',
975
+ }
976
+ return self.safe_string(statuses, status, status)
977
+
978
+ def update_balance(self, balance):
979
+ #
980
+ # {
981
+ # "currency_code": "EUR",
982
+ # "amount": "0.0",
983
+ # "new_available": "0.0",
984
+ # "new_locked": "0.15949533"
985
+ # }
986
+ #
987
+ currencyId = self.safe_string(balance, 'currency_code')
988
+ code = self.safe_currency_code(currencyId)
989
+ account = self.account()
990
+ account['free'] = self.safe_string(balance, 'new_available')
991
+ account['used'] = self.safe_string(balance, 'new_locked')
992
+ self.balance[code] = account
993
+ self.balance = self.safe_balance(self.balance)
994
+
995
+ async def watch_ohlcv(self, symbol: str, timeframe='1m', since: Int = None, limit: Int = None, params={}) -> List[list]:
996
+ """
997
+ :see: https://developers.bitpanda.com/exchange/#candlesticks-channel
998
+ watches historical candlestick data containing the open, high, low, and close price, and the volume of a market
999
+ :param str symbol: unified symbol of the market to fetch OHLCV data for
1000
+ :param str timeframe: the length of time each candle represents
1001
+ :param int [since]: timestamp in ms of the earliest candle to fetch
1002
+ :param int [limit]: the maximum amount of candles to fetch
1003
+ :param dict [params]: extra parameters specific to the exchange API endpoint
1004
+ :returns int[][]: A list of candles ordered, open, high, low, close, volume
1005
+ """
1006
+ await self.load_markets()
1007
+ market = self.market(symbol)
1008
+ symbol = market['symbol']
1009
+ marketId = market['id']
1010
+ url = self.urls['api']['ws']
1011
+ timeframes = self.safe_value(self.options, 'timeframes', {})
1012
+ timeframeId = self.safe_value(timeframes, timeframe)
1013
+ if timeframeId is None:
1014
+ raise NotSupported(self.id + ' self interval is not supported, please provide one of the supported timeframes')
1015
+ messageHash = 'ohlcv.' + symbol + '.' + timeframe
1016
+ subscriptionHash = 'CANDLESTICKS'
1017
+ client = self.safe_value(self.clients, url)
1018
+ type = 'SUBSCRIBE'
1019
+ subscription = {}
1020
+ if client is not None:
1021
+ subscription = self.safe_value(client.subscriptions, subscriptionHash)
1022
+ if subscription is not None:
1023
+ ohlcvMarket = self.safe_value(subscription, marketId, {})
1024
+ marketSubscribed = self.safe_bool(ohlcvMarket, timeframe, False)
1025
+ if not marketSubscribed:
1026
+ type = 'UPDATE_SUBSCRIPTION'
1027
+ client.subscriptions[subscriptionHash] = None
1028
+ else:
1029
+ subscription = {}
1030
+ subscriptionMarketId = self.safe_value(subscription, marketId)
1031
+ if subscriptionMarketId is None:
1032
+ subscription[marketId] = {}
1033
+ subscription[marketId][timeframe] = True
1034
+ properties = []
1035
+ marketIds = list(subscription.keys())
1036
+ for i in range(0, len(marketIds)):
1037
+ marketIdtimeframes = list(subscription[marketIds[i]].keys())
1038
+ for ii in range(0, len(marketIdtimeframes)):
1039
+ marketTimeframeId = self.safe_value(timeframes, timeframe)
1040
+ property: dict = {
1041
+ 'instrument_code': marketIds[i],
1042
+ 'time_granularity': marketTimeframeId,
1043
+ }
1044
+ properties.append(property)
1045
+ request: dict = {
1046
+ 'type': type,
1047
+ 'channels': [
1048
+ {
1049
+ 'name': 'CANDLESTICKS',
1050
+ 'properties': properties,
1051
+ },
1052
+ ],
1053
+ }
1054
+ ohlcv = await self.watch(url, messageHash, self.deep_extend(request, params), subscriptionHash, subscription)
1055
+ if self.newUpdates:
1056
+ limit = ohlcv.getLimit(symbol, limit)
1057
+ return self.filter_by_since_limit(ohlcv, since, limit, 0, True)
1058
+
1059
+ def handle_ohlcv(self, client: Client, message):
1060
+ #
1061
+ # snapshot
1062
+ # {
1063
+ # "instrument_code": "BTC_EUR",
1064
+ # "granularity": {unit: "MONTHS", period: 1},
1065
+ # "high": "29750.81",
1066
+ # "low": "16764.59",
1067
+ # "open": "29556.02",
1068
+ # "close": "20164.55",
1069
+ # "volume": "107518944.610659",
1070
+ # "last_sequence": 2275507,
1071
+ # "channel_name": "CANDLESTICKS",
1072
+ # "type": "CANDLESTICK_SNAPSHOT",
1073
+ # "time": "2022-06-30T23:59:59.999000Z"
1074
+ # }
1075
+ #
1076
+ # update
1077
+ # {
1078
+ # "instrument_code": "BTC_EUR",
1079
+ # "granularity": {
1080
+ # "unit": "MINUTES",
1081
+ # "period": 1
1082
+ # },
1083
+ # "high": "20164.16",
1084
+ # "low": "20164.16",
1085
+ # "open": "20164.16",
1086
+ # "close": "20164.16",
1087
+ # "volume": "3645.2768448",
1088
+ # "last_sequence": 2275511,
1089
+ # "channel_name": "CANDLESTICKS",
1090
+ # "type": "CANDLESTICK",
1091
+ # "time": "2022-06-24T21:20:59.999000Z"
1092
+ # }
1093
+ #
1094
+ marketId = self.safe_string(message, 'instrument_code')
1095
+ symbol = self.safe_symbol(marketId)
1096
+ dateTime = self.safe_string(message, 'time')
1097
+ timeframeId = self.safe_value(message, 'granularity')
1098
+ timeframes = self.safe_value(self.options, 'timeframes', {})
1099
+ timeframe = self.find_timeframe(timeframeId, timeframes)
1100
+ channel = 'ohlcv.' + symbol + '.' + timeframe
1101
+ parsed = [
1102
+ self.parse8601(dateTime),
1103
+ self.safe_number(message, 'open'),
1104
+ self.safe_number(message, 'high'),
1105
+ self.safe_number(message, 'low'),
1106
+ self.safe_number(message, 'close'),
1107
+ self.safe_number(message, 'volume'),
1108
+ ]
1109
+ self.ohlcvs[symbol] = self.safe_value(self.ohlcvs, symbol, {})
1110
+ stored = self.safe_value(self.ohlcvs[symbol], timeframe)
1111
+ if stored is None:
1112
+ limit = self.safe_integer(self.options, 'OHLCVLimit', 1000)
1113
+ stored = ArrayCacheByTimestamp(limit)
1114
+ stored.append(parsed)
1115
+ self.ohlcvs[symbol][timeframe] = stored
1116
+ client.resolve(stored, channel)
1117
+
1118
+ def find_timeframe(self, timeframe, timeframes=None):
1119
+ timeframes = timeframes or self.timeframes
1120
+ keys = list(timeframes.keys())
1121
+ for i in range(0, len(keys)):
1122
+ key = keys[i]
1123
+ if timeframes[key]['unit'] == timeframe['unit'] and timeframes[key]['period'] == timeframe['period']:
1124
+ return key
1125
+ return None
1126
+
1127
+ def handle_subscriptions(self, client: Client, message):
1128
+ #
1129
+ # {
1130
+ # "channels": [{
1131
+ # "instrument_codes": [Array],
1132
+ # "depth": 0,
1133
+ # "name": "ORDER_BOOK"
1134
+ # }],
1135
+ # "type": "SUBSCRIPTIONS",
1136
+ # "time": "2022-06-23T15:36:26.948282Z"
1137
+ # }
1138
+ #
1139
+ return message
1140
+
1141
+ def handle_heartbeat(self, client: Client, message):
1142
+ #
1143
+ # {
1144
+ # "subscription": "SYSTEM",
1145
+ # "channel_name": "SYSTEM",
1146
+ # "type": "HEARTBEAT",
1147
+ # "time": "2022-06-23T16:31:49.170224Z"
1148
+ # }
1149
+ #
1150
+ return message
1151
+
1152
+ def handle_error_message(self, client: Client, message):
1153
+ #
1154
+ # {
1155
+ # "error": "MALFORMED_JSON",
1156
+ # "channel_name": "SYSTEM",
1157
+ # "type": "ERROR",
1158
+ # "time": "2022-06-23T15:38:25.470391Z"
1159
+ # }
1160
+ #
1161
+ raise ExchangeError(self.id + ' ' + self.json(message))
1162
+
1163
+ def handle_message(self, client: Client, message):
1164
+ error = self.safe_value(message, 'error')
1165
+ if error is not None:
1166
+ self.handle_error_message(client, message)
1167
+ return
1168
+ type = self.safe_value(message, 'type')
1169
+ handlers: dict = {
1170
+ 'ORDER_BOOK_UPDATE': self.handle_order_book,
1171
+ 'ORDER_BOOK_SNAPSHOT': self.handle_order_book,
1172
+ 'ACTIVE_ORDERS_SNAPSHOT': self.handle_orders,
1173
+ 'INACTIVE_ORDERS_SNAPSHOT': self.handle_orders,
1174
+ 'ACCOUNT_UPDATE': self.handle_account_update,
1175
+ 'BALANCES_SNAPSHOT': self.handle_balance_snapshot,
1176
+ 'SUBSCRIPTIONS': self.handle_subscriptions,
1177
+ 'SUBSCRIPTION_UPDATED': self.handle_subscriptions,
1178
+ 'PRICE_TICK': self.handle_ticker,
1179
+ 'PRICE_TICK_HISTORY': self.handle_subscriptions,
1180
+ 'HEARTBEAT': self.handle_heartbeat,
1181
+ 'MARKET_TICKER_UPDATES': self.handle_ticker,
1182
+ 'PRICE_POINT_UPDATES': self.handle_price_point_updates,
1183
+ 'CANDLESTICK_SNAPSHOT': self.handle_ohlcv,
1184
+ 'CANDLESTICK': self.handle_ohlcv,
1185
+ 'AUTHENTICATED': self.handle_authentication_message,
1186
+ 'FILL': self.handle_trading,
1187
+ 'DONE': self.handle_trading,
1188
+ 'BOOKED': self.handle_trading,
1189
+ 'UPDATE': self.handle_trading,
1190
+ 'TRACKED': self.handle_trading,
1191
+ 'TRIGGERED': self.handle_trading,
1192
+ 'STOP_TRACKED': self.handle_trading,
1193
+ 'STOP_TRIGGERED': self.handle_trading,
1194
+ }
1195
+ handler = self.safe_value(handlers, type)
1196
+ if handler is not None:
1197
+ handler(client, message)
1198
+
1199
+ def handle_price_point_updates(self, client: Client, message):
1200
+ #
1201
+ # {
1202
+ # "channel_name": "MARKET_TICKER",
1203
+ # "type": "PRICE_POINT_UPDATES",
1204
+ # "time": "2019-03-01T10:59:59.999Z",
1205
+ # "price_updates": [{
1206
+ # "instrument": "BTC_EUR",
1207
+ # "prices": [{
1208
+ # "time": "2019-03-01T08:59:59.999Z",
1209
+ # "close_price": "3580.6"
1210
+ # },
1211
+ # ...
1212
+ # ]
1213
+ # },
1214
+ # ...
1215
+ # ]
1216
+ # }
1217
+ #
1218
+ return message
1219
+
1220
+ def handle_authentication_message(self, client: Client, message):
1221
+ #
1222
+ # {
1223
+ # "channel_name": "SYSTEM",
1224
+ # "type": "AUTHENTICATED",
1225
+ # "time": "2022-06-24T20:45:25.447488Z"
1226
+ # }
1227
+ #
1228
+ future = self.safe_value(client.futures, 'authenticated')
1229
+ if future is not None:
1230
+ future.resolve(True)
1231
+ return message
1232
+
1233
+ async def watch_many(self, messageHash, request, subscriptionHash, symbols: Strings = [], params={}):
1234
+ marketIds = []
1235
+ numSymbols = len(symbols)
1236
+ if numSymbols == 0:
1237
+ marketIds = list(self.markets_by_id.keys())
1238
+ else:
1239
+ marketIds = self.market_ids(symbols)
1240
+ url = self.urls['api']['ws']
1241
+ client = self.safe_value(self.clients, url)
1242
+ type = 'SUBSCRIBE'
1243
+ subscription = {}
1244
+ if client is not None:
1245
+ subscription = self.safe_value(client.subscriptions, subscriptionHash)
1246
+ if subscription is not None:
1247
+ for i in range(0, len(marketIds)):
1248
+ marketId = marketIds[i]
1249
+ marketSubscribed = self.safe_bool(subscription, marketId, False)
1250
+ if not marketSubscribed:
1251
+ type = 'UPDATE_SUBSCRIPTION'
1252
+ client.subscriptions[subscriptionHash] = None
1253
+ else:
1254
+ subscription = {}
1255
+ for i in range(0, len(marketIds)):
1256
+ marketId = marketIds[i]
1257
+ subscription[marketId] = True
1258
+ request['type'] = type
1259
+ request['channels'][0]['instrument_codes'] = list(subscription.keys())
1260
+ return await self.watch(url, messageHash, self.deep_extend(request, params), subscriptionHash, subscription)
1261
+
1262
+ async def authenticate(self, params={}):
1263
+ url = self.urls['api']['ws']
1264
+ client = self.client(url)
1265
+ messageHash = 'authenticated'
1266
+ future = client.future('authenticated')
1267
+ authenticated = self.safe_value(client.subscriptions, messageHash)
1268
+ if authenticated is None:
1269
+ self.check_required_credentials()
1270
+ request: dict = {
1271
+ 'type': 'AUTHENTICATE',
1272
+ 'api_token': self.apiKey,
1273
+ }
1274
+ self.watch(url, messageHash, self.extend(request, params), messageHash)
1275
+ return await future