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,2423 @@
1
+ from __future__ import annotations
2
+
3
+ import itertools
4
+ import logging
5
+ import os
6
+ import random
7
+ import re
8
+ import statistics
9
+ import time
10
+ from dataclasses import dataclass
11
+ from pathlib import Path
12
+ from random import Random
13
+ from typing import Any, Awaitable, Callable, Dict, List, Optional, Set, Tuple, cast
14
+
15
+ import aiohttp
16
+ import aiosqlite
17
+ import pytest
18
+
19
+ from chia._tests.core.data_layer.util import Example, add_0123_example, add_01234567_example
20
+ from chia._tests.util.misc import BenchmarkRunner, Marks, boolean_datacases, datacases
21
+ from chia.data_layer.data_layer_errors import KeyNotFoundError, NodeHashError, TreeGenerationIncrementingError
22
+ from chia.data_layer.data_layer_util import (
23
+ DiffData,
24
+ InternalNode,
25
+ Node,
26
+ NodeType,
27
+ OperationType,
28
+ ProofOfInclusion,
29
+ ProofOfInclusionLayer,
30
+ Root,
31
+ ServerInfo,
32
+ Side,
33
+ Status,
34
+ Subscription,
35
+ TerminalNode,
36
+ _debug_dump,
37
+ leaf_hash,
38
+ )
39
+ from chia.data_layer.data_store import DataStore
40
+ from chia.data_layer.download_data import (
41
+ get_delta_filename_path,
42
+ get_full_tree_filename_path,
43
+ insert_from_delta_file,
44
+ insert_into_data_store_from_file,
45
+ write_files_for_root,
46
+ )
47
+ from chia.types.blockchain_format.program import Program
48
+ from chia.types.blockchain_format.sized_bytes import bytes32
49
+ from chia.util.byte_types import hexstr_to_bytes
50
+ from chia.util.db_wrapper import DBWrapper2, generate_in_memory_db_uri
51
+
52
+ log = logging.getLogger(__name__)
53
+
54
+
55
+ pytestmark = pytest.mark.data_layer
56
+
57
+
58
+ table_columns: Dict[str, List[str]] = {
59
+ "node": ["hash", "node_type", "left", "right", "key", "value"],
60
+ "root": ["tree_id", "generation", "node_hash", "status"],
61
+ }
62
+
63
+
64
+ # TODO: Someday add tests for malformed DB data to make sure we handle it gracefully
65
+ # and with good error messages.
66
+
67
+
68
+ @pytest.mark.anyio
69
+ async def test_valid_node_values_fixture_are_valid(data_store: DataStore, valid_node_values: Dict[str, Any]) -> None:
70
+ async with data_store.db_wrapper.writer() as writer:
71
+ await writer.execute(
72
+ """
73
+ INSERT INTO node(hash, node_type, left, right, key, value)
74
+ VALUES(:hash, :node_type, :left, :right, :key, :value)
75
+ """,
76
+ valid_node_values,
77
+ )
78
+
79
+
80
+ @pytest.mark.parametrize(argnames=["table_name", "expected_columns"], argvalues=table_columns.items())
81
+ @pytest.mark.anyio
82
+ async def test_create_creates_tables_and_columns(
83
+ database_uri: str, table_name: str, expected_columns: List[str]
84
+ ) -> None:
85
+ # Never string-interpolate sql queries... Except maybe in tests when it does not
86
+ # allow you to parametrize the query.
87
+ query = f"pragma table_info({table_name});"
88
+
89
+ async with DBWrapper2.managed(database=database_uri, uri=True, reader_count=1) as db_wrapper:
90
+ async with db_wrapper.reader() as reader:
91
+ cursor = await reader.execute(query)
92
+ columns = await cursor.fetchall()
93
+ assert columns == []
94
+
95
+ async with DataStore.managed(database=database_uri, uri=True):
96
+ async with db_wrapper.reader() as reader:
97
+ cursor = await reader.execute(query)
98
+ columns = await cursor.fetchall()
99
+ assert [column[1] for column in columns] == expected_columns
100
+
101
+
102
+ @pytest.mark.anyio
103
+ async def test_create_tree_accepts_bytes32(raw_data_store: DataStore) -> None:
104
+ store_id = bytes32(b"\0" * 32)
105
+
106
+ await raw_data_store.create_tree(store_id=store_id)
107
+
108
+
109
+ @pytest.mark.parametrize(argnames=["length"], argvalues=[[length] for length in [*range(0, 32), *range(33, 48)]])
110
+ @pytest.mark.anyio
111
+ async def test_create_store_fails_for_not_bytes32(raw_data_store: DataStore, length: int) -> None:
112
+ bad_store_id = b"\0" * length
113
+
114
+ # TODO: require a more specific exception
115
+ with pytest.raises(Exception):
116
+ # type ignore since we are trying to intentionally pass a bad argument
117
+ await raw_data_store.create_tree(store_id=bad_store_id) # type: ignore[arg-type]
118
+
119
+
120
+ @pytest.mark.anyio
121
+ async def test_get_trees(raw_data_store: DataStore) -> None:
122
+ expected_store_ids = set()
123
+
124
+ for n in range(10):
125
+ store_id = bytes32(b"\0" * 31 + bytes([n]))
126
+ await raw_data_store.create_tree(store_id=store_id)
127
+ expected_store_ids.add(store_id)
128
+
129
+ store_ids = await raw_data_store.get_store_ids()
130
+
131
+ assert store_ids == expected_store_ids
132
+
133
+
134
+ @pytest.mark.anyio
135
+ async def test_table_is_empty(data_store: DataStore, store_id: bytes32) -> None:
136
+ is_empty = await data_store.table_is_empty(store_id=store_id)
137
+ assert is_empty
138
+
139
+
140
+ @pytest.mark.anyio
141
+ async def test_table_is_not_empty(data_store: DataStore, store_id: bytes32) -> None:
142
+ key = b"\x01\x02"
143
+ value = b"abc"
144
+
145
+ await data_store.insert(
146
+ key=key,
147
+ value=value,
148
+ store_id=store_id,
149
+ reference_node_hash=None,
150
+ side=None,
151
+ status=Status.COMMITTED,
152
+ )
153
+
154
+ is_empty = await data_store.table_is_empty(store_id=store_id)
155
+ assert not is_empty
156
+
157
+
158
+ # @pytest.mark.anyio
159
+ # async def test_create_root_provides_bytes32(raw_data_store: DataStore, store_id: bytes32) -> None:
160
+ # await raw_data_store.create_tree(store_id=store_id)
161
+ # # TODO: catchup with the node_hash=
162
+ # root_hash = await raw_data_store.create_root(store_id=store_id, node_hash=23)
163
+ #
164
+ # assert isinstance(root_hash, bytes32)
165
+
166
+
167
+ @pytest.mark.anyio
168
+ async def test_insert_over_empty(data_store: DataStore, store_id: bytes32) -> None:
169
+ key = b"\x01\x02"
170
+ value = b"abc"
171
+
172
+ insert_result = await data_store.insert(
173
+ key=key, value=value, store_id=store_id, reference_node_hash=None, side=None
174
+ )
175
+ assert insert_result.node_hash == leaf_hash(key=key, value=value)
176
+
177
+
178
+ @pytest.mark.anyio
179
+ async def test_insert_increments_generation(data_store: DataStore, store_id: bytes32) -> None:
180
+ keys = [b"a", b"b", b"c", b"d"] # efghijklmnopqrstuvwxyz")
181
+ value = b"\x01\x02\x03"
182
+
183
+ generations = []
184
+ expected = []
185
+
186
+ node_hash = None
187
+ for key, expected_generation in zip(keys, itertools.count(start=1)):
188
+ insert_result = await data_store.insert(
189
+ key=key,
190
+ value=value,
191
+ store_id=store_id,
192
+ reference_node_hash=node_hash,
193
+ side=None if node_hash is None else Side.LEFT,
194
+ status=Status.COMMITTED,
195
+ )
196
+ node_hash = insert_result.node_hash
197
+ generation = await data_store.get_tree_generation(store_id=store_id)
198
+ generations.append(generation)
199
+ expected.append(expected_generation)
200
+
201
+ assert generations == expected
202
+
203
+
204
+ @pytest.mark.anyio
205
+ async def test_get_tree_generation_returns_none_when_none_available(
206
+ raw_data_store: DataStore,
207
+ store_id: bytes32,
208
+ ) -> None:
209
+ with pytest.raises(Exception, match=re.escape(f"No generations found for store ID: {store_id.hex()}")):
210
+ await raw_data_store.get_tree_generation(store_id=store_id)
211
+
212
+
213
+ @pytest.mark.anyio
214
+ async def test_insert_internal_node_does_nothing_if_matching(data_store: DataStore, store_id: bytes32) -> None:
215
+ await add_01234567_example(data_store=data_store, store_id=store_id)
216
+
217
+ kv_node = await data_store.get_node_by_key(key=b"\x04", store_id=store_id)
218
+ ancestors = await data_store.get_ancestors(node_hash=kv_node.hash, store_id=store_id)
219
+ parent = ancestors[0]
220
+
221
+ async with data_store.db_wrapper.reader() as reader:
222
+ cursor = await reader.execute("SELECT * FROM node")
223
+ before = await cursor.fetchall()
224
+
225
+ await data_store._insert_internal_node(left_hash=parent.left_hash, right_hash=parent.right_hash)
226
+
227
+ async with data_store.db_wrapper.reader() as reader:
228
+ cursor = await reader.execute("SELECT * FROM node")
229
+ after = await cursor.fetchall()
230
+
231
+ assert after == before
232
+
233
+
234
+ @pytest.mark.anyio
235
+ async def test_insert_terminal_node_does_nothing_if_matching(data_store: DataStore, store_id: bytes32) -> None:
236
+ await add_01234567_example(data_store=data_store, store_id=store_id)
237
+
238
+ kv_node = await data_store.get_node_by_key(key=b"\x04", store_id=store_id)
239
+
240
+ async with data_store.db_wrapper.reader() as reader:
241
+ cursor = await reader.execute("SELECT * FROM node")
242
+ before = await cursor.fetchall()
243
+
244
+ await data_store._insert_terminal_node(key=kv_node.key, value=kv_node.value)
245
+
246
+ async with data_store.db_wrapper.reader() as reader:
247
+ cursor = await reader.execute("SELECT * FROM node")
248
+ after = await cursor.fetchall()
249
+
250
+ assert after == before
251
+
252
+
253
+ @pytest.mark.anyio
254
+ async def test_build_a_tree(
255
+ data_store: DataStore,
256
+ store_id: bytes32,
257
+ create_example: Callable[[DataStore, bytes32], Awaitable[Example]],
258
+ ) -> None:
259
+ example = await create_example(data_store, store_id)
260
+
261
+ await _debug_dump(db=data_store.db_wrapper, description="final")
262
+ actual = await data_store.get_tree_as_nodes(store_id=store_id)
263
+ # print("actual ", actual.as_python())
264
+ # print("expected", example.expected.as_python())
265
+ assert actual == example.expected
266
+
267
+
268
+ @pytest.mark.anyio
269
+ async def test_get_node_by_key(data_store: DataStore, store_id: bytes32) -> None:
270
+ example = await add_0123_example(data_store=data_store, store_id=store_id)
271
+
272
+ key_node_hash = example.terminal_nodes[2]
273
+
274
+ # TODO: make a nicer relationship between the hash and the key
275
+
276
+ actual = await data_store.get_node_by_key(key=b"\x02", store_id=store_id)
277
+ assert actual.hash == key_node_hash
278
+
279
+
280
+ @pytest.mark.anyio
281
+ async def test_get_ancestors(data_store: DataStore, store_id: bytes32) -> None:
282
+ example = await add_0123_example(data_store=data_store, store_id=store_id)
283
+
284
+ reference_node_hash = example.terminal_nodes[2]
285
+
286
+ ancestors = await data_store.get_ancestors(node_hash=reference_node_hash, store_id=store_id)
287
+ hashes = [node.hash.hex() for node in ancestors]
288
+
289
+ # TODO: reverify these are correct
290
+ assert hashes == [
291
+ "3ab212e30b0e746d81a993e39f2cb4ba843412d44b402c1117a500d6451309e3",
292
+ "c852ecd8fb61549a0a42f9eb9dde65e6c94a01934dbd9c1d35ab94e2a0ae58e2",
293
+ ]
294
+
295
+ ancestors_2 = await data_store.get_ancestors_optimized(node_hash=reference_node_hash, store_id=store_id)
296
+ assert ancestors == ancestors_2
297
+
298
+
299
+ @pytest.mark.anyio
300
+ async def test_get_ancestors_optimized(data_store: DataStore, store_id: bytes32) -> None:
301
+ ancestors: List[Tuple[int, bytes32, List[InternalNode]]] = []
302
+ random = Random()
303
+ random.seed(100, version=2)
304
+
305
+ first_insertions = [True, False, True, False, True, True, False, True, False, True, True, False, False, True, False]
306
+ deleted_all = False
307
+ node_count = 0
308
+ for i in range(1000):
309
+ is_insert = False
310
+ if i <= 14:
311
+ is_insert = first_insertions[i]
312
+ if i > 14 and i <= 25:
313
+ is_insert = True
314
+ if i > 25 and i <= 200 and random.randint(0, 4):
315
+ is_insert = True
316
+ if i > 200:
317
+ if not deleted_all:
318
+ while node_count > 0:
319
+ node_count -= 1
320
+ seed = bytes32(b"0" * 32)
321
+ node_hash = await data_store.get_terminal_node_for_seed(store_id, seed)
322
+ assert node_hash is not None
323
+ node = await data_store.get_node(node_hash)
324
+ assert isinstance(node, TerminalNode)
325
+ await data_store.delete(key=node.key, store_id=store_id, status=Status.COMMITTED)
326
+ deleted_all = True
327
+ is_insert = True
328
+ else:
329
+ assert node_count <= 4
330
+ if node_count == 0:
331
+ is_insert = True
332
+ elif node_count < 4 and random.randint(0, 2):
333
+ is_insert = True
334
+ key = (i % 200).to_bytes(4, byteorder="big")
335
+ value = (i % 200).to_bytes(4, byteorder="big")
336
+ seed = Program.to((key, value)).get_tree_hash()
337
+ node_hash = await data_store.get_terminal_node_for_seed(store_id, seed)
338
+ if is_insert:
339
+ node_count += 1
340
+ side = None if node_hash is None else data_store.get_side_for_seed(seed)
341
+
342
+ insert_result = await data_store.insert(
343
+ key=key,
344
+ value=value,
345
+ store_id=store_id,
346
+ reference_node_hash=node_hash,
347
+ side=side,
348
+ use_optimized=False,
349
+ status=Status.COMMITTED,
350
+ )
351
+ node_hash = insert_result.node_hash
352
+ if node_hash is not None:
353
+ generation = await data_store.get_tree_generation(store_id=store_id)
354
+ current_ancestors = await data_store.get_ancestors(node_hash=node_hash, store_id=store_id)
355
+ ancestors.append((generation, node_hash, current_ancestors))
356
+ else:
357
+ node_count -= 1
358
+ assert node_hash is not None
359
+ node = await data_store.get_node(node_hash)
360
+ assert isinstance(node, TerminalNode)
361
+ await data_store.delete(key=node.key, store_id=store_id, use_optimized=False, status=Status.COMMITTED)
362
+
363
+ for generation, node_hash, expected_ancestors in ancestors:
364
+ current_ancestors = await data_store.get_ancestors_optimized(
365
+ node_hash=node_hash, store_id=store_id, generation=generation
366
+ )
367
+ assert current_ancestors == expected_ancestors
368
+
369
+
370
+ @pytest.mark.anyio
371
+ @pytest.mark.parametrize(
372
+ "use_optimized",
373
+ [True, False],
374
+ )
375
+ @pytest.mark.parametrize(
376
+ "num_batches",
377
+ [1, 5, 10, 25],
378
+ )
379
+ async def test_batch_update(
380
+ data_store: DataStore,
381
+ store_id: bytes32,
382
+ use_optimized: bool,
383
+ tmp_path: Path,
384
+ num_batches: int,
385
+ ) -> None:
386
+ total_operations = 1000 if use_optimized else 100
387
+ num_ops_per_batch = total_operations // num_batches
388
+ saved_batches: List[List[Dict[str, Any]]] = []
389
+ saved_kv: List[List[TerminalNode]] = []
390
+ db_uri = generate_in_memory_db_uri()
391
+ async with DataStore.managed(database=db_uri, uri=True) as single_op_data_store:
392
+ await single_op_data_store.create_tree(store_id, status=Status.COMMITTED)
393
+ random = Random()
394
+ random.seed(100, version=2)
395
+
396
+ batch: List[Dict[str, Any]] = []
397
+ keys_values: Dict[bytes, bytes] = {}
398
+ for operation in range(num_batches * num_ops_per_batch):
399
+ [op_type] = random.choices(
400
+ ["insert", "upsert-insert", "upsert-update", "delete"],
401
+ [0.4, 0.2, 0.2, 0.2],
402
+ k=1,
403
+ )
404
+ if op_type == "insert" or op_type == "upsert-insert" or len(keys_values) == 0:
405
+ if len(keys_values) == 0:
406
+ op_type = "insert"
407
+ key = operation.to_bytes(4, byteorder="big")
408
+ value = (2 * operation).to_bytes(4, byteorder="big")
409
+ if op_type == "insert":
410
+ await single_op_data_store.autoinsert(
411
+ key=key,
412
+ value=value,
413
+ store_id=store_id,
414
+ use_optimized=use_optimized,
415
+ status=Status.COMMITTED,
416
+ )
417
+ else:
418
+ await single_op_data_store.upsert(
419
+ key=key,
420
+ new_value=value,
421
+ store_id=store_id,
422
+ use_optimized=use_optimized,
423
+ status=Status.COMMITTED,
424
+ )
425
+ action = "insert" if op_type == "insert" else "upsert"
426
+ batch.append({"action": action, "key": key, "value": value})
427
+ keys_values[key] = value
428
+ elif op_type == "delete":
429
+ key = random.choice(list(keys_values.keys()))
430
+ del keys_values[key]
431
+ await single_op_data_store.delete(
432
+ key=key,
433
+ store_id=store_id,
434
+ use_optimized=use_optimized,
435
+ status=Status.COMMITTED,
436
+ )
437
+ batch.append({"action": "delete", "key": key})
438
+ else:
439
+ assert op_type == "upsert-update"
440
+ key = random.choice(list(keys_values.keys()))
441
+ old_value = keys_values[key]
442
+ new_value_int = int.from_bytes(old_value, byteorder="big") + 1
443
+ new_value = new_value_int.to_bytes(4, byteorder="big")
444
+ await single_op_data_store.upsert(
445
+ key=key,
446
+ new_value=new_value,
447
+ store_id=store_id,
448
+ use_optimized=use_optimized,
449
+ status=Status.COMMITTED,
450
+ )
451
+ keys_values[key] = new_value
452
+ batch.append({"action": "upsert", "key": key, "value": new_value})
453
+ if (operation + 1) % num_ops_per_batch == 0:
454
+ saved_batches.append(batch)
455
+ batch = []
456
+ current_kv = await single_op_data_store.get_keys_values(store_id=store_id)
457
+ assert {kv.key: kv.value for kv in current_kv} == keys_values
458
+ saved_kv.append(current_kv)
459
+
460
+ for batch_number, batch in enumerate(saved_batches):
461
+ assert len(batch) == num_ops_per_batch
462
+ await data_store.insert_batch(store_id, batch, status=Status.COMMITTED)
463
+ root = await data_store.get_tree_root(store_id)
464
+ assert root.generation == batch_number + 1
465
+ assert root.node_hash is not None
466
+ current_kv = await data_store.get_keys_values(store_id=store_id)
467
+ # Get the same keys/values, but possibly stored in other order.
468
+ assert {node.key: node.value for node in current_kv} == {
469
+ node.key: node.value for node in saved_kv[batch_number]
470
+ }
471
+ queue: List[bytes32] = [root.node_hash]
472
+ ancestors: Dict[bytes32, bytes32] = {}
473
+ while len(queue) > 0:
474
+ node_hash = queue.pop(0)
475
+ expected_ancestors = []
476
+ ancestor = node_hash
477
+ while ancestor in ancestors:
478
+ ancestor = ancestors[ancestor]
479
+ expected_ancestors.append(ancestor)
480
+ result_ancestors = await data_store.get_ancestors_optimized(node_hash, store_id)
481
+ assert [node.hash for node in result_ancestors] == expected_ancestors
482
+ node = await data_store.get_node(node_hash)
483
+ if isinstance(node, InternalNode):
484
+ queue.append(node.left_hash)
485
+ queue.append(node.right_hash)
486
+ ancestors[node.left_hash] = node_hash
487
+ ancestors[node.right_hash] = node_hash
488
+
489
+ all_kv = await data_store.get_keys_values(store_id)
490
+ assert {node.key: node.value for node in all_kv} == keys_values
491
+
492
+
493
+ @pytest.mark.anyio
494
+ @pytest.mark.parametrize(
495
+ "use_optimized",
496
+ [True, False],
497
+ )
498
+ async def test_upsert_ignores_existing_arguments(
499
+ data_store: DataStore,
500
+ store_id: bytes32,
501
+ use_optimized: bool,
502
+ ) -> None:
503
+ key = b"key"
504
+ value = b"value1"
505
+
506
+ await data_store.autoinsert(
507
+ key=key,
508
+ value=value,
509
+ store_id=store_id,
510
+ use_optimized=use_optimized,
511
+ status=Status.COMMITTED,
512
+ )
513
+ node = await data_store.get_node_by_key(key, store_id)
514
+ assert node.value == value
515
+
516
+ new_value = b"value2"
517
+ await data_store.upsert(
518
+ key=key,
519
+ new_value=new_value,
520
+ store_id=store_id,
521
+ use_optimized=use_optimized,
522
+ status=Status.COMMITTED,
523
+ )
524
+ node = await data_store.get_node_by_key(key, store_id)
525
+ assert node.value == new_value
526
+
527
+ await data_store.upsert(
528
+ key=key,
529
+ new_value=new_value,
530
+ store_id=store_id,
531
+ use_optimized=use_optimized,
532
+ status=Status.COMMITTED,
533
+ )
534
+ node = await data_store.get_node_by_key(key, store_id)
535
+ assert node.value == new_value
536
+
537
+ key2 = b"key2"
538
+ await data_store.upsert(
539
+ key=key2,
540
+ new_value=value,
541
+ store_id=store_id,
542
+ use_optimized=use_optimized,
543
+ status=Status.COMMITTED,
544
+ )
545
+ node = await data_store.get_node_by_key(key2, store_id)
546
+ assert node.value == value
547
+
548
+
549
+ @pytest.mark.parametrize(argnames="side", argvalues=list(Side))
550
+ @pytest.mark.anyio
551
+ async def test_insert_batch_reference_and_side(
552
+ data_store: DataStore,
553
+ store_id: bytes32,
554
+ side: Side,
555
+ ) -> None:
556
+ insert_result = await data_store.autoinsert(
557
+ key=b"key1",
558
+ value=b"value1",
559
+ store_id=store_id,
560
+ status=Status.COMMITTED,
561
+ )
562
+
563
+ new_root_hash = await data_store.insert_batch(
564
+ store_id=store_id,
565
+ changelist=[
566
+ {
567
+ "action": "insert",
568
+ "key": b"key2",
569
+ "value": b"value2",
570
+ "reference_node_hash": insert_result.node_hash,
571
+ "side": side,
572
+ },
573
+ ],
574
+ )
575
+ assert new_root_hash is not None, "batch insert failed or failed to update root"
576
+
577
+ parent = await data_store.get_node(new_root_hash)
578
+ assert isinstance(parent, InternalNode)
579
+ if side == Side.LEFT:
580
+ child = await data_store.get_node(parent.left_hash)
581
+ assert parent.left_hash == child.hash
582
+ elif side == Side.RIGHT:
583
+ child = await data_store.get_node(parent.right_hash)
584
+ assert parent.right_hash == child.hash
585
+ else: # pragma: no cover
586
+ raise Exception("invalid side for test")
587
+
588
+
589
+ @pytest.mark.anyio
590
+ async def test_ancestor_table_unique_inserts(data_store: DataStore, store_id: bytes32) -> None:
591
+ await add_0123_example(data_store=data_store, store_id=store_id)
592
+ hash_1 = bytes32.from_hexstr("0763561814685fbf92f6ca71fbb1cb11821951450d996375c239979bd63e9535")
593
+ hash_2 = bytes32.from_hexstr("924be8ff27e84cba17f5bc918097f8410fab9824713a4668a21c8e060a8cab40")
594
+ await data_store._insert_ancestor_table(hash_1, hash_2, store_id, 2)
595
+ await data_store._insert_ancestor_table(hash_1, hash_2, store_id, 2)
596
+ with pytest.raises(Exception, match="^Requested insertion of ancestor"):
597
+ await data_store._insert_ancestor_table(hash_1, hash_1, store_id, 2)
598
+ await data_store._insert_ancestor_table(hash_1, hash_2, store_id, 2)
599
+
600
+
601
+ @pytest.mark.anyio
602
+ async def test_get_pairs(
603
+ data_store: DataStore,
604
+ store_id: bytes32,
605
+ create_example: Callable[[DataStore, bytes32], Awaitable[Example]],
606
+ ) -> None:
607
+ example = await create_example(data_store, store_id)
608
+
609
+ pairs = await data_store.get_keys_values(store_id=store_id)
610
+
611
+ assert [node.hash for node in pairs] == example.terminal_nodes
612
+
613
+
614
+ @pytest.mark.anyio
615
+ async def test_get_pairs_when_empty(data_store: DataStore, store_id: bytes32) -> None:
616
+ pairs = await data_store.get_keys_values(store_id=store_id)
617
+
618
+ assert pairs == []
619
+
620
+
621
+ @pytest.mark.parametrize(
622
+ argnames=["first_value", "second_value"],
623
+ argvalues=[[b"\x06", b"\x06"], [b"\x06", b"\x07"]],
624
+ ids=["same values", "different values"],
625
+ )
626
+ @pytest.mark.anyio()
627
+ async def test_inserting_duplicate_key_fails(
628
+ data_store: DataStore,
629
+ store_id: bytes32,
630
+ first_value: bytes,
631
+ second_value: bytes,
632
+ ) -> None:
633
+ key = b"\x05"
634
+
635
+ insert_result = await data_store.insert(
636
+ key=key,
637
+ value=first_value,
638
+ store_id=store_id,
639
+ reference_node_hash=None,
640
+ side=None,
641
+ )
642
+
643
+ # TODO: more specific exception
644
+ with pytest.raises(Exception):
645
+ await data_store.insert(
646
+ key=key,
647
+ value=second_value,
648
+ store_id=store_id,
649
+ reference_node_hash=insert_result.node_hash,
650
+ side=Side.RIGHT,
651
+ )
652
+
653
+ # TODO: more specific exception
654
+ with pytest.raises(Exception):
655
+ await data_store.insert(
656
+ key=key,
657
+ value=second_value,
658
+ store_id=store_id,
659
+ reference_node_hash=insert_result.node_hash,
660
+ side=Side.RIGHT,
661
+ )
662
+
663
+
664
+ @pytest.mark.anyio()
665
+ async def test_inserting_invalid_length_hash_raises_original_exception(
666
+ data_store: DataStore,
667
+ ) -> None:
668
+ with pytest.raises(aiosqlite.IntegrityError):
669
+ # casting since we are testing an invalid case
670
+ await data_store._insert_node(
671
+ node_hash=cast(bytes32, b"\x05"),
672
+ node_type=NodeType.TERMINAL,
673
+ left_hash=None,
674
+ right_hash=None,
675
+ key=b"\x06",
676
+ value=b"\x07",
677
+ )
678
+
679
+
680
+ @pytest.mark.anyio()
681
+ async def test_inserting_invalid_length_ancestor_hash_raises_original_exception(
682
+ data_store: DataStore,
683
+ store_id: bytes32,
684
+ ) -> None:
685
+ with pytest.raises(aiosqlite.IntegrityError):
686
+ # casting since we are testing an invalid case
687
+ await data_store._insert_ancestor_table(
688
+ left_hash=bytes32(b"\x01" * 32),
689
+ right_hash=bytes32(b"\x02" * 32),
690
+ store_id=store_id,
691
+ generation=0,
692
+ )
693
+
694
+
695
+ @pytest.mark.anyio()
696
+ async def test_autoinsert_balances_from_scratch(data_store: DataStore, store_id: bytes32) -> None:
697
+ random = Random()
698
+ random.seed(100, version=2)
699
+ hashes = []
700
+
701
+ for i in range(2000):
702
+ key = (i + 100).to_bytes(4, byteorder="big")
703
+ value = (i + 200).to_bytes(4, byteorder="big")
704
+ insert_result = await data_store.autoinsert(key, value, store_id, status=Status.COMMITTED)
705
+ hashes.append(insert_result.node_hash)
706
+
707
+ heights = {node_hash: len(await data_store.get_ancestors_optimized(node_hash, store_id)) for node_hash in hashes}
708
+ too_tall = {hash: height for hash, height in heights.items() if height > 14}
709
+ assert too_tall == {}
710
+ assert 11 <= statistics.mean(heights.values()) <= 12
711
+
712
+
713
+ @pytest.mark.anyio()
714
+ async def test_autoinsert_balances_gaps(data_store: DataStore, store_id: bytes32) -> None:
715
+ random = Random()
716
+ random.seed(101, version=2)
717
+ hashes = []
718
+
719
+ for i in range(2000):
720
+ key = (i + 100).to_bytes(4, byteorder="big")
721
+ value = (i + 200).to_bytes(4, byteorder="big")
722
+ if i == 0 or i > 10:
723
+ insert_result = await data_store.autoinsert(key, value, store_id, status=Status.COMMITTED)
724
+ else:
725
+ reference_node_hash = await data_store.get_terminal_node_for_seed(store_id, bytes32([0] * 32))
726
+ insert_result = await data_store.insert(
727
+ key=key,
728
+ value=value,
729
+ store_id=store_id,
730
+ reference_node_hash=reference_node_hash,
731
+ side=Side.LEFT,
732
+ status=Status.COMMITTED,
733
+ )
734
+ ancestors = await data_store.get_ancestors_optimized(insert_result.node_hash, store_id)
735
+ assert len(ancestors) == i
736
+ hashes.append(insert_result.node_hash)
737
+
738
+ heights = {node_hash: len(await data_store.get_ancestors_optimized(node_hash, store_id)) for node_hash in hashes}
739
+ too_tall = {hash: height for hash, height in heights.items() if height > 14}
740
+ assert too_tall == {}
741
+ assert 11 <= statistics.mean(heights.values()) <= 12
742
+
743
+
744
+ @pytest.mark.anyio()
745
+ async def test_delete_from_left_both_terminal(data_store: DataStore, store_id: bytes32) -> None:
746
+ await add_01234567_example(data_store=data_store, store_id=store_id)
747
+
748
+ expected = InternalNode.from_child_nodes(
749
+ left=InternalNode.from_child_nodes(
750
+ left=InternalNode.from_child_nodes(
751
+ left=TerminalNode.from_key_value(key=b"\x00", value=b"\x10\x00"),
752
+ right=TerminalNode.from_key_value(key=b"\x01", value=b"\x11\x01"),
753
+ ),
754
+ right=InternalNode.from_child_nodes(
755
+ left=TerminalNode.from_key_value(key=b"\x02", value=b"\x12\x02"),
756
+ right=TerminalNode.from_key_value(key=b"\x03", value=b"\x13\x03"),
757
+ ),
758
+ ),
759
+ right=InternalNode.from_child_nodes(
760
+ left=TerminalNode.from_key_value(key=b"\x05", value=b"\x15\x05"),
761
+ right=InternalNode.from_child_nodes(
762
+ left=TerminalNode.from_key_value(key=b"\x06", value=b"\x16\x06"),
763
+ right=TerminalNode.from_key_value(key=b"\x07", value=b"\x17\x07"),
764
+ ),
765
+ ),
766
+ )
767
+
768
+ await data_store.delete(key=b"\x04", store_id=store_id, status=Status.COMMITTED)
769
+ result = await data_store.get_tree_as_nodes(store_id=store_id)
770
+
771
+ assert result == expected
772
+
773
+
774
+ @pytest.mark.anyio()
775
+ async def test_delete_from_left_other_not_terminal(data_store: DataStore, store_id: bytes32) -> None:
776
+ await add_01234567_example(data_store=data_store, store_id=store_id)
777
+
778
+ expected = InternalNode.from_child_nodes(
779
+ left=InternalNode.from_child_nodes(
780
+ left=InternalNode.from_child_nodes(
781
+ left=TerminalNode.from_key_value(key=b"\x00", value=b"\x10\x00"),
782
+ right=TerminalNode.from_key_value(key=b"\x01", value=b"\x11\x01"),
783
+ ),
784
+ right=InternalNode.from_child_nodes(
785
+ left=TerminalNode.from_key_value(key=b"\x02", value=b"\x12\x02"),
786
+ right=TerminalNode.from_key_value(key=b"\x03", value=b"\x13\x03"),
787
+ ),
788
+ ),
789
+ right=InternalNode.from_child_nodes(
790
+ left=TerminalNode.from_key_value(key=b"\x06", value=b"\x16\x06"),
791
+ right=TerminalNode.from_key_value(key=b"\x07", value=b"\x17\x07"),
792
+ ),
793
+ )
794
+
795
+ await data_store.delete(key=b"\x04", store_id=store_id, status=Status.COMMITTED)
796
+ await data_store.delete(key=b"\x05", store_id=store_id, status=Status.COMMITTED)
797
+ result = await data_store.get_tree_as_nodes(store_id=store_id)
798
+
799
+ assert result == expected
800
+
801
+
802
+ @pytest.mark.anyio()
803
+ async def test_delete_from_right_both_terminal(data_store: DataStore, store_id: bytes32) -> None:
804
+ await add_01234567_example(data_store=data_store, store_id=store_id)
805
+
806
+ expected = InternalNode.from_child_nodes(
807
+ left=InternalNode.from_child_nodes(
808
+ left=InternalNode.from_child_nodes(
809
+ left=TerminalNode.from_key_value(key=b"\x00", value=b"\x10\x00"),
810
+ right=TerminalNode.from_key_value(key=b"\x01", value=b"\x11\x01"),
811
+ ),
812
+ right=TerminalNode.from_key_value(key=b"\x02", value=b"\x12\x02"),
813
+ ),
814
+ right=InternalNode.from_child_nodes(
815
+ left=InternalNode.from_child_nodes(
816
+ left=TerminalNode.from_key_value(key=b"\x04", value=b"\x14\x04"),
817
+ right=TerminalNode.from_key_value(key=b"\x05", value=b"\x15\x05"),
818
+ ),
819
+ right=InternalNode.from_child_nodes(
820
+ left=TerminalNode.from_key_value(key=b"\x06", value=b"\x16\x06"),
821
+ right=TerminalNode.from_key_value(key=b"\x07", value=b"\x17\x07"),
822
+ ),
823
+ ),
824
+ )
825
+
826
+ await data_store.delete(key=b"\x03", store_id=store_id, status=Status.COMMITTED)
827
+ result = await data_store.get_tree_as_nodes(store_id=store_id)
828
+
829
+ assert result == expected
830
+
831
+
832
+ @pytest.mark.anyio()
833
+ async def test_delete_from_right_other_not_terminal(data_store: DataStore, store_id: bytes32) -> None:
834
+ await add_01234567_example(data_store=data_store, store_id=store_id)
835
+
836
+ expected = InternalNode.from_child_nodes(
837
+ left=InternalNode.from_child_nodes(
838
+ left=TerminalNode.from_key_value(key=b"\x00", value=b"\x10\x00"),
839
+ right=TerminalNode.from_key_value(key=b"\x01", value=b"\x11\x01"),
840
+ ),
841
+ right=InternalNode.from_child_nodes(
842
+ left=InternalNode.from_child_nodes(
843
+ left=TerminalNode.from_key_value(key=b"\x04", value=b"\x14\x04"),
844
+ right=TerminalNode.from_key_value(key=b"\x05", value=b"\x15\x05"),
845
+ ),
846
+ right=InternalNode.from_child_nodes(
847
+ left=TerminalNode.from_key_value(key=b"\x06", value=b"\x16\x06"),
848
+ right=TerminalNode.from_key_value(key=b"\x07", value=b"\x17\x07"),
849
+ ),
850
+ ),
851
+ )
852
+
853
+ await data_store.delete(key=b"\x03", store_id=store_id, status=Status.COMMITTED)
854
+ await data_store.delete(key=b"\x02", store_id=store_id, status=Status.COMMITTED)
855
+ result = await data_store.get_tree_as_nodes(store_id=store_id)
856
+
857
+ assert result == expected
858
+
859
+
860
+ @pytest.mark.anyio
861
+ async def test_proof_of_inclusion_by_hash(data_store: DataStore, store_id: bytes32) -> None:
862
+ """A proof of inclusion contains the expected sibling side, sibling hash, combined
863
+ hash, key, value, and root hash values.
864
+ """
865
+ await add_01234567_example(data_store=data_store, store_id=store_id)
866
+ root = await data_store.get_tree_root(store_id=store_id)
867
+ assert root.node_hash is not None
868
+ node = await data_store.get_node_by_key(key=b"\x04", store_id=store_id)
869
+
870
+ proof = await data_store.get_proof_of_inclusion_by_hash(node_hash=node.hash, store_id=store_id)
871
+
872
+ print(node)
873
+ await _debug_dump(db=data_store.db_wrapper)
874
+
875
+ expected_layers = [
876
+ ProofOfInclusionLayer(
877
+ other_hash_side=Side.RIGHT,
878
+ other_hash=bytes32.fromhex("fb66fe539b3eb2020dfbfadfd601fa318521292b41f04c2057c16fca6b947ca1"),
879
+ combined_hash=bytes32.fromhex("36cb1fc56017944213055da8cb0178fb0938c32df3ec4472f5edf0dff85ba4a3"),
880
+ ),
881
+ ProofOfInclusionLayer(
882
+ other_hash_side=Side.RIGHT,
883
+ other_hash=bytes32.fromhex("6d3af8d93db948e8b6aa4386958e137c6be8bab726db86789594b3588b35adcd"),
884
+ combined_hash=bytes32.fromhex("5f67a0ab1976e090b834bf70e5ce2a0f0a9cd474e19a905348c44ae12274d30b"),
885
+ ),
886
+ ProofOfInclusionLayer(
887
+ other_hash_side=Side.LEFT,
888
+ other_hash=bytes32.fromhex("c852ecd8fb61549a0a42f9eb9dde65e6c94a01934dbd9c1d35ab94e2a0ae58e2"),
889
+ combined_hash=bytes32.fromhex("7a5193a4e31a0a72f6623dfeb2876022ab74a48abb5966088a1c6f5451cc5d81"),
890
+ ),
891
+ ]
892
+
893
+ assert proof == ProofOfInclusion(node_hash=node.hash, layers=expected_layers)
894
+
895
+
896
+ @pytest.mark.anyio
897
+ async def test_proof_of_inclusion_by_hash_no_ancestors(data_store: DataStore, store_id: bytes32) -> None:
898
+ """Check proper proof of inclusion creation when the node being proved is the root."""
899
+ await data_store.autoinsert(key=b"\x04", value=b"\x03", store_id=store_id, status=Status.COMMITTED)
900
+ root = await data_store.get_tree_root(store_id=store_id)
901
+ assert root.node_hash is not None
902
+ node = await data_store.get_node_by_key(key=b"\x04", store_id=store_id)
903
+
904
+ proof = await data_store.get_proof_of_inclusion_by_hash(node_hash=node.hash, store_id=store_id)
905
+
906
+ assert proof == ProofOfInclusion(node_hash=node.hash, layers=[])
907
+
908
+
909
+ @pytest.mark.anyio
910
+ async def test_proof_of_inclusion_by_hash_program(data_store: DataStore, store_id: bytes32) -> None:
911
+ """The proof of inclusion program has the expected Python equivalence."""
912
+
913
+ await add_01234567_example(data_store=data_store, store_id=store_id)
914
+ node = await data_store.get_node_by_key(key=b"\x04", store_id=store_id)
915
+
916
+ proof = await data_store.get_proof_of_inclusion_by_hash(node_hash=node.hash, store_id=store_id)
917
+
918
+ assert proof.as_program() == [
919
+ b"\x04",
920
+ [
921
+ bytes32.fromhex("fb66fe539b3eb2020dfbfadfd601fa318521292b41f04c2057c16fca6b947ca1"),
922
+ bytes32.fromhex("6d3af8d93db948e8b6aa4386958e137c6be8bab726db86789594b3588b35adcd"),
923
+ bytes32.fromhex("c852ecd8fb61549a0a42f9eb9dde65e6c94a01934dbd9c1d35ab94e2a0ae58e2"),
924
+ ],
925
+ ]
926
+
927
+
928
+ @pytest.mark.anyio
929
+ async def test_proof_of_inclusion_by_hash_equals_by_key(data_store: DataStore, store_id: bytes32) -> None:
930
+ """The proof of inclusion is equal between hash and key requests."""
931
+
932
+ await add_01234567_example(data_store=data_store, store_id=store_id)
933
+ node = await data_store.get_node_by_key(key=b"\x04", store_id=store_id)
934
+
935
+ proof_by_hash = await data_store.get_proof_of_inclusion_by_hash(node_hash=node.hash, store_id=store_id)
936
+ proof_by_key = await data_store.get_proof_of_inclusion_by_key(key=b"\x04", store_id=store_id)
937
+
938
+ assert proof_by_hash == proof_by_key
939
+
940
+
941
+ @pytest.mark.anyio
942
+ async def test_proof_of_inclusion_by_hash_bytes(data_store: DataStore, store_id: bytes32) -> None:
943
+ """The proof of inclusion provided by the data store is able to be converted to a
944
+ program and subsequently to bytes.
945
+ """
946
+ await add_01234567_example(data_store=data_store, store_id=store_id)
947
+ node = await data_store.get_node_by_key(key=b"\x04", store_id=store_id)
948
+
949
+ proof = await data_store.get_proof_of_inclusion_by_hash(node_hash=node.hash, store_id=store_id)
950
+
951
+ expected = (
952
+ b"\xff\x04\xff\xff\xa0\xfbf\xfeS\x9b>\xb2\x02\r\xfb\xfa\xdf\xd6\x01\xfa1\x85!)"
953
+ b"+A\xf0L W\xc1o\xcak\x94|\xa1\xff\xa0m:\xf8\xd9=\xb9H\xe8\xb6\xaaC\x86\x95"
954
+ b"\x8e\x13|k\xe8\xba\xb7&\xdb\x86x\x95\x94\xb3X\x8b5\xad\xcd\xff\xa0\xc8R\xec"
955
+ b"\xd8\xfbaT\x9a\nB\xf9\xeb\x9d\xdee\xe6\xc9J\x01\x93M\xbd\x9c\x1d5\xab\x94"
956
+ b"\xe2\xa0\xaeX\xe2\x80\x80"
957
+ )
958
+
959
+ assert bytes(proof.as_program()) == expected
960
+
961
+
962
+ # @pytest.mark.anyio
963
+ # async def test_create_first_pair(data_store: DataStore, store_id: bytes) -> None:
964
+ # key = SExp.to([1, 2])
965
+ # value = SExp.to(b'abc')
966
+ #
967
+ # root_hash = await data_store.create_root(store_id=store_id)
968
+ #
969
+ #
970
+ # await data_store.create_pair(key=key, value=value)
971
+
972
+
973
+ def test_all_checks_collected() -> None:
974
+ expected = {value for name, value in vars(DataStore).items() if name.startswith("_check_") and callable(value)}
975
+
976
+ assert set(DataStore._checks) == expected
977
+
978
+
979
+ a_bytes_32 = bytes32(range(32))
980
+ another_bytes_32 = bytes(reversed(a_bytes_32))
981
+
982
+ valid_program_hex = Program.to((b"abc", 2)).as_bin().hex()
983
+ invalid_program_hex = b"\xab\xcd".hex()
984
+
985
+
986
+ @pytest.mark.anyio
987
+ async def test_check_roots_are_incrementing_missing_zero(raw_data_store: DataStore) -> None:
988
+ store_id = hexstr_to_bytes("c954ab71ffaf5b0f129b04b35fdc7c84541f4375167e730e2646bfcfdb7cf2cd")
989
+
990
+ async with raw_data_store.db_wrapper.writer() as writer:
991
+ for generation in range(1, 5):
992
+ await writer.execute(
993
+ """
994
+ INSERT INTO root(tree_id, generation, node_hash, status)
995
+ VALUES(:tree_id, :generation, :node_hash, :status)
996
+ """,
997
+ {
998
+ "tree_id": store_id,
999
+ "generation": generation,
1000
+ "node_hash": None,
1001
+ "status": Status.COMMITTED.value,
1002
+ },
1003
+ )
1004
+
1005
+ with pytest.raises(
1006
+ TreeGenerationIncrementingError,
1007
+ match=r"\n +c954ab71ffaf5b0f129b04b35fdc7c84541f4375167e730e2646bfcfdb7cf2cd$",
1008
+ ):
1009
+ await raw_data_store._check_roots_are_incrementing()
1010
+
1011
+
1012
+ @pytest.mark.anyio
1013
+ async def test_check_roots_are_incrementing_gap(raw_data_store: DataStore) -> None:
1014
+ store_id = hexstr_to_bytes("c954ab71ffaf5b0f129b04b35fdc7c84541f4375167e730e2646bfcfdb7cf2cd")
1015
+
1016
+ async with raw_data_store.db_wrapper.writer() as writer:
1017
+ for generation in [*range(5), *range(6, 10)]:
1018
+ await writer.execute(
1019
+ """
1020
+ INSERT INTO root(tree_id, generation, node_hash, status)
1021
+ VALUES(:tree_id, :generation, :node_hash, :status)
1022
+ """,
1023
+ {
1024
+ "tree_id": store_id,
1025
+ "generation": generation,
1026
+ "node_hash": None,
1027
+ "status": Status.COMMITTED.value,
1028
+ },
1029
+ )
1030
+
1031
+ with pytest.raises(
1032
+ TreeGenerationIncrementingError,
1033
+ match=r"\n +c954ab71ffaf5b0f129b04b35fdc7c84541f4375167e730e2646bfcfdb7cf2cd$",
1034
+ ):
1035
+ await raw_data_store._check_roots_are_incrementing()
1036
+
1037
+
1038
+ @pytest.mark.anyio
1039
+ async def test_check_hashes_internal(raw_data_store: DataStore) -> None:
1040
+ async with raw_data_store.db_wrapper.writer() as writer:
1041
+ await writer.execute(
1042
+ "INSERT INTO node(hash, node_type, left, right) VALUES(:hash, :node_type, :left, :right)",
1043
+ {
1044
+ "hash": a_bytes_32,
1045
+ "node_type": NodeType.INTERNAL,
1046
+ "left": a_bytes_32,
1047
+ "right": a_bytes_32,
1048
+ },
1049
+ )
1050
+
1051
+ with pytest.raises(
1052
+ NodeHashError,
1053
+ match=r"\n +000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1f$",
1054
+ ):
1055
+ await raw_data_store._check_hashes()
1056
+
1057
+
1058
+ @pytest.mark.anyio
1059
+ async def test_check_hashes_terminal(raw_data_store: DataStore) -> None:
1060
+ async with raw_data_store.db_wrapper.writer() as writer:
1061
+ await writer.execute(
1062
+ "INSERT INTO node(hash, node_type, key, value) VALUES(:hash, :node_type, :key, :value)",
1063
+ {
1064
+ "hash": a_bytes_32,
1065
+ "node_type": NodeType.TERMINAL,
1066
+ "key": Program.to((1, 2)).as_bin(),
1067
+ "value": Program.to((1, 2)).as_bin(),
1068
+ },
1069
+ )
1070
+
1071
+ with pytest.raises(
1072
+ NodeHashError,
1073
+ match=r"\n +000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1f$",
1074
+ ):
1075
+ await raw_data_store._check_hashes()
1076
+
1077
+
1078
+ @pytest.mark.anyio
1079
+ async def test_root_state(data_store: DataStore, store_id: bytes32) -> None:
1080
+ key = b"\x01\x02"
1081
+ value = b"abc"
1082
+ await data_store.insert(
1083
+ key=key, value=value, store_id=store_id, reference_node_hash=None, side=None, status=Status.PENDING
1084
+ )
1085
+ is_empty = await data_store.table_is_empty(store_id=store_id)
1086
+ assert is_empty
1087
+
1088
+
1089
+ @pytest.mark.anyio
1090
+ async def test_change_root_state(data_store: DataStore, store_id: bytes32) -> None:
1091
+ key = b"\x01\x02"
1092
+ value = b"abc"
1093
+ await data_store.insert(
1094
+ key=key,
1095
+ value=value,
1096
+ store_id=store_id,
1097
+ reference_node_hash=None,
1098
+ side=None,
1099
+ )
1100
+ root = await data_store.get_pending_root(store_id)
1101
+ assert root is not None
1102
+ assert root.status == Status.PENDING
1103
+ is_empty = await data_store.table_is_empty(store_id=store_id)
1104
+ assert is_empty
1105
+
1106
+ await data_store.change_root_status(root, Status.PENDING_BATCH)
1107
+ root = await data_store.get_pending_root(store_id)
1108
+ assert root is not None
1109
+ assert root.status == Status.PENDING_BATCH
1110
+ is_empty = await data_store.table_is_empty(store_id=store_id)
1111
+ assert is_empty
1112
+
1113
+ await data_store.change_root_status(root, Status.COMMITTED)
1114
+ root = await data_store.get_tree_root(store_id)
1115
+ is_empty = await data_store.table_is_empty(store_id=store_id)
1116
+ assert not is_empty
1117
+ assert root.node_hash is not None
1118
+ root = await data_store.get_pending_root(store_id)
1119
+ assert root is None
1120
+
1121
+
1122
+ @pytest.mark.anyio
1123
+ async def test_kv_diff(data_store: DataStore, store_id: bytes32) -> None:
1124
+ random = Random()
1125
+ random.seed(100, version=2)
1126
+ insertions = 0
1127
+ expected_diff: Set[DiffData] = set()
1128
+ root_start = None
1129
+ for i in range(500):
1130
+ key = (i + 100).to_bytes(4, byteorder="big")
1131
+ value = (i + 200).to_bytes(4, byteorder="big")
1132
+ seed = leaf_hash(key=key, value=value)
1133
+ node_hash = await data_store.get_terminal_node_for_seed(store_id, seed)
1134
+ if random.randint(0, 4) > 0 or insertions < 10:
1135
+ insertions += 1
1136
+ side = None if node_hash is None else data_store.get_side_for_seed(seed)
1137
+
1138
+ await data_store.insert(
1139
+ key=key,
1140
+ value=value,
1141
+ store_id=store_id,
1142
+ reference_node_hash=node_hash,
1143
+ side=side,
1144
+ status=Status.COMMITTED,
1145
+ )
1146
+ if i > 200:
1147
+ expected_diff.add(DiffData(OperationType.INSERT, key, value))
1148
+ else:
1149
+ assert node_hash is not None
1150
+ node = await data_store.get_node(node_hash)
1151
+ assert isinstance(node, TerminalNode)
1152
+ await data_store.delete(key=node.key, store_id=store_id, status=Status.COMMITTED)
1153
+ if i > 200:
1154
+ if DiffData(OperationType.INSERT, node.key, node.value) in expected_diff:
1155
+ expected_diff.remove(DiffData(OperationType.INSERT, node.key, node.value))
1156
+ else:
1157
+ expected_diff.add(DiffData(OperationType.DELETE, node.key, node.value))
1158
+ if i == 200:
1159
+ root_start = await data_store.get_tree_root(store_id)
1160
+
1161
+ root_end = await data_store.get_tree_root(store_id)
1162
+ assert root_start is not None
1163
+ assert root_start.node_hash is not None
1164
+ assert root_end.node_hash is not None
1165
+ diffs = await data_store.get_kv_diff(store_id, root_start.node_hash, root_end.node_hash)
1166
+ assert diffs == expected_diff
1167
+
1168
+
1169
+ @pytest.mark.anyio
1170
+ async def test_kv_diff_2(data_store: DataStore, store_id: bytes32) -> None:
1171
+ insert_result = await data_store.insert(
1172
+ key=b"000",
1173
+ value=b"000",
1174
+ store_id=store_id,
1175
+ reference_node_hash=None,
1176
+ side=None,
1177
+ )
1178
+ empty_hash = bytes32([0] * 32)
1179
+ invalid_hash = bytes32([0] * 31 + [1])
1180
+ diff_1 = await data_store.get_kv_diff(store_id, empty_hash, insert_result.node_hash)
1181
+ assert diff_1 == {DiffData(OperationType.INSERT, b"000", b"000")}
1182
+ diff_2 = await data_store.get_kv_diff(store_id, insert_result.node_hash, empty_hash)
1183
+ assert diff_2 == {DiffData(OperationType.DELETE, b"000", b"000")}
1184
+ with pytest.raises(Exception, match=f"Unable to diff: Can't find keys and values for {invalid_hash.hex()}"):
1185
+ await data_store.get_kv_diff(store_id, invalid_hash, insert_result.node_hash)
1186
+ with pytest.raises(Exception, match=f"Unable to diff: Can't find keys and values for {invalid_hash.hex()}"):
1187
+ await data_store.get_kv_diff(store_id, insert_result.node_hash, invalid_hash)
1188
+
1189
+
1190
+ @pytest.mark.anyio
1191
+ async def test_kv_diff_3(data_store: DataStore, store_id: bytes32) -> None:
1192
+ insert_result = await data_store.autoinsert(
1193
+ key=b"000",
1194
+ value=b"000",
1195
+ store_id=store_id,
1196
+ status=Status.COMMITTED,
1197
+ )
1198
+ await data_store.delete(store_id=store_id, key=b"000", status=Status.COMMITTED)
1199
+ insert_result_2 = await data_store.autoinsert(
1200
+ key=b"000",
1201
+ value=b"001",
1202
+ store_id=store_id,
1203
+ status=Status.COMMITTED,
1204
+ )
1205
+ diff_1 = await data_store.get_kv_diff(store_id, insert_result.node_hash, insert_result_2.node_hash)
1206
+ assert diff_1 == {DiffData(OperationType.DELETE, b"000", b"000"), DiffData(OperationType.INSERT, b"000", b"001")}
1207
+ insert_result_3 = await data_store.upsert(
1208
+ key=b"000",
1209
+ new_value=b"002",
1210
+ store_id=store_id,
1211
+ status=Status.COMMITTED,
1212
+ )
1213
+ diff_2 = await data_store.get_kv_diff(store_id, insert_result_2.node_hash, insert_result_3.node_hash)
1214
+ assert diff_2 == {DiffData(OperationType.DELETE, b"000", b"001"), DiffData(OperationType.INSERT, b"000", b"002")}
1215
+
1216
+
1217
+ @pytest.mark.anyio
1218
+ async def test_rollback_to_generation(data_store: DataStore, store_id: bytes32) -> None:
1219
+ await add_0123_example(data_store, store_id)
1220
+ expected_hashes = []
1221
+ roots = await data_store.get_roots_between(store_id, 1, 5)
1222
+ for generation, root in enumerate(roots):
1223
+ expected_hashes.append((generation + 1, root.node_hash))
1224
+ for generation, expected_hash in reversed(expected_hashes):
1225
+ await data_store.rollback_to_generation(store_id, generation)
1226
+ root = await data_store.get_tree_root(store_id)
1227
+ assert root.node_hash == expected_hash
1228
+
1229
+
1230
+ @pytest.mark.anyio
1231
+ async def test_subscribe_unsubscribe(data_store: DataStore, store_id: bytes32) -> None:
1232
+ await data_store.subscribe(Subscription(store_id, [ServerInfo("http://127:0:0:1/8000", 1, 1)]))
1233
+ subscriptions = await data_store.get_subscriptions()
1234
+ urls = [server_info.url for subscription in subscriptions for server_info in subscription.servers_info]
1235
+ assert urls == ["http://127:0:0:1/8000"]
1236
+
1237
+ await data_store.subscribe(Subscription(store_id, [ServerInfo("http://127:0:0:1/8001", 2, 2)]))
1238
+ subscriptions = await data_store.get_subscriptions()
1239
+ urls = [server_info.url for subscription in subscriptions for server_info in subscription.servers_info]
1240
+ assert urls == ["http://127:0:0:1/8000", "http://127:0:0:1/8001"]
1241
+
1242
+ await data_store.subscribe(
1243
+ Subscription(
1244
+ store_id, [ServerInfo("http://127:0:0:1/8000", 100, 100), ServerInfo("http://127:0:0:1/8001", 200, 200)]
1245
+ )
1246
+ )
1247
+ subscriptions = await data_store.get_subscriptions()
1248
+ assert subscriptions == [
1249
+ Subscription(store_id, [ServerInfo("http://127:0:0:1/8000", 1, 1), ServerInfo("http://127:0:0:1/8001", 2, 2)]),
1250
+ ]
1251
+
1252
+ await data_store.unsubscribe(store_id)
1253
+ assert await data_store.get_subscriptions() == []
1254
+ store_id2 = bytes32([0] * 32)
1255
+
1256
+ await data_store.subscribe(
1257
+ Subscription(
1258
+ store_id, [ServerInfo("http://127:0:0:1/8000", 100, 100), ServerInfo("http://127:0:0:1/8001", 200, 200)]
1259
+ )
1260
+ )
1261
+ await data_store.subscribe(
1262
+ Subscription(
1263
+ store_id2, [ServerInfo("http://127:0:0:1/8000", 300, 300), ServerInfo("http://127:0:0:1/8001", 400, 400)]
1264
+ )
1265
+ )
1266
+ subscriptions = await data_store.get_subscriptions()
1267
+ assert subscriptions == [
1268
+ Subscription(
1269
+ store_id, [ServerInfo("http://127:0:0:1/8000", 100, 100), ServerInfo("http://127:0:0:1/8001", 200, 200)]
1270
+ ),
1271
+ Subscription(
1272
+ store_id2, [ServerInfo("http://127:0:0:1/8000", 300, 300), ServerInfo("http://127:0:0:1/8001", 400, 400)]
1273
+ ),
1274
+ ]
1275
+
1276
+
1277
+ @pytest.mark.anyio
1278
+ async def test_server_selection(data_store: DataStore, store_id: bytes32) -> None:
1279
+ start_timestamp = 1000
1280
+ await data_store.subscribe(
1281
+ Subscription(store_id, [ServerInfo(f"http://127.0.0.1/{port}", 0, 0) for port in range(8000, 8010)])
1282
+ )
1283
+
1284
+ free_servers = {f"http://127.0.0.1/{port}" for port in range(8000, 8010)}
1285
+ tried_servers = 0
1286
+ random = Random()
1287
+ random.seed(100, version=2)
1288
+ while len(free_servers) > 0:
1289
+ servers_info = await data_store.get_available_servers_for_store(store_id=store_id, timestamp=start_timestamp)
1290
+ random.shuffle(servers_info)
1291
+ assert servers_info != []
1292
+ server_info = servers_info[0]
1293
+ assert server_info.ignore_till == 0
1294
+ await data_store.received_incorrect_file(store_id=store_id, server_info=server_info, timestamp=start_timestamp)
1295
+ assert server_info.url in free_servers
1296
+ tried_servers += 1
1297
+ free_servers.remove(server_info.url)
1298
+
1299
+ assert tried_servers == 10
1300
+ servers_info = await data_store.get_available_servers_for_store(store_id=store_id, timestamp=start_timestamp)
1301
+ assert servers_info == []
1302
+
1303
+ current_timestamp = 2000 + 7 * 24 * 3600
1304
+ selected_servers = set()
1305
+ for _ in range(100):
1306
+ servers_info = await data_store.get_available_servers_for_store(store_id=store_id, timestamp=current_timestamp)
1307
+ random.shuffle(servers_info)
1308
+ assert servers_info != []
1309
+ selected_servers.add(servers_info[0].url)
1310
+ assert selected_servers == {f"http://127.0.0.1/{port}" for port in range(8000, 8010)}
1311
+
1312
+ for _ in range(100):
1313
+ servers_info = await data_store.get_available_servers_for_store(store_id=store_id, timestamp=current_timestamp)
1314
+ random.shuffle(servers_info)
1315
+ assert servers_info != []
1316
+ if servers_info[0].url != "http://127.0.0.1/8000":
1317
+ await data_store.received_incorrect_file(
1318
+ store_id=store_id, server_info=servers_info[0], timestamp=current_timestamp
1319
+ )
1320
+
1321
+ servers_info = await data_store.get_available_servers_for_store(store_id=store_id, timestamp=current_timestamp)
1322
+ random.shuffle(servers_info)
1323
+ assert len(servers_info) == 1
1324
+ assert servers_info[0].url == "http://127.0.0.1/8000"
1325
+ await data_store.received_correct_file(store_id=store_id, server_info=servers_info[0])
1326
+
1327
+ ban_times = [5 * 60] * 3 + [15 * 60] * 3 + [30 * 60] * 2 + [60 * 60] * 10
1328
+ for ban_time in ban_times:
1329
+ servers_info = await data_store.get_available_servers_for_store(store_id=store_id, timestamp=current_timestamp)
1330
+ assert len(servers_info) == 1
1331
+ await data_store.server_misses_file(store_id=store_id, server_info=servers_info[0], timestamp=current_timestamp)
1332
+ current_timestamp += ban_time
1333
+ servers_info = await data_store.get_available_servers_for_store(store_id=store_id, timestamp=current_timestamp)
1334
+ assert servers_info == []
1335
+ current_timestamp += 1
1336
+
1337
+
1338
+ @pytest.mark.parametrize(
1339
+ "error",
1340
+ [True, False],
1341
+ )
1342
+ @pytest.mark.anyio
1343
+ async def test_server_http_ban(
1344
+ data_store: DataStore,
1345
+ store_id: bytes32,
1346
+ error: bool,
1347
+ monkeypatch: Any,
1348
+ tmp_path: Path,
1349
+ seeded_random: random.Random,
1350
+ ) -> None:
1351
+ sinfo = ServerInfo("http://127.0.0.1/8003", 0, 0)
1352
+ await data_store.subscribe(Subscription(store_id, [sinfo]))
1353
+
1354
+ async def mock_http_download(
1355
+ target_filename_path: Path,
1356
+ filename: str,
1357
+ proxy_url: str,
1358
+ server_info: ServerInfo,
1359
+ timeout: aiohttp.ClientTimeout,
1360
+ log: logging.Logger,
1361
+ ) -> None:
1362
+ if error:
1363
+ raise aiohttp.ClientConnectionError()
1364
+
1365
+ start_timestamp = int(time.time())
1366
+ with monkeypatch.context() as m:
1367
+ m.setattr("chia.data_layer.download_data.http_download", mock_http_download)
1368
+ success = await insert_from_delta_file(
1369
+ data_store=data_store,
1370
+ store_id=store_id,
1371
+ existing_generation=3,
1372
+ target_generation=4,
1373
+ root_hashes=[bytes32.random(seeded_random)],
1374
+ server_info=sinfo,
1375
+ client_foldername=tmp_path,
1376
+ timeout=aiohttp.ClientTimeout(total=15, sock_connect=5),
1377
+ log=log,
1378
+ proxy_url="",
1379
+ downloader=None,
1380
+ )
1381
+
1382
+ assert success is False
1383
+
1384
+ subscriptions = await data_store.get_subscriptions()
1385
+ sinfo = subscriptions[0].servers_info[0]
1386
+ assert sinfo.num_consecutive_failures == 1
1387
+ assert sinfo.ignore_till >= start_timestamp + 5 * 60 # ban for 5 minutes
1388
+ start_timestamp = sinfo.ignore_till
1389
+
1390
+ with monkeypatch.context() as m:
1391
+ m.setattr("chia.data_layer.download_data.http_download", mock_http_download)
1392
+ success = await insert_from_delta_file(
1393
+ data_store=data_store,
1394
+ store_id=store_id,
1395
+ existing_generation=3,
1396
+ target_generation=4,
1397
+ root_hashes=[bytes32.random(seeded_random)],
1398
+ server_info=sinfo,
1399
+ client_foldername=tmp_path,
1400
+ timeout=aiohttp.ClientTimeout(total=15, sock_connect=5),
1401
+ log=log,
1402
+ proxy_url="",
1403
+ downloader=None,
1404
+ )
1405
+
1406
+ subscriptions = await data_store.get_subscriptions()
1407
+ sinfo = subscriptions[0].servers_info[0]
1408
+ assert sinfo.num_consecutive_failures == 2
1409
+ assert sinfo.ignore_till == start_timestamp # we don't increase on second failure
1410
+
1411
+
1412
+ @pytest.mark.parametrize(
1413
+ "test_delta",
1414
+ [True, False],
1415
+ )
1416
+ @boolean_datacases(name="group_files_by_store", false="group by singleton", true="don't group by singleton")
1417
+ @pytest.mark.anyio
1418
+ async def test_data_server_files(
1419
+ data_store: DataStore,
1420
+ store_id: bytes32,
1421
+ test_delta: bool,
1422
+ group_files_by_store: bool,
1423
+ tmp_path: Path,
1424
+ ) -> None:
1425
+ roots: List[Root] = []
1426
+ num_batches = 10
1427
+ num_ops_per_batch = 100
1428
+
1429
+ db_uri = generate_in_memory_db_uri()
1430
+ async with DataStore.managed(database=db_uri, uri=True) as data_store_server:
1431
+ await data_store_server.create_tree(store_id, status=Status.COMMITTED)
1432
+ random = Random()
1433
+ random.seed(100, version=2)
1434
+
1435
+ keys: List[bytes] = []
1436
+ counter = 0
1437
+
1438
+ for batch in range(num_batches):
1439
+ changelist: List[Dict[str, Any]] = []
1440
+ for operation in range(num_ops_per_batch):
1441
+ if random.randint(0, 4) > 0 or len(keys) == 0:
1442
+ key = counter.to_bytes(4, byteorder="big")
1443
+ value = (2 * counter).to_bytes(4, byteorder="big")
1444
+ keys.append(key)
1445
+ changelist.append({"action": "insert", "key": key, "value": value})
1446
+ else:
1447
+ key = random.choice(keys)
1448
+ keys.remove(key)
1449
+ changelist.append({"action": "delete", "key": key})
1450
+ counter += 1
1451
+ await data_store_server.insert_batch(store_id, changelist, status=Status.COMMITTED)
1452
+ root = await data_store_server.get_tree_root(store_id)
1453
+ await write_files_for_root(
1454
+ data_store_server, store_id, root, tmp_path, 0, group_by_store=group_files_by_store
1455
+ )
1456
+ roots.append(root)
1457
+
1458
+ generation = 1
1459
+ assert len(roots) == num_batches
1460
+ for root in roots:
1461
+ assert root.node_hash is not None
1462
+ if not test_delta:
1463
+ filename = get_full_tree_filename_path(tmp_path, store_id, root.node_hash, generation, group_files_by_store)
1464
+ assert filename.exists()
1465
+ else:
1466
+ filename = get_delta_filename_path(tmp_path, store_id, root.node_hash, generation, group_files_by_store)
1467
+ assert filename.exists()
1468
+ await insert_into_data_store_from_file(data_store, store_id, root.node_hash, tmp_path.joinpath(filename))
1469
+ current_root = await data_store.get_tree_root(store_id=store_id)
1470
+ assert current_root.node_hash == root.node_hash
1471
+ generation += 1
1472
+
1473
+
1474
+ @pytest.mark.anyio
1475
+ @pytest.mark.parametrize("pending_status", [Status.PENDING, Status.PENDING_BATCH])
1476
+ async def test_pending_roots(data_store: DataStore, store_id: bytes32, pending_status: Status) -> None:
1477
+ key = b"\x01\x02"
1478
+ value = b"abc"
1479
+
1480
+ await data_store.insert(
1481
+ key=key,
1482
+ value=value,
1483
+ store_id=store_id,
1484
+ reference_node_hash=None,
1485
+ side=None,
1486
+ status=Status.COMMITTED,
1487
+ )
1488
+
1489
+ key = b"\x01\x03"
1490
+ value = b"abc"
1491
+
1492
+ await data_store.autoinsert(
1493
+ key=key,
1494
+ value=value,
1495
+ store_id=store_id,
1496
+ status=pending_status,
1497
+ )
1498
+ pending_root = await data_store.get_pending_root(store_id=store_id)
1499
+ assert pending_root is not None
1500
+ assert pending_root.generation == 2 and pending_root.status == pending_status
1501
+
1502
+ await data_store.clear_pending_roots(store_id=store_id)
1503
+ pending_root = await data_store.get_pending_root(store_id=store_id)
1504
+ assert pending_root is None
1505
+
1506
+
1507
+ @pytest.mark.anyio
1508
+ @pytest.mark.parametrize("pending_status", [Status.PENDING, Status.PENDING_BATCH])
1509
+ async def test_clear_pending_roots_returns_root(
1510
+ data_store: DataStore, store_id: bytes32, pending_status: Status
1511
+ ) -> None:
1512
+ key = b"\x01\x02"
1513
+ value = b"abc"
1514
+
1515
+ await data_store.insert(
1516
+ key=key,
1517
+ value=value,
1518
+ store_id=store_id,
1519
+ reference_node_hash=None,
1520
+ side=None,
1521
+ status=pending_status,
1522
+ )
1523
+
1524
+ pending_root = await data_store.get_pending_root(store_id=store_id)
1525
+ cleared_root = await data_store.clear_pending_roots(store_id=store_id)
1526
+ assert cleared_root == pending_root
1527
+
1528
+
1529
+ @dataclass
1530
+ class BatchInsertBenchmarkCase:
1531
+ pre: int
1532
+ count: int
1533
+ limit: float
1534
+ marks: Marks = ()
1535
+
1536
+ @property
1537
+ def id(self) -> str:
1538
+ return f"pre={self.pre},count={self.count}"
1539
+
1540
+
1541
+ @dataclass
1542
+ class BatchesInsertBenchmarkCase:
1543
+ count: int
1544
+ batch_count: int
1545
+ limit: float
1546
+ marks: Marks = ()
1547
+
1548
+ @property
1549
+ def id(self) -> str:
1550
+ return f"count={self.count},batch_count={self.batch_count}"
1551
+
1552
+
1553
+ @datacases(
1554
+ BatchInsertBenchmarkCase(
1555
+ pre=0,
1556
+ count=100,
1557
+ limit=2.2,
1558
+ ),
1559
+ BatchInsertBenchmarkCase(
1560
+ pre=1_000,
1561
+ count=100,
1562
+ limit=4,
1563
+ ),
1564
+ BatchInsertBenchmarkCase(
1565
+ pre=0,
1566
+ count=1_000,
1567
+ limit=30,
1568
+ ),
1569
+ BatchInsertBenchmarkCase(
1570
+ pre=1_000,
1571
+ count=1_000,
1572
+ limit=36,
1573
+ ),
1574
+ BatchInsertBenchmarkCase(
1575
+ pre=10_000,
1576
+ count=25_000,
1577
+ limit=52,
1578
+ ),
1579
+ )
1580
+ @pytest.mark.anyio
1581
+ async def test_benchmark_batch_insert_speed(
1582
+ data_store: DataStore,
1583
+ store_id: bytes32,
1584
+ benchmark_runner: BenchmarkRunner,
1585
+ case: BatchInsertBenchmarkCase,
1586
+ ) -> None:
1587
+ r = random.Random()
1588
+ r.seed("shadowlands", version=2)
1589
+
1590
+ changelist = [
1591
+ {
1592
+ "action": "insert",
1593
+ "key": x.to_bytes(32, byteorder="big", signed=False),
1594
+ "value": bytes(r.getrandbits(8) for _ in range(1200)),
1595
+ }
1596
+ for x in range(case.pre + case.count)
1597
+ ]
1598
+
1599
+ pre = changelist[: case.pre]
1600
+ batch = changelist[case.pre : case.pre + case.count]
1601
+
1602
+ if case.pre > 0:
1603
+ await data_store.insert_batch(
1604
+ store_id=store_id,
1605
+ changelist=pre,
1606
+ status=Status.COMMITTED,
1607
+ )
1608
+
1609
+ with benchmark_runner.assert_runtime(seconds=case.limit):
1610
+ await data_store.insert_batch(
1611
+ store_id=store_id,
1612
+ changelist=batch,
1613
+ )
1614
+
1615
+
1616
+ @datacases(
1617
+ BatchesInsertBenchmarkCase(
1618
+ count=50,
1619
+ batch_count=200,
1620
+ limit=195,
1621
+ ),
1622
+ )
1623
+ @pytest.mark.anyio
1624
+ async def test_benchmark_batch_insert_speed_multiple_batches(
1625
+ data_store: DataStore,
1626
+ store_id: bytes32,
1627
+ benchmark_runner: BenchmarkRunner,
1628
+ case: BatchesInsertBenchmarkCase,
1629
+ ) -> None:
1630
+ r = random.Random()
1631
+ r.seed("shadowlands", version=2)
1632
+
1633
+ with benchmark_runner.assert_runtime(seconds=case.limit):
1634
+ for batch in range(case.batch_count):
1635
+ changelist = [
1636
+ {
1637
+ "action": "insert",
1638
+ "key": x.to_bytes(32, byteorder="big", signed=False),
1639
+ "value": bytes(r.getrandbits(8) for _ in range(10000)),
1640
+ }
1641
+ for x in range(batch * case.count, (batch + 1) * case.count)
1642
+ ]
1643
+ await data_store.insert_batch(
1644
+ store_id=store_id,
1645
+ changelist=changelist,
1646
+ status=Status.COMMITTED,
1647
+ )
1648
+
1649
+
1650
+ @pytest.mark.anyio
1651
+ async def test_delete_store_data(raw_data_store: DataStore) -> None:
1652
+ store_id = bytes32(b"\0" * 32)
1653
+ store_id_2 = bytes32(b"\0" * 31 + b"\1")
1654
+ await raw_data_store.create_tree(store_id=store_id, status=Status.COMMITTED)
1655
+ await raw_data_store.create_tree(store_id=store_id_2, status=Status.COMMITTED)
1656
+ total_keys = 4
1657
+ keys = [key.to_bytes(4, byteorder="big") for key in range(total_keys)]
1658
+ batch1 = [
1659
+ {"action": "insert", "key": keys[0], "value": keys[0]},
1660
+ {"action": "insert", "key": keys[1], "value": keys[1]},
1661
+ ]
1662
+ batch2 = batch1.copy()
1663
+ batch1.append({"action": "insert", "key": keys[2], "value": keys[2]})
1664
+ batch2.append({"action": "insert", "key": keys[3], "value": keys[3]})
1665
+ assert batch1 != batch2
1666
+ await raw_data_store.insert_batch(store_id, batch1, status=Status.COMMITTED)
1667
+ await raw_data_store.insert_batch(store_id_2, batch2, status=Status.COMMITTED)
1668
+ keys_values_before = await raw_data_store.get_keys_values(store_id_2)
1669
+ async with raw_data_store.db_wrapper.reader() as reader:
1670
+ result = await reader.execute("SELECT * FROM node")
1671
+ nodes = await result.fetchall()
1672
+ kv_nodes_before = {}
1673
+ for node in nodes:
1674
+ if node["key"] is not None:
1675
+ kv_nodes_before[node["key"]] = node["value"]
1676
+ assert [kv_nodes_before[key] for key in keys] == keys
1677
+ await raw_data_store.delete_store_data(store_id)
1678
+ # Deleting from `node` table doesn't alter other stores.
1679
+ keys_values_after = await raw_data_store.get_keys_values(store_id_2)
1680
+ assert keys_values_before == keys_values_after
1681
+ async with raw_data_store.db_wrapper.reader() as reader:
1682
+ result = await reader.execute("SELECT * FROM node")
1683
+ nodes = await result.fetchall()
1684
+ kv_nodes_after = {}
1685
+ for node in nodes:
1686
+ if node["key"] is not None:
1687
+ kv_nodes_after[node["key"]] = node["value"]
1688
+ for i in range(total_keys):
1689
+ if i != 2:
1690
+ assert kv_nodes_after[keys[i]] == keys[i]
1691
+ else:
1692
+ # `keys[2]` was only present in the first store.
1693
+ assert keys[i] not in kv_nodes_after
1694
+ assert not await raw_data_store.store_id_exists(store_id)
1695
+ await raw_data_store.delete_store_data(store_id_2)
1696
+ async with raw_data_store.db_wrapper.reader() as reader:
1697
+ async with reader.execute("SELECT COUNT(*) FROM node") as cursor:
1698
+ row_count = await cursor.fetchone()
1699
+ assert row_count is not None
1700
+ assert row_count[0] == 0
1701
+ assert not await raw_data_store.store_id_exists(store_id_2)
1702
+
1703
+
1704
+ @pytest.mark.anyio
1705
+ async def test_delete_store_data_multiple_stores(raw_data_store: DataStore) -> None:
1706
+ # Make sure inserting and deleting the same data works
1707
+ for repetition in range(2):
1708
+ num_stores = 50
1709
+ total_keys = 150
1710
+ keys_deleted_per_store = 3
1711
+ store_ids = [bytes32(i.to_bytes(32, byteorder="big")) for i in range(num_stores)]
1712
+ for store_id in store_ids:
1713
+ await raw_data_store.create_tree(store_id=store_id, status=Status.COMMITTED)
1714
+ original_keys = [key.to_bytes(4, byteorder="big") for key in range(total_keys)]
1715
+ batches = []
1716
+ for i in range(num_stores):
1717
+ batch = [
1718
+ {"action": "insert", "key": key, "value": key} for key in original_keys[i * keys_deleted_per_store :]
1719
+ ]
1720
+ batches.append(batch)
1721
+
1722
+ for store_id, batch in zip(store_ids, batches):
1723
+ await raw_data_store.insert_batch(store_id, batch, status=Status.COMMITTED)
1724
+
1725
+ for tree_index in range(num_stores):
1726
+ async with raw_data_store.db_wrapper.reader() as reader:
1727
+ result = await reader.execute("SELECT * FROM node")
1728
+ nodes = await result.fetchall()
1729
+
1730
+ keys = {node["key"] for node in nodes if node["key"] is not None}
1731
+ assert len(keys) == total_keys - tree_index * keys_deleted_per_store
1732
+ keys_after_index = set(original_keys[tree_index * keys_deleted_per_store :])
1733
+ keys_before_index = set(original_keys[: tree_index * keys_deleted_per_store])
1734
+ assert keys_after_index.issubset(keys)
1735
+ assert keys.isdisjoint(keys_before_index)
1736
+ await raw_data_store.delete_store_data(store_ids[tree_index])
1737
+
1738
+ async with raw_data_store.db_wrapper.reader() as reader:
1739
+ async with reader.execute("SELECT COUNT(*) FROM node") as cursor:
1740
+ row_count = await cursor.fetchone()
1741
+ assert row_count is not None
1742
+ assert row_count[0] == 0
1743
+
1744
+
1745
+ @pytest.mark.parametrize("common_keys_count", [1, 250, 499])
1746
+ @pytest.mark.anyio
1747
+ async def test_delete_store_data_with_common_values(raw_data_store: DataStore, common_keys_count: int) -> None:
1748
+ store_id_1 = bytes32(b"\x00" * 31 + b"\x01")
1749
+ store_id_2 = bytes32(b"\x00" * 31 + b"\x02")
1750
+
1751
+ await raw_data_store.create_tree(store_id=store_id_1, status=Status.COMMITTED)
1752
+ await raw_data_store.create_tree(store_id=store_id_2, status=Status.COMMITTED)
1753
+
1754
+ key_offset = 1000
1755
+ total_keys_per_store = 500
1756
+ assert common_keys_count < key_offset
1757
+ common_keys = {key.to_bytes(4, byteorder="big") for key in range(common_keys_count)}
1758
+ unique_keys_1 = {
1759
+ (key + key_offset).to_bytes(4, byteorder="big") for key in range(total_keys_per_store - common_keys_count)
1760
+ }
1761
+ unique_keys_2 = {
1762
+ (key + (2 * key_offset)).to_bytes(4, byteorder="big") for key in range(total_keys_per_store - common_keys_count)
1763
+ }
1764
+
1765
+ batch1 = [{"action": "insert", "key": key, "value": key} for key in common_keys.union(unique_keys_1)]
1766
+ batch2 = [{"action": "insert", "key": key, "value": key} for key in common_keys.union(unique_keys_2)]
1767
+
1768
+ await raw_data_store.insert_batch(store_id_1, batch1, status=Status.COMMITTED)
1769
+ await raw_data_store.insert_batch(store_id_2, batch2, status=Status.COMMITTED)
1770
+
1771
+ await raw_data_store.delete_store_data(store_id_1)
1772
+ async with raw_data_store.db_wrapper.reader() as reader:
1773
+ result = await reader.execute("SELECT * FROM node")
1774
+ nodes = await result.fetchall()
1775
+
1776
+ keys = {node["key"] for node in nodes if node["key"] is not None}
1777
+ # Since one store got all its keys deleted, we're left only with the keys of the other store.
1778
+ assert len(keys) == total_keys_per_store
1779
+ assert keys.intersection(unique_keys_1) == set()
1780
+ assert keys.symmetric_difference(common_keys.union(unique_keys_2)) == set()
1781
+
1782
+
1783
+ @pytest.mark.anyio
1784
+ @pytest.mark.parametrize("pending_status", [Status.PENDING, Status.PENDING_BATCH])
1785
+ async def test_delete_store_data_protects_pending_roots(raw_data_store: DataStore, pending_status: Status) -> None:
1786
+ num_stores = 5
1787
+ total_keys = 15
1788
+ store_ids = [bytes32(i.to_bytes(32, byteorder="big")) for i in range(num_stores)]
1789
+ for store_id in store_ids:
1790
+ await raw_data_store.create_tree(store_id=store_id, status=Status.COMMITTED)
1791
+ original_keys = [key.to_bytes(4, byteorder="big") for key in range(total_keys)]
1792
+ batches = []
1793
+ keys_per_pending_root = 2
1794
+
1795
+ for i in range(num_stores - 1):
1796
+ start_index = i * keys_per_pending_root
1797
+ end_index = (i + 1) * keys_per_pending_root
1798
+ batch = [{"action": "insert", "key": key, "value": key} for key in original_keys[start_index:end_index]]
1799
+ batches.append(batch)
1800
+ for store_id, batch in zip(store_ids, batches):
1801
+ await raw_data_store.insert_batch(store_id, batch, status=pending_status)
1802
+
1803
+ store_id = store_ids[-1]
1804
+ batch = [{"action": "insert", "key": key, "value": key} for key in original_keys]
1805
+ await raw_data_store.insert_batch(store_id, batch, status=Status.COMMITTED)
1806
+
1807
+ async with raw_data_store.db_wrapper.reader() as reader:
1808
+ result = await reader.execute("SELECT * FROM node")
1809
+ nodes = await result.fetchall()
1810
+
1811
+ keys = {node["key"] for node in nodes if node["key"] is not None}
1812
+ assert keys == set(original_keys)
1813
+
1814
+ await raw_data_store.delete_store_data(store_id)
1815
+ async with raw_data_store.db_wrapper.reader() as reader:
1816
+ result = await reader.execute("SELECT * FROM node")
1817
+ nodes = await result.fetchall()
1818
+
1819
+ keys = {node["key"] for node in nodes if node["key"] is not None}
1820
+ assert keys == set(original_keys[: (num_stores - 1) * keys_per_pending_root])
1821
+
1822
+ for index in range(num_stores - 1):
1823
+ store_id = store_ids[index]
1824
+ root = await raw_data_store.get_pending_root(store_id)
1825
+ assert root is not None
1826
+ await raw_data_store.change_root_status(root, Status.COMMITTED)
1827
+ kv = await raw_data_store.get_keys_values(store_id=store_id)
1828
+ start_index = index * keys_per_pending_root
1829
+ end_index = (index + 1) * keys_per_pending_root
1830
+ assert {pair.key for pair in kv} == set(original_keys[start_index:end_index])
1831
+
1832
+
1833
+ @pytest.mark.anyio
1834
+ @boolean_datacases(name="group_files_by_store", true="group by singleton", false="don't group by singleton")
1835
+ @pytest.mark.parametrize("max_full_files", [1, 2, 5])
1836
+ async def test_insert_from_delta_file(
1837
+ data_store: DataStore,
1838
+ store_id: bytes32,
1839
+ monkeypatch: Any,
1840
+ tmp_path: Path,
1841
+ seeded_random: random.Random,
1842
+ group_files_by_store: bool,
1843
+ max_full_files: int,
1844
+ ) -> None:
1845
+ await data_store.create_tree(store_id=store_id, status=Status.COMMITTED)
1846
+ num_files = 5
1847
+ for generation in range(num_files):
1848
+ key = generation.to_bytes(4, byteorder="big")
1849
+ value = generation.to_bytes(4, byteorder="big")
1850
+ await data_store.autoinsert(
1851
+ key=key,
1852
+ value=value,
1853
+ store_id=store_id,
1854
+ status=Status.COMMITTED,
1855
+ )
1856
+
1857
+ root = await data_store.get_tree_root(store_id=store_id)
1858
+ assert root.generation == num_files + 1
1859
+ root_hashes = []
1860
+
1861
+ tmp_path_1 = tmp_path.joinpath("1")
1862
+ tmp_path_2 = tmp_path.joinpath("2")
1863
+
1864
+ for generation in range(1, num_files + 2):
1865
+ root = await data_store.get_tree_root(store_id=store_id, generation=generation)
1866
+ await write_files_for_root(data_store, store_id, root, tmp_path_1, 0, False, group_files_by_store)
1867
+ root_hashes.append(bytes32([0] * 32) if root.node_hash is None else root.node_hash)
1868
+ store_path = tmp_path_1.joinpath(f"{store_id}") if group_files_by_store else tmp_path_1
1869
+ with os.scandir(store_path) as entries:
1870
+ filenames = {entry.name for entry in entries}
1871
+ assert len(filenames) == 2 * (num_files + 1)
1872
+ for filename in filenames:
1873
+ if "full" in filename:
1874
+ store_path.joinpath(filename).unlink()
1875
+ with os.scandir(store_path) as entries:
1876
+ filenames = {entry.name for entry in entries}
1877
+ assert len(filenames) == num_files + 1
1878
+ kv_before = await data_store.get_keys_values(store_id=store_id)
1879
+ await data_store.rollback_to_generation(store_id, 0)
1880
+ root = await data_store.get_tree_root(store_id=store_id)
1881
+ assert root.generation == 0
1882
+ os.rename(store_path, tmp_path_2)
1883
+
1884
+ async def mock_http_download(
1885
+ target_filename_path: Path,
1886
+ filename: str,
1887
+ proxy_url: str,
1888
+ server_info: ServerInfo,
1889
+ timeout: int,
1890
+ log: logging.Logger,
1891
+ ) -> None:
1892
+ pass
1893
+
1894
+ async def mock_http_download_2(
1895
+ target_filename_path: Path,
1896
+ filename: str,
1897
+ proxy_url: str,
1898
+ server_info: ServerInfo,
1899
+ timeout: int,
1900
+ log: logging.Logger,
1901
+ ) -> None:
1902
+ try:
1903
+ os.rmdir(store_path)
1904
+ except OSError:
1905
+ pass
1906
+ os.rename(tmp_path_2, store_path)
1907
+
1908
+ sinfo = ServerInfo("http://127.0.0.1/8003", 0, 0)
1909
+ with monkeypatch.context() as m:
1910
+ m.setattr("chia.data_layer.download_data.http_download", mock_http_download)
1911
+ success = await insert_from_delta_file(
1912
+ data_store=data_store,
1913
+ store_id=store_id,
1914
+ existing_generation=0,
1915
+ target_generation=num_files + 1,
1916
+ root_hashes=root_hashes,
1917
+ server_info=sinfo,
1918
+ client_foldername=tmp_path_1,
1919
+ timeout=aiohttp.ClientTimeout(total=15, sock_connect=5),
1920
+ log=log,
1921
+ proxy_url="",
1922
+ downloader=None,
1923
+ group_files_by_store=group_files_by_store,
1924
+ maximum_full_file_count=max_full_files,
1925
+ )
1926
+ assert not success
1927
+
1928
+ root = await data_store.get_tree_root(store_id=store_id)
1929
+ assert root.generation == 0
1930
+
1931
+ sinfo = ServerInfo("http://127.0.0.1/8003", 0, 0)
1932
+ with monkeypatch.context() as m:
1933
+ m.setattr("chia.data_layer.download_data.http_download", mock_http_download_2)
1934
+ success = await insert_from_delta_file(
1935
+ data_store=data_store,
1936
+ store_id=store_id,
1937
+ existing_generation=0,
1938
+ target_generation=num_files + 1,
1939
+ root_hashes=root_hashes,
1940
+ server_info=sinfo,
1941
+ client_foldername=tmp_path_1,
1942
+ timeout=aiohttp.ClientTimeout(total=15, sock_connect=5),
1943
+ log=log,
1944
+ proxy_url="",
1945
+ downloader=None,
1946
+ group_files_by_store=group_files_by_store,
1947
+ maximum_full_file_count=max_full_files,
1948
+ )
1949
+ assert success
1950
+
1951
+ root = await data_store.get_tree_root(store_id=store_id)
1952
+ assert root.generation == num_files + 1
1953
+ with os.scandir(store_path) as entries:
1954
+ filenames = {entry.name for entry in entries}
1955
+ assert len(filenames) == num_files + 1 + max_full_files # 6 deltas and max_full_files full files
1956
+ kv = await data_store.get_keys_values(store_id=store_id)
1957
+ assert kv == kv_before
1958
+
1959
+
1960
+ @pytest.mark.anyio
1961
+ async def test_get_node_by_key_with_overlapping_keys(raw_data_store: DataStore) -> None:
1962
+ num_stores = 5
1963
+ num_keys = 20
1964
+ values_offset = 10000
1965
+ repetitions = 25
1966
+ random = Random()
1967
+ random.seed(100, version=2)
1968
+
1969
+ store_ids = [bytes32(i.to_bytes(32, byteorder="big")) for i in range(num_stores)]
1970
+ for store_id in store_ids:
1971
+ await raw_data_store.create_tree(store_id=store_id, status=Status.COMMITTED)
1972
+ keys = [key.to_bytes(4, byteorder="big") for key in range(num_keys)]
1973
+ for repetition in range(repetitions):
1974
+ for index, store_id in enumerate(store_ids):
1975
+ values = [
1976
+ (value + values_offset * repetition).to_bytes(4, byteorder="big")
1977
+ for value in range(index * num_keys, (index + 1) * num_keys)
1978
+ ]
1979
+ batch = []
1980
+ for key, value in zip(keys, values):
1981
+ batch.append({"action": "upsert", "key": key, "value": value})
1982
+ await raw_data_store.insert_batch(store_id, batch, status=Status.COMMITTED)
1983
+
1984
+ for index, store_id in enumerate(store_ids):
1985
+ values = [
1986
+ (value + values_offset * repetition).to_bytes(4, byteorder="big")
1987
+ for value in range(index * num_keys, (index + 1) * num_keys)
1988
+ ]
1989
+ for key, value in zip(keys, values):
1990
+ node = await raw_data_store.get_node_by_key(store_id=store_id, key=key)
1991
+ assert node.value == value
1992
+ if random.randint(0, 4) == 0:
1993
+ batch = [{"action": "delete", "key": key}]
1994
+ await raw_data_store.insert_batch(store_id, batch, status=Status.COMMITTED)
1995
+ with pytest.raises(KeyNotFoundError, match=f"Key not found: {key.hex()}"):
1996
+ await raw_data_store.get_node_by_key(store_id=store_id, key=key)
1997
+
1998
+
1999
+ @pytest.mark.anyio
2000
+ @boolean_datacases(name="group_files_by_store", true="group by singleton", false="don't group by singleton")
2001
+ async def test_insert_from_delta_file_correct_file_exists(
2002
+ data_store: DataStore, store_id: bytes32, tmp_path: Path, group_files_by_store: bool
2003
+ ) -> None:
2004
+ await data_store.create_tree(store_id=store_id, status=Status.COMMITTED)
2005
+ num_files = 5
2006
+ for generation in range(num_files):
2007
+ key = generation.to_bytes(4, byteorder="big")
2008
+ value = generation.to_bytes(4, byteorder="big")
2009
+ await data_store.autoinsert(
2010
+ key=key,
2011
+ value=value,
2012
+ store_id=store_id,
2013
+ status=Status.COMMITTED,
2014
+ )
2015
+
2016
+ root = await data_store.get_tree_root(store_id=store_id)
2017
+ assert root.generation == num_files + 1
2018
+ root_hashes = []
2019
+ for generation in range(1, num_files + 2):
2020
+ root = await data_store.get_tree_root(store_id=store_id, generation=generation)
2021
+ await write_files_for_root(data_store, store_id, root, tmp_path, 0, group_by_store=group_files_by_store)
2022
+ root_hashes.append(bytes32([0] * 32) if root.node_hash is None else root.node_hash)
2023
+ store_path = tmp_path.joinpath(f"{store_id}") if group_files_by_store else tmp_path
2024
+ with os.scandir(store_path) as entries:
2025
+ filenames = {entry.name for entry in entries}
2026
+ assert len(filenames) == 2 * (num_files + 1)
2027
+ for filename in filenames:
2028
+ if "full" in filename:
2029
+ store_path.joinpath(filename).unlink()
2030
+ with os.scandir(store_path) as entries:
2031
+ filenames = {entry.name for entry in entries}
2032
+ assert len(filenames) == num_files + 1
2033
+ kv_before = await data_store.get_keys_values(store_id=store_id)
2034
+ await data_store.rollback_to_generation(store_id, 0)
2035
+ root = await data_store.get_tree_root(store_id=store_id)
2036
+ assert root.generation == 0
2037
+
2038
+ sinfo = ServerInfo("http://127.0.0.1/8003", 0, 0)
2039
+ success = await insert_from_delta_file(
2040
+ data_store=data_store,
2041
+ store_id=store_id,
2042
+ existing_generation=0,
2043
+ target_generation=num_files + 1,
2044
+ root_hashes=root_hashes,
2045
+ server_info=sinfo,
2046
+ client_foldername=tmp_path,
2047
+ timeout=aiohttp.ClientTimeout(total=15, sock_connect=5),
2048
+ log=log,
2049
+ proxy_url="",
2050
+ downloader=None,
2051
+ group_files_by_store=group_files_by_store,
2052
+ )
2053
+ assert success
2054
+
2055
+ root = await data_store.get_tree_root(store_id=store_id)
2056
+ assert root.generation == num_files + 1
2057
+ with os.scandir(store_path) as entries:
2058
+ filenames = {entry.name for entry in entries}
2059
+ assert len(filenames) == num_files + 2 # 1 full and 6 deltas
2060
+ kv = await data_store.get_keys_values(store_id=store_id)
2061
+ assert kv == kv_before
2062
+
2063
+
2064
+ @pytest.mark.anyio
2065
+ @boolean_datacases(name="group_files_by_store", true="group by singleton", false="don't group by singleton")
2066
+ async def test_insert_from_delta_file_incorrect_file_exists(
2067
+ data_store: DataStore, store_id: bytes32, tmp_path: Path, group_files_by_store: bool
2068
+ ) -> None:
2069
+ await data_store.create_tree(store_id=store_id, status=Status.COMMITTED)
2070
+ root = await data_store.get_tree_root(store_id=store_id)
2071
+ assert root.generation == 1
2072
+
2073
+ key = b"a"
2074
+ value = b"a"
2075
+ await data_store.autoinsert(
2076
+ key=key,
2077
+ value=value,
2078
+ store_id=store_id,
2079
+ status=Status.COMMITTED,
2080
+ )
2081
+
2082
+ root = await data_store.get_tree_root(store_id=store_id)
2083
+ assert root.generation == 2
2084
+ await write_files_for_root(data_store, store_id, root, tmp_path, 0, group_by_store=group_files_by_store)
2085
+
2086
+ incorrect_root_hash = bytes32([0] * 31 + [1])
2087
+ store_path = tmp_path.joinpath(f"{store_id}") if group_files_by_store else tmp_path
2088
+ with os.scandir(store_path) as entries:
2089
+ filenames = [entry.name for entry in entries]
2090
+ assert len(filenames) == 2
2091
+ os.rename(
2092
+ store_path.joinpath(filenames[0]),
2093
+ get_delta_filename_path(tmp_path, store_id, incorrect_root_hash, 2, group_files_by_store),
2094
+ )
2095
+ os.rename(
2096
+ store_path.joinpath(filenames[1]),
2097
+ get_full_tree_filename_path(tmp_path, store_id, incorrect_root_hash, 2, group_files_by_store),
2098
+ )
2099
+
2100
+ await data_store.rollback_to_generation(store_id, 1)
2101
+ sinfo = ServerInfo("http://127.0.0.1/8003", 0, 0)
2102
+ success = await insert_from_delta_file(
2103
+ data_store=data_store,
2104
+ store_id=store_id,
2105
+ existing_generation=1,
2106
+ target_generation=6,
2107
+ root_hashes=[incorrect_root_hash],
2108
+ server_info=sinfo,
2109
+ client_foldername=tmp_path,
2110
+ timeout=aiohttp.ClientTimeout(total=15, sock_connect=5),
2111
+ log=log,
2112
+ proxy_url="",
2113
+ downloader=None,
2114
+ group_files_by_store=group_files_by_store,
2115
+ )
2116
+ assert not success
2117
+
2118
+ root = await data_store.get_tree_root(store_id=store_id)
2119
+ assert root.generation == 1
2120
+ with os.scandir(store_path) as entries:
2121
+ filenames = [entry.name for entry in entries]
2122
+ assert len(filenames) == 0
2123
+
2124
+
2125
+ @pytest.mark.anyio
2126
+ async def test_insert_key_already_present(data_store: DataStore, store_id: bytes32) -> None:
2127
+ key = b"foo"
2128
+ value = b"bar"
2129
+ await data_store.insert(
2130
+ key=key, value=value, store_id=store_id, reference_node_hash=None, side=None, status=Status.COMMITTED
2131
+ )
2132
+ with pytest.raises(Exception, match=f"Key already present: {key.hex()}"):
2133
+ await data_store.insert(key=key, value=value, store_id=store_id, reference_node_hash=None, side=None)
2134
+
2135
+
2136
+ @pytest.mark.anyio
2137
+ @boolean_datacases(name="use_batch_autoinsert", false="not optimized batch insert", true="optimized batch insert")
2138
+ async def test_batch_insert_key_already_present(
2139
+ data_store: DataStore,
2140
+ store_id: bytes32,
2141
+ use_batch_autoinsert: bool,
2142
+ ) -> None:
2143
+ key = b"foo"
2144
+ value = b"bar"
2145
+ changelist = [{"action": "insert", "key": key, "value": value}]
2146
+ await data_store.insert_batch(store_id, changelist, Status.COMMITTED, use_batch_autoinsert)
2147
+ with pytest.raises(Exception, match=f"Key already present: {key.hex()}"):
2148
+ await data_store.insert_batch(store_id, changelist, Status.COMMITTED, use_batch_autoinsert)
2149
+
2150
+
2151
+ @pytest.mark.anyio
2152
+ @boolean_datacases(name="use_upsert", false="update with delete and insert", true="update with upsert")
2153
+ async def test_update_keys(data_store: DataStore, store_id: bytes32, use_upsert: bool) -> None:
2154
+ num_keys = 10
2155
+ missing_keys = 50
2156
+ num_values = 10
2157
+ new_keys = 10
2158
+ for value in range(num_values):
2159
+ changelist: List[Dict[str, Any]] = []
2160
+ bytes_value = value.to_bytes(4, byteorder="big")
2161
+ if use_upsert:
2162
+ for key in range(num_keys):
2163
+ bytes_key = key.to_bytes(4, byteorder="big")
2164
+ changelist.append({"action": "upsert", "key": bytes_key, "value": bytes_value})
2165
+ else:
2166
+ for key in range(num_keys + missing_keys):
2167
+ bytes_key = key.to_bytes(4, byteorder="big")
2168
+ changelist.append({"action": "delete", "key": bytes_key})
2169
+ for key in range(num_keys):
2170
+ bytes_key = key.to_bytes(4, byteorder="big")
2171
+ changelist.append({"action": "insert", "key": bytes_key, "value": bytes_value})
2172
+
2173
+ await data_store.insert_batch(
2174
+ store_id=store_id,
2175
+ changelist=changelist,
2176
+ status=Status.COMMITTED,
2177
+ )
2178
+ for key in range(num_keys):
2179
+ bytes_key = key.to_bytes(4, byteorder="big")
2180
+ node = await data_store.get_node_by_key(bytes_key, store_id)
2181
+ assert node.value == bytes_value
2182
+ for key in range(num_keys, num_keys + missing_keys):
2183
+ bytes_key = key.to_bytes(4, byteorder="big")
2184
+ with pytest.raises(KeyNotFoundError, match=f"Key not found: {bytes_key.hex()}"):
2185
+ await data_store.get_node_by_key(bytes_key, store_id)
2186
+ num_keys += new_keys
2187
+
2188
+
2189
+ @pytest.mark.anyio
2190
+ async def test_migration_unknown_version(data_store: DataStore) -> None:
2191
+ async with data_store.db_wrapper.writer() as writer:
2192
+ await writer.execute(
2193
+ "INSERT INTO schema(version_id) VALUES(:version_id)",
2194
+ {
2195
+ "version_id": "unknown version",
2196
+ },
2197
+ )
2198
+ with pytest.raises(Exception, match="Unknown version"):
2199
+ await data_store.migrate_db()
2200
+
2201
+
2202
+ async def _check_ancestors(
2203
+ data_store: DataStore, store_id: bytes32, root_hash: bytes32
2204
+ ) -> Dict[bytes32, Optional[bytes32]]:
2205
+ ancestors: Dict[bytes32, Optional[bytes32]] = {}
2206
+ root_node: Node = await data_store.get_node(root_hash)
2207
+ queue: List[Node] = [root_node]
2208
+
2209
+ while queue:
2210
+ node = queue.pop(0)
2211
+ if isinstance(node, InternalNode):
2212
+ left_node = await data_store.get_node(node.left_hash)
2213
+ right_node = await data_store.get_node(node.right_hash)
2214
+ ancestors[left_node.hash] = node.hash
2215
+ ancestors[right_node.hash] = node.hash
2216
+ queue.append(left_node)
2217
+ queue.append(right_node)
2218
+
2219
+ ancestors[root_hash] = None
2220
+ for node_hash, ancestor_hash in ancestors.items():
2221
+ ancestor_node = await data_store._get_one_ancestor(node_hash, store_id)
2222
+ if ancestor_hash is None:
2223
+ assert ancestor_node is None
2224
+ else:
2225
+ assert ancestor_node is not None
2226
+ assert ancestor_node.hash == ancestor_hash
2227
+
2228
+ return ancestors
2229
+
2230
+
2231
+ @pytest.mark.anyio
2232
+ async def test_build_ancestor_table(data_store: DataStore, store_id: bytes32) -> None:
2233
+ num_values = 1000
2234
+ changelist: List[Dict[str, Any]] = []
2235
+ for value in range(num_values):
2236
+ value_bytes = value.to_bytes(4, byteorder="big")
2237
+ changelist.append({"action": "upsert", "key": value_bytes, "value": value_bytes})
2238
+ await data_store.insert_batch(
2239
+ store_id=store_id,
2240
+ changelist=changelist,
2241
+ status=Status.PENDING,
2242
+ )
2243
+
2244
+ pending_root = await data_store.get_pending_root(store_id=store_id)
2245
+ assert pending_root is not None
2246
+ assert pending_root.node_hash is not None
2247
+ await data_store.change_root_status(pending_root, Status.COMMITTED)
2248
+ await data_store.build_ancestor_table_for_latest_root(store_id=store_id)
2249
+
2250
+ assert pending_root.node_hash is not None
2251
+ await _check_ancestors(data_store, store_id, pending_root.node_hash)
2252
+
2253
+
2254
+ @pytest.mark.anyio
2255
+ async def test_sparse_ancestor_table(data_store: DataStore, store_id: bytes32) -> None:
2256
+ num_values = 100
2257
+ for value in range(num_values):
2258
+ value_bytes = value.to_bytes(4, byteorder="big")
2259
+ await data_store.autoinsert(
2260
+ key=value_bytes,
2261
+ value=value_bytes,
2262
+ store_id=store_id,
2263
+ status=Status.COMMITTED,
2264
+ )
2265
+ root = await data_store.get_tree_root(store_id=store_id)
2266
+ assert root.node_hash is not None
2267
+ ancestors = await _check_ancestors(data_store, store_id, root.node_hash)
2268
+
2269
+ # Check the ancestor table is sparse
2270
+ root_generation = root.generation
2271
+ current_generation_count = 0
2272
+ previous_generation_count = 0
2273
+ for node_hash, ancestor_hash in ancestors.items():
2274
+ async with data_store.db_wrapper.reader() as reader:
2275
+ if ancestor_hash is not None:
2276
+ cursor = await reader.execute(
2277
+ "SELECT MAX(generation) AS generation FROM ancestors WHERE hash == :hash AND ancestor == :ancestor",
2278
+ {"hash": node_hash, "ancestor": ancestor_hash},
2279
+ )
2280
+ else:
2281
+ cursor = await reader.execute(
2282
+ "SELECT MAX(generation) AS generation FROM ancestors WHERE hash == :hash AND ancestor IS NULL",
2283
+ {"hash": node_hash},
2284
+ )
2285
+ row = await cursor.fetchone()
2286
+ assert row is not None
2287
+ generation = row["generation"]
2288
+ assert generation <= root_generation
2289
+ if generation == root_generation:
2290
+ current_generation_count += 1
2291
+ else:
2292
+ previous_generation_count += 1
2293
+
2294
+ assert current_generation_count == 15
2295
+ assert previous_generation_count == 184
2296
+
2297
+
2298
+ async def get_all_nodes(data_store: DataStore, store_id: bytes32) -> List[Node]:
2299
+ root = await data_store.get_tree_root(store_id)
2300
+ assert root.node_hash is not None
2301
+ root_node = await data_store.get_node(root.node_hash)
2302
+ nodes: List[Node] = []
2303
+ queue: List[Node] = [root_node]
2304
+
2305
+ while len(queue) > 0:
2306
+ node = queue.pop(0)
2307
+ nodes.append(node)
2308
+ if isinstance(node, InternalNode):
2309
+ left_node = await data_store.get_node(node.left_hash)
2310
+ right_node = await data_store.get_node(node.right_hash)
2311
+ queue.append(left_node)
2312
+ queue.append(right_node)
2313
+
2314
+ return nodes
2315
+
2316
+
2317
+ @pytest.mark.anyio
2318
+ async def test_get_nodes(data_store: DataStore, store_id: bytes32) -> None:
2319
+ num_values = 50
2320
+ changelist: List[Dict[str, Any]] = []
2321
+
2322
+ for value in range(num_values):
2323
+ value_bytes = value.to_bytes(4, byteorder="big")
2324
+ changelist.append({"action": "upsert", "key": value_bytes, "value": value_bytes})
2325
+ await data_store.insert_batch(
2326
+ store_id=store_id,
2327
+ changelist=changelist,
2328
+ status=Status.COMMITTED,
2329
+ )
2330
+
2331
+ expected_nodes = await get_all_nodes(data_store, store_id)
2332
+ nodes = await data_store.get_nodes([node.hash for node in expected_nodes])
2333
+ assert nodes == expected_nodes
2334
+
2335
+ node_hash = bytes32([0] * 32)
2336
+ node_hash_2 = bytes32([0] * 31 + [1])
2337
+ with pytest.raises(Exception, match=f"^Nodes not found for hashes: {node_hash.hex()}, {node_hash_2.hex()}"):
2338
+ await data_store.get_nodes([node_hash, node_hash_2] + [node.hash for node in expected_nodes])
2339
+
2340
+
2341
+ @pytest.mark.anyio
2342
+ @pytest.mark.parametrize("pre", [0, 2048])
2343
+ @pytest.mark.parametrize("batch_size", [25, 100, 500])
2344
+ async def test_get_leaf_at_minimum_height(
2345
+ data_store: DataStore,
2346
+ store_id: bytes32,
2347
+ pre: int,
2348
+ batch_size: int,
2349
+ ) -> None:
2350
+ num_values = 1000
2351
+ value_offset = 1000000
2352
+ all_min_leafs: Set[TerminalNode] = set()
2353
+
2354
+ if pre > 0:
2355
+ # This builds a complete binary tree, in order to test more than one batch in the queue before finding the leaf
2356
+ changelist: List[Dict[str, Any]] = []
2357
+
2358
+ for value in range(pre):
2359
+ value_bytes = (value * value).to_bytes(8, byteorder="big")
2360
+ changelist.append({"action": "upsert", "key": value_bytes, "value": value_bytes})
2361
+ await data_store.insert_batch(
2362
+ store_id=store_id,
2363
+ changelist=changelist,
2364
+ status=Status.COMMITTED,
2365
+ )
2366
+
2367
+ for value in range(num_values):
2368
+ value_bytes = value.to_bytes(4, byteorder="big")
2369
+ # Use autoinsert instead of `insert_batch` to get a more randomly shaped tree
2370
+ await data_store.autoinsert(
2371
+ key=value_bytes,
2372
+ value=value_bytes,
2373
+ store_id=store_id,
2374
+ status=Status.COMMITTED,
2375
+ )
2376
+
2377
+ if (value + 1) % batch_size == 0:
2378
+ hash_to_parent: Dict[bytes32, InternalNode] = {}
2379
+ root = await data_store.get_tree_root(store_id)
2380
+ assert root.node_hash is not None
2381
+ min_leaf = await data_store.get_leaf_at_minimum_height(root.node_hash, hash_to_parent)
2382
+ all_nodes = await get_all_nodes(data_store, store_id)
2383
+ heights: Dict[bytes32, int] = {}
2384
+ heights[root.node_hash] = 0
2385
+ min_leaf_height = None
2386
+
2387
+ for node in all_nodes:
2388
+ if isinstance(node, InternalNode):
2389
+ heights[node.left_hash] = heights[node.hash] + 1
2390
+ heights[node.right_hash] = heights[node.hash] + 1
2391
+ else:
2392
+ if min_leaf_height is not None:
2393
+ min_leaf_height = min(min_leaf_height, heights[node.hash])
2394
+ else:
2395
+ min_leaf_height = heights[node.hash]
2396
+
2397
+ assert min_leaf_height is not None
2398
+ if pre > 0:
2399
+ assert min_leaf_height >= 11
2400
+ for node in all_nodes:
2401
+ if isinstance(node, TerminalNode):
2402
+ assert node == min_leaf
2403
+ assert heights[min_leaf.hash] == min_leaf_height
2404
+ break
2405
+ if node.left_hash in hash_to_parent:
2406
+ assert hash_to_parent[node.left_hash] == node
2407
+ if node.right_hash in hash_to_parent:
2408
+ assert hash_to_parent[node.right_hash] == node
2409
+
2410
+ # Push down the min height leaf, so on the next iteration we get a different leaf
2411
+ pushdown_height = 20
2412
+ for repeat in range(pushdown_height):
2413
+ value_bytes = (value + (repeat + 1) * value_offset).to_bytes(4, byteorder="big")
2414
+ await data_store.insert(
2415
+ key=value_bytes,
2416
+ value=value_bytes,
2417
+ store_id=store_id,
2418
+ reference_node_hash=min_leaf.hash,
2419
+ side=Side.RIGHT,
2420
+ status=Status.COMMITTED,
2421
+ )
2422
+ assert min_leaf not in all_min_leafs
2423
+ all_min_leafs.add(min_leaf)