bitget-api 2.3.6 → 3.0.0

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 (421) hide show
  1. package/README.md +272 -18
  2. package/{lib → dist/cjs}/broker-client.d.ts +3 -2
  3. package/{lib → dist/cjs}/broker-client.js +4 -4
  4. package/dist/cjs/broker-client.js.map +1 -0
  5. package/{lib → dist/cjs}/constants/enum.d.ts +3 -0
  6. package/{lib → dist/cjs}/constants/enum.js +3 -0
  7. package/dist/cjs/constants/enum.js.map +1 -0
  8. package/{lib → dist/cjs}/futures-client.d.ts +5 -2
  9. package/{lib → dist/cjs}/futures-client.js +18 -9
  10. package/dist/cjs/futures-client.js.map +1 -0
  11. package/dist/cjs/index.d.ts +50 -0
  12. package/dist/cjs/index.js +67 -0
  13. package/dist/cjs/index.js.map +1 -0
  14. package/dist/cjs/package.json +3 -0
  15. package/{lib → dist/cjs}/rest-client-v2.d.ts +17 -2
  16. package/{lib → dist/cjs}/rest-client-v2.js +62 -74
  17. package/dist/cjs/rest-client-v2.js.map +1 -0
  18. package/dist/cjs/rest-client-v3.d.ts +440 -0
  19. package/dist/cjs/rest-client-v3.js +573 -0
  20. package/dist/cjs/rest-client-v3.js.map +1 -0
  21. package/{lib → dist/cjs}/spot-client.d.ts +5 -2
  22. package/{lib → dist/cjs}/spot-client.js +21 -23
  23. package/dist/cjs/spot-client.js.map +1 -0
  24. package/{lib → dist/cjs}/types/request/shared.d.ts +4 -4
  25. package/dist/cjs/types/request/shared.js.map +1 -0
  26. package/{lib → dist/cjs}/types/request/v1/brokerV1.d.ts +1 -1
  27. package/dist/cjs/types/request/v1/brokerV1.js.map +1 -0
  28. package/{lib → dist/cjs}/types/request/v1/futuresV1.d.ts +10 -10
  29. package/dist/cjs/types/request/v1/futuresV1.js.map +1 -0
  30. package/{lib → dist/cjs}/types/request/v1/spotV1.d.ts +4 -4
  31. package/dist/cjs/types/request/v1/spotV1.js.map +1 -0
  32. package/{lib → dist/cjs}/types/request/v2/broker.d.ts +3 -3
  33. package/dist/cjs/types/request/v2/broker.js.map +1 -0
  34. package/{lib → dist/cjs}/types/request/v2/common.d.ts +1 -1
  35. package/dist/cjs/types/request/v2/common.js.map +1 -0
  36. package/{lib → dist/cjs}/types/request/v2/copytrading.d.ts +5 -5
  37. package/dist/cjs/types/request/v2/copytrading.js.map +1 -0
  38. package/dist/cjs/types/request/v2/earn.js.map +1 -0
  39. package/{lib → dist/cjs}/types/request/v2/futures.d.ts +9 -9
  40. package/dist/cjs/types/request/v2/futures.js.map +1 -0
  41. package/{lib → dist/cjs}/types/request/v2/margin.d.ts +5 -5
  42. package/dist/cjs/types/request/v2/margin.js.map +1 -0
  43. package/{lib → dist/cjs}/types/request/v2/spot.d.ts +12 -12
  44. package/dist/cjs/types/request/v2/spot.js.map +1 -0
  45. package/dist/cjs/types/request/v3/account.d.ts +162 -0
  46. package/dist/cjs/types/request/v3/account.js +4 -0
  47. package/dist/cjs/types/request/v3/account.js.map +1 -0
  48. package/dist/cjs/types/request/v3/loan.d.ts +31 -0
  49. package/dist/cjs/types/request/v3/loan.js +3 -0
  50. package/dist/cjs/types/request/v3/loan.js.map +1 -0
  51. package/dist/cjs/types/request/v3/public.d.ts +65 -0
  52. package/dist/cjs/types/request/v3/public.js +3 -0
  53. package/dist/cjs/types/request/v3/public.js.map +1 -0
  54. package/dist/cjs/types/request/v3/strategy.d.ts +46 -0
  55. package/dist/cjs/types/request/v3/strategy.js +3 -0
  56. package/dist/cjs/types/request/v3/strategy.js.map +1 -0
  57. package/dist/cjs/types/request/v3/trade.d.ts +112 -0
  58. package/dist/cjs/types/request/v3/trade.js +3 -0
  59. package/dist/cjs/types/request/v3/trade.js.map +1 -0
  60. package/{lib → dist/cjs}/types/response/v1/futures.d.ts +1 -1
  61. package/dist/cjs/types/response/v1/futures.js.map +1 -0
  62. package/dist/cjs/types/response/v1/shared.js.map +1 -0
  63. package/dist/cjs/types/response/v1/spot.js.map +1 -0
  64. package/dist/cjs/types/response/v2/broker.js.map +1 -0
  65. package/dist/cjs/types/response/v2/common.js.map +1 -0
  66. package/dist/cjs/types/response/v2/copy-trading.js.map +1 -0
  67. package/dist/cjs/types/response/v2/earn.js.map +1 -0
  68. package/{lib → dist/cjs}/types/response/v2/futures.d.ts +1 -1
  69. package/dist/cjs/types/response/v2/futures.js.map +1 -0
  70. package/dist/cjs/types/response/v2/margin.js.map +1 -0
  71. package/{lib → dist/cjs}/types/response/v2/spot.d.ts +1 -1
  72. package/dist/cjs/types/response/v2/spot.js.map +1 -0
  73. package/dist/cjs/types/response/v3/account.d.ts +197 -0
  74. package/dist/cjs/types/response/v3/account.js +3 -0
  75. package/dist/cjs/types/response/v3/account.js.map +1 -0
  76. package/dist/cjs/types/response/v3/loan.d.ts +85 -0
  77. package/dist/cjs/types/response/v3/loan.js +3 -0
  78. package/dist/cjs/types/response/v3/loan.js.map +1 -0
  79. package/dist/cjs/types/response/v3/public.d.ts +142 -0
  80. package/dist/cjs/types/response/v3/public.js +3 -0
  81. package/dist/cjs/types/response/v3/public.js.map +1 -0
  82. package/dist/cjs/types/response/v3/strategy.d.ts +26 -0
  83. package/dist/cjs/types/response/v3/strategy.js +3 -0
  84. package/dist/cjs/types/response/v3/strategy.js.map +1 -0
  85. package/dist/cjs/types/response/v3/trade.d.ts +193 -0
  86. package/dist/cjs/types/response/v3/trade.js +3 -0
  87. package/dist/cjs/types/response/v3/trade.js.map +1 -0
  88. package/dist/cjs/types/shared.d.ts +8 -0
  89. package/{lib → dist/cjs}/types/shared.js.map +1 -1
  90. package/dist/cjs/types/websockets/ws-api-request.d.ts +20 -0
  91. package/dist/cjs/types/websockets/ws-api-request.js +3 -0
  92. package/dist/cjs/types/websockets/ws-api-request.js.map +1 -0
  93. package/dist/cjs/types/websockets/ws-api-response.d.ts +6 -0
  94. package/dist/cjs/types/websockets/ws-api-response.js +3 -0
  95. package/dist/cjs/types/websockets/ws-api-response.js.map +1 -0
  96. package/dist/cjs/types/websockets/ws-api.d.ts +91 -0
  97. package/dist/cjs/types/websockets/ws-api.js +11 -0
  98. package/dist/cjs/types/websockets/ws-api.js.map +1 -0
  99. package/{lib/types/websockets/events.d.ts → dist/cjs/types/websockets/ws-events.d.ts} +6 -0
  100. package/dist/cjs/types/websockets/ws-events.js +11 -0
  101. package/dist/cjs/types/websockets/ws-events.js.map +1 -0
  102. package/dist/cjs/types/websockets/ws-general.d.ts +112 -0
  103. package/dist/cjs/types/websockets/ws-general.js +3 -0
  104. package/dist/cjs/types/websockets/ws-general.js.map +1 -0
  105. package/{lib → dist/cjs}/util/BaseRestClient.d.ts +3 -2
  106. package/dist/cjs/util/BaseRestClient.js +274 -0
  107. package/dist/cjs/util/BaseRestClient.js.map +1 -0
  108. package/dist/cjs/util/BaseWSClient.d.ts +197 -0
  109. package/dist/cjs/util/BaseWSClient.js +759 -0
  110. package/dist/cjs/util/BaseWSClient.js.map +1 -0
  111. package/dist/cjs/util/WsStore.d.ts +74 -0
  112. package/dist/cjs/util/WsStore.js +287 -0
  113. package/dist/cjs/util/WsStore.js.map +1 -0
  114. package/{lib → dist/cjs}/util/WsStore.types.d.ts +27 -1
  115. package/{lib → dist/cjs}/util/WsStore.types.js +3 -2
  116. package/dist/cjs/util/WsStore.types.js.map +1 -0
  117. package/dist/cjs/util/logger.d.ts +8 -0
  118. package/{lib → dist/cjs}/util/logger.js +2 -10
  119. package/dist/cjs/util/logger.js.map +1 -0
  120. package/{lib → dist/cjs}/util/requestUtils.d.ts +18 -1
  121. package/{lib → dist/cjs}/util/requestUtils.js +6 -5
  122. package/dist/cjs/util/requestUtils.js.map +1 -0
  123. package/{lib → dist/cjs}/util/type-guards.d.ts +4 -1
  124. package/{lib → dist/cjs}/util/type-guards.js +20 -10
  125. package/dist/cjs/util/type-guards.js.map +1 -0
  126. package/dist/cjs/util/webCryptoAPI.d.ts +14 -0
  127. package/dist/cjs/util/webCryptoAPI.js +80 -0
  128. package/dist/cjs/util/webCryptoAPI.js.map +1 -0
  129. package/dist/cjs/util/websocket-util.d.ts +88 -0
  130. package/dist/cjs/util/websocket-util.js +241 -0
  131. package/dist/cjs/util/websocket-util.js.map +1 -0
  132. package/dist/cjs/websocket-api-client.d.ts +89 -0
  133. package/dist/cjs/websocket-api-client.js +122 -0
  134. package/dist/cjs/websocket-api-client.js.map +1 -0
  135. package/{lib/websocket-client.d.ts → dist/cjs/websocket-client-legacy-v1.d.ts} +6 -6
  136. package/{lib/websocket-client.js → dist/cjs/websocket-client-legacy-v1.js} +211 -156
  137. package/dist/cjs/websocket-client-legacy-v1.js.map +1 -0
  138. package/dist/cjs/websocket-client-v2.d.ts +79 -0
  139. package/dist/cjs/websocket-client-v2.js +339 -0
  140. package/dist/cjs/websocket-client-v2.js.map +1 -0
  141. package/dist/cjs/websocket-client-v3.d.ts +84 -0
  142. package/dist/cjs/websocket-client-v3.js +408 -0
  143. package/dist/cjs/websocket-client-v3.js.map +1 -0
  144. package/dist/mjs/broker-client.d.ts +127 -0
  145. package/dist/mjs/broker-client.js +158 -0
  146. package/dist/mjs/broker-client.js.map +1 -0
  147. package/dist/mjs/constants/enum.d.ts +31 -0
  148. package/dist/mjs/constants/enum.js +32 -0
  149. package/dist/mjs/constants/enum.js.map +1 -0
  150. package/dist/mjs/futures-client.d.ts +184 -0
  151. package/dist/mjs/futures-client.js +473 -0
  152. package/dist/mjs/futures-client.js.map +1 -0
  153. package/dist/mjs/index.d.ts +50 -0
  154. package/dist/mjs/index.js +51 -0
  155. package/dist/mjs/index.js.map +1 -0
  156. package/dist/mjs/package.json +3 -0
  157. package/dist/mjs/rest-client-v2.d.ts +1098 -0
  158. package/dist/mjs/rest-client-v2.js +1092 -0
  159. package/dist/mjs/rest-client-v2.js.map +1 -0
  160. package/dist/mjs/rest-client-v3.d.ts +440 -0
  161. package/dist/mjs/rest-client-v3.js +566 -0
  162. package/dist/mjs/rest-client-v3.js.map +1 -0
  163. package/dist/mjs/spot-client.d.ts +154 -0
  164. package/dist/mjs/spot-client.js +273 -0
  165. package/dist/mjs/spot-client.js.map +1 -0
  166. package/dist/mjs/types/request/shared.d.ts +23 -0
  167. package/dist/mjs/types/request/shared.js +2 -0
  168. package/dist/mjs/types/request/shared.js.map +1 -0
  169. package/dist/mjs/types/request/v1/brokerV1.d.ts +23 -0
  170. package/dist/mjs/types/request/v1/brokerV1.js +2 -0
  171. package/dist/mjs/types/request/v1/brokerV1.js.map +1 -0
  172. package/dist/mjs/types/request/v1/futuresV1.d.ts +168 -0
  173. package/dist/mjs/types/request/v1/futuresV1.js +2 -0
  174. package/dist/mjs/types/request/v1/futuresV1.js.map +1 -0
  175. package/dist/mjs/types/request/v1/spotV1.d.ts +105 -0
  176. package/dist/mjs/types/request/v1/spotV1.js +2 -0
  177. package/dist/mjs/types/request/v1/spotV1.js.map +1 -0
  178. package/dist/mjs/types/request/v2/broker.d.ts +69 -0
  179. package/dist/mjs/types/request/v2/broker.js +7 -0
  180. package/dist/mjs/types/request/v2/broker.js.map +1 -0
  181. package/dist/mjs/types/request/v2/common.d.ts +163 -0
  182. package/dist/mjs/types/request/v2/common.js +2 -0
  183. package/dist/mjs/types/request/v2/common.js.map +1 -0
  184. package/dist/mjs/types/request/v2/copytrading.d.ts +196 -0
  185. package/dist/mjs/types/request/v2/copytrading.js +9 -0
  186. package/dist/mjs/types/request/v2/copytrading.js.map +1 -0
  187. package/dist/mjs/types/request/v2/earn.d.ts +121 -0
  188. package/dist/mjs/types/request/v2/earn.js +9 -0
  189. package/dist/mjs/types/request/v2/earn.js.map +1 -0
  190. package/dist/mjs/types/request/v2/futures.d.ts +363 -0
  191. package/dist/mjs/types/request/v2/futures.js +2 -0
  192. package/dist/mjs/types/request/v2/futures.js.map +1 -0
  193. package/dist/mjs/types/request/v2/margin.d.ts +121 -0
  194. package/dist/mjs/types/request/v2/margin.js +7 -0
  195. package/dist/mjs/types/request/v2/margin.js.map +1 -0
  196. package/dist/mjs/types/request/v2/spot.d.ts +265 -0
  197. package/dist/mjs/types/request/v2/spot.js +2 -0
  198. package/dist/mjs/types/request/v2/spot.js.map +1 -0
  199. package/dist/mjs/types/request/v3/account.d.ts +162 -0
  200. package/dist/mjs/types/request/v3/account.js +3 -0
  201. package/dist/mjs/types/request/v3/account.js.map +1 -0
  202. package/dist/mjs/types/request/v3/loan.d.ts +31 -0
  203. package/dist/mjs/types/request/v3/loan.js +2 -0
  204. package/dist/mjs/types/request/v3/loan.js.map +1 -0
  205. package/dist/mjs/types/request/v3/public.d.ts +65 -0
  206. package/dist/mjs/types/request/v3/public.js +2 -0
  207. package/dist/mjs/types/request/v3/public.js.map +1 -0
  208. package/dist/mjs/types/request/v3/strategy.d.ts +46 -0
  209. package/dist/mjs/types/request/v3/strategy.js +2 -0
  210. package/dist/mjs/types/request/v3/strategy.js.map +1 -0
  211. package/dist/mjs/types/request/v3/trade.d.ts +112 -0
  212. package/dist/mjs/types/request/v3/trade.js +2 -0
  213. package/dist/mjs/types/request/v3/trade.js.map +1 -0
  214. package/dist/mjs/types/response/v1/futures.d.ts +72 -0
  215. package/dist/mjs/types/response/v1/futures.js +2 -0
  216. package/dist/mjs/types/response/v1/futures.js.map +1 -0
  217. package/dist/mjs/types/response/v1/shared.d.ts +15 -0
  218. package/dist/mjs/types/response/v1/shared.js +2 -0
  219. package/dist/mjs/types/response/v1/shared.js.map +1 -0
  220. package/dist/mjs/types/response/v1/spot.d.ts +65 -0
  221. package/dist/mjs/types/response/v1/spot.js +2 -0
  222. package/dist/mjs/types/response/v1/spot.js.map +1 -0
  223. package/dist/mjs/types/response/v2/broker.d.ts +135 -0
  224. package/dist/mjs/types/response/v2/broker.js +7 -0
  225. package/dist/mjs/types/response/v2/broker.js.map +1 -0
  226. package/dist/mjs/types/response/v2/common.d.ts +384 -0
  227. package/dist/mjs/types/response/v2/common.js +7 -0
  228. package/dist/mjs/types/response/v2/common.js.map +1 -0
  229. package/dist/mjs/types/response/v2/copy-trading.d.ts +385 -0
  230. package/dist/mjs/types/response/v2/copy-trading.js +9 -0
  231. package/dist/mjs/types/response/v2/copy-trading.js.map +1 -0
  232. package/dist/mjs/types/response/v2/earn.d.ts +294 -0
  233. package/dist/mjs/types/response/v2/earn.js +9 -0
  234. package/dist/mjs/types/response/v2/earn.js.map +1 -0
  235. package/dist/mjs/types/response/v2/futures.d.ts +546 -0
  236. package/dist/mjs/types/response/v2/futures.js +7 -0
  237. package/dist/mjs/types/response/v2/futures.js.map +1 -0
  238. package/dist/mjs/types/response/v2/margin.d.ts +277 -0
  239. package/dist/mjs/types/response/v2/margin.js +7 -0
  240. package/dist/mjs/types/response/v2/margin.js.map +1 -0
  241. package/dist/mjs/types/response/v2/spot.d.ts +385 -0
  242. package/dist/mjs/types/response/v2/spot.js +7 -0
  243. package/dist/mjs/types/response/v2/spot.js.map +1 -0
  244. package/dist/mjs/types/response/v3/account.d.ts +197 -0
  245. package/dist/mjs/types/response/v3/account.js +2 -0
  246. package/dist/mjs/types/response/v3/account.js.map +1 -0
  247. package/dist/mjs/types/response/v3/loan.d.ts +85 -0
  248. package/dist/mjs/types/response/v3/loan.js +2 -0
  249. package/dist/mjs/types/response/v3/loan.js.map +1 -0
  250. package/dist/mjs/types/response/v3/public.d.ts +142 -0
  251. package/dist/mjs/types/response/v3/public.js +2 -0
  252. package/dist/mjs/types/response/v3/public.js.map +1 -0
  253. package/dist/mjs/types/response/v3/strategy.d.ts +26 -0
  254. package/dist/mjs/types/response/v3/strategy.js +2 -0
  255. package/dist/mjs/types/response/v3/strategy.js.map +1 -0
  256. package/dist/mjs/types/response/v3/trade.d.ts +193 -0
  257. package/dist/mjs/types/response/v3/trade.js +2 -0
  258. package/dist/mjs/types/response/v3/trade.js.map +1 -0
  259. package/dist/mjs/types/shared.d.ts +8 -0
  260. package/dist/mjs/types/shared.js +2 -0
  261. package/{lib/types/request → dist/mjs/types}/shared.js.map +1 -1
  262. package/dist/mjs/types/websockets/ws-api-request.d.ts +20 -0
  263. package/dist/mjs/types/websockets/ws-api-request.js +2 -0
  264. package/dist/mjs/types/websockets/ws-api-request.js.map +1 -0
  265. package/dist/mjs/types/websockets/ws-api-response.d.ts +6 -0
  266. package/dist/mjs/types/websockets/ws-api-response.js +2 -0
  267. package/dist/mjs/types/websockets/ws-api-response.js.map +1 -0
  268. package/dist/mjs/types/websockets/ws-api.d.ts +91 -0
  269. package/dist/mjs/types/websockets/ws-api.js +8 -0
  270. package/dist/mjs/types/websockets/ws-api.js.map +1 -0
  271. package/dist/mjs/types/websockets/ws-events.d.ts +81 -0
  272. package/dist/mjs/types/websockets/ws-events.js +8 -0
  273. package/dist/mjs/types/websockets/ws-events.js.map +1 -0
  274. package/dist/mjs/types/websockets/ws-general.d.ts +112 -0
  275. package/dist/mjs/types/websockets/ws-general.js +2 -0
  276. package/dist/mjs/types/websockets/ws-general.js.map +1 -0
  277. package/dist/mjs/util/BaseRestClient.d.ts +40 -0
  278. package/dist/mjs/util/BaseRestClient.js +268 -0
  279. package/dist/mjs/util/BaseRestClient.js.map +1 -0
  280. package/dist/mjs/util/BaseWSClient.d.ts +197 -0
  281. package/dist/mjs/util/BaseWSClient.js +752 -0
  282. package/dist/mjs/util/BaseWSClient.js.map +1 -0
  283. package/dist/mjs/util/WsStore.d.ts +74 -0
  284. package/dist/mjs/util/WsStore.js +282 -0
  285. package/dist/mjs/util/WsStore.js.map +1 -0
  286. package/dist/mjs/util/WsStore.types.d.ts +52 -0
  287. package/dist/mjs/util/WsStore.types.js +11 -0
  288. package/dist/mjs/util/WsStore.types.js.map +1 -0
  289. package/dist/mjs/util/logger.d.ts +8 -0
  290. package/dist/mjs/util/logger.js +14 -0
  291. package/dist/mjs/util/logger.js.map +1 -0
  292. package/dist/mjs/util/requestUtils.d.ts +61 -0
  293. package/dist/mjs/util/requestUtils.js +50 -0
  294. package/dist/mjs/util/requestUtils.js.map +1 -0
  295. package/dist/mjs/util/type-guards.d.ts +16 -0
  296. package/dist/mjs/util/type-guards.js +62 -0
  297. package/dist/mjs/util/type-guards.js.map +1 -0
  298. package/dist/mjs/util/webCryptoAPI.d.ts +14 -0
  299. package/dist/mjs/util/webCryptoAPI.js +76 -0
  300. package/dist/mjs/util/webCryptoAPI.js.map +1 -0
  301. package/dist/mjs/util/websocket-util.d.ts +88 -0
  302. package/dist/mjs/util/websocket-util.js +227 -0
  303. package/dist/mjs/util/websocket-util.js.map +1 -0
  304. package/dist/mjs/websocket-api-client.d.ts +89 -0
  305. package/dist/mjs/websocket-api-client.js +118 -0
  306. package/dist/mjs/websocket-api-client.js.map +1 -0
  307. package/{lib/util/BaseWSClient.d.ts → dist/mjs/websocket-client-legacy-v1.d.ts} +44 -27
  308. package/dist/mjs/websocket-client-legacy-v1.js +502 -0
  309. package/dist/mjs/websocket-client-legacy-v1.js.map +1 -0
  310. package/dist/mjs/websocket-client-v2.d.ts +79 -0
  311. package/dist/mjs/websocket-client-v2.js +335 -0
  312. package/dist/mjs/websocket-client-v2.js.map +1 -0
  313. package/dist/mjs/websocket-client-v3.d.ts +84 -0
  314. package/dist/mjs/websocket-client-v3.js +404 -0
  315. package/dist/mjs/websocket-client-v3.js.map +1 -0
  316. package/package.json +33 -15
  317. package/index.js +0 -1
  318. package/lib/broker-client.js.map +0 -1
  319. package/lib/constants/enum.js.map +0 -1
  320. package/lib/futures-client.js.map +0 -1
  321. package/lib/index.d.ts +0 -10
  322. package/lib/index.js +0 -27
  323. package/lib/index.js.map +0 -1
  324. package/lib/rest-client-v2.js.map +0 -1
  325. package/lib/spot-client.js.map +0 -1
  326. package/lib/types/index.d.ts +0 -4
  327. package/lib/types/index.js +0 -21
  328. package/lib/types/index.js.map +0 -1
  329. package/lib/types/request/index.d.ts +0 -11
  330. package/lib/types/request/index.js +0 -28
  331. package/lib/types/request/index.js.map +0 -1
  332. package/lib/types/request/v1/brokerV1.js.map +0 -1
  333. package/lib/types/request/v1/futuresV1.js.map +0 -1
  334. package/lib/types/request/v1/spotV1.js.map +0 -1
  335. package/lib/types/request/v2/broker.js.map +0 -1
  336. package/lib/types/request/v2/common.js.map +0 -1
  337. package/lib/types/request/v2/copytrading.js.map +0 -1
  338. package/lib/types/request/v2/earn.js.map +0 -1
  339. package/lib/types/request/v2/futures.js.map +0 -1
  340. package/lib/types/request/v2/margin.js.map +0 -1
  341. package/lib/types/request/v2/spot.js.map +0 -1
  342. package/lib/types/response/index.d.ts +0 -10
  343. package/lib/types/response/index.js +0 -27
  344. package/lib/types/response/index.js.map +0 -1
  345. package/lib/types/response/v1/futures.js.map +0 -1
  346. package/lib/types/response/v1/shared.js.map +0 -1
  347. package/lib/types/response/v1/spot.js.map +0 -1
  348. package/lib/types/response/v2/broker.js.map +0 -1
  349. package/lib/types/response/v2/common.js.map +0 -1
  350. package/lib/types/response/v2/copy-trading.js.map +0 -1
  351. package/lib/types/response/v2/earn.js.map +0 -1
  352. package/lib/types/response/v2/futures.js.map +0 -1
  353. package/lib/types/response/v2/margin.js.map +0 -1
  354. package/lib/types/response/v2/spot.js.map +0 -1
  355. package/lib/types/shared.d.ts +0 -8
  356. package/lib/types/websockets/client.d.ts +0 -76
  357. package/lib/types/websockets/client.js +0 -3
  358. package/lib/types/websockets/client.js.map +0 -1
  359. package/lib/types/websockets/events.js +0 -3
  360. package/lib/types/websockets/events.js.map +0 -1
  361. package/lib/types/websockets/index.d.ts +0 -2
  362. package/lib/types/websockets/index.js +0 -19
  363. package/lib/types/websockets/index.js.map +0 -1
  364. package/lib/util/BaseRestClient.js +0 -221
  365. package/lib/util/BaseRestClient.js.map +0 -1
  366. package/lib/util/BaseWSClient.js +0 -395
  367. package/lib/util/BaseWSClient.js.map +0 -1
  368. package/lib/util/WsStore.d.ts +0 -26
  369. package/lib/util/WsStore.js +0 -126
  370. package/lib/util/WsStore.js.map +0 -1
  371. package/lib/util/WsStore.types.js.map +0 -1
  372. package/lib/util/browser-support.d.ts +0 -1
  373. package/lib/util/browser-support.js +0 -45
  374. package/lib/util/browser-support.js.map +0 -1
  375. package/lib/util/index.d.ts +0 -6
  376. package/lib/util/index.js +0 -23
  377. package/lib/util/index.js.map +0 -1
  378. package/lib/util/logger.d.ts +0 -9
  379. package/lib/util/logger.js.map +0 -1
  380. package/lib/util/node-support.d.ts +0 -2
  381. package/lib/util/node-support.js +0 -34
  382. package/lib/util/node-support.js.map +0 -1
  383. package/lib/util/requestUtils.js.map +0 -1
  384. package/lib/util/type-guards.js.map +0 -1
  385. package/lib/util/websocket-util.d.ts +0 -47
  386. package/lib/util/websocket-util.js +0 -154
  387. package/lib/util/websocket-util.js.map +0 -1
  388. package/lib/websocket-client-v2.d.ts +0 -33
  389. package/lib/websocket-client-v2.js +0 -104
  390. package/lib/websocket-client-v2.js.map +0 -1
  391. package/lib/websocket-client.js.map +0 -1
  392. /package/{lib → dist/cjs}/types/request/shared.js +0 -0
  393. /package/{lib → dist/cjs}/types/request/v1/brokerV1.js +0 -0
  394. /package/{lib → dist/cjs}/types/request/v1/futuresV1.js +0 -0
  395. /package/{lib → dist/cjs}/types/request/v1/spotV1.js +0 -0
  396. /package/{lib → dist/cjs}/types/request/v2/broker.js +0 -0
  397. /package/{lib → dist/cjs}/types/request/v2/common.js +0 -0
  398. /package/{lib → dist/cjs}/types/request/v2/copytrading.js +0 -0
  399. /package/{lib → dist/cjs}/types/request/v2/earn.d.ts +0 -0
  400. /package/{lib → dist/cjs}/types/request/v2/earn.js +0 -0
  401. /package/{lib → dist/cjs}/types/request/v2/futures.js +0 -0
  402. /package/{lib → dist/cjs}/types/request/v2/margin.js +0 -0
  403. /package/{lib → dist/cjs}/types/request/v2/spot.js +0 -0
  404. /package/{lib → dist/cjs}/types/response/v1/futures.js +0 -0
  405. /package/{lib → dist/cjs}/types/response/v1/shared.d.ts +0 -0
  406. /package/{lib → dist/cjs}/types/response/v1/shared.js +0 -0
  407. /package/{lib → dist/cjs}/types/response/v1/spot.d.ts +0 -0
  408. /package/{lib → dist/cjs}/types/response/v1/spot.js +0 -0
  409. /package/{lib → dist/cjs}/types/response/v2/broker.d.ts +0 -0
  410. /package/{lib → dist/cjs}/types/response/v2/broker.js +0 -0
  411. /package/{lib → dist/cjs}/types/response/v2/common.d.ts +0 -0
  412. /package/{lib → dist/cjs}/types/response/v2/common.js +0 -0
  413. /package/{lib → dist/cjs}/types/response/v2/copy-trading.d.ts +0 -0
  414. /package/{lib → dist/cjs}/types/response/v2/copy-trading.js +0 -0
  415. /package/{lib → dist/cjs}/types/response/v2/earn.d.ts +0 -0
  416. /package/{lib → dist/cjs}/types/response/v2/earn.js +0 -0
  417. /package/{lib → dist/cjs}/types/response/v2/futures.js +0 -0
  418. /package/{lib → dist/cjs}/types/response/v2/margin.d.ts +0 -0
  419. /package/{lib → dist/cjs}/types/response/v2/margin.js +0 -0
  420. /package/{lib → dist/cjs}/types/response/v2/spot.js +0 -0
  421. /package/{lib → dist/cjs}/types/shared.js +0 -0
@@ -0,0 +1,752 @@
1
+ /* eslint-disable @typescript-eslint/no-unsafe-declaration-merging */
2
+ import EventEmitter from 'events';
3
+ import WebSocket from 'isomorphic-ws';
4
+ import { isMessageEvent, } from '../types/websockets/ws-events.js';
5
+ import { DefaultLogger } from './logger.js';
6
+ import { getNormalisedTopicRequests, safeTerminateWs, WS_LOGGER_CATEGORY, } from './websocket-util.js';
7
+ import WsStore from './WsStore.js';
8
+ import { WsConnectionStateEnum } from './WsStore.types.js';
9
+ /**
10
+ * Appends wsKey and isWSAPIResponse to all events.
11
+ * Some events are arrays, this handles that nested scenario too.
12
+ */
13
+ function getFinalEmittable(emittable, wsKey, isWSAPIResponse) {
14
+ if (Array.isArray(emittable)) {
15
+ return emittable.map((subEvent) => getFinalEmittable(subEvent, wsKey, isWSAPIResponse));
16
+ }
17
+ if (Array.isArray(emittable.event)) {
18
+ // Some topics just emit an array.
19
+ // This is consistent with how it was before the WS API upgrade:
20
+ return emittable.event.map((subEvent) => getFinalEmittable(subEvent, wsKey, isWSAPIResponse));
21
+ // const { event, ...others } = emittable;
22
+ // return {
23
+ // ...others,
24
+ // event: event.map((subEvent) =>
25
+ // getFinalEmittable(subEvent, wsKey, isWSAPIResponse),
26
+ // ),
27
+ // };
28
+ }
29
+ if (emittable.event) {
30
+ return {
31
+ ...emittable.event,
32
+ wsKey: wsKey,
33
+ isWSAPIResponse: !!isWSAPIResponse,
34
+ };
35
+ }
36
+ return {
37
+ ...emittable,
38
+ wsKey: wsKey,
39
+ isWSAPIResponse: !!isWSAPIResponse,
40
+ };
41
+ }
42
+ export class BaseWebsocketClient extends EventEmitter {
43
+ /**
44
+ * State store to track a list of topics (topic requests) we are expected to be subscribed to if reconnected
45
+ */
46
+ wsStore;
47
+ logger;
48
+ options;
49
+ wsApiRequestId = 0;
50
+ timeOffsetMs = 0;
51
+ /**
52
+ * { [wsKey]: { [requestId]: request } }
53
+ */
54
+ midflightRequestCache = {};
55
+ constructor(options, logger) {
56
+ super();
57
+ this.logger = logger || DefaultLogger;
58
+ this.wsStore = new WsStore(this.logger);
59
+ this.options = {
60
+ demoTrading: false,
61
+ pongTimeout: 1000,
62
+ pingInterval: 10000,
63
+ reconnectTimeout: 500,
64
+ recvWindow: 0,
65
+ // Automatically send an authentication op/request after a connection opens, for private connections.
66
+ authPrivateConnectionsOnConnect: true,
67
+ // Individual requests do not require a signature, so this is disabled.
68
+ authPrivateRequests: false,
69
+ ...options,
70
+ };
71
+ }
72
+ isPrivateWsKey(wsKey) {
73
+ return this.getPrivateWSKeys().includes(wsKey);
74
+ }
75
+ /** Returns auto-incrementing request ID, used to track promise references for async requests */
76
+ getNewRequestId() {
77
+ return ++this.wsApiRequestId;
78
+ }
79
+ getTimeOffsetMs() {
80
+ return this.timeOffsetMs;
81
+ }
82
+ setTimeOffsetMs(newOffset) {
83
+ this.timeOffsetMs = newOffset;
84
+ }
85
+ /**
86
+ * Don't call directly! Use subscribe() instead!
87
+ *
88
+ * Subscribe to one or more topics on a WS connection (identified by WS Key).
89
+ *
90
+ * - Topics are automatically cached
91
+ * - Connections are automatically opened, if not yet connected
92
+ * - Authentication is automatically handled
93
+ * - Topics are automatically resubscribed to, if something happens to the connection, unless you call unsubsribeTopicsForWsKey(topics, key).
94
+ *
95
+ * @param wsRequests array of topics to subscribe to
96
+ * @param wsKey ws key referring to the ws connection these topics should be subscribed on
97
+ */
98
+ async subscribeTopicsForWsKey(wsTopicRequests, wsKey) {
99
+ const normalisedTopicRequests = getNormalisedTopicRequests(wsTopicRequests);
100
+ // Store topics, so future automation (post-auth, post-reconnect) has everything needed to resubscribe automatically
101
+ for (const topic of normalisedTopicRequests) {
102
+ this.wsStore.addTopic(wsKey, topic);
103
+ }
104
+ const isConnected = this.wsStore.isConnectionState(wsKey, WsConnectionStateEnum.CONNECTED);
105
+ const isConnectionInProgress = this.wsStore.isConnectionAttemptInProgress(wsKey);
106
+ // start connection process if it hasn't yet begun. Topics are automatically subscribed to on-connect
107
+ if (!isConnected && !isConnectionInProgress) {
108
+ return this.connect(wsKey);
109
+ }
110
+ // Subscribe should happen automatically once connected, nothing to do here after topics are added to wsStore.
111
+ if (!isConnected) {
112
+ /**
113
+ * Are we in the process of connection? Nothing to send yet.
114
+ */
115
+ this.logger.trace('WS not connected - requests queued for retry once connected.', {
116
+ ...WS_LOGGER_CATEGORY,
117
+ wsKey,
118
+ wsTopicRequests,
119
+ });
120
+ return isConnectionInProgress;
121
+ }
122
+ // We're connected. Check if auth is needed and if already authenticated
123
+ const isPrivateConnection = this.isPrivateWsKey(wsKey);
124
+ const isAuthenticated = this.wsStore.get(wsKey)?.isAuthenticated;
125
+ if (isPrivateConnection && !isAuthenticated) {
126
+ /**
127
+ * If not authenticated yet and auth is required, don't request topics yet.
128
+ *
129
+ * Auth should already automatically be in progress, so no action needed from here. Topics will automatically subscribe post-auth success.
130
+ */
131
+ return false;
132
+ }
133
+ // Finally, request subscription to topics if the connection is healthy and ready
134
+ return this.requestSubscribeTopics(wsKey, normalisedTopicRequests);
135
+ }
136
+ async unsubscribeTopicsForWsKey(wsTopicRequests, wsKey) {
137
+ const normalisedTopicRequests = getNormalisedTopicRequests(wsTopicRequests);
138
+ // Store topics, so future automation (post-auth, post-reconnect) has everything needed to resubscribe automatically
139
+ for (const topic of normalisedTopicRequests) {
140
+ this.wsStore.deleteTopic(wsKey, topic);
141
+ }
142
+ const isConnected = this.wsStore.isConnectionState(wsKey, WsConnectionStateEnum.CONNECTED);
143
+ // If not connected, don't need to do anything.
144
+ // Removing the topic from the store is enough to stop it from being resubscribed to on reconnect.
145
+ if (!isConnected) {
146
+ return;
147
+ }
148
+ // We're connected. Check if auth is needed and if already authenticated
149
+ const isPrivateConnection = this.isPrivateWsKey(wsKey);
150
+ const isAuthenticated = this.wsStore.get(wsKey)?.isAuthenticated;
151
+ if (isPrivateConnection && !isAuthenticated) {
152
+ /**
153
+ * If not authenticated yet and auth is required, don't need to do anything.
154
+ * We don't subscribe to topics until auth is complete anyway.
155
+ */
156
+ return;
157
+ }
158
+ // Finally, request subscription to topics if the connection is healthy and ready
159
+ return this.requestUnsubscribeTopics(wsKey, normalisedTopicRequests);
160
+ }
161
+ /**
162
+ * Splits topic requests into two groups, public & private topic requests
163
+ */
164
+ sortTopicRequestsIntoPublicPrivate(wsTopicRequests, wsKey) {
165
+ const publicTopicRequests = [];
166
+ const privateTopicRequests = [];
167
+ for (const topic of wsTopicRequests) {
168
+ if (this.isPrivateTopicRequest(topic, wsKey)) {
169
+ privateTopicRequests.push(topic);
170
+ }
171
+ else {
172
+ publicTopicRequests.push(topic);
173
+ }
174
+ }
175
+ return {
176
+ publicReqs: publicTopicRequests,
177
+ privateReqs: privateTopicRequests,
178
+ };
179
+ }
180
+ /** Get the WsStore that tracks websockets & topics */
181
+ getWsStore() {
182
+ return this.wsStore;
183
+ }
184
+ close(wsKey, force) {
185
+ this.logger.info('Closing connection', { ...WS_LOGGER_CATEGORY, wsKey });
186
+ this.setWsState(wsKey, WsConnectionStateEnum.CLOSING);
187
+ this.clearTimers(wsKey);
188
+ const ws = this.getWs(wsKey);
189
+ ws?.close();
190
+ if (force) {
191
+ safeTerminateWs(ws);
192
+ }
193
+ }
194
+ closeAll(force) {
195
+ const keys = this.wsStore.getKeys();
196
+ this.logger.info(`Closing all ws connections: ${keys}`);
197
+ keys.forEach((key) => {
198
+ this.close(key, force);
199
+ });
200
+ }
201
+ isConnected(wsKey) {
202
+ return this.wsStore.isConnectionState(wsKey, WsConnectionStateEnum.CONNECTED);
203
+ }
204
+ /**
205
+ * Request connection to a specific websocket, instead of waiting for automatic connection.
206
+ */
207
+ async connect(wsKey, customUrl, throwOnError) {
208
+ try {
209
+ if (this.wsStore.isWsOpen(wsKey)) {
210
+ this.logger.error('Refused to connect to ws with existing active connection', { ...WS_LOGGER_CATEGORY, wsKey });
211
+ return { wsKey, ws: this.wsStore.getWs(wsKey) };
212
+ }
213
+ if (this.wsStore.isConnectionState(wsKey, WsConnectionStateEnum.CONNECTING)) {
214
+ this.logger.error('Refused to connect to ws, connection attempt already active', { ...WS_LOGGER_CATEGORY, wsKey });
215
+ return this.wsStore.getConnectionInProgressPromise(wsKey)?.promise;
216
+ }
217
+ if (!this.wsStore.getConnectionState(wsKey) ||
218
+ this.wsStore.isConnectionState(wsKey, WsConnectionStateEnum.INITIAL)) {
219
+ this.setWsState(wsKey, WsConnectionStateEnum.CONNECTING);
220
+ }
221
+ if (!this.wsStore.getConnectionInProgressPromise(wsKey)) {
222
+ this.wsStore.createConnectionInProgressPromise(wsKey, false);
223
+ }
224
+ const url = customUrl || (await this.getWsUrl(wsKey));
225
+ const ws = this.connectToWsUrl(url, wsKey);
226
+ this.wsStore.setWs(wsKey, ws);
227
+ }
228
+ catch (err) {
229
+ this.parseWsError('Connection failed', err, wsKey);
230
+ this.reconnectWithDelay(wsKey, this.options.reconnectTimeout);
231
+ if (throwOnError) {
232
+ throw err;
233
+ }
234
+ }
235
+ return this.wsStore.getConnectionInProgressPromise(wsKey)?.promise;
236
+ }
237
+ connectToWsUrl(url, wsKey) {
238
+ this.logger.trace(`Opening WS connection to URL: ${url}`, {
239
+ ...WS_LOGGER_CATEGORY,
240
+ wsKey,
241
+ });
242
+ const { protocols = [], ...wsOptions } = this.options.wsOptions || {};
243
+ const ws = new WebSocket(url, protocols, wsOptions);
244
+ ws.onopen = (event) => this.onWsOpen(event, wsKey, url, ws);
245
+ ws.onmessage = (event) => this.onWsMessage(event, wsKey, ws);
246
+ ws.onerror = (event) => this.parseWsError('Websocket onWsError', event, wsKey);
247
+ ws.onclose = (event) => this.onWsClose(event, wsKey);
248
+ // Native ws ping/pong frames are not in use for bitget
249
+ // if (typeof ws.on === 'function') {
250
+ // ws.on('ping', (event) => this.onWsPing(event, wsKey, ws, 'event'));
251
+ // ws.on('pong', (event) => this.onWsPong(event, wsKey, 'event'));
252
+ // }
253
+ // // Not sure these work in the browser, the traditional event listeners are required for ping/pong frames in node
254
+ // ws.onping = (event) => this.onWsPing(event, wsKey, ws, 'function');
255
+ // ws.onpong = (event) => this.onWsPong(event, wsKey, 'function');
256
+ ws.wsKey = wsKey;
257
+ return ws;
258
+ }
259
+ parseWsError(context, error, wsKey) {
260
+ if (!error.message) {
261
+ this.logger.error(`${context} due to unexpected error: `, error);
262
+ this.emit('response', { ...error, wsKey });
263
+ this.emit('exception', { ...error, wsKey });
264
+ return;
265
+ }
266
+ switch (error.message) {
267
+ case 'Unexpected server response: 401':
268
+ this.logger.error(`${context} due to 401 authorization failure.`, {
269
+ ...WS_LOGGER_CATEGORY,
270
+ wsKey,
271
+ });
272
+ break;
273
+ default:
274
+ this.logger.error(`${context} due to unexpected response error: "${error?.msg || error?.message || error}"`, { ...WS_LOGGER_CATEGORY, wsKey, error });
275
+ break;
276
+ }
277
+ this.emit('response', { ...error, wsKey });
278
+ this.emit('exception', { ...error, wsKey });
279
+ }
280
+ /** Get a signature, build the auth request and send it */
281
+ async sendAuthRequest(wsKey) {
282
+ try {
283
+ this.logger.trace('Sending auth request...', {
284
+ ...WS_LOGGER_CATEGORY,
285
+ wsKey,
286
+ });
287
+ await this.assertIsConnected(wsKey);
288
+ if (!this.wsStore.getAuthenticationInProgressPromise(wsKey)) {
289
+ this.wsStore.createAuthenticationInProgressPromise(wsKey, false);
290
+ }
291
+ const request = await this.getWsAuthRequestEvent(wsKey);
292
+ // console.log('ws auth req', request);
293
+ this.tryWsSend(wsKey, JSON.stringify(request));
294
+ return this.wsStore.getAuthenticationInProgressPromise(wsKey)?.promise;
295
+ }
296
+ catch (e) {
297
+ this.logger.trace(e, { ...WS_LOGGER_CATEGORY, wsKey });
298
+ }
299
+ }
300
+ reconnectWithDelay(wsKey, connectionDelayMs) {
301
+ this.clearTimers(wsKey);
302
+ if (this.wsStore.getConnectionState(wsKey) !==
303
+ WsConnectionStateEnum.CONNECTING) {
304
+ this.setWsState(wsKey, WsConnectionStateEnum.RECONNECTING);
305
+ }
306
+ this.wsStore.get(wsKey, true).activeReconnectTimer = setTimeout(() => {
307
+ this.logger.info('Reconnecting to websocket', {
308
+ ...WS_LOGGER_CATEGORY,
309
+ wsKey,
310
+ });
311
+ this.connect(wsKey);
312
+ }, connectionDelayMs);
313
+ }
314
+ ping(wsKey) {
315
+ if (this.wsStore.get(wsKey, true).activePongTimer) {
316
+ return;
317
+ }
318
+ this.clearPongTimer(wsKey);
319
+ this.logger.trace('Sending ping', { ...WS_LOGGER_CATEGORY, wsKey });
320
+ this.tryWsSend(wsKey, 'ping');
321
+ this.wsStore.get(wsKey, true).activePongTimer = setTimeout(() => {
322
+ this.logger.info('Pong timeout - closing socket to reconnect', {
323
+ ...WS_LOGGER_CATEGORY,
324
+ wsKey,
325
+ });
326
+ safeTerminateWs(this.getWs(wsKey), true);
327
+ delete this.wsStore.get(wsKey, true).activePongTimer;
328
+ }, this.options.pongTimeout);
329
+ }
330
+ clearTimers(wsKey) {
331
+ this.clearPingTimer(wsKey);
332
+ this.clearPongTimer(wsKey);
333
+ const wsState = this.wsStore.get(wsKey);
334
+ if (wsState?.activeReconnectTimer) {
335
+ clearTimeout(wsState.activeReconnectTimer);
336
+ }
337
+ }
338
+ // Send a ping at intervals
339
+ clearPingTimer(wsKey) {
340
+ const wsState = this.wsStore.get(wsKey);
341
+ if (wsState?.activePingTimer) {
342
+ clearInterval(wsState.activePingTimer);
343
+ wsState.activePingTimer = undefined;
344
+ }
345
+ }
346
+ // Expect a pong within a time limit
347
+ clearPongTimer(wsKey) {
348
+ const wsState = this.wsStore.get(wsKey);
349
+ if (wsState?.activePongTimer) {
350
+ clearTimeout(wsState.activePongTimer);
351
+ wsState.activePongTimer = undefined;
352
+ }
353
+ }
354
+ /**
355
+ * Returns a list of string events that can be individually sent upstream to complete subscribing/unsubscribing/etc to these topics
356
+ *
357
+ * If events are an object, these should be stringified (`return JSON.stringify(event);`)
358
+ * Each event returned by this will be sent one at a time
359
+ *
360
+ * Events are automatically split into smaller batches, by this method, if needed.
361
+ */
362
+ async getWsOperationEventsForTopics(topics, wsKey, operation) {
363
+ if (!topics.length) {
364
+ return [];
365
+ }
366
+ // Events that are ready to send (usually stringified JSON)
367
+ const requestEvents = [];
368
+ const maxTopicsPerEvent = this.getMaxTopicsPerSubscribeEvent(wsKey);
369
+ if (maxTopicsPerEvent &&
370
+ maxTopicsPerEvent !== null &&
371
+ topics.length > maxTopicsPerEvent) {
372
+ for (let i = 0; i < topics.length; i += maxTopicsPerEvent) {
373
+ const batch = topics.slice(i, i + maxTopicsPerEvent);
374
+ const subscribeRequestEvents = await this.getWsRequestEvents(operation, batch, wsKey);
375
+ requestEvents.push(...subscribeRequestEvents);
376
+ }
377
+ return requestEvents;
378
+ }
379
+ const subscribeRequestEvents = await this.getWsRequestEvents(operation, topics, wsKey);
380
+ return subscribeRequestEvents;
381
+ }
382
+ /**
383
+ * @private Use the `subscribe(topics)` method to subscribe to topics. Send WS message to subscribe to topics.
384
+ */
385
+ async requestSubscribeTopics(wsKey, wsTopicRequests) {
386
+ if (!wsTopicRequests.length) {
387
+ return;
388
+ }
389
+ // Automatically splits requests into smaller batches, if needed
390
+ const subscribeWsMessages = await this.getWsOperationEventsForTopics(wsTopicRequests, wsKey, 'subscribe');
391
+ this.logger.trace(`Subscribing to ${wsTopicRequests.length} "${wsKey}" topics in ${subscribeWsMessages.length} batches.`);
392
+ // console.log(`batches: `, JSON.stringify(subscribeWsMessages, null, 2));
393
+ for (const midflightRequest of subscribeWsMessages) {
394
+ const wsMessage = midflightRequest.requestEvent;
395
+ if (!this.midflightRequestCache[wsKey]) {
396
+ this.midflightRequestCache[wsKey] = {};
397
+ }
398
+ // Cache the request for this call, so we can enrich the response with request info
399
+ // this.midflightRequestCache[wsKey][midflightRequest.requestKey] =
400
+ // midflightRequest.requestEvent;
401
+ this.logger.trace(
402
+ // `Sending batch via message: "${JSON.stringify(wsMessage)}", cached with key "${midflightRequest.requestKey}"`,
403
+ `Sending batch via message: "${JSON.stringify(wsMessage)}"`);
404
+ try {
405
+ this.tryWsSend(wsKey, JSON.stringify(wsMessage), true);
406
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
407
+ }
408
+ catch (e) {
409
+ delete this.midflightRequestCache[wsKey][midflightRequest.requestKey];
410
+ }
411
+ }
412
+ }
413
+ /**
414
+ * @private Use the `unsubscribe(topics)` method to unsubscribe from topics. Send WS message to unsubscribe from topics.
415
+ */
416
+ async requestUnsubscribeTopics(wsKey, wsTopicRequests) {
417
+ if (!wsTopicRequests.length) {
418
+ return;
419
+ }
420
+ const subscribeWsMessages = await this.getWsOperationEventsForTopics(wsTopicRequests, wsKey, 'unsubscribe');
421
+ this.logger.trace(`Unsubscribing to ${wsTopicRequests.length} "${wsKey}" topics in ${subscribeWsMessages.length} batches. Events: "${JSON.stringify(wsTopicRequests)}"`);
422
+ for (const midflightRequest of subscribeWsMessages) {
423
+ const wsMessage = midflightRequest.requestEvent;
424
+ if (!this.midflightRequestCache[wsKey]) {
425
+ this.midflightRequestCache[wsKey] = {};
426
+ }
427
+ // Cache the request for this call, so we can enrich the response with request info
428
+ // this.midflightRequestCache[wsKey][midflightRequest.requestKey] =
429
+ // midflightRequest.requestEvent;
430
+ this.logger.trace(`Sending batch via message: "${wsMessage}"`);
431
+ try {
432
+ this.tryWsSend(wsKey, JSON.stringify(wsMessage));
433
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
434
+ }
435
+ catch (e) {
436
+ delete this.midflightRequestCache[wsKey][midflightRequest.requestKey];
437
+ }
438
+ }
439
+ this.logger.trace(`Finished unsubscribing to ${wsTopicRequests.length} "${wsKey}" topics in ${subscribeWsMessages.length} batches.`);
440
+ // const wsMessage = JSON.stringify({
441
+ // op: 'unsubscribe',
442
+ // args: wsTopicRequests,
443
+ // });
444
+ }
445
+ getCachedMidFlightRequest(wsKey, requestKey) {
446
+ if (!this.midflightRequestCache[wsKey]) {
447
+ this.midflightRequestCache[wsKey] = {};
448
+ }
449
+ return this.midflightRequestCache[wsKey][requestKey];
450
+ }
451
+ // Not in use for Bitget. If desired, call from resolveEmittableEvents() for WS API responses.
452
+ // See binance SDK for reference
453
+ removeCachedMidFlightRequest(wsKey, requestKey) {
454
+ if (this.getCachedMidFlightRequest(wsKey, requestKey)) {
455
+ delete this.midflightRequestCache[wsKey][requestKey];
456
+ }
457
+ }
458
+ tryWsSend(wsKey, wsMessage, throwExceptions) {
459
+ try {
460
+ this.logger.trace('Sending upstream ws message: ', {
461
+ ...WS_LOGGER_CATEGORY,
462
+ wsMessage,
463
+ wsKey,
464
+ });
465
+ if (!wsKey) {
466
+ throw new Error('Cannot send message due to no known websocket for this wsKey');
467
+ }
468
+ const ws = this.getWs(wsKey);
469
+ if (!ws) {
470
+ throw new Error(`${wsKey} socket not connected yet, call "connectAll()" first then try again when the "open" event arrives`);
471
+ }
472
+ ws.send(wsMessage);
473
+ }
474
+ catch (e) {
475
+ this.logger.error('Failed to send WS message', {
476
+ ...WS_LOGGER_CATEGORY,
477
+ wsMessage,
478
+ wsKey,
479
+ exception: e,
480
+ });
481
+ if (throwExceptions) {
482
+ throw e;
483
+ }
484
+ }
485
+ }
486
+ async onWsOpen(event, wsKey, url, ws) {
487
+ const isFreshConnectionAttempt = this.wsStore.isConnectionState(wsKey, WsConnectionStateEnum.CONNECTING);
488
+ const isReconnectionAttempt = this.wsStore.isConnectionState(wsKey, WsConnectionStateEnum.RECONNECTING);
489
+ if (isFreshConnectionAttempt) {
490
+ this.logger.info('Websocket connected', {
491
+ ...WS_LOGGER_CATEGORY,
492
+ wsKey,
493
+ });
494
+ this.emit('open', { wsKey, event, wsUrl: url, ws });
495
+ }
496
+ else if (isReconnectionAttempt) {
497
+ this.logger.info('Websocket reconnected', {
498
+ ...WS_LOGGER_CATEGORY,
499
+ wsKey,
500
+ });
501
+ this.emit('reconnected', { wsKey, event, wsUrl: url, ws });
502
+ }
503
+ this.setWsState(wsKey, WsConnectionStateEnum.CONNECTED);
504
+ this.logger.trace('Enabled ping timer', { ...WS_LOGGER_CATEGORY, wsKey });
505
+ this.wsStore.get(wsKey, true).activePingTimer = setInterval(() => this.ping(wsKey), this.options.pingInterval);
506
+ // Resolve & cleanup deferred "connection attempt in progress" promise
507
+ try {
508
+ const connectionInProgressPromise = this.wsStore.getConnectionInProgressPromise(wsKey);
509
+ if (connectionInProgressPromise?.resolve) {
510
+ connectionInProgressPromise.resolve({
511
+ wsKey,
512
+ ws,
513
+ });
514
+ }
515
+ }
516
+ catch (e) {
517
+ this.logger.error('Exception trying to resolve "connectionInProgress" promise', e);
518
+ }
519
+ // Remove before continuing, in case there's more requests queued
520
+ this.wsStore.removeConnectingInProgressPromise(wsKey);
521
+ // Some websockets require an auth packet to be sent after opening the connection
522
+ if (this.isAuthOnConnectWsKey(wsKey) &&
523
+ this.options.authPrivateConnectionsOnConnect) {
524
+ await this.assertIsAuthenticated(wsKey);
525
+ }
526
+ // Reconnect to topics known before it connected
527
+ const { privateReqs, publicReqs } = this.sortTopicRequestsIntoPublicPrivate([...this.wsStore.getTopics(wsKey)], wsKey);
528
+ // Request sub to public topics, if any
529
+ this.requestSubscribeTopics(wsKey, publicReqs);
530
+ // Request sub to private topics, if auth on connect isn't needed
531
+ // Else, this is automatic after authentication is successfully confirmed
532
+ if (!this.options.authPrivateConnectionsOnConnect) {
533
+ this.requestSubscribeTopics(wsKey, privateReqs);
534
+ }
535
+ }
536
+ /**
537
+ * Handle subscription to private topics _after_ authentication successfully completes asynchronously.
538
+ *
539
+ * Only used for exchanges that require auth before sending private topic subscription requests
540
+ */
541
+ onWsAuthenticated(wsKey, event) {
542
+ const wsState = this.wsStore.get(wsKey, true);
543
+ wsState.isAuthenticated = true;
544
+ // Resolve & cleanup deferred "auth attempt in progress" promise
545
+ try {
546
+ const inProgressPromise = this.wsStore.getAuthenticationInProgressPromise(wsKey);
547
+ if (inProgressPromise?.resolve) {
548
+ inProgressPromise.resolve({
549
+ wsKey,
550
+ event,
551
+ ws: wsState.ws,
552
+ });
553
+ }
554
+ }
555
+ catch (e) {
556
+ this.logger.error('Exception trying to resolve "authenticationInProgress" promise', e);
557
+ }
558
+ // Remove before continuing, in case there's more requests queued
559
+ this.wsStore.removeAuthenticationInProgressPromise(wsKey);
560
+ if (this.options.authPrivateConnectionsOnConnect) {
561
+ const topics = [...this.wsStore.getTopics(wsKey)];
562
+ const privateTopics = topics.filter((topic) => this.isPrivateTopicRequest(topic, wsKey));
563
+ if (privateTopics.length) {
564
+ this.subscribeTopicsForWsKey(privateTopics, wsKey);
565
+ }
566
+ }
567
+ }
568
+ /**
569
+ * Raw incoming event handler. Parsing happens in integration layer via resolveEmittableEvents().
570
+ */
571
+ onWsMessage(event, wsKey, ws) {
572
+ try {
573
+ // console.log('onMessageRaw: ', (event as any).data);
574
+ // any message can clear the pong timer - wouldn't get a message if the ws wasn't working
575
+ this.clearPongTimer(wsKey);
576
+ if (this.isWsPong(event)) {
577
+ this.logger.trace('Received pong', {
578
+ ...WS_LOGGER_CATEGORY,
579
+ wsKey,
580
+ event: event?.data,
581
+ });
582
+ return;
583
+ }
584
+ if (this.isWsPing(event)) {
585
+ this.logger.trace('Received ping', {
586
+ ...WS_LOGGER_CATEGORY,
587
+ wsKey,
588
+ event,
589
+ });
590
+ this.sendPongEvent(wsKey, ws);
591
+ return;
592
+ }
593
+ if (isMessageEvent(event)) {
594
+ const data = event.data;
595
+ const dataType = event.type;
596
+ const emittableEvents = this.resolveEmittableEvents(wsKey, event);
597
+ if (!emittableEvents.length) {
598
+ // console.log(`raw event: `, { data, dataType, emittableEvents });
599
+ this.logger.error('Unhandled/unrecognised ws event message - returned no emittable data', {
600
+ ...WS_LOGGER_CATEGORY,
601
+ message: data || 'no message',
602
+ dataType,
603
+ event,
604
+ wsKey,
605
+ });
606
+ return this.emit('update', { ...event, wsKey });
607
+ }
608
+ for (const emittable of emittableEvents) {
609
+ if (this.isWsPong(emittable)) {
610
+ this.logger.trace('Received pong2', {
611
+ ...WS_LOGGER_CATEGORY,
612
+ wsKey,
613
+ data,
614
+ });
615
+ continue;
616
+ }
617
+ // this.logger.trace(
618
+ // 'getFinalEmittable()->pre(): ',
619
+ // JSON.stringify(emittable),
620
+ // );
621
+ const emittableFinalEvent = getFinalEmittable(emittable, wsKey, emittable.isWSAPIResponse);
622
+ // this.logger.trace(
623
+ // 'getFinalEmittable()->post(): ',
624
+ // JSON.stringify(emittable),
625
+ // );
626
+ if (emittable.eventType === 'authenticated') {
627
+ this.logger.trace('Successfully authenticated', {
628
+ ...WS_LOGGER_CATEGORY,
629
+ wsKey,
630
+ emittable,
631
+ });
632
+ this.emit(emittable.eventType, emittableFinalEvent);
633
+ this.onWsAuthenticated(wsKey, emittable.event);
634
+ continue;
635
+ }
636
+ // Other event types are automatically emitted here
637
+ // this.logger.trace(
638
+ // `onWsMessage().emit(${emittable.eventType})`,
639
+ // emittableFinalEvent,
640
+ // );
641
+ try {
642
+ this.emit(emittable.eventType, emittableFinalEvent);
643
+ }
644
+ catch (e) {
645
+ this.logger.error(`Exception in onWsMessage().emit(${emittable.eventType}) handler:`, e);
646
+ }
647
+ // this.logger.trace(
648
+ // `onWsMessage().emit(${emittable.eventType}).done()`,
649
+ // emittableFinalEvent,
650
+ // );
651
+ }
652
+ return;
653
+ }
654
+ this.logger.error('Unhandled/unrecognised ws event message - unexpected message format', {
655
+ ...WS_LOGGER_CATEGORY,
656
+ message: event || 'no message',
657
+ event,
658
+ wsKey,
659
+ });
660
+ }
661
+ catch (e) {
662
+ this.logger.error('Failed to parse ws event message', {
663
+ ...WS_LOGGER_CATEGORY,
664
+ error: e,
665
+ event,
666
+ wsKey,
667
+ });
668
+ }
669
+ }
670
+ onWsClose(event, wsKey) {
671
+ this.logger.info('Websocket connection closed', {
672
+ ...WS_LOGGER_CATEGORY,
673
+ wsKey,
674
+ });
675
+ const wsState = this.wsStore.get(wsKey, true);
676
+ wsState.isAuthenticated = false;
677
+ if (this.wsStore.getConnectionState(wsKey) !== WsConnectionStateEnum.CLOSING) {
678
+ // unintentional close, attempt recovery
679
+ this.logger.trace(`onWsClose(${wsKey}): rejecting all deferred promises...`);
680
+ // clean up any pending promises for this connection
681
+ this.getWsStore().rejectAllDeferredPromises(wsKey, 'connection lost, reconnecting');
682
+ this.setWsState(wsKey, WsConnectionStateEnum.INITIAL);
683
+ this.reconnectWithDelay(wsKey, this.options.reconnectTimeout);
684
+ this.emit('reconnect', { wsKey, event });
685
+ }
686
+ else {
687
+ // intentional close - clean up
688
+ // clean up any pending promises for this connection
689
+ this.logger.trace(`onWsClose(${wsKey}): rejecting all deferred promises...`);
690
+ this.getWsStore().rejectAllDeferredPromises(wsKey, 'disconnected');
691
+ this.setWsState(wsKey, WsConnectionStateEnum.INITIAL);
692
+ // This was an intentional close, delete all state for this connection, as if it never existed:
693
+ this.wsStore.delete(wsKey);
694
+ this.emit('close', { wsKey, event });
695
+ }
696
+ }
697
+ getWs(wsKey) {
698
+ return this.wsStore.getWs(wsKey);
699
+ }
700
+ setWsState(wsKey, state) {
701
+ this.wsStore.setConnectionState(wsKey, state);
702
+ }
703
+ /**
704
+ * Promise-driven method to assert that a ws has successfully connected (will await until connection is open)
705
+ */
706
+ async assertIsConnected(wsKey) {
707
+ const isConnected = this.getWsStore().isConnectionState(wsKey, WsConnectionStateEnum.CONNECTED);
708
+ if (isConnected) {
709
+ return true;
710
+ }
711
+ const inProgressPromise = this.getWsStore().getConnectionInProgressPromise(wsKey);
712
+ // Already in progress? Await shared promise and retry
713
+ if (inProgressPromise) {
714
+ this.logger.trace('assertIsConnected(): awaiting...');
715
+ await inProgressPromise.promise;
716
+ this.logger.trace('assertIsConnected(): awaiting...connected!');
717
+ return inProgressPromise.promise;
718
+ }
719
+ // Start connection, it should automatically store/return a promise.
720
+ this.logger.trace('assertIsConnected(): connecting...');
721
+ await this.connect(wsKey);
722
+ this.logger.trace('assertIsConnected(): connecting...newly connected!');
723
+ }
724
+ /**
725
+ * Promise-driven method to assert that a ws has been successfully authenticated (will await until auth is confirmed)
726
+ */
727
+ async assertIsAuthenticated(wsKey) {
728
+ const isConnected = this.getWsStore().isConnectionState(wsKey, WsConnectionStateEnum.CONNECTED);
729
+ if (!isConnected) {
730
+ this.logger.trace('assertIsAuthenticated(): connecting...');
731
+ await this.assertIsConnected(wsKey);
732
+ }
733
+ const inProgressPromise = this.getWsStore().getAuthenticationInProgressPromise(wsKey);
734
+ // Already in progress? Await shared promise and retry
735
+ if (inProgressPromise) {
736
+ this.logger.trace('assertIsAuthenticated(): awaiting...');
737
+ await inProgressPromise.promise;
738
+ this.logger.trace('assertIsAuthenticated(): authenticated!');
739
+ return;
740
+ }
741
+ const isAuthenticated = this.wsStore.get(wsKey)?.isAuthenticated;
742
+ if (isAuthenticated) {
743
+ // this.logger.trace('assertIsAuthenticated(): ok');
744
+ return;
745
+ }
746
+ // Start authentication, it should automatically store/return a promise.
747
+ this.logger.trace('assertIsAuthenticated(): authenticating...');
748
+ await this.sendAuthRequest(wsKey);
749
+ this.logger.trace('assertIsAuthenticated(): newly authenticated!');
750
+ }
751
+ }
752
+ //# sourceMappingURL=BaseWSClient.js.map