chia-blockchain 2.5.1rc1__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 (1042) 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 +195 -0
  7. chia/_tests/blockchain/config.py +4 -0
  8. chia/_tests/blockchain/test_augmented_chain.py +145 -0
  9. chia/_tests/blockchain/test_blockchain.py +4202 -0
  10. chia/_tests/blockchain/test_blockchain_transactions.py +1031 -0
  11. chia/_tests/blockchain/test_build_chains.py +59 -0
  12. chia/_tests/blockchain/test_get_block_generator.py +72 -0
  13. chia/_tests/blockchain/test_lookup_fork_chain.py +194 -0
  14. chia/_tests/build-init-files.py +92 -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 +72 -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 +149 -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 +55 -0
  26. chia/_tests/clvm/test_message_conditions.py +184 -0
  27. chia/_tests/clvm/test_program.py +150 -0
  28. chia/_tests/clvm/test_puzzle_compression.py +143 -0
  29. chia/_tests/clvm/test_puzzle_drivers.py +45 -0
  30. chia/_tests/clvm/test_puzzles.py +242 -0
  31. chia/_tests/clvm/test_singletons.py +540 -0
  32. chia/_tests/clvm/test_spend_sim.py +181 -0
  33. chia/_tests/cmds/__init__.py +0 -0
  34. chia/_tests/cmds/cmd_test_utils.py +469 -0
  35. chia/_tests/cmds/config.py +3 -0
  36. chia/_tests/cmds/conftest.py +23 -0
  37. chia/_tests/cmds/test_click_types.py +200 -0
  38. chia/_tests/cmds/test_cmd_framework.py +620 -0
  39. chia/_tests/cmds/test_cmds_util.py +97 -0
  40. chia/_tests/cmds/test_daemon.py +92 -0
  41. chia/_tests/cmds/test_dev_gh.py +131 -0
  42. chia/_tests/cmds/test_farm_cmd.py +66 -0
  43. chia/_tests/cmds/test_show.py +116 -0
  44. chia/_tests/cmds/test_sim.py +207 -0
  45. chia/_tests/cmds/test_timelock_args.py +75 -0
  46. chia/_tests/cmds/test_tx_config_args.py +154 -0
  47. chia/_tests/cmds/testing_classes.py +59 -0
  48. chia/_tests/cmds/wallet/__init__.py +0 -0
  49. chia/_tests/cmds/wallet/test_consts.py +47 -0
  50. chia/_tests/cmds/wallet/test_dao.py +565 -0
  51. chia/_tests/cmds/wallet/test_did.py +403 -0
  52. chia/_tests/cmds/wallet/test_nft.py +471 -0
  53. chia/_tests/cmds/wallet/test_notifications.py +124 -0
  54. chia/_tests/cmds/wallet/test_offer.toffer +1 -0
  55. chia/_tests/cmds/wallet/test_tx_decorators.py +27 -0
  56. chia/_tests/cmds/wallet/test_vcs.py +400 -0
  57. chia/_tests/cmds/wallet/test_wallet.py +1125 -0
  58. chia/_tests/cmds/wallet/test_wallet_check.py +109 -0
  59. chia/_tests/conftest.py +1419 -0
  60. chia/_tests/connection_utils.py +125 -0
  61. chia/_tests/core/__init__.py +0 -0
  62. chia/_tests/core/cmds/__init__.py +0 -0
  63. chia/_tests/core/cmds/test_beta.py +382 -0
  64. chia/_tests/core/cmds/test_keys.py +1734 -0
  65. chia/_tests/core/cmds/test_wallet.py +126 -0
  66. chia/_tests/core/config.py +3 -0
  67. chia/_tests/core/consensus/__init__.py +0 -0
  68. chia/_tests/core/consensus/test_block_creation.py +54 -0
  69. chia/_tests/core/consensus/test_pot_iterations.py +117 -0
  70. chia/_tests/core/custom_types/__init__.py +0 -0
  71. chia/_tests/core/custom_types/test_coin.py +107 -0
  72. chia/_tests/core/custom_types/test_proof_of_space.py +144 -0
  73. chia/_tests/core/custom_types/test_spend_bundle.py +70 -0
  74. chia/_tests/core/daemon/__init__.py +0 -0
  75. chia/_tests/core/daemon/config.py +4 -0
  76. chia/_tests/core/daemon/test_daemon.py +2128 -0
  77. chia/_tests/core/daemon/test_daemon_register.py +109 -0
  78. chia/_tests/core/daemon/test_keychain_proxy.py +101 -0
  79. chia/_tests/core/data_layer/__init__.py +0 -0
  80. chia/_tests/core/data_layer/config.py +5 -0
  81. chia/_tests/core/data_layer/conftest.py +106 -0
  82. chia/_tests/core/data_layer/test_data_cli.py +56 -0
  83. chia/_tests/core/data_layer/test_data_layer.py +83 -0
  84. chia/_tests/core/data_layer/test_data_layer_util.py +218 -0
  85. chia/_tests/core/data_layer/test_data_rpc.py +3847 -0
  86. chia/_tests/core/data_layer/test_data_store.py +2424 -0
  87. chia/_tests/core/data_layer/test_data_store_schema.py +381 -0
  88. chia/_tests/core/data_layer/test_plugin.py +91 -0
  89. chia/_tests/core/data_layer/util.py +233 -0
  90. chia/_tests/core/farmer/__init__.py +0 -0
  91. chia/_tests/core/farmer/config.py +3 -0
  92. chia/_tests/core/farmer/test_farmer_api.py +103 -0
  93. chia/_tests/core/full_node/__init__.py +0 -0
  94. chia/_tests/core/full_node/config.py +4 -0
  95. chia/_tests/core/full_node/dos/__init__.py +0 -0
  96. chia/_tests/core/full_node/dos/config.py +3 -0
  97. chia/_tests/core/full_node/full_sync/__init__.py +0 -0
  98. chia/_tests/core/full_node/full_sync/config.py +4 -0
  99. chia/_tests/core/full_node/full_sync/test_full_sync.py +443 -0
  100. chia/_tests/core/full_node/ram_db.py +27 -0
  101. chia/_tests/core/full_node/stores/__init__.py +0 -0
  102. chia/_tests/core/full_node/stores/config.py +4 -0
  103. chia/_tests/core/full_node/stores/test_block_store.py +590 -0
  104. chia/_tests/core/full_node/stores/test_coin_store.py +897 -0
  105. chia/_tests/core/full_node/stores/test_full_node_store.py +1219 -0
  106. chia/_tests/core/full_node/stores/test_hint_store.py +229 -0
  107. chia/_tests/core/full_node/stores/test_sync_store.py +135 -0
  108. chia/_tests/core/full_node/test_address_manager.py +588 -0
  109. chia/_tests/core/full_node/test_block_height_map.py +556 -0
  110. chia/_tests/core/full_node/test_conditions.py +556 -0
  111. chia/_tests/core/full_node/test_full_node.py +2700 -0
  112. chia/_tests/core/full_node/test_generator_tools.py +82 -0
  113. chia/_tests/core/full_node/test_hint_management.py +104 -0
  114. chia/_tests/core/full_node/test_node_load.py +34 -0
  115. chia/_tests/core/full_node/test_performance.py +179 -0
  116. chia/_tests/core/full_node/test_subscriptions.py +492 -0
  117. chia/_tests/core/full_node/test_transactions.py +203 -0
  118. chia/_tests/core/full_node/test_tx_processing_queue.py +155 -0
  119. chia/_tests/core/large_block.py +2388 -0
  120. chia/_tests/core/make_block_generator.py +70 -0
  121. chia/_tests/core/mempool/__init__.py +0 -0
  122. chia/_tests/core/mempool/config.py +4 -0
  123. chia/_tests/core/mempool/test_mempool.py +3255 -0
  124. chia/_tests/core/mempool/test_mempool_fee_estimator.py +104 -0
  125. chia/_tests/core/mempool/test_mempool_fee_protocol.py +55 -0
  126. chia/_tests/core/mempool/test_mempool_item_queries.py +190 -0
  127. chia/_tests/core/mempool/test_mempool_manager.py +2084 -0
  128. chia/_tests/core/mempool/test_mempool_performance.py +64 -0
  129. chia/_tests/core/mempool/test_singleton_fast_forward.py +567 -0
  130. chia/_tests/core/node_height.py +28 -0
  131. chia/_tests/core/server/__init__.py +0 -0
  132. chia/_tests/core/server/config.py +3 -0
  133. chia/_tests/core/server/flood.py +84 -0
  134. chia/_tests/core/server/serve.py +135 -0
  135. chia/_tests/core/server/test_api_protocol.py +21 -0
  136. chia/_tests/core/server/test_capabilities.py +66 -0
  137. chia/_tests/core/server/test_dos.py +319 -0
  138. chia/_tests/core/server/test_event_loop.py +109 -0
  139. chia/_tests/core/server/test_loop.py +294 -0
  140. chia/_tests/core/server/test_node_discovery.py +73 -0
  141. chia/_tests/core/server/test_rate_limits.py +482 -0
  142. chia/_tests/core/server/test_server.py +226 -0
  143. chia/_tests/core/server/test_upnp.py +8 -0
  144. chia/_tests/core/services/__init__.py +0 -0
  145. chia/_tests/core/services/config.py +3 -0
  146. chia/_tests/core/services/test_services.py +188 -0
  147. chia/_tests/core/ssl/__init__.py +0 -0
  148. chia/_tests/core/ssl/config.py +3 -0
  149. chia/_tests/core/ssl/test_ssl.py +202 -0
  150. chia/_tests/core/test_coins.py +33 -0
  151. chia/_tests/core/test_cost_calculation.py +313 -0
  152. chia/_tests/core/test_crawler.py +175 -0
  153. chia/_tests/core/test_crawler_rpc.py +53 -0
  154. chia/_tests/core/test_daemon_rpc.py +24 -0
  155. chia/_tests/core/test_db_conversion.py +130 -0
  156. chia/_tests/core/test_db_validation.py +162 -0
  157. chia/_tests/core/test_farmer_harvester_rpc.py +505 -0
  158. chia/_tests/core/test_filter.py +35 -0
  159. chia/_tests/core/test_full_node_rpc.py +768 -0
  160. chia/_tests/core/test_merkle_set.py +343 -0
  161. chia/_tests/core/test_program.py +47 -0
  162. chia/_tests/core/test_rpc_util.py +86 -0
  163. chia/_tests/core/test_seeder.py +420 -0
  164. chia/_tests/core/test_setproctitle.py +13 -0
  165. chia/_tests/core/util/__init__.py +0 -0
  166. chia/_tests/core/util/config.py +4 -0
  167. chia/_tests/core/util/test_block_cache.py +44 -0
  168. chia/_tests/core/util/test_cached_bls.py +57 -0
  169. chia/_tests/core/util/test_config.py +337 -0
  170. chia/_tests/core/util/test_file_keyring_synchronization.py +105 -0
  171. chia/_tests/core/util/test_files.py +391 -0
  172. chia/_tests/core/util/test_jsonify.py +146 -0
  173. chia/_tests/core/util/test_keychain.py +522 -0
  174. chia/_tests/core/util/test_keyring_wrapper.py +491 -0
  175. chia/_tests/core/util/test_lockfile.py +380 -0
  176. chia/_tests/core/util/test_log_exceptions.py +187 -0
  177. chia/_tests/core/util/test_lru_cache.py +56 -0
  178. chia/_tests/core/util/test_significant_bits.py +40 -0
  179. chia/_tests/core/util/test_streamable.py +883 -0
  180. chia/_tests/db/__init__.py +0 -0
  181. chia/_tests/db/test_db_wrapper.py +566 -0
  182. chia/_tests/environments/__init__.py +0 -0
  183. chia/_tests/environments/common.py +35 -0
  184. chia/_tests/environments/full_node.py +47 -0
  185. chia/_tests/environments/wallet.py +429 -0
  186. chia/_tests/ether.py +19 -0
  187. chia/_tests/farmer_harvester/__init__.py +0 -0
  188. chia/_tests/farmer_harvester/config.py +3 -0
  189. chia/_tests/farmer_harvester/test_farmer.py +1264 -0
  190. chia/_tests/farmer_harvester/test_farmer_harvester.py +292 -0
  191. chia/_tests/farmer_harvester/test_filter_prefix_bits.py +131 -0
  192. chia/_tests/farmer_harvester/test_third_party_harvesters.py +528 -0
  193. chia/_tests/farmer_harvester/test_third_party_harvesters_data.json +29 -0
  194. chia/_tests/fee_estimation/__init__.py +0 -0
  195. chia/_tests/fee_estimation/config.py +3 -0
  196. chia/_tests/fee_estimation/test_fee_estimation_integration.py +262 -0
  197. chia/_tests/fee_estimation/test_fee_estimation_rpc.py +287 -0
  198. chia/_tests/fee_estimation/test_fee_estimation_unit_tests.py +144 -0
  199. chia/_tests/fee_estimation/test_mempoolitem_height_added.py +146 -0
  200. chia/_tests/generator/__init__.py +0 -0
  201. chia/_tests/generator/puzzles/__init__.py +0 -0
  202. chia/_tests/generator/puzzles/test_generator_deserialize.clsp +3 -0
  203. chia/_tests/generator/puzzles/test_generator_deserialize.clsp.hex +1 -0
  204. chia/_tests/generator/puzzles/test_multiple_generator_input_arguments.clsp +19 -0
  205. chia/_tests/generator/puzzles/test_multiple_generator_input_arguments.clsp.hex +1 -0
  206. chia/_tests/generator/test_compression.py +201 -0
  207. chia/_tests/generator/test_generator_types.py +44 -0
  208. chia/_tests/generator/test_rom.py +180 -0
  209. chia/_tests/plot_sync/__init__.py +0 -0
  210. chia/_tests/plot_sync/config.py +3 -0
  211. chia/_tests/plot_sync/test_delta.py +101 -0
  212. chia/_tests/plot_sync/test_plot_sync.py +618 -0
  213. chia/_tests/plot_sync/test_receiver.py +451 -0
  214. chia/_tests/plot_sync/test_sender.py +116 -0
  215. chia/_tests/plot_sync/test_sync_simulated.py +451 -0
  216. chia/_tests/plot_sync/util.py +68 -0
  217. chia/_tests/plotting/__init__.py +0 -0
  218. chia/_tests/plotting/config.py +3 -0
  219. chia/_tests/plotting/test_plot_manager.py +781 -0
  220. chia/_tests/plotting/util.py +12 -0
  221. chia/_tests/pools/__init__.py +0 -0
  222. chia/_tests/pools/config.py +5 -0
  223. chia/_tests/pools/test_pool_cli_parsing.py +128 -0
  224. chia/_tests/pools/test_pool_cmdline.py +1001 -0
  225. chia/_tests/pools/test_pool_config.py +42 -0
  226. chia/_tests/pools/test_pool_puzzles_lifecycle.py +397 -0
  227. chia/_tests/pools/test_pool_rpc.py +1123 -0
  228. chia/_tests/pools/test_pool_wallet.py +205 -0
  229. chia/_tests/pools/test_wallet_pool_store.py +161 -0
  230. chia/_tests/process_junit.py +348 -0
  231. chia/_tests/rpc/__init__.py +0 -0
  232. chia/_tests/rpc/test_rpc_client.py +138 -0
  233. chia/_tests/rpc/test_rpc_server.py +183 -0
  234. chia/_tests/simulation/__init__.py +0 -0
  235. chia/_tests/simulation/config.py +6 -0
  236. chia/_tests/simulation/test_simulation.py +501 -0
  237. chia/_tests/simulation/test_simulator.py +232 -0
  238. chia/_tests/simulation/test_start_simulator.py +107 -0
  239. chia/_tests/testconfig.py +13 -0
  240. chia/_tests/timelord/__init__.py +0 -0
  241. chia/_tests/timelord/config.py +3 -0
  242. chia/_tests/timelord/test_new_peak.py +437 -0
  243. chia/_tests/timelord/test_timelord.py +11 -0
  244. chia/_tests/tools/1315537.json +170 -0
  245. chia/_tests/tools/1315544.json +160 -0
  246. chia/_tests/tools/1315630.json +150 -0
  247. chia/_tests/tools/300000.json +105 -0
  248. chia/_tests/tools/442734.json +140 -0
  249. chia/_tests/tools/466212.json +130 -0
  250. chia/_tests/tools/__init__.py +0 -0
  251. chia/_tests/tools/config.py +5 -0
  252. chia/_tests/tools/test-blockchain-db.sqlite +0 -0
  253. chia/_tests/tools/test_full_sync.py +30 -0
  254. chia/_tests/tools/test_legacy_keyring.py +82 -0
  255. chia/_tests/tools/test_run_block.py +128 -0
  256. chia/_tests/tools/test_virtual_project.py +591 -0
  257. chia/_tests/util/__init__.py +0 -0
  258. chia/_tests/util/benchmark_cost.py +170 -0
  259. chia/_tests/util/benchmarks.py +153 -0
  260. chia/_tests/util/bip39_test_vectors.json +148 -0
  261. chia/_tests/util/blockchain.py +134 -0
  262. chia/_tests/util/blockchain_mock.py +132 -0
  263. chia/_tests/util/build_network_protocol_files.py +302 -0
  264. chia/_tests/util/clvm_generator.bin +0 -0
  265. chia/_tests/util/config.py +3 -0
  266. chia/_tests/util/constants.py +20 -0
  267. chia/_tests/util/db_connection.py +37 -0
  268. chia/_tests/util/full_sync.py +253 -0
  269. chia/_tests/util/gen_ssl_certs.py +114 -0
  270. chia/_tests/util/generator_tools_testing.py +45 -0
  271. chia/_tests/util/get_name_puzzle_conditions.py +52 -0
  272. chia/_tests/util/key_tool.py +36 -0
  273. chia/_tests/util/misc.py +675 -0
  274. chia/_tests/util/network_protocol_data.py +1072 -0
  275. chia/_tests/util/protocol_messages_bytes-v1.0 +0 -0
  276. chia/_tests/util/protocol_messages_json.py +2701 -0
  277. chia/_tests/util/rpc.py +26 -0
  278. chia/_tests/util/run_block.py +163 -0
  279. chia/_tests/util/setup_nodes.py +481 -0
  280. chia/_tests/util/spend_sim.py +492 -0
  281. chia/_tests/util/split_managers.py +102 -0
  282. chia/_tests/util/temp_file.py +14 -0
  283. chia/_tests/util/test_action_scope.py +144 -0
  284. chia/_tests/util/test_async_pool.py +366 -0
  285. chia/_tests/util/test_build_job_matrix.py +42 -0
  286. chia/_tests/util/test_build_network_protocol_files.py +7 -0
  287. chia/_tests/util/test_chia_version.py +50 -0
  288. chia/_tests/util/test_collection.py +11 -0
  289. chia/_tests/util/test_condition_tools.py +229 -0
  290. chia/_tests/util/test_config.py +426 -0
  291. chia/_tests/util/test_dump_keyring.py +60 -0
  292. chia/_tests/util/test_errors.py +10 -0
  293. chia/_tests/util/test_full_block_utils.py +279 -0
  294. chia/_tests/util/test_installed.py +20 -0
  295. chia/_tests/util/test_limited_semaphore.py +53 -0
  296. chia/_tests/util/test_logging_filter.py +42 -0
  297. chia/_tests/util/test_misc.py +445 -0
  298. chia/_tests/util/test_network.py +73 -0
  299. chia/_tests/util/test_network_protocol_files.py +578 -0
  300. chia/_tests/util/test_network_protocol_json.py +267 -0
  301. chia/_tests/util/test_network_protocol_test.py +256 -0
  302. chia/_tests/util/test_paginator.py +71 -0
  303. chia/_tests/util/test_pprint.py +17 -0
  304. chia/_tests/util/test_priority_mutex.py +488 -0
  305. chia/_tests/util/test_recursive_replace.py +116 -0
  306. chia/_tests/util/test_replace_str_to_bytes.py +137 -0
  307. chia/_tests/util/test_service_groups.py +15 -0
  308. chia/_tests/util/test_ssl_check.py +31 -0
  309. chia/_tests/util/test_testnet_overrides.py +19 -0
  310. chia/_tests/util/test_tests_misc.py +38 -0
  311. chia/_tests/util/test_timing.py +37 -0
  312. chia/_tests/util/test_trusted_peer.py +51 -0
  313. chia/_tests/util/time_out_assert.py +191 -0
  314. chia/_tests/wallet/__init__.py +0 -0
  315. chia/_tests/wallet/cat_wallet/__init__.py +0 -0
  316. chia/_tests/wallet/cat_wallet/config.py +4 -0
  317. chia/_tests/wallet/cat_wallet/test_cat_lifecycle.py +468 -0
  318. chia/_tests/wallet/cat_wallet/test_cat_outer_puzzle.py +69 -0
  319. chia/_tests/wallet/cat_wallet/test_cat_wallet.py +1826 -0
  320. chia/_tests/wallet/cat_wallet/test_offer_lifecycle.py +291 -0
  321. chia/_tests/wallet/cat_wallet/test_trades.py +2600 -0
  322. chia/_tests/wallet/clawback/__init__.py +0 -0
  323. chia/_tests/wallet/clawback/config.py +3 -0
  324. chia/_tests/wallet/clawback/test_clawback_decorator.py +78 -0
  325. chia/_tests/wallet/clawback/test_clawback_lifecycle.py +292 -0
  326. chia/_tests/wallet/clawback/test_clawback_metadata.py +50 -0
  327. chia/_tests/wallet/config.py +4 -0
  328. chia/_tests/wallet/conftest.py +278 -0
  329. chia/_tests/wallet/dao_wallet/__init__.py +0 -0
  330. chia/_tests/wallet/dao_wallet/config.py +3 -0
  331. chia/_tests/wallet/dao_wallet/test_dao_clvm.py +1330 -0
  332. chia/_tests/wallet/dao_wallet/test_dao_wallets.py +3488 -0
  333. chia/_tests/wallet/db_wallet/__init__.py +0 -0
  334. chia/_tests/wallet/db_wallet/config.py +3 -0
  335. chia/_tests/wallet/db_wallet/test_db_graftroot.py +141 -0
  336. chia/_tests/wallet/db_wallet/test_dl_offers.py +491 -0
  337. chia/_tests/wallet/db_wallet/test_dl_wallet.py +823 -0
  338. chia/_tests/wallet/did_wallet/__init__.py +0 -0
  339. chia/_tests/wallet/did_wallet/config.py +4 -0
  340. chia/_tests/wallet/did_wallet/test_did.py +2284 -0
  341. chia/_tests/wallet/nft_wallet/__init__.py +0 -0
  342. chia/_tests/wallet/nft_wallet/config.py +4 -0
  343. chia/_tests/wallet/nft_wallet/test_nft_1_offers.py +1493 -0
  344. chia/_tests/wallet/nft_wallet/test_nft_bulk_mint.py +1024 -0
  345. chia/_tests/wallet/nft_wallet/test_nft_lifecycle.py +375 -0
  346. chia/_tests/wallet/nft_wallet/test_nft_offers.py +1209 -0
  347. chia/_tests/wallet/nft_wallet/test_nft_puzzles.py +172 -0
  348. chia/_tests/wallet/nft_wallet/test_nft_wallet.py +2584 -0
  349. chia/_tests/wallet/nft_wallet/test_ownership_outer_puzzle.py +70 -0
  350. chia/_tests/wallet/rpc/__init__.py +0 -0
  351. chia/_tests/wallet/rpc/config.py +4 -0
  352. chia/_tests/wallet/rpc/test_dl_wallet_rpc.py +285 -0
  353. chia/_tests/wallet/rpc/test_wallet_rpc.py +3153 -0
  354. chia/_tests/wallet/simple_sync/__init__.py +0 -0
  355. chia/_tests/wallet/simple_sync/config.py +3 -0
  356. chia/_tests/wallet/simple_sync/test_simple_sync_protocol.py +718 -0
  357. chia/_tests/wallet/sync/__init__.py +0 -0
  358. chia/_tests/wallet/sync/config.py +4 -0
  359. chia/_tests/wallet/sync/test_wallet_sync.py +1692 -0
  360. chia/_tests/wallet/test_address_type.py +189 -0
  361. chia/_tests/wallet/test_bech32m.py +45 -0
  362. chia/_tests/wallet/test_clvm_streamable.py +244 -0
  363. chia/_tests/wallet/test_coin_management.py +354 -0
  364. chia/_tests/wallet/test_coin_selection.py +588 -0
  365. chia/_tests/wallet/test_conditions.py +400 -0
  366. chia/_tests/wallet/test_debug_spend_bundle.py +218 -0
  367. chia/_tests/wallet/test_new_wallet_protocol.py +1174 -0
  368. chia/_tests/wallet/test_nft_store.py +192 -0
  369. chia/_tests/wallet/test_notifications.py +196 -0
  370. chia/_tests/wallet/test_offer_parsing_performance.py +48 -0
  371. chia/_tests/wallet/test_puzzle_store.py +132 -0
  372. chia/_tests/wallet/test_sign_coin_spends.py +159 -0
  373. chia/_tests/wallet/test_signer_protocol.py +947 -0
  374. chia/_tests/wallet/test_singleton.py +122 -0
  375. chia/_tests/wallet/test_singleton_lifecycle_fast.py +772 -0
  376. chia/_tests/wallet/test_singleton_store.py +152 -0
  377. chia/_tests/wallet/test_taproot.py +19 -0
  378. chia/_tests/wallet/test_transaction_store.py +945 -0
  379. chia/_tests/wallet/test_util.py +185 -0
  380. chia/_tests/wallet/test_wallet.py +2139 -0
  381. chia/_tests/wallet/test_wallet_action_scope.py +85 -0
  382. chia/_tests/wallet/test_wallet_blockchain.py +111 -0
  383. chia/_tests/wallet/test_wallet_coin_store.py +1002 -0
  384. chia/_tests/wallet/test_wallet_interested_store.py +43 -0
  385. chia/_tests/wallet/test_wallet_key_val_store.py +40 -0
  386. chia/_tests/wallet/test_wallet_node.py +780 -0
  387. chia/_tests/wallet/test_wallet_retry.py +95 -0
  388. chia/_tests/wallet/test_wallet_state_manager.py +259 -0
  389. chia/_tests/wallet/test_wallet_test_framework.py +275 -0
  390. chia/_tests/wallet/test_wallet_trade_store.py +218 -0
  391. chia/_tests/wallet/test_wallet_user_store.py +34 -0
  392. chia/_tests/wallet/test_wallet_utils.py +156 -0
  393. chia/_tests/wallet/vc_wallet/__init__.py +0 -0
  394. chia/_tests/wallet/vc_wallet/config.py +3 -0
  395. chia/_tests/wallet/vc_wallet/test_cr_outer_puzzle.py +70 -0
  396. chia/_tests/wallet/vc_wallet/test_vc_lifecycle.py +883 -0
  397. chia/_tests/wallet/vc_wallet/test_vc_wallet.py +830 -0
  398. chia/_tests/wallet/wallet_block_tools.py +327 -0
  399. chia/_tests/weight_proof/__init__.py +0 -0
  400. chia/_tests/weight_proof/config.py +3 -0
  401. chia/_tests/weight_proof/test_weight_proof.py +528 -0
  402. chia/apis.py +19 -0
  403. chia/clvm/__init__.py +0 -0
  404. chia/cmds/__init__.py +0 -0
  405. chia/cmds/beta.py +184 -0
  406. chia/cmds/beta_funcs.py +137 -0
  407. chia/cmds/check_wallet_db.py +420 -0
  408. chia/cmds/chia.py +151 -0
  409. chia/cmds/cmd_classes.py +323 -0
  410. chia/cmds/cmd_helpers.py +242 -0
  411. chia/cmds/cmds_util.py +488 -0
  412. chia/cmds/coin_funcs.py +275 -0
  413. chia/cmds/coins.py +182 -0
  414. chia/cmds/completion.py +49 -0
  415. chia/cmds/configure.py +332 -0
  416. chia/cmds/dao.py +1064 -0
  417. chia/cmds/dao_funcs.py +598 -0
  418. chia/cmds/data.py +708 -0
  419. chia/cmds/data_funcs.py +385 -0
  420. chia/cmds/db.py +87 -0
  421. chia/cmds/db_backup_func.py +77 -0
  422. chia/cmds/db_upgrade_func.py +452 -0
  423. chia/cmds/db_validate_func.py +184 -0
  424. chia/cmds/dev.py +18 -0
  425. chia/cmds/farm.py +100 -0
  426. chia/cmds/farm_funcs.py +200 -0
  427. chia/cmds/gh.py +275 -0
  428. chia/cmds/init.py +63 -0
  429. chia/cmds/init_funcs.py +367 -0
  430. chia/cmds/installers.py +131 -0
  431. chia/cmds/keys.py +527 -0
  432. chia/cmds/keys_funcs.py +863 -0
  433. chia/cmds/netspace.py +50 -0
  434. chia/cmds/netspace_funcs.py +54 -0
  435. chia/cmds/options.py +32 -0
  436. chia/cmds/param_types.py +238 -0
  437. chia/cmds/passphrase.py +131 -0
  438. chia/cmds/passphrase_funcs.py +292 -0
  439. chia/cmds/peer.py +51 -0
  440. chia/cmds/peer_funcs.py +129 -0
  441. chia/cmds/plotnft.py +260 -0
  442. chia/cmds/plotnft_funcs.py +405 -0
  443. chia/cmds/plots.py +230 -0
  444. chia/cmds/plotters.py +18 -0
  445. chia/cmds/rpc.py +208 -0
  446. chia/cmds/show.py +72 -0
  447. chia/cmds/show_funcs.py +215 -0
  448. chia/cmds/signer.py +296 -0
  449. chia/cmds/sim.py +225 -0
  450. chia/cmds/sim_funcs.py +509 -0
  451. chia/cmds/start.py +24 -0
  452. chia/cmds/start_funcs.py +109 -0
  453. chia/cmds/stop.py +62 -0
  454. chia/cmds/units.py +9 -0
  455. chia/cmds/wallet.py +1901 -0
  456. chia/cmds/wallet_funcs.py +1874 -0
  457. chia/consensus/__init__.py +0 -0
  458. chia/consensus/block_body_validation.py +562 -0
  459. chia/consensus/block_creation.py +546 -0
  460. chia/consensus/block_header_validation.py +1059 -0
  461. chia/consensus/block_record.py +31 -0
  462. chia/consensus/block_rewards.py +53 -0
  463. chia/consensus/blockchain.py +1087 -0
  464. chia/consensus/blockchain_interface.py +56 -0
  465. chia/consensus/coinbase.py +30 -0
  466. chia/consensus/condition_costs.py +9 -0
  467. chia/consensus/constants.py +49 -0
  468. chia/consensus/cost_calculator.py +15 -0
  469. chia/consensus/default_constants.py +89 -0
  470. chia/consensus/deficit.py +55 -0
  471. chia/consensus/difficulty_adjustment.py +412 -0
  472. chia/consensus/find_fork_point.py +111 -0
  473. chia/consensus/full_block_to_block_record.py +167 -0
  474. chia/consensus/get_block_challenge.py +106 -0
  475. chia/consensus/get_block_generator.py +27 -0
  476. chia/consensus/make_sub_epoch_summary.py +210 -0
  477. chia/consensus/multiprocess_validation.py +268 -0
  478. chia/consensus/pos_quality.py +19 -0
  479. chia/consensus/pot_iterations.py +67 -0
  480. chia/consensus/puzzles/__init__.py +0 -0
  481. chia/consensus/puzzles/chialisp_deserialisation.clsp +69 -0
  482. chia/consensus/puzzles/chialisp_deserialisation.clsp.hex +1 -0
  483. chia/consensus/puzzles/rom_bootstrap_generator.clsp +37 -0
  484. chia/consensus/puzzles/rom_bootstrap_generator.clsp.hex +1 -0
  485. chia/consensus/vdf_info_computation.py +156 -0
  486. chia/daemon/__init__.py +0 -0
  487. chia/daemon/client.py +252 -0
  488. chia/daemon/keychain_proxy.py +502 -0
  489. chia/daemon/keychain_server.py +365 -0
  490. chia/daemon/server.py +1606 -0
  491. chia/daemon/windows_signal.py +56 -0
  492. chia/data_layer/__init__.py +0 -0
  493. chia/data_layer/data_layer.py +1291 -0
  494. chia/data_layer/data_layer_api.py +33 -0
  495. chia/data_layer/data_layer_errors.py +50 -0
  496. chia/data_layer/data_layer_server.py +170 -0
  497. chia/data_layer/data_layer_util.py +985 -0
  498. chia/data_layer/data_layer_wallet.py +1311 -0
  499. chia/data_layer/data_store.py +2267 -0
  500. chia/data_layer/dl_wallet_store.py +407 -0
  501. chia/data_layer/download_data.py +389 -0
  502. chia/data_layer/puzzles/__init__.py +0 -0
  503. chia/data_layer/puzzles/graftroot_dl_offers.clsp +100 -0
  504. chia/data_layer/puzzles/graftroot_dl_offers.clsp.hex +1 -0
  505. chia/data_layer/s3_plugin_config.yml +33 -0
  506. chia/data_layer/s3_plugin_service.py +468 -0
  507. chia/data_layer/util/__init__.py +0 -0
  508. chia/data_layer/util/benchmark.py +107 -0
  509. chia/data_layer/util/plugin.py +40 -0
  510. chia/farmer/__init__.py +0 -0
  511. chia/farmer/farmer.py +923 -0
  512. chia/farmer/farmer_api.py +820 -0
  513. chia/full_node/__init__.py +0 -0
  514. chia/full_node/bitcoin_fee_estimator.py +85 -0
  515. chia/full_node/block_height_map.py +271 -0
  516. chia/full_node/block_store.py +576 -0
  517. chia/full_node/bundle_tools.py +19 -0
  518. chia/full_node/coin_store.py +647 -0
  519. chia/full_node/fee_estimate.py +54 -0
  520. chia/full_node/fee_estimate_store.py +24 -0
  521. chia/full_node/fee_estimation.py +92 -0
  522. chia/full_node/fee_estimator.py +90 -0
  523. chia/full_node/fee_estimator_constants.py +38 -0
  524. chia/full_node/fee_estimator_interface.py +42 -0
  525. chia/full_node/fee_history.py +25 -0
  526. chia/full_node/fee_tracker.py +564 -0
  527. chia/full_node/full_node.py +3327 -0
  528. chia/full_node/full_node_api.py +2025 -0
  529. chia/full_node/full_node_store.py +1033 -0
  530. chia/full_node/hint_management.py +56 -0
  531. chia/full_node/hint_store.py +93 -0
  532. chia/full_node/mempool.py +589 -0
  533. chia/full_node/mempool_check_conditions.py +146 -0
  534. chia/full_node/mempool_manager.py +853 -0
  535. chia/full_node/pending_tx_cache.py +112 -0
  536. chia/full_node/puzzles/__init__.py +0 -0
  537. chia/full_node/puzzles/block_program_zero.clsp +14 -0
  538. chia/full_node/puzzles/block_program_zero.clsp.hex +1 -0
  539. chia/full_node/puzzles/decompress_coin_spend_entry.clsp +5 -0
  540. chia/full_node/puzzles/decompress_coin_spend_entry.clsp.hex +1 -0
  541. chia/full_node/puzzles/decompress_coin_spend_entry_with_prefix.clsp +7 -0
  542. chia/full_node/puzzles/decompress_coin_spend_entry_with_prefix.clsp.hex +1 -0
  543. chia/full_node/puzzles/decompress_puzzle.clsp +6 -0
  544. chia/full_node/puzzles/decompress_puzzle.clsp.hex +1 -0
  545. chia/full_node/signage_point.py +16 -0
  546. chia/full_node/subscriptions.py +247 -0
  547. chia/full_node/sync_store.py +146 -0
  548. chia/full_node/tx_processing_queue.py +78 -0
  549. chia/full_node/util/__init__.py +0 -0
  550. chia/full_node/weight_proof.py +1720 -0
  551. chia/harvester/__init__.py +0 -0
  552. chia/harvester/harvester.py +272 -0
  553. chia/harvester/harvester_api.py +380 -0
  554. chia/introducer/__init__.py +0 -0
  555. chia/introducer/introducer.py +122 -0
  556. chia/introducer/introducer_api.py +70 -0
  557. chia/legacy/__init__.py +0 -0
  558. chia/legacy/keyring.py +155 -0
  559. chia/plot_sync/__init__.py +0 -0
  560. chia/plot_sync/delta.py +61 -0
  561. chia/plot_sync/exceptions.py +56 -0
  562. chia/plot_sync/receiver.py +386 -0
  563. chia/plot_sync/sender.py +340 -0
  564. chia/plot_sync/util.py +43 -0
  565. chia/plotters/__init__.py +0 -0
  566. chia/plotters/bladebit.py +388 -0
  567. chia/plotters/chiapos.py +63 -0
  568. chia/plotters/madmax.py +224 -0
  569. chia/plotters/plotters.py +577 -0
  570. chia/plotters/plotters_util.py +133 -0
  571. chia/plotting/__init__.py +0 -0
  572. chia/plotting/cache.py +213 -0
  573. chia/plotting/check_plots.py +283 -0
  574. chia/plotting/create_plots.py +278 -0
  575. chia/plotting/manager.py +436 -0
  576. chia/plotting/util.py +336 -0
  577. chia/pools/__init__.py +0 -0
  578. chia/pools/pool_config.py +110 -0
  579. chia/pools/pool_puzzles.py +459 -0
  580. chia/pools/pool_wallet.py +933 -0
  581. chia/pools/pool_wallet_info.py +118 -0
  582. chia/pools/puzzles/__init__.py +0 -0
  583. chia/pools/puzzles/pool_member_innerpuz.clsp +70 -0
  584. chia/pools/puzzles/pool_member_innerpuz.clsp.hex +1 -0
  585. chia/pools/puzzles/pool_waitingroom_innerpuz.clsp +69 -0
  586. chia/pools/puzzles/pool_waitingroom_innerpuz.clsp.hex +1 -0
  587. chia/protocols/__init__.py +0 -0
  588. chia/protocols/farmer_protocol.py +102 -0
  589. chia/protocols/full_node_protocol.py +219 -0
  590. chia/protocols/harvester_protocol.py +216 -0
  591. chia/protocols/introducer_protocol.py +25 -0
  592. chia/protocols/pool_protocol.py +177 -0
  593. chia/protocols/protocol_message_types.py +139 -0
  594. chia/protocols/protocol_state_machine.py +87 -0
  595. chia/protocols/protocol_timing.py +8 -0
  596. chia/protocols/shared_protocol.py +86 -0
  597. chia/protocols/timelord_protocol.py +93 -0
  598. chia/protocols/wallet_protocol.py +401 -0
  599. chia/py.typed +0 -0
  600. chia/rpc/__init__.py +0 -0
  601. chia/rpc/crawler_rpc_api.py +80 -0
  602. chia/rpc/data_layer_rpc_api.py +644 -0
  603. chia/rpc/data_layer_rpc_client.py +188 -0
  604. chia/rpc/data_layer_rpc_util.py +58 -0
  605. chia/rpc/farmer_rpc_api.py +365 -0
  606. chia/rpc/farmer_rpc_client.py +86 -0
  607. chia/rpc/full_node_rpc_api.py +959 -0
  608. chia/rpc/full_node_rpc_client.py +292 -0
  609. chia/rpc/harvester_rpc_api.py +141 -0
  610. chia/rpc/harvester_rpc_client.py +54 -0
  611. chia/rpc/rpc_client.py +164 -0
  612. chia/rpc/rpc_server.py +521 -0
  613. chia/rpc/timelord_rpc_api.py +32 -0
  614. chia/rpc/util.py +93 -0
  615. chia/rpc/wallet_request_types.py +904 -0
  616. chia/rpc/wallet_rpc_api.py +4943 -0
  617. chia/rpc/wallet_rpc_client.py +1814 -0
  618. chia/seeder/__init__.py +0 -0
  619. chia/seeder/crawl_store.py +425 -0
  620. chia/seeder/crawler.py +410 -0
  621. chia/seeder/crawler_api.py +135 -0
  622. chia/seeder/dns_server.py +593 -0
  623. chia/seeder/peer_record.py +146 -0
  624. chia/seeder/start_crawler.py +92 -0
  625. chia/server/__init__.py +0 -0
  626. chia/server/address_manager.py +658 -0
  627. chia/server/address_manager_store.py +237 -0
  628. chia/server/api_protocol.py +116 -0
  629. chia/server/capabilities.py +24 -0
  630. chia/server/chia_policy.py +346 -0
  631. chia/server/introducer_peers.py +76 -0
  632. chia/server/node_discovery.py +714 -0
  633. chia/server/outbound_message.py +33 -0
  634. chia/server/rate_limit_numbers.py +214 -0
  635. chia/server/rate_limits.py +153 -0
  636. chia/server/server.py +741 -0
  637. chia/server/signal_handlers.py +120 -0
  638. chia/server/ssl_context.py +32 -0
  639. chia/server/start_data_layer.py +151 -0
  640. chia/server/start_farmer.py +98 -0
  641. chia/server/start_full_node.py +112 -0
  642. chia/server/start_harvester.py +93 -0
  643. chia/server/start_introducer.py +81 -0
  644. chia/server/start_service.py +316 -0
  645. chia/server/start_timelord.py +89 -0
  646. chia/server/start_wallet.py +113 -0
  647. chia/server/upnp.py +118 -0
  648. chia/server/ws_connection.py +766 -0
  649. chia/simulator/__init__.py +0 -0
  650. chia/simulator/add_blocks_in_batches.py +54 -0
  651. chia/simulator/block_tools.py +2054 -0
  652. chia/simulator/full_node_simulator.py +794 -0
  653. chia/simulator/keyring.py +128 -0
  654. chia/simulator/setup_services.py +506 -0
  655. chia/simulator/simulator_constants.py +13 -0
  656. chia/simulator/simulator_full_node_rpc_api.py +99 -0
  657. chia/simulator/simulator_full_node_rpc_client.py +60 -0
  658. chia/simulator/simulator_protocol.py +29 -0
  659. chia/simulator/simulator_test_tools.py +164 -0
  660. chia/simulator/socket.py +24 -0
  661. chia/simulator/ssl_certs.py +114 -0
  662. chia/simulator/ssl_certs_1.py +697 -0
  663. chia/simulator/ssl_certs_10.py +697 -0
  664. chia/simulator/ssl_certs_2.py +697 -0
  665. chia/simulator/ssl_certs_3.py +697 -0
  666. chia/simulator/ssl_certs_4.py +697 -0
  667. chia/simulator/ssl_certs_5.py +697 -0
  668. chia/simulator/ssl_certs_6.py +697 -0
  669. chia/simulator/ssl_certs_7.py +697 -0
  670. chia/simulator/ssl_certs_8.py +697 -0
  671. chia/simulator/ssl_certs_9.py +697 -0
  672. chia/simulator/start_simulator.py +143 -0
  673. chia/simulator/wallet_tools.py +246 -0
  674. chia/ssl/__init__.py +0 -0
  675. chia/ssl/chia_ca.crt +19 -0
  676. chia/ssl/chia_ca.key +28 -0
  677. chia/ssl/create_ssl.py +249 -0
  678. chia/ssl/dst_root_ca.pem +20 -0
  679. chia/timelord/__init__.py +0 -0
  680. chia/timelord/iters_from_block.py +50 -0
  681. chia/timelord/timelord.py +1226 -0
  682. chia/timelord/timelord_api.py +138 -0
  683. chia/timelord/timelord_launcher.py +190 -0
  684. chia/timelord/timelord_state.py +244 -0
  685. chia/timelord/types.py +22 -0
  686. chia/types/__init__.py +0 -0
  687. chia/types/aliases.py +35 -0
  688. chia/types/block_protocol.py +20 -0
  689. chia/types/blockchain_format/__init__.py +0 -0
  690. chia/types/blockchain_format/classgroup.py +5 -0
  691. chia/types/blockchain_format/coin.py +28 -0
  692. chia/types/blockchain_format/foliage.py +8 -0
  693. chia/types/blockchain_format/pool_target.py +5 -0
  694. chia/types/blockchain_format/program.py +269 -0
  695. chia/types/blockchain_format/proof_of_space.py +135 -0
  696. chia/types/blockchain_format/reward_chain_block.py +6 -0
  697. chia/types/blockchain_format/serialized_program.py +5 -0
  698. chia/types/blockchain_format/sized_bytes.py +11 -0
  699. chia/types/blockchain_format/slots.py +9 -0
  700. chia/types/blockchain_format/sub_epoch_summary.py +5 -0
  701. chia/types/blockchain_format/tree_hash.py +72 -0
  702. chia/types/blockchain_format/vdf.py +86 -0
  703. chia/types/clvm_cost.py +13 -0
  704. chia/types/coin_record.py +43 -0
  705. chia/types/coin_spend.py +115 -0
  706. chia/types/condition_opcodes.py +73 -0
  707. chia/types/condition_with_args.py +16 -0
  708. chia/types/eligible_coin_spends.py +365 -0
  709. chia/types/end_of_slot_bundle.py +5 -0
  710. chia/types/fee_rate.py +38 -0
  711. chia/types/full_block.py +5 -0
  712. chia/types/generator_types.py +13 -0
  713. chia/types/header_block.py +5 -0
  714. chia/types/internal_mempool_item.py +18 -0
  715. chia/types/mempool_inclusion_status.py +9 -0
  716. chia/types/mempool_item.py +85 -0
  717. chia/types/mempool_submission_status.py +30 -0
  718. chia/types/mojos.py +7 -0
  719. chia/types/peer_info.py +64 -0
  720. chia/types/signing_mode.py +29 -0
  721. chia/types/spend_bundle.py +30 -0
  722. chia/types/spend_bundle_conditions.py +7 -0
  723. chia/types/transaction_queue_entry.py +55 -0
  724. chia/types/unfinished_block.py +5 -0
  725. chia/types/unfinished_header_block.py +37 -0
  726. chia/types/validation_state.py +14 -0
  727. chia/types/weight_proof.py +49 -0
  728. chia/util/__init__.py +0 -0
  729. chia/util/action_scope.py +168 -0
  730. chia/util/async_pool.py +226 -0
  731. chia/util/augmented_chain.py +134 -0
  732. chia/util/batches.py +42 -0
  733. chia/util/bech32m.py +126 -0
  734. chia/util/beta_metrics.py +119 -0
  735. chia/util/block_cache.py +56 -0
  736. chia/util/byte_types.py +12 -0
  737. chia/util/check_fork_next_block.py +33 -0
  738. chia/util/chia_logging.py +144 -0
  739. chia/util/chia_version.py +33 -0
  740. chia/util/collection.py +17 -0
  741. chia/util/condition_tools.py +201 -0
  742. chia/util/config.py +367 -0
  743. chia/util/cpu.py +22 -0
  744. chia/util/db_synchronous.py +23 -0
  745. chia/util/db_version.py +32 -0
  746. chia/util/db_wrapper.py +430 -0
  747. chia/util/default_root.py +27 -0
  748. chia/util/dump_keyring.py +93 -0
  749. chia/util/english.txt +2048 -0
  750. chia/util/errors.py +353 -0
  751. chia/util/file_keyring.py +469 -0
  752. chia/util/files.py +97 -0
  753. chia/util/full_block_utils.py +345 -0
  754. chia/util/generator_tools.py +72 -0
  755. chia/util/hash.py +31 -0
  756. chia/util/initial-config.yaml +694 -0
  757. chia/util/inline_executor.py +26 -0
  758. chia/util/ints.py +19 -0
  759. chia/util/ip_address.py +39 -0
  760. chia/util/json_util.py +37 -0
  761. chia/util/keychain.py +676 -0
  762. chia/util/keyring_wrapper.py +327 -0
  763. chia/util/limited_semaphore.py +41 -0
  764. chia/util/lock.py +49 -0
  765. chia/util/log_exceptions.py +32 -0
  766. chia/util/logging.py +36 -0
  767. chia/util/lru_cache.py +31 -0
  768. chia/util/math.py +20 -0
  769. chia/util/network.py +182 -0
  770. chia/util/paginator.py +48 -0
  771. chia/util/path.py +31 -0
  772. chia/util/permissions.py +20 -0
  773. chia/util/prev_transaction_block.py +21 -0
  774. chia/util/priority_mutex.py +95 -0
  775. chia/util/profiler.py +197 -0
  776. chia/util/recursive_replace.py +24 -0
  777. chia/util/safe_cancel_task.py +16 -0
  778. chia/util/service_groups.py +47 -0
  779. chia/util/setproctitle.py +22 -0
  780. chia/util/significant_bits.py +32 -0
  781. chia/util/ssl_check.py +213 -0
  782. chia/util/streamable.py +642 -0
  783. chia/util/task_referencer.py +59 -0
  784. chia/util/task_timing.py +382 -0
  785. chia/util/timing.py +67 -0
  786. chia/util/vdf_prover.py +30 -0
  787. chia/util/virtual_project_analysis.py +540 -0
  788. chia/util/ws_message.py +66 -0
  789. chia/wallet/__init__.py +0 -0
  790. chia/wallet/cat_wallet/__init__.py +0 -0
  791. chia/wallet/cat_wallet/cat_constants.py +75 -0
  792. chia/wallet/cat_wallet/cat_info.py +47 -0
  793. chia/wallet/cat_wallet/cat_outer_puzzle.py +120 -0
  794. chia/wallet/cat_wallet/cat_utils.py +164 -0
  795. chia/wallet/cat_wallet/cat_wallet.py +855 -0
  796. chia/wallet/cat_wallet/dao_cat_info.py +28 -0
  797. chia/wallet/cat_wallet/dao_cat_wallet.py +669 -0
  798. chia/wallet/cat_wallet/lineage_store.py +74 -0
  799. chia/wallet/cat_wallet/puzzles/__init__.py +0 -0
  800. chia/wallet/cat_wallet/puzzles/cat_truths.clib +31 -0
  801. chia/wallet/cat_wallet/puzzles/cat_v2.clsp +397 -0
  802. chia/wallet/cat_wallet/puzzles/cat_v2.clsp.hex +1 -0
  803. chia/wallet/cat_wallet/puzzles/delegated_tail.clsp +25 -0
  804. chia/wallet/cat_wallet/puzzles/delegated_tail.clsp.hex +1 -0
  805. chia/wallet/cat_wallet/puzzles/everything_with_signature.clsp +15 -0
  806. chia/wallet/cat_wallet/puzzles/everything_with_signature.clsp.hex +1 -0
  807. chia/wallet/cat_wallet/puzzles/genesis_by_coin_id.clsp +26 -0
  808. chia/wallet/cat_wallet/puzzles/genesis_by_coin_id.clsp.hex +1 -0
  809. chia/wallet/cat_wallet/puzzles/genesis_by_coin_id_or_singleton.clsp +42 -0
  810. chia/wallet/cat_wallet/puzzles/genesis_by_coin_id_or_singleton.clsp.hex +1 -0
  811. chia/wallet/cat_wallet/puzzles/genesis_by_puzzle_hash.clsp +24 -0
  812. chia/wallet/cat_wallet/puzzles/genesis_by_puzzle_hash.clsp.hex +1 -0
  813. chia/wallet/coin_selection.py +188 -0
  814. chia/wallet/conditions.py +1512 -0
  815. chia/wallet/dao_wallet/__init__.py +0 -0
  816. chia/wallet/dao_wallet/dao_info.py +61 -0
  817. chia/wallet/dao_wallet/dao_utils.py +811 -0
  818. chia/wallet/dao_wallet/dao_wallet.py +2119 -0
  819. chia/wallet/db_wallet/__init__.py +0 -0
  820. chia/wallet/db_wallet/db_wallet_puzzles.py +111 -0
  821. chia/wallet/derivation_record.py +30 -0
  822. chia/wallet/derive_keys.py +146 -0
  823. chia/wallet/did_wallet/__init__.py +0 -0
  824. chia/wallet/did_wallet/did_info.py +39 -0
  825. chia/wallet/did_wallet/did_wallet.py +1494 -0
  826. chia/wallet/did_wallet/did_wallet_puzzles.py +221 -0
  827. chia/wallet/did_wallet/puzzles/__init__.py +0 -0
  828. chia/wallet/did_wallet/puzzles/did_innerpuz.clsp +135 -0
  829. chia/wallet/did_wallet/puzzles/did_innerpuz.clsp.hex +1 -0
  830. chia/wallet/driver_protocol.py +26 -0
  831. chia/wallet/key_val_store.py +55 -0
  832. chia/wallet/lineage_proof.py +58 -0
  833. chia/wallet/nft_wallet/__init__.py +0 -0
  834. chia/wallet/nft_wallet/metadata_outer_puzzle.py +92 -0
  835. chia/wallet/nft_wallet/nft_info.py +120 -0
  836. chia/wallet/nft_wallet/nft_puzzles.py +305 -0
  837. chia/wallet/nft_wallet/nft_wallet.py +1687 -0
  838. chia/wallet/nft_wallet/ownership_outer_puzzle.py +101 -0
  839. chia/wallet/nft_wallet/puzzles/__init__.py +0 -0
  840. chia/wallet/nft_wallet/puzzles/create_nft_launcher_from_did.clsp +6 -0
  841. chia/wallet/nft_wallet/puzzles/create_nft_launcher_from_did.clsp.hex +1 -0
  842. chia/wallet/nft_wallet/puzzles/nft_intermediate_launcher.clsp +6 -0
  843. chia/wallet/nft_wallet/puzzles/nft_intermediate_launcher.clsp.hex +1 -0
  844. chia/wallet/nft_wallet/puzzles/nft_metadata_updater_default.clsp +30 -0
  845. chia/wallet/nft_wallet/puzzles/nft_metadata_updater_default.clsp.hex +1 -0
  846. chia/wallet/nft_wallet/puzzles/nft_metadata_updater_updateable.clsp +28 -0
  847. chia/wallet/nft_wallet/puzzles/nft_metadata_updater_updateable.clsp.hex +1 -0
  848. chia/wallet/nft_wallet/puzzles/nft_ownership_layer.clsp +100 -0
  849. chia/wallet/nft_wallet/puzzles/nft_ownership_layer.clsp.hex +1 -0
  850. chia/wallet/nft_wallet/puzzles/nft_ownership_transfer_program_one_way_claim_with_royalties.clsp +78 -0
  851. chia/wallet/nft_wallet/puzzles/nft_ownership_transfer_program_one_way_claim_with_royalties.clsp.hex +1 -0
  852. chia/wallet/nft_wallet/puzzles/nft_state_layer.clsp +74 -0
  853. chia/wallet/nft_wallet/puzzles/nft_state_layer.clsp.hex +1 -0
  854. chia/wallet/nft_wallet/singleton_outer_puzzle.py +101 -0
  855. chia/wallet/nft_wallet/transfer_program_puzzle.py +82 -0
  856. chia/wallet/nft_wallet/uncurry_nft.py +217 -0
  857. chia/wallet/notification_manager.py +117 -0
  858. chia/wallet/notification_store.py +178 -0
  859. chia/wallet/outer_puzzles.py +84 -0
  860. chia/wallet/payment.py +33 -0
  861. chia/wallet/puzzle_drivers.py +118 -0
  862. chia/wallet/puzzles/__init__.py +0 -0
  863. chia/wallet/puzzles/augmented_condition.clsp +13 -0
  864. chia/wallet/puzzles/augmented_condition.clsp.hex +1 -0
  865. chia/wallet/puzzles/clawback/__init__.py +0 -0
  866. chia/wallet/puzzles/clawback/drivers.py +188 -0
  867. chia/wallet/puzzles/clawback/metadata.py +38 -0
  868. chia/wallet/puzzles/clawback/puzzle_decorator.py +67 -0
  869. chia/wallet/puzzles/condition_codes.clib +77 -0
  870. chia/wallet/puzzles/curry-and-treehash.clib +102 -0
  871. chia/wallet/puzzles/curry.clib +135 -0
  872. chia/wallet/puzzles/curry_by_index.clib +16 -0
  873. chia/wallet/puzzles/dao_cat_eve.clsp +17 -0
  874. chia/wallet/puzzles/dao_cat_eve.clsp.hex +1 -0
  875. chia/wallet/puzzles/dao_cat_launcher.clsp +36 -0
  876. chia/wallet/puzzles/dao_cat_launcher.clsp.hex +1 -0
  877. chia/wallet/puzzles/dao_finished_state.clsp +35 -0
  878. chia/wallet/puzzles/dao_finished_state.clsp.hex +1 -0
  879. chia/wallet/puzzles/dao_finished_state.clsp.hex.sha256tree +1 -0
  880. chia/wallet/puzzles/dao_lockup.clsp +288 -0
  881. chia/wallet/puzzles/dao_lockup.clsp.hex +1 -0
  882. chia/wallet/puzzles/dao_lockup.clsp.hex.sha256tree +1 -0
  883. chia/wallet/puzzles/dao_proposal.clsp +377 -0
  884. chia/wallet/puzzles/dao_proposal.clsp.hex +1 -0
  885. chia/wallet/puzzles/dao_proposal.clsp.hex.sha256tree +1 -0
  886. chia/wallet/puzzles/dao_proposal_timer.clsp +78 -0
  887. chia/wallet/puzzles/dao_proposal_timer.clsp.hex +1 -0
  888. chia/wallet/puzzles/dao_proposal_timer.clsp.hex.sha256tree +1 -0
  889. chia/wallet/puzzles/dao_proposal_validator.clsp +87 -0
  890. chia/wallet/puzzles/dao_proposal_validator.clsp.hex +1 -0
  891. chia/wallet/puzzles/dao_proposal_validator.clsp.hex.sha256tree +1 -0
  892. chia/wallet/puzzles/dao_spend_p2_singleton_v2.clsp +240 -0
  893. chia/wallet/puzzles/dao_spend_p2_singleton_v2.clsp.hex +1 -0
  894. chia/wallet/puzzles/dao_spend_p2_singleton_v2.clsp.hex.sha256tree +1 -0
  895. chia/wallet/puzzles/dao_treasury.clsp +115 -0
  896. chia/wallet/puzzles/dao_treasury.clsp.hex +1 -0
  897. chia/wallet/puzzles/dao_update_proposal.clsp +44 -0
  898. chia/wallet/puzzles/dao_update_proposal.clsp.hex +1 -0
  899. chia/wallet/puzzles/deployed_puzzle_hashes.json +67 -0
  900. chia/wallet/puzzles/json.clib +25 -0
  901. chia/wallet/puzzles/load_clvm.py +161 -0
  902. chia/wallet/puzzles/merkle_utils.clib +18 -0
  903. chia/wallet/puzzles/notification.clsp +7 -0
  904. chia/wallet/puzzles/notification.clsp.hex +1 -0
  905. chia/wallet/puzzles/p2_1_of_n.clsp +22 -0
  906. chia/wallet/puzzles/p2_1_of_n.clsp.hex +1 -0
  907. chia/wallet/puzzles/p2_conditions.clsp +3 -0
  908. chia/wallet/puzzles/p2_conditions.clsp.hex +1 -0
  909. chia/wallet/puzzles/p2_conditions.py +26 -0
  910. chia/wallet/puzzles/p2_delegated_conditions.clsp +18 -0
  911. chia/wallet/puzzles/p2_delegated_conditions.clsp.hex +1 -0
  912. chia/wallet/puzzles/p2_delegated_conditions.py +21 -0
  913. chia/wallet/puzzles/p2_delegated_puzzle.clsp +19 -0
  914. chia/wallet/puzzles/p2_delegated_puzzle.clsp.hex +1 -0
  915. chia/wallet/puzzles/p2_delegated_puzzle.py +34 -0
  916. chia/wallet/puzzles/p2_delegated_puzzle_or_hidden_puzzle.clsp +91 -0
  917. chia/wallet/puzzles/p2_delegated_puzzle_or_hidden_puzzle.clsp.hex +1 -0
  918. chia/wallet/puzzles/p2_delegated_puzzle_or_hidden_puzzle.py +160 -0
  919. chia/wallet/puzzles/p2_m_of_n_delegate_direct.clsp +108 -0
  920. chia/wallet/puzzles/p2_m_of_n_delegate_direct.clsp.hex +1 -0
  921. chia/wallet/puzzles/p2_m_of_n_delegate_direct.py +21 -0
  922. chia/wallet/puzzles/p2_parent.clsp +19 -0
  923. chia/wallet/puzzles/p2_parent.clsp.hex +1 -0
  924. chia/wallet/puzzles/p2_puzzle_hash.clsp +18 -0
  925. chia/wallet/puzzles/p2_puzzle_hash.clsp.hex +1 -0
  926. chia/wallet/puzzles/p2_puzzle_hash.py +27 -0
  927. chia/wallet/puzzles/p2_singleton.clsp +30 -0
  928. chia/wallet/puzzles/p2_singleton.clsp.hex +1 -0
  929. chia/wallet/puzzles/p2_singleton_aggregator.clsp +81 -0
  930. chia/wallet/puzzles/p2_singleton_aggregator.clsp.hex +1 -0
  931. chia/wallet/puzzles/p2_singleton_or_delayed_puzhash.clsp +50 -0
  932. chia/wallet/puzzles/p2_singleton_or_delayed_puzhash.clsp.hex +1 -0
  933. chia/wallet/puzzles/p2_singleton_via_delegated_puzzle.clsp +47 -0
  934. chia/wallet/puzzles/p2_singleton_via_delegated_puzzle.clsp.hex +1 -0
  935. chia/wallet/puzzles/puzzle_utils.py +34 -0
  936. chia/wallet/puzzles/settlement_payments.clsp +49 -0
  937. chia/wallet/puzzles/settlement_payments.clsp.hex +1 -0
  938. chia/wallet/puzzles/sha256tree.clib +11 -0
  939. chia/wallet/puzzles/singleton_launcher.clsp +16 -0
  940. chia/wallet/puzzles/singleton_launcher.clsp.hex +1 -0
  941. chia/wallet/puzzles/singleton_top_layer.clsp +177 -0
  942. chia/wallet/puzzles/singleton_top_layer.clsp.hex +1 -0
  943. chia/wallet/puzzles/singleton_top_layer.py +296 -0
  944. chia/wallet/puzzles/singleton_top_layer_v1_1.clsp +107 -0
  945. chia/wallet/puzzles/singleton_top_layer_v1_1.clsp.hex +1 -0
  946. chia/wallet/puzzles/singleton_top_layer_v1_1.py +345 -0
  947. chia/wallet/puzzles/singleton_truths.clib +21 -0
  948. chia/wallet/puzzles/tails.py +348 -0
  949. chia/wallet/puzzles/utility_macros.clib +48 -0
  950. chia/wallet/signer_protocol.py +125 -0
  951. chia/wallet/singleton.py +106 -0
  952. chia/wallet/singleton_record.py +30 -0
  953. chia/wallet/trade_manager.py +1102 -0
  954. chia/wallet/trade_record.py +67 -0
  955. chia/wallet/trading/__init__.py +0 -0
  956. chia/wallet/trading/offer.py +702 -0
  957. chia/wallet/trading/trade_status.py +13 -0
  958. chia/wallet/trading/trade_store.py +526 -0
  959. chia/wallet/transaction_record.py +158 -0
  960. chia/wallet/transaction_sorting.py +14 -0
  961. chia/wallet/uncurried_puzzle.py +17 -0
  962. chia/wallet/util/__init__.py +0 -0
  963. chia/wallet/util/address_type.py +55 -0
  964. chia/wallet/util/blind_signer_tl.py +164 -0
  965. chia/wallet/util/clvm_streamable.py +203 -0
  966. chia/wallet/util/compute_hints.py +66 -0
  967. chia/wallet/util/compute_memos.py +43 -0
  968. chia/wallet/util/curry_and_treehash.py +91 -0
  969. chia/wallet/util/debug_spend_bundle.py +232 -0
  970. chia/wallet/util/merkle_tree.py +100 -0
  971. chia/wallet/util/merkle_utils.py +102 -0
  972. chia/wallet/util/new_peak_queue.py +82 -0
  973. chia/wallet/util/notifications.py +12 -0
  974. chia/wallet/util/peer_request_cache.py +174 -0
  975. chia/wallet/util/pprint.py +39 -0
  976. chia/wallet/util/puzzle_compression.py +95 -0
  977. chia/wallet/util/puzzle_decorator.py +100 -0
  978. chia/wallet/util/puzzle_decorator_type.py +7 -0
  979. chia/wallet/util/query_filter.py +59 -0
  980. chia/wallet/util/transaction_type.py +23 -0
  981. chia/wallet/util/tx_config.py +158 -0
  982. chia/wallet/util/wallet_sync_utils.py +351 -0
  983. chia/wallet/util/wallet_types.py +72 -0
  984. chia/wallet/vc_wallet/__init__.py +0 -0
  985. chia/wallet/vc_wallet/cr_cat_drivers.py +664 -0
  986. chia/wallet/vc_wallet/cr_cat_wallet.py +877 -0
  987. chia/wallet/vc_wallet/cr_outer_puzzle.py +102 -0
  988. chia/wallet/vc_wallet/cr_puzzles/__init__.py +0 -0
  989. chia/wallet/vc_wallet/cr_puzzles/conditions_w_fee_announce.clsp +3 -0
  990. chia/wallet/vc_wallet/cr_puzzles/conditions_w_fee_announce.clsp.hex +1 -0
  991. chia/wallet/vc_wallet/cr_puzzles/credential_restriction.clsp +304 -0
  992. chia/wallet/vc_wallet/cr_puzzles/credential_restriction.clsp.hex +1 -0
  993. chia/wallet/vc_wallet/cr_puzzles/flag_proofs_checker.clsp +45 -0
  994. chia/wallet/vc_wallet/cr_puzzles/flag_proofs_checker.clsp.hex +1 -0
  995. chia/wallet/vc_wallet/vc_drivers.py +838 -0
  996. chia/wallet/vc_wallet/vc_puzzles/__init__.py +0 -0
  997. chia/wallet/vc_wallet/vc_puzzles/covenant_layer.clsp +30 -0
  998. chia/wallet/vc_wallet/vc_puzzles/covenant_layer.clsp.hex +1 -0
  999. chia/wallet/vc_wallet/vc_puzzles/eml_covenant_morpher.clsp +75 -0
  1000. chia/wallet/vc_wallet/vc_puzzles/eml_covenant_morpher.clsp.hex +1 -0
  1001. chia/wallet/vc_wallet/vc_puzzles/eml_transfer_program_covenant_adapter.clsp +32 -0
  1002. chia/wallet/vc_wallet/vc_puzzles/eml_transfer_program_covenant_adapter.clsp.hex +1 -0
  1003. chia/wallet/vc_wallet/vc_puzzles/eml_update_metadata_with_DID.clsp +80 -0
  1004. chia/wallet/vc_wallet/vc_puzzles/eml_update_metadata_with_DID.clsp.hex +1 -0
  1005. chia/wallet/vc_wallet/vc_puzzles/exigent_metadata_layer.clsp +163 -0
  1006. chia/wallet/vc_wallet/vc_puzzles/exigent_metadata_layer.clsp.hex +1 -0
  1007. chia/wallet/vc_wallet/vc_puzzles/p2_announced_delegated_puzzle.clsp +16 -0
  1008. chia/wallet/vc_wallet/vc_puzzles/p2_announced_delegated_puzzle.clsp.hex +1 -0
  1009. chia/wallet/vc_wallet/vc_puzzles/standard_vc_backdoor_puzzle.clsp +74 -0
  1010. chia/wallet/vc_wallet/vc_puzzles/standard_vc_backdoor_puzzle.clsp.hex +1 -0
  1011. chia/wallet/vc_wallet/vc_puzzles/std_parent_morpher.clsp +23 -0
  1012. chia/wallet/vc_wallet/vc_puzzles/std_parent_morpher.clsp.hex +1 -0
  1013. chia/wallet/vc_wallet/vc_puzzles/viral_backdoor.clsp +64 -0
  1014. chia/wallet/vc_wallet/vc_puzzles/viral_backdoor.clsp.hex +1 -0
  1015. chia/wallet/vc_wallet/vc_store.py +263 -0
  1016. chia/wallet/vc_wallet/vc_wallet.py +638 -0
  1017. chia/wallet/wallet.py +698 -0
  1018. chia/wallet/wallet_action_scope.py +96 -0
  1019. chia/wallet/wallet_blockchain.py +244 -0
  1020. chia/wallet/wallet_coin_record.py +72 -0
  1021. chia/wallet/wallet_coin_store.py +351 -0
  1022. chia/wallet/wallet_info.py +35 -0
  1023. chia/wallet/wallet_interested_store.py +188 -0
  1024. chia/wallet/wallet_nft_store.py +279 -0
  1025. chia/wallet/wallet_node.py +1765 -0
  1026. chia/wallet/wallet_node_api.py +207 -0
  1027. chia/wallet/wallet_pool_store.py +119 -0
  1028. chia/wallet/wallet_protocol.py +90 -0
  1029. chia/wallet/wallet_puzzle_store.py +396 -0
  1030. chia/wallet/wallet_retry_store.py +70 -0
  1031. chia/wallet/wallet_singleton_store.py +259 -0
  1032. chia/wallet/wallet_spend_bundle.py +25 -0
  1033. chia/wallet/wallet_state_manager.py +2819 -0
  1034. chia/wallet/wallet_transaction_store.py +496 -0
  1035. chia/wallet/wallet_user_store.py +110 -0
  1036. chia/wallet/wallet_weight_proof_handler.py +126 -0
  1037. chia_blockchain-2.5.1rc1.dist-info/LICENSE +201 -0
  1038. chia_blockchain-2.5.1rc1.dist-info/METADATA +156 -0
  1039. chia_blockchain-2.5.1rc1.dist-info/RECORD +1042 -0
  1040. chia_blockchain-2.5.1rc1.dist-info/WHEEL +4 -0
  1041. chia_blockchain-2.5.1rc1.dist-info/entry_points.txt +17 -0
  1042. mozilla-ca/cacert.pem +3611 -0
@@ -0,0 +1,1102 @@
1
+ from __future__ import annotations
2
+
3
+ import dataclasses
4
+ import logging
5
+ import time
6
+ from collections import deque
7
+ from typing import TYPE_CHECKING, Any, Optional, Union
8
+
9
+ from typing_extensions import Literal
10
+
11
+ from chia.data_layer.data_layer_wallet import DataLayerWallet
12
+ from chia.protocols.wallet_protocol import CoinState
13
+ from chia.server.ws_connection import WSChiaConnection
14
+ from chia.types.blockchain_format.coin import Coin, coin_as_list
15
+ from chia.types.blockchain_format.program import Program
16
+ from chia.types.blockchain_format.sized_bytes import bytes32
17
+ from chia.types.spend_bundle import estimate_fees
18
+ from chia.util.db_wrapper import DBWrapper2
19
+ from chia.util.hash import std_hash
20
+ from chia.util.ints import uint32, uint64
21
+ from chia.wallet.cat_wallet.cat_wallet import CATWallet
22
+ from chia.wallet.conditions import (
23
+ AssertCoinAnnouncement,
24
+ Condition,
25
+ ConditionValidTimes,
26
+ CreateCoinAnnouncement,
27
+ parse_conditions_non_consensus,
28
+ parse_timelock_info,
29
+ )
30
+ from chia.wallet.db_wallet.db_wallet_puzzles import ACS_MU_PH
31
+ from chia.wallet.nft_wallet.nft_wallet import NFTWallet
32
+ from chia.wallet.outer_puzzles import AssetType
33
+ from chia.wallet.payment import Payment
34
+ from chia.wallet.puzzle_drivers import PuzzleInfo, Solver
35
+ from chia.wallet.puzzles.load_clvm import load_clvm_maybe_recompile
36
+ from chia.wallet.trade_record import TradeRecord
37
+ from chia.wallet.trading.offer import NotarizedPayment, Offer
38
+ from chia.wallet.trading.trade_status import TradeStatus
39
+ from chia.wallet.trading.trade_store import TradeStore
40
+ from chia.wallet.transaction_record import TransactionRecord
41
+ from chia.wallet.uncurried_puzzle import uncurry_puzzle
42
+ from chia.wallet.util.compute_hints import compute_spend_hints_and_additions
43
+ from chia.wallet.util.query_filter import HashFilter
44
+ from chia.wallet.util.transaction_type import TransactionType
45
+ from chia.wallet.util.wallet_types import WalletType
46
+ from chia.wallet.vc_wallet.cr_cat_drivers import ProofsChecker, construct_pending_approval_state
47
+ from chia.wallet.vc_wallet.vc_wallet import VCWallet
48
+ from chia.wallet.wallet import Wallet
49
+ from chia.wallet.wallet_action_scope import WalletActionScope
50
+ from chia.wallet.wallet_coin_record import WalletCoinRecord
51
+ from chia.wallet.wallet_protocol import WalletProtocol
52
+
53
+ if TYPE_CHECKING:
54
+ from chia.wallet.wallet_state_manager import WalletStateManager
55
+ from chia.wallet.wallet_spend_bundle import WalletSpendBundle
56
+
57
+ OFFER_MOD = load_clvm_maybe_recompile("settlement_payments.clsp")
58
+
59
+
60
+ class TradeManager:
61
+ """
62
+ This class is a driver for creating and accepting settlement_payments.clsp style offers.
63
+
64
+ By default, standard XCH is supported but to support other types of assets you must implement certain functions on
65
+ the asset's wallet as well as create a driver for its puzzle(s). Here is a guide to integrating a new types of
66
+ assets with this trade manager:
67
+
68
+ Puzzle Drivers:
69
+ - See chia/wallet/outer_puzzles.py for a full description of how to build these
70
+ - The `solve` method must be able to be solved by a Solver that looks like this:
71
+ Solver(
72
+ {
73
+ "coin": bytes
74
+ "parent_spend": bytes
75
+ "siblings": list[bytes] # other coins of the same type being offered
76
+ "sibling_spends": list[bytes] # The parent spends for the siblings
77
+ "sibling_puzzles": list[Program] # The inner puzzles of the siblings (always OFFER_MOD)
78
+ "sibling_solutions": list[Program] # The inner solution of the siblings
79
+ }
80
+ )
81
+
82
+ Wallet:
83
+ - Segments in this code that call general wallet methods are highlighted by comments: # ATTENTION: new wallets
84
+ - To be able to be traded, a wallet must implement these methods on itself:
85
+ - generate_signed_transaction(...) -> list[TransactionRecord] (See cat_wallet.py for full API)
86
+ - convert_puzzle_hash(puzzle_hash: bytes32) -> bytes32 # Converts a puzzlehash from outer to inner puzzle
87
+ - get_puzzle_info(asset_id: bytes32) -> PuzzleInfo
88
+ - get_coins_to_offer(asset_id: bytes32, amount: uint64) -> set[Coin]
89
+ - If you would like assets from your wallet to be referenced with just a wallet ID, you must also implement:
90
+ - get_asset_id() -> bytes32
91
+ - Finally, you must make sure that your wallet will respond appropriately when these WSM methods are called:
92
+ - get_wallet_for_puzzle_info(puzzle_info: PuzzleInfo) -> <Your wallet>
93
+ - create_wallet_for_puzzle_info(..., puzzle_info: PuzzleInfo) -> <Your wallet> (See cat_wallet.py for full API)
94
+ - get_wallet_for_asset_id(asset_id: bytes32) -> <Your wallet>
95
+ """
96
+
97
+ wallet_state_manager: WalletStateManager
98
+ log: logging.Logger
99
+ trade_store: TradeStore
100
+ most_recently_deserialized_trade: Optional[tuple[bytes32, Offer]]
101
+
102
+ @staticmethod
103
+ async def create(
104
+ wallet_state_manager: Any,
105
+ db_wrapper: DBWrapper2,
106
+ name: Optional[str] = None,
107
+ ) -> TradeManager:
108
+ self = TradeManager()
109
+ if name:
110
+ self.log = logging.getLogger(name)
111
+ else:
112
+ self.log = logging.getLogger(__name__)
113
+
114
+ self.wallet_state_manager = wallet_state_manager
115
+ self.trade_store = await TradeStore.create(db_wrapper)
116
+ self.most_recently_deserialized_trade = None
117
+ return self
118
+
119
+ async def get_offers_with_status(self, status: TradeStatus) -> list[TradeRecord]:
120
+ records = await self.trade_store.get_trade_record_with_status(status)
121
+ return records
122
+
123
+ async def get_coins_of_interest(
124
+ self,
125
+ ) -> set[bytes32]:
126
+ """
127
+ Returns list of coins we want to check if they are included in filter,
128
+ These will include coins that belong to us and coins that that on other side of treade
129
+ """
130
+ coin_ids = await self.trade_store.get_coin_ids_of_interest_with_trade_statuses(
131
+ trade_statuses=[TradeStatus.PENDING_ACCEPT, TradeStatus.PENDING_CONFIRM, TradeStatus.PENDING_CANCEL]
132
+ )
133
+ return coin_ids
134
+
135
+ async def get_trades_by_coin(self, coin: Coin) -> list[TradeRecord]:
136
+ all_trades = await self.get_all_trades()
137
+ trades_by_coin = []
138
+ for trade in all_trades:
139
+ if trade.status == TradeStatus.CANCELLED.value:
140
+ continue
141
+ if coin in trade.coins_of_interest:
142
+ trades_by_coin.append(trade)
143
+ return trades_by_coin
144
+
145
+ async def coins_of_interest_farmed(
146
+ self, coin_state: CoinState, fork_height: Optional[uint32], peer: WSChiaConnection
147
+ ) -> None:
148
+ """
149
+ If both our coins and other coins in trade got removed that means that trade was successfully executed
150
+ If coins from other side of trade got farmed without ours, that means that trade failed because either someone
151
+ else completed trade or other side of trade canceled the trade by doing a spend.
152
+ If our coins got farmed but coins from other side didn't, we successfully canceled trade by spending inputs.
153
+ """
154
+ self.log.info(f"coins_of_interest_farmed: {coin_state}")
155
+ trades = await self.get_trades_by_coin(coin_state.coin)
156
+ for trade in trades:
157
+ if trade is None:
158
+ self.log.error(f"Coin: {coin_state.coin}, not in any trade")
159
+ continue
160
+ if coin_state.spent_height is None:
161
+ self.log.error(f"Coin: {coin_state.coin}, has not been spent so trade can remain valid")
162
+ # Then let's filter the offer into coins that WE offered
163
+ if (
164
+ self.most_recently_deserialized_trade is not None
165
+ and trade.trade_id == self.most_recently_deserialized_trade[0]
166
+ ):
167
+ offer = self.most_recently_deserialized_trade[1]
168
+ else:
169
+ offer = Offer.from_bytes(trade.offer)
170
+ self.most_recently_deserialized_trade = (trade.trade_id, offer)
171
+ primary_coin_ids = [c.name() for c in offer.removals()]
172
+ # TODO: Add `WalletCoinStore.get_coins`.
173
+ result = await self.wallet_state_manager.coin_store.get_coin_records(
174
+ coin_id_filter=HashFilter.include(primary_coin_ids)
175
+ )
176
+ our_primary_coins: list[Coin] = [cr.coin for cr in result.records]
177
+ our_additions: list[Coin] = list(
178
+ filter(lambda c: offer.get_root_removal(c) in our_primary_coins, offer.additions())
179
+ )
180
+ our_addition_ids: list[bytes32] = [c.name() for c in our_additions]
181
+
182
+ # And get all relevant coin states
183
+ coin_states = await self.wallet_state_manager.wallet_node.get_coin_state(
184
+ our_addition_ids,
185
+ peer=peer,
186
+ fork_height=fork_height,
187
+ )
188
+ assert coin_states is not None
189
+ coin_state_names: list[bytes32] = [cs.coin.name() for cs in coin_states]
190
+ # If any of our settlement_payments were spent, this offer was a success!
191
+ if set(our_addition_ids) == set(coin_state_names):
192
+ height = coin_state.spent_height
193
+ assert height is not None
194
+ await self.trade_store.set_status(trade.trade_id, TradeStatus.CONFIRMED, index=height)
195
+ tx_records: list[TransactionRecord] = await self.calculate_tx_records_for_offer(offer, False)
196
+ for tx in tx_records:
197
+ if TradeStatus(trade.status) == TradeStatus.PENDING_ACCEPT:
198
+ await self.wallet_state_manager.add_transaction(
199
+ dataclasses.replace(tx, confirmed_at_height=height, confirmed=True)
200
+ )
201
+
202
+ self.log.info(f"Trade with id: {trade.trade_id} confirmed at height: {height}")
203
+ else:
204
+ # In any other scenario this trade failed
205
+ await self.wallet_state_manager.delete_trade_transactions(trade.trade_id)
206
+ if trade.status == TradeStatus.PENDING_CANCEL.value:
207
+ await self.trade_store.set_status(trade.trade_id, TradeStatus.CANCELLED)
208
+ self.log.info(f"Trade with id: {trade.trade_id} canceled")
209
+ elif trade.status == TradeStatus.PENDING_CONFIRM.value:
210
+ await self.trade_store.set_status(trade.trade_id, TradeStatus.FAILED)
211
+ self.log.warning(f"Trade with id: {trade.trade_id} failed")
212
+
213
+ async def get_locked_coins(self) -> dict[bytes32, WalletCoinRecord]:
214
+ """Returns a dictionary of confirmed coins that are locked by a trade."""
215
+ all_pending = []
216
+ pending_accept = await self.get_offers_with_status(TradeStatus.PENDING_ACCEPT)
217
+ pending_confirm = await self.get_offers_with_status(TradeStatus.PENDING_CONFIRM)
218
+ pending_cancel = await self.get_offers_with_status(TradeStatus.PENDING_CANCEL)
219
+ all_pending.extend(pending_accept)
220
+ all_pending.extend(pending_confirm)
221
+ all_pending.extend(pending_cancel)
222
+
223
+ coins_of_interest = []
224
+ for trade_offer in all_pending:
225
+ coins_of_interest.extend([c.name() for c in trade_offer.coins_of_interest])
226
+
227
+ # TODO:
228
+ # - No need to get the coin records here, we are only interested in the coin_id on the call site.
229
+ return (
230
+ await self.wallet_state_manager.coin_store.get_coin_records(
231
+ coin_id_filter=HashFilter.include(coins_of_interest)
232
+ )
233
+ ).coin_id_to_record
234
+
235
+ async def get_all_trades(self) -> list[TradeRecord]:
236
+ all: list[TradeRecord] = await self.trade_store.get_all_trades()
237
+ return all
238
+
239
+ async def get_trade_by_id(self, trade_id: bytes32) -> Optional[TradeRecord]:
240
+ record = await self.trade_store.get_trade_record(trade_id)
241
+ return record
242
+
243
+ async def fail_pending_offer(self, trade_id: bytes32) -> None:
244
+ await self.trade_store.set_status(trade_id, TradeStatus.FAILED)
245
+ self.wallet_state_manager.state_changed("offer_failed")
246
+
247
+ async def cancel_pending_offers(
248
+ self,
249
+ trade_ids: list[bytes32],
250
+ action_scope: WalletActionScope,
251
+ fee: uint64 = uint64(0),
252
+ secure: bool = True, # Cancel with a transaction on chain
253
+ trade_cache: dict[bytes32, TradeRecord] = {}, # Optional pre-fetched trade records for optimization
254
+ extra_conditions: tuple[Condition, ...] = tuple(),
255
+ ) -> None:
256
+ """This will create a transaction that includes coins that were offered"""
257
+
258
+ # Need to do some pre-figuring of announcements that will be need to be made
259
+ announcement_nonce: bytes32 = std_hash(b"".join(trade_ids))
260
+ trade_records: list[TradeRecord] = []
261
+ all_cancellation_coins: list[list[Coin]] = []
262
+ announcement_creations: deque[CreateCoinAnnouncement] = deque()
263
+ announcement_assertions: deque[AssertCoinAnnouncement] = deque()
264
+ for trade_id in trade_ids:
265
+ if trade_id in trade_cache:
266
+ trade = trade_cache[trade_id]
267
+ else:
268
+ potential_trade = await self.trade_store.get_trade_record(trade_id)
269
+ if potential_trade is None:
270
+ self.log.error(f"Cannot find offer {trade_id.hex()}, skip cancellation.")
271
+ continue
272
+ else:
273
+ trade = potential_trade
274
+
275
+ cancellation_coins = Offer.from_bytes(trade.offer).get_cancellation_coins()
276
+ for coin in cancellation_coins:
277
+ creation = CreateCoinAnnouncement(msg=announcement_nonce, coin_id=coin.name())
278
+ announcement_creations.append(creation)
279
+ announcement_assertions.append(creation.corresponding_assertion())
280
+
281
+ trade_records.append(trade)
282
+ all_cancellation_coins.append(cancellation_coins)
283
+
284
+ # Make every coin assert the announcement from the one before them
285
+ announcement_assertions.rotate(1)
286
+
287
+ all_txs: list[TransactionRecord] = []
288
+ fee_to_pay: uint64 = fee
289
+ for trade, cancellation_coins in zip(trade_records, all_cancellation_coins):
290
+ self.log.info(f"Secure-Cancel pending offer with id trade_id {trade.trade_id.hex()}")
291
+
292
+ if not secure:
293
+ self.wallet_state_manager.state_changed("offer_cancelled")
294
+ await self.trade_store.set_status(trade.trade_id, TradeStatus.CANCELLED)
295
+ continue
296
+
297
+ cancellation_additions: list[Coin] = []
298
+ valid_times: ConditionValidTimes = parse_timelock_info(extra_conditions)
299
+ trades_to_cancel: list[TradeRecord] = []
300
+ for coin in cancellation_coins:
301
+ wallet = await self.wallet_state_manager.get_wallet_for_coin(coin.name())
302
+
303
+ if wallet is None:
304
+ self.log.error(f"Cannot find wallet for offer {trade.trade_id}, skip cancellation.")
305
+ continue
306
+
307
+ new_ph = await wallet.wallet_state_manager.main_wallet.get_puzzle_hash(
308
+ new=(not action_scope.config.tx_config.reuse_puzhash)
309
+ )
310
+
311
+ if len(trade_records) > 1 or len(cancellation_coins) > 1:
312
+ announcement_conditions: tuple[Condition, ...] = (
313
+ announcement_creations.popleft(),
314
+ announcement_assertions.popleft(),
315
+ )
316
+ else:
317
+ announcement_conditions = tuple()
318
+ async with action_scope.use() as interface:
319
+ interface.side_effects.selected_coins.append(coin)
320
+ # This should probably not switch on whether or not we're spending a XCH but it has to for now
321
+ if wallet.type() == WalletType.STANDARD_WALLET:
322
+ assert isinstance(wallet, Wallet)
323
+ if fee_to_pay > coin.amount:
324
+ selected_coins: set[Coin] = await wallet.select_coins(
325
+ uint64(fee_to_pay - coin.amount),
326
+ action_scope,
327
+ )
328
+ selected_coins.add(coin)
329
+ else:
330
+ selected_coins = {coin}
331
+ async with self.wallet_state_manager.new_action_scope(
332
+ action_scope.config.tx_config.override(
333
+ excluded_coin_ids=[],
334
+ ),
335
+ push=False,
336
+ ) as inner_action_scope:
337
+ await wallet.generate_signed_transaction(
338
+ uint64(sum(c.amount for c in selected_coins) - fee_to_pay),
339
+ new_ph,
340
+ inner_action_scope,
341
+ origin_id=coin.name(),
342
+ fee=fee_to_pay,
343
+ coins=selected_coins,
344
+ extra_conditions=(*extra_conditions, *announcement_conditions),
345
+ )
346
+ else:
347
+ # ATTENTION: new_wallets
348
+ assert isinstance(wallet, (CATWallet, DataLayerWallet, NFTWallet))
349
+ async with self.wallet_state_manager.new_action_scope(
350
+ action_scope.config.tx_config.override(
351
+ excluded_coin_ids=[],
352
+ ),
353
+ push=False,
354
+ ) as inner_action_scope:
355
+ await wallet.generate_signed_transaction(
356
+ [coin.amount],
357
+ [new_ph],
358
+ inner_action_scope,
359
+ fee=fee_to_pay,
360
+ coins={coin},
361
+ extra_conditions=(*extra_conditions, *announcement_conditions),
362
+ )
363
+
364
+ cancellation_additions.extend(
365
+ [
366
+ add
367
+ for tx in inner_action_scope.side_effects.transactions
368
+ if tx.spend_bundle is not None
369
+ for add in tx.spend_bundle.additions()
370
+ ]
371
+ )
372
+ all_txs.extend(inner_action_scope.side_effects.transactions)
373
+ fee_to_pay = uint64(0)
374
+ extra_conditions = tuple()
375
+
376
+ incoming_tx = TransactionRecord(
377
+ confirmed_at_height=uint32(0),
378
+ created_at_time=uint64(int(time.time())),
379
+ to_puzzle_hash=new_ph,
380
+ amount=uint64(coin.amount),
381
+ fee_amount=fee,
382
+ confirmed=False,
383
+ sent=uint32(10),
384
+ spend_bundle=None,
385
+ additions=[],
386
+ removals=[coin],
387
+ wallet_id=wallet.id(),
388
+ sent_to=[],
389
+ trade_id=None,
390
+ type=uint32(TransactionType.INCOMING_TX.value),
391
+ name=cancellation_additions[0].name(),
392
+ memos=[],
393
+ valid_times=valid_times,
394
+ )
395
+ all_txs.append(incoming_tx)
396
+
397
+ # The statuses of trades which offer cancellation coin needs to be set to `PENDING_CANCEL`
398
+ trades_to_cancel.extend(await self.get_trades_by_coin(coin))
399
+
400
+ await self.trade_store.set_status(trade.trade_id, TradeStatus.PENDING_CANCEL)
401
+ self.log.info(f"Cancelling trade: {trade.trade_id}")
402
+ for t in trades_to_cancel:
403
+ await self.trade_store.set_status(t.trade_id, TradeStatus.PENDING_CANCEL)
404
+ self.log.info(f"Cancelling trade: {t.trade_id} along with {trade.trade_id}")
405
+
406
+ if secure:
407
+ async with action_scope.use() as interface:
408
+ # We have to combine the spend bundle for these since they are tied with announcements
409
+ all_tx_names = [tx.name for tx in all_txs]
410
+ interface.side_effects.transactions = [
411
+ tx for tx in interface.side_effects.transactions if tx.name not in all_tx_names
412
+ ]
413
+ final_spend_bundle = WalletSpendBundle.aggregate(
414
+ [tx.spend_bundle for tx in all_txs if tx.spend_bundle is not None]
415
+ )
416
+ interface.side_effects.transactions.append(
417
+ dataclasses.replace(all_txs[0], spend_bundle=final_spend_bundle, name=final_spend_bundle.name())
418
+ )
419
+ interface.side_effects.transactions.extend(
420
+ [dataclasses.replace(tx, spend_bundle=None, fee_amount=fee) for tx in all_txs[1:]]
421
+ )
422
+
423
+ async def save_trade(self, trade: TradeRecord, offer: Offer) -> None:
424
+ offer_name: bytes32 = offer.name()
425
+ await self.trade_store.add_trade_record(trade, offer_name)
426
+
427
+ # We want to subscribe to the coin IDs of all coins that are not the ephemeral offer coins
428
+ offered_coins: set[Coin] = {value for values in offer.get_offered_coins().values() for value in values}
429
+ non_offer_additions: set[Coin] = set(offer.additions()) ^ offered_coins
430
+ non_offer_removals: set[Coin] = set(offer.removals()) ^ offered_coins
431
+ await self.wallet_state_manager.add_interested_coin_ids(
432
+ [coin.name() for coin in (*non_offer_removals, *non_offer_additions)]
433
+ )
434
+
435
+ self.wallet_state_manager.state_changed("offer_added")
436
+
437
+ async def create_offer_for_ids(
438
+ self,
439
+ offer: dict[Union[int, bytes32], int],
440
+ action_scope: WalletActionScope,
441
+ driver_dict: Optional[dict[bytes32, PuzzleInfo]] = None,
442
+ solver: Optional[Solver] = None,
443
+ fee: uint64 = uint64(0),
444
+ validate_only: bool = False,
445
+ extra_conditions: tuple[Condition, ...] = tuple(),
446
+ taking: bool = False,
447
+ ) -> Union[tuple[Literal[True], TradeRecord, None], tuple[Literal[False], None, str]]:
448
+ if driver_dict is None:
449
+ driver_dict = {}
450
+ if solver is None:
451
+ solver = Solver({})
452
+ result = await self._create_offer_for_ids(
453
+ offer,
454
+ action_scope,
455
+ driver_dict,
456
+ solver,
457
+ fee=fee,
458
+ extra_conditions=extra_conditions,
459
+ taking=taking,
460
+ )
461
+ if not result[0] or result[1] is None:
462
+ raise Exception(f"Error creating offer: {result[2]}")
463
+
464
+ success, created_offer, error = result
465
+
466
+ now = uint64(int(time.time()))
467
+ trade_offer: TradeRecord = TradeRecord(
468
+ confirmed_at_index=uint32(0),
469
+ accepted_at_time=None,
470
+ created_at_time=now,
471
+ is_my_offer=True,
472
+ sent=uint32(0),
473
+ offer=bytes(created_offer),
474
+ taken_offer=None,
475
+ coins_of_interest=created_offer.get_involved_coins(),
476
+ trade_id=created_offer.name(),
477
+ status=uint32(TradeStatus.PENDING_ACCEPT.value),
478
+ sent_to=[],
479
+ valid_times=parse_timelock_info(extra_conditions),
480
+ )
481
+
482
+ if success is True and trade_offer is not None and not validate_only:
483
+ await self.save_trade(trade_offer, created_offer)
484
+
485
+ return success, trade_offer, error
486
+
487
+ async def _create_offer_for_ids(
488
+ self,
489
+ offer_dict: dict[Union[int, bytes32], int],
490
+ action_scope: WalletActionScope,
491
+ driver_dict: Optional[dict[bytes32, PuzzleInfo]] = None,
492
+ solver: Optional[Solver] = None,
493
+ fee: uint64 = uint64(0),
494
+ extra_conditions: tuple[Condition, ...] = tuple(),
495
+ taking: bool = False,
496
+ ) -> Union[tuple[Literal[True], Offer, None], tuple[Literal[False], None, str]]:
497
+ """
498
+ Offer is dictionary of wallet ids and amount
499
+ """
500
+ if driver_dict is None:
501
+ driver_dict = {}
502
+ if solver is None:
503
+ solver = Solver({})
504
+ try:
505
+ coins_to_offer: dict[Union[int, bytes32], set[Coin]] = {}
506
+ requested_payments: dict[Optional[bytes32], list[Payment]] = {}
507
+ offer_dict_no_ints: dict[Optional[bytes32], int] = {}
508
+ for id, amount in offer_dict.items():
509
+ asset_id: Optional[bytes32] = None
510
+ # asset_id can either be none if asset is XCH or
511
+ # bytes32 if another asset (e.g. NFT, CAT)
512
+ if amount > 0:
513
+ # this is what we are receiving in the trade
514
+ memos: list[bytes] = []
515
+ if isinstance(id, int):
516
+ wallet_id = uint32(id)
517
+ wallet = self.wallet_state_manager.wallets.get(wallet_id)
518
+ assert isinstance(wallet, (CATWallet, Wallet))
519
+ if isinstance(wallet, Wallet):
520
+ p2_ph: bytes32 = await wallet.get_puzzle_hash(
521
+ new=not action_scope.config.tx_config.reuse_puzhash
522
+ )
523
+ else:
524
+ p2_ph = await wallet.standard_wallet.get_puzzle_hash(
525
+ new=not action_scope.config.tx_config.reuse_puzhash
526
+ )
527
+ if wallet.type() != WalletType.STANDARD_WALLET:
528
+ if callable(getattr(wallet, "get_asset_id", None)): # ATTENTION: new wallets
529
+ assert isinstance(wallet, CATWallet)
530
+ asset_id = bytes32(bytes.fromhex(wallet.get_asset_id()))
531
+ memos = [p2_ph]
532
+ else:
533
+ raise ValueError(
534
+ f"Cannot request assets from wallet id {wallet.id()} without more information"
535
+ )
536
+ else:
537
+ p2_ph = await self.wallet_state_manager.main_wallet.get_puzzle_hash(
538
+ new=not action_scope.config.tx_config.reuse_puzhash
539
+ )
540
+ asset_id = id
541
+ wallet = await self.wallet_state_manager.get_wallet_for_asset_id(asset_id.hex())
542
+ memos = [p2_ph]
543
+ requested_payments[asset_id] = [Payment(p2_ph, uint64(amount), memos)]
544
+ elif amount < 0:
545
+ # this is what we are sending in the trade
546
+ if isinstance(id, int):
547
+ wallet_id = uint32(id)
548
+ wallet = self.wallet_state_manager.wallets[wallet_id]
549
+ if wallet.type() != WalletType.STANDARD_WALLET:
550
+ if callable(getattr(wallet, "get_asset_id", None)): # ATTENTION: new wallets
551
+ assert isinstance(wallet, CATWallet)
552
+ asset_id = bytes32(bytes.fromhex(wallet.get_asset_id()))
553
+ else:
554
+ raise ValueError(
555
+ f"Cannot offer assets from wallet id {wallet.id()} without more information"
556
+ )
557
+ else:
558
+ asset_id = id
559
+ wallet = await self.wallet_state_manager.get_wallet_for_asset_id(asset_id.hex())
560
+ assert wallet is not None
561
+ if not callable(getattr(wallet, "get_coins_to_offer", None)): # ATTENTION: new wallets
562
+ raise ValueError(f"Cannot offer coins from wallet id {wallet.id()}")
563
+ # For the XCH wallet also include the fee amount to the coins we use to pay this offer
564
+ amount_to_select = abs(amount)
565
+ if wallet.type() == WalletType.STANDARD_WALLET:
566
+ amount_to_select += fee
567
+ assert isinstance(wallet, (CATWallet, DataLayerWallet, NFTWallet, Wallet))
568
+ if isinstance(wallet, DataLayerWallet):
569
+ assert asset_id is not None
570
+ coins_to_offer[id] = await wallet.get_coins_to_offer(launcher_id=asset_id)
571
+ elif isinstance(wallet, NFTWallet):
572
+ assert asset_id is not None
573
+ coins_to_offer[id] = await wallet.get_coins_to_offer(nft_id=asset_id)
574
+ else:
575
+ coins_to_offer[id] = await wallet.get_coins_to_offer(
576
+ asset_id=asset_id,
577
+ amount=uint64(amount_to_select),
578
+ action_scope=action_scope,
579
+ )
580
+ # Note: if we use check_for_special_offer_making, this is not used.
581
+ elif amount == 0:
582
+ raise ValueError("You cannot offer nor request 0 amount of something")
583
+
584
+ offer_dict_no_ints[asset_id] = amount
585
+
586
+ if asset_id is not None and wallet is not None: # if this asset is not XCH
587
+ if callable(getattr(wallet, "get_puzzle_info", None)):
588
+ assert isinstance(wallet, (CATWallet, DataLayerWallet, NFTWallet))
589
+ puzzle_driver: PuzzleInfo = await wallet.get_puzzle_info(asset_id)
590
+ if asset_id in driver_dict and driver_dict[asset_id] != puzzle_driver:
591
+ # ignore the case if we're an nft transferring the did owner
592
+ if self.check_for_owner_change_in_drivers(puzzle_driver, driver_dict[asset_id]):
593
+ driver_dict[asset_id] = puzzle_driver
594
+ else:
595
+ raise ValueError(
596
+ f"driver_dict specified {driver_dict[asset_id]}, was expecting {puzzle_driver}"
597
+ )
598
+ else:
599
+ driver_dict[asset_id] = puzzle_driver
600
+ else:
601
+ raise ValueError(f"Wallet for asset id {asset_id} is not properly integrated with TradeManager")
602
+
603
+ requested_payments = await self.check_for_requested_payment_modifications(
604
+ requested_payments, driver_dict, taking
605
+ )
606
+
607
+ potential_special_offer: Optional[Offer] = await self.check_for_special_offer_making(
608
+ offer_dict_no_ints,
609
+ driver_dict,
610
+ action_scope,
611
+ solver,
612
+ fee,
613
+ extra_conditions,
614
+ )
615
+
616
+ if potential_special_offer is not None:
617
+ return True, potential_special_offer, None
618
+
619
+ all_coins: list[Coin] = [c for coins in coins_to_offer.values() for c in coins]
620
+ notarized_payments: dict[Optional[bytes32], list[NotarizedPayment]] = Offer.notarize_payments(
621
+ requested_payments, all_coins
622
+ )
623
+ announcements_to_assert = Offer.calculate_announcements(notarized_payments, driver_dict)
624
+
625
+ all_transactions: list[TransactionRecord] = []
626
+ fee_left_to_pay: uint64 = fee
627
+ # The access of the sorted keys here makes sure we create the XCH transaction first to make sure we pay fee
628
+ # with the XCH side of the offer and don't create an extra fee transaction in other wallets.
629
+ for id in sorted(coins_to_offer.keys(), key=lambda id: id != 1):
630
+ selected_coins = coins_to_offer[id]
631
+ if isinstance(id, int):
632
+ wallet = self.wallet_state_manager.wallets.get(uint32(id))
633
+ else:
634
+ wallet = await self.wallet_state_manager.get_wallet_for_asset_id(id.hex())
635
+ async with self.wallet_state_manager.new_action_scope(
636
+ action_scope.config.tx_config, push=False
637
+ ) as inner_action_scope:
638
+ # This should probably not switch on whether or not we're spending XCH but it has to for now
639
+ assert wallet is not None
640
+ if wallet.type() == WalletType.STANDARD_WALLET:
641
+ assert isinstance(wallet, Wallet)
642
+ await wallet.generate_signed_transaction(
643
+ uint64(abs(offer_dict[id])),
644
+ Offer.ph(),
645
+ inner_action_scope,
646
+ fee=fee_left_to_pay,
647
+ coins=selected_coins,
648
+ extra_conditions=(*extra_conditions, *announcements_to_assert),
649
+ )
650
+ elif wallet.type() == WalletType.NFT:
651
+ assert isinstance(wallet, NFTWallet)
652
+ # This is to generate the tx for specific nft assets, i.e. not using
653
+ # wallet_id as the selector which would select any coins from nft_wallet
654
+ amounts = [coin.amount for coin in selected_coins]
655
+ await wallet.generate_signed_transaction(
656
+ # [abs(offer_dict[id])],
657
+ amounts,
658
+ [Offer.ph()],
659
+ inner_action_scope,
660
+ fee=fee_left_to_pay,
661
+ coins=selected_coins,
662
+ extra_conditions=(*extra_conditions, *announcements_to_assert),
663
+ )
664
+ else:
665
+ # ATTENTION: new_wallets
666
+ assert isinstance(wallet, (CATWallet, DataLayerWallet))
667
+ await wallet.generate_signed_transaction(
668
+ [uint64(abs(offer_dict[id]))],
669
+ [Offer.ph()],
670
+ inner_action_scope,
671
+ fee=fee_left_to_pay,
672
+ coins=selected_coins,
673
+ extra_conditions=(*extra_conditions, *announcements_to_assert),
674
+ add_authorizations_to_cr_cats=False,
675
+ )
676
+
677
+ all_transactions.extend(inner_action_scope.side_effects.transactions)
678
+
679
+ fee_left_to_pay = uint64(0)
680
+ extra_conditions = tuple()
681
+
682
+ async with action_scope.use() as interface:
683
+ interface.side_effects.transactions.extend(all_transactions)
684
+
685
+ total_spend_bundle = WalletSpendBundle.aggregate(
686
+ [x.spend_bundle for x in all_transactions if x.spend_bundle is not None]
687
+ )
688
+
689
+ offer = Offer(notarized_payments, total_spend_bundle, driver_dict)
690
+ return True, offer, None
691
+
692
+ except Exception as e:
693
+ self.log.exception("Error creating trade offer")
694
+ return False, None, str(e)
695
+
696
+ async def maybe_create_wallets_for_offer(self, offer: Offer) -> None:
697
+ for key in offer.arbitrage():
698
+ wsm = self.wallet_state_manager
699
+ if key is None:
700
+ continue
701
+ # ATTENTION: new_wallets
702
+ exists = await wsm.get_wallet_for_puzzle_info(offer.driver_dict[key])
703
+ if exists is None:
704
+ await wsm.create_wallet_for_puzzle_info(offer.driver_dict[key])
705
+
706
+ async def check_offer_validity(self, offer: Offer, peer: WSChiaConnection) -> bool:
707
+ all_removals: list[Coin] = offer.removals()
708
+ all_removal_names: list[bytes32] = [c.name() for c in all_removals]
709
+ non_ephemeral_removals: list[Coin] = list(
710
+ filter(lambda c: c.parent_coin_info not in all_removal_names, all_removals)
711
+ )
712
+ coin_states = await self.wallet_state_manager.wallet_node.get_coin_state(
713
+ [c.name() for c in non_ephemeral_removals], peer=peer
714
+ )
715
+
716
+ return len(coin_states) == len(non_ephemeral_removals) and all([cs.spent_height is None for cs in coin_states])
717
+
718
+ async def calculate_tx_records_for_offer(self, offer: Offer, validate: bool) -> list[TransactionRecord]:
719
+ if validate:
720
+ final_spend_bundle: WalletSpendBundle = offer.to_valid_spend()
721
+ hint_dict: dict[bytes32, bytes32] = {}
722
+ additions_dict: dict[bytes32, Coin] = {}
723
+ for hinted_coins, _ in (
724
+ compute_spend_hints_and_additions(spend) for spend in final_spend_bundle.coin_spends
725
+ ):
726
+ hint_dict.update({id: hc.hint for id, hc in hinted_coins.items() if hc.hint is not None})
727
+ additions_dict.update({id: hc.coin for id, hc in hinted_coins.items()})
728
+ all_additions: list[Coin] = list(a for a in additions_dict.values())
729
+ else:
730
+ final_spend_bundle = offer._bundle
731
+ hint_dict = offer.hints()
732
+ all_additions = offer.additions()
733
+
734
+ settlement_coins: list[Coin] = [c for coins in offer.get_offered_coins().values() for c in coins]
735
+ settlement_coin_ids: list[bytes32] = [c.name() for c in settlement_coins]
736
+
737
+ removals: list[Coin] = final_spend_bundle.removals()
738
+ additions: list[Coin] = list(a for a in all_additions if a not in removals)
739
+ valid_times: ConditionValidTimes = parse_timelock_info(
740
+ parse_conditions_non_consensus(
741
+ condition
742
+ for spend in final_spend_bundle.coin_spends
743
+ for condition in spend.puzzle_reveal.to_program().run(spend.solution.to_program()).as_iter()
744
+ )
745
+ )
746
+ # this executes the puzzles again
747
+ all_fees = uint64(estimate_fees(final_spend_bundle))
748
+
749
+ txs = []
750
+
751
+ addition_dict: dict[uint32, list[Coin]] = {}
752
+ for addition in additions:
753
+ wallet_identifier = await self.wallet_state_manager.get_wallet_identifier_for_coin(
754
+ addition,
755
+ hint_dict,
756
+ )
757
+ if wallet_identifier is not None:
758
+ if addition.parent_coin_info in settlement_coin_ids:
759
+ wallet = self.wallet_state_manager.wallets[wallet_identifier.id]
760
+ assert isinstance(wallet, (CATWallet, NFTWallet, Wallet))
761
+ to_puzzle_hash = await wallet.convert_puzzle_hash(addition.puzzle_hash) # ATTENTION: new wallets
762
+ txs.append(
763
+ TransactionRecord(
764
+ confirmed_at_height=uint32(0),
765
+ created_at_time=uint64(int(time.time())),
766
+ to_puzzle_hash=to_puzzle_hash,
767
+ amount=uint64(addition.amount),
768
+ fee_amount=uint64(0),
769
+ confirmed=False,
770
+ sent=uint32(10),
771
+ spend_bundle=None,
772
+ additions=[addition],
773
+ removals=[],
774
+ wallet_id=wallet_identifier.id,
775
+ sent_to=[],
776
+ trade_id=offer.name(),
777
+ type=uint32(TransactionType.INCOMING_TRADE.value),
778
+ name=std_hash(final_spend_bundle.name() + addition.name()),
779
+ memos=[(coin_id, [hint]) for coin_id, hint in hint_dict.items()],
780
+ valid_times=valid_times,
781
+ )
782
+ )
783
+ else: # This is change
784
+ addition_dict.setdefault(wallet_identifier.id, [])
785
+ addition_dict[wallet_identifier.id].append(addition)
786
+
787
+ # While we want additions to show up as separate records, removals of the same wallet should show as one
788
+ removal_dict: dict[uint32, list[Coin]] = {}
789
+ for removal in removals:
790
+ wallet_identifier = await self.wallet_state_manager.get_wallet_identifier_for_coin(
791
+ removal,
792
+ hint_dict,
793
+ )
794
+ if wallet_identifier is not None:
795
+ removal_dict.setdefault(wallet_identifier.id, [])
796
+ removal_dict[wallet_identifier.id].append(removal)
797
+
798
+ all_removals: list[bytes32] = [r.name() for removals in removal_dict.values() for r in removals]
799
+
800
+ for wid, grouped_removals in removal_dict.items():
801
+ wallet = self.wallet_state_manager.wallets[wid]
802
+ to_puzzle_hash = bytes32([1] * 32) # We use all zeros to be clear not to send here
803
+ removal_tree_hash = Program.to([coin_as_list(rem) for rem in grouped_removals]).get_tree_hash()
804
+ # We also need to calculate the sent amount
805
+ removed: int = sum(c.amount for c in grouped_removals)
806
+ removed_ids: list[bytes32] = [c.name() for c in grouped_removals]
807
+ all_additions_from_grouped_removals: list[Coin] = [
808
+ c for c in all_additions if c.parent_coin_info in removed_ids
809
+ ]
810
+ potential_change_coins: list[Coin] = addition_dict[wid] if wid in addition_dict else []
811
+ change_coins: list[Coin] = [c for c in potential_change_coins if c.parent_coin_info in all_removals]
812
+ change_amount: int = sum(c.amount for c in change_coins)
813
+ sent_amount: int = (
814
+ removed
815
+ - change_amount
816
+ - (
817
+ removed - sum(c.amount for c in all_additions_from_grouped_removals) # removals - additions == fees
818
+ if wallet == self.wallet_state_manager.main_wallet
819
+ else 0
820
+ )
821
+ )
822
+ txs.append(
823
+ TransactionRecord(
824
+ confirmed_at_height=uint32(0),
825
+ created_at_time=uint64(int(time.time())),
826
+ to_puzzle_hash=to_puzzle_hash,
827
+ amount=uint64(sent_amount),
828
+ fee_amount=all_fees,
829
+ confirmed=False,
830
+ sent=uint32(10),
831
+ spend_bundle=None,
832
+ additions=change_coins,
833
+ removals=grouped_removals,
834
+ wallet_id=wallet.id(),
835
+ sent_to=[],
836
+ trade_id=offer.name(),
837
+ type=uint32(TransactionType.OUTGOING_TRADE.value),
838
+ name=std_hash(final_spend_bundle.name() + removal_tree_hash),
839
+ memos=[(coin_id, [hint]) for coin_id, hint in hint_dict.items()],
840
+ valid_times=valid_times,
841
+ )
842
+ )
843
+
844
+ return txs
845
+
846
+ async def respond_to_offer(
847
+ self,
848
+ offer: Offer,
849
+ peer: WSChiaConnection,
850
+ action_scope: WalletActionScope,
851
+ solver: Optional[Solver] = None,
852
+ fee: uint64 = uint64(0),
853
+ extra_conditions: tuple[Condition, ...] = tuple(),
854
+ ) -> TradeRecord:
855
+ if solver is None:
856
+ solver = Solver({})
857
+ take_offer_dict: dict[Union[bytes32, int], int] = {}
858
+ arbitrage: dict[Optional[bytes32], int] = offer.arbitrage()
859
+
860
+ for asset_id, amount in arbitrage.items():
861
+ if asset_id is None:
862
+ wallet: Optional[WalletProtocol[Any]] = self.wallet_state_manager.main_wallet
863
+ assert wallet is not None
864
+ key: Union[bytes32, int] = int(wallet.id())
865
+ else:
866
+ # ATTENTION: new wallets
867
+ wallet = await self.wallet_state_manager.get_wallet_for_asset_id(asset_id.hex())
868
+ if wallet is None and amount < 0:
869
+ raise ValueError(f"Do not have a wallet for asset ID: {asset_id} to fulfill offer")
870
+ elif wallet is None or wallet.type() in {WalletType.NFT, WalletType.DATA_LAYER}:
871
+ key = asset_id
872
+ else:
873
+ key = int(wallet.id())
874
+ take_offer_dict[key] = amount
875
+
876
+ # First we validate that all of the coins in this offer exist
877
+ valid: bool = await self.check_offer_validity(offer, peer)
878
+ if not valid:
879
+ raise ValueError("This offer is no longer valid")
880
+ # We need to sandbox the transactions here because we're going to make our own
881
+ async with self.wallet_state_manager.new_action_scope(
882
+ action_scope.config.tx_config, push=False
883
+ ) as inner_action_scope:
884
+ result = await self._create_offer_for_ids(
885
+ take_offer_dict,
886
+ inner_action_scope,
887
+ offer.driver_dict,
888
+ solver,
889
+ fee=fee,
890
+ extra_conditions=extra_conditions,
891
+ taking=True,
892
+ )
893
+ if not result[0] or result[1] is None:
894
+ raise ValueError(result[2])
895
+
896
+ _success, take_offer, _error = result
897
+
898
+ complete_offer, valid_spend_solver = await self.check_for_final_modifications(
899
+ Offer.aggregate([offer, take_offer]), solver, inner_action_scope
900
+ )
901
+ self.log.info("COMPLETE OFFER: %s", complete_offer.to_bech32())
902
+ assert complete_offer.is_valid()
903
+ final_spend_bundle: WalletSpendBundle = complete_offer.to_valid_spend(
904
+ solver=Solver({**valid_spend_solver.info, **solver.info})
905
+ )
906
+ await self.maybe_create_wallets_for_offer(complete_offer)
907
+
908
+ tx_records: list[TransactionRecord] = await self.calculate_tx_records_for_offer(complete_offer, True)
909
+
910
+ trade_record: TradeRecord = TradeRecord(
911
+ confirmed_at_index=uint32(0),
912
+ accepted_at_time=uint64(int(time.time())),
913
+ created_at_time=uint64(int(time.time())),
914
+ is_my_offer=False,
915
+ sent=uint32(0),
916
+ offer=bytes(complete_offer),
917
+ taken_offer=bytes(offer),
918
+ coins_of_interest=complete_offer.get_involved_coins(),
919
+ trade_id=complete_offer.name(),
920
+ status=uint32(TradeStatus.PENDING_CONFIRM.value),
921
+ sent_to=[],
922
+ valid_times=parse_timelock_info(extra_conditions),
923
+ )
924
+
925
+ await self.save_trade(trade_record, offer)
926
+
927
+ async with action_scope.use() as interface:
928
+ interface.side_effects.transactions.extend(tx_records)
929
+ interface.side_effects.extra_spends.append(final_spend_bundle)
930
+
931
+ return trade_record
932
+
933
+ async def check_for_special_offer_making(
934
+ self,
935
+ offer_dict: dict[Optional[bytes32], int],
936
+ driver_dict: dict[bytes32, PuzzleInfo],
937
+ action_scope: WalletActionScope,
938
+ solver: Solver,
939
+ fee: uint64 = uint64(0),
940
+ extra_conditions: tuple[Condition, ...] = tuple(),
941
+ ) -> Optional[Offer]:
942
+ for puzzle_info in driver_dict.values():
943
+ if (
944
+ puzzle_info.check_type([AssetType.SINGLETON.value, AssetType.METADATA.value, AssetType.OWNERSHIP.value])
945
+ and isinstance(puzzle_info.also().also()["transfer_program"], PuzzleInfo) # type: ignore
946
+ and puzzle_info.also().also()["transfer_program"].type() # type: ignore
947
+ == AssetType.ROYALTY_TRANSFER_PROGRAM.value
948
+ ):
949
+ return await NFTWallet.make_nft1_offer(
950
+ self.wallet_state_manager, offer_dict, driver_dict, action_scope, fee, extra_conditions
951
+ )
952
+ elif (
953
+ puzzle_info.check_type(
954
+ [
955
+ AssetType.SINGLETON.value,
956
+ AssetType.METADATA.value,
957
+ ]
958
+ )
959
+ and puzzle_info.also()["updater_hash"] == ACS_MU_PH # type: ignore
960
+ ):
961
+ return await DataLayerWallet.make_update_offer(
962
+ self.wallet_state_manager,
963
+ offer_dict,
964
+ driver_dict,
965
+ solver,
966
+ action_scope,
967
+ fee,
968
+ extra_conditions,
969
+ )
970
+ return None
971
+
972
+ def check_for_owner_change_in_drivers(self, puzzle_info: PuzzleInfo, driver_info: PuzzleInfo) -> bool:
973
+ if puzzle_info.check_type(
974
+ [
975
+ AssetType.SINGLETON.value,
976
+ AssetType.METADATA.value,
977
+ AssetType.OWNERSHIP.value,
978
+ ]
979
+ ) and driver_info.check_type(
980
+ [
981
+ AssetType.SINGLETON.value,
982
+ AssetType.METADATA.value,
983
+ AssetType.OWNERSHIP.value,
984
+ ]
985
+ ):
986
+ old_owner = driver_info.also().also().info["owner"] # type: ignore
987
+ puzzle_info.also().also().info["owner"] = old_owner # type: ignore
988
+ if driver_info == puzzle_info:
989
+ return True
990
+ return False
991
+
992
+ async def get_offer_summary(self, offer: Offer) -> dict[str, Any]:
993
+ for puzzle_info in offer.driver_dict.values():
994
+ if (
995
+ puzzle_info.check_type(
996
+ [
997
+ AssetType.SINGLETON.value,
998
+ AssetType.METADATA.value,
999
+ ]
1000
+ )
1001
+ and puzzle_info.also()["updater_hash"] == ACS_MU_PH # type: ignore
1002
+ ):
1003
+ return await DataLayerWallet.get_offer_summary(offer)
1004
+ # Otherwise just return the same thing as the RPC normally does
1005
+ offered, requested, infos, valid_times = offer.summary()
1006
+ return {
1007
+ "offered": offered,
1008
+ "requested": requested,
1009
+ "fees": offer.fees(),
1010
+ "additions": [c.name().hex() for c in offer.additions()],
1011
+ "removals": [c.name().hex() for c in offer.removals()],
1012
+ "infos": infos,
1013
+ "valid_times": {
1014
+ k: v
1015
+ for k, v in valid_times.to_json_dict().items()
1016
+ if k
1017
+ not in {
1018
+ "max_secs_after_created",
1019
+ "min_secs_since_created",
1020
+ "max_blocks_after_created",
1021
+ "min_blocks_since_created",
1022
+ }
1023
+ },
1024
+ }
1025
+
1026
+ async def check_for_final_modifications(
1027
+ self, offer: Offer, solver: Solver, action_scope: WalletActionScope
1028
+ ) -> tuple[Offer, Solver]:
1029
+ for puzzle_info in offer.driver_dict.values():
1030
+ if (
1031
+ puzzle_info.check_type(
1032
+ [
1033
+ AssetType.SINGLETON.value,
1034
+ AssetType.METADATA.value,
1035
+ ]
1036
+ )
1037
+ and puzzle_info.also()["updater_hash"] == ACS_MU_PH # type: ignore
1038
+ ):
1039
+ return (await DataLayerWallet.finish_graftroot_solutions(offer, solver), Solver({}))
1040
+ elif puzzle_info.check_type(
1041
+ [
1042
+ AssetType.CAT.value,
1043
+ AssetType.CR.value,
1044
+ ]
1045
+ ):
1046
+ # get VC wallet
1047
+ for _, wallet in self.wallet_state_manager.wallets.items():
1048
+ if WalletType(wallet.type()) == WalletType.VC:
1049
+ assert isinstance(wallet, VCWallet)
1050
+ return await wallet.add_vc_authorization(offer, solver, action_scope)
1051
+ raise ValueError("No VCs to approve CR-CATs with") # pragma: no cover
1052
+
1053
+ return offer, Solver({})
1054
+
1055
+ async def check_for_requested_payment_modifications(
1056
+ self,
1057
+ requested_payments: dict[Optional[bytes32], list[Payment]],
1058
+ driver_dict: dict[bytes32, PuzzleInfo],
1059
+ taking: bool,
1060
+ ) -> dict[Optional[bytes32], list[Payment]]:
1061
+ # This function exclusively deals with CR-CATs for now
1062
+ if not taking:
1063
+ for asset_id, puzzle_info in driver_dict.items():
1064
+ if puzzle_info.check_type(
1065
+ [
1066
+ AssetType.CAT.value,
1067
+ AssetType.CR.value,
1068
+ ]
1069
+ ):
1070
+ vc = await (
1071
+ await self.wallet_state_manager.get_or_create_vc_wallet()
1072
+ ).get_vc_with_provider_in_and_proofs(
1073
+ puzzle_info["also"]["authorized_providers"],
1074
+ ProofsChecker.from_program(uncurry_puzzle(puzzle_info["also"]["proofs_checker"])).flags,
1075
+ )
1076
+ if vc is None:
1077
+ raise ValueError("Cannot request CR-CATs that you cannot approve with a VC") # pragma: no cover
1078
+
1079
+ return {
1080
+ asset_id: (
1081
+ [
1082
+ dataclasses.replace(
1083
+ payment,
1084
+ puzzle_hash=construct_pending_approval_state(
1085
+ payment.puzzle_hash, payment.amount
1086
+ ).get_tree_hash(),
1087
+ )
1088
+ for payment in payments
1089
+ ]
1090
+ if asset_id is not None
1091
+ and driver_dict[asset_id].check_type(
1092
+ [
1093
+ AssetType.CAT.value,
1094
+ AssetType.CR.value,
1095
+ ]
1096
+ )
1097
+ else payments
1098
+ )
1099
+ for asset_id, payments in requested_payments.items()
1100
+ }
1101
+ else:
1102
+ return requested_payments