chia-blockchain 2.4.4__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 (1028) hide show
  1. chia/__init__.py +10 -0
  2. chia/__main__.py +5 -0
  3. chia/_tests/README.md +53 -0
  4. chia/_tests/__init__.py +0 -0
  5. chia/_tests/blockchain/__init__.py +0 -0
  6. chia/_tests/blockchain/blockchain_test_utils.py +197 -0
  7. chia/_tests/blockchain/config.py +4 -0
  8. chia/_tests/blockchain/test_augmented_chain.py +147 -0
  9. chia/_tests/blockchain/test_blockchain.py +4100 -0
  10. chia/_tests/blockchain/test_blockchain_transactions.py +1050 -0
  11. chia/_tests/blockchain/test_build_chains.py +61 -0
  12. chia/_tests/blockchain/test_get_block_generator.py +72 -0
  13. chia/_tests/blockchain/test_lookup_fork_chain.py +195 -0
  14. chia/_tests/build-init-files.py +93 -0
  15. chia/_tests/build-job-matrix.py +204 -0
  16. chia/_tests/check_pytest_monitor_output.py +34 -0
  17. chia/_tests/check_sql_statements.py +73 -0
  18. chia/_tests/chia-start-sim +42 -0
  19. chia/_tests/clvm/__init__.py +0 -0
  20. chia/_tests/clvm/benchmark_costs.py +23 -0
  21. chia/_tests/clvm/coin_store.py +147 -0
  22. chia/_tests/clvm/test_chialisp_deserialization.py +101 -0
  23. chia/_tests/clvm/test_clvm_step.py +37 -0
  24. chia/_tests/clvm/test_condition_codes.py +13 -0
  25. chia/_tests/clvm/test_curry_and_treehash.py +57 -0
  26. chia/_tests/clvm/test_program.py +150 -0
  27. chia/_tests/clvm/test_puzzle_compression.py +144 -0
  28. chia/_tests/clvm/test_puzzle_drivers.py +45 -0
  29. chia/_tests/clvm/test_puzzles.py +247 -0
  30. chia/_tests/clvm/test_singletons.py +540 -0
  31. chia/_tests/clvm/test_spend_sim.py +181 -0
  32. chia/_tests/cmds/__init__.py +0 -0
  33. chia/_tests/cmds/cmd_test_utils.py +472 -0
  34. chia/_tests/cmds/config.py +3 -0
  35. chia/_tests/cmds/conftest.py +23 -0
  36. chia/_tests/cmds/test_click_types.py +195 -0
  37. chia/_tests/cmds/test_cmd_framework.py +400 -0
  38. chia/_tests/cmds/test_cmds_util.py +97 -0
  39. chia/_tests/cmds/test_daemon.py +92 -0
  40. chia/_tests/cmds/test_farm_cmd.py +67 -0
  41. chia/_tests/cmds/test_show.py +116 -0
  42. chia/_tests/cmds/test_sim.py +207 -0
  43. chia/_tests/cmds/test_timelock_args.py +75 -0
  44. chia/_tests/cmds/test_tx_config_args.py +153 -0
  45. chia/_tests/cmds/testing_classes.py +59 -0
  46. chia/_tests/cmds/wallet/__init__.py +0 -0
  47. chia/_tests/cmds/wallet/test_coins.py +195 -0
  48. chia/_tests/cmds/wallet/test_consts.py +47 -0
  49. chia/_tests/cmds/wallet/test_dao.py +565 -0
  50. chia/_tests/cmds/wallet/test_did.py +403 -0
  51. chia/_tests/cmds/wallet/test_nft.py +470 -0
  52. chia/_tests/cmds/wallet/test_notifications.py +124 -0
  53. chia/_tests/cmds/wallet/test_offer.toffer +1 -0
  54. chia/_tests/cmds/wallet/test_tx_decorators.py +27 -0
  55. chia/_tests/cmds/wallet/test_vcs.py +376 -0
  56. chia/_tests/cmds/wallet/test_wallet.py +1126 -0
  57. chia/_tests/cmds/wallet/test_wallet_check.py +111 -0
  58. chia/_tests/conftest.py +1304 -0
  59. chia/_tests/connection_utils.py +124 -0
  60. chia/_tests/core/__init__.py +0 -0
  61. chia/_tests/core/cmds/__init__.py +0 -0
  62. chia/_tests/core/cmds/test_beta.py +382 -0
  63. chia/_tests/core/cmds/test_keys.py +1734 -0
  64. chia/_tests/core/cmds/test_wallet.py +126 -0
  65. chia/_tests/core/config.py +3 -0
  66. chia/_tests/core/consensus/__init__.py +0 -0
  67. chia/_tests/core/consensus/test_block_creation.py +56 -0
  68. chia/_tests/core/consensus/test_pot_iterations.py +117 -0
  69. chia/_tests/core/custom_types/__init__.py +0 -0
  70. chia/_tests/core/custom_types/test_coin.py +109 -0
  71. chia/_tests/core/custom_types/test_proof_of_space.py +144 -0
  72. chia/_tests/core/custom_types/test_spend_bundle.py +71 -0
  73. chia/_tests/core/daemon/__init__.py +0 -0
  74. chia/_tests/core/daemon/config.py +4 -0
  75. chia/_tests/core/daemon/test_daemon.py +2128 -0
  76. chia/_tests/core/daemon/test_daemon_register.py +109 -0
  77. chia/_tests/core/daemon/test_keychain_proxy.py +100 -0
  78. chia/_tests/core/data_layer/__init__.py +0 -0
  79. chia/_tests/core/data_layer/config.py +5 -0
  80. chia/_tests/core/data_layer/conftest.py +105 -0
  81. chia/_tests/core/data_layer/test_data_cli.py +57 -0
  82. chia/_tests/core/data_layer/test_data_layer.py +83 -0
  83. chia/_tests/core/data_layer/test_data_layer_util.py +219 -0
  84. chia/_tests/core/data_layer/test_data_rpc.py +3865 -0
  85. chia/_tests/core/data_layer/test_data_store.py +2423 -0
  86. chia/_tests/core/data_layer/test_data_store_schema.py +381 -0
  87. chia/_tests/core/data_layer/test_plugin.py +91 -0
  88. chia/_tests/core/data_layer/util.py +232 -0
  89. chia/_tests/core/farmer/__init__.py +0 -0
  90. chia/_tests/core/farmer/config.py +3 -0
  91. chia/_tests/core/farmer/test_farmer_api.py +101 -0
  92. chia/_tests/core/full_node/__init__.py +0 -0
  93. chia/_tests/core/full_node/config.py +4 -0
  94. chia/_tests/core/full_node/dos/__init__.py +0 -0
  95. chia/_tests/core/full_node/dos/config.py +3 -0
  96. chia/_tests/core/full_node/full_sync/__init__.py +0 -0
  97. chia/_tests/core/full_node/full_sync/config.py +4 -0
  98. chia/_tests/core/full_node/full_sync/test_full_sync.py +448 -0
  99. chia/_tests/core/full_node/ram_db.py +27 -0
  100. chia/_tests/core/full_node/stores/__init__.py +0 -0
  101. chia/_tests/core/full_node/stores/config.py +4 -0
  102. chia/_tests/core/full_node/stores/test_block_store.py +488 -0
  103. chia/_tests/core/full_node/stores/test_coin_store.py +888 -0
  104. chia/_tests/core/full_node/stores/test_full_node_store.py +1215 -0
  105. chia/_tests/core/full_node/stores/test_hint_store.py +230 -0
  106. chia/_tests/core/full_node/stores/test_sync_store.py +135 -0
  107. chia/_tests/core/full_node/test_address_manager.py +588 -0
  108. chia/_tests/core/full_node/test_block_height_map.py +556 -0
  109. chia/_tests/core/full_node/test_conditions.py +558 -0
  110. chia/_tests/core/full_node/test_full_node.py +2445 -0
  111. chia/_tests/core/full_node/test_generator_tools.py +82 -0
  112. chia/_tests/core/full_node/test_hint_management.py +104 -0
  113. chia/_tests/core/full_node/test_node_load.py +34 -0
  114. chia/_tests/core/full_node/test_performance.py +182 -0
  115. chia/_tests/core/full_node/test_subscriptions.py +492 -0
  116. chia/_tests/core/full_node/test_transactions.py +203 -0
  117. chia/_tests/core/full_node/test_tx_processing_queue.py +154 -0
  118. chia/_tests/core/large_block.py +2388 -0
  119. chia/_tests/core/make_block_generator.py +72 -0
  120. chia/_tests/core/mempool/__init__.py +0 -0
  121. chia/_tests/core/mempool/config.py +4 -0
  122. chia/_tests/core/mempool/test_mempool.py +3180 -0
  123. chia/_tests/core/mempool/test_mempool_fee_estimator.py +104 -0
  124. chia/_tests/core/mempool/test_mempool_fee_protocol.py +55 -0
  125. chia/_tests/core/mempool/test_mempool_item_queries.py +192 -0
  126. chia/_tests/core/mempool/test_mempool_manager.py +2054 -0
  127. chia/_tests/core/mempool/test_mempool_performance.py +65 -0
  128. chia/_tests/core/mempool/test_singleton_fast_forward.py +567 -0
  129. chia/_tests/core/node_height.py +28 -0
  130. chia/_tests/core/server/__init__.py +0 -0
  131. chia/_tests/core/server/config.py +3 -0
  132. chia/_tests/core/server/flood.py +82 -0
  133. chia/_tests/core/server/serve.py +132 -0
  134. chia/_tests/core/server/test_capabilities.py +68 -0
  135. chia/_tests/core/server/test_dos.py +320 -0
  136. chia/_tests/core/server/test_event_loop.py +109 -0
  137. chia/_tests/core/server/test_loop.py +290 -0
  138. chia/_tests/core/server/test_node_discovery.py +74 -0
  139. chia/_tests/core/server/test_rate_limits.py +370 -0
  140. chia/_tests/core/server/test_server.py +225 -0
  141. chia/_tests/core/server/test_upnp.py +8 -0
  142. chia/_tests/core/services/__init__.py +0 -0
  143. chia/_tests/core/services/config.py +3 -0
  144. chia/_tests/core/services/test_services.py +166 -0
  145. chia/_tests/core/ssl/__init__.py +0 -0
  146. chia/_tests/core/ssl/config.py +3 -0
  147. chia/_tests/core/ssl/test_ssl.py +198 -0
  148. chia/_tests/core/test_coins.py +33 -0
  149. chia/_tests/core/test_cost_calculation.py +314 -0
  150. chia/_tests/core/test_crawler.py +175 -0
  151. chia/_tests/core/test_crawler_rpc.py +53 -0
  152. chia/_tests/core/test_daemon_rpc.py +24 -0
  153. chia/_tests/core/test_db_conversion.py +129 -0
  154. chia/_tests/core/test_db_validation.py +161 -0
  155. chia/_tests/core/test_farmer_harvester_rpc.py +504 -0
  156. chia/_tests/core/test_filter.py +37 -0
  157. chia/_tests/core/test_full_node_rpc.py +794 -0
  158. chia/_tests/core/test_merkle_set.py +343 -0
  159. chia/_tests/core/test_program.py +49 -0
  160. chia/_tests/core/test_rpc_util.py +87 -0
  161. chia/_tests/core/test_seeder.py +308 -0
  162. chia/_tests/core/test_setproctitle.py +13 -0
  163. chia/_tests/core/util/__init__.py +0 -0
  164. chia/_tests/core/util/config.py +4 -0
  165. chia/_tests/core/util/test_block_cache.py +44 -0
  166. chia/_tests/core/util/test_cached_bls.py +57 -0
  167. chia/_tests/core/util/test_config.py +337 -0
  168. chia/_tests/core/util/test_file_keyring_synchronization.py +105 -0
  169. chia/_tests/core/util/test_files.py +391 -0
  170. chia/_tests/core/util/test_jsonify.py +146 -0
  171. chia/_tests/core/util/test_keychain.py +514 -0
  172. chia/_tests/core/util/test_keyring_wrapper.py +490 -0
  173. chia/_tests/core/util/test_lockfile.py +380 -0
  174. chia/_tests/core/util/test_log_exceptions.py +187 -0
  175. chia/_tests/core/util/test_lru_cache.py +56 -0
  176. chia/_tests/core/util/test_significant_bits.py +40 -0
  177. chia/_tests/core/util/test_streamable.py +883 -0
  178. chia/_tests/db/__init__.py +0 -0
  179. chia/_tests/db/test_db_wrapper.py +565 -0
  180. chia/_tests/environments/__init__.py +0 -0
  181. chia/_tests/environments/common.py +35 -0
  182. chia/_tests/environments/full_node.py +47 -0
  183. chia/_tests/environments/wallet.py +368 -0
  184. chia/_tests/ether.py +19 -0
  185. chia/_tests/farmer_harvester/__init__.py +0 -0
  186. chia/_tests/farmer_harvester/config.py +3 -0
  187. chia/_tests/farmer_harvester/test_farmer.py +1264 -0
  188. chia/_tests/farmer_harvester/test_farmer_harvester.py +292 -0
  189. chia/_tests/farmer_harvester/test_filter_prefix_bits.py +130 -0
  190. chia/_tests/farmer_harvester/test_third_party_harvesters.py +501 -0
  191. chia/_tests/farmer_harvester/test_third_party_harvesters_data.json +29 -0
  192. chia/_tests/fee_estimation/__init__.py +0 -0
  193. chia/_tests/fee_estimation/config.py +3 -0
  194. chia/_tests/fee_estimation/test_fee_estimation_integration.py +262 -0
  195. chia/_tests/fee_estimation/test_fee_estimation_rpc.py +287 -0
  196. chia/_tests/fee_estimation/test_fee_estimation_unit_tests.py +145 -0
  197. chia/_tests/fee_estimation/test_mempoolitem_height_added.py +146 -0
  198. chia/_tests/generator/__init__.py +0 -0
  199. chia/_tests/generator/puzzles/__init__.py +0 -0
  200. chia/_tests/generator/puzzles/test_generator_deserialize.clsp +3 -0
  201. chia/_tests/generator/puzzles/test_generator_deserialize.clsp.hex +1 -0
  202. chia/_tests/generator/puzzles/test_multiple_generator_input_arguments.clsp +19 -0
  203. chia/_tests/generator/puzzles/test_multiple_generator_input_arguments.clsp.hex +1 -0
  204. chia/_tests/generator/test_compression.py +218 -0
  205. chia/_tests/generator/test_generator_types.py +44 -0
  206. chia/_tests/generator/test_rom.py +182 -0
  207. chia/_tests/plot_sync/__init__.py +0 -0
  208. chia/_tests/plot_sync/config.py +3 -0
  209. chia/_tests/plot_sync/test_delta.py +102 -0
  210. chia/_tests/plot_sync/test_plot_sync.py +617 -0
  211. chia/_tests/plot_sync/test_receiver.py +451 -0
  212. chia/_tests/plot_sync/test_sender.py +116 -0
  213. chia/_tests/plot_sync/test_sync_simulated.py +450 -0
  214. chia/_tests/plot_sync/util.py +67 -0
  215. chia/_tests/plotting/__init__.py +0 -0
  216. chia/_tests/plotting/config.py +3 -0
  217. chia/_tests/plotting/test_plot_manager.py +738 -0
  218. chia/_tests/plotting/util.py +13 -0
  219. chia/_tests/pools/__init__.py +0 -0
  220. chia/_tests/pools/config.py +5 -0
  221. chia/_tests/pools/test_pool_cmdline.py +23 -0
  222. chia/_tests/pools/test_pool_config.py +44 -0
  223. chia/_tests/pools/test_pool_puzzles_lifecycle.py +398 -0
  224. chia/_tests/pools/test_pool_rpc.py +1010 -0
  225. chia/_tests/pools/test_pool_wallet.py +201 -0
  226. chia/_tests/pools/test_wallet_pool_store.py +161 -0
  227. chia/_tests/process_junit.py +349 -0
  228. chia/_tests/rpc/__init__.py +0 -0
  229. chia/_tests/rpc/test_rpc_client.py +81 -0
  230. chia/_tests/simulation/__init__.py +0 -0
  231. chia/_tests/simulation/config.py +6 -0
  232. chia/_tests/simulation/test_simulation.py +501 -0
  233. chia/_tests/simulation/test_simulator.py +234 -0
  234. chia/_tests/simulation/test_start_simulator.py +106 -0
  235. chia/_tests/testconfig.py +13 -0
  236. chia/_tests/timelord/__init__.py +0 -0
  237. chia/_tests/timelord/config.py +3 -0
  238. chia/_tests/timelord/test_new_peak.py +437 -0
  239. chia/_tests/timelord/test_timelord.py +11 -0
  240. chia/_tests/tools/1315537.json +170 -0
  241. chia/_tests/tools/1315544.json +160 -0
  242. chia/_tests/tools/1315630.json +150 -0
  243. chia/_tests/tools/300000.json +105 -0
  244. chia/_tests/tools/442734.json +140 -0
  245. chia/_tests/tools/466212.json +130 -0
  246. chia/_tests/tools/__init__.py +0 -0
  247. chia/_tests/tools/config.py +5 -0
  248. chia/_tests/tools/test-blockchain-db.sqlite +0 -0
  249. chia/_tests/tools/test_full_sync.py +30 -0
  250. chia/_tests/tools/test_legacy_keyring.py +82 -0
  251. chia/_tests/tools/test_run_block.py +129 -0
  252. chia/_tests/util/__init__.py +0 -0
  253. chia/_tests/util/benchmark_cost.py +170 -0
  254. chia/_tests/util/benchmarks.py +154 -0
  255. chia/_tests/util/bip39_test_vectors.json +148 -0
  256. chia/_tests/util/blockchain.py +133 -0
  257. chia/_tests/util/blockchain_mock.py +132 -0
  258. chia/_tests/util/build_network_protocol_files.py +302 -0
  259. chia/_tests/util/clvm_generator.bin +0 -0
  260. chia/_tests/util/config.py +3 -0
  261. chia/_tests/util/constants.py +20 -0
  262. chia/_tests/util/db_connection.py +36 -0
  263. chia/_tests/util/full_sync.py +245 -0
  264. chia/_tests/util/gen_ssl_certs.py +115 -0
  265. chia/_tests/util/generator_tools_testing.py +47 -0
  266. chia/_tests/util/key_tool.py +37 -0
  267. chia/_tests/util/misc.py +722 -0
  268. chia/_tests/util/network_protocol_data.py +1074 -0
  269. chia/_tests/util/protocol_messages_bytes-v1.0 +0 -0
  270. chia/_tests/util/protocol_messages_json.py +2700 -0
  271. chia/_tests/util/rpc.py +23 -0
  272. chia/_tests/util/run_block.py +163 -0
  273. chia/_tests/util/setup_nodes.py +479 -0
  274. chia/_tests/util/split_managers.py +99 -0
  275. chia/_tests/util/temp_file.py +14 -0
  276. chia/_tests/util/test_action_scope.py +143 -0
  277. chia/_tests/util/test_async_pool.py +366 -0
  278. chia/_tests/util/test_build_job_matrix.py +43 -0
  279. chia/_tests/util/test_build_network_protocol_files.py +7 -0
  280. chia/_tests/util/test_chia_version.py +50 -0
  281. chia/_tests/util/test_collection.py +11 -0
  282. chia/_tests/util/test_condition_tools.py +231 -0
  283. chia/_tests/util/test_config.py +426 -0
  284. chia/_tests/util/test_dump_keyring.py +60 -0
  285. chia/_tests/util/test_errors.py +10 -0
  286. chia/_tests/util/test_full_block_utils.py +271 -0
  287. chia/_tests/util/test_installed.py +20 -0
  288. chia/_tests/util/test_limited_semaphore.py +52 -0
  289. chia/_tests/util/test_logging_filter.py +43 -0
  290. chia/_tests/util/test_misc.py +444 -0
  291. chia/_tests/util/test_network.py +74 -0
  292. chia/_tests/util/test_network_protocol_files.py +579 -0
  293. chia/_tests/util/test_network_protocol_json.py +266 -0
  294. chia/_tests/util/test_network_protocol_test.py +257 -0
  295. chia/_tests/util/test_paginator.py +72 -0
  296. chia/_tests/util/test_pprint.py +17 -0
  297. chia/_tests/util/test_priority_mutex.py +487 -0
  298. chia/_tests/util/test_recursive_replace.py +116 -0
  299. chia/_tests/util/test_replace_str_to_bytes.py +137 -0
  300. chia/_tests/util/test_service_groups.py +15 -0
  301. chia/_tests/util/test_ssl_check.py +31 -0
  302. chia/_tests/util/test_testnet_overrides.py +19 -0
  303. chia/_tests/util/test_tests_misc.py +38 -0
  304. chia/_tests/util/test_timing.py +37 -0
  305. chia/_tests/util/test_trusted_peer.py +51 -0
  306. chia/_tests/util/time_out_assert.py +154 -0
  307. chia/_tests/wallet/__init__.py +0 -0
  308. chia/_tests/wallet/cat_wallet/__init__.py +0 -0
  309. chia/_tests/wallet/cat_wallet/config.py +4 -0
  310. chia/_tests/wallet/cat_wallet/test_cat_lifecycle.py +468 -0
  311. chia/_tests/wallet/cat_wallet/test_cat_outer_puzzle.py +69 -0
  312. chia/_tests/wallet/cat_wallet/test_cat_wallet.py +1738 -0
  313. chia/_tests/wallet/cat_wallet/test_offer_lifecycle.py +291 -0
  314. chia/_tests/wallet/cat_wallet/test_trades.py +2578 -0
  315. chia/_tests/wallet/clawback/__init__.py +0 -0
  316. chia/_tests/wallet/clawback/config.py +3 -0
  317. chia/_tests/wallet/clawback/test_clawback_decorator.py +80 -0
  318. chia/_tests/wallet/clawback/test_clawback_lifecycle.py +292 -0
  319. chia/_tests/wallet/clawback/test_clawback_metadata.py +51 -0
  320. chia/_tests/wallet/config.py +4 -0
  321. chia/_tests/wallet/conftest.py +217 -0
  322. chia/_tests/wallet/dao_wallet/__init__.py +0 -0
  323. chia/_tests/wallet/dao_wallet/config.py +3 -0
  324. chia/_tests/wallet/dao_wallet/test_dao_clvm.py +1322 -0
  325. chia/_tests/wallet/dao_wallet/test_dao_wallets.py +3488 -0
  326. chia/_tests/wallet/db_wallet/__init__.py +0 -0
  327. chia/_tests/wallet/db_wallet/config.py +3 -0
  328. chia/_tests/wallet/db_wallet/test_db_graftroot.py +143 -0
  329. chia/_tests/wallet/db_wallet/test_dl_offers.py +491 -0
  330. chia/_tests/wallet/db_wallet/test_dl_wallet.py +823 -0
  331. chia/_tests/wallet/did_wallet/__init__.py +0 -0
  332. chia/_tests/wallet/did_wallet/config.py +4 -0
  333. chia/_tests/wallet/did_wallet/test_did.py +1481 -0
  334. chia/_tests/wallet/nft_wallet/__init__.py +0 -0
  335. chia/_tests/wallet/nft_wallet/config.py +4 -0
  336. chia/_tests/wallet/nft_wallet/test_nft_1_offers.py +1492 -0
  337. chia/_tests/wallet/nft_wallet/test_nft_bulk_mint.py +1014 -0
  338. chia/_tests/wallet/nft_wallet/test_nft_lifecycle.py +376 -0
  339. chia/_tests/wallet/nft_wallet/test_nft_offers.py +1209 -0
  340. chia/_tests/wallet/nft_wallet/test_nft_puzzles.py +172 -0
  341. chia/_tests/wallet/nft_wallet/test_nft_wallet.py +2558 -0
  342. chia/_tests/wallet/nft_wallet/test_ownership_outer_puzzle.py +70 -0
  343. chia/_tests/wallet/rpc/__init__.py +0 -0
  344. chia/_tests/wallet/rpc/config.py +4 -0
  345. chia/_tests/wallet/rpc/test_dl_wallet_rpc.py +287 -0
  346. chia/_tests/wallet/rpc/test_wallet_rpc.py +3106 -0
  347. chia/_tests/wallet/simple_sync/__init__.py +0 -0
  348. chia/_tests/wallet/simple_sync/config.py +3 -0
  349. chia/_tests/wallet/simple_sync/test_simple_sync_protocol.py +719 -0
  350. chia/_tests/wallet/sync/__init__.py +0 -0
  351. chia/_tests/wallet/sync/config.py +4 -0
  352. chia/_tests/wallet/sync/test_wallet_sync.py +1529 -0
  353. chia/_tests/wallet/test_address_type.py +189 -0
  354. chia/_tests/wallet/test_bech32m.py +45 -0
  355. chia/_tests/wallet/test_clvm_streamable.py +244 -0
  356. chia/_tests/wallet/test_coin_selection.py +589 -0
  357. chia/_tests/wallet/test_conditions.py +388 -0
  358. chia/_tests/wallet/test_debug_spend_bundle.py +76 -0
  359. chia/_tests/wallet/test_new_wallet_protocol.py +1176 -0
  360. chia/_tests/wallet/test_nft_store.py +193 -0
  361. chia/_tests/wallet/test_notifications.py +196 -0
  362. chia/_tests/wallet/test_offer_parsing_performance.py +48 -0
  363. chia/_tests/wallet/test_puzzle_store.py +133 -0
  364. chia/_tests/wallet/test_sign_coin_spends.py +159 -0
  365. chia/_tests/wallet/test_signer_protocol.py +948 -0
  366. chia/_tests/wallet/test_singleton.py +122 -0
  367. chia/_tests/wallet/test_singleton_lifecycle_fast.py +772 -0
  368. chia/_tests/wallet/test_singleton_store.py +152 -0
  369. chia/_tests/wallet/test_taproot.py +19 -0
  370. chia/_tests/wallet/test_transaction_store.py +941 -0
  371. chia/_tests/wallet/test_util.py +181 -0
  372. chia/_tests/wallet/test_wallet.py +2139 -0
  373. chia/_tests/wallet/test_wallet_action_scope.py +85 -0
  374. chia/_tests/wallet/test_wallet_blockchain.py +113 -0
  375. chia/_tests/wallet/test_wallet_coin_store.py +1002 -0
  376. chia/_tests/wallet/test_wallet_interested_store.py +43 -0
  377. chia/_tests/wallet/test_wallet_key_val_store.py +40 -0
  378. chia/_tests/wallet/test_wallet_node.py +783 -0
  379. chia/_tests/wallet/test_wallet_retry.py +95 -0
  380. chia/_tests/wallet/test_wallet_state_manager.py +252 -0
  381. chia/_tests/wallet/test_wallet_test_framework.py +275 -0
  382. chia/_tests/wallet/test_wallet_trade_store.py +218 -0
  383. chia/_tests/wallet/test_wallet_user_store.py +34 -0
  384. chia/_tests/wallet/test_wallet_utils.py +155 -0
  385. chia/_tests/wallet/vc_wallet/__init__.py +0 -0
  386. chia/_tests/wallet/vc_wallet/config.py +3 -0
  387. chia/_tests/wallet/vc_wallet/test_cr_outer_puzzle.py +70 -0
  388. chia/_tests/wallet/vc_wallet/test_vc_lifecycle.py +883 -0
  389. chia/_tests/wallet/vc_wallet/test_vc_wallet.py +801 -0
  390. chia/_tests/wallet/wallet_block_tools.py +327 -0
  391. chia/_tests/weight_proof/__init__.py +0 -0
  392. chia/_tests/weight_proof/config.py +3 -0
  393. chia/_tests/weight_proof/test_weight_proof.py +528 -0
  394. chia/clvm/__init__.py +0 -0
  395. chia/clvm/spend_sim.py +488 -0
  396. chia/cmds/__init__.py +0 -0
  397. chia/cmds/beta.py +183 -0
  398. chia/cmds/beta_funcs.py +133 -0
  399. chia/cmds/check_wallet_db.py +418 -0
  400. chia/cmds/chia.py +143 -0
  401. chia/cmds/cmd_classes.py +315 -0
  402. chia/cmds/cmds_util.py +498 -0
  403. chia/cmds/coin_funcs.py +260 -0
  404. chia/cmds/coins.py +220 -0
  405. chia/cmds/completion.py +49 -0
  406. chia/cmds/configure.py +331 -0
  407. chia/cmds/dao.py +1008 -0
  408. chia/cmds/dao_funcs.py +576 -0
  409. chia/cmds/data.py +707 -0
  410. chia/cmds/data_funcs.py +380 -0
  411. chia/cmds/db.py +86 -0
  412. chia/cmds/db_backup_func.py +77 -0
  413. chia/cmds/db_upgrade_func.py +452 -0
  414. chia/cmds/db_validate_func.py +184 -0
  415. chia/cmds/dev.py +16 -0
  416. chia/cmds/farm.py +87 -0
  417. chia/cmds/farm_funcs.py +207 -0
  418. chia/cmds/init.py +70 -0
  419. chia/cmds/init_funcs.py +367 -0
  420. chia/cmds/installers.py +129 -0
  421. chia/cmds/keys.py +510 -0
  422. chia/cmds/keys_funcs.py +864 -0
  423. chia/cmds/netspace.py +47 -0
  424. chia/cmds/netspace_funcs.py +53 -0
  425. chia/cmds/options.py +32 -0
  426. chia/cmds/param_types.py +228 -0
  427. chia/cmds/passphrase.py +130 -0
  428. chia/cmds/passphrase_funcs.py +346 -0
  429. chia/cmds/peer.py +50 -0
  430. chia/cmds/peer_funcs.py +129 -0
  431. chia/cmds/plotnft.py +206 -0
  432. chia/cmds/plotnft_funcs.py +374 -0
  433. chia/cmds/plots.py +222 -0
  434. chia/cmds/plotters.py +17 -0
  435. chia/cmds/rpc.py +188 -0
  436. chia/cmds/show.py +71 -0
  437. chia/cmds/show_funcs.py +214 -0
  438. chia/cmds/signer.py +304 -0
  439. chia/cmds/sim.py +217 -0
  440. chia/cmds/sim_funcs.py +509 -0
  441. chia/cmds/start.py +24 -0
  442. chia/cmds/start_funcs.py +112 -0
  443. chia/cmds/stop.py +61 -0
  444. chia/cmds/units.py +11 -0
  445. chia/cmds/wallet.py +1745 -0
  446. chia/cmds/wallet_funcs.py +1800 -0
  447. chia/consensus/__init__.py +0 -0
  448. chia/consensus/block_body_validation.py +515 -0
  449. chia/consensus/block_creation.py +525 -0
  450. chia/consensus/block_header_validation.py +1064 -0
  451. chia/consensus/block_record.py +32 -0
  452. chia/consensus/block_rewards.py +53 -0
  453. chia/consensus/block_root_validation.py +46 -0
  454. chia/consensus/blockchain.py +1100 -0
  455. chia/consensus/blockchain_interface.py +56 -0
  456. chia/consensus/coinbase.py +30 -0
  457. chia/consensus/condition_costs.py +9 -0
  458. chia/consensus/constants.py +49 -0
  459. chia/consensus/cost_calculator.py +15 -0
  460. chia/consensus/default_constants.py +90 -0
  461. chia/consensus/deficit.py +55 -0
  462. chia/consensus/difficulty_adjustment.py +412 -0
  463. chia/consensus/find_fork_point.py +111 -0
  464. chia/consensus/full_block_to_block_record.py +167 -0
  465. chia/consensus/get_block_challenge.py +106 -0
  466. chia/consensus/get_block_generator.py +26 -0
  467. chia/consensus/make_sub_epoch_summary.py +210 -0
  468. chia/consensus/multiprocess_validation.py +365 -0
  469. chia/consensus/pos_quality.py +19 -0
  470. chia/consensus/pot_iterations.py +67 -0
  471. chia/consensus/puzzles/__init__.py +0 -0
  472. chia/consensus/puzzles/chialisp_deserialisation.clsp +69 -0
  473. chia/consensus/puzzles/chialisp_deserialisation.clsp.hex +1 -0
  474. chia/consensus/puzzles/rom_bootstrap_generator.clsp +37 -0
  475. chia/consensus/puzzles/rom_bootstrap_generator.clsp.hex +1 -0
  476. chia/consensus/vdf_info_computation.py +156 -0
  477. chia/daemon/__init__.py +0 -0
  478. chia/daemon/client.py +233 -0
  479. chia/daemon/keychain_proxy.py +501 -0
  480. chia/daemon/keychain_server.py +365 -0
  481. chia/daemon/server.py +1616 -0
  482. chia/daemon/windows_signal.py +56 -0
  483. chia/data_layer/__init__.py +0 -0
  484. chia/data_layer/data_layer.py +1303 -0
  485. chia/data_layer/data_layer_api.py +25 -0
  486. chia/data_layer/data_layer_errors.py +50 -0
  487. chia/data_layer/data_layer_server.py +170 -0
  488. chia/data_layer/data_layer_util.py +985 -0
  489. chia/data_layer/data_layer_wallet.py +1315 -0
  490. chia/data_layer/data_store.py +2267 -0
  491. chia/data_layer/dl_wallet_store.py +407 -0
  492. chia/data_layer/download_data.py +389 -0
  493. chia/data_layer/puzzles/__init__.py +0 -0
  494. chia/data_layer/puzzles/graftroot_dl_offers.clsp +100 -0
  495. chia/data_layer/puzzles/graftroot_dl_offers.clsp.hex +1 -0
  496. chia/data_layer/s3_plugin_config.yml +33 -0
  497. chia/data_layer/s3_plugin_service.py +468 -0
  498. chia/data_layer/util/__init__.py +0 -0
  499. chia/data_layer/util/benchmark.py +108 -0
  500. chia/data_layer/util/plugin.py +41 -0
  501. chia/farmer/__init__.py +0 -0
  502. chia/farmer/farmer.py +920 -0
  503. chia/farmer/farmer_api.py +814 -0
  504. chia/full_node/__init__.py +0 -0
  505. chia/full_node/bitcoin_fee_estimator.py +85 -0
  506. chia/full_node/block_height_map.py +271 -0
  507. chia/full_node/block_store.py +570 -0
  508. chia/full_node/bundle_tools.py +19 -0
  509. chia/full_node/coin_store.py +646 -0
  510. chia/full_node/fee_estimate.py +54 -0
  511. chia/full_node/fee_estimate_store.py +24 -0
  512. chia/full_node/fee_estimation.py +93 -0
  513. chia/full_node/fee_estimator.py +90 -0
  514. chia/full_node/fee_estimator_constants.py +38 -0
  515. chia/full_node/fee_estimator_interface.py +42 -0
  516. chia/full_node/fee_history.py +26 -0
  517. chia/full_node/fee_tracker.py +564 -0
  518. chia/full_node/full_node.py +3052 -0
  519. chia/full_node/full_node_api.py +1974 -0
  520. chia/full_node/full_node_store.py +1033 -0
  521. chia/full_node/hint_management.py +56 -0
  522. chia/full_node/hint_store.py +94 -0
  523. chia/full_node/mempool.py +583 -0
  524. chia/full_node/mempool_check_conditions.py +177 -0
  525. chia/full_node/mempool_manager.py +858 -0
  526. chia/full_node/pending_tx_cache.py +112 -0
  527. chia/full_node/puzzles/__init__.py +0 -0
  528. chia/full_node/puzzles/block_program_zero.clsp +14 -0
  529. chia/full_node/puzzles/block_program_zero.clsp.hex +1 -0
  530. chia/full_node/puzzles/decompress_coin_spend_entry.clsp +5 -0
  531. chia/full_node/puzzles/decompress_coin_spend_entry.clsp.hex +1 -0
  532. chia/full_node/puzzles/decompress_coin_spend_entry_with_prefix.clsp +7 -0
  533. chia/full_node/puzzles/decompress_coin_spend_entry_with_prefix.clsp.hex +1 -0
  534. chia/full_node/puzzles/decompress_puzzle.clsp +6 -0
  535. chia/full_node/puzzles/decompress_puzzle.clsp.hex +1 -0
  536. chia/full_node/signage_point.py +16 -0
  537. chia/full_node/subscriptions.py +248 -0
  538. chia/full_node/sync_store.py +145 -0
  539. chia/full_node/tx_processing_queue.py +78 -0
  540. chia/full_node/weight_proof.py +1719 -0
  541. chia/harvester/__init__.py +0 -0
  542. chia/harvester/harvester.py +271 -0
  543. chia/harvester/harvester_api.py +374 -0
  544. chia/introducer/__init__.py +0 -0
  545. chia/introducer/introducer.py +120 -0
  546. chia/introducer/introducer_api.py +64 -0
  547. chia/legacy/__init__.py +0 -0
  548. chia/legacy/keyring.py +154 -0
  549. chia/plot_sync/__init__.py +0 -0
  550. chia/plot_sync/delta.py +61 -0
  551. chia/plot_sync/exceptions.py +56 -0
  552. chia/plot_sync/receiver.py +385 -0
  553. chia/plot_sync/sender.py +337 -0
  554. chia/plot_sync/util.py +43 -0
  555. chia/plotters/__init__.py +0 -0
  556. chia/plotters/bladebit.py +388 -0
  557. chia/plotters/chiapos.py +63 -0
  558. chia/plotters/madmax.py +224 -0
  559. chia/plotters/plotters.py +577 -0
  560. chia/plotters/plotters_util.py +131 -0
  561. chia/plotting/__init__.py +0 -0
  562. chia/plotting/cache.py +212 -0
  563. chia/plotting/check_plots.py +283 -0
  564. chia/plotting/create_plots.py +278 -0
  565. chia/plotting/manager.py +436 -0
  566. chia/plotting/util.py +324 -0
  567. chia/pools/__init__.py +0 -0
  568. chia/pools/pool_config.py +110 -0
  569. chia/pools/pool_puzzles.py +459 -0
  570. chia/pools/pool_wallet.py +926 -0
  571. chia/pools/pool_wallet_info.py +118 -0
  572. chia/pools/puzzles/__init__.py +0 -0
  573. chia/pools/puzzles/pool_member_innerpuz.clsp +70 -0
  574. chia/pools/puzzles/pool_member_innerpuz.clsp.hex +1 -0
  575. chia/pools/puzzles/pool_waitingroom_innerpuz.clsp +69 -0
  576. chia/pools/puzzles/pool_waitingroom_innerpuz.clsp.hex +1 -0
  577. chia/protocols/__init__.py +0 -0
  578. chia/protocols/farmer_protocol.py +102 -0
  579. chia/protocols/full_node_protocol.py +219 -0
  580. chia/protocols/harvester_protocol.py +216 -0
  581. chia/protocols/introducer_protocol.py +26 -0
  582. chia/protocols/pool_protocol.py +177 -0
  583. chia/protocols/protocol_message_types.py +139 -0
  584. chia/protocols/protocol_state_machine.py +87 -0
  585. chia/protocols/protocol_timing.py +7 -0
  586. chia/protocols/shared_protocol.py +86 -0
  587. chia/protocols/timelord_protocol.py +93 -0
  588. chia/protocols/wallet_protocol.py +401 -0
  589. chia/py.typed +0 -0
  590. chia/rpc/__init__.py +0 -0
  591. chia/rpc/crawler_rpc_api.py +75 -0
  592. chia/rpc/data_layer_rpc_api.py +639 -0
  593. chia/rpc/data_layer_rpc_client.py +188 -0
  594. chia/rpc/data_layer_rpc_util.py +62 -0
  595. chia/rpc/farmer_rpc_api.py +360 -0
  596. chia/rpc/farmer_rpc_client.py +86 -0
  597. chia/rpc/full_node_rpc_api.py +954 -0
  598. chia/rpc/full_node_rpc_client.py +292 -0
  599. chia/rpc/harvester_rpc_api.py +136 -0
  600. chia/rpc/harvester_rpc_client.py +54 -0
  601. chia/rpc/rpc_client.py +144 -0
  602. chia/rpc/rpc_server.py +447 -0
  603. chia/rpc/timelord_rpc_api.py +27 -0
  604. chia/rpc/util.py +293 -0
  605. chia/rpc/wallet_request_types.py +688 -0
  606. chia/rpc/wallet_rpc_api.py +4779 -0
  607. chia/rpc/wallet_rpc_client.py +1844 -0
  608. chia/seeder/__init__.py +0 -0
  609. chia/seeder/crawl_store.py +427 -0
  610. chia/seeder/crawler.py +423 -0
  611. chia/seeder/crawler_api.py +129 -0
  612. chia/seeder/dns_server.py +544 -0
  613. chia/seeder/peer_record.py +146 -0
  614. chia/seeder/start_crawler.py +88 -0
  615. chia/server/__init__.py +0 -0
  616. chia/server/address_manager.py +658 -0
  617. chia/server/address_manager_store.py +237 -0
  618. chia/server/api_protocol.py +11 -0
  619. chia/server/capabilities.py +24 -0
  620. chia/server/chia_policy.py +345 -0
  621. chia/server/introducer_peers.py +76 -0
  622. chia/server/node_discovery.py +718 -0
  623. chia/server/outbound_message.py +33 -0
  624. chia/server/rate_limit_numbers.py +204 -0
  625. chia/server/rate_limits.py +113 -0
  626. chia/server/server.py +720 -0
  627. chia/server/signal_handlers.py +117 -0
  628. chia/server/ssl_context.py +32 -0
  629. chia/server/start_data_layer.py +137 -0
  630. chia/server/start_farmer.py +86 -0
  631. chia/server/start_full_node.py +106 -0
  632. chia/server/start_harvester.py +80 -0
  633. chia/server/start_introducer.py +69 -0
  634. chia/server/start_service.py +328 -0
  635. chia/server/start_timelord.py +82 -0
  636. chia/server/start_wallet.py +109 -0
  637. chia/server/upnp.py +117 -0
  638. chia/server/ws_connection.py +752 -0
  639. chia/simulator/__init__.py +0 -0
  640. chia/simulator/block_tools.py +2053 -0
  641. chia/simulator/full_node_simulator.py +802 -0
  642. chia/simulator/keyring.py +128 -0
  643. chia/simulator/setup_services.py +505 -0
  644. chia/simulator/simulator_constants.py +13 -0
  645. chia/simulator/simulator_full_node_rpc_api.py +101 -0
  646. chia/simulator/simulator_full_node_rpc_client.py +62 -0
  647. chia/simulator/simulator_protocol.py +29 -0
  648. chia/simulator/simulator_test_tools.py +163 -0
  649. chia/simulator/socket.py +27 -0
  650. chia/simulator/ssl_certs.py +114 -0
  651. chia/simulator/ssl_certs_1.py +699 -0
  652. chia/simulator/ssl_certs_10.py +699 -0
  653. chia/simulator/ssl_certs_2.py +699 -0
  654. chia/simulator/ssl_certs_3.py +699 -0
  655. chia/simulator/ssl_certs_4.py +699 -0
  656. chia/simulator/ssl_certs_5.py +699 -0
  657. chia/simulator/ssl_certs_6.py +699 -0
  658. chia/simulator/ssl_certs_7.py +699 -0
  659. chia/simulator/ssl_certs_8.py +699 -0
  660. chia/simulator/ssl_certs_9.py +699 -0
  661. chia/simulator/start_simulator.py +135 -0
  662. chia/simulator/wallet_tools.py +245 -0
  663. chia/ssl/__init__.py +0 -0
  664. chia/ssl/chia_ca.crt +19 -0
  665. chia/ssl/chia_ca.key +28 -0
  666. chia/ssl/create_ssl.py +249 -0
  667. chia/ssl/dst_root_ca.pem +20 -0
  668. chia/timelord/__init__.py +0 -0
  669. chia/timelord/iters_from_block.py +50 -0
  670. chia/timelord/timelord.py +1202 -0
  671. chia/timelord/timelord_api.py +132 -0
  672. chia/timelord/timelord_launcher.py +188 -0
  673. chia/timelord/timelord_state.py +244 -0
  674. chia/timelord/types.py +22 -0
  675. chia/types/__init__.py +0 -0
  676. chia/types/aliases.py +35 -0
  677. chia/types/block_protocol.py +20 -0
  678. chia/types/blockchain_format/__init__.py +0 -0
  679. chia/types/blockchain_format/classgroup.py +5 -0
  680. chia/types/blockchain_format/coin.py +28 -0
  681. chia/types/blockchain_format/foliage.py +8 -0
  682. chia/types/blockchain_format/pool_target.py +5 -0
  683. chia/types/blockchain_format/program.py +270 -0
  684. chia/types/blockchain_format/proof_of_space.py +135 -0
  685. chia/types/blockchain_format/reward_chain_block.py +6 -0
  686. chia/types/blockchain_format/serialized_program.py +5 -0
  687. chia/types/blockchain_format/sized_bytes.py +11 -0
  688. chia/types/blockchain_format/slots.py +9 -0
  689. chia/types/blockchain_format/sub_epoch_summary.py +5 -0
  690. chia/types/blockchain_format/tree_hash.py +72 -0
  691. chia/types/blockchain_format/vdf.py +86 -0
  692. chia/types/clvm_cost.py +13 -0
  693. chia/types/coin_record.py +43 -0
  694. chia/types/coin_spend.py +115 -0
  695. chia/types/condition_opcodes.py +73 -0
  696. chia/types/condition_with_args.py +17 -0
  697. chia/types/eligible_coin_spends.py +364 -0
  698. chia/types/end_of_slot_bundle.py +5 -0
  699. chia/types/fee_rate.py +38 -0
  700. chia/types/full_block.py +5 -0
  701. chia/types/generator_types.py +14 -0
  702. chia/types/header_block.py +5 -0
  703. chia/types/internal_mempool_item.py +19 -0
  704. chia/types/mempool_inclusion_status.py +9 -0
  705. chia/types/mempool_item.py +85 -0
  706. chia/types/mempool_submission_status.py +30 -0
  707. chia/types/mojos.py +7 -0
  708. chia/types/peer_info.py +64 -0
  709. chia/types/signing_mode.py +29 -0
  710. chia/types/spend_bundle.py +31 -0
  711. chia/types/spend_bundle_conditions.py +7 -0
  712. chia/types/transaction_queue_entry.py +55 -0
  713. chia/types/unfinished_block.py +5 -0
  714. chia/types/unfinished_header_block.py +37 -0
  715. chia/types/weight_proof.py +50 -0
  716. chia/util/__init__.py +0 -0
  717. chia/util/action_scope.py +168 -0
  718. chia/util/api_decorators.py +89 -0
  719. chia/util/async_pool.py +224 -0
  720. chia/util/augmented_chain.py +130 -0
  721. chia/util/batches.py +39 -0
  722. chia/util/bech32m.py +123 -0
  723. chia/util/beta_metrics.py +118 -0
  724. chia/util/block_cache.py +56 -0
  725. chia/util/byte_types.py +10 -0
  726. chia/util/check_fork_next_block.py +32 -0
  727. chia/util/chia_logging.py +124 -0
  728. chia/util/chia_version.py +33 -0
  729. chia/util/collection.py +17 -0
  730. chia/util/condition_tools.py +201 -0
  731. chia/util/config.py +366 -0
  732. chia/util/cpu.py +20 -0
  733. chia/util/db_synchronous.py +21 -0
  734. chia/util/db_version.py +30 -0
  735. chia/util/db_wrapper.py +427 -0
  736. chia/util/default_root.py +10 -0
  737. chia/util/dump_keyring.py +93 -0
  738. chia/util/english.txt +2048 -0
  739. chia/util/errors.py +351 -0
  740. chia/util/file_keyring.py +480 -0
  741. chia/util/files.py +95 -0
  742. chia/util/full_block_utils.py +321 -0
  743. chia/util/generator_tools.py +62 -0
  744. chia/util/hash.py +29 -0
  745. chia/util/initial-config.yaml +675 -0
  746. chia/util/inline_executor.py +24 -0
  747. chia/util/ints.py +19 -0
  748. chia/util/json_util.py +41 -0
  749. chia/util/keychain.py +673 -0
  750. chia/util/keyring_wrapper.py +266 -0
  751. chia/util/limited_semaphore.py +39 -0
  752. chia/util/lock.py +47 -0
  753. chia/util/log_exceptions.py +29 -0
  754. chia/util/logging.py +34 -0
  755. chia/util/lru_cache.py +29 -0
  756. chia/util/math.py +20 -0
  757. chia/util/network.py +240 -0
  758. chia/util/paginator.py +46 -0
  759. chia/util/path.py +29 -0
  760. chia/util/permissions.py +19 -0
  761. chia/util/pprint.py +40 -0
  762. chia/util/prev_transaction_block.py +23 -0
  763. chia/util/priority_mutex.py +92 -0
  764. chia/util/profiler.py +194 -0
  765. chia/util/recursive_replace.py +22 -0
  766. chia/util/safe_cancel_task.py +14 -0
  767. chia/util/service_groups.py +47 -0
  768. chia/util/setproctitle.py +20 -0
  769. chia/util/significant_bits.py +30 -0
  770. chia/util/ssl_check.py +213 -0
  771. chia/util/streamable.py +654 -0
  772. chia/util/task_timing.py +378 -0
  773. chia/util/timing.py +64 -0
  774. chia/util/vdf_prover.py +31 -0
  775. chia/util/ws_message.py +66 -0
  776. chia/wallet/__init__.py +0 -0
  777. chia/wallet/cat_wallet/__init__.py +0 -0
  778. chia/wallet/cat_wallet/cat_constants.py +75 -0
  779. chia/wallet/cat_wallet/cat_info.py +47 -0
  780. chia/wallet/cat_wallet/cat_outer_puzzle.py +120 -0
  781. chia/wallet/cat_wallet/cat_utils.py +163 -0
  782. chia/wallet/cat_wallet/cat_wallet.py +869 -0
  783. chia/wallet/cat_wallet/dao_cat_info.py +28 -0
  784. chia/wallet/cat_wallet/dao_cat_wallet.py +669 -0
  785. chia/wallet/cat_wallet/lineage_store.py +74 -0
  786. chia/wallet/cat_wallet/puzzles/__init__.py +0 -0
  787. chia/wallet/cat_wallet/puzzles/cat_truths.clib +31 -0
  788. chia/wallet/cat_wallet/puzzles/cat_v2.clsp +397 -0
  789. chia/wallet/cat_wallet/puzzles/cat_v2.clsp.hex +1 -0
  790. chia/wallet/cat_wallet/puzzles/delegated_tail.clsp +25 -0
  791. chia/wallet/cat_wallet/puzzles/delegated_tail.clsp.hex +1 -0
  792. chia/wallet/cat_wallet/puzzles/everything_with_signature.clsp +15 -0
  793. chia/wallet/cat_wallet/puzzles/everything_with_signature.clsp.hex +1 -0
  794. chia/wallet/cat_wallet/puzzles/genesis_by_coin_id.clsp +26 -0
  795. chia/wallet/cat_wallet/puzzles/genesis_by_coin_id.clsp.hex +1 -0
  796. chia/wallet/cat_wallet/puzzles/genesis_by_coin_id_or_singleton.clsp +42 -0
  797. chia/wallet/cat_wallet/puzzles/genesis_by_coin_id_or_singleton.clsp.hex +1 -0
  798. chia/wallet/cat_wallet/puzzles/genesis_by_puzzle_hash.clsp +24 -0
  799. chia/wallet/cat_wallet/puzzles/genesis_by_puzzle_hash.clsp.hex +1 -0
  800. chia/wallet/coin_selection.py +188 -0
  801. chia/wallet/conditions.py +1326 -0
  802. chia/wallet/dao_wallet/__init__.py +0 -0
  803. chia/wallet/dao_wallet/dao_info.py +61 -0
  804. chia/wallet/dao_wallet/dao_utils.py +810 -0
  805. chia/wallet/dao_wallet/dao_wallet.py +2121 -0
  806. chia/wallet/db_wallet/__init__.py +0 -0
  807. chia/wallet/db_wallet/db_wallet_puzzles.py +107 -0
  808. chia/wallet/derivation_record.py +30 -0
  809. chia/wallet/derive_keys.py +146 -0
  810. chia/wallet/did_wallet/__init__.py +0 -0
  811. chia/wallet/did_wallet/did_info.py +39 -0
  812. chia/wallet/did_wallet/did_wallet.py +1485 -0
  813. chia/wallet/did_wallet/did_wallet_puzzles.py +220 -0
  814. chia/wallet/did_wallet/puzzles/__init__.py +0 -0
  815. chia/wallet/did_wallet/puzzles/did_innerpuz.clsp +135 -0
  816. chia/wallet/did_wallet/puzzles/did_innerpuz.clsp.hex +1 -0
  817. chia/wallet/driver_protocol.py +26 -0
  818. chia/wallet/key_val_store.py +55 -0
  819. chia/wallet/lineage_proof.py +58 -0
  820. chia/wallet/nft_wallet/__init__.py +0 -0
  821. chia/wallet/nft_wallet/metadata_outer_puzzle.py +92 -0
  822. chia/wallet/nft_wallet/nft_info.py +120 -0
  823. chia/wallet/nft_wallet/nft_puzzles.py +305 -0
  824. chia/wallet/nft_wallet/nft_wallet.py +1686 -0
  825. chia/wallet/nft_wallet/ownership_outer_puzzle.py +101 -0
  826. chia/wallet/nft_wallet/puzzles/__init__.py +0 -0
  827. chia/wallet/nft_wallet/puzzles/create_nft_launcher_from_did.clsp +6 -0
  828. chia/wallet/nft_wallet/puzzles/create_nft_launcher_from_did.clsp.hex +1 -0
  829. chia/wallet/nft_wallet/puzzles/nft_intermediate_launcher.clsp +6 -0
  830. chia/wallet/nft_wallet/puzzles/nft_intermediate_launcher.clsp.hex +1 -0
  831. chia/wallet/nft_wallet/puzzles/nft_metadata_updater_default.clsp +30 -0
  832. chia/wallet/nft_wallet/puzzles/nft_metadata_updater_default.clsp.hex +1 -0
  833. chia/wallet/nft_wallet/puzzles/nft_metadata_updater_updateable.clsp +28 -0
  834. chia/wallet/nft_wallet/puzzles/nft_metadata_updater_updateable.clsp.hex +1 -0
  835. chia/wallet/nft_wallet/puzzles/nft_ownership_layer.clsp +100 -0
  836. chia/wallet/nft_wallet/puzzles/nft_ownership_layer.clsp.hex +1 -0
  837. chia/wallet/nft_wallet/puzzles/nft_ownership_transfer_program_one_way_claim_with_royalties.clsp +78 -0
  838. chia/wallet/nft_wallet/puzzles/nft_ownership_transfer_program_one_way_claim_with_royalties.clsp.hex +1 -0
  839. chia/wallet/nft_wallet/puzzles/nft_state_layer.clsp +74 -0
  840. chia/wallet/nft_wallet/puzzles/nft_state_layer.clsp.hex +1 -0
  841. chia/wallet/nft_wallet/singleton_outer_puzzle.py +101 -0
  842. chia/wallet/nft_wallet/transfer_program_puzzle.py +82 -0
  843. chia/wallet/nft_wallet/uncurry_nft.py +217 -0
  844. chia/wallet/notification_manager.py +117 -0
  845. chia/wallet/notification_store.py +178 -0
  846. chia/wallet/outer_puzzles.py +84 -0
  847. chia/wallet/payment.py +34 -0
  848. chia/wallet/puzzle_drivers.py +118 -0
  849. chia/wallet/puzzles/__init__.py +0 -0
  850. chia/wallet/puzzles/augmented_condition.clsp +13 -0
  851. chia/wallet/puzzles/augmented_condition.clsp.hex +1 -0
  852. chia/wallet/puzzles/clawback/__init__.py +0 -0
  853. chia/wallet/puzzles/clawback/drivers.py +188 -0
  854. chia/wallet/puzzles/clawback/metadata.py +38 -0
  855. chia/wallet/puzzles/clawback/puzzle_decorator.py +67 -0
  856. chia/wallet/puzzles/condition_codes.clib +77 -0
  857. chia/wallet/puzzles/curry-and-treehash.clib +102 -0
  858. chia/wallet/puzzles/curry.clib +135 -0
  859. chia/wallet/puzzles/curry_by_index.clib +16 -0
  860. chia/wallet/puzzles/dao_cat_eve.clsp +17 -0
  861. chia/wallet/puzzles/dao_cat_eve.clsp.hex +1 -0
  862. chia/wallet/puzzles/dao_cat_launcher.clsp +36 -0
  863. chia/wallet/puzzles/dao_cat_launcher.clsp.hex +1 -0
  864. chia/wallet/puzzles/dao_finished_state.clsp +35 -0
  865. chia/wallet/puzzles/dao_finished_state.clsp.hex +1 -0
  866. chia/wallet/puzzles/dao_finished_state.clsp.hex.sha256tree +1 -0
  867. chia/wallet/puzzles/dao_lockup.clsp +288 -0
  868. chia/wallet/puzzles/dao_lockup.clsp.hex +1 -0
  869. chia/wallet/puzzles/dao_lockup.clsp.hex.sha256tree +1 -0
  870. chia/wallet/puzzles/dao_proposal.clsp +377 -0
  871. chia/wallet/puzzles/dao_proposal.clsp.hex +1 -0
  872. chia/wallet/puzzles/dao_proposal.clsp.hex.sha256tree +1 -0
  873. chia/wallet/puzzles/dao_proposal_timer.clsp +78 -0
  874. chia/wallet/puzzles/dao_proposal_timer.clsp.hex +1 -0
  875. chia/wallet/puzzles/dao_proposal_timer.clsp.hex.sha256tree +1 -0
  876. chia/wallet/puzzles/dao_proposal_validator.clsp +87 -0
  877. chia/wallet/puzzles/dao_proposal_validator.clsp.hex +1 -0
  878. chia/wallet/puzzles/dao_proposal_validator.clsp.hex.sha256tree +1 -0
  879. chia/wallet/puzzles/dao_spend_p2_singleton_v2.clsp +240 -0
  880. chia/wallet/puzzles/dao_spend_p2_singleton_v2.clsp.hex +1 -0
  881. chia/wallet/puzzles/dao_spend_p2_singleton_v2.clsp.hex.sha256tree +1 -0
  882. chia/wallet/puzzles/dao_treasury.clsp +115 -0
  883. chia/wallet/puzzles/dao_treasury.clsp.hex +1 -0
  884. chia/wallet/puzzles/dao_update_proposal.clsp +44 -0
  885. chia/wallet/puzzles/dao_update_proposal.clsp.hex +1 -0
  886. chia/wallet/puzzles/deployed_puzzle_hashes.json +67 -0
  887. chia/wallet/puzzles/json.clib +25 -0
  888. chia/wallet/puzzles/load_clvm.py +162 -0
  889. chia/wallet/puzzles/merkle_utils.clib +18 -0
  890. chia/wallet/puzzles/notification.clsp +7 -0
  891. chia/wallet/puzzles/notification.clsp.hex +1 -0
  892. chia/wallet/puzzles/p2_1_of_n.clsp +22 -0
  893. chia/wallet/puzzles/p2_1_of_n.clsp.hex +1 -0
  894. chia/wallet/puzzles/p2_conditions.clsp +3 -0
  895. chia/wallet/puzzles/p2_conditions.clsp.hex +1 -0
  896. chia/wallet/puzzles/p2_conditions.py +27 -0
  897. chia/wallet/puzzles/p2_delegated_conditions.clsp +18 -0
  898. chia/wallet/puzzles/p2_delegated_conditions.clsp.hex +1 -0
  899. chia/wallet/puzzles/p2_delegated_conditions.py +22 -0
  900. chia/wallet/puzzles/p2_delegated_puzzle.clsp +19 -0
  901. chia/wallet/puzzles/p2_delegated_puzzle.clsp.hex +1 -0
  902. chia/wallet/puzzles/p2_delegated_puzzle.py +35 -0
  903. chia/wallet/puzzles/p2_delegated_puzzle_or_hidden_puzzle.clsp +91 -0
  904. chia/wallet/puzzles/p2_delegated_puzzle_or_hidden_puzzle.clsp.hex +1 -0
  905. chia/wallet/puzzles/p2_delegated_puzzle_or_hidden_puzzle.py +161 -0
  906. chia/wallet/puzzles/p2_m_of_n_delegate_direct.clsp +108 -0
  907. chia/wallet/puzzles/p2_m_of_n_delegate_direct.clsp.hex +1 -0
  908. chia/wallet/puzzles/p2_m_of_n_delegate_direct.py +22 -0
  909. chia/wallet/puzzles/p2_parent.clsp +19 -0
  910. chia/wallet/puzzles/p2_parent.clsp.hex +1 -0
  911. chia/wallet/puzzles/p2_puzzle_hash.clsp +18 -0
  912. chia/wallet/puzzles/p2_puzzle_hash.clsp.hex +1 -0
  913. chia/wallet/puzzles/p2_puzzle_hash.py +28 -0
  914. chia/wallet/puzzles/p2_singleton.clsp +30 -0
  915. chia/wallet/puzzles/p2_singleton.clsp.hex +1 -0
  916. chia/wallet/puzzles/p2_singleton_aggregator.clsp +81 -0
  917. chia/wallet/puzzles/p2_singleton_aggregator.clsp.hex +1 -0
  918. chia/wallet/puzzles/p2_singleton_or_delayed_puzhash.clsp +50 -0
  919. chia/wallet/puzzles/p2_singleton_or_delayed_puzhash.clsp.hex +1 -0
  920. chia/wallet/puzzles/p2_singleton_via_delegated_puzzle.clsp +47 -0
  921. chia/wallet/puzzles/p2_singleton_via_delegated_puzzle.clsp.hex +1 -0
  922. chia/wallet/puzzles/puzzle_utils.py +34 -0
  923. chia/wallet/puzzles/settlement_payments.clsp +49 -0
  924. chia/wallet/puzzles/settlement_payments.clsp.hex +1 -0
  925. chia/wallet/puzzles/sha256tree.clib +11 -0
  926. chia/wallet/puzzles/singleton_launcher.clsp +16 -0
  927. chia/wallet/puzzles/singleton_launcher.clsp.hex +1 -0
  928. chia/wallet/puzzles/singleton_top_layer.clsp +177 -0
  929. chia/wallet/puzzles/singleton_top_layer.clsp.hex +1 -0
  930. chia/wallet/puzzles/singleton_top_layer.py +295 -0
  931. chia/wallet/puzzles/singleton_top_layer_v1_1.clsp +107 -0
  932. chia/wallet/puzzles/singleton_top_layer_v1_1.clsp.hex +1 -0
  933. chia/wallet/puzzles/singleton_top_layer_v1_1.py +344 -0
  934. chia/wallet/puzzles/singleton_truths.clib +21 -0
  935. chia/wallet/puzzles/tails.py +344 -0
  936. chia/wallet/puzzles/utility_macros.clib +48 -0
  937. chia/wallet/signer_protocol.py +126 -0
  938. chia/wallet/singleton.py +106 -0
  939. chia/wallet/singleton_record.py +30 -0
  940. chia/wallet/trade_manager.py +1088 -0
  941. chia/wallet/trade_record.py +67 -0
  942. chia/wallet/trading/__init__.py +0 -0
  943. chia/wallet/trading/offer.py +703 -0
  944. chia/wallet/trading/trade_status.py +13 -0
  945. chia/wallet/trading/trade_store.py +526 -0
  946. chia/wallet/transaction_record.py +143 -0
  947. chia/wallet/transaction_sorting.py +14 -0
  948. chia/wallet/uncurried_puzzle.py +17 -0
  949. chia/wallet/util/__init__.py +0 -0
  950. chia/wallet/util/address_type.py +55 -0
  951. chia/wallet/util/blind_signer_tl.py +168 -0
  952. chia/wallet/util/clvm_streamable.py +203 -0
  953. chia/wallet/util/compute_hints.py +66 -0
  954. chia/wallet/util/compute_memos.py +45 -0
  955. chia/wallet/util/curry_and_treehash.py +90 -0
  956. chia/wallet/util/debug_spend_bundle.py +234 -0
  957. chia/wallet/util/merkle_tree.py +100 -0
  958. chia/wallet/util/merkle_utils.py +102 -0
  959. chia/wallet/util/new_peak_queue.py +82 -0
  960. chia/wallet/util/notifications.py +12 -0
  961. chia/wallet/util/peer_request_cache.py +174 -0
  962. chia/wallet/util/puzzle_compression.py +96 -0
  963. chia/wallet/util/puzzle_decorator.py +100 -0
  964. chia/wallet/util/puzzle_decorator_type.py +7 -0
  965. chia/wallet/util/query_filter.py +60 -0
  966. chia/wallet/util/transaction_type.py +23 -0
  967. chia/wallet/util/tx_config.py +158 -0
  968. chia/wallet/util/wallet_sync_utils.py +348 -0
  969. chia/wallet/util/wallet_types.py +65 -0
  970. chia/wallet/vc_wallet/__init__.py +0 -0
  971. chia/wallet/vc_wallet/cr_cat_drivers.py +663 -0
  972. chia/wallet/vc_wallet/cr_cat_wallet.py +875 -0
  973. chia/wallet/vc_wallet/cr_outer_puzzle.py +102 -0
  974. chia/wallet/vc_wallet/cr_puzzles/__init__.py +0 -0
  975. chia/wallet/vc_wallet/cr_puzzles/conditions_w_fee_announce.clsp +3 -0
  976. chia/wallet/vc_wallet/cr_puzzles/conditions_w_fee_announce.clsp.hex +1 -0
  977. chia/wallet/vc_wallet/cr_puzzles/credential_restriction.clsp +304 -0
  978. chia/wallet/vc_wallet/cr_puzzles/credential_restriction.clsp.hex +1 -0
  979. chia/wallet/vc_wallet/cr_puzzles/flag_proofs_checker.clsp +45 -0
  980. chia/wallet/vc_wallet/cr_puzzles/flag_proofs_checker.clsp.hex +1 -0
  981. chia/wallet/vc_wallet/vc_drivers.py +838 -0
  982. chia/wallet/vc_wallet/vc_puzzles/__init__.py +0 -0
  983. chia/wallet/vc_wallet/vc_puzzles/covenant_layer.clsp +30 -0
  984. chia/wallet/vc_wallet/vc_puzzles/covenant_layer.clsp.hex +1 -0
  985. chia/wallet/vc_wallet/vc_puzzles/eml_covenant_morpher.clsp +75 -0
  986. chia/wallet/vc_wallet/vc_puzzles/eml_covenant_morpher.clsp.hex +1 -0
  987. chia/wallet/vc_wallet/vc_puzzles/eml_transfer_program_covenant_adapter.clsp +32 -0
  988. chia/wallet/vc_wallet/vc_puzzles/eml_transfer_program_covenant_adapter.clsp.hex +1 -0
  989. chia/wallet/vc_wallet/vc_puzzles/eml_update_metadata_with_DID.clsp +80 -0
  990. chia/wallet/vc_wallet/vc_puzzles/eml_update_metadata_with_DID.clsp.hex +1 -0
  991. chia/wallet/vc_wallet/vc_puzzles/exigent_metadata_layer.clsp +163 -0
  992. chia/wallet/vc_wallet/vc_puzzles/exigent_metadata_layer.clsp.hex +1 -0
  993. chia/wallet/vc_wallet/vc_puzzles/p2_announced_delegated_puzzle.clsp +16 -0
  994. chia/wallet/vc_wallet/vc_puzzles/p2_announced_delegated_puzzle.clsp.hex +1 -0
  995. chia/wallet/vc_wallet/vc_puzzles/standard_vc_backdoor_puzzle.clsp +74 -0
  996. chia/wallet/vc_wallet/vc_puzzles/standard_vc_backdoor_puzzle.clsp.hex +1 -0
  997. chia/wallet/vc_wallet/vc_puzzles/std_parent_morpher.clsp +23 -0
  998. chia/wallet/vc_wallet/vc_puzzles/std_parent_morpher.clsp.hex +1 -0
  999. chia/wallet/vc_wallet/vc_puzzles/viral_backdoor.clsp +64 -0
  1000. chia/wallet/vc_wallet/vc_puzzles/viral_backdoor.clsp.hex +1 -0
  1001. chia/wallet/vc_wallet/vc_store.py +263 -0
  1002. chia/wallet/vc_wallet/vc_wallet.py +638 -0
  1003. chia/wallet/wallet.py +698 -0
  1004. chia/wallet/wallet_action_scope.py +95 -0
  1005. chia/wallet/wallet_blockchain.py +244 -0
  1006. chia/wallet/wallet_coin_record.py +72 -0
  1007. chia/wallet/wallet_coin_store.py +351 -0
  1008. chia/wallet/wallet_info.py +36 -0
  1009. chia/wallet/wallet_interested_store.py +188 -0
  1010. chia/wallet/wallet_nft_store.py +279 -0
  1011. chia/wallet/wallet_node.py +1769 -0
  1012. chia/wallet/wallet_node_api.py +201 -0
  1013. chia/wallet/wallet_pool_store.py +120 -0
  1014. chia/wallet/wallet_protocol.py +90 -0
  1015. chia/wallet/wallet_puzzle_store.py +365 -0
  1016. chia/wallet/wallet_retry_store.py +70 -0
  1017. chia/wallet/wallet_singleton_store.py +258 -0
  1018. chia/wallet/wallet_spend_bundle.py +41 -0
  1019. chia/wallet/wallet_state_manager.py +2820 -0
  1020. chia/wallet/wallet_transaction_store.py +470 -0
  1021. chia/wallet/wallet_user_store.py +110 -0
  1022. chia/wallet/wallet_weight_proof_handler.py +126 -0
  1023. chia_blockchain-2.4.4.dist-info/LICENSE +201 -0
  1024. chia_blockchain-2.4.4.dist-info/METADATA +161 -0
  1025. chia_blockchain-2.4.4.dist-info/RECORD +1028 -0
  1026. chia_blockchain-2.4.4.dist-info/WHEEL +4 -0
  1027. chia_blockchain-2.4.4.dist-info/entry_points.txt +17 -0
  1028. mozilla-ca/cacert.pem +3666 -0
@@ -0,0 +1,1303 @@
1
+ from __future__ import annotations
2
+
3
+ import asyncio
4
+ import contextlib
5
+ import dataclasses
6
+ import functools
7
+ import json
8
+ import logging
9
+ import os
10
+ import random
11
+ import time
12
+ import traceback
13
+ from pathlib import Path
14
+ from typing import (
15
+ TYPE_CHECKING,
16
+ Any,
17
+ AsyncIterator,
18
+ Awaitable,
19
+ ClassVar,
20
+ Dict,
21
+ List,
22
+ Optional,
23
+ Set,
24
+ Tuple,
25
+ Union,
26
+ cast,
27
+ final,
28
+ )
29
+
30
+ import aiohttp
31
+
32
+ from chia.data_layer.data_layer_errors import KeyNotFoundError
33
+ from chia.data_layer.data_layer_util import (
34
+ DiffData,
35
+ InternalNode,
36
+ KeysPaginationData,
37
+ KeysValuesPaginationData,
38
+ KeyValue,
39
+ KVDiffPaginationData,
40
+ Layer,
41
+ Offer,
42
+ OfferStore,
43
+ PluginRemote,
44
+ PluginStatus,
45
+ Proof,
46
+ ProofOfInclusion,
47
+ ProofOfInclusionLayer,
48
+ Root,
49
+ ServerInfo,
50
+ Status,
51
+ StoreProofs,
52
+ Subscription,
53
+ SyncStatus,
54
+ TerminalNode,
55
+ Unspecified,
56
+ UnsubscribeData,
57
+ leaf_hash,
58
+ unspecified,
59
+ )
60
+ from chia.data_layer.data_layer_wallet import DataLayerWallet, Mirror, SingletonRecord, verify_offer
61
+ from chia.data_layer.data_store import DataStore
62
+ from chia.data_layer.download_data import (
63
+ delete_full_file_if_exists,
64
+ get_delta_filename_path,
65
+ get_full_tree_filename_path,
66
+ insert_from_delta_file,
67
+ write_files_for_root,
68
+ )
69
+ from chia.rpc.rpc_server import StateChangedProtocol, default_get_connections
70
+ from chia.rpc.wallet_request_types import LogIn
71
+ from chia.rpc.wallet_rpc_client import WalletRpcClient
72
+ from chia.server.outbound_message import NodeType
73
+ from chia.server.server import ChiaServer
74
+ from chia.server.ws_connection import WSChiaConnection
75
+ from chia.types.blockchain_format.sized_bytes import bytes32
76
+ from chia.util.async_pool import Job, QueuedAsyncPool
77
+ from chia.util.ints import uint32, uint64
78
+ from chia.util.path import path_from_root
79
+ from chia.wallet.trade_record import TradeRecord
80
+ from chia.wallet.trading.offer import Offer as TradingOffer
81
+ from chia.wallet.transaction_record import TransactionRecord
82
+ from chia.wallet.util.tx_config import DEFAULT_TX_CONFIG
83
+
84
+
85
+ async def get_plugin_info(plugin_remote: PluginRemote) -> Tuple[PluginRemote, Dict[str, Any]]:
86
+ try:
87
+ async with aiohttp.ClientSession() as session:
88
+ async with session.post(
89
+ plugin_remote.url + "/plugin_info",
90
+ json={},
91
+ headers=plugin_remote.headers,
92
+ ) as response:
93
+ ret = {"status": response.status}
94
+ if response.status == 200:
95
+ ret["response"] = json.loads(await response.text())
96
+ return plugin_remote, ret
97
+ except aiohttp.ClientError as e:
98
+ return plugin_remote, {"error": f"ClientError: {e}"}
99
+
100
+
101
+ @final
102
+ @dataclasses.dataclass
103
+ class DataLayer:
104
+ if TYPE_CHECKING:
105
+ from chia.rpc.rpc_server import RpcServiceProtocol
106
+
107
+ _protocol_check: ClassVar[RpcServiceProtocol] = cast("DataLayer", None)
108
+
109
+ db_path: Path
110
+ config: Dict[str, Any]
111
+ root_path: Path
112
+ log: logging.Logger
113
+ wallet_rpc_init: Awaitable[WalletRpcClient]
114
+ downloaders: List[PluginRemote]
115
+ uploaders: List[PluginRemote]
116
+ maximum_full_file_count: int
117
+ server_files_location: Path
118
+ unsubscribe_data_queue: List[UnsubscribeData]
119
+ _server: Optional[ChiaServer] = None
120
+ none_bytes: bytes32 = bytes32([0] * 32)
121
+ initialized: bool = False
122
+ _data_store: Optional[DataStore] = None
123
+ state_changed_callback: Optional[StateChangedProtocol] = None
124
+ _shut_down: bool = False
125
+ periodically_manage_data_task: Optional[asyncio.Task[None]] = None
126
+ _wallet_rpc: Optional[WalletRpcClient] = None
127
+ subscription_lock: asyncio.Lock = dataclasses.field(default_factory=asyncio.Lock)
128
+ subscription_update_concurrency: int = 5
129
+ client_timeout: aiohttp.ClientTimeout = dataclasses.field(
130
+ default_factory=functools.partial(aiohttp.ClientTimeout, total=45, sock_connect=5)
131
+ )
132
+ group_files_by_store: bool = False
133
+
134
+ @property
135
+ def server(self) -> ChiaServer:
136
+ # This is a stop gap until the class usage is refactored such the values of
137
+ # integral attributes are known at creation of the instance.
138
+ if self._server is None:
139
+ raise RuntimeError("server not assigned")
140
+
141
+ return self._server
142
+
143
+ @property
144
+ def data_store(self) -> DataStore:
145
+ # This is a stop gap until the class usage is refactored such the values of
146
+ # integral attributes are known at creation of the instance.
147
+ if self._data_store is None:
148
+ raise RuntimeError("data_store not assigned")
149
+
150
+ return self._data_store
151
+
152
+ @property
153
+ def wallet_rpc(self) -> WalletRpcClient:
154
+ # This is a stop gap until the class usage is refactored such the values of
155
+ # integral attributes are known at creation of the instance.
156
+ if self._wallet_rpc is None:
157
+ raise RuntimeError("wallet_rpc not assigned")
158
+
159
+ return self._wallet_rpc
160
+
161
+ @classmethod
162
+ def create(
163
+ cls,
164
+ config: Dict[str, Any],
165
+ root_path: Path,
166
+ wallet_rpc_init: Awaitable[WalletRpcClient],
167
+ downloaders: List[PluginRemote],
168
+ uploaders: List[PluginRemote], # dont add FilesystemUploader to this, it is the default uploader
169
+ name: Optional[str] = None,
170
+ ) -> DataLayer:
171
+ if name == "":
172
+ # TODO: If no code depends on "" counting as 'unspecified' then we do not
173
+ # need this.
174
+ name = None
175
+
176
+ server_files_replaced: str = config.get(
177
+ "server_files_location", "data_layer/db/server_files_location_CHALLENGE"
178
+ ).replace("CHALLENGE", config["selected_network"])
179
+
180
+ db_path_replaced: str = config["database_path"].replace("CHALLENGE", config["selected_network"])
181
+
182
+ self = cls(
183
+ config=config,
184
+ root_path=root_path,
185
+ wallet_rpc_init=wallet_rpc_init,
186
+ log=logging.getLogger(name if name is None else __name__),
187
+ db_path=path_from_root(root_path, db_path_replaced),
188
+ server_files_location=path_from_root(root_path, server_files_replaced),
189
+ downloaders=downloaders,
190
+ uploaders=uploaders,
191
+ maximum_full_file_count=config.get("maximum_full_file_count", 1),
192
+ subscription_update_concurrency=config.get("subscription_update_concurrency", 5),
193
+ unsubscribe_data_queue=[],
194
+ client_timeout=aiohttp.ClientTimeout(
195
+ total=config.get("client_timeout", 45), sock_connect=config.get("connect_timeout", 5)
196
+ ),
197
+ group_files_by_store=config.get("group_files_by_store", False),
198
+ )
199
+
200
+ self.db_path.parent.mkdir(parents=True, exist_ok=True)
201
+ self.server_files_location.mkdir(parents=True, exist_ok=True)
202
+
203
+ return self
204
+
205
+ @contextlib.asynccontextmanager
206
+ async def manage(self) -> AsyncIterator[None]:
207
+ sql_log_path: Optional[Path] = None
208
+ if self.config.get("log_sqlite_cmds", False):
209
+ sql_log_path = path_from_root(self.root_path, "log/data_sql.log")
210
+ self.log.info(f"logging SQL commands to {sql_log_path}")
211
+
212
+ async with DataStore.managed(database=self.db_path, sql_log_path=sql_log_path) as self._data_store:
213
+ self._wallet_rpc = await self.wallet_rpc_init
214
+
215
+ await self._data_store.migrate_db()
216
+ self.periodically_manage_data_task = asyncio.create_task(self.periodically_manage_data())
217
+ try:
218
+ yield
219
+ finally:
220
+ # TODO: review for anything else we need to do here
221
+ self._shut_down = True
222
+ if self._wallet_rpc is not None:
223
+ self.wallet_rpc.close()
224
+
225
+ if self.periodically_manage_data_task is not None:
226
+ try:
227
+ self.periodically_manage_data_task.cancel()
228
+ except asyncio.CancelledError:
229
+ pass
230
+ if self._wallet_rpc is not None:
231
+ await self.wallet_rpc.await_closed()
232
+
233
+ def _set_state_changed_callback(self, callback: StateChangedProtocol) -> None:
234
+ self.state_changed_callback = callback
235
+
236
+ async def on_connect(self, connection: WSChiaConnection) -> None:
237
+ pass
238
+
239
+ def get_connections(self, request_node_type: Optional[NodeType]) -> List[Dict[str, Any]]:
240
+ return default_get_connections(server=self.server, request_node_type=request_node_type)
241
+
242
+ def set_server(self, server: ChiaServer) -> None:
243
+ self._server = server
244
+
245
+ async def wallet_log_in(self, fingerprint: int) -> int:
246
+ try:
247
+ result = await self.wallet_rpc.log_in(LogIn(uint32(fingerprint)))
248
+ except ValueError as e:
249
+ raise Exception(f"DataLayer wallet RPC log in request failed: {e.args[0]}")
250
+
251
+ return result.fingerprint
252
+
253
+ async def create_store(
254
+ self, fee: uint64, root: bytes32 = bytes32([0] * 32)
255
+ ) -> Tuple[List[TransactionRecord], bytes32]:
256
+ txs, store_id = await self.wallet_rpc.create_new_dl(root, fee)
257
+ res = await self.data_store.create_tree(store_id=store_id)
258
+ if res is None:
259
+ self.log.fatal("failed creating store")
260
+ self.initialized = True
261
+ return txs, store_id
262
+
263
+ async def batch_update(
264
+ self,
265
+ store_id: bytes32,
266
+ changelist: List[Dict[str, Any]],
267
+ fee: uint64,
268
+ submit_on_chain: bool = True,
269
+ ) -> Optional[TransactionRecord]:
270
+ status = Status.PENDING if submit_on_chain else Status.PENDING_BATCH
271
+ await self.batch_insert(store_id=store_id, changelist=changelist, status=status)
272
+ await self.data_store.clean_node_table()
273
+
274
+ if submit_on_chain:
275
+ return await self.publish_update(store_id=store_id, fee=fee)
276
+ else:
277
+ return None
278
+
279
+ async def _get_publishable_root_hash(self, store_id: bytes32) -> bytes32:
280
+ pending_root: Optional[Root] = await self.data_store.get_pending_root(store_id=store_id)
281
+ if pending_root is None:
282
+ raise Exception("Latest root is already confirmed.")
283
+ if pending_root.status == Status.PENDING_BATCH:
284
+ raise Exception("Unable to publish on chain, batch update set still open.")
285
+
286
+ return self.none_bytes if pending_root.node_hash is None else pending_root.node_hash
287
+
288
+ async def multistore_batch_update(
289
+ self,
290
+ store_updates: List[Dict[str, Any]],
291
+ fee: uint64,
292
+ submit_on_chain: bool = True,
293
+ ) -> List[TransactionRecord]:
294
+ store_ids: Set[bytes32] = set()
295
+ for update in store_updates:
296
+ store_id = update["store_id"]
297
+ changelist = update["changelist"]
298
+
299
+ if store_id in store_ids:
300
+ raise Exception(f"Store id {store_id.hex()} must appear in a single update")
301
+ store_ids.add(store_id)
302
+
303
+ status = Status.PENDING if submit_on_chain else Status.PENDING_BATCH
304
+ await self.batch_insert(store_id=store_id, changelist=changelist, status=status)
305
+
306
+ await self.data_store.clean_node_table()
307
+
308
+ if submit_on_chain:
309
+ update_dictionary: Dict[bytes32, bytes32] = {}
310
+ for store_id in store_ids:
311
+ await self._update_confirmation_status(store_id=store_id)
312
+ root_hash = await self._get_publishable_root_hash(store_id=store_id)
313
+ update_dictionary[store_id] = root_hash
314
+ transaction_records = await self.wallet_rpc.dl_update_multiple(update_dictionary=update_dictionary, fee=fee)
315
+ return transaction_records
316
+ else:
317
+ return []
318
+
319
+ async def submit_pending_root(
320
+ self,
321
+ store_id: bytes32,
322
+ fee: uint64,
323
+ ) -> TransactionRecord:
324
+ await self._update_confirmation_status(store_id=store_id)
325
+
326
+ pending_root: Optional[Root] = await self.data_store.get_pending_root(store_id=store_id)
327
+ if pending_root is None:
328
+ raise Exception("Latest root is already confirmed.")
329
+ if pending_root.status == Status.PENDING:
330
+ raise Exception("Pending root is already submitted.")
331
+
332
+ await self.data_store.change_root_status(pending_root, Status.PENDING)
333
+ return await self.publish_update(store_id, fee)
334
+
335
+ async def submit_all_pending_roots(self, fee: uint64) -> List[TransactionRecord]:
336
+ pending_roots = await self.data_store.get_all_pending_batches_roots()
337
+ update_dictionary: Dict[bytes32, bytes32] = {}
338
+ if len(pending_roots) == 0:
339
+ raise Exception("No pending roots found to submit")
340
+ for pending_root in pending_roots:
341
+ root_hash = pending_root.node_hash if pending_root.node_hash is not None else self.none_bytes
342
+ update_dictionary[pending_root.store_id] = root_hash
343
+ await self.data_store.change_root_status(pending_root, Status.PENDING)
344
+ transaction_records = await self.wallet_rpc.dl_update_multiple(update_dictionary=update_dictionary, fee=fee)
345
+ return transaction_records
346
+
347
+ async def batch_insert(
348
+ self,
349
+ store_id: bytes32,
350
+ changelist: List[Dict[str, Any]],
351
+ status: Status = Status.PENDING,
352
+ enable_batch_autoinsert: Optional[bool] = None,
353
+ ) -> bytes32:
354
+ await self._update_confirmation_status(store_id=store_id)
355
+
356
+ async with self.data_store.transaction():
357
+ pending_root: Optional[Root] = await self.data_store.get_pending_root(store_id=store_id)
358
+ if pending_root is not None and pending_root.status == Status.PENDING:
359
+ raise Exception("Already have a pending root waiting for confirmation.")
360
+
361
+ # check before any DL changes that this singleton is currently owned by this wallet
362
+ singleton_records: List[SingletonRecord] = await self.get_owned_stores()
363
+ if not any(store_id == singleton.launcher_id for singleton in singleton_records):
364
+ raise ValueError(f"Singleton with launcher ID {store_id} is not owned by DL Wallet")
365
+
366
+ t1 = time.monotonic()
367
+ if enable_batch_autoinsert is None:
368
+ enable_batch_autoinsert = self.config.get("enable_batch_autoinsert", True)
369
+ batch_hash = await self.data_store.insert_batch(store_id, changelist, status, enable_batch_autoinsert)
370
+ t2 = time.monotonic()
371
+ self.log.info(f"Data store batch update process time: {t2 - t1}.")
372
+ # todo return empty node hash from get_tree_root
373
+ if batch_hash is not None:
374
+ node_hash = batch_hash
375
+ else:
376
+ node_hash = self.none_bytes # todo change
377
+
378
+ return node_hash
379
+
380
+ async def publish_update(
381
+ self,
382
+ store_id: bytes32,
383
+ fee: uint64,
384
+ ) -> TransactionRecord:
385
+ await self._update_confirmation_status(store_id=store_id)
386
+ root_hash = await self._get_publishable_root_hash(store_id=store_id)
387
+ transaction_record = await self.wallet_rpc.dl_update_root(
388
+ launcher_id=store_id,
389
+ new_root=root_hash,
390
+ fee=fee,
391
+ )
392
+ return transaction_record
393
+
394
+ async def get_key_value_hash(
395
+ self,
396
+ store_id: bytes32,
397
+ key: bytes,
398
+ root_hash: Union[bytes32, Unspecified] = unspecified,
399
+ ) -> bytes32:
400
+ await self._update_confirmation_status(store_id=store_id)
401
+
402
+ async with self.data_store.transaction():
403
+ node = await self.data_store.get_node_by_key(store_id=store_id, key=key, root_hash=root_hash)
404
+ return node.hash
405
+
406
+ async def get_value(
407
+ self, store_id: bytes32, key: bytes, root_hash: Union[bytes32, Unspecified] = unspecified
408
+ ) -> bytes:
409
+ await self._update_confirmation_status(store_id=store_id)
410
+
411
+ async with self.data_store.transaction():
412
+ # this either returns the node or raises an exception
413
+ res = await self.data_store.get_node_by_key(store_id=store_id, key=key, root_hash=root_hash)
414
+ return res.value
415
+
416
+ async def get_keys_values(
417
+ self,
418
+ store_id: bytes32,
419
+ root_hash: Union[bytes32, Unspecified],
420
+ ) -> List[TerminalNode]:
421
+ await self._update_confirmation_status(store_id=store_id)
422
+
423
+ res = await self.data_store.get_keys_values(store_id, root_hash)
424
+ if res is None:
425
+ self.log.error("Failed to fetch keys values")
426
+ return res
427
+
428
+ async def get_keys_values_paginated(
429
+ self,
430
+ store_id: bytes32,
431
+ root_hash: Union[bytes32, Unspecified],
432
+ page: int,
433
+ max_page_size: Optional[int] = None,
434
+ ) -> KeysValuesPaginationData:
435
+ await self._update_confirmation_status(store_id=store_id)
436
+
437
+ if max_page_size is None:
438
+ max_page_size = 40 * 1024 * 1024
439
+ res = await self.data_store.get_keys_values_paginated(store_id, page, max_page_size, root_hash)
440
+ return res
441
+
442
+ async def get_keys(self, store_id: bytes32, root_hash: Union[bytes32, Unspecified]) -> List[bytes]:
443
+ await self._update_confirmation_status(store_id=store_id)
444
+
445
+ res = await self.data_store.get_keys(store_id, root_hash)
446
+ return res
447
+
448
+ async def get_keys_paginated(
449
+ self,
450
+ store_id: bytes32,
451
+ root_hash: Union[bytes32, Unspecified],
452
+ page: int,
453
+ max_page_size: Optional[int] = None,
454
+ ) -> KeysPaginationData:
455
+ await self._update_confirmation_status(store_id=store_id)
456
+
457
+ if max_page_size is None:
458
+ max_page_size = 40 * 1024 * 1024
459
+ res = await self.data_store.get_keys_paginated(store_id, page, max_page_size, root_hash)
460
+ return res
461
+
462
+ async def get_ancestors(self, node_hash: bytes32, store_id: bytes32) -> List[InternalNode]:
463
+ await self._update_confirmation_status(store_id=store_id)
464
+
465
+ res = await self.data_store.get_ancestors(node_hash=node_hash, store_id=store_id)
466
+ if res is None:
467
+ self.log.error("Failed to get ancestors")
468
+ return res
469
+
470
+ async def get_root(self, store_id: bytes32) -> Optional[SingletonRecord]:
471
+ latest = await self.wallet_rpc.dl_latest_singleton(store_id, True)
472
+ if latest is None:
473
+ self.log.error(f"Failed to get root for {store_id.hex()}")
474
+ return latest
475
+
476
+ async def get_local_root(self, store_id: bytes32) -> Optional[bytes32]:
477
+ await self._update_confirmation_status(store_id=store_id)
478
+
479
+ res = await self.data_store.get_tree_root(store_id=store_id)
480
+ if res is None:
481
+ self.log.error(f"Failed to get root for {store_id.hex()}")
482
+ return None
483
+ return res.node_hash
484
+
485
+ async def get_root_history(self, store_id: bytes32) -> List[SingletonRecord]:
486
+ records = await self.wallet_rpc.dl_history(store_id)
487
+ if records is None:
488
+ self.log.error(f"Failed to get root history for {store_id.hex()}")
489
+ root_history = []
490
+ prev: Optional[SingletonRecord] = None
491
+ for record in records:
492
+ if prev is None or record.root != prev.root:
493
+ root_history.append(record)
494
+ prev = record
495
+ return root_history
496
+
497
+ async def _update_confirmation_status(self, store_id: bytes32) -> None:
498
+ async with self.data_store.transaction():
499
+ try:
500
+ root = await self.data_store.get_tree_root(store_id=store_id)
501
+ except Exception:
502
+ root = None
503
+ singleton_record: Optional[SingletonRecord] = await self.wallet_rpc.dl_latest_singleton(store_id, True)
504
+ if singleton_record is None:
505
+ return
506
+ if root is None:
507
+ pending_root = await self.data_store.get_pending_root(store_id=store_id)
508
+ if pending_root is not None and pending_root.status == Status.PENDING:
509
+ if pending_root.generation == 0 and pending_root.node_hash is None:
510
+ await self.data_store.change_root_status(pending_root, Status.COMMITTED)
511
+ await self.data_store.clear_pending_roots(store_id=store_id)
512
+ return
513
+ else:
514
+ root = None
515
+ if root is None:
516
+ self.log.info(f"Don't have pending root for {store_id}.")
517
+ return
518
+ if root.generation == singleton_record.generation:
519
+ return
520
+ if root.generation > singleton_record.generation:
521
+ self.log.info(
522
+ f"Local root ahead of chain root: {root.generation} {singleton_record.generation}. "
523
+ "Maybe we're doing a batch update."
524
+ )
525
+ return
526
+ wallet_history = await self.wallet_rpc.dl_history(
527
+ launcher_id=store_id,
528
+ min_generation=uint32(root.generation + 1),
529
+ max_generation=singleton_record.generation,
530
+ )
531
+ new_hashes = [record.root for record in reversed(wallet_history)]
532
+ root_hash = self.none_bytes if root.node_hash is None else root.node_hash
533
+ generation_shift = 0
534
+ while len(new_hashes) > 0 and new_hashes[0] == root_hash:
535
+ generation_shift += 1
536
+ new_hashes.pop(0)
537
+ if generation_shift > 0:
538
+ await self.data_store.clear_pending_roots(store_id=store_id)
539
+ await self.data_store.shift_root_generations(store_id=store_id, shift_size=generation_shift)
540
+ else:
541
+ expected_root_hash = None if new_hashes[0] == self.none_bytes else new_hashes[0]
542
+ pending_root = await self.data_store.get_pending_root(store_id=store_id)
543
+ if (
544
+ pending_root is not None
545
+ and pending_root.generation == root.generation + 1
546
+ and pending_root.node_hash == expected_root_hash
547
+ and pending_root.status == Status.PENDING
548
+ ):
549
+ await self.data_store.change_root_status(pending_root, Status.COMMITTED)
550
+ await self.data_store.build_ancestor_table_for_latest_root(store_id=store_id)
551
+ await self.data_store.clear_pending_roots(store_id=store_id)
552
+
553
+ async def fetch_and_validate(self, store_id: bytes32) -> None:
554
+ singleton_record: Optional[SingletonRecord] = await self.wallet_rpc.dl_latest_singleton(store_id, True)
555
+ if singleton_record is None:
556
+ self.log.info(f"Fetch data: No singleton record for {store_id}.")
557
+ return
558
+ if singleton_record.generation == uint32(0):
559
+ self.log.info(f"Fetch data: No data on chain for {store_id}.")
560
+ return
561
+
562
+ await self._update_confirmation_status(store_id=store_id)
563
+
564
+ if not await self.data_store.store_id_exists(store_id=store_id):
565
+ await self.data_store.create_tree(store_id=store_id, status=Status.COMMITTED)
566
+
567
+ timestamp = int(time.time())
568
+ servers_info = await self.data_store.get_available_servers_for_store(store_id, timestamp)
569
+ # TODO: maybe append a random object to the whole DataLayer class?
570
+ random.shuffle(servers_info)
571
+ success = False
572
+ for server_info in servers_info:
573
+ url = server_info.url
574
+
575
+ root = await self.data_store.get_tree_root(store_id=store_id)
576
+ if root.generation > singleton_record.generation:
577
+ self.log.info(
578
+ "Fetch data: local DL store is ahead of chain generation. "
579
+ f"Local root: {root}. Singleton: {singleton_record}"
580
+ )
581
+ break
582
+ if root.generation == singleton_record.generation:
583
+ self.log.info(f"Fetch data: wallet generation matching on-chain generation: {store_id}.")
584
+ break
585
+
586
+ self.log.info(
587
+ f"Downloading files {store_id}. "
588
+ f"Current wallet generation: {root.generation}. "
589
+ f"Target wallet generation: {singleton_record.generation}. "
590
+ f"Server used: {url}."
591
+ )
592
+
593
+ to_download = await self.wallet_rpc.dl_history(
594
+ launcher_id=store_id,
595
+ min_generation=uint32(root.generation + 1),
596
+ max_generation=singleton_record.generation,
597
+ )
598
+ try:
599
+ proxy_url = self.config.get("proxy_url", None)
600
+ success = await insert_from_delta_file(
601
+ self.data_store,
602
+ store_id,
603
+ root.generation,
604
+ target_generation=singleton_record.generation,
605
+ root_hashes=[record.root for record in reversed(to_download)],
606
+ server_info=server_info,
607
+ client_foldername=self.server_files_location,
608
+ timeout=self.client_timeout,
609
+ log=self.log,
610
+ proxy_url=proxy_url,
611
+ downloader=await self.get_downloader(store_id, url),
612
+ group_files_by_store=self.group_files_by_store,
613
+ maximum_full_file_count=self.maximum_full_file_count,
614
+ )
615
+ if success:
616
+ self.log.info(
617
+ f"Finished downloading and validating {store_id}. "
618
+ f"Wallet generation saved: {singleton_record.generation}. "
619
+ f"Root hash saved: {singleton_record.root}."
620
+ )
621
+ break
622
+ except aiohttp.client_exceptions.ClientConnectorError:
623
+ self.log.warning(f"Server {url} unavailable for {store_id}.")
624
+ except Exception as e:
625
+ self.log.warning(f"Exception while downloading files for {store_id}: {e} {traceback.format_exc()}.")
626
+
627
+ # if there aren't any servers then don't try to write the full tree
628
+ if not success and len(servers_info) > 0:
629
+ root = await self.data_store.get_tree_root(store_id=store_id)
630
+ if root.node_hash is None:
631
+ return
632
+ filename_full_tree = get_full_tree_filename_path(
633
+ foldername=self.server_files_location,
634
+ store_id=store_id,
635
+ node_hash=root.node_hash,
636
+ generation=root.generation,
637
+ group_by_store=self.group_files_by_store,
638
+ )
639
+ # Had trouble with this generation, so generate full file for the generation we currently have
640
+ if not os.path.exists(filename_full_tree):
641
+ with open(filename_full_tree, "wb") as writer:
642
+ await self.data_store.write_tree_to_file(
643
+ root=root,
644
+ node_hash=root.node_hash,
645
+ store_id=store_id,
646
+ deltas_only=False,
647
+ writer=writer,
648
+ )
649
+ self.log.info(f"Successfully written full tree filename {filename_full_tree}.")
650
+
651
+ async def get_downloader(self, store_id: bytes32, url: str) -> Optional[PluginRemote]:
652
+ request_json = {"store_id": store_id.hex(), "url": url}
653
+ for d in self.downloaders:
654
+ async with aiohttp.ClientSession() as session:
655
+ try:
656
+ async with session.post(
657
+ d.url + "/handle_download",
658
+ json=request_json,
659
+ headers=d.headers,
660
+ ) as response:
661
+ res_json = await response.json()
662
+ if res_json["handle_download"]:
663
+ return d
664
+ except Exception as e:
665
+ self.log.error(f"get_downloader could not get response: {type(e).__name__}: {e}")
666
+ return None
667
+
668
+ async def clean_old_full_tree_files(self, store_id: bytes32) -> None:
669
+ singleton_record: Optional[SingletonRecord] = await self.wallet_rpc.dl_latest_singleton(store_id, True)
670
+ if singleton_record is None:
671
+ return
672
+ await self._update_confirmation_status(store_id=store_id)
673
+
674
+ root = await self.data_store.get_tree_root(store_id=store_id)
675
+ latest_generation = root.generation
676
+ full_tree_first_publish_generation = max(0, latest_generation - self.maximum_full_file_count + 1)
677
+ foldername = self.server_files_location
678
+
679
+ for generation in range(full_tree_first_publish_generation - 1, 0, -1):
680
+ root = await self.data_store.get_tree_root(store_id=store_id, generation=generation)
681
+ file_exists = delete_full_file_if_exists(foldername, store_id, root)
682
+ if not file_exists:
683
+ break
684
+
685
+ async def upload_files(self, store_id: bytes32) -> None:
686
+ uploaders = await self.get_uploaders(store_id)
687
+ singleton_record: Optional[SingletonRecord] = await self.wallet_rpc.dl_latest_singleton(store_id, True)
688
+ if singleton_record is None:
689
+ self.log.info(f"Upload files: no on-chain record for {store_id}.")
690
+ return
691
+ await self._update_confirmation_status(store_id=store_id)
692
+
693
+ root = await self.data_store.get_tree_root(store_id=store_id)
694
+ latest_generation = root.generation
695
+ # Don't store full tree files before this generation.
696
+ full_tree_first_publish_generation = max(0, latest_generation - self.maximum_full_file_count + 1)
697
+ publish_generation = min(singleton_record.generation, 0 if root is None else root.generation)
698
+ # If we make some batch updates, which get confirmed to the chain, we need to create the files.
699
+ # We iterate back and write the missing files, until we find the files already written.
700
+ root = await self.data_store.get_tree_root(store_id=store_id, generation=publish_generation)
701
+ while publish_generation > 0:
702
+ write_file_result = await write_files_for_root(
703
+ self.data_store,
704
+ store_id,
705
+ root,
706
+ self.server_files_location,
707
+ full_tree_first_publish_generation,
708
+ group_by_store=self.group_files_by_store,
709
+ )
710
+ if not write_file_result.result:
711
+ # this particular return only happens if the files already exist, no need to log anything
712
+ break
713
+ try:
714
+ if uploaders is not None and len(uploaders) > 0:
715
+ request_json = {
716
+ "store_id": store_id.hex(),
717
+ "diff_filename": write_file_result.diff_tree.name,
718
+ "group_files_by_store": self.group_files_by_store,
719
+ }
720
+ if write_file_result.full_tree is not None:
721
+ request_json["full_tree_filename"] = write_file_result.full_tree.name
722
+
723
+ for uploader in uploaders:
724
+ self.log.info(f"Using uploader {uploader} for store {store_id.hex()}")
725
+ async with aiohttp.ClientSession() as session:
726
+ async with session.post(
727
+ uploader.url + "/upload",
728
+ json=request_json,
729
+ headers=uploader.headers,
730
+ ) as response:
731
+ res_json = await response.json()
732
+ if res_json["uploaded"]:
733
+ self.log.info(
734
+ f"Uploaded files to {uploader} for store {store_id.hex()} "
735
+ f"generation {publish_generation}"
736
+ )
737
+ else:
738
+ self.log.error(
739
+ f"Failed to upload files to, will retry later: {uploader} : {res_json}"
740
+ )
741
+ except Exception as e:
742
+ self.log.error(f"Exception uploading files, will retry later: store id {store_id}")
743
+ self.log.debug(f"Failed to upload files, cleaning local files: {type(e).__name__}: {e}")
744
+ if write_file_result.full_tree is not None:
745
+ os.remove(write_file_result.full_tree)
746
+ os.remove(write_file_result.diff_tree)
747
+ publish_generation -= 1
748
+ root = await self.data_store.get_tree_root(store_id=store_id, generation=publish_generation)
749
+
750
+ async def add_missing_files(self, store_id: bytes32, overwrite: bool, foldername: Optional[Path]) -> None:
751
+ root = await self.data_store.get_tree_root(store_id=store_id)
752
+ latest_generation = root.generation
753
+ full_tree_first_publish_generation = max(0, latest_generation - self.maximum_full_file_count + 1)
754
+ singleton_record: Optional[SingletonRecord] = await self.wallet_rpc.dl_latest_singleton(store_id, True)
755
+ if singleton_record is None:
756
+ self.log.error(f"No singleton record found for: {store_id}")
757
+ return
758
+ max_generation = min(singleton_record.generation, 0 if root is None else root.generation)
759
+ server_files_location = foldername if foldername is not None else self.server_files_location
760
+ files = []
761
+ for generation in range(1, max_generation + 1):
762
+ root = await self.data_store.get_tree_root(store_id=store_id, generation=generation)
763
+ res = await write_files_for_root(
764
+ self.data_store,
765
+ store_id,
766
+ root,
767
+ server_files_location,
768
+ full_tree_first_publish_generation,
769
+ overwrite,
770
+ self.group_files_by_store,
771
+ )
772
+ files.append(res.diff_tree.name)
773
+ if res.full_tree is not None:
774
+ files.append(res.full_tree.name)
775
+
776
+ uploaders = await self.get_uploaders(store_id)
777
+ if uploaders is not None and len(uploaders) > 0:
778
+ request_json = {
779
+ "store_id": store_id.hex(),
780
+ "files": json.dumps(files),
781
+ "group_files_by_store": self.group_files_by_store,
782
+ }
783
+ for uploader in uploaders:
784
+ async with aiohttp.ClientSession() as session:
785
+ async with session.post(
786
+ uploader.url + "/add_missing_files",
787
+ json=request_json,
788
+ headers=uploader.headers,
789
+ ) as response:
790
+ res_json = await response.json()
791
+ if not res_json["uploaded"]:
792
+ self.log.error(f"failed to upload to uploader {uploader}")
793
+ else:
794
+ self.log.debug(f"uploaded to uploader {uploader}")
795
+
796
+ async def subscribe(self, store_id: bytes32, urls: List[str]) -> Subscription:
797
+ parsed_urls = [url.rstrip("/") for url in urls]
798
+ subscription = Subscription(store_id, [ServerInfo(url, 0, 0) for url in parsed_urls])
799
+ await self.wallet_rpc.dl_track_new(subscription.store_id)
800
+ async with self.subscription_lock:
801
+ await self.data_store.subscribe(subscription)
802
+ self.log.info(f"Done adding subscription: {subscription.store_id}")
803
+ return subscription
804
+
805
+ async def remove_subscriptions(self, store_id: bytes32, urls: List[str]) -> None:
806
+ parsed_urls = [url.rstrip("/") for url in urls]
807
+ async with self.subscription_lock:
808
+ await self.data_store.remove_subscriptions(store_id, parsed_urls)
809
+
810
+ async def unsubscribe(self, store_id: bytes32, retain_data: bool) -> None:
811
+ async with self.subscription_lock:
812
+ subscriptions = await self.data_store.get_subscriptions()
813
+ if store_id not in (subscription.store_id for subscription in subscriptions):
814
+ raise RuntimeError("No subscription found for the given store_id.")
815
+
816
+ # Unsubscribe is processed later, after all fetching of data is done, to avoid races.
817
+ self.unsubscribe_data_queue.append(UnsubscribeData(store_id, retain_data))
818
+
819
+ async def process_unsubscribe(self, store_id: bytes32, retain_data: bool) -> None:
820
+ # This function already acquired `subscriptions_lock`.
821
+ subscriptions = await self.data_store.get_subscriptions()
822
+ if store_id not in (subscription.store_id for subscription in subscriptions):
823
+ raise RuntimeError("No subscription found for the given store_id.")
824
+ paths: List[Path] = []
825
+ if await self.data_store.store_id_exists(store_id) and not retain_data:
826
+ generation = await self.data_store.get_tree_generation(store_id)
827
+ all_roots = await self.data_store.get_roots_between(store_id, 1, generation + 1)
828
+ for root in all_roots:
829
+ root_hash = root.node_hash if root.node_hash is not None else self.none_bytes
830
+ for group_by_store in (True, False):
831
+ paths.append(
832
+ get_full_tree_filename_path(
833
+ self.server_files_location,
834
+ store_id,
835
+ root_hash,
836
+ root.generation,
837
+ group_by_store,
838
+ )
839
+ )
840
+ paths.append(
841
+ get_delta_filename_path(
842
+ self.server_files_location,
843
+ store_id,
844
+ root_hash,
845
+ root.generation,
846
+ group_by_store,
847
+ )
848
+ )
849
+
850
+ # stop tracking first, then unsubscribe from the data store
851
+ await self.wallet_rpc.dl_stop_tracking(store_id)
852
+ await self.data_store.unsubscribe(store_id)
853
+ if not retain_data:
854
+ await self.data_store.delete_store_data(store_id)
855
+
856
+ self.log.info(f"Unsubscribed to {store_id}")
857
+ for file_path in paths:
858
+ try:
859
+ file_path.unlink()
860
+ except FileNotFoundError:
861
+ pass
862
+
863
+ async def get_subscriptions(self) -> List[Subscription]:
864
+ async with self.subscription_lock:
865
+ return await self.data_store.get_subscriptions()
866
+
867
+ async def add_mirror(self, store_id: bytes32, urls: List[str], amount: uint64, fee: uint64) -> None:
868
+ if not urls:
869
+ raise RuntimeError("URL list can't be empty")
870
+ bytes_urls = [bytes(url, "utf8") for url in urls]
871
+ await self.wallet_rpc.dl_new_mirror(store_id, amount, bytes_urls, fee)
872
+
873
+ async def delete_mirror(self, coin_id: bytes32, fee: uint64) -> None:
874
+ await self.wallet_rpc.dl_delete_mirror(coin_id, fee)
875
+
876
+ async def get_mirrors(self, store_id: bytes32) -> List[Mirror]:
877
+ mirrors: List[Mirror] = await self.wallet_rpc.dl_get_mirrors(store_id)
878
+ return [mirror for mirror in mirrors if mirror.urls]
879
+
880
+ async def update_subscriptions_from_wallet(self, store_id: bytes32) -> None:
881
+ mirrors: List[Mirror] = await self.wallet_rpc.dl_get_mirrors(store_id)
882
+ urls: List[str] = []
883
+ for mirror in mirrors:
884
+ urls = urls + [url.decode("utf8") for url in mirror.urls]
885
+ urls = [url.rstrip("/") for url in urls]
886
+ await self.data_store.update_subscriptions_from_wallet(store_id, urls)
887
+
888
+ async def get_owned_stores(self) -> List[SingletonRecord]:
889
+ return await self.wallet_rpc.dl_owned_singletons()
890
+
891
+ async def get_kv_diff(self, store_id: bytes32, hash_1: bytes32, hash_2: bytes32) -> Set[DiffData]:
892
+ return await self.data_store.get_kv_diff(store_id, hash_1, hash_2)
893
+
894
+ async def get_kv_diff_paginated(
895
+ self,
896
+ store_id: bytes32,
897
+ # NOTE: empty is expressed as zeros
898
+ hash_1: bytes32,
899
+ hash_2: bytes32,
900
+ page: int,
901
+ max_page_size: Optional[int] = None,
902
+ ) -> KVDiffPaginationData:
903
+ if max_page_size is None:
904
+ max_page_size = 40 * 1024 * 1024
905
+ return await self.data_store.get_kv_diff_paginated(store_id, page, max_page_size, hash_1, hash_2)
906
+
907
+ async def periodically_manage_data(self) -> None:
908
+ manage_data_interval = self.config.get("manage_data_interval", 60)
909
+ while not self._shut_down:
910
+ async with self.subscription_lock:
911
+ try:
912
+ subscriptions = await self.data_store.get_subscriptions()
913
+ for subscription in subscriptions:
914
+ await self.wallet_rpc.dl_track_new(subscription.store_id)
915
+ break
916
+ except aiohttp.client_exceptions.ClientConnectorError:
917
+ pass
918
+ except Exception as e:
919
+ self.log.error(f"Exception while requesting wallet track subscription: {type(e)} {e}")
920
+
921
+ self.log.warning("Cannot connect to the wallet. Retrying in 3s.")
922
+
923
+ delay_until = time.monotonic() + 3
924
+ while time.monotonic() < delay_until:
925
+ if self._shut_down:
926
+ break
927
+ await asyncio.sleep(0.1)
928
+
929
+ while not self._shut_down:
930
+ # Add existing subscriptions
931
+ async with self.subscription_lock:
932
+ subscriptions = await self.data_store.get_subscriptions()
933
+
934
+ # pseudo-subscribe to all unsubscribed owned stores
935
+ # Need this to make sure we process updates and generate DAT files
936
+ try:
937
+ owned_stores = await self.get_owned_stores()
938
+ except ValueError:
939
+ # Sometimes the DL wallet isn't available, so we can't get the owned stores.
940
+ # We'll try again next time.
941
+ owned_stores = []
942
+ subscription_store_ids = {subscription.store_id for subscription in subscriptions}
943
+ for record in owned_stores:
944
+ store_id = record.launcher_id
945
+ if store_id not in subscription_store_ids:
946
+ try:
947
+ # don't actually subscribe, just add to the list
948
+ subscriptions.insert(0, Subscription(store_id=store_id, servers_info=[]))
949
+ except Exception as e:
950
+ self.log.info(
951
+ f"Can't subscribe to owned store {store_id}: {type(e)} {e} {traceback.format_exc()}"
952
+ )
953
+
954
+ # Optionally
955
+ # Subscribe to all local non-owned store_ids that we can find on chain.
956
+ # This is the prior behavior where all local stores, both owned and not owned, are subscribed to.
957
+ if self.config.get("auto_subscribe_to_local_stores", False):
958
+ local_store_ids = await self.data_store.get_store_ids()
959
+ subscription_store_ids = {subscription.store_id for subscription in subscriptions}
960
+ for local_id in local_store_ids:
961
+ if local_id not in subscription_store_ids:
962
+ try:
963
+ subscription = await self.subscribe(local_id, [])
964
+ subscriptions.insert(0, subscription)
965
+ except Exception as e:
966
+ self.log.info(
967
+ f"Can't subscribe to local store {local_id}: {type(e)} {e} {traceback.format_exc()}"
968
+ )
969
+
970
+ work_queue: asyncio.Queue[Job[Subscription]] = asyncio.Queue()
971
+ async with QueuedAsyncPool.managed(
972
+ name="DataLayer subscription update pool",
973
+ worker_async_callable=self.update_subscription,
974
+ job_queue=work_queue,
975
+ target_worker_count=self.subscription_update_concurrency,
976
+ log=self.log,
977
+ ):
978
+ jobs = [Job(input=subscription) for subscription in subscriptions]
979
+ for job in jobs:
980
+ await work_queue.put(job)
981
+
982
+ await asyncio.gather(*(job.done.wait() for job in jobs), return_exceptions=True)
983
+
984
+ # Do unsubscribes after the fetching of data is complete, to avoid races.
985
+ async with self.subscription_lock:
986
+ for unsubscribe_data in self.unsubscribe_data_queue:
987
+ await self.process_unsubscribe(unsubscribe_data.store_id, unsubscribe_data.retain_data)
988
+ self.unsubscribe_data_queue.clear()
989
+ await asyncio.sleep(manage_data_interval)
990
+
991
+ async def update_subscription(
992
+ self,
993
+ worker_id: int,
994
+ job: Job[Subscription],
995
+ ) -> None:
996
+ subscription = job.input
997
+
998
+ try:
999
+ await self.update_subscriptions_from_wallet(subscription.store_id)
1000
+ await self.fetch_and_validate(subscription.store_id)
1001
+ await self.upload_files(subscription.store_id)
1002
+ await self.clean_old_full_tree_files(subscription.store_id)
1003
+ except Exception as e:
1004
+ self.log.error(f"Exception while fetching data: {type(e)} {e} {traceback.format_exc()}.")
1005
+
1006
+ async def build_offer_changelist(
1007
+ self,
1008
+ store_id: bytes32,
1009
+ inclusions: Tuple[KeyValue, ...],
1010
+ ) -> List[Dict[str, Any]]:
1011
+ async with self.data_store.transaction():
1012
+ changelist: List[Dict[str, Any]] = []
1013
+ for entry in inclusions:
1014
+ try:
1015
+ existing_value = await self.get_value(store_id=store_id, key=entry.key)
1016
+ except KeyNotFoundError:
1017
+ existing_value = None
1018
+
1019
+ if existing_value == entry.value:
1020
+ # already present, nothing needed
1021
+ continue
1022
+
1023
+ if existing_value is not None:
1024
+ # upsert, delete the existing key and value
1025
+ changelist.append(
1026
+ {
1027
+ "action": "delete",
1028
+ "key": entry.key,
1029
+ }
1030
+ )
1031
+
1032
+ changelist.append(
1033
+ {
1034
+ "action": "insert",
1035
+ "key": entry.key,
1036
+ "value": entry.value,
1037
+ }
1038
+ )
1039
+
1040
+ return changelist
1041
+
1042
+ async def process_offered_stores(self, offer_stores: Tuple[OfferStore, ...]) -> Dict[bytes32, StoreProofs]:
1043
+ for offer_store in offer_stores:
1044
+ await self._update_confirmation_status(store_id=offer_store.store_id)
1045
+
1046
+ async with self.data_store.transaction():
1047
+ our_store_proofs: Dict[bytes32, StoreProofs] = {}
1048
+ for offer_store in offer_stores:
1049
+ changelist = await self.build_offer_changelist(
1050
+ store_id=offer_store.store_id,
1051
+ inclusions=offer_store.inclusions,
1052
+ )
1053
+
1054
+ if len(changelist) > 0:
1055
+ new_root_hash = await self.batch_insert(
1056
+ store_id=offer_store.store_id,
1057
+ changelist=changelist,
1058
+ enable_batch_autoinsert=False,
1059
+ )
1060
+ else:
1061
+ existing_root = await self.get_root(store_id=offer_store.store_id)
1062
+ if existing_root is None:
1063
+ raise Exception(f"store id not available: {offer_store.store_id.hex()}")
1064
+ new_root_hash = existing_root.root
1065
+
1066
+ if new_root_hash is None:
1067
+ raise Exception("only inserts are supported so a None root hash should not be possible")
1068
+
1069
+ proofs: List[Proof] = []
1070
+ for entry in offer_store.inclusions:
1071
+ node_hash = await self.get_key_value_hash(
1072
+ store_id=offer_store.store_id,
1073
+ key=entry.key,
1074
+ root_hash=new_root_hash,
1075
+ )
1076
+ proof_of_inclusion = await self.data_store.get_proof_of_inclusion_by_hash(
1077
+ node_hash=node_hash,
1078
+ store_id=offer_store.store_id,
1079
+ root_hash=new_root_hash,
1080
+ )
1081
+ proof = Proof(
1082
+ key=entry.key,
1083
+ value=entry.value,
1084
+ node_hash=proof_of_inclusion.node_hash,
1085
+ layers=tuple(
1086
+ Layer(
1087
+ other_hash_side=layer.other_hash_side,
1088
+ other_hash=layer.other_hash,
1089
+ combined_hash=layer.combined_hash,
1090
+ )
1091
+ for layer in proof_of_inclusion.layers
1092
+ ),
1093
+ )
1094
+ proofs.append(proof)
1095
+ store_proof = StoreProofs(store_id=offer_store.store_id, proofs=tuple(proofs))
1096
+ our_store_proofs[offer_store.store_id] = store_proof
1097
+ return our_store_proofs
1098
+
1099
+ async def make_offer(
1100
+ self,
1101
+ maker: Tuple[OfferStore, ...],
1102
+ taker: Tuple[OfferStore, ...],
1103
+ fee: uint64,
1104
+ ) -> Offer:
1105
+ async with self.data_store.transaction():
1106
+ our_store_proofs = await self.process_offered_stores(offer_stores=maker)
1107
+
1108
+ offer_dict: Dict[Union[uint32, str], int] = {
1109
+ **{offer_store.store_id.hex(): -1 for offer_store in maker},
1110
+ **{offer_store.store_id.hex(): 1 for offer_store in taker},
1111
+ }
1112
+
1113
+ solver: Dict[str, Any] = {
1114
+ "0x"
1115
+ + our_offer_store.store_id.hex(): {
1116
+ "new_root": "0x" + our_store_proofs[our_offer_store.store_id].proofs[0].root().hex(),
1117
+ "dependencies": [
1118
+ {
1119
+ "launcher_id": "0x" + their_offer_store.store_id.hex(),
1120
+ "values_to_prove": [
1121
+ "0x" + leaf_hash(key=entry.key, value=entry.value).hex()
1122
+ for entry in their_offer_store.inclusions
1123
+ ],
1124
+ }
1125
+ for their_offer_store in taker
1126
+ ],
1127
+ }
1128
+ for our_offer_store in maker
1129
+ }
1130
+
1131
+ res = await self.wallet_rpc.create_offer_for_ids(
1132
+ offer_dict=offer_dict,
1133
+ solver=solver,
1134
+ driver_dict={},
1135
+ fee=fee,
1136
+ validate_only=False,
1137
+ # TODO: probably shouldn't be default but due to peculiarities in the RPC, we're using a stop gap.
1138
+ # This is not a change in behavior, the default was already implicit.
1139
+ tx_config=DEFAULT_TX_CONFIG,
1140
+ )
1141
+
1142
+ offer = Offer(
1143
+ trade_id=res.trade_record.trade_id,
1144
+ offer=bytes(res.offer),
1145
+ taker=taker,
1146
+ maker=tuple(our_store_proofs.values()),
1147
+ )
1148
+
1149
+ # being extra careful and verifying the offer before returning it
1150
+ trading_offer = TradingOffer.from_bytes(offer.offer)
1151
+ summary = await DataLayerWallet.get_offer_summary(offer=trading_offer)
1152
+
1153
+ verify_offer(maker=offer.maker, taker=offer.taker, summary=summary)
1154
+
1155
+ await self.data_store.clean_node_table()
1156
+ return offer
1157
+
1158
+ async def take_offer(
1159
+ self,
1160
+ offer_bytes: bytes,
1161
+ taker: Tuple[OfferStore, ...],
1162
+ maker: Tuple[StoreProofs, ...],
1163
+ fee: uint64,
1164
+ ) -> TradeRecord:
1165
+ async with self.data_store.transaction():
1166
+ our_store_proofs = await self.process_offered_stores(offer_stores=taker)
1167
+
1168
+ offer = TradingOffer.from_bytes(offer_bytes)
1169
+ summary = await DataLayerWallet.get_offer_summary(offer=offer)
1170
+
1171
+ verify_offer(maker=maker, taker=taker, summary=summary)
1172
+
1173
+ all_store_proofs: Dict[bytes32, StoreProofs] = {
1174
+ store_proofs.proofs[0].root(): store_proofs for store_proofs in [*maker, *our_store_proofs.values()]
1175
+ }
1176
+ proofs_of_inclusion: List[Tuple[str, str, List[str]]] = []
1177
+ for root, store_proofs in all_store_proofs.items():
1178
+ for proof in store_proofs.proofs:
1179
+ layers = [
1180
+ ProofOfInclusionLayer(
1181
+ combined_hash=layer.combined_hash,
1182
+ other_hash_side=layer.other_hash_side,
1183
+ other_hash=layer.other_hash,
1184
+ )
1185
+ for layer in proof.layers
1186
+ ]
1187
+ proof_of_inclusion = ProofOfInclusion(node_hash=proof.node_hash, layers=layers)
1188
+ sibling_sides_integer = proof_of_inclusion.sibling_sides_integer()
1189
+ proofs_of_inclusion.append(
1190
+ (
1191
+ root.hex(),
1192
+ str(sibling_sides_integer),
1193
+ ["0x" + sibling_hash.hex() for sibling_hash in proof_of_inclusion.sibling_hashes()],
1194
+ )
1195
+ )
1196
+
1197
+ solver: Dict[str, Any] = {
1198
+ "proofs_of_inclusion": proofs_of_inclusion,
1199
+ **{
1200
+ "0x"
1201
+ + our_offer_store.store_id.hex(): {
1202
+ "new_root": "0x" + root.hex(),
1203
+ "dependencies": [
1204
+ {
1205
+ "launcher_id": "0x" + their_offer_store.store_id.hex(),
1206
+ "values_to_prove": ["0x" + entry.node_hash.hex() for entry in their_offer_store.proofs],
1207
+ }
1208
+ for their_offer_store in maker
1209
+ ],
1210
+ }
1211
+ for our_offer_store in taker
1212
+ },
1213
+ }
1214
+
1215
+ await self.data_store.clean_node_table()
1216
+
1217
+ # Excluding wallet from transaction since failures in the wallet may occur
1218
+ # after the transaction is submitted to the chain. If we roll back data we
1219
+ # may lose published data.
1220
+
1221
+ trade_record = (
1222
+ await self.wallet_rpc.take_offer(
1223
+ offer=offer,
1224
+ solver=solver,
1225
+ fee=fee,
1226
+ # TODO: probably shouldn't be default but due to peculiarities in the RPC, we're using a stop gap.
1227
+ # This is not a change in behavior, the default was already implicit.
1228
+ tx_config=DEFAULT_TX_CONFIG,
1229
+ )
1230
+ ).trade_record
1231
+
1232
+ return trade_record
1233
+
1234
+ async def cancel_offer(self, trade_id: bytes32, secure: bool, fee: uint64) -> None:
1235
+ store_ids: List[bytes32] = []
1236
+
1237
+ if not secure:
1238
+ trade_record = await self.wallet_rpc.get_offer(trade_id=trade_id, file_contents=True)
1239
+ trading_offer = TradingOffer.from_bytes(trade_record.offer)
1240
+ summary = await DataLayerWallet.get_offer_summary(offer=trading_offer)
1241
+ store_ids = [bytes32.from_hexstr(offered["launcher_id"]) for offered in summary["offered"]]
1242
+
1243
+ await self.wallet_rpc.cancel_offer(
1244
+ trade_id=trade_id,
1245
+ secure=secure,
1246
+ fee=fee,
1247
+ # TODO: probably shouldn't be default but due to peculiarities in the RPC, we're using a stop gap.
1248
+ # This is not a change in behavior, the default was already implicit.
1249
+ tx_config=DEFAULT_TX_CONFIG,
1250
+ )
1251
+
1252
+ if not secure:
1253
+ for store_id in store_ids:
1254
+ await self.data_store.clear_pending_roots(store_id=store_id)
1255
+
1256
+ async def get_sync_status(self, store_id: bytes32) -> SyncStatus:
1257
+ await self._update_confirmation_status(store_id=store_id)
1258
+
1259
+ if not await self.data_store.store_id_exists(store_id=store_id):
1260
+ raise Exception(f"No store id stored in the local database for {store_id}")
1261
+ root = await self.data_store.get_tree_root(store_id=store_id)
1262
+ singleton_record = await self.wallet_rpc.dl_latest_singleton(store_id, True)
1263
+ if singleton_record is None:
1264
+ raise Exception(f"No singleton found for {store_id}")
1265
+
1266
+ return SyncStatus(
1267
+ root_hash=self.none_bytes if root.node_hash is None else root.node_hash,
1268
+ generation=root.generation,
1269
+ target_root_hash=singleton_record.root,
1270
+ target_generation=singleton_record.generation,
1271
+ )
1272
+
1273
+ async def get_uploaders(self, store_id: bytes32) -> List[PluginRemote]:
1274
+ uploaders = []
1275
+ for uploader in self.uploaders:
1276
+ async with aiohttp.ClientSession() as session:
1277
+ try:
1278
+ async with session.post(
1279
+ uploader.url + "/handle_upload",
1280
+ json={"store_id": store_id.hex()},
1281
+ headers=uploader.headers,
1282
+ ) as response:
1283
+ res_json = await response.json()
1284
+ if res_json["handle_upload"]:
1285
+ uploaders.append(uploader)
1286
+ except Exception as e:
1287
+ self.log.error(f"get_uploader could not get response {e}")
1288
+ return uploaders
1289
+
1290
+ async def check_plugins(self) -> PluginStatus:
1291
+ coros = [get_plugin_info(plugin_remote=plugin) for plugin in {*self.uploaders, *self.downloaders}]
1292
+ results = dict(await asyncio.gather(*coros))
1293
+
1294
+ unknown = {
1295
+ "name": "unknown",
1296
+ "version": "unknown",
1297
+ "instance": "unknown",
1298
+ }
1299
+
1300
+ uploader_status = {uploader.url: results.get(uploader, unknown) for uploader in self.uploaders}
1301
+ downloader_status = {downloader.url: results.get(downloader, unknown) for downloader in self.downloaders}
1302
+
1303
+ return PluginStatus(uploaders=uploader_status, downloaders=downloader_status)