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,1102 @@
|
|
|
1
|
+
from __future__ import annotations
|
|
2
|
+
|
|
3
|
+
import dataclasses
|
|
4
|
+
import logging
|
|
5
|
+
import time
|
|
6
|
+
from collections import deque
|
|
7
|
+
from typing import TYPE_CHECKING, Any, Optional, Union
|
|
8
|
+
|
|
9
|
+
from typing_extensions import Literal
|
|
10
|
+
|
|
11
|
+
from chia.data_layer.data_layer_wallet import DataLayerWallet
|
|
12
|
+
from chia.protocols.wallet_protocol import CoinState
|
|
13
|
+
from chia.server.ws_connection import WSChiaConnection
|
|
14
|
+
from chia.types.blockchain_format.coin import Coin, coin_as_list
|
|
15
|
+
from chia.types.blockchain_format.program import Program
|
|
16
|
+
from chia.types.blockchain_format.sized_bytes import bytes32
|
|
17
|
+
from chia.types.spend_bundle import estimate_fees
|
|
18
|
+
from chia.util.db_wrapper import DBWrapper2
|
|
19
|
+
from chia.util.hash import std_hash
|
|
20
|
+
from chia.util.ints import uint32, uint64
|
|
21
|
+
from chia.wallet.cat_wallet.cat_wallet import CATWallet
|
|
22
|
+
from chia.wallet.conditions import (
|
|
23
|
+
AssertCoinAnnouncement,
|
|
24
|
+
Condition,
|
|
25
|
+
ConditionValidTimes,
|
|
26
|
+
CreateCoinAnnouncement,
|
|
27
|
+
parse_conditions_non_consensus,
|
|
28
|
+
parse_timelock_info,
|
|
29
|
+
)
|
|
30
|
+
from chia.wallet.db_wallet.db_wallet_puzzles import ACS_MU_PH
|
|
31
|
+
from chia.wallet.nft_wallet.nft_wallet import NFTWallet
|
|
32
|
+
from chia.wallet.outer_puzzles import AssetType
|
|
33
|
+
from chia.wallet.payment import Payment
|
|
34
|
+
from chia.wallet.puzzle_drivers import PuzzleInfo, Solver
|
|
35
|
+
from chia.wallet.puzzles.load_clvm import load_clvm_maybe_recompile
|
|
36
|
+
from chia.wallet.trade_record import TradeRecord
|
|
37
|
+
from chia.wallet.trading.offer import NotarizedPayment, Offer
|
|
38
|
+
from chia.wallet.trading.trade_status import TradeStatus
|
|
39
|
+
from chia.wallet.trading.trade_store import TradeStore
|
|
40
|
+
from chia.wallet.transaction_record import TransactionRecord
|
|
41
|
+
from chia.wallet.uncurried_puzzle import uncurry_puzzle
|
|
42
|
+
from chia.wallet.util.compute_hints import compute_spend_hints_and_additions
|
|
43
|
+
from chia.wallet.util.query_filter import HashFilter
|
|
44
|
+
from chia.wallet.util.transaction_type import TransactionType
|
|
45
|
+
from chia.wallet.util.wallet_types import WalletType
|
|
46
|
+
from chia.wallet.vc_wallet.cr_cat_drivers import ProofsChecker, construct_pending_approval_state
|
|
47
|
+
from chia.wallet.vc_wallet.vc_wallet import VCWallet
|
|
48
|
+
from chia.wallet.wallet import Wallet
|
|
49
|
+
from chia.wallet.wallet_action_scope import WalletActionScope
|
|
50
|
+
from chia.wallet.wallet_coin_record import WalletCoinRecord
|
|
51
|
+
from chia.wallet.wallet_protocol import WalletProtocol
|
|
52
|
+
|
|
53
|
+
if TYPE_CHECKING:
|
|
54
|
+
from chia.wallet.wallet_state_manager import WalletStateManager
|
|
55
|
+
from chia.wallet.wallet_spend_bundle import WalletSpendBundle
|
|
56
|
+
|
|
57
|
+
OFFER_MOD = load_clvm_maybe_recompile("settlement_payments.clsp")
|
|
58
|
+
|
|
59
|
+
|
|
60
|
+
class TradeManager:
|
|
61
|
+
"""
|
|
62
|
+
This class is a driver for creating and accepting settlement_payments.clsp style offers.
|
|
63
|
+
|
|
64
|
+
By default, standard XCH is supported but to support other types of assets you must implement certain functions on
|
|
65
|
+
the asset's wallet as well as create a driver for its puzzle(s). Here is a guide to integrating a new types of
|
|
66
|
+
assets with this trade manager:
|
|
67
|
+
|
|
68
|
+
Puzzle Drivers:
|
|
69
|
+
- See chia/wallet/outer_puzzles.py for a full description of how to build these
|
|
70
|
+
- The `solve` method must be able to be solved by a Solver that looks like this:
|
|
71
|
+
Solver(
|
|
72
|
+
{
|
|
73
|
+
"coin": bytes
|
|
74
|
+
"parent_spend": bytes
|
|
75
|
+
"siblings": list[bytes] # other coins of the same type being offered
|
|
76
|
+
"sibling_spends": list[bytes] # The parent spends for the siblings
|
|
77
|
+
"sibling_puzzles": list[Program] # The inner puzzles of the siblings (always OFFER_MOD)
|
|
78
|
+
"sibling_solutions": list[Program] # The inner solution of the siblings
|
|
79
|
+
}
|
|
80
|
+
)
|
|
81
|
+
|
|
82
|
+
Wallet:
|
|
83
|
+
- Segments in this code that call general wallet methods are highlighted by comments: # ATTENTION: new wallets
|
|
84
|
+
- To be able to be traded, a wallet must implement these methods on itself:
|
|
85
|
+
- generate_signed_transaction(...) -> list[TransactionRecord] (See cat_wallet.py for full API)
|
|
86
|
+
- convert_puzzle_hash(puzzle_hash: bytes32) -> bytes32 # Converts a puzzlehash from outer to inner puzzle
|
|
87
|
+
- get_puzzle_info(asset_id: bytes32) -> PuzzleInfo
|
|
88
|
+
- get_coins_to_offer(asset_id: bytes32, amount: uint64) -> set[Coin]
|
|
89
|
+
- If you would like assets from your wallet to be referenced with just a wallet ID, you must also implement:
|
|
90
|
+
- get_asset_id() -> bytes32
|
|
91
|
+
- Finally, you must make sure that your wallet will respond appropriately when these WSM methods are called:
|
|
92
|
+
- get_wallet_for_puzzle_info(puzzle_info: PuzzleInfo) -> <Your wallet>
|
|
93
|
+
- create_wallet_for_puzzle_info(..., puzzle_info: PuzzleInfo) -> <Your wallet> (See cat_wallet.py for full API)
|
|
94
|
+
- get_wallet_for_asset_id(asset_id: bytes32) -> <Your wallet>
|
|
95
|
+
"""
|
|
96
|
+
|
|
97
|
+
wallet_state_manager: WalletStateManager
|
|
98
|
+
log: logging.Logger
|
|
99
|
+
trade_store: TradeStore
|
|
100
|
+
most_recently_deserialized_trade: Optional[tuple[bytes32, Offer]]
|
|
101
|
+
|
|
102
|
+
@staticmethod
|
|
103
|
+
async def create(
|
|
104
|
+
wallet_state_manager: Any,
|
|
105
|
+
db_wrapper: DBWrapper2,
|
|
106
|
+
name: Optional[str] = None,
|
|
107
|
+
) -> TradeManager:
|
|
108
|
+
self = TradeManager()
|
|
109
|
+
if name:
|
|
110
|
+
self.log = logging.getLogger(name)
|
|
111
|
+
else:
|
|
112
|
+
self.log = logging.getLogger(__name__)
|
|
113
|
+
|
|
114
|
+
self.wallet_state_manager = wallet_state_manager
|
|
115
|
+
self.trade_store = await TradeStore.create(db_wrapper)
|
|
116
|
+
self.most_recently_deserialized_trade = None
|
|
117
|
+
return self
|
|
118
|
+
|
|
119
|
+
async def get_offers_with_status(self, status: TradeStatus) -> list[TradeRecord]:
|
|
120
|
+
records = await self.trade_store.get_trade_record_with_status(status)
|
|
121
|
+
return records
|
|
122
|
+
|
|
123
|
+
async def get_coins_of_interest(
|
|
124
|
+
self,
|
|
125
|
+
) -> set[bytes32]:
|
|
126
|
+
"""
|
|
127
|
+
Returns list of coins we want to check if they are included in filter,
|
|
128
|
+
These will include coins that belong to us and coins that that on other side of treade
|
|
129
|
+
"""
|
|
130
|
+
coin_ids = await self.trade_store.get_coin_ids_of_interest_with_trade_statuses(
|
|
131
|
+
trade_statuses=[TradeStatus.PENDING_ACCEPT, TradeStatus.PENDING_CONFIRM, TradeStatus.PENDING_CANCEL]
|
|
132
|
+
)
|
|
133
|
+
return coin_ids
|
|
134
|
+
|
|
135
|
+
async def get_trades_by_coin(self, coin: Coin) -> list[TradeRecord]:
|
|
136
|
+
all_trades = await self.get_all_trades()
|
|
137
|
+
trades_by_coin = []
|
|
138
|
+
for trade in all_trades:
|
|
139
|
+
if trade.status == TradeStatus.CANCELLED.value:
|
|
140
|
+
continue
|
|
141
|
+
if coin in trade.coins_of_interest:
|
|
142
|
+
trades_by_coin.append(trade)
|
|
143
|
+
return trades_by_coin
|
|
144
|
+
|
|
145
|
+
async def coins_of_interest_farmed(
|
|
146
|
+
self, coin_state: CoinState, fork_height: Optional[uint32], peer: WSChiaConnection
|
|
147
|
+
) -> None:
|
|
148
|
+
"""
|
|
149
|
+
If both our coins and other coins in trade got removed that means that trade was successfully executed
|
|
150
|
+
If coins from other side of trade got farmed without ours, that means that trade failed because either someone
|
|
151
|
+
else completed trade or other side of trade canceled the trade by doing a spend.
|
|
152
|
+
If our coins got farmed but coins from other side didn't, we successfully canceled trade by spending inputs.
|
|
153
|
+
"""
|
|
154
|
+
self.log.info(f"coins_of_interest_farmed: {coin_state}")
|
|
155
|
+
trades = await self.get_trades_by_coin(coin_state.coin)
|
|
156
|
+
for trade in trades:
|
|
157
|
+
if trade is None:
|
|
158
|
+
self.log.error(f"Coin: {coin_state.coin}, not in any trade")
|
|
159
|
+
continue
|
|
160
|
+
if coin_state.spent_height is None:
|
|
161
|
+
self.log.error(f"Coin: {coin_state.coin}, has not been spent so trade can remain valid")
|
|
162
|
+
# Then let's filter the offer into coins that WE offered
|
|
163
|
+
if (
|
|
164
|
+
self.most_recently_deserialized_trade is not None
|
|
165
|
+
and trade.trade_id == self.most_recently_deserialized_trade[0]
|
|
166
|
+
):
|
|
167
|
+
offer = self.most_recently_deserialized_trade[1]
|
|
168
|
+
else:
|
|
169
|
+
offer = Offer.from_bytes(trade.offer)
|
|
170
|
+
self.most_recently_deserialized_trade = (trade.trade_id, offer)
|
|
171
|
+
primary_coin_ids = [c.name() for c in offer.removals()]
|
|
172
|
+
# TODO: Add `WalletCoinStore.get_coins`.
|
|
173
|
+
result = await self.wallet_state_manager.coin_store.get_coin_records(
|
|
174
|
+
coin_id_filter=HashFilter.include(primary_coin_ids)
|
|
175
|
+
)
|
|
176
|
+
our_primary_coins: list[Coin] = [cr.coin for cr in result.records]
|
|
177
|
+
our_additions: list[Coin] = list(
|
|
178
|
+
filter(lambda c: offer.get_root_removal(c) in our_primary_coins, offer.additions())
|
|
179
|
+
)
|
|
180
|
+
our_addition_ids: list[bytes32] = [c.name() for c in our_additions]
|
|
181
|
+
|
|
182
|
+
# And get all relevant coin states
|
|
183
|
+
coin_states = await self.wallet_state_manager.wallet_node.get_coin_state(
|
|
184
|
+
our_addition_ids,
|
|
185
|
+
peer=peer,
|
|
186
|
+
fork_height=fork_height,
|
|
187
|
+
)
|
|
188
|
+
assert coin_states is not None
|
|
189
|
+
coin_state_names: list[bytes32] = [cs.coin.name() for cs in coin_states]
|
|
190
|
+
# If any of our settlement_payments were spent, this offer was a success!
|
|
191
|
+
if set(our_addition_ids) == set(coin_state_names):
|
|
192
|
+
height = coin_state.spent_height
|
|
193
|
+
assert height is not None
|
|
194
|
+
await self.trade_store.set_status(trade.trade_id, TradeStatus.CONFIRMED, index=height)
|
|
195
|
+
tx_records: list[TransactionRecord] = await self.calculate_tx_records_for_offer(offer, False)
|
|
196
|
+
for tx in tx_records:
|
|
197
|
+
if TradeStatus(trade.status) == TradeStatus.PENDING_ACCEPT:
|
|
198
|
+
await self.wallet_state_manager.add_transaction(
|
|
199
|
+
dataclasses.replace(tx, confirmed_at_height=height, confirmed=True)
|
|
200
|
+
)
|
|
201
|
+
|
|
202
|
+
self.log.info(f"Trade with id: {trade.trade_id} confirmed at height: {height}")
|
|
203
|
+
else:
|
|
204
|
+
# In any other scenario this trade failed
|
|
205
|
+
await self.wallet_state_manager.delete_trade_transactions(trade.trade_id)
|
|
206
|
+
if trade.status == TradeStatus.PENDING_CANCEL.value:
|
|
207
|
+
await self.trade_store.set_status(trade.trade_id, TradeStatus.CANCELLED)
|
|
208
|
+
self.log.info(f"Trade with id: {trade.trade_id} canceled")
|
|
209
|
+
elif trade.status == TradeStatus.PENDING_CONFIRM.value:
|
|
210
|
+
await self.trade_store.set_status(trade.trade_id, TradeStatus.FAILED)
|
|
211
|
+
self.log.warning(f"Trade with id: {trade.trade_id} failed")
|
|
212
|
+
|
|
213
|
+
async def get_locked_coins(self) -> dict[bytes32, WalletCoinRecord]:
|
|
214
|
+
"""Returns a dictionary of confirmed coins that are locked by a trade."""
|
|
215
|
+
all_pending = []
|
|
216
|
+
pending_accept = await self.get_offers_with_status(TradeStatus.PENDING_ACCEPT)
|
|
217
|
+
pending_confirm = await self.get_offers_with_status(TradeStatus.PENDING_CONFIRM)
|
|
218
|
+
pending_cancel = await self.get_offers_with_status(TradeStatus.PENDING_CANCEL)
|
|
219
|
+
all_pending.extend(pending_accept)
|
|
220
|
+
all_pending.extend(pending_confirm)
|
|
221
|
+
all_pending.extend(pending_cancel)
|
|
222
|
+
|
|
223
|
+
coins_of_interest = []
|
|
224
|
+
for trade_offer in all_pending:
|
|
225
|
+
coins_of_interest.extend([c.name() for c in trade_offer.coins_of_interest])
|
|
226
|
+
|
|
227
|
+
# TODO:
|
|
228
|
+
# - No need to get the coin records here, we are only interested in the coin_id on the call site.
|
|
229
|
+
return (
|
|
230
|
+
await self.wallet_state_manager.coin_store.get_coin_records(
|
|
231
|
+
coin_id_filter=HashFilter.include(coins_of_interest)
|
|
232
|
+
)
|
|
233
|
+
).coin_id_to_record
|
|
234
|
+
|
|
235
|
+
async def get_all_trades(self) -> list[TradeRecord]:
|
|
236
|
+
all: list[TradeRecord] = await self.trade_store.get_all_trades()
|
|
237
|
+
return all
|
|
238
|
+
|
|
239
|
+
async def get_trade_by_id(self, trade_id: bytes32) -> Optional[TradeRecord]:
|
|
240
|
+
record = await self.trade_store.get_trade_record(trade_id)
|
|
241
|
+
return record
|
|
242
|
+
|
|
243
|
+
async def fail_pending_offer(self, trade_id: bytes32) -> None:
|
|
244
|
+
await self.trade_store.set_status(trade_id, TradeStatus.FAILED)
|
|
245
|
+
self.wallet_state_manager.state_changed("offer_failed")
|
|
246
|
+
|
|
247
|
+
async def cancel_pending_offers(
|
|
248
|
+
self,
|
|
249
|
+
trade_ids: list[bytes32],
|
|
250
|
+
action_scope: WalletActionScope,
|
|
251
|
+
fee: uint64 = uint64(0),
|
|
252
|
+
secure: bool = True, # Cancel with a transaction on chain
|
|
253
|
+
trade_cache: dict[bytes32, TradeRecord] = {}, # Optional pre-fetched trade records for optimization
|
|
254
|
+
extra_conditions: tuple[Condition, ...] = tuple(),
|
|
255
|
+
) -> None:
|
|
256
|
+
"""This will create a transaction that includes coins that were offered"""
|
|
257
|
+
|
|
258
|
+
# Need to do some pre-figuring of announcements that will be need to be made
|
|
259
|
+
announcement_nonce: bytes32 = std_hash(b"".join(trade_ids))
|
|
260
|
+
trade_records: list[TradeRecord] = []
|
|
261
|
+
all_cancellation_coins: list[list[Coin]] = []
|
|
262
|
+
announcement_creations: deque[CreateCoinAnnouncement] = deque()
|
|
263
|
+
announcement_assertions: deque[AssertCoinAnnouncement] = deque()
|
|
264
|
+
for trade_id in trade_ids:
|
|
265
|
+
if trade_id in trade_cache:
|
|
266
|
+
trade = trade_cache[trade_id]
|
|
267
|
+
else:
|
|
268
|
+
potential_trade = await self.trade_store.get_trade_record(trade_id)
|
|
269
|
+
if potential_trade is None:
|
|
270
|
+
self.log.error(f"Cannot find offer {trade_id.hex()}, skip cancellation.")
|
|
271
|
+
continue
|
|
272
|
+
else:
|
|
273
|
+
trade = potential_trade
|
|
274
|
+
|
|
275
|
+
cancellation_coins = Offer.from_bytes(trade.offer).get_cancellation_coins()
|
|
276
|
+
for coin in cancellation_coins:
|
|
277
|
+
creation = CreateCoinAnnouncement(msg=announcement_nonce, coin_id=coin.name())
|
|
278
|
+
announcement_creations.append(creation)
|
|
279
|
+
announcement_assertions.append(creation.corresponding_assertion())
|
|
280
|
+
|
|
281
|
+
trade_records.append(trade)
|
|
282
|
+
all_cancellation_coins.append(cancellation_coins)
|
|
283
|
+
|
|
284
|
+
# Make every coin assert the announcement from the one before them
|
|
285
|
+
announcement_assertions.rotate(1)
|
|
286
|
+
|
|
287
|
+
all_txs: list[TransactionRecord] = []
|
|
288
|
+
fee_to_pay: uint64 = fee
|
|
289
|
+
for trade, cancellation_coins in zip(trade_records, all_cancellation_coins):
|
|
290
|
+
self.log.info(f"Secure-Cancel pending offer with id trade_id {trade.trade_id.hex()}")
|
|
291
|
+
|
|
292
|
+
if not secure:
|
|
293
|
+
self.wallet_state_manager.state_changed("offer_cancelled")
|
|
294
|
+
await self.trade_store.set_status(trade.trade_id, TradeStatus.CANCELLED)
|
|
295
|
+
continue
|
|
296
|
+
|
|
297
|
+
cancellation_additions: list[Coin] = []
|
|
298
|
+
valid_times: ConditionValidTimes = parse_timelock_info(extra_conditions)
|
|
299
|
+
trades_to_cancel: list[TradeRecord] = []
|
|
300
|
+
for coin in cancellation_coins:
|
|
301
|
+
wallet = await self.wallet_state_manager.get_wallet_for_coin(coin.name())
|
|
302
|
+
|
|
303
|
+
if wallet is None:
|
|
304
|
+
self.log.error(f"Cannot find wallet for offer {trade.trade_id}, skip cancellation.")
|
|
305
|
+
continue
|
|
306
|
+
|
|
307
|
+
new_ph = await wallet.wallet_state_manager.main_wallet.get_puzzle_hash(
|
|
308
|
+
new=(not action_scope.config.tx_config.reuse_puzhash)
|
|
309
|
+
)
|
|
310
|
+
|
|
311
|
+
if len(trade_records) > 1 or len(cancellation_coins) > 1:
|
|
312
|
+
announcement_conditions: tuple[Condition, ...] = (
|
|
313
|
+
announcement_creations.popleft(),
|
|
314
|
+
announcement_assertions.popleft(),
|
|
315
|
+
)
|
|
316
|
+
else:
|
|
317
|
+
announcement_conditions = tuple()
|
|
318
|
+
async with action_scope.use() as interface:
|
|
319
|
+
interface.side_effects.selected_coins.append(coin)
|
|
320
|
+
# This should probably not switch on whether or not we're spending a XCH but it has to for now
|
|
321
|
+
if wallet.type() == WalletType.STANDARD_WALLET:
|
|
322
|
+
assert isinstance(wallet, Wallet)
|
|
323
|
+
if fee_to_pay > coin.amount:
|
|
324
|
+
selected_coins: set[Coin] = await wallet.select_coins(
|
|
325
|
+
uint64(fee_to_pay - coin.amount),
|
|
326
|
+
action_scope,
|
|
327
|
+
)
|
|
328
|
+
selected_coins.add(coin)
|
|
329
|
+
else:
|
|
330
|
+
selected_coins = {coin}
|
|
331
|
+
async with self.wallet_state_manager.new_action_scope(
|
|
332
|
+
action_scope.config.tx_config.override(
|
|
333
|
+
excluded_coin_ids=[],
|
|
334
|
+
),
|
|
335
|
+
push=False,
|
|
336
|
+
) as inner_action_scope:
|
|
337
|
+
await wallet.generate_signed_transaction(
|
|
338
|
+
uint64(sum(c.amount for c in selected_coins) - fee_to_pay),
|
|
339
|
+
new_ph,
|
|
340
|
+
inner_action_scope,
|
|
341
|
+
origin_id=coin.name(),
|
|
342
|
+
fee=fee_to_pay,
|
|
343
|
+
coins=selected_coins,
|
|
344
|
+
extra_conditions=(*extra_conditions, *announcement_conditions),
|
|
345
|
+
)
|
|
346
|
+
else:
|
|
347
|
+
# ATTENTION: new_wallets
|
|
348
|
+
assert isinstance(wallet, (CATWallet, DataLayerWallet, NFTWallet))
|
|
349
|
+
async with self.wallet_state_manager.new_action_scope(
|
|
350
|
+
action_scope.config.tx_config.override(
|
|
351
|
+
excluded_coin_ids=[],
|
|
352
|
+
),
|
|
353
|
+
push=False,
|
|
354
|
+
) as inner_action_scope:
|
|
355
|
+
await wallet.generate_signed_transaction(
|
|
356
|
+
[coin.amount],
|
|
357
|
+
[new_ph],
|
|
358
|
+
inner_action_scope,
|
|
359
|
+
fee=fee_to_pay,
|
|
360
|
+
coins={coin},
|
|
361
|
+
extra_conditions=(*extra_conditions, *announcement_conditions),
|
|
362
|
+
)
|
|
363
|
+
|
|
364
|
+
cancellation_additions.extend(
|
|
365
|
+
[
|
|
366
|
+
add
|
|
367
|
+
for tx in inner_action_scope.side_effects.transactions
|
|
368
|
+
if tx.spend_bundle is not None
|
|
369
|
+
for add in tx.spend_bundle.additions()
|
|
370
|
+
]
|
|
371
|
+
)
|
|
372
|
+
all_txs.extend(inner_action_scope.side_effects.transactions)
|
|
373
|
+
fee_to_pay = uint64(0)
|
|
374
|
+
extra_conditions = tuple()
|
|
375
|
+
|
|
376
|
+
incoming_tx = TransactionRecord(
|
|
377
|
+
confirmed_at_height=uint32(0),
|
|
378
|
+
created_at_time=uint64(int(time.time())),
|
|
379
|
+
to_puzzle_hash=new_ph,
|
|
380
|
+
amount=uint64(coin.amount),
|
|
381
|
+
fee_amount=fee,
|
|
382
|
+
confirmed=False,
|
|
383
|
+
sent=uint32(10),
|
|
384
|
+
spend_bundle=None,
|
|
385
|
+
additions=[],
|
|
386
|
+
removals=[coin],
|
|
387
|
+
wallet_id=wallet.id(),
|
|
388
|
+
sent_to=[],
|
|
389
|
+
trade_id=None,
|
|
390
|
+
type=uint32(TransactionType.INCOMING_TX.value),
|
|
391
|
+
name=cancellation_additions[0].name(),
|
|
392
|
+
memos=[],
|
|
393
|
+
valid_times=valid_times,
|
|
394
|
+
)
|
|
395
|
+
all_txs.append(incoming_tx)
|
|
396
|
+
|
|
397
|
+
# The statuses of trades which offer cancellation coin needs to be set to `PENDING_CANCEL`
|
|
398
|
+
trades_to_cancel.extend(await self.get_trades_by_coin(coin))
|
|
399
|
+
|
|
400
|
+
await self.trade_store.set_status(trade.trade_id, TradeStatus.PENDING_CANCEL)
|
|
401
|
+
self.log.info(f"Cancelling trade: {trade.trade_id}")
|
|
402
|
+
for t in trades_to_cancel:
|
|
403
|
+
await self.trade_store.set_status(t.trade_id, TradeStatus.PENDING_CANCEL)
|
|
404
|
+
self.log.info(f"Cancelling trade: {t.trade_id} along with {trade.trade_id}")
|
|
405
|
+
|
|
406
|
+
if secure:
|
|
407
|
+
async with action_scope.use() as interface:
|
|
408
|
+
# We have to combine the spend bundle for these since they are tied with announcements
|
|
409
|
+
all_tx_names = [tx.name for tx in all_txs]
|
|
410
|
+
interface.side_effects.transactions = [
|
|
411
|
+
tx for tx in interface.side_effects.transactions if tx.name not in all_tx_names
|
|
412
|
+
]
|
|
413
|
+
final_spend_bundle = WalletSpendBundle.aggregate(
|
|
414
|
+
[tx.spend_bundle for tx in all_txs if tx.spend_bundle is not None]
|
|
415
|
+
)
|
|
416
|
+
interface.side_effects.transactions.append(
|
|
417
|
+
dataclasses.replace(all_txs[0], spend_bundle=final_spend_bundle, name=final_spend_bundle.name())
|
|
418
|
+
)
|
|
419
|
+
interface.side_effects.transactions.extend(
|
|
420
|
+
[dataclasses.replace(tx, spend_bundle=None, fee_amount=fee) for tx in all_txs[1:]]
|
|
421
|
+
)
|
|
422
|
+
|
|
423
|
+
async def save_trade(self, trade: TradeRecord, offer: Offer) -> None:
|
|
424
|
+
offer_name: bytes32 = offer.name()
|
|
425
|
+
await self.trade_store.add_trade_record(trade, offer_name)
|
|
426
|
+
|
|
427
|
+
# We want to subscribe to the coin IDs of all coins that are not the ephemeral offer coins
|
|
428
|
+
offered_coins: set[Coin] = {value for values in offer.get_offered_coins().values() for value in values}
|
|
429
|
+
non_offer_additions: set[Coin] = set(offer.additions()) ^ offered_coins
|
|
430
|
+
non_offer_removals: set[Coin] = set(offer.removals()) ^ offered_coins
|
|
431
|
+
await self.wallet_state_manager.add_interested_coin_ids(
|
|
432
|
+
[coin.name() for coin in (*non_offer_removals, *non_offer_additions)]
|
|
433
|
+
)
|
|
434
|
+
|
|
435
|
+
self.wallet_state_manager.state_changed("offer_added")
|
|
436
|
+
|
|
437
|
+
async def create_offer_for_ids(
|
|
438
|
+
self,
|
|
439
|
+
offer: dict[Union[int, bytes32], int],
|
|
440
|
+
action_scope: WalletActionScope,
|
|
441
|
+
driver_dict: Optional[dict[bytes32, PuzzleInfo]] = None,
|
|
442
|
+
solver: Optional[Solver] = None,
|
|
443
|
+
fee: uint64 = uint64(0),
|
|
444
|
+
validate_only: bool = False,
|
|
445
|
+
extra_conditions: tuple[Condition, ...] = tuple(),
|
|
446
|
+
taking: bool = False,
|
|
447
|
+
) -> Union[tuple[Literal[True], TradeRecord, None], tuple[Literal[False], None, str]]:
|
|
448
|
+
if driver_dict is None:
|
|
449
|
+
driver_dict = {}
|
|
450
|
+
if solver is None:
|
|
451
|
+
solver = Solver({})
|
|
452
|
+
result = await self._create_offer_for_ids(
|
|
453
|
+
offer,
|
|
454
|
+
action_scope,
|
|
455
|
+
driver_dict,
|
|
456
|
+
solver,
|
|
457
|
+
fee=fee,
|
|
458
|
+
extra_conditions=extra_conditions,
|
|
459
|
+
taking=taking,
|
|
460
|
+
)
|
|
461
|
+
if not result[0] or result[1] is None:
|
|
462
|
+
raise Exception(f"Error creating offer: {result[2]}")
|
|
463
|
+
|
|
464
|
+
success, created_offer, error = result
|
|
465
|
+
|
|
466
|
+
now = uint64(int(time.time()))
|
|
467
|
+
trade_offer: TradeRecord = TradeRecord(
|
|
468
|
+
confirmed_at_index=uint32(0),
|
|
469
|
+
accepted_at_time=None,
|
|
470
|
+
created_at_time=now,
|
|
471
|
+
is_my_offer=True,
|
|
472
|
+
sent=uint32(0),
|
|
473
|
+
offer=bytes(created_offer),
|
|
474
|
+
taken_offer=None,
|
|
475
|
+
coins_of_interest=created_offer.get_involved_coins(),
|
|
476
|
+
trade_id=created_offer.name(),
|
|
477
|
+
status=uint32(TradeStatus.PENDING_ACCEPT.value),
|
|
478
|
+
sent_to=[],
|
|
479
|
+
valid_times=parse_timelock_info(extra_conditions),
|
|
480
|
+
)
|
|
481
|
+
|
|
482
|
+
if success is True and trade_offer is not None and not validate_only:
|
|
483
|
+
await self.save_trade(trade_offer, created_offer)
|
|
484
|
+
|
|
485
|
+
return success, trade_offer, error
|
|
486
|
+
|
|
487
|
+
async def _create_offer_for_ids(
|
|
488
|
+
self,
|
|
489
|
+
offer_dict: dict[Union[int, bytes32], int],
|
|
490
|
+
action_scope: WalletActionScope,
|
|
491
|
+
driver_dict: Optional[dict[bytes32, PuzzleInfo]] = None,
|
|
492
|
+
solver: Optional[Solver] = None,
|
|
493
|
+
fee: uint64 = uint64(0),
|
|
494
|
+
extra_conditions: tuple[Condition, ...] = tuple(),
|
|
495
|
+
taking: bool = False,
|
|
496
|
+
) -> Union[tuple[Literal[True], Offer, None], tuple[Literal[False], None, str]]:
|
|
497
|
+
"""
|
|
498
|
+
Offer is dictionary of wallet ids and amount
|
|
499
|
+
"""
|
|
500
|
+
if driver_dict is None:
|
|
501
|
+
driver_dict = {}
|
|
502
|
+
if solver is None:
|
|
503
|
+
solver = Solver({})
|
|
504
|
+
try:
|
|
505
|
+
coins_to_offer: dict[Union[int, bytes32], set[Coin]] = {}
|
|
506
|
+
requested_payments: dict[Optional[bytes32], list[Payment]] = {}
|
|
507
|
+
offer_dict_no_ints: dict[Optional[bytes32], int] = {}
|
|
508
|
+
for id, amount in offer_dict.items():
|
|
509
|
+
asset_id: Optional[bytes32] = None
|
|
510
|
+
# asset_id can either be none if asset is XCH or
|
|
511
|
+
# bytes32 if another asset (e.g. NFT, CAT)
|
|
512
|
+
if amount > 0:
|
|
513
|
+
# this is what we are receiving in the trade
|
|
514
|
+
memos: list[bytes] = []
|
|
515
|
+
if isinstance(id, int):
|
|
516
|
+
wallet_id = uint32(id)
|
|
517
|
+
wallet = self.wallet_state_manager.wallets.get(wallet_id)
|
|
518
|
+
assert isinstance(wallet, (CATWallet, Wallet))
|
|
519
|
+
if isinstance(wallet, Wallet):
|
|
520
|
+
p2_ph: bytes32 = await wallet.get_puzzle_hash(
|
|
521
|
+
new=not action_scope.config.tx_config.reuse_puzhash
|
|
522
|
+
)
|
|
523
|
+
else:
|
|
524
|
+
p2_ph = await wallet.standard_wallet.get_puzzle_hash(
|
|
525
|
+
new=not action_scope.config.tx_config.reuse_puzhash
|
|
526
|
+
)
|
|
527
|
+
if wallet.type() != WalletType.STANDARD_WALLET:
|
|
528
|
+
if callable(getattr(wallet, "get_asset_id", None)): # ATTENTION: new wallets
|
|
529
|
+
assert isinstance(wallet, CATWallet)
|
|
530
|
+
asset_id = bytes32(bytes.fromhex(wallet.get_asset_id()))
|
|
531
|
+
memos = [p2_ph]
|
|
532
|
+
else:
|
|
533
|
+
raise ValueError(
|
|
534
|
+
f"Cannot request assets from wallet id {wallet.id()} without more information"
|
|
535
|
+
)
|
|
536
|
+
else:
|
|
537
|
+
p2_ph = await self.wallet_state_manager.main_wallet.get_puzzle_hash(
|
|
538
|
+
new=not action_scope.config.tx_config.reuse_puzhash
|
|
539
|
+
)
|
|
540
|
+
asset_id = id
|
|
541
|
+
wallet = await self.wallet_state_manager.get_wallet_for_asset_id(asset_id.hex())
|
|
542
|
+
memos = [p2_ph]
|
|
543
|
+
requested_payments[asset_id] = [Payment(p2_ph, uint64(amount), memos)]
|
|
544
|
+
elif amount < 0:
|
|
545
|
+
# this is what we are sending in the trade
|
|
546
|
+
if isinstance(id, int):
|
|
547
|
+
wallet_id = uint32(id)
|
|
548
|
+
wallet = self.wallet_state_manager.wallets[wallet_id]
|
|
549
|
+
if wallet.type() != WalletType.STANDARD_WALLET:
|
|
550
|
+
if callable(getattr(wallet, "get_asset_id", None)): # ATTENTION: new wallets
|
|
551
|
+
assert isinstance(wallet, CATWallet)
|
|
552
|
+
asset_id = bytes32(bytes.fromhex(wallet.get_asset_id()))
|
|
553
|
+
else:
|
|
554
|
+
raise ValueError(
|
|
555
|
+
f"Cannot offer assets from wallet id {wallet.id()} without more information"
|
|
556
|
+
)
|
|
557
|
+
else:
|
|
558
|
+
asset_id = id
|
|
559
|
+
wallet = await self.wallet_state_manager.get_wallet_for_asset_id(asset_id.hex())
|
|
560
|
+
assert wallet is not None
|
|
561
|
+
if not callable(getattr(wallet, "get_coins_to_offer", None)): # ATTENTION: new wallets
|
|
562
|
+
raise ValueError(f"Cannot offer coins from wallet id {wallet.id()}")
|
|
563
|
+
# For the XCH wallet also include the fee amount to the coins we use to pay this offer
|
|
564
|
+
amount_to_select = abs(amount)
|
|
565
|
+
if wallet.type() == WalletType.STANDARD_WALLET:
|
|
566
|
+
amount_to_select += fee
|
|
567
|
+
assert isinstance(wallet, (CATWallet, DataLayerWallet, NFTWallet, Wallet))
|
|
568
|
+
if isinstance(wallet, DataLayerWallet):
|
|
569
|
+
assert asset_id is not None
|
|
570
|
+
coins_to_offer[id] = await wallet.get_coins_to_offer(launcher_id=asset_id)
|
|
571
|
+
elif isinstance(wallet, NFTWallet):
|
|
572
|
+
assert asset_id is not None
|
|
573
|
+
coins_to_offer[id] = await wallet.get_coins_to_offer(nft_id=asset_id)
|
|
574
|
+
else:
|
|
575
|
+
coins_to_offer[id] = await wallet.get_coins_to_offer(
|
|
576
|
+
asset_id=asset_id,
|
|
577
|
+
amount=uint64(amount_to_select),
|
|
578
|
+
action_scope=action_scope,
|
|
579
|
+
)
|
|
580
|
+
# Note: if we use check_for_special_offer_making, this is not used.
|
|
581
|
+
elif amount == 0:
|
|
582
|
+
raise ValueError("You cannot offer nor request 0 amount of something")
|
|
583
|
+
|
|
584
|
+
offer_dict_no_ints[asset_id] = amount
|
|
585
|
+
|
|
586
|
+
if asset_id is not None and wallet is not None: # if this asset is not XCH
|
|
587
|
+
if callable(getattr(wallet, "get_puzzle_info", None)):
|
|
588
|
+
assert isinstance(wallet, (CATWallet, DataLayerWallet, NFTWallet))
|
|
589
|
+
puzzle_driver: PuzzleInfo = await wallet.get_puzzle_info(asset_id)
|
|
590
|
+
if asset_id in driver_dict and driver_dict[asset_id] != puzzle_driver:
|
|
591
|
+
# ignore the case if we're an nft transferring the did owner
|
|
592
|
+
if self.check_for_owner_change_in_drivers(puzzle_driver, driver_dict[asset_id]):
|
|
593
|
+
driver_dict[asset_id] = puzzle_driver
|
|
594
|
+
else:
|
|
595
|
+
raise ValueError(
|
|
596
|
+
f"driver_dict specified {driver_dict[asset_id]}, was expecting {puzzle_driver}"
|
|
597
|
+
)
|
|
598
|
+
else:
|
|
599
|
+
driver_dict[asset_id] = puzzle_driver
|
|
600
|
+
else:
|
|
601
|
+
raise ValueError(f"Wallet for asset id {asset_id} is not properly integrated with TradeManager")
|
|
602
|
+
|
|
603
|
+
requested_payments = await self.check_for_requested_payment_modifications(
|
|
604
|
+
requested_payments, driver_dict, taking
|
|
605
|
+
)
|
|
606
|
+
|
|
607
|
+
potential_special_offer: Optional[Offer] = await self.check_for_special_offer_making(
|
|
608
|
+
offer_dict_no_ints,
|
|
609
|
+
driver_dict,
|
|
610
|
+
action_scope,
|
|
611
|
+
solver,
|
|
612
|
+
fee,
|
|
613
|
+
extra_conditions,
|
|
614
|
+
)
|
|
615
|
+
|
|
616
|
+
if potential_special_offer is not None:
|
|
617
|
+
return True, potential_special_offer, None
|
|
618
|
+
|
|
619
|
+
all_coins: list[Coin] = [c for coins in coins_to_offer.values() for c in coins]
|
|
620
|
+
notarized_payments: dict[Optional[bytes32], list[NotarizedPayment]] = Offer.notarize_payments(
|
|
621
|
+
requested_payments, all_coins
|
|
622
|
+
)
|
|
623
|
+
announcements_to_assert = Offer.calculate_announcements(notarized_payments, driver_dict)
|
|
624
|
+
|
|
625
|
+
all_transactions: list[TransactionRecord] = []
|
|
626
|
+
fee_left_to_pay: uint64 = fee
|
|
627
|
+
# The access of the sorted keys here makes sure we create the XCH transaction first to make sure we pay fee
|
|
628
|
+
# with the XCH side of the offer and don't create an extra fee transaction in other wallets.
|
|
629
|
+
for id in sorted(coins_to_offer.keys(), key=lambda id: id != 1):
|
|
630
|
+
selected_coins = coins_to_offer[id]
|
|
631
|
+
if isinstance(id, int):
|
|
632
|
+
wallet = self.wallet_state_manager.wallets.get(uint32(id))
|
|
633
|
+
else:
|
|
634
|
+
wallet = await self.wallet_state_manager.get_wallet_for_asset_id(id.hex())
|
|
635
|
+
async with self.wallet_state_manager.new_action_scope(
|
|
636
|
+
action_scope.config.tx_config, push=False
|
|
637
|
+
) as inner_action_scope:
|
|
638
|
+
# This should probably not switch on whether or not we're spending XCH but it has to for now
|
|
639
|
+
assert wallet is not None
|
|
640
|
+
if wallet.type() == WalletType.STANDARD_WALLET:
|
|
641
|
+
assert isinstance(wallet, Wallet)
|
|
642
|
+
await wallet.generate_signed_transaction(
|
|
643
|
+
uint64(abs(offer_dict[id])),
|
|
644
|
+
Offer.ph(),
|
|
645
|
+
inner_action_scope,
|
|
646
|
+
fee=fee_left_to_pay,
|
|
647
|
+
coins=selected_coins,
|
|
648
|
+
extra_conditions=(*extra_conditions, *announcements_to_assert),
|
|
649
|
+
)
|
|
650
|
+
elif wallet.type() == WalletType.NFT:
|
|
651
|
+
assert isinstance(wallet, NFTWallet)
|
|
652
|
+
# This is to generate the tx for specific nft assets, i.e. not using
|
|
653
|
+
# wallet_id as the selector which would select any coins from nft_wallet
|
|
654
|
+
amounts = [coin.amount for coin in selected_coins]
|
|
655
|
+
await wallet.generate_signed_transaction(
|
|
656
|
+
# [abs(offer_dict[id])],
|
|
657
|
+
amounts,
|
|
658
|
+
[Offer.ph()],
|
|
659
|
+
inner_action_scope,
|
|
660
|
+
fee=fee_left_to_pay,
|
|
661
|
+
coins=selected_coins,
|
|
662
|
+
extra_conditions=(*extra_conditions, *announcements_to_assert),
|
|
663
|
+
)
|
|
664
|
+
else:
|
|
665
|
+
# ATTENTION: new_wallets
|
|
666
|
+
assert isinstance(wallet, (CATWallet, DataLayerWallet))
|
|
667
|
+
await wallet.generate_signed_transaction(
|
|
668
|
+
[uint64(abs(offer_dict[id]))],
|
|
669
|
+
[Offer.ph()],
|
|
670
|
+
inner_action_scope,
|
|
671
|
+
fee=fee_left_to_pay,
|
|
672
|
+
coins=selected_coins,
|
|
673
|
+
extra_conditions=(*extra_conditions, *announcements_to_assert),
|
|
674
|
+
add_authorizations_to_cr_cats=False,
|
|
675
|
+
)
|
|
676
|
+
|
|
677
|
+
all_transactions.extend(inner_action_scope.side_effects.transactions)
|
|
678
|
+
|
|
679
|
+
fee_left_to_pay = uint64(0)
|
|
680
|
+
extra_conditions = tuple()
|
|
681
|
+
|
|
682
|
+
async with action_scope.use() as interface:
|
|
683
|
+
interface.side_effects.transactions.extend(all_transactions)
|
|
684
|
+
|
|
685
|
+
total_spend_bundle = WalletSpendBundle.aggregate(
|
|
686
|
+
[x.spend_bundle for x in all_transactions if x.spend_bundle is not None]
|
|
687
|
+
)
|
|
688
|
+
|
|
689
|
+
offer = Offer(notarized_payments, total_spend_bundle, driver_dict)
|
|
690
|
+
return True, offer, None
|
|
691
|
+
|
|
692
|
+
except Exception as e:
|
|
693
|
+
self.log.exception("Error creating trade offer")
|
|
694
|
+
return False, None, str(e)
|
|
695
|
+
|
|
696
|
+
async def maybe_create_wallets_for_offer(self, offer: Offer) -> None:
|
|
697
|
+
for key in offer.arbitrage():
|
|
698
|
+
wsm = self.wallet_state_manager
|
|
699
|
+
if key is None:
|
|
700
|
+
continue
|
|
701
|
+
# ATTENTION: new_wallets
|
|
702
|
+
exists = await wsm.get_wallet_for_puzzle_info(offer.driver_dict[key])
|
|
703
|
+
if exists is None:
|
|
704
|
+
await wsm.create_wallet_for_puzzle_info(offer.driver_dict[key])
|
|
705
|
+
|
|
706
|
+
async def check_offer_validity(self, offer: Offer, peer: WSChiaConnection) -> bool:
|
|
707
|
+
all_removals: list[Coin] = offer.removals()
|
|
708
|
+
all_removal_names: list[bytes32] = [c.name() for c in all_removals]
|
|
709
|
+
non_ephemeral_removals: list[Coin] = list(
|
|
710
|
+
filter(lambda c: c.parent_coin_info not in all_removal_names, all_removals)
|
|
711
|
+
)
|
|
712
|
+
coin_states = await self.wallet_state_manager.wallet_node.get_coin_state(
|
|
713
|
+
[c.name() for c in non_ephemeral_removals], peer=peer
|
|
714
|
+
)
|
|
715
|
+
|
|
716
|
+
return len(coin_states) == len(non_ephemeral_removals) and all([cs.spent_height is None for cs in coin_states])
|
|
717
|
+
|
|
718
|
+
async def calculate_tx_records_for_offer(self, offer: Offer, validate: bool) -> list[TransactionRecord]:
|
|
719
|
+
if validate:
|
|
720
|
+
final_spend_bundle: WalletSpendBundle = offer.to_valid_spend()
|
|
721
|
+
hint_dict: dict[bytes32, bytes32] = {}
|
|
722
|
+
additions_dict: dict[bytes32, Coin] = {}
|
|
723
|
+
for hinted_coins, _ in (
|
|
724
|
+
compute_spend_hints_and_additions(spend) for spend in final_spend_bundle.coin_spends
|
|
725
|
+
):
|
|
726
|
+
hint_dict.update({id: hc.hint for id, hc in hinted_coins.items() if hc.hint is not None})
|
|
727
|
+
additions_dict.update({id: hc.coin for id, hc in hinted_coins.items()})
|
|
728
|
+
all_additions: list[Coin] = list(a for a in additions_dict.values())
|
|
729
|
+
else:
|
|
730
|
+
final_spend_bundle = offer._bundle
|
|
731
|
+
hint_dict = offer.hints()
|
|
732
|
+
all_additions = offer.additions()
|
|
733
|
+
|
|
734
|
+
settlement_coins: list[Coin] = [c for coins in offer.get_offered_coins().values() for c in coins]
|
|
735
|
+
settlement_coin_ids: list[bytes32] = [c.name() for c in settlement_coins]
|
|
736
|
+
|
|
737
|
+
removals: list[Coin] = final_spend_bundle.removals()
|
|
738
|
+
additions: list[Coin] = list(a for a in all_additions if a not in removals)
|
|
739
|
+
valid_times: ConditionValidTimes = parse_timelock_info(
|
|
740
|
+
parse_conditions_non_consensus(
|
|
741
|
+
condition
|
|
742
|
+
for spend in final_spend_bundle.coin_spends
|
|
743
|
+
for condition in spend.puzzle_reveal.to_program().run(spend.solution.to_program()).as_iter()
|
|
744
|
+
)
|
|
745
|
+
)
|
|
746
|
+
# this executes the puzzles again
|
|
747
|
+
all_fees = uint64(estimate_fees(final_spend_bundle))
|
|
748
|
+
|
|
749
|
+
txs = []
|
|
750
|
+
|
|
751
|
+
addition_dict: dict[uint32, list[Coin]] = {}
|
|
752
|
+
for addition in additions:
|
|
753
|
+
wallet_identifier = await self.wallet_state_manager.get_wallet_identifier_for_coin(
|
|
754
|
+
addition,
|
|
755
|
+
hint_dict,
|
|
756
|
+
)
|
|
757
|
+
if wallet_identifier is not None:
|
|
758
|
+
if addition.parent_coin_info in settlement_coin_ids:
|
|
759
|
+
wallet = self.wallet_state_manager.wallets[wallet_identifier.id]
|
|
760
|
+
assert isinstance(wallet, (CATWallet, NFTWallet, Wallet))
|
|
761
|
+
to_puzzle_hash = await wallet.convert_puzzle_hash(addition.puzzle_hash) # ATTENTION: new wallets
|
|
762
|
+
txs.append(
|
|
763
|
+
TransactionRecord(
|
|
764
|
+
confirmed_at_height=uint32(0),
|
|
765
|
+
created_at_time=uint64(int(time.time())),
|
|
766
|
+
to_puzzle_hash=to_puzzle_hash,
|
|
767
|
+
amount=uint64(addition.amount),
|
|
768
|
+
fee_amount=uint64(0),
|
|
769
|
+
confirmed=False,
|
|
770
|
+
sent=uint32(10),
|
|
771
|
+
spend_bundle=None,
|
|
772
|
+
additions=[addition],
|
|
773
|
+
removals=[],
|
|
774
|
+
wallet_id=wallet_identifier.id,
|
|
775
|
+
sent_to=[],
|
|
776
|
+
trade_id=offer.name(),
|
|
777
|
+
type=uint32(TransactionType.INCOMING_TRADE.value),
|
|
778
|
+
name=std_hash(final_spend_bundle.name() + addition.name()),
|
|
779
|
+
memos=[(coin_id, [hint]) for coin_id, hint in hint_dict.items()],
|
|
780
|
+
valid_times=valid_times,
|
|
781
|
+
)
|
|
782
|
+
)
|
|
783
|
+
else: # This is change
|
|
784
|
+
addition_dict.setdefault(wallet_identifier.id, [])
|
|
785
|
+
addition_dict[wallet_identifier.id].append(addition)
|
|
786
|
+
|
|
787
|
+
# While we want additions to show up as separate records, removals of the same wallet should show as one
|
|
788
|
+
removal_dict: dict[uint32, list[Coin]] = {}
|
|
789
|
+
for removal in removals:
|
|
790
|
+
wallet_identifier = await self.wallet_state_manager.get_wallet_identifier_for_coin(
|
|
791
|
+
removal,
|
|
792
|
+
hint_dict,
|
|
793
|
+
)
|
|
794
|
+
if wallet_identifier is not None:
|
|
795
|
+
removal_dict.setdefault(wallet_identifier.id, [])
|
|
796
|
+
removal_dict[wallet_identifier.id].append(removal)
|
|
797
|
+
|
|
798
|
+
all_removals: list[bytes32] = [r.name() for removals in removal_dict.values() for r in removals]
|
|
799
|
+
|
|
800
|
+
for wid, grouped_removals in removal_dict.items():
|
|
801
|
+
wallet = self.wallet_state_manager.wallets[wid]
|
|
802
|
+
to_puzzle_hash = bytes32([1] * 32) # We use all zeros to be clear not to send here
|
|
803
|
+
removal_tree_hash = Program.to([coin_as_list(rem) for rem in grouped_removals]).get_tree_hash()
|
|
804
|
+
# We also need to calculate the sent amount
|
|
805
|
+
removed: int = sum(c.amount for c in grouped_removals)
|
|
806
|
+
removed_ids: list[bytes32] = [c.name() for c in grouped_removals]
|
|
807
|
+
all_additions_from_grouped_removals: list[Coin] = [
|
|
808
|
+
c for c in all_additions if c.parent_coin_info in removed_ids
|
|
809
|
+
]
|
|
810
|
+
potential_change_coins: list[Coin] = addition_dict[wid] if wid in addition_dict else []
|
|
811
|
+
change_coins: list[Coin] = [c for c in potential_change_coins if c.parent_coin_info in all_removals]
|
|
812
|
+
change_amount: int = sum(c.amount for c in change_coins)
|
|
813
|
+
sent_amount: int = (
|
|
814
|
+
removed
|
|
815
|
+
- change_amount
|
|
816
|
+
- (
|
|
817
|
+
removed - sum(c.amount for c in all_additions_from_grouped_removals) # removals - additions == fees
|
|
818
|
+
if wallet == self.wallet_state_manager.main_wallet
|
|
819
|
+
else 0
|
|
820
|
+
)
|
|
821
|
+
)
|
|
822
|
+
txs.append(
|
|
823
|
+
TransactionRecord(
|
|
824
|
+
confirmed_at_height=uint32(0),
|
|
825
|
+
created_at_time=uint64(int(time.time())),
|
|
826
|
+
to_puzzle_hash=to_puzzle_hash,
|
|
827
|
+
amount=uint64(sent_amount),
|
|
828
|
+
fee_amount=all_fees,
|
|
829
|
+
confirmed=False,
|
|
830
|
+
sent=uint32(10),
|
|
831
|
+
spend_bundle=None,
|
|
832
|
+
additions=change_coins,
|
|
833
|
+
removals=grouped_removals,
|
|
834
|
+
wallet_id=wallet.id(),
|
|
835
|
+
sent_to=[],
|
|
836
|
+
trade_id=offer.name(),
|
|
837
|
+
type=uint32(TransactionType.OUTGOING_TRADE.value),
|
|
838
|
+
name=std_hash(final_spend_bundle.name() + removal_tree_hash),
|
|
839
|
+
memos=[(coin_id, [hint]) for coin_id, hint in hint_dict.items()],
|
|
840
|
+
valid_times=valid_times,
|
|
841
|
+
)
|
|
842
|
+
)
|
|
843
|
+
|
|
844
|
+
return txs
|
|
845
|
+
|
|
846
|
+
async def respond_to_offer(
|
|
847
|
+
self,
|
|
848
|
+
offer: Offer,
|
|
849
|
+
peer: WSChiaConnection,
|
|
850
|
+
action_scope: WalletActionScope,
|
|
851
|
+
solver: Optional[Solver] = None,
|
|
852
|
+
fee: uint64 = uint64(0),
|
|
853
|
+
extra_conditions: tuple[Condition, ...] = tuple(),
|
|
854
|
+
) -> TradeRecord:
|
|
855
|
+
if solver is None:
|
|
856
|
+
solver = Solver({})
|
|
857
|
+
take_offer_dict: dict[Union[bytes32, int], int] = {}
|
|
858
|
+
arbitrage: dict[Optional[bytes32], int] = offer.arbitrage()
|
|
859
|
+
|
|
860
|
+
for asset_id, amount in arbitrage.items():
|
|
861
|
+
if asset_id is None:
|
|
862
|
+
wallet: Optional[WalletProtocol[Any]] = self.wallet_state_manager.main_wallet
|
|
863
|
+
assert wallet is not None
|
|
864
|
+
key: Union[bytes32, int] = int(wallet.id())
|
|
865
|
+
else:
|
|
866
|
+
# ATTENTION: new wallets
|
|
867
|
+
wallet = await self.wallet_state_manager.get_wallet_for_asset_id(asset_id.hex())
|
|
868
|
+
if wallet is None and amount < 0:
|
|
869
|
+
raise ValueError(f"Do not have a wallet for asset ID: {asset_id} to fulfill offer")
|
|
870
|
+
elif wallet is None or wallet.type() in {WalletType.NFT, WalletType.DATA_LAYER}:
|
|
871
|
+
key = asset_id
|
|
872
|
+
else:
|
|
873
|
+
key = int(wallet.id())
|
|
874
|
+
take_offer_dict[key] = amount
|
|
875
|
+
|
|
876
|
+
# First we validate that all of the coins in this offer exist
|
|
877
|
+
valid: bool = await self.check_offer_validity(offer, peer)
|
|
878
|
+
if not valid:
|
|
879
|
+
raise ValueError("This offer is no longer valid")
|
|
880
|
+
# We need to sandbox the transactions here because we're going to make our own
|
|
881
|
+
async with self.wallet_state_manager.new_action_scope(
|
|
882
|
+
action_scope.config.tx_config, push=False
|
|
883
|
+
) as inner_action_scope:
|
|
884
|
+
result = await self._create_offer_for_ids(
|
|
885
|
+
take_offer_dict,
|
|
886
|
+
inner_action_scope,
|
|
887
|
+
offer.driver_dict,
|
|
888
|
+
solver,
|
|
889
|
+
fee=fee,
|
|
890
|
+
extra_conditions=extra_conditions,
|
|
891
|
+
taking=True,
|
|
892
|
+
)
|
|
893
|
+
if not result[0] or result[1] is None:
|
|
894
|
+
raise ValueError(result[2])
|
|
895
|
+
|
|
896
|
+
_success, take_offer, _error = result
|
|
897
|
+
|
|
898
|
+
complete_offer, valid_spend_solver = await self.check_for_final_modifications(
|
|
899
|
+
Offer.aggregate([offer, take_offer]), solver, inner_action_scope
|
|
900
|
+
)
|
|
901
|
+
self.log.info("COMPLETE OFFER: %s", complete_offer.to_bech32())
|
|
902
|
+
assert complete_offer.is_valid()
|
|
903
|
+
final_spend_bundle: WalletSpendBundle = complete_offer.to_valid_spend(
|
|
904
|
+
solver=Solver({**valid_spend_solver.info, **solver.info})
|
|
905
|
+
)
|
|
906
|
+
await self.maybe_create_wallets_for_offer(complete_offer)
|
|
907
|
+
|
|
908
|
+
tx_records: list[TransactionRecord] = await self.calculate_tx_records_for_offer(complete_offer, True)
|
|
909
|
+
|
|
910
|
+
trade_record: TradeRecord = TradeRecord(
|
|
911
|
+
confirmed_at_index=uint32(0),
|
|
912
|
+
accepted_at_time=uint64(int(time.time())),
|
|
913
|
+
created_at_time=uint64(int(time.time())),
|
|
914
|
+
is_my_offer=False,
|
|
915
|
+
sent=uint32(0),
|
|
916
|
+
offer=bytes(complete_offer),
|
|
917
|
+
taken_offer=bytes(offer),
|
|
918
|
+
coins_of_interest=complete_offer.get_involved_coins(),
|
|
919
|
+
trade_id=complete_offer.name(),
|
|
920
|
+
status=uint32(TradeStatus.PENDING_CONFIRM.value),
|
|
921
|
+
sent_to=[],
|
|
922
|
+
valid_times=parse_timelock_info(extra_conditions),
|
|
923
|
+
)
|
|
924
|
+
|
|
925
|
+
await self.save_trade(trade_record, offer)
|
|
926
|
+
|
|
927
|
+
async with action_scope.use() as interface:
|
|
928
|
+
interface.side_effects.transactions.extend(tx_records)
|
|
929
|
+
interface.side_effects.extra_spends.append(final_spend_bundle)
|
|
930
|
+
|
|
931
|
+
return trade_record
|
|
932
|
+
|
|
933
|
+
async def check_for_special_offer_making(
|
|
934
|
+
self,
|
|
935
|
+
offer_dict: dict[Optional[bytes32], int],
|
|
936
|
+
driver_dict: dict[bytes32, PuzzleInfo],
|
|
937
|
+
action_scope: WalletActionScope,
|
|
938
|
+
solver: Solver,
|
|
939
|
+
fee: uint64 = uint64(0),
|
|
940
|
+
extra_conditions: tuple[Condition, ...] = tuple(),
|
|
941
|
+
) -> Optional[Offer]:
|
|
942
|
+
for puzzle_info in driver_dict.values():
|
|
943
|
+
if (
|
|
944
|
+
puzzle_info.check_type([AssetType.SINGLETON.value, AssetType.METADATA.value, AssetType.OWNERSHIP.value])
|
|
945
|
+
and isinstance(puzzle_info.also().also()["transfer_program"], PuzzleInfo) # type: ignore
|
|
946
|
+
and puzzle_info.also().also()["transfer_program"].type() # type: ignore
|
|
947
|
+
== AssetType.ROYALTY_TRANSFER_PROGRAM.value
|
|
948
|
+
):
|
|
949
|
+
return await NFTWallet.make_nft1_offer(
|
|
950
|
+
self.wallet_state_manager, offer_dict, driver_dict, action_scope, fee, extra_conditions
|
|
951
|
+
)
|
|
952
|
+
elif (
|
|
953
|
+
puzzle_info.check_type(
|
|
954
|
+
[
|
|
955
|
+
AssetType.SINGLETON.value,
|
|
956
|
+
AssetType.METADATA.value,
|
|
957
|
+
]
|
|
958
|
+
)
|
|
959
|
+
and puzzle_info.also()["updater_hash"] == ACS_MU_PH # type: ignore
|
|
960
|
+
):
|
|
961
|
+
return await DataLayerWallet.make_update_offer(
|
|
962
|
+
self.wallet_state_manager,
|
|
963
|
+
offer_dict,
|
|
964
|
+
driver_dict,
|
|
965
|
+
solver,
|
|
966
|
+
action_scope,
|
|
967
|
+
fee,
|
|
968
|
+
extra_conditions,
|
|
969
|
+
)
|
|
970
|
+
return None
|
|
971
|
+
|
|
972
|
+
def check_for_owner_change_in_drivers(self, puzzle_info: PuzzleInfo, driver_info: PuzzleInfo) -> bool:
|
|
973
|
+
if puzzle_info.check_type(
|
|
974
|
+
[
|
|
975
|
+
AssetType.SINGLETON.value,
|
|
976
|
+
AssetType.METADATA.value,
|
|
977
|
+
AssetType.OWNERSHIP.value,
|
|
978
|
+
]
|
|
979
|
+
) and driver_info.check_type(
|
|
980
|
+
[
|
|
981
|
+
AssetType.SINGLETON.value,
|
|
982
|
+
AssetType.METADATA.value,
|
|
983
|
+
AssetType.OWNERSHIP.value,
|
|
984
|
+
]
|
|
985
|
+
):
|
|
986
|
+
old_owner = driver_info.also().also().info["owner"] # type: ignore
|
|
987
|
+
puzzle_info.also().also().info["owner"] = old_owner # type: ignore
|
|
988
|
+
if driver_info == puzzle_info:
|
|
989
|
+
return True
|
|
990
|
+
return False
|
|
991
|
+
|
|
992
|
+
async def get_offer_summary(self, offer: Offer) -> dict[str, Any]:
|
|
993
|
+
for puzzle_info in offer.driver_dict.values():
|
|
994
|
+
if (
|
|
995
|
+
puzzle_info.check_type(
|
|
996
|
+
[
|
|
997
|
+
AssetType.SINGLETON.value,
|
|
998
|
+
AssetType.METADATA.value,
|
|
999
|
+
]
|
|
1000
|
+
)
|
|
1001
|
+
and puzzle_info.also()["updater_hash"] == ACS_MU_PH # type: ignore
|
|
1002
|
+
):
|
|
1003
|
+
return await DataLayerWallet.get_offer_summary(offer)
|
|
1004
|
+
# Otherwise just return the same thing as the RPC normally does
|
|
1005
|
+
offered, requested, infos, valid_times = offer.summary()
|
|
1006
|
+
return {
|
|
1007
|
+
"offered": offered,
|
|
1008
|
+
"requested": requested,
|
|
1009
|
+
"fees": offer.fees(),
|
|
1010
|
+
"additions": [c.name().hex() for c in offer.additions()],
|
|
1011
|
+
"removals": [c.name().hex() for c in offer.removals()],
|
|
1012
|
+
"infos": infos,
|
|
1013
|
+
"valid_times": {
|
|
1014
|
+
k: v
|
|
1015
|
+
for k, v in valid_times.to_json_dict().items()
|
|
1016
|
+
if k
|
|
1017
|
+
not in {
|
|
1018
|
+
"max_secs_after_created",
|
|
1019
|
+
"min_secs_since_created",
|
|
1020
|
+
"max_blocks_after_created",
|
|
1021
|
+
"min_blocks_since_created",
|
|
1022
|
+
}
|
|
1023
|
+
},
|
|
1024
|
+
}
|
|
1025
|
+
|
|
1026
|
+
async def check_for_final_modifications(
|
|
1027
|
+
self, offer: Offer, solver: Solver, action_scope: WalletActionScope
|
|
1028
|
+
) -> tuple[Offer, Solver]:
|
|
1029
|
+
for puzzle_info in offer.driver_dict.values():
|
|
1030
|
+
if (
|
|
1031
|
+
puzzle_info.check_type(
|
|
1032
|
+
[
|
|
1033
|
+
AssetType.SINGLETON.value,
|
|
1034
|
+
AssetType.METADATA.value,
|
|
1035
|
+
]
|
|
1036
|
+
)
|
|
1037
|
+
and puzzle_info.also()["updater_hash"] == ACS_MU_PH # type: ignore
|
|
1038
|
+
):
|
|
1039
|
+
return (await DataLayerWallet.finish_graftroot_solutions(offer, solver), Solver({}))
|
|
1040
|
+
elif puzzle_info.check_type(
|
|
1041
|
+
[
|
|
1042
|
+
AssetType.CAT.value,
|
|
1043
|
+
AssetType.CR.value,
|
|
1044
|
+
]
|
|
1045
|
+
):
|
|
1046
|
+
# get VC wallet
|
|
1047
|
+
for _, wallet in self.wallet_state_manager.wallets.items():
|
|
1048
|
+
if WalletType(wallet.type()) == WalletType.VC:
|
|
1049
|
+
assert isinstance(wallet, VCWallet)
|
|
1050
|
+
return await wallet.add_vc_authorization(offer, solver, action_scope)
|
|
1051
|
+
raise ValueError("No VCs to approve CR-CATs with") # pragma: no cover
|
|
1052
|
+
|
|
1053
|
+
return offer, Solver({})
|
|
1054
|
+
|
|
1055
|
+
async def check_for_requested_payment_modifications(
|
|
1056
|
+
self,
|
|
1057
|
+
requested_payments: dict[Optional[bytes32], list[Payment]],
|
|
1058
|
+
driver_dict: dict[bytes32, PuzzleInfo],
|
|
1059
|
+
taking: bool,
|
|
1060
|
+
) -> dict[Optional[bytes32], list[Payment]]:
|
|
1061
|
+
# This function exclusively deals with CR-CATs for now
|
|
1062
|
+
if not taking:
|
|
1063
|
+
for asset_id, puzzle_info in driver_dict.items():
|
|
1064
|
+
if puzzle_info.check_type(
|
|
1065
|
+
[
|
|
1066
|
+
AssetType.CAT.value,
|
|
1067
|
+
AssetType.CR.value,
|
|
1068
|
+
]
|
|
1069
|
+
):
|
|
1070
|
+
vc = await (
|
|
1071
|
+
await self.wallet_state_manager.get_or_create_vc_wallet()
|
|
1072
|
+
).get_vc_with_provider_in_and_proofs(
|
|
1073
|
+
puzzle_info["also"]["authorized_providers"],
|
|
1074
|
+
ProofsChecker.from_program(uncurry_puzzle(puzzle_info["also"]["proofs_checker"])).flags,
|
|
1075
|
+
)
|
|
1076
|
+
if vc is None:
|
|
1077
|
+
raise ValueError("Cannot request CR-CATs that you cannot approve with a VC") # pragma: no cover
|
|
1078
|
+
|
|
1079
|
+
return {
|
|
1080
|
+
asset_id: (
|
|
1081
|
+
[
|
|
1082
|
+
dataclasses.replace(
|
|
1083
|
+
payment,
|
|
1084
|
+
puzzle_hash=construct_pending_approval_state(
|
|
1085
|
+
payment.puzzle_hash, payment.amount
|
|
1086
|
+
).get_tree_hash(),
|
|
1087
|
+
)
|
|
1088
|
+
for payment in payments
|
|
1089
|
+
]
|
|
1090
|
+
if asset_id is not None
|
|
1091
|
+
and driver_dict[asset_id].check_type(
|
|
1092
|
+
[
|
|
1093
|
+
AssetType.CAT.value,
|
|
1094
|
+
AssetType.CR.value,
|
|
1095
|
+
]
|
|
1096
|
+
)
|
|
1097
|
+
else payments
|
|
1098
|
+
)
|
|
1099
|
+
for asset_id, payments in requested_payments.items()
|
|
1100
|
+
}
|
|
1101
|
+
else:
|
|
1102
|
+
return requested_payments
|