chia-blockchain 2.4.4__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (1028) hide show
  1. chia/__init__.py +10 -0
  2. chia/__main__.py +5 -0
  3. chia/_tests/README.md +53 -0
  4. chia/_tests/__init__.py +0 -0
  5. chia/_tests/blockchain/__init__.py +0 -0
  6. chia/_tests/blockchain/blockchain_test_utils.py +197 -0
  7. chia/_tests/blockchain/config.py +4 -0
  8. chia/_tests/blockchain/test_augmented_chain.py +147 -0
  9. chia/_tests/blockchain/test_blockchain.py +4100 -0
  10. chia/_tests/blockchain/test_blockchain_transactions.py +1050 -0
  11. chia/_tests/blockchain/test_build_chains.py +61 -0
  12. chia/_tests/blockchain/test_get_block_generator.py +72 -0
  13. chia/_tests/blockchain/test_lookup_fork_chain.py +195 -0
  14. chia/_tests/build-init-files.py +93 -0
  15. chia/_tests/build-job-matrix.py +204 -0
  16. chia/_tests/check_pytest_monitor_output.py +34 -0
  17. chia/_tests/check_sql_statements.py +73 -0
  18. chia/_tests/chia-start-sim +42 -0
  19. chia/_tests/clvm/__init__.py +0 -0
  20. chia/_tests/clvm/benchmark_costs.py +23 -0
  21. chia/_tests/clvm/coin_store.py +147 -0
  22. chia/_tests/clvm/test_chialisp_deserialization.py +101 -0
  23. chia/_tests/clvm/test_clvm_step.py +37 -0
  24. chia/_tests/clvm/test_condition_codes.py +13 -0
  25. chia/_tests/clvm/test_curry_and_treehash.py +57 -0
  26. chia/_tests/clvm/test_program.py +150 -0
  27. chia/_tests/clvm/test_puzzle_compression.py +144 -0
  28. chia/_tests/clvm/test_puzzle_drivers.py +45 -0
  29. chia/_tests/clvm/test_puzzles.py +247 -0
  30. chia/_tests/clvm/test_singletons.py +540 -0
  31. chia/_tests/clvm/test_spend_sim.py +181 -0
  32. chia/_tests/cmds/__init__.py +0 -0
  33. chia/_tests/cmds/cmd_test_utils.py +472 -0
  34. chia/_tests/cmds/config.py +3 -0
  35. chia/_tests/cmds/conftest.py +23 -0
  36. chia/_tests/cmds/test_click_types.py +195 -0
  37. chia/_tests/cmds/test_cmd_framework.py +400 -0
  38. chia/_tests/cmds/test_cmds_util.py +97 -0
  39. chia/_tests/cmds/test_daemon.py +92 -0
  40. chia/_tests/cmds/test_farm_cmd.py +67 -0
  41. chia/_tests/cmds/test_show.py +116 -0
  42. chia/_tests/cmds/test_sim.py +207 -0
  43. chia/_tests/cmds/test_timelock_args.py +75 -0
  44. chia/_tests/cmds/test_tx_config_args.py +153 -0
  45. chia/_tests/cmds/testing_classes.py +59 -0
  46. chia/_tests/cmds/wallet/__init__.py +0 -0
  47. chia/_tests/cmds/wallet/test_coins.py +195 -0
  48. chia/_tests/cmds/wallet/test_consts.py +47 -0
  49. chia/_tests/cmds/wallet/test_dao.py +565 -0
  50. chia/_tests/cmds/wallet/test_did.py +403 -0
  51. chia/_tests/cmds/wallet/test_nft.py +470 -0
  52. chia/_tests/cmds/wallet/test_notifications.py +124 -0
  53. chia/_tests/cmds/wallet/test_offer.toffer +1 -0
  54. chia/_tests/cmds/wallet/test_tx_decorators.py +27 -0
  55. chia/_tests/cmds/wallet/test_vcs.py +376 -0
  56. chia/_tests/cmds/wallet/test_wallet.py +1126 -0
  57. chia/_tests/cmds/wallet/test_wallet_check.py +111 -0
  58. chia/_tests/conftest.py +1304 -0
  59. chia/_tests/connection_utils.py +124 -0
  60. chia/_tests/core/__init__.py +0 -0
  61. chia/_tests/core/cmds/__init__.py +0 -0
  62. chia/_tests/core/cmds/test_beta.py +382 -0
  63. chia/_tests/core/cmds/test_keys.py +1734 -0
  64. chia/_tests/core/cmds/test_wallet.py +126 -0
  65. chia/_tests/core/config.py +3 -0
  66. chia/_tests/core/consensus/__init__.py +0 -0
  67. chia/_tests/core/consensus/test_block_creation.py +56 -0
  68. chia/_tests/core/consensus/test_pot_iterations.py +117 -0
  69. chia/_tests/core/custom_types/__init__.py +0 -0
  70. chia/_tests/core/custom_types/test_coin.py +109 -0
  71. chia/_tests/core/custom_types/test_proof_of_space.py +144 -0
  72. chia/_tests/core/custom_types/test_spend_bundle.py +71 -0
  73. chia/_tests/core/daemon/__init__.py +0 -0
  74. chia/_tests/core/daemon/config.py +4 -0
  75. chia/_tests/core/daemon/test_daemon.py +2128 -0
  76. chia/_tests/core/daemon/test_daemon_register.py +109 -0
  77. chia/_tests/core/daemon/test_keychain_proxy.py +100 -0
  78. chia/_tests/core/data_layer/__init__.py +0 -0
  79. chia/_tests/core/data_layer/config.py +5 -0
  80. chia/_tests/core/data_layer/conftest.py +105 -0
  81. chia/_tests/core/data_layer/test_data_cli.py +57 -0
  82. chia/_tests/core/data_layer/test_data_layer.py +83 -0
  83. chia/_tests/core/data_layer/test_data_layer_util.py +219 -0
  84. chia/_tests/core/data_layer/test_data_rpc.py +3865 -0
  85. chia/_tests/core/data_layer/test_data_store.py +2423 -0
  86. chia/_tests/core/data_layer/test_data_store_schema.py +381 -0
  87. chia/_tests/core/data_layer/test_plugin.py +91 -0
  88. chia/_tests/core/data_layer/util.py +232 -0
  89. chia/_tests/core/farmer/__init__.py +0 -0
  90. chia/_tests/core/farmer/config.py +3 -0
  91. chia/_tests/core/farmer/test_farmer_api.py +101 -0
  92. chia/_tests/core/full_node/__init__.py +0 -0
  93. chia/_tests/core/full_node/config.py +4 -0
  94. chia/_tests/core/full_node/dos/__init__.py +0 -0
  95. chia/_tests/core/full_node/dos/config.py +3 -0
  96. chia/_tests/core/full_node/full_sync/__init__.py +0 -0
  97. chia/_tests/core/full_node/full_sync/config.py +4 -0
  98. chia/_tests/core/full_node/full_sync/test_full_sync.py +448 -0
  99. chia/_tests/core/full_node/ram_db.py +27 -0
  100. chia/_tests/core/full_node/stores/__init__.py +0 -0
  101. chia/_tests/core/full_node/stores/config.py +4 -0
  102. chia/_tests/core/full_node/stores/test_block_store.py +488 -0
  103. chia/_tests/core/full_node/stores/test_coin_store.py +888 -0
  104. chia/_tests/core/full_node/stores/test_full_node_store.py +1215 -0
  105. chia/_tests/core/full_node/stores/test_hint_store.py +230 -0
  106. chia/_tests/core/full_node/stores/test_sync_store.py +135 -0
  107. chia/_tests/core/full_node/test_address_manager.py +588 -0
  108. chia/_tests/core/full_node/test_block_height_map.py +556 -0
  109. chia/_tests/core/full_node/test_conditions.py +558 -0
  110. chia/_tests/core/full_node/test_full_node.py +2445 -0
  111. chia/_tests/core/full_node/test_generator_tools.py +82 -0
  112. chia/_tests/core/full_node/test_hint_management.py +104 -0
  113. chia/_tests/core/full_node/test_node_load.py +34 -0
  114. chia/_tests/core/full_node/test_performance.py +182 -0
  115. chia/_tests/core/full_node/test_subscriptions.py +492 -0
  116. chia/_tests/core/full_node/test_transactions.py +203 -0
  117. chia/_tests/core/full_node/test_tx_processing_queue.py +154 -0
  118. chia/_tests/core/large_block.py +2388 -0
  119. chia/_tests/core/make_block_generator.py +72 -0
  120. chia/_tests/core/mempool/__init__.py +0 -0
  121. chia/_tests/core/mempool/config.py +4 -0
  122. chia/_tests/core/mempool/test_mempool.py +3180 -0
  123. chia/_tests/core/mempool/test_mempool_fee_estimator.py +104 -0
  124. chia/_tests/core/mempool/test_mempool_fee_protocol.py +55 -0
  125. chia/_tests/core/mempool/test_mempool_item_queries.py +192 -0
  126. chia/_tests/core/mempool/test_mempool_manager.py +2054 -0
  127. chia/_tests/core/mempool/test_mempool_performance.py +65 -0
  128. chia/_tests/core/mempool/test_singleton_fast_forward.py +567 -0
  129. chia/_tests/core/node_height.py +28 -0
  130. chia/_tests/core/server/__init__.py +0 -0
  131. chia/_tests/core/server/config.py +3 -0
  132. chia/_tests/core/server/flood.py +82 -0
  133. chia/_tests/core/server/serve.py +132 -0
  134. chia/_tests/core/server/test_capabilities.py +68 -0
  135. chia/_tests/core/server/test_dos.py +320 -0
  136. chia/_tests/core/server/test_event_loop.py +109 -0
  137. chia/_tests/core/server/test_loop.py +290 -0
  138. chia/_tests/core/server/test_node_discovery.py +74 -0
  139. chia/_tests/core/server/test_rate_limits.py +370 -0
  140. chia/_tests/core/server/test_server.py +225 -0
  141. chia/_tests/core/server/test_upnp.py +8 -0
  142. chia/_tests/core/services/__init__.py +0 -0
  143. chia/_tests/core/services/config.py +3 -0
  144. chia/_tests/core/services/test_services.py +166 -0
  145. chia/_tests/core/ssl/__init__.py +0 -0
  146. chia/_tests/core/ssl/config.py +3 -0
  147. chia/_tests/core/ssl/test_ssl.py +198 -0
  148. chia/_tests/core/test_coins.py +33 -0
  149. chia/_tests/core/test_cost_calculation.py +314 -0
  150. chia/_tests/core/test_crawler.py +175 -0
  151. chia/_tests/core/test_crawler_rpc.py +53 -0
  152. chia/_tests/core/test_daemon_rpc.py +24 -0
  153. chia/_tests/core/test_db_conversion.py +129 -0
  154. chia/_tests/core/test_db_validation.py +161 -0
  155. chia/_tests/core/test_farmer_harvester_rpc.py +504 -0
  156. chia/_tests/core/test_filter.py +37 -0
  157. chia/_tests/core/test_full_node_rpc.py +794 -0
  158. chia/_tests/core/test_merkle_set.py +343 -0
  159. chia/_tests/core/test_program.py +49 -0
  160. chia/_tests/core/test_rpc_util.py +87 -0
  161. chia/_tests/core/test_seeder.py +308 -0
  162. chia/_tests/core/test_setproctitle.py +13 -0
  163. chia/_tests/core/util/__init__.py +0 -0
  164. chia/_tests/core/util/config.py +4 -0
  165. chia/_tests/core/util/test_block_cache.py +44 -0
  166. chia/_tests/core/util/test_cached_bls.py +57 -0
  167. chia/_tests/core/util/test_config.py +337 -0
  168. chia/_tests/core/util/test_file_keyring_synchronization.py +105 -0
  169. chia/_tests/core/util/test_files.py +391 -0
  170. chia/_tests/core/util/test_jsonify.py +146 -0
  171. chia/_tests/core/util/test_keychain.py +514 -0
  172. chia/_tests/core/util/test_keyring_wrapper.py +490 -0
  173. chia/_tests/core/util/test_lockfile.py +380 -0
  174. chia/_tests/core/util/test_log_exceptions.py +187 -0
  175. chia/_tests/core/util/test_lru_cache.py +56 -0
  176. chia/_tests/core/util/test_significant_bits.py +40 -0
  177. chia/_tests/core/util/test_streamable.py +883 -0
  178. chia/_tests/db/__init__.py +0 -0
  179. chia/_tests/db/test_db_wrapper.py +565 -0
  180. chia/_tests/environments/__init__.py +0 -0
  181. chia/_tests/environments/common.py +35 -0
  182. chia/_tests/environments/full_node.py +47 -0
  183. chia/_tests/environments/wallet.py +368 -0
  184. chia/_tests/ether.py +19 -0
  185. chia/_tests/farmer_harvester/__init__.py +0 -0
  186. chia/_tests/farmer_harvester/config.py +3 -0
  187. chia/_tests/farmer_harvester/test_farmer.py +1264 -0
  188. chia/_tests/farmer_harvester/test_farmer_harvester.py +292 -0
  189. chia/_tests/farmer_harvester/test_filter_prefix_bits.py +130 -0
  190. chia/_tests/farmer_harvester/test_third_party_harvesters.py +501 -0
  191. chia/_tests/farmer_harvester/test_third_party_harvesters_data.json +29 -0
  192. chia/_tests/fee_estimation/__init__.py +0 -0
  193. chia/_tests/fee_estimation/config.py +3 -0
  194. chia/_tests/fee_estimation/test_fee_estimation_integration.py +262 -0
  195. chia/_tests/fee_estimation/test_fee_estimation_rpc.py +287 -0
  196. chia/_tests/fee_estimation/test_fee_estimation_unit_tests.py +145 -0
  197. chia/_tests/fee_estimation/test_mempoolitem_height_added.py +146 -0
  198. chia/_tests/generator/__init__.py +0 -0
  199. chia/_tests/generator/puzzles/__init__.py +0 -0
  200. chia/_tests/generator/puzzles/test_generator_deserialize.clsp +3 -0
  201. chia/_tests/generator/puzzles/test_generator_deserialize.clsp.hex +1 -0
  202. chia/_tests/generator/puzzles/test_multiple_generator_input_arguments.clsp +19 -0
  203. chia/_tests/generator/puzzles/test_multiple_generator_input_arguments.clsp.hex +1 -0
  204. chia/_tests/generator/test_compression.py +218 -0
  205. chia/_tests/generator/test_generator_types.py +44 -0
  206. chia/_tests/generator/test_rom.py +182 -0
  207. chia/_tests/plot_sync/__init__.py +0 -0
  208. chia/_tests/plot_sync/config.py +3 -0
  209. chia/_tests/plot_sync/test_delta.py +102 -0
  210. chia/_tests/plot_sync/test_plot_sync.py +617 -0
  211. chia/_tests/plot_sync/test_receiver.py +451 -0
  212. chia/_tests/plot_sync/test_sender.py +116 -0
  213. chia/_tests/plot_sync/test_sync_simulated.py +450 -0
  214. chia/_tests/plot_sync/util.py +67 -0
  215. chia/_tests/plotting/__init__.py +0 -0
  216. chia/_tests/plotting/config.py +3 -0
  217. chia/_tests/plotting/test_plot_manager.py +738 -0
  218. chia/_tests/plotting/util.py +13 -0
  219. chia/_tests/pools/__init__.py +0 -0
  220. chia/_tests/pools/config.py +5 -0
  221. chia/_tests/pools/test_pool_cmdline.py +23 -0
  222. chia/_tests/pools/test_pool_config.py +44 -0
  223. chia/_tests/pools/test_pool_puzzles_lifecycle.py +398 -0
  224. chia/_tests/pools/test_pool_rpc.py +1010 -0
  225. chia/_tests/pools/test_pool_wallet.py +201 -0
  226. chia/_tests/pools/test_wallet_pool_store.py +161 -0
  227. chia/_tests/process_junit.py +349 -0
  228. chia/_tests/rpc/__init__.py +0 -0
  229. chia/_tests/rpc/test_rpc_client.py +81 -0
  230. chia/_tests/simulation/__init__.py +0 -0
  231. chia/_tests/simulation/config.py +6 -0
  232. chia/_tests/simulation/test_simulation.py +501 -0
  233. chia/_tests/simulation/test_simulator.py +234 -0
  234. chia/_tests/simulation/test_start_simulator.py +106 -0
  235. chia/_tests/testconfig.py +13 -0
  236. chia/_tests/timelord/__init__.py +0 -0
  237. chia/_tests/timelord/config.py +3 -0
  238. chia/_tests/timelord/test_new_peak.py +437 -0
  239. chia/_tests/timelord/test_timelord.py +11 -0
  240. chia/_tests/tools/1315537.json +170 -0
  241. chia/_tests/tools/1315544.json +160 -0
  242. chia/_tests/tools/1315630.json +150 -0
  243. chia/_tests/tools/300000.json +105 -0
  244. chia/_tests/tools/442734.json +140 -0
  245. chia/_tests/tools/466212.json +130 -0
  246. chia/_tests/tools/__init__.py +0 -0
  247. chia/_tests/tools/config.py +5 -0
  248. chia/_tests/tools/test-blockchain-db.sqlite +0 -0
  249. chia/_tests/tools/test_full_sync.py +30 -0
  250. chia/_tests/tools/test_legacy_keyring.py +82 -0
  251. chia/_tests/tools/test_run_block.py +129 -0
  252. chia/_tests/util/__init__.py +0 -0
  253. chia/_tests/util/benchmark_cost.py +170 -0
  254. chia/_tests/util/benchmarks.py +154 -0
  255. chia/_tests/util/bip39_test_vectors.json +148 -0
  256. chia/_tests/util/blockchain.py +133 -0
  257. chia/_tests/util/blockchain_mock.py +132 -0
  258. chia/_tests/util/build_network_protocol_files.py +302 -0
  259. chia/_tests/util/clvm_generator.bin +0 -0
  260. chia/_tests/util/config.py +3 -0
  261. chia/_tests/util/constants.py +20 -0
  262. chia/_tests/util/db_connection.py +36 -0
  263. chia/_tests/util/full_sync.py +245 -0
  264. chia/_tests/util/gen_ssl_certs.py +115 -0
  265. chia/_tests/util/generator_tools_testing.py +47 -0
  266. chia/_tests/util/key_tool.py +37 -0
  267. chia/_tests/util/misc.py +722 -0
  268. chia/_tests/util/network_protocol_data.py +1074 -0
  269. chia/_tests/util/protocol_messages_bytes-v1.0 +0 -0
  270. chia/_tests/util/protocol_messages_json.py +2700 -0
  271. chia/_tests/util/rpc.py +23 -0
  272. chia/_tests/util/run_block.py +163 -0
  273. chia/_tests/util/setup_nodes.py +479 -0
  274. chia/_tests/util/split_managers.py +99 -0
  275. chia/_tests/util/temp_file.py +14 -0
  276. chia/_tests/util/test_action_scope.py +143 -0
  277. chia/_tests/util/test_async_pool.py +366 -0
  278. chia/_tests/util/test_build_job_matrix.py +43 -0
  279. chia/_tests/util/test_build_network_protocol_files.py +7 -0
  280. chia/_tests/util/test_chia_version.py +50 -0
  281. chia/_tests/util/test_collection.py +11 -0
  282. chia/_tests/util/test_condition_tools.py +231 -0
  283. chia/_tests/util/test_config.py +426 -0
  284. chia/_tests/util/test_dump_keyring.py +60 -0
  285. chia/_tests/util/test_errors.py +10 -0
  286. chia/_tests/util/test_full_block_utils.py +271 -0
  287. chia/_tests/util/test_installed.py +20 -0
  288. chia/_tests/util/test_limited_semaphore.py +52 -0
  289. chia/_tests/util/test_logging_filter.py +43 -0
  290. chia/_tests/util/test_misc.py +444 -0
  291. chia/_tests/util/test_network.py +74 -0
  292. chia/_tests/util/test_network_protocol_files.py +579 -0
  293. chia/_tests/util/test_network_protocol_json.py +266 -0
  294. chia/_tests/util/test_network_protocol_test.py +257 -0
  295. chia/_tests/util/test_paginator.py +72 -0
  296. chia/_tests/util/test_pprint.py +17 -0
  297. chia/_tests/util/test_priority_mutex.py +487 -0
  298. chia/_tests/util/test_recursive_replace.py +116 -0
  299. chia/_tests/util/test_replace_str_to_bytes.py +137 -0
  300. chia/_tests/util/test_service_groups.py +15 -0
  301. chia/_tests/util/test_ssl_check.py +31 -0
  302. chia/_tests/util/test_testnet_overrides.py +19 -0
  303. chia/_tests/util/test_tests_misc.py +38 -0
  304. chia/_tests/util/test_timing.py +37 -0
  305. chia/_tests/util/test_trusted_peer.py +51 -0
  306. chia/_tests/util/time_out_assert.py +154 -0
  307. chia/_tests/wallet/__init__.py +0 -0
  308. chia/_tests/wallet/cat_wallet/__init__.py +0 -0
  309. chia/_tests/wallet/cat_wallet/config.py +4 -0
  310. chia/_tests/wallet/cat_wallet/test_cat_lifecycle.py +468 -0
  311. chia/_tests/wallet/cat_wallet/test_cat_outer_puzzle.py +69 -0
  312. chia/_tests/wallet/cat_wallet/test_cat_wallet.py +1738 -0
  313. chia/_tests/wallet/cat_wallet/test_offer_lifecycle.py +291 -0
  314. chia/_tests/wallet/cat_wallet/test_trades.py +2578 -0
  315. chia/_tests/wallet/clawback/__init__.py +0 -0
  316. chia/_tests/wallet/clawback/config.py +3 -0
  317. chia/_tests/wallet/clawback/test_clawback_decorator.py +80 -0
  318. chia/_tests/wallet/clawback/test_clawback_lifecycle.py +292 -0
  319. chia/_tests/wallet/clawback/test_clawback_metadata.py +51 -0
  320. chia/_tests/wallet/config.py +4 -0
  321. chia/_tests/wallet/conftest.py +217 -0
  322. chia/_tests/wallet/dao_wallet/__init__.py +0 -0
  323. chia/_tests/wallet/dao_wallet/config.py +3 -0
  324. chia/_tests/wallet/dao_wallet/test_dao_clvm.py +1322 -0
  325. chia/_tests/wallet/dao_wallet/test_dao_wallets.py +3488 -0
  326. chia/_tests/wallet/db_wallet/__init__.py +0 -0
  327. chia/_tests/wallet/db_wallet/config.py +3 -0
  328. chia/_tests/wallet/db_wallet/test_db_graftroot.py +143 -0
  329. chia/_tests/wallet/db_wallet/test_dl_offers.py +491 -0
  330. chia/_tests/wallet/db_wallet/test_dl_wallet.py +823 -0
  331. chia/_tests/wallet/did_wallet/__init__.py +0 -0
  332. chia/_tests/wallet/did_wallet/config.py +4 -0
  333. chia/_tests/wallet/did_wallet/test_did.py +1481 -0
  334. chia/_tests/wallet/nft_wallet/__init__.py +0 -0
  335. chia/_tests/wallet/nft_wallet/config.py +4 -0
  336. chia/_tests/wallet/nft_wallet/test_nft_1_offers.py +1492 -0
  337. chia/_tests/wallet/nft_wallet/test_nft_bulk_mint.py +1014 -0
  338. chia/_tests/wallet/nft_wallet/test_nft_lifecycle.py +376 -0
  339. chia/_tests/wallet/nft_wallet/test_nft_offers.py +1209 -0
  340. chia/_tests/wallet/nft_wallet/test_nft_puzzles.py +172 -0
  341. chia/_tests/wallet/nft_wallet/test_nft_wallet.py +2558 -0
  342. chia/_tests/wallet/nft_wallet/test_ownership_outer_puzzle.py +70 -0
  343. chia/_tests/wallet/rpc/__init__.py +0 -0
  344. chia/_tests/wallet/rpc/config.py +4 -0
  345. chia/_tests/wallet/rpc/test_dl_wallet_rpc.py +287 -0
  346. chia/_tests/wallet/rpc/test_wallet_rpc.py +3106 -0
  347. chia/_tests/wallet/simple_sync/__init__.py +0 -0
  348. chia/_tests/wallet/simple_sync/config.py +3 -0
  349. chia/_tests/wallet/simple_sync/test_simple_sync_protocol.py +719 -0
  350. chia/_tests/wallet/sync/__init__.py +0 -0
  351. chia/_tests/wallet/sync/config.py +4 -0
  352. chia/_tests/wallet/sync/test_wallet_sync.py +1529 -0
  353. chia/_tests/wallet/test_address_type.py +189 -0
  354. chia/_tests/wallet/test_bech32m.py +45 -0
  355. chia/_tests/wallet/test_clvm_streamable.py +244 -0
  356. chia/_tests/wallet/test_coin_selection.py +589 -0
  357. chia/_tests/wallet/test_conditions.py +388 -0
  358. chia/_tests/wallet/test_debug_spend_bundle.py +76 -0
  359. chia/_tests/wallet/test_new_wallet_protocol.py +1176 -0
  360. chia/_tests/wallet/test_nft_store.py +193 -0
  361. chia/_tests/wallet/test_notifications.py +196 -0
  362. chia/_tests/wallet/test_offer_parsing_performance.py +48 -0
  363. chia/_tests/wallet/test_puzzle_store.py +133 -0
  364. chia/_tests/wallet/test_sign_coin_spends.py +159 -0
  365. chia/_tests/wallet/test_signer_protocol.py +948 -0
  366. chia/_tests/wallet/test_singleton.py +122 -0
  367. chia/_tests/wallet/test_singleton_lifecycle_fast.py +772 -0
  368. chia/_tests/wallet/test_singleton_store.py +152 -0
  369. chia/_tests/wallet/test_taproot.py +19 -0
  370. chia/_tests/wallet/test_transaction_store.py +941 -0
  371. chia/_tests/wallet/test_util.py +181 -0
  372. chia/_tests/wallet/test_wallet.py +2139 -0
  373. chia/_tests/wallet/test_wallet_action_scope.py +85 -0
  374. chia/_tests/wallet/test_wallet_blockchain.py +113 -0
  375. chia/_tests/wallet/test_wallet_coin_store.py +1002 -0
  376. chia/_tests/wallet/test_wallet_interested_store.py +43 -0
  377. chia/_tests/wallet/test_wallet_key_val_store.py +40 -0
  378. chia/_tests/wallet/test_wallet_node.py +783 -0
  379. chia/_tests/wallet/test_wallet_retry.py +95 -0
  380. chia/_tests/wallet/test_wallet_state_manager.py +252 -0
  381. chia/_tests/wallet/test_wallet_test_framework.py +275 -0
  382. chia/_tests/wallet/test_wallet_trade_store.py +218 -0
  383. chia/_tests/wallet/test_wallet_user_store.py +34 -0
  384. chia/_tests/wallet/test_wallet_utils.py +155 -0
  385. chia/_tests/wallet/vc_wallet/__init__.py +0 -0
  386. chia/_tests/wallet/vc_wallet/config.py +3 -0
  387. chia/_tests/wallet/vc_wallet/test_cr_outer_puzzle.py +70 -0
  388. chia/_tests/wallet/vc_wallet/test_vc_lifecycle.py +883 -0
  389. chia/_tests/wallet/vc_wallet/test_vc_wallet.py +801 -0
  390. chia/_tests/wallet/wallet_block_tools.py +327 -0
  391. chia/_tests/weight_proof/__init__.py +0 -0
  392. chia/_tests/weight_proof/config.py +3 -0
  393. chia/_tests/weight_proof/test_weight_proof.py +528 -0
  394. chia/clvm/__init__.py +0 -0
  395. chia/clvm/spend_sim.py +488 -0
  396. chia/cmds/__init__.py +0 -0
  397. chia/cmds/beta.py +183 -0
  398. chia/cmds/beta_funcs.py +133 -0
  399. chia/cmds/check_wallet_db.py +418 -0
  400. chia/cmds/chia.py +143 -0
  401. chia/cmds/cmd_classes.py +315 -0
  402. chia/cmds/cmds_util.py +498 -0
  403. chia/cmds/coin_funcs.py +260 -0
  404. chia/cmds/coins.py +220 -0
  405. chia/cmds/completion.py +49 -0
  406. chia/cmds/configure.py +331 -0
  407. chia/cmds/dao.py +1008 -0
  408. chia/cmds/dao_funcs.py +576 -0
  409. chia/cmds/data.py +707 -0
  410. chia/cmds/data_funcs.py +380 -0
  411. chia/cmds/db.py +86 -0
  412. chia/cmds/db_backup_func.py +77 -0
  413. chia/cmds/db_upgrade_func.py +452 -0
  414. chia/cmds/db_validate_func.py +184 -0
  415. chia/cmds/dev.py +16 -0
  416. chia/cmds/farm.py +87 -0
  417. chia/cmds/farm_funcs.py +207 -0
  418. chia/cmds/init.py +70 -0
  419. chia/cmds/init_funcs.py +367 -0
  420. chia/cmds/installers.py +129 -0
  421. chia/cmds/keys.py +510 -0
  422. chia/cmds/keys_funcs.py +864 -0
  423. chia/cmds/netspace.py +47 -0
  424. chia/cmds/netspace_funcs.py +53 -0
  425. chia/cmds/options.py +32 -0
  426. chia/cmds/param_types.py +228 -0
  427. chia/cmds/passphrase.py +130 -0
  428. chia/cmds/passphrase_funcs.py +346 -0
  429. chia/cmds/peer.py +50 -0
  430. chia/cmds/peer_funcs.py +129 -0
  431. chia/cmds/plotnft.py +206 -0
  432. chia/cmds/plotnft_funcs.py +374 -0
  433. chia/cmds/plots.py +222 -0
  434. chia/cmds/plotters.py +17 -0
  435. chia/cmds/rpc.py +188 -0
  436. chia/cmds/show.py +71 -0
  437. chia/cmds/show_funcs.py +214 -0
  438. chia/cmds/signer.py +304 -0
  439. chia/cmds/sim.py +217 -0
  440. chia/cmds/sim_funcs.py +509 -0
  441. chia/cmds/start.py +24 -0
  442. chia/cmds/start_funcs.py +112 -0
  443. chia/cmds/stop.py +61 -0
  444. chia/cmds/units.py +11 -0
  445. chia/cmds/wallet.py +1745 -0
  446. chia/cmds/wallet_funcs.py +1800 -0
  447. chia/consensus/__init__.py +0 -0
  448. chia/consensus/block_body_validation.py +515 -0
  449. chia/consensus/block_creation.py +525 -0
  450. chia/consensus/block_header_validation.py +1064 -0
  451. chia/consensus/block_record.py +32 -0
  452. chia/consensus/block_rewards.py +53 -0
  453. chia/consensus/block_root_validation.py +46 -0
  454. chia/consensus/blockchain.py +1100 -0
  455. chia/consensus/blockchain_interface.py +56 -0
  456. chia/consensus/coinbase.py +30 -0
  457. chia/consensus/condition_costs.py +9 -0
  458. chia/consensus/constants.py +49 -0
  459. chia/consensus/cost_calculator.py +15 -0
  460. chia/consensus/default_constants.py +90 -0
  461. chia/consensus/deficit.py +55 -0
  462. chia/consensus/difficulty_adjustment.py +412 -0
  463. chia/consensus/find_fork_point.py +111 -0
  464. chia/consensus/full_block_to_block_record.py +167 -0
  465. chia/consensus/get_block_challenge.py +106 -0
  466. chia/consensus/get_block_generator.py +26 -0
  467. chia/consensus/make_sub_epoch_summary.py +210 -0
  468. chia/consensus/multiprocess_validation.py +365 -0
  469. chia/consensus/pos_quality.py +19 -0
  470. chia/consensus/pot_iterations.py +67 -0
  471. chia/consensus/puzzles/__init__.py +0 -0
  472. chia/consensus/puzzles/chialisp_deserialisation.clsp +69 -0
  473. chia/consensus/puzzles/chialisp_deserialisation.clsp.hex +1 -0
  474. chia/consensus/puzzles/rom_bootstrap_generator.clsp +37 -0
  475. chia/consensus/puzzles/rom_bootstrap_generator.clsp.hex +1 -0
  476. chia/consensus/vdf_info_computation.py +156 -0
  477. chia/daemon/__init__.py +0 -0
  478. chia/daemon/client.py +233 -0
  479. chia/daemon/keychain_proxy.py +501 -0
  480. chia/daemon/keychain_server.py +365 -0
  481. chia/daemon/server.py +1616 -0
  482. chia/daemon/windows_signal.py +56 -0
  483. chia/data_layer/__init__.py +0 -0
  484. chia/data_layer/data_layer.py +1303 -0
  485. chia/data_layer/data_layer_api.py +25 -0
  486. chia/data_layer/data_layer_errors.py +50 -0
  487. chia/data_layer/data_layer_server.py +170 -0
  488. chia/data_layer/data_layer_util.py +985 -0
  489. chia/data_layer/data_layer_wallet.py +1315 -0
  490. chia/data_layer/data_store.py +2267 -0
  491. chia/data_layer/dl_wallet_store.py +407 -0
  492. chia/data_layer/download_data.py +389 -0
  493. chia/data_layer/puzzles/__init__.py +0 -0
  494. chia/data_layer/puzzles/graftroot_dl_offers.clsp +100 -0
  495. chia/data_layer/puzzles/graftroot_dl_offers.clsp.hex +1 -0
  496. chia/data_layer/s3_plugin_config.yml +33 -0
  497. chia/data_layer/s3_plugin_service.py +468 -0
  498. chia/data_layer/util/__init__.py +0 -0
  499. chia/data_layer/util/benchmark.py +108 -0
  500. chia/data_layer/util/plugin.py +41 -0
  501. chia/farmer/__init__.py +0 -0
  502. chia/farmer/farmer.py +920 -0
  503. chia/farmer/farmer_api.py +814 -0
  504. chia/full_node/__init__.py +0 -0
  505. chia/full_node/bitcoin_fee_estimator.py +85 -0
  506. chia/full_node/block_height_map.py +271 -0
  507. chia/full_node/block_store.py +570 -0
  508. chia/full_node/bundle_tools.py +19 -0
  509. chia/full_node/coin_store.py +646 -0
  510. chia/full_node/fee_estimate.py +54 -0
  511. chia/full_node/fee_estimate_store.py +24 -0
  512. chia/full_node/fee_estimation.py +93 -0
  513. chia/full_node/fee_estimator.py +90 -0
  514. chia/full_node/fee_estimator_constants.py +38 -0
  515. chia/full_node/fee_estimator_interface.py +42 -0
  516. chia/full_node/fee_history.py +26 -0
  517. chia/full_node/fee_tracker.py +564 -0
  518. chia/full_node/full_node.py +3052 -0
  519. chia/full_node/full_node_api.py +1974 -0
  520. chia/full_node/full_node_store.py +1033 -0
  521. chia/full_node/hint_management.py +56 -0
  522. chia/full_node/hint_store.py +94 -0
  523. chia/full_node/mempool.py +583 -0
  524. chia/full_node/mempool_check_conditions.py +177 -0
  525. chia/full_node/mempool_manager.py +858 -0
  526. chia/full_node/pending_tx_cache.py +112 -0
  527. chia/full_node/puzzles/__init__.py +0 -0
  528. chia/full_node/puzzles/block_program_zero.clsp +14 -0
  529. chia/full_node/puzzles/block_program_zero.clsp.hex +1 -0
  530. chia/full_node/puzzles/decompress_coin_spend_entry.clsp +5 -0
  531. chia/full_node/puzzles/decompress_coin_spend_entry.clsp.hex +1 -0
  532. chia/full_node/puzzles/decompress_coin_spend_entry_with_prefix.clsp +7 -0
  533. chia/full_node/puzzles/decompress_coin_spend_entry_with_prefix.clsp.hex +1 -0
  534. chia/full_node/puzzles/decompress_puzzle.clsp +6 -0
  535. chia/full_node/puzzles/decompress_puzzle.clsp.hex +1 -0
  536. chia/full_node/signage_point.py +16 -0
  537. chia/full_node/subscriptions.py +248 -0
  538. chia/full_node/sync_store.py +145 -0
  539. chia/full_node/tx_processing_queue.py +78 -0
  540. chia/full_node/weight_proof.py +1719 -0
  541. chia/harvester/__init__.py +0 -0
  542. chia/harvester/harvester.py +271 -0
  543. chia/harvester/harvester_api.py +374 -0
  544. chia/introducer/__init__.py +0 -0
  545. chia/introducer/introducer.py +120 -0
  546. chia/introducer/introducer_api.py +64 -0
  547. chia/legacy/__init__.py +0 -0
  548. chia/legacy/keyring.py +154 -0
  549. chia/plot_sync/__init__.py +0 -0
  550. chia/plot_sync/delta.py +61 -0
  551. chia/plot_sync/exceptions.py +56 -0
  552. chia/plot_sync/receiver.py +385 -0
  553. chia/plot_sync/sender.py +337 -0
  554. chia/plot_sync/util.py +43 -0
  555. chia/plotters/__init__.py +0 -0
  556. chia/plotters/bladebit.py +388 -0
  557. chia/plotters/chiapos.py +63 -0
  558. chia/plotters/madmax.py +224 -0
  559. chia/plotters/plotters.py +577 -0
  560. chia/plotters/plotters_util.py +131 -0
  561. chia/plotting/__init__.py +0 -0
  562. chia/plotting/cache.py +212 -0
  563. chia/plotting/check_plots.py +283 -0
  564. chia/plotting/create_plots.py +278 -0
  565. chia/plotting/manager.py +436 -0
  566. chia/plotting/util.py +324 -0
  567. chia/pools/__init__.py +0 -0
  568. chia/pools/pool_config.py +110 -0
  569. chia/pools/pool_puzzles.py +459 -0
  570. chia/pools/pool_wallet.py +926 -0
  571. chia/pools/pool_wallet_info.py +118 -0
  572. chia/pools/puzzles/__init__.py +0 -0
  573. chia/pools/puzzles/pool_member_innerpuz.clsp +70 -0
  574. chia/pools/puzzles/pool_member_innerpuz.clsp.hex +1 -0
  575. chia/pools/puzzles/pool_waitingroom_innerpuz.clsp +69 -0
  576. chia/pools/puzzles/pool_waitingroom_innerpuz.clsp.hex +1 -0
  577. chia/protocols/__init__.py +0 -0
  578. chia/protocols/farmer_protocol.py +102 -0
  579. chia/protocols/full_node_protocol.py +219 -0
  580. chia/protocols/harvester_protocol.py +216 -0
  581. chia/protocols/introducer_protocol.py +26 -0
  582. chia/protocols/pool_protocol.py +177 -0
  583. chia/protocols/protocol_message_types.py +139 -0
  584. chia/protocols/protocol_state_machine.py +87 -0
  585. chia/protocols/protocol_timing.py +7 -0
  586. chia/protocols/shared_protocol.py +86 -0
  587. chia/protocols/timelord_protocol.py +93 -0
  588. chia/protocols/wallet_protocol.py +401 -0
  589. chia/py.typed +0 -0
  590. chia/rpc/__init__.py +0 -0
  591. chia/rpc/crawler_rpc_api.py +75 -0
  592. chia/rpc/data_layer_rpc_api.py +639 -0
  593. chia/rpc/data_layer_rpc_client.py +188 -0
  594. chia/rpc/data_layer_rpc_util.py +62 -0
  595. chia/rpc/farmer_rpc_api.py +360 -0
  596. chia/rpc/farmer_rpc_client.py +86 -0
  597. chia/rpc/full_node_rpc_api.py +954 -0
  598. chia/rpc/full_node_rpc_client.py +292 -0
  599. chia/rpc/harvester_rpc_api.py +136 -0
  600. chia/rpc/harvester_rpc_client.py +54 -0
  601. chia/rpc/rpc_client.py +144 -0
  602. chia/rpc/rpc_server.py +447 -0
  603. chia/rpc/timelord_rpc_api.py +27 -0
  604. chia/rpc/util.py +293 -0
  605. chia/rpc/wallet_request_types.py +688 -0
  606. chia/rpc/wallet_rpc_api.py +4779 -0
  607. chia/rpc/wallet_rpc_client.py +1844 -0
  608. chia/seeder/__init__.py +0 -0
  609. chia/seeder/crawl_store.py +427 -0
  610. chia/seeder/crawler.py +423 -0
  611. chia/seeder/crawler_api.py +129 -0
  612. chia/seeder/dns_server.py +544 -0
  613. chia/seeder/peer_record.py +146 -0
  614. chia/seeder/start_crawler.py +88 -0
  615. chia/server/__init__.py +0 -0
  616. chia/server/address_manager.py +658 -0
  617. chia/server/address_manager_store.py +237 -0
  618. chia/server/api_protocol.py +11 -0
  619. chia/server/capabilities.py +24 -0
  620. chia/server/chia_policy.py +345 -0
  621. chia/server/introducer_peers.py +76 -0
  622. chia/server/node_discovery.py +718 -0
  623. chia/server/outbound_message.py +33 -0
  624. chia/server/rate_limit_numbers.py +204 -0
  625. chia/server/rate_limits.py +113 -0
  626. chia/server/server.py +720 -0
  627. chia/server/signal_handlers.py +117 -0
  628. chia/server/ssl_context.py +32 -0
  629. chia/server/start_data_layer.py +137 -0
  630. chia/server/start_farmer.py +86 -0
  631. chia/server/start_full_node.py +106 -0
  632. chia/server/start_harvester.py +80 -0
  633. chia/server/start_introducer.py +69 -0
  634. chia/server/start_service.py +328 -0
  635. chia/server/start_timelord.py +82 -0
  636. chia/server/start_wallet.py +109 -0
  637. chia/server/upnp.py +117 -0
  638. chia/server/ws_connection.py +752 -0
  639. chia/simulator/__init__.py +0 -0
  640. chia/simulator/block_tools.py +2053 -0
  641. chia/simulator/full_node_simulator.py +802 -0
  642. chia/simulator/keyring.py +128 -0
  643. chia/simulator/setup_services.py +505 -0
  644. chia/simulator/simulator_constants.py +13 -0
  645. chia/simulator/simulator_full_node_rpc_api.py +101 -0
  646. chia/simulator/simulator_full_node_rpc_client.py +62 -0
  647. chia/simulator/simulator_protocol.py +29 -0
  648. chia/simulator/simulator_test_tools.py +163 -0
  649. chia/simulator/socket.py +27 -0
  650. chia/simulator/ssl_certs.py +114 -0
  651. chia/simulator/ssl_certs_1.py +699 -0
  652. chia/simulator/ssl_certs_10.py +699 -0
  653. chia/simulator/ssl_certs_2.py +699 -0
  654. chia/simulator/ssl_certs_3.py +699 -0
  655. chia/simulator/ssl_certs_4.py +699 -0
  656. chia/simulator/ssl_certs_5.py +699 -0
  657. chia/simulator/ssl_certs_6.py +699 -0
  658. chia/simulator/ssl_certs_7.py +699 -0
  659. chia/simulator/ssl_certs_8.py +699 -0
  660. chia/simulator/ssl_certs_9.py +699 -0
  661. chia/simulator/start_simulator.py +135 -0
  662. chia/simulator/wallet_tools.py +245 -0
  663. chia/ssl/__init__.py +0 -0
  664. chia/ssl/chia_ca.crt +19 -0
  665. chia/ssl/chia_ca.key +28 -0
  666. chia/ssl/create_ssl.py +249 -0
  667. chia/ssl/dst_root_ca.pem +20 -0
  668. chia/timelord/__init__.py +0 -0
  669. chia/timelord/iters_from_block.py +50 -0
  670. chia/timelord/timelord.py +1202 -0
  671. chia/timelord/timelord_api.py +132 -0
  672. chia/timelord/timelord_launcher.py +188 -0
  673. chia/timelord/timelord_state.py +244 -0
  674. chia/timelord/types.py +22 -0
  675. chia/types/__init__.py +0 -0
  676. chia/types/aliases.py +35 -0
  677. chia/types/block_protocol.py +20 -0
  678. chia/types/blockchain_format/__init__.py +0 -0
  679. chia/types/blockchain_format/classgroup.py +5 -0
  680. chia/types/blockchain_format/coin.py +28 -0
  681. chia/types/blockchain_format/foliage.py +8 -0
  682. chia/types/blockchain_format/pool_target.py +5 -0
  683. chia/types/blockchain_format/program.py +270 -0
  684. chia/types/blockchain_format/proof_of_space.py +135 -0
  685. chia/types/blockchain_format/reward_chain_block.py +6 -0
  686. chia/types/blockchain_format/serialized_program.py +5 -0
  687. chia/types/blockchain_format/sized_bytes.py +11 -0
  688. chia/types/blockchain_format/slots.py +9 -0
  689. chia/types/blockchain_format/sub_epoch_summary.py +5 -0
  690. chia/types/blockchain_format/tree_hash.py +72 -0
  691. chia/types/blockchain_format/vdf.py +86 -0
  692. chia/types/clvm_cost.py +13 -0
  693. chia/types/coin_record.py +43 -0
  694. chia/types/coin_spend.py +115 -0
  695. chia/types/condition_opcodes.py +73 -0
  696. chia/types/condition_with_args.py +17 -0
  697. chia/types/eligible_coin_spends.py +364 -0
  698. chia/types/end_of_slot_bundle.py +5 -0
  699. chia/types/fee_rate.py +38 -0
  700. chia/types/full_block.py +5 -0
  701. chia/types/generator_types.py +14 -0
  702. chia/types/header_block.py +5 -0
  703. chia/types/internal_mempool_item.py +19 -0
  704. chia/types/mempool_inclusion_status.py +9 -0
  705. chia/types/mempool_item.py +85 -0
  706. chia/types/mempool_submission_status.py +30 -0
  707. chia/types/mojos.py +7 -0
  708. chia/types/peer_info.py +64 -0
  709. chia/types/signing_mode.py +29 -0
  710. chia/types/spend_bundle.py +31 -0
  711. chia/types/spend_bundle_conditions.py +7 -0
  712. chia/types/transaction_queue_entry.py +55 -0
  713. chia/types/unfinished_block.py +5 -0
  714. chia/types/unfinished_header_block.py +37 -0
  715. chia/types/weight_proof.py +50 -0
  716. chia/util/__init__.py +0 -0
  717. chia/util/action_scope.py +168 -0
  718. chia/util/api_decorators.py +89 -0
  719. chia/util/async_pool.py +224 -0
  720. chia/util/augmented_chain.py +130 -0
  721. chia/util/batches.py +39 -0
  722. chia/util/bech32m.py +123 -0
  723. chia/util/beta_metrics.py +118 -0
  724. chia/util/block_cache.py +56 -0
  725. chia/util/byte_types.py +10 -0
  726. chia/util/check_fork_next_block.py +32 -0
  727. chia/util/chia_logging.py +124 -0
  728. chia/util/chia_version.py +33 -0
  729. chia/util/collection.py +17 -0
  730. chia/util/condition_tools.py +201 -0
  731. chia/util/config.py +366 -0
  732. chia/util/cpu.py +20 -0
  733. chia/util/db_synchronous.py +21 -0
  734. chia/util/db_version.py +30 -0
  735. chia/util/db_wrapper.py +427 -0
  736. chia/util/default_root.py +10 -0
  737. chia/util/dump_keyring.py +93 -0
  738. chia/util/english.txt +2048 -0
  739. chia/util/errors.py +351 -0
  740. chia/util/file_keyring.py +480 -0
  741. chia/util/files.py +95 -0
  742. chia/util/full_block_utils.py +321 -0
  743. chia/util/generator_tools.py +62 -0
  744. chia/util/hash.py +29 -0
  745. chia/util/initial-config.yaml +675 -0
  746. chia/util/inline_executor.py +24 -0
  747. chia/util/ints.py +19 -0
  748. chia/util/json_util.py +41 -0
  749. chia/util/keychain.py +673 -0
  750. chia/util/keyring_wrapper.py +266 -0
  751. chia/util/limited_semaphore.py +39 -0
  752. chia/util/lock.py +47 -0
  753. chia/util/log_exceptions.py +29 -0
  754. chia/util/logging.py +34 -0
  755. chia/util/lru_cache.py +29 -0
  756. chia/util/math.py +20 -0
  757. chia/util/network.py +240 -0
  758. chia/util/paginator.py +46 -0
  759. chia/util/path.py +29 -0
  760. chia/util/permissions.py +19 -0
  761. chia/util/pprint.py +40 -0
  762. chia/util/prev_transaction_block.py +23 -0
  763. chia/util/priority_mutex.py +92 -0
  764. chia/util/profiler.py +194 -0
  765. chia/util/recursive_replace.py +22 -0
  766. chia/util/safe_cancel_task.py +14 -0
  767. chia/util/service_groups.py +47 -0
  768. chia/util/setproctitle.py +20 -0
  769. chia/util/significant_bits.py +30 -0
  770. chia/util/ssl_check.py +213 -0
  771. chia/util/streamable.py +654 -0
  772. chia/util/task_timing.py +378 -0
  773. chia/util/timing.py +64 -0
  774. chia/util/vdf_prover.py +31 -0
  775. chia/util/ws_message.py +66 -0
  776. chia/wallet/__init__.py +0 -0
  777. chia/wallet/cat_wallet/__init__.py +0 -0
  778. chia/wallet/cat_wallet/cat_constants.py +75 -0
  779. chia/wallet/cat_wallet/cat_info.py +47 -0
  780. chia/wallet/cat_wallet/cat_outer_puzzle.py +120 -0
  781. chia/wallet/cat_wallet/cat_utils.py +163 -0
  782. chia/wallet/cat_wallet/cat_wallet.py +869 -0
  783. chia/wallet/cat_wallet/dao_cat_info.py +28 -0
  784. chia/wallet/cat_wallet/dao_cat_wallet.py +669 -0
  785. chia/wallet/cat_wallet/lineage_store.py +74 -0
  786. chia/wallet/cat_wallet/puzzles/__init__.py +0 -0
  787. chia/wallet/cat_wallet/puzzles/cat_truths.clib +31 -0
  788. chia/wallet/cat_wallet/puzzles/cat_v2.clsp +397 -0
  789. chia/wallet/cat_wallet/puzzles/cat_v2.clsp.hex +1 -0
  790. chia/wallet/cat_wallet/puzzles/delegated_tail.clsp +25 -0
  791. chia/wallet/cat_wallet/puzzles/delegated_tail.clsp.hex +1 -0
  792. chia/wallet/cat_wallet/puzzles/everything_with_signature.clsp +15 -0
  793. chia/wallet/cat_wallet/puzzles/everything_with_signature.clsp.hex +1 -0
  794. chia/wallet/cat_wallet/puzzles/genesis_by_coin_id.clsp +26 -0
  795. chia/wallet/cat_wallet/puzzles/genesis_by_coin_id.clsp.hex +1 -0
  796. chia/wallet/cat_wallet/puzzles/genesis_by_coin_id_or_singleton.clsp +42 -0
  797. chia/wallet/cat_wallet/puzzles/genesis_by_coin_id_or_singleton.clsp.hex +1 -0
  798. chia/wallet/cat_wallet/puzzles/genesis_by_puzzle_hash.clsp +24 -0
  799. chia/wallet/cat_wallet/puzzles/genesis_by_puzzle_hash.clsp.hex +1 -0
  800. chia/wallet/coin_selection.py +188 -0
  801. chia/wallet/conditions.py +1326 -0
  802. chia/wallet/dao_wallet/__init__.py +0 -0
  803. chia/wallet/dao_wallet/dao_info.py +61 -0
  804. chia/wallet/dao_wallet/dao_utils.py +810 -0
  805. chia/wallet/dao_wallet/dao_wallet.py +2121 -0
  806. chia/wallet/db_wallet/__init__.py +0 -0
  807. chia/wallet/db_wallet/db_wallet_puzzles.py +107 -0
  808. chia/wallet/derivation_record.py +30 -0
  809. chia/wallet/derive_keys.py +146 -0
  810. chia/wallet/did_wallet/__init__.py +0 -0
  811. chia/wallet/did_wallet/did_info.py +39 -0
  812. chia/wallet/did_wallet/did_wallet.py +1485 -0
  813. chia/wallet/did_wallet/did_wallet_puzzles.py +220 -0
  814. chia/wallet/did_wallet/puzzles/__init__.py +0 -0
  815. chia/wallet/did_wallet/puzzles/did_innerpuz.clsp +135 -0
  816. chia/wallet/did_wallet/puzzles/did_innerpuz.clsp.hex +1 -0
  817. chia/wallet/driver_protocol.py +26 -0
  818. chia/wallet/key_val_store.py +55 -0
  819. chia/wallet/lineage_proof.py +58 -0
  820. chia/wallet/nft_wallet/__init__.py +0 -0
  821. chia/wallet/nft_wallet/metadata_outer_puzzle.py +92 -0
  822. chia/wallet/nft_wallet/nft_info.py +120 -0
  823. chia/wallet/nft_wallet/nft_puzzles.py +305 -0
  824. chia/wallet/nft_wallet/nft_wallet.py +1686 -0
  825. chia/wallet/nft_wallet/ownership_outer_puzzle.py +101 -0
  826. chia/wallet/nft_wallet/puzzles/__init__.py +0 -0
  827. chia/wallet/nft_wallet/puzzles/create_nft_launcher_from_did.clsp +6 -0
  828. chia/wallet/nft_wallet/puzzles/create_nft_launcher_from_did.clsp.hex +1 -0
  829. chia/wallet/nft_wallet/puzzles/nft_intermediate_launcher.clsp +6 -0
  830. chia/wallet/nft_wallet/puzzles/nft_intermediate_launcher.clsp.hex +1 -0
  831. chia/wallet/nft_wallet/puzzles/nft_metadata_updater_default.clsp +30 -0
  832. chia/wallet/nft_wallet/puzzles/nft_metadata_updater_default.clsp.hex +1 -0
  833. chia/wallet/nft_wallet/puzzles/nft_metadata_updater_updateable.clsp +28 -0
  834. chia/wallet/nft_wallet/puzzles/nft_metadata_updater_updateable.clsp.hex +1 -0
  835. chia/wallet/nft_wallet/puzzles/nft_ownership_layer.clsp +100 -0
  836. chia/wallet/nft_wallet/puzzles/nft_ownership_layer.clsp.hex +1 -0
  837. chia/wallet/nft_wallet/puzzles/nft_ownership_transfer_program_one_way_claim_with_royalties.clsp +78 -0
  838. chia/wallet/nft_wallet/puzzles/nft_ownership_transfer_program_one_way_claim_with_royalties.clsp.hex +1 -0
  839. chia/wallet/nft_wallet/puzzles/nft_state_layer.clsp +74 -0
  840. chia/wallet/nft_wallet/puzzles/nft_state_layer.clsp.hex +1 -0
  841. chia/wallet/nft_wallet/singleton_outer_puzzle.py +101 -0
  842. chia/wallet/nft_wallet/transfer_program_puzzle.py +82 -0
  843. chia/wallet/nft_wallet/uncurry_nft.py +217 -0
  844. chia/wallet/notification_manager.py +117 -0
  845. chia/wallet/notification_store.py +178 -0
  846. chia/wallet/outer_puzzles.py +84 -0
  847. chia/wallet/payment.py +34 -0
  848. chia/wallet/puzzle_drivers.py +118 -0
  849. chia/wallet/puzzles/__init__.py +0 -0
  850. chia/wallet/puzzles/augmented_condition.clsp +13 -0
  851. chia/wallet/puzzles/augmented_condition.clsp.hex +1 -0
  852. chia/wallet/puzzles/clawback/__init__.py +0 -0
  853. chia/wallet/puzzles/clawback/drivers.py +188 -0
  854. chia/wallet/puzzles/clawback/metadata.py +38 -0
  855. chia/wallet/puzzles/clawback/puzzle_decorator.py +67 -0
  856. chia/wallet/puzzles/condition_codes.clib +77 -0
  857. chia/wallet/puzzles/curry-and-treehash.clib +102 -0
  858. chia/wallet/puzzles/curry.clib +135 -0
  859. chia/wallet/puzzles/curry_by_index.clib +16 -0
  860. chia/wallet/puzzles/dao_cat_eve.clsp +17 -0
  861. chia/wallet/puzzles/dao_cat_eve.clsp.hex +1 -0
  862. chia/wallet/puzzles/dao_cat_launcher.clsp +36 -0
  863. chia/wallet/puzzles/dao_cat_launcher.clsp.hex +1 -0
  864. chia/wallet/puzzles/dao_finished_state.clsp +35 -0
  865. chia/wallet/puzzles/dao_finished_state.clsp.hex +1 -0
  866. chia/wallet/puzzles/dao_finished_state.clsp.hex.sha256tree +1 -0
  867. chia/wallet/puzzles/dao_lockup.clsp +288 -0
  868. chia/wallet/puzzles/dao_lockup.clsp.hex +1 -0
  869. chia/wallet/puzzles/dao_lockup.clsp.hex.sha256tree +1 -0
  870. chia/wallet/puzzles/dao_proposal.clsp +377 -0
  871. chia/wallet/puzzles/dao_proposal.clsp.hex +1 -0
  872. chia/wallet/puzzles/dao_proposal.clsp.hex.sha256tree +1 -0
  873. chia/wallet/puzzles/dao_proposal_timer.clsp +78 -0
  874. chia/wallet/puzzles/dao_proposal_timer.clsp.hex +1 -0
  875. chia/wallet/puzzles/dao_proposal_timer.clsp.hex.sha256tree +1 -0
  876. chia/wallet/puzzles/dao_proposal_validator.clsp +87 -0
  877. chia/wallet/puzzles/dao_proposal_validator.clsp.hex +1 -0
  878. chia/wallet/puzzles/dao_proposal_validator.clsp.hex.sha256tree +1 -0
  879. chia/wallet/puzzles/dao_spend_p2_singleton_v2.clsp +240 -0
  880. chia/wallet/puzzles/dao_spend_p2_singleton_v2.clsp.hex +1 -0
  881. chia/wallet/puzzles/dao_spend_p2_singleton_v2.clsp.hex.sha256tree +1 -0
  882. chia/wallet/puzzles/dao_treasury.clsp +115 -0
  883. chia/wallet/puzzles/dao_treasury.clsp.hex +1 -0
  884. chia/wallet/puzzles/dao_update_proposal.clsp +44 -0
  885. chia/wallet/puzzles/dao_update_proposal.clsp.hex +1 -0
  886. chia/wallet/puzzles/deployed_puzzle_hashes.json +67 -0
  887. chia/wallet/puzzles/json.clib +25 -0
  888. chia/wallet/puzzles/load_clvm.py +162 -0
  889. chia/wallet/puzzles/merkle_utils.clib +18 -0
  890. chia/wallet/puzzles/notification.clsp +7 -0
  891. chia/wallet/puzzles/notification.clsp.hex +1 -0
  892. chia/wallet/puzzles/p2_1_of_n.clsp +22 -0
  893. chia/wallet/puzzles/p2_1_of_n.clsp.hex +1 -0
  894. chia/wallet/puzzles/p2_conditions.clsp +3 -0
  895. chia/wallet/puzzles/p2_conditions.clsp.hex +1 -0
  896. chia/wallet/puzzles/p2_conditions.py +27 -0
  897. chia/wallet/puzzles/p2_delegated_conditions.clsp +18 -0
  898. chia/wallet/puzzles/p2_delegated_conditions.clsp.hex +1 -0
  899. chia/wallet/puzzles/p2_delegated_conditions.py +22 -0
  900. chia/wallet/puzzles/p2_delegated_puzzle.clsp +19 -0
  901. chia/wallet/puzzles/p2_delegated_puzzle.clsp.hex +1 -0
  902. chia/wallet/puzzles/p2_delegated_puzzle.py +35 -0
  903. chia/wallet/puzzles/p2_delegated_puzzle_or_hidden_puzzle.clsp +91 -0
  904. chia/wallet/puzzles/p2_delegated_puzzle_or_hidden_puzzle.clsp.hex +1 -0
  905. chia/wallet/puzzles/p2_delegated_puzzle_or_hidden_puzzle.py +161 -0
  906. chia/wallet/puzzles/p2_m_of_n_delegate_direct.clsp +108 -0
  907. chia/wallet/puzzles/p2_m_of_n_delegate_direct.clsp.hex +1 -0
  908. chia/wallet/puzzles/p2_m_of_n_delegate_direct.py +22 -0
  909. chia/wallet/puzzles/p2_parent.clsp +19 -0
  910. chia/wallet/puzzles/p2_parent.clsp.hex +1 -0
  911. chia/wallet/puzzles/p2_puzzle_hash.clsp +18 -0
  912. chia/wallet/puzzles/p2_puzzle_hash.clsp.hex +1 -0
  913. chia/wallet/puzzles/p2_puzzle_hash.py +28 -0
  914. chia/wallet/puzzles/p2_singleton.clsp +30 -0
  915. chia/wallet/puzzles/p2_singleton.clsp.hex +1 -0
  916. chia/wallet/puzzles/p2_singleton_aggregator.clsp +81 -0
  917. chia/wallet/puzzles/p2_singleton_aggregator.clsp.hex +1 -0
  918. chia/wallet/puzzles/p2_singleton_or_delayed_puzhash.clsp +50 -0
  919. chia/wallet/puzzles/p2_singleton_or_delayed_puzhash.clsp.hex +1 -0
  920. chia/wallet/puzzles/p2_singleton_via_delegated_puzzle.clsp +47 -0
  921. chia/wallet/puzzles/p2_singleton_via_delegated_puzzle.clsp.hex +1 -0
  922. chia/wallet/puzzles/puzzle_utils.py +34 -0
  923. chia/wallet/puzzles/settlement_payments.clsp +49 -0
  924. chia/wallet/puzzles/settlement_payments.clsp.hex +1 -0
  925. chia/wallet/puzzles/sha256tree.clib +11 -0
  926. chia/wallet/puzzles/singleton_launcher.clsp +16 -0
  927. chia/wallet/puzzles/singleton_launcher.clsp.hex +1 -0
  928. chia/wallet/puzzles/singleton_top_layer.clsp +177 -0
  929. chia/wallet/puzzles/singleton_top_layer.clsp.hex +1 -0
  930. chia/wallet/puzzles/singleton_top_layer.py +295 -0
  931. chia/wallet/puzzles/singleton_top_layer_v1_1.clsp +107 -0
  932. chia/wallet/puzzles/singleton_top_layer_v1_1.clsp.hex +1 -0
  933. chia/wallet/puzzles/singleton_top_layer_v1_1.py +344 -0
  934. chia/wallet/puzzles/singleton_truths.clib +21 -0
  935. chia/wallet/puzzles/tails.py +344 -0
  936. chia/wallet/puzzles/utility_macros.clib +48 -0
  937. chia/wallet/signer_protocol.py +126 -0
  938. chia/wallet/singleton.py +106 -0
  939. chia/wallet/singleton_record.py +30 -0
  940. chia/wallet/trade_manager.py +1088 -0
  941. chia/wallet/trade_record.py +67 -0
  942. chia/wallet/trading/__init__.py +0 -0
  943. chia/wallet/trading/offer.py +703 -0
  944. chia/wallet/trading/trade_status.py +13 -0
  945. chia/wallet/trading/trade_store.py +526 -0
  946. chia/wallet/transaction_record.py +143 -0
  947. chia/wallet/transaction_sorting.py +14 -0
  948. chia/wallet/uncurried_puzzle.py +17 -0
  949. chia/wallet/util/__init__.py +0 -0
  950. chia/wallet/util/address_type.py +55 -0
  951. chia/wallet/util/blind_signer_tl.py +168 -0
  952. chia/wallet/util/clvm_streamable.py +203 -0
  953. chia/wallet/util/compute_hints.py +66 -0
  954. chia/wallet/util/compute_memos.py +45 -0
  955. chia/wallet/util/curry_and_treehash.py +90 -0
  956. chia/wallet/util/debug_spend_bundle.py +234 -0
  957. chia/wallet/util/merkle_tree.py +100 -0
  958. chia/wallet/util/merkle_utils.py +102 -0
  959. chia/wallet/util/new_peak_queue.py +82 -0
  960. chia/wallet/util/notifications.py +12 -0
  961. chia/wallet/util/peer_request_cache.py +174 -0
  962. chia/wallet/util/puzzle_compression.py +96 -0
  963. chia/wallet/util/puzzle_decorator.py +100 -0
  964. chia/wallet/util/puzzle_decorator_type.py +7 -0
  965. chia/wallet/util/query_filter.py +60 -0
  966. chia/wallet/util/transaction_type.py +23 -0
  967. chia/wallet/util/tx_config.py +158 -0
  968. chia/wallet/util/wallet_sync_utils.py +348 -0
  969. chia/wallet/util/wallet_types.py +65 -0
  970. chia/wallet/vc_wallet/__init__.py +0 -0
  971. chia/wallet/vc_wallet/cr_cat_drivers.py +663 -0
  972. chia/wallet/vc_wallet/cr_cat_wallet.py +875 -0
  973. chia/wallet/vc_wallet/cr_outer_puzzle.py +102 -0
  974. chia/wallet/vc_wallet/cr_puzzles/__init__.py +0 -0
  975. chia/wallet/vc_wallet/cr_puzzles/conditions_w_fee_announce.clsp +3 -0
  976. chia/wallet/vc_wallet/cr_puzzles/conditions_w_fee_announce.clsp.hex +1 -0
  977. chia/wallet/vc_wallet/cr_puzzles/credential_restriction.clsp +304 -0
  978. chia/wallet/vc_wallet/cr_puzzles/credential_restriction.clsp.hex +1 -0
  979. chia/wallet/vc_wallet/cr_puzzles/flag_proofs_checker.clsp +45 -0
  980. chia/wallet/vc_wallet/cr_puzzles/flag_proofs_checker.clsp.hex +1 -0
  981. chia/wallet/vc_wallet/vc_drivers.py +838 -0
  982. chia/wallet/vc_wallet/vc_puzzles/__init__.py +0 -0
  983. chia/wallet/vc_wallet/vc_puzzles/covenant_layer.clsp +30 -0
  984. chia/wallet/vc_wallet/vc_puzzles/covenant_layer.clsp.hex +1 -0
  985. chia/wallet/vc_wallet/vc_puzzles/eml_covenant_morpher.clsp +75 -0
  986. chia/wallet/vc_wallet/vc_puzzles/eml_covenant_morpher.clsp.hex +1 -0
  987. chia/wallet/vc_wallet/vc_puzzles/eml_transfer_program_covenant_adapter.clsp +32 -0
  988. chia/wallet/vc_wallet/vc_puzzles/eml_transfer_program_covenant_adapter.clsp.hex +1 -0
  989. chia/wallet/vc_wallet/vc_puzzles/eml_update_metadata_with_DID.clsp +80 -0
  990. chia/wallet/vc_wallet/vc_puzzles/eml_update_metadata_with_DID.clsp.hex +1 -0
  991. chia/wallet/vc_wallet/vc_puzzles/exigent_metadata_layer.clsp +163 -0
  992. chia/wallet/vc_wallet/vc_puzzles/exigent_metadata_layer.clsp.hex +1 -0
  993. chia/wallet/vc_wallet/vc_puzzles/p2_announced_delegated_puzzle.clsp +16 -0
  994. chia/wallet/vc_wallet/vc_puzzles/p2_announced_delegated_puzzle.clsp.hex +1 -0
  995. chia/wallet/vc_wallet/vc_puzzles/standard_vc_backdoor_puzzle.clsp +74 -0
  996. chia/wallet/vc_wallet/vc_puzzles/standard_vc_backdoor_puzzle.clsp.hex +1 -0
  997. chia/wallet/vc_wallet/vc_puzzles/std_parent_morpher.clsp +23 -0
  998. chia/wallet/vc_wallet/vc_puzzles/std_parent_morpher.clsp.hex +1 -0
  999. chia/wallet/vc_wallet/vc_puzzles/viral_backdoor.clsp +64 -0
  1000. chia/wallet/vc_wallet/vc_puzzles/viral_backdoor.clsp.hex +1 -0
  1001. chia/wallet/vc_wallet/vc_store.py +263 -0
  1002. chia/wallet/vc_wallet/vc_wallet.py +638 -0
  1003. chia/wallet/wallet.py +698 -0
  1004. chia/wallet/wallet_action_scope.py +95 -0
  1005. chia/wallet/wallet_blockchain.py +244 -0
  1006. chia/wallet/wallet_coin_record.py +72 -0
  1007. chia/wallet/wallet_coin_store.py +351 -0
  1008. chia/wallet/wallet_info.py +36 -0
  1009. chia/wallet/wallet_interested_store.py +188 -0
  1010. chia/wallet/wallet_nft_store.py +279 -0
  1011. chia/wallet/wallet_node.py +1769 -0
  1012. chia/wallet/wallet_node_api.py +201 -0
  1013. chia/wallet/wallet_pool_store.py +120 -0
  1014. chia/wallet/wallet_protocol.py +90 -0
  1015. chia/wallet/wallet_puzzle_store.py +365 -0
  1016. chia/wallet/wallet_retry_store.py +70 -0
  1017. chia/wallet/wallet_singleton_store.py +258 -0
  1018. chia/wallet/wallet_spend_bundle.py +41 -0
  1019. chia/wallet/wallet_state_manager.py +2820 -0
  1020. chia/wallet/wallet_transaction_store.py +470 -0
  1021. chia/wallet/wallet_user_store.py +110 -0
  1022. chia/wallet/wallet_weight_proof_handler.py +126 -0
  1023. chia_blockchain-2.4.4.dist-info/LICENSE +201 -0
  1024. chia_blockchain-2.4.4.dist-info/METADATA +161 -0
  1025. chia_blockchain-2.4.4.dist-info/RECORD +1028 -0
  1026. chia_blockchain-2.4.4.dist-info/WHEEL +4 -0
  1027. chia_blockchain-2.4.4.dist-info/entry_points.txt +17 -0
  1028. mozilla-ca/cacert.pem +3666 -0
@@ -0,0 +1,1202 @@
1
+ from __future__ import annotations
2
+
3
+ import asyncio
4
+ import contextlib
5
+ import dataclasses
6
+ import io
7
+ import logging
8
+ import os
9
+ import random
10
+ import time
11
+ import traceback
12
+ from concurrent.futures import ThreadPoolExecutor
13
+ from pathlib import Path
14
+ from typing import TYPE_CHECKING, Any, AsyncIterator, ClassVar, Dict, List, Optional, Set, Tuple, cast
15
+
16
+ from chiavdf import create_discriminant, prove
17
+
18
+ from chia.consensus.constants import ConsensusConstants
19
+ from chia.consensus.pot_iterations import calculate_sp_iters, is_overflow_block
20
+ from chia.protocols import timelord_protocol
21
+ from chia.protocols.protocol_message_types import ProtocolMessageTypes
22
+ from chia.rpc.rpc_server import StateChangedProtocol, default_get_connections
23
+ from chia.server.outbound_message import NodeType, make_msg
24
+ from chia.server.server import ChiaServer
25
+ from chia.server.ws_connection import WSChiaConnection
26
+ from chia.timelord.iters_from_block import iters_from_block
27
+ from chia.timelord.timelord_state import LastState
28
+ from chia.timelord.types import Chain, IterationType, StateType
29
+ from chia.types.blockchain_format.classgroup import ClassgroupElement
30
+ from chia.types.blockchain_format.reward_chain_block import RewardChainBlock
31
+ from chia.types.blockchain_format.sized_bytes import bytes32
32
+ from chia.types.blockchain_format.slots import (
33
+ ChallengeChainSubSlot,
34
+ InfusedChallengeChainSubSlot,
35
+ RewardChainSubSlot,
36
+ SubSlotProofs,
37
+ )
38
+ from chia.types.blockchain_format.sub_epoch_summary import SubEpochSummary
39
+ from chia.types.blockchain_format.vdf import VDFInfo, VDFProof, validate_vdf
40
+ from chia.types.end_of_slot_bundle import EndOfSubSlotBundle
41
+ from chia.util.ints import uint8, uint16, uint32, uint64, uint128
42
+ from chia.util.streamable import Streamable, streamable
43
+
44
+ log = logging.getLogger(__name__)
45
+
46
+
47
+ @streamable
48
+ @dataclasses.dataclass(frozen=True)
49
+ class BlueboxProcessData(Streamable):
50
+ challenge: bytes32
51
+ size_bits: uint16
52
+ iters: uint64
53
+
54
+
55
+ def prove_bluebox_slow(payload: bytes) -> bytes:
56
+ bluebox_process_data = BlueboxProcessData.from_bytes(payload)
57
+ initial_el = b"\x08" + (b"\x00" * 99)
58
+ return cast(
59
+ bytes,
60
+ prove(
61
+ bluebox_process_data.challenge,
62
+ initial_el,
63
+ bluebox_process_data.size_bits,
64
+ bluebox_process_data.iters,
65
+ ),
66
+ )
67
+
68
+
69
+ class Timelord:
70
+ if TYPE_CHECKING:
71
+ from chia.rpc.rpc_server import RpcServiceProtocol
72
+
73
+ _protocol_check: ClassVar[RpcServiceProtocol] = cast("Timelord", None)
74
+
75
+ @property
76
+ def server(self) -> ChiaServer:
77
+ # This is a stop gap until the class usage is refactored such the values of
78
+ # integral attributes are known at creation of the instance.
79
+ if self._server is None:
80
+ raise RuntimeError("server not assigned")
81
+
82
+ return self._server
83
+
84
+ def __init__(self, root_path: Path, config: Dict[str, Any], constants: ConsensusConstants) -> None:
85
+ self.config = config
86
+ self.root_path = root_path
87
+ self.constants = constants
88
+ self._shut_down = False
89
+ self.free_clients: List[Tuple[str, asyncio.StreamReader, asyncio.StreamWriter]] = []
90
+ self.ip_whitelist = self.config["vdf_clients"]["ip"]
91
+ self._server: Optional[ChiaServer] = None
92
+ self.chain_type_to_stream: Dict[Chain, Tuple[str, asyncio.StreamReader, asyncio.StreamWriter]] = {}
93
+ self.chain_start_time: Dict[Chain, float] = {}
94
+ # Chains that currently don't have a vdf_client.
95
+ self.unspawned_chains: List[Chain] = [
96
+ Chain.CHALLENGE_CHAIN,
97
+ Chain.REWARD_CHAIN,
98
+ Chain.INFUSED_CHALLENGE_CHAIN,
99
+ ]
100
+ # Chains that currently accept iterations.
101
+ self.allows_iters: List[Chain] = []
102
+ # Last peak received, None if it's already processed.
103
+ self.new_peak: Optional[timelord_protocol.NewPeakTimelord] = None
104
+ # Last state received. Can either be a new peak or a new EndOfSubslotBundle.
105
+ # Unfinished block info, iters adjusted to the last peak.
106
+ self.unfinished_blocks: List[timelord_protocol.NewUnfinishedBlockTimelord] = []
107
+ # Signage points iters, adjusted to the last peak.
108
+ self.signage_point_iters: List[Tuple[uint64, uint8]] = []
109
+ # For each chain, send those info when the process spawns.
110
+ self.iters_to_submit: Dict[Chain, List[uint64]] = {}
111
+ self.iters_submitted: Dict[Chain, List[uint64]] = {}
112
+ self.iters_finished: Set[uint64] = set()
113
+ # For each iteration submitted, know if it's a signage point, an infusion point or an end of slot.
114
+ self.iteration_to_proof_type: Dict[uint64, IterationType] = {}
115
+ # List of proofs finished.
116
+ self.proofs_finished: List[Tuple[Chain, VDFInfo, VDFProof, int]] = []
117
+ # Data to send at vdf_client initialization.
118
+ self.overflow_blocks: List[timelord_protocol.NewUnfinishedBlockTimelord] = []
119
+ # Incremented each time `reset_chains` has been called.
120
+ # Used to label proofs in `finished_proofs` and to only filter proofs corresponding to the most recent state.
121
+ self.num_resets: int = 0
122
+
123
+ self.process_communication_tasks: List[asyncio.Task[None]] = []
124
+ self.main_loop: Optional[asyncio.Task[None]] = None
125
+ self.vdf_server: Optional[asyncio.base_events.Server] = None
126
+ self._shut_down = False
127
+ self.vdf_failures: List[Tuple[Chain, Optional[int]]] = []
128
+ self.vdf_failures_count: int = 0
129
+ self.vdf_failure_time: float = 0
130
+ self.total_unfinished: int = 0
131
+ self.total_infused: int = 0
132
+ self.state_changed_callback: Optional[StateChangedProtocol] = None
133
+ self.bluebox_mode = self.config.get("bluebox_mode", False)
134
+ # Support backwards compatibility for the old `config.yaml` that has field `sanitizer_mode`.
135
+ if not self.bluebox_mode:
136
+ self.bluebox_mode = self.config.get("sanitizer_mode", False)
137
+ self.pending_bluebox_info: List[Tuple[float, timelord_protocol.RequestCompactProofOfTime]] = []
138
+ self.last_active_time = time.time()
139
+ self.max_allowed_inactivity_time = 60
140
+ self.bluebox_pool: Optional[ThreadPoolExecutor] = None
141
+
142
+ @contextlib.asynccontextmanager
143
+ async def manage(self) -> AsyncIterator[None]:
144
+ self.lock: asyncio.Lock = asyncio.Lock()
145
+ self.vdf_server = await asyncio.start_server(
146
+ self._handle_client,
147
+ self.config["vdf_server"]["host"],
148
+ int(self.config["vdf_server"]["port"]),
149
+ )
150
+ self.last_state: LastState = LastState(self.constants)
151
+ slow_bluebox = self.config.get("slow_bluebox", False)
152
+ if not self.bluebox_mode:
153
+ self.main_loop = asyncio.create_task(self._manage_chains())
154
+ else:
155
+ if os.name == "nt" or slow_bluebox:
156
+ # `vdf_client` doesn't build on windows, use `prove()` from chiavdf.
157
+ workers = self.config.get("slow_bluebox_process_count", 1)
158
+ self.bluebox_pool = ThreadPoolExecutor(
159
+ max_workers=workers,
160
+ )
161
+ self.main_loop = asyncio.create_task(
162
+ self._start_manage_discriminant_queue_sanitizer_slow(self.bluebox_pool, workers)
163
+ )
164
+ else:
165
+ self.main_loop = asyncio.create_task(self._manage_discriminant_queue_sanitizer())
166
+ log.info(f"Started timelord, listening on port {self.get_vdf_server_port()}")
167
+ try:
168
+ yield
169
+ finally:
170
+ self._shut_down = True
171
+ for task in self.process_communication_tasks:
172
+ task.cancel()
173
+ if self.main_loop is not None:
174
+ self.main_loop.cancel()
175
+ if self.bluebox_pool is not None:
176
+ self.bluebox_pool.shutdown()
177
+
178
+ def get_connections(self, request_node_type: Optional[NodeType]) -> List[Dict[str, Any]]:
179
+ return default_get_connections(server=self.server, request_node_type=request_node_type)
180
+
181
+ async def on_connect(self, connection: WSChiaConnection) -> None:
182
+ pass
183
+
184
+ def get_vdf_server_port(self) -> Optional[uint16]:
185
+ if self.vdf_server is not None:
186
+ return uint16(self.vdf_server.sockets[0].getsockname()[1])
187
+ return None
188
+
189
+ def _set_state_changed_callback(self, callback: StateChangedProtocol) -> None:
190
+ self.state_changed_callback = callback
191
+
192
+ def state_changed(self, change: str, change_data: Optional[Dict[str, Any]] = None) -> None:
193
+ if self.state_changed_callback is not None:
194
+ self.state_changed_callback(change, change_data)
195
+
196
+ def set_server(self, server: ChiaServer) -> None:
197
+ self._server = server
198
+
199
+ async def _handle_client(self, reader: asyncio.StreamReader, writer: asyncio.StreamWriter) -> None:
200
+ async with self.lock:
201
+ client_ip = writer.get_extra_info("peername")[0]
202
+ log.debug(f"New timelord connection from client: {client_ip}.")
203
+ if client_ip in self.ip_whitelist:
204
+ self.free_clients.append((client_ip, reader, writer))
205
+ log.debug(f"Added new VDF client {client_ip}.")
206
+
207
+ async def _stop_chain(self, chain: Chain) -> None:
208
+ try:
209
+ _, _, stop_writer = self.chain_type_to_stream[chain]
210
+ if chain in self.allows_iters:
211
+ stop_writer.write(b"010")
212
+ await stop_writer.drain()
213
+ self.allows_iters.remove(chain)
214
+ else:
215
+ log.error(f"Trying to stop {chain} before its initialization.")
216
+ stop_writer.close()
217
+ await stop_writer.wait_closed()
218
+ if chain not in self.unspawned_chains:
219
+ self.unspawned_chains.append(chain)
220
+ del self.chain_type_to_stream[chain]
221
+ except ConnectionResetError as e:
222
+ log.error(f"{e}")
223
+ except Exception as e:
224
+ log.error(f"Exception in stop chain: {type(e)} {e}")
225
+
226
+ def get_height(self) -> uint32:
227
+ if self.last_state.state_type == StateType.FIRST_SUB_SLOT:
228
+ return uint32(0)
229
+ else:
230
+ return uint32(self.last_state.get_height() + 1)
231
+
232
+ def _can_infuse_unfinished_block(self, block: timelord_protocol.NewUnfinishedBlockTimelord) -> Optional[uint64]:
233
+ assert self.last_state is not None
234
+ sub_slot_iters = self.last_state.get_sub_slot_iters()
235
+ difficulty = self.last_state.get_difficulty()
236
+ ip_iters = self.last_state.get_last_ip()
237
+ rc_block = block.reward_chain_block
238
+ try:
239
+ block_sp_iters, block_ip_iters = iters_from_block(
240
+ self.constants,
241
+ rc_block,
242
+ sub_slot_iters,
243
+ difficulty,
244
+ self.get_height(),
245
+ )
246
+ except Exception as e:
247
+ log.warning(f"Received invalid unfinished block: {e}.")
248
+ return None
249
+ block_sp_total_iters = self.last_state.total_iters - ip_iters + block_sp_iters
250
+ if is_overflow_block(self.constants, block.reward_chain_block.signage_point_index):
251
+ block_sp_total_iters -= self.last_state.get_sub_slot_iters()
252
+ found_index = -1
253
+ for index, (rc, total_iters) in enumerate(self.last_state.reward_challenge_cache):
254
+ if rc == block.rc_prev:
255
+ found_index = index
256
+ break
257
+ if found_index == -1:
258
+ log.warning(f"Will not infuse {block.rc_prev} because its reward chain challenge is not in the chain")
259
+ return None
260
+ if ip_iters > block_ip_iters:
261
+ log.warning("Too late to infuse block")
262
+ return None
263
+
264
+ new_block_iters = uint64(block_ip_iters - ip_iters)
265
+ if len(self.last_state.reward_challenge_cache) > found_index + 1:
266
+ if self.last_state.reward_challenge_cache[found_index + 1][1] < block_sp_total_iters:
267
+ log.warning(
268
+ f"Will not infuse unfinished block {block.rc_prev} sp total iters {block_sp_total_iters}, "
269
+ f"because there is another infusion before its SP"
270
+ )
271
+ return None
272
+ if self.last_state.reward_challenge_cache[found_index][1] > block_sp_total_iters:
273
+ if not is_overflow_block(self.constants, block.reward_chain_block.signage_point_index):
274
+ log.error(
275
+ f"Will not infuse unfinished block {block.rc_prev}, sp total iters: {block_sp_total_iters}, "
276
+ f"because its iters are too low"
277
+ )
278
+ return None
279
+
280
+ if new_block_iters > 0:
281
+ return new_block_iters
282
+ return None
283
+
284
+ async def _reset_chains(self, *, first_run: bool = False, only_eos: bool = False) -> None:
285
+ # First, stop all chains.
286
+ self.last_active_time = time.time()
287
+ log.debug("Resetting chains")
288
+ ip_iters = self.last_state.get_last_ip()
289
+ sub_slot_iters = self.last_state.get_sub_slot_iters()
290
+
291
+ if not first_run:
292
+ for chain in list(self.chain_type_to_stream.keys()):
293
+ await self._stop_chain(chain)
294
+
295
+ # Adjust all signage points iterations to the peak.
296
+ iters_per_signage = uint64(sub_slot_iters // self.constants.NUM_SPS_SUB_SLOT)
297
+ self.signage_point_iters = [
298
+ (uint64(k * iters_per_signage - ip_iters), uint8(k))
299
+ for k in range(1, self.constants.NUM_SPS_SUB_SLOT)
300
+ if k * iters_per_signage - ip_iters > 0
301
+ ]
302
+ for sp, k in self.signage_point_iters:
303
+ assert k * iters_per_signage > 0
304
+ assert k * iters_per_signage < sub_slot_iters
305
+ # Adjust all unfinished blocks iterations to the peak.
306
+ new_unfinished_blocks = []
307
+ self.iters_finished = set()
308
+ self.proofs_finished = []
309
+ self.num_resets += 1
310
+ for chain in [Chain.CHALLENGE_CHAIN, Chain.REWARD_CHAIN, Chain.INFUSED_CHALLENGE_CHAIN]:
311
+ self.iters_to_submit[chain] = []
312
+ self.iters_submitted[chain] = []
313
+ self.iteration_to_proof_type = {}
314
+ if not only_eos:
315
+ for block in self.unfinished_blocks + self.overflow_blocks:
316
+ new_block_iters: Optional[uint64] = self._can_infuse_unfinished_block(block)
317
+ # Does not add duplicates, or blocks that we cannot infuse
318
+ if new_block_iters and new_block_iters not in self.iters_to_submit[Chain.CHALLENGE_CHAIN]:
319
+ if block not in self.unfinished_blocks:
320
+ self.total_unfinished += 1
321
+ new_unfinished_blocks.append(block)
322
+ for chain in [Chain.REWARD_CHAIN, Chain.CHALLENGE_CHAIN]:
323
+ self.iters_to_submit[chain].append(new_block_iters)
324
+ if self.last_state.get_deficit() < self.constants.MIN_BLOCKS_PER_CHALLENGE_BLOCK:
325
+ self.iters_to_submit[Chain.INFUSED_CHALLENGE_CHAIN].append(new_block_iters)
326
+ self.iteration_to_proof_type[new_block_iters] = IterationType.INFUSION_POINT
327
+ # Remove all unfinished blocks that have already passed.
328
+ self.unfinished_blocks = new_unfinished_blocks
329
+ # Signage points.
330
+ if not only_eos and len(self.signage_point_iters) > 0:
331
+ count_signage = 0
332
+ for signage, k in self.signage_point_iters:
333
+ for chain in [Chain.CHALLENGE_CHAIN, Chain.REWARD_CHAIN]:
334
+ self.iters_to_submit[chain].append(signage)
335
+ self.iteration_to_proof_type[signage] = IterationType.SIGNAGE_POINT
336
+ count_signage += 1
337
+ if count_signage == 3:
338
+ break
339
+ left_subslot_iters = uint64(sub_slot_iters - ip_iters)
340
+ assert left_subslot_iters > 0
341
+
342
+ if self.last_state.get_deficit() < self.constants.MIN_BLOCKS_PER_CHALLENGE_BLOCK:
343
+ self.iters_to_submit[Chain.INFUSED_CHALLENGE_CHAIN].append(left_subslot_iters)
344
+ self.iters_to_submit[Chain.CHALLENGE_CHAIN].append(left_subslot_iters)
345
+ self.iters_to_submit[Chain.REWARD_CHAIN].append(left_subslot_iters)
346
+ self.iteration_to_proof_type[left_subslot_iters] = IterationType.END_OF_SUBSLOT
347
+
348
+ for chain, iters in self.iters_to_submit.items():
349
+ for iteration in iters:
350
+ assert iteration > 0
351
+
352
+ async def _handle_new_peak(self) -> None:
353
+ assert self.new_peak is not None
354
+ self.last_state.set_state(self.new_peak)
355
+
356
+ if self.total_unfinished > 0:
357
+ remove_unfinished = []
358
+ for unf_block_timelord in self.unfinished_blocks + self.overflow_blocks:
359
+ if (
360
+ unf_block_timelord.reward_chain_block.get_hash()
361
+ == self.new_peak.reward_chain_block.get_unfinished().get_hash()
362
+ ):
363
+ if unf_block_timelord not in self.unfinished_blocks:
364
+ # We never got the EOS for this, but we have the block in overflow list
365
+ self.total_unfinished += 1
366
+
367
+ remove_unfinished.append(unf_block_timelord)
368
+ if len(remove_unfinished) > 0:
369
+ self.total_infused += 1
370
+ for block in remove_unfinished:
371
+ if block in self.unfinished_blocks:
372
+ self.unfinished_blocks.remove(block)
373
+ if block in self.overflow_blocks:
374
+ self.overflow_blocks.remove(block)
375
+ infusion_rate = round(self.total_infused / self.total_unfinished * 100.0, 2)
376
+ log.info(
377
+ f"Total unfinished blocks: {self.total_unfinished}. "
378
+ f"Total infused blocks: {self.total_infused}. "
379
+ f"Infusion rate: {infusion_rate}%."
380
+ )
381
+
382
+ self.new_peak = None
383
+ await self._reset_chains()
384
+
385
+ async def _map_chains_with_vdf_clients(self) -> None:
386
+ while not self._shut_down:
387
+ picked_chain = None
388
+ async with self.lock:
389
+ if len(self.free_clients) == 0:
390
+ break
391
+ ip, reader, writer = self.free_clients[0]
392
+ for chain_type in self.unspawned_chains:
393
+ challenge = self.last_state.get_challenge(chain_type)
394
+ initial_form = self.last_state.get_initial_form(chain_type)
395
+ if challenge is not None and initial_form is not None:
396
+ picked_chain = chain_type
397
+ break
398
+ if picked_chain is None:
399
+ break
400
+ picked_chain = self.unspawned_chains[0]
401
+ self.chain_type_to_stream[picked_chain] = (ip, reader, writer)
402
+ self.free_clients = self.free_clients[1:]
403
+ self.unspawned_chains = self.unspawned_chains[1:]
404
+ self.chain_start_time[picked_chain] = time.time()
405
+
406
+ log.debug(f"Mapping free vdf_client with chain: {picked_chain}.")
407
+ assert challenge is not None
408
+ assert initial_form is not None
409
+ self.process_communication_tasks.append(
410
+ asyncio.create_task(
411
+ self._do_process_communication(
412
+ picked_chain, challenge, initial_form, ip, reader, writer, proof_label=self.num_resets
413
+ )
414
+ )
415
+ )
416
+
417
+ async def _submit_iterations(self) -> None:
418
+ for chain in [Chain.CHALLENGE_CHAIN, Chain.REWARD_CHAIN, Chain.INFUSED_CHALLENGE_CHAIN]:
419
+ if chain in self.allows_iters:
420
+ _, _, writer = self.chain_type_to_stream[chain]
421
+ for iteration in self.iters_to_submit[chain]:
422
+ if iteration in self.iters_submitted[chain]:
423
+ continue
424
+ self.iters_submitted[chain].append(iteration)
425
+ log.debug(f"Submitting iterations to {chain}: {iteration}")
426
+ assert iteration > 0
427
+ prefix = str(len(str(iteration)))
428
+ if len(str(iteration)) < 10:
429
+ prefix = "0" + prefix
430
+ iter_str = prefix + str(iteration)
431
+ writer.write(iter_str.encode())
432
+ await writer.drain()
433
+
434
+ def _clear_proof_list(self, iters: uint64) -> List[Tuple[Chain, VDFInfo, VDFProof, int]]:
435
+ return [
436
+ (chain, info, proof, label)
437
+ for chain, info, proof, label in self.proofs_finished
438
+ if info.number_of_iterations != iters
439
+ ]
440
+
441
+ async def _check_for_new_sp(self, iter_to_look_for: uint64) -> None:
442
+ signage_iters = [
443
+ iteration for iteration, t in self.iteration_to_proof_type.items() if t == IterationType.SIGNAGE_POINT
444
+ ]
445
+ if len(signage_iters) == 0:
446
+ return
447
+ to_remove = []
448
+ for potential_sp_iters, signage_point_index in self.signage_point_iters:
449
+ if potential_sp_iters not in signage_iters or potential_sp_iters != iter_to_look_for:
450
+ continue
451
+ signage_iter = potential_sp_iters
452
+ proofs_with_iter = [
453
+ (chain, info, proof)
454
+ for chain, info, proof, label in self.proofs_finished
455
+ if info.number_of_iterations == signage_iter and label == self.num_resets
456
+ ]
457
+ # Wait for both cc and rc to have the signage point.
458
+ if len(proofs_with_iter) == 2:
459
+ cc_info: Optional[VDFInfo] = None
460
+ cc_proof: Optional[VDFProof] = None
461
+ rc_info: Optional[VDFInfo] = None
462
+ rc_proof: Optional[VDFProof] = None
463
+ for chain, info, proof in proofs_with_iter:
464
+ if chain == Chain.CHALLENGE_CHAIN:
465
+ cc_info = info
466
+ cc_proof = proof
467
+ if chain == Chain.REWARD_CHAIN:
468
+ rc_info = info
469
+ rc_proof = proof
470
+ if cc_info is None or cc_proof is None or rc_info is None or rc_proof is None:
471
+ log.error(f"Insufficient signage point data {signage_iter}")
472
+ continue
473
+ self.iters_finished.add(iter_to_look_for)
474
+ self.last_active_time = time.time()
475
+
476
+ rc_challenge = self.last_state.get_challenge(Chain.REWARD_CHAIN)
477
+ if rc_info.challenge != rc_challenge:
478
+ assert rc_challenge is not None
479
+ log.warning(f"SP: Do not have correct challenge {rc_challenge.hex()} has {rc_info.challenge.hex()}")
480
+ # This proof is on an outdated challenge, so don't use it
481
+ continue
482
+ iters_from_sub_slot_start = uint64(cc_info.number_of_iterations + self.last_state.get_last_ip())
483
+ response = timelord_protocol.NewSignagePointVDF(
484
+ signage_point_index,
485
+ cc_info.replace(number_of_iterations=iters_from_sub_slot_start),
486
+ cc_proof,
487
+ rc_info,
488
+ rc_proof,
489
+ )
490
+ if self._server is not None:
491
+ msg = make_msg(ProtocolMessageTypes.new_signage_point_vdf, response)
492
+ await self.server.send_to_all([msg], NodeType.FULL_NODE)
493
+ # Cleanup the signage point from memory.
494
+ to_remove.append((signage_iter, signage_point_index))
495
+
496
+ self.proofs_finished = self._clear_proof_list(signage_iter)
497
+ # Send the next 3 signage point to the chains.
498
+ next_iters_count = 0
499
+ for next_sp, k in self.signage_point_iters:
500
+ for chain in [Chain.CHALLENGE_CHAIN, Chain.REWARD_CHAIN]:
501
+ if next_sp not in self.iters_submitted[chain] and next_sp not in self.iters_to_submit[chain]:
502
+ self.iters_to_submit[chain].append(next_sp)
503
+ self.iteration_to_proof_type[next_sp] = IterationType.SIGNAGE_POINT
504
+ next_iters_count += 1
505
+ if next_iters_count == 10:
506
+ break
507
+
508
+ # Break so we alternate between checking SP and IP
509
+ break
510
+ for r in to_remove:
511
+ self.signage_point_iters.remove(r)
512
+
513
+ async def _check_for_new_ip(self, iter_to_look_for: uint64) -> None:
514
+ if len(self.unfinished_blocks) == 0:
515
+ return
516
+ infusion_iters = [
517
+ iteration for iteration, t in self.iteration_to_proof_type.items() if t == IterationType.INFUSION_POINT
518
+ ]
519
+ for iteration in infusion_iters:
520
+ if iteration != iter_to_look_for:
521
+ continue
522
+ proofs_with_iter = [
523
+ (chain, info, proof)
524
+ for chain, info, proof, label in self.proofs_finished
525
+ if info.number_of_iterations == iteration and label == self.num_resets
526
+ ]
527
+ if self.last_state.get_challenge(Chain.INFUSED_CHALLENGE_CHAIN) is not None:
528
+ chain_count = 3
529
+ else:
530
+ chain_count = 2
531
+ if len(proofs_with_iter) == chain_count:
532
+ block = None
533
+ ip_iters = None
534
+ for unfinished_block in self.unfinished_blocks:
535
+ try:
536
+ _, ip_iters = iters_from_block(
537
+ self.constants,
538
+ unfinished_block.reward_chain_block,
539
+ self.last_state.get_sub_slot_iters(),
540
+ self.last_state.get_difficulty(),
541
+ self.get_height(),
542
+ )
543
+ except Exception as e:
544
+ log.error(f"Error {e}")
545
+ continue
546
+ if ip_iters - self.last_state.get_last_ip() == iteration:
547
+ block = unfinished_block
548
+ break
549
+ assert ip_iters is not None
550
+ if block is not None:
551
+ ip_total_iters = self.last_state.get_total_iters() + iteration
552
+ challenge = block.reward_chain_block.get_hash()
553
+ icc_info: Optional[VDFInfo] = None
554
+ icc_proof: Optional[VDFProof] = None
555
+ cc_info: Optional[VDFInfo] = None
556
+ cc_proof: Optional[VDFProof] = None
557
+ rc_info: Optional[VDFInfo] = None
558
+ rc_proof: Optional[VDFProof] = None
559
+ for chain, info, proof in proofs_with_iter:
560
+ if chain == Chain.CHALLENGE_CHAIN:
561
+ cc_info = info
562
+ cc_proof = proof
563
+ if chain == Chain.REWARD_CHAIN:
564
+ rc_info = info
565
+ rc_proof = proof
566
+ if chain == Chain.INFUSED_CHALLENGE_CHAIN:
567
+ icc_info = info
568
+ icc_proof = proof
569
+ if cc_info is None or cc_proof is None or rc_info is None or rc_proof is None:
570
+ log.error(f"Insufficient VDF proofs for infusion point ch: {challenge} iterations:{iteration}")
571
+ return
572
+
573
+ rc_challenge = self.last_state.get_challenge(Chain.REWARD_CHAIN)
574
+ if rc_info.challenge != rc_challenge:
575
+ assert rc_challenge is not None
576
+ log.warning(
577
+ f"Do not have correct challenge {rc_challenge.hex()} "
578
+ f"has {rc_info.challenge.hex()}, partial hash {block.reward_chain_block.get_hash()}"
579
+ )
580
+ # This proof is on an outdated challenge, so don't use it
581
+ continue
582
+
583
+ self.iters_finished.add(iter_to_look_for)
584
+ self.last_active_time = time.time()
585
+ log.debug(f"Generated infusion point for challenge: {challenge} iterations: {iteration}.")
586
+
587
+ overflow = is_overflow_block(self.constants, block.reward_chain_block.signage_point_index)
588
+
589
+ if not self.last_state.can_infuse_block(overflow):
590
+ log.warning("Too many blocks, or overflow in new epoch, cannot infuse, discarding")
591
+ return
592
+
593
+ cc_info = cc_info.replace(number_of_iterations=ip_iters)
594
+ response = timelord_protocol.NewInfusionPointVDF(
595
+ challenge,
596
+ cc_info,
597
+ cc_proof,
598
+ rc_info,
599
+ rc_proof,
600
+ icc_info,
601
+ icc_proof,
602
+ )
603
+ msg = make_msg(ProtocolMessageTypes.new_infusion_point_vdf, response)
604
+ if self._server is not None:
605
+ await self.server.send_to_all([msg], NodeType.FULL_NODE)
606
+
607
+ self.proofs_finished = self._clear_proof_list(iteration)
608
+
609
+ if (
610
+ self.last_state.get_last_block_total_iters() is None
611
+ and not self.last_state.state_type == StateType.FIRST_SUB_SLOT
612
+ ):
613
+ # We don't know when the last block was, so we can't make peaks
614
+ return
615
+
616
+ sp_total_iters = (
617
+ ip_total_iters
618
+ - ip_iters
619
+ + calculate_sp_iters(
620
+ self.constants,
621
+ block.sub_slot_iters,
622
+ block.reward_chain_block.signage_point_index,
623
+ )
624
+ - (block.sub_slot_iters if overflow else 0)
625
+ )
626
+ if self.last_state.state_type == StateType.FIRST_SUB_SLOT:
627
+ is_transaction_block = True
628
+ height: uint32 = uint32(0)
629
+ else:
630
+ last_block_ti = self.last_state.get_last_block_total_iters()
631
+ assert last_block_ti is not None
632
+ is_transaction_block = last_block_ti < sp_total_iters
633
+ height = uint32(self.last_state.get_height() + 1)
634
+
635
+ if height < 5:
636
+ # Don't directly update our state for the first few blocks, because we cannot validate
637
+ # whether the pre-farm is correct
638
+ return
639
+
640
+ new_reward_chain_block = RewardChainBlock(
641
+ uint128(self.last_state.get_weight() + block.difficulty),
642
+ height,
643
+ uint128(ip_total_iters),
644
+ block.reward_chain_block.signage_point_index,
645
+ block.reward_chain_block.pos_ss_cc_challenge_hash,
646
+ block.reward_chain_block.proof_of_space,
647
+ block.reward_chain_block.challenge_chain_sp_vdf,
648
+ block.reward_chain_block.challenge_chain_sp_signature,
649
+ cc_info,
650
+ block.reward_chain_block.reward_chain_sp_vdf,
651
+ block.reward_chain_block.reward_chain_sp_signature,
652
+ rc_info,
653
+ icc_info,
654
+ is_transaction_block,
655
+ )
656
+ if self.last_state.state_type == StateType.FIRST_SUB_SLOT:
657
+ # Genesis
658
+ new_deficit = self.constants.MIN_BLOCKS_PER_CHALLENGE_BLOCK - 1
659
+ elif overflow and self.last_state.deficit == self.constants.MIN_BLOCKS_PER_CHALLENGE_BLOCK:
660
+ if self.last_state.peak is not None:
661
+ assert self.last_state.subslot_end is None
662
+ # This means the previous block is also an overflow block, and did not manage
663
+ # to lower the deficit, therefore we cannot lower it either. (new slot)
664
+ new_deficit = self.constants.MIN_BLOCKS_PER_CHALLENGE_BLOCK
665
+ else:
666
+ # This means we are the first infusion in this sub-slot. This may be a new slot or not.
667
+ assert self.last_state.subslot_end is not None
668
+ if self.last_state.subslot_end.infused_challenge_chain is None:
669
+ # There is no ICC, which means we are not finishing a slot. We can reduce the deficit.
670
+ new_deficit = self.constants.MIN_BLOCKS_PER_CHALLENGE_BLOCK - 1
671
+ else:
672
+ # There is an ICC, which means we are finishing a slot. Different slot, so can't change
673
+ # the deficit
674
+ new_deficit = self.constants.MIN_BLOCKS_PER_CHALLENGE_BLOCK
675
+ else:
676
+ new_deficit = max(self.last_state.deficit - 1, 0)
677
+
678
+ if new_deficit == self.constants.MIN_BLOCKS_PER_CHALLENGE_BLOCK - 1:
679
+ last_csb_or_eos = ip_total_iters
680
+ else:
681
+ last_csb_or_eos = self.last_state.last_challenge_sb_or_eos_total_iters
682
+
683
+ if self.last_state.just_infused_sub_epoch_summary():
684
+ new_sub_epoch_summary = None
685
+ passed_ses_height_but_not_yet_included = False
686
+ else:
687
+ new_sub_epoch_summary = block.sub_epoch_summary
688
+ if new_reward_chain_block.height % self.constants.SUB_EPOCH_BLOCKS == 0:
689
+ passed_ses_height_but_not_yet_included = True
690
+ else:
691
+ passed_ses_height_but_not_yet_included = (
692
+ self.last_state.get_passed_ses_height_but_not_yet_included()
693
+ )
694
+
695
+ self.new_peak = timelord_protocol.NewPeakTimelord(
696
+ new_reward_chain_block,
697
+ block.difficulty,
698
+ uint8(new_deficit),
699
+ block.sub_slot_iters,
700
+ new_sub_epoch_summary,
701
+ self.last_state.reward_challenge_cache,
702
+ uint128(last_csb_or_eos),
703
+ passed_ses_height_but_not_yet_included,
704
+ )
705
+
706
+ await self._handle_new_peak()
707
+ # Break so we alternate between checking SP and IP
708
+ break
709
+
710
+ async def _check_for_end_of_subslot(self, iter_to_look_for: uint64) -> None:
711
+ left_subslot_iters = [
712
+ iteration for iteration, t in self.iteration_to_proof_type.items() if t == IterationType.END_OF_SUBSLOT
713
+ ]
714
+ if len(left_subslot_iters) == 0:
715
+ return
716
+ if left_subslot_iters[0] != iter_to_look_for:
717
+ return
718
+ chains_finished = [
719
+ (chain, info, proof)
720
+ for chain, info, proof, label in self.proofs_finished
721
+ if info.number_of_iterations == left_subslot_iters[0] and label == self.num_resets
722
+ ]
723
+ if self.last_state.get_challenge(Chain.INFUSED_CHALLENGE_CHAIN) is not None:
724
+ chain_count = 3
725
+ else:
726
+ chain_count = 2
727
+ if len(chains_finished) == chain_count:
728
+ icc_ip_vdf: Optional[VDFInfo] = None
729
+ icc_ip_proof: Optional[VDFProof] = None
730
+ cc_vdf: Optional[VDFInfo] = None
731
+ cc_proof: Optional[VDFProof] = None
732
+ rc_vdf: Optional[VDFInfo] = None
733
+ rc_proof: Optional[VDFProof] = None
734
+ for chain, info, proof in chains_finished:
735
+ if chain == Chain.CHALLENGE_CHAIN:
736
+ cc_vdf = info
737
+ cc_proof = proof
738
+ if chain == Chain.REWARD_CHAIN:
739
+ rc_vdf = info
740
+ rc_proof = proof
741
+ if chain == Chain.INFUSED_CHALLENGE_CHAIN:
742
+ icc_ip_vdf = info
743
+ icc_ip_proof = proof
744
+ assert cc_proof is not None and rc_proof is not None and cc_vdf is not None and rc_vdf is not None
745
+
746
+ rc_challenge = self.last_state.get_challenge(Chain.REWARD_CHAIN)
747
+ if rc_vdf.challenge != rc_challenge:
748
+ assert rc_challenge is not None
749
+ log.warning(f"Do not have correct challenge {rc_challenge.hex()} has {rc_vdf.challenge.hex()}")
750
+ # This proof is on an outdated challenge, so don't use it
751
+ return
752
+ log.debug("Collected end of subslot vdfs.")
753
+ self.iters_finished.add(iter_to_look_for)
754
+ self.last_active_time = time.time()
755
+ iters_from_sub_slot_start = uint64(cc_vdf.number_of_iterations + self.last_state.get_last_ip())
756
+ cc_vdf = cc_vdf.replace(number_of_iterations=iters_from_sub_slot_start)
757
+ if icc_ip_vdf is not None:
758
+ if self.last_state.peak is not None:
759
+ total_iters = (
760
+ self.last_state.get_total_iters()
761
+ - self.last_state.get_last_ip()
762
+ + self.last_state.get_sub_slot_iters()
763
+ )
764
+ else:
765
+ total_iters = self.last_state.get_total_iters() + self.last_state.get_sub_slot_iters()
766
+ iters_from_cb = uint64(total_iters - self.last_state.last_challenge_sb_or_eos_total_iters)
767
+ if iters_from_cb > self.last_state.sub_slot_iters:
768
+ log.error(f"{self.last_state.peak}")
769
+ log.error(f"{self.last_state.subslot_end}")
770
+ assert False
771
+ assert iters_from_cb <= self.last_state.sub_slot_iters
772
+ icc_ip_vdf = icc_ip_vdf.replace(number_of_iterations=iters_from_cb)
773
+
774
+ icc_sub_slot: Optional[InfusedChallengeChainSubSlot] = (
775
+ None if icc_ip_vdf is None else InfusedChallengeChainSubSlot(icc_ip_vdf)
776
+ )
777
+ icc_sub_slot_hash: Optional[bytes32]
778
+ if self.last_state.get_deficit() == 0:
779
+ assert icc_sub_slot is not None
780
+ icc_sub_slot_hash = icc_sub_slot.get_hash()
781
+ else:
782
+ icc_sub_slot_hash = None
783
+ next_ses: Optional[SubEpochSummary] = self.last_state.get_next_sub_epoch_summary()
784
+ ses_hash: Optional[bytes32]
785
+ if next_ses is not None:
786
+ log.info(f"Including sub epoch summary{next_ses}")
787
+ ses_hash = next_ses.get_hash()
788
+ new_sub_slot_iters = next_ses.new_sub_slot_iters
789
+ new_difficulty = next_ses.new_difficulty
790
+ else:
791
+ ses_hash = None
792
+ new_sub_slot_iters = None
793
+ new_difficulty = None
794
+ cc_sub_slot = ChallengeChainSubSlot(cc_vdf, icc_sub_slot_hash, ses_hash, new_sub_slot_iters, new_difficulty)
795
+ eos_deficit: uint8 = (
796
+ self.last_state.get_deficit()
797
+ if self.constants.MIN_BLOCKS_PER_CHALLENGE_BLOCK > self.last_state.get_deficit() > 0
798
+ else self.constants.MIN_BLOCKS_PER_CHALLENGE_BLOCK
799
+ )
800
+ rc_sub_slot = RewardChainSubSlot(
801
+ rc_vdf,
802
+ cc_sub_slot.get_hash(),
803
+ icc_sub_slot.get_hash() if icc_sub_slot is not None else None,
804
+ eos_deficit,
805
+ )
806
+ eos_bundle = EndOfSubSlotBundle(
807
+ cc_sub_slot,
808
+ icc_sub_slot,
809
+ rc_sub_slot,
810
+ SubSlotProofs(cc_proof, icc_ip_proof, rc_proof),
811
+ )
812
+ if self._server is not None:
813
+ msg = make_msg(
814
+ ProtocolMessageTypes.new_end_of_sub_slot_vdf,
815
+ timelord_protocol.NewEndOfSubSlotVDF(eos_bundle),
816
+ )
817
+ await self.server.send_to_all([msg], NodeType.FULL_NODE)
818
+
819
+ log.info(
820
+ f"Built end of subslot bundle. cc hash: {eos_bundle.challenge_chain.get_hash()}. New_difficulty: "
821
+ f"{eos_bundle.challenge_chain.new_difficulty} New ssi: {eos_bundle.challenge_chain.new_sub_slot_iters}"
822
+ )
823
+
824
+ if next_ses is None or next_ses.new_difficulty is None:
825
+ self.unfinished_blocks = self.overflow_blocks.copy()
826
+ else:
827
+ # No overflow blocks in a new epoch
828
+ self.unfinished_blocks = []
829
+ self.overflow_blocks = []
830
+
831
+ self.last_state.set_state(eos_bundle)
832
+ for block in self.unfinished_blocks:
833
+ if self._can_infuse_unfinished_block(block) is not None:
834
+ self.total_unfinished += 1
835
+ await self._reset_chains()
836
+
837
+ async def _handle_failures(self) -> None:
838
+ if len(self.vdf_failures) > 0:
839
+ # This can happen if one of the VDF processes has an issue. In this case, we abort all other
840
+ # infusion points and signage points, and go straight to the end of slot, so we avoid potential
841
+ # issues with the number of iterations that failed.
842
+
843
+ failed_chain, proof_label = self.vdf_failures[0]
844
+ log.error(
845
+ f"Vdf clients failed {self.vdf_failures_count} times. Last failure: {failed_chain}, "
846
+ f"label {proof_label}, current: {self.num_resets}"
847
+ )
848
+ if proof_label == self.num_resets:
849
+ await self._reset_chains(only_eos=True)
850
+ self.vdf_failure_time = time.time()
851
+ self.vdf_failures = []
852
+
853
+ # If something goes wrong in the VDF client due to a failed thread, we might get stuck in a situation where we
854
+ # are waiting for that client to finish. Usually other peers will finish the VDFs and reset us. In the case that
855
+ # there are no other timelords, this reset should bring the timelord back to a running state.
856
+ if time.time() - self.vdf_failure_time < self.constants.SUB_SLOT_TIME_TARGET * 3:
857
+ # If we have recently had a failure, allow some more time to finish the slot (we can be up to 3x slower)
858
+ active_time_threshold = self.constants.SUB_SLOT_TIME_TARGET * 3
859
+ else:
860
+ # If there were no failures recently trigger a reset after 60 seconds of no activity.
861
+ # Signage points should be every 9 seconds
862
+ active_time_threshold = self.max_allowed_inactivity_time
863
+ if time.time() - self.last_active_time > active_time_threshold:
864
+ log.error(f"Not active for {active_time_threshold} seconds, restarting all chains")
865
+ self.max_allowed_inactivity_time = min(self.max_allowed_inactivity_time * 2, 1800)
866
+ await self._reset_chains()
867
+
868
+ async def _manage_chains(self) -> None:
869
+ async with self.lock:
870
+ await asyncio.sleep(5)
871
+ await self._reset_chains(first_run=True)
872
+ while not self._shut_down:
873
+ try:
874
+ await asyncio.sleep(0.1)
875
+ async with self.lock:
876
+ await self._handle_failures()
877
+ # We've got a new peak, process it.
878
+ if self.new_peak is not None:
879
+ await self._handle_new_peak()
880
+ # Map free vdf_clients to unspawned chains.
881
+ await self._map_chains_with_vdf_clients()
882
+ async with self.lock:
883
+ # Submit pending iterations.
884
+ await self._submit_iterations()
885
+
886
+ not_finished_iters = [
887
+ it for it in self.iters_submitted[Chain.REWARD_CHAIN] if it not in self.iters_finished
888
+ ]
889
+ if len(not_finished_iters) == 0:
890
+ await asyncio.sleep(0.1)
891
+ continue
892
+ selected_iter = min(not_finished_iters)
893
+
894
+ # Check for new infusion point and broadcast it if present.
895
+ await self._check_for_new_ip(selected_iter)
896
+ # Check for new signage point and broadcast it if present.
897
+ await self._check_for_new_sp(selected_iter)
898
+ # Check for end of subslot, respawn chains and build EndOfSubslotBundle.
899
+ await self._check_for_end_of_subslot(selected_iter)
900
+
901
+ except Exception:
902
+ tb = traceback.format_exc()
903
+ log.error(f"Error while handling message: {tb}")
904
+
905
+ async def _do_process_communication(
906
+ self,
907
+ chain: Chain,
908
+ challenge: bytes32,
909
+ initial_form: ClassgroupElement,
910
+ ip: str,
911
+ reader: asyncio.StreamReader,
912
+ writer: asyncio.StreamWriter,
913
+ # Data specific only when running in bluebox mode.
914
+ bluebox_iteration: Optional[uint64] = None,
915
+ header_hash: Optional[bytes32] = None,
916
+ height: Optional[uint32] = None,
917
+ field_vdf: Optional[uint8] = None,
918
+ # Labels a proof to the current state only
919
+ proof_label: Optional[int] = None,
920
+ ) -> None:
921
+ disc: int = create_discriminant(challenge, self.constants.DISCRIMINANT_SIZE_BITS)
922
+
923
+ try:
924
+ # Depending on the flags 'fast_algorithm' and 'bluebox_mode',
925
+ # the timelord tells the vdf_client what to execute.
926
+ async with self.lock:
927
+ if self.bluebox_mode:
928
+ writer.write(b"S")
929
+ else:
930
+ if self.config["fast_algorithm"]:
931
+ # Run n-wesolowski (fast) algorithm.
932
+ writer.write(b"N")
933
+ else:
934
+ # Run two-wesolowski (slow) algorithm.
935
+ writer.write(b"T")
936
+ await writer.drain()
937
+
938
+ prefix = str(len(str(disc)))
939
+ if len(prefix) == 1:
940
+ prefix = "00" + prefix
941
+ if len(prefix) == 2:
942
+ prefix = "0" + prefix
943
+ async with self.lock:
944
+ writer.write((prefix + str(disc)).encode())
945
+ await writer.drain()
946
+
947
+ # Send initial_form prefixed with its length.
948
+ async with self.lock:
949
+ writer.write(bytes([len(initial_form.data)]) + initial_form.data)
950
+ await writer.drain()
951
+ try:
952
+ ok = await reader.readexactly(2)
953
+ except (asyncio.IncompleteReadError, ConnectionResetError, Exception) as e:
954
+ log.warning(f"{type(e)} {e}")
955
+ async with self.lock:
956
+ self.vdf_failures.append((chain, proof_label))
957
+ self.vdf_failures_count += 1
958
+ return
959
+
960
+ if ok.decode() != "OK":
961
+ return
962
+
963
+ log.debug("Got handshake with VDF client.")
964
+ if not self.bluebox_mode:
965
+ async with self.lock:
966
+ self.allows_iters.append(chain)
967
+ else:
968
+ async with self.lock:
969
+ assert chain is Chain.BLUEBOX
970
+ assert bluebox_iteration is not None
971
+ prefix = str(len(str(bluebox_iteration)))
972
+ if len(str(bluebox_iteration)) < 10:
973
+ prefix = "0" + prefix
974
+ iter_str = prefix + str(bluebox_iteration)
975
+ writer.write(iter_str.encode())
976
+ await writer.drain()
977
+
978
+ # Listen to the client until "STOP" is received.
979
+ while True:
980
+ try:
981
+ data = await reader.readexactly(4)
982
+ except (
983
+ asyncio.IncompleteReadError,
984
+ ConnectionResetError,
985
+ Exception,
986
+ ) as e:
987
+ log.warning(f"{type(e)} {e}")
988
+ async with self.lock:
989
+ self.vdf_failures.append((chain, proof_label))
990
+ self.vdf_failures_count += 1
991
+ break
992
+
993
+ if data == b"STOP":
994
+ log.debug(f"Stopped client running on ip {ip}.")
995
+ async with self.lock:
996
+ writer.write(b"ACK")
997
+ await writer.drain()
998
+ break
999
+ try:
1000
+ # This must be a proof, 4 bytes is length prefix
1001
+ length = int.from_bytes(data, "big")
1002
+ proof = await reader.readexactly(length)
1003
+ stdout_bytes_io: io.BytesIO = io.BytesIO(bytes.fromhex(proof.decode()))
1004
+ except (
1005
+ asyncio.IncompleteReadError,
1006
+ ConnectionResetError,
1007
+ Exception,
1008
+ ) as e:
1009
+ log.warning(f"{type(e)} {e}")
1010
+ async with self.lock:
1011
+ self.vdf_failures.append((chain, proof_label))
1012
+ self.vdf_failures_count += 1
1013
+ break
1014
+
1015
+ iterations_needed = uint64(int.from_bytes(stdout_bytes_io.read(8), "big", signed=True))
1016
+
1017
+ y_size_bytes = stdout_bytes_io.read(8)
1018
+ y_size = uint64(int.from_bytes(y_size_bytes, "big", signed=True))
1019
+
1020
+ y_bytes = stdout_bytes_io.read(y_size)
1021
+ witness_type = uint8(int.from_bytes(stdout_bytes_io.read(1), "big", signed=True))
1022
+ proof_bytes: bytes = stdout_bytes_io.read()
1023
+
1024
+ # Verifies our own proof just in case
1025
+
1026
+ form_size = ClassgroupElement.get_size()
1027
+ output = ClassgroupElement.create(y_bytes[:form_size])
1028
+ # default value so that it's always set for state_changed later
1029
+ ips: float = 0
1030
+ if not self.bluebox_mode:
1031
+ time_taken = time.time() - self.chain_start_time[chain]
1032
+ ips = int(iterations_needed / time_taken * 10) / 10
1033
+ log.info(
1034
+ f"Finished PoT chall:{challenge[:10].hex()}.. {iterations_needed}"
1035
+ f" iters, "
1036
+ f"Estimated IPS: {ips}, Chain: {chain}"
1037
+ )
1038
+
1039
+ vdf_info: VDFInfo = VDFInfo(
1040
+ challenge,
1041
+ iterations_needed,
1042
+ output,
1043
+ )
1044
+ vdf_proof: VDFProof = VDFProof(
1045
+ witness_type,
1046
+ proof_bytes,
1047
+ self.bluebox_mode,
1048
+ )
1049
+
1050
+ if not validate_vdf(vdf_proof, self.constants, initial_form, vdf_info):
1051
+ log.error("Invalid proof of time!")
1052
+ if not self.bluebox_mode:
1053
+ async with self.lock:
1054
+ assert proof_label is not None
1055
+ self.proofs_finished.append((chain, vdf_info, vdf_proof, proof_label))
1056
+ self.state_changed(
1057
+ "finished_pot",
1058
+ {
1059
+ "estimated_ips": ips,
1060
+ "iterations_needed": iterations_needed,
1061
+ "chain": chain.value,
1062
+ "vdf_info": vdf_info,
1063
+ "vdf_proof": vdf_proof,
1064
+ },
1065
+ )
1066
+ else:
1067
+ async with self.lock:
1068
+ writer.write(b"010")
1069
+ await writer.drain()
1070
+ assert header_hash is not None
1071
+ assert field_vdf is not None
1072
+ assert height is not None
1073
+ response = timelord_protocol.RespondCompactProofOfTime(
1074
+ vdf_info, vdf_proof, header_hash, height, field_vdf
1075
+ )
1076
+ if self._server is not None:
1077
+ message = make_msg(ProtocolMessageTypes.respond_compact_proof_of_time, response)
1078
+ await self.server.send_to_all([message], NodeType.FULL_NODE)
1079
+ self.state_changed(
1080
+ "new_compact_proof", {"header_hash": header_hash, "height": height, "field_vdf": field_vdf}
1081
+ )
1082
+
1083
+ except ConnectionResetError as e:
1084
+ log.debug(f"Connection reset with VDF client {e}")
1085
+
1086
+ async def _manage_discriminant_queue_sanitizer(self) -> None:
1087
+ while not self._shut_down:
1088
+ async with self.lock:
1089
+ try:
1090
+ while len(self.pending_bluebox_info) > 0 and len(self.free_clients) > 0:
1091
+ # Select randomly the field_vdf we're creating a compact vdf for.
1092
+ # This is done because CC_SP and CC_IP are more frequent than
1093
+ # CC_EOS and ICC_EOS. This guarantees everything is picked uniformly.
1094
+ target_field_vdf = random.randint(1, 4)
1095
+ info = next(
1096
+ (info for info in self.pending_bluebox_info if info[1].field_vdf == target_field_vdf),
1097
+ None,
1098
+ )
1099
+ if info is None:
1100
+ # Nothing found with target_field_vdf, just pick the first VDFInfo.
1101
+ info = self.pending_bluebox_info[0]
1102
+ ip, reader, writer = self.free_clients[0]
1103
+ self.process_communication_tasks.append(
1104
+ asyncio.create_task(
1105
+ self._do_process_communication(
1106
+ Chain.BLUEBOX,
1107
+ info[1].new_proof_of_time.challenge,
1108
+ ClassgroupElement.get_default_element(),
1109
+ ip,
1110
+ reader,
1111
+ writer,
1112
+ info[1].new_proof_of_time.number_of_iterations,
1113
+ info[1].header_hash,
1114
+ info[1].height,
1115
+ info[1].field_vdf,
1116
+ )
1117
+ )
1118
+ )
1119
+ self.pending_bluebox_info.remove(info)
1120
+ self.free_clients = self.free_clients[1:]
1121
+ except Exception as e:
1122
+ log.error(f"Exception manage discriminant queue: {e}")
1123
+ await asyncio.sleep(0.1)
1124
+
1125
+ async def _start_manage_discriminant_queue_sanitizer_slow(self, pool: ThreadPoolExecutor, counter: int) -> None:
1126
+ tasks = []
1127
+ for _ in range(counter):
1128
+ tasks.append(asyncio.create_task(self._manage_discriminant_queue_sanitizer_slow(pool)))
1129
+ for task in tasks:
1130
+ await task
1131
+
1132
+ async def _manage_discriminant_queue_sanitizer_slow(self, pool: ThreadPoolExecutor) -> None:
1133
+ log.info("Started task for managing bluebox queue.")
1134
+ while not self._shut_down:
1135
+ picked_info = None
1136
+ async with self.lock:
1137
+ try:
1138
+ if len(self.pending_bluebox_info) > 0:
1139
+ # Select randomly the field_vdf we're creating a compact vdf for.
1140
+ # This is done because CC_SP and CC_IP are more frequent than
1141
+ # CC_EOS and ICC_EOS. This guarantees everything is picked uniformly.
1142
+ target_field_vdf = random.randint(1, 4)
1143
+ info = next(
1144
+ (info for info in self.pending_bluebox_info if info[1].field_vdf == target_field_vdf),
1145
+ None,
1146
+ )
1147
+ if info is None:
1148
+ # Nothing found with target_field_vdf, just pick the first VDFInfo.
1149
+ info = self.pending_bluebox_info[0]
1150
+ self.pending_bluebox_info.remove(info)
1151
+ picked_info = info[1]
1152
+ except Exception as e:
1153
+ log.error(f"Exception manage discriminant queue: {e}")
1154
+ if picked_info is not None:
1155
+ try:
1156
+ t1 = time.time()
1157
+ log.info(
1158
+ f"Working on compact proof for height: {picked_info.height}. "
1159
+ f"Iters: {picked_info.new_proof_of_time.number_of_iterations}."
1160
+ )
1161
+ bluebox_process_data = BlueboxProcessData(
1162
+ picked_info.new_proof_of_time.challenge,
1163
+ uint16(self.constants.DISCRIMINANT_SIZE_BITS),
1164
+ picked_info.new_proof_of_time.number_of_iterations,
1165
+ )
1166
+ proof = await asyncio.get_running_loop().run_in_executor(
1167
+ pool,
1168
+ prove_bluebox_slow,
1169
+ bytes(bluebox_process_data),
1170
+ )
1171
+ t2 = time.time()
1172
+ delta = t2 - t1
1173
+ if delta > 0:
1174
+ ips = picked_info.new_proof_of_time.number_of_iterations / delta
1175
+ else:
1176
+ ips = 0
1177
+ log.info(f"Finished compact proof: {picked_info.height}. Time: {delta}s. IPS: {ips}.")
1178
+ output = proof[:100]
1179
+ proof_part = proof[100:200]
1180
+ if ClassgroupElement.create(output) != picked_info.new_proof_of_time.output:
1181
+ log.error("Expected vdf output different than produced one. Stopping.")
1182
+ return
1183
+ vdf_proof = VDFProof(uint8(0), proof_part, True)
1184
+ initial_form = ClassgroupElement.get_default_element()
1185
+ if not validate_vdf(vdf_proof, self.constants, initial_form, picked_info.new_proof_of_time):
1186
+ log.error("Invalid compact proof of time!")
1187
+ return
1188
+ response = timelord_protocol.RespondCompactProofOfTime(
1189
+ picked_info.new_proof_of_time,
1190
+ vdf_proof,
1191
+ picked_info.header_hash,
1192
+ picked_info.height,
1193
+ picked_info.field_vdf,
1194
+ )
1195
+ if self._server is not None:
1196
+ message = make_msg(ProtocolMessageTypes.respond_compact_proof_of_time, response)
1197
+ await self.server.send_to_all([message], NodeType.FULL_NODE)
1198
+ except Exception as e:
1199
+ log.error(f"Exception manage discriminant queue: {e}")
1200
+ tb = traceback.format_exc()
1201
+ log.error(f"Error while handling message: {tb}")
1202
+ await asyncio.sleep(0.1)