ccxt 4.2.10 → 4.2.12

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 (383) hide show
  1. package/README.md +5 -5
  2. package/build.sh +4 -4
  3. package/dist/ccxt.browser.js +695 -282
  4. package/dist/ccxt.browser.min.js +3 -3
  5. package/dist/cjs/_virtual/agent.js +7 -0
  6. package/dist/cjs/_virtual/parse-proxy-response.js +7 -0
  7. package/dist/cjs/_virtual/promisify.js +7 -0
  8. package/dist/cjs/ccxt.js +1 -1
  9. package/dist/cjs/js/ccxt.js +474 -0
  10. package/dist/cjs/js/src/abstract/ace.js +9 -0
  11. package/dist/cjs/js/src/abstract/alpaca.js +9 -0
  12. package/dist/cjs/js/src/abstract/ascendex.js +9 -0
  13. package/dist/cjs/js/src/abstract/bigone.js +9 -0
  14. package/dist/cjs/js/src/abstract/binance.js +9 -0
  15. package/dist/cjs/js/src/abstract/bingx.js +9 -0
  16. package/dist/cjs/js/src/abstract/bit2c.js +9 -0
  17. package/dist/cjs/js/src/abstract/bitbank.js +9 -0
  18. package/dist/cjs/js/src/abstract/bitbns.js +9 -0
  19. package/dist/cjs/js/src/abstract/bitfinex.js +9 -0
  20. package/dist/cjs/js/src/abstract/bitfinex2.js +9 -0
  21. package/dist/cjs/js/src/abstract/bitflyer.js +9 -0
  22. package/dist/cjs/js/src/abstract/bitforex.js +9 -0
  23. package/dist/cjs/js/src/abstract/bitget.js +9 -0
  24. package/dist/cjs/js/src/abstract/bithumb.js +9 -0
  25. package/dist/cjs/js/src/abstract/bitmart.js +9 -0
  26. package/dist/cjs/js/src/abstract/bitmex.js +9 -0
  27. package/dist/cjs/js/src/abstract/bitopro.js +9 -0
  28. package/dist/cjs/js/src/abstract/bitpanda.js +9 -0
  29. package/dist/cjs/js/src/abstract/bitrue.js +9 -0
  30. package/dist/cjs/js/src/abstract/bitso.js +9 -0
  31. package/dist/cjs/js/src/abstract/bitstamp.js +9 -0
  32. package/dist/cjs/js/src/abstract/bitteam.js +9 -0
  33. package/dist/cjs/js/src/abstract/bitvavo.js +9 -0
  34. package/dist/cjs/js/src/abstract/bl3p.js +9 -0
  35. package/dist/cjs/js/src/abstract/blockchaincom.js +9 -0
  36. package/dist/cjs/js/src/abstract/btcalpha.js +9 -0
  37. package/dist/cjs/js/src/abstract/btcbox.js +9 -0
  38. package/dist/cjs/js/src/abstract/btcmarkets.js +9 -0
  39. package/dist/cjs/js/src/abstract/btcturk.js +9 -0
  40. package/dist/cjs/js/src/abstract/bybit.js +9 -0
  41. package/dist/cjs/js/src/abstract/cex.js +9 -0
  42. package/dist/cjs/js/src/abstract/coinbase.js +9 -0
  43. package/dist/cjs/js/src/abstract/coinbasepro.js +9 -0
  44. package/dist/cjs/js/src/abstract/coincheck.js +9 -0
  45. package/dist/cjs/js/src/abstract/coinex.js +9 -0
  46. package/dist/cjs/js/src/abstract/coinlist.js +9 -0
  47. package/dist/cjs/js/src/abstract/coinmate.js +9 -0
  48. package/dist/cjs/js/src/abstract/coinone.js +9 -0
  49. package/dist/cjs/js/src/abstract/coinsph.js +9 -0
  50. package/dist/cjs/js/src/abstract/coinspot.js +9 -0
  51. package/dist/cjs/js/src/abstract/cryptocom.js +9 -0
  52. package/dist/cjs/js/src/abstract/currencycom.js +9 -0
  53. package/dist/cjs/js/src/abstract/delta.js +9 -0
  54. package/dist/cjs/js/src/abstract/deribit.js +9 -0
  55. package/dist/cjs/js/src/abstract/digifinex.js +9 -0
  56. package/dist/cjs/js/src/abstract/exmo.js +9 -0
  57. package/dist/cjs/js/src/abstract/gate.js +9 -0
  58. package/dist/cjs/js/src/abstract/gemini.js +9 -0
  59. package/dist/cjs/js/src/abstract/hitbtc.js +9 -0
  60. package/dist/cjs/js/src/abstract/hollaex.js +9 -0
  61. package/dist/cjs/js/src/abstract/htx.js +9 -0
  62. package/dist/cjs/js/src/abstract/huobijp.js +9 -0
  63. package/dist/cjs/js/src/abstract/idex.js +9 -0
  64. package/dist/cjs/js/src/abstract/independentreserve.js +9 -0
  65. package/dist/cjs/js/src/abstract/indodax.js +9 -0
  66. package/dist/cjs/js/src/abstract/kraken.js +9 -0
  67. package/dist/cjs/js/src/abstract/krakenfutures.js +9 -0
  68. package/dist/cjs/js/src/abstract/kucoin.js +9 -0
  69. package/dist/cjs/js/src/abstract/kucoinfutures.js +9 -0
  70. package/dist/cjs/js/src/abstract/kuna.js +9 -0
  71. package/dist/cjs/js/src/abstract/latoken.js +9 -0
  72. package/dist/cjs/js/src/abstract/lbank.js +9 -0
  73. package/dist/cjs/js/src/abstract/luno.js +9 -0
  74. package/dist/cjs/js/src/abstract/lykke.js +9 -0
  75. package/dist/cjs/js/src/abstract/mercado.js +9 -0
  76. package/dist/cjs/js/src/abstract/mexc.js +9 -0
  77. package/dist/cjs/js/src/abstract/ndax.js +9 -0
  78. package/dist/cjs/js/src/abstract/novadax.js +9 -0
  79. package/dist/cjs/js/src/abstract/oceanex.js +9 -0
  80. package/dist/cjs/js/src/abstract/okcoin.js +9 -0
  81. package/dist/cjs/js/src/abstract/okx.js +9 -0
  82. package/dist/cjs/js/src/abstract/p2b.js +9 -0
  83. package/dist/cjs/js/src/abstract/paymium.js +9 -0
  84. package/dist/cjs/js/src/abstract/phemex.js +9 -0
  85. package/dist/cjs/js/src/abstract/poloniex.js +9 -0
  86. package/dist/cjs/js/src/abstract/poloniexfutures.js +9 -0
  87. package/dist/cjs/js/src/abstract/probit.js +9 -0
  88. package/dist/cjs/js/src/abstract/timex.js +9 -0
  89. package/dist/cjs/js/src/abstract/tokocrypto.js +9 -0
  90. package/dist/cjs/js/src/abstract/upbit.js +9 -0
  91. package/dist/cjs/js/src/abstract/wavesexchange.js +9 -0
  92. package/dist/cjs/js/src/abstract/wazirx.js +9 -0
  93. package/dist/cjs/js/src/abstract/whitebit.js +9 -0
  94. package/dist/cjs/js/src/abstract/woo.js +9 -0
  95. package/dist/cjs/js/src/abstract/yobit.js +9 -0
  96. package/dist/cjs/js/src/abstract/zaif.js +9 -0
  97. package/dist/cjs/js/src/abstract/zonda.js +9 -0
  98. package/dist/cjs/js/src/ace.js +1058 -0
  99. package/dist/cjs/js/src/alpaca.js +1125 -0
  100. package/dist/cjs/js/src/ascendex.js +3360 -0
  101. package/dist/cjs/js/src/base/Exchange.js +5110 -0
  102. package/dist/cjs/js/src/base/Precise.js +263 -0
  103. package/dist/cjs/js/src/base/errors.js +299 -0
  104. package/dist/cjs/js/src/base/functions/crypto.js +78 -0
  105. package/dist/cjs/js/src/base/functions/encode.js +44 -0
  106. package/dist/cjs/js/src/base/functions/generic.js +193 -0
  107. package/dist/cjs/js/src/base/functions/misc.js +96 -0
  108. package/dist/cjs/js/src/base/functions/number.js +297 -0
  109. package/dist/cjs/js/src/base/functions/platform.js +28 -0
  110. package/dist/cjs/js/src/base/functions/rsa.js +34 -0
  111. package/dist/cjs/js/src/base/functions/string.js +48 -0
  112. package/dist/cjs/js/src/base/functions/throttle.js +66 -0
  113. package/dist/cjs/js/src/base/functions/time.js +187 -0
  114. package/dist/cjs/js/src/base/functions/totp.js +24 -0
  115. package/dist/cjs/js/src/base/functions/type.js +162 -0
  116. package/dist/cjs/js/src/base/functions.js +157 -0
  117. package/dist/cjs/js/src/base/ws/Cache.js +254 -0
  118. package/dist/cjs/js/src/base/ws/Client.js +299 -0
  119. package/dist/cjs/js/src/base/ws/Future.js +34 -0
  120. package/dist/cjs/js/src/base/ws/OrderBook.js +107 -0
  121. package/dist/cjs/js/src/base/ws/OrderBookSide.js +281 -0
  122. package/dist/cjs/js/src/base/ws/WsClient.js +69 -0
  123. package/dist/cjs/js/src/bequant.js +33 -0
  124. package/dist/cjs/js/src/bigone.js +2142 -0
  125. package/dist/cjs/js/src/binance.js +9729 -0
  126. package/dist/cjs/js/src/binancecoinm.js +45 -0
  127. package/dist/cjs/js/src/binanceus.js +84 -0
  128. package/dist/cjs/js/src/binanceusdm.js +58 -0
  129. package/dist/cjs/js/src/bingx.js +3737 -0
  130. package/dist/cjs/js/src/bit2c.js +916 -0
  131. package/dist/cjs/js/src/bitbank.js +1000 -0
  132. package/dist/cjs/js/src/bitbay.js +17 -0
  133. package/dist/cjs/js/src/bitbns.js +1220 -0
  134. package/dist/cjs/js/src/bitcoincom.js +17 -0
  135. package/dist/cjs/js/src/bitfinex.js +1670 -0
  136. package/dist/cjs/js/src/bitfinex2.js +2990 -0
  137. package/dist/cjs/js/src/bitflyer.js +1045 -0
  138. package/dist/cjs/js/src/bitforex.js +852 -0
  139. package/dist/cjs/js/src/bitget.js +8284 -0
  140. package/dist/cjs/js/src/bithumb.js +1090 -0
  141. package/dist/cjs/js/src/bitmart.js +4454 -0
  142. package/dist/cjs/js/src/bitmex.js +2884 -0
  143. package/dist/cjs/js/src/bitopro.js +1724 -0
  144. package/dist/cjs/js/src/bitpanda.js +2002 -0
  145. package/dist/cjs/js/src/bitrue.js +3253 -0
  146. package/dist/cjs/js/src/bitso.js +1753 -0
  147. package/dist/cjs/js/src/bitstamp.js +2188 -0
  148. package/dist/cjs/js/src/bitteam.js +2309 -0
  149. package/dist/cjs/js/src/bitvavo.js +1968 -0
  150. package/dist/cjs/js/src/bl3p.js +447 -0
  151. package/dist/cjs/js/src/blockchaincom.js +1160 -0
  152. package/dist/cjs/js/src/btcalpha.js +929 -0
  153. package/dist/cjs/js/src/btcbox.js +565 -0
  154. package/dist/cjs/js/src/btcmarkets.js +1237 -0
  155. package/dist/cjs/js/src/btcturk.js +929 -0
  156. package/dist/cjs/js/src/bybit.js +7646 -0
  157. package/dist/cjs/js/src/cex.js +1693 -0
  158. package/dist/cjs/js/src/coinbase.js +3274 -0
  159. package/dist/cjs/js/src/coinbasepro.js +1866 -0
  160. package/dist/cjs/js/src/coincheck.js +843 -0
  161. package/dist/cjs/js/src/coinex.js +5414 -0
  162. package/dist/cjs/js/src/coinlist.js +2329 -0
  163. package/dist/cjs/js/src/coinmate.js +989 -0
  164. package/dist/cjs/js/src/coinone.js +1185 -0
  165. package/dist/cjs/js/src/coinsph.js +1933 -0
  166. package/dist/cjs/js/src/coinspot.js +548 -0
  167. package/dist/cjs/js/src/cryptocom.js +3007 -0
  168. package/dist/cjs/js/src/currencycom.js +2015 -0
  169. package/dist/cjs/js/src/delta.js +3256 -0
  170. package/dist/cjs/js/src/deribit.js +3306 -0
  171. package/dist/cjs/js/src/digifinex.js +4307 -0
  172. package/dist/cjs/js/src/exmo.js +2645 -0
  173. package/dist/cjs/js/src/fmfwio.js +34 -0
  174. package/dist/cjs/js/src/gate.js +7054 -0
  175. package/dist/cjs/js/src/gateio.js +16 -0
  176. package/dist/cjs/js/src/gemini.js +1801 -0
  177. package/dist/cjs/js/src/hitbtc.js +3660 -0
  178. package/dist/cjs/js/src/hitbtc3.js +19 -0
  179. package/dist/cjs/js/src/hollaex.js +1882 -0
  180. package/dist/cjs/js/src/htx.js +9024 -0
  181. package/dist/cjs/js/src/huobi.js +16 -0
  182. package/dist/cjs/js/src/huobijp.js +1918 -0
  183. package/dist/cjs/js/src/idex.js +1770 -0
  184. package/dist/cjs/js/src/independentreserve.js +759 -0
  185. package/dist/cjs/js/src/indodax.js +1069 -0
  186. package/dist/cjs/js/src/kraken.js +2861 -0
  187. package/dist/cjs/js/src/krakenfutures.js +2407 -0
  188. package/dist/cjs/js/src/kucoin.js +4489 -0
  189. package/dist/cjs/js/src/kucoinfutures.js +2475 -0
  190. package/dist/cjs/js/src/kuna.js +1949 -0
  191. package/dist/cjs/js/src/latoken.js +1729 -0
  192. package/dist/cjs/js/src/lbank.js +2825 -0
  193. package/dist/cjs/js/src/luno.js +1044 -0
  194. package/dist/cjs/js/src/lykke.js +1303 -0
  195. package/dist/cjs/js/src/mercado.js +897 -0
  196. package/dist/cjs/js/src/mexc.js +5407 -0
  197. package/dist/cjs/js/src/ndax.js +2450 -0
  198. package/dist/cjs/js/src/novadax.js +1556 -0
  199. package/dist/cjs/js/src/oceanex.js +964 -0
  200. package/dist/cjs/js/src/okcoin.js +3115 -0
  201. package/dist/cjs/js/src/okx.js +7329 -0
  202. package/dist/cjs/js/src/p2b.js +1243 -0
  203. package/dist/cjs/js/src/paymium.js +597 -0
  204. package/dist/cjs/js/src/phemex.js +4715 -0
  205. package/dist/cjs/js/src/poloniex.js +2356 -0
  206. package/dist/cjs/js/src/poloniexfutures.js +1794 -0
  207. package/dist/cjs/js/src/pro/alpaca.js +714 -0
  208. package/dist/cjs/js/src/pro/ascendex.js +957 -0
  209. package/dist/cjs/js/src/pro/bequant.js +33 -0
  210. package/dist/cjs/js/src/pro/binance.js +2796 -0
  211. package/dist/cjs/js/src/pro/binancecoinm.js +23 -0
  212. package/dist/cjs/js/src/pro/binanceus.js +51 -0
  213. package/dist/cjs/js/src/pro/binanceusdm.js +32 -0
  214. package/dist/cjs/js/src/pro/bingx.js +944 -0
  215. package/dist/cjs/js/src/pro/bitcoincom.js +29 -0
  216. package/dist/cjs/js/src/pro/bitfinex.js +672 -0
  217. package/dist/cjs/js/src/pro/bitfinex2.js +1159 -0
  218. package/dist/cjs/js/src/pro/bitget.js +1733 -0
  219. package/dist/cjs/js/src/pro/bitmart.js +1486 -0
  220. package/dist/cjs/js/src/pro/bitmex.js +1576 -0
  221. package/dist/cjs/js/src/pro/bitopro.js +327 -0
  222. package/dist/cjs/js/src/pro/bitpanda.js +1341 -0
  223. package/dist/cjs/js/src/pro/bitrue.js +462 -0
  224. package/dist/cjs/js/src/pro/bitstamp.js +547 -0
  225. package/dist/cjs/js/src/pro/bitvavo.js +704 -0
  226. package/dist/cjs/js/src/pro/blockchaincom.js +794 -0
  227. package/dist/cjs/js/src/pro/bybit.js +1843 -0
  228. package/dist/cjs/js/src/pro/cex.js +1510 -0
  229. package/dist/cjs/js/src/pro/coinbase.js +561 -0
  230. package/dist/cjs/js/src/pro/coinbasepro.js +968 -0
  231. package/dist/cjs/js/src/pro/coinex.js +1095 -0
  232. package/dist/cjs/js/src/pro/cryptocom.js +1020 -0
  233. package/dist/cjs/js/src/pro/currencycom.js +563 -0
  234. package/dist/cjs/js/src/pro/deribit.js +825 -0
  235. package/dist/cjs/js/src/pro/exmo.js +658 -0
  236. package/dist/cjs/js/src/pro/gate.js +1316 -0
  237. package/dist/cjs/js/src/pro/gateio.js +16 -0
  238. package/dist/cjs/js/src/pro/gemini.js +649 -0
  239. package/dist/cjs/js/src/pro/hitbtc.js +1293 -0
  240. package/dist/cjs/js/src/pro/hollaex.js +597 -0
  241. package/dist/cjs/js/src/pro/htx.js +2383 -0
  242. package/dist/cjs/js/src/pro/huobi.js +16 -0
  243. package/dist/cjs/js/src/pro/huobijp.js +606 -0
  244. package/dist/cjs/js/src/pro/idex.js +714 -0
  245. package/dist/cjs/js/src/pro/independentreserve.js +280 -0
  246. package/dist/cjs/js/src/pro/kraken.js +1364 -0
  247. package/dist/cjs/js/src/pro/krakenfutures.js +1500 -0
  248. package/dist/cjs/js/src/pro/kucoin.js +1052 -0
  249. package/dist/cjs/js/src/pro/kucoinfutures.js +981 -0
  250. package/dist/cjs/js/src/pro/luno.js +322 -0
  251. package/dist/cjs/js/src/pro/mexc.js +1170 -0
  252. package/dist/cjs/js/src/pro/ndax.js +545 -0
  253. package/dist/cjs/js/src/pro/okcoin.js +760 -0
  254. package/dist/cjs/js/src/pro/okx.js +1558 -0
  255. package/dist/cjs/js/src/pro/phemex.js +1511 -0
  256. package/dist/cjs/js/src/pro/poloniex.js +1253 -0
  257. package/dist/cjs/js/src/pro/poloniexfutures.js +1014 -0
  258. package/dist/cjs/js/src/pro/probit.js +586 -0
  259. package/dist/cjs/js/src/pro/upbit.js +234 -0
  260. package/dist/cjs/js/src/pro/wazirx.js +776 -0
  261. package/dist/cjs/js/src/pro/whitebit.js +927 -0
  262. package/dist/cjs/js/src/pro/woo.js +769 -0
  263. package/dist/cjs/js/src/probit.js +1865 -0
  264. package/dist/cjs/js/src/static_dependencies/fflake/browser.js +401 -0
  265. package/dist/cjs/js/src/static_dependencies/jsencrypt/JSEncrypt.js +195 -0
  266. package/dist/cjs/js/src/static_dependencies/jsencrypt/JSEncryptRSAKey.js +308 -0
  267. package/dist/cjs/js/src/static_dependencies/jsencrypt/lib/asn1js/asn1.js +554 -0
  268. package/dist/cjs/js/src/static_dependencies/jsencrypt/lib/asn1js/base64.js +94 -0
  269. package/dist/cjs/js/src/static_dependencies/jsencrypt/lib/asn1js/hex.js +70 -0
  270. package/dist/cjs/js/src/static_dependencies/jsencrypt/lib/asn1js/int10.js +91 -0
  271. package/dist/cjs/js/src/static_dependencies/jsencrypt/lib/jsbn/base64.js +16 -0
  272. package/dist/cjs/js/src/static_dependencies/jsencrypt/lib/jsbn/jsbn.js +1760 -0
  273. package/dist/cjs/js/src/static_dependencies/jsencrypt/lib/jsbn/prng4.js +52 -0
  274. package/dist/cjs/js/src/static_dependencies/jsencrypt/lib/jsbn/rng.js +81 -0
  275. package/dist/cjs/js/src/static_dependencies/jsencrypt/lib/jsbn/rsa.js +376 -0
  276. package/dist/cjs/js/src/static_dependencies/jsencrypt/lib/jsbn/util.js +70 -0
  277. package/dist/cjs/js/src/static_dependencies/jsencrypt/lib/jsrsasign/asn1-1.0.js +1580 -0
  278. package/dist/cjs/js/src/static_dependencies/jsencrypt/lib/jsrsasign/yahoo.js +74 -0
  279. package/dist/cjs/js/src/static_dependencies/noble-curves/_shortw_utils.js +24 -0
  280. package/dist/cjs/js/src/static_dependencies/noble-curves/abstract/curve.js +158 -0
  281. package/dist/cjs/js/src/static_dependencies/noble-curves/abstract/edwards.js +429 -0
  282. package/dist/cjs/js/src/static_dependencies/noble-curves/abstract/hash-to-curve.js +176 -0
  283. package/dist/cjs/js/src/static_dependencies/noble-curves/abstract/modular.js +324 -0
  284. package/dist/cjs/js/src/static_dependencies/noble-curves/abstract/montgomery.js +163 -0
  285. package/dist/cjs/js/src/static_dependencies/noble-curves/abstract/utils.js +245 -0
  286. package/dist/cjs/js/src/static_dependencies/noble-curves/abstract/weierstrass.js +1018 -0
  287. package/dist/cjs/js/src/static_dependencies/noble-curves/ed25519.js +383 -0
  288. package/dist/cjs/js/src/static_dependencies/noble-curves/secp256k1.js +258 -0
  289. package/dist/cjs/js/src/static_dependencies/noble-hashes/_assert.js +53 -0
  290. package/dist/cjs/js/src/static_dependencies/noble-hashes/_sha2.js +120 -0
  291. package/dist/cjs/js/src/static_dependencies/noble-hashes/_u64.js +69 -0
  292. package/dist/cjs/js/src/static_dependencies/noble-hashes/crypto.js +7 -0
  293. package/dist/cjs/js/src/static_dependencies/noble-hashes/hmac.js +83 -0
  294. package/dist/cjs/js/src/static_dependencies/noble-hashes/md5.js +240 -0
  295. package/dist/cjs/js/src/static_dependencies/noble-hashes/sha1.js +91 -0
  296. package/dist/cjs/js/src/static_dependencies/noble-hashes/sha256.js +130 -0
  297. package/dist/cjs/js/src/static_dependencies/noble-hashes/sha3.js +214 -0
  298. package/dist/cjs/js/src/static_dependencies/noble-hashes/sha512.js +239 -0
  299. package/dist/cjs/js/src/static_dependencies/noble-hashes/utils.js +93 -0
  300. package/dist/cjs/js/src/static_dependencies/node-fetch/body.js +354 -0
  301. package/dist/cjs/js/src/static_dependencies/node-fetch/errors/abort-error.js +16 -0
  302. package/dist/cjs/js/src/static_dependencies/node-fetch/errors/base.js +20 -0
  303. package/dist/cjs/js/src/static_dependencies/node-fetch/errors/fetch-error.js +30 -0
  304. package/dist/cjs/js/src/static_dependencies/node-fetch/headers.js +239 -0
  305. package/dist/cjs/js/src/static_dependencies/node-fetch/index.js +372 -0
  306. package/dist/cjs/js/src/static_dependencies/node-fetch/request.js +273 -0
  307. package/dist/cjs/js/src/static_dependencies/node-fetch/response.js +139 -0
  308. package/dist/cjs/js/src/static_dependencies/node-fetch/utils/get-search.js +14 -0
  309. package/dist/cjs/js/src/static_dependencies/node-fetch/utils/is-redirect.js +16 -0
  310. package/dist/cjs/js/src/static_dependencies/node-fetch/utils/is.js +81 -0
  311. package/dist/cjs/js/src/static_dependencies/node-fetch/utils/referrer.js +292 -0
  312. package/dist/cjs/js/src/static_dependencies/proxies/agent-base/index.js +103 -0
  313. package/dist/cjs/js/src/static_dependencies/proxies/http-proxy-agent/index.js +140 -0
  314. package/dist/cjs/js/src/static_dependencies/proxies/https-proxy-agent/index.js +175 -0
  315. package/dist/cjs/js/src/static_dependencies/proxies/https-proxy-agent/parse-proxy-response.js +95 -0
  316. package/dist/cjs/js/src/static_dependencies/qs/index.cjs.js +7 -0
  317. package/dist/cjs/js/src/static_dependencies/scure-base/index.js +383 -0
  318. package/dist/cjs/js/src/timex.js +1562 -0
  319. package/dist/cjs/js/src/tokocrypto.js +2542 -0
  320. package/dist/cjs/js/src/upbit.js +1844 -0
  321. package/dist/cjs/js/src/wavesexchange.js +2607 -0
  322. package/dist/cjs/js/src/wazirx.js +953 -0
  323. package/dist/cjs/js/src/whitebit.js +2309 -0
  324. package/dist/cjs/js/src/woo.js +2715 -0
  325. package/dist/cjs/js/src/yobit.js +1314 -0
  326. package/dist/cjs/js/src/zaif.js +736 -0
  327. package/dist/cjs/js/src/zonda.js +1883 -0
  328. package/dist/cjs/src/base/Exchange.js +11 -0
  329. package/dist/cjs/src/bingx.js +0 -10
  330. package/dist/cjs/src/bitget.js +14 -5
  331. package/dist/cjs/src/bybit.js +1 -1
  332. package/dist/cjs/src/kucoin.js +29 -5
  333. package/js/ccxt.d.ts +1 -1
  334. package/js/ccxt.js +1 -1
  335. package/js/src/abstract/bigone.d.ts +18 -0
  336. package/js/src/abstract/binance.d.ts +2 -0
  337. package/js/src/abstract/binancecoinm.d.ts +2 -0
  338. package/js/src/abstract/binanceus.d.ts +2 -0
  339. package/js/src/abstract/binanceusdm.d.ts +2 -0
  340. package/js/src/abstract/bybit.d.ts +1 -0
  341. package/js/src/abstract/gate.d.ts +11 -0
  342. package/js/src/abstract/gateio.d.ts +11 -0
  343. package/js/src/alpaca.js +18 -18
  344. package/js/src/base/Exchange.d.ts +6 -1
  345. package/js/src/base/Exchange.js +112 -12
  346. package/js/src/bigone.d.ts +1 -2
  347. package/js/src/bigone.js +340 -145
  348. package/js/src/binance.js +15 -8
  349. package/js/src/bingx.d.ts +0 -1
  350. package/js/src/bingx.js +9 -12
  351. package/js/src/bitfinex.d.ts +2 -2
  352. package/js/src/bitfinex.js +2 -3
  353. package/js/src/bitget.js +35 -13
  354. package/js/src/bitmart.d.ts +2 -2
  355. package/js/src/bitmart.js +3 -3
  356. package/js/src/bitmex.js +1 -0
  357. package/js/src/bybit.js +3 -1
  358. package/js/src/coinlist.js +2 -3
  359. package/js/src/coinsph.js +2 -3
  360. package/js/src/deribit.js +1 -0
  361. package/js/src/gate.d.ts +4 -4
  362. package/js/src/gate.js +22 -3
  363. package/js/src/hitbtc.d.ts +4 -4
  364. package/js/src/hitbtc.js +2 -3
  365. package/js/src/htx.js +4 -7
  366. package/js/src/huobijp.js +2 -3
  367. package/js/src/kraken.js +1 -0
  368. package/js/src/kucoin.js +29 -5
  369. package/js/src/mexc.js +2 -1
  370. package/js/src/okx.js +13 -3
  371. package/js/src/pro/binance.d.ts +2 -23
  372. package/js/src/pro/binance.js +58 -22
  373. package/js/src/pro/coinbase.d.ts +2 -2
  374. package/js/src/pro/coinbase.js +4 -1
  375. package/js/src/pro/coinbasepro.d.ts +2 -2
  376. package/js/src/pro/hitbtc.d.ts +2 -2
  377. package/js/src/pro/poloniex.d.ts +2 -2
  378. package/js/src/upbit.d.ts +3 -101
  379. package/js/src/upbit.js +12 -12
  380. package/js/src/woo.js +2 -0
  381. package/package.json +11 -11
  382. package/skip-tests.json +5 -0
  383. package/tests-manager.sh +2 -2
@@ -0,0 +1,1558 @@
1
+ 'use strict';
2
+
3
+ var okx$1 = require('../okx.js');
4
+ var errors = require('../base/errors.js');
5
+ var Cache = require('../base/ws/Cache.js');
6
+ var sha256 = require('../static_dependencies/noble-hashes/sha256.js');
7
+
8
+ // ---------------------------------------------------------------------------
9
+ // ---------------------------------------------------------------------------
10
+ class okx extends okx$1 {
11
+ describe() {
12
+ return this.deepExtend(super.describe(), {
13
+ 'has': {
14
+ 'ws': true,
15
+ 'watchTicker': true,
16
+ 'watchTickers': true,
17
+ 'watchOrderBook': true,
18
+ 'watchTrades': true,
19
+ 'watchTradesForSymbols': true,
20
+ 'watchOrderBookForSymbols': true,
21
+ 'watchBalance': true,
22
+ 'watchOHLCV': true,
23
+ 'watchOrders': true,
24
+ 'watchMyTrades': true,
25
+ 'watchPositions': true,
26
+ 'createOrderWs': true,
27
+ 'editOrderWs': true,
28
+ 'cancelOrderWs': true,
29
+ 'cancelOrdersWs': true,
30
+ 'cancelAllOrdersWs': true,
31
+ },
32
+ 'urls': {
33
+ 'api': {
34
+ 'ws': 'wss://ws.okx.com:8443/ws/v5',
35
+ },
36
+ 'test': {
37
+ 'ws': 'wss://wspap.okx.com:8443/ws/v5',
38
+ },
39
+ },
40
+ 'options': {
41
+ 'watchOrderBook': {
42
+ //
43
+ // bbo-tbt
44
+ // 1. Newly added channel that sends tick-by-tick Level 1 data
45
+ // 2. All API users can subscribe
46
+ // 3. Public depth channel, verification not required
47
+ //
48
+ // books-l2-tbt
49
+ // 1. Only users who're VIP5 and above can subscribe
50
+ // 2. Identity verification required before subscription
51
+ //
52
+ // books50-l2-tbt
53
+ // 1. Only users who're VIP4 and above can subscribe
54
+ // 2. Identity verification required before subscription
55
+ //
56
+ // books
57
+ // 1. All API users can subscribe
58
+ // 2. Public depth channel, verification not required
59
+ //
60
+ // books5
61
+ // 1. All API users can subscribe
62
+ // 2. Public depth channel, verification not required
63
+ // 3. Data feeds will be delivered every 100ms (vs. every 200ms now)
64
+ //
65
+ 'depth': 'books',
66
+ },
67
+ 'watchBalance': 'spot',
68
+ 'watchTicker': {
69
+ 'channel': 'tickers', // tickers, sprd-tickers, index-tickers, block-tickers
70
+ },
71
+ 'watchTickers': {
72
+ 'channel': 'tickers', // tickers, sprd-tickers, index-tickers, block-tickers
73
+ },
74
+ 'watchOrders': {
75
+ 'type': 'ANY', // SPOT, MARGIN, SWAP, FUTURES, OPTION, ANY
76
+ },
77
+ 'watchMyTrades': {
78
+ 'type': 'ANY', // SPOT, MARGIN, SWAP, FUTURES, OPTION, ANY
79
+ },
80
+ 'createOrderWs': {
81
+ 'op': 'batch-orders', // order, batch-orders
82
+ },
83
+ 'editOrderWs': {
84
+ 'op': 'amend-order', // amend-order, batch-amend-orders
85
+ },
86
+ 'ws': {
87
+ // 'inflate': true,
88
+ },
89
+ 'checksum': true,
90
+ },
91
+ 'streaming': {
92
+ // okex does not support built-in ws protocol-level ping-pong
93
+ // instead it requires a custom text-based ping-pong
94
+ 'ping': this.ping,
95
+ 'keepAlive': 20000,
96
+ },
97
+ });
98
+ }
99
+ getUrl(channel, access = 'public') {
100
+ // for context: https://www.okx.com/help-center/changes-to-v5-api-websocket-subscription-parameter-and-url
101
+ const isSandbox = this.options['sandboxMode'];
102
+ const sandboxSuffix = isSandbox ? '?brokerId=9999' : '';
103
+ const isBusiness = (access === 'business');
104
+ const isPublic = (access === 'public');
105
+ const url = this.urls['api']['ws'];
106
+ if (isBusiness || (channel.indexOf('candle') > -1) || (channel === 'orders-algo')) {
107
+ return url + '/business' + sandboxSuffix;
108
+ }
109
+ else if (isPublic) {
110
+ return url + '/public' + sandboxSuffix;
111
+ }
112
+ return url + '/private' + sandboxSuffix;
113
+ }
114
+ async subscribeMultiple(access, channel, symbols = undefined, params = {}) {
115
+ await this.loadMarkets();
116
+ if (symbols === undefined) {
117
+ symbols = this.symbols;
118
+ }
119
+ symbols = this.marketSymbols(symbols);
120
+ const url = this.getUrl(channel, access);
121
+ let messageHash = channel;
122
+ const args = [];
123
+ messageHash += '::' + symbols.join(',');
124
+ for (let i = 0; i < symbols.length; i++) {
125
+ const marketId = this.marketId(symbols[i]);
126
+ const arg = {
127
+ 'channel': channel,
128
+ 'instId': marketId,
129
+ };
130
+ args.push(this.extend(arg, params));
131
+ }
132
+ const request = {
133
+ 'op': 'subscribe',
134
+ 'args': args,
135
+ };
136
+ return await this.watch(url, messageHash, request, messageHash);
137
+ }
138
+ async subscribe(access, messageHash, channel, symbol, params = {}) {
139
+ await this.loadMarkets();
140
+ const url = this.getUrl(channel, access);
141
+ const firstArgument = {
142
+ 'channel': channel,
143
+ };
144
+ if (symbol !== undefined) {
145
+ const market = this.market(symbol);
146
+ messageHash += ':' + market['id'];
147
+ firstArgument['instId'] = market['id'];
148
+ }
149
+ const request = {
150
+ 'op': 'subscribe',
151
+ 'args': [
152
+ this.deepExtend(firstArgument, params),
153
+ ],
154
+ };
155
+ return await this.watch(url, messageHash, request, messageHash);
156
+ }
157
+ async watchTrades(symbol, since = undefined, limit = undefined, params = {}) {
158
+ /**
159
+ * @method
160
+ * @name okx#watchTrades
161
+ * @description get the list of most recent trades for a particular symbol
162
+ * @param {string} symbol unified symbol of the market to fetch trades for
163
+ * @param {int} [since] timestamp in ms of the earliest trade to fetch
164
+ * @param {int} [limit] the maximum amount of trades to fetch
165
+ * @param {object} [params] extra parameters specific to the exchange API endpoint
166
+ * @returns {object[]} a list of [trade structures]{@link https://docs.ccxt.com/#/?id=public-trades}
167
+ */
168
+ return await this.watchTradesForSymbols([symbol], since, limit, params);
169
+ }
170
+ async watchTradesForSymbols(symbols, since = undefined, limit = undefined, params = {}) {
171
+ /**
172
+ * @method
173
+ * @name okx#watchTradesForSymbols
174
+ * @description get the list of most recent trades for a particular symbol
175
+ * @param {string} symbol unified symbol of the market to fetch trades for
176
+ * @param {int} [since] timestamp in ms of the earliest trade to fetch
177
+ * @param {int} [limit] the maximum amount of trades to fetch
178
+ * @param {object} [params] extra parameters specific to the exchange API endpoint
179
+ * @returns {object[]} a list of [trade structures]{@link https://docs.ccxt.com/#/?id=public-trades}
180
+ */
181
+ const symbolsLength = symbols.length;
182
+ if (symbolsLength === 0) {
183
+ throw new errors.ArgumentsRequired(this.id + ' watchTradesForSymbols() requires a non-empty array of symbols');
184
+ }
185
+ await this.loadMarkets();
186
+ symbols = this.marketSymbols(symbols);
187
+ const channel = 'trades';
188
+ const topics = [];
189
+ const messageHashes = [];
190
+ for (let i = 0; i < symbols.length; i++) {
191
+ const symbol = symbols[i];
192
+ messageHashes.push(channel + ':' + symbol);
193
+ const marketId = this.marketId(symbol);
194
+ const topic = {
195
+ 'channel': channel,
196
+ 'instId': marketId,
197
+ };
198
+ topics.push(topic);
199
+ }
200
+ const request = {
201
+ 'op': 'subscribe',
202
+ 'args': topics,
203
+ };
204
+ const url = this.getUrl(channel, 'public');
205
+ const trades = await this.watchMultiple(url, messageHashes, request, messageHashes);
206
+ if (this.newUpdates) {
207
+ const first = this.safeValue(trades, 0);
208
+ const tradeSymbol = this.safeString(first, 'symbol');
209
+ limit = trades.getLimit(tradeSymbol, limit);
210
+ }
211
+ return this.filterBySinceLimit(trades, since, limit, 'timestamp', true);
212
+ }
213
+ handleTrades(client, message) {
214
+ //
215
+ // {
216
+ // "arg": { channel: "trades", instId: "BTC-USDT" },
217
+ // "data": [
218
+ // {
219
+ // "instId": "BTC-USDT",
220
+ // "tradeId": "216970876",
221
+ // "px": "31684.5",
222
+ // "sz": "0.00001186",
223
+ // "side": "buy",
224
+ // "ts": "1626531038288"
225
+ // }
226
+ // ]
227
+ // }
228
+ //
229
+ const arg = this.safeValue(message, 'arg', {});
230
+ const channel = this.safeString(arg, 'channel');
231
+ const marketId = this.safeString(arg, 'instId');
232
+ const symbol = this.safeSymbol(marketId);
233
+ const data = this.safeValue(message, 'data', []);
234
+ const tradesLimit = this.safeInteger(this.options, 'tradesLimit', 1000);
235
+ for (let i = 0; i < data.length; i++) {
236
+ const trade = this.parseTrade(data[i]);
237
+ const messageHash = channel + ':' + symbol;
238
+ let stored = this.safeValue(this.trades, symbol);
239
+ if (stored === undefined) {
240
+ stored = new Cache.ArrayCache(tradesLimit);
241
+ this.trades[symbol] = stored;
242
+ }
243
+ stored.append(trade);
244
+ client.resolve(stored, messageHash);
245
+ }
246
+ }
247
+ async watchTicker(symbol, params = {}) {
248
+ /**
249
+ * @method
250
+ * @name okx#watchTicker
251
+ * @see https://www.okx.com/docs-v5/en/#order-book-trading-market-data-ws-tickers-channel
252
+ * @description watches a price ticker, a statistical calculation with the information calculated over the past 24 hours for a specific market
253
+ * @param {string} symbol unified symbol of the market to fetch the ticker for
254
+ * @param {object} [params] extra parameters specific to the exchange API endpoint
255
+ * @param {string} [params.channel] the channel to subscribe to, tickers by default. Can be tickers, sprd-tickers, index-tickers, block-tickers
256
+ * @returns {object} a [ticker structure]{@link https://docs.ccxt.com/#/?id=ticker-structure}
257
+ */
258
+ let channel = undefined;
259
+ [channel, params] = this.handleOptionAndParams(params, 'watchTicker', 'channel', 'tickers');
260
+ params['channel'] = channel;
261
+ const ticker = await this.watchTickers([symbol], params);
262
+ return this.safeValue(ticker, symbol);
263
+ }
264
+ async watchTickers(symbols = undefined, params = {}) {
265
+ /**
266
+ * @method
267
+ * @name okx#watchTickers
268
+ * @see https://www.okx.com/docs-v5/en/#order-book-trading-market-data-ws-tickers-channel
269
+ * @description watches a price ticker, a statistical calculation with the information calculated over the past 24 hours for all markets of a specific list
270
+ * @param {string[]} [symbols] unified symbol of the market to fetch the ticker for
271
+ * @param {object} [params] extra parameters specific to the exchange API endpoint
272
+ * @param {string} [params.channel] the channel to subscribe to, tickers by default. Can be tickers, sprd-tickers, index-tickers, block-tickers
273
+ * @returns {object} a [ticker structure]{@link https://docs.ccxt.com/#/?id=ticker-structure}
274
+ */
275
+ if (this.isEmpty(symbols)) {
276
+ throw new errors.ArgumentsRequired(this.id + ' watchTickers requires a list of symbols');
277
+ }
278
+ let channel = undefined;
279
+ [channel, params] = this.handleOptionAndParams(params, 'watchTickers', 'channel', 'tickers');
280
+ const newTickers = await this.subscribeMultiple('public', channel, symbols, params);
281
+ if (this.newUpdates) {
282
+ return newTickers;
283
+ }
284
+ return this.filterByArray(this.tickers, 'symbol', symbols);
285
+ }
286
+ handleTicker(client, message) {
287
+ //
288
+ // {
289
+ // "arg": { channel: "tickers", instId: "BTC-USDT" },
290
+ // "data": [
291
+ // {
292
+ // "instType": "SPOT",
293
+ // "instId": "BTC-USDT",
294
+ // "last": "31500.1",
295
+ // "lastSz": "0.00001754",
296
+ // "askPx": "31500.1",
297
+ // "askSz": "0.00998144",
298
+ // "bidPx": "31500",
299
+ // "bidSz": "3.05652439",
300
+ // "open24h": "31697",
301
+ // "high24h": "32248",
302
+ // "low24h": "31165.6",
303
+ // "sodUtc0": "31385.5",
304
+ // "sodUtc8": "32134.9",
305
+ // "volCcy24h": "503403597.38138519",
306
+ // "vol24h": "15937.10781721",
307
+ // "ts": "1626526618762"
308
+ // }
309
+ // ]
310
+ // }
311
+ //
312
+ const arg = this.safeValue(message, 'arg', {});
313
+ const channel = this.safeString(arg, 'channel');
314
+ const data = this.safeValue(message, 'data', []);
315
+ const newTickers = [];
316
+ for (let i = 0; i < data.length; i++) {
317
+ const ticker = this.parseTicker(data[i]);
318
+ const symbol = ticker['symbol'];
319
+ this.tickers[symbol] = ticker;
320
+ newTickers.push(ticker);
321
+ }
322
+ const messageHashes = this.findMessageHashes(client, channel + '::');
323
+ for (let i = 0; i < messageHashes.length; i++) {
324
+ const messageHash = messageHashes[i];
325
+ const parts = messageHash.split('::');
326
+ const symbolsString = parts[1];
327
+ const symbols = symbolsString.split(',');
328
+ const tickers = this.filterByArray(newTickers, 'symbol', symbols);
329
+ const tickersSymbols = Object.keys(tickers);
330
+ const numTickers = tickersSymbols.length;
331
+ if (numTickers > 0) {
332
+ client.resolve(tickers, messageHash);
333
+ }
334
+ }
335
+ return message;
336
+ }
337
+ async watchOHLCV(symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {
338
+ /**
339
+ * @method
340
+ * @name okx#watchOHLCV
341
+ * @description watches historical candlestick data containing the open, high, low, and close price, and the volume of a market
342
+ * @param {string} symbol unified symbol of the market to fetch OHLCV data for
343
+ * @param {string} timeframe the length of time each candle represents
344
+ * @param {int} [since] timestamp in ms of the earliest candle to fetch
345
+ * @param {int} [limit] the maximum amount of candles to fetch
346
+ * @param {object} [params] extra parameters specific to the exchange API endpoint
347
+ * @returns {int[][]} A list of candles ordered as timestamp, open, high, low, close, volume
348
+ */
349
+ await this.loadMarkets();
350
+ symbol = this.symbol(symbol);
351
+ const interval = this.safeString(this.timeframes, timeframe, timeframe);
352
+ const name = 'candle' + interval;
353
+ const ohlcv = await this.subscribe('public', name, name, symbol, params);
354
+ if (this.newUpdates) {
355
+ limit = ohlcv.getLimit(symbol, limit);
356
+ }
357
+ return this.filterBySinceLimit(ohlcv, since, limit, 0, true);
358
+ }
359
+ handleOHLCV(client, message) {
360
+ //
361
+ // {
362
+ // "arg": { channel: "candle1m", instId: "BTC-USDT" },
363
+ // "data": [
364
+ // [
365
+ // "1626690720000",
366
+ // "31334",
367
+ // "31334",
368
+ // "31334",
369
+ // "31334",
370
+ // "0.0077",
371
+ // "241.2718"
372
+ // ]
373
+ // ]
374
+ // }
375
+ //
376
+ const arg = this.safeValue(message, 'arg', {});
377
+ const channel = this.safeString(arg, 'channel');
378
+ const data = this.safeValue(message, 'data', []);
379
+ const marketId = this.safeString(arg, 'instId');
380
+ const market = this.safeMarket(marketId);
381
+ const symbol = market['id'];
382
+ const interval = channel.replace('candle', '');
383
+ // use a reverse lookup in a static map instead
384
+ const timeframe = this.findTimeframe(interval);
385
+ for (let i = 0; i < data.length; i++) {
386
+ const parsed = this.parseOHLCV(data[i], market);
387
+ this.ohlcvs[symbol] = this.safeValue(this.ohlcvs, symbol, {});
388
+ let stored = this.safeValue(this.ohlcvs[symbol], timeframe);
389
+ if (stored === undefined) {
390
+ const limit = this.safeInteger(this.options, 'OHLCVLimit', 1000);
391
+ stored = new Cache.ArrayCacheByTimestamp(limit);
392
+ this.ohlcvs[symbol][timeframe] = stored;
393
+ }
394
+ stored.append(parsed);
395
+ const messageHash = channel + ':' + marketId;
396
+ client.resolve(stored, messageHash);
397
+ }
398
+ }
399
+ async watchOrderBook(symbol, limit = undefined, params = {}) {
400
+ /**
401
+ * @method
402
+ * @name okx#watchOrderBook
403
+ * @description watches information on open orders with bid (buy) and ask (sell) prices, volumes and other data
404
+ * @param {string} symbol unified symbol of the market to fetch the order book for
405
+ * @param {int} [limit] the maximum amount of order book entries to return
406
+ * @param {object} [params] extra parameters specific to the exchange API endpoint
407
+ * @returns {object} A dictionary of [order book structures]{@link https://docs.ccxt.com/#/?id=order-book-structure} indexed by market symbols
408
+ */
409
+ //
410
+ // bbo-tbt
411
+ // 1. Newly added channel that sends tick-by-tick Level 1 data
412
+ // 2. All API users can subscribe
413
+ // 3. Public depth channel, verification not required
414
+ //
415
+ // books-l2-tbt
416
+ // 1. Only users who're VIP5 and above can subscribe
417
+ // 2. Identity verification required before subscription
418
+ //
419
+ // books50-l2-tbt
420
+ // 1. Only users who're VIP4 and above can subscribe
421
+ // 2. Identity verification required before subscription
422
+ //
423
+ // books
424
+ // 1. All API users can subscribe
425
+ // 2. Public depth channel, verification not required
426
+ //
427
+ // books5
428
+ // 1. All API users can subscribe
429
+ // 2. Public depth channel, verification not required
430
+ // 3. Data feeds will be delivered every 100ms (vs. every 200ms now)
431
+ //
432
+ return await this.watchOrderBookForSymbols([symbol], limit, params);
433
+ }
434
+ async watchOrderBookForSymbols(symbols, limit = undefined, params = {}) {
435
+ /**
436
+ * @method
437
+ * @name okx#watchOrderBookForSymbols
438
+ * @description watches information on open orders with bid (buy) and ask (sell) prices, volumes and other data
439
+ * @param {string[]} symbols unified array of symbols
440
+ * @param {int} [limit] the maximum amount of order book entries to return
441
+ * @param {object} [params] extra parameters specific to the exchange API endpoint
442
+ * @returns {object} A dictionary of [order book structures]{@link https://docs.ccxt.com/#/?id=order-book-structure} indexed by market symbols
443
+ */
444
+ await this.loadMarkets();
445
+ symbols = this.marketSymbols(symbols);
446
+ const options = this.safeValue(this.options, 'watchOrderBook', {});
447
+ const depth = this.safeString(options, 'depth', 'books');
448
+ if ((depth === 'books-l2-tbt') || (depth === 'books50-l2-tbt')) {
449
+ await this.authenticate({ 'access': 'public' });
450
+ }
451
+ const topics = [];
452
+ const messageHashes = [];
453
+ for (let i = 0; i < symbols.length; i++) {
454
+ const symbol = symbols[i];
455
+ messageHashes.push(depth + ':' + symbol);
456
+ const marketId = this.marketId(symbol);
457
+ const topic = {
458
+ 'channel': depth,
459
+ 'instId': marketId,
460
+ };
461
+ topics.push(topic);
462
+ }
463
+ const request = {
464
+ 'op': 'subscribe',
465
+ 'args': topics,
466
+ };
467
+ const url = this.getUrl(depth, 'public');
468
+ const orderbook = await this.watchMultiple(url, messageHashes, request, messageHashes);
469
+ return orderbook.limit();
470
+ }
471
+ handleDelta(bookside, delta) {
472
+ //
473
+ // [
474
+ // "31685", // price
475
+ // "0.78069158", // amount
476
+ // "0", // liquidated orders
477
+ // "17" // orders
478
+ // ]
479
+ //
480
+ const price = this.safeFloat(delta, 0);
481
+ const amount = this.safeFloat(delta, 1);
482
+ bookside.store(price, amount);
483
+ }
484
+ handleDeltas(bookside, deltas) {
485
+ for (let i = 0; i < deltas.length; i++) {
486
+ this.handleDelta(bookside, deltas[i]);
487
+ }
488
+ }
489
+ handleOrderBookMessage(client, message, orderbook, messageHash) {
490
+ //
491
+ // {
492
+ // "asks": [
493
+ // [ '31738.3', '0.05973179', "0", "3" ],
494
+ // [ '31738.5', '0.11035404', "0", "2" ],
495
+ // [ '31739.6', '0.01', "0", "1" ],
496
+ // ],
497
+ // "bids": [
498
+ // [ '31738.2', '0.67557666', "0", "9" ],
499
+ // [ '31738', '0.02466947', "0", "2" ],
500
+ // [ '31736.3', '0.01705046', "0", "2" ],
501
+ // ],
502
+ // "instId": "BTC-USDT",
503
+ // "ts": "1626537446491"
504
+ // }
505
+ //
506
+ const asks = this.safeValue(message, 'asks', []);
507
+ const bids = this.safeValue(message, 'bids', []);
508
+ const storedAsks = orderbook['asks'];
509
+ const storedBids = orderbook['bids'];
510
+ this.handleDeltas(storedAsks, asks);
511
+ this.handleDeltas(storedBids, bids);
512
+ const checksum = this.safeValue(this.options, 'checksum', true);
513
+ if (checksum) {
514
+ const asksLength = storedAsks.length;
515
+ const bidsLength = storedBids.length;
516
+ const payloadArray = [];
517
+ for (let i = 0; i < 25; i++) {
518
+ if (i < bidsLength) {
519
+ payloadArray.push(this.numberToString(storedBids[i][0]));
520
+ payloadArray.push(this.numberToString(storedBids[i][1]));
521
+ }
522
+ if (i < asksLength) {
523
+ payloadArray.push(this.numberToString(storedAsks[i][0]));
524
+ payloadArray.push(this.numberToString(storedAsks[i][1]));
525
+ }
526
+ }
527
+ const payload = payloadArray.join(':');
528
+ const responseChecksum = this.safeInteger(message, 'checksum');
529
+ const localChecksum = this.crc32(payload, true);
530
+ if (responseChecksum !== localChecksum) {
531
+ const error = new errors.InvalidNonce(this.id + ' invalid checksum');
532
+ client.reject(error, messageHash);
533
+ }
534
+ }
535
+ const timestamp = this.safeInteger(message, 'ts');
536
+ orderbook['timestamp'] = timestamp;
537
+ orderbook['datetime'] = this.iso8601(timestamp);
538
+ return orderbook;
539
+ }
540
+ handleOrderBook(client, message) {
541
+ //
542
+ // snapshot
543
+ //
544
+ // {
545
+ // "arg": { channel: 'books-l2-tbt', instId: "BTC-USDT" },
546
+ // "action": "snapshot",
547
+ // "data": [
548
+ // {
549
+ // "asks": [
550
+ // [ '31685', '0.78069158', "0", "17" ],
551
+ // [ '31685.1', '0.0001', "0", "1" ],
552
+ // [ '31685.6', '0.04543165', "0", "1" ],
553
+ // ],
554
+ // "bids": [
555
+ // [ '31684.9', '0.01', "0", "1" ],
556
+ // [ '31682.9', '0.0001', "0", "1" ],
557
+ // [ '31680.7', '0.01', "0", "1" ],
558
+ // ],
559
+ // "ts": "1626532416403",
560
+ // "checksum": -1023440116
561
+ // }
562
+ // ]
563
+ // }
564
+ //
565
+ // update
566
+ //
567
+ // {
568
+ // "arg": { channel: 'books-l2-tbt', instId: "BTC-USDT" },
569
+ // "action": "update",
570
+ // "data": [
571
+ // {
572
+ // "asks": [
573
+ // [ '31657.7', '0', "0", "0" ],
574
+ // [ '31659.7', '0.01', "0", "1" ],
575
+ // [ '31987.3', '0.01', "0", "1" ]
576
+ // ],
577
+ // "bids": [
578
+ // [ '31642.9', '0.50296385', "0", "4" ],
579
+ // [ '31639.9', '0', "0", "0" ],
580
+ // [ '31638.7', '0.01', "0", "1" ],
581
+ // ],
582
+ // "ts": "1626535709008",
583
+ // "checksum": 830931827
584
+ // }
585
+ // ]
586
+ // }
587
+ //
588
+ // books5
589
+ //
590
+ // {
591
+ // "arg": { channel: "books5", instId: "BTC-USDT" },
592
+ // "data": [
593
+ // {
594
+ // "asks": [
595
+ // [ '31738.3', '0.05973179', "0", "3" ],
596
+ // [ '31738.5', '0.11035404', "0", "2" ],
597
+ // [ '31739.6', '0.01', "0", "1" ],
598
+ // ],
599
+ // "bids": [
600
+ // [ '31738.2', '0.67557666', "0", "9" ],
601
+ // [ '31738', '0.02466947', "0", "2" ],
602
+ // [ '31736.3', '0.01705046', "0", "2" ],
603
+ // ],
604
+ // "instId": "BTC-USDT",
605
+ // "ts": "1626537446491"
606
+ // }
607
+ // ]
608
+ // }
609
+ //
610
+ // bbo-tbt
611
+ //
612
+ // {
613
+ // "arg":{
614
+ // "channel":"bbo-tbt",
615
+ // "instId":"BTC-USDT"
616
+ // },
617
+ // "data":[
618
+ // {
619
+ // "asks":[["36232.2","1.8826134","0","17"]],
620
+ // "bids":[["36232.1","0.00572212","0","2"]],
621
+ // "ts":"1651826598363"
622
+ // }
623
+ // ]
624
+ // }
625
+ //
626
+ const arg = this.safeValue(message, 'arg', {});
627
+ const channel = this.safeString(arg, 'channel');
628
+ const action = this.safeString(message, 'action');
629
+ const data = this.safeValue(message, 'data', []);
630
+ const marketId = this.safeString(arg, 'instId');
631
+ const market = this.safeMarket(marketId);
632
+ const symbol = market['symbol'];
633
+ const depths = {
634
+ 'bbo-tbt': 1,
635
+ 'books': 400,
636
+ 'books5': 5,
637
+ 'books-l2-tbt': 400,
638
+ 'books50-l2-tbt': 50,
639
+ };
640
+ const limit = this.safeInteger(depths, channel);
641
+ const messageHash = channel + ':' + symbol;
642
+ if (action === 'snapshot') {
643
+ for (let i = 0; i < data.length; i++) {
644
+ const update = data[i];
645
+ const orderbook = this.orderBook({}, limit);
646
+ this.orderbooks[symbol] = orderbook;
647
+ orderbook['symbol'] = symbol;
648
+ this.handleOrderBookMessage(client, update, orderbook, messageHash);
649
+ client.resolve(orderbook, messageHash);
650
+ }
651
+ }
652
+ else if (action === 'update') {
653
+ if (symbol in this.orderbooks) {
654
+ const orderbook = this.orderbooks[symbol];
655
+ for (let i = 0; i < data.length; i++) {
656
+ const update = data[i];
657
+ this.handleOrderBookMessage(client, update, orderbook, messageHash);
658
+ client.resolve(orderbook, messageHash);
659
+ }
660
+ }
661
+ }
662
+ else if ((channel === 'books5') || (channel === 'bbo-tbt')) {
663
+ let orderbook = this.safeValue(this.orderbooks, symbol);
664
+ if (orderbook === undefined) {
665
+ orderbook = this.orderBook({}, limit);
666
+ }
667
+ this.orderbooks[symbol] = orderbook;
668
+ for (let i = 0; i < data.length; i++) {
669
+ const update = data[i];
670
+ const timestamp = this.safeInteger(update, 'ts');
671
+ const snapshot = this.parseOrderBook(update, symbol, timestamp, 'bids', 'asks', 0, 1);
672
+ orderbook.reset(snapshot);
673
+ client.resolve(orderbook, messageHash);
674
+ }
675
+ }
676
+ return message;
677
+ }
678
+ async authenticate(params = {}) {
679
+ this.checkRequiredCredentials();
680
+ const access = this.safeString(params, 'access', 'private');
681
+ params = this.omit(params, ['access']);
682
+ const url = this.getUrl('users', access);
683
+ const messageHash = 'authenticated';
684
+ const client = this.client(url);
685
+ const future = client.future(messageHash);
686
+ const authenticated = this.safeValue(client.subscriptions, messageHash);
687
+ if (authenticated === undefined) {
688
+ const timestamp = this.seconds().toString();
689
+ const method = 'GET';
690
+ const path = '/users/self/verify';
691
+ const auth = timestamp + method + path;
692
+ const signature = this.hmac(this.encode(auth), this.encode(this.secret), sha256.sha256, 'base64');
693
+ const operation = 'login';
694
+ const request = {
695
+ 'op': operation,
696
+ 'args': [
697
+ {
698
+ 'apiKey': this.apiKey,
699
+ 'passphrase': this.password,
700
+ 'timestamp': timestamp,
701
+ 'sign': signature,
702
+ },
703
+ ],
704
+ };
705
+ const message = this.extend(request, params);
706
+ this.watch(url, messageHash, message, messageHash);
707
+ }
708
+ return future;
709
+ }
710
+ async watchBalance(params = {}) {
711
+ /**
712
+ * @method
713
+ * @name okx#watchBalance
714
+ * @description watch balance and get the amount of funds available for trading or funds locked in orders
715
+ * @param {object} [params] extra parameters specific to the exchange API endpoint
716
+ * @returns {object} a [balance structure]{@link https://docs.ccxt.com/#/?id=balance-structure}
717
+ */
718
+ await this.loadMarkets();
719
+ await this.authenticate();
720
+ return await this.subscribe('private', 'account', 'account', undefined, params);
721
+ }
722
+ handleBalance(client, message) {
723
+ //
724
+ // {
725
+ // "arg": { channel: "account" },
726
+ // "data": [
727
+ // {
728
+ // "adjEq": '',
729
+ // "details": [
730
+ // {
731
+ // "availBal": '',
732
+ // "availEq": "8.21009913",
733
+ // "cashBal": "8.21009913",
734
+ // "ccy": "USDT",
735
+ // "coinUsdPrice": "0.99994",
736
+ // "crossLiab": '',
737
+ // "disEq": "8.2096065240522",
738
+ // "eq": "8.21009913",
739
+ // "eqUsd": "8.2096065240522",
740
+ // "frozenBal": "0",
741
+ // "interest": '',
742
+ // "isoEq": "0",
743
+ // "isoLiab": '',
744
+ // "liab": '',
745
+ // "maxLoan": '',
746
+ // "mgnRatio": '',
747
+ // "notionalLever": "0",
748
+ // "ordFrozen": "0",
749
+ // "twap": "0",
750
+ // "uTime": "1621927314996",
751
+ // "upl": "0"
752
+ // },
753
+ // ],
754
+ // "imr": '',
755
+ // "isoEq": "0",
756
+ // "mgnRatio": '',
757
+ // "mmr": '',
758
+ // "notionalUsd": '',
759
+ // "ordFroz": '',
760
+ // "totalEq": "22.1930992296832",
761
+ // "uTime": "1626692120916"
762
+ // }
763
+ // ]
764
+ // }
765
+ //
766
+ const arg = this.safeValue(message, 'arg', {});
767
+ const channel = this.safeString(arg, 'channel');
768
+ const type = 'spot';
769
+ const balance = this.parseTradingBalance(message);
770
+ const oldBalance = this.safeValue(this.balance, type, {});
771
+ const newBalance = this.deepExtend(oldBalance, balance);
772
+ this.balance[type] = this.safeBalance(newBalance);
773
+ client.resolve(this.balance[type], channel);
774
+ }
775
+ orderToTrade(order, market = undefined) {
776
+ const info = this.safeValue(order, 'info', {});
777
+ const timestamp = this.safeInteger(info, 'fillTime');
778
+ const feeMarketId = this.safeString(info, 'fillFeeCcy');
779
+ const isTaker = this.safeString(info, 'execType', '') === 'T';
780
+ return this.safeTrade({
781
+ 'info': info,
782
+ 'timestamp': timestamp,
783
+ 'datetime': this.iso8601(timestamp),
784
+ 'symbol': this.safeString(order, 'symbol'),
785
+ 'id': this.safeString(info, 'tradeId'),
786
+ 'order': this.safeString(order, 'id'),
787
+ 'type': this.safeString(order, 'type'),
788
+ 'takerOrMaker': (isTaker) ? 'taker' : 'maker',
789
+ 'side': this.safeString(order, 'side'),
790
+ 'price': this.safeNumber(info, 'fillPx'),
791
+ 'amount': this.safeNumber(info, 'fillSz'),
792
+ 'cost': this.safeNumber(order, 'cost'),
793
+ 'fee': {
794
+ 'cost': this.safeNumber(info, 'fillFee'),
795
+ 'currency': this.safeCurrencyCode(feeMarketId),
796
+ },
797
+ }, market);
798
+ }
799
+ async watchMyTrades(symbol = undefined, since = undefined, limit = undefined, params = {}) {
800
+ /**
801
+ * @method
802
+ * @name okx#watchMyTrades
803
+ * @see https://www.okx.com/docs-v5/en/#order-book-trading-trade-ws-order-channel
804
+ * @description watches information on multiple trades made by the user
805
+ * @param {string} [symbol] unified market symbol of the market trades were made in
806
+ * @param {int} [since] the earliest time in ms to fetch trades for
807
+ * @param {int} [limit] the maximum number of trade structures to retrieve
808
+ * @param {object} [params] extra parameters specific to the exchange API endpoint
809
+ * @param {bool} [params.stop] true if fetching trigger or conditional trades
810
+ * @returns {object[]} a list of [trade structures]{@link https://docs.ccxt.com/#/?id=trade-structure
811
+ */
812
+ // By default, receive order updates from any instrument type
813
+ let type = undefined;
814
+ [type, params] = this.handleOptionAndParams(params, 'watchMyTrades', 'type', 'ANY');
815
+ const isStop = this.safeValue(params, 'stop', false);
816
+ params = this.omit(params, ['stop']);
817
+ await this.loadMarkets();
818
+ await this.authenticate({ 'access': isStop ? 'business' : 'private' });
819
+ const channel = isStop ? 'orders-algo' : 'orders';
820
+ let messageHash = channel + '::myTrades';
821
+ let market = undefined;
822
+ if (symbol !== undefined) {
823
+ market = this.market(symbol);
824
+ symbol = market['symbol'];
825
+ type = market['type'];
826
+ messageHash = messageHash + '::' + symbol;
827
+ }
828
+ if (type === 'future') {
829
+ type = 'futures';
830
+ }
831
+ const uppercaseType = type.toUpperCase();
832
+ const request = {
833
+ 'instType': uppercaseType,
834
+ };
835
+ const orders = await this.subscribe('private', messageHash, channel, undefined, this.extend(request, params));
836
+ if (this.newUpdates) {
837
+ limit = orders.getLimit(symbol, limit);
838
+ }
839
+ return this.filterBySymbolSinceLimit(orders, symbol, since, limit, true);
840
+ }
841
+ async watchPositions(symbols = undefined, since = undefined, limit = undefined, params = {}) {
842
+ /**
843
+ * @method
844
+ * @name okx#watchPositions
845
+ * @see https://www.okx.com/docs-v5/en/#trading-account-websocket-positions-channel
846
+ * @description watch all open positions
847
+ * @param {string[]|undefined} symbols list of unified market symbols
848
+ * @param {object} params extra parameters specific to the exchange API endpoint
849
+ * @returns {object[]} a list of [position structure]{@link https://docs.ccxt.com/en/latest/manual.html#position-structure}
850
+ */
851
+ if (this.isEmpty(symbols)) {
852
+ throw new errors.ArgumentsRequired(this.id + ' watchPositions requires a list of symbols');
853
+ }
854
+ await this.loadMarkets();
855
+ await this.authenticate(params);
856
+ symbols = this.marketSymbols(symbols);
857
+ const request = {
858
+ 'instType': 'ANY',
859
+ };
860
+ const channel = 'positions';
861
+ const newPositions = await this.subscribeMultiple('private', channel, symbols, this.extend(request, params));
862
+ if (this.newUpdates) {
863
+ return newPositions;
864
+ }
865
+ return this.filterBySymbolsSinceLimit(this.positions, symbols, since, limit, true);
866
+ }
867
+ handlePositions(client, message) {
868
+ //
869
+ // {
870
+ // arg: {
871
+ // channel: 'positions',
872
+ // instType: 'ANY',
873
+ // instId: 'XRP-USDT-SWAP',
874
+ // uid: '464737184507959869'
875
+ // },
876
+ // data: [{
877
+ // adl: '1',
878
+ // availPos: '',
879
+ // avgPx: '0.52668',
880
+ // baseBal: '',
881
+ // baseBorrowed: '',
882
+ // baseInterest: '',
883
+ // bizRefId: '',
884
+ // bizRefType: '',
885
+ // cTime: '1693151444408',
886
+ // ccy: 'USDT',
887
+ // closeOrderAlgo: [],
888
+ // deltaBS: '',
889
+ // deltaPA: '',
890
+ // gammaBS: '',
891
+ // gammaPA: '',
892
+ // idxPx: '0.52683',
893
+ // imr: '17.564000000000004',
894
+ // instId: 'XRP-USDT-SWAP',
895
+ // instType: 'SWAP',
896
+ // interest: '',
897
+ // last: '0.52691',
898
+ // lever: '3',
899
+ // liab: '',
900
+ // liabCcy: '',
901
+ // liqPx: '0.3287514731020614',
902
+ // margin: '',
903
+ // markPx: '0.52692',
904
+ // mgnMode: 'cross',
905
+ // mgnRatio: '69.00363001456147',
906
+ // mmr: '0.26346',
907
+ // notionalUsd: '52.68620388000001',
908
+ // optVal: '',
909
+ // pTime: '1693151906023',
910
+ // pendingCloseOrdLiabVal: '',
911
+ // pos: '1',
912
+ // posCcy: '',
913
+ // posId: '616057041198907393',
914
+ // posSide: 'net',
915
+ // quoteBal: '',
916
+ // quoteBorrowed: '',
917
+ // quoteInterest: '',
918
+ // spotInUseAmt: '',
919
+ // spotInUseCcy: '',
920
+ // thetaBS: '',
921
+ // thetaPA: '',
922
+ // tradeId: '138745402',
923
+ // uTime: '1693151444408',
924
+ // upl: '0.0240000000000018',
925
+ // uplLastPx: '0.0229999999999952',
926
+ // uplRatio: '0.0013670539986328',
927
+ // uplRatioLastPx: '0.001310093415356',
928
+ // usdPx: '',
929
+ // vegaBS: '',
930
+ // vegaPA: ''
931
+ // }]
932
+ // }
933
+ //
934
+ const arg = this.safeValue(message, 'arg', {});
935
+ const channel = this.safeString(arg, 'channel', '');
936
+ const data = this.safeValue(message, 'data', []);
937
+ if (this.positions === undefined) {
938
+ this.positions = new Cache.ArrayCacheBySymbolBySide();
939
+ }
940
+ const cache = this.positions;
941
+ const newPositions = [];
942
+ for (let i = 0; i < data.length; i++) {
943
+ const rawPosition = data[i];
944
+ const position = this.parsePosition(rawPosition);
945
+ newPositions.push(position);
946
+ cache.append(position);
947
+ }
948
+ const messageHashes = this.findMessageHashes(client, channel + '::');
949
+ for (let i = 0; i < messageHashes.length; i++) {
950
+ const messageHash = messageHashes[i];
951
+ const parts = messageHash.split('::');
952
+ const symbolsString = parts[1];
953
+ const symbols = symbolsString.split(',');
954
+ const positions = this.filterByArray(newPositions, 'symbol', symbols, false);
955
+ if (!this.isEmpty(positions)) {
956
+ client.resolve(positions, messageHash);
957
+ }
958
+ }
959
+ }
960
+ async watchOrders(symbol = undefined, since = undefined, limit = undefined, params = {}) {
961
+ /**
962
+ * @method
963
+ * @name okx#watchOrders
964
+ * @see https://www.okx.com/docs-v5/en/#order-book-trading-trade-ws-order-channel
965
+ * @description watches information on multiple orders made by the user
966
+ * @param {string} [symbol] unified market symbol of the market orders were made in
967
+ * @param {int} [since] the earliest time in ms to fetch orders for
968
+ * @param {int} [limit] the maximum number of order structures to retrieve
969
+ * @param {object} [params] extra parameters specific to the exchange API endpoint
970
+ * @param {bool} [params.stop] true if fetching trigger or conditional orders
971
+ * @returns {object[]} a list of [order structures]{@link https://docs.ccxt.com/#/?id=order-structure}
972
+ */
973
+ let type = undefined;
974
+ // By default, receive order updates from any instrument type
975
+ [type, params] = this.handleOptionAndParams(params, 'watchOrders', 'type', 'ANY');
976
+ const isStop = this.safeValue2(params, 'stop', 'trigger', false);
977
+ params = this.omit(params, ['stop', 'trigger']);
978
+ await this.loadMarkets();
979
+ await this.authenticate({ 'access': isStop ? 'business' : 'private' });
980
+ let market = undefined;
981
+ if (symbol !== undefined) {
982
+ market = this.market(symbol);
983
+ symbol = market['symbol'];
984
+ type = market['type'];
985
+ }
986
+ if (type === 'future') {
987
+ type = 'futures';
988
+ }
989
+ const uppercaseType = type.toUpperCase();
990
+ const request = {
991
+ 'instType': uppercaseType,
992
+ };
993
+ const channel = isStop ? 'orders-algo' : 'orders';
994
+ const orders = await this.subscribe('private', channel, channel, symbol, this.extend(request, params));
995
+ if (this.newUpdates) {
996
+ limit = orders.getLimit(symbol, limit);
997
+ }
998
+ return this.filterBySymbolSinceLimit(orders, symbol, since, limit, true);
999
+ }
1000
+ handleOrders(client, message, subscription = undefined) {
1001
+ //
1002
+ // {
1003
+ // "arg":{
1004
+ // "channel":"orders",
1005
+ // "instType":"SPOT"
1006
+ // },
1007
+ // "data":[
1008
+ // {
1009
+ // "accFillSz":"0",
1010
+ // "amendResult":"",
1011
+ // "avgPx":"",
1012
+ // "cTime":"1634548275191",
1013
+ // "category":"normal",
1014
+ // "ccy":"",
1015
+ // "clOrdId":"e847386590ce4dBC330547db94a08ba0",
1016
+ // "code":"0",
1017
+ // "execType":"",
1018
+ // "fee":"0",
1019
+ // "feeCcy":"USDT",
1020
+ // "fillFee":"0",
1021
+ // "fillFeeCcy":"",
1022
+ // "fillNotionalUsd":"",
1023
+ // "fillPx":"",
1024
+ // "fillSz":"0",
1025
+ // "fillTime":"",
1026
+ // "instId":"ETH-USDT",
1027
+ // "instType":"SPOT",
1028
+ // "lever":"",
1029
+ // "msg":"",
1030
+ // "notionalUsd":"451.4516256",
1031
+ // "ordId":"370257534141235201",
1032
+ // "ordType":"limit",
1033
+ // "pnl":"0",
1034
+ // "posSide":"",
1035
+ // "px":"60000",
1036
+ // "rebate":"0",
1037
+ // "rebateCcy":"ETH",
1038
+ // "reqId":"",
1039
+ // "side":"sell",
1040
+ // "slOrdPx":"",
1041
+ // "slTriggerPx":"",
1042
+ // "state":"live",
1043
+ // "sz":"0.007526",
1044
+ // "tag":"",
1045
+ // "tdMode":"cash",
1046
+ // "tgtCcy":"",
1047
+ // "tpOrdPx":"",
1048
+ // "tpTriggerPx":"",
1049
+ // "tradeId":"",
1050
+ // "uTime":"1634548275191"
1051
+ // }
1052
+ // ]
1053
+ // }
1054
+ //
1055
+ this.handleMyTrades(client, message);
1056
+ const arg = this.safeValue(message, 'arg', {});
1057
+ const channel = this.safeString(arg, 'channel');
1058
+ const orders = this.safeValue(message, 'data', []);
1059
+ const ordersLength = orders.length;
1060
+ if (ordersLength > 0) {
1061
+ const limit = this.safeInteger(this.options, 'ordersLimit', 1000);
1062
+ if (this.orders === undefined) {
1063
+ this.orders = new Cache.ArrayCacheBySymbolById(limit);
1064
+ this.triggerOrders = new Cache.ArrayCacheBySymbolById(limit);
1065
+ }
1066
+ const stored = (channel === 'orders-algo') ? this.triggerOrders : this.orders;
1067
+ const marketIds = [];
1068
+ const parsed = this.parseOrders(orders);
1069
+ for (let i = 0; i < parsed.length; i++) {
1070
+ const order = parsed[i];
1071
+ stored.append(order);
1072
+ const symbol = order['symbol'];
1073
+ const market = this.market(symbol);
1074
+ marketIds.push(market['id']);
1075
+ }
1076
+ client.resolve(stored, channel);
1077
+ for (let i = 0; i < marketIds.length; i++) {
1078
+ const messageHash = channel + ':' + marketIds[i];
1079
+ client.resolve(stored, messageHash);
1080
+ }
1081
+ }
1082
+ }
1083
+ handleMyTrades(client, message) {
1084
+ //
1085
+ // {
1086
+ // "arg":{
1087
+ // "channel":"orders",
1088
+ // "instType":"SPOT"
1089
+ // },
1090
+ // "data":[
1091
+ // {
1092
+ // "accFillSz":"0",
1093
+ // "amendResult":"",
1094
+ // "avgPx":"",
1095
+ // "cTime":"1634548275191",
1096
+ // "category":"normal",
1097
+ // "ccy":"",
1098
+ // "clOrdId":"e847386590ce4dBC330547db94a08ba0",
1099
+ // "code":"0",
1100
+ // "execType":"",
1101
+ // "fee":"0",
1102
+ // "feeCcy":"USDT",
1103
+ // "fillFee":"0",
1104
+ // "fillFeeCcy":"",
1105
+ // "fillNotionalUsd":"",
1106
+ // "fillPx":"",
1107
+ // "fillSz":"0",
1108
+ // "fillTime":"",
1109
+ // "instId":"ETH-USDT",
1110
+ // "instType":"SPOT",
1111
+ // "lever":"",
1112
+ // "msg":"",
1113
+ // "notionalUsd":"451.4516256",
1114
+ // "ordId":"370257534141235201",
1115
+ // "ordType":"limit",
1116
+ // "pnl":"0",
1117
+ // "posSide":"",
1118
+ // "px":"60000",
1119
+ // "rebate":"0",
1120
+ // "rebateCcy":"ETH",
1121
+ // "reqId":"",
1122
+ // "side":"sell",
1123
+ // "slOrdPx":"",
1124
+ // "slTriggerPx":"",
1125
+ // "state":"live",
1126
+ // "sz":"0.007526",
1127
+ // "tag":"",
1128
+ // "tdMode":"cash",
1129
+ // "tgtCcy":"",
1130
+ // "tpOrdPx":"",
1131
+ // "tpTriggerPx":"",
1132
+ // "tradeId":"",
1133
+ // "uTime":"1634548275191"
1134
+ // }
1135
+ // ]
1136
+ // }
1137
+ //
1138
+ const arg = this.safeValue(message, 'arg', {});
1139
+ const channel = this.safeString(arg, 'channel');
1140
+ const rawOrders = this.safeValue(message, 'data', []);
1141
+ const filteredOrders = [];
1142
+ // filter orders with no last trade id
1143
+ for (let i = 0; i < rawOrders.length; i++) {
1144
+ const rawOrder = rawOrders[i];
1145
+ const tradeId = this.safeString(rawOrder, 'tradeId', '');
1146
+ if (tradeId.length > 0) {
1147
+ const order = this.parseOrder(rawOrder);
1148
+ filteredOrders.push(order);
1149
+ }
1150
+ }
1151
+ const tradesLength = filteredOrders.length;
1152
+ if (tradesLength === 0) {
1153
+ return;
1154
+ }
1155
+ if (this.myTrades === undefined) {
1156
+ const limit = this.safeInteger(this.options, 'tradesLimit', 1000);
1157
+ this.myTrades = new Cache.ArrayCacheBySymbolById(limit);
1158
+ }
1159
+ const myTrades = this.myTrades;
1160
+ const symbols = {};
1161
+ for (let i = 0; i < filteredOrders.length; i++) {
1162
+ const rawTrade = filteredOrders[i];
1163
+ const trade = this.orderToTrade(rawTrade);
1164
+ myTrades.append(trade);
1165
+ const symbol = trade['symbol'];
1166
+ symbols[symbol] = true;
1167
+ }
1168
+ const messageHash = channel + '::myTrades';
1169
+ client.resolve(this.myTrades, messageHash);
1170
+ const tradeSymbols = Object.keys(symbols);
1171
+ for (let i = 0; i < tradeSymbols.length; i++) {
1172
+ const symbolMessageHash = messageHash + '::' + tradeSymbols[i];
1173
+ client.resolve(this.orders, symbolMessageHash);
1174
+ }
1175
+ }
1176
+ async createOrderWs(symbol, type, side, amount, price = undefined, params = {}) {
1177
+ /**
1178
+ * @method
1179
+ * @name okx#createOrderWs
1180
+ * @see https://www.okx.com/docs-v5/en/#websocket-api-trade-place-order
1181
+ * @description create a trade order
1182
+ * @param {string} symbol unified symbol of the market to create an order in
1183
+ * @param {string} type 'market' or 'limit'
1184
+ * @param {string} side 'buy' or 'sell'
1185
+ * @param {float} amount how much of currency you want to trade in units of base currency
1186
+ * @param {float|undefined} [price] the price at which the order is to be fullfilled, in units of the quote currency, ignored in market orders
1187
+ * @param {object} [params] extra parameters specific to the exchange API endpoint
1188
+ * @param {boolean} params.test test order, default false
1189
+ * @returns {object} an [order structure]{@link https://docs.ccxt.com/#/?id=order-structure}
1190
+ */
1191
+ await this.loadMarkets();
1192
+ await this.authenticate();
1193
+ const url = this.getUrl('private', 'private');
1194
+ const messageHash = this.nonce().toString();
1195
+ let op = undefined;
1196
+ [op, params] = this.handleOptionAndParams(params, 'createOrderWs', 'op', 'batch-orders');
1197
+ const args = this.createOrderRequest(symbol, type, side, amount, price, params);
1198
+ const ordType = this.safeString(args, 'ordType');
1199
+ if ((ordType === 'trigger') || (ordType === 'conditional') || (type === 'oco') || (type === 'move_order_stop') || (type === 'iceberg') || (type === 'twap')) {
1200
+ throw new errors.BadRequest(this.id + ' createOrderWs() does not support algo trading. this.options["createOrderWs"]["op"] must be either order or batch-order');
1201
+ }
1202
+ if ((op !== 'order') && (op !== 'batch-orders')) {
1203
+ throw new errors.BadRequest(this.id + ' createOrderWs() does not support algo trading. this.options["createOrderWs"]["op"] must be either order or privatePostTradeOrder or privatePostTradeOrderAlgo');
1204
+ }
1205
+ const request = {
1206
+ 'id': messageHash,
1207
+ 'op': op,
1208
+ 'args': [args],
1209
+ };
1210
+ return await this.watch(url, messageHash, request, messageHash);
1211
+ }
1212
+ handlePlaceOrders(client, message) {
1213
+ //
1214
+ // batch-orders/order/cancel-order
1215
+ // {
1216
+ // "id": "1689281055",
1217
+ // "op": "batch-orders",
1218
+ // "code": "0",
1219
+ // "msg": '',
1220
+ // "data": [{
1221
+ // "tag": "e847386590ce4dBC",
1222
+ // "ordId": "599823446566084608",
1223
+ // "clOrdId": "e847386590ce4dBCb939511604f394b0",
1224
+ // "sCode": "0",
1225
+ // "sMsg": "Order successfully placed."
1226
+ // },
1227
+ // ...
1228
+ // ]
1229
+ // }
1230
+ //
1231
+ const messageHash = this.safeString(message, 'id');
1232
+ let args = this.safeValue(message, 'data', []);
1233
+ // filter out partial errors
1234
+ args = this.filterBy(args, 'sCode', '0');
1235
+ // if empty means request failed and handle error
1236
+ if (this.isEmpty(args)) {
1237
+ const method = this.safeString(message, 'op');
1238
+ const stringMsg = this.json(message);
1239
+ this.handleErrors(undefined, undefined, client.url, method, undefined, stringMsg, stringMsg, undefined, undefined);
1240
+ }
1241
+ const orders = this.parseOrders(args, undefined, undefined, undefined);
1242
+ client.resolve(orders, messageHash);
1243
+ }
1244
+ async editOrderWs(id, symbol, type, side, amount, price = undefined, params = {}) {
1245
+ /**
1246
+ * @method
1247
+ * @name okx#editOrderWs
1248
+ * @description edit a trade order
1249
+ * @see https://www.okx.com/docs-v5/en/#order-book-trading-trade-ws-amend-order
1250
+ * @see https://www.okx.com/docs-v5/en/#order-book-trading-trade-ws-amend-multiple-orders
1251
+ * @param {string} id order id
1252
+ * @param {string} symbol unified symbol of the market to create an order in
1253
+ * @param {string} type 'market' or 'limit'
1254
+ * @param {string} side 'buy' or 'sell'
1255
+ * @param {float} amount how much of the currency you want to trade in units of the base currency
1256
+ * @param {float|undefined} [price] the price at which the order is to be fullfilled, in units of the quote currency, ignored in market orders
1257
+ * @param {object} [params] extra parameters specific to the exchange API endpoint
1258
+ * @returns {object} an [order structure]{@link https://docs.ccxt.com/#/?id=order-structure}
1259
+ */
1260
+ await this.loadMarkets();
1261
+ await this.authenticate();
1262
+ const url = this.getUrl('private', 'private');
1263
+ const messageHash = this.nonce().toString();
1264
+ let op = undefined;
1265
+ [op, params] = this.handleOptionAndParams(params, 'editOrderWs', 'op', 'amend-order');
1266
+ const args = this.editOrderRequest(id, symbol, type, side, amount, price, params);
1267
+ const request = {
1268
+ 'id': messageHash,
1269
+ 'op': op,
1270
+ 'args': [args],
1271
+ };
1272
+ return await this.watch(url, messageHash, this.extend(request, params), messageHash);
1273
+ }
1274
+ async cancelOrderWs(id, symbol = undefined, params = {}) {
1275
+ /**
1276
+ * @method
1277
+ * @name okx#cancelOrderWs
1278
+ * @see https://okx-docs.github.io/apidocs/websocket_api/en/#cancel-order-trade
1279
+ * @description cancel multiple orders
1280
+ * @param {string} id order id
1281
+ * @param {string} symbol unified market symbol, default is undefined
1282
+ * @param {object} [params] extra parameters specific to the exchange API endpoint
1283
+ * @param {string} [params.clOrdId] client order id
1284
+ * @returns {object} an list of [order structures]{@link https://docs.ccxt.com/#/?id=order-structure}
1285
+ */
1286
+ if (symbol === undefined) {
1287
+ throw new errors.BadRequest(this.id + ' cancelOrderWs() requires a symbol argument');
1288
+ }
1289
+ await this.loadMarkets();
1290
+ await this.authenticate();
1291
+ const url = this.getUrl('private', 'private');
1292
+ const messageHash = this.nonce().toString();
1293
+ const clientOrderId = this.safeString2(params, 'clOrdId', 'clientOrderId');
1294
+ params = this.omit(params, ['clientOrderId', 'clOrdId']);
1295
+ const arg = {
1296
+ 'instId': this.marketId(symbol),
1297
+ };
1298
+ if (clientOrderId !== undefined) {
1299
+ arg['clOrdId'] = clientOrderId;
1300
+ }
1301
+ else {
1302
+ arg['ordId'] = id;
1303
+ }
1304
+ const request = {
1305
+ 'id': messageHash,
1306
+ 'op': 'cancel-order',
1307
+ 'args': [this.extend(arg, params)],
1308
+ };
1309
+ return await this.watch(url, messageHash, request, messageHash);
1310
+ }
1311
+ async cancelOrdersWs(ids, symbol = undefined, params = {}) {
1312
+ /**
1313
+ * @method
1314
+ * @name okx#cancelOrdersWs
1315
+ * @see https://www.okx.com/docs-v5/en/#order-book-trading-trade-ws-mass-cancel-order
1316
+ * @description cancel multiple orders
1317
+ * @param {string[]} ids order ids
1318
+ * @param {string} symbol unified market symbol, default is undefined
1319
+ * @param {object} [params] extra parameters specific to the exchange API endpoint
1320
+ * @returns {object} an list of [order structures]{@link https://docs.ccxt.com/#/?id=order-structure}
1321
+ */
1322
+ const idsLength = ids.length;
1323
+ if (idsLength > 20) {
1324
+ throw new errors.BadRequest(this.id + ' cancelOrdersWs() accepts up to 20 ids at a time');
1325
+ }
1326
+ if (symbol === undefined) {
1327
+ throw new errors.BadRequest(this.id + ' cancelOrdersWs() requires a symbol argument');
1328
+ }
1329
+ await this.loadMarkets();
1330
+ await this.authenticate();
1331
+ const url = this.getUrl('private', 'private');
1332
+ const messageHash = this.nonce().toString();
1333
+ const args = [];
1334
+ for (let i = 0; i < idsLength; i++) {
1335
+ const arg = {
1336
+ 'instId': this.marketId(symbol),
1337
+ 'ordId': ids[i],
1338
+ };
1339
+ args.push(arg);
1340
+ }
1341
+ const request = {
1342
+ 'id': messageHash,
1343
+ 'op': 'batch-cancel-orders',
1344
+ 'args': args,
1345
+ };
1346
+ return await this.watch(url, messageHash, this.deepExtend(request, params), messageHash);
1347
+ }
1348
+ async cancelAllOrdersWs(symbol = undefined, params = {}) {
1349
+ /**
1350
+ * @method
1351
+ * @name okx#cancelAllOrdersWs
1352
+ * @see https://docs.okx.com/websockets/#message-cancelAll
1353
+ * @description cancel all open orders of a type. Only applicable to Option in Portfolio Margin mode, and MMP privilege is required.
1354
+ * @param {string} symbol unified market symbol, only orders in the market of this symbol are cancelled when symbol is not undefined
1355
+ * @param {object} [params] extra parameters specific to the exchange API endpoint
1356
+ * @returns {object[]} a list of [order structures]{@link https://docs.ccxt.com/#/?id=order-structure}
1357
+ */
1358
+ if (symbol === undefined) {
1359
+ throw new errors.BadRequest(this.id + ' cancelAllOrdersWs() requires a symbol argument');
1360
+ }
1361
+ await this.loadMarkets();
1362
+ await this.authenticate();
1363
+ const market = this.market(symbol);
1364
+ if (market['type'] !== 'option') {
1365
+ throw new errors.BadRequest(this.id + 'cancelAllOrdersWs is only applicable to Option in Portfolio Margin mode, and MMP privilege is required.');
1366
+ }
1367
+ const url = this.getUrl('private', 'private');
1368
+ const messageHash = this.nonce().toString();
1369
+ const request = {
1370
+ 'id': messageHash,
1371
+ 'op': 'mass-cancel',
1372
+ 'args': [this.extend({
1373
+ 'instType': 'OPTION',
1374
+ 'instFamily': market['id'],
1375
+ }, params)],
1376
+ };
1377
+ return await this.watch(url, messageHash, request, messageHash);
1378
+ }
1379
+ handleCancelAllOrders(client, message) {
1380
+ //
1381
+ // {
1382
+ // "id": "1512",
1383
+ // "op": "mass-cancel",
1384
+ // "data": [
1385
+ // {
1386
+ // "result": true
1387
+ // }
1388
+ // ],
1389
+ // "code": "0",
1390
+ // "msg": ""
1391
+ // }
1392
+ //
1393
+ const messageHash = this.safeString(message, 'id');
1394
+ const data = this.safeValue(message, 'data', []);
1395
+ client.resolve(data, messageHash);
1396
+ }
1397
+ handleSubscriptionStatus(client, message) {
1398
+ //
1399
+ // { event: 'subscribe', arg: { channel: "tickers", instId: "BTC-USDT" } }
1400
+ //
1401
+ // const channel = this.safeString (message, "channel");
1402
+ // client.subscriptions[channel] = message;
1403
+ return message;
1404
+ }
1405
+ handleAuthenticate(client, message) {
1406
+ //
1407
+ // { event: "login", success: true }
1408
+ //
1409
+ const future = this.safeValue(client.futures, 'authenticated');
1410
+ future.resolve(true);
1411
+ }
1412
+ ping(client) {
1413
+ // okex does not support built-in ws protocol-level ping-pong
1414
+ // instead it requires custom text-based ping-pong
1415
+ return 'ping';
1416
+ }
1417
+ handlePong(client, message) {
1418
+ client.lastPong = this.milliseconds();
1419
+ return message;
1420
+ }
1421
+ handleErrorMessage(client, message) {
1422
+ //
1423
+ // { event: 'error', msg: "Illegal request: {"op":"subscribe","args":["spot/ticker:BTC-USDT"]}", code: "60012" }
1424
+ // { event: 'error", msg: "channel:ticker,instId:BTC-USDT doesn"t exist", code: "60018" }
1425
+ //
1426
+ const errorCode = this.safeInteger(message, 'code');
1427
+ try {
1428
+ if (errorCode) {
1429
+ const feedback = this.id + ' ' + this.json(message);
1430
+ this.throwExactlyMatchedException(this.exceptions['exact'], errorCode, feedback);
1431
+ const messageString = this.safeValue(message, 'msg');
1432
+ if (messageString !== undefined) {
1433
+ this.throwBroadlyMatchedException(this.exceptions['broad'], messageString, feedback);
1434
+ }
1435
+ }
1436
+ }
1437
+ catch (e) {
1438
+ if (e instanceof errors.AuthenticationError) {
1439
+ const messageHash = 'authenticated';
1440
+ client.reject(e, messageHash);
1441
+ if (messageHash in client.subscriptions) {
1442
+ delete client.subscriptions[messageHash];
1443
+ }
1444
+ return false;
1445
+ }
1446
+ else {
1447
+ client.reject(e);
1448
+ }
1449
+ }
1450
+ return message;
1451
+ }
1452
+ handleMessage(client, message) {
1453
+ if (!this.handleErrorMessage(client, message)) {
1454
+ return;
1455
+ }
1456
+ //
1457
+ // { event: 'subscribe', arg: { channel: "tickers", instId: "BTC-USDT" } }
1458
+ // { event: 'login", msg: '", code: "0" }
1459
+ //
1460
+ // {
1461
+ // "arg": { channel: "tickers", instId: "BTC-USDT" },
1462
+ // "data": [
1463
+ // {
1464
+ // "instType": "SPOT",
1465
+ // "instId": "BTC-USDT",
1466
+ // "last": "31500.1",
1467
+ // "lastSz": "0.00001754",
1468
+ // "askPx": "31500.1",
1469
+ // "askSz": "0.00998144",
1470
+ // "bidPx": "31500",
1471
+ // "bidSz": "3.05652439",
1472
+ // "open24h": "31697",
1473
+ // "high24h": "32248",
1474
+ // "low24h": "31165.6",
1475
+ // "sodUtc0": "31385.5",
1476
+ // "sodUtc8": "32134.9",
1477
+ // "volCcy24h": "503403597.38138519",
1478
+ // "vol24h": "15937.10781721",
1479
+ // "ts": "1626526618762"
1480
+ // }
1481
+ // ]
1482
+ // }
1483
+ //
1484
+ // { event: 'error', msg: "Illegal request: {"op":"subscribe","args":["spot/ticker:BTC-USDT"]}", code: "60012" }
1485
+ // { event: 'error", msg: "channel:ticker,instId:BTC-USDT doesn"t exist", code: "60018" }
1486
+ // { event: 'error', msg: "Invalid OK_ACCESS_KEY", code: "60005" }
1487
+ // {
1488
+ // "event": "error",
1489
+ // "msg": "Illegal request: {"op":"login","args":["de89b035-b233-44b2-9a13-0ccdd00bda0e","7KUcc8YzQhnxBE3K","1626691289","H57N99mBt5NvW8U19FITrPdOxycAERFMaapQWRqLaSE="]}",
1490
+ // "code": "60012"
1491
+ // }
1492
+ //
1493
+ //
1494
+ //
1495
+ if (message === 'pong') {
1496
+ return this.handlePong(client, message);
1497
+ }
1498
+ // const table = this.safeString (message, 'table');
1499
+ // if (table === undefined) {
1500
+ const event = this.safeString2(message, 'event', 'op');
1501
+ if (event !== undefined) {
1502
+ const methods = {
1503
+ // 'info': this.handleSystemStatus,
1504
+ // 'book': 'handleOrderBook',
1505
+ 'login': this.handleAuthenticate,
1506
+ 'subscribe': this.handleSubscriptionStatus,
1507
+ 'order': this.handlePlaceOrders,
1508
+ 'batch-orders': this.handlePlaceOrders,
1509
+ 'amend-order': this.handlePlaceOrders,
1510
+ 'batch-amend-orders': this.handlePlaceOrders,
1511
+ 'cancel-order': this.handlePlaceOrders,
1512
+ 'mass-cancel': this.handleCancelAllOrders,
1513
+ };
1514
+ const method = this.safeValue(methods, event);
1515
+ if (method === undefined) {
1516
+ return message;
1517
+ }
1518
+ else {
1519
+ return method.call(this, client, message);
1520
+ }
1521
+ }
1522
+ else {
1523
+ const arg = this.safeValue(message, 'arg', {});
1524
+ const channel = this.safeString(arg, 'channel');
1525
+ const methods = {
1526
+ 'bbo-tbt': this.handleOrderBook,
1527
+ 'books': this.handleOrderBook,
1528
+ 'books5': this.handleOrderBook,
1529
+ 'books50-l2-tbt': this.handleOrderBook,
1530
+ 'books-l2-tbt': this.handleOrderBook,
1531
+ 'tickers': this.handleTicker,
1532
+ 'positions': this.handlePositions,
1533
+ 'index-tickers': this.handleTicker,
1534
+ 'sprd-tickers': this.handleTicker,
1535
+ 'block-tickers': this.handleTicker,
1536
+ 'trades': this.handleTrades,
1537
+ 'account': this.handleBalance,
1538
+ // 'margin_account': this.handleBalance,
1539
+ 'orders': this.handleOrders,
1540
+ 'orders-algo': this.handleOrders,
1541
+ };
1542
+ const method = this.safeValue(methods, channel);
1543
+ if (method === undefined) {
1544
+ if (channel.indexOf('candle') === 0) {
1545
+ this.handleOHLCV(client, message);
1546
+ }
1547
+ else {
1548
+ return message;
1549
+ }
1550
+ }
1551
+ else {
1552
+ return method.call(this, client, message);
1553
+ }
1554
+ }
1555
+ }
1556
+ }
1557
+
1558
+ module.exports = okx;