chia-blockchain 2.5.1rc1__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- chia/__init__.py +10 -0
- chia/__main__.py +5 -0
- chia/_tests/README.md +53 -0
- chia/_tests/__init__.py +0 -0
- chia/_tests/blockchain/__init__.py +0 -0
- chia/_tests/blockchain/blockchain_test_utils.py +195 -0
- chia/_tests/blockchain/config.py +4 -0
- chia/_tests/blockchain/test_augmented_chain.py +145 -0
- chia/_tests/blockchain/test_blockchain.py +4202 -0
- chia/_tests/blockchain/test_blockchain_transactions.py +1031 -0
- chia/_tests/blockchain/test_build_chains.py +59 -0
- chia/_tests/blockchain/test_get_block_generator.py +72 -0
- chia/_tests/blockchain/test_lookup_fork_chain.py +194 -0
- chia/_tests/build-init-files.py +92 -0
- chia/_tests/build-job-matrix.py +204 -0
- chia/_tests/check_pytest_monitor_output.py +34 -0
- chia/_tests/check_sql_statements.py +72 -0
- chia/_tests/chia-start-sim +42 -0
- chia/_tests/clvm/__init__.py +0 -0
- chia/_tests/clvm/benchmark_costs.py +23 -0
- chia/_tests/clvm/coin_store.py +149 -0
- chia/_tests/clvm/test_chialisp_deserialization.py +101 -0
- chia/_tests/clvm/test_clvm_step.py +37 -0
- chia/_tests/clvm/test_condition_codes.py +13 -0
- chia/_tests/clvm/test_curry_and_treehash.py +55 -0
- chia/_tests/clvm/test_message_conditions.py +184 -0
- chia/_tests/clvm/test_program.py +150 -0
- chia/_tests/clvm/test_puzzle_compression.py +143 -0
- chia/_tests/clvm/test_puzzle_drivers.py +45 -0
- chia/_tests/clvm/test_puzzles.py +242 -0
- chia/_tests/clvm/test_singletons.py +540 -0
- chia/_tests/clvm/test_spend_sim.py +181 -0
- chia/_tests/cmds/__init__.py +0 -0
- chia/_tests/cmds/cmd_test_utils.py +469 -0
- chia/_tests/cmds/config.py +3 -0
- chia/_tests/cmds/conftest.py +23 -0
- chia/_tests/cmds/test_click_types.py +200 -0
- chia/_tests/cmds/test_cmd_framework.py +620 -0
- chia/_tests/cmds/test_cmds_util.py +97 -0
- chia/_tests/cmds/test_daemon.py +92 -0
- chia/_tests/cmds/test_dev_gh.py +131 -0
- chia/_tests/cmds/test_farm_cmd.py +66 -0
- chia/_tests/cmds/test_show.py +116 -0
- chia/_tests/cmds/test_sim.py +207 -0
- chia/_tests/cmds/test_timelock_args.py +75 -0
- chia/_tests/cmds/test_tx_config_args.py +154 -0
- chia/_tests/cmds/testing_classes.py +59 -0
- chia/_tests/cmds/wallet/__init__.py +0 -0
- chia/_tests/cmds/wallet/test_consts.py +47 -0
- chia/_tests/cmds/wallet/test_dao.py +565 -0
- chia/_tests/cmds/wallet/test_did.py +403 -0
- chia/_tests/cmds/wallet/test_nft.py +471 -0
- chia/_tests/cmds/wallet/test_notifications.py +124 -0
- chia/_tests/cmds/wallet/test_offer.toffer +1 -0
- chia/_tests/cmds/wallet/test_tx_decorators.py +27 -0
- chia/_tests/cmds/wallet/test_vcs.py +400 -0
- chia/_tests/cmds/wallet/test_wallet.py +1125 -0
- chia/_tests/cmds/wallet/test_wallet_check.py +109 -0
- chia/_tests/conftest.py +1419 -0
- chia/_tests/connection_utils.py +125 -0
- chia/_tests/core/__init__.py +0 -0
- chia/_tests/core/cmds/__init__.py +0 -0
- chia/_tests/core/cmds/test_beta.py +382 -0
- chia/_tests/core/cmds/test_keys.py +1734 -0
- chia/_tests/core/cmds/test_wallet.py +126 -0
- chia/_tests/core/config.py +3 -0
- chia/_tests/core/consensus/__init__.py +0 -0
- chia/_tests/core/consensus/test_block_creation.py +54 -0
- chia/_tests/core/consensus/test_pot_iterations.py +117 -0
- chia/_tests/core/custom_types/__init__.py +0 -0
- chia/_tests/core/custom_types/test_coin.py +107 -0
- chia/_tests/core/custom_types/test_proof_of_space.py +144 -0
- chia/_tests/core/custom_types/test_spend_bundle.py +70 -0
- chia/_tests/core/daemon/__init__.py +0 -0
- chia/_tests/core/daemon/config.py +4 -0
- chia/_tests/core/daemon/test_daemon.py +2128 -0
- chia/_tests/core/daemon/test_daemon_register.py +109 -0
- chia/_tests/core/daemon/test_keychain_proxy.py +101 -0
- chia/_tests/core/data_layer/__init__.py +0 -0
- chia/_tests/core/data_layer/config.py +5 -0
- chia/_tests/core/data_layer/conftest.py +106 -0
- chia/_tests/core/data_layer/test_data_cli.py +56 -0
- chia/_tests/core/data_layer/test_data_layer.py +83 -0
- chia/_tests/core/data_layer/test_data_layer_util.py +218 -0
- chia/_tests/core/data_layer/test_data_rpc.py +3847 -0
- chia/_tests/core/data_layer/test_data_store.py +2424 -0
- chia/_tests/core/data_layer/test_data_store_schema.py +381 -0
- chia/_tests/core/data_layer/test_plugin.py +91 -0
- chia/_tests/core/data_layer/util.py +233 -0
- chia/_tests/core/farmer/__init__.py +0 -0
- chia/_tests/core/farmer/config.py +3 -0
- chia/_tests/core/farmer/test_farmer_api.py +103 -0
- chia/_tests/core/full_node/__init__.py +0 -0
- chia/_tests/core/full_node/config.py +4 -0
- chia/_tests/core/full_node/dos/__init__.py +0 -0
- chia/_tests/core/full_node/dos/config.py +3 -0
- chia/_tests/core/full_node/full_sync/__init__.py +0 -0
- chia/_tests/core/full_node/full_sync/config.py +4 -0
- chia/_tests/core/full_node/full_sync/test_full_sync.py +443 -0
- chia/_tests/core/full_node/ram_db.py +27 -0
- chia/_tests/core/full_node/stores/__init__.py +0 -0
- chia/_tests/core/full_node/stores/config.py +4 -0
- chia/_tests/core/full_node/stores/test_block_store.py +590 -0
- chia/_tests/core/full_node/stores/test_coin_store.py +897 -0
- chia/_tests/core/full_node/stores/test_full_node_store.py +1219 -0
- chia/_tests/core/full_node/stores/test_hint_store.py +229 -0
- chia/_tests/core/full_node/stores/test_sync_store.py +135 -0
- chia/_tests/core/full_node/test_address_manager.py +588 -0
- chia/_tests/core/full_node/test_block_height_map.py +556 -0
- chia/_tests/core/full_node/test_conditions.py +556 -0
- chia/_tests/core/full_node/test_full_node.py +2700 -0
- chia/_tests/core/full_node/test_generator_tools.py +82 -0
- chia/_tests/core/full_node/test_hint_management.py +104 -0
- chia/_tests/core/full_node/test_node_load.py +34 -0
- chia/_tests/core/full_node/test_performance.py +179 -0
- chia/_tests/core/full_node/test_subscriptions.py +492 -0
- chia/_tests/core/full_node/test_transactions.py +203 -0
- chia/_tests/core/full_node/test_tx_processing_queue.py +155 -0
- chia/_tests/core/large_block.py +2388 -0
- chia/_tests/core/make_block_generator.py +70 -0
- chia/_tests/core/mempool/__init__.py +0 -0
- chia/_tests/core/mempool/config.py +4 -0
- chia/_tests/core/mempool/test_mempool.py +3255 -0
- chia/_tests/core/mempool/test_mempool_fee_estimator.py +104 -0
- chia/_tests/core/mempool/test_mempool_fee_protocol.py +55 -0
- chia/_tests/core/mempool/test_mempool_item_queries.py +190 -0
- chia/_tests/core/mempool/test_mempool_manager.py +2084 -0
- chia/_tests/core/mempool/test_mempool_performance.py +64 -0
- chia/_tests/core/mempool/test_singleton_fast_forward.py +567 -0
- chia/_tests/core/node_height.py +28 -0
- chia/_tests/core/server/__init__.py +0 -0
- chia/_tests/core/server/config.py +3 -0
- chia/_tests/core/server/flood.py +84 -0
- chia/_tests/core/server/serve.py +135 -0
- chia/_tests/core/server/test_api_protocol.py +21 -0
- chia/_tests/core/server/test_capabilities.py +66 -0
- chia/_tests/core/server/test_dos.py +319 -0
- chia/_tests/core/server/test_event_loop.py +109 -0
- chia/_tests/core/server/test_loop.py +294 -0
- chia/_tests/core/server/test_node_discovery.py +73 -0
- chia/_tests/core/server/test_rate_limits.py +482 -0
- chia/_tests/core/server/test_server.py +226 -0
- chia/_tests/core/server/test_upnp.py +8 -0
- chia/_tests/core/services/__init__.py +0 -0
- chia/_tests/core/services/config.py +3 -0
- chia/_tests/core/services/test_services.py +188 -0
- chia/_tests/core/ssl/__init__.py +0 -0
- chia/_tests/core/ssl/config.py +3 -0
- chia/_tests/core/ssl/test_ssl.py +202 -0
- chia/_tests/core/test_coins.py +33 -0
- chia/_tests/core/test_cost_calculation.py +313 -0
- chia/_tests/core/test_crawler.py +175 -0
- chia/_tests/core/test_crawler_rpc.py +53 -0
- chia/_tests/core/test_daemon_rpc.py +24 -0
- chia/_tests/core/test_db_conversion.py +130 -0
- chia/_tests/core/test_db_validation.py +162 -0
- chia/_tests/core/test_farmer_harvester_rpc.py +505 -0
- chia/_tests/core/test_filter.py +35 -0
- chia/_tests/core/test_full_node_rpc.py +768 -0
- chia/_tests/core/test_merkle_set.py +343 -0
- chia/_tests/core/test_program.py +47 -0
- chia/_tests/core/test_rpc_util.py +86 -0
- chia/_tests/core/test_seeder.py +420 -0
- chia/_tests/core/test_setproctitle.py +13 -0
- chia/_tests/core/util/__init__.py +0 -0
- chia/_tests/core/util/config.py +4 -0
- chia/_tests/core/util/test_block_cache.py +44 -0
- chia/_tests/core/util/test_cached_bls.py +57 -0
- chia/_tests/core/util/test_config.py +337 -0
- chia/_tests/core/util/test_file_keyring_synchronization.py +105 -0
- chia/_tests/core/util/test_files.py +391 -0
- chia/_tests/core/util/test_jsonify.py +146 -0
- chia/_tests/core/util/test_keychain.py +522 -0
- chia/_tests/core/util/test_keyring_wrapper.py +491 -0
- chia/_tests/core/util/test_lockfile.py +380 -0
- chia/_tests/core/util/test_log_exceptions.py +187 -0
- chia/_tests/core/util/test_lru_cache.py +56 -0
- chia/_tests/core/util/test_significant_bits.py +40 -0
- chia/_tests/core/util/test_streamable.py +883 -0
- chia/_tests/db/__init__.py +0 -0
- chia/_tests/db/test_db_wrapper.py +566 -0
- chia/_tests/environments/__init__.py +0 -0
- chia/_tests/environments/common.py +35 -0
- chia/_tests/environments/full_node.py +47 -0
- chia/_tests/environments/wallet.py +429 -0
- chia/_tests/ether.py +19 -0
- chia/_tests/farmer_harvester/__init__.py +0 -0
- chia/_tests/farmer_harvester/config.py +3 -0
- chia/_tests/farmer_harvester/test_farmer.py +1264 -0
- chia/_tests/farmer_harvester/test_farmer_harvester.py +292 -0
- chia/_tests/farmer_harvester/test_filter_prefix_bits.py +131 -0
- chia/_tests/farmer_harvester/test_third_party_harvesters.py +528 -0
- chia/_tests/farmer_harvester/test_third_party_harvesters_data.json +29 -0
- chia/_tests/fee_estimation/__init__.py +0 -0
- chia/_tests/fee_estimation/config.py +3 -0
- chia/_tests/fee_estimation/test_fee_estimation_integration.py +262 -0
- chia/_tests/fee_estimation/test_fee_estimation_rpc.py +287 -0
- chia/_tests/fee_estimation/test_fee_estimation_unit_tests.py +144 -0
- chia/_tests/fee_estimation/test_mempoolitem_height_added.py +146 -0
- chia/_tests/generator/__init__.py +0 -0
- chia/_tests/generator/puzzles/__init__.py +0 -0
- chia/_tests/generator/puzzles/test_generator_deserialize.clsp +3 -0
- chia/_tests/generator/puzzles/test_generator_deserialize.clsp.hex +1 -0
- chia/_tests/generator/puzzles/test_multiple_generator_input_arguments.clsp +19 -0
- chia/_tests/generator/puzzles/test_multiple_generator_input_arguments.clsp.hex +1 -0
- chia/_tests/generator/test_compression.py +201 -0
- chia/_tests/generator/test_generator_types.py +44 -0
- chia/_tests/generator/test_rom.py +180 -0
- chia/_tests/plot_sync/__init__.py +0 -0
- chia/_tests/plot_sync/config.py +3 -0
- chia/_tests/plot_sync/test_delta.py +101 -0
- chia/_tests/plot_sync/test_plot_sync.py +618 -0
- chia/_tests/plot_sync/test_receiver.py +451 -0
- chia/_tests/plot_sync/test_sender.py +116 -0
- chia/_tests/plot_sync/test_sync_simulated.py +451 -0
- chia/_tests/plot_sync/util.py +68 -0
- chia/_tests/plotting/__init__.py +0 -0
- chia/_tests/plotting/config.py +3 -0
- chia/_tests/plotting/test_plot_manager.py +781 -0
- chia/_tests/plotting/util.py +12 -0
- chia/_tests/pools/__init__.py +0 -0
- chia/_tests/pools/config.py +5 -0
- chia/_tests/pools/test_pool_cli_parsing.py +128 -0
- chia/_tests/pools/test_pool_cmdline.py +1001 -0
- chia/_tests/pools/test_pool_config.py +42 -0
- chia/_tests/pools/test_pool_puzzles_lifecycle.py +397 -0
- chia/_tests/pools/test_pool_rpc.py +1123 -0
- chia/_tests/pools/test_pool_wallet.py +205 -0
- chia/_tests/pools/test_wallet_pool_store.py +161 -0
- chia/_tests/process_junit.py +348 -0
- chia/_tests/rpc/__init__.py +0 -0
- chia/_tests/rpc/test_rpc_client.py +138 -0
- chia/_tests/rpc/test_rpc_server.py +183 -0
- chia/_tests/simulation/__init__.py +0 -0
- chia/_tests/simulation/config.py +6 -0
- chia/_tests/simulation/test_simulation.py +501 -0
- chia/_tests/simulation/test_simulator.py +232 -0
- chia/_tests/simulation/test_start_simulator.py +107 -0
- chia/_tests/testconfig.py +13 -0
- chia/_tests/timelord/__init__.py +0 -0
- chia/_tests/timelord/config.py +3 -0
- chia/_tests/timelord/test_new_peak.py +437 -0
- chia/_tests/timelord/test_timelord.py +11 -0
- chia/_tests/tools/1315537.json +170 -0
- chia/_tests/tools/1315544.json +160 -0
- chia/_tests/tools/1315630.json +150 -0
- chia/_tests/tools/300000.json +105 -0
- chia/_tests/tools/442734.json +140 -0
- chia/_tests/tools/466212.json +130 -0
- chia/_tests/tools/__init__.py +0 -0
- chia/_tests/tools/config.py +5 -0
- chia/_tests/tools/test-blockchain-db.sqlite +0 -0
- chia/_tests/tools/test_full_sync.py +30 -0
- chia/_tests/tools/test_legacy_keyring.py +82 -0
- chia/_tests/tools/test_run_block.py +128 -0
- chia/_tests/tools/test_virtual_project.py +591 -0
- chia/_tests/util/__init__.py +0 -0
- chia/_tests/util/benchmark_cost.py +170 -0
- chia/_tests/util/benchmarks.py +153 -0
- chia/_tests/util/bip39_test_vectors.json +148 -0
- chia/_tests/util/blockchain.py +134 -0
- chia/_tests/util/blockchain_mock.py +132 -0
- chia/_tests/util/build_network_protocol_files.py +302 -0
- chia/_tests/util/clvm_generator.bin +0 -0
- chia/_tests/util/config.py +3 -0
- chia/_tests/util/constants.py +20 -0
- chia/_tests/util/db_connection.py +37 -0
- chia/_tests/util/full_sync.py +253 -0
- chia/_tests/util/gen_ssl_certs.py +114 -0
- chia/_tests/util/generator_tools_testing.py +45 -0
- chia/_tests/util/get_name_puzzle_conditions.py +52 -0
- chia/_tests/util/key_tool.py +36 -0
- chia/_tests/util/misc.py +675 -0
- chia/_tests/util/network_protocol_data.py +1072 -0
- chia/_tests/util/protocol_messages_bytes-v1.0 +0 -0
- chia/_tests/util/protocol_messages_json.py +2701 -0
- chia/_tests/util/rpc.py +26 -0
- chia/_tests/util/run_block.py +163 -0
- chia/_tests/util/setup_nodes.py +481 -0
- chia/_tests/util/spend_sim.py +492 -0
- chia/_tests/util/split_managers.py +102 -0
- chia/_tests/util/temp_file.py +14 -0
- chia/_tests/util/test_action_scope.py +144 -0
- chia/_tests/util/test_async_pool.py +366 -0
- chia/_tests/util/test_build_job_matrix.py +42 -0
- chia/_tests/util/test_build_network_protocol_files.py +7 -0
- chia/_tests/util/test_chia_version.py +50 -0
- chia/_tests/util/test_collection.py +11 -0
- chia/_tests/util/test_condition_tools.py +229 -0
- chia/_tests/util/test_config.py +426 -0
- chia/_tests/util/test_dump_keyring.py +60 -0
- chia/_tests/util/test_errors.py +10 -0
- chia/_tests/util/test_full_block_utils.py +279 -0
- chia/_tests/util/test_installed.py +20 -0
- chia/_tests/util/test_limited_semaphore.py +53 -0
- chia/_tests/util/test_logging_filter.py +42 -0
- chia/_tests/util/test_misc.py +445 -0
- chia/_tests/util/test_network.py +73 -0
- chia/_tests/util/test_network_protocol_files.py +578 -0
- chia/_tests/util/test_network_protocol_json.py +267 -0
- chia/_tests/util/test_network_protocol_test.py +256 -0
- chia/_tests/util/test_paginator.py +71 -0
- chia/_tests/util/test_pprint.py +17 -0
- chia/_tests/util/test_priority_mutex.py +488 -0
- chia/_tests/util/test_recursive_replace.py +116 -0
- chia/_tests/util/test_replace_str_to_bytes.py +137 -0
- chia/_tests/util/test_service_groups.py +15 -0
- chia/_tests/util/test_ssl_check.py +31 -0
- chia/_tests/util/test_testnet_overrides.py +19 -0
- chia/_tests/util/test_tests_misc.py +38 -0
- chia/_tests/util/test_timing.py +37 -0
- chia/_tests/util/test_trusted_peer.py +51 -0
- chia/_tests/util/time_out_assert.py +191 -0
- chia/_tests/wallet/__init__.py +0 -0
- chia/_tests/wallet/cat_wallet/__init__.py +0 -0
- chia/_tests/wallet/cat_wallet/config.py +4 -0
- chia/_tests/wallet/cat_wallet/test_cat_lifecycle.py +468 -0
- chia/_tests/wallet/cat_wallet/test_cat_outer_puzzle.py +69 -0
- chia/_tests/wallet/cat_wallet/test_cat_wallet.py +1826 -0
- chia/_tests/wallet/cat_wallet/test_offer_lifecycle.py +291 -0
- chia/_tests/wallet/cat_wallet/test_trades.py +2600 -0
- chia/_tests/wallet/clawback/__init__.py +0 -0
- chia/_tests/wallet/clawback/config.py +3 -0
- chia/_tests/wallet/clawback/test_clawback_decorator.py +78 -0
- chia/_tests/wallet/clawback/test_clawback_lifecycle.py +292 -0
- chia/_tests/wallet/clawback/test_clawback_metadata.py +50 -0
- chia/_tests/wallet/config.py +4 -0
- chia/_tests/wallet/conftest.py +278 -0
- chia/_tests/wallet/dao_wallet/__init__.py +0 -0
- chia/_tests/wallet/dao_wallet/config.py +3 -0
- chia/_tests/wallet/dao_wallet/test_dao_clvm.py +1330 -0
- chia/_tests/wallet/dao_wallet/test_dao_wallets.py +3488 -0
- chia/_tests/wallet/db_wallet/__init__.py +0 -0
- chia/_tests/wallet/db_wallet/config.py +3 -0
- chia/_tests/wallet/db_wallet/test_db_graftroot.py +141 -0
- chia/_tests/wallet/db_wallet/test_dl_offers.py +491 -0
- chia/_tests/wallet/db_wallet/test_dl_wallet.py +823 -0
- chia/_tests/wallet/did_wallet/__init__.py +0 -0
- chia/_tests/wallet/did_wallet/config.py +4 -0
- chia/_tests/wallet/did_wallet/test_did.py +2284 -0
- chia/_tests/wallet/nft_wallet/__init__.py +0 -0
- chia/_tests/wallet/nft_wallet/config.py +4 -0
- chia/_tests/wallet/nft_wallet/test_nft_1_offers.py +1493 -0
- chia/_tests/wallet/nft_wallet/test_nft_bulk_mint.py +1024 -0
- chia/_tests/wallet/nft_wallet/test_nft_lifecycle.py +375 -0
- chia/_tests/wallet/nft_wallet/test_nft_offers.py +1209 -0
- chia/_tests/wallet/nft_wallet/test_nft_puzzles.py +172 -0
- chia/_tests/wallet/nft_wallet/test_nft_wallet.py +2584 -0
- chia/_tests/wallet/nft_wallet/test_ownership_outer_puzzle.py +70 -0
- chia/_tests/wallet/rpc/__init__.py +0 -0
- chia/_tests/wallet/rpc/config.py +4 -0
- chia/_tests/wallet/rpc/test_dl_wallet_rpc.py +285 -0
- chia/_tests/wallet/rpc/test_wallet_rpc.py +3153 -0
- chia/_tests/wallet/simple_sync/__init__.py +0 -0
- chia/_tests/wallet/simple_sync/config.py +3 -0
- chia/_tests/wallet/simple_sync/test_simple_sync_protocol.py +718 -0
- chia/_tests/wallet/sync/__init__.py +0 -0
- chia/_tests/wallet/sync/config.py +4 -0
- chia/_tests/wallet/sync/test_wallet_sync.py +1692 -0
- chia/_tests/wallet/test_address_type.py +189 -0
- chia/_tests/wallet/test_bech32m.py +45 -0
- chia/_tests/wallet/test_clvm_streamable.py +244 -0
- chia/_tests/wallet/test_coin_management.py +354 -0
- chia/_tests/wallet/test_coin_selection.py +588 -0
- chia/_tests/wallet/test_conditions.py +400 -0
- chia/_tests/wallet/test_debug_spend_bundle.py +218 -0
- chia/_tests/wallet/test_new_wallet_protocol.py +1174 -0
- chia/_tests/wallet/test_nft_store.py +192 -0
- chia/_tests/wallet/test_notifications.py +196 -0
- chia/_tests/wallet/test_offer_parsing_performance.py +48 -0
- chia/_tests/wallet/test_puzzle_store.py +132 -0
- chia/_tests/wallet/test_sign_coin_spends.py +159 -0
- chia/_tests/wallet/test_signer_protocol.py +947 -0
- chia/_tests/wallet/test_singleton.py +122 -0
- chia/_tests/wallet/test_singleton_lifecycle_fast.py +772 -0
- chia/_tests/wallet/test_singleton_store.py +152 -0
- chia/_tests/wallet/test_taproot.py +19 -0
- chia/_tests/wallet/test_transaction_store.py +945 -0
- chia/_tests/wallet/test_util.py +185 -0
- chia/_tests/wallet/test_wallet.py +2139 -0
- chia/_tests/wallet/test_wallet_action_scope.py +85 -0
- chia/_tests/wallet/test_wallet_blockchain.py +111 -0
- chia/_tests/wallet/test_wallet_coin_store.py +1002 -0
- chia/_tests/wallet/test_wallet_interested_store.py +43 -0
- chia/_tests/wallet/test_wallet_key_val_store.py +40 -0
- chia/_tests/wallet/test_wallet_node.py +780 -0
- chia/_tests/wallet/test_wallet_retry.py +95 -0
- chia/_tests/wallet/test_wallet_state_manager.py +259 -0
- chia/_tests/wallet/test_wallet_test_framework.py +275 -0
- chia/_tests/wallet/test_wallet_trade_store.py +218 -0
- chia/_tests/wallet/test_wallet_user_store.py +34 -0
- chia/_tests/wallet/test_wallet_utils.py +156 -0
- chia/_tests/wallet/vc_wallet/__init__.py +0 -0
- chia/_tests/wallet/vc_wallet/config.py +3 -0
- chia/_tests/wallet/vc_wallet/test_cr_outer_puzzle.py +70 -0
- chia/_tests/wallet/vc_wallet/test_vc_lifecycle.py +883 -0
- chia/_tests/wallet/vc_wallet/test_vc_wallet.py +830 -0
- chia/_tests/wallet/wallet_block_tools.py +327 -0
- chia/_tests/weight_proof/__init__.py +0 -0
- chia/_tests/weight_proof/config.py +3 -0
- chia/_tests/weight_proof/test_weight_proof.py +528 -0
- chia/apis.py +19 -0
- chia/clvm/__init__.py +0 -0
- chia/cmds/__init__.py +0 -0
- chia/cmds/beta.py +184 -0
- chia/cmds/beta_funcs.py +137 -0
- chia/cmds/check_wallet_db.py +420 -0
- chia/cmds/chia.py +151 -0
- chia/cmds/cmd_classes.py +323 -0
- chia/cmds/cmd_helpers.py +242 -0
- chia/cmds/cmds_util.py +488 -0
- chia/cmds/coin_funcs.py +275 -0
- chia/cmds/coins.py +182 -0
- chia/cmds/completion.py +49 -0
- chia/cmds/configure.py +332 -0
- chia/cmds/dao.py +1064 -0
- chia/cmds/dao_funcs.py +598 -0
- chia/cmds/data.py +708 -0
- chia/cmds/data_funcs.py +385 -0
- chia/cmds/db.py +87 -0
- chia/cmds/db_backup_func.py +77 -0
- chia/cmds/db_upgrade_func.py +452 -0
- chia/cmds/db_validate_func.py +184 -0
- chia/cmds/dev.py +18 -0
- chia/cmds/farm.py +100 -0
- chia/cmds/farm_funcs.py +200 -0
- chia/cmds/gh.py +275 -0
- chia/cmds/init.py +63 -0
- chia/cmds/init_funcs.py +367 -0
- chia/cmds/installers.py +131 -0
- chia/cmds/keys.py +527 -0
- chia/cmds/keys_funcs.py +863 -0
- chia/cmds/netspace.py +50 -0
- chia/cmds/netspace_funcs.py +54 -0
- chia/cmds/options.py +32 -0
- chia/cmds/param_types.py +238 -0
- chia/cmds/passphrase.py +131 -0
- chia/cmds/passphrase_funcs.py +292 -0
- chia/cmds/peer.py +51 -0
- chia/cmds/peer_funcs.py +129 -0
- chia/cmds/plotnft.py +260 -0
- chia/cmds/plotnft_funcs.py +405 -0
- chia/cmds/plots.py +230 -0
- chia/cmds/plotters.py +18 -0
- chia/cmds/rpc.py +208 -0
- chia/cmds/show.py +72 -0
- chia/cmds/show_funcs.py +215 -0
- chia/cmds/signer.py +296 -0
- chia/cmds/sim.py +225 -0
- chia/cmds/sim_funcs.py +509 -0
- chia/cmds/start.py +24 -0
- chia/cmds/start_funcs.py +109 -0
- chia/cmds/stop.py +62 -0
- chia/cmds/units.py +9 -0
- chia/cmds/wallet.py +1901 -0
- chia/cmds/wallet_funcs.py +1874 -0
- chia/consensus/__init__.py +0 -0
- chia/consensus/block_body_validation.py +562 -0
- chia/consensus/block_creation.py +546 -0
- chia/consensus/block_header_validation.py +1059 -0
- chia/consensus/block_record.py +31 -0
- chia/consensus/block_rewards.py +53 -0
- chia/consensus/blockchain.py +1087 -0
- chia/consensus/blockchain_interface.py +56 -0
- chia/consensus/coinbase.py +30 -0
- chia/consensus/condition_costs.py +9 -0
- chia/consensus/constants.py +49 -0
- chia/consensus/cost_calculator.py +15 -0
- chia/consensus/default_constants.py +89 -0
- chia/consensus/deficit.py +55 -0
- chia/consensus/difficulty_adjustment.py +412 -0
- chia/consensus/find_fork_point.py +111 -0
- chia/consensus/full_block_to_block_record.py +167 -0
- chia/consensus/get_block_challenge.py +106 -0
- chia/consensus/get_block_generator.py +27 -0
- chia/consensus/make_sub_epoch_summary.py +210 -0
- chia/consensus/multiprocess_validation.py +268 -0
- chia/consensus/pos_quality.py +19 -0
- chia/consensus/pot_iterations.py +67 -0
- chia/consensus/puzzles/__init__.py +0 -0
- chia/consensus/puzzles/chialisp_deserialisation.clsp +69 -0
- chia/consensus/puzzles/chialisp_deserialisation.clsp.hex +1 -0
- chia/consensus/puzzles/rom_bootstrap_generator.clsp +37 -0
- chia/consensus/puzzles/rom_bootstrap_generator.clsp.hex +1 -0
- chia/consensus/vdf_info_computation.py +156 -0
- chia/daemon/__init__.py +0 -0
- chia/daemon/client.py +252 -0
- chia/daemon/keychain_proxy.py +502 -0
- chia/daemon/keychain_server.py +365 -0
- chia/daemon/server.py +1606 -0
- chia/daemon/windows_signal.py +56 -0
- chia/data_layer/__init__.py +0 -0
- chia/data_layer/data_layer.py +1291 -0
- chia/data_layer/data_layer_api.py +33 -0
- chia/data_layer/data_layer_errors.py +50 -0
- chia/data_layer/data_layer_server.py +170 -0
- chia/data_layer/data_layer_util.py +985 -0
- chia/data_layer/data_layer_wallet.py +1311 -0
- chia/data_layer/data_store.py +2267 -0
- chia/data_layer/dl_wallet_store.py +407 -0
- chia/data_layer/download_data.py +389 -0
- chia/data_layer/puzzles/__init__.py +0 -0
- chia/data_layer/puzzles/graftroot_dl_offers.clsp +100 -0
- chia/data_layer/puzzles/graftroot_dl_offers.clsp.hex +1 -0
- chia/data_layer/s3_plugin_config.yml +33 -0
- chia/data_layer/s3_plugin_service.py +468 -0
- chia/data_layer/util/__init__.py +0 -0
- chia/data_layer/util/benchmark.py +107 -0
- chia/data_layer/util/plugin.py +40 -0
- chia/farmer/__init__.py +0 -0
- chia/farmer/farmer.py +923 -0
- chia/farmer/farmer_api.py +820 -0
- chia/full_node/__init__.py +0 -0
- chia/full_node/bitcoin_fee_estimator.py +85 -0
- chia/full_node/block_height_map.py +271 -0
- chia/full_node/block_store.py +576 -0
- chia/full_node/bundle_tools.py +19 -0
- chia/full_node/coin_store.py +647 -0
- chia/full_node/fee_estimate.py +54 -0
- chia/full_node/fee_estimate_store.py +24 -0
- chia/full_node/fee_estimation.py +92 -0
- chia/full_node/fee_estimator.py +90 -0
- chia/full_node/fee_estimator_constants.py +38 -0
- chia/full_node/fee_estimator_interface.py +42 -0
- chia/full_node/fee_history.py +25 -0
- chia/full_node/fee_tracker.py +564 -0
- chia/full_node/full_node.py +3327 -0
- chia/full_node/full_node_api.py +2025 -0
- chia/full_node/full_node_store.py +1033 -0
- chia/full_node/hint_management.py +56 -0
- chia/full_node/hint_store.py +93 -0
- chia/full_node/mempool.py +589 -0
- chia/full_node/mempool_check_conditions.py +146 -0
- chia/full_node/mempool_manager.py +853 -0
- chia/full_node/pending_tx_cache.py +112 -0
- chia/full_node/puzzles/__init__.py +0 -0
- chia/full_node/puzzles/block_program_zero.clsp +14 -0
- chia/full_node/puzzles/block_program_zero.clsp.hex +1 -0
- chia/full_node/puzzles/decompress_coin_spend_entry.clsp +5 -0
- chia/full_node/puzzles/decompress_coin_spend_entry.clsp.hex +1 -0
- chia/full_node/puzzles/decompress_coin_spend_entry_with_prefix.clsp +7 -0
- chia/full_node/puzzles/decompress_coin_spend_entry_with_prefix.clsp.hex +1 -0
- chia/full_node/puzzles/decompress_puzzle.clsp +6 -0
- chia/full_node/puzzles/decompress_puzzle.clsp.hex +1 -0
- chia/full_node/signage_point.py +16 -0
- chia/full_node/subscriptions.py +247 -0
- chia/full_node/sync_store.py +146 -0
- chia/full_node/tx_processing_queue.py +78 -0
- chia/full_node/util/__init__.py +0 -0
- chia/full_node/weight_proof.py +1720 -0
- chia/harvester/__init__.py +0 -0
- chia/harvester/harvester.py +272 -0
- chia/harvester/harvester_api.py +380 -0
- chia/introducer/__init__.py +0 -0
- chia/introducer/introducer.py +122 -0
- chia/introducer/introducer_api.py +70 -0
- chia/legacy/__init__.py +0 -0
- chia/legacy/keyring.py +155 -0
- chia/plot_sync/__init__.py +0 -0
- chia/plot_sync/delta.py +61 -0
- chia/plot_sync/exceptions.py +56 -0
- chia/plot_sync/receiver.py +386 -0
- chia/plot_sync/sender.py +340 -0
- chia/plot_sync/util.py +43 -0
- chia/plotters/__init__.py +0 -0
- chia/plotters/bladebit.py +388 -0
- chia/plotters/chiapos.py +63 -0
- chia/plotters/madmax.py +224 -0
- chia/plotters/plotters.py +577 -0
- chia/plotters/plotters_util.py +133 -0
- chia/plotting/__init__.py +0 -0
- chia/plotting/cache.py +213 -0
- chia/plotting/check_plots.py +283 -0
- chia/plotting/create_plots.py +278 -0
- chia/plotting/manager.py +436 -0
- chia/plotting/util.py +336 -0
- chia/pools/__init__.py +0 -0
- chia/pools/pool_config.py +110 -0
- chia/pools/pool_puzzles.py +459 -0
- chia/pools/pool_wallet.py +933 -0
- chia/pools/pool_wallet_info.py +118 -0
- chia/pools/puzzles/__init__.py +0 -0
- chia/pools/puzzles/pool_member_innerpuz.clsp +70 -0
- chia/pools/puzzles/pool_member_innerpuz.clsp.hex +1 -0
- chia/pools/puzzles/pool_waitingroom_innerpuz.clsp +69 -0
- chia/pools/puzzles/pool_waitingroom_innerpuz.clsp.hex +1 -0
- chia/protocols/__init__.py +0 -0
- chia/protocols/farmer_protocol.py +102 -0
- chia/protocols/full_node_protocol.py +219 -0
- chia/protocols/harvester_protocol.py +216 -0
- chia/protocols/introducer_protocol.py +25 -0
- chia/protocols/pool_protocol.py +177 -0
- chia/protocols/protocol_message_types.py +139 -0
- chia/protocols/protocol_state_machine.py +87 -0
- chia/protocols/protocol_timing.py +8 -0
- chia/protocols/shared_protocol.py +86 -0
- chia/protocols/timelord_protocol.py +93 -0
- chia/protocols/wallet_protocol.py +401 -0
- chia/py.typed +0 -0
- chia/rpc/__init__.py +0 -0
- chia/rpc/crawler_rpc_api.py +80 -0
- chia/rpc/data_layer_rpc_api.py +644 -0
- chia/rpc/data_layer_rpc_client.py +188 -0
- chia/rpc/data_layer_rpc_util.py +58 -0
- chia/rpc/farmer_rpc_api.py +365 -0
- chia/rpc/farmer_rpc_client.py +86 -0
- chia/rpc/full_node_rpc_api.py +959 -0
- chia/rpc/full_node_rpc_client.py +292 -0
- chia/rpc/harvester_rpc_api.py +141 -0
- chia/rpc/harvester_rpc_client.py +54 -0
- chia/rpc/rpc_client.py +164 -0
- chia/rpc/rpc_server.py +521 -0
- chia/rpc/timelord_rpc_api.py +32 -0
- chia/rpc/util.py +93 -0
- chia/rpc/wallet_request_types.py +904 -0
- chia/rpc/wallet_rpc_api.py +4943 -0
- chia/rpc/wallet_rpc_client.py +1814 -0
- chia/seeder/__init__.py +0 -0
- chia/seeder/crawl_store.py +425 -0
- chia/seeder/crawler.py +410 -0
- chia/seeder/crawler_api.py +135 -0
- chia/seeder/dns_server.py +593 -0
- chia/seeder/peer_record.py +146 -0
- chia/seeder/start_crawler.py +92 -0
- chia/server/__init__.py +0 -0
- chia/server/address_manager.py +658 -0
- chia/server/address_manager_store.py +237 -0
- chia/server/api_protocol.py +116 -0
- chia/server/capabilities.py +24 -0
- chia/server/chia_policy.py +346 -0
- chia/server/introducer_peers.py +76 -0
- chia/server/node_discovery.py +714 -0
- chia/server/outbound_message.py +33 -0
- chia/server/rate_limit_numbers.py +214 -0
- chia/server/rate_limits.py +153 -0
- chia/server/server.py +741 -0
- chia/server/signal_handlers.py +120 -0
- chia/server/ssl_context.py +32 -0
- chia/server/start_data_layer.py +151 -0
- chia/server/start_farmer.py +98 -0
- chia/server/start_full_node.py +112 -0
- chia/server/start_harvester.py +93 -0
- chia/server/start_introducer.py +81 -0
- chia/server/start_service.py +316 -0
- chia/server/start_timelord.py +89 -0
- chia/server/start_wallet.py +113 -0
- chia/server/upnp.py +118 -0
- chia/server/ws_connection.py +766 -0
- chia/simulator/__init__.py +0 -0
- chia/simulator/add_blocks_in_batches.py +54 -0
- chia/simulator/block_tools.py +2054 -0
- chia/simulator/full_node_simulator.py +794 -0
- chia/simulator/keyring.py +128 -0
- chia/simulator/setup_services.py +506 -0
- chia/simulator/simulator_constants.py +13 -0
- chia/simulator/simulator_full_node_rpc_api.py +99 -0
- chia/simulator/simulator_full_node_rpc_client.py +60 -0
- chia/simulator/simulator_protocol.py +29 -0
- chia/simulator/simulator_test_tools.py +164 -0
- chia/simulator/socket.py +24 -0
- chia/simulator/ssl_certs.py +114 -0
- chia/simulator/ssl_certs_1.py +697 -0
- chia/simulator/ssl_certs_10.py +697 -0
- chia/simulator/ssl_certs_2.py +697 -0
- chia/simulator/ssl_certs_3.py +697 -0
- chia/simulator/ssl_certs_4.py +697 -0
- chia/simulator/ssl_certs_5.py +697 -0
- chia/simulator/ssl_certs_6.py +697 -0
- chia/simulator/ssl_certs_7.py +697 -0
- chia/simulator/ssl_certs_8.py +697 -0
- chia/simulator/ssl_certs_9.py +697 -0
- chia/simulator/start_simulator.py +143 -0
- chia/simulator/wallet_tools.py +246 -0
- chia/ssl/__init__.py +0 -0
- chia/ssl/chia_ca.crt +19 -0
- chia/ssl/chia_ca.key +28 -0
- chia/ssl/create_ssl.py +249 -0
- chia/ssl/dst_root_ca.pem +20 -0
- chia/timelord/__init__.py +0 -0
- chia/timelord/iters_from_block.py +50 -0
- chia/timelord/timelord.py +1226 -0
- chia/timelord/timelord_api.py +138 -0
- chia/timelord/timelord_launcher.py +190 -0
- chia/timelord/timelord_state.py +244 -0
- chia/timelord/types.py +22 -0
- chia/types/__init__.py +0 -0
- chia/types/aliases.py +35 -0
- chia/types/block_protocol.py +20 -0
- chia/types/blockchain_format/__init__.py +0 -0
- chia/types/blockchain_format/classgroup.py +5 -0
- chia/types/blockchain_format/coin.py +28 -0
- chia/types/blockchain_format/foliage.py +8 -0
- chia/types/blockchain_format/pool_target.py +5 -0
- chia/types/blockchain_format/program.py +269 -0
- chia/types/blockchain_format/proof_of_space.py +135 -0
- chia/types/blockchain_format/reward_chain_block.py +6 -0
- chia/types/blockchain_format/serialized_program.py +5 -0
- chia/types/blockchain_format/sized_bytes.py +11 -0
- chia/types/blockchain_format/slots.py +9 -0
- chia/types/blockchain_format/sub_epoch_summary.py +5 -0
- chia/types/blockchain_format/tree_hash.py +72 -0
- chia/types/blockchain_format/vdf.py +86 -0
- chia/types/clvm_cost.py +13 -0
- chia/types/coin_record.py +43 -0
- chia/types/coin_spend.py +115 -0
- chia/types/condition_opcodes.py +73 -0
- chia/types/condition_with_args.py +16 -0
- chia/types/eligible_coin_spends.py +365 -0
- chia/types/end_of_slot_bundle.py +5 -0
- chia/types/fee_rate.py +38 -0
- chia/types/full_block.py +5 -0
- chia/types/generator_types.py +13 -0
- chia/types/header_block.py +5 -0
- chia/types/internal_mempool_item.py +18 -0
- chia/types/mempool_inclusion_status.py +9 -0
- chia/types/mempool_item.py +85 -0
- chia/types/mempool_submission_status.py +30 -0
- chia/types/mojos.py +7 -0
- chia/types/peer_info.py +64 -0
- chia/types/signing_mode.py +29 -0
- chia/types/spend_bundle.py +30 -0
- chia/types/spend_bundle_conditions.py +7 -0
- chia/types/transaction_queue_entry.py +55 -0
- chia/types/unfinished_block.py +5 -0
- chia/types/unfinished_header_block.py +37 -0
- chia/types/validation_state.py +14 -0
- chia/types/weight_proof.py +49 -0
- chia/util/__init__.py +0 -0
- chia/util/action_scope.py +168 -0
- chia/util/async_pool.py +226 -0
- chia/util/augmented_chain.py +134 -0
- chia/util/batches.py +42 -0
- chia/util/bech32m.py +126 -0
- chia/util/beta_metrics.py +119 -0
- chia/util/block_cache.py +56 -0
- chia/util/byte_types.py +12 -0
- chia/util/check_fork_next_block.py +33 -0
- chia/util/chia_logging.py +144 -0
- chia/util/chia_version.py +33 -0
- chia/util/collection.py +17 -0
- chia/util/condition_tools.py +201 -0
- chia/util/config.py +367 -0
- chia/util/cpu.py +22 -0
- chia/util/db_synchronous.py +23 -0
- chia/util/db_version.py +32 -0
- chia/util/db_wrapper.py +430 -0
- chia/util/default_root.py +27 -0
- chia/util/dump_keyring.py +93 -0
- chia/util/english.txt +2048 -0
- chia/util/errors.py +353 -0
- chia/util/file_keyring.py +469 -0
- chia/util/files.py +97 -0
- chia/util/full_block_utils.py +345 -0
- chia/util/generator_tools.py +72 -0
- chia/util/hash.py +31 -0
- chia/util/initial-config.yaml +694 -0
- chia/util/inline_executor.py +26 -0
- chia/util/ints.py +19 -0
- chia/util/ip_address.py +39 -0
- chia/util/json_util.py +37 -0
- chia/util/keychain.py +676 -0
- chia/util/keyring_wrapper.py +327 -0
- chia/util/limited_semaphore.py +41 -0
- chia/util/lock.py +49 -0
- chia/util/log_exceptions.py +32 -0
- chia/util/logging.py +36 -0
- chia/util/lru_cache.py +31 -0
- chia/util/math.py +20 -0
- chia/util/network.py +182 -0
- chia/util/paginator.py +48 -0
- chia/util/path.py +31 -0
- chia/util/permissions.py +20 -0
- chia/util/prev_transaction_block.py +21 -0
- chia/util/priority_mutex.py +95 -0
- chia/util/profiler.py +197 -0
- chia/util/recursive_replace.py +24 -0
- chia/util/safe_cancel_task.py +16 -0
- chia/util/service_groups.py +47 -0
- chia/util/setproctitle.py +22 -0
- chia/util/significant_bits.py +32 -0
- chia/util/ssl_check.py +213 -0
- chia/util/streamable.py +642 -0
- chia/util/task_referencer.py +59 -0
- chia/util/task_timing.py +382 -0
- chia/util/timing.py +67 -0
- chia/util/vdf_prover.py +30 -0
- chia/util/virtual_project_analysis.py +540 -0
- chia/util/ws_message.py +66 -0
- chia/wallet/__init__.py +0 -0
- chia/wallet/cat_wallet/__init__.py +0 -0
- chia/wallet/cat_wallet/cat_constants.py +75 -0
- chia/wallet/cat_wallet/cat_info.py +47 -0
- chia/wallet/cat_wallet/cat_outer_puzzle.py +120 -0
- chia/wallet/cat_wallet/cat_utils.py +164 -0
- chia/wallet/cat_wallet/cat_wallet.py +855 -0
- chia/wallet/cat_wallet/dao_cat_info.py +28 -0
- chia/wallet/cat_wallet/dao_cat_wallet.py +669 -0
- chia/wallet/cat_wallet/lineage_store.py +74 -0
- chia/wallet/cat_wallet/puzzles/__init__.py +0 -0
- chia/wallet/cat_wallet/puzzles/cat_truths.clib +31 -0
- chia/wallet/cat_wallet/puzzles/cat_v2.clsp +397 -0
- chia/wallet/cat_wallet/puzzles/cat_v2.clsp.hex +1 -0
- chia/wallet/cat_wallet/puzzles/delegated_tail.clsp +25 -0
- chia/wallet/cat_wallet/puzzles/delegated_tail.clsp.hex +1 -0
- chia/wallet/cat_wallet/puzzles/everything_with_signature.clsp +15 -0
- chia/wallet/cat_wallet/puzzles/everything_with_signature.clsp.hex +1 -0
- chia/wallet/cat_wallet/puzzles/genesis_by_coin_id.clsp +26 -0
- chia/wallet/cat_wallet/puzzles/genesis_by_coin_id.clsp.hex +1 -0
- chia/wallet/cat_wallet/puzzles/genesis_by_coin_id_or_singleton.clsp +42 -0
- chia/wallet/cat_wallet/puzzles/genesis_by_coin_id_or_singleton.clsp.hex +1 -0
- chia/wallet/cat_wallet/puzzles/genesis_by_puzzle_hash.clsp +24 -0
- chia/wallet/cat_wallet/puzzles/genesis_by_puzzle_hash.clsp.hex +1 -0
- chia/wallet/coin_selection.py +188 -0
- chia/wallet/conditions.py +1512 -0
- chia/wallet/dao_wallet/__init__.py +0 -0
- chia/wallet/dao_wallet/dao_info.py +61 -0
- chia/wallet/dao_wallet/dao_utils.py +811 -0
- chia/wallet/dao_wallet/dao_wallet.py +2119 -0
- chia/wallet/db_wallet/__init__.py +0 -0
- chia/wallet/db_wallet/db_wallet_puzzles.py +111 -0
- chia/wallet/derivation_record.py +30 -0
- chia/wallet/derive_keys.py +146 -0
- chia/wallet/did_wallet/__init__.py +0 -0
- chia/wallet/did_wallet/did_info.py +39 -0
- chia/wallet/did_wallet/did_wallet.py +1494 -0
- chia/wallet/did_wallet/did_wallet_puzzles.py +221 -0
- chia/wallet/did_wallet/puzzles/__init__.py +0 -0
- chia/wallet/did_wallet/puzzles/did_innerpuz.clsp +135 -0
- chia/wallet/did_wallet/puzzles/did_innerpuz.clsp.hex +1 -0
- chia/wallet/driver_protocol.py +26 -0
- chia/wallet/key_val_store.py +55 -0
- chia/wallet/lineage_proof.py +58 -0
- chia/wallet/nft_wallet/__init__.py +0 -0
- chia/wallet/nft_wallet/metadata_outer_puzzle.py +92 -0
- chia/wallet/nft_wallet/nft_info.py +120 -0
- chia/wallet/nft_wallet/nft_puzzles.py +305 -0
- chia/wallet/nft_wallet/nft_wallet.py +1687 -0
- chia/wallet/nft_wallet/ownership_outer_puzzle.py +101 -0
- chia/wallet/nft_wallet/puzzles/__init__.py +0 -0
- chia/wallet/nft_wallet/puzzles/create_nft_launcher_from_did.clsp +6 -0
- chia/wallet/nft_wallet/puzzles/create_nft_launcher_from_did.clsp.hex +1 -0
- chia/wallet/nft_wallet/puzzles/nft_intermediate_launcher.clsp +6 -0
- chia/wallet/nft_wallet/puzzles/nft_intermediate_launcher.clsp.hex +1 -0
- chia/wallet/nft_wallet/puzzles/nft_metadata_updater_default.clsp +30 -0
- chia/wallet/nft_wallet/puzzles/nft_metadata_updater_default.clsp.hex +1 -0
- chia/wallet/nft_wallet/puzzles/nft_metadata_updater_updateable.clsp +28 -0
- chia/wallet/nft_wallet/puzzles/nft_metadata_updater_updateable.clsp.hex +1 -0
- chia/wallet/nft_wallet/puzzles/nft_ownership_layer.clsp +100 -0
- chia/wallet/nft_wallet/puzzles/nft_ownership_layer.clsp.hex +1 -0
- chia/wallet/nft_wallet/puzzles/nft_ownership_transfer_program_one_way_claim_with_royalties.clsp +78 -0
- chia/wallet/nft_wallet/puzzles/nft_ownership_transfer_program_one_way_claim_with_royalties.clsp.hex +1 -0
- chia/wallet/nft_wallet/puzzles/nft_state_layer.clsp +74 -0
- chia/wallet/nft_wallet/puzzles/nft_state_layer.clsp.hex +1 -0
- chia/wallet/nft_wallet/singleton_outer_puzzle.py +101 -0
- chia/wallet/nft_wallet/transfer_program_puzzle.py +82 -0
- chia/wallet/nft_wallet/uncurry_nft.py +217 -0
- chia/wallet/notification_manager.py +117 -0
- chia/wallet/notification_store.py +178 -0
- chia/wallet/outer_puzzles.py +84 -0
- chia/wallet/payment.py +33 -0
- chia/wallet/puzzle_drivers.py +118 -0
- chia/wallet/puzzles/__init__.py +0 -0
- chia/wallet/puzzles/augmented_condition.clsp +13 -0
- chia/wallet/puzzles/augmented_condition.clsp.hex +1 -0
- chia/wallet/puzzles/clawback/__init__.py +0 -0
- chia/wallet/puzzles/clawback/drivers.py +188 -0
- chia/wallet/puzzles/clawback/metadata.py +38 -0
- chia/wallet/puzzles/clawback/puzzle_decorator.py +67 -0
- chia/wallet/puzzles/condition_codes.clib +77 -0
- chia/wallet/puzzles/curry-and-treehash.clib +102 -0
- chia/wallet/puzzles/curry.clib +135 -0
- chia/wallet/puzzles/curry_by_index.clib +16 -0
- chia/wallet/puzzles/dao_cat_eve.clsp +17 -0
- chia/wallet/puzzles/dao_cat_eve.clsp.hex +1 -0
- chia/wallet/puzzles/dao_cat_launcher.clsp +36 -0
- chia/wallet/puzzles/dao_cat_launcher.clsp.hex +1 -0
- chia/wallet/puzzles/dao_finished_state.clsp +35 -0
- chia/wallet/puzzles/dao_finished_state.clsp.hex +1 -0
- chia/wallet/puzzles/dao_finished_state.clsp.hex.sha256tree +1 -0
- chia/wallet/puzzles/dao_lockup.clsp +288 -0
- chia/wallet/puzzles/dao_lockup.clsp.hex +1 -0
- chia/wallet/puzzles/dao_lockup.clsp.hex.sha256tree +1 -0
- chia/wallet/puzzles/dao_proposal.clsp +377 -0
- chia/wallet/puzzles/dao_proposal.clsp.hex +1 -0
- chia/wallet/puzzles/dao_proposal.clsp.hex.sha256tree +1 -0
- chia/wallet/puzzles/dao_proposal_timer.clsp +78 -0
- chia/wallet/puzzles/dao_proposal_timer.clsp.hex +1 -0
- chia/wallet/puzzles/dao_proposal_timer.clsp.hex.sha256tree +1 -0
- chia/wallet/puzzles/dao_proposal_validator.clsp +87 -0
- chia/wallet/puzzles/dao_proposal_validator.clsp.hex +1 -0
- chia/wallet/puzzles/dao_proposal_validator.clsp.hex.sha256tree +1 -0
- chia/wallet/puzzles/dao_spend_p2_singleton_v2.clsp +240 -0
- chia/wallet/puzzles/dao_spend_p2_singleton_v2.clsp.hex +1 -0
- chia/wallet/puzzles/dao_spend_p2_singleton_v2.clsp.hex.sha256tree +1 -0
- chia/wallet/puzzles/dao_treasury.clsp +115 -0
- chia/wallet/puzzles/dao_treasury.clsp.hex +1 -0
- chia/wallet/puzzles/dao_update_proposal.clsp +44 -0
- chia/wallet/puzzles/dao_update_proposal.clsp.hex +1 -0
- chia/wallet/puzzles/deployed_puzzle_hashes.json +67 -0
- chia/wallet/puzzles/json.clib +25 -0
- chia/wallet/puzzles/load_clvm.py +161 -0
- chia/wallet/puzzles/merkle_utils.clib +18 -0
- chia/wallet/puzzles/notification.clsp +7 -0
- chia/wallet/puzzles/notification.clsp.hex +1 -0
- chia/wallet/puzzles/p2_1_of_n.clsp +22 -0
- chia/wallet/puzzles/p2_1_of_n.clsp.hex +1 -0
- chia/wallet/puzzles/p2_conditions.clsp +3 -0
- chia/wallet/puzzles/p2_conditions.clsp.hex +1 -0
- chia/wallet/puzzles/p2_conditions.py +26 -0
- chia/wallet/puzzles/p2_delegated_conditions.clsp +18 -0
- chia/wallet/puzzles/p2_delegated_conditions.clsp.hex +1 -0
- chia/wallet/puzzles/p2_delegated_conditions.py +21 -0
- chia/wallet/puzzles/p2_delegated_puzzle.clsp +19 -0
- chia/wallet/puzzles/p2_delegated_puzzle.clsp.hex +1 -0
- chia/wallet/puzzles/p2_delegated_puzzle.py +34 -0
- chia/wallet/puzzles/p2_delegated_puzzle_or_hidden_puzzle.clsp +91 -0
- chia/wallet/puzzles/p2_delegated_puzzle_or_hidden_puzzle.clsp.hex +1 -0
- chia/wallet/puzzles/p2_delegated_puzzle_or_hidden_puzzle.py +160 -0
- chia/wallet/puzzles/p2_m_of_n_delegate_direct.clsp +108 -0
- chia/wallet/puzzles/p2_m_of_n_delegate_direct.clsp.hex +1 -0
- chia/wallet/puzzles/p2_m_of_n_delegate_direct.py +21 -0
- chia/wallet/puzzles/p2_parent.clsp +19 -0
- chia/wallet/puzzles/p2_parent.clsp.hex +1 -0
- chia/wallet/puzzles/p2_puzzle_hash.clsp +18 -0
- chia/wallet/puzzles/p2_puzzle_hash.clsp.hex +1 -0
- chia/wallet/puzzles/p2_puzzle_hash.py +27 -0
- chia/wallet/puzzles/p2_singleton.clsp +30 -0
- chia/wallet/puzzles/p2_singleton.clsp.hex +1 -0
- chia/wallet/puzzles/p2_singleton_aggregator.clsp +81 -0
- chia/wallet/puzzles/p2_singleton_aggregator.clsp.hex +1 -0
- chia/wallet/puzzles/p2_singleton_or_delayed_puzhash.clsp +50 -0
- chia/wallet/puzzles/p2_singleton_or_delayed_puzhash.clsp.hex +1 -0
- chia/wallet/puzzles/p2_singleton_via_delegated_puzzle.clsp +47 -0
- chia/wallet/puzzles/p2_singleton_via_delegated_puzzle.clsp.hex +1 -0
- chia/wallet/puzzles/puzzle_utils.py +34 -0
- chia/wallet/puzzles/settlement_payments.clsp +49 -0
- chia/wallet/puzzles/settlement_payments.clsp.hex +1 -0
- chia/wallet/puzzles/sha256tree.clib +11 -0
- chia/wallet/puzzles/singleton_launcher.clsp +16 -0
- chia/wallet/puzzles/singleton_launcher.clsp.hex +1 -0
- chia/wallet/puzzles/singleton_top_layer.clsp +177 -0
- chia/wallet/puzzles/singleton_top_layer.clsp.hex +1 -0
- chia/wallet/puzzles/singleton_top_layer.py +296 -0
- chia/wallet/puzzles/singleton_top_layer_v1_1.clsp +107 -0
- chia/wallet/puzzles/singleton_top_layer_v1_1.clsp.hex +1 -0
- chia/wallet/puzzles/singleton_top_layer_v1_1.py +345 -0
- chia/wallet/puzzles/singleton_truths.clib +21 -0
- chia/wallet/puzzles/tails.py +348 -0
- chia/wallet/puzzles/utility_macros.clib +48 -0
- chia/wallet/signer_protocol.py +125 -0
- chia/wallet/singleton.py +106 -0
- chia/wallet/singleton_record.py +30 -0
- chia/wallet/trade_manager.py +1102 -0
- chia/wallet/trade_record.py +67 -0
- chia/wallet/trading/__init__.py +0 -0
- chia/wallet/trading/offer.py +702 -0
- chia/wallet/trading/trade_status.py +13 -0
- chia/wallet/trading/trade_store.py +526 -0
- chia/wallet/transaction_record.py +158 -0
- chia/wallet/transaction_sorting.py +14 -0
- chia/wallet/uncurried_puzzle.py +17 -0
- chia/wallet/util/__init__.py +0 -0
- chia/wallet/util/address_type.py +55 -0
- chia/wallet/util/blind_signer_tl.py +164 -0
- chia/wallet/util/clvm_streamable.py +203 -0
- chia/wallet/util/compute_hints.py +66 -0
- chia/wallet/util/compute_memos.py +43 -0
- chia/wallet/util/curry_and_treehash.py +91 -0
- chia/wallet/util/debug_spend_bundle.py +232 -0
- chia/wallet/util/merkle_tree.py +100 -0
- chia/wallet/util/merkle_utils.py +102 -0
- chia/wallet/util/new_peak_queue.py +82 -0
- chia/wallet/util/notifications.py +12 -0
- chia/wallet/util/peer_request_cache.py +174 -0
- chia/wallet/util/pprint.py +39 -0
- chia/wallet/util/puzzle_compression.py +95 -0
- chia/wallet/util/puzzle_decorator.py +100 -0
- chia/wallet/util/puzzle_decorator_type.py +7 -0
- chia/wallet/util/query_filter.py +59 -0
- chia/wallet/util/transaction_type.py +23 -0
- chia/wallet/util/tx_config.py +158 -0
- chia/wallet/util/wallet_sync_utils.py +351 -0
- chia/wallet/util/wallet_types.py +72 -0
- chia/wallet/vc_wallet/__init__.py +0 -0
- chia/wallet/vc_wallet/cr_cat_drivers.py +664 -0
- chia/wallet/vc_wallet/cr_cat_wallet.py +877 -0
- chia/wallet/vc_wallet/cr_outer_puzzle.py +102 -0
- chia/wallet/vc_wallet/cr_puzzles/__init__.py +0 -0
- chia/wallet/vc_wallet/cr_puzzles/conditions_w_fee_announce.clsp +3 -0
- chia/wallet/vc_wallet/cr_puzzles/conditions_w_fee_announce.clsp.hex +1 -0
- chia/wallet/vc_wallet/cr_puzzles/credential_restriction.clsp +304 -0
- chia/wallet/vc_wallet/cr_puzzles/credential_restriction.clsp.hex +1 -0
- chia/wallet/vc_wallet/cr_puzzles/flag_proofs_checker.clsp +45 -0
- chia/wallet/vc_wallet/cr_puzzles/flag_proofs_checker.clsp.hex +1 -0
- chia/wallet/vc_wallet/vc_drivers.py +838 -0
- chia/wallet/vc_wallet/vc_puzzles/__init__.py +0 -0
- chia/wallet/vc_wallet/vc_puzzles/covenant_layer.clsp +30 -0
- chia/wallet/vc_wallet/vc_puzzles/covenant_layer.clsp.hex +1 -0
- chia/wallet/vc_wallet/vc_puzzles/eml_covenant_morpher.clsp +75 -0
- chia/wallet/vc_wallet/vc_puzzles/eml_covenant_morpher.clsp.hex +1 -0
- chia/wallet/vc_wallet/vc_puzzles/eml_transfer_program_covenant_adapter.clsp +32 -0
- chia/wallet/vc_wallet/vc_puzzles/eml_transfer_program_covenant_adapter.clsp.hex +1 -0
- chia/wallet/vc_wallet/vc_puzzles/eml_update_metadata_with_DID.clsp +80 -0
- chia/wallet/vc_wallet/vc_puzzles/eml_update_metadata_with_DID.clsp.hex +1 -0
- chia/wallet/vc_wallet/vc_puzzles/exigent_metadata_layer.clsp +163 -0
- chia/wallet/vc_wallet/vc_puzzles/exigent_metadata_layer.clsp.hex +1 -0
- chia/wallet/vc_wallet/vc_puzzles/p2_announced_delegated_puzzle.clsp +16 -0
- chia/wallet/vc_wallet/vc_puzzles/p2_announced_delegated_puzzle.clsp.hex +1 -0
- chia/wallet/vc_wallet/vc_puzzles/standard_vc_backdoor_puzzle.clsp +74 -0
- chia/wallet/vc_wallet/vc_puzzles/standard_vc_backdoor_puzzle.clsp.hex +1 -0
- chia/wallet/vc_wallet/vc_puzzles/std_parent_morpher.clsp +23 -0
- chia/wallet/vc_wallet/vc_puzzles/std_parent_morpher.clsp.hex +1 -0
- chia/wallet/vc_wallet/vc_puzzles/viral_backdoor.clsp +64 -0
- chia/wallet/vc_wallet/vc_puzzles/viral_backdoor.clsp.hex +1 -0
- chia/wallet/vc_wallet/vc_store.py +263 -0
- chia/wallet/vc_wallet/vc_wallet.py +638 -0
- chia/wallet/wallet.py +698 -0
- chia/wallet/wallet_action_scope.py +96 -0
- chia/wallet/wallet_blockchain.py +244 -0
- chia/wallet/wallet_coin_record.py +72 -0
- chia/wallet/wallet_coin_store.py +351 -0
- chia/wallet/wallet_info.py +35 -0
- chia/wallet/wallet_interested_store.py +188 -0
- chia/wallet/wallet_nft_store.py +279 -0
- chia/wallet/wallet_node.py +1765 -0
- chia/wallet/wallet_node_api.py +207 -0
- chia/wallet/wallet_pool_store.py +119 -0
- chia/wallet/wallet_protocol.py +90 -0
- chia/wallet/wallet_puzzle_store.py +396 -0
- chia/wallet/wallet_retry_store.py +70 -0
- chia/wallet/wallet_singleton_store.py +259 -0
- chia/wallet/wallet_spend_bundle.py +25 -0
- chia/wallet/wallet_state_manager.py +2819 -0
- chia/wallet/wallet_transaction_store.py +496 -0
- chia/wallet/wallet_user_store.py +110 -0
- chia/wallet/wallet_weight_proof_handler.py +126 -0
- chia_blockchain-2.5.1rc1.dist-info/LICENSE +201 -0
- chia_blockchain-2.5.1rc1.dist-info/METADATA +156 -0
- chia_blockchain-2.5.1rc1.dist-info/RECORD +1042 -0
- chia_blockchain-2.5.1rc1.dist-info/WHEEL +4 -0
- chia_blockchain-2.5.1rc1.dist-info/entry_points.txt +17 -0
- mozilla-ca/cacert.pem +3611 -0
|
@@ -0,0 +1,855 @@
|
|
|
1
|
+
from __future__ import annotations
|
|
2
|
+
|
|
3
|
+
import dataclasses
|
|
4
|
+
import logging
|
|
5
|
+
import time
|
|
6
|
+
import traceback
|
|
7
|
+
from typing import TYPE_CHECKING, Any, ClassVar, Optional, cast
|
|
8
|
+
|
|
9
|
+
from chia_rs import G1Element
|
|
10
|
+
from typing_extensions import Unpack
|
|
11
|
+
|
|
12
|
+
from chia.consensus.default_constants import DEFAULT_CONSTANTS
|
|
13
|
+
from chia.server.ws_connection import WSChiaConnection
|
|
14
|
+
from chia.types.blockchain_format.coin import Coin
|
|
15
|
+
from chia.types.blockchain_format.program import Program
|
|
16
|
+
from chia.types.blockchain_format.sized_bytes import bytes32
|
|
17
|
+
from chia.types.coin_spend import compute_additions_with_cost
|
|
18
|
+
from chia.types.condition_opcodes import ConditionOpcode
|
|
19
|
+
from chia.util.byte_types import hexstr_to_bytes
|
|
20
|
+
from chia.util.errors import Err, ValidationError
|
|
21
|
+
from chia.util.hash import std_hash
|
|
22
|
+
from chia.util.ints import uint32, uint64, uint128
|
|
23
|
+
from chia.wallet.cat_wallet.cat_constants import DEFAULT_CATS
|
|
24
|
+
from chia.wallet.cat_wallet.cat_info import CATCoinData, CATInfo, LegacyCATInfo
|
|
25
|
+
from chia.wallet.cat_wallet.cat_utils import (
|
|
26
|
+
CAT_MOD,
|
|
27
|
+
SpendableCAT,
|
|
28
|
+
construct_cat_puzzle,
|
|
29
|
+
match_cat_puzzle,
|
|
30
|
+
unsigned_spend_bundle_for_spendable_cats,
|
|
31
|
+
)
|
|
32
|
+
from chia.wallet.cat_wallet.lineage_store import CATLineageStore
|
|
33
|
+
from chia.wallet.coin_selection import select_coins
|
|
34
|
+
from chia.wallet.conditions import (
|
|
35
|
+
AssertCoinAnnouncement,
|
|
36
|
+
Condition,
|
|
37
|
+
ConditionValidTimes,
|
|
38
|
+
CreateCoinAnnouncement,
|
|
39
|
+
UnknownCondition,
|
|
40
|
+
parse_timelock_info,
|
|
41
|
+
)
|
|
42
|
+
from chia.wallet.derivation_record import DerivationRecord
|
|
43
|
+
from chia.wallet.lineage_proof import LineageProof
|
|
44
|
+
from chia.wallet.outer_puzzles import AssetType
|
|
45
|
+
from chia.wallet.payment import Payment
|
|
46
|
+
from chia.wallet.puzzle_drivers import PuzzleInfo
|
|
47
|
+
from chia.wallet.puzzles.tails import ALL_LIMITATIONS_PROGRAMS
|
|
48
|
+
from chia.wallet.transaction_record import TransactionRecord
|
|
49
|
+
from chia.wallet.uncurried_puzzle import uncurry_puzzle
|
|
50
|
+
from chia.wallet.util.compute_memos import compute_memos
|
|
51
|
+
from chia.wallet.util.curry_and_treehash import calculate_hash_of_quoted_mod_hash, curry_and_treehash
|
|
52
|
+
from chia.wallet.util.transaction_type import TransactionType
|
|
53
|
+
from chia.wallet.util.wallet_sync_utils import fetch_coin_spend_for_coin_state
|
|
54
|
+
from chia.wallet.util.wallet_types import WalletType
|
|
55
|
+
from chia.wallet.wallet import Wallet
|
|
56
|
+
from chia.wallet.wallet_action_scope import WalletActionScope
|
|
57
|
+
from chia.wallet.wallet_coin_record import WalletCoinRecord
|
|
58
|
+
from chia.wallet.wallet_info import WalletInfo
|
|
59
|
+
from chia.wallet.wallet_protocol import GSTOptionalArgs, WalletProtocol
|
|
60
|
+
from chia.wallet.wallet_spend_bundle import WalletSpendBundle
|
|
61
|
+
|
|
62
|
+
if TYPE_CHECKING:
|
|
63
|
+
from chia.wallet.wallet_state_manager import WalletStateManager
|
|
64
|
+
|
|
65
|
+
# This should probably not live in this file but it's for experimental right now
|
|
66
|
+
|
|
67
|
+
CAT_MOD_HASH = CAT_MOD.get_tree_hash()
|
|
68
|
+
CAT_MOD_HASH_HASH = Program.to(CAT_MOD_HASH).get_tree_hash()
|
|
69
|
+
QUOTED_MOD_HASH = calculate_hash_of_quoted_mod_hash(CAT_MOD_HASH)
|
|
70
|
+
|
|
71
|
+
|
|
72
|
+
def not_ephemeral_additions(sp: WalletSpendBundle) -> list[Coin]:
|
|
73
|
+
removals: set[Coin] = set()
|
|
74
|
+
for cs in sp.coin_spends:
|
|
75
|
+
removals.add(cs.coin)
|
|
76
|
+
|
|
77
|
+
additions: list[Coin] = []
|
|
78
|
+
max_cost = int(DEFAULT_CONSTANTS.MAX_BLOCK_COST_CLVM)
|
|
79
|
+
for cs in sp.coin_spends:
|
|
80
|
+
coins, cost = compute_additions_with_cost(cs, max_cost=max_cost)
|
|
81
|
+
max_cost -= cost
|
|
82
|
+
if max_cost < 0:
|
|
83
|
+
raise ValidationError(Err.BLOCK_COST_EXCEEDS_MAX, "non_ephemeral_additions() for SpendBundle")
|
|
84
|
+
for c in coins:
|
|
85
|
+
if c not in removals:
|
|
86
|
+
additions.append(c)
|
|
87
|
+
return additions
|
|
88
|
+
|
|
89
|
+
|
|
90
|
+
class CATWallet:
|
|
91
|
+
if TYPE_CHECKING:
|
|
92
|
+
_protocol_check: ClassVar[WalletProtocol[CATCoinData]] = cast("CATWallet", None)
|
|
93
|
+
|
|
94
|
+
wallet_state_manager: WalletStateManager
|
|
95
|
+
log: logging.Logger
|
|
96
|
+
wallet_info: WalletInfo
|
|
97
|
+
cat_info: CATInfo
|
|
98
|
+
standard_wallet: Wallet
|
|
99
|
+
lineage_store: CATLineageStore
|
|
100
|
+
|
|
101
|
+
@staticmethod
|
|
102
|
+
def default_wallet_name_for_unknown_cat(limitations_program_hash_hex: str) -> str:
|
|
103
|
+
return f"CAT {limitations_program_hash_hex[:16]}..."
|
|
104
|
+
|
|
105
|
+
@staticmethod
|
|
106
|
+
async def create_new_cat_wallet(
|
|
107
|
+
wallet_state_manager: WalletStateManager,
|
|
108
|
+
wallet: Wallet,
|
|
109
|
+
cat_tail_info: dict[str, Any],
|
|
110
|
+
amount: uint64,
|
|
111
|
+
action_scope: WalletActionScope,
|
|
112
|
+
fee: uint64 = uint64(0),
|
|
113
|
+
name: Optional[str] = None,
|
|
114
|
+
push: bool = True,
|
|
115
|
+
) -> CATWallet:
|
|
116
|
+
self = CATWallet()
|
|
117
|
+
self.standard_wallet = wallet
|
|
118
|
+
self.log = logging.getLogger(__name__)
|
|
119
|
+
std_wallet_id = self.standard_wallet.wallet_id
|
|
120
|
+
bal = await wallet_state_manager.get_confirmed_balance_for_wallet(std_wallet_id)
|
|
121
|
+
if amount > bal:
|
|
122
|
+
raise ValueError("Not enough balance")
|
|
123
|
+
self.wallet_state_manager = wallet_state_manager
|
|
124
|
+
|
|
125
|
+
# We use 00 bytes because it's not optional. We must check this is overridden during issuance.
|
|
126
|
+
empty_bytes = bytes32(32 * b"\0")
|
|
127
|
+
self.cat_info = CATInfo(empty_bytes, None)
|
|
128
|
+
info_as_string = bytes(self.cat_info).hex()
|
|
129
|
+
# If the name is not provided, it will be autogenerated based on the resulting tail hash.
|
|
130
|
+
# For now, give the wallet a temporary name "CAT WALLET" until we get the tail hash
|
|
131
|
+
original_name = name
|
|
132
|
+
if name is None:
|
|
133
|
+
name = "CAT WALLET"
|
|
134
|
+
|
|
135
|
+
self.wallet_info = await wallet_state_manager.user_store.create_wallet(name, WalletType.CAT, info_as_string)
|
|
136
|
+
|
|
137
|
+
try:
|
|
138
|
+
spend_bundle = await ALL_LIMITATIONS_PROGRAMS[cat_tail_info["identifier"]].generate_issuance_bundle(
|
|
139
|
+
self,
|
|
140
|
+
cat_tail_info,
|
|
141
|
+
amount,
|
|
142
|
+
action_scope,
|
|
143
|
+
fee,
|
|
144
|
+
)
|
|
145
|
+
assert self.cat_info.limitations_program_hash != empty_bytes
|
|
146
|
+
except Exception:
|
|
147
|
+
await wallet_state_manager.delete_wallet(self.id())
|
|
148
|
+
raise
|
|
149
|
+
if spend_bundle is None:
|
|
150
|
+
await wallet_state_manager.delete_wallet(self.id())
|
|
151
|
+
raise ValueError("Failed to create spend.")
|
|
152
|
+
|
|
153
|
+
await self.wallet_state_manager.add_new_wallet(self)
|
|
154
|
+
|
|
155
|
+
# If the new CAT name wasn't originally provided, we used a temporary name before issuance
|
|
156
|
+
# since we didn't yet know the TAIL. Now we know the TAIL, we can update the name
|
|
157
|
+
# according to the template name for unknown/new CATs.
|
|
158
|
+
if original_name is None:
|
|
159
|
+
name = self.default_wallet_name_for_unknown_cat(self.cat_info.limitations_program_hash.hex())
|
|
160
|
+
await self.set_name(name)
|
|
161
|
+
|
|
162
|
+
# Change and actual CAT coin
|
|
163
|
+
non_ephemeral_coins: list[Coin] = not_ephemeral_additions(spend_bundle)
|
|
164
|
+
cat_coin = None
|
|
165
|
+
puzzle_store = self.wallet_state_manager.puzzle_store
|
|
166
|
+
for c in non_ephemeral_coins:
|
|
167
|
+
wallet_identifier = await puzzle_store.get_wallet_identifier_for_puzzle_hash(c.puzzle_hash)
|
|
168
|
+
if wallet_identifier is None:
|
|
169
|
+
raise ValueError("Internal Error")
|
|
170
|
+
if wallet_identifier.id == self.id():
|
|
171
|
+
cat_coin = c
|
|
172
|
+
|
|
173
|
+
if cat_coin is None:
|
|
174
|
+
raise ValueError("Internal Error, unable to generate new CAT coin")
|
|
175
|
+
cat_pid: bytes32 = cat_coin.parent_coin_info
|
|
176
|
+
|
|
177
|
+
cat_record = TransactionRecord(
|
|
178
|
+
confirmed_at_height=uint32(0),
|
|
179
|
+
created_at_time=uint64(int(time.time())),
|
|
180
|
+
to_puzzle_hash=(await self.convert_puzzle_hash(cat_coin.puzzle_hash)),
|
|
181
|
+
amount=uint64(cat_coin.amount),
|
|
182
|
+
fee_amount=fee,
|
|
183
|
+
confirmed=False,
|
|
184
|
+
sent=uint32(10),
|
|
185
|
+
spend_bundle=spend_bundle,
|
|
186
|
+
additions=[cat_coin],
|
|
187
|
+
removals=list(filter(lambda rem: rem.name() == cat_pid, spend_bundle.removals())),
|
|
188
|
+
wallet_id=self.id(),
|
|
189
|
+
sent_to=[],
|
|
190
|
+
trade_id=None,
|
|
191
|
+
type=uint32(TransactionType.INCOMING_TX.value),
|
|
192
|
+
name=spend_bundle.name(),
|
|
193
|
+
memos=[],
|
|
194
|
+
valid_times=ConditionValidTimes(),
|
|
195
|
+
)
|
|
196
|
+
async with action_scope.use() as interface:
|
|
197
|
+
interface.side_effects.transactions.append(cat_record)
|
|
198
|
+
return self
|
|
199
|
+
|
|
200
|
+
@staticmethod
|
|
201
|
+
async def get_or_create_wallet_for_cat(
|
|
202
|
+
wallet_state_manager: WalletStateManager,
|
|
203
|
+
wallet: Wallet,
|
|
204
|
+
limitations_program_hash_hex: str,
|
|
205
|
+
name: Optional[str] = None,
|
|
206
|
+
) -> CATWallet:
|
|
207
|
+
self = CATWallet()
|
|
208
|
+
self.standard_wallet = wallet
|
|
209
|
+
self.log = logging.getLogger(__name__)
|
|
210
|
+
|
|
211
|
+
limitations_program_hash_hex = bytes32.from_hexstr(limitations_program_hash_hex).hex() # Normalize the format
|
|
212
|
+
|
|
213
|
+
for id, w in wallet_state_manager.wallets.items():
|
|
214
|
+
if w.type() == CATWallet.type():
|
|
215
|
+
assert isinstance(w, CATWallet)
|
|
216
|
+
if w.get_asset_id() == limitations_program_hash_hex:
|
|
217
|
+
self.log.warning("Not creating wallet for already existing CAT wallet")
|
|
218
|
+
return w
|
|
219
|
+
|
|
220
|
+
self.wallet_state_manager = wallet_state_manager
|
|
221
|
+
if limitations_program_hash_hex in DEFAULT_CATS:
|
|
222
|
+
cat_info = DEFAULT_CATS[limitations_program_hash_hex]
|
|
223
|
+
name = cat_info["name"]
|
|
224
|
+
elif name is None:
|
|
225
|
+
name = self.default_wallet_name_for_unknown_cat(limitations_program_hash_hex)
|
|
226
|
+
|
|
227
|
+
limitations_program_hash = bytes32.from_hexstr(limitations_program_hash_hex)
|
|
228
|
+
self.cat_info = CATInfo(limitations_program_hash, None)
|
|
229
|
+
info_as_string = bytes(self.cat_info).hex()
|
|
230
|
+
self.wallet_info = await wallet_state_manager.user_store.create_wallet(name, WalletType.CAT, info_as_string)
|
|
231
|
+
|
|
232
|
+
self.lineage_store = await CATLineageStore.create(self.wallet_state_manager.db_wrapper, self.get_asset_id())
|
|
233
|
+
await self.wallet_state_manager.add_new_wallet(self)
|
|
234
|
+
|
|
235
|
+
delete: bool = False
|
|
236
|
+
for state in await self.wallet_state_manager.interested_store.get_unacknowledged_states_for_asset_id(
|
|
237
|
+
limitations_program_hash
|
|
238
|
+
):
|
|
239
|
+
new_peer = self.wallet_state_manager.wallet_node.get_full_node_peer()
|
|
240
|
+
if new_peer is not None:
|
|
241
|
+
delete = True
|
|
242
|
+
peer_id: bytes32 = new_peer.peer_node_id
|
|
243
|
+
await self.wallet_state_manager.retry_store.add_state(state[0], peer_id, state[1])
|
|
244
|
+
|
|
245
|
+
if delete:
|
|
246
|
+
await self.wallet_state_manager.interested_store.delete_unacknowledged_states_for_asset_id(
|
|
247
|
+
limitations_program_hash
|
|
248
|
+
)
|
|
249
|
+
|
|
250
|
+
return self
|
|
251
|
+
|
|
252
|
+
@classmethod
|
|
253
|
+
async def create_from_puzzle_info(
|
|
254
|
+
cls,
|
|
255
|
+
wallet_state_manager: WalletStateManager,
|
|
256
|
+
wallet: Wallet,
|
|
257
|
+
puzzle_driver: PuzzleInfo,
|
|
258
|
+
name: Optional[str] = None,
|
|
259
|
+
# We're hinting this as Any for mypy by should explore adding this to the wallet protocol and hinting properly
|
|
260
|
+
potential_subclasses: dict[AssetType, Any] = {},
|
|
261
|
+
) -> Any:
|
|
262
|
+
next_layer: Optional[PuzzleInfo] = puzzle_driver.also()
|
|
263
|
+
if next_layer is not None:
|
|
264
|
+
if AssetType(next_layer.type()) in potential_subclasses:
|
|
265
|
+
return await potential_subclasses[AssetType(next_layer.type())].create_from_puzzle_info(
|
|
266
|
+
wallet_state_manager,
|
|
267
|
+
wallet,
|
|
268
|
+
puzzle_driver,
|
|
269
|
+
name,
|
|
270
|
+
potential_subclasses,
|
|
271
|
+
)
|
|
272
|
+
return await cls.get_or_create_wallet_for_cat(
|
|
273
|
+
wallet_state_manager,
|
|
274
|
+
wallet,
|
|
275
|
+
puzzle_driver["tail"].hex(),
|
|
276
|
+
name,
|
|
277
|
+
)
|
|
278
|
+
|
|
279
|
+
@staticmethod
|
|
280
|
+
async def create(
|
|
281
|
+
wallet_state_manager: WalletStateManager,
|
|
282
|
+
wallet: Wallet,
|
|
283
|
+
wallet_info: WalletInfo,
|
|
284
|
+
) -> CATWallet:
|
|
285
|
+
self = CATWallet()
|
|
286
|
+
|
|
287
|
+
self.log = logging.getLogger(__name__)
|
|
288
|
+
|
|
289
|
+
self.wallet_state_manager = wallet_state_manager
|
|
290
|
+
self.wallet_info = wallet_info
|
|
291
|
+
self.standard_wallet = wallet
|
|
292
|
+
try:
|
|
293
|
+
self.cat_info = CATInfo.from_bytes(hexstr_to_bytes(self.wallet_info.data))
|
|
294
|
+
self.lineage_store = await CATLineageStore.create(self.wallet_state_manager.db_wrapper, self.get_asset_id())
|
|
295
|
+
except AssertionError:
|
|
296
|
+
# Do a migration of the lineage proofs
|
|
297
|
+
cat_info = LegacyCATInfo.from_bytes(hexstr_to_bytes(self.wallet_info.data))
|
|
298
|
+
self.cat_info = CATInfo(cat_info.limitations_program_hash, cat_info.my_tail)
|
|
299
|
+
self.lineage_store = await CATLineageStore.create(self.wallet_state_manager.db_wrapper, self.get_asset_id())
|
|
300
|
+
for coin_id, lineage in cat_info.lineage_proofs:
|
|
301
|
+
await self.add_lineage(coin_id, lineage)
|
|
302
|
+
await self.save_info(self.cat_info)
|
|
303
|
+
|
|
304
|
+
return self
|
|
305
|
+
|
|
306
|
+
@classmethod
|
|
307
|
+
def type(cls) -> WalletType:
|
|
308
|
+
return WalletType.CAT
|
|
309
|
+
|
|
310
|
+
def id(self) -> uint32:
|
|
311
|
+
return self.wallet_info.id
|
|
312
|
+
|
|
313
|
+
async def get_confirmed_balance(self, record_list: Optional[set[WalletCoinRecord]] = None) -> uint128:
|
|
314
|
+
if record_list is None:
|
|
315
|
+
record_list = await self.wallet_state_manager.coin_store.get_unspent_coins_for_wallet(self.id())
|
|
316
|
+
|
|
317
|
+
amount: uint128 = uint128(0)
|
|
318
|
+
for record in record_list:
|
|
319
|
+
lineage = await self.get_lineage_proof_for_coin(record.coin)
|
|
320
|
+
if lineage is not None:
|
|
321
|
+
amount = uint128(amount + record.coin.amount)
|
|
322
|
+
|
|
323
|
+
self.log.info(f"Confirmed balance for cat wallet {self.id()} is {amount}")
|
|
324
|
+
return uint128(amount)
|
|
325
|
+
|
|
326
|
+
async def get_unconfirmed_balance(self, unspent_records: Optional[set[WalletCoinRecord]] = None) -> uint128:
|
|
327
|
+
return await self.wallet_state_manager.get_unconfirmed_balance(self.id(), unspent_records)
|
|
328
|
+
|
|
329
|
+
@property
|
|
330
|
+
def cost_of_single_tx(self) -> int:
|
|
331
|
+
return 30000000 # Estimate
|
|
332
|
+
|
|
333
|
+
@property
|
|
334
|
+
def max_send_quantity(self) -> int:
|
|
335
|
+
# avoid full block TXs
|
|
336
|
+
return int(self.wallet_state_manager.constants.MAX_BLOCK_COST_CLVM / 2 / self.cost_of_single_tx)
|
|
337
|
+
|
|
338
|
+
async def get_max_spendable_coins(self, records: Optional[set[WalletCoinRecord]] = None) -> set[WalletCoinRecord]:
|
|
339
|
+
spendable: list[WalletCoinRecord] = list(
|
|
340
|
+
await self.wallet_state_manager.get_spendable_coins_for_wallet(self.id(), records)
|
|
341
|
+
)
|
|
342
|
+
spendable.sort(reverse=True, key=lambda record: record.coin.amount)
|
|
343
|
+
return set(spendable[0 : min(len(spendable), self.max_send_quantity)])
|
|
344
|
+
|
|
345
|
+
async def get_max_send_amount(self, records: Optional[set[WalletCoinRecord]] = None) -> uint128:
|
|
346
|
+
return uint128(sum(cr.coin.amount for cr in await self.get_max_spendable_coins()))
|
|
347
|
+
|
|
348
|
+
def get_name(self) -> str:
|
|
349
|
+
return self.wallet_info.name
|
|
350
|
+
|
|
351
|
+
async def set_name(self, new_name: str) -> None:
|
|
352
|
+
new_info = dataclasses.replace(self.wallet_info, name=new_name)
|
|
353
|
+
self.wallet_info = new_info
|
|
354
|
+
await self.wallet_state_manager.user_store.update_wallet(self.wallet_info)
|
|
355
|
+
|
|
356
|
+
def get_asset_id(self) -> str:
|
|
357
|
+
return bytes(self.cat_info.limitations_program_hash).hex()
|
|
358
|
+
|
|
359
|
+
async def set_tail_program(self, tail_program: str) -> None:
|
|
360
|
+
assert Program.fromhex(tail_program).get_tree_hash() == self.cat_info.limitations_program_hash
|
|
361
|
+
await self.save_info(
|
|
362
|
+
CATInfo(
|
|
363
|
+
self.cat_info.limitations_program_hash,
|
|
364
|
+
Program.fromhex(tail_program),
|
|
365
|
+
)
|
|
366
|
+
)
|
|
367
|
+
|
|
368
|
+
async def coin_added(
|
|
369
|
+
self, coin: Coin, height: uint32, peer: WSChiaConnection, parent_coin_data: Optional[CATCoinData]
|
|
370
|
+
) -> None:
|
|
371
|
+
"""Notification from wallet state manager that wallet has been received."""
|
|
372
|
+
self.log.info(f"CAT wallet has been notified that {coin.name().hex()} was added")
|
|
373
|
+
|
|
374
|
+
inner_puzzle = await self.inner_puzzle_for_cat_puzhash(coin.puzzle_hash)
|
|
375
|
+
lineage_proof = LineageProof(coin.parent_coin_info, inner_puzzle.get_tree_hash(), uint64(coin.amount))
|
|
376
|
+
await self.add_lineage(coin.name(), lineage_proof)
|
|
377
|
+
|
|
378
|
+
lineage = await self.get_lineage_proof_for_coin(coin)
|
|
379
|
+
|
|
380
|
+
if lineage is None:
|
|
381
|
+
try:
|
|
382
|
+
if parent_coin_data is None:
|
|
383
|
+
# The method is not triggered after the determine_coin_type, no pre-fetched data
|
|
384
|
+
coin_state = await self.wallet_state_manager.wallet_node.get_coin_state(
|
|
385
|
+
[coin.parent_coin_info], peer=peer
|
|
386
|
+
)
|
|
387
|
+
assert coin_state[0].coin.name() == coin.parent_coin_info
|
|
388
|
+
coin_spend = await fetch_coin_spend_for_coin_state(coin_state[0], peer)
|
|
389
|
+
cat_curried_args = match_cat_puzzle(uncurry_puzzle(coin_spend.puzzle_reveal))
|
|
390
|
+
if cat_curried_args is not None:
|
|
391
|
+
cat_mod_hash, tail_program_hash, cat_inner_puzzle = cat_curried_args
|
|
392
|
+
parent_coin_data = CATCoinData(
|
|
393
|
+
bytes32(cat_mod_hash.as_atom()),
|
|
394
|
+
bytes32(tail_program_hash.as_atom()),
|
|
395
|
+
cat_inner_puzzle,
|
|
396
|
+
coin_state[0].coin.parent_coin_info,
|
|
397
|
+
uint64(coin_state[0].coin.amount),
|
|
398
|
+
)
|
|
399
|
+
await self.puzzle_solution_received(coin, parent_coin_data)
|
|
400
|
+
except Exception as e:
|
|
401
|
+
self.log.debug(f"Exception: {e}, traceback: {traceback.format_exc()}")
|
|
402
|
+
|
|
403
|
+
async def puzzle_solution_received(self, coin: Coin, parent_coin_data: Optional[CATCoinData]) -> None:
|
|
404
|
+
coin_name = coin.parent_coin_info
|
|
405
|
+
if parent_coin_data is not None:
|
|
406
|
+
assert isinstance(parent_coin_data, CATCoinData)
|
|
407
|
+
data: CATCoinData = parent_coin_data
|
|
408
|
+
self.log.info(f"parent: {coin_name.hex()} inner_puzzle for parent is {data.inner_puzzle}")
|
|
409
|
+
|
|
410
|
+
await self.add_lineage(
|
|
411
|
+
coin_name,
|
|
412
|
+
LineageProof(data.parent_coin_id, data.inner_puzzle.get_tree_hash(), data.amount),
|
|
413
|
+
)
|
|
414
|
+
else:
|
|
415
|
+
# The parent is not a CAT which means we need to scrub all of its children from our DB
|
|
416
|
+
child_coin_records = await self.wallet_state_manager.coin_store.get_coin_records_by_parent_id(coin_name)
|
|
417
|
+
if len(child_coin_records) > 0:
|
|
418
|
+
for record in child_coin_records:
|
|
419
|
+
if record.wallet_id == self.id():
|
|
420
|
+
await self.wallet_state_manager.coin_store.delete_coin_record(record.coin.name())
|
|
421
|
+
await self.remove_lineage(record.coin.name())
|
|
422
|
+
# We also need to make sure there's no record of the transaction
|
|
423
|
+
await self.wallet_state_manager.tx_store.delete_transaction_record(record.coin.name())
|
|
424
|
+
|
|
425
|
+
def require_derivation_paths(self) -> bool:
|
|
426
|
+
return True
|
|
427
|
+
|
|
428
|
+
def puzzle_for_pk(self, pubkey: G1Element) -> Program:
|
|
429
|
+
inner_puzzle = self.standard_wallet.puzzle_for_pk(pubkey)
|
|
430
|
+
cat_puzzle: Program = construct_cat_puzzle(CAT_MOD, self.cat_info.limitations_program_hash, inner_puzzle)
|
|
431
|
+
return cat_puzzle
|
|
432
|
+
|
|
433
|
+
def puzzle_hash_for_pk(self, pubkey: G1Element) -> bytes32:
|
|
434
|
+
inner_puzzle_hash = self.standard_wallet.puzzle_hash_for_pk(pubkey)
|
|
435
|
+
limitations_program_hash_hash = Program.to(self.cat_info.limitations_program_hash).get_tree_hash()
|
|
436
|
+
return curry_and_treehash(QUOTED_MOD_HASH, CAT_MOD_HASH_HASH, limitations_program_hash_hash, inner_puzzle_hash)
|
|
437
|
+
|
|
438
|
+
async def get_cat_puzzle_hash(self, new: bool) -> bytes32:
|
|
439
|
+
if new:
|
|
440
|
+
return (await self.wallet_state_manager.get_unused_derivation_record(self.id())).puzzle_hash
|
|
441
|
+
else:
|
|
442
|
+
derivation_record = await self.wallet_state_manager.get_current_derivation_record_for_wallet(self.id())
|
|
443
|
+
if derivation_record is None:
|
|
444
|
+
return (await self.wallet_state_manager.get_unused_derivation_record(self.id())).puzzle_hash
|
|
445
|
+
|
|
446
|
+
return derivation_record.puzzle_hash
|
|
447
|
+
|
|
448
|
+
async def get_spendable_balance(self, records: Optional[set[WalletCoinRecord]] = None) -> uint128:
|
|
449
|
+
coins = await self.get_cat_spendable_coins(records)
|
|
450
|
+
amount = 0
|
|
451
|
+
for record in coins:
|
|
452
|
+
amount += record.coin.amount
|
|
453
|
+
|
|
454
|
+
return uint128(amount)
|
|
455
|
+
|
|
456
|
+
async def get_pending_change_balance(self) -> uint64:
|
|
457
|
+
unconfirmed_tx = await self.wallet_state_manager.tx_store.get_unconfirmed_for_wallet(self.id())
|
|
458
|
+
addition_amount = 0
|
|
459
|
+
for record in unconfirmed_tx:
|
|
460
|
+
if not record.is_in_mempool() and record.spend_bundle is not None:
|
|
461
|
+
continue
|
|
462
|
+
our_spend = False
|
|
463
|
+
for coin in record.removals:
|
|
464
|
+
if await self.wallet_state_manager.does_coin_belong_to_wallet(coin, self.id()):
|
|
465
|
+
our_spend = True
|
|
466
|
+
break
|
|
467
|
+
|
|
468
|
+
if our_spend is not True:
|
|
469
|
+
continue
|
|
470
|
+
|
|
471
|
+
for coin in record.additions:
|
|
472
|
+
hint_dict = {
|
|
473
|
+
coin_id: bytes32(memos[0])
|
|
474
|
+
for coin_id, memos in record.memos
|
|
475
|
+
if len(memos) > 0 and len(memos[0]) == 32
|
|
476
|
+
}
|
|
477
|
+
if await self.wallet_state_manager.does_coin_belong_to_wallet(coin, self.id(), hint_dict=hint_dict):
|
|
478
|
+
addition_amount += coin.amount
|
|
479
|
+
|
|
480
|
+
return uint64(addition_amount)
|
|
481
|
+
|
|
482
|
+
async def get_cat_spendable_coins(self, records: Optional[set[WalletCoinRecord]] = None) -> list[WalletCoinRecord]:
|
|
483
|
+
result: list[WalletCoinRecord] = []
|
|
484
|
+
|
|
485
|
+
record_list: set[WalletCoinRecord] = await self.wallet_state_manager.get_spendable_coins_for_wallet(
|
|
486
|
+
self.id(), records
|
|
487
|
+
)
|
|
488
|
+
|
|
489
|
+
for record in record_list:
|
|
490
|
+
lineage = await self.get_lineage_proof_for_coin(record.coin)
|
|
491
|
+
if lineage is not None and not lineage.is_none():
|
|
492
|
+
result.append(record)
|
|
493
|
+
|
|
494
|
+
return list(await self.get_max_spendable_coins(set(result)))
|
|
495
|
+
|
|
496
|
+
async def select_coins(
|
|
497
|
+
self,
|
|
498
|
+
amount: uint64,
|
|
499
|
+
action_scope: WalletActionScope,
|
|
500
|
+
) -> set[Coin]:
|
|
501
|
+
"""
|
|
502
|
+
Returns a set of coins that can be used for generating a new transaction.
|
|
503
|
+
Note: Must be called under wallet state manager lock
|
|
504
|
+
"""
|
|
505
|
+
spendable_amount: uint128 = await self.get_spendable_balance()
|
|
506
|
+
spendable_coins: list[WalletCoinRecord] = await self.get_cat_spendable_coins()
|
|
507
|
+
|
|
508
|
+
# Try to use coins from the store, if there isn't enough of "unused"
|
|
509
|
+
# coins use change coins that are not confirmed yet
|
|
510
|
+
unconfirmed_removals: dict[bytes32, Coin] = await self.wallet_state_manager.unconfirmed_removals_for_wallet(
|
|
511
|
+
self.id()
|
|
512
|
+
)
|
|
513
|
+
async with action_scope.use() as interface:
|
|
514
|
+
coins = await select_coins(
|
|
515
|
+
spendable_amount,
|
|
516
|
+
action_scope.config.adjust_for_side_effects(interface.side_effects).tx_config.coin_selection_config,
|
|
517
|
+
spendable_coins,
|
|
518
|
+
unconfirmed_removals,
|
|
519
|
+
self.log,
|
|
520
|
+
uint128(amount),
|
|
521
|
+
)
|
|
522
|
+
interface.side_effects.selected_coins.extend([*coins])
|
|
523
|
+
assert sum(c.amount for c in coins) >= amount
|
|
524
|
+
return coins
|
|
525
|
+
|
|
526
|
+
async def inner_puzzle_for_cat_puzhash(self, cat_hash: bytes32) -> Program:
|
|
527
|
+
record: Optional[
|
|
528
|
+
DerivationRecord
|
|
529
|
+
] = await self.wallet_state_manager.puzzle_store.get_derivation_record_for_puzzle_hash(cat_hash)
|
|
530
|
+
if record is None:
|
|
531
|
+
raise RuntimeError(f"Missing Derivation Record for CAT puzzle_hash {cat_hash}")
|
|
532
|
+
inner_puzzle: Program = self.standard_wallet.puzzle_for_pk(record.pubkey)
|
|
533
|
+
return inner_puzzle
|
|
534
|
+
|
|
535
|
+
async def convert_puzzle_hash(self, puzzle_hash: bytes32) -> bytes32:
|
|
536
|
+
record: Optional[
|
|
537
|
+
DerivationRecord
|
|
538
|
+
] = await self.wallet_state_manager.puzzle_store.get_derivation_record_for_puzzle_hash(puzzle_hash)
|
|
539
|
+
if record is None:
|
|
540
|
+
return puzzle_hash # TODO: check if we have a test for this case!
|
|
541
|
+
else:
|
|
542
|
+
return (await self.inner_puzzle_for_cat_puzhash(puzzle_hash)).get_tree_hash()
|
|
543
|
+
|
|
544
|
+
async def get_lineage_proof_for_coin(self, coin: Coin) -> Optional[LineageProof]:
|
|
545
|
+
return await self.lineage_store.get_lineage_proof(coin.parent_coin_info)
|
|
546
|
+
|
|
547
|
+
async def create_tandem_xch_tx(
|
|
548
|
+
self,
|
|
549
|
+
fee: uint64,
|
|
550
|
+
amount_to_claim: uint64,
|
|
551
|
+
action_scope: WalletActionScope,
|
|
552
|
+
extra_conditions: tuple[Condition, ...] = tuple(),
|
|
553
|
+
) -> Optional[AssertCoinAnnouncement]:
|
|
554
|
+
"""
|
|
555
|
+
This function creates a non-CAT transaction to pay fees, contribute funds for issuance, and absorb melt value.
|
|
556
|
+
It is meant to be called in `generate_unsigned_spendbundle` and as such should be called under the
|
|
557
|
+
wallet_state_manager lock
|
|
558
|
+
"""
|
|
559
|
+
announcement: Optional[AssertCoinAnnouncement] = None
|
|
560
|
+
async with self.wallet_state_manager.new_action_scope(
|
|
561
|
+
action_scope.config.tx_config, push=False
|
|
562
|
+
) as inner_action_scope:
|
|
563
|
+
if fee > amount_to_claim:
|
|
564
|
+
chia_coins = await self.standard_wallet.select_coins(
|
|
565
|
+
fee,
|
|
566
|
+
action_scope,
|
|
567
|
+
)
|
|
568
|
+
origin_id = next(iter(chia_coins)).name()
|
|
569
|
+
await self.standard_wallet.generate_signed_transaction(
|
|
570
|
+
uint64(0),
|
|
571
|
+
(await self.standard_wallet.get_puzzle_hash(not action_scope.config.tx_config.reuse_puzhash)),
|
|
572
|
+
inner_action_scope,
|
|
573
|
+
fee=uint64(fee - amount_to_claim),
|
|
574
|
+
coins=chia_coins,
|
|
575
|
+
origin_id=origin_id, # We specify this so that we know the coin that is making the announcement
|
|
576
|
+
negative_change_allowed=False,
|
|
577
|
+
extra_conditions=extra_conditions,
|
|
578
|
+
)
|
|
579
|
+
else:
|
|
580
|
+
chia_coins = await self.standard_wallet.select_coins(
|
|
581
|
+
fee,
|
|
582
|
+
action_scope,
|
|
583
|
+
)
|
|
584
|
+
origin_id = next(iter(chia_coins)).name()
|
|
585
|
+
selected_amount = sum(c.amount for c in chia_coins)
|
|
586
|
+
await self.standard_wallet.generate_signed_transaction(
|
|
587
|
+
uint64(selected_amount + amount_to_claim - fee),
|
|
588
|
+
(await self.standard_wallet.get_puzzle_hash(not action_scope.config.tx_config.reuse_puzhash)),
|
|
589
|
+
inner_action_scope,
|
|
590
|
+
coins=chia_coins,
|
|
591
|
+
negative_change_allowed=True,
|
|
592
|
+
extra_conditions=extra_conditions,
|
|
593
|
+
)
|
|
594
|
+
|
|
595
|
+
message = None
|
|
596
|
+
for tx in inner_action_scope.side_effects.transactions:
|
|
597
|
+
if tx.spend_bundle is None:
|
|
598
|
+
continue
|
|
599
|
+
for spend in tx.spend_bundle.coin_spends:
|
|
600
|
+
if spend.coin.name() == origin_id:
|
|
601
|
+
conditions = spend.puzzle_reveal.to_program().run(spend.solution.to_program()).as_python()
|
|
602
|
+
for condition in conditions:
|
|
603
|
+
if condition[0] == ConditionOpcode.CREATE_COIN_ANNOUNCEMENT:
|
|
604
|
+
message = condition[1]
|
|
605
|
+
|
|
606
|
+
assert message is not None
|
|
607
|
+
announcement = AssertCoinAnnouncement(asserted_id=origin_id, asserted_msg=message)
|
|
608
|
+
|
|
609
|
+
async with action_scope.use() as interface:
|
|
610
|
+
interface.side_effects.transactions.extend(inner_action_scope.side_effects.transactions)
|
|
611
|
+
|
|
612
|
+
return announcement
|
|
613
|
+
|
|
614
|
+
async def generate_unsigned_spendbundle(
|
|
615
|
+
self,
|
|
616
|
+
payments: list[Payment],
|
|
617
|
+
action_scope: WalletActionScope,
|
|
618
|
+
fee: uint64 = uint64(0),
|
|
619
|
+
cat_discrepancy: Optional[tuple[int, Program, Program]] = None, # (extra_delta, tail_reveal, tail_solution)
|
|
620
|
+
coins: Optional[set[Coin]] = None,
|
|
621
|
+
extra_conditions: tuple[Condition, ...] = tuple(),
|
|
622
|
+
) -> WalletSpendBundle:
|
|
623
|
+
if cat_discrepancy is not None:
|
|
624
|
+
extra_delta, tail_reveal, tail_solution = cat_discrepancy
|
|
625
|
+
else:
|
|
626
|
+
extra_delta, tail_reveal, tail_solution = 0, Program.to([]), Program.to([])
|
|
627
|
+
payment_amount: int = sum(p.amount for p in payments)
|
|
628
|
+
starting_amount: int = payment_amount - extra_delta
|
|
629
|
+
if coins is None:
|
|
630
|
+
cat_coins = await self.select_coins(
|
|
631
|
+
uint64(starting_amount),
|
|
632
|
+
action_scope,
|
|
633
|
+
)
|
|
634
|
+
else:
|
|
635
|
+
cat_coins = coins
|
|
636
|
+
|
|
637
|
+
selected_cat_amount = sum(c.amount for c in cat_coins)
|
|
638
|
+
assert selected_cat_amount >= starting_amount
|
|
639
|
+
|
|
640
|
+
# Figure out if we need to absorb/melt some XCH as part of this
|
|
641
|
+
regular_chia_to_claim: int = 0
|
|
642
|
+
if payment_amount > starting_amount:
|
|
643
|
+
fee = uint64(fee + payment_amount - starting_amount)
|
|
644
|
+
elif payment_amount < starting_amount:
|
|
645
|
+
regular_chia_to_claim = payment_amount
|
|
646
|
+
|
|
647
|
+
need_chia_transaction = (fee > 0 or regular_chia_to_claim > 0) and (fee - regular_chia_to_claim != 0)
|
|
648
|
+
|
|
649
|
+
# Calculate standard puzzle solutions
|
|
650
|
+
change = selected_cat_amount - starting_amount
|
|
651
|
+
primaries = payments.copy()
|
|
652
|
+
|
|
653
|
+
if change > 0:
|
|
654
|
+
derivation_record = await self.wallet_state_manager.puzzle_store.get_derivation_record_for_puzzle_hash(
|
|
655
|
+
next(iter(cat_coins)).puzzle_hash
|
|
656
|
+
)
|
|
657
|
+
if derivation_record is not None and action_scope.config.tx_config.reuse_puzhash:
|
|
658
|
+
change_puzhash = self.standard_wallet.puzzle_hash_for_pk(derivation_record.pubkey)
|
|
659
|
+
for payment in payments:
|
|
660
|
+
if change_puzhash == payment.puzzle_hash and change == payment.amount:
|
|
661
|
+
# We cannot create two coins has same id, create a new puzhash for the change
|
|
662
|
+
change_puzhash = await self.standard_wallet.get_puzzle_hash(new=True)
|
|
663
|
+
break
|
|
664
|
+
else:
|
|
665
|
+
change_puzhash = await self.standard_wallet.get_puzzle_hash(new=True)
|
|
666
|
+
primaries.append(Payment(change_puzhash, uint64(change), [change_puzhash]))
|
|
667
|
+
|
|
668
|
+
# Loop through the coins we've selected and gather the information we need to spend them
|
|
669
|
+
spendable_cat_list = []
|
|
670
|
+
first = True
|
|
671
|
+
announcement: CreateCoinAnnouncement
|
|
672
|
+
|
|
673
|
+
for coin in cat_coins:
|
|
674
|
+
if cat_discrepancy is not None:
|
|
675
|
+
cat_condition = UnknownCondition(
|
|
676
|
+
opcode=Program.to(51),
|
|
677
|
+
args=[
|
|
678
|
+
Program.to(None),
|
|
679
|
+
Program.to(-113),
|
|
680
|
+
tail_reveal,
|
|
681
|
+
tail_solution,
|
|
682
|
+
],
|
|
683
|
+
)
|
|
684
|
+
if first:
|
|
685
|
+
extra_conditions = (*extra_conditions, cat_condition)
|
|
686
|
+
if first:
|
|
687
|
+
first = False
|
|
688
|
+
announcement = CreateCoinAnnouncement(std_hash(b"".join([c.name() for c in cat_coins])), coin.name())
|
|
689
|
+
if need_chia_transaction:
|
|
690
|
+
if fee > regular_chia_to_claim:
|
|
691
|
+
await self.create_tandem_xch_tx(
|
|
692
|
+
fee,
|
|
693
|
+
uint64(regular_chia_to_claim),
|
|
694
|
+
action_scope,
|
|
695
|
+
extra_conditions=(announcement.corresponding_assertion(),),
|
|
696
|
+
)
|
|
697
|
+
innersol = self.standard_wallet.make_solution(
|
|
698
|
+
primaries=primaries,
|
|
699
|
+
conditions=(*extra_conditions, announcement),
|
|
700
|
+
)
|
|
701
|
+
elif regular_chia_to_claim > fee: # pragma: no cover
|
|
702
|
+
xch_announcement = await self.create_tandem_xch_tx(
|
|
703
|
+
fee,
|
|
704
|
+
uint64(regular_chia_to_claim),
|
|
705
|
+
action_scope,
|
|
706
|
+
)
|
|
707
|
+
assert xch_announcement is not None
|
|
708
|
+
innersol = self.standard_wallet.make_solution(
|
|
709
|
+
primaries=primaries,
|
|
710
|
+
conditions=(*extra_conditions, xch_announcement, announcement),
|
|
711
|
+
)
|
|
712
|
+
else:
|
|
713
|
+
# TODO: what about when they are equal?
|
|
714
|
+
raise Exception("Equality not handled")
|
|
715
|
+
else:
|
|
716
|
+
innersol = self.standard_wallet.make_solution(
|
|
717
|
+
primaries=primaries,
|
|
718
|
+
conditions=(*extra_conditions, announcement),
|
|
719
|
+
)
|
|
720
|
+
else:
|
|
721
|
+
innersol = self.standard_wallet.make_solution(
|
|
722
|
+
primaries=[], conditions=(announcement.corresponding_assertion(),)
|
|
723
|
+
)
|
|
724
|
+
inner_puzzle = await self.inner_puzzle_for_cat_puzhash(coin.puzzle_hash)
|
|
725
|
+
lineage_proof = await self.get_lineage_proof_for_coin(coin)
|
|
726
|
+
assert lineage_proof is not None
|
|
727
|
+
new_spendable_cat = SpendableCAT(
|
|
728
|
+
coin,
|
|
729
|
+
self.cat_info.limitations_program_hash,
|
|
730
|
+
inner_puzzle,
|
|
731
|
+
innersol,
|
|
732
|
+
limitations_solution=tail_solution,
|
|
733
|
+
extra_delta=extra_delta,
|
|
734
|
+
lineage_proof=lineage_proof,
|
|
735
|
+
limitations_program_reveal=tail_reveal,
|
|
736
|
+
)
|
|
737
|
+
spendable_cat_list.append(new_spendable_cat)
|
|
738
|
+
|
|
739
|
+
cat_spend_bundle = unsigned_spend_bundle_for_spendable_cats(CAT_MOD, spendable_cat_list)
|
|
740
|
+
|
|
741
|
+
return cat_spend_bundle
|
|
742
|
+
|
|
743
|
+
async def generate_signed_transaction(
|
|
744
|
+
self,
|
|
745
|
+
amounts: list[uint64],
|
|
746
|
+
puzzle_hashes: list[bytes32],
|
|
747
|
+
action_scope: WalletActionScope,
|
|
748
|
+
fee: uint64 = uint64(0),
|
|
749
|
+
coins: Optional[set[Coin]] = None,
|
|
750
|
+
memos: Optional[list[list[bytes]]] = None,
|
|
751
|
+
extra_conditions: tuple[Condition, ...] = tuple(),
|
|
752
|
+
**kwargs: Unpack[GSTOptionalArgs],
|
|
753
|
+
) -> None:
|
|
754
|
+
# (extra_delta, tail_reveal, tail_solution)
|
|
755
|
+
cat_discrepancy: Optional[tuple[int, Program, Program]] = kwargs.get("cat_discrepancy", None)
|
|
756
|
+
if memos is None:
|
|
757
|
+
memos = [[] for _ in range(len(puzzle_hashes))]
|
|
758
|
+
|
|
759
|
+
if not (len(memos) == len(puzzle_hashes) == len(amounts)):
|
|
760
|
+
raise ValueError("Memos, puzzle_hashes, and amounts must have the same length")
|
|
761
|
+
|
|
762
|
+
payments = []
|
|
763
|
+
for amount, puzhash, memo_list in zip(amounts, puzzle_hashes, memos):
|
|
764
|
+
memos_with_hint: list[bytes] = [puzhash]
|
|
765
|
+
memos_with_hint.extend(memo_list)
|
|
766
|
+
payments.append(Payment(puzhash, amount, memos_with_hint))
|
|
767
|
+
|
|
768
|
+
payment_sum = sum(p.amount for p in payments)
|
|
769
|
+
spend_bundle = await self.generate_unsigned_spendbundle(
|
|
770
|
+
payments,
|
|
771
|
+
action_scope,
|
|
772
|
+
fee,
|
|
773
|
+
cat_discrepancy=cat_discrepancy, # (extra_delta, tail_reveal, tail_solution)
|
|
774
|
+
coins=coins,
|
|
775
|
+
extra_conditions=extra_conditions,
|
|
776
|
+
)
|
|
777
|
+
|
|
778
|
+
async with action_scope.use() as interface:
|
|
779
|
+
other_tx_removals: set[Coin] = {
|
|
780
|
+
removal for tx in interface.side_effects.transactions for removal in tx.removals
|
|
781
|
+
}
|
|
782
|
+
other_tx_additions: set[Coin] = {
|
|
783
|
+
removal for tx in interface.side_effects.transactions for removal in tx.additions
|
|
784
|
+
}
|
|
785
|
+
interface.side_effects.transactions.append(
|
|
786
|
+
TransactionRecord(
|
|
787
|
+
confirmed_at_height=uint32(0),
|
|
788
|
+
created_at_time=uint64(int(time.time())),
|
|
789
|
+
to_puzzle_hash=puzzle_hashes[0],
|
|
790
|
+
amount=uint64(payment_sum),
|
|
791
|
+
fee_amount=fee,
|
|
792
|
+
confirmed=False,
|
|
793
|
+
sent=uint32(0),
|
|
794
|
+
spend_bundle=spend_bundle,
|
|
795
|
+
additions=list(set(spend_bundle.additions()) - other_tx_additions),
|
|
796
|
+
removals=list(set(spend_bundle.removals()) - other_tx_removals),
|
|
797
|
+
wallet_id=self.id(),
|
|
798
|
+
sent_to=[],
|
|
799
|
+
trade_id=None,
|
|
800
|
+
type=uint32(TransactionType.OUTGOING_TX.value),
|
|
801
|
+
name=spend_bundle.name(),
|
|
802
|
+
memos=list(compute_memos(spend_bundle).items()),
|
|
803
|
+
valid_times=parse_timelock_info(extra_conditions),
|
|
804
|
+
)
|
|
805
|
+
)
|
|
806
|
+
|
|
807
|
+
async def add_lineage(self, name: bytes32, lineage: Optional[LineageProof]) -> None:
|
|
808
|
+
"""
|
|
809
|
+
Lineage proofs are stored as a list of parent coins and the lineage proof you will need if they are the
|
|
810
|
+
parent of the coin you are trying to spend. 'If I'm your parent, here's the info you need to spend yourself'
|
|
811
|
+
"""
|
|
812
|
+
self.log.info(f"Adding parent {name.hex()}: {lineage}")
|
|
813
|
+
if lineage is not None:
|
|
814
|
+
await self.lineage_store.add_lineage_proof(name, lineage)
|
|
815
|
+
|
|
816
|
+
async def remove_lineage(self, name: bytes32) -> None:
|
|
817
|
+
self.log.info(f"Removing parent {name} (probably had a non-CAT parent)")
|
|
818
|
+
await self.lineage_store.remove_lineage_proof(name)
|
|
819
|
+
|
|
820
|
+
async def save_info(self, cat_info: CATInfo) -> None:
|
|
821
|
+
self.cat_info = cat_info
|
|
822
|
+
current_info = self.wallet_info
|
|
823
|
+
data_str = bytes(cat_info).hex()
|
|
824
|
+
wallet_info = WalletInfo(current_info.id, current_info.name, current_info.type, data_str)
|
|
825
|
+
self.wallet_info = wallet_info
|
|
826
|
+
await self.wallet_state_manager.user_store.update_wallet(wallet_info)
|
|
827
|
+
|
|
828
|
+
async def match_puzzle_info(self, puzzle_driver: PuzzleInfo) -> bool:
|
|
829
|
+
return (
|
|
830
|
+
AssetType(puzzle_driver.type()) == AssetType.CAT
|
|
831
|
+
and puzzle_driver["tail"] == bytes.fromhex(self.get_asset_id())
|
|
832
|
+
and puzzle_driver.also() is None
|
|
833
|
+
)
|
|
834
|
+
|
|
835
|
+
async def get_puzzle_info(self, asset_id: bytes32) -> PuzzleInfo:
|
|
836
|
+
return PuzzleInfo({"type": AssetType.CAT.value, "tail": "0x" + self.get_asset_id()})
|
|
837
|
+
|
|
838
|
+
async def get_coins_to_offer(
|
|
839
|
+
self,
|
|
840
|
+
asset_id: Optional[bytes32],
|
|
841
|
+
amount: uint64,
|
|
842
|
+
action_scope: WalletActionScope,
|
|
843
|
+
) -> set[Coin]:
|
|
844
|
+
balance = await self.get_confirmed_balance()
|
|
845
|
+
if balance < amount:
|
|
846
|
+
raise Exception(f"insufficient funds in wallet {self.id()}")
|
|
847
|
+
# We need to sandbox this because this method isn't supposed to lock up the coins
|
|
848
|
+
async with self.wallet_state_manager.new_action_scope(action_scope.config.tx_config) as sandbox:
|
|
849
|
+
return await self.select_coins(amount, sandbox)
|
|
850
|
+
|
|
851
|
+
async def match_hinted_coin(self, coin: Coin, hint: bytes32) -> bool:
|
|
852
|
+
return (
|
|
853
|
+
construct_cat_puzzle(CAT_MOD, self.cat_info.limitations_program_hash, hint).get_tree_hash_precalc(hint)
|
|
854
|
+
== coin.puzzle_hash
|
|
855
|
+
)
|