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/bitmart.py ADDED
@@ -0,0 +1,1449 @@
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, ArrayCacheBySymbolBySide, ArrayCacheByTimestamp
8
+ from ccxt.async_support.base.ws.order_book_side import Asks, Bids
9
+ import hashlib
10
+ from ccxt.base.types import Balances, Int, Market, Order, OrderBook, Position, Str, Strings, Ticker, Tickers, Trade
11
+ from ccxt.async_support.base.ws.client import Client
12
+ from typing import List
13
+ from ccxt.base.errors import ExchangeError
14
+ from ccxt.base.errors import AuthenticationError
15
+ from ccxt.base.errors import NotSupported
16
+
17
+
18
+ class bitmart(ccxt.async_support.bitmart):
19
+
20
+ def describe(self):
21
+ return self.deep_extend(super(bitmart, self).describe(), {
22
+ 'has': {
23
+ 'createOrderWs': False,
24
+ 'editOrderWs': False,
25
+ 'fetchOpenOrdersWs': False,
26
+ 'fetchOrderWs': False,
27
+ 'cancelOrderWs': False,
28
+ 'cancelOrdersWs': False,
29
+ 'cancelAllOrdersWs': False,
30
+ 'ws': True,
31
+ 'watchBalance': True,
32
+ 'watchTicker': True,
33
+ 'watchTickers': True,
34
+ 'watchOrderBook': True,
35
+ 'watchOrderBookForSymbols': True,
36
+ 'watchOrders': True,
37
+ 'watchTrades': True,
38
+ 'watchTradesForSymbols': True,
39
+ 'watchOHLCV': True,
40
+ 'watchPosition': 'emulated',
41
+ 'watchPositions': True,
42
+ },
43
+ 'urls': {
44
+ 'api': {
45
+ 'ws': {
46
+ 'spot': {
47
+ 'public': 'wss://ws-manager-compress.{hostname}/api?protocol=1.1',
48
+ 'private': 'wss://ws-manager-compress.{hostname}/user?protocol=1.1',
49
+ },
50
+ 'swap': {
51
+ 'public': 'wss://openapi-ws.{hostname}/api?protocol=1.1',
52
+ 'private': 'wss://openapi-ws.{hostname}/user?protocol=1.1',
53
+ },
54
+ },
55
+ },
56
+ },
57
+ 'options': {
58
+ 'defaultType': 'spot',
59
+ 'watchBalance': {
60
+ 'fetchBalanceSnapshot': True, # or False
61
+ 'awaitBalanceSnapshot': False, # whether to wait for the balance snapshot before providing updates
62
+ },
63
+ #
64
+ # orderbook channels can have:
65
+ # - 'depth5', 'depth20', 'depth50' # these endpoints emit full Orderbooks once in every 500ms
66
+ # - 'depth/increase100' # self endpoint is preferred, because it emits once in 100ms. however, when self value is chosen, it only affects spot-market, but contracts markets automatically `depth50` will be being used
67
+ 'watchOrderBook': {
68
+ 'depth': 'depth/increase100',
69
+ },
70
+ 'watchOrderBookForSymbols': {
71
+ 'depth': 'depth/increase100',
72
+ },
73
+ 'ws': {
74
+ 'inflate': True,
75
+ },
76
+ 'timeframes': {
77
+ '1m': '1m',
78
+ '3m': '3m',
79
+ '5m': '5m',
80
+ '15m': '15m',
81
+ '30m': '30m',
82
+ '45m': '45m',
83
+ '1h': '1H',
84
+ '2h': '2H',
85
+ '3h': '3H',
86
+ '4h': '4H',
87
+ '1d': '1D',
88
+ '1w': '1W',
89
+ '1M': '1M',
90
+ },
91
+ },
92
+ 'streaming': {
93
+ 'keepAlive': 15000,
94
+ },
95
+ })
96
+
97
+ async def subscribe(self, channel, symbol, type, params={}):
98
+ market = self.market(symbol)
99
+ url = self.implode_hostname(self.urls['api']['ws'][type]['public'])
100
+ request = {}
101
+ messageHash = None
102
+ if type == 'spot':
103
+ messageHash = 'spot/' + channel + ':' + market['id']
104
+ request = {
105
+ 'op': 'subscribe',
106
+ 'args': [messageHash],
107
+ }
108
+ else:
109
+ messageHash = 'futures/' + channel + ':' + market['id']
110
+ request = {
111
+ 'action': 'subscribe',
112
+ 'args': [messageHash],
113
+ }
114
+ return await self.watch(url, messageHash, self.deep_extend(request, params), messageHash)
115
+
116
+ async def subscribe_multiple(self, channel: str, type: str, symbols: Strings = None, params={}):
117
+ symbols = self.market_symbols(symbols, type, False, True)
118
+ url = self.implode_hostname(self.urls['api']['ws'][type]['public'])
119
+ channelType = 'spot' if (type == 'spot') else 'futures'
120
+ actionType = 'op' if (type == 'spot') else 'action'
121
+ rawSubscriptions = []
122
+ messageHashes = []
123
+ for i in range(0, len(symbols)):
124
+ market = self.market(symbols[i])
125
+ message = channelType + '/' + channel + ':' + market['id']
126
+ rawSubscriptions.append(message)
127
+ messageHashes.append(channel + ':' + market['symbol'])
128
+ # exclusion, futures "tickers" need one generic request for all symbols
129
+ if (type != 'spot') and (channel == 'ticker'):
130
+ rawSubscriptions = [channelType + '/' + channel]
131
+ request: dict = {
132
+ 'args': rawSubscriptions,
133
+ }
134
+ request[actionType] = 'subscribe'
135
+ return await self.watch_multiple(url, messageHashes, self.deep_extend(request, params), rawSubscriptions)
136
+
137
+ async def watch_balance(self, params={}) -> Balances:
138
+ """
139
+ :see: https://developer-pro.bitmart.com/en/spot/#private-balance-change
140
+ :see: https://developer-pro.bitmart.com/en/futures/#private-assets-channel
141
+ watch balance and get the amount of funds available for trading or funds locked in orders
142
+ :param dict [params]: extra parameters specific to the exchange API endpoint
143
+ :returns dict: a `balance structure <https://docs.ccxt.com/#/?id=balance-structure>`
144
+ """
145
+ await self.load_markets()
146
+ type = 'spot'
147
+ type, params = self.handle_market_type_and_params('watchBalance', None, params)
148
+ await self.authenticate(type, params)
149
+ request = {}
150
+ if type == 'spot':
151
+ request = {
152
+ 'op': 'subscribe',
153
+ 'args': ['spot/user/balance:BALANCE_UPDATE'],
154
+ }
155
+ else:
156
+ request = {
157
+ 'action': 'subscribe',
158
+ 'args': ['futures/asset:USDT', 'futures/asset:BTC', 'futures/asset:ETH'],
159
+ }
160
+ messageHash = 'balance:' + type
161
+ url = self.implode_hostname(self.urls['api']['ws'][type]['private'])
162
+ client = self.client(url)
163
+ self.set_balance_cache(client, type, messageHash)
164
+ fetchBalanceSnapshot = None
165
+ awaitBalanceSnapshot = None
166
+ fetchBalanceSnapshot, params = self.handle_option_and_params(self.options, 'watchBalance', 'fetchBalanceSnapshot', True)
167
+ awaitBalanceSnapshot, params = self.handle_option_and_params(self.options, 'watchBalance', 'awaitBalanceSnapshot', False)
168
+ if fetchBalanceSnapshot and awaitBalanceSnapshot:
169
+ await client.future(type + ':fetchBalanceSnapshot')
170
+ return await self.watch(url, messageHash, self.deep_extend(request, params), messageHash)
171
+
172
+ def set_balance_cache(self, client: Client, type, subscribeHash):
173
+ if subscribeHash in client.subscriptions:
174
+ return
175
+ options = self.safe_value(self.options, 'watchBalance')
176
+ snapshot = self.safe_bool(options, 'fetchBalanceSnapshot', True)
177
+ if snapshot:
178
+ messageHash = type + ':' + 'fetchBalanceSnapshot'
179
+ if not (messageHash in client.futures):
180
+ client.future(messageHash)
181
+ self.spawn(self.load_balance_snapshot, client, messageHash, type)
182
+ self.balance[type] = {}
183
+ # without self comment, transpilation breaks for some reason...
184
+
185
+ async def load_balance_snapshot(self, client, messageHash, type):
186
+ response = await self.fetch_balance({'type': type})
187
+ self.balance[type] = self.extend(response, self.safe_value(self.balance, type, {}))
188
+ # don't remove the future from the .futures cache
189
+ future = client.futures[messageHash]
190
+ future.resolve()
191
+ client.resolve(self.balance[type], 'balance:' + type)
192
+
193
+ def handle_balance(self, client: Client, message):
194
+ #
195
+ # spot
196
+ # {
197
+ # "data":[
198
+ # {
199
+ # "balance_details":[
200
+ # {
201
+ # "av_bal":"0.206000000000000000000000000000",
202
+ # "ccy":"LTC",
203
+ # "fz_bal":"0.100000000000000000000000000000"
204
+ # }
205
+ # ],
206
+ # "event_time":"1701632345415",
207
+ # "event_type":"TRANSACTION_COMPLETED"
208
+ # }
209
+ # ],
210
+ # "table":"spot/user/balance"
211
+ # }
212
+ # swap
213
+ # {
214
+ # group: 'futures/asset:USDT',
215
+ # data: {
216
+ # currency: 'USDT',
217
+ # available_balance: '37.19688649135',
218
+ # position_deposit: '0.788687546',
219
+ # frozen_balance: '0'
220
+ # }
221
+ # }
222
+ #
223
+ channel = self.safe_string_2(message, 'table', 'group')
224
+ data = self.safe_value(message, 'data')
225
+ if data is None:
226
+ return
227
+ isSpot = (channel.find('spot') >= 0)
228
+ type = 'spot' if isSpot else 'swap'
229
+ self.balance[type]['info'] = message
230
+ if isSpot:
231
+ if not isinstance(data, list):
232
+ return
233
+ for i in range(0, len(data)):
234
+ timestamp = self.safe_integer(message, 'event_time')
235
+ self.balance[type]['timestamp'] = timestamp
236
+ self.balance[type]['datetime'] = self.iso8601(timestamp)
237
+ balanceDetails = self.safe_value(data[i], 'balance_details', [])
238
+ for ii in range(0, len(balanceDetails)):
239
+ rawBalance = balanceDetails[i]
240
+ account = self.account()
241
+ currencyId = self.safe_string(rawBalance, 'ccy')
242
+ code = self.safe_currency_code(currencyId)
243
+ account['free'] = self.safe_string(rawBalance, 'av_bal')
244
+ account['used'] = self.safe_string(rawBalance, 'fz_bal')
245
+ self.balance[type][code] = account
246
+ else:
247
+ currencyId = self.safe_string(data, 'currency')
248
+ code = self.safe_currency_code(currencyId)
249
+ account = self.account()
250
+ account['free'] = self.safe_string(data, 'available_balance')
251
+ account['used'] = self.safe_string(data, 'frozen_balance')
252
+ self.balance[type][code] = account
253
+ self.balance[type] = self.safe_balance(self.balance[type])
254
+ messageHash = 'balance:' + type
255
+ client.resolve(self.balance[type], messageHash)
256
+
257
+ async def watch_trades(self, symbol: str, since: Int = None, limit: Int = None, params={}) -> List[Trade]:
258
+ """
259
+ :see: https://developer-pro.bitmart.com/en/spot/#public-trade-channel
260
+ :see: https://developer-pro.bitmart.com/en/futures/#public-trade-channel
261
+ get the list of most recent trades for a particular symbol
262
+ :param str symbol: unified symbol of the market to fetch trades for
263
+ :param int [since]: timestamp in ms of the earliest trade to fetch
264
+ :param int [limit]: the maximum amount of trades to fetch
265
+ :param dict [params]: extra parameters specific to the exchange API endpoint
266
+ :returns dict[]: a list of `trade structures <https://docs.ccxt.com/#/?id=public-trades>`
267
+ """
268
+ return await self.watch_trades_for_symbols([symbol], since, limit, params)
269
+
270
+ async def watch_trades_for_symbols(self, symbols: List[str], since: Int = None, limit: Int = None, params={}) -> List[Trade]:
271
+ """
272
+ :see: https://developer-pro.bitmart.com/en/spot/#public-trade-channel
273
+ get the list of most recent trades for a list of symbols
274
+ :param str[] symbols: unified symbol of the market to fetch trades for
275
+ :param int [since]: timestamp in ms of the earliest trade to fetch
276
+ :param int [limit]: the maximum amount of trades to fetch
277
+ :param dict [params]: extra parameters specific to the exchange API endpoint
278
+ :returns dict[]: a list of `trade structures <https://docs.ccxt.com/#/?id=public-trades>`
279
+ """
280
+ await self.load_markets()
281
+ marketType = None
282
+ symbols, marketType, params = self.get_params_for_multiple_sub('watchTradesForSymbols', symbols, limit, params)
283
+ channelName = 'trade'
284
+ trades = await self.subscribe_multiple(channelName, marketType, symbols, params)
285
+ if self.newUpdates:
286
+ first = self.safe_dict(trades, 0)
287
+ tradeSymbol = self.safe_string(first, 'symbol')
288
+ limit = trades.getLimit(tradeSymbol, limit)
289
+ return self.filter_by_since_limit(trades, since, limit, 'timestamp', True)
290
+
291
+ def get_params_for_multiple_sub(self, methodName: str, symbols: List[str], limit: Int = None, params={}):
292
+ symbols = self.market_symbols(symbols, None, False, True)
293
+ length = len(symbols)
294
+ if length > 20:
295
+ raise NotSupported(self.id + ' ' + methodName + '() accepts a maximum of 20 symbols in one request')
296
+ market = self.market(symbols[0])
297
+ marketType = None
298
+ marketType, params = self.handle_market_type_and_params(methodName, market, params)
299
+ return [symbols, marketType, params]
300
+
301
+ async def watch_ticker(self, symbol: str, params={}) -> Ticker:
302
+ """
303
+ :see: https://developer-pro.bitmart.com/en/spot/#public-ticker-channel
304
+ watches a price ticker, a statistical calculation with the information calculated over the past 24 hours for a specific market
305
+ :param str symbol: unified symbol of the market to fetch the ticker for
306
+ :param dict [params]: extra parameters specific to the exchange API endpoint
307
+ :returns dict: a `ticker structure <https://docs.ccxt.com/#/?id=ticker-structure>`
308
+ """
309
+ await self.load_markets()
310
+ symbol = self.symbol(symbol)
311
+ tickers = await self.watch_tickers([symbol], params)
312
+ return tickers[symbol]
313
+
314
+ async def watch_tickers(self, symbols: Strings = None, params={}) -> Tickers:
315
+ """
316
+ :see: https://developer-pro.bitmart.com/en/futures/#overview
317
+ watches a price ticker, a statistical calculation with the information calculated over the past 24 hours for all markets of a specific list
318
+ :param str[] symbols: unified symbol of the market to fetch the ticker for
319
+ :param dict [params]: extra parameters specific to the exchange API endpoint
320
+ :returns dict: a `ticker structure <https://docs.ccxt.com/#/?id=ticker-structure>`
321
+ """
322
+ await self.load_markets()
323
+ market = self.get_market_from_symbols(symbols)
324
+ marketType = None
325
+ marketType, params = self.handle_market_type_and_params('watchTickers', market, params)
326
+ ticker = await self.subscribe_multiple('ticker', marketType, symbols, params)
327
+ if self.newUpdates:
328
+ tickers: dict = {}
329
+ tickers[ticker['symbol']] = ticker
330
+ return tickers
331
+ return self.filter_by_array(self.tickers, 'symbol', symbols)
332
+
333
+ async def watch_orders(self, symbol: Str = None, since: Int = None, limit: Int = None, params={}) -> List[Order]:
334
+ """
335
+ watches information on multiple orders made by the user
336
+ :see: https://developer-pro.bitmart.com/en/spot/#private-order-progress
337
+ :see: https://developer-pro.bitmart.com/en/futures/#private-order-channel
338
+ :param str symbol: unified market symbol of the market orders were made in
339
+ :param int [since]: the earliest time in ms to fetch orders for
340
+ :param int [limit]: the maximum number of order structures to retrieve
341
+ :param dict [params]: extra parameters specific to the exchange API endpoint
342
+ :returns dict[]: a list of `order structures <https://docs.ccxt.com/#/?id=order-structure>`
343
+ """
344
+ await self.load_markets()
345
+ market = None
346
+ messageHash = 'orders'
347
+ if symbol is not None:
348
+ symbol = self.symbol(symbol)
349
+ market = self.market(symbol)
350
+ messageHash = 'orders::' + symbol
351
+ type = 'spot'
352
+ type, params = self.handle_market_type_and_params('watchOrders', market, params)
353
+ await self.authenticate(type, params)
354
+ request = None
355
+ if type == 'spot':
356
+ argsRequest = 'spot/user/order:'
357
+ if symbol is not None:
358
+ argsRequest += market['id']
359
+ else:
360
+ argsRequest = 'spot/user/orders:ALL_SYMBOLS'
361
+ request = {
362
+ 'op': 'subscribe',
363
+ 'args': [argsRequest],
364
+ }
365
+ else:
366
+ request = {
367
+ 'action': 'subscribe',
368
+ 'args': ['futures/order'],
369
+ }
370
+ url = self.implode_hostname(self.urls['api']['ws'][type]['private'])
371
+ newOrders = await self.watch(url, messageHash, self.deep_extend(request, params), messageHash)
372
+ if self.newUpdates:
373
+ return newOrders
374
+ return self.filter_by_symbol_since_limit(self.orders, symbol, since, limit, True)
375
+
376
+ def handle_orders(self, client: Client, message):
377
+ #
378
+ # spot
379
+ # {
380
+ # "data":[
381
+ # {
382
+ # "symbol": "LTC_USDT",
383
+ # "notional": '',
384
+ # "side": "buy",
385
+ # "last_fill_time": "0",
386
+ # "ms_t": "1646216634000",
387
+ # "type": "limit",
388
+ # "filled_notional": "0.000000000000000000000000000000",
389
+ # "last_fill_price": "0",
390
+ # "size": "0.500000000000000000000000000000",
391
+ # "price": "50.000000000000000000000000000000",
392
+ # "last_fill_count": "0",
393
+ # "filled_size": "0.000000000000000000000000000000",
394
+ # "margin_trading": "0",
395
+ # "state": "8",
396
+ # "order_id": "24807076628",
397
+ # "order_type": "0"
398
+ # }
399
+ # ],
400
+ # "table":"spot/user/order"
401
+ # }
402
+ # swap
403
+ # {
404
+ # "group":"futures/order",
405
+ # "data":[
406
+ # {
407
+ # "action":2,
408
+ # "order":{
409
+ # "order_id":"2312045036986775",
410
+ # "client_order_id":"",
411
+ # "price":"71.61707928",
412
+ # "size":"1",
413
+ # "symbol":"LTCUSDT",
414
+ # "state":1,
415
+ # "side":4,
416
+ # "type":"market",
417
+ # "leverage":"1",
418
+ # "open_type":"cross",
419
+ # "deal_avg_price":"0",
420
+ # "deal_size":"0",
421
+ # "create_time":1701625324646,
422
+ # "update_time":1701625324640,
423
+ # "plan_order_id":"",
424
+ # "last_trade":null
425
+ # }
426
+ # }
427
+ # ]
428
+ # }
429
+ #
430
+ orders = self.safe_value(message, 'data')
431
+ if orders is None:
432
+ return
433
+ ordersLength = len(orders)
434
+ newOrders = []
435
+ symbols: dict = {}
436
+ if ordersLength > 0:
437
+ limit = self.safe_integer(self.options, 'ordersLimit', 1000)
438
+ if self.orders is None:
439
+ self.orders = ArrayCacheBySymbolById(limit)
440
+ stored = self.orders
441
+ for i in range(0, len(orders)):
442
+ order = self.parse_ws_order(orders[i])
443
+ stored.append(order)
444
+ newOrders.append(order)
445
+ symbol = order['symbol']
446
+ symbols[symbol] = True
447
+ messageHash = 'orders'
448
+ symbolKeys = list(symbols.keys())
449
+ for i in range(0, len(symbolKeys)):
450
+ symbol = symbolKeys[i]
451
+ symbolSpecificMessageHash = messageHash + '::' + symbol
452
+ client.resolve(newOrders, symbolSpecificMessageHash)
453
+ client.resolve(newOrders, messageHash)
454
+
455
+ def parse_ws_order(self, order, market: Market = None):
456
+ #
457
+ # spot
458
+ # {
459
+ # "symbol": "LTC_USDT",
460
+ # "notional": '',
461
+ # "side": "buy",
462
+ # "last_fill_time": "0",
463
+ # "ms_t": "1646216634000",
464
+ # "type": "limit",
465
+ # "filled_notional": "0.000000000000000000000000000000",
466
+ # "last_fill_price": "0",
467
+ # "size": "0.500000000000000000000000000000",
468
+ # "price": "50.000000000000000000000000000000",
469
+ # "last_fill_count": "0",
470
+ # "filled_size": "0.000000000000000000000000000000",
471
+ # "margin_trading": "0",
472
+ # "state": "8",
473
+ # "order_id": "24807076628",
474
+ # "order_type": "0"
475
+ # }
476
+ # swap
477
+ # {
478
+ # "action":2,
479
+ # "order":{
480
+ # "order_id":"2312045036986775",
481
+ # "client_order_id":"",
482
+ # "price":"71.61707928",
483
+ # "size":"1",
484
+ # "symbol":"LTCUSDT",
485
+ # "state":1,
486
+ # "side":4,
487
+ # "type":"market",
488
+ # "leverage":"1",
489
+ # "open_type":"cross",
490
+ # "deal_avg_price":"0",
491
+ # "deal_size":"0",
492
+ # "create_time":1701625324646,
493
+ # "update_time":1701625324640,
494
+ # "plan_order_id":"",
495
+ # "last_trade":null
496
+ # }
497
+ # }
498
+ #
499
+ action = self.safe_number(order, 'action')
500
+ isSpot = (action is None)
501
+ if isSpot:
502
+ marketId = self.safe_string(order, 'symbol')
503
+ market = self.safe_market(marketId, market, '_', 'spot')
504
+ id = self.safe_string(order, 'order_id')
505
+ clientOrderId = self.safe_string(order, 'clientOid')
506
+ price = self.safe_string(order, 'price')
507
+ filled = self.safe_string(order, 'filled_size')
508
+ amount = self.safe_string(order, 'size')
509
+ type = self.safe_string(order, 'type')
510
+ rawState = self.safe_string(order, 'state')
511
+ status = self.parse_order_status_by_type(market['type'], rawState)
512
+ timestamp = self.safe_integer(order, 'ms_t')
513
+ symbol = market['symbol']
514
+ side = self.safe_string_lower(order, 'side')
515
+ return self.safe_order({
516
+ 'info': order,
517
+ 'symbol': symbol,
518
+ 'id': id,
519
+ 'clientOrderId': clientOrderId,
520
+ 'timestamp': None,
521
+ 'datetime': None,
522
+ 'lastTradeTimestamp': timestamp,
523
+ 'type': type,
524
+ 'timeInForce': None,
525
+ 'postOnly': None,
526
+ 'side': side,
527
+ 'price': price,
528
+ 'stopPrice': None,
529
+ 'triggerPrice': None,
530
+ 'amount': amount,
531
+ 'cost': None,
532
+ 'average': None,
533
+ 'filled': filled,
534
+ 'remaining': None,
535
+ 'status': status,
536
+ 'fee': None,
537
+ 'trades': None,
538
+ }, market)
539
+ else:
540
+ orderInfo = self.safe_value(order, 'order')
541
+ marketId = self.safe_string(orderInfo, 'symbol')
542
+ symbol = self.safe_symbol(marketId, market, '', 'swap')
543
+ orderId = self.safe_string(orderInfo, 'order_id')
544
+ timestamp = self.safe_integer(orderInfo, 'create_time')
545
+ updatedTimestamp = self.safe_integer(orderInfo, 'update_time')
546
+ lastTrade = self.safe_value(orderInfo, 'last_trade')
547
+ cachedOrders = self.orders
548
+ orders = self.safe_value(cachedOrders.hashmap, symbol, {})
549
+ cachedOrder = self.safe_value(orders, orderId)
550
+ trades = None
551
+ if cachedOrder is not None:
552
+ trades = self.safe_value(order, 'trades')
553
+ if lastTrade is not None:
554
+ if trades is None:
555
+ trades = []
556
+ trades.append(lastTrade)
557
+ return self.safe_order({
558
+ 'info': order,
559
+ 'symbol': symbol,
560
+ 'id': orderId,
561
+ 'clientOrderId': self.safe_string(orderInfo, 'client_order_id'),
562
+ 'timestamp': timestamp,
563
+ 'datetime': self.iso8601(timestamp),
564
+ 'lastTradeTimestamp': updatedTimestamp,
565
+ 'type': self.safe_string(orderInfo, 'type'),
566
+ 'timeInForce': None,
567
+ 'postOnly': None,
568
+ 'side': self.parse_ws_order_side(self.safe_string(orderInfo, 'side')),
569
+ 'price': self.safe_string(orderInfo, 'price'),
570
+ 'stopPrice': None,
571
+ 'triggerPrice': None,
572
+ 'amount': self.safe_string(orderInfo, 'size'),
573
+ 'cost': None,
574
+ 'average': self.safe_string(orderInfo, 'deal_avg_price'),
575
+ 'filled': self.safe_string(orderInfo, 'deal_size'),
576
+ 'remaining': None,
577
+ 'status': self.parse_ws_order_status(self.safe_string(order, 'action')),
578
+ 'fee': None,
579
+ 'trades': trades,
580
+ }, market)
581
+
582
+ def parse_ws_order_status(self, statusId):
583
+ statuses: dict = {
584
+ '1': 'closed', # match deal
585
+ '2': 'open', # submit order
586
+ '3': 'canceled', # cancel order
587
+ '4': 'closed', # liquidate cancel order
588
+ '5': 'canceled', # adl cancel order
589
+ '6': 'open', # part liquidate
590
+ '7': 'open', # bankrupty order
591
+ '8': 'closed', # passive adl match deal
592
+ '9': 'closed', # active adl match deal
593
+ }
594
+ return self.safe_string(statuses, statusId, statusId)
595
+
596
+ def parse_ws_order_side(self, sideId):
597
+ sides: dict = {
598
+ '1': 'buy', # buy_open_long
599
+ '2': 'buy', # buy_close_short
600
+ '3': 'sell', # sell_close_long
601
+ '4': 'sell', # sell_open_short
602
+ }
603
+ return self.safe_string(sides, sideId, sideId)
604
+
605
+ async def watch_positions(self, symbols: Strings = None, since: Int = None, limit: Int = None, params={}) -> List[Position]:
606
+ """
607
+ :see: https://developer-pro.bitmart.com/en/futures/#private-position-channel
608
+ watch all open positions
609
+ :param str[]|None symbols: list of unified market symbols
610
+ :param dict params: extra parameters specific to the exchange API endpoint
611
+ :returns dict[]: a list of `position structure <https://docs.ccxt.com/en/latest/manual.html#position-structure>`
612
+ """
613
+ await self.load_markets()
614
+ type = 'swap'
615
+ await self.authenticate(type, params)
616
+ symbols = self.market_symbols(symbols, 'swap', True, True, False)
617
+ messageHash = 'positions'
618
+ if symbols is not None:
619
+ messageHash += '::' + ','.join(symbols)
620
+ subscriptionHash = 'futures/position'
621
+ request: dict = {
622
+ 'action': 'subscribe',
623
+ 'args': ['futures/position'],
624
+ }
625
+ url = self.implode_hostname(self.urls['api']['ws'][type]['private'])
626
+ newPositions = await self.watch(url, messageHash, self.deep_extend(request, params), subscriptionHash)
627
+ if self.newUpdates:
628
+ return newPositions
629
+ return self.filter_by_symbols_since_limit(self.positions, symbols, since, limit)
630
+
631
+ def handle_positions(self, client: Client, message):
632
+ #
633
+ # {
634
+ # "group":"futures/position",
635
+ # "data":[
636
+ # {
637
+ # "symbol":"LTCUSDT",
638
+ # "hold_volume":"5",
639
+ # "position_type":2,
640
+ # "open_type":2,
641
+ # "frozen_volume":"0",
642
+ # "close_volume":"0",
643
+ # "hold_avg_price":"71.582",
644
+ # "close_avg_price":"0",
645
+ # "open_avg_price":"71.582",
646
+ # "liquidate_price":"0",
647
+ # "create_time":1701623327513,
648
+ # "update_time":1701627620439
649
+ # },
650
+ # {
651
+ # "symbol":"LTCUSDT",
652
+ # "hold_volume":"6",
653
+ # "position_type":1,
654
+ # "open_type":2,
655
+ # "frozen_volume":"0",
656
+ # "close_volume":"0",
657
+ # "hold_avg_price":"71.681666666666666667",
658
+ # "close_avg_price":"0",
659
+ # "open_avg_price":"71.681666666666666667",
660
+ # "liquidate_price":"0",
661
+ # "create_time":1701621167225,
662
+ # "update_time":1701628152614
663
+ # }
664
+ # ]
665
+ # }
666
+ #
667
+ data = self.safe_value(message, 'data', [])
668
+ if self.positions is None:
669
+ self.positions = ArrayCacheBySymbolBySide()
670
+ cache = self.positions
671
+ newPositions = []
672
+ for i in range(0, len(data)):
673
+ rawPosition = data[i]
674
+ position = self.parse_ws_position(rawPosition)
675
+ newPositions.append(position)
676
+ cache.append(position)
677
+ messageHashes = self.find_message_hashes(client, 'positions::')
678
+ for i in range(0, len(messageHashes)):
679
+ messageHash = messageHashes[i]
680
+ parts = messageHash.split('::')
681
+ symbolsString = parts[1]
682
+ symbols = symbolsString.split(',')
683
+ positions = self.filter_by_array(newPositions, 'symbol', symbols, False)
684
+ if not self.is_empty(positions):
685
+ client.resolve(positions, messageHash)
686
+ client.resolve(newPositions, 'positions')
687
+
688
+ def parse_ws_position(self, position, market: Market = None):
689
+ #
690
+ # {
691
+ # "symbol":"LTCUSDT",
692
+ # "hold_volume":"6",
693
+ # "position_type":1,
694
+ # "open_type":2,
695
+ # "frozen_volume":"0",
696
+ # "close_volume":"0",
697
+ # "hold_avg_price":"71.681666666666666667",
698
+ # "close_avg_price":"0",
699
+ # "open_avg_price":"71.681666666666666667",
700
+ # "liquidate_price":"0",
701
+ # "create_time":1701621167225,
702
+ # "update_time":1701628152614
703
+ # }
704
+ #
705
+ marketId = self.safe_string(position, 'symbol')
706
+ market = self.safe_market(marketId, market, None, 'swap')
707
+ symbol = market['symbol']
708
+ openTimestamp = self.safe_integer(position, 'create_time')
709
+ timestamp = self.safe_integer(position, 'update_time')
710
+ side = self.safe_number(position, 'position_type')
711
+ marginModeId = self.safe_number(position, 'open_type')
712
+ return self.safe_position({
713
+ 'info': position,
714
+ 'id': None,
715
+ 'symbol': symbol,
716
+ 'timestamp': openTimestamp,
717
+ 'datetime': self.iso8601(openTimestamp),
718
+ 'lastUpdateTimestamp': timestamp,
719
+ 'hedged': None,
720
+ 'side': 'long' if (side == 1) else 'short',
721
+ 'contracts': self.safe_number(position, 'hold_volume'),
722
+ 'contractSize': self.safe_number(market, 'contractSize'),
723
+ 'entryPrice': self.safe_number(position, 'open_avg_price'),
724
+ 'markPrice': self.safe_number(position, 'hold_avg_price'),
725
+ 'lastPrice': None,
726
+ 'notional': None,
727
+ 'leverage': None,
728
+ 'collateral': None,
729
+ 'initialMargin': None,
730
+ 'initialMarginPercentage': None,
731
+ 'maintenanceMargin': None,
732
+ 'maintenanceMarginPercentage': None,
733
+ 'unrealizedPnl': None,
734
+ 'realizedPnl': None,
735
+ 'liquidationPrice': self.safe_number(position, 'liquidate_price'),
736
+ 'marginMode': 'isolated' if (marginModeId == 1) else 'cross',
737
+ 'percentage': None,
738
+ 'marginRatio': None,
739
+ 'stopLossPrice': None,
740
+ 'takeProfitPrice': None,
741
+ })
742
+
743
+ def handle_trade(self, client: Client, message):
744
+ #
745
+ # spot
746
+ # {
747
+ # "table": "spot/trade",
748
+ # "data": [
749
+ # {
750
+ # "price": "52700.50",
751
+ # "s_t": 1630982050,
752
+ # "side": "buy",
753
+ # "size": "0.00112",
754
+ # "symbol": "BTC_USDT"
755
+ # },
756
+ # ]
757
+ # }
758
+ #
759
+ # swap
760
+ # {
761
+ # "group":"futures/trade:BTCUSDT",
762
+ # "data":[
763
+ # {
764
+ # "trade_id":6798697637,
765
+ # "contract_id":1,
766
+ # "symbol":"BTCUSDT",
767
+ # "deal_price":"39735.8",
768
+ # "deal_vol":"2",
769
+ # "type":0,
770
+ # "way":1,
771
+ # "create_time":1701618503,
772
+ # "create_time_mill":1701618503517,
773
+ # "created_at":"2023-12-03T15:48:23.517518538Z"
774
+ # }
775
+ # ]
776
+ # }
777
+ #
778
+ data = self.safe_value(message, 'data')
779
+ if data is None:
780
+ return
781
+ symbol = None
782
+ length = len(data)
783
+ isSwap = ('group' in message)
784
+ if isSwap:
785
+ # in swap, chronologically decreasing: 1709536849322, 1709536848954,
786
+ for i in range(0, length):
787
+ index = length - i - 1
788
+ symbol = self.handle_trade_loop(data[index])
789
+ else:
790
+ # in spot, chronologically increasing: 1709536771200, 1709536771226,
791
+ for i in range(0, length):
792
+ symbol = self.handle_trade_loop(data[i])
793
+ client.resolve(self.trades[symbol], 'trade:' + symbol)
794
+
795
+ def handle_trade_loop(self, entry):
796
+ trade = self.parse_ws_trade(entry)
797
+ symbol = trade['symbol']
798
+ tradesLimit = self.safe_integer(self.options, 'tradesLimit', 1000)
799
+ if self.safe_value(self.trades, symbol) is None:
800
+ self.trades[symbol] = ArrayCache(tradesLimit)
801
+ stored = self.trades[symbol]
802
+ stored.append(trade)
803
+ return symbol
804
+
805
+ def parse_ws_trade(self, trade, market: Market = None):
806
+ # spot
807
+ # {
808
+ # "price": "52700.50",
809
+ # "s_t": 1630982050,
810
+ # "side": "buy",
811
+ # "size": "0.00112",
812
+ # "symbol": "BTC_USDT"
813
+ # }
814
+ # swap
815
+ # {
816
+ # "trade_id":6798697637,
817
+ # "contract_id":1,
818
+ # "symbol":"BTCUSDT",
819
+ # "deal_price":"39735.8",
820
+ # "deal_vol":"2",
821
+ # "type":0,
822
+ # "way":1,
823
+ # "create_time":1701618503,
824
+ # "create_time_mill":1701618503517,
825
+ # "created_at":"2023-12-03T15:48:23.517518538Z"
826
+ # }
827
+ #
828
+ contractId = self.safe_string(trade, 'contract_id')
829
+ marketType = 'spot' if (contractId is None) else 'swap'
830
+ marketDelimiter = '_' if (marketType == 'spot') else ''
831
+ timestamp = self.safe_integer(trade, 'create_time_mill', self.safe_timestamp(trade, 's_t'))
832
+ marketId = self.safe_string(trade, 'symbol')
833
+ return self.safe_trade({
834
+ 'info': trade,
835
+ 'id': self.safe_string(trade, 'trade_id'),
836
+ 'order': None,
837
+ 'timestamp': timestamp,
838
+ 'datetime': self.iso8601(timestamp),
839
+ 'symbol': self.safe_symbol(marketId, market, marketDelimiter, marketType),
840
+ 'type': None,
841
+ 'side': self.safe_string(trade, 'side'),
842
+ 'price': self.safe_string_2(trade, 'price', 'deal_price'),
843
+ 'amount': self.safe_string_2(trade, 'size', 'deal_vol'),
844
+ 'cost': None,
845
+ 'takerOrMaker': None,
846
+ 'fee': None,
847
+ }, market)
848
+
849
+ def handle_ticker(self, client: Client, message):
850
+ #
851
+ # {
852
+ # "data": [
853
+ # {
854
+ # "base_volume_24h": "78615593.81",
855
+ # "high_24h": "52756.97",
856
+ # "last_price": "52638.31",
857
+ # "low_24h": "50991.35",
858
+ # "open_24h": "51692.03",
859
+ # "s_t": 1630981727,
860
+ # "symbol": "BTC_USDT"
861
+ # }
862
+ # ],
863
+ # "table": "spot/ticker"
864
+ # }
865
+ # {
866
+ # "group":"futures/ticker",
867
+ # "data":{
868
+ # "symbol":"BTCUSDT",
869
+ # "volume_24":"117387.58",
870
+ # "fair_price":"146.24",
871
+ # "last_price":"146.24",
872
+ # "range":"147.17",
873
+ # "ask_price": "147.11",
874
+ # "ask_vol": "1",
875
+ # "bid_price": "142.11",
876
+ # "bid_vol": "1"
877
+ # }
878
+ # }
879
+ #
880
+ table = self.safe_string(message, 'table')
881
+ isSpot = (table is not None)
882
+ rawTickers = []
883
+ if isSpot:
884
+ rawTickers = self.safe_list(message, 'data', [])
885
+ else:
886
+ rawTickers = [self.safe_value(message, 'data', {})]
887
+ if not len(rawTickers):
888
+ return
889
+ for i in range(0, len(rawTickers)):
890
+ ticker = self.parse_ticker(rawTickers[i]) if isSpot else self.parse_ws_swap_ticker(rawTickers[i])
891
+ symbol = ticker['symbol']
892
+ self.tickers[symbol] = ticker
893
+ messageHash = 'ticker:' + symbol
894
+ client.resolve(ticker, messageHash)
895
+
896
+ def parse_ws_swap_ticker(self, ticker, market: Market = None):
897
+ #
898
+ # {
899
+ # "symbol":"BTCUSDT",
900
+ # "volume_24":"117387.58",
901
+ # "fair_price":"146.24",
902
+ # "last_price":"146.24",
903
+ # "range":"147.17",
904
+ # "ask_price": "147.11",
905
+ # "ask_vol": "1",
906
+ # "bid_price": "142.11",
907
+ # "bid_vol": "1"
908
+ # }
909
+ marketId = self.safe_string(ticker, 'symbol')
910
+ return self.safe_ticker({
911
+ 'symbol': self.safe_symbol(marketId, market, '', 'swap'),
912
+ 'timestamp': None,
913
+ 'datetime': None,
914
+ 'high': None,
915
+ 'low': None,
916
+ 'bid': self.safe_string(ticker, 'bid_price'),
917
+ 'bidVolume': self.safe_string(ticker, 'bid_vol'),
918
+ 'ask': self.safe_string(ticker, 'ask_price'),
919
+ 'askVolume': self.safe_string(ticker, 'ask_vol'),
920
+ 'vwap': None,
921
+ 'open': None,
922
+ 'close': None,
923
+ 'last': self.safe_string(ticker, 'last_price'),
924
+ 'previousClose': None,
925
+ 'change': None,
926
+ 'percentage': None,
927
+ 'average': self.safe_string(ticker, 'fair_price'),
928
+ 'baseVolume': None,
929
+ 'quoteVolume': self.safe_string(ticker, 'volume_24'),
930
+ 'info': ticker,
931
+ }, market)
932
+
933
+ async def watch_ohlcv(self, symbol: str, timeframe='1m', since: Int = None, limit: Int = None, params={}) -> List[list]:
934
+ """
935
+ :see: https://developer-pro.bitmart.com/en/spot/#public-kline-channel
936
+ :see: https://developer-pro.bitmart.com/en/futures/#public-klinebin-channel
937
+ watches historical candlestick data containing the open, high, low, and close price, and the volume of a market
938
+ :param str symbol: unified symbol of the market to fetch OHLCV data for
939
+ :param str timeframe: the length of time each candle represents
940
+ :param int [since]: timestamp in ms of the earliest candle to fetch
941
+ :param int [limit]: the maximum amount of candles to fetch
942
+ :param dict [params]: extra parameters specific to the exchange API endpoint
943
+ :returns int[][]: A list of candles ordered, open, high, low, close, volume
944
+ """
945
+ await self.load_markets()
946
+ symbol = self.symbol(symbol)
947
+ market = self.market(symbol)
948
+ type = 'spot'
949
+ type, params = self.handle_market_type_and_params('watchOrderBook', market, params)
950
+ timeframes = self.safe_value(self.options, 'timeframes', {})
951
+ interval = self.safe_string(timeframes, timeframe)
952
+ name = None
953
+ if type == 'spot':
954
+ name = 'kline' + interval
955
+ else:
956
+ name = 'klineBin' + interval
957
+ ohlcv = await self.subscribe(name, symbol, type, params)
958
+ if self.newUpdates:
959
+ limit = ohlcv.getLimit(symbol, limit)
960
+ return self.filter_by_since_limit(ohlcv, since, limit, 0, True)
961
+
962
+ def handle_ohlcv(self, client: Client, message):
963
+ #
964
+ # {
965
+ # "data": [
966
+ # {
967
+ # "candle": [
968
+ # 1631056350,
969
+ # "46532.83",
970
+ # "46555.71",
971
+ # "46511.41",
972
+ # "46555.71",
973
+ # "0.25"
974
+ # ],
975
+ # "symbol": "BTC_USDT"
976
+ # }
977
+ # ],
978
+ # "table": "spot/kline1m"
979
+ # }
980
+ # swap
981
+ # {
982
+ # "group":"futures/klineBin1m:BTCUSDT",
983
+ # "data":{
984
+ # "symbol":"BTCUSDT",
985
+ # "items":[
986
+ # {
987
+ # "o":"39635.8",
988
+ # "h":"39636",
989
+ # "l":"39614.4",
990
+ # "c":"39629.7",
991
+ # "v":"31852",
992
+ # "ts":1701617761
993
+ # }
994
+ # ]
995
+ # }
996
+ # }
997
+ #
998
+ channel = self.safe_string_2(message, 'table', 'group')
999
+ isSpot = (channel.find('spot') >= 0)
1000
+ data = self.safe_value(message, 'data')
1001
+ if data is None:
1002
+ return
1003
+ parts = channel.split('/')
1004
+ part1 = self.safe_string(parts, 1, '')
1005
+ interval = part1.replace('kline', '')
1006
+ interval = interval.replace('Bin', '')
1007
+ intervalParts = interval.split(':')
1008
+ interval = self.safe_string(intervalParts, 0)
1009
+ # use a reverse lookup in a static map instead
1010
+ timeframes = self.safe_value(self.options, 'timeframes', {})
1011
+ timeframe = self.find_timeframe(interval, timeframes)
1012
+ duration = self.parse_timeframe(timeframe)
1013
+ durationInMs = duration * 1000
1014
+ if isSpot:
1015
+ for i in range(0, len(data)):
1016
+ marketId = self.safe_string(data[i], 'symbol')
1017
+ market = self.safe_market(marketId)
1018
+ symbol = market['symbol']
1019
+ rawOHLCV = self.safe_value(data[i], 'candle')
1020
+ parsed = self.parse_ohlcv(rawOHLCV, market)
1021
+ parsed[0] = self.parse_to_int(parsed[0] / durationInMs) * durationInMs
1022
+ self.ohlcvs[symbol] = self.safe_value(self.ohlcvs, symbol, {})
1023
+ stored = self.safe_value(self.ohlcvs[symbol], timeframe)
1024
+ if stored is None:
1025
+ limit = self.safe_integer(self.options, 'OHLCVLimit', 1000)
1026
+ stored = ArrayCacheByTimestamp(limit)
1027
+ self.ohlcvs[symbol][timeframe] = stored
1028
+ stored.append(parsed)
1029
+ messageHash = channel + ':' + marketId
1030
+ client.resolve(stored, messageHash)
1031
+ else:
1032
+ marketId = self.safe_string(data, 'symbol')
1033
+ market = self.safe_market(marketId, None, None, 'swap')
1034
+ symbol = market['symbol']
1035
+ items = self.safe_value(data, 'items', [])
1036
+ self.ohlcvs[symbol] = self.safe_value(self.ohlcvs, symbol, {})
1037
+ stored = self.safe_value(self.ohlcvs[symbol], timeframe)
1038
+ if stored is None:
1039
+ limit = self.safe_integer(self.options, 'OHLCVLimit', 1000)
1040
+ stored = ArrayCacheByTimestamp(limit)
1041
+ self.ohlcvs[symbol][timeframe] = stored
1042
+ for i in range(0, len(items)):
1043
+ candle = items[i]
1044
+ parsed = self.parse_ohlcv(candle, market)
1045
+ stored.append(parsed)
1046
+ client.resolve(stored, channel)
1047
+
1048
+ async def watch_order_book(self, symbol: str, limit: Int = None, params={}) -> OrderBook:
1049
+ """
1050
+ :see: https://developer-pro.bitmart.com/en/spot/#public-depth-all-channel
1051
+ :see: https://developer-pro.bitmart.com/en/spot/#public-depth-increase-channel
1052
+ :see: https://developer-pro.bitmart.com/en/futures/#public-depth-channel
1053
+ watches information on open orders with bid(buy) and ask(sell) prices, volumes and other data
1054
+ :param str symbol: unified symbol of the market to fetch the order book for
1055
+ :param int [limit]: the maximum amount of order book entries to return
1056
+ :param dict [params]: extra parameters specific to the exchange API endpoint
1057
+ :returns dict: A dictionary of `order book structures <https://docs.ccxt.com/#/?id=order-book-structure>` indexed by market symbols
1058
+ """
1059
+ await self.load_markets()
1060
+ options = self.safe_value(self.options, 'watchOrderBook', {})
1061
+ depth = self.safe_string(options, 'depth', 'depth/increase100')
1062
+ symbol = self.symbol(symbol)
1063
+ market = self.market(symbol)
1064
+ type = 'spot'
1065
+ type, params = self.handle_market_type_and_params('watchOrderBook', market, params)
1066
+ if type == 'swap' and depth == 'depth/increase100':
1067
+ depth = 'depth50'
1068
+ orderbook = await self.subscribe(depth, symbol, type, params)
1069
+ return orderbook.limit()
1070
+
1071
+ def handle_delta(self, bookside, delta):
1072
+ price = self.safe_float(delta, 0)
1073
+ amount = self.safe_float(delta, 1)
1074
+ bookside.store(price, amount)
1075
+
1076
+ def handle_deltas(self, bookside, deltas):
1077
+ for i in range(0, len(deltas)):
1078
+ self.handle_delta(bookside, deltas[i])
1079
+
1080
+ def handle_order_book_message(self, client: Client, message, orderbook):
1081
+ #
1082
+ # {
1083
+ # "asks": [
1084
+ # ['46828.38', "0.21847"],
1085
+ # ['46830.68', "0.08232"],
1086
+ # ['46832.08', "0.09285"],
1087
+ # ['46837.82', "0.02028"],
1088
+ # ['46839.43', "0.15068"]
1089
+ # ],
1090
+ # "bids": [
1091
+ # ['46820.78', "0.00444"],
1092
+ # ['46814.33', "0.00234"],
1093
+ # ['46813.50', "0.05021"],
1094
+ # ['46808.14', "0.00217"],
1095
+ # ['46808.04', "0.00013"]
1096
+ # ],
1097
+ # "ms_t": 1631044962431,
1098
+ # "symbol": "BTC_USDT"
1099
+ # }
1100
+ #
1101
+ asks = self.safe_list(message, 'asks', [])
1102
+ bids = self.safe_list(message, 'bids', [])
1103
+ self.handle_deltas(orderbook['asks'], asks)
1104
+ self.handle_deltas(orderbook['bids'], bids)
1105
+ timestamp = self.safe_integer(message, 'ms_t')
1106
+ marketId = self.safe_string(message, 'symbol')
1107
+ symbol = self.safe_symbol(marketId)
1108
+ orderbook['symbol'] = symbol
1109
+ orderbook['timestamp'] = timestamp
1110
+ orderbook['datetime'] = self.iso8601(timestamp)
1111
+ return orderbook
1112
+
1113
+ def handle_order_book(self, client: Client, message):
1114
+ #
1115
+ # spot depth-all
1116
+ #
1117
+ # {
1118
+ # "data": [
1119
+ # {
1120
+ # "asks": [
1121
+ # ['46828.38', "0.21847"],
1122
+ # ['46830.68', "0.08232"],
1123
+ # ...
1124
+ # ],
1125
+ # "bids": [
1126
+ # ['46820.78', "0.00444"],
1127
+ # ['46814.33', "0.00234"],
1128
+ # ...
1129
+ # ],
1130
+ # "ms_t": 1631044962431,
1131
+ # "symbol": "BTC_USDT"
1132
+ # }
1133
+ # ],
1134
+ # "table": "spot/depth5"
1135
+ # }
1136
+ #
1137
+ # spot increse depth snapshot
1138
+ #
1139
+ # {
1140
+ # "data":[
1141
+ # {
1142
+ # "asks":[
1143
+ # ["43652.52", "0.02039"],
1144
+ # ...
1145
+ # ],
1146
+ # "bids":[
1147
+ # ["43652.51", "0.00500"],
1148
+ # ...
1149
+ # ],
1150
+ # "ms_t":1703376836487,
1151
+ # "symbol":"BTC_USDT",
1152
+ # "type":"snapshot", # or update
1153
+ # "version":2141731
1154
+ # }
1155
+ # ],
1156
+ # "table":"spot/depth/increase100"
1157
+ # }
1158
+ #
1159
+ # swap
1160
+ #
1161
+ # {
1162
+ # "group":"futures/depth50:BTCUSDT",
1163
+ # "data":{
1164
+ # "symbol":"BTCUSDT",
1165
+ # "way":1,
1166
+ # "depths":[
1167
+ # {
1168
+ # "price":"39509.8",
1169
+ # "vol":"2379"
1170
+ # },
1171
+ # {
1172
+ # "price":"39509.6",
1173
+ # "vol":"6815"
1174
+ # },
1175
+ # ...
1176
+ # ],
1177
+ # "ms_t":1701566021194
1178
+ # }
1179
+ # }
1180
+ #
1181
+ isSpot = ('table' in message)
1182
+ datas = []
1183
+ if isSpot:
1184
+ datas = self.safe_list(message, 'data', datas)
1185
+ else:
1186
+ orderBookEntry = self.safe_dict(message, 'data')
1187
+ if orderBookEntry is not None:
1188
+ datas.append(orderBookEntry)
1189
+ length = len(datas)
1190
+ if length <= 0:
1191
+ return
1192
+ channelName = self.safe_string_2(message, 'table', 'group')
1193
+ # find limit subscribed to
1194
+ limitsToCheck = ['100', '50', '20', '10', '5']
1195
+ limit = 0
1196
+ for i in range(0, len(limitsToCheck)):
1197
+ limitString = limitsToCheck[i]
1198
+ if channelName.find(limitString) >= 0:
1199
+ limit = self.parse_to_int(limitString)
1200
+ break
1201
+ if isSpot:
1202
+ channel = channelName.replace('spot/', '')
1203
+ for i in range(0, len(datas)):
1204
+ update = datas[i]
1205
+ marketId = self.safe_string(update, 'symbol')
1206
+ symbol = self.safe_symbol(marketId)
1207
+ if not (symbol in self.orderbooks):
1208
+ ob = self.order_book({}, limit)
1209
+ ob['symbol'] = symbol
1210
+ self.orderbooks[symbol] = ob
1211
+ orderbook = self.orderbooks[symbol]
1212
+ type = self.safe_string(update, 'type')
1213
+ if (type == 'snapshot') or (not(channelName.find('increase') >= 0)):
1214
+ orderbook.reset({})
1215
+ self.handle_order_book_message(client, update, orderbook)
1216
+ timestamp = self.safe_integer(update, 'ms_t')
1217
+ if orderbook['timestamp'] is None:
1218
+ orderbook['timestamp'] = timestamp
1219
+ orderbook['datetime'] = self.iso8601(timestamp)
1220
+ messageHash = channelName + ':' + marketId
1221
+ client.resolve(orderbook, messageHash)
1222
+ # resolve ForSymbols
1223
+ messageHashForMulti = channel + ':' + symbol
1224
+ client.resolve(orderbook, messageHashForMulti)
1225
+ else:
1226
+ tableParts = channelName.split(':')
1227
+ channel = tableParts[0].replace('futures/', '')
1228
+ data = datas[0] # contract markets always contain only one member
1229
+ depths = data['depths']
1230
+ marketId = self.safe_string(data, 'symbol')
1231
+ symbol = self.safe_symbol(marketId)
1232
+ if not (symbol in self.orderbooks):
1233
+ ob = self.order_book({}, limit)
1234
+ ob['symbol'] = symbol
1235
+ self.orderbooks[symbol] = ob
1236
+ orderbook = self.orderbooks[symbol]
1237
+ way = self.safe_number(data, 'way')
1238
+ side = 'bids' if (way == 1) else 'asks'
1239
+ if way == 1:
1240
+ orderbook[side] = Bids([], limit)
1241
+ else:
1242
+ orderbook[side] = Asks([], limit)
1243
+ for i in range(0, len(depths)):
1244
+ depth = depths[i]
1245
+ price = self.safe_number(depth, 'price')
1246
+ amount = self.safe_number(depth, 'vol')
1247
+ orderbookSide = self.safe_value(orderbook, side)
1248
+ orderbookSide.store(price, amount)
1249
+ bidsLength = len(orderbook['bids'])
1250
+ asksLength = len(orderbook['asks'])
1251
+ if (bidsLength == 0) or (asksLength == 0):
1252
+ return
1253
+ timestamp = self.safe_integer(data, 'ms_t')
1254
+ orderbook['timestamp'] = timestamp
1255
+ orderbook['datetime'] = self.iso8601(timestamp)
1256
+ messageHash = channelName
1257
+ client.resolve(orderbook, messageHash)
1258
+ # resolve ForSymbols
1259
+ messageHashForMulti = channel + ':' + symbol
1260
+ client.resolve(orderbook, messageHashForMulti)
1261
+
1262
+ async def watch_order_book_for_symbols(self, symbols: List[str], limit: Int = None, params={}) -> OrderBook:
1263
+ """
1264
+ watches information on open orders with bid(buy) and ask(sell) prices, volumes and other data
1265
+ :see: https://developer-pro.bitmart.com/en/spot/#public-depth-increase-channel
1266
+ :param str[] symbols: unified array of symbols
1267
+ :param int [limit]: the maximum amount of order book entries to return
1268
+ :param dict [params]: extra parameters specific to the exchange API endpoint
1269
+ :param str [params.depth]: the type of order book to subscribe to, default is 'depth/increase100', also accepts 'depth5' or 'depth20' or depth50
1270
+ :returns dict: A dictionary of `order book structures <https://docs.ccxt.com/#/?id=order-book-structure>` indexed by market symbols
1271
+ """
1272
+ await self.load_markets()
1273
+ type = None
1274
+ symbols, type, params = self.get_params_for_multiple_sub('watchOrderBookForSymbols', symbols, limit, params)
1275
+ channel = None
1276
+ channel, params = self.handle_option_and_params(params, 'watchOrderBookForSymbols', 'depth', 'depth/increase100')
1277
+ if type == 'swap' and channel == 'depth/increase100':
1278
+ channel = 'depth50'
1279
+ orderbook = await self.subscribe_multiple(channel, type, symbols, params)
1280
+ return orderbook.limit()
1281
+
1282
+ async def authenticate(self, type, params={}):
1283
+ self.check_required_credentials()
1284
+ url = self.implode_hostname(self.urls['api']['ws'][type]['private'])
1285
+ messageHash = 'authenticated'
1286
+ client = self.client(url)
1287
+ future = client.future(messageHash)
1288
+ authenticated = self.safe_value(client.subscriptions, messageHash)
1289
+ if authenticated is None:
1290
+ timestamp = str(self.milliseconds())
1291
+ memo = self.uid
1292
+ path = 'bitmart.WebSocket'
1293
+ auth = timestamp + '#' + memo + '#' + path
1294
+ signature = self.hmac(self.encode(auth), self.encode(self.secret), hashlib.sha256)
1295
+ request = None
1296
+ if type == 'spot':
1297
+ request = {
1298
+ 'op': 'login',
1299
+ 'args': [
1300
+ self.apiKey,
1301
+ timestamp,
1302
+ signature,
1303
+ ],
1304
+ }
1305
+ else:
1306
+ request = {
1307
+ 'action': 'access',
1308
+ 'args': [
1309
+ self.apiKey,
1310
+ timestamp,
1311
+ signature,
1312
+ 'web',
1313
+ ],
1314
+ }
1315
+ message = self.extend(request, params)
1316
+ self.watch(url, messageHash, message, messageHash)
1317
+ return await future
1318
+
1319
+ def handle_subscription_status(self, client: Client, message):
1320
+ #
1321
+ # {"event":"subscribe","channel":"spot/depth:BTC-USDT"}
1322
+ #
1323
+ return message
1324
+
1325
+ def handle_authenticate(self, client: Client, message):
1326
+ #
1327
+ # spot
1328
+ # {event: "login"}
1329
+ # swap
1330
+ # {action: 'access', success: True}
1331
+ #
1332
+ messageHash = 'authenticated'
1333
+ future = self.safe_value(client.futures, messageHash)
1334
+ future.resolve(True)
1335
+
1336
+ def handle_error_message(self, client: Client, message):
1337
+ #
1338
+ # {event: "error", message: "Invalid sign", errorCode: 30013}
1339
+ # {"event":"error","message":"Unrecognized request: {\"event\":\"subscribe\",\"channel\":\"spot/depth:BTC-USDT\"}","errorCode":30039}
1340
+ # {
1341
+ # action: '',
1342
+ # group: 'futures/trade:BTCUSDT',
1343
+ # success: False,
1344
+ # request: {action: '', args: ['futures/trade:BTCUSDT']},
1345
+ # error: 'Invalid action [] for group [futures/trade:BTCUSDT]'
1346
+ # }
1347
+ #
1348
+ errorCode = self.safe_string(message, 'errorCode')
1349
+ error = self.safe_string(message, 'error')
1350
+ try:
1351
+ if errorCode is not None or error is not None:
1352
+ feedback = self.id + ' ' + self.json(message)
1353
+ self.throw_exactly_matched_exception(self.exceptions['exact'], errorCode, feedback)
1354
+ messageString = self.safe_value(message, 'message', error)
1355
+ self.throw_broadly_matched_exception(self.exceptions['broad'], messageString, feedback)
1356
+ action = self.safe_string(message, 'action')
1357
+ if action == 'access':
1358
+ raise AuthenticationError(feedback)
1359
+ raise ExchangeError(feedback)
1360
+ return False
1361
+ except Exception as e:
1362
+ if (isinstance(e, AuthenticationError)):
1363
+ messageHash = 'authenticated'
1364
+ client.reject(e, messageHash)
1365
+ if messageHash in client.subscriptions:
1366
+ del client.subscriptions[messageHash]
1367
+ client.reject(e)
1368
+ return True
1369
+
1370
+ def handle_message(self, client: Client, message):
1371
+ if self.handle_error_message(client, message):
1372
+ return
1373
+ #
1374
+ # {"event":"error","message":"Unrecognized request: {\"event\":\"subscribe\",\"channel\":\"spot/depth:BTC-USDT\"}","errorCode":30039}
1375
+ #
1376
+ # subscribe events on spot:
1377
+ #
1378
+ # {"event":"subscribe", "topic":"spot/kline1m:BTC_USDT"}
1379
+ #
1380
+ # subscribe on contracts:
1381
+ #
1382
+ # {"action":"subscribe", "group":"futures/klineBin1m:BTCUSDT", "success":true, "request":{"action":"subscribe", "args":["futures/klineBin1m:BTCUSDT"]}}
1383
+ #
1384
+ # regular updates - spot
1385
+ #
1386
+ # {
1387
+ # "table": "spot/depth",
1388
+ # "action": "partial",
1389
+ # "data": [
1390
+ # {
1391
+ # "instrument_id": "BTC-USDT",
1392
+ # "asks": [
1393
+ # ["5301.8", "0.03763319", "1"],
1394
+ # ["5302.4", "0.00305", "2"],
1395
+ # ],
1396
+ # "bids": [
1397
+ # ["5301.7", "0.58911427", "6"],
1398
+ # ["5301.6", "0.01222922", "4"],
1399
+ # ],
1400
+ # "timestamp": "2020-03-16T03:25:00.440Z",
1401
+ # "checksum": -2088736623
1402
+ # }
1403
+ # ]
1404
+ # }
1405
+ #
1406
+ # regular updates - contracts
1407
+ #
1408
+ # {
1409
+ # group: "futures/klineBin1m:BTCUSDT",
1410
+ # data: {
1411
+ # symbol: "BTCUSDT",
1412
+ # items: [{o: "67944.7", "h": ....}],
1413
+ # },
1414
+ # }
1415
+ #
1416
+ # {data: '', table: "spot/user/order"}
1417
+ #
1418
+ # the only realiable way(for both spot & swap) is to check 'data' key
1419
+ isDataUpdate = ('data' in message)
1420
+ if not isDataUpdate:
1421
+ event = self.safe_string_2(message, 'event', 'action')
1422
+ if event is not None:
1423
+ methods: dict = {
1424
+ # 'info': self.handleSystemStatus,
1425
+ 'login': self.handle_authenticate,
1426
+ 'access': self.handle_authenticate,
1427
+ 'subscribe': self.handle_subscription_status,
1428
+ }
1429
+ method = self.safe_value(methods, event)
1430
+ if method is not None:
1431
+ method(client, message)
1432
+ else:
1433
+ channel = self.safe_string_2(message, 'table', 'group')
1434
+ methods: dict = {
1435
+ 'depth': self.handle_order_book,
1436
+ 'ticker': self.handle_ticker,
1437
+ 'trade': self.handle_trade,
1438
+ 'kline': self.handle_ohlcv,
1439
+ 'order': self.handle_orders,
1440
+ 'position': self.handle_positions,
1441
+ 'balance': self.handle_balance,
1442
+ 'asset': self.handle_balance,
1443
+ }
1444
+ keys = list(methods.keys())
1445
+ for i in range(0, len(keys)):
1446
+ key = keys[i]
1447
+ if channel.find(key) >= 0:
1448
+ method = self.safe_value(methods, key)
1449
+ method(client, message)