ccxt 4.2.76__py2.py3-none-any.whl → 4.4.48__py2.py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (546) hide show
  1. ccxt/__init__.py +36 -14
  2. ccxt/abstract/alpaca.py +4 -0
  3. ccxt/abstract/bigone.py +1 -1
  4. ccxt/abstract/binance.py +112 -48
  5. ccxt/abstract/binancecoinm.py +112 -48
  6. ccxt/abstract/binanceus.py +147 -83
  7. ccxt/abstract/binanceusdm.py +112 -48
  8. ccxt/abstract/bingx.py +133 -78
  9. ccxt/abstract/bitbank.py +5 -0
  10. ccxt/abstract/bitfinex.py +136 -65
  11. ccxt/abstract/bitfinex1.py +69 -0
  12. ccxt/abstract/bitflyer.py +1 -0
  13. ccxt/abstract/bitget.py +8 -1
  14. ccxt/abstract/bitmart.py +13 -1
  15. ccxt/abstract/bitopro.py +1 -0
  16. ccxt/abstract/bitpanda.py +0 -12
  17. ccxt/abstract/bitrue.py +3 -3
  18. ccxt/abstract/bitstamp.py +26 -3
  19. ccxt/abstract/blofin.py +24 -0
  20. ccxt/abstract/btcbox.py +1 -0
  21. ccxt/abstract/bybit.py +29 -14
  22. ccxt/abstract/cex.py +28 -29
  23. ccxt/abstract/coinbase.py +6 -0
  24. ccxt/abstract/coinbaseadvanced.py +94 -0
  25. ccxt/abstract/{coinbasepro.py → coinbaseexchange.py} +1 -0
  26. ccxt/abstract/coinbaseinternational.py +1 -1
  27. ccxt/abstract/coincatch.py +94 -0
  28. ccxt/abstract/coinex.py +233 -123
  29. ccxt/abstract/coinmetro.py +1 -0
  30. ccxt/abstract/cryptocom.py +14 -0
  31. ccxt/abstract/defx.py +69 -0
  32. ccxt/abstract/deribit.py +1 -0
  33. ccxt/abstract/digifinex.py +1 -0
  34. ccxt/abstract/ellipx.py +25 -0
  35. ccxt/abstract/gate.py +20 -0
  36. ccxt/abstract/gateio.py +20 -0
  37. ccxt/abstract/gemini.py +1 -0
  38. ccxt/abstract/hashkey.py +67 -0
  39. ccxt/abstract/hyperliquid.py +1 -1
  40. ccxt/abstract/independentreserve.py +6 -0
  41. ccxt/abstract/kraken.py +4 -3
  42. ccxt/abstract/krakenfutures.py +4 -0
  43. ccxt/abstract/kucoin.py +25 -0
  44. ccxt/abstract/kucoinfutures.py +35 -0
  45. ccxt/abstract/luno.py +2 -0
  46. ccxt/abstract/mexc.py +4 -0
  47. ccxt/abstract/myokx.py +340 -0
  48. ccxt/abstract/oceanex.py +5 -0
  49. ccxt/abstract/okx.py +30 -0
  50. ccxt/abstract/onetrading.py +0 -12
  51. ccxt/abstract/oxfun.py +34 -0
  52. ccxt/abstract/paradex.py +40 -0
  53. ccxt/abstract/phemex.py +1 -0
  54. ccxt/abstract/upbit.py +4 -0
  55. ccxt/abstract/vertex.py +19 -0
  56. ccxt/abstract/whitebit.py +31 -1
  57. ccxt/abstract/woo.py +6 -2
  58. ccxt/abstract/woofipro.py +119 -0
  59. ccxt/abstract/xt.py +153 -0
  60. ccxt/abstract/zonda.py +6 -0
  61. ccxt/ace.py +164 -60
  62. ccxt/alpaca.py +727 -63
  63. ccxt/ascendex.py +395 -249
  64. ccxt/async_support/__init__.py +36 -14
  65. ccxt/async_support/ace.py +164 -60
  66. ccxt/async_support/alpaca.py +727 -63
  67. ccxt/async_support/ascendex.py +396 -249
  68. ccxt/async_support/base/exchange.py +531 -155
  69. ccxt/async_support/base/ws/aiohttp_client.py +28 -5
  70. ccxt/async_support/base/ws/cache.py +3 -2
  71. ccxt/async_support/base/ws/client.py +26 -5
  72. ccxt/async_support/base/ws/fast_client.py +4 -3
  73. ccxt/async_support/base/ws/functions.py +1 -1
  74. ccxt/async_support/base/ws/future.py +40 -31
  75. ccxt/async_support/base/ws/order_book_side.py +3 -0
  76. ccxt/async_support/bequant.py +1 -1
  77. ccxt/async_support/bigone.py +329 -202
  78. ccxt/async_support/binance.py +3513 -1511
  79. ccxt/async_support/binancecoinm.py +2 -1
  80. ccxt/async_support/binanceus.py +12 -1
  81. ccxt/async_support/binanceusdm.py +3 -1
  82. ccxt/async_support/bingx.py +3105 -881
  83. ccxt/async_support/bit2c.py +119 -38
  84. ccxt/async_support/bitbank.py +215 -76
  85. ccxt/async_support/bitbns.py +124 -53
  86. ccxt/async_support/bitfinex.py +3236 -1078
  87. ccxt/async_support/bitfinex1.py +1711 -0
  88. ccxt/async_support/bitflyer.py +239 -50
  89. ccxt/async_support/bitget.py +1513 -563
  90. ccxt/async_support/bithumb.py +201 -67
  91. ccxt/async_support/bitmart.py +1320 -435
  92. ccxt/async_support/bitmex.py +308 -111
  93. ccxt/async_support/bitopro.py +256 -96
  94. ccxt/async_support/bitrue.py +365 -233
  95. ccxt/async_support/bitso.py +201 -89
  96. ccxt/async_support/bitstamp.py +438 -269
  97. ccxt/async_support/bitteam.py +179 -73
  98. ccxt/async_support/bitvavo.py +180 -70
  99. ccxt/async_support/bl3p.py +92 -25
  100. ccxt/async_support/blockchaincom.py +193 -79
  101. ccxt/async_support/blofin.py +403 -150
  102. ccxt/async_support/btcalpha.py +161 -55
  103. ccxt/async_support/btcbox.py +250 -34
  104. ccxt/async_support/btcmarkets.py +232 -85
  105. ccxt/async_support/btcturk.py +159 -60
  106. ccxt/async_support/bybit.py +2326 -1255
  107. ccxt/async_support/cex.py +1409 -1329
  108. ccxt/async_support/coinbase.py +1455 -288
  109. ccxt/async_support/coinbaseadvanced.py +17 -0
  110. ccxt/async_support/{coinbasepro.py → coinbaseexchange.py} +233 -99
  111. ccxt/async_support/coinbaseinternational.py +428 -88
  112. ccxt/async_support/coincatch.py +5152 -0
  113. ccxt/async_support/coincheck.py +121 -38
  114. ccxt/async_support/coinex.py +4020 -3339
  115. ccxt/async_support/coinlist.py +273 -116
  116. ccxt/async_support/coinmate.py +204 -97
  117. ccxt/async_support/coinmetro.py +203 -110
  118. ccxt/async_support/coinone.py +142 -68
  119. ccxt/async_support/coinsph.py +206 -89
  120. ccxt/async_support/coinspot.py +137 -62
  121. ccxt/async_support/cryptocom.py +515 -185
  122. ccxt/async_support/currencycom.py +203 -85
  123. ccxt/async_support/defx.py +2066 -0
  124. ccxt/async_support/delta.py +467 -158
  125. ccxt/async_support/deribit.py +558 -324
  126. ccxt/async_support/digifinex.py +340 -223
  127. ccxt/async_support/ellipx.py +1826 -0
  128. ccxt/async_support/exmo.py +259 -128
  129. ccxt/async_support/gate.py +1473 -464
  130. ccxt/async_support/gemini.py +206 -84
  131. ccxt/async_support/hashkey.py +4164 -0
  132. ccxt/async_support/hitbtc.py +334 -178
  133. ccxt/async_support/hollaex.py +134 -83
  134. ccxt/async_support/htx.py +1095 -563
  135. ccxt/async_support/huobijp.py +105 -56
  136. ccxt/async_support/hyperliquid.py +1634 -269
  137. ccxt/async_support/idex.py +148 -95
  138. ccxt/async_support/independentreserve.py +236 -31
  139. ccxt/async_support/indodax.py +165 -62
  140. ccxt/async_support/kraken.py +871 -354
  141. ccxt/async_support/krakenfutures.py +324 -100
  142. ccxt/async_support/kucoin.py +1050 -355
  143. ccxt/async_support/kucoinfutures.py +1004 -149
  144. ccxt/async_support/kuna.py +138 -106
  145. ccxt/async_support/latoken.py +135 -79
  146. ccxt/async_support/lbank.py +290 -113
  147. ccxt/async_support/luno.py +112 -62
  148. ccxt/async_support/lykke.py +104 -55
  149. ccxt/async_support/mercado.py +36 -29
  150. ccxt/async_support/mexc.py +995 -429
  151. ccxt/async_support/myokx.py +43 -0
  152. ccxt/async_support/ndax.py +163 -82
  153. ccxt/async_support/novadax.py +121 -75
  154. ccxt/async_support/oceanex.py +175 -59
  155. ccxt/async_support/okcoin.py +222 -163
  156. ccxt/async_support/okx.py +1777 -455
  157. ccxt/async_support/onetrading.py +132 -414
  158. ccxt/async_support/oxfun.py +2832 -0
  159. ccxt/async_support/p2b.py +79 -51
  160. ccxt/async_support/paradex.py +2017 -0
  161. ccxt/async_support/paymium.py +56 -32
  162. ccxt/async_support/phemex.py +572 -196
  163. ccxt/async_support/poloniex.py +218 -95
  164. ccxt/async_support/poloniexfutures.py +260 -92
  165. ccxt/async_support/probit.py +143 -110
  166. ccxt/async_support/timex.py +123 -70
  167. ccxt/async_support/tokocrypto.py +129 -93
  168. ccxt/async_support/tradeogre.py +39 -25
  169. ccxt/async_support/upbit.py +322 -113
  170. ccxt/async_support/vertex.py +2983 -0
  171. ccxt/async_support/wavesexchange.py +227 -173
  172. ccxt/async_support/wazirx.py +145 -65
  173. ccxt/async_support/whitebit.py +533 -138
  174. ccxt/async_support/woo.py +1155 -295
  175. ccxt/async_support/woofipro.py +2716 -0
  176. ccxt/async_support/xt.py +4628 -0
  177. ccxt/async_support/yobit.py +160 -92
  178. ccxt/async_support/zaif.py +80 -33
  179. ccxt/async_support/zonda.py +140 -69
  180. ccxt/base/errors.py +51 -20
  181. ccxt/base/exchange.py +1729 -482
  182. ccxt/base/precise.py +10 -0
  183. ccxt/base/types.py +223 -4
  184. ccxt/bequant.py +1 -1
  185. ccxt/bigone.py +329 -202
  186. ccxt/binance.py +3513 -1511
  187. ccxt/binancecoinm.py +2 -1
  188. ccxt/binanceus.py +12 -1
  189. ccxt/binanceusdm.py +3 -1
  190. ccxt/bingx.py +3105 -881
  191. ccxt/bit2c.py +119 -38
  192. ccxt/bitbank.py +215 -76
  193. ccxt/bitbns.py +124 -53
  194. ccxt/bitfinex.py +3235 -1078
  195. ccxt/bitfinex1.py +1710 -0
  196. ccxt/bitflyer.py +239 -50
  197. ccxt/bitget.py +1513 -563
  198. ccxt/bithumb.py +200 -67
  199. ccxt/bitmart.py +1320 -435
  200. ccxt/bitmex.py +308 -111
  201. ccxt/bitopro.py +256 -96
  202. ccxt/bitrue.py +365 -233
  203. ccxt/bitso.py +201 -89
  204. ccxt/bitstamp.py +438 -269
  205. ccxt/bitteam.py +179 -73
  206. ccxt/bitvavo.py +180 -70
  207. ccxt/bl3p.py +92 -25
  208. ccxt/blockchaincom.py +193 -79
  209. ccxt/blofin.py +403 -150
  210. ccxt/btcalpha.py +161 -55
  211. ccxt/btcbox.py +250 -34
  212. ccxt/btcmarkets.py +232 -85
  213. ccxt/btcturk.py +159 -60
  214. ccxt/bybit.py +2326 -1255
  215. ccxt/cex.py +1408 -1329
  216. ccxt/coinbase.py +1455 -288
  217. ccxt/coinbaseadvanced.py +17 -0
  218. ccxt/{coinbasepro.py → coinbaseexchange.py} +233 -99
  219. ccxt/coinbaseinternational.py +428 -88
  220. ccxt/coincatch.py +5152 -0
  221. ccxt/coincheck.py +121 -38
  222. ccxt/coinex.py +4020 -3339
  223. ccxt/coinlist.py +273 -116
  224. ccxt/coinmate.py +204 -97
  225. ccxt/coinmetro.py +203 -110
  226. ccxt/coinone.py +142 -68
  227. ccxt/coinsph.py +206 -89
  228. ccxt/coinspot.py +137 -62
  229. ccxt/cryptocom.py +515 -185
  230. ccxt/currencycom.py +203 -85
  231. ccxt/defx.py +2065 -0
  232. ccxt/delta.py +467 -158
  233. ccxt/deribit.py +558 -324
  234. ccxt/digifinex.py +340 -223
  235. ccxt/ellipx.py +1826 -0
  236. ccxt/exmo.py +259 -128
  237. ccxt/gate.py +1473 -464
  238. ccxt/gemini.py +206 -84
  239. ccxt/hashkey.py +4164 -0
  240. ccxt/hitbtc.py +334 -178
  241. ccxt/hollaex.py +134 -83
  242. ccxt/htx.py +1095 -563
  243. ccxt/huobijp.py +105 -56
  244. ccxt/hyperliquid.py +1633 -269
  245. ccxt/idex.py +148 -95
  246. ccxt/independentreserve.py +235 -31
  247. ccxt/indodax.py +165 -62
  248. ccxt/kraken.py +871 -354
  249. ccxt/krakenfutures.py +324 -100
  250. ccxt/kucoin.py +1050 -355
  251. ccxt/kucoinfutures.py +1004 -149
  252. ccxt/kuna.py +138 -106
  253. ccxt/latoken.py +135 -79
  254. ccxt/lbank.py +290 -113
  255. ccxt/luno.py +112 -62
  256. ccxt/lykke.py +104 -55
  257. ccxt/mercado.py +36 -29
  258. ccxt/mexc.py +994 -429
  259. ccxt/myokx.py +43 -0
  260. ccxt/ndax.py +163 -82
  261. ccxt/novadax.py +121 -75
  262. ccxt/oceanex.py +175 -59
  263. ccxt/okcoin.py +222 -163
  264. ccxt/okx.py +1777 -455
  265. ccxt/onetrading.py +132 -414
  266. ccxt/oxfun.py +2831 -0
  267. ccxt/p2b.py +79 -51
  268. ccxt/paradex.py +2017 -0
  269. ccxt/paymium.py +56 -32
  270. ccxt/phemex.py +572 -196
  271. ccxt/poloniex.py +218 -95
  272. ccxt/poloniexfutures.py +260 -92
  273. ccxt/pro/__init__.py +29 -5
  274. ccxt/pro/alpaca.py +32 -17
  275. ccxt/pro/ascendex.py +63 -15
  276. ccxt/pro/bequant.py +4 -0
  277. ccxt/pro/binance.py +1596 -329
  278. ccxt/pro/binancecoinm.py +1 -0
  279. ccxt/pro/binanceus.py +2 -9
  280. ccxt/pro/binanceusdm.py +2 -0
  281. ccxt/pro/bingx.py +527 -134
  282. ccxt/pro/bitcoincom.py +4 -1
  283. ccxt/pro/bitfinex.py +731 -266
  284. ccxt/pro/bitfinex1.py +635 -0
  285. ccxt/pro/bitget.py +726 -357
  286. ccxt/pro/bithumb.py +380 -0
  287. ccxt/pro/bitmart.py +138 -39
  288. ccxt/pro/bitmex.py +199 -40
  289. ccxt/pro/bitopro.py +25 -13
  290. ccxt/pro/bitrue.py +31 -32
  291. ccxt/pro/bitstamp.py +7 -6
  292. ccxt/pro/bitvavo.py +204 -82
  293. ccxt/pro/blockchaincom.py +30 -17
  294. ccxt/pro/blofin.py +692 -0
  295. ccxt/pro/bybit.py +791 -82
  296. ccxt/pro/cex.py +99 -51
  297. ccxt/pro/coinbase.py +220 -30
  298. ccxt/{async_support/hitbtc3.py → pro/coinbaseadvanced.py} +5 -5
  299. ccxt/pro/{coinbasepro.py → coinbaseexchange.py} +19 -19
  300. ccxt/pro/coinbaseinternational.py +193 -30
  301. ccxt/pro/coincatch.py +1464 -0
  302. ccxt/pro/coincheck.py +11 -6
  303. ccxt/pro/coinex.py +967 -661
  304. ccxt/pro/coinone.py +17 -10
  305. ccxt/pro/cryptocom.py +446 -66
  306. ccxt/pro/currencycom.py +11 -10
  307. ccxt/pro/defx.py +832 -0
  308. ccxt/pro/deribit.py +168 -32
  309. ccxt/pro/exmo.py +253 -21
  310. ccxt/pro/gate.py +729 -64
  311. ccxt/pro/gemini.py +44 -26
  312. ccxt/pro/hashkey.py +802 -0
  313. ccxt/pro/hitbtc.py +208 -103
  314. ccxt/pro/hollaex.py +25 -9
  315. ccxt/pro/htx.py +83 -39
  316. ccxt/pro/huobijp.py +17 -16
  317. ccxt/pro/hyperliquid.py +502 -31
  318. ccxt/pro/idex.py +28 -13
  319. ccxt/pro/independentreserve.py +21 -16
  320. ccxt/pro/kraken.py +298 -51
  321. ccxt/pro/krakenfutures.py +166 -75
  322. ccxt/pro/kucoin.py +395 -77
  323. ccxt/pro/kucoinfutures.py +400 -99
  324. ccxt/pro/lbank.py +52 -31
  325. ccxt/pro/luno.py +12 -10
  326. ccxt/pro/mexc.py +400 -50
  327. ccxt/pro/myokx.py +28 -0
  328. ccxt/pro/ndax.py +25 -12
  329. ccxt/pro/okcoin.py +28 -9
  330. ccxt/pro/okx.py +935 -124
  331. ccxt/pro/onetrading.py +41 -24
  332. ccxt/pro/oxfun.py +1054 -0
  333. ccxt/pro/p2b.py +100 -24
  334. ccxt/pro/paradex.py +352 -0
  335. ccxt/pro/phemex.py +93 -34
  336. ccxt/pro/poloniex.py +129 -50
  337. ccxt/pro/poloniexfutures.py +53 -32
  338. ccxt/pro/probit.py +93 -86
  339. ccxt/pro/upbit.py +401 -8
  340. ccxt/pro/vertex.py +943 -0
  341. ccxt/pro/wazirx.py +46 -28
  342. ccxt/pro/whitebit.py +65 -12
  343. ccxt/pro/woo.py +486 -70
  344. ccxt/pro/woofipro.py +1271 -0
  345. ccxt/pro/xt.py +1067 -0
  346. ccxt/probit.py +143 -110
  347. ccxt/static_dependencies/__init__.py +1 -1
  348. ccxt/static_dependencies/lark/__init__.py +38 -0
  349. ccxt/static_dependencies/lark/__pyinstaller/__init__.py +6 -0
  350. ccxt/static_dependencies/lark/__pyinstaller/hook-lark.py +14 -0
  351. ccxt/static_dependencies/lark/ast_utils.py +59 -0
  352. ccxt/static_dependencies/lark/common.py +86 -0
  353. ccxt/static_dependencies/lark/exceptions.py +292 -0
  354. ccxt/static_dependencies/lark/grammar.py +130 -0
  355. ccxt/static_dependencies/lark/grammars/__init__.py +0 -0
  356. ccxt/static_dependencies/lark/indenter.py +143 -0
  357. ccxt/static_dependencies/lark/lark.py +658 -0
  358. ccxt/static_dependencies/lark/lexer.py +678 -0
  359. ccxt/static_dependencies/lark/load_grammar.py +1428 -0
  360. ccxt/static_dependencies/lark/parse_tree_builder.py +391 -0
  361. ccxt/static_dependencies/lark/parser_frontends.py +257 -0
  362. ccxt/static_dependencies/lark/parsers/__init__.py +0 -0
  363. ccxt/static_dependencies/lark/parsers/cyk.py +340 -0
  364. ccxt/static_dependencies/lark/parsers/earley.py +314 -0
  365. ccxt/static_dependencies/lark/parsers/earley_common.py +42 -0
  366. ccxt/static_dependencies/lark/parsers/earley_forest.py +801 -0
  367. ccxt/static_dependencies/lark/parsers/grammar_analysis.py +203 -0
  368. ccxt/static_dependencies/lark/parsers/lalr_analysis.py +332 -0
  369. ccxt/static_dependencies/lark/parsers/lalr_interactive_parser.py +158 -0
  370. ccxt/static_dependencies/lark/parsers/lalr_parser.py +122 -0
  371. ccxt/static_dependencies/lark/parsers/lalr_parser_state.py +110 -0
  372. ccxt/static_dependencies/lark/parsers/xearley.py +165 -0
  373. ccxt/static_dependencies/lark/py.typed +0 -0
  374. ccxt/static_dependencies/lark/reconstruct.py +107 -0
  375. ccxt/static_dependencies/lark/tools/__init__.py +70 -0
  376. ccxt/static_dependencies/lark/tools/nearley.py +202 -0
  377. ccxt/static_dependencies/lark/tools/serialize.py +32 -0
  378. ccxt/static_dependencies/lark/tools/standalone.py +196 -0
  379. ccxt/static_dependencies/lark/tree.py +267 -0
  380. ccxt/static_dependencies/lark/tree_matcher.py +186 -0
  381. ccxt/static_dependencies/lark/tree_templates.py +180 -0
  382. ccxt/static_dependencies/lark/utils.py +343 -0
  383. ccxt/static_dependencies/lark/visitors.py +596 -0
  384. ccxt/static_dependencies/marshmallow/__init__.py +81 -0
  385. ccxt/static_dependencies/marshmallow/base.py +65 -0
  386. ccxt/static_dependencies/marshmallow/class_registry.py +94 -0
  387. ccxt/static_dependencies/marshmallow/decorators.py +231 -0
  388. ccxt/static_dependencies/marshmallow/error_store.py +60 -0
  389. ccxt/static_dependencies/marshmallow/exceptions.py +71 -0
  390. ccxt/static_dependencies/marshmallow/fields.py +2114 -0
  391. ccxt/static_dependencies/marshmallow/orderedset.py +89 -0
  392. ccxt/static_dependencies/marshmallow/py.typed +0 -0
  393. ccxt/static_dependencies/marshmallow/schema.py +1228 -0
  394. ccxt/static_dependencies/marshmallow/types.py +12 -0
  395. ccxt/static_dependencies/marshmallow/utils.py +378 -0
  396. ccxt/static_dependencies/marshmallow/validate.py +678 -0
  397. ccxt/static_dependencies/marshmallow/warnings.py +2 -0
  398. ccxt/static_dependencies/marshmallow_dataclass/__init__.py +1047 -0
  399. ccxt/static_dependencies/marshmallow_dataclass/collection_field.py +51 -0
  400. ccxt/static_dependencies/marshmallow_dataclass/lazy_class_attribute.py +45 -0
  401. ccxt/static_dependencies/marshmallow_dataclass/mypy.py +71 -0
  402. ccxt/static_dependencies/marshmallow_dataclass/py.typed +0 -0
  403. ccxt/static_dependencies/marshmallow_dataclass/typing.py +14 -0
  404. ccxt/static_dependencies/marshmallow_dataclass/union_field.py +82 -0
  405. ccxt/static_dependencies/marshmallow_oneofschema/__init__.py +1 -0
  406. ccxt/static_dependencies/marshmallow_oneofschema/one_of_schema.py +193 -0
  407. ccxt/static_dependencies/marshmallow_oneofschema/py.typed +0 -0
  408. ccxt/static_dependencies/starknet/__init__.py +0 -0
  409. ccxt/static_dependencies/starknet/cairo/__init__.py +0 -0
  410. ccxt/static_dependencies/starknet/cairo/data_types.py +123 -0
  411. ccxt/static_dependencies/starknet/cairo/deprecated_parse/__init__.py +0 -0
  412. ccxt/static_dependencies/starknet/cairo/deprecated_parse/cairo_types.py +77 -0
  413. ccxt/static_dependencies/starknet/cairo/deprecated_parse/parser.py +46 -0
  414. ccxt/static_dependencies/starknet/cairo/deprecated_parse/parser_transformer.py +138 -0
  415. ccxt/static_dependencies/starknet/cairo/felt.py +64 -0
  416. ccxt/static_dependencies/starknet/cairo/type_parser.py +121 -0
  417. ccxt/static_dependencies/starknet/cairo/v1/__init__.py +0 -0
  418. ccxt/static_dependencies/starknet/cairo/v1/type_parser.py +59 -0
  419. ccxt/static_dependencies/starknet/cairo/v2/__init__.py +0 -0
  420. ccxt/static_dependencies/starknet/cairo/v2/type_parser.py +77 -0
  421. ccxt/static_dependencies/starknet/ccxt_utils.py +7 -0
  422. ccxt/static_dependencies/starknet/common.py +15 -0
  423. ccxt/static_dependencies/starknet/constants.py +39 -0
  424. ccxt/static_dependencies/starknet/hash/__init__.py +0 -0
  425. ccxt/static_dependencies/starknet/hash/address.py +79 -0
  426. ccxt/static_dependencies/starknet/hash/compiled_class_hash_objects.py +111 -0
  427. ccxt/static_dependencies/starknet/hash/selector.py +16 -0
  428. ccxt/static_dependencies/starknet/hash/storage.py +12 -0
  429. ccxt/static_dependencies/starknet/hash/utils.py +78 -0
  430. ccxt/static_dependencies/starknet/models/__init__.py +0 -0
  431. ccxt/static_dependencies/starknet/models/typed_data.py +45 -0
  432. ccxt/static_dependencies/starknet/serialization/__init__.py +24 -0
  433. ccxt/static_dependencies/starknet/serialization/_calldata_reader.py +40 -0
  434. ccxt/static_dependencies/starknet/serialization/_context.py +142 -0
  435. ccxt/static_dependencies/starknet/serialization/data_serializers/__init__.py +10 -0
  436. ccxt/static_dependencies/starknet/serialization/data_serializers/_common.py +82 -0
  437. ccxt/static_dependencies/starknet/serialization/data_serializers/array_serializer.py +43 -0
  438. ccxt/static_dependencies/starknet/serialization/data_serializers/bool_serializer.py +37 -0
  439. ccxt/static_dependencies/starknet/serialization/data_serializers/byte_array_serializer.py +66 -0
  440. ccxt/static_dependencies/starknet/serialization/data_serializers/cairo_data_serializer.py +71 -0
  441. ccxt/static_dependencies/starknet/serialization/data_serializers/enum_serializer.py +71 -0
  442. ccxt/static_dependencies/starknet/serialization/data_serializers/felt_serializer.py +50 -0
  443. ccxt/static_dependencies/starknet/serialization/data_serializers/named_tuple_serializer.py +58 -0
  444. ccxt/static_dependencies/starknet/serialization/data_serializers/option_serializer.py +43 -0
  445. ccxt/static_dependencies/starknet/serialization/data_serializers/output_serializer.py +40 -0
  446. ccxt/static_dependencies/starknet/serialization/data_serializers/payload_serializer.py +72 -0
  447. ccxt/static_dependencies/starknet/serialization/data_serializers/struct_serializer.py +36 -0
  448. ccxt/static_dependencies/starknet/serialization/data_serializers/tuple_serializer.py +36 -0
  449. ccxt/static_dependencies/starknet/serialization/data_serializers/uint256_serializer.py +76 -0
  450. ccxt/static_dependencies/starknet/serialization/data_serializers/uint_serializer.py +100 -0
  451. ccxt/static_dependencies/starknet/serialization/data_serializers/unit_serializer.py +32 -0
  452. ccxt/static_dependencies/starknet/serialization/errors.py +10 -0
  453. ccxt/static_dependencies/starknet/serialization/factory.py +229 -0
  454. ccxt/static_dependencies/starknet/serialization/function_serialization_adapter.py +110 -0
  455. ccxt/static_dependencies/starknet/serialization/tuple_dataclass.py +59 -0
  456. ccxt/static_dependencies/starknet/utils/__init__.py +0 -0
  457. ccxt/static_dependencies/starknet/utils/constructor_args_translator.py +86 -0
  458. ccxt/static_dependencies/starknet/utils/iterable.py +13 -0
  459. ccxt/static_dependencies/starknet/utils/schema.py +13 -0
  460. ccxt/static_dependencies/starknet/utils/typed_data.py +182 -0
  461. ccxt/static_dependencies/starkware/__init__.py +0 -0
  462. ccxt/static_dependencies/starkware/crypto/__init__.py +0 -0
  463. ccxt/static_dependencies/starkware/crypto/fast_pedersen_hash.py +50 -0
  464. ccxt/static_dependencies/starkware/crypto/math_utils.py +78 -0
  465. ccxt/static_dependencies/starkware/crypto/signature.py +2344 -0
  466. ccxt/static_dependencies/starkware/crypto/utils.py +63 -0
  467. ccxt/static_dependencies/sympy/__init__.py +0 -0
  468. ccxt/static_dependencies/sympy/core/__init__.py +0 -0
  469. ccxt/static_dependencies/sympy/core/intfunc.py +35 -0
  470. ccxt/static_dependencies/sympy/external/__init__.py +0 -0
  471. ccxt/static_dependencies/sympy/external/gmpy.py +345 -0
  472. ccxt/static_dependencies/sympy/external/importtools.py +187 -0
  473. ccxt/static_dependencies/sympy/external/ntheory.py +637 -0
  474. ccxt/static_dependencies/sympy/external/pythonmpq.py +341 -0
  475. ccxt/static_dependencies/typing_inspect/__init__.py +0 -0
  476. ccxt/static_dependencies/typing_inspect/typing_inspect.py +851 -0
  477. ccxt/test/{test_async.py → tests_async.py} +465 -407
  478. ccxt/test/tests_helpers.py +285 -0
  479. ccxt/test/tests_init.py +39 -0
  480. ccxt/test/{test_sync.py → tests_sync.py} +465 -409
  481. ccxt/timex.py +123 -70
  482. ccxt/tokocrypto.py +129 -93
  483. ccxt/tradeogre.py +39 -25
  484. ccxt/upbit.py +322 -113
  485. ccxt/vertex.py +2983 -0
  486. ccxt/wavesexchange.py +227 -173
  487. ccxt/wazirx.py +145 -65
  488. ccxt/whitebit.py +533 -138
  489. ccxt/woo.py +1155 -295
  490. ccxt/woofipro.py +2716 -0
  491. ccxt/xt.py +4627 -0
  492. ccxt/yobit.py +159 -92
  493. ccxt/zaif.py +80 -33
  494. ccxt/zonda.py +140 -69
  495. ccxt-4.4.48.dist-info/LICENSE.txt +21 -0
  496. ccxt-4.4.48.dist-info/METADATA +646 -0
  497. ccxt-4.4.48.dist-info/RECORD +669 -0
  498. {ccxt-4.2.76.dist-info → ccxt-4.4.48.dist-info}/WHEEL +1 -1
  499. ccxt/abstract/bitbay.py +0 -47
  500. ccxt/abstract/bitfinex2.py +0 -139
  501. ccxt/abstract/hitbtc3.py +0 -115
  502. ccxt/async_support/bitbay.py +0 -17
  503. ccxt/async_support/bitfinex2.py +0 -3496
  504. ccxt/async_support/flowbtc.py +0 -34
  505. ccxt/bitbay.py +0 -17
  506. ccxt/bitfinex2.py +0 -3496
  507. ccxt/flowbtc.py +0 -34
  508. ccxt/hitbtc3.py +0 -16
  509. ccxt/pro/bitfinex2.py +0 -1081
  510. ccxt/test/base/__init__.py +0 -28
  511. ccxt/test/base/test_account.py +0 -26
  512. ccxt/test/base/test_balance.py +0 -56
  513. ccxt/test/base/test_borrow_interest.py +0 -35
  514. ccxt/test/base/test_borrow_rate.py +0 -32
  515. ccxt/test/base/test_calculate_fee.py +0 -51
  516. ccxt/test/base/test_crypto.py +0 -127
  517. ccxt/test/base/test_currency.py +0 -76
  518. ccxt/test/base/test_datetime.py +0 -103
  519. ccxt/test/base/test_decimal_to_precision.py +0 -392
  520. ccxt/test/base/test_deep_extend.py +0 -68
  521. ccxt/test/base/test_deposit_withdrawal.py +0 -50
  522. ccxt/test/base/test_exchange_datetime_functions.py +0 -76
  523. ccxt/test/base/test_funding_rate_history.py +0 -29
  524. ccxt/test/base/test_last_price.py +0 -32
  525. ccxt/test/base/test_ledger_entry.py +0 -45
  526. ccxt/test/base/test_ledger_item.py +0 -48
  527. ccxt/test/base/test_leverage_tier.py +0 -33
  528. ccxt/test/base/test_margin_mode.py +0 -24
  529. ccxt/test/base/test_margin_modification.py +0 -35
  530. ccxt/test/base/test_market.py +0 -190
  531. ccxt/test/base/test_number.py +0 -411
  532. ccxt/test/base/test_ohlcv.py +0 -32
  533. ccxt/test/base/test_open_interest.py +0 -32
  534. ccxt/test/base/test_order.py +0 -64
  535. ccxt/test/base/test_order_book.py +0 -63
  536. ccxt/test/base/test_position.py +0 -60
  537. ccxt/test/base/test_shared_methods.py +0 -345
  538. ccxt/test/base/test_status.py +0 -24
  539. ccxt/test/base/test_throttle.py +0 -126
  540. ccxt/test/base/test_ticker.py +0 -86
  541. ccxt/test/base/test_trade.py +0 -47
  542. ccxt/test/base/test_trading_fee.py +0 -26
  543. ccxt/test/base/test_transaction.py +0 -39
  544. ccxt-4.2.76.dist-info/METADATA +0 -626
  545. ccxt-4.2.76.dist-info/RECORD +0 -534
  546. {ccxt-4.2.76.dist-info → ccxt-4.4.48.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,801 @@
1
+ """"This module implements an SPPF implementation
2
+
3
+ This is used as the primary output mechanism for the Earley parser
4
+ in order to store complex ambiguities.
5
+
6
+ Full reference and more details is here:
7
+ https://web.archive.org/web/20190616123959/http://www.bramvandersanden.com/post/2014/06/shared-packed-parse-forest/
8
+ """
9
+
10
+ from typing import Type, AbstractSet
11
+ from random import randint
12
+ from collections import deque
13
+ from operator import attrgetter
14
+ from importlib import import_module
15
+ from functools import partial
16
+
17
+ from ..parse_tree_builder import AmbiguousIntermediateExpander
18
+ from ..visitors import Discard
19
+ from ..utils import logger, OrderedSet
20
+ from ..tree import Tree
21
+
22
+ class ForestNode:
23
+ pass
24
+
25
+ class SymbolNode(ForestNode):
26
+ """
27
+ A Symbol Node represents a symbol (or Intermediate LR0).
28
+
29
+ Symbol nodes are keyed by the symbol (s). For intermediate nodes
30
+ s will be an LR0, stored as a tuple of (rule, ptr). For completed symbol
31
+ nodes, s will be a string representing the non-terminal origin (i.e.
32
+ the left hand side of the rule).
33
+
34
+ The children of a Symbol or Intermediate Node will always be Packed Nodes;
35
+ with each Packed Node child representing a single derivation of a production.
36
+
37
+ Hence a Symbol Node with a single child is unambiguous.
38
+
39
+ Parameters:
40
+ s: A Symbol, or a tuple of (rule, ptr) for an intermediate node.
41
+ start: For dynamic lexers, the index of the start of the substring matched by this symbol (inclusive).
42
+ end: For dynamic lexers, the index of the end of the substring matched by this symbol (exclusive).
43
+
44
+ Properties:
45
+ is_intermediate: True if this node is an intermediate node.
46
+ priority: The priority of the node's symbol.
47
+ """
48
+ Set: Type[AbstractSet] = set # Overridden by StableSymbolNode
49
+ __slots__ = ('s', 'start', 'end', '_children', 'paths', 'paths_loaded', 'priority', 'is_intermediate')
50
+ def __init__(self, s, start, end):
51
+ self.s = s
52
+ self.start = start
53
+ self.end = end
54
+ self._children = self.Set()
55
+ self.paths = self.Set()
56
+ self.paths_loaded = False
57
+
58
+ ### We use inf here as it can be safely negated without resorting to conditionals,
59
+ # unlike None or float('NaN'), and sorts appropriately.
60
+ self.priority = float('-inf')
61
+ self.is_intermediate = isinstance(s, tuple)
62
+
63
+ def add_family(self, lr0, rule, start, left, right):
64
+ self._children.add(PackedNode(self, lr0, rule, start, left, right))
65
+
66
+ def add_path(self, transitive, node):
67
+ self.paths.add((transitive, node))
68
+
69
+ def load_paths(self):
70
+ for transitive, node in self.paths:
71
+ if transitive.next_titem is not None:
72
+ vn = type(self)(transitive.next_titem.s, transitive.next_titem.start, self.end)
73
+ vn.add_path(transitive.next_titem, node)
74
+ self.add_family(transitive.reduction.rule.origin, transitive.reduction.rule, transitive.reduction.start, transitive.reduction.node, vn)
75
+ else:
76
+ self.add_family(transitive.reduction.rule.origin, transitive.reduction.rule, transitive.reduction.start, transitive.reduction.node, node)
77
+ self.paths_loaded = True
78
+
79
+ @property
80
+ def is_ambiguous(self):
81
+ """Returns True if this node is ambiguous."""
82
+ return len(self.children) > 1
83
+
84
+ @property
85
+ def children(self):
86
+ """Returns a list of this node's children sorted from greatest to
87
+ least priority."""
88
+ if not self.paths_loaded:
89
+ self.load_paths()
90
+ return sorted(self._children, key=attrgetter('sort_key'))
91
+
92
+ def __iter__(self):
93
+ return iter(self._children)
94
+
95
+ def __repr__(self):
96
+ if self.is_intermediate:
97
+ rule = self.s[0]
98
+ ptr = self.s[1]
99
+ before = ( expansion.name for expansion in rule.expansion[:ptr] )
100
+ after = ( expansion.name for expansion in rule.expansion[ptr:] )
101
+ symbol = "{} ::= {}* {}".format(rule.origin.name, ' '.join(before), ' '.join(after))
102
+ else:
103
+ symbol = self.s.name
104
+ return "({}, {}, {}, {})".format(symbol, self.start, self.end, self.priority)
105
+
106
+ class StableSymbolNode(SymbolNode):
107
+ "A version of SymbolNode that uses OrderedSet for output stability"
108
+ Set = OrderedSet
109
+
110
+ class PackedNode(ForestNode):
111
+ """
112
+ A Packed Node represents a single derivation in a symbol node.
113
+
114
+ Parameters:
115
+ rule: The rule associated with this node.
116
+ parent: The parent of this node.
117
+ left: The left child of this node. ``None`` if one does not exist.
118
+ right: The right child of this node. ``None`` if one does not exist.
119
+ priority: The priority of this node.
120
+ """
121
+ __slots__ = ('parent', 's', 'rule', 'start', 'left', 'right', 'priority', '_hash')
122
+ def __init__(self, parent, s, rule, start, left, right):
123
+ self.parent = parent
124
+ self.s = s
125
+ self.start = start
126
+ self.rule = rule
127
+ self.left = left
128
+ self.right = right
129
+ self.priority = float('-inf')
130
+ self._hash = hash((self.left, self.right))
131
+
132
+ @property
133
+ def is_empty(self):
134
+ return self.left is None and self.right is None
135
+
136
+ @property
137
+ def sort_key(self):
138
+ """
139
+ Used to sort PackedNode children of SymbolNodes.
140
+ A SymbolNode has multiple PackedNodes if it matched
141
+ ambiguously. Hence, we use the sort order to identify
142
+ the order in which ambiguous children should be considered.
143
+ """
144
+ return self.is_empty, -self.priority, self.rule.order
145
+
146
+ @property
147
+ def children(self):
148
+ """Returns a list of this node's children."""
149
+ return [x for x in [self.left, self.right] if x is not None]
150
+
151
+ def __iter__(self):
152
+ yield self.left
153
+ yield self.right
154
+
155
+ def __eq__(self, other):
156
+ if not isinstance(other, PackedNode):
157
+ return False
158
+ return self is other or (self.left == other.left and self.right == other.right)
159
+
160
+ def __hash__(self):
161
+ return self._hash
162
+
163
+ def __repr__(self):
164
+ if isinstance(self.s, tuple):
165
+ rule = self.s[0]
166
+ ptr = self.s[1]
167
+ before = ( expansion.name for expansion in rule.expansion[:ptr] )
168
+ after = ( expansion.name for expansion in rule.expansion[ptr:] )
169
+ symbol = "{} ::= {}* {}".format(rule.origin.name, ' '.join(before), ' '.join(after))
170
+ else:
171
+ symbol = self.s.name
172
+ return "({}, {}, {}, {})".format(symbol, self.start, self.priority, self.rule.order)
173
+
174
+ class TokenNode(ForestNode):
175
+ """
176
+ A Token Node represents a matched terminal and is always a leaf node.
177
+
178
+ Parameters:
179
+ token: The Token associated with this node.
180
+ term: The TerminalDef matched by the token.
181
+ priority: The priority of this node.
182
+ """
183
+ __slots__ = ('token', 'term', 'priority', '_hash')
184
+ def __init__(self, token, term, priority=None):
185
+ self.token = token
186
+ self.term = term
187
+ if priority is not None:
188
+ self.priority = priority
189
+ else:
190
+ self.priority = term.priority if term is not None else 0
191
+ self._hash = hash(token)
192
+
193
+ def __eq__(self, other):
194
+ if not isinstance(other, TokenNode):
195
+ return False
196
+ return self is other or (self.token == other.token)
197
+
198
+ def __hash__(self):
199
+ return self._hash
200
+
201
+ def __repr__(self):
202
+ return repr(self.token)
203
+
204
+ class ForestVisitor:
205
+ """
206
+ An abstract base class for building forest visitors.
207
+
208
+ This class performs a controllable depth-first walk of an SPPF.
209
+ The visitor will not enter cycles and will backtrack if one is encountered.
210
+ Subclasses are notified of cycles through the ``on_cycle`` method.
211
+
212
+ Behavior for visit events is defined by overriding the
213
+ ``visit*node*`` functions.
214
+
215
+ The walk is controlled by the return values of the ``visit*node_in``
216
+ methods. Returning a node(s) will schedule them to be visited. The visitor
217
+ will begin to backtrack if no nodes are returned.
218
+
219
+ Parameters:
220
+ single_visit: If ``True``, non-Token nodes will only be visited once.
221
+ """
222
+
223
+ def __init__(self, single_visit=False):
224
+ self.single_visit = single_visit
225
+
226
+ def visit_token_node(self, node):
227
+ """Called when a ``Token`` is visited. ``Token`` nodes are always leaves."""
228
+ pass
229
+
230
+ def visit_symbol_node_in(self, node):
231
+ """Called when a symbol node is visited. Nodes that are returned
232
+ will be scheduled to be visited. If ``visit_intermediate_node_in``
233
+ is not implemented, this function will be called for intermediate
234
+ nodes as well."""
235
+ pass
236
+
237
+ def visit_symbol_node_out(self, node):
238
+ """Called after all nodes returned from a corresponding ``visit_symbol_node_in``
239
+ call have been visited. If ``visit_intermediate_node_out``
240
+ is not implemented, this function will be called for intermediate
241
+ nodes as well."""
242
+ pass
243
+
244
+ def visit_packed_node_in(self, node):
245
+ """Called when a packed node is visited. Nodes that are returned
246
+ will be scheduled to be visited. """
247
+ pass
248
+
249
+ def visit_packed_node_out(self, node):
250
+ """Called after all nodes returned from a corresponding ``visit_packed_node_in``
251
+ call have been visited."""
252
+ pass
253
+
254
+ def on_cycle(self, node, path):
255
+ """Called when a cycle is encountered.
256
+
257
+ Parameters:
258
+ node: The node that causes a cycle.
259
+ path: The list of nodes being visited: nodes that have been
260
+ entered but not exited. The first element is the root in a forest
261
+ visit, and the last element is the node visited most recently.
262
+ ``path`` should be treated as read-only.
263
+ """
264
+ pass
265
+
266
+ def get_cycle_in_path(self, node, path):
267
+ """A utility function for use in ``on_cycle`` to obtain a slice of
268
+ ``path`` that only contains the nodes that make up the cycle."""
269
+ index = len(path) - 1
270
+ while id(path[index]) != id(node):
271
+ index -= 1
272
+ return path[index:]
273
+
274
+ def visit(self, root):
275
+ # Visiting is a list of IDs of all symbol/intermediate nodes currently in
276
+ # the stack. It serves two purposes: to detect when we 'recurse' in and out
277
+ # of a symbol/intermediate so that we can process both up and down. Also,
278
+ # since the SPPF can have cycles it allows us to detect if we're trying
279
+ # to recurse into a node that's already on the stack (infinite recursion).
280
+ visiting = set()
281
+
282
+ # set of all nodes that have been visited
283
+ visited = set()
284
+
285
+ # a list of nodes that are currently being visited
286
+ # used for the `on_cycle` callback
287
+ path = []
288
+
289
+ # We do not use recursion here to walk the Forest due to the limited
290
+ # stack size in python. Therefore input_stack is essentially our stack.
291
+ input_stack = deque([root])
292
+
293
+ # It is much faster to cache these as locals since they are called
294
+ # many times in large parses.
295
+ vpno = getattr(self, 'visit_packed_node_out')
296
+ vpni = getattr(self, 'visit_packed_node_in')
297
+ vsno = getattr(self, 'visit_symbol_node_out')
298
+ vsni = getattr(self, 'visit_symbol_node_in')
299
+ vino = getattr(self, 'visit_intermediate_node_out', vsno)
300
+ vini = getattr(self, 'visit_intermediate_node_in', vsni)
301
+ vtn = getattr(self, 'visit_token_node')
302
+ oc = getattr(self, 'on_cycle')
303
+
304
+ while input_stack:
305
+ current = next(reversed(input_stack))
306
+ try:
307
+ next_node = next(current)
308
+ except StopIteration:
309
+ input_stack.pop()
310
+ continue
311
+ except TypeError:
312
+ ### If the current object is not an iterator, pass through to Token/SymbolNode
313
+ pass
314
+ else:
315
+ if next_node is None:
316
+ continue
317
+
318
+ if id(next_node) in visiting:
319
+ oc(next_node, path)
320
+ continue
321
+
322
+ input_stack.append(next_node)
323
+ continue
324
+
325
+ if isinstance(current, TokenNode):
326
+ vtn(current.token)
327
+ input_stack.pop()
328
+ continue
329
+
330
+ current_id = id(current)
331
+ if current_id in visiting:
332
+ if isinstance(current, PackedNode):
333
+ vpno(current)
334
+ elif current.is_intermediate:
335
+ vino(current)
336
+ else:
337
+ vsno(current)
338
+ input_stack.pop()
339
+ path.pop()
340
+ visiting.remove(current_id)
341
+ visited.add(current_id)
342
+ elif self.single_visit and current_id in visited:
343
+ input_stack.pop()
344
+ else:
345
+ visiting.add(current_id)
346
+ path.append(current)
347
+ if isinstance(current, PackedNode):
348
+ next_node = vpni(current)
349
+ elif current.is_intermediate:
350
+ next_node = vini(current)
351
+ else:
352
+ next_node = vsni(current)
353
+ if next_node is None:
354
+ continue
355
+
356
+ if not isinstance(next_node, ForestNode):
357
+ next_node = iter(next_node)
358
+ elif id(next_node) in visiting:
359
+ oc(next_node, path)
360
+ continue
361
+
362
+ input_stack.append(next_node)
363
+
364
+ class ForestTransformer(ForestVisitor):
365
+ """The base class for a bottom-up forest transformation. Most users will
366
+ want to use ``TreeForestTransformer`` instead as it has a friendlier
367
+ interface and covers most use cases.
368
+
369
+ Transformations are applied via inheritance and overriding of the
370
+ ``transform*node`` methods.
371
+
372
+ ``transform_token_node`` receives a ``Token`` as an argument.
373
+ All other methods receive the node that is being transformed and
374
+ a list of the results of the transformations of that node's children.
375
+ The return value of these methods are the resulting transformations.
376
+
377
+ If ``Discard`` is raised in a node's transformation, no data from that node
378
+ will be passed to its parent's transformation.
379
+ """
380
+
381
+ def __init__(self):
382
+ super(ForestTransformer, self).__init__()
383
+ # results of transformations
384
+ self.data = dict()
385
+ # used to track parent nodes
386
+ self.node_stack = deque()
387
+
388
+ def transform(self, root):
389
+ """Perform a transformation on an SPPF."""
390
+ self.node_stack.append('result')
391
+ self.data['result'] = []
392
+ self.visit(root)
393
+ assert len(self.data['result']) <= 1
394
+ if self.data['result']:
395
+ return self.data['result'][0]
396
+
397
+ def transform_symbol_node(self, node, data):
398
+ """Transform a symbol node."""
399
+ return node
400
+
401
+ def transform_intermediate_node(self, node, data):
402
+ """Transform an intermediate node."""
403
+ return node
404
+
405
+ def transform_packed_node(self, node, data):
406
+ """Transform a packed node."""
407
+ return node
408
+
409
+ def transform_token_node(self, node):
410
+ """Transform a ``Token``."""
411
+ return node
412
+
413
+ def visit_symbol_node_in(self, node):
414
+ self.node_stack.append(id(node))
415
+ self.data[id(node)] = []
416
+ return node.children
417
+
418
+ def visit_packed_node_in(self, node):
419
+ self.node_stack.append(id(node))
420
+ self.data[id(node)] = []
421
+ return node.children
422
+
423
+ def visit_token_node(self, node):
424
+ transformed = self.transform_token_node(node)
425
+ if transformed is not Discard:
426
+ self.data[self.node_stack[-1]].append(transformed)
427
+
428
+ def _visit_node_out_helper(self, node, method):
429
+ self.node_stack.pop()
430
+ transformed = method(node, self.data[id(node)])
431
+ if transformed is not Discard:
432
+ self.data[self.node_stack[-1]].append(transformed)
433
+ del self.data[id(node)]
434
+
435
+ def visit_symbol_node_out(self, node):
436
+ self._visit_node_out_helper(node, self.transform_symbol_node)
437
+
438
+ def visit_intermediate_node_out(self, node):
439
+ self._visit_node_out_helper(node, self.transform_intermediate_node)
440
+
441
+ def visit_packed_node_out(self, node):
442
+ self._visit_node_out_helper(node, self.transform_packed_node)
443
+
444
+
445
+ class ForestSumVisitor(ForestVisitor):
446
+ """
447
+ A visitor for prioritizing ambiguous parts of the Forest.
448
+
449
+ This visitor is used when support for explicit priorities on
450
+ rules is requested (whether normal, or invert). It walks the
451
+ forest (or subsets thereof) and cascades properties upwards
452
+ from the leaves.
453
+
454
+ It would be ideal to do this during parsing, however this would
455
+ require processing each Earley item multiple times. That's
456
+ a big performance drawback; so running a forest walk is the
457
+ lesser of two evils: there can be significantly more Earley
458
+ items created during parsing than there are SPPF nodes in the
459
+ final tree.
460
+ """
461
+ def __init__(self):
462
+ super(ForestSumVisitor, self).__init__(single_visit=True)
463
+
464
+ def visit_packed_node_in(self, node):
465
+ yield node.left
466
+ yield node.right
467
+
468
+ def visit_symbol_node_in(self, node):
469
+ return iter(node.children)
470
+
471
+ def visit_packed_node_out(self, node):
472
+ priority = node.rule.options.priority if not node.parent.is_intermediate and node.rule.options.priority else 0
473
+ priority += getattr(node.right, 'priority', 0)
474
+ priority += getattr(node.left, 'priority', 0)
475
+ node.priority = priority
476
+
477
+ def visit_symbol_node_out(self, node):
478
+ node.priority = max(child.priority for child in node.children)
479
+
480
+ class PackedData():
481
+ """Used in transformationss of packed nodes to distinguish the data
482
+ that comes from the left child and the right child.
483
+ """
484
+
485
+ class _NoData():
486
+ pass
487
+
488
+ NO_DATA = _NoData()
489
+
490
+ def __init__(self, node, data):
491
+ self.left = self.NO_DATA
492
+ self.right = self.NO_DATA
493
+ if data:
494
+ if node.left is not None:
495
+ self.left = data[0]
496
+ if len(data) > 1:
497
+ self.right = data[1]
498
+ else:
499
+ self.right = data[0]
500
+
501
+ class ForestToParseTree(ForestTransformer):
502
+ """Used by the earley parser when ambiguity equals 'resolve' or
503
+ 'explicit'. Transforms an SPPF into an (ambiguous) parse tree.
504
+
505
+ Parameters:
506
+ tree_class: The tree class to use for construction
507
+ callbacks: A dictionary of rules to functions that output a tree
508
+ prioritizer: A ``ForestVisitor`` that manipulates the priorities of ForestNodes
509
+ resolve_ambiguity: If True, ambiguities will be resolved based on
510
+ priorities. Otherwise, `_ambig` nodes will be in the resulting tree.
511
+ use_cache: If True, the results of packed node transformations will be cached.
512
+ """
513
+
514
+ def __init__(self, tree_class=Tree, callbacks=dict(), prioritizer=ForestSumVisitor(), resolve_ambiguity=True, use_cache=True):
515
+ super(ForestToParseTree, self).__init__()
516
+ self.tree_class = tree_class
517
+ self.callbacks = callbacks
518
+ self.prioritizer = prioritizer
519
+ self.resolve_ambiguity = resolve_ambiguity
520
+ self._use_cache = use_cache
521
+ self._cache = {}
522
+ self._on_cycle_retreat = False
523
+ self._cycle_node = None
524
+ self._successful_visits = set()
525
+
526
+ def visit(self, root):
527
+ if self.prioritizer:
528
+ self.prioritizer.visit(root)
529
+ super(ForestToParseTree, self).visit(root)
530
+ self._cache = {}
531
+
532
+ def on_cycle(self, node, path):
533
+ logger.debug("Cycle encountered in the SPPF at node: %s. "
534
+ "As infinite ambiguities cannot be represented in a tree, "
535
+ "this family of derivations will be discarded.", node)
536
+ self._cycle_node = node
537
+ self._on_cycle_retreat = True
538
+
539
+ def _check_cycle(self, node):
540
+ if self._on_cycle_retreat:
541
+ if id(node) == id(self._cycle_node) or id(node) in self._successful_visits:
542
+ self._cycle_node = None
543
+ self._on_cycle_retreat = False
544
+ else:
545
+ return Discard
546
+
547
+ def _collapse_ambig(self, children):
548
+ new_children = []
549
+ for child in children:
550
+ if hasattr(child, 'data') and child.data == '_ambig':
551
+ new_children += child.children
552
+ else:
553
+ new_children.append(child)
554
+ return new_children
555
+
556
+ def _call_rule_func(self, node, data):
557
+ # called when transforming children of symbol nodes
558
+ # data is a list of trees or tokens that correspond to the
559
+ # symbol's rule expansion
560
+ return self.callbacks[node.rule](data)
561
+
562
+ def _call_ambig_func(self, node, data):
563
+ # called when transforming a symbol node
564
+ # data is a list of trees where each tree's data is
565
+ # equal to the name of the symbol or one of its aliases.
566
+ if len(data) > 1:
567
+ return self.tree_class('_ambig', data)
568
+ elif data:
569
+ return data[0]
570
+ return Discard
571
+
572
+ def transform_symbol_node(self, node, data):
573
+ if id(node) not in self._successful_visits:
574
+ return Discard
575
+ r = self._check_cycle(node)
576
+ if r is Discard:
577
+ return r
578
+ self._successful_visits.remove(id(node))
579
+ data = self._collapse_ambig(data)
580
+ return self._call_ambig_func(node, data)
581
+
582
+ def transform_intermediate_node(self, node, data):
583
+ if id(node) not in self._successful_visits:
584
+ return Discard
585
+ r = self._check_cycle(node)
586
+ if r is Discard:
587
+ return r
588
+ self._successful_visits.remove(id(node))
589
+ if len(data) > 1:
590
+ children = [self.tree_class('_inter', c) for c in data]
591
+ return self.tree_class('_iambig', children)
592
+ return data[0]
593
+
594
+ def transform_packed_node(self, node, data):
595
+ r = self._check_cycle(node)
596
+ if r is Discard:
597
+ return r
598
+ if self.resolve_ambiguity and id(node.parent) in self._successful_visits:
599
+ return Discard
600
+ if self._use_cache and id(node) in self._cache:
601
+ return self._cache[id(node)]
602
+ children = []
603
+ assert len(data) <= 2
604
+ data = PackedData(node, data)
605
+ if data.left is not PackedData.NO_DATA:
606
+ if node.left.is_intermediate and isinstance(data.left, list):
607
+ children += data.left
608
+ else:
609
+ children.append(data.left)
610
+ if data.right is not PackedData.NO_DATA:
611
+ children.append(data.right)
612
+ if node.parent.is_intermediate:
613
+ return self._cache.setdefault(id(node), children)
614
+ return self._cache.setdefault(id(node), self._call_rule_func(node, children))
615
+
616
+ def visit_symbol_node_in(self, node):
617
+ super(ForestToParseTree, self).visit_symbol_node_in(node)
618
+ if self._on_cycle_retreat:
619
+ return
620
+ return node.children
621
+
622
+ def visit_packed_node_in(self, node):
623
+ self._on_cycle_retreat = False
624
+ to_visit = super(ForestToParseTree, self).visit_packed_node_in(node)
625
+ if not self.resolve_ambiguity or id(node.parent) not in self._successful_visits:
626
+ if not self._use_cache or id(node) not in self._cache:
627
+ return to_visit
628
+
629
+ def visit_packed_node_out(self, node):
630
+ super(ForestToParseTree, self).visit_packed_node_out(node)
631
+ if not self._on_cycle_retreat:
632
+ self._successful_visits.add(id(node.parent))
633
+
634
+ def handles_ambiguity(func):
635
+ """Decorator for methods of subclasses of ``TreeForestTransformer``.
636
+ Denotes that the method should receive a list of transformed derivations."""
637
+ func.handles_ambiguity = True
638
+ return func
639
+
640
+ class TreeForestTransformer(ForestToParseTree):
641
+ """A ``ForestTransformer`` with a tree ``Transformer``-like interface.
642
+ By default, it will construct a tree.
643
+
644
+ Methods provided via inheritance are called based on the rule/symbol
645
+ names of nodes in the forest.
646
+
647
+ Methods that act on rules will receive a list of the results of the
648
+ transformations of the rule's children. By default, trees and tokens.
649
+
650
+ Methods that act on tokens will receive a token.
651
+
652
+ Alternatively, methods that act on rules may be annotated with
653
+ ``handles_ambiguity``. In this case, the function will receive a list
654
+ of all the transformations of all the derivations of the rule.
655
+ By default, a list of trees where each tree.data is equal to the
656
+ rule name or one of its aliases.
657
+
658
+ Non-tree transformations are made possible by override of
659
+ ``__default__``, ``__default_token__``, and ``__default_ambig__``.
660
+
661
+ Note:
662
+ Tree shaping features such as inlined rules and token filtering are
663
+ not built into the transformation. Positions are also not propagated.
664
+
665
+ Parameters:
666
+ tree_class: The tree class to use for construction
667
+ prioritizer: A ``ForestVisitor`` that manipulates the priorities of nodes in the SPPF.
668
+ resolve_ambiguity: If True, ambiguities will be resolved based on priorities.
669
+ use_cache (bool): If True, caches the results of some transformations,
670
+ potentially improving performance when ``resolve_ambiguity==False``.
671
+ Only use if you know what you are doing: i.e. All transformation
672
+ functions are pure and referentially transparent.
673
+ """
674
+
675
+ def __init__(self, tree_class=Tree, prioritizer=ForestSumVisitor(), resolve_ambiguity=True, use_cache=False):
676
+ super(TreeForestTransformer, self).__init__(tree_class, dict(), prioritizer, resolve_ambiguity, use_cache)
677
+
678
+ def __default__(self, name, data):
679
+ """Default operation on tree (for override).
680
+
681
+ Returns a tree with name with data as children.
682
+ """
683
+ return self.tree_class(name, data)
684
+
685
+ def __default_ambig__(self, name, data):
686
+ """Default operation on ambiguous rule (for override).
687
+
688
+ Wraps data in an '_ambig_' node if it contains more than
689
+ one element.
690
+ """
691
+ if len(data) > 1:
692
+ return self.tree_class('_ambig', data)
693
+ elif data:
694
+ return data[0]
695
+ return Discard
696
+
697
+ def __default_token__(self, node):
698
+ """Default operation on ``Token`` (for override).
699
+
700
+ Returns ``node``.
701
+ """
702
+ return node
703
+
704
+ def transform_token_node(self, node):
705
+ return getattr(self, node.type, self.__default_token__)(node)
706
+
707
+ def _call_rule_func(self, node, data):
708
+ name = node.rule.alias or node.rule.options.template_source or node.rule.origin.name
709
+ user_func = getattr(self, name, self.__default__)
710
+ if user_func == self.__default__ or hasattr(user_func, 'handles_ambiguity'):
711
+ user_func = partial(self.__default__, name)
712
+ if not self.resolve_ambiguity:
713
+ wrapper = partial(AmbiguousIntermediateExpander, self.tree_class)
714
+ user_func = wrapper(user_func)
715
+ return user_func(data)
716
+
717
+ def _call_ambig_func(self, node, data):
718
+ name = node.s.name
719
+ user_func = getattr(self, name, self.__default_ambig__)
720
+ if user_func == self.__default_ambig__ or not hasattr(user_func, 'handles_ambiguity'):
721
+ user_func = partial(self.__default_ambig__, name)
722
+ return user_func(data)
723
+
724
+ class ForestToPyDotVisitor(ForestVisitor):
725
+ """
726
+ A Forest visitor which writes the SPPF to a PNG.
727
+
728
+ The SPPF can get really large, really quickly because
729
+ of the amount of meta-data it stores, so this is probably
730
+ only useful for trivial trees and learning how the SPPF
731
+ is structured.
732
+ """
733
+ def __init__(self, rankdir="TB"):
734
+ super(ForestToPyDotVisitor, self).__init__(single_visit=True)
735
+ self.pydot = import_module('pydot')
736
+ self.graph = self.pydot.Dot(graph_type='digraph', rankdir=rankdir)
737
+
738
+ def visit(self, root, filename):
739
+ super(ForestToPyDotVisitor, self).visit(root)
740
+ try:
741
+ self.graph.write_png(filename)
742
+ except FileNotFoundError as e:
743
+ logger.error("Could not write png: ", e)
744
+
745
+ def visit_token_node(self, node):
746
+ graph_node_id = str(id(node))
747
+ graph_node_label = "\"{}\"".format(node.value.replace('"', '\\"'))
748
+ graph_node_color = 0x808080
749
+ graph_node_style = "\"filled,rounded\""
750
+ graph_node_shape = "diamond"
751
+ graph_node = self.pydot.Node(graph_node_id, style=graph_node_style, fillcolor="#{:06x}".format(graph_node_color), shape=graph_node_shape, label=graph_node_label)
752
+ self.graph.add_node(graph_node)
753
+
754
+ def visit_packed_node_in(self, node):
755
+ graph_node_id = str(id(node))
756
+ graph_node_label = repr(node)
757
+ graph_node_color = 0x808080
758
+ graph_node_style = "filled"
759
+ graph_node_shape = "diamond"
760
+ graph_node = self.pydot.Node(graph_node_id, style=graph_node_style, fillcolor="#{:06x}".format(graph_node_color), shape=graph_node_shape, label=graph_node_label)
761
+ self.graph.add_node(graph_node)
762
+ yield node.left
763
+ yield node.right
764
+
765
+ def visit_packed_node_out(self, node):
766
+ graph_node_id = str(id(node))
767
+ graph_node = self.graph.get_node(graph_node_id)[0]
768
+ for child in [node.left, node.right]:
769
+ if child is not None:
770
+ child_graph_node_id = str(id(child.token if isinstance(child, TokenNode) else child))
771
+ child_graph_node = self.graph.get_node(child_graph_node_id)[0]
772
+ self.graph.add_edge(self.pydot.Edge(graph_node, child_graph_node))
773
+ else:
774
+ #### Try and be above the Python object ID range; probably impl. specific, but maybe this is okay.
775
+ child_graph_node_id = str(randint(100000000000000000000000000000,123456789012345678901234567890))
776
+ child_graph_node_style = "invis"
777
+ child_graph_node = self.pydot.Node(child_graph_node_id, style=child_graph_node_style, label="None")
778
+ child_edge_style = "invis"
779
+ self.graph.add_node(child_graph_node)
780
+ self.graph.add_edge(self.pydot.Edge(graph_node, child_graph_node, style=child_edge_style))
781
+
782
+ def visit_symbol_node_in(self, node):
783
+ graph_node_id = str(id(node))
784
+ graph_node_label = repr(node)
785
+ graph_node_color = 0x808080
786
+ graph_node_style = "\"filled\""
787
+ if node.is_intermediate:
788
+ graph_node_shape = "ellipse"
789
+ else:
790
+ graph_node_shape = "rectangle"
791
+ graph_node = self.pydot.Node(graph_node_id, style=graph_node_style, fillcolor="#{:06x}".format(graph_node_color), shape=graph_node_shape, label=graph_node_label)
792
+ self.graph.add_node(graph_node)
793
+ return iter(node.children)
794
+
795
+ def visit_symbol_node_out(self, node):
796
+ graph_node_id = str(id(node))
797
+ graph_node = self.graph.get_node(graph_node_id)[0]
798
+ for child in node.children:
799
+ child_graph_node_id = str(id(child))
800
+ child_graph_node = self.graph.get_node(child_graph_node_id)[0]
801
+ self.graph.add_edge(self.pydot.Edge(graph_node, child_graph_node))