From 304f9d5df4f847f72e285085bb694ee2e372cf20 Mon Sep 17 00:00:00 2001 From: lubkoll <11710767+lubkoll@users.noreply.github.com> Date: Thu, 1 Aug 2024 09:55:24 +0000 Subject: [PATCH] Remove commented out contracts (#731) --- smart-contracts/osmosis/Cargo.toml | 19 +- .../osmosis/contracts/airdrop/.cargo/config | 4 - .../osmosis/contracts/airdrop/.gitignore | 2 - .../osmosis/contracts/airdrop/Cargo.toml | 32 - .../contracts/airdrop/examples/schema.rs | 10 - .../osmosis/contracts/airdrop/src/admin.rs | 459 ---- .../osmosis/contracts/airdrop/src/contract.rs | 168 -- .../osmosis/contracts/airdrop/src/error.rs | 60 - .../osmosis/contracts/airdrop/src/helpers.rs | 242 -- .../osmosis/contracts/airdrop/src/lib.rs | 10 - .../osmosis/contracts/airdrop/src/msg.rs | 90 - .../osmosis/contracts/airdrop/src/query.rs | 131 - .../osmosis/contracts/airdrop/src/state.rs | 74 - .../osmosis/contracts/airdrop/src/users.rs | 63 - .../contracts/basic-vault/.cargo/config | 6 - .../osmosis/contracts/basic-vault/.gitignore | 2 - .../contracts/basic-vault/CHANGELOG.md | 20 - .../osmosis/contracts/basic-vault/Cargo.toml | 47 - .../osmosis/contracts/basic-vault/NOTICE | 15 - .../osmosis/contracts/basic-vault/README.md | 25 - .../contracts/basic-vault/examples/schema.rs | 10 - .../contracts/basic-vault/src/callback.rs | 400 --- .../contracts/basic-vault/src/contract.rs | 614 ----- .../contracts/basic-vault/src/error.rs | 146 -- .../contracts/basic-vault/src/execute.rs | 1098 -------- .../contracts/basic-vault/src/helpers.rs | 65 - .../osmosis/contracts/basic-vault/src/lib.rs | 17 - .../osmosis/contracts/basic-vault/src/msg.rs | 316 --- .../basic-vault/src/multitest/common.rs | 63 - .../basic-vault/src/multitest/mod.rs | 3 - .../basic-vault/src/multitest/suite.rs | 322 --- .../basic-vault/src/multitest/vault.rs | 23 - .../contracts/basic-vault/src/query.rs | 120 - .../contracts/basic-vault/src/state.rs | 194 -- .../contracts/basic-vault/src/tests.rs | 2228 ----------------- .../contracts/basic-vault/src/types.rs | 11 - .../osmosis/contracts/cw-4626/.cargo/config | 6 - .../osmosis/contracts/cw-4626/Cargo.lock | 745 ------ .../osmosis/contracts/cw-4626/Cargo.toml | 40 - .../osmosis/contracts/cw-4626/NOTICE | 14 - .../osmosis/contracts/cw-4626/README.md | 61 - .../contracts/cw-4626/examples/schema.rs | 37 - .../cw-4626/schema/all_accounts_response.json | 16 - .../schema/all_allowances_response.json | 99 - .../cw-4626/schema/allowance_response.json | 81 - .../cw-4626/schema/asset_response.json | 13 - .../cw-4626/schema/balance_response.json | 19 - .../schema/convert_to_assets_response.json | 37 - .../schema/convert_to_shares_response.json | 19 - .../cw-4626/schema/cw20_execute_msg.json | 442 ---- .../cw-4626/schema/instantiate_msg.json | 240 -- .../cw-4626/schema/max_deposit_response.json | 38 - .../contracts/cw-4626/schema/query_msg.json | 168 -- .../cw-4626/schema/token_info_response.json | 33 - .../cw-4626/schema/total_asset_response.json | 19 - .../cw-4626/schema/vault_info_response.json | 23 - .../osmosis/contracts/cw-4626/src/contract.rs | 958 ------- .../osmosis/contracts/cw-4626/src/error.rs | 17 - .../osmosis/contracts/cw-4626/src/lib.rs | 6 - .../osmosis/contracts/cw-4626/src/msg.rs | 283 --- .../osmosis/contracts/cw-4626/src/state.rs | 22 - .../contracts/ibc-transfer/.cargo/config | 6 - .../osmosis/contracts/ibc-transfer/Cargo.toml | 54 - .../osmosis/contracts/ibc-transfer/README.md | 14 - .../contracts/ibc-transfer/examples/schema.rs | 30 - .../ibc-transfer/schema/execute_msg.json | 32 - .../ibc-transfer/schema/instantiate_msg.json | 5 - .../contracts/ibc-transfer/src/contract.rs | 128 - .../contracts/ibc-transfer/src/error.rs | 34 - .../contracts/ibc-transfer/src/helpers.rs | 87 - .../osmosis/contracts/ibc-transfer/src/ibc.rs | 269 -- .../osmosis/contracts/ibc-transfer/src/lib.rs | 21 - .../contracts/ibc-transfer/src/state.rs | 30 - .../osmosis/contracts/ica/.cargo/config | 5 - .../osmosis/contracts/ica/Cargo.toml | 37 - smart-contracts/osmosis/contracts/ica/NOTICE | 14 - .../osmosis/contracts/ica/README.md | 32 - .../contracts/ica/create_and_execute.sh | 32 - .../osmosis/contracts/ica/examples/schema.rs | 23 - .../ica/schema/channel_response.json | 61 - .../contracts/ica/schema/execute_msg.json | 67 - .../contracts/ica/schema/i_c_q_query_msg.json | 60 - .../contracts/ica/schema/init_msg.json | 16 - .../ica/schema/list_channels_response.json | 59 - .../contracts/ica/schema/port_response.json | 13 - .../contracts/ica/schema/query_msg.json | 66 - .../osmosis/contracts/ica/src/contract.rs | 194 -- .../osmosis/contracts/ica/src/error.rs | 53 - .../osmosis/contracts/ica/src/helpers.rs | 85 - .../osmosis/contracts/ica/src/ibc.rs | 225 -- .../osmosis/contracts/ica/src/lib.rs | 14 - .../osmosis/contracts/ica/src/msg.rs | 95 - .../osmosis/contracts/ica/src/proto.rs | 15 - .../osmosis/contracts/ica/src/state.rs | 40 - .../osmosis/contracts/ica/src/test_helpers.rs | 72 - .../osmosis/contracts/icq/.cargo/config | 5 - .../osmosis/contracts/icq/Cargo.toml | 37 - smart-contracts/osmosis/contracts/icq/NOTICE | 14 - .../osmosis/contracts/icq/README.md | 32 - .../osmosis/contracts/icq/examples/schema.rs | 23 - .../icq/schema/channel_response.json | 61 - .../contracts/icq/schema/execute_msg.json | 148 -- .../contracts/icq/schema/i_c_q_query_msg.json | 60 - .../contracts/icq/schema/init_msg.json | 16 - .../icq/schema/list_channels_response.json | 59 - .../contracts/icq/schema/port_response.json | 13 - .../contracts/icq/schema/query_msg.json | 66 - .../osmosis/contracts/icq/src/contract.rs | 324 --- .../osmosis/contracts/icq/src/error.rs | 50 - .../osmosis/contracts/icq/src/helpers.rs | 147 -- .../osmosis/contracts/icq/src/ibc.rs | 221 -- .../osmosis/contracts/icq/src/lib.rs | 14 - .../osmosis/contracts/icq/src/msg.rs | 100 - .../osmosis/contracts/icq/src/proto.rs | 15 - .../osmosis/contracts/icq/src/state.rs | 32 - .../osmosis/contracts/icq/src/test_helpers.rs | 72 - .../intergamm-bindings-test/.cargo/config | 4 - .../intergamm-bindings-test/.editorconfig | 11 - .../intergamm-bindings-test/.gitignore | 15 - .../.gitpod.Dockerfile | 17 - .../intergamm-bindings-test/.gitpod.yml | 10 - .../intergamm-bindings-test/Cargo.lock | 738 ------ .../intergamm-bindings-test/Cargo.toml | 61 - .../intergamm-bindings-test/Developing.md | 104 - .../intergamm-bindings-test/Importing.md | 62 - .../contracts/intergamm-bindings-test/LICENSE | 202 -- .../contracts/intergamm-bindings-test/NOTICE | 13 - .../intergamm-bindings-test/Publishing.md | 115 - .../intergamm-bindings-test/README.md | 33 - .../examples/schema.rs | 17 - .../intergamm-bindings-test/rustfmt.toml | 15 - .../schema/execute_msg.json | 516 ---- .../schema/instantiate_msg.json | 13 - .../schema/query_msg.json | 30 - .../intergamm-bindings-test/src/contract.rs | 341 --- .../intergamm-bindings-test/src/error.rs | 22 - .../src/integration_tests.rs | 70 - .../intergamm-bindings-test/src/lib.rs | 6 - .../intergamm-bindings-test/src/msg.rs | 98 - .../intergamm-bindings-test/src/state.rs | 23 - .../contracts/intergamm-bindings-test/test.sh | 0 .../contracts/lp-strategy/.cargo/config | 5 - .../contracts/lp-strategy/CHANGELOG.md | 64 - .../osmosis/contracts/lp-strategy/Cargo.toml | 38 - .../osmosis/contracts/lp-strategy/README.md | 19 - .../contracts/lp-strategy/examples/schema.rs | 10 - .../lp-strategy/migration_primitive1.json | 10 - .../lp-strategy/migration_primitive3.json | 22 - .../lp-strategy/proptest-regressions/icq.txt | 8 - .../proptest-regressions/proptests.txt | 10 - .../contracts/lp-strategy/src/admin.rs | 158 -- .../osmosis/contracts/lp-strategy/src/bond.rs | 554 ---- .../contracts/lp-strategy/src/contract.rs | 1331 ---------- .../contracts/lp-strategy/src/error.rs | 141 -- .../lp-strategy/src/error_recovery.rs | 298 --- .../contracts/lp-strategy/src/execute.rs | 1007 -------- .../contracts/lp-strategy/src/helpers.rs | 365 --- .../osmosis/contracts/lp-strategy/src/ibc.rs | 1179 --------- .../contracts/lp-strategy/src/ibc_lock.rs | 192 -- .../contracts/lp-strategy/src/ibc_util.rs | 397 --- .../osmosis/contracts/lp-strategy/src/icq.rs | 410 --- .../lp-strategy/src/integration_tests.rs | 420 ---- .../osmosis/contracts/lp-strategy/src/lib.rs | 31 - .../osmosis/contracts/lp-strategy/src/msg.rs | 285 --- .../contracts/lp-strategy/src/proptests.rs | 291 --- .../contracts/lp-strategy/src/queries.rs | 336 --- .../contracts/lp-strategy/src/reply.rs | 370 --- .../contracts/lp-strategy/src/start_unbond.rs | 819 ------ .../contracts/lp-strategy/src/state.rs | 330 --- .../contracts/lp-strategy/src/test_helpers.rs | 237 -- .../contracts/lp-strategy/src/unbond.rs | 659 ----- .../contracts/multihop-router/.cargo/config | 4 - .../contracts/multihop-router/.editorconfig | 11 - .../contracts/multihop-router/.gitignore | 16 - .../contracts/multihop-router/Cargo.toml | 62 - .../osmosis/contracts/multihop-router/LICENSE | 202 -- .../osmosis/contracts/multihop-router/NOTICE | 13 - .../contracts/multihop-router/README.md | 9 - .../multihop-router/src/bin/schema.rs | 11 - .../contracts/multihop-router/src/contract.rs | 261 -- .../contracts/multihop-router/src/error.rs | 19 - .../contracts/multihop-router/src/helpers.rs | 45 - .../contracts/multihop-router/src/lib.rs | 11 - .../contracts/multihop-router/src/msg.rs | 65 - .../multihop-router/src/multitest/common.rs | 21 - .../multihop-router/src/multitest/mod.rs | 3 - .../multihop-router/src/multitest/suite.rs | 251 -- .../multihop-router/src/multitest/test.rs | 99 - .../contracts/multihop-router/src/route.rs | 379 --- .../contracts/multihop-router/src/state.rs | 7 - .../contracts/vault-rewards/.cargo/config | 4 - .../contracts/vault-rewards/.gitignore | 1 - .../contracts/vault-rewards/CHANGELOG.md | 83 - .../contracts/vault-rewards/Cargo.toml | 54 - .../vault-rewards/examples/schema.rs | 17 - .../vault-rewards/schema/config.json | 92 - .../vault-rewards/schema/execute_msg.json | 250 -- .../vault-rewards/schema/instantiate_msg.json | 85 - .../vault-rewards/schema/query_msg.json | 31 - .../contracts/vault-rewards/src/contract.rs | 96 - .../contracts/vault-rewards/src/error.rs | 41 - .../vault-rewards/src/execute/admin.rs | 111 - .../vault-rewards/src/execute/mock_querier.rs | 104 - .../vault-rewards/src/execute/mod.rs | 6 - .../vault-rewards/src/execute/user.rs | 505 ---- .../vault-rewards/src/execute/vault.rs | 42 - .../contracts/vault-rewards/src/helpers.rs | 47 - .../contracts/vault-rewards/src/lib.rs | 9 - .../contracts/vault-rewards/src/msg.rs | 70 - .../contracts/vault-rewards/src/query.rs | 65 - .../contracts/vault-rewards/src/state.rs | 225 -- 211 files changed, 1 insertion(+), 30000 deletions(-) delete mode 100644 smart-contracts/osmosis/contracts/airdrop/.cargo/config delete mode 100644 smart-contracts/osmosis/contracts/airdrop/.gitignore delete mode 100644 smart-contracts/osmosis/contracts/airdrop/Cargo.toml delete mode 100644 smart-contracts/osmosis/contracts/airdrop/examples/schema.rs delete mode 100644 smart-contracts/osmosis/contracts/airdrop/src/admin.rs delete mode 100644 smart-contracts/osmosis/contracts/airdrop/src/contract.rs delete mode 100644 smart-contracts/osmosis/contracts/airdrop/src/error.rs delete mode 100644 smart-contracts/osmosis/contracts/airdrop/src/helpers.rs delete mode 100644 smart-contracts/osmosis/contracts/airdrop/src/lib.rs delete mode 100644 smart-contracts/osmosis/contracts/airdrop/src/msg.rs delete mode 100644 smart-contracts/osmosis/contracts/airdrop/src/query.rs delete mode 100644 smart-contracts/osmosis/contracts/airdrop/src/state.rs delete mode 100644 smart-contracts/osmosis/contracts/airdrop/src/users.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/.cargo/config delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/.gitignore delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/CHANGELOG.md delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/Cargo.toml delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/NOTICE delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/README.md delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/examples/schema.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/src/callback.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/src/contract.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/src/error.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/src/execute.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/src/helpers.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/src/lib.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/src/msg.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/src/multitest/common.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/src/multitest/mod.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/src/multitest/suite.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/src/multitest/vault.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/src/query.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/src/state.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/src/tests.rs delete mode 100644 smart-contracts/osmosis/contracts/basic-vault/src/types.rs delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/.cargo/config delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/Cargo.lock delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/Cargo.toml delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/NOTICE delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/README.md delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/examples/schema.rs delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/schema/all_accounts_response.json delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/schema/all_allowances_response.json delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/schema/allowance_response.json delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/schema/asset_response.json delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/schema/balance_response.json delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/schema/convert_to_assets_response.json delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/schema/convert_to_shares_response.json delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/schema/cw20_execute_msg.json delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/schema/instantiate_msg.json delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/schema/max_deposit_response.json delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/schema/query_msg.json delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/schema/token_info_response.json delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/schema/total_asset_response.json delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/schema/vault_info_response.json delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/src/contract.rs delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/src/error.rs delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/src/lib.rs delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/src/msg.rs delete mode 100644 smart-contracts/osmosis/contracts/cw-4626/src/state.rs delete mode 100644 smart-contracts/osmosis/contracts/ibc-transfer/.cargo/config delete mode 100644 smart-contracts/osmosis/contracts/ibc-transfer/Cargo.toml delete mode 100644 smart-contracts/osmosis/contracts/ibc-transfer/README.md delete mode 100644 smart-contracts/osmosis/contracts/ibc-transfer/examples/schema.rs delete mode 100644 smart-contracts/osmosis/contracts/ibc-transfer/schema/execute_msg.json delete mode 100644 smart-contracts/osmosis/contracts/ibc-transfer/schema/instantiate_msg.json delete mode 100644 smart-contracts/osmosis/contracts/ibc-transfer/src/contract.rs delete mode 100644 smart-contracts/osmosis/contracts/ibc-transfer/src/error.rs delete mode 100644 smart-contracts/osmosis/contracts/ibc-transfer/src/helpers.rs delete mode 100644 smart-contracts/osmosis/contracts/ibc-transfer/src/ibc.rs delete mode 100644 smart-contracts/osmosis/contracts/ibc-transfer/src/lib.rs delete mode 100644 smart-contracts/osmosis/contracts/ibc-transfer/src/state.rs delete mode 100644 smart-contracts/osmosis/contracts/ica/.cargo/config delete mode 100644 smart-contracts/osmosis/contracts/ica/Cargo.toml delete mode 100644 smart-contracts/osmosis/contracts/ica/NOTICE delete mode 100644 smart-contracts/osmosis/contracts/ica/README.md delete mode 100755 smart-contracts/osmosis/contracts/ica/create_and_execute.sh delete mode 100644 smart-contracts/osmosis/contracts/ica/examples/schema.rs delete mode 100644 smart-contracts/osmosis/contracts/ica/schema/channel_response.json delete mode 100644 smart-contracts/osmosis/contracts/ica/schema/execute_msg.json delete mode 100644 smart-contracts/osmosis/contracts/ica/schema/i_c_q_query_msg.json delete mode 100644 smart-contracts/osmosis/contracts/ica/schema/init_msg.json delete mode 100644 smart-contracts/osmosis/contracts/ica/schema/list_channels_response.json delete mode 100644 smart-contracts/osmosis/contracts/ica/schema/port_response.json delete mode 100644 smart-contracts/osmosis/contracts/ica/schema/query_msg.json delete mode 100644 smart-contracts/osmosis/contracts/ica/src/contract.rs delete mode 100644 smart-contracts/osmosis/contracts/ica/src/error.rs delete mode 100644 smart-contracts/osmosis/contracts/ica/src/helpers.rs delete mode 100644 smart-contracts/osmosis/contracts/ica/src/ibc.rs delete mode 100644 smart-contracts/osmosis/contracts/ica/src/lib.rs delete mode 100644 smart-contracts/osmosis/contracts/ica/src/msg.rs delete mode 100644 smart-contracts/osmosis/contracts/ica/src/proto.rs delete mode 100644 smart-contracts/osmosis/contracts/ica/src/state.rs delete mode 100644 smart-contracts/osmosis/contracts/ica/src/test_helpers.rs delete mode 100644 smart-contracts/osmosis/contracts/icq/.cargo/config delete mode 100644 smart-contracts/osmosis/contracts/icq/Cargo.toml delete mode 100644 smart-contracts/osmosis/contracts/icq/NOTICE delete mode 100644 smart-contracts/osmosis/contracts/icq/README.md delete mode 100644 smart-contracts/osmosis/contracts/icq/examples/schema.rs delete mode 100644 smart-contracts/osmosis/contracts/icq/schema/channel_response.json delete mode 100644 smart-contracts/osmosis/contracts/icq/schema/execute_msg.json delete mode 100644 smart-contracts/osmosis/contracts/icq/schema/i_c_q_query_msg.json delete mode 100644 smart-contracts/osmosis/contracts/icq/schema/init_msg.json delete mode 100644 smart-contracts/osmosis/contracts/icq/schema/list_channels_response.json delete mode 100644 smart-contracts/osmosis/contracts/icq/schema/port_response.json delete mode 100644 smart-contracts/osmosis/contracts/icq/schema/query_msg.json delete mode 100644 smart-contracts/osmosis/contracts/icq/src/contract.rs delete mode 100644 smart-contracts/osmosis/contracts/icq/src/error.rs delete mode 100644 smart-contracts/osmosis/contracts/icq/src/helpers.rs delete mode 100644 smart-contracts/osmosis/contracts/icq/src/ibc.rs delete mode 100644 smart-contracts/osmosis/contracts/icq/src/lib.rs delete mode 100644 smart-contracts/osmosis/contracts/icq/src/msg.rs delete mode 100644 smart-contracts/osmosis/contracts/icq/src/proto.rs delete mode 100644 smart-contracts/osmosis/contracts/icq/src/state.rs delete mode 100644 smart-contracts/osmosis/contracts/icq/src/test_helpers.rs delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/.cargo/config delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/.editorconfig delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/.gitignore delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/.gitpod.Dockerfile delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/.gitpod.yml delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/Cargo.lock delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/Cargo.toml delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/Developing.md delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/Importing.md delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/LICENSE delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/NOTICE delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/Publishing.md delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/README.md delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/examples/schema.rs delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/rustfmt.toml delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/schema/execute_msg.json delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/schema/instantiate_msg.json delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/schema/query_msg.json delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/src/contract.rs delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/src/error.rs delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/src/integration_tests.rs delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/src/lib.rs delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/src/msg.rs delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/src/state.rs delete mode 100644 smart-contracts/osmosis/contracts/intergamm-bindings-test/test.sh delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/.cargo/config delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/CHANGELOG.md delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/Cargo.toml delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/README.md delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/examples/schema.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/migration_primitive1.json delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/migration_primitive3.json delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/proptest-regressions/icq.txt delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/proptest-regressions/proptests.txt delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/admin.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/bond.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/contract.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/error.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/error_recovery.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/execute.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/helpers.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/ibc.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/ibc_lock.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/ibc_util.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/icq.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/integration_tests.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/lib.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/msg.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/proptests.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/queries.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/reply.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/start_unbond.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/state.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/test_helpers.rs delete mode 100644 smart-contracts/osmosis/contracts/lp-strategy/src/unbond.rs delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/.cargo/config delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/.editorconfig delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/.gitignore delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/Cargo.toml delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/LICENSE delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/NOTICE delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/README.md delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/src/bin/schema.rs delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/src/contract.rs delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/src/error.rs delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/src/helpers.rs delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/src/lib.rs delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/src/msg.rs delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/src/multitest/common.rs delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/src/multitest/mod.rs delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/src/multitest/suite.rs delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/src/multitest/test.rs delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/src/route.rs delete mode 100644 smart-contracts/osmosis/contracts/multihop-router/src/state.rs delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/.cargo/config delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/.gitignore delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/CHANGELOG.md delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/Cargo.toml delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/examples/schema.rs delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/schema/config.json delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/schema/execute_msg.json delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/schema/instantiate_msg.json delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/schema/query_msg.json delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/src/contract.rs delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/src/error.rs delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/src/execute/admin.rs delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/src/execute/mock_querier.rs delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/src/execute/mod.rs delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/src/execute/user.rs delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/src/execute/vault.rs delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/src/helpers.rs delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/src/lib.rs delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/src/msg.rs delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/src/query.rs delete mode 100644 smart-contracts/osmosis/contracts/vault-rewards/src/state.rs diff --git a/smart-contracts/osmosis/Cargo.toml b/smart-contracts/osmosis/Cargo.toml index 3b783311b..f50ea5c37 100644 --- a/smart-contracts/osmosis/Cargo.toml +++ b/smart-contracts/osmosis/Cargo.toml @@ -1,24 +1,7 @@ [workspace] resolver = "2" -members = [ - # "contracts/icq", - # "contracts/ibc-transfer", - # "contracts/intergamm-bindings-test", - # "contracts/airdrop", - # "contracts/lp-strategy", - # "contracts/basic-vault", - # "contracts/vault-rewards", - # "contracts/multihop-router", - "contracts/token-burner", - "contracts/cl-vault", - "contracts/merkle-incentives", - "contracts/range-middleware", - "contracts/dex-router-osmosis", - "contracts/lst-dex-adapter-osmosis", - "contracts/lst-adapter-osmosis", - "packages/*", -] +members = ["contracts/*", "packages/*"] [workspace.dependencies] # CosmWasm diff --git a/smart-contracts/osmosis/contracts/airdrop/.cargo/config b/smart-contracts/osmosis/contracts/airdrop/.cargo/config deleted file mode 100644 index 624255c74..000000000 --- a/smart-contracts/osmosis/contracts/airdrop/.cargo/config +++ /dev/null @@ -1,4 +0,0 @@ -[alias] -wasm = "build --release --target wasm32-unknown-unknown" -unit-test = "test --lib" -schema = "run --example schema" \ No newline at end of file diff --git a/smart-contracts/osmosis/contracts/airdrop/.gitignore b/smart-contracts/osmosis/contracts/airdrop/.gitignore deleted file mode 100644 index 360bb05b1..000000000 --- a/smart-contracts/osmosis/contracts/airdrop/.gitignore +++ /dev/null @@ -1,2 +0,0 @@ -ts -schema diff --git a/smart-contracts/osmosis/contracts/airdrop/Cargo.toml b/smart-contracts/osmosis/contracts/airdrop/Cargo.toml deleted file mode 100644 index 91a101ee4..000000000 --- a/smart-contracts/osmosis/contracts/airdrop/Cargo.toml +++ /dev/null @@ -1,32 +0,0 @@ -[package] -name = "airdrop" -version = "0.1.0" -edition = "2021" - -[lib] -crate-type = ["cdylib", "rlib"] - -[features] -# for more explicit tests, cargo test --features=backtraces -backtraces = ["cosmwasm-std/backtraces"] -# use library feature to disable all instantiate/execute/query exports -library = [] - - -# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html - -[dependencies] -cosmwasm-std = { workspace = true } -osmosis-std = { workspace = true } -osmosis-std-derive = { workspace = true } -cosmwasm-schema = { workspace = true } -cw-storage-plus ={ workspace = true } -schemars ={ workspace = true } -serde ={ workspace = true } -thiserror ={ workspace = true } -cw2 = { workspace = true } -cw20 = { workspace = true } -cw20-base = { workspace = true } -cw-asset = { workspace = true } -num_enum = { workspace = true } -itertools = { workspace = true } \ No newline at end of file diff --git a/smart-contracts/osmosis/contracts/airdrop/examples/schema.rs b/smart-contracts/osmosis/contracts/airdrop/examples/schema.rs deleted file mode 100644 index 0f41c1516..000000000 --- a/smart-contracts/osmosis/contracts/airdrop/examples/schema.rs +++ /dev/null @@ -1,10 +0,0 @@ -use airdrop::msg::{ExecuteMsg, InstantiateMsg, QueryMsg}; -use cosmwasm_schema::write_api; - -fn main() { - write_api! { - instantiate: InstantiateMsg, - query: QueryMsg, - execute: ExecuteMsg, - } -} diff --git a/smart-contracts/osmosis/contracts/airdrop/src/admin.rs b/smart-contracts/osmosis/contracts/airdrop/src/admin.rs deleted file mode 100644 index 48b586c64..000000000 --- a/smart-contracts/osmosis/contracts/airdrop/src/admin.rs +++ /dev/null @@ -1,459 +0,0 @@ -use std::string::String; - -use cosmwasm_std::{Attribute, DepsMut, Env, Event, Response, Uint128}; -use cw_asset::Asset; - -use crate::helpers::{ - check_amounts_and_airdrop_size, get_total_in_user_info, validate_amount, validate_update_config, -}; -use crate::msg::User; -use crate::state::{AirdropConfig, UserInfo, AIRDROP_CONFIG, USER_INFO}; -use crate::AirdropErrors; - -/// Updates the airdrop configuration of the contract. -/// -/// # Arguments -/// -/// * `deps` - Dependencies to access storage and external data. -/// * `env` - The current contract execution environment. -/// * `config` - The new airdrop configuration to be set. -/// -/// # Errors -/// -/// Returns an error if the airdrop has already ended or if the new configuration is invalid. -/// -/// # Returns -/// -/// Returns a response indicating the success of the update operation and includes -/// relevant attributes in the event. -pub fn execute_update_airdrop_config( - deps: DepsMut, - env: Env, - config: AirdropConfig, -) -> Result { - let current_airdrop_config = AIRDROP_CONFIG.load(deps.storage)?; - - if current_airdrop_config.is_airdrop_active(env.block.height) { - return Err(AirdropErrors::InvalidChangeInAirdropConfig {}); - } - - // Check if an airdrop has been executed on the contract and if the update is allowed - if current_airdrop_config.end_height != 0 - && env.block.height > current_airdrop_config.end_height - { - return Err(AirdropErrors::InvalidChangeInAirdropConfig {}); - } - - validate_update_config(config.clone(), deps.storage, deps.querier, env)?; - AIRDROP_CONFIG.save(deps.storage, &config)?; - - // Return a default response to indicate success with an "update_airdrop_config" event - Ok(Response::new().add_event( - Event::new("update_airdrop_config") - .add_attribute( - "description".to_string(), - config.airdrop_description.to_string(), - ) - .add_attribute( - "airdrop_amount".to_string(), - config.airdrop_amount.to_string(), - ) - .add_attribute( - "airdrop_asset".to_string(), - config.airdrop_asset.to_string(), - ) - .add_attribute("claimed".to_string(), config.total_claimed.to_string()) - .add_attribute("start_height".to_string(), config.start_height.to_string()) - .add_attribute("end_height".to_string(), config.end_height.to_string()), - )) -} - -/// Adds new users and their respective amounts to the airdrop configuration. -/// -/// # Arguments -/// -/// * `deps` - Dependencies to access storage and external data. -/// * `users` - A vector of user addresses to be added. -/// * `amounts` - A vector of amounts to be allocated to each user. -/// -/// # Errors -/// -/// Returns an error if the airdrop window is not open, the number of users and amounts provided do not match, -/// or if any of the provided users already have existing claims or allocations. -/// -/// # Returns -/// -/// Returns a response indicating the success of the addition operation and includes relevant attributes -/// in the event. -pub fn execute_add_users(deps: DepsMut, users: Vec) -> Result { - let current_airdrop_config = AIRDROP_CONFIG.load(deps.storage)?; - - // Check if the current airdrop window is not open (start_height or end_height not zero) - if current_airdrop_config.start_height != 0 || current_airdrop_config.end_height != 0 { - return Err(AirdropErrors::InvalidChangeUserInfo {}); - } - - let mut attributes: Vec = Vec::new(); - - // Loop through the provided users and amounts - for user in users { - // Validate the user's address - deps.api.addr_validate(&user.address)?; - - // Validate that the amount is not zero - validate_amount(user.clone())?; - - // Attempt to load user_info from storage - let maybe_user_info = USER_INFO.may_load(deps.storage, user.address.clone())?; - - // Check if the user_info exists (is not empty) - if let Some(user_info) = maybe_user_info { - // User info exists, perform your checks here - if user_info.get_claimable_amount() != Uint128::zero() || user_info.get_claimed_flag() { - // Handle the case where user_info exists - return Err(AirdropErrors::AlreadyExists { user: user.address }); - } - } else { - // User info does not exist, create a new entry - let new_user_info = UserInfo { - claimable_amount: user.amount, - claimed_flag: false, - }; - USER_INFO.save(deps.storage, user.address.to_string(), &new_user_info)?; - - // Add user and amount to attributes for the event - attributes.push(Attribute { - key: "address".to_string(), - value: user.address.to_string(), - }); - attributes.push(Attribute { - key: "amount".to_string(), - value: user.amount.to_string(), - }); - } - } - - // config is invalid of total claimable assigned to users is greater than amount assigned to the airdrop - check_amounts_and_airdrop_size( - get_total_in_user_info(deps.storage), - current_airdrop_config.airdrop_amount, - )?; - - // Return a default response if all checks pass with an "airdrop_add_users" event - Ok(Response::new().add_event(Event::new("airdrop_add_users").add_attributes(attributes))) -} - -/// Sets or updates the allocation of claimable amounts for a list of users in the airdrop configuration. -/// -/// # Arguments -/// -/// * `deps` - Dependencies to access storage and external data. -/// * `users` - A vector of user addresses to set or update allocations for. -/// * `amounts` - A vector of amounts to be allocated to each user. -/// -/// # Errors -/// -/// Returns an error if the airdrop window is not open (start_height or end_height not zero), -/// the number of users and amounts provided do not match, or if any of the provided users have claimed their allocations. -/// -/// # Returns -/// -/// Returns a response indicating the success of the set/update operation and includes relevant attributes -/// in the event. -pub fn execute_set_users(deps: DepsMut, users: Vec) -> Result { - let current_airdrop_config = AIRDROP_CONFIG.load(deps.storage)?; - - // Check if the current airdrop window is not open (start_height or end_height not zero) - if current_airdrop_config.start_height != 0 || current_airdrop_config.end_height != 0 { - return Err(AirdropErrors::InvalidChangeUserInfo {}); - } - - let mut attributes: Vec = Vec::new(); - - for user in users { - // Validate the user's address - deps.api.addr_validate(&user.address)?; - - // Validate that the amount is not zero - validate_amount(user.clone())?; - - // Load the user's current information from storage - let user_info = USER_INFO.load(deps.storage, user.address.to_string())?; - - // Check if the user has not claimed - if !user_info.get_claimed_flag() { - // Update all the users with the given info - let new_user_info = UserInfo { - claimable_amount: user.amount, - claimed_flag: false, - }; - USER_INFO.save(deps.storage, user.address.to_string(), &new_user_info)?; - - // Add user and amount to attributes for the event - attributes.push(Attribute { - key: "address".to_string(), - value: user.address.to_string(), - }); - attributes.push(Attribute { - key: "amount".to_string(), - value: user.amount.to_string(), - }) - } - } - - // config is invalid of total claimable assigned to users is greater than amount assigned to the airdrop - check_amounts_and_airdrop_size( - get_total_in_user_info(deps.storage), - current_airdrop_config.airdrop_amount, - )?; - - // Return a default response if all checks pass with an "airdrop_set_users" event - Ok(Response::new().add_event(Event::new("airdrop_set_users").add_attributes(attributes))) -} - -/// Removes specified users from the airdrop configuration if they have not claimed their allocations. -/// -/// # Arguments -/// -/// * `deps` - Dependencies to access storage and external data. -/// * `users` - A vector of user addresses to remove from the airdrop configuration. -/// -/// # Errors -/// -/// Returns an error if the airdrop window is not open (start_height or end_height not zero). -/// -/// # Returns -/// -/// Returns a response indicating the success of the removal operation and includes relevant attributes -/// in the event for each removed user. -pub fn execute_remove_users(deps: DepsMut, users: Vec) -> Result { - let current_airdrop_config = AIRDROP_CONFIG.load(deps.storage)?; - - // Check if the current airdrop window is not open (start_height or end_height not zero) - if current_airdrop_config.start_height != 0 || current_airdrop_config.end_height != 0 { - return Err(AirdropErrors::InvalidChangeUserInfo {}); - } - - // Initialize vectors to store removed users and attributes for the event - let mut removed_users: Vec = Vec::new(); - let mut attributes: Vec = Vec::new(); - - // Iterate through the list of users to be removed - for user in users.iter() { - // Validate the user's address - deps.api.addr_validate(user)?; - - // Load the user_info entry from storage - let user_info = USER_INFO.load(deps.storage, user.to_string())?; - - // Check if the claimed flag is false, indicating that the user has not claimed - if !user_info.get_claimed_flag() { - // Add the user's address to the list of removed users - removed_users.push(user.to_string()); - - // Remove the user's entry from the USER_INFO map - USER_INFO.remove(deps.storage, user.to_string()); - - // Add user address as an attribute for the event - attributes.push(Attribute { - key: "address".to_string(), - value: user.to_string(), - }); - } - } - - // Return a default response if all checks pass with an "airdrop_remove_users" event - Ok(Response::new().add_event(Event::new("airdrop_remove_users").add_attributes(attributes))) -} - -/// Withdraws airdrop funds to the specified address after the airdrop window has ended. -/// -/// # Arguments -/// -/// * `deps` - Dependencies to access storage, external data, and assets. -/// * `env` - Environment information including the current block height. -/// * `withdraw_address` - The address to which the airdrop funds will be withdrawn. -/// -/// # Errors -/// -/// Returns an error if the current block height is not within the airdrop window or the window is open-ended. -/// Also returns an error if the withdrawal address is invalid. -/// -/// # Returns -/// -/// Returns a response indicating the success of the withdrawal and includes attributes in the response for tracking. -pub fn execute_withdraw_funds( - deps: DepsMut, - env: Env, - withdraw_address: String, -) -> Result { - let current_airdrop_config = AIRDROP_CONFIG.load(deps.storage)?; - - // Check if the current block height is within the airdrop window or the window is open-ended - if env.block.height < current_airdrop_config.end_height - || current_airdrop_config.end_height == 0 - || current_airdrop_config.start_height == 0 - { - return Err(AirdropErrors::InvalidWithdraw {}); - } - - // Validate the withdrawal address - deps.api.addr_validate(&withdraw_address)?; - - // Load the current airdrop configuration again to ensure consistency - let current_airdrop_config = AIRDROP_CONFIG.load(deps.storage)?; - - // Query the contract's balance of the airdrop asset - let contract_balance = current_airdrop_config - .airdrop_asset - .query_balance(&deps.querier, &env.contract.address)?; - - // Transfer the airdrop asset to the withdrawal address - let withdraw = Asset::new(current_airdrop_config.airdrop_asset, contract_balance) - .transfer_msg(&withdraw_address)?; - - // Return a response and add the withdraw transfer message - Ok(Response::new().add_message(withdraw).add_attributes(vec![ - ("action", "withdraw"), - ("address", env.contract.address.as_ref()), - ("amount", &contract_balance.to_string()), - ])) -} - -// Import necessary modules for testing -#[cfg(test)] -mod tests { - // Import the necessary items for testing - use super::*; - use cosmwasm_std::testing::{mock_dependencies, mock_env, mock_info}; - use cosmwasm_std::{Addr, Coin}; - use cw_asset::AssetInfo; - - use crate::contract::instantiate; - use crate::msg::InstantiateMsg; - - // Define a helper function to create a mock contract configuration - fn mock_config_1() -> AirdropConfig { - AirdropConfig { - airdrop_title: "Test Title".to_string(), - airdrop_description: "Test Airdrop".to_string(), - airdrop_amount: Uint128::new(1000000), // Adjust this value as needed - airdrop_asset: AssetInfo::Native("uqsr".parse().unwrap()), - total_claimed: Uint128::new(0), - start_height: 12346, // Adjust this value as needed - end_height: 14567, // Adjust this value as needed - } - } - - // Define a helper function to create a mock contract configuration - fn mock_config_2() -> AirdropConfig { - AirdropConfig { - airdrop_title: "Test Title".to_string(), - airdrop_description: "Test Airdrop".to_string(), - airdrop_amount: Uint128::new(1000000), // Adjust this value as needed - airdrop_asset: AssetInfo::Native("uqsr".parse().unwrap()), - total_claimed: Uint128::new(0), - start_height: 0, // Adjust this value as needed - end_height: 0, // Adjust this value as needed - } - } - - // Define a test case for updating the airdrop configuration - #[test] - fn test_execute_update_airdrop_config() { - // Create mock dependencies, environment, and config - let mut deps = mock_dependencies(); - let env = mock_env(); - let info = mock_info("admin", &[Coin::new(100000000, "uqsr")]); - let config = mock_config_1(); - - // Execute the instantiate function to set up the initial state (if needed) - let instantiate_msg_1 = InstantiateMsg { - config: config.clone(), - }; - instantiate(deps.as_mut(), env.clone(), info.clone(), instantiate_msg_1).unwrap_err(); - - // instantiate with a correct config - let instantiate_msg_2 = InstantiateMsg { - config: mock_config_2(), - }; - instantiate(deps.as_mut(), env.clone(), info.clone(), instantiate_msg_2).unwrap(); - - // try to update config with wrong conditions - execute_update_airdrop_config(deps.as_mut(), env.clone(), config.clone()).unwrap_err(); - - // add users to the airdrop - let users: Vec = vec![ - User { - address: "user1".to_string(), - amount: Uint128::new(330000), - }, - User { - address: "user2".to_string(), - amount: Uint128::new(330000), - }, - User { - address: "user3".to_string(), - amount: Uint128::new(330000), - }, - ]; - let add_users_response = execute_add_users(deps.as_mut(), users).unwrap(); - assert_eq!(add_users_response.events[0].attributes.len(), 6); - - // set a user so that the total is higher than airdrop size - let users1: Vec = vec![User { - address: "user1".to_string(), - amount: Uint128::new(630000), - }]; - execute_set_users(deps.as_mut(), users1).unwrap_err(); - - // set users with new values and the amount should be less than the airdrop size - let users2: Vec = vec![ - User { - address: "user1".to_string(), - amount: Uint128::new(230000), - }, - User { - address: "user2".to_string(), - amount: Uint128::new(430000), - }, - ]; - let set_users_response = execute_set_users(deps.as_mut(), users2).unwrap(); - assert_eq!(set_users_response.events[0].attributes.len(), 4); - - // remove user1 which should be successful - let users3: Vec = vec!["user1".to_string()]; - let set_users_response = execute_remove_users(deps.as_mut(), users3).unwrap(); - assert_eq!(set_users_response.events[0].attributes.len(), 1); - - // remove user4 which should result into an error - let users4: Vec = vec!["user4".to_string()]; - execute_remove_users(deps.as_mut(), users4).unwrap_err(); - - // add the user1 again - let users5: Vec = vec![User { - address: "user1".to_string(), - amount: Uint128::new(230000), - }]; - - let set_users_response = execute_add_users(deps.as_mut(), users5).unwrap(); - assert_eq!(set_users_response.events[0].attributes.len(), 2); - - // update the airdrop config with - let new_balance: Vec = vec![Coin { - denom: "uqsr".to_string(), - amount: Uint128::new(1000000), - }]; - let address = Addr::unchecked("cosmos2contract"); - deps.querier.update_balance(address, new_balance); - let execute_response = - execute_update_airdrop_config(deps.as_mut(), env.clone(), config.clone()).unwrap(); - - // Ensure that the response is successful - assert_eq!(execute_response.events[0].attributes.len(), 6); // Check for expected attributes - - // Verify that the new configuration is stored - let stored_config = AIRDROP_CONFIG.load(deps.as_ref().storage); - assert_eq!(stored_config.unwrap(), config); - } -} diff --git a/smart-contracts/osmosis/contracts/airdrop/src/contract.rs b/smart-contracts/osmosis/contracts/airdrop/src/contract.rs deleted file mode 100644 index 0961eb2b7..000000000 --- a/smart-contracts/osmosis/contracts/airdrop/src/contract.rs +++ /dev/null @@ -1,168 +0,0 @@ -use cosmwasm_std::{ - entry_point, to_json_binary, Binary, Deps, DepsMut, Env, Event, MessageInfo, Response, - StdResult, Uint128, -}; -use cw2::set_contract_version; -use cw20_base::msg::MigrateMsg; - -use crate::admin::{ - execute_add_users, execute_remove_users, execute_set_users, execute_update_airdrop_config, - execute_withdraw_funds, -}; -use crate::error::AirdropErrors; -use crate::helpers::is_contract_admin; -use crate::msg::{AdminExecuteMsg, ExecuteMsg, InstantiateMsg, QueryMsg}; -use crate::query::{ - query_airdrop_readiness, query_config, query_contract_state, query_user, query_users_stats, -}; -use crate::state::AIRDROP_CONFIG; -use crate::users::execute_claim; - -// version info for migration info -const CONTRACT_NAME: &str = "quasar_airdrop"; -const CONTRACT_VERSION: &str = env!("CARGO_PKG_VERSION"); - -/// Instantiates the airdrop contract with the provided configuration. -/// -/// # Arguments -/// -/// * `deps` - Dependencies to access storage and external data. -/// * `_env` - Environment information, not used in this function. -/// * `_info` - Message sender's information, not used in this function. -/// * `msg` - Instantiate message containing the airdrop configuration. -/// -/// # Errors -/// -/// Returns an error if the airdrop configuration is invalid, specifically if start height, -/// end height, and total claimed are not set to zero. -/// -/// # Returns -/// -/// Returns a response indicating the success of contract instantiation and includes attributes -/// describing the airdrop configuration. -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn instantiate( - deps: DepsMut, - _env: Env, - _info: MessageInfo, - msg: InstantiateMsg, -) -> Result { - // Set the contract version in storage - set_contract_version(deps.storage, CONTRACT_NAME, CONTRACT_VERSION)?; - - // Check if the contract should not be instantiated due to an invalid airdrop window - if msg.config.start_height != 0 - || msg.config.end_height != 0 - || msg.config.total_claimed != Uint128::zero() - { - return Err(AirdropErrors::InvalidAirdropWindow {}); - } - - // Save the new airdrop configuration to storage - AIRDROP_CONFIG.save(deps.storage, &msg.config)?; - - // Return a response indicating successful contract instantiation with attributes - Ok(Response::new().add_event( - Event::new("instantiate_airdrop_contract") - .add_attribute( - "description".to_string(), - msg.config.airdrop_description.to_string(), - ) - .add_attribute( - "airdrop_amount".to_string(), - msg.config.airdrop_amount.to_string(), - ) - .add_attribute( - "airdrop_asset".to_string(), - msg.config.airdrop_asset.to_string(), - ) - .add_attribute("claimed".to_string(), msg.config.total_claimed.to_string()) - .add_attribute( - "start_height".to_string(), - msg.config.start_height.to_string(), - ) - .add_attribute("end_height".to_string(), msg.config.end_height.to_string()), - )) -} - -/// Executes contract operations based on the received message. -/// -/// # Arguments -/// -/// * `deps` - Dependencies to access storage and external data. -/// * `env` - Environment information. -/// * `info` - Message sender's information. -/// * `msg` - Execute message to determine the operation. -/// -/// # Returns -/// -/// Returns a response based on the executed operation or an error if the operation fails. -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn execute( - deps: DepsMut, - env: Env, - info: MessageInfo, - msg: ExecuteMsg, -) -> Result { - match msg { - ExecuteMsg::Admin(admin_msg) => { - // Check if the sender is a contract admin - is_contract_admin(&deps.querier, &env, &info.sender)?; - - match admin_msg { - AdminExecuteMsg::UpdateAirdropConfig(airdrop_config) => { - // Call the function to update the airdrop configuration - execute_update_airdrop_config(deps, env, airdrop_config) - } - AdminExecuteMsg::AddUsers { users } => { - // Call the function to add users and their amounts - execute_add_users(deps, users) - } - AdminExecuteMsg::RemoveUsers(users) => execute_remove_users(deps, users), - AdminExecuteMsg::WithdrawFunds(withdraw_address) => { - execute_withdraw_funds(deps, env, withdraw_address) - } - AdminExecuteMsg::SetUsers { users } => execute_set_users(deps, users), - } - } - ExecuteMsg::ClaimAirdrop() => execute_claim(deps, env, info.sender), - } -} - -/// Queries contract state based on the received query message. -/// -/// # Arguments -/// -/// * `deps` - Dependencies to access storage and external data. -/// * `env` - Environment information. -/// * `msg` - Query message to determine the requested information. -/// -/// # Returns -/// -/// Returns a binary response containing the queried information or an error if the query fails. -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn query(deps: Deps, env: Env, msg: QueryMsg) -> StdResult { - match msg { - QueryMsg::AirdropConfigQuery {} => to_json_binary(&query_config(deps)?), - QueryMsg::UserInfoQuery { user } => to_json_binary(&query_user(deps, user)?), - QueryMsg::ContractStateQuery {} => to_json_binary(&query_contract_state(deps)?), - QueryMsg::SanityCheckQuery {} => to_json_binary(&query_airdrop_readiness(deps, env)?), - QueryMsg::UsersStatsQuery {} => to_json_binary(&query_users_stats(deps)?), - } -} - -/// Migrates the contract to a new version (not implemented). -/// -/// # Arguments -/// -/// * `_deps` - Dependencies to access storage and external data, not used in this function. -/// * `_env` - Environment information, not used in this function. -/// * `_msg` - Migrate message, not used in this function. -/// -/// # Returns -/// -/// Returns a response indicating a successful migration (not implemented). -#[entry_point] -pub fn migrate(_deps: DepsMut, _env: Env, _msg: MigrateMsg) -> Result { - Ok(Response::new().add_attribute("migrate", "successful")) -} diff --git a/smart-contracts/osmosis/contracts/airdrop/src/error.rs b/smart-contracts/osmosis/contracts/airdrop/src/error.rs deleted file mode 100644 index d7757c1f9..000000000 --- a/smart-contracts/osmosis/contracts/airdrop/src/error.rs +++ /dev/null @@ -1,60 +0,0 @@ -use cosmwasm_std::{StdError, Uint128}; -use cw_asset::AssetError; -use thiserror::Error; - -#[derive(Error, Debug, PartialEq)] -pub enum AirdropErrors { - #[error("{0}")] - Std(#[from] StdError), - - #[error("{0}")] - Asset(#[from] AssetError), - - #[error("Unauthorized")] - Unauthorized {}, - - #[error("Invalid airdrop window")] - InvalidAirdropWindow {}, - - #[error("Airdrop config cannot be changed once airdrop is active")] - InvalidChangeInConfig {}, - - #[error("User info cannot be changed once airdrop is active")] - InvalidChangeUserInfo {}, - - #[error("Withdraw is in an invalid window")] - InvalidWithdraw {}, - - #[error("Claim is in an invalid window")] - InvalidClaim {}, - - #[error("Already claimed")] - AlreadyClaimed {}, - - #[error("Already exists : {user:?}")] - AlreadyExists { user: String }, - - #[error("Invalid change in airdrop config")] - InvalidChangeInAirdropConfig {}, - - #[error("Insufficient funds in contract account. Balance: {balance:?}")] - InsufficientFundsInContractAccount { balance: Uint128 }, - - #[error("Total amount in the given user amounts {total_in_user_info:?} is greater than {current_airdrop_amount:?}")] - UserAmountIsGreaterThanTotal { - total_in_user_info: Uint128, - current_airdrop_amount: Uint128, - }, - - #[error("Failed due to config has less amount than the amount allowed to the users to claim")] - ConfigAmountLessThanTotalClaimable {}, - - #[error("Failed as total claimed is non zero")] - NonZeroClaimedAmount {}, - - #[error("Given number of users do not match the given number of amounts which results into a mismatch")] - UnequalLengths {}, - - #[error("Amount for address {address:?} is zero")] - ZeroAmount { address: String }, -} diff --git a/smart-contracts/osmosis/contracts/airdrop/src/helpers.rs b/smart-contracts/osmosis/contracts/airdrop/src/helpers.rs deleted file mode 100644 index 7ac09cc23..000000000 --- a/smart-contracts/osmosis/contracts/airdrop/src/helpers.rs +++ /dev/null @@ -1,242 +0,0 @@ -use cosmwasm_std::{Addr, Env, Order, QuerierWrapper, Response, Storage, Uint128}; - -use crate::error::AirdropErrors; -use crate::msg::User; -use crate::state::{AirdropConfig, AIRDROP_CONFIG, USER_INFO}; - -/// Checks if the sender is the contract admin. Returns an error if not authorized. -/// -/// # Arguments -/// -/// * `querier` - QuerierWrapper to query contract admin information. -/// * `env` - Environment information. -/// * `sus_admin` - Address of the sender. -/// -/// # Returns -/// -/// Returns `Ok(())` if the sender is authorized as the contract admin, otherwise returns an Unauthorized error. -pub fn is_contract_admin( - querier: &QuerierWrapper, - env: &Env, - sus_admin: &Addr, -) -> Result<(), AirdropErrors> { - // Get the contract admin address from the contract's information - let contract_admin = querier - .query_wasm_contract_info(&env.contract.address)? - .admin; - - // Check if the contract admin address exists - if let Some(contract_admin) = contract_admin { - // Compare the contract admin address with the provided sus_admin address - if contract_admin != *sus_admin { - // If they don't match, return an Unauthorized error - return Err(AirdropErrors::Unauthorized {}); - } - } else { - // If the contract admin address doesn't exist, return an Unauthorized error - return Err(AirdropErrors::Unauthorized {}); - } - - // If all checks pass, return Ok() to indicate success - Ok(()) -} - -/// Checks if the total claimable amount exceeds the airdrop amount. -/// -/// # Arguments -/// -/// * `total_in_user_info` - Total claimable amount from all users. -/// * `current_airdrop_amount` - Current airdrop amount in the contract. -/// -/// # Returns -/// -/// Returns a default response if the total claimable amount does not exceed the airdrop amount, -/// otherwise returns an error indicating insufficient funds. -pub fn check_amounts_and_airdrop_size( - total_in_user_info: Uint128, - current_airdrop_amount: Uint128, -) -> Result { - // Check if the total claimable amount exceeds the airdrop amount - if total_in_user_info > current_airdrop_amount { - return Err(AirdropErrors::UserAmountIsGreaterThanTotal { - total_in_user_info, - current_airdrop_amount, - }); - } - Ok(Response::default()) -} - -/// Validates that an amount is not zero. -/// -/// # Arguments -/// -/// * `amount` - Amount to validate. -/// * `index` - Index of the amount in a list (used for error message). -/// -/// # Returns -/// -/// Returns a default response if the amount is not zero, otherwise returns an error indicating a zero amount. -pub fn validate_amount(user: User) -> Result { - // Check if the amount is not zero - if user.amount == Uint128::zero() { - return Err(AirdropErrors::ZeroAmount { - address: user.address, - }); - } - Ok(Response::default()) -} - -/// Validates and checks the airdrop configuration update. -/// -/// # Arguments -/// -/// * `config` - New airdrop configuration to validate. -/// * `storage` - Storage to access contract state. -/// * `querier` - QuerierWrapper to query contract state. -/// * `env` - Environment information. -/// -/// # Returns -/// -/// Returns a default response if the configuration update is valid, otherwise returns an error -/// indicating the reason for the validation failure. -pub fn validate_update_config( - config: AirdropConfig, - storage: &dyn Storage, - querier: QuerierWrapper, - env: Env, -) -> Result { - // Check if the start height and end height are not zero, - // indicating a valid airdrop window - if config.total_claimed == Uint128::zero() { - if config.start_height != 0 || config.end_height != 0 { - // Check if the current block height is less than the start height - // and if the start height is less than the end height - if env.block.height < config.start_height && config.start_height < config.end_height { - // Check if the airdrop amount is sufficient to supply all users - if config.airdrop_amount >= get_total_in_user_info(storage) { - // Get the contract's bank balance - let current_airdrop_config = AIRDROP_CONFIG.load(storage)?; - let contract_balance = current_airdrop_config - .airdrop_asset - .query_balance(&querier, &env.contract.address)?; - - // Check if the contract has enough funds for the airdrop - if contract_balance < config.airdrop_amount { - return Err(AirdropErrors::InsufficientFundsInContractAccount { - balance: contract_balance, - }); - } - } else { - return Err(AirdropErrors::ConfigAmountLessThanTotalClaimable {}); - } - } else { - return Err(AirdropErrors::InvalidAirdropWindow {}); - } - } - } else { - return Err(AirdropErrors::NonZeroClaimedAmount {}); - } - Ok(Response::default()) -} - -/// Calculates the total claimable amount from all users. -/// -/// # Arguments -/// -/// * `storage` - Storage to access user information. -/// -/// # Returns -/// -/// Returns the total claimable amount from all users. -pub fn get_total_in_user_info(storage: &dyn Storage) -> Uint128 { - let mut total_claimable_amount = Uint128::zero(); - - for res in USER_INFO.range(storage, None, None, Order::Ascending) { - let claimed = res.as_ref().unwrap().1.get_claimed_flag(); - if !claimed { - total_claimable_amount += res.unwrap().1.get_claimable_amount() - } - } - - // Return the total claimable amount - total_claimable_amount -} - -#[cfg(test)] -mod tests { - use cosmwasm_std::testing::mock_dependencies; - - use crate::state::UserInfo; - - use super::*; - - #[test] - fn test_get_total_in_user_info() { - // Create a mock context and storage - let mut deps = mock_dependencies(); - - // Initialize the USER_INFO map in the mock storage with sample data - USER_INFO - .save( - &mut deps.storage, - "user1".parse().unwrap(), - &UserInfo { - claimable_amount: Uint128::new(100), - claimed_flag: false, - }, - ) - .unwrap(); - USER_INFO - .save( - &mut deps.storage, - "user2".parse().unwrap(), - &UserInfo { - claimable_amount: Uint128::new(200), - claimed_flag: false, - }, - ) - .unwrap(); - - // Call the function to be tested - let total_claimable_amount = get_total_in_user_info(deps.as_mut().storage); - - // Check the result against the expected total claimable amount - assert_eq!(total_claimable_amount, Uint128::new(300)); - } - - #[test] - fn test_validate_amount() { - let err = validate_amount(User { - address: "test".to_string(), - amount: Uint128::new(0), - }) - .unwrap_err(); - assert_eq!( - AirdropErrors::ZeroAmount { - address: "test".to_string() - }, - err - ); - let _resp = validate_amount(User { - address: "test".to_string(), - amount: Uint128::new(10), - }) - .unwrap(); - } - - // #[test] - // fn test_is_contract_admin() { - // let mut deps = mock_dependencies(); - // let env = mock_env(); - // let info = mock_info("admin", &[Coin::new(100000000, "uqsr")]); - // - // - // let qx: MockQuerier = MockQuerier::new(&[]); - // let q = QuerierWrapper::new(&qx); - // - // - // - // let a= is_contract_admin(&q, &env, &Addr::unchecked("admin")); - // println!("{:?}", a); - // } -} diff --git a/smart-contracts/osmosis/contracts/airdrop/src/lib.rs b/smart-contracts/osmosis/contracts/airdrop/src/lib.rs deleted file mode 100644 index b0c16682e..000000000 --- a/smart-contracts/osmosis/contracts/airdrop/src/lib.rs +++ /dev/null @@ -1,10 +0,0 @@ -pub mod admin; -pub mod contract; -pub mod helpers; -pub mod msg; -pub mod query; -pub mod state; -pub mod users; - -mod error; -pub use crate::error::AirdropErrors; diff --git a/smart-contracts/osmosis/contracts/airdrop/src/msg.rs b/smart-contracts/osmosis/contracts/airdrop/src/msg.rs deleted file mode 100644 index a7d1b5b80..000000000 --- a/smart-contracts/osmosis/contracts/airdrop/src/msg.rs +++ /dev/null @@ -1,90 +0,0 @@ -use cosmwasm_schema::{cw_serde, QueryResponses}; -use cosmwasm_std::Uint128; - -use crate::state::{AirdropConfig, UserInfo}; - -#[cw_serde] -pub struct InstantiateMsg { - pub config: AirdropConfig, -} - -#[cw_serde] -pub enum ExecuteMsg { - /// admin contains all the messages that can be executed by admin permission only - Admin(AdminExecuteMsg), - - /// claim airdrop is for the users to execute a specific airdrop id - ClaimAirdrop(), -} - -#[cw_serde] -pub enum AdminExecuteMsg { - /// updates airdrop config given by the admin - UpdateAirdropConfig(AirdropConfig), - - /// add users to the airdrop with the given amounts - AddUsers { users: Vec }, - - /// updates the existing users with the given address and amounts - SetUsers { users: Vec }, - - /// remove a list of users from an airdrop - RemoveUsers(Vec), - - /// sends back the remaining funds to the quasar funding address - WithdrawFunds(String), -} - -#[cw_serde] -#[derive(QueryResponses)] -pub enum QueryMsg { - /// airdrop config shows the current config set on the contract - #[returns(ConfigResponse)] - AirdropConfigQuery {}, - - #[returns(UserInfoResponse)] - UserInfoQuery { user: String }, - - #[returns(ContractStateResponse)] - ContractStateQuery {}, - - #[returns(SanityCheckResponse)] - SanityCheckQuery {}, - - #[returns(UsersStatsResponse)] - UsersStatsQuery {}, -} - -#[cw_serde] -pub struct ConfigResponse { - pub airdrop_config: AirdropConfig, -} - -#[cw_serde] -pub struct UserInfoResponse { - pub user_info: UserInfo, -} - -#[cw_serde] -pub struct ContractStateResponse { - pub airdrop_config: AirdropConfig, - pub user_info: Vec<(String, UserInfo)>, -} - -#[cw_serde] -pub struct SanityCheckResponse { - pub response: bool, -} - -#[cw_serde] -pub struct UsersStatsResponse { - pub claimed_users_count: u64, - pub unclaimed_users_count: u64, - pub total_users_count: u64, -} - -#[cw_serde] -pub struct User { - pub address: String, - pub amount: Uint128, -} diff --git a/smart-contracts/osmosis/contracts/airdrop/src/query.rs b/smart-contracts/osmosis/contracts/airdrop/src/query.rs deleted file mode 100644 index da22477da..000000000 --- a/smart-contracts/osmosis/contracts/airdrop/src/query.rs +++ /dev/null @@ -1,131 +0,0 @@ -use std::string::String; - -use crate::helpers::get_total_in_user_info; -use cosmwasm_std::{Deps, Env, Order, StdResult}; - -use crate::msg::{ - ConfigResponse, ContractStateResponse, SanityCheckResponse, UserInfoResponse, - UsersStatsResponse, -}; -use crate::state::{UserInfo, AIRDROP_CONFIG, USER_INFO}; - -/// Queries and returns the current airdrop configuration. -/// -/// # Arguments -/// -/// * `deps` - Deps is a struct providing access to the contract's dependencies like storage. -/// -/// # Returns -/// -/// Returns a `ConfigResponse` containing the current airdrop configuration. -pub fn query_config(deps: Deps) -> StdResult { - let config = AIRDROP_CONFIG.load(deps.storage)?; - Ok(ConfigResponse { - airdrop_config: config, - }) -} - -/// Queries and returns the information of a specific user. -/// -/// # Arguments -/// -/// * `deps` - Deps is a struct providing access to the contract's dependencies like storage. -/// * `user` - The address of the user for which to retrieve information. -/// -/// # Returns -/// -/// Returns a `UserInfoResponse` containing the user's information. -pub fn query_user(deps: Deps, user: String) -> StdResult { - let user_addr = deps.api.addr_validate(&user)?; - let user_info = USER_INFO.load(deps.storage, user_addr.to_string())?; - Ok(UserInfoResponse { user_info }) -} - -/// Queries and returns the entire contract state, including airdrop configuration and user information. -/// -/// # Arguments -/// -/// * `deps` - Deps is a struct providing access to the contract's dependencies like storage. -/// -/// # Returns -/// -/// Returns a `ContractStateResponse` containing the airdrop configuration and user information. -pub fn query_contract_state(deps: Deps) -> StdResult { - let config = AIRDROP_CONFIG.load(deps.storage)?; - let mut user_infos: Vec<(String, UserInfo)> = Vec::new(); - for res in USER_INFO.range(deps.storage, None, None, Order::Ascending) { - let unwrapped_res = res.unwrap(); - user_infos.push((unwrapped_res.0, unwrapped_res.1)) - } - Ok(ContractStateResponse { - airdrop_config: config, - user_info: user_infos, - }) -} - -/// Performs a sanity check to verify if there are sufficient funds for airdrop payments. -/// -/// # Arguments -/// -/// * `deps` - Deps is a struct providing access to the contract's dependencies like storage and querier. -/// * `env` - Environment information. -/// -/// # Returns -/// -/// Returns a `SanityCheckResponse` indicating whether there are sufficient funds for airdrop payments. -pub fn query_airdrop_readiness(deps: Deps, env: Env) -> StdResult { - // Check if the airdrop amount is sufficient to supply all users - let airdrop_config = AIRDROP_CONFIG.load(deps.storage)?; - if airdrop_config.airdrop_amount >= get_total_in_user_info(deps.storage) { - // Get the contract's bank balance - let contract_balance = airdrop_config - .airdrop_asset - .query_balance(&deps.querier, env.contract.address) - .unwrap(); - - // Check if the contract has enough funds for the airdrop - if contract_balance < airdrop_config.airdrop_amount { - return Ok(SanityCheckResponse { response: false }); - } - } else { - return Ok(SanityCheckResponse { response: false }); - } - Ok(SanityCheckResponse { response: true }) -} - -/// Query user statistics including the number of claimed and unclaimed users. -/// -/// # Arguments -/// -/// - `deps`: A reference to the dependencies needed for the query. -/// -/// # Returns -/// -/// A result containing a `UsersStatsResponse`, which includes the counts of -/// claimed and unclaimed users, and the total number of users. -/// -/// # Errors -/// -/// This function returns a `StdResult` that can hold a `CosmosSDK` error. -/// -pub fn query_users_stats(deps: Deps) -> StdResult { - // Initialize counters for claimed and total users. - let mut claimed_users_count = 0u64; - let mut total_users_count = 0u64; - - // Iterate through user info and count claimed and total users. - for res in USER_INFO.range(deps.storage, None, None, Order::Ascending) { - let claimed = res.as_ref().unwrap().1.get_claimed_flag(); - if claimed { - claimed_users_count += 1; - } - total_users_count += 1; - } - - // Create and return the UsersStatsResponse. - Ok(UsersStatsResponse { - claimed_users_count, - unclaimed_users_count: total_users_count - claimed_users_count, - total_users_count, - }) -} diff --git a/smart-contracts/osmosis/contracts/airdrop/src/state.rs b/smart-contracts/osmosis/contracts/airdrop/src/state.rs deleted file mode 100644 index def711562..000000000 --- a/smart-contracts/osmosis/contracts/airdrop/src/state.rs +++ /dev/null @@ -1,74 +0,0 @@ -use std::string::String; - -use cosmwasm_schema::cw_serde; -use cosmwasm_std::Uint128; -use cw_asset::AssetInfo; -use cw_storage_plus::{Item, Map}; - -pub const AIRDROP_CONFIG: Item = Item::new("airdrop_config"); -pub const USER_INFO: Map = Map::new("user_info"); - -//---------------------------------------------------------------------------------------- -// Storage types -//---------------------------------------------------------------------------------------- - -#[cw_serde] -pub struct AirdropConfig { - pub airdrop_title: String, - /// every airdrop contains a description of it - pub airdrop_description: String, - /// token amount to be airdropped - pub airdrop_amount: Uint128, - /// token denom to be airdropped - pub airdrop_asset: AssetInfo, - /// total claimed amount, zero initially - pub total_claimed: Uint128, - /// starting time from which users can claim airdrop - pub start_height: u64, - /// end time after which users cannot claim airdrop - pub end_height: u64, -} - -#[cw_serde] -pub struct UserInfo { - /// total airdrop tokens claimable by the user - pub claimable_amount: Uint128, - /// boolean value indicating if the user has withdrawn the remaining tokens - pub claimed_flag: bool, -} - -impl AirdropConfig { - pub fn get_start_height(&self) -> u64 { - self.start_height - } - pub fn get_end_heights(&self) -> u64 { - self.start_height - } - pub fn get_config(self) -> AirdropConfig { - self - } - pub fn is_airdrop_active(&self, current_height: u64) -> bool { - if self.start_height <= current_height && self.end_height >= current_height { - return true; - } - false - } -} - -impl UserInfo { - pub fn get_claimable_amount(&self) -> Uint128 { - self.claimable_amount - } - pub fn get_claimed_flag(&self) -> bool { - self.claimed_flag - } -} - -impl Default for UserInfo { - fn default() -> Self { - UserInfo { - claimable_amount: Uint128::zero(), - claimed_flag: false, - } - } -} diff --git a/smart-contracts/osmosis/contracts/airdrop/src/users.rs b/smart-contracts/osmosis/contracts/airdrop/src/users.rs deleted file mode 100644 index 2172c18d5..000000000 --- a/smart-contracts/osmosis/contracts/airdrop/src/users.rs +++ /dev/null @@ -1,63 +0,0 @@ -use cosmwasm_std::{Addr, DepsMut, Env, Response}; -use cw_asset::Asset; - -use crate::state::{AIRDROP_CONFIG, USER_INFO}; -use crate::AirdropErrors; - -// Define a function to process airdrop claims for a user -pub fn execute_claim(deps: DepsMut, env: Env, user: Addr) -> Result { - let current_airdrop_config = AIRDROP_CONFIG.load(deps.storage)?; - - // Check if the airdrop window is open and the user is eligible to claim - if current_airdrop_config.start_height == 0 - || current_airdrop_config.end_height == 0 - || env.block.height > current_airdrop_config.end_height - || env.block.height < current_airdrop_config.start_height - { - return Err(AirdropErrors::InvalidClaim {}); - } - - // Load the user's airdrop information from storage - let mut user_info = USER_INFO.load(deps.storage, user.to_string())?; - - // Check if the user has already claimed the airdrop - if user_info.get_claimed_flag() { - return Err(AirdropErrors::AlreadyClaimed {}); - } - - // Get the admin address of the contract - let current_airdrop_config = AIRDROP_CONFIG.load(deps.storage)?; - let contract_balance = current_airdrop_config - .airdrop_asset - .query_balance(&deps.querier, &env.contract.address)?; - - // Check if the user's claimable amount exceeds the contract's balance - if user_info.get_claimable_amount() > contract_balance { - return Err(AirdropErrors::InsufficientFundsInContractAccount { - balance: contract_balance, - }); - } - - // Transfer the airdrop asset to the withdrawal address - let claim = Asset::new( - current_airdrop_config.airdrop_asset, - user_info.claimable_amount, - ) - .transfer_msg(user.clone())?; - - // Mark the user as claimed - user_info.claimed_flag = true; - USER_INFO.save(deps.storage, user.to_string(), &user_info)?; - - // Update the airdrop configuration by increasing the total claimed amount - let mut airdrop_config = AIRDROP_CONFIG.load(deps.storage)?; - airdrop_config.total_claimed += user_info.claimable_amount; - AIRDROP_CONFIG.save(deps.storage, &airdrop_config)?; - - // Return a default response if all checks pass - Ok(Response::new().add_message(claim).add_attributes(vec![ - ("action", "claim"), - ("user", user.as_ref()), - ("amount", &user_info.claimable_amount.to_string()), - ])) -} diff --git a/smart-contracts/osmosis/contracts/basic-vault/.cargo/config b/smart-contracts/osmosis/contracts/basic-vault/.cargo/config deleted file mode 100644 index 8d4bc738b..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/.cargo/config +++ /dev/null @@ -1,6 +0,0 @@ -[alias] -wasm = "build --release --target wasm32-unknown-unknown" -wasm-debug = "build --target wasm32-unknown-unknown" -unit-test = "test --lib" -integration-test = "test --test integration" -schema = "run --example schema" diff --git a/smart-contracts/osmosis/contracts/basic-vault/.gitignore b/smart-contracts/osmosis/contracts/basic-vault/.gitignore deleted file mode 100644 index 360bb05b1..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/.gitignore +++ /dev/null @@ -1,2 +0,0 @@ -ts -schema diff --git a/smart-contracts/osmosis/contracts/basic-vault/CHANGELOG.md b/smart-contracts/osmosis/contracts/basic-vault/CHANGELOG.md deleted file mode 100644 index 466ae582b..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/CHANGELOG.md +++ /dev/null @@ -1,20 +0,0 @@ -# CHANGELOG - -## V0.1.1 -### Initial version -### Dependencies -### API breaking -### State breaking -### Improvements -### Features -### Bugfixes -- Update user index for rewards contract on cw20 TransferFrom and SendFrom - -## V0.1.0 - \ No newline at end of file diff --git a/smart-contracts/osmosis/contracts/basic-vault/Cargo.toml b/smart-contracts/osmosis/contracts/basic-vault/Cargo.toml deleted file mode 100644 index 2ea4bcef9..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/Cargo.toml +++ /dev/null @@ -1,47 +0,0 @@ -[package] -authors = ["N ", "Laurens Kubat "] -description = "a basic vault implementation to " -documentation = "https://docs.quasar.fi/" -edition = "2018" -homepage = "https://quasar.fi" -license = "Apache-2.0" -name = "basic-vault" -repository = "https://github.com/quasar-finance/quasar" -version = "0.1.1" - -# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html - -[lib] -crate-type = ["cdylib", "rlib"] - -[features] -backtraces = ["cosmwasm-std/backtraces"] -# use library feature to disable all instantiate/execute/query exports -library = [] - -[dependencies] -thiserror = { workspace = true } -cosmwasm-schema = { workspace = true } -cosmwasm-std = { workspace = true } -cw-storage-plus = { workspace = true } -osmosis-std = { workspace = true } -cw-controllers = { workspace = true } -cw-utils = { workspace = true } -cw2 = { workspace = true } -cw20 = { workspace = true } -cw20-base = { workspace = true } -serde-json-wasm = { workspace = true } -cw-asset = { workspace = true } -cw20-staking = { workspace = true } - -# Quasar contracts and packages -lp-strategy = {path = "../lp-strategy", features = ["library"], default-features = false} -vault-rewards = {path = "../vault-rewards", features = ["library"], default-features = false} -quasar-types = {path = "../../packages/quasar-types"} - -[dev-dependencies] -cw-multi-test = { workspace = true } -prost = { workspace = true } - -anyhow = "1" -derivative = "2" diff --git a/smart-contracts/osmosis/contracts/basic-vault/NOTICE b/smart-contracts/osmosis/contracts/basic-vault/NOTICE deleted file mode 100644 index 55148b76b..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/NOTICE +++ /dev/null @@ -1,15 +0,0 @@ -CW20-Staking: Staking Derivatives as a CW20 token -Copyright 2020-21 Ethan Frey -Copyright 2021-22 Confio GmbH - -Licensed under the Apache License, Version 2.0 (the "License"); -you may not use this file except in compliance with the License. -You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - -Unless required by applicable law or agreed to in writing, software -distributed under the License is distributed on an "AS IS" BASIS, -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -See the License for the specific language governing permissions and -limitations under the License. diff --git a/smart-contracts/osmosis/contracts/basic-vault/README.md b/smart-contracts/osmosis/contracts/basic-vault/README.md deleted file mode 100644 index 9a23707dc..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/README.md +++ /dev/null @@ -1,25 +0,0 @@ -# Staking Derivatives - -This is a sample contract that releases a minimal form of staking derivatives. -This is to be used for integration tests and as a foundation for other to build -more complex logic upon. - -## Functionality - -On one side, this acts as a CW20 token, holding a list of -balances for multiple addresses, and exposing queries and transfers (no -allowances and "transfer from" to focus the logic on the staking stuff). -However, it has no initial balance. Instead, it mints and burns them based on -delegations. - -For such a "bonding curve" we expose two additional message types. A "bond" -message sends native staking tokens to the contract to be bonded to a validator -and credits the user with the appropriate amount of derivative tokens. Likewise -you can burn some of your derivative tokens, and the contract will unbond the -proportional amount of stake to the user's account (after typical 21-day -unbonding period). - -To show an example of charging for such a service, we allow the contract owner -to take a small exit tax, thus maybe 98% of the tokens will be unbonded and sent -to the original account, and 2% of the tokens are not unbonded, but rather -transferred to the owners account. (The ownership can also be transferred). diff --git a/smart-contracts/osmosis/contracts/basic-vault/examples/schema.rs b/smart-contracts/osmosis/contracts/basic-vault/examples/schema.rs deleted file mode 100644 index f98d5bbfe..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/examples/schema.rs +++ /dev/null @@ -1,10 +0,0 @@ -use basic_vault::msg::{ExecuteMsg, InstantiateMsg, QueryMsg}; -use cosmwasm_schema::write_api; - -fn main() { - write_api! { - instantiate: InstantiateMsg, - query: QueryMsg, - execute: ExecuteMsg, - } -} diff --git a/smart-contracts/osmosis/contracts/basic-vault/src/callback.rs b/smart-contracts/osmosis/contracts/basic-vault/src/callback.rs deleted file mode 100644 index 29bf2866f..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/src/callback.rs +++ /dev/null @@ -1,400 +0,0 @@ -use cosmwasm_std::{ - Addr, BankMsg, DepsMut, Env, MessageInfo, Response, SubMsg, Timestamp, Uint128, -}; -use cw20_base::contract::execute_mint; -use quasar_types::callback::{BondResponse, UnbondResponse}; - -use crate::{ - helpers::update_user_reward_index, - state::{ - Unbond, BONDING_SEQ_TO_ADDR, BOND_STATE, DEBUG_TOOL, INVESTMENT, PENDING_BOND_IDS, - PENDING_UNBOND_IDS, UNBOND_STATE, - }, - ContractError, -}; - -pub fn on_bond( - deps: DepsMut, - env: Env, - info: MessageInfo, - share_amount: Uint128, - bond_id: String, -) -> Result { - DEBUG_TOOL.save( - deps.storage, - &format!("We hit on_bond with bond_id: {bond_id}"), - )?; - - // load investment info - let invest = INVESTMENT.load(deps.storage)?; - - let mut bond_stubs = BOND_STATE.load(deps.storage, bond_id.clone())?; - - // lets find the primitive for this response - let primitive_config = invest.primitives.iter().find(|p| p.address == info.sender); - - // if we don't find a primitive, this is an unauthorized call - if primitive_config.is_none() { - return Err(ContractError::Unauthorized {}); - } - - // if we find a bond_stub coming from a primitive that already sent us one. fail - if bond_stubs - .iter() - .any(|s| s.address == info.sender && s.bond_response.is_some()) - { - return Err(ContractError::DuplicateBondResponse { bond_id }); - } - - // update deposit state here before doing anything else & save! - bond_stubs.iter_mut().for_each(|s| { - if s.address == info.sender { - // we should probably return the primitive value in the bond response - let primitive_value: lp_strategy::msg::IcaBalanceResponse = deps - .querier - .query_wasm_smart( - info.sender.clone(), - &lp_strategy::msg::QueryMsg::IcaBalance {}, - ) - .unwrap(); - s.bond_response = Some(BondResponse { - share_amount, - bond_id: bond_id.clone(), - }); - s.primitive_value = Some(primitive_value.amount.amount); - } - }); - - BOND_STATE.save(deps.storage, bond_id.clone(), &bond_stubs)?; - - // if still waiting on successful bonds, then return - if bond_stubs.iter().any(|s| s.bond_response.is_none()) { - return Ok(Response::new() - .add_attribute("action", "on_bond") - .add_attribute( - "state", - bond_stubs - .iter() - .fold(0u32, |acc, stub| { - if stub.bond_response.is_none() { - acc + 1 - } else { - acc - } - }) - .to_string() - + "pending bonds", - )); - } - // at this point we know that the deposit has succeeded fully, and we can mint shares - - let user_address = BONDING_SEQ_TO_ADDR.load(deps.storage, bond_id.clone())?; - let validated_user_address = deps.api.addr_validate(&user_address)?; - // lets updated all pending deposit info - PENDING_BOND_IDS.update(deps.storage, validated_user_address.clone(), |ids| { - if let Some(mut bond_ids) = ids { - let bond_index = bond_ids.iter().position(|id| id.eq(&bond_id)).ok_or( - ContractError::IncorrectCallbackId { - expected: bond_id.clone(), - ids: bond_ids.clone(), - }, - )?; - bond_ids.remove(bond_index); - Ok::, ContractError>(bond_ids) - } else { - Ok(vec![]) - } - })?; - - BOND_STATE.save(deps.storage, bond_id, &bond_stubs)?; - - // calculate the shares to mint by value - // at the time of bonding, we want to figure out what percentage of value in the vault the user has - - // User value per primitive = BondResponse Primitive Shares / Total Primitive Shares * ICA_BALANCE or funds send in the bond - // Total Vault Value = Total Vault shares in Primitive / Total Primitive shares * ICA_BALANCE - let total_vault_value = bond_stubs.iter().try_fold( - Uint128::zero(), - |acc, stub| -> Result { - Ok(acc - + stub - .primitive_value - .ok_or(ContractError::BondResponseIsEmpty {})?) - }, - )?; - - // User Vault Shares = Sum(user value per primitive) / Total Vault value * Total Vault Shares - let total_user_value = bond_stubs - .iter() - .fold(Uint128::zero(), |acc, stub| acc + stub.amount); - - let token_info = cw20_base::contract::query_token_info(deps.as_ref())?; - // equal to the cw20 base total supply - let total_vault_shares: Uint128 = token_info.total_supply; - - //if either is zero, then we just mint the user value - let shares_to_mint = if total_vault_shares.is_zero() || total_vault_value.is_zero() { - total_user_value - } else { - total_user_value - .checked_multiply_ratio(total_vault_shares, total_vault_value - total_user_value)? - }; - - // call into cw20-base to mint the token, call as self as no one else is allowed - let sub_info = MessageInfo { - sender: env.contract.address.clone(), - funds: vec![], - }; - - let update_user_rewards_idx_msg = - update_user_reward_index(deps.as_ref().storage, &validated_user_address)?; - execute_mint(deps, env, sub_info, user_address, shares_to_mint)?; - - let res = Response::new() - .add_submessage(SubMsg::new(update_user_rewards_idx_msg)) - .add_attribute("action", "on_bond") - .add_attribute("from", info.sender) - .add_attribute("minted", shares_to_mint); - Ok(res) -} - -pub fn on_start_unbond( - deps: DepsMut, - _env: Env, - info: MessageInfo, - unbond_id: String, - unlock_time: Timestamp, -) -> Result { - let invest = INVESTMENT.load(deps.storage)?; - let primitive_config = invest.primitives.iter().find(|p| p.address == info.sender); - - // if we don't find a primitive, this is an unauthorized call - if primitive_config.is_none() { - return Err(ContractError::Unauthorized {}); - } - - UNBOND_STATE.update( - deps.storage, - unbond_id.clone(), - |s: Option| -> Result { - let mut unbond = s.ok_or(ContractError::UnbondIsEmpty {})?; - // update the stub where the address is the same as message sender with the unlock time - - unbond - .stub - .iter_mut() - .find(|s| s.address == info.sender) - .ok_or(ContractError::UnbondStubIsEmpty {})? - .unlock_time = Option::Some(unlock_time); - Ok(Unbond { - stub: unbond.stub, - shares: unbond.shares, - }) - }, - )?; - - Ok(Response::new() - .add_attribute("action", "on_start_unbond") - .add_attribute("unbond_id", unbond_id) - .add_attribute("unlock_time", unlock_time.to_string())) -} - -pub fn on_unbond( - deps: DepsMut, - _env: Env, - info: MessageInfo, - unbond_id: String, -) -> Result { - let invest = INVESTMENT.load(deps.storage)?; - let primitive_config = invest.primitives.iter().find(|p| p.address == info.sender); - - // if we don't find a primitive, this is an unauthorized call - if primitive_config.is_none() { - return Err(ContractError::Unauthorized {}); - } - - let mut unbond_stubs = UNBOND_STATE.load(deps.storage, unbond_id.clone())?; - - // edit and save the stub where the address is the same as message sender with the unbond response - let unbonding_stub = unbond_stubs - .stub - .iter_mut() - .find(|s| s.address == info.sender) - .ok_or(ContractError::UnbondStubIsEmpty {})?; - - // update info - unbonding_stub.unbond_response = Option::Some(UnbondResponse { - unbond_id: unbond_id.clone(), - }); - unbonding_stub.unbond_funds = info.funds; - - UNBOND_STATE.save(deps.storage, unbond_id.clone(), &unbond_stubs)?; - - // if still waiting on successful unbonds, then return - // todo: should we eagerly send back funds? - if unbond_stubs - .stub - .iter() - .any(|s| s.unbond_response.is_none()) - { - return Ok(Response::new()); - } - - let user_address = BONDING_SEQ_TO_ADDR.load(deps.storage, unbond_id.clone())?; - // Construct message to return these funds to the user - let return_msgs: Vec = unbond_stubs - .stub - .iter() - .map(|s| BankMsg::Send { - to_address: user_address.to_string(), - amount: s.unbond_funds.clone(), - }) - .collect(); - - // delete this pending unbond id from the state - UNBOND_STATE.remove(deps.storage, unbond_id.clone()); - PENDING_UNBOND_IDS.update( - deps.storage, - Addr::unchecked(user_address), - |ids| -> Result, ContractError> { - Ok(ids - .ok_or(ContractError::NoPendingUnbonds {})? - .into_iter() - .filter(|id| id != &unbond_id) - .collect()) - }, - )?; - - Ok(Response::new() - .add_messages(return_msgs) - .add_attribute("action", "on_unbond") - .add_attribute("unbond_id", unbond_id)) -} - -#[cfg(test)] -mod test { - use std::str::FromStr; - - use crate::multitest::common::PrimitiveInstantiateMsg; - use crate::state::{BondingStub, BOND_STATE}; - use crate::tests::mock_deps_with_primitives; - use crate::{ - callback::on_bond, - msg::PrimitiveConfig, - multitest::common::{DENOM, LOCAL_DENOM}, - state::{InvestmentInfo, INVESTMENT}, - ContractError, - }; - use cosmwasm_std::{ - testing::{mock_env, mock_info}, - Decimal, - }; - use cosmwasm_std::{Addr, Uint128}; - - #[test] - fn fail_if_duplicate_bond_id() { - let primitive_states = vec![ - ( - "addr00001".to_string(), - LOCAL_DENOM.to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - ), - ( - "addr00002".to_string(), - LOCAL_DENOM.to_string(), - Uint128::from(200u128), - Uint128::from(400u128), - ), - ]; - // mock the queries so the primitives exist - let mut deps = mock_deps_with_primitives(primitive_states); - let env = mock_env(); - let info = mock_info("addr00001", &[]); - - INVESTMENT - .save( - &mut deps.storage, - &InvestmentInfo { - primitives: vec![ - PrimitiveConfig { - weight: Decimal::from_str("0.33333333333").unwrap(), - address: "addr00001".to_string(), - init: crate::msg::PrimitiveInitMsg::LP(PrimitiveInstantiateMsg { - lock_period: 64, - pool_id: 1, - pool_denom: "gamm/pool/1".to_string(), - local_denom: LOCAL_DENOM.to_string(), - base_denom: DENOM.to_string(), - quote_denom: "uatom".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }), - }, - PrimitiveConfig { - weight: Decimal::from_str("0.33333333333").unwrap(), - address: "addr00002".to_string(), - init: crate::msg::PrimitiveInitMsg::LP(PrimitiveInstantiateMsg { - lock_period: 64, - pool_id: 2, - pool_denom: "gamm/pool/1".to_string(), - local_denom: LOCAL_DENOM.to_string(), - base_denom: DENOM.to_string(), - quote_denom: "uatom".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }), - }, - ], - owner: Addr::unchecked("owner".to_string()), - min_withdrawal: 1u128.into(), - deposit_denom: LOCAL_DENOM.to_string(), - }, - ) - .unwrap(); - - let share_amount = 100u128; - let bond_id = "1".to_string(); - - BOND_STATE - .save( - &mut deps.storage, - bond_id.clone(), - &vec![ - BondingStub { - address: "addr00001".to_string(), - bond_response: None, - primitive_value: None, - amount: Uint128::one(), - }, - BondingStub { - address: "addr00002".to_string(), - bond_response: None, - primitive_value: None, - amount: Uint128::one(), - }, - ], - ) - .unwrap(); - - // first bond should work - let res = on_bond( - deps.as_mut(), - env.clone(), - info.clone(), - share_amount.into(), - bond_id.clone(), - ) - .unwrap(); - assert_eq!(0, res.messages.len()); - - // second bond should fail - let res = on_bond(deps.as_mut(), env, info, share_amount.into(), bond_id).unwrap_err(); - match res { - ContractError::DuplicateBondResponse { .. } => {} - _ => panic!("Unexpected error: {:?}", res), - } - } -} diff --git a/smart-contracts/osmosis/contracts/basic-vault/src/contract.rs b/smart-contracts/osmosis/contracts/basic-vault/src/contract.rs deleted file mode 100644 index 255659366..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/src/contract.rs +++ /dev/null @@ -1,614 +0,0 @@ -#[cfg(not(feature = "library"))] -use cosmwasm_std::entry_point; -use cosmwasm_std::{ - to_json_binary, Binary, CosmosMsg, Deps, DepsMut, Env, MessageInfo, Order, Reply, Response, - StdError, StdResult, SubMsg, SubMsgResult, Uint128, WasmMsg, -}; - -use cw2::set_contract_version; -use cw20_base::allowances::{ - execute_burn_from, execute_decrease_allowance, execute_increase_allowance, execute_send_from, - execute_transfer_from, query_allowance, -}; -use cw20_base::contract::{ - execute_burn, execute_send, execute_transfer, query_balance, query_token_info, -}; -use cw20_base::state::{MinterData, TokenInfo, TOKEN_INFO}; -use cw_utils::parse_instantiate_response_data; -use lp_strategy::msg::ConfigResponse; -use vault_rewards::msg::InstantiateMsg as VaultRewardsInstantiateMsg; - -use crate::callback::{on_bond, on_start_unbond, on_unbond}; -use crate::error::ContractError; -use crate::execute::{bond, claim, unbond, update_cap}; -use crate::helpers::update_user_reward_index; -use crate::msg::{ - ExecuteMsg, GetCapResponse, GetDebugResponse, InstantiateMsg, MigrateMsg, PrimitiveConfig, - QueryMsg, VaultTokenInfoResponse, -}; -use crate::query::{ - query_deposit_ratio, query_investment, query_pending_bonds, query_pending_bonds_by_id, - query_pending_unbonds, query_pending_unbonds_by_id, query_tvl_info, -}; -use crate::state::{ - AdditionalTokenInfo, Cap, InvestmentInfo, ADDITIONAL_TOKEN_INFO, BONDING_SEQ, CAP, CLAIMS, - CONTRACT_NAME, CONTRACT_VERSION, DEBUG_TOOL, INVESTMENT, VAULT_REWARDS, -}; - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn instantiate( - deps: DepsMut, - env: Env, - info: MessageInfo, - msg: InstantiateMsg, -) -> Result { - set_contract_version(deps.storage, CONTRACT_NAME, CONTRACT_VERSION)?; - - // store token info using cw20-base format - let token_info = TokenInfo { - name: msg.name, - symbol: msg.symbol, - decimals: msg.decimals, - total_supply: Uint128::zero(), - // set self as minter, so we can properly execute mint and burn - mint: Some(MinterData { - minter: env.contract.address.clone(), - cap: None, - }), - }; - let additional_info = AdditionalTokenInfo { - creation_time: env.block.time, - thesis: msg.thesis, - }; - TOKEN_INFO.save(deps.storage, &token_info)?; - ADDITIONAL_TOKEN_INFO.save(deps.storage, &additional_info)?; - - CAP.save(deps.storage, &Cap::new(info.sender.clone(), msg.total_cap))?; - - for prim in msg.primitives.iter() { - let config: ConfigResponse = deps - .querier - .query_wasm_smart(&prim.address, &lp_strategy::msg::QueryMsg::Config {})?; - match &prim.init { - crate::msg::PrimitiveInitMsg::LP(init) => { - assert_eq!(config.config.base_denom, init.base_denom); - assert_eq!(config.config.expected_connection, init.expected_connection); - assert_eq!(config.config.local_denom, init.local_denom); - assert_eq!(config.config.lock_period, init.lock_period); - assert_eq!(config.config.pool_denom, init.pool_denom); - assert_eq!(config.config.pool_id, init.pool_id); - assert_eq!(config.config.quote_denom, init.quote_denom); - assert_eq!( - config.config.return_source_channel, - init.return_source_channel - ); - assert_eq!(config.config.transfer_channel, init.transfer_channel); - } - } - } - - let mut invest = InvestmentInfo { - owner: info.sender.clone(), - min_withdrawal: msg.min_withdrawal, - primitives: msg.primitives, - deposit_denom: msg.deposit_denom, - }; - invest.normalize_primitive_weights(); - INVESTMENT.save(deps.storage, &invest)?; - - // initialize bonding sequence num - BONDING_SEQ.save(deps.storage, &Uint128::one())?; - - DEBUG_TOOL.save(deps.storage, &"Empty".to_string())?; - - let init_vault_rewards = SubMsg::reply_always( - WasmMsg::Instantiate { - admin: Some(info.sender.to_string()), - code_id: msg.vault_rewards_code_id, - msg: to_json_binary(&VaultRewardsInstantiateMsg { - vault_token: env.contract.address.to_string(), - reward_token: msg.reward_token, - distribution_schedules: msg.reward_distribution_schedules, - })?, - funds: vec![], - label: "vault-rewards".to_string(), - }, - REPLY_INIT_VAULT_REWARDS, - ); - - Ok(Response::new().add_submessage(init_vault_rewards)) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn execute( - deps: DepsMut, - env: Env, - info: MessageInfo, - msg: ExecuteMsg, -) -> Result { - match msg { - ExecuteMsg::Bond { recipient } => bond(deps, env, info, recipient), - ExecuteMsg::Unbond { amount } => unbond(deps, env, info, amount), - ExecuteMsg::Claim {} => claim(deps, env, info), - - // calls the TryIcq ExecuteMsg in the lp-strategy contract - ExecuteMsg::ClearCache {} => { - let try_icq_msg = lp_strategy::msg::ExecuteMsg::TryIcq {}; - - let mut msgs: Vec = vec![]; - - let primitives = INVESTMENT.load(deps.storage)?.primitives; - primitives.iter().try_for_each( - |pc: &PrimitiveConfig| -> Result<(), ContractError> { - let clear_cache_msg = WasmMsg::Execute { - contract_addr: pc.address.to_string(), - funds: vec![], - msg: to_json_binary(&try_icq_msg)?, - }; - msgs.push(clear_cache_msg); - Ok(()) - }, - )?; - Ok(Response::new().add_messages(msgs)) - } - - // Callbacks entrypoint - // you cant do this - DONT TRY IT (unless you know what you're doing) - // ExecuteMsg::Callback(callback_msg) => handle_callback(deps, env, info, callback_msg), - ExecuteMsg::BondResponse(bond_response) => on_bond( - deps, - env, - info, - bond_response.share_amount, - bond_response.bond_id, - ), - ExecuteMsg::StartUnbondResponse(start_unbond_response) => on_start_unbond( - deps, - env, - info, - start_unbond_response.unbond_id, - start_unbond_response.unlock_time, - ), - ExecuteMsg::UnbondResponse(unbond_response) => { - on_unbond(deps, env, info, unbond_response.unbond_id) - } - - // Admin messages - ExecuteMsg::SetCap { - new_total, - new_cap_admin, - } => update_cap(deps, env, info, new_total, new_cap_admin), - - // these all come from cw20-base to implement the cw20 standard - ExecuteMsg::Transfer { recipient, amount } => { - let recipient = deps.api.addr_validate(&recipient)?; - let update_user_reward_indexes = vec![ - update_user_reward_index(deps.storage, &info.sender)?, - update_user_reward_index(deps.storage, &recipient)?, - ]; - Ok( - execute_transfer(deps, env, info, recipient.to_string(), amount)? - .add_messages(update_user_reward_indexes), - ) - } - ExecuteMsg::Burn { amount } => { - let update_user_reward_index = update_user_reward_index(deps.storage, &info.sender)?; - Ok(execute_burn(deps, env, info, amount)?.add_message(update_user_reward_index)) - } - ExecuteMsg::Send { - contract, - amount, - msg, - } => { - let contract = deps.api.addr_validate(&contract)?; - let update_user_reward_indexes = vec![ - update_user_reward_index(deps.storage, &info.sender)?, - update_user_reward_index(deps.storage, &contract)?, - ]; - Ok( - execute_send(deps, env, info, contract.to_string(), amount, msg)? - .add_messages(update_user_reward_indexes), - ) - } - ExecuteMsg::IncreaseAllowance { - spender, - amount, - expires, - } => Ok(execute_increase_allowance( - deps, env, info, spender, amount, expires, - )?), - ExecuteMsg::DecreaseAllowance { - spender, - amount, - expires, - } => Ok(execute_decrease_allowance( - deps, env, info, spender, amount, expires, - )?), - ExecuteMsg::TransferFrom { - owner, - recipient, - amount, - } => { - let recipient = deps.api.addr_validate(&recipient)?; - let update_user_reward_indexes = vec![ - update_user_reward_index(deps.storage, &deps.api.addr_validate(&owner)?)?, - update_user_reward_index(deps.storage, &recipient)?, - ]; - Ok( - execute_transfer_from(deps, env, info, owner, recipient.to_string(), amount)? - .add_messages(update_user_reward_indexes), - ) - } - ExecuteMsg::BurnFrom { owner, amount } => { - let owner = deps.api.addr_validate(&owner)?; - let update_user_reward_index = update_user_reward_index(deps.storage, &owner)?; - Ok( - execute_burn_from(deps, env, info, owner.to_string(), amount)? - .add_message(update_user_reward_index), - ) - } - ExecuteMsg::SendFrom { - owner, - contract, - amount, - msg, - } => { - let contract = deps.api.addr_validate(&contract)?; - let update_user_reward_indexes = vec![ - update_user_reward_index(deps.storage, &deps.api.addr_validate(&owner)?)?, - update_user_reward_index(deps.storage, &contract)?, - ]; - Ok( - execute_send_from(deps, env, info, owner, contract.to_string(), amount, msg)? - .add_messages(update_user_reward_indexes), - ) - } - } -} - -pub const REPLY_INIT_VAULT_REWARDS: u64 = 777; - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn reply(deps: DepsMut, _env: Env, msg: Reply) -> Result { - match msg.id { - REPLY_INIT_VAULT_REWARDS => match msg.result { - SubMsgResult::Ok(res) => { - let vault_rewards = - parse_instantiate_response_data(res.data.unwrap().as_slice()).unwrap(); - update_rewards_contract(deps, vault_rewards.contract_address) - } - SubMsgResult::Err(e) => Err(StdError::generic_err(format!( - "error instantiating vault rewards contract: {e:?}" - )))?, - }, - _ => { - unimplemented!() - } - } -} - -fn update_rewards_contract( - deps: DepsMut, - rewards_contract: String, -) -> Result { - deps.api.addr_validate(&rewards_contract)?; - - VAULT_REWARDS.save(deps.storage, &deps.api.addr_validate(&rewards_contract)?)?; - - Ok(Response::default().add_attributes(vec![ - ("action", "init_vault_rewards"), - ("contract_address", rewards_contract.as_str()), - ])) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn query(deps: Deps, _env: Env, msg: QueryMsg) -> StdResult { - match msg { - QueryMsg::Claims { address } => { - to_json_binary(&CLAIMS.query_claims(deps, &deps.api.addr_validate(&address)?)?) - } - QueryMsg::Investment {} => to_json_binary(&query_investment(deps)?), - QueryMsg::TokenInfo {} => to_json_binary(&query_token_info(deps)?), - QueryMsg::AdditionalTokenInfo {} => to_json_binary(&query_vault_token_info(deps)?), - QueryMsg::Balance { address } => to_json_binary(&query_balance(deps, address)?), - QueryMsg::Allowance { owner, spender } => { - to_json_binary(&query_allowance(deps, owner, spender)?) - } - QueryMsg::DepositRatio { funds } => to_json_binary(&query_deposit_ratio(deps, funds)?), - QueryMsg::PendingBonds { address } => to_json_binary(&query_pending_bonds(deps, address)?), - QueryMsg::GetDebug {} => to_json_binary(&query_debug_string(deps)?), - QueryMsg::GetTvlInfo {} => to_json_binary(&query_tvl_info(deps)?), - QueryMsg::PendingUnbonds { address } => { - to_json_binary(&query_pending_unbonds(deps, address)?) - } - QueryMsg::GetCap {} => to_json_binary(&query_cap(deps)?), - QueryMsg::PendingBondsById { bond_id } => { - to_json_binary(&query_pending_bonds_by_id(deps, bond_id)?) - } - QueryMsg::PendingUnbondsById { bond_id } => { - to_json_binary(&query_pending_unbonds_by_id(deps, bond_id)?) - } - } -} - -fn query_cap(deps: Deps) -> StdResult { - Ok(GetCapResponse { - cap: CAP.load(deps.storage)?, - }) -} - -pub fn query_vault_token_info(deps: Deps) -> StdResult { - let token_info = TOKEN_INFO.load(deps.storage)?; - let additional_info = ADDITIONAL_TOKEN_INFO.load(deps.storage)?; - let res = VaultTokenInfoResponse { - name: token_info.name, - thesis: additional_info.thesis, - symbol: token_info.symbol, - decimals: token_info.decimals, - total_supply: token_info.total_supply, - creation_time: additional_info.creation_time, - }; - Ok(res) -} - -pub fn query_debug_string(deps: Deps) -> StdResult { - let debug_string = DEBUG_TOOL.load(deps.storage)?; - - Ok(GetDebugResponse { - debug: debug_string, - }) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn migrate(deps: DepsMut, _env: Env, _msg: MigrateMsg) -> Result { - let msgs: Result, StdError> = cw20_base::state::BALANCES - .range(deps.storage, None, None, Order::Ascending) - .map(|val| { - let (addr, _) = val?; - update_user_reward_index(deps.storage, &addr) - }) - .collect(); - - let wrapped_msges = msgs?.into_iter().map(CosmosMsg::Wasm); - - Ok(Response::new() - .add_attribute( - "updated-rewards-indexes-msges", - wrapped_msges.len().to_string(), - ) - .add_messages(wrapped_msges)) -} - -#[cfg(test)] -mod test { - use cosmwasm_std::{ - testing::{mock_dependencies, mock_env, mock_info}, - Addr, ContractResult, Decimal, QuerierResult, - }; - - use crate::msg::PrimitiveConfig; - - use super::*; - - #[test] - fn instantiate_works() { - let mut deps = mock_dependencies(); - let env = mock_env(); - - let info = MessageInfo { - sender: Addr::unchecked("owner"), - funds: vec![], - }; - - let msg = InstantiateMsg { - name: "vault".to_string(), - thesis: "to generate yield, I guess".to_string(), - symbol: "VLT".to_string(), - decimals: 6, - min_withdrawal: Uint128::new(100), - primitives: vec![ - PrimitiveConfig { - weight: Decimal::from_ratio(Uint128::one(), Uint128::new(3)), - address: "prim1".to_string(), - init: crate::msg::PrimitiveInitMsg::LP(lp_strategy::msg::InstantiateMsg { - lock_period: 300, - pool_id: 1, - pool_denom: "gamm/pool/1".to_string(), - local_denom: "ibc/SOME_DENOM".to_string(), - base_denom: "uosmo".to_string(), - quote_denom: "uqsr".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }), - }, - PrimitiveConfig { - weight: Decimal::from_ratio(Uint128::one(), Uint128::new(3)), - address: "prim2".to_string(), - init: crate::msg::PrimitiveInitMsg::LP(lp_strategy::msg::InstantiateMsg { - lock_period: 300, - pool_id: 1, - pool_denom: "gamm/pool/2".to_string(), - local_denom: "ibc/SOME_DENOM".to_string(), - base_denom: "uqsr".to_string(), - quote_denom: "uosmo".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }), - }, - PrimitiveConfig { - weight: Decimal::from_ratio(Uint128::one(), Uint128::new(3)), - address: "prim3".to_string(), - init: crate::msg::PrimitiveInitMsg::LP(lp_strategy::msg::InstantiateMsg { - lock_period: 300, - pool_id: 1, - pool_denom: "gamm/pool/3".to_string(), - local_denom: "ibc/SOME_DENOM".to_string(), - base_denom: "uatom".to_string(), - quote_denom: "uqsr".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }), - }, - ], - vault_rewards_code_id: 123, - reward_token: cw_asset::AssetInfoBase::Native("uqsr".to_string()), - reward_distribution_schedules: vec![vault_rewards::state::DistributionSchedule { - start: 0, - end: 500, - amount: Uint128::from(1000u128), - }], - total_cap: Uint128::new(10_000_000_000_000), - deposit_denom: "ibc/SOME_DENOM".to_string(), - }; - - // prepare 3 mock configs for prim1, prim2 and prim3 - deps.querier.update_wasm(|wq| match wq { - cosmwasm_std::WasmQuery::Smart { - contract_addr, - msg: _, - } => { - if contract_addr == "prim1" { - QuerierResult::Ok(ContractResult::Ok( - to_json_binary(&lp_strategy::msg::ConfigResponse { - config: lp_strategy::state::Config { - lock_period: 300, - pool_id: 1, - pool_denom: "gamm/pool/1".to_string(), - local_denom: "ibc/SOME_DENOM".to_string(), - base_denom: "uosmo".to_string(), - quote_denom: "uqsr".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }, - }) - .unwrap(), - )) - } else if contract_addr == "prim2" { - QuerierResult::Ok(ContractResult::Ok( - to_json_binary(&lp_strategy::msg::ConfigResponse { - config: lp_strategy::state::Config { - lock_period: 300, - pool_id: 1, - pool_denom: "gamm/pool/2".to_string(), - local_denom: "ibc/SOME_DENOM".to_string(), - base_denom: "uqsr".to_string(), - quote_denom: "uosmo".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }, - }) - .unwrap(), - )) - } else if contract_addr == "prim3" { - QuerierResult::Ok(ContractResult::Ok( - to_json_binary(&lp_strategy::msg::ConfigResponse { - config: lp_strategy::state::Config { - lock_period: 300, - pool_id: 1, - pool_denom: "gamm/pool/3".to_string(), - local_denom: "ibc/SOME_DENOM".to_string(), - base_denom: "uatom".to_string(), - quote_denom: "uqsr".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }, - }) - .unwrap(), - )) - } else { - QuerierResult::Err(cosmwasm_std::SystemError::NoSuchContract { - addr: contract_addr.to_string(), - }) - } - } - cosmwasm_std::WasmQuery::Raw { - contract_addr: _, - key: _, - } => QuerierResult::Err(cosmwasm_std::SystemError::Unknown {}), - cosmwasm_std::WasmQuery::ContractInfo { contract_addr: _ } => { - QuerierResult::Err(cosmwasm_std::SystemError::Unknown {}) - } - _ => panic!("Unimplemented query path"), - }); - - instantiate(deps.as_mut(), env, info, msg).unwrap(); - } - - #[test] - fn test_try_clear_cache() { - let mut deps = mock_dependencies(); - let env = mock_env(); - let info = mock_info("lulu", &[]); - - let primitive_configs = vec![ - PrimitiveConfig { - weight: Decimal::from_ratio(Uint128::one(), Uint128::new(3)), - address: "prim1".to_string(), - init: crate::msg::PrimitiveInitMsg::LP(lp_strategy::msg::InstantiateMsg { - lock_period: 300, - pool_id: 1, - pool_denom: "gamm/pool/1".to_string(), - local_denom: "ibc/SOME_DENOM".to_string(), - base_denom: "uosmo".to_string(), - quote_denom: "uqsr".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }), - }, - PrimitiveConfig { - weight: Decimal::from_ratio(Uint128::one(), Uint128::new(3)), - address: "prim2".to_string(), - init: crate::msg::PrimitiveInitMsg::LP(lp_strategy::msg::InstantiateMsg { - lock_period: 300, - pool_id: 1, - pool_denom: "gamm/pool/2".to_string(), - local_denom: "ibc/SOME_DENOM".to_string(), - base_denom: "uqsr".to_string(), - quote_denom: "uosmo".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }), - }, - ]; - - let investment_info = InvestmentInfo { - owner: Addr::unchecked("lulu"), - min_withdrawal: Uint128::from(100u128), - primitives: primitive_configs, - deposit_denom: "ibc/SOME_DENOM".to_string(), - }; - - INVESTMENT - .save(deps.as_mut().storage, &investment_info) - .unwrap(); - - let msg = ExecuteMsg::ClearCache {}; - let res = execute(deps.as_mut(), env, info, msg).unwrap(); - - assert_eq!(res.messages.len(), 2); - assert_eq!( - res.messages[0], - SubMsg::new(WasmMsg::Execute { - contract_addr: "prim1".to_string(), - funds: vec![], - msg: to_json_binary(&lp_strategy::msg::ExecuteMsg::TryIcq {}).unwrap(), - }) - ); - assert_eq!( - res.messages[1], - SubMsg::new(WasmMsg::Execute { - contract_addr: "prim2".to_string(), - funds: vec![], - msg: to_json_binary(&lp_strategy::msg::ExecuteMsg::TryIcq {}).unwrap(), - }) - ); - } -} diff --git a/smart-contracts/osmosis/contracts/basic-vault/src/error.rs b/smart-contracts/osmosis/contracts/basic-vault/src/error.rs deleted file mode 100644 index dd06951db..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/src/error.rs +++ /dev/null @@ -1,146 +0,0 @@ -use cosmwasm_std::{ - CheckedFromRatioError, CheckedMultiplyRatioError, OverflowError, StdError, Uint128, -}; -use quasar_types::error::Error as QError; -use thiserror::Error; - -#[derive(Error, Debug, PartialEq)] -pub enum ContractError { - #[error("{0}")] - Std(#[from] StdError), - - #[error("Unauthorized")] - Unauthorized {}, - - #[error("Vault is over cap")] - OverCap {}, - - #[error("Validator '{validator}' not in current validator set")] - NotInValidatorSet { validator: String }, - - #[error("Different denominations in bonds: '{denom1}' vs. '{denom2}'")] - DifferentBondDenom { denom1: String, denom2: String }, - - #[error("Stored bonded {stored}, but query bonded {queried}")] - BondedMismatch { stored: Uint128, queried: Uint128 }, - - #[error("Incorrect bonding ratio")] - IncorrectBondingRatio {}, - - #[error("No {denom} tokens sent")] - EmptyBalance { denom: String }, - - #[error("Must unbond at least {min_bonded}")] - UnbondTooSmall { min_bonded: Uint128 }, - - #[error("Cannot withdraw without vault tokens")] - NoFunds {}, - - #[error("Insufficient balance in contract to process claim")] - BalanceTooSmall {}, - - #[error("No claims that can be released currently")] - NothingToClaim {}, - - #[error("Cannot set to own account")] - CannotSetOwnAccount {}, - - #[error("Invalid expiration")] - InvalidExpiration {}, - - #[error("Invalid zero amount")] - InvalidZeroAmount {}, - - #[error("Duplicate bond response for bond_id {bond_id}")] - DuplicateBondResponse { bond_id: String }, - - #[error("Allowance is expired")] - Expired {}, - - #[error("No allowance for this account")] - NoAllowance {}, - - #[error("Minting cannot exceed the cap")] - CannotExceedCap {}, - - #[error("Duplicate initial balance addresses")] - DuplicateInitialBalanceAddresses {}, - - #[error("Incorrect callback id, expected: {expected}, got: {:?}", ids)] - IncorrectCallbackId { expected: String, ids: Vec }, - - #[error("Multiply ratio error: {0}")] - MultiplyRatioError(String), - - #[error("Missing bond response")] - MissingBondResponse {}, - - #[error("Token weight vector is empty")] - TokenWeightsIsEMpty {}, - - #[error("Coins vector is empty")] - CoinsVectorIsEmpty {}, - - #[error("Denom not found in coins vector")] - DenomNotFoundInCoinsVector {}, - - #[error("User does not have pending unbonds")] - NoPendingUnbonds {}, - - #[error("Bond response is empty")] - BondResponseIsEmpty {}, - - #[error("Unbond is empty")] - UnbondIsEmpty {}, - - #[error("Unbond stub is empty")] - UnbondStubIsEmpty {}, - - #[error("Coins weight vector is empty")] - CoinsWeightVectorIsEmpty {}, - - #[error("{0}")] - QError(#[from] QError), - - #[error("{0}")] - PaymentError(#[from] cw_utils::PaymentError), - - #[error("{0}")] - CheckedMultiplyRatioError(#[from] CheckedMultiplyRatioError), - - #[error("{0}")] - OverflowError(#[from] OverflowError), -} - -impl From for ContractError { - fn from(err: CheckedFromRatioError) -> Self { - ContractError::MultiplyRatioError(err.to_string()) - } -} - -impl From for ContractError { - fn from(err: cw20_base::ContractError) -> Self { - match err { - cw20_base::ContractError::Std(error) => ContractError::Std(error), - cw20_base::ContractError::Unauthorized {} => ContractError::Unauthorized {}, - cw20_base::ContractError::CannotSetOwnAccount {} => { - ContractError::CannotSetOwnAccount {} - } - cw20_base::ContractError::InvalidExpiration {} => ContractError::InvalidExpiration {}, - #[allow(deprecated)] - cw20_base::ContractError::InvalidZeroAmount {} => ContractError::InvalidZeroAmount {}, - cw20_base::ContractError::Expired {} => ContractError::Expired {}, - cw20_base::ContractError::NoAllowance {} => ContractError::NoAllowance {}, - cw20_base::ContractError::CannotExceedCap {} => ContractError::CannotExceedCap {}, - // This should never happen, as this contract doesn't use logo - cw20_base::ContractError::LogoTooBig {} - | cw20_base::ContractError::InvalidPngHeader {} - | cw20_base::ContractError::InvalidXmlPreamble {} => { - ContractError::Std(StdError::generic_err(err.to_string())) - } - cw20_base::ContractError::DuplicateInitialBalanceAddresses {} => { - ContractError::DuplicateInitialBalanceAddresses {} - } - } - } -} diff --git a/smart-contracts/osmosis/contracts/basic-vault/src/execute.rs b/smart-contracts/osmosis/contracts/basic-vault/src/execute.rs deleted file mode 100644 index c7641da45..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/src/execute.rs +++ /dev/null @@ -1,1098 +0,0 @@ -use cosmwasm_std::{ - coin, to_json_binary, Attribute, Coin, Decimal, Deps, DepsMut, Env, MessageInfo, Response, - StdError, Uint128, WasmMsg, -}; - -use cw20::BalanceResponse; -use cw20_base::contract::execute_burn; -use cw_utils::{must_pay, nonpayable, PaymentError}; - -use lp_strategy::msg::{IcaBalanceResponse, PrimitiveSharesResponse}; -use quasar_types::types::{CoinRatio, CoinWeight}; - -use crate::error::ContractError; -use crate::helpers::{can_unbond_from_primitive, is_contract_admin, update_user_reward_index}; -use crate::msg::PrimitiveConfig; -use crate::state::{ - BondingStub, Cap, InvestmentInfo, Unbond, UnbondingStub, BONDING_SEQ, BONDING_SEQ_TO_ADDR, - BOND_STATE, CAP, INVESTMENT, PENDING_BOND_IDS, PENDING_UNBOND_IDS, UNBOND_STATE, -}; -use crate::types::FromUint128; - -// returns amount if the coin is found and amount is non-zero -// errors otherwise -pub fn must_pay_multi(funds: &[Coin], denom: &str) -> Result { - match funds.iter().find(|c| c.denom == denom) { - Some(coin) => { - if coin.amount.is_zero() { - Err(PaymentError::MissingDenom(denom.to_string())) - } else { - Ok(coin.amount) - } - } - None => Err(PaymentError::MissingDenom(denom.to_string())), - } -} - -pub fn get_deposit_amount_weights( - deps: &Deps, - primitives: &[PrimitiveConfig], -) -> Result { - let weights = primitives - .iter() - .map(|pc| -> Result { - let balance: IcaBalanceResponse = deps.querier.query_wasm_smart( - pc.address.clone(), - &lp_strategy::msg::QueryMsg::IcaBalance {}, - )?; - let supply: PrimitiveSharesResponse = deps.querier.query_wasm_smart( - pc.address.clone(), - &lp_strategy::msg::QueryMsg::PrimitiveShares {}, - )?; - - // if only one of the two is zero, we should error - if (supply.total.is_zero() && !balance.amount.amount.is_zero()) || (!supply.total.is_zero() && balance.amount.amount.is_zero()) { - return Err(ContractError::Std(StdError::GenericErr { - msg: "Unexpected primitive state, either both supply and balance should be zero, or neither.".to_string(), - })); - } - - let ratio = match supply.total.is_zero() { - true => Decimal::one(), - false => Decimal::from_ratio(balance.amount.amount, supply.total), - }; - - Ok(CoinWeight { - weight: ratio.checked_mul(pc.weight)?, - denom: balance.amount.denom, - }) - }) - .collect::, ContractError>>()?; - - let mut ratio = CoinRatio { ratio: weights }; - ratio.normalize()?; - - Ok(ratio) -} - -pub fn get_token_amount_weights( - deposit_amount_weights: &[CoinWeight], -) -> Result, ContractError> { - deposit_amount_weights.iter().try_fold( - vec![], - |mut acc: Vec, - coin_weight: &CoinWeight| - -> Result, ContractError> { - // look through acc for existing denom and add weight, or else push it to the back of the vec - // todo: verify this works for multiple tokens, this might not overwrite when two primitives have the same token - let existing_weight = acc.iter_mut().find(|cw| cw.denom == coin_weight.denom); - match existing_weight { - Some(weight) => weight.weight = weight.weight.checked_add(coin_weight.weight)?, - None => acc.push(coin_weight.clone()), - }; - Ok(acc) - }, - ) -} - -pub fn get_max_bond( - funds: &[Coin], - token_weights: &Vec, -) -> Result { - let mut max_bond = Decimal::MAX; - for coin_weight in token_weights { - let amount = must_pay_multi(funds, &coin_weight.denom)?; - let bond_for_token = Decimal::from_uint128(amount).checked_div(coin_weight.weight)?; - - if bond_for_token < max_bond { - max_bond = bond_for_token; - } - } - Ok(max_bond) -} - -pub fn get_deposit_and_remainder_for_ratio( - funds: &[Coin], - max_bond: Decimal, - ratio: &CoinRatio, -) -> Result<(Vec, Vec), ContractError> { - // verify that >0 of each token in ratio is passed in, return (funds, remainder)) - // where funds is the max amount we can use in compliance with the ratio - // and remainder is the change to return to user - let mut remainder = funds.to_owned(); - - let coins: Result, ContractError> = ratio - .ratio - .iter() - .filter(|r| r.weight > Decimal::zero()) - .map(|r| { - let amount = Decimal::from_uint128(must_pay_multi(funds, &r.denom)?); - let expected_amount = max_bond.checked_mul(r.weight)?; - - if expected_amount > amount { - return Err(ContractError::IncorrectBondingRatio {}); - } - - remainder = remainder - .iter() - .map(|c| -> Result { - if c.denom == r.denom { - Ok(Coin { - amount: c.amount.checked_sub(expected_amount.to_uint_floor())?, - denom: c.denom.clone(), - }) - } else { - Ok(c.clone()) - } - }) - .collect::, ContractError>>()?; - - Ok(Coin { - denom: r.denom.clone(), - amount: expected_amount.to_uint_floor(), - }) - }) - .collect(); - - Ok((coins?, remainder)) -} - -pub fn divide_by_ratio( - funds: Coin, - invest: InvestmentInfo, -) -> Result, ContractError> { - let coins: Result, cosmwasm_std::OverflowError> = invest - .primitives - .iter() - .map( - |config| -> Result<(Coin, String), cosmwasm_std::OverflowError> { - config - .weight - .checked_mul(Decimal::from_uint128(funds.amount)) - .map(|dec| { - ( - coin(dec.to_uint_floor().u128(), funds.denom.as_str()), - config.address.clone(), - ) - }) - }, - ) - .collect(); - Ok(coins?) -} - -pub fn may_pay_with_ratio( - deps: &Deps, - funds: &[Coin], - mut invest: InvestmentInfo, -) -> Result<(Vec, Vec), ContractError> { - // normalize primitives - invest.normalize_primitive_weights(); - - // load cached balance of primitive contracts - let deposit_amount_ratio = get_deposit_amount_weights(deps, &invest.primitives)?; - - if deposit_amount_ratio - .ratio - .first() - .ok_or(ContractError::CoinsWeightVectorIsEmpty {})? - .weight - == Decimal::zero() - { - return Err(ContractError::Std(StdError::GenericErr { - msg: "Deposit amount weight for primitive is zero".to_string(), - })); - } - - let token_weights: Vec = get_token_amount_weights(&deposit_amount_ratio.ratio)?; - - if token_weights - .first() - .ok_or(ContractError::TokenWeightsIsEMpty {})? - .weight - == Decimal::zero() - { - return Err(ContractError::Std(StdError::GenericErr { - msg: format!( - "token weight is zero for {}", - token_weights.first().unwrap().denom - ), - })); - } - - let max_bond = get_max_bond(funds, &token_weights)?; - - if max_bond == Decimal::zero() || max_bond == Decimal::MAX { - return Err(ContractError::Std(StdError::GenericErr { - msg: format!("Unable to correctly determine max_bond, value: {max_bond}"), - })); - } - - let (coins, remainder) = - get_deposit_and_remainder_for_ratio(funds, max_bond, &deposit_amount_ratio)?; - - if coins - .first() - .ok_or(ContractError::CoinsVectorIsEmpty {})? - .amount - == Uint128::zero() - { - return Err(ContractError::Std(StdError::GenericErr { - msg: "we failed here".to_string(), - })); - } - - Ok((coins, remainder)) -} - -pub fn bond( - deps: DepsMut, - _env: Env, - info: MessageInfo, - recipient: Option, -) -> Result { - let invest = INVESTMENT.load(deps.storage)?; - - // load vault info & sequence number - let bond_seq = BONDING_SEQ.load(deps.storage)?; - - // get the deposited funds - let amount = must_pay(&info, invest.deposit_denom.as_str())?; - - // find recipient - let recipient_addr = match recipient { - Some(r) => deps.api.addr_validate(&r)?, - None => info.sender, - }; - - let mut deposit_stubs = vec![]; - let divided = divide_by_ratio(coin(amount.u128(), invest.deposit_denom.as_str()), invest)?; - - CAP.update( - deps.storage, - |cap| -> Result { cap.update_current(amount) }, - )?; - - let bond_msgs: Result, ContractError> = divided - .into_iter() - .map(|(coin, prim_addr)| { - let deposit_stub = BondingStub { - address: prim_addr.clone(), - bond_response: None, - primitive_value: None, - amount: coin.amount, - }; - deposit_stubs.push(deposit_stub); - - Ok(WasmMsg::Execute { - contract_addr: prim_addr, - msg: to_json_binary(&lp_strategy::msg::ExecuteMsg::Bond { - id: bond_seq.to_string(), - })?, - funds: vec![coin], - }) - }) - .collect(); - - // let (primitive_funding_amounts, remainder) = - // may_pay_with_ratio(&deps.as_ref(), &info.funds, invest.clone())?; - - // let bond_msgs: Result, ContractError> = invest - // .primitives - // .iter() - // .zip(primitive_funding_amounts) - // .map(|(pc, funds)| { - // let deposit_stub = BondingStub { - // address: pc.address.clone(), - // bond_response: None, - // primitive_value: None, - // amount: funds.amount, - // }; - // deposit_stubs.push(deposit_stub); - - // Ok(WasmMsg::Execute { - // contract_addr: pc.address.clone(), - // msg: to_json_binary(&lp_strategy::msg::ExecuteMsg::Bond { - // id: bond_seq.to_string(), - // })?, - // funds: vec![funds], - // }) - // }) - // .collect(); - - // save bonding state for use during the callback - PENDING_BOND_IDS.update(deps.storage, recipient_addr.clone(), |ids| match ids { - Some(mut bond_ids) => { - bond_ids.push(bond_seq.to_string()); - Ok::, ContractError>(bond_ids) - } - None => Ok(vec![bond_seq.to_string()]), - })?; - BOND_STATE.save(deps.storage, bond_seq.to_string(), &deposit_stubs)?; - BONDING_SEQ_TO_ADDR.save( - deps.storage, - bond_seq.to_string(), - &recipient_addr.to_string(), - )?; - BONDING_SEQ.save(deps.storage, &bond_seq.checked_add(Uint128::new(1))?)?; - - // let remainder_msg = BankMsg::Send { - // to_address: recipient_addr.to_string(), - // amount: remainder - // .iter() - // .filter(|c| !c.amount.is_zero()) - // .map(|r| Coin { - // denom: r.denom.clone(), - // amount: r.amount, - // }) - // .collect(), - // }; - - Ok(Response::new() - .add_attribute("bond_id", bond_seq.to_string()) - .add_messages(bond_msgs?)) - // .add_message(remainder_msg)) -} - -pub fn unbond( - mut deps: DepsMut, - env: Env, - info: MessageInfo, - amount: Option, -) -> Result { - nonpayable(&info)?; - - let start_unbond_response = - do_start_unbond(deps.branch(), &env, &info, amount)?.unwrap_or(Response::new()); - - let start_unbond_msgs = start_unbond_response - .messages - .iter() - .map(|sm| sm.msg.clone()); - - Ok(Response::new() - .add_messages(start_unbond_msgs) - .add_attributes(start_unbond_response.attributes)) -} - -pub fn do_start_unbond( - mut deps: DepsMut, - env: &Env, - info: &MessageInfo, - amount: Option, -) -> Result, ContractError> { - let unbond_amount = amount.unwrap_or(Uint128::zero()); - if unbond_amount.is_zero() { - // skip start unbond - return Ok(None); - } - - let invest = INVESTMENT.load(deps.storage)?; - let bond_seq = BONDING_SEQ.load(deps.storage)?; - - // check that user has vault tokens and the amount is > min_withdrawal - if unbond_amount < invest.min_withdrawal { - return Err(ContractError::UnbondTooSmall { - min_bonded: invest.min_withdrawal, - }); - } - - // burn if balance is more than or equal to amount (handled in execute_burn) - let update_user_rewards_idx_msg = - update_user_reward_index(deps.as_ref().storage, &info.sender)?; - - let mut unbonding_stubs = vec![]; - let supply = cw20_base::contract::query_token_info(deps.as_ref())?.total_supply; - - let start_unbond_msgs: Vec = invest - .primitives - .iter() - .map(|pc| -> Result { - // get this vaults primitive share balance - let our_balance: BalanceResponse = deps.querier.query_wasm_smart( - pc.address.clone(), - &lp_strategy::msg::QueryMsg::Balance { - address: env.contract.address.to_string(), - }, - )?; - - // lets get the amount of tokens to unbond for this primitive: p_unbond_amount = (v_unbond_amount / v_total_supply) * our_p_balance - let primitive_share_amount = Decimal::from_ratio(unbond_amount, supply) - .checked_mul(Decimal::from_uint128(our_balance.balance))? - .to_uint_floor(); - - unbonding_stubs.push(UnbondingStub { - address: pc.address.clone(), - unlock_time: None, - unbond_response: None, - unbond_funds: vec![], - }); - - Ok(WasmMsg::Execute { - contract_addr: pc.address.clone(), - msg: to_json_binary(&lp_strategy::msg::ExecuteMsg::StartUnbond { - id: bond_seq.to_string(), - share_amount: primitive_share_amount, - })?, - funds: vec![], - }) - }) - .collect::, ContractError>>()?; - - // We need to save the unbonding state for use during the callback - PENDING_UNBOND_IDS.update(deps.storage, info.sender.clone(), |ids| match ids { - Some(mut bond_ids) => { - bond_ids.push(bond_seq.to_string()); - Ok::, ContractError>(bond_ids) - } - None => Ok(vec![bond_seq.to_string()]), - })?; - UNBOND_STATE.save( - deps.storage, - bond_seq.to_string(), - &Unbond { - stub: unbonding_stubs, - shares: unbond_amount, - }, - )?; - BONDING_SEQ_TO_ADDR.save(deps.storage, bond_seq.to_string(), &info.sender.to_string())?; - BONDING_SEQ.save(deps.storage, &bond_seq.checked_add(Uint128::from(1u128))?)?; - - execute_burn(deps.branch(), env.clone(), info.clone(), unbond_amount)?; - Ok(Some( - Response::new() - .add_messages(start_unbond_msgs) - .add_message(update_user_rewards_idx_msg) - .add_attributes(vec![ - Attribute { - key: "action".to_string(), - value: "start_unbond".to_string(), - }, - Attribute { - key: "from".to_string(), - value: info.sender.to_string(), - }, - Attribute { - key: "burnt".to_string(), - value: unbond_amount.to_string(), - }, - Attribute { - key: "bond_id".to_string(), - value: bond_seq.to_string(), - }, - ]), - )) -} - -// find all unbondable pending unbonds where unlock_time < env.block.time -// then trigger unbonds -pub fn do_unbond( - mut deps: DepsMut, - env: &Env, - info: &MessageInfo, -) -> Result, ContractError> { - let pending_unbond_ids_opt = PENDING_UNBOND_IDS.may_load(deps.storage, info.sender.clone())?; - - match pending_unbond_ids_opt { - Some(pending_unbond_ids) => { - let mut unbond_msgs: Vec = vec![]; - for unbond_id in pending_unbond_ids.iter() { - let unbond_stubs_opt = UNBOND_STATE.may_load(deps.storage, unbond_id.clone())?; - if let Some(unbond_stubs) = unbond_stubs_opt { - let mut current_unbond_msgs = find_and_return_unbondable_msgs( - deps.branch(), - env, - info, - unbond_id, - unbond_stubs.stub, - )?; - unbond_msgs.append(current_unbond_msgs.as_mut()); - } - } - - Ok(Some( - Response::new() - .add_messages(unbond_msgs.clone()) - .add_attributes(vec![ - Attribute { - key: "action".to_string(), - value: "unbond".to_string(), - }, - Attribute { - key: "from".to_string(), - value: info.sender.to_string(), - }, - Attribute { - key: "num_unbondable_ids".to_string(), - value: unbond_msgs.len().to_string(), - }, - ]), - )) - } - None => Ok(None), - } -} - -pub fn find_and_return_unbondable_msgs( - deps: DepsMut, - env: &Env, - _info: &MessageInfo, - unbond_id: &str, - unbond_stubs: Vec, -) -> Result, ContractError> { - // go through unbond_stubs and find ones where unlock_time < env.block.time and execute - let mut unbond_msgs = vec![]; - - for stub in unbond_stubs.iter() { - let can_unbond = can_unbond_from_primitive(deps.as_ref(), env, unbond_id, stub)?; - - if can_unbond { - unbond_msgs.push(WasmMsg::Execute { - contract_addr: stub.address.clone(), - msg: to_json_binary(&lp_strategy::msg::ExecuteMsg::Unbond { - id: unbond_id.to_string(), - })?, - funds: vec![], - }) - } - } - - Ok(unbond_msgs) -} - -// claim is equivalent to calling unbond with amount: 0 -pub fn claim(deps: DepsMut, env: Env, info: MessageInfo) -> Result { - nonpayable(&info)?; - - Ok(do_unbond(deps, &env, &info)?.unwrap_or(Response::new())) -} - -pub fn update_cap( - deps: DepsMut, - env: Env, - info: MessageInfo, - new_total: Option, - new_cap_admin: Option, -) -> Result { - nonpayable(&info)?; - is_contract_admin(&deps.querier, &env, &info.sender)?; - let mut attributes = vec![]; - - if let Some(new_total) = new_total { - CAP.update(deps.storage, |c| -> Result { - Ok(c.update_total_cap(new_total)) - })?; - attributes.push(Attribute { - key: "new_total".to_string(), - value: new_total.to_string(), - }) - } - - if let Some(new_cap_admin) = new_cap_admin { - let new_cap_admin_validated = deps.api.addr_validate(&new_cap_admin)?; - CAP.update(deps.storage, |c| -> Result { - Ok(c.update_cap_admin(new_cap_admin_validated)) - })?; - attributes.push(Attribute { - key: "new_cap_admin".to_string(), - value: new_cap_admin, - }) - } - - Ok(Response::new() - .add_attribute("action", "update_cap") - .add_attributes(attributes) - .add_attribute("success", "true")) -} - -#[cfg(test)] -mod tests { - use crate::callback::on_bond; - use crate::msg::PrimitiveInitMsg; - use crate::state::VAULT_REWARDS; - use crate::tests::{mock_deps_with_primitives, TEST_ADMIN}; - - use super::*; - use cosmwasm_std::testing::{mock_env, mock_info}; - use cosmwasm_std::{attr, Addr, Coin, CosmosMsg, Uint128}; - - use cw20_base::state::{MinterData, TokenInfo, TOKEN_INFO}; - use lp_strategy::msg::InstantiateMsg; - - // this test tests 2 on_bond callbacks and a start unbond. The amounts returned slightly 'weird'. The main idea is that after the on_bond callbacks, - // our user owns 10% of the vault. When we then start to unbond, we expect the user to get 10% of the value in each primitive - #[test] - fn test_correct_start_unbond_amount() { - let primitive_states = vec![ - ( - "contract1".to_string(), - "ibc/ED07".to_string(), - // we init state with 1 primitve share being 10 tokens - Uint128::from(500u128), - Uint128::from(5000u128), - ), - ( - "contract2".to_string(), - "ibc/ED07".to_string(), - Uint128::from(500u128), - Uint128::from(5000u128), - ), - ]; - // mock the queries so the primitives exist - let mut deps = mock_deps_with_primitives(primitive_states); - let env = mock_env(); - let info = mock_info("user", &[Coin::new(10000, "token")]); - - let instantiate_msg_1 = InstantiateMsg { - lock_period: 3600, - pool_id: 2, - pool_denom: "gamm/pool/2".to_string(), - local_denom: "ibc/ED07".to_string(), - base_denom: "uosmo".to_string(), - quote_denom: "usdc".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-1".to_string(), - expected_connection: "connection-0".to_string(), - }; - - let instantiate_msg_2 = InstantiateMsg { - lock_period: 7200, - pool_id: 5, - pool_denom: "gamm/pool/5".to_string(), - local_denom: "ibc/ED07".to_string(), - base_denom: "uosmo".to_string(), - quote_denom: "uatom".to_string(), - transfer_channel: "channel-2".to_string(), - return_source_channel: "channel-3".to_string(), - expected_connection: "connection-1".to_string(), - }; - - // set up the contract state - let min_withdrawal = Uint128::new(100); - let invest = InvestmentInfo { - primitives: vec![ - PrimitiveConfig { - address: "contract1".to_string(), - weight: Decimal::percent(50), - init: PrimitiveInitMsg::LP(instantiate_msg_1), - }, - PrimitiveConfig { - address: "contract2".to_string(), - weight: Decimal::percent(50), - init: PrimitiveInitMsg::LP(instantiate_msg_2), - }, - ], - min_withdrawal, - owner: Addr::unchecked("bob"), - deposit_denom: "ibc/ED07".to_string(), - }; - let bond_seq = Uint128::new(1); - - INVESTMENT.save(deps.as_mut().storage, &invest).unwrap(); - BONDING_SEQ.save(deps.as_mut().storage, &bond_seq).unwrap(); - VAULT_REWARDS - .save(deps.as_mut().storage, &Addr::unchecked("rewards-contract")) - .unwrap(); - - // store token info using cw20-base format - let token_info = TokenInfo { - name: "token".to_string(), - symbol: "token".to_string(), - decimals: 6, - total_supply: Uint128::new(5000), - // set self as minter, so we can properly execute mint and burn - mint: Some(MinterData { - minter: env.contract.address.clone(), - cap: None, - }), - }; - TOKEN_INFO.save(deps.as_mut().storage, &token_info).unwrap(); - - BONDING_SEQ_TO_ADDR - .save(deps.as_mut().storage, "1".to_string(), &"user".to_string()) - .unwrap(); - - // mock an unfilfilled stub, do 2 callbacks to fullfill the stubs, and mint shares for the user such that the user owns 50% of the shares - BOND_STATE - .save( - deps.as_mut().storage, - "1".to_string(), - &vec![ - BondingStub { - address: "contract1".to_string(), - bond_response: None, - amount: Uint128::new(5000), - primitive_value: None, - }, - BondingStub { - address: "contract2".to_string(), - bond_response: None, - amount: Uint128::new(5000), - primitive_value: None, - }, - ], - ) - .unwrap(); - - // update the querier to return underlying shares of the vault, in total our vault has 5000 internal shares - // we expect the vault to unbond 10% of the shares it owns in each primitive, so if contract 1 returns - // 4000 shares and contract 2 returns 3000 shares, and we unbond 10% of the shares, we should unbond 400 and 300 shares respectively - deps.querier.update_state(vec![ - // the primitives were mocked with 500 shares and 5000 tokens, we should have deposited 5000 more tokens so get 500 more prim shares - ("contract1", Uint128::new(1000), Uint128::new(10000)), - ("contract2", Uint128::new(1000), Uint128::new(10000)), - ]); - - // we do 2 callbacks, reflecting the updated state of the primitives - on_bond( - deps.as_mut(), - env.clone(), - MessageInfo { - sender: Addr::unchecked("contract1"), - funds: vec![], - }, - Uint128::new(500), - "1".to_string(), - ) - .unwrap(); - on_bond( - deps.as_mut(), - env.clone(), - MessageInfo { - sender: Addr::unchecked("contract2"), - funds: vec![], - }, - Uint128::new(500), - "1".to_string(), - ) - .unwrap(); - - // start trying withdrawals - // our succesful withdrawal should show that it is possible for the vault contract to unbond a different amount than 350 and 150 shares - - // case 1: amount is zero, skip start unbond - let amount = None; - let res = do_start_unbond(deps.as_mut(), &env, &info, amount).unwrap(); - assert_eq!(res, None); - - // case 2: amount is less than min_withdrawal, error - let amount = Some(Uint128::new(50)); - let res = do_start_unbond(deps.as_mut(), &env, &info, amount); - assert!(res.is_err()); - assert_eq!( - res.unwrap_err(), - ContractError::UnbondTooSmall { - min_bonded: min_withdrawal - } - ); - - // case 3: amount is valid, execute start unbond on all primitive contracts - let amount = cw20_base::contract::query_balance(deps.as_ref(), "user".to_string()) - .unwrap() - .balance; - - let res = do_start_unbond(deps.as_mut(), &env, &info, Some(amount)) - .unwrap() - .unwrap(); - assert_eq!(res.attributes.len(), 4); - assert_eq!(res.messages.len(), 3); - - // check the messages sent to each primitive contract - let msg1 = &res.messages[0]; - let msg2 = &res.messages[1]; - - // start unbond is independent of the amounts in the callback, but is dependent on the vault's amount of shares in the primitive - assert_eq!( - msg1.msg, - CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr: "contract1".to_string(), - msg: to_json_binary(&lp_strategy::msg::ExecuteMsg::StartUnbond { - id: bond_seq.to_string(), - share_amount: Uint128::new(500), - }) - .unwrap(), - funds: vec![], - }) - ); - assert_eq!( - msg2.msg, - CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr: "contract2".to_string(), - msg: to_json_binary(&lp_strategy::msg::ExecuteMsg::StartUnbond { - id: bond_seq.to_string(), - share_amount: Uint128::new(500), - }) - .unwrap(), - funds: vec![], - }) - ); - } - - #[test] - fn test_correct_start_unbond_amount_uneven_weights() { - let primitive_states = vec![ - ( - "contract1".to_string(), - "ibc/ED07".to_string(), - // we init state with 1 primitve share being 10 tokens - Uint128::from(900u128), - Uint128::from(9000u128), - ), - ( - "contract2".to_string(), - "ibc/ED07".to_string(), - Uint128::from(100u128), - Uint128::from(1000u128), - ), - ]; - // mock the queries so the primitives exist - let mut deps = mock_deps_with_primitives(primitive_states); - let env = mock_env(); - let info = mock_info("user", &[Coin::new(10000, "token")]); - - let instantiate_msg_1 = InstantiateMsg { - lock_period: 3600, - pool_id: 2, - pool_denom: "gamm/pool/2".to_string(), - local_denom: "ibc/ED07".to_string(), - base_denom: "uosmo".to_string(), - quote_denom: "usdc".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-1".to_string(), - expected_connection: "connection-0".to_string(), - }; - - let instantiate_msg_2 = InstantiateMsg { - lock_period: 7200, - pool_id: 5, - pool_denom: "gamm/pool/5".to_string(), - local_denom: "ibc/ED07".to_string(), - base_denom: "uosmo".to_string(), - quote_denom: "uatom".to_string(), - transfer_channel: "channel-2".to_string(), - return_source_channel: "channel-3".to_string(), - expected_connection: "connection-1".to_string(), - }; - - // set up the contract state - let min_withdrawal = Uint128::new(100); - let invest = InvestmentInfo { - primitives: vec![ - PrimitiveConfig { - address: "contract1".to_string(), - weight: Decimal::percent(90), - init: PrimitiveInitMsg::LP(instantiate_msg_1), - }, - PrimitiveConfig { - address: "contract2".to_string(), - weight: Decimal::percent(10), - init: PrimitiveInitMsg::LP(instantiate_msg_2), - }, - ], - min_withdrawal, - owner: Addr::unchecked("bob"), - deposit_denom: "ibc/ED07".to_string(), - }; - let bond_seq = Uint128::new(1); - - INVESTMENT.save(deps.as_mut().storage, &invest).unwrap(); - BONDING_SEQ.save(deps.as_mut().storage, &bond_seq).unwrap(); - VAULT_REWARDS - .save(deps.as_mut().storage, &Addr::unchecked("rewards-contract")) - .unwrap(); - - // store token info using cw20-base format - let token_info = TokenInfo { - name: "token".to_string(), - symbol: "token".to_string(), - decimals: 6, - total_supply: Uint128::new(5000), - // set self as minter, so we can properly execute mint and burn - mint: Some(MinterData { - minter: env.contract.address.clone(), - cap: None, - }), - }; - TOKEN_INFO.save(deps.as_mut().storage, &token_info).unwrap(); - - BONDING_SEQ_TO_ADDR - .save(deps.as_mut().storage, "1".to_string(), &"user".to_string()) - .unwrap(); - - BOND_STATE - .save( - deps.as_mut().storage, - "1".to_string(), - &vec![ - BondingStub { - address: "contract1".to_string(), - bond_response: None, - amount: Uint128::new(9000), - primitive_value: None, - }, - BondingStub { - address: "contract2".to_string(), - bond_response: None, - amount: Uint128::new(1000), - primitive_value: None, - }, - ], - ) - .unwrap(); - // update the querier to return underlying shares of the vault, in total our vault has 5000 internal shares - // we expect the vault to unbond 10% of the shares it owns in each primitive, so if contract 1 returns - // 4000 shares and contract 2 returns 3000 shares, and we unbond 10% of the shares, we should unbond 400 and 300 shares respectively - deps.querier.update_state(vec![ - // the primitives were mocked with 500 shares and 5000 tokens, we should have deposited 5000 more tokens so get 500 more prim shares - ("contract1", Uint128::new(1800), Uint128::new(18000)), - ("contract2", Uint128::new(200), Uint128::new(2000)), - ]); - - // we do 2 callbacks, reflecting the user deposit and the primitive state update - on_bond( - deps.as_mut(), - env.clone(), - MessageInfo { - sender: Addr::unchecked("contract1"), - funds: vec![], - }, - Uint128::new(900), - "1".to_string(), - ) - .unwrap(); - on_bond( - deps.as_mut(), - env.clone(), - MessageInfo { - sender: Addr::unchecked("contract2"), - funds: vec![], - }, - Uint128::new(100), - "1".to_string(), - ) - .unwrap(); - - // start trying withdrawals - // our succesful withdrawal should show that it is possible for the vault contract to unbond a different amount than 350 and 150 shares - - // case 1: amount is zero, skip start unbond - let amount = None; - let res = do_start_unbond(deps.as_mut(), &env, &info, amount).unwrap(); - assert_eq!(res, None); - - // case 2: amount is less than min_withdrawal, error - let amount = Some(Uint128::new(50)); - let res = do_start_unbond(deps.as_mut(), &env, &info, amount); - assert!(res.is_err()); - assert_eq!( - res.unwrap_err(), - ContractError::UnbondTooSmall { - min_bonded: min_withdrawal - } - ); - - // case 3: amount is valid, execute start unbond on all primitive contracts - let amount = cw20_base::contract::query_balance(deps.as_ref(), "user".to_string()) - .unwrap() - .balance; - let _total_supply = cw20_base::contract::query_token_info(deps.as_ref()).unwrap(); - - let res = do_start_unbond(deps.as_mut(), &env, &info, Some(amount)) - .unwrap() - .unwrap(); - assert_eq!(res.attributes.len(), 4); - assert_eq!(res.messages.len(), 3); - - // check the messages sent to each primitive contract - let msg1 = &res.messages[0]; - let msg2 = &res.messages[1]; - - // start unbond is independent of the amounts in the callback, but is dependent on the vault's amount of shares in the primitive - // TODO make sure these numbers make sense - assert_eq!( - msg1.msg, - CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr: "contract1".to_string(), - msg: to_json_binary(&lp_strategy::msg::ExecuteMsg::StartUnbond { - id: bond_seq.to_string(), - share_amount: Uint128::new(900), - }) - .unwrap(), - funds: vec![], - }) - ); - assert_eq!( - msg2.msg, - CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr: "contract2".to_string(), - msg: to_json_binary(&lp_strategy::msg::ExecuteMsg::StartUnbond { - id: bond_seq.to_string(), - share_amount: Uint128::new(100), - }) - .unwrap(), - funds: vec![], - }) - ); - } - - #[test] - fn test_proper_update_cap() { - let mut deps = mock_deps_with_primitives(vec![( - "abc".to_string(), - "123".to_string(), - 100u128.into(), - 100u128.into(), - )]); - let env = mock_env(); - let info = mock_info(TEST_ADMIN, &[]); - CAP.save( - &mut deps.storage, - &Cap::new(Addr::unchecked(TEST_ADMIN.to_string()), Uint128::new(100)), - ) - .unwrap(); - - let cap = Uint128::new(1000); - let res = update_cap(deps.as_mut(), env.clone(), info.clone(), Some(cap), None).unwrap(); - assert_eq!(res.attributes.len(), 3); - assert_eq!(res.attributes[0], attr("action", "update_cap")); - assert_eq!(res.attributes[1], attr("new_total", cap.to_string())); - assert_eq!(res.messages.len(), 0); - - // update again - let cap = Uint128::new(5000); - let res = update_cap(deps.as_mut(), env.clone(), info.clone(), Some(cap), None).unwrap(); - assert_eq!(res.attributes.len(), 3); - assert_eq!(res.attributes[0], attr("action", "update_cap")); - assert_eq!(res.attributes[1], attr("new_total", cap.to_string())); - assert_eq!(res.messages.len(), 0); - - // clear cap - let res = update_cap(deps.as_mut(), env, info, None, None).unwrap(); - assert_eq!(res.attributes.len(), 2); - assert_eq!(res.attributes[0], attr("action", "update_cap")); - assert_eq!(res.messages.len(), 0); - } - - #[test] - fn test_unauthorized_update_cap() { - let mut deps = mock_deps_with_primitives(vec![( - "abc".to_string(), - "123".to_string(), - 100u128.into(), - 100u128.into(), - )]); - let env = mock_env(); - let info = mock_info("not_admin", &[]); - - CAP.save( - &mut deps.storage, - &Cap::new(Addr::unchecked(TEST_ADMIN.to_string()), Uint128::new(100)), - ) - .unwrap(); - - let cap = Uint128::new(1000); - let res = update_cap(deps.as_mut(), env, info, Some(cap), None); - assert!(res.is_err()); - assert_eq!(res.unwrap_err(), ContractError::Unauthorized {}); - } -} diff --git a/smart-contracts/osmosis/contracts/basic-vault/src/helpers.rs b/smart-contracts/osmosis/contracts/basic-vault/src/helpers.rs deleted file mode 100644 index b66296a39..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/src/helpers.rs +++ /dev/null @@ -1,65 +0,0 @@ -use cosmwasm_std::{wasm_execute, Addr, Deps, Env, QuerierWrapper, StdResult, Storage, WasmMsg}; -use lp_strategy::msg::UnbondingClaimResponse; -use vault_rewards::msg::{ExecuteMsg as VaultRewardsExecuteMsg, VaultExecuteMsg}; - -use crate::state::VAULT_REWARDS; -use crate::{state::UnbondingStub, ContractError}; - -pub fn can_unbond_from_primitive( - deps: Deps, - env: &Env, - unbond_id: &str, - stub: &UnbondingStub, -) -> Result { - // only attempt if we already know we passed unlock time. - if !stub - .unlock_time - .map_or(false, |unlock_time| unlock_time < env.block.time) - { - return Ok(false); - } - - let unbonding_claim_query = lp_strategy::msg::QueryMsg::UnbondingClaim { - addr: env.contract.address.clone(), - id: unbond_id.to_string(), - }; - let unbonding_claim: UnbondingClaimResponse = deps - .querier - .query_wasm_smart(stub.address.clone(), &unbonding_claim_query)?; - - // if we attempted to unbond, don't attempt again - if let Some(unbond) = unbonding_claim.unbond { - match unbond.attempted { - true => Ok(false), - false => Ok(unbond.unlock_time < env.block.time), - } - } else { - Ok(true) - } -} - -pub fn update_user_reward_index(storage: &dyn Storage, user: &Addr) -> StdResult { - wasm_execute( - VAULT_REWARDS.load(storage)?, - &VaultRewardsExecuteMsg::Vault(VaultExecuteMsg::UpdateUserRewardIndex(user.to_string())), - vec![], - ) -} - -pub fn is_contract_admin( - querier: &QuerierWrapper, - env: &Env, - sus_admin: &Addr, -) -> Result<(), ContractError> { - let contract_admin = querier - .query_wasm_contract_info(&env.contract.address)? - .admin; - if let Some(contract_admin) = contract_admin { - if contract_admin != *sus_admin { - return Err(ContractError::Unauthorized {}); - } - } else { - return Err(ContractError::Unauthorized {}); - } - Ok(()) -} diff --git a/smart-contracts/osmosis/contracts/basic-vault/src/lib.rs b/smart-contracts/osmosis/contracts/basic-vault/src/lib.rs deleted file mode 100644 index f6bbe7c55..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/src/lib.rs +++ /dev/null @@ -1,17 +0,0 @@ -mod callback; -pub mod contract; -mod error; -mod execute; -mod helpers; -pub mod msg; -mod query; -pub mod state; -pub mod types; - -pub use crate::error::ContractError; - -#[cfg(test)] -pub mod multitest; - -#[cfg(test)] -pub mod tests; diff --git a/smart-contracts/osmosis/contracts/basic-vault/src/msg.rs b/smart-contracts/osmosis/contracts/basic-vault/src/msg.rs deleted file mode 100644 index 8b0f9c94c..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/src/msg.rs +++ /dev/null @@ -1,316 +0,0 @@ -use cosmwasm_schema::{cw_serde, QueryResponses}; - -use cosmwasm_std::{Binary, Coin, Decimal, Timestamp, Uint128}; - -use cw20::{AllowanceResponse, BalanceResponse}; -use cw20::{Expiration, TokenInfoResponse}; -use cw_asset::AssetInfo; -pub use cw_controllers::ClaimsResponse; -use lp_strategy::state::LpCache; -use quasar_types::callback::{BondResponse, StartUnbondResponse, UnbondResponse}; -use vault_rewards::state::DistributionSchedule; - -use crate::state::{BondingStub, Cap, InvestmentInfo, Unbond}; - -#[cw_serde] -pub enum PrimitiveInitMsg { - LP(lp_strategy::msg::InstantiateMsg), -} - -#[cw_serde] -pub struct PrimitiveConfig { - // the weighting of this strategy that the vault should subscribe to (e.g. 30%) - // weights are normalized accross strategies, so values don't need to add up to 100% - pub weight: Decimal, - // the contract address of the stored primitive contract on the chain - pub address: String, - // the Instantiation message for the primitive. - pub init: PrimitiveInitMsg, -} - -#[cw_serde] -pub struct InstantiateMsg { - /// name of the derivative token - pub name: String, - /// description of the derivative token - pub thesis: String, - /// symbol / ticker of the derivative token - pub symbol: String, - /// decimal places of the derivative token (for UI) - pub decimals: u8, - - /// This is the minimum amount we will pull out to reinvest, as well as a minimum - /// that can be unbonded (to avoid needless staking tx) - pub min_withdrawal: Uint128, - /// the denom in which users can deposit - pub deposit_denom: String, - // the array of primitives to subscribe to for this vault - pub primitives: Vec, - // the total amount of tokens that can be deposited, eg: max uosmo of the contract - pub total_cap: Uint128, - - // vault rewards contract code id - pub vault_rewards_code_id: u64, - // vault reward token - pub reward_token: AssetInfo, - // vault reward token distribution schedule - pub reward_distribution_schedules: Vec, -} - -#[cw_serde] -pub enum ExecuteMsg { - /// Bond will bond all staking tokens sent with the message and release derivative tokens - /// recipient will receive the minted vault tokens - Bond { - recipient: Option, - }, - /// Unbond will "burn" the given amount of derivative tokens and send the unbonded - /// staking tokens to the message sender (after exit tax is deducted) - Unbond { - amount: Option, - }, - /// Claim is used to claim your native tokens that you previously "unbonded" - /// after the chain-defined waiting period (eg. 3 weeks) - Claim {}, - /// Clear cache can be triggered by anyone and hits each internal primitive with - /// the TryICQ message. - ClearCache {}, - - // Callback(Callback), - BondResponse(BondResponse), - StartUnbondResponse(StartUnbondResponse), - UnbondResponse(UnbondResponse), - - /// Admin Messages - /// Set Cap is used to set the total cap of the vault, it can also be used to - /// updated admin - SetCap { - new_total: Option, - new_cap_admin: Option, - }, - - /// CW20 Messges - - /// Implements CW20. Transfer is a base message to move tokens to another account without triggering actions - Transfer { - recipient: String, - amount: Uint128, - }, - /// Implements CW20. Burn is a base message to destroy tokens forever - Burn { - amount: Uint128, - }, - /// Implements CW20. Send is a base message to transfer tokens to a contract and trigger an action - /// on the receiving contract. - Send { - contract: String, - amount: Uint128, - msg: Binary, - }, - /// Implements CW20 "approval" extension. Allows spender to access an additional amount tokens - /// from the owner's (env.sender) account. If expires is Some(), overwrites current allowance - /// expiration with this one. - IncreaseAllowance { - spender: String, - amount: Uint128, - expires: Option, - }, - /// Implements CW20 "approval" extension. Lowers the spender's access of tokens - /// from the owner's (env.sender) account by amount. If expires is Some(), overwrites current - /// allowance expiration with this one. - DecreaseAllowance { - spender: String, - amount: Uint128, - expires: Option, - }, - /// Implements CW20 "approval" extension. Transfers amount tokens from owner -> recipient - /// if `env.sender` has sufficient pre-approval. - TransferFrom { - owner: String, - recipient: String, - amount: Uint128, - }, - /// Implements CW20 "approval" extension. Sends amount tokens from owner -> contract - /// if `env.sender` has sufficient pre-approval. - SendFrom { - owner: String, - contract: String, - amount: Uint128, - msg: Binary, - }, - /// Implements CW20 "approval" extension. Destroys tokens forever - BurnFrom { - owner: String, - amount: Uint128, - }, -} - -#[cw_serde] -#[derive(QueryResponses)] -pub enum QueryMsg { - /// Claims shows the number of tokens this address can access when they are done unbonding - #[returns(ClaimsResponse)] - Claims { address: String }, - - #[returns(GetCapResponse)] - GetCap {}, - /// Investment shows metadata on the staking info of the contract - #[returns(InvestmentResponse)] - Investment {}, - /// DepositRatio shows the ratio of tokens that should be sent for a deposit given list of available tokens - #[returns(DepositRatioResponse)] - DepositRatio { funds: Vec }, - - /// PendingBonds shows the bonds that are currently in the process of being deposited for a user - #[returns(PendingBondsResponse)] - PendingBonds { address: String }, - - /// PendingBondsById shows the bonds that are currently in the process of being deposited for a bond id - #[returns(PendingBondsByIdResponse)] - PendingBondsById { bond_id: String }, - - /// GetTvlInfo gets all info necessary for - #[returns(TvlInfoResponse)] - GetTvlInfo {}, - - /// Get all unbonding claims of a user - #[returns(PendingUnbondsResponse)] - PendingUnbonds { address: String }, - - /// Get all unbonding claims for an id - #[returns(PendingUnbondsByIdResponse)] - PendingUnbondsById { bond_id: String }, - - /// GetDebug shows us debug string info - #[returns(GetDebugResponse)] - GetDebug {}, - - /// Implements CW20. Returns the current balance of the given address, 0 if unset. - #[returns(BalanceResponse)] - Balance { address: String }, - /// Implements CW20. Returns metadata on the contract - name, decimals, supply, etc. - #[returns(TokenInfoResponse)] - TokenInfo {}, - /// Additional token metadata, includes regular token info too - #[returns(VaultTokenInfoResponse)] - AdditionalTokenInfo {}, - /// Implements CW20 "allowance" extension. - /// Returns how much spender can use from owner account, 0 if unset. - #[returns(AllowanceResponse)] - Allowance { owner: String, spender: String }, -} - -#[cw_serde] -pub struct MigrateMsg {} - -#[cw_serde] -pub struct InvestmentResponse { - pub info: InvestmentInfo, -} - -#[cw_serde] -pub struct GetCapResponse { - pub cap: Cap, -} - -#[cw_serde] -pub struct DepositRatioResponse { - /// the ratio of tokens that should be sent for a deposit given list of available tokens - pub primitive_funding_amounts: Vec, - pub remainder: Vec, -} - -#[cw_serde] -pub struct PendingBondsResponse { - /// the bonds that are currently in the process of being deposited for a user - pub pending_bonds: Vec, - /// the bond ids that are registered as pending for a user - pub pending_bond_ids: Vec, -} - -#[cw_serde] -pub struct PendingBondsByIdResponse { - /// the bonds that are currently in the process of being deposited for a user - pub pending_bonds: Vec, -} - -#[cw_serde] -pub struct PendingUnbondsByIdResponse { - /// the unbonds that are currently in the process of being withdrawn by an user - pub pending_unbonds: Unbond, -} - -#[cw_serde] -pub struct PendingUnbondsResponse { - /// the unbonds that are currently in the process of being withdrawn by an user - pub pending_unbonds: Vec, - /// the bond ids that are registered as pending for a user - pub pending_unbond_ids: Vec, -} - -#[cw_serde] -pub struct VaultTokenInfoResponse { - pub name: String, - pub thesis: String, - pub symbol: String, - pub decimals: u8, - pub total_supply: Uint128, - pub creation_time: Timestamp, -} - -#[cw_serde] -pub struct GetDebugResponse { - /// the debug string - pub debug: String, -} - -#[cw_serde] -pub struct TvlInfoResponse { - pub primitives: Vec, -} - -#[cw_serde] -pub struct PrimitiveInfo { - pub ica_address: String, - pub base_denom: String, - pub quote_denom: String, - pub lp_denom: String, - pub lp_shares: LpCache, -} - -#[cfg(test)] -mod tests { - use cosmwasm_std::Timestamp; - - use super::*; - - #[test] - fn callback_equals_execute() { - let bond_response = BondResponse { - share_amount: Uint128::one(), - bond_id: "id".to_string(), - }; - let cb = quasar_types::callback::Callback::BondResponse(bond_response.clone()); - let se = serde_json_wasm::to_string(&cb).unwrap(); - let msg: ExecuteMsg = serde_json_wasm::from_str(se.as_str()).unwrap(); - assert_eq!(msg, ExecuteMsg::BondResponse(bond_response)); - - let start_unbond_response = StartUnbondResponse { - unbond_id: "id".to_string(), - unlock_time: Timestamp::from_seconds(100), - }; - let cb = - quasar_types::callback::Callback::StartUnbondResponse(start_unbond_response.clone()); - let se = serde_json_wasm::to_string(&cb).unwrap(); - let msg: ExecuteMsg = serde_json_wasm::from_str(se.as_str()).unwrap(); - assert_eq!(msg, ExecuteMsg::StartUnbondResponse(start_unbond_response)); - - let unbond_response = UnbondResponse { - unbond_id: "id".to_string(), - }; - let cb = quasar_types::callback::Callback::UnbondResponse(unbond_response.clone()); - let se = serde_json_wasm::to_string(&cb).unwrap(); - let msg: ExecuteMsg = serde_json_wasm::from_str(se.as_str()).unwrap(); - assert_eq!(msg, ExecuteMsg::UnbondResponse(unbond_response)); - } -} diff --git a/smart-contracts/osmosis/contracts/basic-vault/src/multitest/common.rs b/smart-contracts/osmosis/contracts/basic-vault/src/multitest/common.rs deleted file mode 100644 index ed098f6f5..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/src/multitest/common.rs +++ /dev/null @@ -1,63 +0,0 @@ -pub use anyhow::Result; -pub use derivative::Derivative; - -pub use crate::contract::{ - execute as execute_vault, instantiate as instantiate_vault, query as query_vault, - reply as reply_vault, -}; -pub use crate::{ - error::ContractError as VaultContractError, - msg::{ - ExecuteMsg as VaultExecuteMsg, InstantiateMsg as VaultInstantiateMsg, - QueryMsg as VaultQueryMsg, - }, -}; -pub use cosmwasm_std::{coin, BlockInfo, Coin, Decimal, Empty, StdResult, Uint128}; -pub use cw_multi_test::{App, AppResponse, Contract, ContractWrapper, Executor}; - -pub use cw_utils::Duration; - -pub use lp_strategy::{ - contract::{execute as execute_primitive, instantiate as instantiate_primitive}, - msg::{ - ExecuteMsg as PrimitiveExecuteMsg, InstantiateMsg as PrimitiveInstantiateMsg, - QueryMsg as PrimitiveQueryMsg, - }, - queries::query as query_primitive, -}; -pub use vault_rewards::{ - contract::{ - execute as execute_vault_rewards, instantiate as instantiate_vault_rewards, - query as query_vault_rewards, - }, - msg::{ - ExecuteMsg as VaultRewardsExecuteMsg, InstantiateMsg as VaultRewardsInstantiateMsg, - QueryMsg as VaultRewardsQueryMsg, - }, -}; - -pub const USER: &str = "user"; -pub const DEPLOYER: &str = "deployer"; -pub const EXECUTOR: &str = "executor"; -pub const DENOM: &str = "uosmo"; -pub const LOCAL_DENOM: &str = "ibc/ilovemymom"; - -pub fn contract_vault() -> Box> { - let contract = - ContractWrapper::new(execute_vault, instantiate_vault, query_vault).with_reply(reply_vault); - Box::new(contract) -} - -pub fn contract_primitive() -> Box> { - let contract = ContractWrapper::new(execute_primitive, instantiate_primitive, query_primitive); - Box::new(contract) -} - -pub fn contract_vault_rewards() -> Box> { - let contract = ContractWrapper::new( - execute_vault_rewards, - instantiate_vault_rewards, - query_vault_rewards, - ); - Box::new(contract) -} diff --git a/smart-contracts/osmosis/contracts/basic-vault/src/multitest/mod.rs b/smart-contracts/osmosis/contracts/basic-vault/src/multitest/mod.rs deleted file mode 100644 index bb18c4535..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/src/multitest/mod.rs +++ /dev/null @@ -1,3 +0,0 @@ -pub mod common; -pub mod suite; -pub mod vault; diff --git a/smart-contracts/osmosis/contracts/basic-vault/src/multitest/suite.rs b/smart-contracts/osmosis/contracts/basic-vault/src/multitest/suite.rs deleted file mode 100644 index 5e8cb8b18..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/src/multitest/suite.rs +++ /dev/null @@ -1,322 +0,0 @@ -use std::str::FromStr; - -use crate::{ - msg::{DepositRatioResponse, PrimitiveConfig}, - multitest::common::*, -}; -use cosmwasm_schema::{schemars, serde}; -use cosmwasm_std::{ - testing::MockApi, Addr, Binary, IbcChannel, IbcEndpoint, IbcMsg, IbcOrder, IbcQuery, - MemoryStorage, StdError, -}; -use cw_multi_test::{ - App, AppBuilder, BankKeeper, CosmosRouter, DistributionKeeper, FailingModule, Ibc, Module, - StakeKeeper, WasmKeeper, -}; -use vault_rewards::state::DistributionSchedule; - -pub type QuasarVaultApp = App< - BankKeeper, - MockApi, - MemoryStorage, - FailingModule, - WasmKeeper, - StakeKeeper, - DistributionKeeper, - AcceptingModule, ->; - -#[derive(Derivative)] -#[derivative(Debug)] -pub struct QuasarVaultSuite { - #[derivative(Debug = "ignore")] - pub app: QuasarVaultApp, - // The account that deploys everything - pub deployer: Addr, - // executor address - pub executor: Addr, - // user address - pub user: Addr, - // vault address - pub vault: Addr, - // primitive address - pub primitive: Addr, -} - -pub struct AcceptingModule; - -impl Module for AcceptingModule { - type ExecT = IbcMsg; - type QueryT = IbcQuery; - type SudoT = Empty; - - fn execute( - &self, - _api: &dyn cosmwasm_std::Api, - _storage: &mut dyn cosmwasm_std::Storage, - _router: &dyn CosmosRouter, - _block: &cosmwasm_std::BlockInfo, - _sender: cosmwasm_std::Addr, - _msg: Self::ExecT, - ) -> anyhow::Result - where - ExecC: std::fmt::Debug - + Clone - + PartialEq - + schemars::JsonSchema - + serde::de::DeserializeOwned - + 'static, - QueryC: cosmwasm_std::CustomQuery + serde::de::DeserializeOwned + 'static, - { - Ok(AppResponse::default()) - } - - fn sudo( - &self, - _api: &dyn cosmwasm_std::Api, - _storage: &mut dyn cosmwasm_std::Storage, - _router: &dyn CosmosRouter, - _block: &cosmwasm_std::BlockInfo, - _msg: Self::SudoT, - ) -> anyhow::Result - where - ExecC: std::fmt::Debug - + Clone - + PartialEq - + schemars::JsonSchema - + serde::de::DeserializeOwned - + 'static, - QueryC: cosmwasm_std::CustomQuery + serde::de::DeserializeOwned + 'static, - { - Ok(AppResponse::default()) - } - - fn query( - &self, - _api: &dyn cosmwasm_std::Api, - _storage: &dyn cosmwasm_std::Storage, - _querier: &dyn cosmwasm_std::Querier, - _block: &cosmwasm_std::BlockInfo, - _request: Self::QueryT, - ) -> anyhow::Result { - Ok(Binary::default()) - } -} - -impl Ibc for AcceptingModule {} - -impl QuasarVaultSuite { - pub fn init( - init_msg: Option, - funds: Option>, - ) -> Result { - let genesis_funds = vec![coin(150000, DENOM), coin(150000, LOCAL_DENOM)]; - let deployer = Addr::unchecked(DEPLOYER); - let executor = Addr::unchecked(EXECUTOR); - let user = Addr::unchecked(USER); - let mut app = AppBuilder::new() - .with_ibc(AcceptingModule) - .build(|router, _, storage| { - router - .bank - .init_balance(storage, &deployer, genesis_funds) - .unwrap(); - }); - // let mut app = App::new(|router, _, storage| { - // router - // .bank - // .init_balance(storage, &deployer, genesis_funds) - // .unwrap(); - // }); - app.send_tokens( - deployer.clone(), - user.clone(), - &[coin(50000, DENOM), coin(50000, LOCAL_DENOM)], - )?; - app.send_tokens( - deployer.clone(), - executor.clone(), - &[coin(50000, DENOM), coin(50000, LOCAL_DENOM)], - )?; - - let vault_id = app.store_code(contract_vault()); - let primitive_id = app.store_code(contract_primitive()); - - let primitive = app - .instantiate_contract( - primitive_id, - deployer.clone(), - &PrimitiveInstantiateMsg { - lock_period: 64, - pool_id: 1, - pool_denom: "gamm/pool/1".to_string(), - local_denom: LOCAL_DENOM.to_string(), - base_denom: DENOM.to_string(), - quote_denom: "uatom".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }, - &[], - "router_contract", - Some(deployer.to_string()), - ) - .unwrap(); - - // IbcChannelOpenMsg::OpenInit { channel: () } - // app.wasm_sudo(contract_addr, msg) - let endpoint = IbcEndpoint { - port_id: "wasm.my_addr".to_string(), - channel_id: "channel-1".to_string(), - }; - let counterparty_endpoint = IbcEndpoint { - port_id: "icahost".to_string(), - channel_id: "channel-2".to_string(), - }; - - let version = r#"{"version":"ics27-1","encoding":"proto3","tx_type":"sdk_multi_msg","controller_connection_id":"connection-0","host_connection_id":"connection-0"}"#.to_string(); - let _channel = IbcChannel::new( - endpoint, - counterparty_endpoint, - IbcOrder::Ordered, - version, - "connection-0".to_string(), - ); - - // Todo: keep track of this issue for ibc mock support in cw-multi-test: https://github.com/CosmWasm/cw-multi-test/issues/27 - // let ibc_channel_open_msg = IbcChannelOpenMsg::OpenInit { channel }; - // let res = app.execute( - // primitive.clone(), - // CosmosMsg::Ibc(IbcMsg::SendPacket { - // channel_id: "channel-0".to_string(), - // data: to_json_binary(&ibc_channel_open_msg)?, - // timeout: IbcTimeout::with_block(IbcTimeoutBlock { - // revision: 1, - // height: app.block_info().height + 5, - // }), - // }), - // ); - // res.unwrap(); - // IbcChannelConnectMsg::OpenConfirm { channel: () } - - let vault_rewards_id = app.store_code(contract_vault_rewards()); - - let vault = app - .instantiate_contract( - vault_id, - deployer.clone(), - &init_msg.unwrap_or(VaultInstantiateMsg { - name: "orion".to_string(), - thesis: "to generate yield, I guess".to_string(), - symbol: "ORN".to_string(), - decimals: 6, - min_withdrawal: 1u128.into(), - total_cap: 100000000u128.into(), - primitives: vec![PrimitiveConfig { - weight: Decimal::from_str("0.33333333333")?, - address: primitive.to_string(), - init: crate::msg::PrimitiveInitMsg::LP(PrimitiveInstantiateMsg { - lock_period: 64, - pool_id: 1, - pool_denom: "gamm/pool/1".to_string(), - local_denom: LOCAL_DENOM.to_string(), - base_denom: DENOM.to_string(), - quote_denom: "uatom".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }), - }], - vault_rewards_code_id: vault_rewards_id, - reward_token: cw_asset::AssetInfoBase::Native("uqsr".to_string()), - reward_distribution_schedules: vec![ - vault_rewards::state::DistributionSchedule { - start: app.block_info().height + 1, - end: app.block_info().height + 501, - amount: Uint128::from(1000u128), - }, - ], - deposit_denom: LOCAL_DENOM.to_string(), - }), - &funds.unwrap_or_default(), - "vault_contract", - Some(deployer.to_string()), // admin: Option, will need this for upgrading - ) - .unwrap(); - - // set depositor on primitive as the vault address - let msg = PrimitiveExecuteMsg::SetDepositor { - depositor: vault.to_string(), - }; - app.execute_contract(deployer.clone(), primitive.clone(), &msg, &[]) - .unwrap(); - - Ok(QuasarVaultSuite { - app, - user, - executor, - deployer, - primitive, - vault, - }) - } - - pub fn query_balance(&self, addr: &Addr) -> StdResult { - self.app.wrap().query_balance(addr.as_str(), DENOM) - } - - pub fn unbond( - &mut self, - sender: &Addr, - unbond_amount: Option, - ) -> Result<(), VaultContractError> { - let msg = VaultExecuteMsg::Unbond { - amount: unbond_amount, - }; - self.app - .execute_contract(sender.clone(), self.vault.clone(), &msg, &[]) - .map_err(|err| err.downcast().unwrap()) - .map(|_| ()) - } - - pub fn bond(&mut self, sender: &Addr, funds: Vec) -> Result<(), VaultContractError> { - let msg = VaultExecuteMsg::Bond { - recipient: Option::None, - }; - self.app - .execute_contract(sender.clone(), self.vault.clone(), &msg, &funds) - .map_err(|err| match err.downcast::() { - Ok(err_unwrapped) => err_unwrapped, - Err(e) => VaultContractError::Std(StdError::GenericErr { - msg: e.root_cause().to_string(), - }), - }) - .map(|_| ()) - } - - pub fn query_deposit_ratio(&self, funds: Vec) -> StdResult { - let msg = VaultQueryMsg::DepositRatio { funds }; - self.app.wrap().query_wasm_smart(self.vault.clone(), &msg) - } - - pub fn add_distribution_schedule(&mut self, schedule: DistributionSchedule) { - let msg = VaultRewardsExecuteMsg::Admin( - vault_rewards::msg::AdminExecuteMsg::AddDistributionSchedule(schedule), - ); - self.app - .execute_contract(self.deployer.clone(), self.vault.clone(), &msg, &[]) - .unwrap(); - } - - pub fn fast_forward_block_time(&mut self, forward_time_sec: u64) { - let block = self.app.block_info(); - - let mock_block = BlockInfo { - height: block.height + 10, - chain_id: block.chain_id, - time: block.time.plus_seconds(forward_time_sec), - }; - - self.app.set_block(mock_block); - } -} diff --git a/smart-contracts/osmosis/contracts/basic-vault/src/multitest/vault.rs b/smart-contracts/osmosis/contracts/basic-vault/src/multitest/vault.rs deleted file mode 100644 index 626478699..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/src/multitest/vault.rs +++ /dev/null @@ -1,23 +0,0 @@ -use crate::multitest::common::*; -use crate::multitest::suite::*; - -#[test] -fn try_bond() { - let mut suite = QuasarVaultSuite::init(None, None).unwrap(); - let err = suite - .bond( - &suite.user.clone(), - vec![Coin { - denom: LOCAL_DENOM.to_string(), - amount: Uint128::from(1000u128), - }], - ) - .unwrap_err(); - // this error happens because our ibc channel is not open yet - assert_eq!( - err, - VaultContractError::Std(cosmwasm_std::StdError::GenericErr { - msg: "type: alloc::string::String; key: [69, 63, 71, 5F, 63, 68, 61, 6E, 6E, 65, 6C] not found".to_string() - }) - ); -} diff --git a/smart-contracts/osmosis/contracts/basic-vault/src/query.rs b/smart-contracts/osmosis/contracts/basic-vault/src/query.rs deleted file mode 100644 index 9904a7d09..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/src/query.rs +++ /dev/null @@ -1,120 +0,0 @@ -use cosmwasm_std::{Addr, Coin, Deps, StdResult}; -use lp_strategy::msg::{ConfigResponse, IcaAddressResponse, LpSharesResponse, QueryMsg}; - -use crate::{ - execute::may_pay_with_ratio, - msg::{ - DepositRatioResponse, InvestmentResponse, PendingBondsByIdResponse, PendingBondsResponse, - PendingUnbondsByIdResponse, PendingUnbondsResponse, PrimitiveInfo, TvlInfoResponse, - }, - state::{Unbond, BOND_STATE, INVESTMENT, PENDING_BOND_IDS, PENDING_UNBOND_IDS, UNBOND_STATE}, -}; - -pub fn query_tvl_info(deps: Deps) -> StdResult { - let primitives = INVESTMENT.load(deps.storage)?.primitives; - let mut prim_infos: Vec = Vec::new(); - for prim in primitives { - let addr = deps.api.addr_validate(prim.address.as_str())?; - let ica = deps - .querier - .query_wasm_smart::(addr.as_str(), &QueryMsg::IcaAddress {})?; - let lp_shares = deps - .querier - .query_wasm_smart::(addr.as_str(), &QueryMsg::LpShares {})? - .lp_shares; - let config = deps - .querier - .query_wasm_smart::(addr.as_str(), &QueryMsg::Config {})? - .config; - prim_infos.push(PrimitiveInfo { - ica_address: ica.address, - base_denom: config.base_denom, - quote_denom: config.quote_denom, - lp_denom: config.pool_denom, - lp_shares, - }) - } - Ok(TvlInfoResponse { - primitives: prim_infos, - }) -} - -pub fn query_investment(deps: Deps) -> StdResult { - let invest = INVESTMENT.load(deps.storage)?; - - let res = InvestmentResponse { info: invest }; - Ok(res) -} - -pub fn query_deposit_ratio(deps: Deps, funds: Vec) -> StdResult { - let invest = INVESTMENT.load(deps.storage)?; - - let (primitive_funding_amounts, remainder) = may_pay_with_ratio(&deps, &funds, invest).unwrap(); - - let res = DepositRatioResponse { - primitive_funding_amounts, - remainder, - }; - Ok(res) -} - -pub fn query_pending_bonds(deps: Deps, address: String) -> StdResult { - let pending_bond_ids = PENDING_BOND_IDS.may_load(deps.storage, Addr::unchecked(address))?; - let mut pending_bonds = vec![]; - - pending_bond_ids.clone().unwrap().iter().for_each(|id| { - let mut deposit_stubs = BOND_STATE.load(deps.storage, id.to_string()).unwrap(); - - pending_bonds.append(deposit_stubs.as_mut()); - }); - - Ok(PendingBondsResponse { - pending_bonds, - pending_bond_ids: pending_bond_ids.unwrap(), - }) -} - -pub fn query_pending_unbonds(deps: Deps, address: String) -> StdResult { - let pending_unbond_ids = PENDING_UNBOND_IDS.may_load(deps.storage, Addr::unchecked(address))?; - let mut pending_unbonds: Vec = vec![]; - - if pending_unbond_ids.is_none() { - return Ok(PendingUnbondsResponse { - pending_unbonds, - pending_unbond_ids: vec![], - }); - } - - pending_unbond_ids - .clone() - .unwrap() - .iter() - .for_each(|id: &String| { - let unbond_stubs: Unbond = UNBOND_STATE.load(deps.storage, id.to_string()).unwrap(); - pending_unbonds.push(unbond_stubs); - }); - - Ok(PendingUnbondsResponse { - pending_unbonds, - pending_unbond_ids: pending_unbond_ids.unwrap(), - }) -} - -pub fn query_pending_bonds_by_id(deps: Deps, id: String) -> StdResult { - let deposit_stubs = BOND_STATE.load(deps.storage, id).unwrap(); - - Ok(PendingBondsByIdResponse { - pending_bonds: deposit_stubs, - }) -} - -pub fn query_pending_unbonds_by_id( - deps: Deps, - id: String, -) -> StdResult { - let unbond_stubs = UNBOND_STATE.load(deps.storage, id).unwrap(); - - Ok(PendingUnbondsByIdResponse { - pending_unbonds: unbond_stubs, - }) -} diff --git a/smart-contracts/osmosis/contracts/basic-vault/src/state.rs b/smart-contracts/osmosis/contracts/basic-vault/src/state.rs deleted file mode 100644 index 8b1d76ffe..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/src/state.rs +++ /dev/null @@ -1,194 +0,0 @@ -use cosmwasm_schema::cw_serde; -use cosmwasm_std::{Addr, Coin, Decimal, Timestamp, Uint128}; - -use cw_controllers::Claims; -use cw_storage_plus::{Item, Map}; -use quasar_types::callback::{BondResponse, UnbondResponse}; - -use crate::{msg::PrimitiveConfig, ContractError}; - -// constants -pub const FALLBACK_RATIO: Decimal = Decimal::one(); - -// reply ids -pub const STRATEGY_BOND_ID: u64 = 80085; - -// version info for migration info -pub const CONTRACT_NAME: &str = "basic-vault"; -pub const CONTRACT_VERSION: &str = env!("CARGO_PKG_VERSION"); - -pub const CLAIMS: Claims = Claims::new("claims"); - -/// Investment info is fixed at instantiation, and is used to control the function of the contract -#[cw_serde] -pub struct InvestmentInfo { - /// Owner created the contract and takes a cut - pub owner: Addr, - /// This is the minimum amount we will pull out to reinvest, as well as a minimum - /// that can be unbonded (to avoid needless staking tx) - pub min_withdrawal: Uint128, - /// the denom accepted by the vault - pub deposit_denom: String, - /// this is the array of primitives that this vault will subscribe to - pub primitives: Vec, -} - -pub const CAP: Item = Item::new("cap"); - -#[cw_serde] -pub struct Cap { - cap_admin: Addr, - total: Uint128, - current: Uint128, -} - -impl Cap { - pub fn new(admin: Addr, total: Uint128) -> Self { - Self { - cap_admin: admin, - total, - current: Uint128::zero(), - } - } - - pub fn update_cap_admin(mut self, new_cap_admin: Addr) -> Self { - self.cap_admin = new_cap_admin; - self - } - - pub fn update_total_cap(mut self, new_total: Uint128) -> Self { - self.total = new_total; - self - } - - pub fn update_current(mut self, to_add: Uint128) -> Result { - let new_total = self.current.checked_add(to_add)?; - // if we go over cap, reject - if new_total > self.total { - return Err(ContractError::OverCap {}); - }; - self.current = new_total; - Ok(self) - } -} - -#[cw_serde] -pub struct AdditionalTokenInfo { - pub thesis: String, - pub creation_time: Timestamp, -} - -pub const ADDITIONAL_TOKEN_INFO: Item = Item::new("additional_token_info"); -pub const OLD_INVESTMENT: Item = Item::new("invest"); -pub const INVESTMENT: Item = Item::new("new_invest"); - -/// OldInvestmentInfo is a premigration version without the single_denom -#[cw_serde] -pub struct OldInvestmentInfo { - /// Owner created the contract and takes a cut - pub owner: Addr, - /// This is the minimum amount we will pull out to reinvest, as well as a minimum - /// that can be unbonded (to avoid needless staking tx) - pub min_withdrawal: Uint128, - /// this is the array of primitives that this vault will subscribe to - pub primitives: Vec, -} - -#[cw_serde] -#[derive(Default)] -pub struct BondingStub { - // the contract address of the primitive - pub address: String, - // the response of the primitive upon successful bond or error - pub bond_response: Option, - // primitive value at the time of receiving the bond_response - pub primitive_value: Option, - // the amount sent with the Bond - pub amount: Uint128, -} - -#[cw_serde] -#[derive(Default)] -pub struct Unbond { - pub stub: Vec, - pub shares: Uint128, -} - -#[cw_serde] -#[derive(Default)] -pub struct UnbondingStub { - // the contract address of the primitive - pub address: String, - // the response of the primitive upon successful bond or error - pub unlock_time: Option, - // response of the unbond, if this is present then we have finished unbonding - pub unbond_response: Option, - // funds attached to the unbond_response - pub unbond_funds: Vec, -} - -// (un)bonding sequence number (to map primitive responses to the right bond action) -pub const BONDING_SEQ: Item = Item::new("bond_seq"); -// mapping from bonding sequence number to depositor/withdrawer address -pub const BONDING_SEQ_TO_ADDR: Map = Map::new("bond_seq_to_addr"); -// current bonds pending for a user -pub const PENDING_BOND_IDS: Map> = Map::new("pending_bond_ids"); -// current unbonds pending for a user -pub const PENDING_UNBOND_IDS: Map> = Map::new("pending_unbond_ids"); -// map of bond id to bond state -pub const BOND_STATE: Map> = Map::new("bond_state"); -// map of unbond id to unbond state -pub const UNBOND_STATE: Map = Map::new("unbond_state"); - -pub const DEBUG_TOOL: Item = Item::new("debug_tool"); - -// vault rewards contract -pub const VAULT_REWARDS: Item = Item::new("vault_rewards"); - -impl InvestmentInfo { - pub fn normalize_primitive_weights(&mut self) { - let mut total_weight = Decimal::zero(); - for p in &self.primitives { - total_weight += p.weight; - } - for p in &mut self.primitives { - p.weight /= total_weight; - } - } -} - -#[cfg(test)] -mod tests { - use super::*; - - use crate::msg::{PrimitiveConfig, PrimitiveInitMsg}; - - #[test] - fn test_investment_info() { - let mut invest = InvestmentInfo { - owner: Addr::unchecked("owner".to_string()), - min_withdrawal: Uint128::from(1000u128), - primitives: vec![ - PrimitiveConfig { - address: "primitive".to_string(), - weight: Decimal::percent(50), // passing in unnormalized - init: PrimitiveInitMsg::LP(lp_strategy::msg::InstantiateMsg { - lock_period: 1, - pool_id: 1, - pool_denom: "gamm/pool/1".to_string(), - local_denom: "uosmo".to_string(), - base_denom: "ibc/blah".to_string(), - quote_denom: "ibc/blah2".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }), - }; - 4 - ], - deposit_denom: "ibc/osmo".to_string(), - }; - invest.normalize_primitive_weights(); - assert_eq!(invest.primitives[0].weight, Decimal::percent(25)); - } -} diff --git a/smart-contracts/osmosis/contracts/basic-vault/src/tests.rs b/smart-contracts/osmosis/contracts/basic-vault/src/tests.rs deleted file mode 100644 index c034f1d02..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/src/tests.rs +++ /dev/null @@ -1,2228 +0,0 @@ -use core::panic; -use std::{marker::PhantomData, str::FromStr}; - -use cosmwasm_std::{ - coins, from_json, - testing::{mock_dependencies, mock_env, mock_info, MockApi, MockStorage}, - to_json_binary, Addr, Attribute, BankMsg, Binary, Coin, ContractInfoResponse, ContractResult, - CosmosMsg, Decimal, DepsMut, Empty, Env, Fraction, MessageInfo, OwnedDeps, Querier, - QuerierResult, QueryRequest, Reply, Response, StdError, StdResult, SubMsgResponse, - SubMsgResult, Timestamp, Uint128, WasmMsg, -}; -use cw20::BalanceResponse; - -use cw_asset::AssetInfoBase; -use cw_utils::PaymentError; -use lp_strategy::{ - msg::{ConfigResponse, IcaBalanceResponse, PrimitiveSharesResponse, UnbondingClaimResponse}, - state::{Config, Unbond}, -}; -use prost::Message; -use quasar_types::{ - callback::{BondResponse, StartUnbondResponse, UnbondResponse}, - types::{CoinRatio, CoinWeight}, -}; - -use crate::{ - callback::on_bond, - contract::execute, - contract::instantiate, - contract::query, - contract::{reply, REPLY_INIT_VAULT_REWARDS}, - execute::{ - get_deposit_amount_weights, get_deposit_and_remainder_for_ratio, get_max_bond, - get_token_amount_weights, may_pay_with_ratio, - }, - msg::{ExecuteMsg, InstantiateMsg, InvestmentResponse, PrimitiveConfig, PrimitiveInitMsg}, - state::{BONDING_SEQ, VAULT_REWARDS}, -}; - -#[derive(Clone, PartialEq, prost::Message)] -struct MsgInstantiateContractResponse { - #[prost(string, tag = "1")] - pub contract_address: ::prost::alloc::string::String, - #[prost(bytes, tag = "2")] - pub data: ::prost::alloc::vec::Vec, -} - -pub struct QuasarQuerier { - // address, denom, share, balance - pub primitive_states: Vec<(String, String, Uint128, Uint128)>, - // address, unlock_time, attempted - pub primitive_unlock_times: Vec<(String, Timestamp, bool)>, -} - -impl QuasarQuerier { - pub fn new(primitive_states: Vec<(String, String, Uint128, Uint128)>) -> QuasarQuerier { - QuasarQuerier { - primitive_states, - primitive_unlock_times: vec![], - } - } - - /// update the state of the quasar querier, expects denom, shares, balance - pub fn update_state(&mut self, new_states: Vec<(&str, Uint128, Uint128)>) { - new_states - .into_iter() - .for_each(|(address, shares, balance)| { - let val = self - .primitive_states - .iter_mut() - .find(|(prim_addr, _, _, _)| prim_addr == address) - .unwrap(); - val.2 = shares; - val.3 = balance; - }) - } - - pub fn find_states_for_primitive(&self, address: String) -> (String, Uint128, Uint128) { - let mut total_share = Uint128::zero(); - let mut total_balance = Uint128::zero(); - let mut this_denom = "".to_string(); - for (addr, denom, share, balance) in &self.primitive_states { - if addr.eq(&address) { - this_denom = denom.to_string(); - total_share = *share; - total_balance = *balance; - } - } - (this_denom, total_share, total_balance) - } - - pub fn set_unbonding_claim_for_primitive( - &mut self, - address: String, - time: Timestamp, - attempted: bool, - ) { - let put = self - .primitive_unlock_times - .iter_mut() - .find(|put| put.0 == address); - match put { - Some(p) => { - p.1 = time; - p.2 = attempted; - } - None => self.primitive_unlock_times.push((address, time, attempted)), - } - } - - pub fn get_unbonding_claim_for_primitive( - &self, - address: String, - ) -> StdResult<(Timestamp, bool)> { - let prim = self.primitive_unlock_times.iter().find(|p| p.0 == address); - - match prim { - Some(p) => Ok((p.1, p.2)), - None => Err(StdError::GenericErr { - msg: "Unbonding claim not found".to_owned(), - }), - } - } -} - -impl Querier for QuasarQuerier { - fn raw_query(&self, bin_request: &[u8]) -> cosmwasm_std::QuerierResult { - let request: QueryRequest = from_json(&Binary::from(bin_request)).unwrap(); - match request { - QueryRequest::Wasm(wasm_query) => match wasm_query { - cosmwasm_std::WasmQuery::Smart { contract_addr, msg } => { - let primitive_query = from_json::(&msg).unwrap(); - - let (this_denom, total_share, total_balance) = - self.find_states_for_primitive(contract_addr.clone()); - match primitive_query { - lp_strategy::msg::QueryMsg::PrimitiveShares {} => { - let response = PrimitiveSharesResponse { total: total_share }; - QuerierResult::Ok(ContractResult::Ok( - to_json_binary(&response).unwrap(), - )) - } - lp_strategy::msg::QueryMsg::IcaBalance {} => { - let response = IcaBalanceResponse { - amount: Coin { - denom: this_denom, - amount: total_balance, - }, - }; - QuerierResult::Ok(ContractResult::Ok( - to_json_binary(&response).unwrap(), - )) - } - lp_strategy::msg::QueryMsg::Config {} => { - let config = Config { - lock_period: 14, - pool_id: 1, - pool_denom: "gamm/pool/1".to_string(), - local_denom: this_denom, - base_denom: "uosmo".to_string(), - quote_denom: "uatom".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }; - QuerierResult::Ok(ContractResult::Ok( - to_json_binary(&ConfigResponse { config }).unwrap(), - )) - } - lp_strategy::msg::QueryMsg::UnbondingClaim { addr: _, id } => { - let query_result = - self.get_unbonding_claim_for_primitive(contract_addr); - QuerierResult::Ok(match query_result { - Ok((unlock_time, attempted)) => ContractResult::Ok( - to_json_binary(&UnbondingClaimResponse { - unbond: Some(Unbond { - lp_shares: Uint128::from(1u128), - unlock_time, - attempted, - owner: Addr::unchecked(TEST_CREATOR), - id, - }), - }) - .unwrap(), - ), - Err(error) => ContractResult::Err(error.to_string()), - }) - } - lp_strategy::msg::QueryMsg::Balance { address: _ } => { - let response = BalanceResponse { - balance: total_share, - }; - QuerierResult::Ok(ContractResult::Ok( - to_json_binary(&response).unwrap(), - )) - } - _ => QuerierResult::Err(cosmwasm_std::SystemError::UnsupportedRequest { - kind: format!("Unmocked primitive query type: {primitive_query:?}"), - }), - } - } - cosmwasm_std::WasmQuery::ContractInfo { contract_addr: _ } => { - let mut response = ContractInfoResponse::default(); - response.admin = Some(TEST_ADMIN.to_string()); - QuerierResult::Ok(ContractResult::Ok(to_json_binary(&response).unwrap())) - } - _ => QuerierResult::Err(cosmwasm_std::SystemError::UnsupportedRequest { - kind: format!("Unmocked wasm query type: {wasm_query:?}"), - }), - }, - _ => QuerierResult::Err(cosmwasm_std::SystemError::UnsupportedRequest { - kind: format!("Unmocked query type: {request:?}"), - }), - } - // QuerierResult::Ok(ContractResult::Ok(to_json_binary(&"hello").unwrap())) - } -} - -pub fn mock_deps_with_primitives( - primitive_states: Vec<(String, String, Uint128, Uint128)>, -) -> OwnedDeps { - OwnedDeps { - storage: MockStorage::default(), - api: MockApi::default(), - querier: QuasarQuerier::new(primitive_states), - custom_query_type: PhantomData, - } -} - -pub const TEST_ADMIN: &str = "admin"; -pub const TEST_CREATOR: &str = "creator"; -pub const TEST_DEPOSITOR: &str = "depositor"; - -fn init_msg() -> InstantiateMsg { - InstantiateMsg { - name: "Blazar Vault".to_string(), - thesis: "to generate yield, I guess".to_string(), - symbol: "BLZR".to_string(), - decimals: 6, - min_withdrawal: Uint128::one(), - primitives: vec![PrimitiveConfig { - weight: Decimal::from_str("1.0").unwrap(), - address: "quasar123".to_string(), - init: PrimitiveInitMsg::LP(lp_strategy::msg::InstantiateMsg { - lock_period: 14, // this is supposed to be nanos i think - pool_id: 1, - pool_denom: "gamm/pool/1".to_string(), - local_denom: "ibc/uosmo".to_string(), - base_denom: "uosmo".to_string(), - quote_denom: "uatom".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }), - }], - vault_rewards_code_id: 123, - reward_token: cw_asset::AssetInfoBase::Native("uqsr".to_string()), - reward_distribution_schedules: vec![vault_rewards::state::DistributionSchedule { - start: 1, - end: 501, - amount: Uint128::from(1000u128), - }], - total_cap: Uint128::new(10_000_000_000_000), - deposit_denom: "ibc/uosmo".to_string(), - } -} - -fn reply_msg() -> Reply { - let instantiate_reply = MsgInstantiateContractResponse { - contract_address: "vault_rewards_addr".to_string(), - data: vec![], - }; - let mut encoded_instantiate_reply = Vec::::with_capacity(instantiate_reply.encoded_len()); - instantiate_reply - .encode(&mut encoded_instantiate_reply) - .unwrap(); - - // reply to init our map for the vault rewards contract - Reply { - id: REPLY_INIT_VAULT_REWARDS, - result: SubMsgResult::Ok(SubMsgResponse { - events: vec![], - data: Some(encoded_instantiate_reply.into()), - }), - } -} - -// convenience function to init primitives with a vec of tuples which are (local_denom, weight) - -fn init_msg_with_primitive_details( - primitive_details: Vec<(String, String, Decimal)>, -) -> InstantiateMsg { - InstantiateMsg { - name: "Blazar Vault".to_string(), - thesis: "to generate yield, I guess".to_string(), - symbol: "BLZR".to_string(), - decimals: 6, - min_withdrawal: Uint128::one(), - vault_rewards_code_id: 123, - reward_token: cw_asset::AssetInfoBase::Native("uqsr".to_string()), - reward_distribution_schedules: vec![vault_rewards::state::DistributionSchedule { - start: 1, - end: 501, - amount: Uint128::from(1000u128), - }], - primitives: primitive_details - .iter() - .map(|pd| { - PrimitiveConfig { - weight: pd.2, - address: pd.0.clone(), - init: PrimitiveInitMsg::LP(lp_strategy::msg::InstantiateMsg { - lock_period: 14, // this is supposed to be nanos i think - pool_id: 1, - pool_denom: "gamm/pool/1".to_string(), - local_denom: pd.1.clone(), - base_denom: "uosmo".to_string(), - quote_denom: "uatom".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }), - } - }) - .collect(), - total_cap: Uint128::new(10_000_000_000_000), - // TODO this is kind of sloppy and should be a param - deposit_denom: primitive_details[0].1.clone(), - } -} - -fn init(deps: DepsMut, msg: &InstantiateMsg, env: &Env, info: &MessageInfo) -> Response { - instantiate(deps, env.clone(), info.clone(), msg.clone()).unwrap() -} - -#[test] -fn proper_initialization() { - let mut deps = mock_deps_with_primitives(even_primitives()); - let msg = init_msg(); - let info = mock_info(TEST_CREATOR, &[]); - let env = mock_env(); - let res = init(deps.as_mut(), &msg, &env, &info); - - println!("res: {res:?}"); - assert_eq!(1, res.messages.len()); - - if let CosmosMsg::Wasm(WasmMsg::Instantiate { - code_id, - msg, - funds, - admin, - label, - }) = &res.messages[0].msg - { - assert_eq!(123, *code_id); - assert_eq!(0, funds.len()); - assert_eq!(admin.clone().unwrap(), TEST_CREATOR); - assert_eq!(label, "vault-rewards"); - let msg: vault_rewards::msg::InstantiateMsg = from_json(msg).unwrap(); - assert_eq!( - Uint128::from(1000u128), - msg.distribution_schedules[0].amount - ); - if let AssetInfoBase::Native(native) = &msg.reward_token { - assert_eq!("uqsr", native); - } else { - panic!("Unexpected reward token type"); - } - } else { - panic!("Unexpected message type"); - } -} - -#[test] -fn proper_bond_with_one_primitive() { - let mut deps = mock_deps_with_primitives(vec![( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - )]); - let msg = init_msg(); - let info = mock_info(TEST_CREATOR, &[]); - let env = mock_env(); - let _res = init(deps.as_mut(), &msg, &env, &info); - - let deposit_info = mock_info( - TEST_DEPOSITOR, - &[Coin { - denom: "ibc/uosmo".to_string(), - amount: Uint128::from(100u128), - }], - ); - let deposit_msg = ExecuteMsg::Bond { - recipient: Option::None, - }; - - let res = execute(deps.as_mut(), env, deposit_info, deposit_msg).unwrap(); - assert_eq!(res.messages.len(), 1); - assert_eq!(res.attributes.first().unwrap().value, "1"); - - if let CosmosMsg::Wasm(wasm_msg) = &res.messages.first().unwrap().msg { - if let WasmMsg::Execute { - contract_addr, - msg: _, - funds, - } = wasm_msg - { - assert_eq!(contract_addr, "quasar123"); - assert_eq!(funds[0].amount, Uint128::from(100u128)); - } else { - panic!("Wrong message type") - } - } -} - -fn even_primitives() -> Vec<(String, String, Uint128, Uint128)> { - vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - ), - ( - "quasar125".to_string(), - "ibc/ustars".to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - ), - ] -} - -fn even_primitives_single_token() -> Vec<(String, String, Uint128, Uint128)> { - vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - ), - ( - "quasar124".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - ), - ( - "quasar125".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - ), - ] -} - -fn even_primitive_details() -> Vec<(String, String, Decimal)> { - vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Decimal::one(), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Decimal::one(), - ), - ( - "quasar125".to_string(), - "ibc/ustars".to_string(), - Decimal::one(), - ), - ] -} - -fn even_primitive_details_single_token() -> Vec<(String, String, Decimal)> { - vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Decimal::one(), - ), - ( - "quasar124".to_string(), - "ibc/uosmo".to_string(), - Decimal::one(), - ), - ( - "quasar125".to_string(), - "ibc/uosmo".to_string(), - Decimal::one(), - ), - ] -} - -fn even_deposit() -> Vec { - vec![ - Coin { - denom: "ibc/uosmo".to_string(), - amount: Uint128::from(100u128), - }, - Coin { - denom: "ibc/uatom".to_string(), - amount: Uint128::from(100u128), - }, - Coin { - denom: "ibc/ustars".to_string(), - amount: Uint128::from(100u128), - }, - ] -} - -fn even_deposit_single_token() -> Vec { - vec![Coin { - denom: "ibc/uosmo".to_string(), - amount: Uint128::from(300u128), - }] -} - -fn _uneven_primitives() -> Vec<(String, String, Uint128, Uint128)> { - vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Uint128::from(200u128), - Uint128::from(400u128), - ), - ] -} - -fn _uneven_primitive_details() -> Vec<(String, String, Decimal)> { - vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Decimal::from_str("2.0").unwrap(), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Decimal::one(), - ), - ] -} - -fn _uneven_deposit() -> Vec { - vec![ - Coin { - denom: "ibc/uosmo".to_string(), - amount: Uint128::from(1000u128), - }, - Coin { - denom: "ibc/uatom".to_string(), - amount: Uint128::from(200u128), - }, - ] -} - -#[test] -fn test_get_deposit_amount_weights() { - let primitive_states = vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Uint128::from(200u128), - Uint128::from(400u128), - ), - ]; - let primitive_deets = vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Decimal::from_str("3.5").unwrap(), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Decimal::one(), - ), - ]; - - let mut deps = mock_deps_with_primitives(primitive_states.clone()); - let init_msg = init_msg_with_primitive_details(primitive_deets.clone()); - - let info = mock_info(TEST_CREATOR, &[]); - let env = mock_env(); - let res = init(deps.as_mut(), &init_msg, &env, &info); - assert_eq!(1, res.messages.len()); - - let invest_query = crate::msg::QueryMsg::Investment {}; - let query_res = query(deps.as_ref(), env, invest_query).unwrap(); - - let investment_response: InvestmentResponse = from_json(&query_res).unwrap(); - - let weights = - get_deposit_amount_weights(&deps.as_ref(), &investment_response.info.primitives).unwrap(); - - let first_weight = Decimal::from_ratio( - primitive_deets[0].2 * primitive_states[0].3, - primitive_states[0].2, - ); - let second_weight = Decimal::from_ratio( - primitive_deets[1].2 * primitive_states[1].3, - primitive_states[1].2, - ); - - let total = first_weight + second_weight; - - let expected_first_weight = Decimal::from_ratio( - first_weight.numerator() * total.denominator(), - first_weight.denominator() * total.numerator(), - ); - let expected_second_weight = Decimal::from_ratio( - second_weight.numerator() * total.denominator(), - second_weight.denominator() * total.numerator(), - ); - - assert_eq!(weights.ratio[0].weight, expected_first_weight); - assert_eq!(weights.ratio[1].weight, expected_second_weight); -} - -#[test] -fn test_get_token_amount_weights() { - let primitive_states = vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Uint128::from(200u128), - Uint128::from(400u128), - ), - ]; - let primitive_deets = vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Decimal::from_str("3.5").unwrap(), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Decimal::one(), - ), - ]; - - let mut deps = mock_deps_with_primitives(primitive_states.clone()); - let init_msg = init_msg_with_primitive_details(primitive_deets.clone()); - - let info = mock_info(TEST_CREATOR, &[]); - let env = mock_env(); - let res = init(deps.as_mut(), &init_msg, &env, &info); - assert_eq!(1, res.messages.len()); - - let invest_query = crate::msg::QueryMsg::Investment {}; - let query_res = query(deps.as_ref(), env, invest_query).unwrap(); - - let investment_response: InvestmentResponse = from_json(&query_res).unwrap(); - - let _weights = - get_deposit_amount_weights(&deps.as_ref(), &investment_response.info.primitives).unwrap(); - - let first_weight = Decimal::from_ratio( - primitive_deets[0].2 * primitive_states[0].3, - primitive_states[0].2, - ); - let second_weight = Decimal::from_ratio( - primitive_deets[1].2 * primitive_states[1].3, - primitive_states[1].2, - ); - - let total = first_weight + second_weight; - - let expected_first_weight = Decimal::from_ratio( - first_weight.numerator() * total.denominator(), - first_weight.denominator() * total.numerator(), - ); - let expected_second_weight = Decimal::from_ratio( - second_weight.numerator() * total.denominator(), - second_weight.denominator() * total.numerator(), - ); - - let token_weights = get_token_amount_weights(&[ - CoinWeight { - denom: "ibc/uosmo".to_string(), - weight: expected_first_weight, - }, - CoinWeight { - denom: "ibc/uatom".to_string(), - weight: expected_second_weight, - }, - ]) - .unwrap(); - - assert_eq!(token_weights[0].weight, expected_first_weight); - assert_eq!(token_weights[1].weight, expected_second_weight); -} - -#[test] -fn test_get_token_amount_weights_duplicate_tokens() {} - -#[test] -fn test_get_max_bond() { - let primitive_states = vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Uint128::from(200u128), - Uint128::from(400u128), - ), - ]; - let primitive_deets = vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Decimal::from_str("3.5").unwrap(), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Decimal::one(), - ), - ]; - - let funds = vec![ - Coin { - denom: "ibc/uosmo".to_string(), - amount: Uint128::from(1000u128), - }, - Coin { - denom: "ibc/uatom".to_string(), - amount: Uint128::from(200u128), - }, - ]; - - let mut deps = mock_deps_with_primitives(primitive_states.clone()); - let init_msg = init_msg_with_primitive_details(primitive_deets.clone()); - - let info = mock_info(TEST_CREATOR, &[]); - let env = mock_env(); - let res = init(deps.as_mut(), &init_msg, &env, &info); - assert_eq!(1, res.messages.len()); - - let invest_query = crate::msg::QueryMsg::Investment {}; - let query_res = query(deps.as_ref(), env, invest_query).unwrap(); - - let investment_response: InvestmentResponse = from_json(&query_res).unwrap(); - - let weights = - get_deposit_amount_weights(&deps.as_ref(), &investment_response.info.primitives).unwrap(); - - let first_weight = Decimal::from_ratio( - primitive_deets[0].2 * primitive_states[0].3, - primitive_states[0].2, - ); - let second_weight = Decimal::from_ratio( - primitive_deets[1].2 * primitive_states[1].3, - primitive_states[1].2, - ); - - let total = first_weight + second_weight; - - let expected_first_weight = Decimal::from_ratio( - first_weight.numerator() * total.denominator(), - first_weight.denominator() * total.numerator(), - ); - let expected_second_weight = Decimal::from_ratio( - second_weight.numerator() * total.denominator(), - second_weight.denominator() * total.numerator(), - ); - - assert_eq!(weights.ratio[0].weight, expected_first_weight); - assert_eq!(weights.ratio[1].weight, expected_second_weight); - - let token_weights = get_token_amount_weights(&[ - CoinWeight { - denom: "ibc/uosmo".to_string(), - weight: expected_first_weight, - }, - CoinWeight { - denom: "ibc/uatom".to_string(), - weight: expected_second_weight, - }, - ]) - .unwrap(); - - assert_eq!(token_weights[0].weight, expected_first_weight); - assert_eq!(token_weights[1].weight, expected_second_weight); - - let expected_max_bond = std::cmp::min( - Decimal::from_ratio( - funds[0].amount * token_weights[0].weight.denominator(), - token_weights[0].weight.numerator(), - ) - .to_uint_floor(), - Decimal::from_ratio( - funds[1].amount * token_weights[1].weight.denominator(), - token_weights[1].weight.numerator(), - ) - .to_uint_floor(), - ); - - let max_bond = get_max_bond(&funds, &token_weights).unwrap(); - - assert_eq!(max_bond.to_uint_floor(), expected_max_bond); -} - -#[test] -fn test_get_deposit_and_remainder_for_ratio() { - let primitive_states = vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Uint128::from(200u128), - Uint128::from(400u128), - ), - ]; - let primitive_deets = vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Decimal::from_str("3.5").unwrap(), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Decimal::one(), - ), - ]; - - let funds = vec![ - Coin { - denom: "ibc/uosmo".to_string(), - amount: Uint128::from(1000u128), - }, - Coin { - denom: "ibc/uatom".to_string(), - amount: Uint128::from(200u128), - }, - ]; - - let mut deps = mock_deps_with_primitives(primitive_states.clone()); - let init_msg = init_msg_with_primitive_details(primitive_deets.clone()); - - let info = mock_info(TEST_CREATOR, &[]); - let env = mock_env(); - let res = init(deps.as_mut(), &init_msg, &env, &info); - assert_eq!(1, res.messages.len()); - - let invest_query = crate::msg::QueryMsg::Investment {}; - let query_res = query(deps.as_ref(), env, invest_query).unwrap(); - - let investment_response: InvestmentResponse = from_json(&query_res).unwrap(); - - let _weights = - get_deposit_amount_weights(&deps.as_ref(), &investment_response.info.primitives).unwrap(); - - let first_weight = Decimal::from_ratio( - primitive_deets[0].2 * primitive_states[0].3, - primitive_states[0].2, - ); - let second_weight = Decimal::from_ratio( - primitive_deets[1].2 * primitive_states[1].3, - primitive_states[1].2, - ); - - let total = first_weight + second_weight; - - let expected_first_weight = Decimal::from_ratio( - first_weight.numerator() * total.denominator(), - first_weight.denominator() * total.numerator(), - ); - let expected_second_weight = Decimal::from_ratio( - second_weight.numerator() * total.denominator(), - second_weight.denominator() * total.numerator(), - ); - - let token_weights = get_token_amount_weights(&[ - CoinWeight { - denom: "ibc/uosmo".to_string(), - weight: expected_first_weight, - }, - CoinWeight { - denom: "ibc/uatom".to_string(), - weight: expected_second_weight, - }, - ]) - .unwrap(); - - assert_eq!(token_weights[0].weight, expected_first_weight); - assert_eq!(token_weights[1].weight, expected_second_weight); - - let expected_max_bond = std::cmp::min( - Decimal::from_ratio( - funds[0].amount * token_weights[0].weight.denominator(), - token_weights[0].weight.numerator(), - ) - .to_uint_floor(), - Decimal::from_ratio( - funds[1].amount * token_weights[1].weight.denominator(), - token_weights[1].weight.numerator(), - ) - .to_uint_floor(), - ); - - let max_bond = get_max_bond(&funds, &token_weights).unwrap(); - - assert_eq!(max_bond.to_uint_floor(), expected_max_bond); - - let (deposit, remainder) = get_deposit_and_remainder_for_ratio( - &funds, - max_bond, - &CoinRatio { - ratio: token_weights.clone(), - }, - ) - .unwrap(); - - let expected_first_deposit = Decimal::from_ratio( - token_weights[0].weight.numerator() * max_bond, - token_weights[0].weight.denominator(), - ); - let expected_second_deposit = Decimal::from_ratio( - token_weights[1].weight.numerator() * max_bond, - token_weights[1].weight.denominator(), - ); - - assert_eq!(deposit[0].amount, expected_first_deposit.to_uint_floor()); - assert_eq!(deposit[1].amount, expected_second_deposit.to_uint_floor()); - - assert_eq!( - remainder[0].amount, - funds[0].amount - expected_first_deposit.to_uint_floor() - ); - assert_eq!( - remainder[1].amount, - funds[1].amount - expected_second_deposit.to_uint_floor() - ); -} - -#[test] -fn test_get_deposit_and_remainder_for_ratio_three_primitives() { - let primitive_states = vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Uint128::from(200u128), - Uint128::from(400u128), - ), - ( - "quasar125".to_string(), - "ibc/ustars".to_string(), - Uint128::from(600u128), - Uint128::from(450u128), - ), - ]; - let primitive_deets = vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Decimal::from_str("3.5").unwrap(), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Decimal::one(), - ), - ( - "quasar125".to_string(), - "ibc/ustars".to_string(), - Decimal::from_str("0.5").unwrap(), - ), - ]; - - let funds = vec![ - Coin { - denom: "ibc/uosmo".to_string(), - amount: Uint128::from(1000u128), - }, - Coin { - denom: "ibc/uatom".to_string(), - amount: Uint128::from(200u128), - }, - Coin { - denom: "ibc/ustars".to_string(), - amount: Uint128::from(200u128), - }, - ]; - - let mut deps = mock_deps_with_primitives(primitive_states.clone()); - let init_msg = init_msg_with_primitive_details(primitive_deets.clone()); - - let info = mock_info(TEST_CREATOR, &[]); - let env = mock_env(); - let res = init(deps.as_mut(), &init_msg, &env, &info); - assert_eq!(1, res.messages.len()); - - let invest_query = crate::msg::QueryMsg::Investment {}; - let query_res = query(deps.as_ref(), env, invest_query).unwrap(); - - let investment_response: InvestmentResponse = from_json(&query_res).unwrap(); - - let weights = - get_deposit_amount_weights(&deps.as_ref(), &investment_response.info.primitives).unwrap(); - - let first_weight = Decimal::from_ratio( - primitive_deets[0].2 * primitive_states[0].3, - primitive_states[0].2, - ); - let second_weight = Decimal::from_ratio( - primitive_deets[1].2 * primitive_states[1].3, - primitive_states[1].2, - ); - let third_weight = Decimal::from_ratio( - primitive_deets[2].2 * primitive_states[2].3, - primitive_states[2].2, - ); - - let total = first_weight + second_weight + third_weight; - - let expected_first_weight = Decimal::from_ratio( - first_weight.numerator() * total.denominator(), - first_weight.denominator() * total.numerator(), - ); - let expected_second_weight = Decimal::from_ratio( - second_weight.numerator() * total.denominator(), - second_weight.denominator() * total.numerator(), - ); - let expected_third_weight = Decimal::from_ratio( - third_weight.numerator() * total.denominator(), - third_weight.denominator() * total.numerator(), - ); - - assert_eq!(weights.ratio[0].weight, expected_first_weight); - assert_eq!(weights.ratio[1].weight, expected_second_weight); - assert_eq!(weights.ratio[2].weight, expected_third_weight); - - let token_weights = get_token_amount_weights(&[ - CoinWeight { - denom: "ibc/uosmo".to_string(), - weight: expected_first_weight, - }, - CoinWeight { - denom: "ibc/uatom".to_string(), - weight: expected_second_weight, - }, - CoinWeight { - denom: "ibc/ustars".to_string(), - weight: expected_third_weight, - }, - ]) - .unwrap(); - - assert_eq!(token_weights[0].weight, expected_first_weight); - assert_eq!(token_weights[1].weight, expected_second_weight); - assert_eq!(token_weights[2].weight, expected_third_weight); - - let expected_max_bond = std::cmp::min( - Decimal::from_ratio( - funds[0].amount * token_weights[0].weight.denominator(), - token_weights[0].weight.numerator(), - ) - .to_uint_floor(), - Decimal::from_ratio( - funds[1].amount * token_weights[1].weight.denominator(), - token_weights[1].weight.numerator(), - ) - .to_uint_floor(), - ); - - let max_bond = get_max_bond(&funds, &token_weights).unwrap(); - - assert_eq!(max_bond.to_uint_floor(), expected_max_bond); - - let (deposit, remainder) = get_deposit_and_remainder_for_ratio( - &funds, - max_bond, - &CoinRatio { - ratio: token_weights.clone(), - }, - ) - .unwrap(); - - let expected_first_deposit = Decimal::from_ratio( - token_weights[0].weight.numerator() * max_bond, - token_weights[0].weight.denominator(), - ); - let expected_second_deposit = Decimal::from_ratio( - token_weights[1].weight.numerator() * max_bond, - token_weights[1].weight.denominator(), - ); - let expected_third_deposit = Decimal::from_ratio( - token_weights[2].weight.numerator() * max_bond, - token_weights[2].weight.denominator(), - ); - - assert_eq!(deposit[0].amount, expected_first_deposit.to_uint_floor()); - assert_eq!(deposit[1].amount, expected_second_deposit.to_uint_floor()); - assert_eq!(deposit[2].amount, expected_third_deposit.to_uint_floor()); - - println!("remainder: {remainder:?}"); - println!("deposit: {deposit:?}"); - assert_eq!( - remainder[0].amount, - funds[0].amount - expected_first_deposit.to_uint_floor() - ); - assert_eq!( - remainder[1].amount, - funds[1].amount - expected_second_deposit.to_uint_floor() - ); - assert_eq!( - remainder[2].amount, - funds[2].amount - expected_third_deposit.to_uint_floor() - ); -} - -#[test] -fn test_may_pay_with_one_primitive() { - let mut deps = mock_deps_with_primitives(vec![( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - )]); - let init_msg = init_msg_with_primitive_details(vec![( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Decimal::from_str("2.0").unwrap(), - )]); - - let info = mock_info(TEST_CREATOR, &[]); - let env = mock_env(); - let res = init(deps.as_mut(), &init_msg, &env, &info); - assert_eq!(1, res.messages.len()); - - let invest_query = crate::msg::QueryMsg::Investment {}; - let query_res = query(deps.as_ref(), env, invest_query).unwrap(); - - let investment_response: InvestmentResponse = from_json(&query_res).unwrap(); - - let funds = &[Coin { - denom: "ibc/uosmo".to_string(), - amount: Uint128::from(200u128), - }]; - - // load cached balance of primitive contracts - let deposit_amount_ratio = - get_deposit_amount_weights(&deps.as_ref(), &investment_response.info.primitives).unwrap(); - - let token_weights: Vec = - get_token_amount_weights(&deposit_amount_ratio.ratio).unwrap(); - - let max_bond = get_max_bond(funds, &token_weights).unwrap(); - - let (coins, remainder) = - get_deposit_and_remainder_for_ratio(funds, max_bond, &deposit_amount_ratio).unwrap(); - - assert_eq!(coins.len(), 1); - assert_eq!(coins[0].amount, Uint128::from(200u128)); - - assert_eq!(remainder.len(), 1); - assert_eq!(remainder[0].amount, Uint128::from(0u128)); -} - -#[test] -fn test_may_pay_with_even_ratio() { - let mut deps = mock_deps_with_primitives(even_primitives()); - let init_msg = init_msg_with_primitive_details(even_primitive_details()); - let info = mock_info(TEST_CREATOR, &[]); - let env = mock_env(); - let res = init(deps.as_mut(), &init_msg, &env, &info); - assert_eq!(1, res.messages.len()); - - let invest_query = crate::msg::QueryMsg::Investment {}; - let query_res = query(deps.as_ref(), env, invest_query).unwrap(); - - let investment_response: InvestmentResponse = from_json(&query_res).unwrap(); - - let (coins, remainder) = - may_pay_with_ratio(&deps.as_ref(), &even_deposit(), investment_response.info).unwrap(); - - assert_eq!(coins.len(), 3); - assert_eq!(coins[0].amount, Uint128::from(99u128)); // 99 because 0.33333 results in coins getting floored - assert_eq!(coins[1].amount, Uint128::from(99u128)); - assert_eq!(coins[2].amount, Uint128::from(99u128)); - - assert_eq!(remainder.len(), 3); - assert_eq!(remainder[0].amount, Uint128::from(1u128)); - assert_eq!(remainder[1].amount, Uint128::from(1u128)); - assert_eq!(remainder[2].amount, Uint128::from(1u128)); -} - -#[test] -fn test_may_pay_with_uneven_ratio() { - let mut deps = mock_deps_with_primitives(vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(100u128), - Uint128::from(100u128), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Uint128::from(150u128), - Uint128::from(200u128), - ), - ( - "quasar125".to_string(), - "ibc/ustars".to_string(), - Uint128::from(250u128), - Uint128::from(300u128), - ), - ]); - let init_msg = init_msg_with_primitive_details(vec![ - ( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Decimal::from_str("2.0").unwrap(), - ), - ( - "quasar124".to_string(), - "ibc/uatom".to_string(), - Decimal::one(), - ), - ( - "quasar125".to_string(), - "ibc/ustars".to_string(), - Decimal::one(), - ), - ]); - - let info = mock_info(TEST_CREATOR, &[]); - let env = mock_env(); - let res = init(deps.as_mut(), &init_msg, &env, &info); - assert_eq!(1, res.messages.len()); - - let invest_query = crate::msg::QueryMsg::Investment {}; - let query_res = query(deps.as_ref(), env, invest_query).unwrap(); - - let investment_response: InvestmentResponse = from_json(&query_res).unwrap(); - - let funds = &[ - Coin { - denom: "ibc/uosmo".to_string(), - amount: Uint128::from(200u128), - }, - Coin { - denom: "ibc/uatom".to_string(), - amount: Uint128::from(150u128), - }, - Coin { - denom: "ibc/ustars".to_string(), - amount: Uint128::from(140u128), - }, - ]; - - // load cached balance of primitive contracts - let deposit_amount_ratio = - get_deposit_amount_weights(&deps.as_ref(), &investment_response.info.primitives).unwrap(); - - let token_weights: Vec = - get_token_amount_weights(&deposit_amount_ratio.ratio).unwrap(); - - let max_bond = get_max_bond(funds, &token_weights).unwrap(); - - let (coins, remainder) = - get_deposit_and_remainder_for_ratio(funds, max_bond, &deposit_amount_ratio).unwrap(); - - assert_eq!(coins.len(), 3); - assert_eq!(coins[0].amount, Uint128::from(199u128)); // these numbers have been verified - assert_eq!(coins[1].amount, Uint128::from(133u128)); - assert_eq!(coins[2].amount, Uint128::from(119u128)); - - assert_eq!(remainder.len(), 3); - assert_eq!(remainder[0].amount, Uint128::from(1u128)); - assert_eq!(remainder[1].amount, Uint128::from(17u128)); - assert_eq!(remainder[2].amount, Uint128::from(21u128)); -} - -#[test] -fn proper_bond() { - let mut deps = mock_deps_with_primitives(even_primitives_single_token()); - let init_msg = init_msg_with_primitive_details(even_primitive_details_single_token()); - let info = mock_info(TEST_CREATOR, &[]); - let env = mock_env(); - let res = init(deps.as_mut(), &init_msg, &env, &info); - assert_eq!(1, res.messages.len()); - - let deposit_info = mock_info(TEST_DEPOSITOR, &even_deposit_single_token()); - let deposit_msg = ExecuteMsg::Bond { - recipient: Option::None, - }; - let res = execute(deps.as_mut(), env, deposit_info, deposit_msg).unwrap(); - assert_eq!(res.messages.len(), 3); - assert_eq!(res.attributes.first().unwrap().value, "1"); - - if let CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr, - msg, - funds, - }) = &res.messages[0].msg - { - assert_eq!(contract_addr, "quasar123"); - assert_eq!(funds.len(), 1); - assert_eq!(funds[0].denom, "ibc/uosmo"); - assert_eq!(funds[0].amount, Uint128::from(99u128)); - if let lp_strategy::msg::ExecuteMsg::Bond { id } = from_json(msg).unwrap() { - assert_eq!(id, "1") - } else { - panic!("expected Bond msg") - } - } else { - panic!("expected wasm msg") - } - - if let CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr, - msg, - funds, - }) = &res.messages[1].msg - { - assert_eq!(contract_addr, "quasar124"); - assert_eq!(funds.len(), 1); - assert_eq!(funds[0].denom, "ibc/uosmo"); - assert_eq!(funds[0].amount, Uint128::from(99u128)); - if let lp_strategy::msg::ExecuteMsg::Bond { id } = from_json(msg).unwrap() { - assert_eq!(id, "1") - } else { - panic!("expected Bond msg") - } - } else { - panic!("expected Wasm msg") - } - - if let CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr, - msg, - funds, - }) = &res.messages[2].msg - { - assert_eq!(contract_addr, "quasar125"); - assert_eq!(funds.len(), 1); - assert_eq!(funds[0].denom, "ibc/uosmo"); - assert_eq!(funds[0].amount, Uint128::from(99u128)); - if let lp_strategy::msg::ExecuteMsg::Bond { id } = from_json(msg).unwrap() { - assert_eq!(id, "1") - } else { - panic!("expected Bond msg") - } - } else { - panic!("expected wasm msg") - } - - // This BankMsg should not exist since we deprecated the dust send back - // if let CosmosMsg::Bank(BankMsg::Send { to_address, amount }) = &res.messages[3].msg { - // assert_eq!(to_address, TEST_DEPOSITOR); - // assert_eq!(amount.len(), 3); - // assert_eq!(amount[0].amount, Uint128::from(1u128)); - // assert_eq!(amount[1].amount, Uint128::from(1u128)); - // assert_eq!(amount[2].amount, Uint128::from(1u128)); - // } else { - // panic!("unexpected message"); - // } -} - -#[test] -fn proper_bond_with_zero_primitive_balance() { - let mut deps = mock_deps_with_primitives(vec![( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(0u128), - Uint128::from(0u128), - )]); - let init_msg = init_msg_with_primitive_details(vec![( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Decimal::one(), - )]); - let info = mock_info(TEST_CREATOR, &[]); - let env = mock_env(); - let res = init(deps.as_mut(), &init_msg, &env, &info); - assert_eq!(1, res.messages.len()); - - let deposit_info = mock_info(TEST_DEPOSITOR, &even_deposit_single_token()); - let deposit_msg = ExecuteMsg::Bond { - recipient: Option::None, - }; - let res = execute(deps.as_mut(), env, deposit_info, deposit_msg).unwrap(); - assert_eq!(res.messages.len(), 1); - assert_eq!(res.attributes.first().unwrap().value, "1"); -} - -#[test] -fn test_bond_with_zero_primitive_state() { - let mut deps_1 = mock_deps_with_primitives(vec![( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(0u128), - Uint128::from(1u128), - )]); - let mut deps_2 = mock_deps_with_primitives(vec![( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Uint128::from(1u128), - Uint128::from(0u128), - )]); - let init_msg: InstantiateMsg = init_msg_with_primitive_details(vec![( - "quasar123".to_string(), - "ibc/uosmo".to_string(), - Decimal::one(), - )]); - let info = mock_info(TEST_CREATOR, &[]); - let env = mock_env(); - let _ = init(deps_1.as_mut(), &init_msg, &env, &info); - let _ = init(deps_2.as_mut(), &init_msg, &env, &info); - - // mock the rewards_contract address - VAULT_REWARDS - .save( - deps_1.as_mut().storage, - &Addr::unchecked("rewards_contract"), - ) - .unwrap(); - VAULT_REWARDS - .save( - deps_2.as_mut().storage, - &Addr::unchecked("rewards_contract"), - ) - .unwrap(); - - let deposit_info = mock_info(TEST_DEPOSITOR, &even_deposit_single_token()); - let deposit_msg = ExecuteMsg::Bond { - recipient: Option::None, - }; - let bond_seq_1 = BONDING_SEQ.load(deps_1.as_ref().storage).unwrap(); - let _res_1 = execute( - deps_1.as_mut(), - env.clone(), - deposit_info.clone(), - deposit_msg.clone(), - ) - .unwrap(); - - // with changing to minting by ica_balance, if either is zero, we expect to mint user value, so we should be able to callback these primitives - // and receive even deposit single token amount of shares, aka 300 - let res_1 = on_bond( - deps_1.as_mut(), - env.clone(), - mock_info("quasar123", &[]), - Uint128::new(250), - bond_seq_1.to_string(), - ) - .unwrap(); - assert!(res_1.attributes.contains(&Attribute::new("minted", "300"))); - - let bond_seq_2 = BONDING_SEQ.load(deps_2.as_ref().storage).unwrap(); - let _res_2 = execute(deps_2.as_mut(), env.clone(), deposit_info, deposit_msg).unwrap(); - - let res_2 = on_bond( - deps_2.as_mut(), - env, - mock_info("quasar123", &[]), - Uint128::new(250), - bond_seq_2.to_string(), - ) - .unwrap(); - assert!(res_2.attributes.contains(&Attribute::new("minted", "300"))); - // assert_eq!(res_2.to_string(), "Generic error: Unexpected primitive state, either both supply and balance should be zero, or neither."); -} - -#[test] -fn proper_bond_response_callback_single_token() { - let mut deps = mock_deps_with_primitives(even_primitives_single_token()); - let init_msg = init_msg_with_primitive_details(even_primitive_details_single_token()); - let info = mock_info(TEST_CREATOR, &[]); - let env = mock_env(); - let res = init(deps.as_mut(), &init_msg, &env, &info); - assert_eq!(1, res.messages.len()); - - let reply_msg = reply_msg(); - let res = reply(deps.as_mut(), env.clone(), reply_msg).unwrap(); - assert_eq!(res.messages.len(), 0); - - let deposit_info = mock_info(TEST_DEPOSITOR, &even_deposit_single_token()); - let deposit_msg = ExecuteMsg::Bond { - recipient: Option::None, - }; - let res = execute(deps.as_mut(), env.clone(), deposit_info, deposit_msg).unwrap(); - - println!("messages: {:#?}", res.messages); - // assert_eq!(res.messages.len(), 4); - // assert_eq!(res.attributes.first().unwrap().value, "1"); - - // in this scenario we expect 1000/1000 * 100 = 100 shares back from each primitive - let primitive_1_info = mock_info("quasar123", &[]); - let primitive_1_msg = ExecuteMsg::BondResponse(BondResponse { - share_amount: 100u128.into(), - bond_id: "1".to_string(), - }); - let p1_res = execute( - deps.as_mut(), - env.clone(), - primitive_1_info, - primitive_1_msg, - ) - .unwrap(); - assert_eq!(p1_res.messages.len(), 0); - - let primitive_2_info = mock_info("quasar124", &[]); - let primitive_2_msg = ExecuteMsg::BondResponse(BondResponse { - share_amount: 100u128.into(), - bond_id: "1".to_string(), - }); - let p2_res = execute( - deps.as_mut(), - env.clone(), - primitive_2_info, - primitive_2_msg, - ) - .unwrap(); - assert_eq!(p2_res.messages.len(), 0); - - let primitive_3_info = mock_info("quasar125", &[]); - let primitive_3_msg = ExecuteMsg::BondResponse(BondResponse { - share_amount: 100u128.into(), - bond_id: "1".to_string(), - }); - let p3_res = execute( - deps.as_mut(), - env.clone(), - primitive_3_info, - primitive_3_msg, - ) - .unwrap(); - assert_eq!(p3_res.messages.len(), 1); - - let balance_query = crate::msg::QueryMsg::Balance { - address: TEST_DEPOSITOR.to_string(), - }; - let balance_res = query(deps.as_ref(), env, balance_query).unwrap(); - let balance: BalanceResponse = from_json(&balance_res).unwrap(); - - assert_eq!(balance.balance, Uint128::from(297u128)); -} - -// this looks to be a duplicate now of proper_bond_response_callback_single_token, so should no longer be supported -// #[test] -// fn proper_bond_response_callback() { -// let mut deps = mock_deps_with_primitives(even_primitives()); -// let init_msg = init_msg_with_primitive_details(even_primitive_details()); -// let info = mock_info(TEST_CREATOR, &[]); -// let env = mock_env(); -// let res = init(deps.as_mut(), &init_msg, &env, &info); -// assert_eq!(1, res.messages.len()); - -// let reply_msg = reply_msg(); -// let res = reply(deps.as_mut(), env.clone(), reply_msg).unwrap(); -// assert_eq!(res.messages.len(), 0); - -// let deposit_info = mock_info(TEST_DEPOSITOR, &even_deposit()); -// let deposit_msg = ExecuteMsg::Bond { -// recipient: Option::None, -// }; -// let res = execute(deps.as_mut(), env.clone(), deposit_info, deposit_msg).unwrap(); - -// println!("messages: {:#?}", res.messages); -// // assert_eq!(res.messages.len(), 4); -// // assert_eq!(res.attributes.first().unwrap().value, "1"); - -// // in this scenario we expect 1000/1000 * 100 = 100 shares back from each primitive -// let primitive_1_info = mock_info("quasar123", &[]); -// let primitive_1_msg = ExecuteMsg::BondResponse(BondResponse { -// share_amount: 100u128.into(), -// bond_id: "1".to_string(), -// }); -// let p1_res = execute( -// deps.as_mut(), -// env.clone(), -// primitive_1_info, -// primitive_1_msg, -// ) -// .unwrap(); -// assert_eq!(p1_res.messages.len(), 0); - -// let primitive_2_info = mock_info("quasar124", &[]); -// let primitive_2_msg = ExecuteMsg::BondResponse(BondResponse { -// share_amount: 100u128.into(), -// bond_id: "1".to_string(), -// }); -// let p2_res = execute( -// deps.as_mut(), -// env.clone(), -// primitive_2_info, -// primitive_2_msg, -// ) -// .unwrap(); -// assert_eq!(p2_res.messages.len(), 0); - -// let primitive_3_info = mock_info("quasar125", &[]); -// let primitive_3_msg = ExecuteMsg::BondResponse(BondResponse { -// share_amount: 100u128.into(), -// bond_id: "1".to_string(), -// }); -// let p3_res = execute( -// deps.as_mut(), -// env.clone(), -// primitive_3_info, -// primitive_3_msg, -// ) -// .unwrap(); -// assert_eq!(p3_res.messages.len(), 1); - -// let balance_query = crate::msg::QueryMsg::Balance { -// address: TEST_DEPOSITOR.to_string(), -// }; -// let balance_res = query(deps.as_ref(), env, balance_query).unwrap(); -// let balance: BalanceResponse = from_json(&balance_res).unwrap(); - -// assert_eq!(balance.balance, Uint128::from(300u128)); -// } - -#[test] -fn proper_unbond() { - let mut deps = mock_deps_with_primitives(even_primitives()); - let init_msg = init_msg_with_primitive_details(even_primitive_details()); - let info = mock_info(TEST_CREATOR, &[]); - let mut env = mock_env(); - let res = init(deps.as_mut(), &init_msg, &env, &info); - assert_eq!(1, res.messages.len()); - - let reply_msg = reply_msg(); - let _res = reply(deps.as_mut(), env.clone(), reply_msg).unwrap(); - - let deposit_info = mock_info(TEST_DEPOSITOR, &even_deposit_single_token()); - let deposit_msg = ExecuteMsg::Bond { - recipient: Option::None, - }; - let res = execute(deps.as_mut(), env.clone(), deposit_info, deposit_msg).unwrap(); - assert_eq!(res.messages.len(), 3); - assert_eq!(res.attributes.first().unwrap().value, "1"); - - // in this scenario we expect 1000/1000 * 100 = 100 shares back from each primitive - let primitive_1_info = mock_info("quasar123", &[]); - let primitive_1_msg = ExecuteMsg::BondResponse(BondResponse { - share_amount: 100u128.into(), - bond_id: "1".to_string(), - }); - let p1_res = execute( - deps.as_mut(), - env.clone(), - primitive_1_info.clone(), - primitive_1_msg, - ) - .unwrap(); - assert_eq!(p1_res.messages.len(), 0); - - let primitive_2_info = mock_info("quasar124", &[]); - let primitive_2_msg = ExecuteMsg::BondResponse(BondResponse { - share_amount: 100u128.into(), - bond_id: "1".to_string(), - }); - let p2_res = execute( - deps.as_mut(), - env.clone(), - primitive_2_info.clone(), - primitive_2_msg, - ) - .unwrap(); - assert_eq!(p2_res.messages.len(), 0); - - let primitive_3_info = mock_info("quasar125", &[]); - let primitive_3_msg = ExecuteMsg::BondResponse(BondResponse { - share_amount: 100u128.into(), - bond_id: "1".to_string(), - }); - let p3_res = execute( - deps.as_mut(), - env.clone(), - primitive_3_info.clone(), - primitive_3_msg, - ) - .unwrap(); - println!("p3_res: {p3_res:?}"); - assert_eq!(p3_res.messages.len(), 1); - if let CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr, - msg, - funds: _, - }) = &p3_res.messages[0].msg - { - assert_eq!(contract_addr, "vault_rewards_addr"); - if let vault_rewards::msg::ExecuteMsg::Vault( - vault_rewards::msg::VaultExecuteMsg::UpdateUserRewardIndex(user_reward_index), - ) = from_json(msg).unwrap() - { - assert_eq!(user_reward_index, TEST_DEPOSITOR); - } else { - panic!("wrong message"); - } - } else { - panic!("wrong message"); - } - - let balance_query = crate::msg::QueryMsg::Balance { - address: TEST_DEPOSITOR.to_string(), - }; - let balance_res = query(deps.as_ref(), env.clone(), balance_query).unwrap(); - let balance: BalanceResponse = from_json(&balance_res).unwrap(); - - assert_eq!(balance.balance, Uint128::from(297u128)); - - // start unbond - let unbond_info = mock_info(TEST_DEPOSITOR, &[]); - let unbond_msg = ExecuteMsg::Unbond { - amount: Option::Some(balance.balance), - }; - let unbond_res = execute(deps.as_mut(), env.clone(), unbond_info, unbond_msg).unwrap(); - assert_eq!(unbond_res.messages.len(), 4); - assert_eq!(unbond_res.attributes[2].key, "burnt"); - assert_eq!(unbond_res.attributes[2].value, "297"); - assert_eq!(unbond_res.attributes[3].key, "bond_id"); - assert_eq!(unbond_res.attributes[3].value, "2"); - - // todo replace with a macro - if let CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr, - msg, - funds, - }) = &unbond_res.messages[0].msg - { - assert_eq!(contract_addr, "quasar123"); - assert!(funds.is_empty()); - if let lp_strategy::msg::ExecuteMsg::StartUnbond { id, share_amount } = - from_json(msg).unwrap() - { - assert_eq!(id, "2"); - assert_eq!(share_amount, Uint128::from(100u128)); - } else { - panic!("expected start unbond") - } - } else { - panic!("expected wasm msg") - } - if let CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr, - msg, - funds, - }) = &unbond_res.messages[1].msg - { - assert_eq!(contract_addr, "quasar124"); - assert!(funds.is_empty()); - if let lp_strategy::msg::ExecuteMsg::StartUnbond { id, share_amount } = - from_json(msg).unwrap() - { - assert_eq!(id, "2"); - assert_eq!(share_amount, Uint128::from(100u128)); - } else { - panic!("expected start unbond") - } - } else { - panic!("expected wasm msg") - } - if let CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr, - msg, - funds, - }) = &unbond_res.messages[2].msg - { - assert_eq!(contract_addr, "quasar125"); - assert!(funds.is_empty()); - if let lp_strategy::msg::ExecuteMsg::StartUnbond { id, share_amount } = - from_json(msg).unwrap() - { - assert_eq!(id, "2"); - assert_eq!(share_amount, Uint128::from(100u128)); - } else { - panic!("expected start unbond") - } - } else { - panic!("expected wasm msg") - } - if let CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr, - msg, - funds, - }) = &unbond_res.messages[3].msg - { - assert_eq!(contract_addr, "vault_rewards_addr"); - assert!(funds.is_empty()); - if let vault_rewards::msg::ExecuteMsg::Vault( - vault_rewards::msg::VaultExecuteMsg::UpdateUserRewardIndex(user_addr), - ) = from_json(msg).unwrap() - { - assert_eq!(user_addr, TEST_DEPOSITOR); - } else { - panic!("expected update user reward index") - } - } else { - panic!("expected wasm message") - } - - // get callbacks back - let start_unbond_msg_p1 = ExecuteMsg::StartUnbondResponse(StartUnbondResponse { - unbond_id: "2".to_string(), - unlock_time: Timestamp::from_seconds(env.block.time.seconds() + 5), - }); - let start_unbond_res = execute( - deps.as_mut(), - env.clone(), - primitive_1_info, - start_unbond_msg_p1, - ) - .unwrap(); - assert_eq!(start_unbond_res.messages.len(), 0); - - let start_unbond_msg_p2 = ExecuteMsg::StartUnbondResponse(StartUnbondResponse { - unbond_id: "2".to_string(), - unlock_time: Timestamp::from_seconds(env.block.time.seconds() + 5), - }); - let start_unbond_res = execute( - deps.as_mut(), - env.clone(), - primitive_2_info, - start_unbond_msg_p2, - ) - .unwrap(); - assert_eq!(start_unbond_res.messages.len(), 0); - - let start_unbond_msg_p3 = ExecuteMsg::StartUnbondResponse(StartUnbondResponse { - unbond_id: "2".to_string(), - unlock_time: Timestamp::from_seconds(env.block.time.seconds() + 60), - }); - let start_unbond_res = execute( - deps.as_mut(), - env.clone(), - primitive_3_info, - start_unbond_msg_p3, - ) - .unwrap(); - assert_eq!(start_unbond_res.messages.len(), 0); - - // do unbond - let do_unbond_info = mock_info(TEST_DEPOSITOR, &[]); - let do_unbond_msg = ExecuteMsg::Unbond { amount: None }; - let do_unbond_res = execute( - deps.as_mut(), - env.clone(), - do_unbond_info.clone(), - do_unbond_msg.clone(), - ) - .unwrap(); - - assert_eq!(do_unbond_res.messages.len(), 0); - - env.block.height += 4; - env.block.time = env.block.time.plus_seconds(30); - - // set two of the primitives to be unbondable - deps.querier.set_unbonding_claim_for_primitive( - "quasar123".to_owned(), - env.block.time.minus_seconds(5), - false, - ); - deps.querier.set_unbonding_claim_for_primitive( - "quasar124".to_owned(), - env.block.time.minus_seconds(5), - false, - ); - - // unbond and see that 2 are unbondable - let do_unbond_res = execute( - deps.as_mut(), - env.clone(), - do_unbond_info.clone(), - do_unbond_msg, - ) - .unwrap(); - // nothing here - assert_eq!(do_unbond_res.messages.len(), 0); - - let claim_msg = ExecuteMsg::Claim {}; - let claim_res = execute( - deps.as_mut(), - env.clone(), - do_unbond_info.clone(), - claim_msg, - ) - .unwrap(); - - assert_eq!(claim_res.messages.len(), 2); - assert_eq!(claim_res.attributes[2].key, "num_unbondable_ids"); - assert_eq!(claim_res.attributes[2].value, "2"); - - if let CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr, - msg, - funds, - }) = &claim_res.messages[0].msg - { - assert_eq!(contract_addr, "quasar123"); - assert!(funds.is_empty()); - if let lp_strategy::msg::ExecuteMsg::Unbond { id } = from_json(msg).unwrap() { - assert_eq!(id, "2"); - } else { - panic!("expected unbond") - } - } else { - panic!("expected wasm msg") - } - if let CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr, - msg, - funds, - }) = &claim_res.messages[1].msg - { - assert_eq!(contract_addr, "quasar124"); - assert!(funds.is_empty()); - if let lp_strategy::msg::ExecuteMsg::Unbond { id } = from_json(msg).unwrap() { - assert_eq!(id, "2"); - } else { - panic!("expected unbond") - } - } else { - panic!("expected wasm msg") - } - // set these two primitive unbonds to have been attempted already - deps.querier.set_unbonding_claim_for_primitive( - "quasar123".to_owned(), - env.block.time.minus_seconds(5), - true, - ); - deps.querier.set_unbonding_claim_for_primitive( - "quasar124".to_owned(), - env.block.time.minus_seconds(5), - true, - ); - - env.block.height += 5; - env.block.time = env.block.time.plus_seconds(40); - - // set last of the primitives to be unbondable - deps.querier.set_unbonding_claim_for_primitive( - "quasar125".to_owned(), - env.block.time.minus_seconds(5), - false, - ); - - // test that claim works the same way as unbond(amount:0) - let claim_msg = ExecuteMsg::Claim {}; - let claim_res = execute(deps.as_mut(), env.clone(), do_unbond_info, claim_msg).unwrap(); - - // todo: This assertion will change because we should ideally only expect one here, pending arch discussion - assert_eq!(claim_res.messages.len(), 1); - assert_eq!(claim_res.attributes[2].key, "num_unbondable_ids"); - assert_eq!(claim_res.attributes[2].value, "1"); - - if let CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr, - msg, - funds, - }) = &claim_res.messages[0].msg - { - // todo: This assertion will change because we should ideally only expect one here, pending arch discussions - assert_eq!(contract_addr, "quasar125"); - assert!(funds.is_empty()); - if let lp_strategy::msg::ExecuteMsg::Unbond { id } = from_json(msg).unwrap() { - assert_eq!(id, "2"); - } else { - panic!("expected unbond") - } - } else { - panic!("ex[ected wasm msg") - } - - // start callbacks from primitives for unbond - let unbond_callback_msg = ExecuteMsg::UnbondResponse(UnbondResponse { - unbond_id: "2".to_string(), - }); - let p1_unbond_callback_info = mock_info( - "quasar123", - &[Coin { - denom: "ibc/uosmo".to_string(), - amount: Uint128::from(100u128), - }], - ); - let p1_unbond_callback_res = execute( - deps.as_mut(), - env.clone(), - p1_unbond_callback_info, - unbond_callback_msg.clone(), - ) - .unwrap(); - assert_eq!(p1_unbond_callback_res.messages.len(), 0); - - let p2_unbond_callback_info = mock_info( - "quasar124", - &[Coin { - denom: "ibc/uatom".to_string(), - amount: Uint128::from(100u128), - }], - ); - let p2_unbond_callback_res = execute( - deps.as_mut(), - env.clone(), - p2_unbond_callback_info, - unbond_callback_msg.clone(), - ) - .unwrap(); - assert_eq!(p2_unbond_callback_res.messages.len(), 0); - - let p3_unbond_callback_info = mock_info( - "quasar125", - &[Coin { - denom: "ibc/ustars".to_string(), - amount: Uint128::from(100u128), - }], - ); - let p3_unbond_callback_res = execute( - deps.as_mut(), - env, - p3_unbond_callback_info, - unbond_callback_msg, - ) - .unwrap(); - assert_eq!(p3_unbond_callback_res.messages.len(), 3); - - if let CosmosMsg::Bank(bank_msg) = &p3_unbond_callback_res.messages[0].msg { - if let BankMsg::Send { to_address, amount } = bank_msg { - assert_eq!(to_address, TEST_DEPOSITOR); - assert_eq!(amount.len(), 1); - assert_eq!(amount[0].denom, "ibc/uosmo"); - assert_eq!(amount[0].amount, Uint128::from(100u128)); - } else { - panic!("unexpected bank message"); - } - } else { - panic!("unexpected message"); - } - - if let CosmosMsg::Bank(bank_msg) = &p3_unbond_callback_res.messages[1].msg { - if let BankMsg::Send { to_address, amount } = bank_msg { - assert_eq!(to_address, TEST_DEPOSITOR); - assert_eq!(amount.len(), 1); - assert_eq!(amount[0].denom, "ibc/uatom"); - assert_eq!(amount[0].amount, Uint128::from(100u128)); - } else { - panic!("unexpected bank message"); - } - } else { - panic!("unexpected message"); - } - - if let CosmosMsg::Bank(bank_msg) = &p3_unbond_callback_res.messages[2].msg { - if let BankMsg::Send { to_address, amount } = bank_msg { - assert_eq!(to_address, TEST_DEPOSITOR); - assert_eq!(amount.len(), 1); - assert_eq!(amount[0].denom, "ibc/ustars"); - assert_eq!(amount[0].amount, Uint128::from(100u128)); - } else { - panic!("unexpected bank message"); - } - } else { - panic!("unexpected message"); - } -} - -#[test] -fn test_multi_user_bond_unbond() {} - -#[test] -fn test_recipient_not_sender() {} - -#[test] -fn test_dup_token_deposits() { - let env = mock_env(); - const ADDRESS: &str = "quasar"; - const DENOM_LOCAL_CHAIN: &str = "ibc/uosmo"; - const SHARES: Uint128 = Uint128::new(100); - const BALANCE: Uint128 = Uint128::new(100); - - let deposit_amounts = vec![ - Uint128::new(10), - Uint128::new(1_000), - Uint128::new(1_000_000_000), - ]; - - for deposit_amount in deposit_amounts { - // test params - let weights = [ - Decimal::from_str("0.2").unwrap(), - Decimal::from_str("0.3").unwrap(), - Decimal::from_str("0.5").unwrap(), - ]; - - let mut primitive_states: Vec<(String, String, Uint128, Uint128)> = Vec::new(); - let mut primitive_details: Vec<(String, String, Decimal)> = Vec::new(); - for i in 1..=3 { - primitive_states.push(( - format!("{ADDRESS}{i}"), - DENOM_LOCAL_CHAIN.to_string(), - SHARES, - BALANCE, - )); - primitive_details.push(( - format!("{ADDRESS}{i}"), - DENOM_LOCAL_CHAIN.to_string(), - weights[i - 1], - )); - } - let mut deps = mock_deps_with_primitives(primitive_states); - let init_info = mock_info(TEST_CREATOR, &[]); - - let init_msg = init_msg_with_primitive_details(primitive_details); - let init_res = init(deps.as_mut(), &init_msg, &env, &init_info); - assert_eq!(1, init_res.messages.len()); - - // deposit 3 times to the same vault - let deposit_info = mock_info( - TEST_CREATOR, - &[Coin { - denom: DENOM_LOCAL_CHAIN.to_string(), - amount: deposit_amount, - }], - ); - let deposit_msg = ExecuteMsg::Bond { - recipient: Option::None, - }; - let deposit_res = execute( - deps.as_mut(), - env.clone(), - deposit_info.clone(), - deposit_msg, - ) - .unwrap(); - - assert_eq!(deposit_res.messages.len(), init_msg.primitives.len()); - - // total money sent to the vault - let total_money = deposit_info.funds[0].amount; - - // total weight from init msg - let total_weight: Decimal = init_msg.primitives.iter().map(|p| p.weight).sum(); - assert_eq!(total_weight, Decimal::one()); - - for (i, msg) in deposit_res.messages.iter().enumerate() { - if let CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr: _, - funds, - msg: _, - }) = &msg.msg - { - // weight[i] / total_weight * total_money = money_output[i] - let expected = init_msg.primitives[i].weight / total_weight * total_money; - assert_eq!(expected, funds[0].amount); - } - } - } -} - -#[test] -fn test_unbond_with_funds_for_none_and_some() { - let mut deps = mock_dependencies(); - let env = mock_env(); - let info = mock_info("pepe", &coins(420, "uqsr")); - - let msg = ExecuteMsg::Unbond { amount: None }; - let res = execute(deps.as_mut(), env.clone(), info.clone(), msg); - assert_eq!(res.unwrap_err(), PaymentError::NonPayable {}.into()); - - let msg = ExecuteMsg::Unbond { - amount: Some(Uint128::new(420)), - }; - let res = execute(deps.as_mut(), env, info, msg); - assert_eq!(res.unwrap_err(), PaymentError::NonPayable {}.into()); -} - -#[test] -fn test_claim_with_funds() { - let mut deps = mock_dependencies(); - let env = mock_env(); - let info = mock_info("juan", &coins(420, "uqsr")); - - let msg = ExecuteMsg::Claim {}; - let res = execute(deps.as_mut(), env, info, msg); - assert_eq!(res.unwrap_err(), PaymentError::NonPayable {}.into()); -} diff --git a/smart-contracts/osmosis/contracts/basic-vault/src/types.rs b/smart-contracts/osmosis/contracts/basic-vault/src/types.rs deleted file mode 100644 index 3d540d501..000000000 --- a/smart-contracts/osmosis/contracts/basic-vault/src/types.rs +++ /dev/null @@ -1,11 +0,0 @@ -use cosmwasm_std::{Decimal, Uint128}; - -pub trait FromUint128 { - fn from_uint128(value: Uint128) -> Self; -} - -impl FromUint128 for Decimal { - fn from_uint128(value: Uint128) -> Self { - Decimal::from_ratio(value, 1u128) - } -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/.cargo/config b/smart-contracts/osmosis/contracts/cw-4626/.cargo/config deleted file mode 100644 index 8d4bc738b..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/.cargo/config +++ /dev/null @@ -1,6 +0,0 @@ -[alias] -wasm = "build --release --target wasm32-unknown-unknown" -wasm-debug = "build --target wasm32-unknown-unknown" -unit-test = "test --lib" -integration-test = "test --test integration" -schema = "run --example schema" diff --git a/smart-contracts/osmosis/contracts/cw-4626/Cargo.lock b/smart-contracts/osmosis/contracts/cw-4626/Cargo.lock deleted file mode 100644 index 089cfe022..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/Cargo.lock +++ /dev/null @@ -1,745 +0,0 @@ -# This file is automatically @generated by Cargo. -# It is not intended for manual editing. -version = 3 - -[[package]] -name = "arrayvec" -version = "0.7.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8da52d66c7071e2e3fa2a1e5c6d088fec47b593032b254f5e980de8ea54454d6" - -[[package]] -name = "autocfg" -version = "1.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa" - -[[package]] -name = "base16ct" -version = "0.1.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "349a06037c7bf932dd7e7d1f653678b2038b9ad46a74102f1fc7bd7872678cce" - -[[package]] -name = "base64" -version = "0.13.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "904dfeac50f3cdaba28fc6f57fdcddb75f49ed61346676a78c4ffe55877802fd" - -[[package]] -name = "base64ct" -version = "1.5.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dea908e7347a8c64e378c17e30ef880ad73e3b4498346b055c2c00ea342f3179" - -[[package]] -name = "block-buffer" -version = "0.9.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4152116fd6e9dadb291ae18fc1ec3575ed6d84c29642d97890f4b4a3417297e4" -dependencies = [ - "generic-array", -] - -[[package]] -name = "byteorder" -version = "1.4.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "14c189c53d098945499cdfa7ecc63567cf3886b3332b312a5b4585d8d3a6a610" - -[[package]] -name = "cfg-if" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" - -[[package]] -name = "const-oid" -version = "0.7.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e4c78c047431fee22c1a7bb92e00ad095a02a983affe4d8a72e2a2c62c1b94f3" - -[[package]] -name = "cosmwasm-crypto" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5eb0afef2325df81aadbf9be1233f522ed8f6e91df870c764bc44cca2b1415bd" -dependencies = [ - "digest", - "ed25519-zebra", - "k256", - "rand_core 0.6.3", - "thiserror", -] - -[[package]] -name = "cosmwasm-derive" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4b36e527620a2a3e00e46b6e731ab6c9b68d11069c986f7d7be8eba79ef081a4" -dependencies = [ - "syn", -] - -[[package]] -name = "cosmwasm-schema" -version = "1.0.0-rc.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0f95658fed27bc64d4547da6fdbd2f7375bbf84e2f4c6ee6f0cab544294ecaca" -dependencies = [ - "schemars", - "serde_json", -] - -[[package]] -name = "cosmwasm-std" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "875994993c2082a6fcd406937bf0fca21c349e4a624f3810253a14fa83a3a195" -dependencies = [ - "base64", - "cosmwasm-crypto", - "cosmwasm-derive", - "forward_ref", - "schemars", - "serde", - "serde-json-wasm", - "thiserror", - "uint", -] - -[[package]] -name = "cpufeatures" -version = "0.2.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "59a6001667ab124aebae2a495118e11d30984c3a653e99d86d58971708cf5e4b" -dependencies = [ - "libc", -] - -[[package]] -name = "crunchy" -version = "0.2.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7a81dae078cea95a014a339291cec439d2f232ebe854a9d672b796c6afafa9b7" - -[[package]] -name = "crypto-bigint" -version = "0.3.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "03c6a1d5fa1de37e071642dfa44ec552ca5b299adb128fab16138e24b548fd21" -dependencies = [ - "generic-array", - "rand_core 0.6.3", - "subtle", - "zeroize", -] - -[[package]] -name = "crypto-mac" -version = "0.11.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b1d1a86f49236c215f271d40892d5fc950490551400b02ef360692c29815c714" -dependencies = [ - "generic-array", - "subtle", -] - -[[package]] -name = "curve25519-dalek" -version = "3.2.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0b9fdf9972b2bd6af2d913799d9ebc165ea4d2e65878e329d9c6b372c4491b61" -dependencies = [ - "byteorder", - "digest", - "rand_core 0.5.1", - "subtle", - "zeroize", -] - -[[package]] -name = "cw-4626" -version = "0.13.4" -dependencies = [ - "cosmwasm-schema", - "cosmwasm-std", - "cw-storage-plus", - "cw-utils", - "cw2", - "cw20", - "cw20-base", - "quasar-traits", - "quasar-types", - "schemars", - "serde", - "share-distributor", - "thiserror", -] - -[[package]] -name = "cw-storage-plus" -version = "0.13.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "648b1507290bbc03a8d88463d7cd9b04b1fa0155e5eef366c4fa052b9caaac7a" -dependencies = [ - "cosmwasm-std", - "schemars", - "serde", -] - -[[package]] -name = "cw-utils" -version = "0.13.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9dbaecb78c8e8abfd6b4258c7f4fbeb5c49a5e45ee4d910d3240ee8e1d714e1b" -dependencies = [ - "cosmwasm-std", - "schemars", - "serde", - "thiserror", -] - -[[package]] -name = "cw2" -version = "0.13.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "04cf4639517490dd36b333bbd6c4fbd92e325fd0acf4683b41753bc5eb63bfc1" -dependencies = [ - "cosmwasm-std", - "cw-storage-plus", - "schemars", - "serde", -] - -[[package]] -name = "cw20" -version = "0.13.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4cb782b8f110819a4eb5dbbcfed25ffba49ec16bbe32b4ad8da50a5ce68fec05" -dependencies = [ - "cosmwasm-std", - "cw-utils", - "schemars", - "serde", -] - -[[package]] -name = "cw20-base" -version = "0.13.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0306e606581f4fb45e82bcbb7f0333179ed53dd949c6523f01a99b4bfc1475a0" -dependencies = [ - "cosmwasm-std", - "cw-storage-plus", - "cw-utils", - "cw2", - "cw20", - "schemars", - "serde", - "thiserror", -] - -[[package]] -name = "der" -version = "0.5.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6919815d73839e7ad218de758883aae3a257ba6759ce7a9992501efbb53d705c" -dependencies = [ - "const-oid", -] - -[[package]] -name = "digest" -version = "0.9.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d3dd60d1080a57a05ab032377049e0591415d2b31afd7028356dbf3cc6dcb066" -dependencies = [ - "generic-array", -] - -[[package]] -name = "dyn-clone" -version = "1.0.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "21e50f3adc76d6a43f5ed73b698a87d0760ca74617f60f7c3b879003536fdd28" - -[[package]] -name = "ecdsa" -version = "0.13.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d0d69ae62e0ce582d56380743515fefaf1a8c70cec685d9677636d7e30ae9dc9" -dependencies = [ - "der", - "elliptic-curve", - "rfc6979", - "signature", -] - -[[package]] -name = "ed25519-zebra" -version = "3.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "403ef3e961ab98f0ba902771d29f842058578bb1ce7e3c59dad5a6a93e784c69" -dependencies = [ - "curve25519-dalek", - "hex", - "rand_core 0.6.3", - "serde", - "sha2", - "thiserror", - "zeroize", -] - -[[package]] -name = "elliptic-curve" -version = "0.11.12" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "25b477563c2bfed38a3b7a60964c49e058b2510ad3f12ba3483fd8f62c2306d6" -dependencies = [ - "base16ct", - "crypto-bigint", - "der", - "ff", - "generic-array", - "group", - "rand_core 0.6.3", - "sec1", - "subtle", - "zeroize", -] - -[[package]] -name = "ff" -version = "0.11.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "131655483be284720a17d74ff97592b8e76576dc25563148601df2d7c9080924" -dependencies = [ - "rand_core 0.6.3", - "subtle", -] - -[[package]] -name = "forward_ref" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c8cbd1169bd7b4a0a20d92b9af7a7e0422888bd38a6f5ec29c1fd8c1558a272e" - -[[package]] -name = "generic-array" -version = "0.14.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fd48d33ec7f05fbfa152300fdad764757cbded343c1aa1cff2fbaf4134851803" -dependencies = [ - "typenum", - "version_check", -] - -[[package]] -name = "getrandom" -version = "0.1.16" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8fc3cb4d91f53b50155bdcfd23f6a4c39ae1969c2ae85982b135750cccaf5fce" -dependencies = [ - "cfg-if", - "libc", - "wasi 0.9.0+wasi-snapshot-preview1", -] - -[[package]] -name = "getrandom" -version = "0.2.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9be70c98951c83b8d2f8f60d7065fa6d5146873094452a1008da8c2f1e4205ad" -dependencies = [ - "cfg-if", - "libc", - "wasi 0.10.2+wasi-snapshot-preview1", -] - -[[package]] -name = "group" -version = "0.11.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bc5ac374b108929de78460075f3dc439fa66df9d8fc77e8f12caa5165fcf0c89" -dependencies = [ - "ff", - "rand_core 0.6.3", - "subtle", -] - -[[package]] -name = "hex" -version = "0.4.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70" - -[[package]] -name = "hmac" -version = "0.11.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2a2a2320eb7ec0ebe8da8f744d7812d9fc4cb4d09344ac01898dbcb6a20ae69b" -dependencies = [ - "crypto-mac", - "digest", -] - -[[package]] -name = "itoa" -version = "1.0.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1aab8fc367588b89dcee83ab0fd66b72b50b72fa1904d7095045ace2b0c81c35" - -[[package]] -name = "k256" -version = "0.10.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "19c3a5e0a0b8450278feda242592512e09f61c72e018b8cd5c859482802daf2d" -dependencies = [ - "cfg-if", - "ecdsa", - "elliptic-curve", - "sec1", - "sha2", -] - -[[package]] -name = "libc" -version = "0.2.125" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5916d2ae698f6de9bfb891ad7a8d65c09d232dc58cc4ac433c7da3b2fd84bc2b" - -[[package]] -name = "num-traits" -version = "0.2.15" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "578ede34cf02f8924ab9447f50c28075b4d3e5b269972345e7e0372b38c6cdcd" -dependencies = [ - "autocfg", -] - -[[package]] -name = "opaque-debug" -version = "0.3.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "624a8340c38c1b80fd549087862da4ba43e08858af025b236e509b6649fc13d5" - -[[package]] -name = "pkcs8" -version = "0.8.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7cabda3fb821068a9a4fab19a683eac3af12edf0f34b94a8be53c4972b8149d0" -dependencies = [ - "der", - "spki", - "zeroize", -] - -[[package]] -name = "proc-macro2" -version = "1.0.38" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9027b48e9d4c9175fa2218adf3557f91c1137021739951d4932f5f8268ac48aa" -dependencies = [ - "unicode-xid", -] - -[[package]] -name = "quasar-traits" -version = "0.1.0" -dependencies = [ - "cosmwasm-std", - "cw20", - "schemars", - "serde", -] - -[[package]] -name = "quasar-types" -version = "0.1.0" -dependencies = [ - "cosmwasm-std", - "cw20", - "quasar-traits", - "rust_decimal", - "schemars", - "serde", -] - -[[package]] -name = "quote" -version = "1.0.18" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a1feb54ed693b93a84e14094943b84b7c4eae204c512b7ccb95ab0c66d278ad1" -dependencies = [ - "proc-macro2", -] - -[[package]] -name = "rand_core" -version = "0.5.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "90bde5296fc891b0cef12a6d03ddccc162ce7b2aff54160af9338f8d40df6d19" -dependencies = [ - "getrandom 0.1.16", -] - -[[package]] -name = "rand_core" -version = "0.6.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d34f1408f55294453790c48b2f1ebbb1c5b4b7563eb1f418bcfcfdbb06ebb4e7" -dependencies = [ - "getrandom 0.2.6", -] - -[[package]] -name = "rfc6979" -version = "0.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "96ef608575f6392792f9ecf7890c00086591d29a83910939d430753f7c050525" -dependencies = [ - "crypto-bigint", - "hmac", - "zeroize", -] - -[[package]] -name = "rust_decimal" -version = "1.25.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "34a3bb58e85333f1ab191bf979104b586ebd77475bc6681882825f4532dfe87c" -dependencies = [ - "arrayvec", - "num-traits", - "serde", -] - -[[package]] -name = "ryu" -version = "1.0.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "73b4b750c782965c211b42f022f59af1fbceabdd026623714f104152f1ec149f" - -[[package]] -name = "schemars" -version = "0.8.8" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c6b5a3c80cea1ab61f4260238409510e814e38b4b563c06044edf91e7dc070e3" -dependencies = [ - "dyn-clone", - "schemars_derive", - "serde", - "serde_json", -] - -[[package]] -name = "schemars_derive" -version = "0.8.8" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "41ae4dce13e8614c46ac3c38ef1c0d668b101df6ac39817aebdaa26642ddae9b" -dependencies = [ - "proc-macro2", - "quote", - "serde_derive_internals", - "syn", -] - -[[package]] -name = "sec1" -version = "0.2.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "08da66b8b0965a5555b6bd6639e68ccba85e1e2506f5fbb089e93f8a04e1a2d1" -dependencies = [ - "der", - "generic-array", - "pkcs8", - "subtle", - "zeroize", -] - -[[package]] -name = "serde" -version = "1.0.137" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "61ea8d54c77f8315140a05f4c7237403bf38b72704d031543aa1d16abbf517d1" -dependencies = [ - "serde_derive", -] - -[[package]] -name = "serde-json-wasm" -version = "0.4.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "479b4dbc401ca13ee8ce902851b834893251404c4f3c65370a49e047a6be09a5" -dependencies = [ - "serde", -] - -[[package]] -name = "serde_derive" -version = "1.0.137" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1f26faba0c3959972377d3b2d306ee9f71faee9714294e41bb777f83f88578be" -dependencies = [ - "proc-macro2", - "quote", - "syn", -] - -[[package]] -name = "serde_derive_internals" -version = "0.25.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1dbab34ca63057a1f15280bdf3c39f2b1eb1b54c17e98360e511637aef7418c6" -dependencies = [ - "proc-macro2", - "quote", - "syn", -] - -[[package]] -name = "serde_json" -version = "1.0.81" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9b7ce2b32a1aed03c558dc61a5cd328f15aff2dbc17daad8fb8af04d2100e15c" -dependencies = [ - "itoa", - "ryu", - "serde", -] - -[[package]] -name = "sha2" -version = "0.9.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4d58a1e1bf39749807d89cf2d98ac2dfa0ff1cb3faa38fbb64dd88ac8013d800" -dependencies = [ - "block-buffer", - "cfg-if", - "cpufeatures", - "digest", - "opaque-debug", -] - -[[package]] -name = "share-distributor" -version = "0.1.0" -dependencies = [ - "cosmwasm-std", - "cw20", - "quasar-traits", - "schemars", - "serde", -] - -[[package]] -name = "signature" -version = "1.4.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "02658e48d89f2bec991f9a78e69cfa4c316f8d6a6c4ec12fae1aeb263d486788" -dependencies = [ - "digest", - "rand_core 0.6.3", -] - -[[package]] -name = "spki" -version = "0.5.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "44d01ac02a6ccf3e07db148d2be087da624fea0221a16152ed01f0496a6b0a27" -dependencies = [ - "base64ct", - "der", -] - -[[package]] -name = "static_assertions" -version = "1.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f" - -[[package]] -name = "subtle" -version = "2.4.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6bdef32e8150c2a081110b42772ffe7d7c9032b606bc226c8260fd97e0976601" - -[[package]] -name = "syn" -version = "1.0.94" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a07e33e919ebcd69113d5be0e4d70c5707004ff45188910106854f38b960df4a" -dependencies = [ - "proc-macro2", - "quote", - "unicode-xid", -] - -[[package]] -name = "thiserror" -version = "1.0.31" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bd829fe32373d27f76265620b5309d0340cb8550f523c1dda251d6298069069a" -dependencies = [ - "thiserror-impl", -] - -[[package]] -name = "thiserror-impl" -version = "1.0.31" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0396bc89e626244658bef819e22d0cc459e795a5ebe878e6ec336d1674a8d79a" -dependencies = [ - "proc-macro2", - "quote", - "syn", -] - -[[package]] -name = "typenum" -version = "1.15.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dcf81ac59edc17cc8697ff311e8f5ef2d99fcbd9817b34cec66f90b6c3dfd987" - -[[package]] -name = "uint" -version = "0.9.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "12f03af7ccf01dd611cc450a0d10dbc9b745770d096473e2faf0ca6e2d66d1e0" -dependencies = [ - "byteorder", - "crunchy", - "hex", - "static_assertions", -] - -[[package]] -name = "unicode-xid" -version = "0.2.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "957e51f3646910546462e67d5f7599b9e4fb8acdd304b087a6494730f9eebf04" - -[[package]] -name = "version_check" -version = "0.9.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "49874b5167b65d7193b8aba1567f5c7d93d001cafc34600cee003eda787e483f" - -[[package]] -name = "wasi" -version = "0.9.0+wasi-snapshot-preview1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cccddf32554fecc6acb585f82a32a72e28b48f8c4c1883ddfeeeaa96f7d8e519" - -[[package]] -name = "wasi" -version = "0.10.2+wasi-snapshot-preview1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fd6fbd9a79829dd1ad0cc20627bf1ed606756a7f77edff7b66b7064f9cb327c6" - -[[package]] -name = "zeroize" -version = "1.5.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "94693807d016b2f2d2e14420eb3bfcca689311ff775dcf113d74ea624b7cdf07" diff --git a/smart-contracts/osmosis/contracts/cw-4626/Cargo.toml b/smart-contracts/osmosis/contracts/cw-4626/Cargo.toml deleted file mode 100644 index 836d41947..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/Cargo.toml +++ /dev/null @@ -1,40 +0,0 @@ -[package] -name = "cw-4626" -version = "0.13.4" -authors = ["Laurens Kubat "] -edition = "2021" -description = "Basic implementation of the Quasar vault" -license = "Apache-2.0" -repository = "https://github.com/quasar-finance/quasar" -homepage = "https://www.quasar.fi/" -documentation = "" - -[lib] -crate-type = ["cdylib", "rlib"] - -[features] -backtraces = ["cosmwasm-std/backtraces"] -# use library feature to disable all instantiate/execute/query exports -library = [] - -[dependencies] -cosmwasm-std = { workspace = true } -osmosis-std = { workspace = true } -osmosis-std-derive = { workspace = true } -cosmwasm-schema = { workspace = true } -cw-storage-plus ={ workspace = true } -schemars ={ workspace = true } -serde = { workspace = true } -serde_json = { workspace = true } -thiserror = { workspace = true } -cw-utils = { workspace = true } -cw2 = { workspace = true } -cw20 = { workspace = true } -cw20-base = { workspace = true } - -quasar-traits = { path = "../../packages/quasar-traits" } -quasar-types = { path = "../../packages/quasar-types"} -strategy = { path = "../strategy", features = ["library"]} - -[dev-dependencies] -cosmwasm-schema = { workspace = true } diff --git a/smart-contracts/osmosis/contracts/cw-4626/NOTICE b/smart-contracts/osmosis/contracts/cw-4626/NOTICE deleted file mode 100644 index 84b1c2103..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/NOTICE +++ /dev/null @@ -1,14 +0,0 @@ -CW20-Base: A reference implementation for fungible token on CosmWasm -Copyright (C) 2020 Confio OÜ - -Licensed under the Apache License, Version 2.0 (the "License"); -you may not use this file except in compliance with the License. -You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - -Unless required by applicable law or agreed to in writing, software -distributed under the License is distributed on an "AS IS" BASIS, -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -See the License for the specific language governing permissions and -limitations under the License. diff --git a/smart-contracts/osmosis/contracts/cw-4626/README.md b/smart-contracts/osmosis/contracts/cw-4626/README.md deleted file mode 100644 index 7ab7f7aa3..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/README.md +++ /dev/null @@ -1,61 +0,0 @@ -# CW-4626 - CW20 Base - -Built on the [CW20 Base](https://github.com/CosmWasm/cw-plus/tree/main/contracts/cw20-base) contract, this contract -is a basic implementation of [eip-4626](https://eips.ethereum.org/EIPS/eip-4626). -This is a basic implementation of a cw20 contract. It implements -the [CW20 spec](quasar-finance/quasar/smart-contracts/cw-plus/packages/cw20/README.md) and is designed to -be deployed as is, or imported into other contracts to easily build -cw20-compatible tokens with custom logic. - -Implements: - -- [x] CW20 Base -- [x] Mintable extension -- [x] Allowances extension -- [ ] EIP-4626 implementation - - [ ] Change CW-20 asset to vault share - - [ ] implement EIP-4626 interface for cosmwasm - - [x] Implement token whitelist - - [ ] Add share minting logic - - [ ] Add share burning logic - - [ ] Add support for all queries - - [ ] Add support for all execute messages -- [ ] Multi-Asset Deposit -- [ ] Multi-Asset Withdrawal -- [ ] Strategy trait - -## Running this contract - -You will need Rust 1.44.1+ with `wasm32-unknown-unknown` target installed. - -You can run unit tests on this via: - -`cargo test` - -Once you are happy with the content, you can compile it to wasm via: - -``` -RUSTFLAGS='-C link-arg=-s' cargo wasm -cp ../../target/wasm32-unknown-unknown/release/cw20_base.wasm . -ls -l cw20_base.wasm -sha256sum cw20_base.wasm -``` - -Or for a production-ready (optimized) build, run a build command in the -the repository root: https://github.com/CosmWasm/cw-plus#compiling. - -## Importing this contract - -You can also import much of the logic of this contract to build another -ERC20-contract, such as a bonding curve, overiding or extending what you -need. - -Basically, you just need to write your handle function and import -`cw20_base::contract::handle_transfer`, etc and dispatch to them. -This allows you to use custom `ExecuteMsg` and `QueryMsg` with your additional -calls, but then use the underlying implementation for the standard cw20 -messages you want to support. The same with `QueryMsg`. You *could* reuse `instantiate` -as it, but it is likely you will want to change it. And it is rather simple. - -Look at [`cw20-staking`](https://github.com/CosmWasm/cw-tokens/tree/main/contracts/cw20-staking) for an example of how to "inherit" -all this token functionality and combine it with custom logic. diff --git a/smart-contracts/osmosis/contracts/cw-4626/examples/schema.rs b/smart-contracts/osmosis/contracts/cw-4626/examples/schema.rs deleted file mode 100644 index ebc8fc33e..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/examples/schema.rs +++ /dev/null @@ -1,37 +0,0 @@ -use std::env::current_dir; -use std::fs::create_dir_all; -extern crate cw_4626; - -use cosmwasm_schema::{export_schema, remove_schemas, schema_for}; -use cw20::{ - AllAccountsResponse, AllAllowancesResponse, AllowanceResponse, BalanceResponse, - TokenInfoResponse, -}; -use cw20_base::msg::{ExecuteMsg, QueryMsg}; -use cw_4626::msg::{ - AssetResponse, ConvertToAssetsResponse, ConvertToSharesResponse, InstantiateMsg, - MaxDepositResponse, TotalAssetResponse, VaultInfoResponse, -}; - -fn main() { - let mut out_dir = current_dir().unwrap(); - out_dir.push("schema"); - create_dir_all(&out_dir).unwrap(); - remove_schemas(&out_dir).unwrap(); - - export_schema(&schema_for!(InstantiateMsg), &out_dir); - export_schema(&schema_for!(ExecuteMsg), &out_dir); - export_schema(&schema_for!(QueryMsg), &out_dir); - export_schema(&schema_for!(AllowanceResponse), &out_dir); - export_schema(&schema_for!(BalanceResponse), &out_dir); - export_schema(&schema_for!(TokenInfoResponse), &out_dir); - export_schema(&schema_for!(AllAllowancesResponse), &out_dir); - export_schema(&schema_for!(AllAccountsResponse), &out_dir); - // the cw-4626 schemas - export_schema(&schema_for!(AssetResponse), &out_dir); - export_schema(&schema_for!(TotalAssetResponse), &out_dir); - export_schema(&schema_for!(ConvertToSharesResponse), &out_dir); - export_schema(&schema_for!(ConvertToAssetsResponse), &out_dir); - export_schema(&schema_for!(MaxDepositResponse), &out_dir); - export_schema(&schema_for!(VaultInfoResponse), &out_dir); -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/schema/all_accounts_response.json b/smart-contracts/osmosis/contracts/cw-4626/schema/all_accounts_response.json deleted file mode 100644 index cea50fba4..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/schema/all_accounts_response.json +++ /dev/null @@ -1,16 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "AllAccountsResponse", - "type": "object", - "required": [ - "accounts" - ], - "properties": { - "accounts": { - "type": "array", - "items": { - "type": "string" - } - } - } -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/schema/all_allowances_response.json b/smart-contracts/osmosis/contracts/cw-4626/schema/all_allowances_response.json deleted file mode 100644 index 6bb2291ce..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/schema/all_allowances_response.json +++ /dev/null @@ -1,99 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "AllAllowancesResponse", - "type": "object", - "required": [ - "allowances" - ], - "properties": { - "allowances": { - "type": "array", - "items": { - "$ref": "#/definitions/AllowanceInfo" - } - } - }, - "definitions": { - "AllowanceInfo": { - "type": "object", - "required": [ - "allowance", - "expires", - "spender" - ], - "properties": { - "allowance": { - "$ref": "#/definitions/Uint128" - }, - "expires": { - "$ref": "#/definitions/Expiration" - }, - "spender": { - "type": "string" - } - } - }, - "Expiration": { - "description": "Expiration represents a point in time when some event happens. It can compare with a BlockInfo and will return is_expired() == true once the condition is hit (and for every block in the future)", - "oneOf": [ - { - "description": "AtHeight will expire when `env.block.height` >= height", - "type": "object", - "required": [ - "at_height" - ], - "properties": { - "at_height": { - "type": "integer", - "format": "uint64", - "minimum": 0.0 - } - }, - "additionalProperties": false - }, - { - "description": "AtTime will expire when `env.block.time` >= time", - "type": "object", - "required": [ - "at_time" - ], - "properties": { - "at_time": { - "$ref": "#/definitions/Timestamp" - } - }, - "additionalProperties": false - }, - { - "description": "Never will never expire. Used to express the empty variant", - "type": "object", - "required": [ - "never" - ], - "properties": { - "never": { - "type": "object" - } - }, - "additionalProperties": false - } - ] - }, - "Timestamp": { - "description": "A point in time in nanosecond precision.\n\nThis type can represent times from 1970-01-01T00:00:00Z to 2554-07-21T23:34:33Z.\n\n## Examples\n\n``` # use cosmwasm_std::Timestamp; let ts = Timestamp::from_nanos(1_000_000_202); assert_eq!(ts.nanos(), 1_000_000_202); assert_eq!(ts.seconds(), 1); assert_eq!(ts.subsec_nanos(), 202);\n\nlet ts = ts.plus_seconds(2); assert_eq!(ts.nanos(), 3_000_000_202); assert_eq!(ts.seconds(), 3); assert_eq!(ts.subsec_nanos(), 202); ```", - "allOf": [ - { - "$ref": "#/definitions/Uint64" - } - ] - }, - "Uint128": { - "description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```", - "type": "string" - }, - "Uint64": { - "description": "A thin wrapper around u64 that is using strings for JSON encoding/decoding, such that the full u64 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u64` to get the value out:\n\n``` # use cosmwasm_std::Uint64; let a = Uint64::from(42u64); assert_eq!(a.u64(), 42);\n\nlet b = Uint64::from(70u32); assert_eq!(b.u64(), 70); ```", - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/schema/allowance_response.json b/smart-contracts/osmosis/contracts/cw-4626/schema/allowance_response.json deleted file mode 100644 index c4f98d6fc..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/schema/allowance_response.json +++ /dev/null @@ -1,81 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "AllowanceResponse", - "type": "object", - "required": [ - "allowance", - "expires" - ], - "properties": { - "allowance": { - "$ref": "#/definitions/Uint128" - }, - "expires": { - "$ref": "#/definitions/Expiration" - } - }, - "definitions": { - "Expiration": { - "description": "Expiration represents a point in time when some event happens. It can compare with a BlockInfo and will return is_expired() == true once the condition is hit (and for every block in the future)", - "oneOf": [ - { - "description": "AtHeight will expire when `env.block.height` >= height", - "type": "object", - "required": [ - "at_height" - ], - "properties": { - "at_height": { - "type": "integer", - "format": "uint64", - "minimum": 0.0 - } - }, - "additionalProperties": false - }, - { - "description": "AtTime will expire when `env.block.time` >= time", - "type": "object", - "required": [ - "at_time" - ], - "properties": { - "at_time": { - "$ref": "#/definitions/Timestamp" - } - }, - "additionalProperties": false - }, - { - "description": "Never will never expire. Used to express the empty variant", - "type": "object", - "required": [ - "never" - ], - "properties": { - "never": { - "type": "object" - } - }, - "additionalProperties": false - } - ] - }, - "Timestamp": { - "description": "A point in time in nanosecond precision.\n\nThis type can represent times from 1970-01-01T00:00:00Z to 2554-07-21T23:34:33Z.\n\n## Examples\n\n``` # use cosmwasm_std::Timestamp; let ts = Timestamp::from_nanos(1_000_000_202); assert_eq!(ts.nanos(), 1_000_000_202); assert_eq!(ts.seconds(), 1); assert_eq!(ts.subsec_nanos(), 202);\n\nlet ts = ts.plus_seconds(2); assert_eq!(ts.nanos(), 3_000_000_202); assert_eq!(ts.seconds(), 3); assert_eq!(ts.subsec_nanos(), 202); ```", - "allOf": [ - { - "$ref": "#/definitions/Uint64" - } - ] - }, - "Uint128": { - "description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```", - "type": "string" - }, - "Uint64": { - "description": "A thin wrapper around u64 that is using strings for JSON encoding/decoding, such that the full u64 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u64` to get the value out:\n\n``` # use cosmwasm_std::Uint64; let a = Uint64::from(42u64); assert_eq!(a.u64(), 42);\n\nlet b = Uint64::from(70u32); assert_eq!(b.u64(), 70); ```", - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/schema/asset_response.json b/smart-contracts/osmosis/contracts/cw-4626/schema/asset_response.json deleted file mode 100644 index e5ff52aa7..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/schema/asset_response.json +++ /dev/null @@ -1,13 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "AssetResponse", - "type": "object", - "required": [ - "denom" - ], - "properties": { - "denom": { - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/schema/balance_response.json b/smart-contracts/osmosis/contracts/cw-4626/schema/balance_response.json deleted file mode 100644 index 4e1a0be2b..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/schema/balance_response.json +++ /dev/null @@ -1,19 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "BalanceResponse", - "type": "object", - "required": [ - "balance" - ], - "properties": { - "balance": { - "$ref": "#/definitions/Uint128" - } - }, - "definitions": { - "Uint128": { - "description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```", - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/schema/convert_to_assets_response.json b/smart-contracts/osmosis/contracts/cw-4626/schema/convert_to_assets_response.json deleted file mode 100644 index 803477018..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/schema/convert_to_assets_response.json +++ /dev/null @@ -1,37 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "ConvertToAssetsResponse", - "type": "object", - "required": [ - "assets" - ], - "properties": { - "assets": { - "type": "array", - "items": { - "$ref": "#/definitions/Coin" - } - } - }, - "definitions": { - "Coin": { - "type": "object", - "required": [ - "amount", - "denom" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - }, - "denom": { - "type": "string" - } - } - }, - "Uint128": { - "description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```", - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/schema/convert_to_shares_response.json b/smart-contracts/osmosis/contracts/cw-4626/schema/convert_to_shares_response.json deleted file mode 100644 index fcf1dd75c..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/schema/convert_to_shares_response.json +++ /dev/null @@ -1,19 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "ConvertToSharesResponse", - "type": "object", - "required": [ - "amount" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - } - }, - "definitions": { - "Uint128": { - "description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```", - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/schema/cw20_execute_msg.json b/smart-contracts/osmosis/contracts/cw-4626/schema/cw20_execute_msg.json deleted file mode 100644 index e4bc559cd..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/schema/cw20_execute_msg.json +++ /dev/null @@ -1,442 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "Cw20ExecuteMsg", - "oneOf": [ - { - "description": "Transfer is a base message to move tokens to another account without triggering actions", - "type": "object", - "required": [ - "transfer" - ], - "properties": { - "transfer": { - "type": "object", - "required": [ - "amount", - "recipient" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - }, - "recipient": { - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "description": "Burn is a base message to destroy tokens forever", - "type": "object", - "required": [ - "burn" - ], - "properties": { - "burn": { - "type": "object", - "required": [ - "amount" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - } - } - } - }, - "additionalProperties": false - }, - { - "description": "Send is a base message to transfer tokens to a contract and trigger an action on the receiving contract.", - "type": "object", - "required": [ - "send" - ], - "properties": { - "send": { - "type": "object", - "required": [ - "amount", - "contract", - "msg" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - }, - "contract": { - "type": "string" - }, - "msg": { - "$ref": "#/definitions/Binary" - } - } - } - }, - "additionalProperties": false - }, - { - "description": "Only with \"approval\" extension. Allows spender to access an additional amount tokens from the owner's (env.sender) account. If expires is Some(), overwrites current allowance expiration with this one.", - "type": "object", - "required": [ - "increase_allowance" - ], - "properties": { - "increase_allowance": { - "type": "object", - "required": [ - "amount", - "spender" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - }, - "expires": { - "anyOf": [ - { - "$ref": "#/definitions/Expiration" - }, - { - "type": "null" - } - ] - }, - "spender": { - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "description": "Only with \"approval\" extension. Lowers the spender's access of tokens from the owner's (env.sender) account by amount. If expires is Some(), overwrites current allowance expiration with this one.", - "type": "object", - "required": [ - "decrease_allowance" - ], - "properties": { - "decrease_allowance": { - "type": "object", - "required": [ - "amount", - "spender" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - }, - "expires": { - "anyOf": [ - { - "$ref": "#/definitions/Expiration" - }, - { - "type": "null" - } - ] - }, - "spender": { - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "description": "Only with \"approval\" extension. Transfers amount tokens from owner -> recipient if `env.sender` has sufficient pre-approval.", - "type": "object", - "required": [ - "transfer_from" - ], - "properties": { - "transfer_from": { - "type": "object", - "required": [ - "amount", - "owner", - "recipient" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - }, - "owner": { - "type": "string" - }, - "recipient": { - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "description": "Only with \"approval\" extension. Sends amount tokens from owner -> contract if `env.sender` has sufficient pre-approval.", - "type": "object", - "required": [ - "send_from" - ], - "properties": { - "send_from": { - "type": "object", - "required": [ - "amount", - "contract", - "msg", - "owner" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - }, - "contract": { - "type": "string" - }, - "msg": { - "$ref": "#/definitions/Binary" - }, - "owner": { - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "description": "Only with \"approval\" extension. Destroys tokens forever", - "type": "object", - "required": [ - "burn_from" - ], - "properties": { - "burn_from": { - "type": "object", - "required": [ - "amount", - "owner" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - }, - "owner": { - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "description": "Only with the \"mintable\" extension. If authorized, creates amount new tokens and adds to the recipient balance.", - "type": "object", - "required": [ - "mint" - ], - "properties": { - "mint": { - "type": "object", - "required": [ - "amount", - "recipient" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - }, - "recipient": { - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "description": "Only with the \"marketing\" extension. If authorized, updates marketing metadata. Setting None/null for any of these will leave it unchanged. Setting Some(\"\") will clear this field on the contract storage", - "type": "object", - "required": [ - "update_marketing" - ], - "properties": { - "update_marketing": { - "type": "object", - "properties": { - "description": { - "description": "A longer description of the token and it's utility. Designed for tooltips or such", - "type": [ - "string", - "null" - ] - }, - "marketing": { - "description": "The address (if any) who can update this data structure", - "type": [ - "string", - "null" - ] - }, - "project": { - "description": "A URL pointing to the project behind this token.", - "type": [ - "string", - "null" - ] - } - } - } - }, - "additionalProperties": false - }, - { - "description": "If set as the \"marketing\" role on the contract, upload a new URL, SVG, or PNG for the token", - "type": "object", - "required": [ - "upload_logo" - ], - "properties": { - "upload_logo": { - "$ref": "#/definitions/Logo" - } - }, - "additionalProperties": false - } - ], - "definitions": { - "Binary": { - "description": "Binary is a wrapper around Vec to add base64 de/serialization with serde. It also adds some helper methods to help encode inline.\n\nThis is only needed as serde-json-{core,wasm} has a horrible encoding for Vec", - "type": "string" - }, - "EmbeddedLogo": { - "description": "This is used to store the logo on the blockchain in an accepted format. Enforce maximum size of 5KB on all variants.", - "oneOf": [ - { - "description": "Store the Logo as an SVG file. The content must conform to the spec at https://en.wikipedia.org/wiki/Scalable_Vector_Graphics (The contract should do some light-weight sanity-check validation)", - "type": "object", - "required": [ - "svg" - ], - "properties": { - "svg": { - "$ref": "#/definitions/Binary" - } - }, - "additionalProperties": false - }, - { - "description": "Store the Logo as a PNG file. This will likely only support up to 64x64 or so within the 5KB limit.", - "type": "object", - "required": [ - "png" - ], - "properties": { - "png": { - "$ref": "#/definitions/Binary" - } - }, - "additionalProperties": false - } - ] - }, - "Expiration": { - "description": "Expiration represents a point in time when some event happens. It can compare with a BlockInfo and will return is_expired() == true once the condition is hit (and for every block in the future)", - "oneOf": [ - { - "description": "AtHeight will expire when `env.block.height` >= height", - "type": "object", - "required": [ - "at_height" - ], - "properties": { - "at_height": { - "type": "integer", - "format": "uint64", - "minimum": 0.0 - } - }, - "additionalProperties": false - }, - { - "description": "AtTime will expire when `env.block.time` >= time", - "type": "object", - "required": [ - "at_time" - ], - "properties": { - "at_time": { - "$ref": "#/definitions/Timestamp" - } - }, - "additionalProperties": false - }, - { - "description": "Never will never expire. Used to express the empty variant", - "type": "object", - "required": [ - "never" - ], - "properties": { - "never": { - "type": "object" - } - }, - "additionalProperties": false - } - ] - }, - "Logo": { - "description": "This is used for uploading logo data, or setting it in InstantiateData", - "oneOf": [ - { - "description": "A reference to an externally hosted logo. Must be a valid HTTP or HTTPS URL.", - "type": "object", - "required": [ - "url" - ], - "properties": { - "url": { - "type": "string" - } - }, - "additionalProperties": false - }, - { - "description": "Logo content stored on the blockchain. Enforce maximum size of 5KB on all variants", - "type": "object", - "required": [ - "embedded" - ], - "properties": { - "embedded": { - "$ref": "#/definitions/EmbeddedLogo" - } - }, - "additionalProperties": false - } - ] - }, - "Timestamp": { - "description": "A point in time in nanosecond precision.\n\nThis type can represent times from 1970-01-01T00:00:00Z to 2554-07-21T23:34:33Z.\n\n## Examples\n\n``` # use cosmwasm_std::Timestamp; let ts = Timestamp::from_nanos(1_000_000_202); assert_eq!(ts.nanos(), 1_000_000_202); assert_eq!(ts.seconds(), 1); assert_eq!(ts.subsec_nanos(), 202);\n\nlet ts = ts.plus_seconds(2); assert_eq!(ts.nanos(), 3_000_000_202); assert_eq!(ts.seconds(), 3); assert_eq!(ts.subsec_nanos(), 202); ```", - "allOf": [ - { - "$ref": "#/definitions/Uint64" - } - ] - }, - "Uint128": { - "description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```", - "type": "string" - }, - "Uint64": { - "description": "A thin wrapper around u64 that is using strings for JSON encoding/decoding, such that the full u64 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u64` to get the value out:\n\n``` # use cosmwasm_std::Uint64; let a = Uint64::from(42u64); assert_eq!(a.u64(), 42);\n\nlet b = Uint64::from(70u32); assert_eq!(b.u64(), 70); ```", - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/schema/instantiate_msg.json b/smart-contracts/osmosis/contracts/cw-4626/schema/instantiate_msg.json deleted file mode 100644 index 859785a73..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/schema/instantiate_msg.json +++ /dev/null @@ -1,240 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "InstantiateMsg", - "type": "object", - "required": [ - "curve_type", - "initial_balances", - "name", - "reserve_decimals", - "reserve_denom", - "reserve_total_supply", - "supply_decimals", - "symbol" - ], - "properties": { - "curve_type": { - "$ref": "#/definitions/CurveType" - }, - "initial_balances": { - "type": "array", - "items": { - "$ref": "#/definitions/Cw20Coin" - } - }, - "marketing": { - "anyOf": [ - { - "$ref": "#/definitions/InstantiateMarketingInfo" - }, - { - "type": "null" - } - ] - }, - "mint": { - "anyOf": [ - { - "$ref": "#/definitions/MinterResponse" - }, - { - "type": "null" - } - ] - }, - "name": { - "type": "string" - }, - "reserve_decimals": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "reserve_denom": { - "type": "string" - }, - "reserve_total_supply": { - "$ref": "#/definitions/Uint128" - }, - "supply_decimals": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "symbol": { - "type": "string" - } - }, - "definitions": { - "Binary": { - "description": "Binary is a wrapper around Vec to add base64 de/serialization with serde. It also adds some helper methods to help encode inline.\n\nThis is only needed as serde-json-{core,wasm} has a horrible encoding for Vec", - "type": "string" - }, - "CurveType": { - "oneOf": [ - { - "type": "object", - "required": [ - "constant" - ], - "properties": { - "constant": { - "type": "object", - "required": [ - "scale", - "value" - ], - "properties": { - "scale": { - "type": "integer", - "format": "uint32", - "minimum": 0.0 - }, - "value": { - "$ref": "#/definitions/Uint128" - } - } - } - }, - "additionalProperties": false - } - ] - }, - "Cw20Coin": { - "type": "object", - "required": [ - "address", - "amount" - ], - "properties": { - "address": { - "type": "string" - }, - "amount": { - "$ref": "#/definitions/Uint128" - } - } - }, - "EmbeddedLogo": { - "description": "This is used to store the logo on the blockchain in an accepted format. Enforce maximum size of 5KB on all variants.", - "oneOf": [ - { - "description": "Store the Logo as an SVG file. The content must conform to the spec at https://en.wikipedia.org/wiki/Scalable_Vector_Graphics (The contract should do some light-weight sanity-check validation)", - "type": "object", - "required": [ - "svg" - ], - "properties": { - "svg": { - "$ref": "#/definitions/Binary" - } - }, - "additionalProperties": false - }, - { - "description": "Store the Logo as a PNG file. This will likely only support up to 64x64 or so within the 5KB limit.", - "type": "object", - "required": [ - "png" - ], - "properties": { - "png": { - "$ref": "#/definitions/Binary" - } - }, - "additionalProperties": false - } - ] - }, - "InstantiateMarketingInfo": { - "type": "object", - "properties": { - "description": { - "type": [ - "string", - "null" - ] - }, - "logo": { - "anyOf": [ - { - "$ref": "#/definitions/Logo" - }, - { - "type": "null" - } - ] - }, - "marketing": { - "type": [ - "string", - "null" - ] - }, - "project": { - "type": [ - "string", - "null" - ] - } - } - }, - "Logo": { - "description": "This is used for uploading logo data, or setting it in InstantiateData", - "oneOf": [ - { - "description": "A reference to an externally hosted logo. Must be a valid HTTP or HTTPS URL.", - "type": "object", - "required": [ - "url" - ], - "properties": { - "url": { - "type": "string" - } - }, - "additionalProperties": false - }, - { - "description": "Logo content stored on the blockchain. Enforce maximum size of 5KB on all variants", - "type": "object", - "required": [ - "embedded" - ], - "properties": { - "embedded": { - "$ref": "#/definitions/EmbeddedLogo" - } - }, - "additionalProperties": false - } - ] - }, - "MinterResponse": { - "type": "object", - "required": [ - "minter" - ], - "properties": { - "cap": { - "description": "cap is a hard cap on total supply that can be achieved by minting. Note that this refers to total_supply. If None, there is unlimited cap.", - "anyOf": [ - { - "$ref": "#/definitions/Uint128" - }, - { - "type": "null" - } - ] - }, - "minter": { - "type": "string" - } - } - }, - "Uint128": { - "description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```", - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/schema/max_deposit_response.json b/smart-contracts/osmosis/contracts/cw-4626/schema/max_deposit_response.json deleted file mode 100644 index e442f80e7..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/schema/max_deposit_response.json +++ /dev/null @@ -1,38 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "MaxDepositResponse", - "description": "A None response indicates that the vault has no cap an thus no max deposit", - "type": "object", - "properties": { - "max_assets": { - "type": [ - "array", - "null" - ], - "items": { - "$ref": "#/definitions/Coin" - } - } - }, - "definitions": { - "Coin": { - "type": "object", - "required": [ - "amount", - "denom" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - }, - "denom": { - "type": "string" - } - } - }, - "Uint128": { - "description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```", - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/schema/query_msg.json b/smart-contracts/osmosis/contracts/cw-4626/schema/query_msg.json deleted file mode 100644 index 7c33c352d..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/schema/query_msg.json +++ /dev/null @@ -1,168 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "QueryMsg", - "oneOf": [ - { - "description": "Returns the current balance of the given address, 0 if unset. Return type: BalanceResponse.", - "type": "object", - "required": [ - "balance" - ], - "properties": { - "balance": { - "type": "object", - "required": [ - "address" - ], - "properties": { - "address": { - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "description": "Returns metadata on the contract - name, decimals, supply, etc. Return type: TokenInfoResponse.", - "type": "object", - "required": [ - "token_info" - ], - "properties": { - "token_info": { - "type": "object" - } - }, - "additionalProperties": false - }, - { - "description": "Only with \"mintable\" extension. Returns who can mint and the hard cap on maximum tokens after minting. Return type: MinterResponse.", - "type": "object", - "required": [ - "minter" - ], - "properties": { - "minter": { - "type": "object" - } - }, - "additionalProperties": false - }, - { - "description": "Only with \"allowance\" extension. Returns how much spender can use from owner account, 0 if unset. Return type: AllowanceResponse.", - "type": "object", - "required": [ - "allowance" - ], - "properties": { - "allowance": { - "type": "object", - "required": [ - "owner", - "spender" - ], - "properties": { - "owner": { - "type": "string" - }, - "spender": { - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "description": "Only with \"enumerable\" extension (and \"allowances\") Returns all allowances this owner has approved. Supports pagination. Return type: AllAllowancesResponse.", - "type": "object", - "required": [ - "all_allowances" - ], - "properties": { - "all_allowances": { - "type": "object", - "required": [ - "owner" - ], - "properties": { - "limit": { - "type": [ - "integer", - "null" - ], - "format": "uint32", - "minimum": 0.0 - }, - "owner": { - "type": "string" - }, - "start_after": { - "type": [ - "string", - "null" - ] - } - } - } - }, - "additionalProperties": false - }, - { - "description": "Only with \"enumerable\" extension Returns all accounts that have balances. Supports pagination. Return type: AllAccountsResponse.", - "type": "object", - "required": [ - "all_accounts" - ], - "properties": { - "all_accounts": { - "type": "object", - "properties": { - "limit": { - "type": [ - "integer", - "null" - ], - "format": "uint32", - "minimum": 0.0 - }, - "start_after": { - "type": [ - "string", - "null" - ] - } - } - } - }, - "additionalProperties": false - }, - { - "description": "Only with \"marketing\" extension Returns more metadata on the contract to display in the client: - description, logo, project url, etc. Return type: MarketingInfoResponse", - "type": "object", - "required": [ - "marketing_info" - ], - "properties": { - "marketing_info": { - "type": "object" - } - }, - "additionalProperties": false - }, - { - "description": "Only with \"marketing\" extension Downloads the embedded logo data (if stored on chain). Errors if no logo data is stored for this contract. Return type: DownloadLogoResponse.", - "type": "object", - "required": [ - "download_logo" - ], - "properties": { - "download_logo": { - "type": "object" - } - }, - "additionalProperties": false - } - ] -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/schema/token_info_response.json b/smart-contracts/osmosis/contracts/cw-4626/schema/token_info_response.json deleted file mode 100644 index 9920c841f..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/schema/token_info_response.json +++ /dev/null @@ -1,33 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "TokenInfoResponse", - "type": "object", - "required": [ - "decimals", - "name", - "symbol", - "total_supply" - ], - "properties": { - "decimals": { - "type": "integer", - "format": "uint8", - "minimum": 0.0 - }, - "name": { - "type": "string" - }, - "symbol": { - "type": "string" - }, - "total_supply": { - "$ref": "#/definitions/Uint128" - } - }, - "definitions": { - "Uint128": { - "description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```", - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/schema/total_asset_response.json b/smart-contracts/osmosis/contracts/cw-4626/schema/total_asset_response.json deleted file mode 100644 index fb1624ad4..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/schema/total_asset_response.json +++ /dev/null @@ -1,19 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "TotalAssetResponse", - "type": "object", - "required": [ - "total_managed_assets" - ], - "properties": { - "total_managed_assets": { - "$ref": "#/definitions/Uint128" - } - }, - "definitions": { - "Uint128": { - "description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```", - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/schema/vault_info_response.json b/smart-contracts/osmosis/contracts/cw-4626/schema/vault_info_response.json deleted file mode 100644 index 1a8f605cf..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/schema/vault_info_response.json +++ /dev/null @@ -1,23 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "VaultInfoResponse", - "type": "object", - "required": [ - "reserve_denom", - "total_supply" - ], - "properties": { - "reserve_denom": { - "type": "string" - }, - "total_supply": { - "$ref": "#/definitions/Uint128" - } - }, - "definitions": { - "Uint128": { - "description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```", - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/src/contract.rs b/smart-contracts/osmosis/contracts/cw-4626/src/contract.rs deleted file mode 100644 index 466cfe70f..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/src/contract.rs +++ /dev/null @@ -1,958 +0,0 @@ -#[cfg(not(feature = "library"))] -use cosmwasm_std::entry_point; -use cosmwasm_std::{ - coins, to_json_binary, Binary, Coin, Deps, DepsMut, Env, MessageInfo, Response, StdError, StdResult, - Uint128, -}; -use cw2::set_contract_version; -use cw20::{EmbeddedLogo, Logo, LogoInfo, MarketingInfoResponse}; - -// TODO decide if we want to use deduct allowance -use cw20_base::allowances::{ - execute_decrease_allowance, execute_increase_allowance, execute_send_from, - execute_transfer_from, query_allowance, -}; -use cw20_base::contract::{ - execute_burn, execute_mint, execute_send, execute_transfer, execute_update_marketing, - execute_upload_logo, query_balance, query_download_logo, query_marketing_info, query_minter, - query_token_info, -}; -use cw20_base::enumerable::{query_all_accounts, query_all_allowances}; -use cw20_base::state::{MinterData, TokenInfo, LOGO, MARKETING_INFO, TOKEN_INFO}; -use cw_utils::{must_pay, nonpayable}; - -use quasar_types::curve::DecimalPlaces; -use strategy::contract::{execute_deposit as execute_strategy_deposit, execute_withdraw_request}; - -use crate::error::ContractError; -use crate::msg::{ - AssetResponse, ConvertToAssetsResponse, ConvertToSharesResponse, ExecuteMsg, InstantiateMsg, - MaxDepositResponse, QueryMsg, TotalAssetResponse, VaultInfoResponse, -}; -use crate::state::{VaultInfo, VAULT_CURVE, VAULT_INFO}; - -// version info for migration info -const CONTRACT_NAME: &str = "crates.io:cw-4626"; -const CONTRACT_VERSION: &str = env!("CARGO_PKG_VERSION"); - -const LOGO_SIZE_CAP: usize = 5 * 1024; - -/// Checks if data starts with XML preamble -fn verify_xml_preamble(data: &[u8]) -> Result<(), cw20_base::ContractError> { - // The easiest way to perform this check would be just match on regex, however regex - // compilation is heavy and probably not worth it. - - let preamble = data - .split_inclusive(|c| *c == b'>') - .next() - .ok_or(cw20_base::ContractError::InvalidXmlPreamble {})?; - - const PREFIX: &[u8] = b""; - - if !(preamble.starts_with(PREFIX) && preamble.ends_with(POSTFIX)) { - Err(cw20_base::ContractError::InvalidXmlPreamble {}) - } else { - Ok(()) - } - - // Additionally attributes format could be validated as they are well defined, as well as - // comments presence inside of preable, but it is probably not worth it. -} - -/// Validates XML logo -fn verify_xml_logo(logo: &[u8]) -> Result<(), cw20_base::ContractError> { - verify_xml_preamble(logo)?; - - if logo.len() > LOGO_SIZE_CAP { - Err(cw20_base::ContractError::LogoTooBig {}) - } else { - Ok(()) - } -} - -/// Validates png logo -fn verify_png_logo(logo: &[u8]) -> Result<(), cw20_base::ContractError> { - // PNG header format: - // 0x89 - magic byte, out of ASCII table to fail on 7-bit systems - // "PNG" ascii representation - // [0x0d, 0x0a] - dos style line ending - // 0x1a - dos control character, stop displaying rest of the file - // 0x0a - unix style line ending - const HEADER: [u8; 8] = [0x89, b'P', b'N', b'G', 0x0d, 0x0a, 0x1a, 0x0a]; - if logo.len() > LOGO_SIZE_CAP { - Err(cw20_base::ContractError::LogoTooBig {}) - } else if !logo.starts_with(&HEADER) { - Err(cw20_base::ContractError::InvalidPngHeader {}) - } else { - Ok(()) - } -} - -/// Checks if passed logo is correct, and if not, returns an error -fn verify_logo(logo: &Logo) -> Result<(), cw20_base::ContractError> { - match logo { - Logo::Embedded(EmbeddedLogo::Svg(logo)) => verify_xml_logo(logo), - Logo::Embedded(EmbeddedLogo::Png(logo)) => verify_png_logo(logo), - Logo::Url(_) => Ok(()), // Any reasonable url validation would be regex based, probably not worth it - } -} - -// TODO add the curve of the vault here -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn instantiate( - deps: DepsMut, - env: Env, - _info: MessageInfo, - msg: InstantiateMsg, -) -> Result { - set_contract_version(deps.storage, CONTRACT_NAME, CONTRACT_VERSION)?; - // check valid token info - msg.validate()?; - - // set the share distributor of the vault to the single token distributor - VAULT_INFO.save( - deps.storage, - &VaultInfo { - reserve_denom: msg.reserve_denom.to_string(), - total_supply: msg.reserve_total_supply, - decimals: DecimalPlaces { - supply: msg.supply_decimals as u32, - reserve: msg.reserve_decimals as u32, - }, - }, - )?; - - // store token info - let data = TokenInfo { - name: msg.name, - symbol: msg.symbol, - decimals: msg.supply_decimals, - total_supply: Uint128::zero(), - // set self as minter, so we can properly execute mint and burn - mint: Some(MinterData { - minter: env.contract.address, - cap: None, - }), - }; - TOKEN_INFO.save(deps.storage, &data)?; - - if let Some(marketing) = msg.marketing { - let logo = if let Some(logo) = marketing.logo { - verify_logo(&logo)?; - LOGO.save(deps.storage, &logo)?; - - match logo { - Logo::Url(url) => Some(LogoInfo::Url(url)), - Logo::Embedded(_) => Some(LogoInfo::Embedded), - } - } else { - None - }; - - let data = MarketingInfoResponse { - project: marketing.project, - description: marketing.description, - marketing: marketing - .marketing - .map(|addr| deps.api.addr_validate(&addr)) - .transpose()?, - logo, - }; - MARKETING_INFO.save(deps.storage, &data)?; - } - - VAULT_CURVE.save(deps.storage, &msg.curve_type)?; - - Ok(Response::default()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn execute( - deps: DepsMut, - env: Env, - info: MessageInfo, - msg: ExecuteMsg, -) -> Result { - match msg { - // the vault execute messages - ExecuteMsg::Deposit {} => execute_deposit(deps, env, info), - ExecuteMsg::Withdraw { amount } => execute_withdraw(deps, env, info, amount), - // the cw-20 execute messages - ExecuteMsg::Transfer { recipient, amount } => { - Ok(execute_transfer(deps, env, info, recipient, amount)?) - } - ExecuteMsg::Burn { amount } => Ok(execute_burn(deps, env, info, amount)?), - ExecuteMsg::Send { - contract, - amount, - msg, - } => Ok(execute_send(deps, env, info, contract, amount, msg)?), - ExecuteMsg::Mint { recipient, amount } => { - Ok(execute_mint(deps, env, info, recipient, amount)?) - } - ExecuteMsg::IncreaseAllowance { - spender, - amount, - expires, - } => Ok(execute_increase_allowance( - deps, env, info, spender, amount, expires, - )?), - ExecuteMsg::DecreaseAllowance { - spender, - amount, - expires, - } => Ok(execute_decrease_allowance( - deps, env, info, spender, amount, expires, - )?), - ExecuteMsg::TransferFrom { - owner, - recipient, - amount, - } => Ok(execute_transfer_from( - deps, env, info, owner, recipient, amount, - )?), - ExecuteMsg::BurnFrom { owner, amount } => { - Ok(execute_burn_from(deps, env, info, owner, amount)?) - } - ExecuteMsg::SendFrom { - owner, - contract, - amount, - msg, - } => Ok(execute_send_from( - deps, env, info, owner, contract, amount, msg, - )?), - ExecuteMsg::UpdateMarketing { - project, - description, - marketing, - } => Ok(execute_update_marketing( - deps, - env, - info, - project, - description, - marketing, - )?), - ExecuteMsg::UploadLogo(logo) => Ok(execute_upload_logo(deps, env, info, logo)?), - ExecuteMsg::Receive { .. } => { - todo!() - } - } -} - -pub fn execute_deposit( - mut deps: DepsMut, - env: Env, - info: MessageInfo, -) -> Result { - // do all reasonable checks - let vault_info = VAULT_INFO.load(deps.storage)?; - let curve = VAULT_CURVE.load(deps.storage)?; - let curve_fn = curve.to_curve_fn(); - let curve = curve_fn(vault_info.decimals); - // only accept one token, change this for multi asset - - // TODO this is a hardcoded amount of a linear 1-1 price, we need to change this with customizable curves - let amount = must_pay(&info, vault_info.reserve_denom.as_str())?; - - // deposit the funds into the strategy - // TODO remove clones and decide whether we directly pass info - execute_strategy_deposit(deps.branch(), env.clone(), info.clone())?; - - let shares = curve.deposit(&amount); - // call into cw20-base to mint the token, call as self as no one else is allowed - let sub_info = MessageInfo { - sender: env.contract.address.clone(), - funds: vec![], - }; - // exchange all the free balance for shares, mint shares of the underlying cw-20 - execute_mint(deps, env, sub_info, info.sender.to_string(), shares)?; - - let res = Response::new() - .add_attribute("action", "buy") - .add_attribute("from", info.sender) - .add_attribute("reserve", amount) - .add_attribute("shares", shares); - Ok(res) -} - -// TODO decide on whether to add owner here for allowance support -pub fn execute_withdraw( - mut deps: DepsMut, - env: Env, - info: MessageInfo, - amount: Option, -) -> Result { - // check that no funds are sent with the withdraw - nonpayable(&info)?; - - // TODO make this a bit nicer with a helper - let vault_info = VAULT_INFO.load(deps.storage)?; - let curve_type = VAULT_CURVE.load(deps.storage)?; - let curve_fn = curve_type.to_curve_fn(); - let curve = curve_fn(vault_info.decimals); - - // if amount is None, sell all shares of sender - let shares = if let Some(value) = amount { - value - } else { - let addr = &info.sender.to_string(); - query_balance(deps.as_ref(), addr.clone())?.balance - }; - - let withdraw_amount = curve.withdraw(&shares); - - // TODO here we need to withdraw from the strategy, if the strategy has the funds, we can simply withdraw - // if the withdraw is queued, do we already burn and mint again if we need to reverse or not burn at all? - let withdraw_res = execute_withdraw_request( - deps.branch(), - env.clone(), - info.clone(), - info.sender.to_string(), - vault_info.reserve_denom, - withdraw_amount, - )?; - - // execute_burn will error if the sender does not have enough tokens to burn - // TODO make sure that we can discard the result of execute_burn - execute_burn(deps.branch(), env, info.clone(), shares)?; - - // we have created the Send bankmessage in the strategy contract, thus we pass res.messages here - let res = Response::new() - .add_submessages(withdraw_res.messages) - .add_attributes(withdraw_res.attributes) - .add_attribute("action", "sell") - .add_attribute("to", &info.sender) - .add_attribute("amount", shares); - Ok(res) -} - -// TODO implement burn_from to support the cw20 allowances -// execute_burn_from tries to burn the shares from owner if sender has an allowance over those tokens -// in the cw20 contract. The underlying reserve tokens should be returned to who? probably the sender -// and not the owner -pub fn execute_burn_from( - _deps: DepsMut, - _env: Env, - _info: MessageInfo, - _owner: String, - _amount: Uint128, -) -> Result { - todo!() -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn query(deps: Deps, env: Env, msg: QueryMsg) -> StdResult { - match msg { - QueryMsg::Balance { address } => to_json_binary(&query_balance(deps, address)?), - QueryMsg::TokenInfo {} => to_json_binary(&query_token_info(deps)?), - QueryMsg::Minter {} => to_json_binary(&query_minter(deps)?), - QueryMsg::Allowance { owner, spender } => { - to_json_binary(&query_allowance(deps, owner, spender)?) - } - QueryMsg::AllAllowances { - owner, - start_after, - limit, - } => to_json_binary(&query_all_allowances(deps, owner, start_after, limit)?), - QueryMsg::AllAccounts { start_after, limit } => { - to_json_binary(&query_all_accounts(deps, start_after, limit)?) - } - QueryMsg::MarketingInfo {} => to_json_binary(&query_marketing_info(deps)?), - QueryMsg::DownloadLogo {} => to_json_binary(&query_download_logo(deps)?), - QueryMsg::Asset {} => to_json_binary(&query_asset(deps)?), - QueryMsg::TotalAssets {} => to_json_binary(&query_total_assets(deps, env)?), - QueryMsg::ConvertToShares { assets } => to_json_binary(&query_convert_to_shares(deps, assets)?), - QueryMsg::ConvertToAssets { shares } => to_json_binary(&query_convert_to_assets(deps, shares)?), - QueryMsg::MaxDeposit { - receiver: _receiver, - } => { - // max deposit needs to check the underlying cw-20 token for the maximum supply, convert that - // to the amount - to_json_binary(&query_max_deposit(deps)?) - } - QueryMsg::PreviewDeposit { .. } => { - todo!() - } - QueryMsg::MaxMint { .. } => { - todo!() - } - QueryMsg::PreviewMint { .. } => { - todo!() - } - QueryMsg::MaxWithdraw { .. } => { - todo!() - } - QueryMsg::PreviewWithdraw { .. } => { - todo!() - } - QueryMsg::MaxRedeem { .. } => { - todo!() - } - QueryMsg::PreviewRedeem { .. } => { - todo!() - } - QueryMsg::VaultInfo {} => to_json_binary(&query_vault_info(deps)?), - } -} - -pub fn query_asset(deps: Deps) -> StdResult { - let vault_info = VAULT_INFO.load(deps.storage)?; - Ok(AssetResponse { - denom: vault_info.reserve_denom, - }) -} - -pub fn query_total_assets(deps: Deps, env: Env) -> StdResult { - let vault_info = VAULT_INFO.load(deps.storage)?; - let balance = deps - .querier - .query_balance(env.contract.address, vault_info.reserve_denom)?; - Ok(TotalAssetResponse { - total_managed_assets: balance.amount, - }) -} - -pub fn query_convert_to_shares( - deps: Deps, - assets: Vec, -) -> StdResult { - let vault_info = VAULT_INFO.load(deps.storage)?; - let curve_type = VAULT_CURVE.load(deps.storage)?; - let curve_fn = curve_type.to_curve_fn(); - let curve = curve_fn(vault_info.decimals); - - // error on wrong amount of assets - if assets.len() != 1 { - return Err(StdError::generic_err("Query only supports one asset")); - } - - // error on wrong denom - if assets[0].denom != vault_info.reserve_denom { - return Err(StdError::generic_err(format!( - "Expected {} instead of {}", - vault_info.reserve_denom, assets[0].denom - ))); - } - - let shares = curve.deposit(&assets[0].amount); - - Ok(ConvertToSharesResponse { amount: shares }) -} - -pub fn query_convert_to_assets(deps: Deps, shares: Uint128) -> StdResult { - let vault_info = VAULT_INFO.load(deps.storage)?; - let curve_type = VAULT_CURVE.load(deps.storage)?; - let curve_fn = curve_type.to_curve_fn(); - let curve = curve_fn(vault_info.decimals); - - let amount = curve.withdraw(&shares); - Ok(ConvertToAssetsResponse { - assets: coins(amount.u128(), vault_info.reserve_denom), - }) -} - -fn query_max_deposit(deps: Deps) -> StdResult { - let vault_info = VAULT_INFO.load(deps.storage)?; - let curve_type = VAULT_CURVE.load(deps.storage)?; - let curve_fn = curve_type.to_curve_fn(); - let curve = curve_fn(vault_info.decimals); - - let minter = query_minter(deps)?; - let free_tokens: Uint128; - if let Some(min) = minter { - // calculate the outstanding tokens - let accounts = query_all_accounts(deps, None, None)?; - let mut outstanding = Uint128::zero(); - for acc in accounts.accounts { - let balance = query_balance(deps, acc)?; - outstanding = outstanding.checked_add(balance.balance)? - } - // if we have a cap, calculate the difference between current outstanding supply and cap - if min.cap.is_some() { - free_tokens = min.cap.unwrap() - outstanding; - } else { - // if there is no cap, what do we do? We can return Uint128::MAX - outstanding tokens - // This does create an issue with normalizing so that probably is not best. - // thus we have to return None here to indicate the vault has no cap - return Ok(MaxDepositResponse { max_assets: None }); - } - } else { - return Err(StdError::generic_err("no minter found in vault")); - } - // in order to get this many shares, we calculate F^-1(X), or withdraw, since withdraw and deposit are reversible - let asset = curve.deposit(&free_tokens); - Ok(MaxDepositResponse { - max_assets: Some(coins(asset.u128(), vault_info.reserve_denom)), - }) -} - -pub fn query_vault_info(deps: Deps) -> StdResult { - let info = VAULT_INFO.load(deps.storage)?; - let res = VaultInfoResponse { - reserve_denom: info.reserve_denom, - total_supply: info.total_supply, - }; - Ok(res) -} - -#[cfg(test)] -mod tests { - use super::*; - use cosmwasm_std::testing::{mock_dependencies, mock_env, mock_info}; - use cosmwasm_std::{ - coin, BankMsg, Decimal, OverflowError, OverflowOperation, StdError, SubMsg, - }; - use cw_utils::PaymentError; - use quasar_types::curve::CurveType; - use std::borrow::BorrowMut; - - const DENOM: &str = "satoshi"; - const CREATOR: &str = "creator"; - const INVESTOR: &str = "investor"; - const BUYER: &str = "buyer"; - - fn default_instantiate( - supply_decimals: u8, - reserve_decimals: u8, - reserve_supply: Uint128, - ) -> InstantiateMsg { - InstantiateMsg { - name: "Bonded".to_string(), - symbol: "EPOXY".to_string(), - reserve_denom: DENOM.to_string(), - reserve_total_supply: reserve_supply, - reserve_decimals, - supply_decimals, - initial_balances: vec![], - // initiate with a constant 1 to 1 curve - curve_type: CurveType::Constant { - value: Uint128::new(10), - scale: 1, - }, - mint: None, - marketing: None, - } - } - - fn get_balance>(deps: Deps, addr: U) -> Uint128 { - query_balance(deps, addr.into()).unwrap().balance - } - - fn setup_test( - deps: DepsMut, - supply_decimals: u8, - reserve_decimals: u8, - reserve_supply: Uint128, - ) { - // this matches `constant_curve` test case from curves.rs - let creator = String::from(CREATOR); - let msg = default_instantiate(supply_decimals, reserve_decimals, reserve_supply); - let info = mock_info(&creator, &[]); - - // make sure we can instantiate with this - let res = instantiate(deps, mock_env(), info, msg).unwrap(); - assert_eq!(0, res.messages.len()); - } - - fn setup_test_with_deposit( - mut deps: DepsMut, - env: Env, - supply_decimals: u8, - reserve_decimals: u8, - reserve_supply: Uint128, - deposited_funds: u128, - received_shares: u128, - ) { - // this matches `constant_curve` test case from curves.rs - let creator = String::from(CREATOR); - let msg = default_instantiate(supply_decimals, reserve_decimals, reserve_supply); - let info = mock_info(&creator, &[]); - - // make sure we can instantiate with this - let res = instantiate(deps.branch(), env.clone(), info, msg).unwrap(); - assert_eq!(0, res.messages.len()); - - let alice: &str = "alice"; - let bob: &str = "bobbyb"; - let carl: &str = "carl"; - - // setup_test() defaults to a 1-1 curve - // bob buys some shares, spends 45 9 decimal coins (45_000_000_000) and receives 45 6 decimal (45_000_000) shares - let info = mock_info(bob, &coins(deposited_funds, DENOM)); - let buy = ExecuteMsg::Deposit {}; - execute(deps.branch(), env, info, buy).unwrap(); - - // check that bob has the shares - assert_eq!( - get_balance(deps.as_ref(), bob), - Uint128::new(received_shares) - ); - } - - #[test] - fn proper_instantiation() { - let mut deps = mock_dependencies(); - - // this matches `linear_curve` test case from curves.rs - let creator = String::from("creator"); - let msg = default_instantiate(2, 8, Uint128::MAX); - let info = mock_info(&creator, &[]); - - // make sure we can instantiate with this - let res = instantiate(deps.as_mut(), mock_env(), info, msg.clone()).unwrap(); - assert_eq!(0, res.messages.len()); - - // token info is proper - let token = query_token_info(deps.as_ref()).unwrap(); - assert_eq!(&token.name, &msg.name); - assert_eq!(&token.symbol, &msg.symbol); - assert_eq!(token.decimals, 2); - assert_eq!(token.total_supply, Uint128::zero()); - // - // // curve state is sensible - // let state = query_curve_info(deps.as_ref(), curve_type.to_curve_fn()).unwrap(); - let state = VAULT_INFO.load(deps.storage.borrow_mut()).unwrap(); - assert_eq!(state.total_supply, Uint128::MAX); - assert_eq!(state.reserve_denom.as_str(), DENOM); - // spot price 0 as supply is 0 - // assert_eq!(state.spot_price, Decimal::zero()); - - // no balance - assert_eq!(get_balance(deps.as_ref(), &creator), Uint128::zero()); - } - - #[test] - fn deposit_works() { - let mut deps = mock_dependencies(); - setup_test(deps.as_mut(), 9, 6, Uint128::MAX); - - let alice: &str = "alice"; - let bob: &str = "bobbyb"; - let carl: &str = "carl"; - - // setup_test() defaults to a 1-1 curve - // bob buys some shares, spends 45 9 decimal coins (45_000_000_000) and receives 45 6 decimal (45_000_000) shares - let info = mock_info(bob, &coins(45_000_000_000, DENOM)); - let buy = ExecuteMsg::Deposit {}; - execute(deps.as_mut(), mock_env(), info, buy).unwrap(); - - // check that bob has the shares - assert_eq!(get_balance(deps.as_ref(), bob), Uint128::new(45_000_000)); - } - - #[test] - fn deposit_needs_right_denom() { - let mut deps = mock_dependencies(); - setup_test(deps.as_mut(), 9, 6, Uint128::MAX); - - let alice: &str = "alice"; - let bob: &str = "bobbyb"; - let carl: &str = "carl"; - - // setup_test() defaults to a 1-1 curve - // bob buys some shares, spends 45 9 decimal coins (45_000_000_000) and receives 45 6 decimal (45_000_000) shares - let info = mock_info(bob, &coins(45_000_000_000, "wrong-denom")); - let buy = ExecuteMsg::Deposit {}; - execute(deps.as_mut(), mock_env(), info, buy).unwrap_err(); - - // check that bob has no shares - assert_eq!(get_balance(deps.as_ref(), bob), Uint128::new(0)); - } - - fn deposit_needs_funds() { - let mut deps = mock_dependencies(); - setup_test(deps.as_mut(), 9, 6, Uint128::MAX); - - let alice: &str = "alice"; - let bob: &str = "bobbyb"; - let carl: &str = "carl"; - - // setup_test() defaults to a 1-1 curve - // bob tries to buy some shares without any funds - let info = mock_info(bob, &[]); - let buy = ExecuteMsg::Deposit {}; - execute(deps.as_mut(), mock_env(), info, buy).unwrap_err(); - - // check that bob has no shares - assert_eq!(get_balance(deps.as_ref(), bob), Uint128::new(0)); - } - - #[test] - fn withdraw_works() { - let mut deps = mock_dependencies(); - // setup_test() defaults to a 1-1 curve - // bob buys some shares, spends 45 9 decimal coins (45_000_000_000) and receives 45 6 decimal (45_000_000) shares - setup_test_with_deposit( - deps.as_mut(), - mock_env(), - 9, - 6, - Uint128::MAX, - 45_000_000_000, - 45_000_000, - ); - - deps.querier - .update_balance(mock_env().contract.address, coins(45_000_000_000, DENOM)); - - let alice: &str = "alice"; - let bob: &str = "bobbyb"; - let carl: &str = "carl"; - - // check that bob has the shares - assert_eq!(get_balance(deps.as_ref(), bob), Uint128::new(45_000_000)); - - let info = mock_info(bob, &[]); - // withdraw all shares - let sell = ExecuteMsg::Withdraw { amount: None }; - let res = execute(deps.as_mut(), mock_env(), info, sell).unwrap(); - - // check that bob's balance is completely gone - assert_eq!(get_balance(deps.as_ref(), bob), Uint128::zero()); - - assert_eq!(res.messages.len(), 1); - // check that bob received a bankmessage containing funds - assert_eq!( - res.messages[0], - SubMsg::new(BankMsg::Send { - to_address: bob.to_string(), - amount: coins(45_000_000_000, DENOM) - }) - ) - } - - #[test] - fn cannot_withdraw_too_many_funds() { - let mut deps = mock_dependencies(); - setup_test_with_deposit( - deps.as_mut(), - mock_env(), - 9, - 6, - Uint128::MAX, - 45_000_000_000, - 45_000_000, - ); - - let alice: &str = "alice"; - let bob: &str = "bobbyb"; - let carl: &str = "carl"; - - // check that bob has the shares - assert_eq!(get_balance(deps.as_ref(), bob), Uint128::new(45_000_000)); - - let info = mock_info(bob, &[]); - // withdraw too many shares - let sell = ExecuteMsg::Withdraw { - amount: Some(Uint128::new(999_000_000)), - }; - execute(deps.as_mut(), mock_env(), info, sell).unwrap_err(); - - // check that bob's balance has not changed - assert_eq!(get_balance(deps.as_ref(), bob), Uint128::new(45_000_000)) - } - - #[test] - fn cannot_send_funds_with_withdraw() { - let mut deps = mock_dependencies(); - setup_test_with_deposit( - deps.as_mut(), - mock_env(), - 9, - 6, - Uint128::MAX, - 45_000_000_000, - 45_000_000, - ); - - let alice: &str = "alice"; - let bob: &str = "bobbyb"; - let carl: &str = "carl"; - - // check that bob has the shares - assert_eq!(get_balance(deps.as_ref(), bob), Uint128::new(45_000_000)); - - // check that bob has the shares - assert_eq!(get_balance(deps.as_ref(), bob), Uint128::new(45_000_000)); - - let info = mock_info(bob, &coins(45_000_000_000, DENOM)); - // withdraw too many shares - let sell = ExecuteMsg::Withdraw { - amount: Some(Uint128::new(999_000_000)), - }; - execute(deps.as_mut(), mock_env(), info, sell).unwrap_err(); - - // check that bob's balance has not changed - assert_eq!(get_balance(deps.as_ref(), bob), Uint128::new(45_000_000)) - } - - #[test] - fn cw20_imports_work() { - let mut deps = mock_dependencies(); - setup_test(deps.as_mut(), 9, 6, Uint128::MAX); - - let alice: &str = "alice"; - let bob: &str = "bobby"; - let carl: &str = "carl"; - - // setup_test() defaults to a 1-1 curve, the supply has 9 decimals and the shares 6 decimals - // spend 45_000_000 uatom for 45_000 shares - let info = mock_info(bob, &coins(45_000_000, DENOM)); - let buy = ExecuteMsg::Deposit {}; - execute(deps.as_mut(), mock_env(), info, buy).unwrap(); - - // check balances - assert_eq!(get_balance(deps.as_ref(), bob), Uint128::new(45_000)); - assert_eq!(get_balance(deps.as_ref(), carl), Uint128::zero()); - - // send coins to carl - let bob_info = mock_info(bob, &[]); - let transfer = ExecuteMsg::Transfer { - recipient: carl.into(), - amount: Uint128::new(20_000), - }; - execute(deps.as_mut(), mock_env(), bob_info.clone(), transfer).unwrap(); - assert_eq!(get_balance(deps.as_ref(), bob), Uint128::new(25_000)); - assert_eq!(get_balance(deps.as_ref(), carl), Uint128::new(20_000)); - - // allow alice - let allow = ExecuteMsg::IncreaseAllowance { - spender: alice.into(), - amount: Uint128::new(10_000), - expires: None, - }; - execute(deps.as_mut(), mock_env(), bob_info, allow).unwrap(); - assert_eq!(get_balance(deps.as_ref(), bob), Uint128::new(25_000)); - assert_eq!(get_balance(deps.as_ref(), alice), Uint128::zero()); - assert_eq!( - query_allowance(deps.as_ref(), bob.into(), alice.into()) - .unwrap() - .allowance, - Uint128::new(10_000) - ); - - // alice takes some for herself - let self_pay = ExecuteMsg::TransferFrom { - owner: bob.into(), - recipient: alice.into(), - amount: Uint128::new(5_000), - }; - let alice_info = mock_info(alice, &[]); - execute(deps.as_mut(), mock_env(), alice_info, self_pay).unwrap(); - assert_eq!(get_balance(deps.as_ref(), bob), Uint128::new(20_000)); - assert_eq!(get_balance(deps.as_ref(), alice), Uint128::new(5_000)); - assert_eq!(get_balance(deps.as_ref(), carl), Uint128::new(20_000)); - assert_eq!( - query_allowance(deps.as_ref(), bob.into(), alice.into()) - .unwrap() - .allowance, - Uint128::new(5_000) - ); - - // test burn from works properly (burn tested in burning_sends_reserve) - // cannot burn more than they have - - // TODO burn_from needs to be implemented and these tests added back here - - // let info = mock_info(alice, &[]); - // let burn_from = ExecuteMsg::BurnFrom { - // owner: bob.into(), - // amount: Uint128::new(3_300_000), - // }; - // let err = execute(deps.as_mut(), mock_env(), info, burn_from).unwrap_err(); - // assert_eq!( - // err, - // ContractError::Base(cw20_base::ContractError::Std(StdError::overflow( - // OverflowError::new(OverflowOperation::Sub, 5000 as u128, 3300000 as u128) - // ))) - // ); - - // burn 1_500 EPOXY to get back 1_500 DENOM (constant curve) - // let info = mock_info(alice, &[]); - // let burn_from = ExecuteMsg::BurnFrom { - // owner: bob.into(), - // amount: Uint128::new(1_500), - // }; - // let res = execute(deps.as_mut(), mock_env(), info, burn_from).unwrap(); - - // bob balance is lower, not alice - // assert_eq!(get_balance(deps.as_ref(), alice), Uint128::new(5000)); - // assert_eq!(get_balance(deps.as_ref(), bob), Uint128::new(18500)); - - // ensure alice got our money back - // TODO, currently this is not supported, we will need to support it, in cw-20 bonding curve, this is routed to sell from - // assert_eq!(1, res.messages.len()); - // assert_eq!( - // &res.messages[0], - // &SubMsg::new(BankMsg::Send { - // to_address: alice.into(), - // amount: coins(1_500, DENOM), - // }) - // ); - } - - #[test] - fn query_asset_works() { - let mut deps = mock_dependencies(); - setup_test(deps.as_mut(), 9, 6, Uint128::MAX); - - let asset = query_asset(deps.as_ref()).unwrap(); - assert_eq!(asset.denom, "satoshi".to_string()) - } - - #[test] - fn query_total_assets_works() { - let mut deps = mock_dependencies(); - setup_test(deps.as_mut(), 9, 6, Uint128::MAX); - - // check that total_assets is zero upon instantiation - let total_assets = query_total_assets(deps.as_ref(), mock_env()).unwrap(); - assert_eq!(total_assets.total_managed_assets, Uint128::new(0)); - - // deposit some funds in different accounts and check assets after each deposit - let alice: &str = "alice"; - let bob: &str = "bobbyb"; - let carl: &str = "carl"; - - deps.querier - .update_balance(mock_env().contract.address, coins(45_000_000_000, DENOM)); - - let total_assets = query_total_assets(deps.as_ref(), mock_env()).unwrap(); - assert_eq!( - total_assets.total_managed_assets, - Uint128::new(45_000_000_000) - ); - } - - #[test] - fn query_convert_to_shares_works() { - let mut deps = mock_dependencies(); - setup_test(deps.as_mut(), 9, 6, Uint128::MAX); - - let response = - query_convert_to_shares(deps.as_ref(), coins(45_000_000_000, DENOM)).unwrap(); - assert_eq!(response.amount, Uint128::new(45_000_000)) - } - - #[test] - fn query_convert_to_assets_works() { - let mut deps = mock_dependencies(); - setup_test(deps.as_mut(), 9, 6, Uint128::MAX); - - let response = query_convert_to_assets(deps.as_ref(), Uint128::new(45_000_000)).unwrap(); - assert_eq!(response.assets, coins(45_000_000_000, DENOM)) - } - - #[test] - fn query_max_deposit_works() { - let mut deps = mock_dependencies(); - setup_test(deps.as_mut(), 9, 6, Uint128::MAX); - - let response = query_max_deposit(deps.as_ref()).unwrap(); - // the vault has no cap to minting by default, thus we expect None - assert_eq!(response.max_assets, None) - } -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/src/error.rs b/smart-contracts/osmosis/contracts/cw-4626/src/error.rs deleted file mode 100644 index 5cb8514fb..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/src/error.rs +++ /dev/null @@ -1,17 +0,0 @@ -use cosmwasm_std::StdError; -use thiserror::Error; - -#[derive(Error, Debug, PartialEq)] -pub enum ContractError { - #[error("{0}")] - Std(#[from] StdError), - - #[error("{0}")] - Base(#[from] cw20_base::ContractError), - - #[error("{0}")] - PaymentError(#[from] cw_utils::PaymentError), - - #[error("{0}")] - StrategyError(#[from] strategy::error::ContractError), -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/src/lib.rs b/smart-contracts/osmosis/contracts/cw-4626/src/lib.rs deleted file mode 100644 index dfedc9dc6..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/src/lib.rs +++ /dev/null @@ -1,6 +0,0 @@ -pub mod contract; -mod error; -pub mod msg; -pub mod state; - -pub use crate::error::ContractError; diff --git a/smart-contracts/osmosis/contracts/cw-4626/src/msg.rs b/smart-contracts/osmosis/contracts/cw-4626/src/msg.rs deleted file mode 100644 index 554e3bb48..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/src/msg.rs +++ /dev/null @@ -1,283 +0,0 @@ -use cosmwasm_std::{Binary, Coin, StdError, StdResult, Uint128, Uint256}; -use cw20::{Cw20Coin, Logo, MinterResponse}; -use cw_utils::Expiration; -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -#[derive(Serialize, Deserialize, JsonSchema, Debug, Clone, PartialEq)] -pub struct InstantiateMarketingInfo { - pub project: Option, - pub description: Option, - pub marketing: Option, - pub logo: Option, -} - -#[derive(Serialize, Deserialize, JsonSchema, Debug, Clone, PartialEq)] -pub struct InstantiateMsg { - pub name: String, - pub symbol: String, - // the token accepted by the contract. - pub reserve_denom: String, - // the total amount of tokens that can be put in the reserve. - // TODO change to Option to allow for no cap on reserve supply - pub reserve_total_supply: Uint128, - pub reserve_decimals: u8, - // supply is the amount of tokens this vault has issued. - // supply_decimals is the amount of decimals of the supply token - pub supply_decimals: u8, - // TODO should this be an Option to not need the field in the initialization json - pub initial_balances: Vec, - pub mint: Option, - pub marketing: Option, - - pub curve_type: quasar_types::curve::CurveType, -} - -impl InstantiateMsg { - pub fn get_cap(&self) -> Option { - self.mint.as_ref().and_then(|v| v.cap) - } - - pub fn validate(&self) -> StdResult<()> { - // Check name, symbol, decimals - if !is_valid_name(&self.name) { - return Err(StdError::generic_err( - "Name is not in the expected format (3-50 UTF-8 bytes)", - )); - } - if !is_valid_symbol(&self.symbol) { - return Err(StdError::generic_err( - "Ticker symbol is not in expected format [a-zA-Z\\-]{3,12}", - )); - } - if self.reserve_denom.is_empty() { - return Err(StdError::generic_err("No reserve denom")); - } - if self.supply_decimals > 18 { - return Err(StdError::generic_err("Decimals must not exceed 18")); - } - Ok(()) - } -} - -fn is_valid_name(name: &str) -> bool { - let bytes = name.as_bytes(); - if bytes.len() < 3 || bytes.len() > 50 { - return false; - } - true -} - -fn is_valid_symbol(symbol: &str) -> bool { - let bytes = symbol.as_bytes(); - if bytes.len() < 3 || bytes.len() > 12 { - return false; - } - for byte in bytes.iter() { - if (*byte != 45) && (*byte < 65 || *byte > 90) && (*byte < 97 || *byte > 122) { - return false; - } - } - true -} - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -#[serde(rename_all = "snake_case")] -pub enum QueryMsg { - /// Returns the denomination of the vaults reserve token - /// Return type: AssetResponse - Asset {}, - /// Returns the total amount of underlying reserve assets that is managed by the vault - /// Return type: TotalAssetsResponse - TotalAssets {}, - /// Returns the current balance of shares of the given address, 0 if unset. - /// Return type: BalanceResponse. - Balance { address: String }, - /// Returns the amount of shares the vault would exchange for the underlying reserve asset, in the ideal scenario - /// Return type: TODO - ConvertToShares { assets: Vec }, - /// Returns the amount of assets the vault would exchange for the amount of shares, in the ideal scenario - /// Return type: ConvertToSharesResponse - ConvertToAssets { shares: Uint128 }, - /// Returns the maximum amount of the underlying asset that can be deposited into the Vault for the receiver, through a deposit call. - /// If None is returned in the response, the vault does not have a cap - /// Return type: ConvertToAssetsResponse - MaxDeposit { receiver: String }, - /// Allows an on-chain or off-chain user to simulate the effects of their deposit at the current block, given current on-chain conditions. - /// Return type: TODO - PreviewDeposit { assets: Uint256 }, - /// Return the maximum amount of shares that can be minted from the vault for the receiver, through a mint call - /// Return type: TODO - MaxMint { receiver: String }, - /// Allows an on-chain or off-chain user to simulate the effects of their mint at the current block, given current on-chain conditions. - /// Return type: TODO - PreviewMint { shares: Uint256 }, - /// Returns the maximum amount of the underlying asset that can be withdrawn from the owner balance in the Vault, through a withdraw call. - /// Return type: TODO - MaxWithdraw { owner: String }, - /// Allows an on-chain or off-chain user to simulate the effects of their withdrawal at the current block, given current on-chain conditions. - /// Return type: TODO - PreviewWithdraw { assets: Uint256 }, - /// Returns the maximum amount of Vault shares that can be redeemed from the owner balance in the Vault, through a redeem call. - MaxRedeem { owner: String }, - /// Allows an on-chain or off-chain user to simulate the effects of their redeemption at the current block, given current on-chain conditions. - PreviewRedeem { shares: Uint256 }, - /// Returns metadata on the contract - name, decimals, supply, etc. - /// Return type: TokenInfoResponse. - TokenInfo {}, - /// Returns the metadata on the vault - whitelist, etc. - /// Return type: VaultInfoResponse. - VaultInfo {}, - /// Only with "mintable" extension. - /// Returns who can mint and the hard cap on maximum tokens after minting. - /// Return type: MinterResponse. - Minter {}, - /// Only with "allowance" extension. - /// Returns how much spender can use from owner account, 0 if unset. - /// Return type: AllowanceResponse. - Allowance { owner: String, spender: String }, - /// Only with "enumerable" extension (and "allowances") - /// Returns all allowances this owner has approved. Supports pagination. - /// Return type: AllAllowancesResponse. - AllAllowances { - owner: String, - start_after: Option, - limit: Option, - }, - /// Only with "enumerable" extension - /// Returns all accounts that have balances. Supports pagination. - /// Return type: AllAccountsResponse. - AllAccounts { - start_after: Option, - limit: Option, - }, - /// Only with "marketing" extension - /// Returns more metadata on the contract to display in the client: - /// - description, logo, project url, etc. - /// Return type: MarketingInfoResponse - MarketingInfo {}, - /// Only with "marketing" extension - /// Downloads the embedded logo data (if stored on chain). Errors if no logo data is stored for this - /// contract. - /// Return type: DownloadLogoResponse. - DownloadLogo {}, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub struct AssetResponse { - pub denom: String, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub struct TotalAssetResponse { - pub total_managed_assets: Uint128, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub struct ConvertToSharesResponse { - pub amount: Uint128, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub struct ConvertToAssetsResponse { - pub assets: Vec, -} - -/// A None response indicates that the vault has no cap an thus no max deposit -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub struct MaxDepositResponse { - pub max_assets: Option>, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub struct VaultInfoResponse { - pub reserve_denom: String, - pub total_supply: Uint128, -} - -// we give our own ExecuteMsg instead of the cw-20 executeMsg so we can easily extend it -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub enum ExecuteMsg { - /// Deposits assets and mints shares - Deposit {}, - /// Burns shares from owner and sends exactly assets of underlying tokens to receiver. - /// If amount is None, all shares are sold - Withdraw { amount: Option }, - - //cw20-base messages - /// Transfer is a base message to move tokens to another account without triggering actions - Transfer { recipient: String, amount: Uint128 }, - /// Burn is a base message to destroy tokens forever - Burn { amount: Uint128 }, - /// Send is a base message to transfer tokens to a contract and trigger an action - /// on the receiving contract. - Send { - contract: String, - amount: Uint128, - msg: Binary, - }, - /// Receive is a base message to receive tokens from another contract and trigger an action on - /// this contract. the address of the contract is stored in env.sender. Sender should match the - /// contract we expect to handle. Sender is the original account moving the tokens. The vault - /// needs to support sender if we expect people to be able to move their shares out of the vault. - Receive { - sender: String, - amount: Uint128, - msg: Binary, - }, - /// Only with "approval" extension. Allows spender to access an additional amount tokens - /// from the owner's (env.sender) account. If expires is Some(), overwrites current allowance - /// expiration with this one. - IncreaseAllowance { - spender: String, - amount: Uint128, - expires: Option, - }, - /// Only with "approval" extension. Lowers the spender's access of tokens - /// from the owner's (env.sender) account by amount. If expires is Some(), overwrites current - /// allowance expiration with this one. - DecreaseAllowance { - spender: String, - amount: Uint128, - expires: Option, - }, - /// Only with "approval" extension. Transfers amount tokens from owner -> recipient - /// if `env.sender` has sufficient pre-approval. - TransferFrom { - owner: String, - recipient: String, - amount: Uint128, - }, - /// Only with "approval" extension. Sends amount tokens from owner -> contract - /// if `env.sender` has sufficient pre-approval. - SendFrom { - owner: String, - contract: String, - amount: Uint128, - msg: Binary, - }, - /// Only with "approval" extension. Destroys tokens forever - BurnFrom { owner: String, amount: Uint128 }, - /// Only with the "mintable" extension. If authorized, creates amount new tokens - /// and adds to the recipient balance. - Mint { recipient: String, amount: Uint128 }, - /// Only with the "marketing" extension. If authorized, updates marketing metadata. - /// Setting None/null for any of these will leave it unchanged. - /// Setting Some("") will clear this field on the contract storage - UpdateMarketing { - /// A URL pointing to the project behind this token. - project: Option, - /// A longer description of the token and it's utility. Designed for tooltips or such - description: Option, - /// The address (if any) who can update this data structure - marketing: Option, - }, - /// If set as the "marketing" role on the contract, upload a new URL, SVG, or PNG for the token - UploadLogo(Logo), -} diff --git a/smart-contracts/osmosis/contracts/cw-4626/src/state.rs b/smart-contracts/osmosis/contracts/cw-4626/src/state.rs deleted file mode 100644 index a2e198cd3..000000000 --- a/smart-contracts/osmosis/contracts/cw-4626/src/state.rs +++ /dev/null @@ -1,22 +0,0 @@ -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; -use std::fmt::Debug; - -use cosmwasm_std::Uint128; -use cw_storage_plus::Item; -use quasar_types::curve::{CurveType, DecimalPlaces}; - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub struct VaultInfo { - // reserve_denom is the denomination accepted by this vault. If the accepted token should be - // a cw20-token, one should wrap the token using the tokenfactory and the wrapping contract - pub reserve_denom: String, - // total_supply is the total supply of shares this contract - pub total_supply: Uint128, - // the decimals of the vault supply and reserve - pub decimals: DecimalPlaces, -} - -pub const VAULT_INFO: Item = Item::new("vault_info"); -pub const VAULT_CURVE: Item = Item::new("vault_curve"); diff --git a/smart-contracts/osmosis/contracts/ibc-transfer/.cargo/config b/smart-contracts/osmosis/contracts/ibc-transfer/.cargo/config deleted file mode 100644 index 7c115322a..000000000 --- a/smart-contracts/osmosis/contracts/ibc-transfer/.cargo/config +++ /dev/null @@ -1,6 +0,0 @@ -[alias] -wasm = "build --release --target wasm32-unknown-unknown" -wasm-debug = "build --target wasm32-unknown-unknown" -unit-test = "test --lib --features backtraces" -integration-test = "test --test integration" -schema = "run --example schema" diff --git a/smart-contracts/osmosis/contracts/ibc-transfer/Cargo.toml b/smart-contracts/osmosis/contracts/ibc-transfer/Cargo.toml deleted file mode 100644 index 99618bb5b..000000000 --- a/smart-contracts/osmosis/contracts/ibc-transfer/Cargo.toml +++ /dev/null @@ -1,54 +0,0 @@ -[package] -name = "ibc_transfer" -version = "0.1.0" -edition = "2021" - - -exclude = [ - # Those files are rust-optimizer artifacts. You might want to commit them for convenience but they should not be part of the source code publication. - "contract.wasm", - "hash.txt", -] - -# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html - -[lib] -crate-type = ["cdylib", "rlib"] - -[profile.release] -opt-level = 3 -debug = false -rpath = false -lto = true -debug-assertions = false -codegen-units = 1 -panic = 'abort' -incremental = false -overflow-checks = true - -[features] -# for quicker tests, cargo test --lib -# for more explicit tests, cargo test --features=backtraces -backtraces = ["cosmwasm-std/backtraces"] -library = [] - -[dependencies] -cosmwasm-std = { workspace = true } -osmosis-std = { workspace = true } -osmosis-std-derive = { workspace = true } -cosmwasm-schema = { workspace = true } -cw-storage-plus ={ workspace = true } -schemars ={ workspace = true } -serde = { workspace = true } -thiserror = { workspace = true } -cw-utils = { workspace = true } -cw20-base = { workspace = true } -serde-json-wasm = { workspace = true } - -# Quasar packages -quasar-types = { path = "../../packages/quasar-types", version = "0.1.0"} -quasar-traits = { path = "../../packages/quasar-traits", version = "0.1.0"} -quasar-bindings = { path = "../../packages/quasar-bindings", version = "0.1.0"} - -[dev-dependencies] -cosmwasm-schema = { workspace = true } diff --git a/smart-contracts/osmosis/contracts/ibc-transfer/README.md b/smart-contracts/osmosis/contracts/ibc-transfer/README.md deleted file mode 100644 index da7dc9a95..000000000 --- a/smart-contracts/osmosis/contracts/ibc-transfer/README.md +++ /dev/null @@ -1,14 +0,0 @@ -# ibc-transfer demo contract -This contract is modified from Neutrons ibc-transfer demo contract. - -The main difference is they use a Sudo message to re-enter the contract, while we comply with the ibc standards for it - -## IBC transfer contract -Interacting with counterpart chain via ibc transfer is two phases process. -1. Send ibc transfer message -2. Accept and process ibc acknowlegement(ibc_packet_ack call) - -to run the contract you need to init two chain network connected with go relayer - - -## See demos/ibc-transfer-test/README.md for running instructions \ No newline at end of file diff --git a/smart-contracts/osmosis/contracts/ibc-transfer/examples/schema.rs b/smart-contracts/osmosis/contracts/ibc-transfer/examples/schema.rs deleted file mode 100644 index 873570c92..000000000 --- a/smart-contracts/osmosis/contracts/ibc-transfer/examples/schema.rs +++ /dev/null @@ -1,30 +0,0 @@ -// Copyright 2022 Neutron -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. - -use std::env::current_dir; -use std::fs::create_dir_all; - -use cosmwasm_schema::{export_schema, remove_schemas, schema_for}; - -use ibc_transfer::contract::{ExecuteMsg, InstantiateMsg}; - -fn main() { - let mut out_dir = current_dir().unwrap(); - out_dir.push("schema"); - create_dir_all(&out_dir).unwrap(); - remove_schemas(&out_dir).unwrap(); - - export_schema(&schema_for!(InstantiateMsg), &out_dir); - export_schema(&schema_for!(ExecuteMsg), &out_dir); -} diff --git a/smart-contracts/osmosis/contracts/ibc-transfer/schema/execute_msg.json b/smart-contracts/osmosis/contracts/ibc-transfer/schema/execute_msg.json deleted file mode 100644 index 5c6683e85..000000000 --- a/smart-contracts/osmosis/contracts/ibc-transfer/schema/execute_msg.json +++ /dev/null @@ -1,32 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "ExecuteMsg", - "oneOf": [ - { - "type": "object", - "required": [ - "send" - ], - "properties": { - "send": { - "type": "object", - "required": [ - "amount", - "to" - ], - "properties": { - "amount": { - "type": "integer", - "format": "uint128", - "minimum": 0.0 - }, - "to": { - "type": "string" - } - } - } - }, - "additionalProperties": false - } - ] -} diff --git a/smart-contracts/osmosis/contracts/ibc-transfer/schema/instantiate_msg.json b/smart-contracts/osmosis/contracts/ibc-transfer/schema/instantiate_msg.json deleted file mode 100644 index 44588cf22..000000000 --- a/smart-contracts/osmosis/contracts/ibc-transfer/schema/instantiate_msg.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "InstantiateMsg", - "type": "object" -} diff --git a/smart-contracts/osmosis/contracts/ibc-transfer/src/contract.rs b/smart-contracts/osmosis/contracts/ibc-transfer/src/contract.rs deleted file mode 100644 index 7da5d5b6f..000000000 --- a/smart-contracts/osmosis/contracts/ibc-transfer/src/contract.rs +++ /dev/null @@ -1,128 +0,0 @@ -use cosmwasm_std::{ - entry_point, to_json_binary, Binary, CosmosMsg, Deps, DepsMut, Env, IbcBasicResponse, IbcMsg, - IbcTimeout, MessageInfo, Reply, Response, StdError, StdResult, Storage, -}; - -use crate::{ - error::ContractError, - helpers::{create_reply, parse_seq, IbcMsgKind, MsgKind}, - state::{State, STATE}, -}; - -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -use crate::state::{PENDING_ACK, REPLIES}; - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)] -pub struct InstantiateMsg {} - -#[entry_point] -pub fn instantiate( - deps: DepsMut, - _env: Env, - _info: MessageInfo, - _msg: InstantiateMsg, -) -> StdResult { - deps.api.debug("WASMDEBUG: Instantiate"); - - let initial_state = State { - transfer_happened: false, - }; - - STATE.save(deps.storage, &initial_state)?; - - Ok(Response::default()) -} - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)] -#[serde(rename_all = "snake_case")] -pub enum ExecuteMsg { - Transfer { to_address: String, channel: String }, -} - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)] -#[serde(rename_all = "snake_case")] -pub enum QueryMsg { - // this will just confirm that a transfer happened - State {}, -} - -#[entry_point] -pub fn execute(deps: DepsMut, env: Env, info: MessageInfo, msg: ExecuteMsg) -> StdResult { - deps.api - .debug(format!("WASMDEBUG: execute: received msg: {msg:?}").as_str()); - match msg { - ExecuteMsg::Transfer { - to_address, - channel, - } => execute_transfer(deps, env, info, channel, to_address), - } -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn reply(deps: DepsMut, _env: Env, msg: Reply) -> StdResult { - // Save the ibc message together with the sequence number, to be handled properly later at the ack - let kind = REPLIES.load(deps.storage, msg.id)?; - match kind { - MsgKind::Ibc(ibc_kind) => { - let seq = parse_seq(msg)?; - PENDING_ACK.save(deps.storage, seq, &ibc_kind)?; - } - } - Ok(Response::default()) -} - -pub fn do_ibc_lock_tokens( - _deps: &mut dyn Storage, - _token_amount: String, -) -> Result { - todo!() -} - -fn execute_transfer( - deps: DepsMut, - env: Env, - info: MessageInfo, - channel: String, - to_address: String, -) -> StdResult { - if info.funds.len() != 1 { - return Err(StdError::GenericErr { - msg: "invalid number of denoms sent (send one)".to_string(), - }); - } - - let funds = info.funds[0].clone(); - let transfer = IbcMsg::Transfer { - channel_id: channel.clone(), - to_address: to_address.clone(), - amount: funds, - timeout: IbcTimeout::with_timestamp(env.block.time.plus_seconds(300)), - }; - Ok( - create_reply(deps.storage, MsgKind::Ibc(IbcMsgKind::Transfer), transfer)? - .add_attribute("ibc-tranfer-channel", channel) - .add_attribute("ibc-transfer-receiver", to_address), - ) -} - -pub fn confirm_transfer(deps: DepsMut) -> Result { - let mut state = STATE.load(deps.storage)?; - state.transfer_happened = true; - STATE.save(deps.storage, &state)?; - Ok(IbcBasicResponse::default()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn query(deps: Deps, _env: Env, msg: QueryMsg) -> StdResult { - match msg { - QueryMsg::State {} => to_json_binary(&query_state(deps)?), - } -} - -pub fn query_state(deps: Deps) -> StdResult { - let state = STATE.load(deps.storage)?; - - Ok(state) -} diff --git a/smart-contracts/osmosis/contracts/ibc-transfer/src/error.rs b/smart-contracts/osmosis/contracts/ibc-transfer/src/error.rs deleted file mode 100644 index 90548bab0..000000000 --- a/smart-contracts/osmosis/contracts/ibc-transfer/src/error.rs +++ /dev/null @@ -1,34 +0,0 @@ -use cosmwasm_std::StdError; -use quasar_types::error::Error as QError; -use thiserror::Error; - -/// Never is a placeholder to ensure we don't return any errors -#[derive(Error, Debug)] -pub enum Never {} - -#[derive(Error, Debug, PartialEq)] -pub enum ContractError { - #[error("{0}")] - Std(#[from] StdError), - - #[error("{0}")] - Base(#[from] cw20_base::ContractError), - - #[error("{0}")] - QError(#[from] QError), - - #[error("{0}")] - PaymentError(#[from] cw_utils::PaymentError), - - #[error("{0}")] - QueueError(String), - - #[error("no counterpart ica address found")] - NoCounterpartyIcaAddress, - - #[error("channel is not an ica channel")] - NoIcaChannel, - - #[error("not enough funds in the strategy to withdraw")] - InsufficientOutStandingFunds, -} diff --git a/smart-contracts/osmosis/contracts/ibc-transfer/src/helpers.rs b/smart-contracts/osmosis/contracts/ibc-transfer/src/helpers.rs deleted file mode 100644 index f2938110b..000000000 --- a/smart-contracts/osmosis/contracts/ibc-transfer/src/helpers.rs +++ /dev/null @@ -1,87 +0,0 @@ -use crate::state::REPLIES; -use cosmwasm_std::{CosmosMsg, Order, Reply, Response, StdError, Storage, SubMsg}; - -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -pub fn create_reply( - storage: &mut dyn Storage, - msg_kind: MsgKind, - msg: impl Into, -) -> Result { - let last = REPLIES.range(storage, None, None, Order::Descending).next(); - let mut id: u64 = 0; - if let Some(val) = last { - id = val?.0; - } - // register the message in the replies for handling - REPLIES.save(storage, id, &msg_kind)?; - Ok(Response::new().add_submessage(SubMsg::reply_always(msg, id))) -} - -pub fn create_submsg( - storage: &mut dyn Storage, - msg_kind: MsgKind, - msg: impl Into, -) -> Result { - let last = REPLIES.range(storage, None, None, Order::Descending).next(); - let mut id: u64 = 0; - if let Some(val) = last { - id = val?.0; - } - // register the message in the replies for handling - REPLIES.save(storage, id, &msg_kind)?; - Ok(SubMsg::reply_always(msg, id)) -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub enum IbcMsgKind { - Transfer, - Ica(IcaMessages), - Icq, -} - -// All enums supported by this contract -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub enum IcaMessages { - JoinSwapExternAmountIn, - LockTokens, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub enum MsgKind { - Ibc(IbcMsgKind), -} - -pub(crate) fn parse_seq(reply: Reply) -> Result { - reply - .result - .into_result() - .map_err(|msg| StdError::GenericErr { msg })? - .events - .iter() - .find(|e| e.ty == "send_packet") - .ok_or(StdError::NotFound { - kind: "send_packet_event".into(), - })? - .attributes - .iter() - .find(|attr| attr.key == "packet_sequence") - .ok_or(StdError::NotFound { - kind: "packet_sequence".into(), - })? - .value - .parse::() - .map_err(|e| StdError::ParseErr { - target_type: "u64".into(), - msg: e.to_string(), - }) -} - -#[cfg(test)] -mod tests { - // TODO write some tests for helpers -} diff --git a/smart-contracts/osmosis/contracts/ibc-transfer/src/ibc.rs b/smart-contracts/osmosis/contracts/ibc-transfer/src/ibc.rs deleted file mode 100644 index 853c63412..000000000 --- a/smart-contracts/osmosis/contracts/ibc-transfer/src/ibc.rs +++ /dev/null @@ -1,269 +0,0 @@ -use crate::contract::{confirm_transfer, do_ibc_lock_tokens}; -use crate::error::{ContractError, Never}; -use crate::helpers::{create_submsg, IbcMsgKind, IcaMessages, MsgKind}; -use crate::state::{CHANNELS, PENDING_ACK}; -use osmosis_std::types::osmosis::gamm::v1beta1::MsgJoinSwapExternAmountInResponse; -use quasar_types::error::Error as QError; -use quasar_types::ibc::{ - enforce_order_and_version, ChannelInfo, ChannelType, HandshakeState, IcsAck, -}; -use quasar_types::ica::handshake::enforce_ica_order_and_metadata; -use quasar_types::icq::ICQ_ORDERING; -use quasar_types::{ibc, ica::handshake::IcaMetadata, icq::ICQ_VERSION}; - -use cosmwasm_std::{ - entry_point, from_json, Binary, DepsMut, Env, IbcBasicResponse, IbcChannel, - IbcChannelCloseMsg, IbcChannelConnectMsg, IbcChannelOpenMsg, IbcPacket, IbcPacketAckMsg, - IbcPacketReceiveMsg, IbcPacketTimeoutMsg, IbcReceiveResponse, StdError, -}; - -#[cfg_attr(not(feature = "library"), entry_point)] -/// enforces ordering and versioning constraints, this combines ChanOpenInit and ChanOpenTry -pub fn ibc_channel_open( - deps: DepsMut, - _env: Env, - msg: IbcChannelOpenMsg, -) -> Result<(), ContractError> { - // save the channel as an channel in ChanOpenInit, we support inits from icq and ica channels - if msg.channel().version == ICQ_VERSION { - handle_icq_channel(deps, msg.channel().clone())?; - } else { - handle_ica_channel(deps, msg.channel().clone())?; - } - - Ok(()) -} - -fn handle_icq_channel(deps: DepsMut, channel: IbcChannel) -> Result<(), ContractError> { - ibc::enforce_order_and_version(&channel, None, &channel.version, channel.order.clone())?; - // save the channel state here - let info = ChannelInfo { - id: channel.endpoint.channel_id.clone(), - counterparty_endpoint: channel.counterparty_endpoint, - connection_id: channel.connection_id, - channel_type: ChannelType::Icq { - channel_ty: channel.version, - }, - handshake_state: HandshakeState::Init, - }; - CHANNELS.save(deps.storage, channel.endpoint.channel_id, &info)?; - Ok(()) -} - -fn handle_ica_channel(deps: DepsMut, channel: IbcChannel) -> Result<(), ContractError> { - let metadata: IcaMetadata = serde_json_wasm::from_str(&channel.version).map_err(|error| { - QError::InvalidIcaMetadata { - raw_metadata: channel.version.clone(), - error: error.to_string(), - } - })?; - - enforce_ica_order_and_metadata(&channel, None, &metadata)?; - // save the current state of the initializing channel - let info = ChannelInfo { - id: channel.endpoint.channel_id.clone(), - counterparty_endpoint: channel.counterparty_endpoint, - connection_id: channel.connection_id, - channel_type: ChannelType::Ica { - channel_ty: metadata, - counter_party_address: None, - }, - handshake_state: HandshakeState::Init, - }; - CHANNELS.save(deps.storage, channel.endpoint.channel_id, &info)?; - Ok(()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -/// record the channel in CHANNEL_INFO, this combines the ChanOpenAck and ChanOpenConfirm steps -pub fn ibc_channel_connect( - deps: DepsMut, - _env: Env, - msg: IbcChannelConnectMsg, -) -> Result { - // try to fetch the connecting channel, we should error if it does not exist\ - let mut info: ChannelInfo = CHANNELS - .load(deps.storage, msg.channel().endpoint.channel_id.clone()) - .map_err(|err| StdError::GenericErr { - msg: err.to_string(), - })?; - // we need to check the counter party version in try and ack (sometimes here) - // TODO we can wrap this match in a function in our ibc package - - // TODO think of a better datastructure so we dont have to parse ICA channels like this - match info.channel_type { - ChannelType::Icq { ref channel_ty } => enforce_order_and_version( - msg.channel(), - msg.counterparty_version(), - channel_ty.as_str(), - ICQ_ORDERING, - )?, - ChannelType::Ica { - channel_ty, - counter_party_address: _, - } => { - let counter_party_metadata = enforce_ica_order_and_metadata( - msg.channel(), - msg.counterparty_version(), - &channel_ty, - )?; - - let counter_party = counter_party_metadata - .ok_or(ContractError::QError(QError::NoCounterpartyIcaAddress))?; - // at this point, we expect a counterparty address, if it's none, we have to error - if counter_party.address().is_none() { - return Err(ContractError::NoCounterpartyIcaAddress); - } - let addr = counter_party.address(); - if addr.is_none() { - return Err(ContractError::NoCounterpartyIcaAddress); - } - info.channel_type = ChannelType::Ica { - channel_ty, - counter_party_address: addr, - } - } - ChannelType::Ics20 { channel_ty: _ } => todo!(), - } - - info.handshake_state = HandshakeState::Open; - - CHANNELS.save( - deps.storage, - msg.channel().endpoint.channel_id.clone(), - &info, - )?; - - Ok(IbcBasicResponse::default()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn ibc_channel_close( - _deps: DepsMut, - _env: Env, - _channel: IbcChannelCloseMsg, -) -> Result { - // TODO: what to do here? - // we will have locked funds that need to be returned somehow - unimplemented!(); -} - -#[cfg_attr(not(feature = "library"), entry_point)] -/// Check to see if we have any balance here -/// We should not return an error if possible, but rather an acknowledgement of failure -pub fn ibc_packet_receive( - _deps: DepsMut, - _env: Env, - _msg: IbcPacketReceiveMsg, -) -> Result { - // Contract does not handle packets/queries. - unimplemented!(); -} - -#[cfg_attr(not(feature = "library"), entry_point)] -/// check if success or failure and update balance, or return funds -pub fn ibc_packet_ack( - deps: DepsMut, - env: Env, - msg: IbcPacketAckMsg, -) -> Result { - // TODO: trap error like in receive? - let ack: IcsAck = from_json(&msg.acknowledgement.data)?; - match ack { - IcsAck::Result(val) => handle_succesful_ack(deps, env, msg, val), - IcsAck::Error(err) => handle_failing_ack(deps, env, msg, err), - } -} - -pub fn handle_succesful_ack( - deps: DepsMut, - _env: Env, - pkt: IbcPacketAckMsg, - ack_bin: Binary, -) -> Result { - let kind = PENDING_ACK.load(deps.storage, pkt.original_packet.sequence)?; - match kind { - crate::helpers::IbcMsgKind::Transfer => confirm_transfer(deps), - crate::helpers::IbcMsgKind::Ica(ica_kind) => match ica_kind { - crate::helpers::IcaMessages::JoinSwapExternAmountIn => { - let response: MsgJoinSwapExternAmountInResponse = from_json(&ack_bin)?; - let msg = do_ibc_lock_tokens(deps.storage, response.share_out_amount)?; - let msg_kind = MsgKind::Ibc(IbcMsgKind::Ica(IcaMessages::LockTokens)); - Ok(IbcBasicResponse::new().add_submessage(create_submsg( - deps.storage, - msg_kind, - msg, - )?)) - } - crate::helpers::IcaMessages::LockTokens => todo!(), - }, - crate::helpers::IbcMsgKind::Icq => todo!(), - } -} - -pub fn handle_failing_ack( - _deps: DepsMut, - _env: Env, - _pkt: IbcPacketAckMsg, - error: String, -) -> Result { - // TODO we can expand error handling here to fetch the packet by the - Ok(IbcBasicResponse::new().add_attribute("ibc-error", error.as_str())) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -/// return fund to original sender (same as failure in ibc_packet_ack) -pub fn ibc_packet_timeout( - deps: DepsMut, - _env: Env, - msg: IbcPacketTimeoutMsg, -) -> Result { - // TODO: trap error like in receive? - on_packet_failure(deps, msg.packet, "timeout".to_string())?; - Ok(IbcBasicResponse::default()) -} - -fn on_packet_failure( - _deps: DepsMut, - _packet: IbcPacket, - _error: String, -) -> Result { - todo!() -} - -// #[cfg(test)] -// mod test { -// use super::*; -// use crate::test_helpers::*; - -// use crate::contract::query_channel; -// use cosmwasm_std::testing::mock_env; -// use cosmwasm_std::{coins, to_vec, IbcEndpoint}; - -// #[test] -// fn check_ack_json() { -// let success = StrategyAck::Result(b"1".into()); -// let fail = StrategyAck::Error("bad coin".into()); - -// let success_json = String::from_utf8(to_vec(&success).unwrap()).unwrap(); -// assert_eq!(r#"{"result":"MQ=="}"#, success_json.as_str()); - -// let fail_json = String::from_utf8(to_vec(&fail).unwrap()).unwrap(); -// assert_eq!(r#"{"error":"bad coin"}"#, fail_json.as_str()); -// } - -// #[test] -// fn check_packet_json() { -// let packet = StrategyPacket::new( -// Uint128(12345), -// "ucosm", -// "cosmos1zedxv25ah8fksmg2lzrndrpkvsjqgk4zt5ff7n", -// "wasm1fucynrfkrt684pm8jrt8la5h2csvs5cnldcgqc", -// ); -// // Example message generated from the SDK -// let expected = r#"{"amount":"12345","denom":"ucosm","receiver":"wasm1fucynrfkrt684pm8jrt8la5h2csvs5cnldcgqc","sender":"cosmos1zedxv25ah8fksmg2lzrndrpkvsjqgk4zt5ff7n"}"#; - -// let encdoded = String::from_utf8(to_vec(&packet).unwrap()).unwrap(); -// assert_eq!(expected, encdoded.as_str()); -// } -// } diff --git a/smart-contracts/osmosis/contracts/ibc-transfer/src/lib.rs b/smart-contracts/osmosis/contracts/ibc-transfer/src/lib.rs deleted file mode 100644 index 7468ac82e..000000000 --- a/smart-contracts/osmosis/contracts/ibc-transfer/src/lib.rs +++ /dev/null @@ -1,21 +0,0 @@ -// Copyright 2022 Neutron -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. - -#![warn(clippy::unwrap_used, clippy::expect_used)] - -pub mod contract; -mod error; -pub mod helpers; -pub mod ibc; -pub mod state; diff --git a/smart-contracts/osmosis/contracts/ibc-transfer/src/state.rs b/smart-contracts/osmosis/contracts/ibc-transfer/src/state.rs deleted file mode 100644 index 3c2c06dec..000000000 --- a/smart-contracts/osmosis/contracts/ibc-transfer/src/state.rs +++ /dev/null @@ -1,30 +0,0 @@ -use quasar_types::ibc::ChannelInfo; -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -use cw_storage_plus::{Item, Map}; - -use crate::helpers::{IbcMsgKind, MsgKind}; - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub enum Origin { - Sample, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub struct Config { - pub default_timeout: u64, -} - -pub(crate) const CHANNELS: Map = Map::new("channels"); - -pub(crate) const REPLIES: Map = Map::new("replies"); - -pub(crate) const PENDING_ACK: Map = Map::new("pending_acks"); - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub struct State { - pub transfer_happened: bool, -} - -pub(crate) const STATE: Item = Item::new("state"); diff --git a/smart-contracts/osmosis/contracts/ica/.cargo/config b/smart-contracts/osmosis/contracts/ica/.cargo/config deleted file mode 100644 index dba2c3f7a..000000000 --- a/smart-contracts/osmosis/contracts/ica/.cargo/config +++ /dev/null @@ -1,5 +0,0 @@ -[alias] -wasm = "build --release --target wasm32-unknown-unknown" -unit-test = "test --lib" -schema = "run --example schema" -lint = "clippy -- -D warnings" \ No newline at end of file diff --git a/smart-contracts/osmosis/contracts/ica/Cargo.toml b/smart-contracts/osmosis/contracts/ica/Cargo.toml deleted file mode 100644 index 4ed7a8fc3..000000000 --- a/smart-contracts/osmosis/contracts/ica/Cargo.toml +++ /dev/null @@ -1,37 +0,0 @@ -[package] -name = "ica" -version = "0.1.0" -authors = ["Laurens Kubat "] -edition = "2018" -description = "IBC Enabled contracts for interchain queries" -license = "Apache-2.0" -repository = "https://github.com/CosmWasm/cw-plus" -homepage = "https://cosmwasm.com" -documentation = "https://docs.cosmwasm.com" - -[lib] -crate-type = ["cdylib", "rlib"] - -[features] -# backtraces = ["cosmwasm-std/backtraces"] -# use library feature to disable all init/handle/query exports -library = [] - -[dependencies] -cosmwasm-std = { workspace = true } -osmosis-std = { workspace = true } -osmosis-std-derive = { workspace = true } -cosmwasm-schema = { workspace = true } -cw-storage-plus ={ workspace = true } -schemars ={ workspace = true } -serde = { workspace = true } -thiserror = { workspace = true } -prost = { workspace = true } -cw-controllers = { workspace = true } -cw-utils = { workspace = true } -cw2 = { workspace = true } -cw20 = { workspace = true } -cosmos-sdk-proto = { workspace = true } -base64 = { workspace = true } - -semver = "1" diff --git a/smart-contracts/osmosis/contracts/ica/NOTICE b/smart-contracts/osmosis/contracts/ica/NOTICE deleted file mode 100644 index bc266454e..000000000 --- a/smart-contracts/osmosis/contracts/ica/NOTICE +++ /dev/null @@ -1,14 +0,0 @@ -CW20-ICS20: IBC Enabled contracts that receives CW20 tokens and sends them over ICS20 to a remote chain -Copyright (C) 2020 Confio OÜ - -Licensed under the Apache License, Version 2.0 (the "License"); -you may not use this file except in compliance with the License. -You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - -Unless required by applicable law or agreed to in writing, software -distributed under the License is distributed on an "AS IS" BASIS, -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -See the License for the specific language governing permissions and -limitations under the License. diff --git a/smart-contracts/osmosis/contracts/ica/README.md b/smart-contracts/osmosis/contracts/ica/README.md deleted file mode 100644 index 3f5f43452..000000000 --- a/smart-contracts/osmosis/contracts/ica/README.md +++ /dev/null @@ -1,32 +0,0 @@ -# ICQ - -This is an *IBC Enabled* contract that allows us to send ICQ queries from one chain over the standard ICQ -protocol to the bank module of another chain. - -## Workflow - -The contract starts with minimal state. It just stores a default timeout in seconds for all packets it sends. -Most importantly it binds a local IBC port to enable channel connections. - -An external party first needs to make one or more channels using this contract as one endpoint. It will use standard -unordered channels for the version negotiation. Once established, it manages a list of known channels. - -After there is at least one channel, you can send any ICQ query to this contract. It may optionally include a custom timeout. - -## Messages - -It only accepts ICQQueryMsg. The data sent along with that message must be a JSON-serialized -TransferMsg: - -## Queries - -Queries only make sense relative to the established channels of this contract. - -* `Port{}` - returns the port ID this contract has bound, so you can create channels. This info can be queried - via wasmd contract info query, but we expose another query here for convenience. -* `ListChannels{}` - returns a (currently unpaginated) list of all channels that have been created on this contract. - Returns their local channelId along with some basic metadata, like the remote port/channel and the connection they - run on top of. -* `Channel{id}` - returns more detailed information on one specific channel. In addition to the information available - in the list view, it returns the current outstanding balance on that channel, as well as the total amount that - has ever been sent on the channel. diff --git a/smart-contracts/osmosis/contracts/ica/create_and_execute.sh b/smart-contracts/osmosis/contracts/ica/create_and_execute.sh deleted file mode 100755 index 4cb96b2e4..000000000 --- a/smart-contracts/osmosis/contracts/ica/create_and_execute.sh +++ /dev/null @@ -1,32 +0,0 @@ -#!/bin/sh - -CHAIN_ID="quasar" -TESTNET_NAME="quasar" -FEE_DENOM="uqsr" -RPC="http://127.0.0.1:26659" -NODE="--node $RPC" -TXFLAG="$NODE --chain-id $CHAIN_ID --gas-prices 10$FEE_DENOM --gas auto --gas-adjustment 1.3" -echo $NODE -INIT='{"default_timeout": 600}' -# quasar <-> osmosis channel is connection 2 in current setup -# MSG='{"register_interchain_account":{"connection_id":"connection-0"}}' - -cd ../../ - -docker run --rm -v "$(pwd)":/code --mount type=volume,source="$(basename "$(pwd)")_cache",target=/code/target --mount type=volume,source=registry_cache,target=/usr/local/cargo/registry cosmwasm/rust-optimizer:0.12.6 - -echo "Running store code" -RES=$(quasarnoded tx wasm store artifacts/icq.wasm --from alice --keyring-backend test -y --output json -b block $TXFLAG) -CODE_ID=$(echo $RES | jq -r '.logs[0].events[-1].attributes[0].value') -echo "Got CODE_ID = $CODE_ID" - -echo "Deploying contract" -# swallow output -OUT=$(quasarnoded tx wasm instantiate $CODE_ID "$INIT" --from alice --keyring-backend test --label "my first contract" --gas-prices 10$FEE_DENOM --gas auto --gas-adjustment 1.3 -b block -y --no-admin $NODE --chain-id $CHAIN_ID) -ADDR=$(quasarnoded query wasm list-contract-by-code $CODE_ID --output json $NODE | jq -r '.contracts[0]') -echo "Got address of deployed contract = $ADDR" - -# echo "Executing register ica message... ('$MSG')" -# quasarnoded tx wasm execute $ADDR "$MSG" -y --from alice --keyring-backend test --gas-prices 10$FEE_DENOM --gas auto --gas-adjustment 1.3 $NODE --chain-id $CHAIN_ID - -cd - \ No newline at end of file diff --git a/smart-contracts/osmosis/contracts/ica/examples/schema.rs b/smart-contracts/osmosis/contracts/ica/examples/schema.rs deleted file mode 100644 index 4948e9c07..000000000 --- a/smart-contracts/osmosis/contracts/ica/examples/schema.rs +++ /dev/null @@ -1,23 +0,0 @@ -use std::env::current_dir; -use std::fs::create_dir_all; - -use cosmwasm_schema::{export_schema, remove_schemas, schema_for}; - -use ica::msg::{ - ChannelResponse, ExecuteMsg, ICQQueryMsg, InitMsg, ListChannelsResponse, PortResponse, QueryMsg, -}; - -fn main() { - let mut out_dir = current_dir().unwrap(); - out_dir.push("schema"); - create_dir_all(&out_dir).unwrap(); - remove_schemas(&out_dir).unwrap(); - - export_schema(&schema_for!(InitMsg), &out_dir); - export_schema(&schema_for!(ExecuteMsg), &out_dir); - export_schema(&schema_for!(QueryMsg), &out_dir); - export_schema(&schema_for!(ICQQueryMsg), &out_dir); - export_schema(&schema_for!(ChannelResponse), &out_dir); - export_schema(&schema_for!(ListChannelsResponse), &out_dir); - export_schema(&schema_for!(PortResponse), &out_dir); -} diff --git a/smart-contracts/osmosis/contracts/ica/schema/channel_response.json b/smart-contracts/osmosis/contracts/ica/schema/channel_response.json deleted file mode 100644 index 1744539e6..000000000 --- a/smart-contracts/osmosis/contracts/ica/schema/channel_response.json +++ /dev/null @@ -1,61 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "ChannelResponse", - "type": "object", - "required": [ - "info" - ], - "properties": { - "info": { - "description": "Information on the channel's connection", - "allOf": [ - { - "$ref": "#/definitions/ChannelInfo" - } - ] - } - }, - "definitions": { - "ChannelInfo": { - "type": "object", - "required": [ - "connection_id", - "counterparty_endpoint", - "id" - ], - "properties": { - "connection_id": { - "description": "the connection this exists on (you can use to query client/consensus info)", - "type": "string" - }, - "counterparty_endpoint": { - "description": "the remote channel/port we connect to", - "allOf": [ - { - "$ref": "#/definitions/IbcEndpoint" - } - ] - }, - "id": { - "description": "id of this channel", - "type": "string" - } - } - }, - "IbcEndpoint": { - "type": "object", - "required": [ - "channel_id", - "port_id" - ], - "properties": { - "channel_id": { - "type": "string" - }, - "port_id": { - "type": "string" - } - } - } - } -} diff --git a/smart-contracts/osmosis/contracts/ica/schema/execute_msg.json b/smart-contracts/osmosis/contracts/ica/schema/execute_msg.json deleted file mode 100644 index 858ddc8c3..000000000 --- a/smart-contracts/osmosis/contracts/ica/schema/execute_msg.json +++ /dev/null @@ -1,67 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "ExecuteMsg", - "oneOf": [ - { - "type": "object", - "required": [ - "join_pool" - ], - "properties": { - "join_pool": { - "type": "object", - "required": [ - "channel", - "pool_id", - "sender", - "share_out_amount", - "token_in_maxs" - ], - "properties": { - "channel": { - "type": "string" - }, - "pool_id": { - "$ref": "#/definitions/Uint64" - }, - "sender": { - "type": "string" - }, - "share_out_amount": { - "type": "string" - }, - "token_in_maxs": { - "type": "array", - "items": { - "$ref": "#/definitions/Coin" - } - } - } - } - }, - "additionalProperties": false - } - ], - "definitions": { - "Coin": { - "description": "Coin defines a token with a denomination and an amount.\n\nNOTE: The amount field is an Int which implements the custom method signatures required by gogoproto.", - "type": "object", - "required": [ - "amount", - "denom" - ], - "properties": { - "amount": { - "type": "string" - }, - "denom": { - "type": "string" - } - } - }, - "Uint64": { - "description": "A thin wrapper around u64 that is using strings for JSON encoding/decoding, such that the full u64 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u64` to get the value out:\n\n``` # use cosmwasm_std::Uint64; let a = Uint64::from(42u64); assert_eq!(a.u64(), 42);\n\nlet b = Uint64::from(70u32); assert_eq!(b.u64(), 70); ```", - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/ica/schema/i_c_q_query_msg.json b/smart-contracts/osmosis/contracts/ica/schema/i_c_q_query_msg.json deleted file mode 100644 index 316cd71ce..000000000 --- a/smart-contracts/osmosis/contracts/ica/schema/i_c_q_query_msg.json +++ /dev/null @@ -1,60 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "ICQQueryMsg", - "description": "This is the message we accept via Receive", - "type": "object", - "required": [ - "channel", - "requests" - ], - "properties": { - "channel": { - "description": "The local channel to send the packets on", - "type": "string" - }, - "requests": { - "type": "array", - "items": { - "$ref": "#/definitions/RequestQueryJSON" - } - }, - "timeout": { - "type": [ - "integer", - "null" - ], - "format": "uint64", - "minimum": 0.0 - } - }, - "definitions": { - "Binary": { - "description": "Binary is a wrapper around Vec to add base64 de/serialization with serde. It also adds some helper methods to help encode inline.\n\nThis is only needed as serde-json-{core,wasm} has a horrible encoding for Vec. See also .", - "type": "string" - }, - "RequestQueryJSON": { - "type": "object", - "required": [ - "data", - "height", - "path", - "prove" - ], - "properties": { - "data": { - "$ref": "#/definitions/Binary" - }, - "height": { - "type": "integer", - "format": "int64" - }, - "path": { - "type": "string" - }, - "prove": { - "type": "boolean" - } - } - } - } -} diff --git a/smart-contracts/osmosis/contracts/ica/schema/init_msg.json b/smart-contracts/osmosis/contracts/ica/schema/init_msg.json deleted file mode 100644 index 33a291fc3..000000000 --- a/smart-contracts/osmosis/contracts/ica/schema/init_msg.json +++ /dev/null @@ -1,16 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "InitMsg", - "type": "object", - "required": [ - "default_timeout" - ], - "properties": { - "default_timeout": { - "description": "Default timeout for icq packets, specified in seconds", - "type": "integer", - "format": "uint64", - "minimum": 0.0 - } - } -} diff --git a/smart-contracts/osmosis/contracts/ica/schema/list_channels_response.json b/smart-contracts/osmosis/contracts/ica/schema/list_channels_response.json deleted file mode 100644 index 69f020c93..000000000 --- a/smart-contracts/osmosis/contracts/ica/schema/list_channels_response.json +++ /dev/null @@ -1,59 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "ListChannelsResponse", - "type": "object", - "required": [ - "channels" - ], - "properties": { - "channels": { - "type": "array", - "items": { - "$ref": "#/definitions/ChannelInfo" - } - } - }, - "definitions": { - "ChannelInfo": { - "type": "object", - "required": [ - "connection_id", - "counterparty_endpoint", - "id" - ], - "properties": { - "connection_id": { - "description": "the connection this exists on (you can use to query client/consensus info)", - "type": "string" - }, - "counterparty_endpoint": { - "description": "the remote channel/port we connect to", - "allOf": [ - { - "$ref": "#/definitions/IbcEndpoint" - } - ] - }, - "id": { - "description": "id of this channel", - "type": "string" - } - } - }, - "IbcEndpoint": { - "type": "object", - "required": [ - "channel_id", - "port_id" - ], - "properties": { - "channel_id": { - "type": "string" - }, - "port_id": { - "type": "string" - } - } - } - } -} diff --git a/smart-contracts/osmosis/contracts/ica/schema/port_response.json b/smart-contracts/osmosis/contracts/ica/schema/port_response.json deleted file mode 100644 index 4561b3a72..000000000 --- a/smart-contracts/osmosis/contracts/ica/schema/port_response.json +++ /dev/null @@ -1,13 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "PortResponse", - "type": "object", - "required": [ - "port_id" - ], - "properties": { - "port_id": { - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/ica/schema/query_msg.json b/smart-contracts/osmosis/contracts/ica/schema/query_msg.json deleted file mode 100644 index be42f505f..000000000 --- a/smart-contracts/osmosis/contracts/ica/schema/query_msg.json +++ /dev/null @@ -1,66 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "QueryMsg", - "oneOf": [ - { - "description": "Return the port ID bound by this contract. Returns PortResponse", - "type": "object", - "required": [ - "port" - ], - "properties": { - "port": { - "type": "object" - } - }, - "additionalProperties": false - }, - { - "description": "Show all channels we have connected to. Return type is ListChannelsResponse.", - "type": "object", - "required": [ - "list_channels" - ], - "properties": { - "list_channels": { - "type": "object" - } - }, - "additionalProperties": false - }, - { - "description": "Returns the details of the name channel, error if not created. Return type: ChannelResponse.", - "type": "object", - "required": [ - "channel" - ], - "properties": { - "channel": { - "type": "object", - "required": [ - "id" - ], - "properties": { - "id": { - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "description": "Show the Config. Returns ConfigResponse (currently including admin as well)", - "type": "object", - "required": [ - "config" - ], - "properties": { - "config": { - "type": "object" - } - }, - "additionalProperties": false - } - ] -} diff --git a/smart-contracts/osmosis/contracts/ica/src/contract.rs b/smart-contracts/osmosis/contracts/ica/src/contract.rs deleted file mode 100644 index 400a55a69..000000000 --- a/smart-contracts/osmosis/contracts/ica/src/contract.rs +++ /dev/null @@ -1,194 +0,0 @@ -use cosmos_sdk_proto::ibc::applications::interchain_accounts::v1::InterchainAccountPacketData; -#[cfg(not(feature = "library"))] -use cosmwasm_std::entry_point; -use cosmwasm_std::{ - to_json_binary, Binary, Deps, DepsMut, Env, IbcMsg, IbcQuery, IbcTimeout, MessageInfo, Order, - PortIdResponse, Reply, Response, StdResult, SubMsg, Uint64, -}; -use prost::Message; - -use osmosis_std::types::cosmos::base::v1beta1::Coin as OsmoCoin; -use osmosis_std::types::osmosis::gamm::v1beta1::MsgJoinPool; - -use cw2::set_contract_version; - -use crate::error::ContractError; -use crate::helpers::{handle_reply_sample, set_reply}; -use crate::msg::{ - ChannelResponse, ConfigResponse, ExecuteMsg, InitMsg, ListChannelsResponse, MigrateMsg, - PortResponse, QueryMsg, -}; -use crate::state::{Config, Origin, CHANNEL_INFO, CONFIG, QUERY_RESULT_COUNTER, REPLIES}; - -// version info for migration info -const CONTRACT_NAME: &str = "crates.io:icq"; -const CONTRACT_VERSION: &str = env!("CARGO_PKG_VERSION"); - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn instantiate( - deps: DepsMut, - _env: Env, - _info: MessageInfo, - msg: InitMsg, -) -> Result { - set_contract_version(deps.storage, CONTRACT_NAME, CONTRACT_VERSION)?; - let cfg = Config { - default_timeout: msg.default_timeout, - }; - CONFIG.save(deps.storage, &cfg)?; - QUERY_RESULT_COUNTER.save(deps.storage, &0)?; - Ok(Response::default()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn execute( - deps: DepsMut, - env: Env, - _info: MessageInfo, - msg: ExecuteMsg, -) -> Result { - match msg { - ExecuteMsg::JoinPool { - channel, - sender, - pool_id, - share_out_amount, - token_in_maxs, - } => execute_bank_send( - deps, - env, - channel, - sender, - pool_id, - share_out_amount, - token_in_maxs, - ), - } -} - -fn execute_bank_send( - deps: DepsMut, - env: Env, - channel: String, - sender: String, - pool_id: Uint64, - share_out_amount: String, - token_in_maxs: Vec, -) -> Result { - let msg = MsgJoinPool { - sender, - pool_id: pool_id.u64(), - share_out_amount, - token_in_maxs, - }; - let packet = InterchainAccountPacketData { - r#type: 1, - data: msg.encode_to_vec(), - memo: "".into(), - }; - - let send_packet_msg = IbcMsg::SendPacket { - channel_id: channel, - data: to_json_binary(&packet.encode_to_vec())?, - timeout: IbcTimeout::with_timestamp(env.block.time.plus_seconds(300)), - }; - - let id = set_reply(deps, &Origin::Sample)?; - let res = Response::new() - .add_submessage(SubMsg::reply_on_success(send_packet_msg, id)) - .add_attribute("action", "query"); - Ok(res) -} - -#[entry_point] -pub fn reply(deps: DepsMut, _env: Env, msg: Reply) -> StdResult { - let origin = REPLIES.load(deps.storage, msg.id)?; - match origin { - Origin::Sample => handle_reply_sample(deps, msg), - } -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn migrate(_deps: DepsMut, _env: Env, _msg: MigrateMsg) -> Result { - Ok(Response::new()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn query(deps: Deps, _env: Env, msg: QueryMsg) -> StdResult { - match msg { - QueryMsg::Port {} => to_json_binary(&query_port(deps)?), - QueryMsg::ListChannels {} => to_json_binary(&query_list(deps)?), - QueryMsg::Channel { id } => to_json_binary(&query_channel(deps, id)?), - QueryMsg::Config {} => to_json_binary(&query_config(deps)?), - } -} - -fn query_port(deps: Deps) -> StdResult { - let query = IbcQuery::PortId {}.into(); - let PortIdResponse { port_id } = deps.querier.query(&query)?; - Ok(PortResponse { port_id }) -} - -fn query_list(deps: Deps) -> StdResult { - let channels = CHANNEL_INFO - .range_raw(deps.storage, None, None, Order::Ascending) - .map(|r| r.map(|(_, v)| v)) - .collect::>()?; - Ok(ListChannelsResponse { channels }) -} - -// make public for ibc tests -pub fn query_channel(deps: Deps, id: String) -> StdResult { - let info = CHANNEL_INFO.load(deps.storage, &id)?; - Ok(ChannelResponse { info }) -} - -fn query_config(deps: Deps) -> StdResult { - let cfg = CONFIG.load(deps.storage)?; - let res = ConfigResponse { - default_timeout: cfg.default_timeout, - }; - Ok(res) -} - -#[cfg(test)] -mod test { - use super::*; - - use crate::test_helpers::*; - - use cosmwasm_std::testing::mock_env; - use cosmwasm_std::{from_json, StdError}; - - #[test] - fn setup_and_query() { - let deps = setup(&["channel-3", "channel-7"]); - - let raw_list = query(deps.as_ref(), mock_env(), QueryMsg::ListChannels {}).unwrap(); - let list_res: ListChannelsResponse = from_json(&raw_list).unwrap(); - assert_eq!(2, list_res.channels.len()); - assert_eq!(mock_channel_info("channel-3"), list_res.channels[0]); - assert_eq!(mock_channel_info("channel-7"), list_res.channels[1]); - - let raw_channel = query( - deps.as_ref(), - mock_env(), - QueryMsg::Channel { - id: "channel-3".to_string(), - }, - ) - .unwrap(); - let chan_res: ChannelResponse = from_json(&raw_channel).unwrap(); - assert_eq!(chan_res.info, mock_channel_info("channel-3")); - - let err = query( - deps.as_ref(), - mock_env(), - QueryMsg::Channel { - id: "channel-10".to_string(), - }, - ) - .unwrap_err(); - assert_eq!(err, StdError::not_found("icq::state::ChannelInfo")); - } -} diff --git a/smart-contracts/osmosis/contracts/ica/src/error.rs b/smart-contracts/osmosis/contracts/ica/src/error.rs deleted file mode 100644 index 10c38ef77..000000000 --- a/smart-contracts/osmosis/contracts/ica/src/error.rs +++ /dev/null @@ -1,53 +0,0 @@ -use std::string::FromUtf8Error; -use thiserror::Error; - -use cosmwasm_std::StdError; - -/// Never is a placeholder to ensure we don't return any errors -#[derive(Error, Debug)] -pub enum Never {} - -#[derive(Error, Debug, PartialEq)] -pub enum ContractError { - #[error("{0}")] - Std(#[from] StdError), - - #[error("Channel doesn't exist: {id}")] - NoSuchChannel { id: String }, - - #[error("Only supports channel with ibc version {contract_version}, got {version}")] - InvalidIbcVersion { - contract_version: String, - version: String, - }, - - #[error("Only supports unordered channel")] - OnlyOrderedChannel {}, - - #[error("Parsed port from denom ({port}) doesn't match packet")] - FromOtherPort { port: String }, - - #[error("Parsed channel from denom ({channel}) doesn't match packet")] - FromOtherChannel { channel: String }, - - #[error("Cannot migrate from different contract type: {previous_contract}")] - CannotMigrate { previous_contract: String }, - - #[error("Cannot migrate from unsupported version: {previous_version}")] - CannotMigrateVersion { previous_version: String }, - - #[error("Failed to proto encode")] - EncodingFail, - - #[error("Failed to proto decode")] - DecodingFail, - - #[error("Only the governance contract can do this")] - Unauthorized, -} - -impl From for ContractError { - fn from(_: FromUtf8Error) -> Self { - ContractError::Std(StdError::invalid_utf8("parsing denom key")) - } -} diff --git a/smart-contracts/osmosis/contracts/ica/src/helpers.rs b/smart-contracts/osmosis/contracts/ica/src/helpers.rs deleted file mode 100644 index bee098683..000000000 --- a/smart-contracts/osmosis/contracts/ica/src/helpers.rs +++ /dev/null @@ -1,85 +0,0 @@ -use crate::{ - proto::CosmosResponse, - state::{Origin, PENDING_QUERIES, QUERY_RESULT_COUNTER, REPLIES}, - ContractError, -}; -use cosmwasm_std::{ - attr, DepsMut, Env, IbcBasicResponse, IbcPacket, Order, Reply, Response, StdError, StdResult, -}; - -pub(crate) fn handle_reply_sample(deps: DepsMut, msg: Reply) -> StdResult { - let val = msg - .result - .into_result() - .map_err(|msg| StdError::GenericErr { msg })?; - - let event = val - .events - .iter() - .find(|e| e.ty == "send_packet") - .ok_or(StdError::NotFound { - kind: "send_packet_event".into(), - })?; - - // here we can do further stuff with a succesful package if necessary, in this case we can simply - // save the package, under the sequence number and channel id - let seq = event - .attributes - .iter() - .find(|attr| attr.key == "packet_sequence") - .ok_or(StdError::NotFound { - kind: "packet_sequence".into(), - })?; - let s = seq.value.parse::().map_err(|e| StdError::ParseErr { - target_type: "u64".into(), - msg: e.to_string(), - })?; - let channel = event - .attributes - .iter() - .find(|attr| attr.key == "packet_src_channel") - .ok_or(StdError::NotFound { - kind: "packet_src_channel".into(), - })?; - - PENDING_QUERIES.save(deps.storage, (s, &channel.value), &Origin::Sample)?; - Ok(Response::new().add_attribute("reply_registered", msg.id.to_string())) -} - -pub fn set_reply(deps: DepsMut, origin: &Origin) -> Result { - let last = REPLIES - .range(deps.storage, None, None, Order::Descending) - .next(); - let mut id: u64 = 0; - if let Some(val) = last { - id = val?.0; - } - - // register the message in the replies for handling - REPLIES.save(deps.storage, id, origin)?; - // send response - Ok(id) -} - -// for our sample origin callback, we increment the query counter and leave it at that -pub fn handle_sample_callback( - deps: DepsMut, - _env: Env, - response: CosmosResponse, - _original: IbcPacket, -) -> Result { - let attrs = vec![ - attr("action", "acknowledge"), - attr("num_messages", response.responses.len().to_string()), - attr("success", "true"), - ]; - - // Store result counter. - let mut counter = QUERY_RESULT_COUNTER.load(deps.storage)?; - counter += response.responses.len() as u64; - QUERY_RESULT_COUNTER.save(deps.storage, &counter)?; - Ok(IbcBasicResponse::new().add_attributes(attrs)) -} - -#[cfg(test)] -mod test {} diff --git a/smart-contracts/osmosis/contracts/ica/src/ibc.rs b/smart-contracts/osmosis/contracts/ica/src/ibc.rs deleted file mode 100644 index 45da2da41..000000000 --- a/smart-contracts/osmosis/contracts/ica/src/ibc.rs +++ /dev/null @@ -1,225 +0,0 @@ -use prost::bytes::Bytes; -use prost::Message; -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -use cosmwasm_std::{ - attr, entry_point, from_json, Binary, DepsMut, Env, IbcBasicResponse, IbcChannel, - IbcChannelCloseMsg, IbcChannelConnectMsg, IbcChannelOpenMsg, IbcOrder, IbcPacket, - IbcPacketAckMsg, IbcPacketReceiveMsg, IbcPacketTimeoutMsg, IbcReceiveResponse, -}; - -use crate::error::{ContractError, Never}; -use crate::helpers::handle_sample_callback; -use crate::proto::CosmosResponse; -use crate::state::{ChannelInfo, Origin, CHANNEL_INFO, PENDING_QUERIES}; - -pub const ICA_VERSION: &str = "{\"version\":\"ics-20\"}"; -pub const ICA_ORDERING: IbcOrder = IbcOrder::Ordered; - -/// This is compatible with the JSON serialization -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug, Default)] -pub struct InterchainQueryPacketAck { - pub data: Binary, -} - -impl InterchainQueryPacketAck { - pub fn new(data: Binary) -> Self { - InterchainQueryPacketAck { data } - } - - pub fn validate(&self) -> Result<(), ContractError> { - Ok(()) - } -} - -/// This is a generic ICS acknowledgement format. -/// Proto defined here: https://github.com/cosmos/cosmos-sdk/blob/v0.42.0/proto/ibc/core/channel/v1/channel.proto#L141-L147 -/// This is compatible with the JSON serialization -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub enum IcsAck { - Result(Binary), - Error(String), -} - -#[cfg_attr(not(feature = "library"), entry_point)] -/// enforces ordering and versioning constraints -pub fn ibc_channel_open( - _deps: DepsMut, - _env: Env, - msg: IbcChannelOpenMsg, -) -> Result<(), ContractError> { - enforce_order_and_version(msg.channel(), msg.counterparty_version())?; - Ok(()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -/// record the channel in CHANNEL_INFO -pub fn ibc_channel_connect( - deps: DepsMut, - _env: Env, - msg: IbcChannelConnectMsg, -) -> Result { - // we need to check the counter party version in try and ack (sometimes here) - enforce_order_and_version(msg.channel(), msg.counterparty_version())?; - - let channel: IbcChannel = msg.into(); - let info = ChannelInfo { - id: channel.endpoint.channel_id, - counterparty_endpoint: channel.counterparty_endpoint, - connection_id: channel.connection_id, - }; - CHANNEL_INFO.save(deps.storage, &info.id, &info)?; - - Ok(IbcBasicResponse::default()) -} - -fn enforce_order_and_version( - channel: &IbcChannel, - _counterparty_version: Option<&str>, -) -> Result<(), ContractError> { - // if channel.version != ICA_VERSION { - // return Err(ContractError::InvalidIbcVersion { - // contract_version: ICA_VERSION.to_string(), - // version: channel.version.clone(), - // }); - // } - // if let Some(version) = counterparty_version { - // if version != ICA_VERSION { - // return Err(ContractError::InvalidIbcVersion { - // contract_version: ICA_VERSION.to_string(), - // version: version.to_string(), - // }); - // } - // } - if channel.order != ICA_ORDERING { - return Err(ContractError::OnlyOrderedChannel {}); - } - Ok(()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn ibc_channel_close( - _deps: DepsMut, - _env: Env, - _channel: IbcChannelCloseMsg, -) -> Result { - // TODO: what to do here? - // we will have locked funds that need to be returned somehow - unimplemented!(); -} - -#[cfg_attr(not(feature = "library"), entry_point)] -/// Check to see if we have any balance here -/// We should not return an error if possible, but rather an acknowledgement of failure -pub fn ibc_packet_receive( - _deps: DepsMut, - _env: Env, - _msg: IbcPacketReceiveMsg, -) -> Result { - // Contract does not handle packets/queries. - unimplemented!(); -} - -#[cfg_attr(not(feature = "library"), entry_point)] -/// check if success or failure and update balance, or return funds -pub fn ibc_packet_ack( - deps: DepsMut, - env: Env, - msg: IbcPacketAckMsg, -) -> Result { - // Design decision: should we trap error like in receive? - // TODO: unsure... as it is now a failed ack handling would revert the tx and would be - // retried again and again. is that good? - let ics_msg: IcsAck = from_json(&msg.acknowledgement.data)?; - match ics_msg { - IcsAck::Result(data) => on_packet_success(deps, data, msg.original_packet, env), - IcsAck::Error(err) => on_packet_failure(deps, msg.original_packet, err), - } -} - -#[cfg_attr(not(feature = "library"), entry_point)] -/// return fund to original sender (same as failure in ibc_packet_ack) -pub fn ibc_packet_timeout( - deps: DepsMut, - _env: Env, - msg: IbcPacketTimeoutMsg, -) -> Result { - // TODO: trap error like in receive? (same question as ack above) - let packet = msg.packet; - on_packet_failure(deps, packet, "timeout".to_string()) -} - -fn on_packet_success( - deps: DepsMut, - data: Binary, - original: IbcPacket, - env: Env, -) -> Result { - let ack: InterchainQueryPacketAck = from_json(&data)?; - - let buf = Bytes::copy_from_slice(ack.data.as_slice()); - let resp: CosmosResponse = match CosmosResponse::decode(buf) { - Ok(resp) => resp, - Err(_) => return Err(ContractError::DecodingFail {}), - }; - - // load the msg from the pending queries so we know what to do - let origin = - PENDING_QUERIES.load(deps.storage, (original.sequence, &original.src.channel_id))?; - match origin { - Origin::Sample => Ok(handle_sample_callback(deps, env, resp, original)?), - } -} - -fn on_packet_failure( - _deps: DepsMut, - _packet: IbcPacket, - err: String, -) -> Result { - let attributes = vec![ - attr("action", "acknowledge"), - attr("success", "false"), - attr("error", err), - ]; - - Ok(IbcBasicResponse::new().add_attributes(attributes)) -} - -#[cfg(test)] -mod test { - use super::*; - use crate::state::QUERY_RESULT_COUNTER; - use crate::ContractError; - - use cosmwasm_std::testing::{mock_dependencies, mock_env}; - use cosmwasm_std::{Binary, IbcAcknowledgement, IbcEndpoint, IbcPacket, IbcTimeout, Timestamp}; - - #[test] - fn test_ibc_packet_ack() -> Result<(), ContractError> { - let mut deps = mock_dependencies(); - let timeout = IbcTimeout::with_timestamp(Timestamp::from_nanos(0)); - let src = IbcEndpoint { - port_id: "port-0".to_string(), - channel_id: "channel-0".to_string(), - }; - let dest = IbcEndpoint { - port_id: "port-1".to_string(), - channel_id: "channel-1".to_string(), - }; - let packet = IbcPacket::new(Binary::default(), src, dest, 0, timeout); - let ack = IbcAcknowledgement::new(Binary::from_base64( - "eyJyZXN1bHQiOiJleUprWVhSaElqb2lRMmRaU1VWcmFXUnpaMFU5SW4wPSJ9", - )?); - let msg = IbcPacketAckMsg::new(ack, packet); - QUERY_RESULT_COUNTER.save(deps.as_mut().storage, &0)?; // Save a 0 - match ibc_packet_ack(deps.as_mut(), mock_env(), msg) { - Ok(_) => { - assert_eq!(QUERY_RESULT_COUNTER.load(deps.as_mut().storage)?, 1); - Ok(()) - } - Err(err) => Err(err), - } - } -} diff --git a/smart-contracts/osmosis/contracts/ica/src/lib.rs b/smart-contracts/osmosis/contracts/ica/src/lib.rs deleted file mode 100644 index c785d5eef..000000000 --- a/smart-contracts/osmosis/contracts/ica/src/lib.rs +++ /dev/null @@ -1,14 +0,0 @@ -pub mod contract; -mod error; -mod helpers; -pub mod ibc; -pub mod msg; -mod proto; -pub mod state; -mod test_helpers; - -pub use crate::error::ContractError; - -// pub mod items { -// include!(concat!(env!("OUT_DIR"), "/queries.rs")); -// } diff --git a/smart-contracts/osmosis/contracts/ica/src/msg.rs b/smart-contracts/osmosis/contracts/ica/src/msg.rs deleted file mode 100644 index 88c2a670a..000000000 --- a/smart-contracts/osmosis/contracts/ica/src/msg.rs +++ /dev/null @@ -1,95 +0,0 @@ -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -use crate::state::ChannelInfo; -use cosmwasm_std::{Binary, Uint64}; -use osmosis_std::types::cosmos::base::v1beta1::Coin as OsmoCoin; - -#[derive(Serialize, Deserialize, Clone, Debug, JsonSchema)] -pub struct InitMsg { - /// Default timeout for icq packets, specified in seconds - pub default_timeout: u64, -} - -#[derive(Serialize, Deserialize, Clone, Debug, JsonSchema)] -pub struct MigrateMsg { - pub default_gas_limit: Option, -} - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -#[serde(rename_all = "snake_case")] -pub enum ExecuteMsg { - JoinPool { - channel: String, - sender: String, - pool_id: Uint64, - share_out_amount: String, - token_in_maxs: Vec, - }, -} - -/// This is the message we accept via Receive -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -pub struct ICQQueryMsg { - /// The local channel to send the packets on - pub channel: String, - pub requests: Vec, - // How long the packet lives in seconds. If not specified, use default_timeout - pub timeout: Option, -} - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -pub struct RequestQueryJSON { - pub data: Binary, - pub path: String, - pub height: i64, - pub prove: bool, -} - -// ResponseQuery does not contain all of the response fields. -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -pub struct ResponseQuery { - pub key: Binary, - pub value: String, - pub height: i64, -} - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -pub struct InterchainQueryPacketData { - pub data: Vec, -} - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -#[serde(rename_all = "snake_case")] -pub enum QueryMsg { - /// Return the port ID bound by this contract. Returns PortResponse - Port {}, - /// Show all channels we have connected to. Return type is ListChannelsResponse. - ListChannels {}, - /// Returns the details of the name channel, error if not created. - /// Return type: ChannelResponse. - Channel { id: String }, - /// Show the Config. Returns ConfigResponse (currently including admin as well) - Config {}, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub struct ListChannelsResponse { - pub channels: Vec, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub struct ChannelResponse { - /// Information on the channel's connection - pub info: ChannelInfo, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub struct PortResponse { - pub port_id: String, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub struct ConfigResponse { - pub default_timeout: u64, -} diff --git a/smart-contracts/osmosis/contracts/ica/src/proto.rs b/smart-contracts/osmosis/contracts/ica/src/proto.rs deleted file mode 100644 index 31cd1090c..000000000 --- a/smart-contracts/osmosis/contracts/ica/src/proto.rs +++ /dev/null @@ -1,15 +0,0 @@ -use cosmos_sdk_proto::tendermint::abci::{RequestQuery, ResponseQuery}; - -/// CosmosQuery contains a list of tendermint ABCI query requests. It should be used when sending queries to an SDK host chain. -#[derive(Clone, PartialEq, ::prost::Message)] -pub struct CosmosQuery { - #[prost(message, repeated, tag = "1")] - pub requests: ::prost::alloc::vec::Vec, -} - -/// CosmosResponse contains a list of tendermint ABCI query responses. It should be used when receiving responses from an SDK host chain. -#[derive(Clone, PartialEq, ::prost::Message)] -pub struct CosmosResponse { - #[prost(message, repeated, tag = "1")] - pub responses: ::prost::alloc::vec::Vec, -} diff --git a/smart-contracts/osmosis/contracts/ica/src/state.rs b/smart-contracts/osmosis/contracts/ica/src/state.rs deleted file mode 100644 index 1e555d458..000000000 --- a/smart-contracts/osmosis/contracts/ica/src/state.rs +++ /dev/null @@ -1,40 +0,0 @@ -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -use cosmwasm_std::IbcEndpoint; -use cw_storage_plus::{Item, Map}; - -pub const CONFIG: Item = Item::new("icq_config"); - -/// static info on one channel that doesn't change -pub const CHANNEL_INFO: Map<&str, ChannelInfo> = Map::new("channel_info"); - -pub const QUERY_RESULT_COUNTER: Item = Item::new("query_result_counter"); - -// pending queries lets us register a sequence number on a specific channel and set a corresponding enum. -// using the Origin enum, we can write a function for the callback on acknowledgement. -// we want to use an enum and some form of state here so we support callbacks for multiple queries batches together -// and different callbacks for the same set of queries from a different origin -pub const PENDING_QUERIES: Map<(u64, &str), Origin> = Map::new("pending_queries"); - -pub const REPLIES: Map = Map::new("replies"); - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub enum Origin { - Sample, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub struct Config { - pub default_timeout: u64, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub struct ChannelInfo { - /// id of this channel - pub id: String, - /// the remote channel/port we connect to - pub counterparty_endpoint: IbcEndpoint, - /// the connection this exists on (you can use to query client/consensus info) - pub connection_id: String, -} diff --git a/smart-contracts/osmosis/contracts/ica/src/test_helpers.rs b/smart-contracts/osmosis/contracts/ica/src/test_helpers.rs deleted file mode 100644 index 2dfac85af..000000000 --- a/smart-contracts/osmosis/contracts/ica/src/test_helpers.rs +++ /dev/null @@ -1,72 +0,0 @@ -#![cfg(test)] - -use crate::contract::instantiate; -use crate::ibc::{ibc_channel_connect, ibc_channel_open, ICA_ORDERING, ICA_VERSION}; -use crate::state::ChannelInfo; - -use cosmwasm_std::testing::{ - mock_dependencies, mock_env, mock_info, MockApi, MockQuerier, MockStorage, -}; -use cosmwasm_std::{ - DepsMut, IbcChannel, IbcChannelConnectMsg, IbcChannelOpenMsg, IbcEndpoint, OwnedDeps, -}; - -use crate::msg::InitMsg; - -pub const DEFAULT_TIMEOUT: u64 = 3600; // 1 hour, -pub const CONTRACT_PORT: &str = "ibc:wasm1234567890abcdef"; -pub const REMOTE_PORT: &str = "icahost"; -pub const CONNECTION_ID: &str = "connection-2"; - -pub fn mock_channel(channel_id: &str) -> IbcChannel { - IbcChannel::new( - IbcEndpoint { - port_id: CONTRACT_PORT.into(), - channel_id: channel_id.into(), - }, - IbcEndpoint { - port_id: REMOTE_PORT.into(), - channel_id: format!("{}5", channel_id), - }, - ICA_ORDERING, - ICA_VERSION, - CONNECTION_ID, - ) -} - -pub fn mock_channel_info(channel_id: &str) -> ChannelInfo { - ChannelInfo { - id: channel_id.to_string(), - counterparty_endpoint: IbcEndpoint { - port_id: REMOTE_PORT.into(), - channel_id: format!("{}5", channel_id), - }, - connection_id: CONNECTION_ID.into(), - } -} - -// we simulate instantiate and ack here -pub fn add_channel(mut deps: DepsMut, channel_id: &str) { - let channel = mock_channel(channel_id); - let open_msg = IbcChannelOpenMsg::new_init(channel.clone()); - ibc_channel_open(deps.branch(), mock_env(), open_msg).unwrap(); - let connect_msg = IbcChannelConnectMsg::new_ack(channel, ICA_VERSION); - ibc_channel_connect(deps.branch(), mock_env(), connect_msg).unwrap(); -} - -pub fn setup(channels: &[&str]) -> OwnedDeps { - let mut deps = mock_dependencies(); - - // instantiate an empty contract - let instantiate_msg = InitMsg { - default_timeout: DEFAULT_TIMEOUT, - }; - let info = mock_info(&String::from("anyone"), &[]); - let res = instantiate(deps.as_mut(), mock_env(), info, instantiate_msg).unwrap(); - assert_eq!(0, res.messages.len()); - - for channel in channels { - add_channel(deps.as_mut(), channel); - } - deps -} diff --git a/smart-contracts/osmosis/contracts/icq/.cargo/config b/smart-contracts/osmosis/contracts/icq/.cargo/config deleted file mode 100644 index dba2c3f7a..000000000 --- a/smart-contracts/osmosis/contracts/icq/.cargo/config +++ /dev/null @@ -1,5 +0,0 @@ -[alias] -wasm = "build --release --target wasm32-unknown-unknown" -unit-test = "test --lib" -schema = "run --example schema" -lint = "clippy -- -D warnings" \ No newline at end of file diff --git a/smart-contracts/osmosis/contracts/icq/Cargo.toml b/smart-contracts/osmosis/contracts/icq/Cargo.toml deleted file mode 100644 index ade605545..000000000 --- a/smart-contracts/osmosis/contracts/icq/Cargo.toml +++ /dev/null @@ -1,37 +0,0 @@ -[package] -name = "icq" -version = "0.1.0" -authors = ["Bo Du "] -edition = "2018" -description = "IBC Enabled contracts for interchain queries" -license = "Apache-2.0" -repository = "https://github.com/CosmWasm/cw-plus" -homepage = "https://cosmwasm.com" -documentation = "https://docs.cosmwasm.com" - -[lib] -crate-type = ["cdylib", "rlib"] - -[features] -# backtraces = ["cosmwasm-std/backtraces"] -# use library feature to disable all init/handle/query exports -library = [] - -[dependencies] -cosmwasm-std = { workspace = true } -osmosis-std = { workspace = true } -osmosis-std-derive = { workspace = true } -cosmwasm-schema = { workspace = true } -cw-storage-plus ={ workspace = true } -schemars ={ workspace = true } -serde = { workspace = true } -thiserror = { workspace = true } -prost = { workspace = true } -cw-controllers = { workspace = true } -cw-utils = { workspace = true } -cw2 = { workspace = true } -cw20 = { workspace = true } -cosmos-sdk-proto = { workspace = true } -base64 = { workspace = true } - -semver = "1" \ No newline at end of file diff --git a/smart-contracts/osmosis/contracts/icq/NOTICE b/smart-contracts/osmosis/contracts/icq/NOTICE deleted file mode 100644 index bc266454e..000000000 --- a/smart-contracts/osmosis/contracts/icq/NOTICE +++ /dev/null @@ -1,14 +0,0 @@ -CW20-ICS20: IBC Enabled contracts that receives CW20 tokens and sends them over ICS20 to a remote chain -Copyright (C) 2020 Confio OÜ - -Licensed under the Apache License, Version 2.0 (the "License"); -you may not use this file except in compliance with the License. -You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - -Unless required by applicable law or agreed to in writing, software -distributed under the License is distributed on an "AS IS" BASIS, -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -See the License for the specific language governing permissions and -limitations under the License. diff --git a/smart-contracts/osmosis/contracts/icq/README.md b/smart-contracts/osmosis/contracts/icq/README.md deleted file mode 100644 index 3f5f43452..000000000 --- a/smart-contracts/osmosis/contracts/icq/README.md +++ /dev/null @@ -1,32 +0,0 @@ -# ICQ - -This is an *IBC Enabled* contract that allows us to send ICQ queries from one chain over the standard ICQ -protocol to the bank module of another chain. - -## Workflow - -The contract starts with minimal state. It just stores a default timeout in seconds for all packets it sends. -Most importantly it binds a local IBC port to enable channel connections. - -An external party first needs to make one or more channels using this contract as one endpoint. It will use standard -unordered channels for the version negotiation. Once established, it manages a list of known channels. - -After there is at least one channel, you can send any ICQ query to this contract. It may optionally include a custom timeout. - -## Messages - -It only accepts ICQQueryMsg. The data sent along with that message must be a JSON-serialized -TransferMsg: - -## Queries - -Queries only make sense relative to the established channels of this contract. - -* `Port{}` - returns the port ID this contract has bound, so you can create channels. This info can be queried - via wasmd contract info query, but we expose another query here for convenience. -* `ListChannels{}` - returns a (currently unpaginated) list of all channels that have been created on this contract. - Returns their local channelId along with some basic metadata, like the remote port/channel and the connection they - run on top of. -* `Channel{id}` - returns more detailed information on one specific channel. In addition to the information available - in the list view, it returns the current outstanding balance on that channel, as well as the total amount that - has ever been sent on the channel. diff --git a/smart-contracts/osmosis/contracts/icq/examples/schema.rs b/smart-contracts/osmosis/contracts/icq/examples/schema.rs deleted file mode 100644 index 645b8b312..000000000 --- a/smart-contracts/osmosis/contracts/icq/examples/schema.rs +++ /dev/null @@ -1,23 +0,0 @@ -use std::env::current_dir; -use std::fs::create_dir_all; - -use cosmwasm_schema::{export_schema, remove_schemas, schema_for}; - -use icq::msg::{ - ChannelResponse, ExecuteMsg, ICQQueryMsg, InitMsg, ListChannelsResponse, PortResponse, QueryMsg, -}; - -fn main() { - let mut out_dir = current_dir().unwrap(); - out_dir.push("schema"); - create_dir_all(&out_dir).unwrap(); - remove_schemas(&out_dir).unwrap(); - - export_schema(&schema_for!(InitMsg), &out_dir); - export_schema(&schema_for!(ExecuteMsg), &out_dir); - export_schema(&schema_for!(QueryMsg), &out_dir); - export_schema(&schema_for!(ICQQueryMsg), &out_dir); - export_schema(&schema_for!(ChannelResponse), &out_dir); - export_schema(&schema_for!(ListChannelsResponse), &out_dir); - export_schema(&schema_for!(PortResponse), &out_dir); -} diff --git a/smart-contracts/osmosis/contracts/icq/schema/channel_response.json b/smart-contracts/osmosis/contracts/icq/schema/channel_response.json deleted file mode 100644 index 1744539e6..000000000 --- a/smart-contracts/osmosis/contracts/icq/schema/channel_response.json +++ /dev/null @@ -1,61 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "ChannelResponse", - "type": "object", - "required": [ - "info" - ], - "properties": { - "info": { - "description": "Information on the channel's connection", - "allOf": [ - { - "$ref": "#/definitions/ChannelInfo" - } - ] - } - }, - "definitions": { - "ChannelInfo": { - "type": "object", - "required": [ - "connection_id", - "counterparty_endpoint", - "id" - ], - "properties": { - "connection_id": { - "description": "the connection this exists on (you can use to query client/consensus info)", - "type": "string" - }, - "counterparty_endpoint": { - "description": "the remote channel/port we connect to", - "allOf": [ - { - "$ref": "#/definitions/IbcEndpoint" - } - ] - }, - "id": { - "description": "id of this channel", - "type": "string" - } - } - }, - "IbcEndpoint": { - "type": "object", - "required": [ - "channel_id", - "port_id" - ], - "properties": { - "channel_id": { - "type": "string" - }, - "port_id": { - "type": "string" - } - } - } - } -} diff --git a/smart-contracts/osmosis/contracts/icq/schema/execute_msg.json b/smart-contracts/osmosis/contracts/icq/schema/execute_msg.json deleted file mode 100644 index 87c427c4c..000000000 --- a/smart-contracts/osmosis/contracts/icq/schema/execute_msg.json +++ /dev/null @@ -1,148 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "ExecuteMsg", - "oneOf": [ - { - "type": "object", - "required": [ - "query" - ], - "properties": { - "query": { - "$ref": "#/definitions/ICQQueryMsg" - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "query_balance" - ], - "properties": { - "query_balance": { - "type": "object", - "required": [ - "address", - "channel", - "denom" - ], - "properties": { - "address": { - "type": "string" - }, - "channel": { - "type": "string" - }, - "denom": { - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "query_all_balance" - ], - "properties": { - "query_all_balance": { - "type": "object", - "required": [ - "address", - "channel" - ], - "properties": { - "address": { - "type": "string" - }, - "channel": { - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "query_mint" - ], - "properties": { - "query_mint": { - "type": "object", - "required": [ - "channel" - ], - "properties": { - "channel": { - "type": "string" - } - } - } - }, - "additionalProperties": false - } - ], - "definitions": { - "Binary": { - "description": "Binary is a wrapper around Vec to add base64 de/serialization with serde. It also adds some helper methods to help encode inline.\n\nThis is only needed as serde-json-{core,wasm} has a horrible encoding for Vec. See also .", - "type": "string" - }, - "ICQQueryMsg": { - "description": "This is the message we accept via Receive", - "type": "object", - "required": [ - "channel", - "requests" - ], - "properties": { - "channel": { - "description": "The local channel to send the packets on", - "type": "string" - }, - "requests": { - "type": "array", - "items": { - "$ref": "#/definitions/RequestQueryJSON" - } - }, - "timeout": { - "type": [ - "integer", - "null" - ], - "format": "uint64", - "minimum": 0.0 - } - } - }, - "RequestQueryJSON": { - "type": "object", - "required": [ - "data", - "height", - "path", - "prove" - ], - "properties": { - "data": { - "$ref": "#/definitions/Binary" - }, - "height": { - "type": "integer", - "format": "int64" - }, - "path": { - "type": "string" - }, - "prove": { - "type": "boolean" - } - } - } - } -} diff --git a/smart-contracts/osmosis/contracts/icq/schema/i_c_q_query_msg.json b/smart-contracts/osmosis/contracts/icq/schema/i_c_q_query_msg.json deleted file mode 100644 index 316cd71ce..000000000 --- a/smart-contracts/osmosis/contracts/icq/schema/i_c_q_query_msg.json +++ /dev/null @@ -1,60 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "ICQQueryMsg", - "description": "This is the message we accept via Receive", - "type": "object", - "required": [ - "channel", - "requests" - ], - "properties": { - "channel": { - "description": "The local channel to send the packets on", - "type": "string" - }, - "requests": { - "type": "array", - "items": { - "$ref": "#/definitions/RequestQueryJSON" - } - }, - "timeout": { - "type": [ - "integer", - "null" - ], - "format": "uint64", - "minimum": 0.0 - } - }, - "definitions": { - "Binary": { - "description": "Binary is a wrapper around Vec to add base64 de/serialization with serde. It also adds some helper methods to help encode inline.\n\nThis is only needed as serde-json-{core,wasm} has a horrible encoding for Vec. See also .", - "type": "string" - }, - "RequestQueryJSON": { - "type": "object", - "required": [ - "data", - "height", - "path", - "prove" - ], - "properties": { - "data": { - "$ref": "#/definitions/Binary" - }, - "height": { - "type": "integer", - "format": "int64" - }, - "path": { - "type": "string" - }, - "prove": { - "type": "boolean" - } - } - } - } -} diff --git a/smart-contracts/osmosis/contracts/icq/schema/init_msg.json b/smart-contracts/osmosis/contracts/icq/schema/init_msg.json deleted file mode 100644 index 33a291fc3..000000000 --- a/smart-contracts/osmosis/contracts/icq/schema/init_msg.json +++ /dev/null @@ -1,16 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "InitMsg", - "type": "object", - "required": [ - "default_timeout" - ], - "properties": { - "default_timeout": { - "description": "Default timeout for icq packets, specified in seconds", - "type": "integer", - "format": "uint64", - "minimum": 0.0 - } - } -} diff --git a/smart-contracts/osmosis/contracts/icq/schema/list_channels_response.json b/smart-contracts/osmosis/contracts/icq/schema/list_channels_response.json deleted file mode 100644 index 69f020c93..000000000 --- a/smart-contracts/osmosis/contracts/icq/schema/list_channels_response.json +++ /dev/null @@ -1,59 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "ListChannelsResponse", - "type": "object", - "required": [ - "channels" - ], - "properties": { - "channels": { - "type": "array", - "items": { - "$ref": "#/definitions/ChannelInfo" - } - } - }, - "definitions": { - "ChannelInfo": { - "type": "object", - "required": [ - "connection_id", - "counterparty_endpoint", - "id" - ], - "properties": { - "connection_id": { - "description": "the connection this exists on (you can use to query client/consensus info)", - "type": "string" - }, - "counterparty_endpoint": { - "description": "the remote channel/port we connect to", - "allOf": [ - { - "$ref": "#/definitions/IbcEndpoint" - } - ] - }, - "id": { - "description": "id of this channel", - "type": "string" - } - } - }, - "IbcEndpoint": { - "type": "object", - "required": [ - "channel_id", - "port_id" - ], - "properties": { - "channel_id": { - "type": "string" - }, - "port_id": { - "type": "string" - } - } - } - } -} diff --git a/smart-contracts/osmosis/contracts/icq/schema/port_response.json b/smart-contracts/osmosis/contracts/icq/schema/port_response.json deleted file mode 100644 index 4561b3a72..000000000 --- a/smart-contracts/osmosis/contracts/icq/schema/port_response.json +++ /dev/null @@ -1,13 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "PortResponse", - "type": "object", - "required": [ - "port_id" - ], - "properties": { - "port_id": { - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/icq/schema/query_msg.json b/smart-contracts/osmosis/contracts/icq/schema/query_msg.json deleted file mode 100644 index be42f505f..000000000 --- a/smart-contracts/osmosis/contracts/icq/schema/query_msg.json +++ /dev/null @@ -1,66 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "QueryMsg", - "oneOf": [ - { - "description": "Return the port ID bound by this contract. Returns PortResponse", - "type": "object", - "required": [ - "port" - ], - "properties": { - "port": { - "type": "object" - } - }, - "additionalProperties": false - }, - { - "description": "Show all channels we have connected to. Return type is ListChannelsResponse.", - "type": "object", - "required": [ - "list_channels" - ], - "properties": { - "list_channels": { - "type": "object" - } - }, - "additionalProperties": false - }, - { - "description": "Returns the details of the name channel, error if not created. Return type: ChannelResponse.", - "type": "object", - "required": [ - "channel" - ], - "properties": { - "channel": { - "type": "object", - "required": [ - "id" - ], - "properties": { - "id": { - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "description": "Show the Config. Returns ConfigResponse (currently including admin as well)", - "type": "object", - "required": [ - "config" - ], - "properties": { - "config": { - "type": "object" - } - }, - "additionalProperties": false - } - ] -} diff --git a/smart-contracts/osmosis/contracts/icq/src/contract.rs b/smart-contracts/osmosis/contracts/icq/src/contract.rs deleted file mode 100644 index b7e8485c2..000000000 --- a/smart-contracts/osmosis/contracts/icq/src/contract.rs +++ /dev/null @@ -1,324 +0,0 @@ -use cosmos_sdk_proto::cosmos::bank::v1beta1::{QueryAllBalancesRequest, QueryBalanceRequest}; -use cosmos_sdk_proto::tendermint::abci::RequestQuery; -#[cfg(not(feature = "library"))] -use cosmwasm_std::entry_point; -use cosmwasm_std::{ - to_json_binary, Binary, Deps, DepsMut, Env, IbcMsg, IbcQuery, MessageInfo, Order, PortIdResponse, - Reply, Response, StdResult, -}; -use osmosis_std::types::osmosis::gamm::v1beta1::QueryNumPoolsRequest; -use prost::Message; - -use cw2::set_contract_version; - -use crate::error::ContractError; -use crate::helpers::{prepare_query, Query}; -use crate::msg::{ - ChannelResponse, ConfigResponse, ExecuteMsg, ICQQueryMsg, InitMsg, InterchainQueryPacketData, - ListChannelsResponse, MigrateMsg, PortResponse, QueryMsg, -}; -use crate::proto::CosmosQuery; -use crate::state::{Config, CHANNEL_INFO, CONFIG, QUERY_RESULT_COUNTER}; - -// version info for migration info -const CONTRACT_NAME: &str = "crates.io:icq"; -const CONTRACT_VERSION: &str = env!("CARGO_PKG_VERSION"); - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn instantiate( - deps: DepsMut, - _env: Env, - _info: MessageInfo, - msg: InitMsg, -) -> Result { - set_contract_version(deps.storage, CONTRACT_NAME, CONTRACT_VERSION)?; - let cfg = Config { - default_timeout: msg.default_timeout, - }; - CONFIG.save(deps.storage, &cfg)?; - QUERY_RESULT_COUNTER.save(deps.storage, &0)?; - Ok(Response::default()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn execute( - deps: DepsMut, - env: Env, - _info: MessageInfo, - msg: ExecuteMsg, -) -> Result { - match msg { - ExecuteMsg::Query(msg) => execute_query(deps, env, msg), - ExecuteMsg::QueryBalance { - address, - denom, - channel, - } => execute_balance_query(deps, env, address, denom, channel), - ExecuteMsg::QueryAllBalance { address, channel } => { - execute_all_balance_query(deps, env, address, channel) - } - ExecuteMsg::QueryMint { channel } => execute_mint_params_query(deps, env, channel), - } -} - -pub fn execute_balance_query( - deps: DepsMut, - env: Env, - address: String, - denom: String, - channel: String, -) -> Result { - let timeout = prepare_query(deps.storage, env, &channel)?; - let query = QueryBalanceRequest { address, denom }; - let packet = Query::new() - .add_request( - query.encode_to_vec(), - "/cosmos.bank.v1beta1.Query/Balance".into(), - ) - .encode_pkt(); - - // prepare ibc message - let send_packet_msg = IbcMsg::SendPacket { - channel_id: channel, - data: to_json_binary(&packet)?, - timeout: timeout.into(), - }; - - let res = Response::new() - .add_message(send_packet_msg) - .add_attribute("action", "query"); - Ok(res) -} - -pub fn execute_all_balance_query( - deps: DepsMut, - env: Env, - address: String, - channel: String, -) -> Result { - let timeout = prepare_query(deps.storage, env, &channel)?; - - let query = QueryAllBalancesRequest { - address, - pagination: None, - }; - - let packet = Query::new() - .add_request( - query.encode_to_vec(), - "/cosmos.bank.v1beta1.Query/AllBalances".into(), - ) - .encode_pkt(); - - // prepare ibc message - let send_packet_msg = IbcMsg::SendPacket { - channel_id: channel, - data: to_json_binary(&packet)?, - timeout: timeout.into(), - }; - - let res = Response::new() - .add_message(send_packet_msg) - .add_attribute("action", "query"); - Ok(res) -} - -fn execute_mint_params_query( - deps: DepsMut, - env: Env, - channel: String, -) -> Result { - let timeout = prepare_query(deps.storage, env, &channel)?; - let data = QueryNumPoolsRequest {}; - let packet = Query::new() - .add_request( - data.encode_to_vec(), - "/osmosis.gamm.v1beta1.Query/NumPools".into(), - ) - .encode_pkt(); - - // prepare ibc message - let send_packet_msg = IbcMsg::SendPacket { - channel_id: channel, - data: to_json_binary(&packet)?, - timeout: timeout.into(), - }; - - let res = Response::new() - .add_message(send_packet_msg) - .add_attribute("action", "query"); - Ok(res) -} - -pub fn execute_query(deps: DepsMut, env: Env, msg: ICQQueryMsg) -> Result { - // ensure the requested channel is registered - if !CHANNEL_INFO.has(deps.storage, &msg.channel) { - return Err(ContractError::NoSuchChannel { id: msg.channel }); - } - let config = CONFIG.load(deps.storage)?; - // delta from user is in seconds - let timeout_delta = match msg.timeout { - Some(t) => t, - None => config.default_timeout, - }; - // timeout is in nanoseconds - let timeout = env.block.time.plus_seconds(timeout_delta); - let num_requests = msg.requests.len(); - - let q = CosmosQuery { - requests: msg - .requests - .iter() - .map(|req| RequestQuery { - data: req.data.clone().into(), - path: req.path.clone(), - height: req.height, - prove: req.prove, - }) - .collect::>(), - }; - let mut data = Vec::new(); - if q.encode(&mut data).is_err() { - return Err(ContractError::EncodingFail {}); - } - - let packet = InterchainQueryPacketData { data }; - // prepare ibc message - let send_packet_msg = IbcMsg::SendPacket { - channel_id: msg.channel, - data: to_json_binary(&packet)?, - timeout: timeout.into(), - }; - - // send response - let res = Response::new() - .add_message(send_packet_msg) - .add_attribute("action", "query") - .add_attribute("num_requests", num_requests.to_string()); - Ok(res) -} - -#[entry_point] -pub fn reply(_deps: DepsMut, _env: Env, _msg: Reply) -> StdResult { - Ok(Response::default()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn migrate(_deps: DepsMut, _env: Env, _msg: MigrateMsg) -> Result { - Ok(Response::new()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn query(deps: Deps, _env: Env, msg: QueryMsg) -> StdResult { - match msg { - QueryMsg::Port {} => to_json_binary(&query_port(deps)?), - QueryMsg::ListChannels {} => to_json_binary(&query_list(deps)?), - QueryMsg::Channel { id } => to_json_binary(&query_channel(deps, id)?), - QueryMsg::Config {} => to_json_binary(&query_config(deps)?), - } -} - -fn query_port(deps: Deps) -> StdResult { - let query = IbcQuery::PortId {}.into(); - let PortIdResponse { port_id } = deps.querier.query(&query)?; - Ok(PortResponse { port_id }) -} - -fn query_list(deps: Deps) -> StdResult { - let channels = CHANNEL_INFO - .range_raw(deps.storage, None, None, Order::Ascending) - .map(|r| r.map(|(_, v)| v)) - .collect::>()?; - Ok(ListChannelsResponse { channels }) -} - -// make public for ibc tests -pub fn query_channel(deps: Deps, id: String) -> StdResult { - let info = CHANNEL_INFO.load(deps.storage, &id)?; - Ok(ChannelResponse { info }) -} - -fn query_config(deps: Deps) -> StdResult { - let cfg = CONFIG.load(deps.storage)?; - let res = ConfigResponse { - default_timeout: cfg.default_timeout, - }; - Ok(res) -} - -#[cfg(test)] -mod test { - use super::*; - use crate::msg::RequestQueryJSON; - use crate::test_helpers::*; - - use cosmwasm_std::testing::{mock_env, mock_info}; - use cosmwasm_std::{from_json, CosmosMsg, StdError}; - - #[test] - fn setup_and_query() { - let deps = setup(&["channel-3", "channel-7"]); - - let raw_list = query(deps.as_ref(), mock_env(), QueryMsg::ListChannels {}).unwrap(); - let list_res: ListChannelsResponse = from_json(&raw_list).unwrap(); - assert_eq!(2, list_res.channels.len()); - assert_eq!(mock_channel_info("channel-3"), list_res.channels[0]); - assert_eq!(mock_channel_info("channel-7"), list_res.channels[1]); - - let raw_channel = query( - deps.as_ref(), - mock_env(), - QueryMsg::Channel { - id: "channel-3".to_string(), - }, - ) - .unwrap(); - let chan_res: ChannelResponse = from_json(&raw_channel).unwrap(); - assert_eq!(chan_res.info, mock_channel_info("channel-3")); - - let err = query( - deps.as_ref(), - mock_env(), - QueryMsg::Channel { - id: "channel-10".to_string(), - }, - ) - .unwrap_err(); - assert_eq!(err, StdError::not_found("icq::state::ChannelInfo")); - } - - #[test] - fn execute_query_success() { - let send_channel = "channel-5"; - let mut deps = setup(&[send_channel, "channel-10"]); - - let requests = vec![RequestQueryJSON { - data: Binary::from([0, 1, 0, 1]), - path: "/path".to_string(), - height: 0, - prove: false, - }]; - let q = ICQQueryMsg { - channel: send_channel.to_string(), - requests, - timeout: None, - }; - - let msg = ExecuteMsg::Query(q); - let info = mock_info("foobar", &[]); - let res = execute(deps.as_mut(), mock_env(), info, msg).unwrap(); - if let CosmosMsg::Ibc(IbcMsg::SendPacket { - channel_id, - data, - timeout, - }) = &res.messages[0].msg - { - let expected_timeout = mock_env().block.time.plus_seconds(DEFAULT_TIMEOUT); - assert_eq!(timeout, &expected_timeout.into()); - assert_eq!(channel_id.as_str(), send_channel); - let _: InterchainQueryPacketData = from_json(data).unwrap(); - } else { - panic!("Unexpected return message: {:?}", res.messages[0]); - } - } -} diff --git a/smart-contracts/osmosis/contracts/icq/src/error.rs b/smart-contracts/osmosis/contracts/icq/src/error.rs deleted file mode 100644 index e531295a8..000000000 --- a/smart-contracts/osmosis/contracts/icq/src/error.rs +++ /dev/null @@ -1,50 +0,0 @@ -use std::string::FromUtf8Error; -use thiserror::Error; - -use cosmwasm_std::StdError; - -/// Never is a placeholder to ensure we don't return any errors -#[derive(Error, Debug)] -pub enum Never {} - -#[derive(Error, Debug, PartialEq)] -pub enum ContractError { - #[error("{0}")] - Std(#[from] StdError), - - #[error("Channel doesn't exist: {id}")] - NoSuchChannel { id: String }, - - #[error("Only supports channel with ibc version icq-1, got {version}")] - InvalidIbcVersion { version: String }, - - #[error("Only supports unordered channel")] - OnlyOrderedChannel {}, - - #[error("Parsed port from denom ({port}) doesn't match packet")] - FromOtherPort { port: String }, - - #[error("Parsed channel from denom ({channel}) doesn't match packet")] - FromOtherChannel { channel: String }, - - #[error("Cannot migrate from different contract type: {previous_contract}")] - CannotMigrate { previous_contract: String }, - - #[error("Cannot migrate from unsupported version: {previous_version}")] - CannotMigrateVersion { previous_version: String }, - - #[error("Failed to proto encode")] - EncodingFail, - - #[error("Failed to proto decode")] - DecodingFail, - - #[error("Only the governance contract can do this")] - Unauthorized, -} - -impl From for ContractError { - fn from(_: FromUtf8Error) -> Self { - ContractError::Std(StdError::invalid_utf8("parsing denom key")) - } -} diff --git a/smart-contracts/osmosis/contracts/icq/src/helpers.rs b/smart-contracts/osmosis/contracts/icq/src/helpers.rs deleted file mode 100644 index 264f46647..000000000 --- a/smart-contracts/osmosis/contracts/icq/src/helpers.rs +++ /dev/null @@ -1,147 +0,0 @@ -use crate::{ - msg::InterchainQueryPacketData, - proto::{CosmosQuery, CosmosResponse}, - state::{CHANNEL_INFO, CONFIG, QUERY_RESULT_COUNTER}, - ContractError, -}; -use cosmos_sdk_proto::tendermint::abci::RequestQuery; -use cosmwasm_std::{attr, DepsMut, Env, IbcBasicResponse, IbcPacket, Storage, Timestamp}; -use prost::Message; - -pub fn prepare_query( - storage: &dyn Storage, - env: Env, - channel: &str, -) -> Result { - // ensure the requested channel is registered - if !CHANNEL_INFO.has(storage, channel) { - return Err(ContractError::NoSuchChannel { id: channel.into() }); - } - let config = CONFIG.load(storage)?; - // delta from user is in seconds - let timeout_delta = config.default_timeout; - - // timeout is in nanoseconds - Ok(env.block.time.plus_seconds(timeout_delta)) -} - -// for our sample origin callback, we increment the query counter and leave it at that -pub fn handle_sample_callback( - deps: DepsMut, - _env: Env, - response: CosmosResponse, - _original: IbcPacket, -) -> Result { - let attrs = vec![ - attr("action", "acknowledge"), - attr("num_messages", response.responses.len().to_string()), - attr("success", "true"), - ]; - - // Store result counter. - let mut counter = QUERY_RESULT_COUNTER.load(deps.storage)?; - counter += response.responses.len() as u64; - QUERY_RESULT_COUNTER.save(deps.storage, &counter)?; - Ok(IbcBasicResponse::new().add_attributes(attrs)) -} - -#[derive(Clone, Debug, PartialEq)] -pub struct Query { - requests: Vec, -} - -impl Query { - pub fn new() -> Query { - Query { - requests: Vec::new(), - } - } - - pub fn add_request(mut self, data: Vec, path: String) -> Self { - self.requests.push(RequestQuery { - data, - path, - height: 0, - prove: false, - }); - self - } - - pub fn encode(self) -> Vec { - CosmosQuery { - requests: self.requests, - } - .encode_to_vec() - } - - pub fn encode_pkt(self) -> InterchainQueryPacketData { - InterchainQueryPacketData { - data: self.encode(), - } - } -} - -#[cfg(test)] -mod test { - use super::*; - - #[test] - pub fn single_query_works() { - let req = RequestQuery { - data: vec![1, 0, 1, 0], - path: "/cosmos.bank.v1beta1.Query/AllBalances".into(), - height: 0, - prove: false, - }; - - let data = Query::new().add_request(req.data.clone(), req.path.clone()); - - assert_eq!( - data, - Query { - requests: vec![req.clone()] - } - ); - assert_eq!( - data.encode(), - CosmosQuery { - requests: vec![req] - } - .encode_to_vec() - ) - } - - #[test] - pub fn multiple_query_works() { - let req1 = RequestQuery { - data: vec![1, 0, 1, 0], - path: "/cosmos.bank.v1beta1.Query/AllBalances".into(), - height: 0, - prove: false, - }; - let req2 = RequestQuery { - data: vec![1, 0, 0, 0], - path: "/cosmos.bank.v1beta1.Query/Balance".into(), - height: 0, - prove: false, - }; - - let data = Query::new() - .add_request(req1.data.clone(), req1.path.clone()) - .add_request(req2.data.clone(), req2.path.clone()); - - assert_eq!( - data, - Query { - requests: vec![req1.clone(), req2.clone()] - } - ); - assert_eq!( - data.encode(), - CosmosQuery { - requests: vec![req1, req2] - } - .encode_to_vec() - ) - } -} diff --git a/smart-contracts/osmosis/contracts/icq/src/ibc.rs b/smart-contracts/osmosis/contracts/icq/src/ibc.rs deleted file mode 100644 index 1cd280716..000000000 --- a/smart-contracts/osmosis/contracts/icq/src/ibc.rs +++ /dev/null @@ -1,221 +0,0 @@ -use prost::bytes::Bytes; -use prost::Message; -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -use cosmwasm_std::{ - attr, entry_point, from_json, Binary, DepsMut, Env, IbcBasicResponse, IbcChannel, - IbcChannelCloseMsg, IbcChannelConnectMsg, IbcChannelOpenMsg, IbcOrder, IbcPacket, - IbcPacketAckMsg, IbcPacketReceiveMsg, IbcPacketTimeoutMsg, IbcReceiveResponse, -}; - -use crate::error::{ContractError, Never}; -use crate::helpers::handle_sample_callback; -use crate::proto::CosmosResponse; -use crate::state::{ChannelInfo, CHANNEL_INFO}; - -pub const ICQ_VERSION: &str = "icq-1"; -pub const ICQ_ORDERING: IbcOrder = IbcOrder::Unordered; - -/// This is compatible with the JSON serialization -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug, Default)] -pub struct InterchainQueryPacketAck { - pub data: Binary, -} - -impl InterchainQueryPacketAck { - pub fn new(data: Binary) -> Self { - InterchainQueryPacketAck { data } - } - - pub fn validate(&self) -> Result<(), ContractError> { - Ok(()) - } -} - -/// This is a generic ICS acknowledgement format. -/// Proto defined here: https://github.com/cosmos/cosmos-sdk/blob/v0.42.0/proto/ibc/core/channel/v1/channel.proto#L141-L147 -/// This is compatible with the JSON serialization -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub enum IcsAck { - Result(Binary), - Error(String), -} - -#[cfg_attr(not(feature = "library"), entry_point)] -/// enforces ordering and versioning constraints -pub fn ibc_channel_open( - _deps: DepsMut, - _env: Env, - msg: IbcChannelOpenMsg, -) -> Result<(), ContractError> { - enforce_order_and_version(msg.channel(), msg.counterparty_version())?; - Ok(()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -/// record the channel in CHANNEL_INFO -pub fn ibc_channel_connect( - deps: DepsMut, - _env: Env, - msg: IbcChannelConnectMsg, -) -> Result { - // we need to check the counter party version in try and ack (sometimes here) - enforce_order_and_version(msg.channel(), msg.counterparty_version())?; - - let channel: IbcChannel = msg.into(); - let info = ChannelInfo { - id: channel.endpoint.channel_id, - counterparty_endpoint: channel.counterparty_endpoint, - connection_id: channel.connection_id, - }; - CHANNEL_INFO.save(deps.storage, &info.id, &info)?; - - Ok(IbcBasicResponse::default()) -} - -fn enforce_order_and_version( - channel: &IbcChannel, - counterparty_version: Option<&str>, -) -> Result<(), ContractError> { - if channel.version != ICQ_VERSION { - return Err(ContractError::InvalidIbcVersion { - version: channel.version.clone(), - }); - } - if let Some(version) = counterparty_version { - if version != ICQ_VERSION { - return Err(ContractError::InvalidIbcVersion { - version: version.to_string(), - }); - } - } - if channel.order != ICQ_ORDERING { - return Err(ContractError::OnlyOrderedChannel {}); - } - Ok(()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn ibc_channel_close( - _deps: DepsMut, - _env: Env, - _channel: IbcChannelCloseMsg, -) -> Result { - // TODO: what to do here? - // we will have locked funds that need to be returned somehow - unimplemented!(); -} - -#[cfg_attr(not(feature = "library"), entry_point)] -/// Check to see if we have any balance here -/// We should not return an error if possible, but rather an acknowledgement of failure -pub fn ibc_packet_receive( - _deps: DepsMut, - _env: Env, - _msg: IbcPacketReceiveMsg, -) -> Result { - // Contract does not handle packets/queries. - unimplemented!(); -} - -#[cfg_attr(not(feature = "library"), entry_point)] -/// check if success or failure and update balance, or return funds -pub fn ibc_packet_ack( - deps: DepsMut, - env: Env, - msg: IbcPacketAckMsg, -) -> Result { - // Design decision: should we trap error like in receive? - // TODO: unsure... as it is now a failed ack handling would revert the tx and would be - // retried again and again. is that good? - let ics_msg: IcsAck = from_json(&msg.acknowledgement.data)?; - match ics_msg { - IcsAck::Result(data) => on_packet_success(deps, data, msg.original_packet, env), - IcsAck::Error(err) => on_packet_failure(deps, msg.original_packet, err), - } -} - -#[cfg_attr(not(feature = "library"), entry_point)] -/// return fund to original sender (same as failure in ibc_packet_ack) -pub fn ibc_packet_timeout( - deps: DepsMut, - _env: Env, - msg: IbcPacketTimeoutMsg, -) -> Result { - // TODO: trap error like in receive? (same question as ack above) - let packet = msg.packet; - on_packet_failure(deps, packet, "timeout".to_string()) -} - -fn on_packet_success( - deps: DepsMut, - data: Binary, - original: IbcPacket, - env: Env, -) -> Result { - let ack: InterchainQueryPacketAck = from_json(&data)?; - - let buf = Bytes::copy_from_slice(ack.data.as_slice()); - let resp: CosmosResponse = match CosmosResponse::decode(buf) { - Ok(resp) => resp, - Err(_) => return Err(ContractError::DecodingFail {}), - }; - - handle_sample_callback(deps, env, resp, original) -} - -fn on_packet_failure( - _deps: DepsMut, - _packet: IbcPacket, - err: String, -) -> Result { - let attributes = vec![ - attr("action", "acknowledge"), - attr("success", "false"), - attr("error", err), - ]; - - Ok(IbcBasicResponse::new().add_attributes(attributes)) -} - -#[cfg(test)] -mod test { - - use cosmwasm_std::{ - testing::{mock_dependencies, mock_env}, - Binary, IbcAcknowledgement, IbcEndpoint, IbcPacket, IbcPacketAckMsg, IbcTimeout, Timestamp, - }; - - use crate::{state::QUERY_RESULT_COUNTER, ContractError}; - - use super::ibc_packet_ack; - - #[test] - fn test_ibc_packet_ack() -> Result<(), ContractError> { - let mut deps = mock_dependencies(); - let timeout = IbcTimeout::with_timestamp(Timestamp::from_nanos(0)); - let src = IbcEndpoint { - port_id: "port-0".to_string(), - channel_id: "channel-0".to_string(), - }; - let dest = IbcEndpoint { - port_id: "port-1".to_string(), - channel_id: "channel-1".to_string(), - }; - let packet = IbcPacket::new(Binary::default(), src, dest, 0, timeout); - let ack = IbcAcknowledgement::new(Binary::from_base64( - "eyJyZXN1bHQiOiJleUprWVhSaElqb2lRMmRaU1VWcmFXUnpaMFU5SW4wPSJ9", - )?); - let msg = IbcPacketAckMsg::new(ack, packet); - QUERY_RESULT_COUNTER.save(deps.as_mut().storage, &0)?; // Save a 0 - match ibc_packet_ack(deps.as_mut(), mock_env(), msg) { - Ok(_) => { - assert_eq!(QUERY_RESULT_COUNTER.load(deps.as_mut().storage)?, 1); - Ok(()) - } - Err(err) => Err(err), - } - } -} diff --git a/smart-contracts/osmosis/contracts/icq/src/lib.rs b/smart-contracts/osmosis/contracts/icq/src/lib.rs deleted file mode 100644 index c785d5eef..000000000 --- a/smart-contracts/osmosis/contracts/icq/src/lib.rs +++ /dev/null @@ -1,14 +0,0 @@ -pub mod contract; -mod error; -mod helpers; -pub mod ibc; -pub mod msg; -mod proto; -pub mod state; -mod test_helpers; - -pub use crate::error::ContractError; - -// pub mod items { -// include!(concat!(env!("OUT_DIR"), "/queries.rs")); -// } diff --git a/smart-contracts/osmosis/contracts/icq/src/msg.rs b/smart-contracts/osmosis/contracts/icq/src/msg.rs deleted file mode 100644 index 6fc91dac2..000000000 --- a/smart-contracts/osmosis/contracts/icq/src/msg.rs +++ /dev/null @@ -1,100 +0,0 @@ -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -use crate::state::ChannelInfo; -use cosmwasm_std::Binary; - -#[derive(Serialize, Deserialize, Clone, Debug, JsonSchema)] -pub struct InitMsg { - /// Default timeout for icq packets, specified in seconds - pub default_timeout: u64, -} - -#[derive(Serialize, Deserialize, Clone, Debug, JsonSchema)] -pub struct MigrateMsg { - pub default_gas_limit: Option, -} - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -#[serde(rename_all = "snake_case")] -pub enum ExecuteMsg { - Query(ICQQueryMsg), - QueryBalance { - address: String, - denom: String, - channel: String, - }, - QueryAllBalance { - address: String, - channel: String, - }, - QueryMint { - channel: String, - }, -} - -/// This is the message we accept via Receive -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -pub struct ICQQueryMsg { - /// The local channel to send the packets on - pub channel: String, - pub requests: Vec, - // How long the packet lives in seconds. If not specified, use default_timeout - pub timeout: Option, -} - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -pub struct RequestQueryJSON { - pub data: Binary, - pub path: String, - pub height: i64, - pub prove: bool, -} - -// ResponseQuery does not contain all of the response fields. -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -pub struct ResponseQuery { - pub key: Binary, - pub value: String, - pub height: i64, -} - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -pub struct InterchainQueryPacketData { - pub data: Vec, -} - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -#[serde(rename_all = "snake_case")] -pub enum QueryMsg { - /// Return the port ID bound by this contract. Returns PortResponse - Port {}, - /// Show all channels we have connected to. Return type is ListChannelsResponse. - ListChannels {}, - /// Returns the details of the name channel, error if not created. - /// Return type: ChannelResponse. - Channel { id: String }, - /// Show the Config. Returns ConfigResponse (currently including admin as well) - Config {}, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub struct ListChannelsResponse { - pub channels: Vec, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub struct ChannelResponse { - /// Information on the channel's connection - pub info: ChannelInfo, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub struct PortResponse { - pub port_id: String, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub struct ConfigResponse { - pub default_timeout: u64, -} diff --git a/smart-contracts/osmosis/contracts/icq/src/proto.rs b/smart-contracts/osmosis/contracts/icq/src/proto.rs deleted file mode 100644 index 31cd1090c..000000000 --- a/smart-contracts/osmosis/contracts/icq/src/proto.rs +++ /dev/null @@ -1,15 +0,0 @@ -use cosmos_sdk_proto::tendermint::abci::{RequestQuery, ResponseQuery}; - -/// CosmosQuery contains a list of tendermint ABCI query requests. It should be used when sending queries to an SDK host chain. -#[derive(Clone, PartialEq, ::prost::Message)] -pub struct CosmosQuery { - #[prost(message, repeated, tag = "1")] - pub requests: ::prost::alloc::vec::Vec, -} - -/// CosmosResponse contains a list of tendermint ABCI query responses. It should be used when receiving responses from an SDK host chain. -#[derive(Clone, PartialEq, ::prost::Message)] -pub struct CosmosResponse { - #[prost(message, repeated, tag = "1")] - pub responses: ::prost::alloc::vec::Vec, -} diff --git a/smart-contracts/osmosis/contracts/icq/src/state.rs b/smart-contracts/osmosis/contracts/icq/src/state.rs deleted file mode 100644 index 1d8a115dd..000000000 --- a/smart-contracts/osmosis/contracts/icq/src/state.rs +++ /dev/null @@ -1,32 +0,0 @@ -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -use cosmwasm_std::IbcEndpoint; -use cw_storage_plus::{Item, Map}; - -pub const CONFIG: Item = Item::new("icq_config"); - -/// static info on one channel that doesn't change -pub const CHANNEL_INFO: Map<&str, ChannelInfo> = Map::new("channel_info"); - -pub const QUERY_RESULT_COUNTER: Item = Item::new("query_result_counter"); - -// pending queries lets us register a sequence number on a specific channel and set a corresponding enum. -// using the Origin enum, we can write a function for the callback on acknowledgement. -// we want to use an enum and some form of state here so we support callbacks for multiple queries batches together -// and different callbacks for the same set of queries from a different origin - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub struct Config { - pub default_timeout: u64, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -pub struct ChannelInfo { - /// id of this channel - pub id: String, - /// the remote channel/port we connect to - pub counterparty_endpoint: IbcEndpoint, - /// the connection this exists on (you can use to query client/consensus info) - pub connection_id: String, -} diff --git a/smart-contracts/osmosis/contracts/icq/src/test_helpers.rs b/smart-contracts/osmosis/contracts/icq/src/test_helpers.rs deleted file mode 100644 index 0c7ced3de..000000000 --- a/smart-contracts/osmosis/contracts/icq/src/test_helpers.rs +++ /dev/null @@ -1,72 +0,0 @@ -#![cfg(test)] - -use crate::contract::instantiate; -use crate::ibc::{ibc_channel_connect, ibc_channel_open, ICQ_ORDERING, ICQ_VERSION}; -use crate::state::ChannelInfo; - -use cosmwasm_std::testing::{ - mock_dependencies, mock_env, mock_info, MockApi, MockQuerier, MockStorage, -}; -use cosmwasm_std::{ - DepsMut, IbcChannel, IbcChannelConnectMsg, IbcChannelOpenMsg, IbcEndpoint, OwnedDeps, -}; - -use crate::msg::InitMsg; - -pub const DEFAULT_TIMEOUT: u64 = 3600; // 1 hour, -pub const CONTRACT_PORT: &str = "ibc:wasm1234567890abcdef"; -pub const REMOTE_PORT: &str = "icqhost"; -pub const CONNECTION_ID: &str = "connection-2"; - -pub fn mock_channel(channel_id: &str) -> IbcChannel { - IbcChannel::new( - IbcEndpoint { - port_id: CONTRACT_PORT.into(), - channel_id: channel_id.into(), - }, - IbcEndpoint { - port_id: REMOTE_PORT.into(), - channel_id: format!("{channel_id}5"), - }, - ICQ_ORDERING, - ICQ_VERSION, - CONNECTION_ID, - ) -} - -pub fn mock_channel_info(channel_id: &str) -> ChannelInfo { - ChannelInfo { - id: channel_id.to_string(), - counterparty_endpoint: IbcEndpoint { - port_id: REMOTE_PORT.into(), - channel_id: format!("{channel_id}5"), - }, - connection_id: CONNECTION_ID.into(), - } -} - -// we simulate instantiate and ack here -pub fn add_channel(mut deps: DepsMut, channel_id: &str) { - let channel = mock_channel(channel_id); - let open_msg = IbcChannelOpenMsg::new_init(channel.clone()); - ibc_channel_open(deps.branch(), mock_env(), open_msg).unwrap(); - let connect_msg = IbcChannelConnectMsg::new_ack(channel, ICQ_VERSION); - ibc_channel_connect(deps.branch(), mock_env(), connect_msg).unwrap(); -} - -pub fn setup(channels: &[&str]) -> OwnedDeps { - let mut deps = mock_dependencies(); - - // instantiate an empty contract - let instantiate_msg = InitMsg { - default_timeout: DEFAULT_TIMEOUT, - }; - let info = mock_info(&String::from("anyone"), &[]); - let res = instantiate(deps.as_mut(), mock_env(), info, instantiate_msg).unwrap(); - assert_eq!(0, res.messages.len()); - - for channel in channels { - add_channel(deps.as_mut(), channel); - } - deps -} diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/.cargo/config b/smart-contracts/osmosis/contracts/intergamm-bindings-test/.cargo/config deleted file mode 100644 index 336b618a1..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/.cargo/config +++ /dev/null @@ -1,4 +0,0 @@ -[alias] -wasm = "build --release --target wasm32-unknown-unknown" -unit-test = "test --lib" -schema = "run --example schema" diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/.editorconfig b/smart-contracts/osmosis/contracts/intergamm-bindings-test/.editorconfig deleted file mode 100644 index 3d36f20b1..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/.editorconfig +++ /dev/null @@ -1,11 +0,0 @@ -root = true - -[*] -indent_style = space -indent_size = 2 -charset = utf-8 -trim_trailing_whitespace = true -insert_final_newline = true - -[*.rs] -indent_size = 4 diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/.gitignore b/smart-contracts/osmosis/contracts/intergamm-bindings-test/.gitignore deleted file mode 100644 index dfdaaa6bc..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/.gitignore +++ /dev/null @@ -1,15 +0,0 @@ -# Build results -/target - -# Cargo+Git helper file (https://github.com/rust-lang/cargo/blob/0.44.1/src/cargo/sources/git/utils.rs#L320-L327) -.cargo-ok - -# Text file backups -**/*.rs.bk - -# macOS -.DS_Store - -# IDEs -*.iml -.idea diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/.gitpod.Dockerfile b/smart-contracts/osmosis/contracts/intergamm-bindings-test/.gitpod.Dockerfile deleted file mode 100644 index bff8bc536..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/.gitpod.Dockerfile +++ /dev/null @@ -1,17 +0,0 @@ -### wasmd ### -FROM cosmwasm/wasmd:v0.18.0 as wasmd - -### rust-optimizer ### -FROM cosmwasm/rust-optimizer:0.11.5 as rust-optimizer - -FROM gitpod/workspace-full:latest - -COPY --from=wasmd /usr/bin/wasmd /usr/local/bin/wasmd -COPY --from=wasmd /opt/* /opt/ - -RUN sudo apt-get update \ - && sudo apt-get install -y jq \ - && sudo rm -rf /var/lib/apt/lists/* - -RUN rustup update stable \ - && rustup target add wasm32-unknown-unknown diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/.gitpod.yml b/smart-contracts/osmosis/contracts/intergamm-bindings-test/.gitpod.yml deleted file mode 100644 index d03610c21..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/.gitpod.yml +++ /dev/null @@ -1,10 +0,0 @@ -image: cosmwasm/cw-gitpod-base:v0.16 - -vscode: - extensions: - - rust-lang.rust - -tasks: - - name: Dependencies & Build - init: | - cargo build diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/Cargo.lock b/smart-contracts/osmosis/contracts/intergamm-bindings-test/Cargo.lock deleted file mode 100644 index 91fb09ff8..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/Cargo.lock +++ /dev/null @@ -1,738 +0,0 @@ -# This file is automatically @generated by Cargo. -# It is not intended for manual editing. -version = 3 - -[[package]] -name = "anyhow" -version = "1.0.57" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "08f9b8508dccb7687a1d6c4ce66b2b0ecef467c94667de27d8d7fe1f8d2a9cdc" - -[[package]] -name = "base16ct" -version = "0.1.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "349a06037c7bf932dd7e7d1f653678b2038b9ad46a74102f1fc7bd7872678cce" - -[[package]] -name = "base64" -version = "0.13.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "904dfeac50f3cdaba28fc6f57fdcddb75f49ed61346676a78c4ffe55877802fd" - -[[package]] -name = "base64ct" -version = "1.5.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dea908e7347a8c64e378c17e30ef880ad73e3b4498346b055c2c00ea342f3179" - -[[package]] -name = "block-buffer" -version = "0.9.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4152116fd6e9dadb291ae18fc1ec3575ed6d84c29642d97890f4b4a3417297e4" -dependencies = [ - "generic-array", -] - -[[package]] -name = "byteorder" -version = "1.4.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "14c189c53d098945499cdfa7ecc63567cf3886b3332b312a5b4585d8d3a6a610" - -[[package]] -name = "bytes" -version = "1.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c4872d67bab6358e59559027aa3b9157c53d9358c51423c17554809a8858e0f8" - -[[package]] -name = "cfg-if" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" - -[[package]] -name = "const-oid" -version = "0.7.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e4c78c047431fee22c1a7bb92e00ad095a02a983affe4d8a72e2a2c62c1b94f3" - -[[package]] -name = "cosmwasm-crypto" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5eb0afef2325df81aadbf9be1233f522ed8f6e91df870c764bc44cca2b1415bd" -dependencies = [ - "digest", - "ed25519-zebra", - "k256", - "rand_core 0.6.3", - "thiserror", -] - -[[package]] -name = "cosmwasm-derive" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4b36e527620a2a3e00e46b6e731ab6c9b68d11069c986f7d7be8eba79ef081a4" -dependencies = [ - "syn", -] - -[[package]] -name = "cosmwasm-schema" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "772e80bbad231a47a2068812b723a1ff81dd4a0d56c9391ac748177bea3a61da" -dependencies = [ - "schemars", - "serde_json", -] - -[[package]] -name = "cosmwasm-std" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "875994993c2082a6fcd406937bf0fca21c349e4a624f3810253a14fa83a3a195" -dependencies = [ - "base64", - "cosmwasm-crypto", - "cosmwasm-derive", - "forward_ref", - "schemars", - "serde", - "serde-json-wasm", - "thiserror", - "uint", -] - -[[package]] -name = "cosmwasm-storage" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d18403b07304d15d304dad11040d45bbcaf78d603b4be3fb5e2685c16f9229b5" -dependencies = [ - "cosmwasm-std", - "serde", -] - -[[package]] -name = "cpufeatures" -version = "0.2.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "59a6001667ab124aebae2a495118e11d30984c3a653e99d86d58971708cf5e4b" -dependencies = [ - "libc", -] - -[[package]] -name = "crunchy" -version = "0.2.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7a81dae078cea95a014a339291cec439d2f232ebe854a9d672b796c6afafa9b7" - -[[package]] -name = "crypto-bigint" -version = "0.3.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "03c6a1d5fa1de37e071642dfa44ec552ca5b299adb128fab16138e24b548fd21" -dependencies = [ - "generic-array", - "rand_core 0.6.3", - "subtle", - "zeroize", -] - -[[package]] -name = "crypto-mac" -version = "0.11.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b1d1a86f49236c215f271d40892d5fc950490551400b02ef360692c29815c714" -dependencies = [ - "generic-array", - "subtle", -] - -[[package]] -name = "curve25519-dalek" -version = "3.2.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0b9fdf9972b2bd6af2d913799d9ebc165ea4d2e65878e329d9c6b372c4491b61" -dependencies = [ - "byteorder", - "digest", - "rand_core 0.5.1", - "subtle", - "zeroize", -] - -[[package]] -name = "cw-multi-test" -version = "0.13.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cbea57e5be4a682268a5eca1a57efece57a54ff216bfd87603d5e864aad40e12" -dependencies = [ - "anyhow", - "cosmwasm-std", - "cosmwasm-storage", - "cw-storage-plus", - "cw-utils", - "derivative", - "itertools", - "prost", - "schemars", - "serde", - "thiserror", -] - -[[package]] -name = "cw-storage-plus" -version = "0.13.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9336ecef1e19d56cf6e3e932475fc6a3dee35eec5a386e07917a1d1ba6bb0e35" -dependencies = [ - "cosmwasm-std", - "schemars", - "serde", -] - -[[package]] -name = "cw-utils" -version = "0.13.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "babd2c090f39d07ce5bf2556962305e795daa048ce20a93709eb591476e4a29e" -dependencies = [ - "cosmwasm-std", - "schemars", - "serde", - "thiserror", -] - -[[package]] -name = "cw2" -version = "0.13.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "993df11574f29574dd443eb0c189484bb91bc0638b6de3e32ab7f9319c92122d" -dependencies = [ - "cosmwasm-std", - "cw-storage-plus", - "schemars", - "serde", -] - -[[package]] -name = "der" -version = "0.5.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6919815d73839e7ad218de758883aae3a257ba6759ce7a9992501efbb53d705c" -dependencies = [ - "const-oid", -] - -[[package]] -name = "derivative" -version = "2.2.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fcc3dd5e9e9c0b295d6e1e4d811fb6f157d5ffd784b8d202fc62eac8035a770b" -dependencies = [ - "proc-macro2", - "quote", - "syn", -] - -[[package]] -name = "digest" -version = "0.9.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d3dd60d1080a57a05ab032377049e0591415d2b31afd7028356dbf3cc6dcb066" -dependencies = [ - "generic-array", -] - -[[package]] -name = "dyn-clone" -version = "1.0.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "21e50f3adc76d6a43f5ed73b698a87d0760ca74617f60f7c3b879003536fdd28" - -[[package]] -name = "ecdsa" -version = "0.13.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d0d69ae62e0ce582d56380743515fefaf1a8c70cec685d9677636d7e30ae9dc9" -dependencies = [ - "der", - "elliptic-curve", - "rfc6979", - "signature", -] - -[[package]] -name = "ed25519-zebra" -version = "3.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "403ef3e961ab98f0ba902771d29f842058578bb1ce7e3c59dad5a6a93e784c69" -dependencies = [ - "curve25519-dalek", - "hex", - "rand_core 0.6.3", - "serde", - "sha2", - "thiserror", - "zeroize", -] - -[[package]] -name = "either" -version = "1.6.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e78d4f1cc4ae33bbfc157ed5d5a5ef3bc29227303d595861deb238fcec4e9457" - -[[package]] -name = "elliptic-curve" -version = "0.11.12" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "25b477563c2bfed38a3b7a60964c49e058b2510ad3f12ba3483fd8f62c2306d6" -dependencies = [ - "base16ct", - "crypto-bigint", - "der", - "ff", - "generic-array", - "group", - "rand_core 0.6.3", - "sec1", - "subtle", - "zeroize", -] - -[[package]] -name = "ff" -version = "0.11.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "131655483be284720a17d74ff97592b8e76576dc25563148601df2d7c9080924" -dependencies = [ - "rand_core 0.6.3", - "subtle", -] - -[[package]] -name = "forward_ref" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c8cbd1169bd7b4a0a20d92b9af7a7e0422888bd38a6f5ec29c1fd8c1558a272e" - -[[package]] -name = "generic-array" -version = "0.14.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fd48d33ec7f05fbfa152300fdad764757cbded343c1aa1cff2fbaf4134851803" -dependencies = [ - "typenum", - "version_check", -] - -[[package]] -name = "getrandom" -version = "0.1.16" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8fc3cb4d91f53b50155bdcfd23f6a4c39ae1969c2ae85982b135750cccaf5fce" -dependencies = [ - "cfg-if", - "libc", - "wasi 0.9.0+wasi-snapshot-preview1", -] - -[[package]] -name = "getrandom" -version = "0.2.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9be70c98951c83b8d2f8f60d7065fa6d5146873094452a1008da8c2f1e4205ad" -dependencies = [ - "cfg-if", - "libc", - "wasi 0.10.2+wasi-snapshot-preview1", -] - -[[package]] -name = "group" -version = "0.11.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bc5ac374b108929de78460075f3dc439fa66df9d8fc77e8f12caa5165fcf0c89" -dependencies = [ - "ff", - "rand_core 0.6.3", - "subtle", -] - -[[package]] -name = "hex" -version = "0.4.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70" - -[[package]] -name = "hmac" -version = "0.11.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2a2a2320eb7ec0ebe8da8f744d7812d9fc4cb4d09344ac01898dbcb6a20ae69b" -dependencies = [ - "crypto-mac", - "digest", -] - -[[package]] -name = "itertools" -version = "0.10.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a9a9d19fa1e79b6215ff29b9d6880b706147f16e9b1dbb1e4e5947b5b02bc5e3" -dependencies = [ - "either", -] - -[[package]] -name = "itoa" -version = "1.0.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "112c678d4050afce233f4f2852bb2eb519230b3cf12f33585275537d7e41578d" - -[[package]] -name = "k256" -version = "0.10.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "19c3a5e0a0b8450278feda242592512e09f61c72e018b8cd5c859482802daf2d" -dependencies = [ - "cfg-if", - "ecdsa", - "elliptic-curve", - "sec1", - "sha2", -] - -[[package]] -name = "libc" -version = "0.2.126" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "349d5a591cd28b49e1d1037471617a32ddcda5731b99419008085f72d5a53836" - -[[package]] -name = "opaque-debug" -version = "0.3.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "624a8340c38c1b80fd549087862da4ba43e08858af025b236e509b6649fc13d5" - -[[package]] -name = "pkcs8" -version = "0.8.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7cabda3fb821068a9a4fab19a683eac3af12edf0f34b94a8be53c4972b8149d0" -dependencies = [ - "der", - "spki", - "zeroize", -] - -[[package]] -name = "proc-macro2" -version = "1.0.39" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c54b25569025b7fc9651de43004ae593a75ad88543b17178aa5e1b9c4f15f56f" -dependencies = [ - "unicode-ident", -] - -[[package]] -name = "prost" -version = "0.9.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "444879275cb4fd84958b1a1d5420d15e6fcf7c235fe47f053c9c2a80aceb6001" -dependencies = [ - "bytes", - "prost-derive", -] - -[[package]] -name = "prost-derive" -version = "0.9.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f9cc1a3263e07e0bf68e96268f37665207b49560d98739662cdfaae215c720fe" -dependencies = [ - "anyhow", - "itertools", - "proc-macro2", - "quote", - "syn", -] - -[[package]] -name = "quote" -version = "1.0.18" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a1feb54ed693b93a84e14094943b84b7c4eae204c512b7ccb95ab0c66d278ad1" -dependencies = [ - "proc-macro2", -] - -[[package]] -name = "rand_core" -version = "0.5.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "90bde5296fc891b0cef12a6d03ddccc162ce7b2aff54160af9338f8d40df6d19" -dependencies = [ - "getrandom 0.1.16", -] - -[[package]] -name = "rand_core" -version = "0.6.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d34f1408f55294453790c48b2f1ebbb1c5b4b7563eb1f418bcfcfdbb06ebb4e7" -dependencies = [ - "getrandom 0.2.6", -] - -[[package]] -name = "rfc6979" -version = "0.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "96ef608575f6392792f9ecf7890c00086591d29a83910939d430753f7c050525" -dependencies = [ - "crypto-bigint", - "hmac", - "zeroize", -] - -[[package]] -name = "ryu" -version = "1.0.10" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f3f6f92acf49d1b98f7a81226834412ada05458b7364277387724a237f062695" - -[[package]] -name = "schemars" -version = "0.8.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7d21ecb263bf75fc69d5e74b0f2a60b6dd80cfd9fb0eba15c4b9d1104ceffc77" -dependencies = [ - "dyn-clone", - "schemars_derive", - "serde", - "serde_json", -] - -[[package]] -name = "schemars_derive" -version = "0.8.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "219b924f5b39f25b7d7c9203873a2698fdac8db2b396aaea6fa099b699cc40e9" -dependencies = [ - "proc-macro2", - "quote", - "serde_derive_internals", - "syn", -] - -[[package]] -name = "sec1" -version = "0.2.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "08da66b8b0965a5555b6bd6639e68ccba85e1e2506f5fbb089e93f8a04e1a2d1" -dependencies = [ - "der", - "generic-array", - "pkcs8", - "subtle", - "zeroize", -] - -[[package]] -name = "serde" -version = "1.0.137" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "61ea8d54c77f8315140a05f4c7237403bf38b72704d031543aa1d16abbf517d1" -dependencies = [ - "serde_derive", -] - -[[package]] -name = "serde-json-wasm" -version = "0.4.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "479b4dbc401ca13ee8ce902851b834893251404c4f3c65370a49e047a6be09a5" -dependencies = [ - "serde", -] - -[[package]] -name = "serde_derive" -version = "1.0.137" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1f26faba0c3959972377d3b2d306ee9f71faee9714294e41bb777f83f88578be" -dependencies = [ - "proc-macro2", - "quote", - "syn", -] - -[[package]] -name = "serde_derive_internals" -version = "0.26.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "85bf8229e7920a9f636479437026331ce11aa132b4dde37d121944a44d6e5f3c" -dependencies = [ - "proc-macro2", - "quote", - "syn", -] - -[[package]] -name = "serde_json" -version = "1.0.81" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9b7ce2b32a1aed03c558dc61a5cd328f15aff2dbc17daad8fb8af04d2100e15c" -dependencies = [ - "itoa", - "ryu", - "serde", -] - -[[package]] -name = "sha2" -version = "0.9.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4d58a1e1bf39749807d89cf2d98ac2dfa0ff1cb3faa38fbb64dd88ac8013d800" -dependencies = [ - "block-buffer", - "cfg-if", - "cpufeatures", - "digest", - "opaque-debug", -] - -[[package]] -name = "signature" -version = "1.4.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "02658e48d89f2bec991f9a78e69cfa4c316f8d6a6c4ec12fae1aeb263d486788" -dependencies = [ - "digest", - "rand_core 0.6.3", -] - -[[package]] -name = "spki" -version = "0.5.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "44d01ac02a6ccf3e07db148d2be087da624fea0221a16152ed01f0496a6b0a27" -dependencies = [ - "base64ct", - "der", -] - -[[package]] -name = "static_assertions" -version = "1.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f" - -[[package]] -name = "subtle" -version = "2.4.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6bdef32e8150c2a081110b42772ffe7d7c9032b606bc226c8260fd97e0976601" - -[[package]] -name = "syn" -version = "1.0.95" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fbaf6116ab8924f39d52792136fb74fd60a80194cf1b1c6ffa6453eef1c3f942" -dependencies = [ - "proc-macro2", - "quote", - "unicode-ident", -] - -[[package]] -name = "testgen-local" -version = "0.1.0" -dependencies = [ - "cosmwasm-schema", - "cosmwasm-std", - "cosmwasm-storage", - "cw-multi-test", - "cw-storage-plus", - "cw2", - "schemars", - "serde", - "thiserror", -] - -[[package]] -name = "thiserror" -version = "1.0.31" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bd829fe32373d27f76265620b5309d0340cb8550f523c1dda251d6298069069a" -dependencies = [ - "thiserror-impl", -] - -[[package]] -name = "thiserror-impl" -version = "1.0.31" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0396bc89e626244658bef819e22d0cc459e795a5ebe878e6ec336d1674a8d79a" -dependencies = [ - "proc-macro2", - "quote", - "syn", -] - -[[package]] -name = "typenum" -version = "1.15.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dcf81ac59edc17cc8697ff311e8f5ef2d99fcbd9817b34cec66f90b6c3dfd987" - -[[package]] -name = "uint" -version = "0.9.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "12f03af7ccf01dd611cc450a0d10dbc9b745770d096473e2faf0ca6e2d66d1e0" -dependencies = [ - "byteorder", - "crunchy", - "hex", - "static_assertions", -] - -[[package]] -name = "unicode-ident" -version = "1.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d22af068fba1eb5edcb4aea19d382b2a3deb4c8f9d475c589b6ada9e0fd493ee" - -[[package]] -name = "version_check" -version = "0.9.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "49874b5167b65d7193b8aba1567f5c7d93d001cafc34600cee003eda787e483f" - -[[package]] -name = "wasi" -version = "0.9.0+wasi-snapshot-preview1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cccddf32554fecc6acb585f82a32a72e28b48f8c4c1883ddfeeeaa96f7d8e519" - -[[package]] -name = "wasi" -version = "0.10.2+wasi-snapshot-preview1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fd6fbd9a79829dd1ad0cc20627bf1ed606756a7f77edff7b66b7064f9cb327c6" - -[[package]] -name = "zeroize" -version = "1.5.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "94693807d016b2f2d2e14420eb3bfcca689311ff775dcf113d74ea624b7cdf07" diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/Cargo.toml b/smart-contracts/osmosis/contracts/intergamm-bindings-test/Cargo.toml deleted file mode 100644 index df06f6249..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/Cargo.toml +++ /dev/null @@ -1,61 +0,0 @@ -[package] -name = "intergamm-bindings-test" -version = "0.1.0" -authors = ["LaurensKubat "] -edition = "2018" - -exclude = [ - # Those files are rust-optimizer artifacts. You might want to commit them for convenience but they should not be part of the source code publication. - "contract.wasm", - "hash.txt", -] - -# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html - -[lib] -crate-type = ["cdylib", "rlib"] - -[profile.release] -opt-level = 3 -debug = false -rpath = false -lto = true -debug-assertions = false -codegen-units = 1 -panic = 'abort' -incremental = false -overflow-checks = true - -[features] -# for more explicit tests, cargo test --features=backtraces -backtraces = ["cosmwasm-std/backtraces"] -# use library feature to disable all instantiate/execute/query exports -library = [] - -# Replaced by justfile -# [package.metadata.scripts] -# optimize = """docker run --rm -v "$(pwd)":/code \ -# --mount type=volume,source="$(basename "$(pwd)")_cache",target=/code/target \ -# --mount type=volume,source=registry_cache,target=/usr/local/cargo/registry \ -# cosmwasm/rust-optimizer:0.12.6 -# """ - -[dependencies] -cosmwasm-std = { workspace = true } -osmosis-std = { workspace = true } -osmosis-std-derive = { workspace = true } -cosmwasm-schema = { workspace = true } -cw-storage-plus ={ workspace = true } -schemars ={ workspace = true } -serde = { workspace = true } -thiserror = { workspace = true } -cw-utils = { workspace = true } -cw2 = { workspace = true } - -# Quasar packages -intergamm-bindings = { path = "../../packages/intergamm-bindings", version = "0.1.0"} - -[dev-dependencies] -cosmwasm-schema = { workspace = true } -cw-multi-test = {workspace = true} - diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/Developing.md b/smart-contracts/osmosis/contracts/intergamm-bindings-test/Developing.md deleted file mode 100644 index 2dd572cee..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/Developing.md +++ /dev/null @@ -1,104 +0,0 @@ -# Developing - -If you have recently created a contract with this template, you probably could use some -help on how to build and test the contract, as well as prepare it for production. This -file attempts to provide a brief overview, assuming you have installed a recent -version of Rust already (eg. 1.58.1+). - -## Prerequisites - -Before starting, make sure you have [rustup](https://rustup.rs/) along with a -recent `rustc` and `cargo` version installed. Currently, we are testing on 1.58.1+. - -And you need to have the `wasm32-unknown-unknown` target installed as well. - -You can check that via: - -```sh -rustc --version -cargo --version -rustup target list --installed -# if wasm32 is not listed above, run this -rustup target add wasm32-unknown-unknown -``` - -## Compiling and running tests - -Now that you created your custom contract, make sure you can compile and run it before -making any changes. Go into the repository and do: - -```sh -# this will produce a wasm build in ./target/wasm32-unknown-unknown/release/YOUR_NAME_HERE.wasm -cargo wasm - -# this runs unit tests with helpful backtraces -RUST_BACKTRACE=1 cargo unit-test - -# auto-generate json schema -cargo schema -``` - -### Understanding the tests - -The main code is in `src/contract.rs` and the unit tests there run in pure rust, -which makes them very quick to execute and give nice output on failures, especially -if you do `RUST_BACKTRACE=1 cargo unit-test`. - -We consider testing critical for anything on a blockchain, and recommend to always keep -the tests up to date. - -## Generating JSON Schema - -While the Wasm calls (`instantiate`, `execute`, `query`) accept JSON, this is not enough -information to use it. We need to expose the schema for the expected messages to the -clients. You can generate this schema by calling `cargo schema`, which will output -4 files in `./schema`, corresponding to the 3 message types the contract accepts, -as well as the internal `State`. - -These files are in standard json-schema format, which should be usable by various -client side tools, either to auto-generate codecs, or just to validate incoming -json wrt. the defined schema. - -## Preparing the Wasm bytecode for production - -Before we upload it to a chain, we need to ensure the smallest output size possible, -as this will be included in the body of a transaction. We also want to have a -reproducible build process, so third parties can verify that the uploaded Wasm -code did indeed come from the claimed rust code. - -To solve both these issues, we have produced `rust-optimizer`, a docker image to -produce an extremely small build output in a consistent manner. The suggest way -to run it is this: - -```sh -docker run --rm -v "$(pwd)":/code \ - --mount type=volume,source="$(basename "$(pwd)")_cache",target=/code/target \ - --mount type=volume,source=registry_cache,target=/usr/local/cargo/registry \ - cosmwasm/rust-optimizer:0.12.6 -``` - -Or, If you're on an arm64 machine, you should use a docker image built with arm64. -```sh -docker run --rm -v "$(pwd)":/code \ - --mount type=volume,source="$(basename "$(pwd)")_cache",target=/code/target \ - --mount type=volume,source=registry_cache,target=/usr/local/cargo/registry \ - cosmwasm/rust-optimizer-arm64:0.12.6 -``` - -We must mount the contract code to `/code`. You can use a absolute path instead -of `$(pwd)` if you don't want to `cd` to the directory first. The other two -volumes are nice for speedup. Mounting `/code/target` in particular is useful -to avoid docker overwriting your local dev files with root permissions. -Note the `/code/target` cache is unique for each contract being compiled to limit -interference, while the registry cache is global. - -This is rather slow compared to local compilations, especially the first compile -of a given contract. The use of the two volume caches is very useful to speed up -following compiles of the same contract. - -This produces an `artifacts` directory with a `PROJECT_NAME.wasm`, as well as -`checksums.txt`, containing the Sha256 hash of the wasm file. -The wasm file is compiled deterministically (anyone else running the same -docker on the same git commit should get the identical file with the same Sha256 hash). -It is also stripped and minimized for upload to a blockchain (we will also -gzip it in the uploading process to make it even smaller). diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/Importing.md b/smart-contracts/osmosis/contracts/intergamm-bindings-test/Importing.md deleted file mode 100644 index a572ba05e..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/Importing.md +++ /dev/null @@ -1,62 +0,0 @@ -# Importing - -In [Publishing](quasar-finance/quasar/smart-contracts/contracts/intergamm-bindings-testt/Publishing.md), we discussed how you can publish your contract to the world. -This looks at the flip-side, how can you use someone else's contract (which is the same -question as how they will use your contract). Let's go through the various stages. - -## Verifying Artifacts - -Before using remote code, you most certainly want to verify it is honest. - -The simplest audit of the repo is to simply check that the artifacts in the repo -are correct. This involves recompiling the claimed source with the claimed builder -and validating that the locally compiled code (hash) matches the code hash that was -uploaded. This will verify that the source code is the correct preimage. Which allows -one to audit the original (Rust) source code, rather than looking at wasm bytecode. - -We have a script to do this automatic verification steps that can -easily be run by many individuals. Please check out -[`cosmwasm-verify`](https://github.com/CosmWasm/cosmwasm-verify/blob/master/README.md) -to see a simple shell script that does all these steps and easily allows you to verify -any uploaded contract. - -## Reviewing - -Once you have done the quick programatic checks, it is good to give at least a quick -look through the code. A glance at `examples/schema.rs` to make sure it is outputing -all relevant structs from `contract.rs`, and also ensure `src/lib.rs` is just the -default wrapper (nothing funny going on there). After this point, we can dive into -the contract code itself. Check the flows for the execute methods, any invariants and -permission checks that should be there, and a reasonable data storage format. - -You can dig into the contract as far as you want, but it is important to make sure there -are no obvious backdoors at least. - -## Decentralized Verification - -It's not very practical to do a deep code review on every dependency you want to use, -which is a big reason for the popularity of code audits in the blockchain world. We trust -some experts review in lieu of doing the work ourselves. But wouldn't it be nice to do this -in a decentralized manner and peer-review each other's contracts? Bringing in deeper domain -knowledge and saving fees. - -Luckily, there is an amazing project called [crev](https://github.com/crev-dev/cargo-crev/blob/master/cargo-crev/README.md) -that provides `A cryptographically verifiable code review system for the cargo (Rust) package manager`. - -I highly recommend that CosmWasm contract developers get set up with this. At minimum, we -can all add a review on a package that programmatically checked out that the json schemas -and wasm bytecode do match the code, and publish our claim, so we don't all rely on some -central server to say it validated this. As we go on, we can add deeper reviews on standard -packages. - -If you want to use `cargo-crev`, please follow their -[getting started guide](https://github.com/crev-dev/cargo-crev/blob/master/cargo-crev/src/doc/getting_started.md) -and once you have made your own *proof repository* with at least one *trust proof*, -please make a PR to the [`cawesome-wasm`]() repo with a link to your repo and -some public name or pseudonym that people know you by. This allows people who trust you -to also reuse your proofs. - -There is a [standard list of proof repos](https://github.com/crev-dev/cargo-crev/wiki/List-of-Proof-Repositories) -with some strong rust developers in there. This may cover dependencies like `serde` and `snafu` -but will not hit any CosmWasm-related modules, so we look to bootstrap a very focused -review community. diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/LICENSE b/smart-contracts/osmosis/contracts/intergamm-bindings-test/LICENSE deleted file mode 100644 index d64569567..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/LICENSE +++ /dev/null @@ -1,202 +0,0 @@ - - Apache License - Version 2.0, January 2004 - http://www.apache.org/licenses/ - - TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION - - 1. Definitions. - - "License" shall mean the terms and conditions for use, reproduction, - and distribution as defined by Sections 1 through 9 of this document. - - "Licensor" shall mean the copyright owner or entity authorized by - the copyright owner that is granting the License. - - "Legal Entity" shall mean the union of the acting entity and all - other entities that control, are controlled by, or are under common - control with that entity. For the purposes of this definition, - "control" means (i) the power, direct or indirect, to cause the - direction or management of such entity, whether by contract or - otherwise, or (ii) ownership of fifty percent (50%) or more of the - outstanding shares, or (iii) beneficial ownership of such entity. - - "You" (or "Your") shall mean an individual or Legal Entity - exercising permissions granted by this License. - - "Source" form shall mean the preferred form for making modifications, - including but not limited to software source code, documentation - source, and configuration files. - - "Object" form shall mean any form resulting from mechanical - transformation or translation of a Source form, including but - not limited to compiled object code, generated documentation, - and conversions to other media types. - - "Work" shall mean the work of authorship, whether in Source or - Object form, made available under the License, as indicated by a - copyright notice that is included in or attached to the work - (an example is provided in the Appendix below). - - "Derivative Works" shall mean any work, whether in Source or Object - form, that is based on (or derived from) the Work and for which the - editorial revisions, annotations, elaborations, or other modifications - represent, as a whole, an original work of authorship. For the purposes - of this License, Derivative Works shall not include works that remain - separable from, or merely link (or bind by name) to the interfaces of, - the Work and Derivative Works thereof. - - "Contribution" shall mean any work of authorship, including - the original version of the Work and any modifications or additions - to that Work or Derivative Works thereof, that is intentionally - submitted to Licensor for inclusion in the Work by the copyright owner - or by an individual or Legal Entity authorized to submit on behalf of - the copyright owner. For the purposes of this definition, "submitted" - means any form of electronic, verbal, or written communication sent - to the Licensor or its representatives, including but not limited to - communication on electronic mailing lists, source code control systems, - and issue tracking systems that are managed by, or on behalf of, the - Licensor for the purpose of discussing and improving the Work, but - excluding communication that is conspicuously marked or otherwise - designated in writing by the copyright owner as "Not a Contribution." - - "Contributor" shall mean Licensor and any individual or Legal Entity - on behalf of whom a Contribution has been received by Licensor and - subsequently incorporated within the Work. - - 2. Grant of Copyright License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - copyright license to reproduce, prepare Derivative Works of, - publicly display, publicly perform, sublicense, and distribute the - Work and such Derivative Works in Source or Object form. - - 3. Grant of Patent License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - (except as stated in this section) patent license to make, have made, - use, offer to sell, sell, import, and otherwise transfer the Work, - where such license applies only to those patent claims licensable - by such Contributor that are necessarily infringed by their - Contribution(s) alone or by combination of their Contribution(s) - with the Work to which such Contribution(s) was submitted. If You - institute patent litigation against any entity (including a - cross-claim or counterclaim in a lawsuit) alleging that the Work - or a Contribution incorporated within the Work constitutes direct - or contributory patent infringement, then any patent licenses - granted to You under this License for that Work shall terminate - as of the date such litigation is filed. - - 4. Redistribution. You may reproduce and distribute copies of the - Work or Derivative Works thereof in any medium, with or without - modifications, and in Source or Object form, provided that You - meet the following conditions: - - (a) You must give any other recipients of the Work or - Derivative Works a copy of this License; and - - (b) You must cause any modified files to carry prominent notices - stating that You changed the files; and - - (c) You must retain, in the Source form of any Derivative Works - that You distribute, all copyright, patent, trademark, and - attribution notices from the Source form of the Work, - excluding those notices that do not pertain to any part of - the Derivative Works; and - - (d) If the Work includes a "NOTICE" text file as part of its - distribution, then any Derivative Works that You distribute must - include a readable copy of the attribution notices contained - within such NOTICE file, excluding those notices that do not - pertain to any part of the Derivative Works, in at least one - of the following places: within a NOTICE text file distributed - as part of the Derivative Works; within the Source form or - documentation, if provided along with the Derivative Works; or, - within a display generated by the Derivative Works, if and - wherever such third-party notices normally appear. The contents - of the NOTICE file are for informational purposes only and - do not modify the License. You may add Your own attribution - notices within Derivative Works that You distribute, alongside - or as an addendum to the NOTICE text from the Work, provided - that such additional attribution notices cannot be construed - as modifying the License. - - You may add Your own copyright statement to Your modifications and - may provide additional or different license terms and conditions - for use, reproduction, or distribution of Your modifications, or - for any such Derivative Works as a whole, provided Your use, - reproduction, and distribution of the Work otherwise complies with - the conditions stated in this License. - - 5. Submission of Contributions. Unless You explicitly state otherwise, - any Contribution intentionally submitted for inclusion in the Work - by You to the Licensor shall be under the terms and conditions of - this License, without any additional terms or conditions. - Notwithstanding the above, nothing herein shall supersede or modify - the terms of any separate license agreement you may have executed - with Licensor regarding such Contributions. - - 6. Trademarks. This License does not grant permission to use the trade - names, trademarks, service marks, or product names of the Licensor, - except as required for reasonable and customary use in describing the - origin of the Work and reproducing the content of the NOTICE file. - - 7. Disclaimer of Warranty. Unless required by applicable law or - agreed to in writing, Licensor provides the Work (and each - Contributor provides its Contributions) on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or - implied, including, without limitation, any warranties or conditions - of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A - PARTICULAR PURPOSE. You are solely responsible for determining the - appropriateness of using or redistributing the Work and assume any - risks associated with Your exercise of permissions under this License. - - 8. Limitation of Liability. In no event and under no legal theory, - whether in tort (including negligence), contract, or otherwise, - unless required by applicable law (such as deliberate and grossly - negligent acts) or agreed to in writing, shall any Contributor be - liable to You for damages, including any direct, indirect, special, - incidental, or consequential damages of any character arising as a - result of this License or out of the use or inability to use the - Work (including but not limited to damages for loss of goodwill, - work stoppage, computer failure or malfunction, or any and all - other commercial damages or losses), even if such Contributor - has been advised of the possibility of such damages. - - 9. Accepting Warranty or Additional Liability. While redistributing - the Work or Derivative Works thereof, You may choose to offer, - and charge a fee for, acceptance of support, warranty, indemnity, - or other liability obligations and/or rights consistent with this - License. However, in accepting such obligations, You may act only - on Your own behalf and on Your sole responsibility, not on behalf - of any other Contributor, and only if You agree to indemnify, - defend, and hold each Contributor harmless for any liability - incurred by, or claims asserted against, such Contributor by reason - of your accepting any such warranty or additional liability. - - END OF TERMS AND CONDITIONS - - APPENDIX: How to apply the Apache License to your work. - - To apply the Apache License to your work, attach the following - boilerplate notice, with the fields enclosed by brackets "[]" - replaced with your own identifying information. (Don't include - the brackets!) The text should be enclosed in the appropriate - comment syntax for the file format. We also recommend that a - file or class name and description of purpose be included on the - same "printed page" as the copyright notice for easier - identification within third-party archives. - - Copyright [yyyy] [name of copyright owner] - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/NOTICE b/smart-contracts/osmosis/contracts/intergamm-bindings-test/NOTICE deleted file mode 100644 index 1270a9c2f..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/NOTICE +++ /dev/null @@ -1,13 +0,0 @@ -Copyright 2022 LaurensKubat - -Licensed under the Apache License, Version 2.0 (the "License"); -you may not use this file except in compliance with the License. -You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - -Unless required by applicable law or agreed to in writing, software -distributed under the License is distributed on an "AS IS" BASIS, -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -See the License for the specific language governing permissions and -limitations under the License. diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/Publishing.md b/smart-contracts/osmosis/contracts/intergamm-bindings-test/Publishing.md deleted file mode 100644 index c94f8675d..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/Publishing.md +++ /dev/null @@ -1,115 +0,0 @@ -# Publishing Contracts - -This is an overview of how to publish the contract's source code in this repo. -We use Cargo's default registry [crates.io](https://crates.io/) for publishing contracts written in Rust. - -## Preparation - -Ensure the `Cargo.toml` file in the repo is properly configured. In particular, you want to -choose a name starting with `cw-`, which will help a lot finding CosmWasm contracts when -searching on crates.io. For the first publication, you will probably want version `0.1.0`. -If you have tested this on a public net already and/or had an audit on the code, -you can start with `1.0.0`, but that should imply some level of stability and confidence. -You will want entries like the following in `Cargo.toml`: - -```toml -name = "cw-escrow" -version = "0.1.0" -description = "Simple CosmWasm contract for an escrow with arbiter and timeout" -repository = "https://github.com/confio/cosmwasm-examples" -``` - -You will also want to add a valid [SPDX license statement](https://spdx.org/licenses/), -so others know the rules for using this crate. You can use any license you wish, -even a commercial license, but we recommend choosing one of the following, unless you have -specific requirements. - -* Permissive: [`Apache-2.0`](https://spdx.org/licenses/Apache-2.0.html#licenseText) or [`MIT`](https://spdx.org/licenses/MIT.html#licenseText) -* Copyleft: [`GPL-3.0-or-later`](https://spdx.org/licenses/GPL-3.0-or-later.html#licenseText) or [`AGPL-3.0-or-later`](https://spdx.org/licenses/AGPL-3.0-or-later.html#licenseText) -* Commercial license: `Commercial` (not sure if this works, I cannot find examples) - -It is also helpful to download the LICENSE text (linked to above) and store this -in a LICENSE file in your repo. Now, you have properly configured your crate for use -in a larger ecosystem. - -### Updating schema - -To allow easy use of the contract, we can publish the schema (`schema/*.json`) together -with the source code. - -```sh -cargo schema -``` - -Ensure you check in all the schema files, and make a git commit with the final state. -This commit will be published and should be tagged. Generally, you will want to -tag with the version (eg. `v0.1.0`), but in the `cosmwasm-examples` repo, we have -multiple contracts and label it like `escrow-0.1.0`. Don't forget a -`git push && git push --tags` - -### Note on build results - -Build results like Wasm bytecode or expected hash don't need to be updated since -they don't belong to the source publication. However, they are excluded from packaging -in `Cargo.toml` which allows you to commit them to your git repository if you like. - -```toml -exclude = ["artifacts"] -``` - -A single source code can be built with multiple different optimizers, so -we should not make any strict assumptions on the tooling that will be used. - -## Publishing - -Now that your package is properly configured and all artifacts are committed, it -is time to share it with the world. -Please refer to the [complete instructions for any questions](https://rurust.github.io/cargo-docs-ru/crates-io.html), -but I will try to give a quick overview of the happy path here. - -### Registry - -You will need an account on [crates.io](https://crates.io) to publish a rust crate. -If you don't have one already, just click on "Log in with GitHub" in the top-right -to quickly set up a free account. Once inside, click on your username (top-right), -then "Account Settings". On the bottom, there is a section called "API Access". -If you don't have this set up already, create a new token and use `cargo login` -to set it up. This will now authenticate you with the `cargo` cli tool and allow -you to publish. - -### Uploading - -Once this is set up, make sure you commit the current state you want to publish. -Then try `cargo publish --dry-run`. If that works well, review the files that -will be published via `cargo package --list`. If you are satisfied, you can now -officially publish it via `cargo publish`. - -Congratulations, your package is public to the world. - -### Sharing - -Once you have published your package, people can now find it by -[searching for "cw-" on crates.io](https://crates.io/search?q=cw). -But that isn't exactly the simplest way. To make things easier and help -keep the ecosystem together, we suggest making a PR to add your package -to the [`cawesome-wasm`](https://github.com/cosmwasm/cawesome-wasm) list. - -### Organizations - -Many times you are writing a contract not as a solo developer, but rather as -part of an organization. You will want to allow colleagues to upload new -versions of the contract to crates.io when you are on holiday. -[These instructions show how]() you can set up your crate to allow multiple maintainers. - -You can add another owner to the crate by specifying their github user. Note, you will -now both have complete control of the crate, and they can remove you: - -`cargo owner --add ethanfrey` - -You can also add an existing github team inside your organization: - -`cargo owner --add github:confio:developers` - -The team will allow anyone who is currently in the team to publish new versions of the crate. -And this is automatically updated when you make changes on github. However, it will not allow -anyone in the team to add or remove other owners. diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/README.md b/smart-contracts/osmosis/contracts/intergamm-bindings-test/README.md deleted file mode 100644 index aaecb7101..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/README.md +++ /dev/null @@ -1,33 +0,0 @@ -# Intergamm test contract -This contract demonstrates the usage of the intergamm - -## Prerequisites -A local quasarnoded, osmosisd and gaiad binary. -The quasarnoded binary can be installed by running -``` -go install go install ../../../cmd/quasarnoded/ -``` - -For the osmosisd and gaiad, you need to build those from their respective source, which can be found here: -[Osmosis](https://github.com/osmosis-labs/osmosis) -[Gaia](https://github.com/cosmos/gaia) - -## setting up -Run the chains using the [run_all](../../../demos/orion-manual-demo/run_all.sh) script from the orion manual demo in a separate window: -``` -./run_all.sh -``` - -once you see `starting relaying`, in a second terminal run the [create_and_execute](../../../demos/orion-manual-demo/create_and_execute_contract.sh) script - -``` -./create_and_execute.sh -``` - -Now a new contract should be deployed with a registered interchain account. The address of this contract can be found in the output of [create_and_execute](../../../demos/orion-manual-demo/create_and_execute_contract.sh) - -The created contract contains multiple execute messages to use the different intergamm messages from the intergamm-bindings package. The easiest way to get the correct funds on the interchain account address on cosmos is to send funds from alice/bob to the newly created interchain account. -A sample pool can be made using -``` -osmosisd tx gamm create-pool --pool-file ./sample_pool.json --node tcp://localhost:26679 --from bob --chain-id osmosis --gas auto -``` \ No newline at end of file diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/examples/schema.rs b/smart-contracts/osmosis/contracts/intergamm-bindings-test/examples/schema.rs deleted file mode 100644 index 21bfcfe76..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/examples/schema.rs +++ /dev/null @@ -1,17 +0,0 @@ -use std::env::current_dir; -use std::fs::create_dir_all; - -use cosmwasm_schema::{export_schema, remove_schemas, schema_for}; - -use intergamm_bindings_test::msg::{ExecuteMsg, InstantiateMsg, QueryMsg}; - -fn main() { - let mut out_dir = current_dir().unwrap(); - out_dir.push("schema"); - create_dir_all(&out_dir).unwrap(); - remove_schemas(&out_dir).unwrap(); - - export_schema(&schema_for!(InstantiateMsg), &out_dir); - export_schema(&schema_for!(ExecuteMsg), &out_dir); - export_schema(&schema_for!(QueryMsg), &out_dir); -} diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/rustfmt.toml b/smart-contracts/osmosis/contracts/intergamm-bindings-test/rustfmt.toml deleted file mode 100644 index 11a85e6a9..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/rustfmt.toml +++ /dev/null @@ -1,15 +0,0 @@ -# stable -newline_style = "unix" -hard_tabs = false -tab_spaces = 4 - -# unstable... should we require `rustup run nightly cargo fmt` ? -# or just update the style guide when they are stable? -#fn_single_line = true -#format_code_in_doc_comments = true -#overflow_delimited_expr = true -#reorder_impl_items = true -#struct_field_align_threshold = 20 -#struct_lit_single_line = true -#report_todo = "Always" - diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/schema/execute_msg.json b/smart-contracts/osmosis/contracts/intergamm-bindings-test/schema/execute_msg.json deleted file mode 100644 index 9fcd1ba27..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/schema/execute_msg.json +++ /dev/null @@ -1,516 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "ExecuteMsg", - "oneOf": [ - { - "type": "object", - "required": [ - "send_token" - ], - "properties": { - "send_token": { - "type": "object", - "required": [ - "destination_local_zone_id" - ], - "properties": { - "destination_local_zone_id": { - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "send_token_ibc" - ], - "properties": { - "send_token_ibc": { - "type": "object", - "required": [ - "amount", - "channel_id", - "to_address" - ], - "properties": { - "amount": { - "description": "packet data only supports one coin https://github.com/cosmos/cosmos-sdk/blob/v0.40.0/proto/ibc/applications/transfer/v1/transfer.proto#L11-L20", - "allOf": [ - { - "$ref": "#/definitions/Coin" - } - ] - }, - "channel_id": { - "description": "exisiting channel to send the tokens over", - "type": "string" - }, - "to_address": { - "description": "address on the remote chain to receive these tokens", - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "deposit" - ], - "properties": { - "deposit": { - "type": "object" - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "register_interchain_account" - ], - "properties": { - "register_interchain_account": { - "type": "object", - "required": [ - "connection_id" - ], - "properties": { - "connection_id": { - "type": "string" - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "join_pool" - ], - "properties": { - "join_pool": { - "type": "object", - "required": [ - "connection_id", - "pool_id", - "share_out_amount", - "timeout_timestamp", - "token_in_maxs" - ], - "properties": { - "connection_id": { - "type": "string" - }, - "pool_id": { - "$ref": "#/definitions/Uint64" - }, - "share_out_amount": { - "type": "integer", - "format": "int64" - }, - "timeout_timestamp": { - "$ref": "#/definitions/Uint64" - }, - "token_in_maxs": { - "type": "array", - "items": { - "$ref": "#/definitions/Coin" - } - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "exit_pool" - ], - "properties": { - "exit_pool": { - "type": "object", - "required": [ - "connection_id", - "pool_id", - "share_in_amount", - "timeout_timestamp", - "token_out_mins" - ], - "properties": { - "connection_id": { - "type": "string" - }, - "pool_id": { - "$ref": "#/definitions/Uint64" - }, - "share_in_amount": { - "type": "integer", - "format": "int64" - }, - "timeout_timestamp": { - "$ref": "#/definitions/Uint64" - }, - "token_out_mins": { - "type": "array", - "items": { - "$ref": "#/definitions/Coin" - } - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "lock_tokens" - ], - "properties": { - "lock_tokens": { - "type": "object", - "required": [ - "coins", - "connection_id", - "duration", - "timeout_timestamp" - ], - "properties": { - "coins": { - "type": "array", - "items": { - "$ref": "#/definitions/Coin" - } - }, - "connection_id": { - "type": "string" - }, - "duration": { - "$ref": "#/definitions/Uint64" - }, - "timeout_timestamp": { - "$ref": "#/definitions/Uint64" - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "join_swap_extern_amount_in" - ], - "properties": { - "join_swap_extern_amount_in": { - "type": "object", - "required": [ - "connection_id", - "pool_id", - "share_out_min_amount", - "token_in" - ], - "properties": { - "connection_id": { - "type": "string" - }, - "pool_id": { - "$ref": "#/definitions/Uint64" - }, - "share_out_min_amount": { - "type": "integer", - "format": "int64" - }, - "token_in": { - "$ref": "#/definitions/Coin" - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "exit_swap_extern_amount_out" - ], - "properties": { - "exit_swap_extern_amount_out": { - "type": "object", - "required": [ - "connection_id", - "pool_id", - "share_in_amount", - "timeout_timestamp", - "token_out_mins" - ], - "properties": { - "connection_id": { - "type": "string" - }, - "pool_id": { - "$ref": "#/definitions/Uint64" - }, - "share_in_amount": { - "type": "integer", - "format": "int64" - }, - "timeout_timestamp": { - "$ref": "#/definitions/Uint64" - }, - "token_out_mins": { - "$ref": "#/definitions/Coin" - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "begin_unlocking" - ], - "properties": { - "begin_unlocking": { - "type": "object", - "required": [ - "coins", - "connection_id", - "id", - "timeout_timestamp" - ], - "properties": { - "coins": { - "type": "array", - "items": { - "$ref": "#/definitions/Coin" - } - }, - "connection_id": { - "type": "string" - }, - "id": { - "$ref": "#/definitions/Uint64" - }, - "timeout_timestamp": { - "$ref": "#/definitions/Uint64" - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "test_ica_scenario" - ], - "properties": { - "test_ica_scenario": { - "type": "object" - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "ack" - ], - "properties": { - "ack": { - "type": "object", - "required": [ - "sequence_number" - ], - "properties": { - "error": { - "type": [ - "string", - "null" - ] - }, - "response": { - "anyOf": [ - { - "$ref": "#/definitions/AckResponse" - }, - { - "type": "null" - } - ] - }, - "sequence_number": { - "type": "integer", - "format": "uint64", - "minimum": 0.0 - } - } - } - }, - "additionalProperties": false - } - ], - "definitions": { - "AckResponse": { - "oneOf": [ - { - "type": "object", - "required": [ - "join_swap_extern_amount_in" - ], - "properties": { - "join_swap_extern_amount_in": { - "type": "object", - "required": [ - "shareOutAmount" - ], - "properties": { - "shareOutAmount": { - "$ref": "#/definitions/Uint256" - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "exit_swap_extern_amount_out" - ], - "properties": { - "exit_swap_extern_amount_out": { - "type": "object", - "required": [ - "shareInAmount" - ], - "properties": { - "shareInAmount": { - "$ref": "#/definitions/Uint256" - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "join_swap_share_amount_out" - ], - "properties": { - "join_swap_share_amount_out": { - "type": "object", - "required": [ - "tokenInAmount" - ], - "properties": { - "tokenInAmount": { - "$ref": "#/definitions/Uint256" - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "exit_swap_share_amount_in" - ], - "properties": { - "exit_swap_share_amount_in": { - "type": "object", - "required": [ - "tokenOutAmount" - ], - "properties": { - "tokenOutAmount": { - "$ref": "#/definitions/Uint256" - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "lock_tokens" - ], - "properties": { - "lock_tokens": { - "type": "object", - "required": [ - "ID" - ], - "properties": { - "ID": { - "$ref": "#/definitions/Uint64" - } - } - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "begin_unlocking" - ], - "properties": { - "begin_unlocking": { - "type": "object", - "required": [ - "Success" - ], - "properties": { - "Success": { - "type": "boolean" - } - } - } - }, - "additionalProperties": false - } - ] - }, - "Coin": { - "type": "object", - "required": [ - "amount", - "denom" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - }, - "denom": { - "type": "string" - } - } - }, - "Uint128": { - "description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```", - "type": "string" - }, - "Uint256": { - "description": "An implementation of u256 that is using strings for JSON encoding/decoding, such that the full u256 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances out of primitive uint types or `new` to provide big endian bytes:\n\n``` # use cosmwasm_std::Uint256; let a = Uint256::from(258u128); let b = Uint256::new([ 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 0u8, 1u8, 2u8, ]); assert_eq!(a, b); ```", - "type": "string" - }, - "Uint64": { - "description": "A thin wrapper around u64 that is using strings for JSON encoding/decoding, such that the full u64 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u64` to get the value out:\n\n``` # use cosmwasm_std::Uint64; let a = Uint64::from(42u64); assert_eq!(a.u64(), 42);\n\nlet b = Uint64::from(70u32); assert_eq!(b.u64(), 70); ```", - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/schema/instantiate_msg.json b/smart-contracts/osmosis/contracts/intergamm-bindings-test/schema/instantiate_msg.json deleted file mode 100644 index ab0fcab20..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/schema/instantiate_msg.json +++ /dev/null @@ -1,13 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "InstantiateMsg", - "type": "object", - "required": [ - "callback_address" - ], - "properties": { - "callback_address": { - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/schema/query_msg.json b/smart-contracts/osmosis/contracts/intergamm-bindings-test/schema/query_msg.json deleted file mode 100644 index 5f93056d6..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/schema/query_msg.json +++ /dev/null @@ -1,30 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "QueryMsg", - "oneOf": [ - { - "type": "object", - "required": [ - "pending_acks" - ], - "properties": { - "pending_acks": { - "type": "object" - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "acks" - ], - "properties": { - "acks": { - "type": "object" - } - }, - "additionalProperties": false - } - ] -} diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/contract.rs b/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/contract.rs deleted file mode 100644 index 85834112e..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/contract.rs +++ /dev/null @@ -1,341 +0,0 @@ -use cosmwasm_std::{ - entry_point, to_json_binary, Binary, Coin, Deps, DepsMut, Env, IbcMsg, IbcTimeout, MessageInfo, - Order, Reply, Response, StdError, StdResult, Uint64, -}; -use cw2::set_contract_version; -use intergamm_bindings::helper::{ - ack, check_callback_addr, create_intergamm_msg, handle_reply, set_callback_addr, -}; -use intergamm_bindings::msg::IntergammMsg; - -use crate::error::ContractError; -use crate::msg::{AcksResponse, ExecuteMsg, InstantiateMsg, PendingAcksResponse, QueryMsg}; - -// version info for migration info -const CONTRACT_NAME: &str = "crates.io:intergamm-bindings-test"; -const CONTRACT_VERSION: &str = env!("CARGO_PKG_VERSION"); - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn instantiate( - deps: DepsMut, - _env: Env, - _info: MessageInfo, - msg: InstantiateMsg, -) -> Result { - set_contract_version(deps.storage, CONTRACT_NAME, CONTRACT_VERSION)?; - set_callback_addr(deps, &msg.callback_address)?; - Ok(Response::default()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn reply(deps: DepsMut, env: Env, msg: Reply) -> StdResult { - handle_reply(deps.storage, env, msg) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn execute( - deps: DepsMut, - env: Env, - info: MessageInfo, - msg: ExecuteMsg, -) -> Result, ContractError> { - match msg { - ExecuteMsg::SendToken { - destination_local_zone_id, - receiver, - coin, - } => execute_send_token(destination_local_zone_id, receiver, coin), - - ExecuteMsg::SendTokenIbc { - channel_id, - to_address, - amount, - } => execute_send_token_ibc(channel_id, to_address, amount, env), - ExecuteMsg::RegisterIcaOnZone { zone_id } => execute_register_ica_on_zone(zone_id, deps), - ExecuteMsg::JoinSwapExternAmountIn { - connection_id, - pool_id, - share_out_min_amount, - token_in, - } => execute_join_swap_extern_amount_in( - connection_id, - pool_id, - share_out_min_amount, - token_in, - deps, - env, - ), - ExecuteMsg::TestIcaScenario {} => execute_test_scenario("registerIca".to_string()), - ExecuteMsg::Ack { - sequence_number, - error, - response, - } => do_ibc_packet_ack(deps, env, info, sequence_number, error, response), - ExecuteMsg::Deposit {} => execute_deposit(info), - ExecuteMsg::JoinPool { - connection_id, - timeout_timestamp, - pool_id, - share_out_amount, - token_in_maxs, - } => execute_join_pool( - connection_id, - timeout_timestamp, - pool_id, - share_out_amount, - token_in_maxs, - deps, - ), - ExecuteMsg::ExitPool { - connection_id, - timeout_timestamp, - pool_id, - share_in_amount, - token_out_mins, - } => execute_exit_pool( - connection_id, - timeout_timestamp, - pool_id, - share_in_amount, - token_out_mins, - deps, - ), - ExecuteMsg::LockTokens { - connection_id, - timeout_timestamp, - duration, - coins, - } => execute_lock_tokens(connection_id, timeout_timestamp, duration, coins, deps), - ExecuteMsg::ExitSwapExternAmountOut { - connection_id, - timeout_timestamp, - pool_id, - share_in_amount, - token_out_mins, - } => execute_exit_swap_extern_amount_out( - connection_id, - timeout_timestamp, - pool_id, - share_in_amount, - token_out_mins, - deps, - ), - ExecuteMsg::BeginUnlocking { - connection_id, - timeout_timestamp, - id, - coins, - } => execute_begin_unlocking(connection_id, timeout_timestamp, id, coins, deps), - } -} - -pub fn execute_exit_pool( - connection_id: String, - timeout_timestamp: Uint64, - pool_id: Uint64, - share_in_amount: i64, - token_out_mins: Vec, - deps: DepsMut, -) -> Result, ContractError> { - let msg = IntergammMsg::ExitPool { - connection_id, - timeout_timestamp: timeout_timestamp.u64(), - pool_id: pool_id.u64(), - share_in_amount, - token_out_mins, - }; - create_intergamm_msg(deps.storage, msg).map_err(ContractError::Std) -} - -pub fn execute_join_pool( - connection_id: String, - timeout_timestamp: Uint64, - pool_id: Uint64, - share_out_amount: i64, - token_in_maxs: Vec, - deps: DepsMut, -) -> Result, ContractError> { - let msg = IntergammMsg::JoinPool { - connection_id, - timeout_timestamp: timeout_timestamp.u64(), - pool_id: pool_id.u64(), - share_out_amount, - token_in_maxs, - }; - create_intergamm_msg(deps.storage, msg).map_err(ContractError::Std) -} - -pub fn execute_lock_tokens( - connection_id: String, - timeout_timestamp: Uint64, - duration: Uint64, - coins: Vec, - deps: DepsMut, -) -> Result, ContractError> { - let msg = IntergammMsg::LockTokens { - connection_id, - timeout_timestamp: timeout_timestamp.u64(), - duration: duration.u64(), - coins, - }; - create_intergamm_msg(deps.storage, msg).map_err(ContractError::Std) -} - -pub fn execute_begin_unlocking( - connection_id: String, - timeout_timestamp: Uint64, - id: Uint64, - coins: Vec, - deps: DepsMut, -) -> Result, ContractError> { - let msg = IntergammMsg::BeginUnlocking { - connection_id, - timeout_timestamp: timeout_timestamp.u64(), - id: id.u64(), - coins, - }; - create_intergamm_msg(deps.storage, msg).map_err(ContractError::Std) -} - -pub fn execute_exit_swap_extern_amount_out( - connection_id: String, - timeout_timestamp: Uint64, - pool_id: Uint64, - share_in_amount: i64, - token_out_mins: Coin, - deps: DepsMut, -) -> Result, ContractError> { - let msg = IntergammMsg::ExitSwapExternAmountOut { - connection_id, - timeout_timestamp: timeout_timestamp.u64(), - pool_id: pool_id.u64(), - share_in_amount, - token_out_mins, - }; - create_intergamm_msg(deps.storage, msg).map_err(ContractError::Std) -} - -pub fn execute_send_token( - destination_local_zone_id: String, - receiver: String, - coin: Coin, -) -> Result, ContractError> { - // receiver is an address on a different chain, so we can't parse it. - Ok(Response::new() - .add_attribute( - "send_tokens", - format!( - "{} {} to {}", - coin.amount, coin.denom, destination_local_zone_id - ), - ) - .add_message(IntergammMsg::SendToken { - destination_local_zone_id, - receiver, - coin, - })) -} - -pub fn execute_send_token_ibc( - channel_id: String, - to_address: String, - amount: Coin, - env: Env, -) -> Result, ContractError> { - // timeout in 600 seconds after current block timestamp - let timeout = IbcTimeout::with_timestamp(env.block.time.plus_seconds(600)); - Ok(Response::new().add_message(IbcMsg::Transfer { - channel_id, - to_address, - amount, - timeout, - })) -} - -pub fn execute_test_scenario(scenario: String) -> Result, ContractError> { - Ok(Response::new().add_message(IntergammMsg::TestScenario { scenario })) -} - -pub fn execute_register_ica_on_zone( - zone_id: String, - deps: DepsMut, -) -> Result, ContractError> { - let msg = IntergammMsg::RegisterIcaOnZone { zone_id }; - create_intergamm_msg(deps.storage, msg).map_err(ContractError::Std) -} - -// join pool requires us to have a pool on the remote chain and funds in the interchain account of this contract -pub fn execute_join_swap_extern_amount_in( - connection_id: String, - pool_id: Uint64, - share_out_min_amount: i64, - token_in: Coin, - deps: DepsMut, - env: Env, -) -> Result, ContractError> { - let msg = IntergammMsg::JoinSwapExternAmountIn { - connection_id, - // timeout in 10 minutes - timeout_timestamp: env.block.time.plus_seconds(600).nanos(), - pool_id: pool_id.u64(), - share_out_min_amount, - token_in, - }; - create_intergamm_msg(deps.storage, msg).map_err(ContractError::Std) -} - -pub fn execute_deposit(info: MessageInfo) -> Result, ContractError> { - let funds = cw_utils::one_coin(&info)?; - if funds.denom != "uqsr" && funds.denom != "stake" { - return Err(ContractError::PaymentError( - cw_utils::PaymentError::MissingDenom("uqsr/stake".into()), - )); - } - // we dont do anything else with the funds since we solely use them for testing and don't need to deposit - Ok(Response::new() - .add_attribute("deposit_amount", funds.amount) - .add_attribute("deposit_denom", funds.denom)) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn query(deps: Deps, _env: Env, msg: QueryMsg) -> StdResult { - match msg { - QueryMsg::Acks {} => to_json_binary(&query_acks(deps)?), - QueryMsg::PendingAcks {} => to_json_binary(&query_pending_acks(deps)?), - } -} - -pub fn query_acks(deps: Deps) -> StdResult { - let acks: Result, StdError> = - intergamm_bindings::state::ACKS - .range(deps.storage, None, None, Order::Ascending) - .collect(); - Ok(AcksResponse { acks: acks? }) -} - -pub fn query_pending_acks(deps: Deps) -> StdResult { - let pending: Result, StdError> = - intergamm_bindings::state::PENDINGACKS - .range(deps.storage, None, None, Order::Ascending) - .collect(); - Ok(PendingAcksResponse { pending: pending? }) -} - -pub fn do_ibc_packet_ack( - deps: DepsMut, - _env: Env, - info: MessageInfo, - sequence: u64, - error: Option, - response: Option, -) -> Result, ContractError> { - check_callback_addr(deps.as_ref(), info.sender)?; - ack(deps, sequence, &error, &response)?; - // Insert any further neede logic to handle acks here - Ok(Response::new() - .add_attribute("error", error.unwrap_or_else(|| "none".into())) - .add_attribute("response", format!("{response:?}"))) -} - -#[cfg(test)] -mod tests {} diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/error.rs b/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/error.rs deleted file mode 100644 index 71e1bd749..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/error.rs +++ /dev/null @@ -1,22 +0,0 @@ -use cosmwasm_std::StdError; -use thiserror::Error; - -#[derive(Error, Debug)] -pub enum ContractError { - #[error("{0}")] - Std(#[from] StdError), - - #[error("Unauthorized")] - Unauthorized {}, - - #[error("{0}")] - PaymentError(#[from] cw_utils::PaymentError), - - #[error("bindings {0}")] - Bindings(#[from] intergamm_bindings::error::ContractError), - - #[error("Custom Error val: {val:?}")] - CustomError { val: String }, - // Add any other custom errors you like here. - // Look at https://docs.rs/thiserror/1.0.21/thiserror/ for details. -} diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/integration_tests.rs b/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/integration_tests.rs deleted file mode 100644 index ae09e2d44..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/integration_tests.rs +++ /dev/null @@ -1,70 +0,0 @@ -#[cfg(test)] -mod tests { - use crate::msg::InstantiateMsg; - use cosmwasm_std::{Addr, Coin, Empty, Uint128}; - use cw_multi_test::{App, AppBuilder, Contract, ContractWrapper, Executor}; - - pub fn contract_template() -> Box> { - let contract = ContractWrapper::new( - crate::contract::execute, - crate::contract::instantiate, - crate::contract::query, - ); - Box::new(contract) - } - - const USER: &str = "USER"; - const ADMIN: &str = "ADMIN"; - const NATIVE_DENOM: &str = "denom"; - - fn mock_app() -> App { - AppBuilder::new().build(|router, _, storage| { - router - .bank - .init_balance( - storage, - &Addr::unchecked(USER), - vec![Coin { - denom: NATIVE_DENOM.to_string(), - amount: Uint128::new(1), - }], - ) - .unwrap(); - }) - } - - fn proper_instantiate() -> (App, CwTemplateContract) { - let mut app = mock_app(); - let cw_template_id = app.store_code(contract_template()); - - let msg = InstantiateMsg { }; - let cw_template_contract_addr = app - .instantiate_contract( - cw_template_id, - Addr::unchecked(ADMIN), - &msg, - &[], - "test", - None, - ) - .unwrap(); - - let cw_template_contract = CwTemplateContract(cw_template_contract_addr); - - (app, cw_template_contract) - } - - mod count { - use super::*; - use crate::msg::ExecuteMsg; - - #[test] - fn count() { - let (mut app, cw_template_contract) = proper_instantiate(); - - let msg = ExecuteMsg::Increment {}; - let cosmos_msg = cw_template_contract.call(msg).unwrap(); - app.execute(Addr::unchecked(USER), cosmos_msg).unwrap(); - } - } -} diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/lib.rs b/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/lib.rs deleted file mode 100644 index dfedc9dc6..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/lib.rs +++ /dev/null @@ -1,6 +0,0 @@ -pub mod contract; -mod error; -pub mod msg; -pub mod state; - -pub use crate::error::ContractError; diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/msg.rs b/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/msg.rs deleted file mode 100644 index 6c2bb1b0c..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/msg.rs +++ /dev/null @@ -1,98 +0,0 @@ -use cosmwasm_std::{Coin, Uint64}; -use intergamm_bindings::msg::IntergammMsg; -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -pub struct InstantiateMsg { - pub callback_address: String, -} - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -#[serde(rename_all = "snake_case")] -pub enum ExecuteMsg { - SendToken { - destination_local_zone_id: String, - receiver: String, - coin: Coin, - }, - SendTokenIbc { - /// exisiting channel to send the tokens over - channel_id: String, - /// address on the remote chain to receive these tokens - to_address: String, - /// packet data only supports one coin - /// https://github.com/cosmos/cosmos-sdk/blob/v0.40.0/proto/ibc/applications/transfer/v1/transfer.proto#L11-L20 - amount: Coin, - // when packet times out, measured on remote chain, for now we hardcode the timeout - // timeout: IbcTimeout, - }, - Deposit {}, - RegisterIcaOnZone { - zone_id: String, - }, - JoinPool { - connection_id: String, - timeout_timestamp: Uint64, - pool_id: Uint64, - share_out_amount: i64, - token_in_maxs: Vec, - }, - ExitPool { - connection_id: String, - timeout_timestamp: Uint64, - pool_id: Uint64, - share_in_amount: i64, - token_out_mins: Vec, - }, - LockTokens { - connection_id: String, - timeout_timestamp: Uint64, - duration: Uint64, - coins: Vec, - }, - JoinSwapExternAmountIn { - connection_id: String, - pool_id: Uint64, - share_out_min_amount: i64, - token_in: Coin, - }, - ExitSwapExternAmountOut { - connection_id: String, - timeout_timestamp: Uint64, - pool_id: Uint64, - share_in_amount: i64, - token_out_mins: Coin, - }, - BeginUnlocking { - connection_id: String, - timeout_timestamp: Uint64, - id: Uint64, - coins: Vec, - }, - TestIcaScenario {}, - Ack { - sequence_number: u64, - error: Option, - response: Option, - }, -} - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -#[serde(rename_all = "snake_case")] -pub enum QueryMsg { - PendingAcks {}, - Acks {}, -} - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -#[serde(rename_all = "snake_case")] -pub struct AcksResponse { - pub acks: Vec<(u64, intergamm_bindings::msg::AckValue)>, -} - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -#[serde(rename_all = "snake_case")] -pub struct PendingAcksResponse { - pub pending: Vec<(u64, IntergammMsg)>, -} diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/state.rs b/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/state.rs deleted file mode 100644 index 0c9da70e7..000000000 --- a/smart-contracts/osmosis/contracts/intergamm-bindings-test/src/state.rs +++ /dev/null @@ -1,23 +0,0 @@ -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -use cosmwasm_std::Addr; -use cw_storage_plus::Item; - -use crate::ContractError; - -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -pub struct State { - pub count: i32, - pub owner: Addr, -} - -pub const ACKTRIGGERED: Item = Item::new("ack_triggered"); -pub enum Status { - Succes, - // TODO saving the error as part of the sequence might not be desirable for the future - Error { reason: ContractError }, - InProgress, -} - -pub const STATE: Item = Item::new("state"); diff --git a/smart-contracts/osmosis/contracts/intergamm-bindings-test/test.sh b/smart-contracts/osmosis/contracts/intergamm-bindings-test/test.sh deleted file mode 100644 index e69de29bb..000000000 diff --git a/smart-contracts/osmosis/contracts/lp-strategy/.cargo/config b/smart-contracts/osmosis/contracts/lp-strategy/.cargo/config deleted file mode 100644 index dba2c3f7a..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/.cargo/config +++ /dev/null @@ -1,5 +0,0 @@ -[alias] -wasm = "build --release --target wasm32-unknown-unknown" -unit-test = "test --lib" -schema = "run --example schema" -lint = "clippy -- -D warnings" \ No newline at end of file diff --git a/smart-contracts/osmosis/contracts/lp-strategy/CHANGELOG.md b/smart-contracts/osmosis/contracts/lp-strategy/CHANGELOG.md deleted file mode 100644 index 19336e815..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/CHANGELOG.md +++ /dev/null @@ -1,64 +0,0 @@ -# CHANGELOG - -## Unreleased - -### Dependencies - -### API breaking - -### State breaking - -- Delete old pending acks from the state -- Delete failed traps from the state - -### Improvements - -- Add bond queue duplicate key check -- Add testing to try_icq -- Remove unnecessary load from try_icq -- Changed the locking on the execute calls to lock correctly depending on queue state -- Delete pending ack entry after succesful ack handling -- Added some doc comments -- Created execute.rs file and created retry exit pool fn there -- Added proptests for retry join pool - -### Features - -- Added permission-less retry entrypoint to handle exit pool errors -- Added permission-less retry entrypoint to handle join pool errors - -### Bugfixes - -- divide quote denom by spotprice instead of multiply with spotprice -- readd proper lock behaviour -- Do not allow opentry messages to clog up our state -- Compare users' shares to their owned amount of queued shares instead of all queued shares -- make it so that the primitive compounds -- using only the unbonds amount to calculate slippage (previously using total shares amount) -- fixed math on consolidate_exit_pool_amount_into_local_denom - -## V0.1.1 08-05-2023 - -### Dependencies - -### API breaking - -### State breaking - -- Recover user bonds with manual callbacks - -### Improvements - -### Features - -### Bugfixes - -### Notes - -Migrations (branch names) performed with this source: -migration-004/recover-bonds -migration-005/recover-bonds-again - -## V0.1.0 - -### Initial version diff --git a/smart-contracts/osmosis/contracts/lp-strategy/Cargo.toml b/smart-contracts/osmosis/contracts/lp-strategy/Cargo.toml deleted file mode 100644 index e5696410e..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/Cargo.toml +++ /dev/null @@ -1,38 +0,0 @@ -[package] -authors = ["Laurens Kubat "] -edition = "2021" -name = "lp-strategy" -version = "0.1.1" - -[lib] -crate-type = ["cdylib", "rlib"] - -# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html -[features] -backtraces = ["cosmwasm-std/backtraces"] -# use library feature to disable all instantiate/execute/query exports -library = [] - -[dependencies] -cosmwasm-std = { workspace = true } -osmosis-std = { workspace = true } -osmosis-std-derive = { workspace = true } -cosmwasm-schema = { workspace = true } -cw-storage-plus ={ workspace = true } -schemars ={ workspace = true } -serde = { workspace = true } -thiserror = { workspace = true } -prost = { workspace = true } -cw-utils = { workspace = true } -cw2 = { workspace = true } -cw20 = { workspace = true } -cw20-base = { workspace = true } -cosmos-sdk-proto = { workspace = true } -serde-json-wasm = { workspace = true } - -uuid = {version = "1.1.2", default-features = false, features = ["v4", "js"]} -quasar-types = {path = "../../packages/quasar-types"} - -[dev-dependencies] -cw-multi-test = { workspace = true } -proptest = "1.0.0" diff --git a/smart-contracts/osmosis/contracts/lp-strategy/README.md b/smart-contracts/osmosis/contracts/lp-strategy/README.md deleted file mode 100644 index dcdcb383d..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/README.md +++ /dev/null @@ -1,19 +0,0 @@ -# LP-strategy - -## current state -Currently, the strategy supports receiving funds from a depositor, transferring those funds over a channel to it's own ICA address, joining a predetermined pool with those funds and locking those funds for two weeks - -## to be added -The strategy currently lacks the following that need to be added -- [ ] bookkeeping -- [ ] withdraws (dependant on ibc hooks on quasar, memo field in ibc transfer over ICA) -- [ ] move channel logic to quasar-types - -## how does the contract work -In ibc.rs, we have the cosmwasm entrypoints to setup the different IBC connections, this contract currently supports ICQ and ICA channels, but there is currently no use case for ICQ. - -After channels are setup, users can call the `TransferJoinLock` and `DepositAndLockTokens` execute Msgs, `TransferJoinLock`, transfers funds, joins the pool and locks the tokens, `DepositAndLockTokens` only joins the pool and locks the tokens - -## Testing locally -In demos/orion-manual-demo, `run_all.sh` sets up all chain needed, and `create_and_execute_contract` sets up a contract and starts trying to `TransferJoinLock` for Alice - diff --git a/smart-contracts/osmosis/contracts/lp-strategy/examples/schema.rs b/smart-contracts/osmosis/contracts/lp-strategy/examples/schema.rs deleted file mode 100644 index 819abb062..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/examples/schema.rs +++ /dev/null @@ -1,10 +0,0 @@ -use cosmwasm_schema::write_api; -use lp_strategy::msg::{ExecuteMsg, InstantiateMsg, QueryMsg}; - -fn main() { - write_api! { - instantiate: InstantiateMsg, - query: QueryMsg, - execute: ExecuteMsg, - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/migration_primitive1.json b/smart-contracts/osmosis/contracts/lp-strategy/migration_primitive1.json deleted file mode 100644 index 53f4565dc..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/migration_primitive1.json +++ /dev/null @@ -1,10 +0,0 @@ -[ - {"claim_amount":"28971416","raw_amount":{"local_denom":"6982413"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"13"}, - {"claim_amount":"145064310","raw_amount":{"local_denom":"26615807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"21"}, - {"claim_amount":"4061797","raw_amount":{"local_denom":"745242"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"22"}, - {"claim_amount":"521529251","raw_amount":{"local_denom":"95688056"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"23"}, - {"claim_amount":"5814419","raw_amount":{"local_denom":"1066806"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"24"}, - {"claim_amount":"3953805","raw_amount":{"local_denom":"725428"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"25"}, - {"claim_amount":"409062074","raw_amount":{"local_denom":"75053038"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"26"}, - {"claim_amount":"1503723755","raw_amount":{"local_denom":"79003403"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1468"} -] \ No newline at end of file diff --git a/smart-contracts/osmosis/contracts/lp-strategy/migration_primitive3.json b/smart-contracts/osmosis/contracts/lp-strategy/migration_primitive3.json deleted file mode 100644 index 3324cacff..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/migration_primitive3.json +++ /dev/null @@ -1,22 +0,0 @@ -[ - {"claim_amount":"43881823","raw_amount":{"local_denom":"32201796"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"20"},{"claim_amount":"88650150","raw_amount":{"local_denom":"65054135"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"21"},{"claim_amount":"2482203","raw_amount":{"local_denom":"1821515"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"22"},{"claim_amount":"315431924","raw_amount":{"local_denom":"231473390"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"23"},{"claim_amount":"2063730","raw_amount":{"local_denom":"1514427"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"25"},{"claim_amount":"248209338","raw_amount":{"local_denom":"182143444"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"26"}, - {"claim_amount":"4043454","raw_amount":{"local_denom":"5877677"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"491"}, - {"claim_amount":"12776164","raw_amount":{"local_denom":"8615945"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"492"}, - {"claim_amount":"2874701","raw_amount":{"local_denom":"1938631"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"493"}, - {"claim_amount":"15970566","raw_amount":{"local_denom":"10770175"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"494"}, - {"claim_amount":"63882268","raw_amount":{"local_denom":"43080701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"495"}, - {"claim_amount":"1213762","raw_amount":{"local_denom":"818533"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"496"},{"claim_amount":"12776453","raw_amount":{"local_denom":"8616140"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"497"},{"claim_amount":"89435176","raw_amount":{"local_denom":"60312982"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"498"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"499"},{"claim_amount":"4152346","raw_amount":{"local_denom":"2800245"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"500"},{"claim_amount":"91351644","raw_amount":{"local_denom":"61605403"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"501"},{"claim_amount":"2874702108","raw_amount":{"local_denom":"1938631576"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"502"},{"claim_amount":"37690537","raw_amount":{"local_denom":"25417613"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"503"},{"claim_amount":"2171995","raw_amount":{"local_denom":"1464743"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"504"}, - {"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"505"},{"claim_amount":"1659661350","raw_amount":{"local_denom":"1119236630"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"506"},{"claim_amount":"165455076","raw_amount":{"local_denom":"111579017"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"507"},{"claim_amount":"1916467","raw_amount":{"local_denom":"1292421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"508"},{"claim_amount":"766586","raw_amount":{"local_denom":"516968"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"509"},{"claim_amount":"6707637","raw_amount":{"local_denom":"4523473"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"510"},{"claim_amount":"8943517","raw_amount":{"local_denom":"6031298"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"511"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"512"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"513"},{"claim_amount":"5110581","raw_amount":{"local_denom":"3446456"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"514"}, - {"claim_amount":"638183866","raw_amount":{"local_denom":"430376209"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"515"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"516"},{"claim_amount":"4471758","raw_amount":{"local_denom":"3015649"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"517"},{"claim_amount":"11498807","raw_amount":{"local_denom":"7754526"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"518"},{"claim_amount":"111155146","raw_amount":{"local_denom":"74960420"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"519"},{"claim_amount":"1916467","raw_amount":{"local_denom":"1292421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"520"},{"claim_amount":"63946150","raw_amount":{"local_denom":"43123782"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"521"},{"claim_amount":"44717588","raw_amount":{"local_denom":"30156491"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"522"},{"claim_amount":"2427525","raw_amount":{"local_denom":"1637066"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"523"},{"claim_amount":"63882268","raw_amount":{"local_denom":"43080701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"524"}, - {"claim_amount":"8240811","raw_amount":{"local_denom":"5557410"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"525"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"526"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"527"},{"claim_amount":"3171575","raw_amount":{"local_denom":"2138836"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"528"},{"claim_amount":"2171995","raw_amount":{"local_denom":"1464743"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"529"},{"claim_amount":"32579955","raw_amount":{"local_denom":"21971157"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"530"},{"claim_amount":"91287761","raw_amount":{"local_denom":"61562322"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"531"},{"claim_amount":"7090930","raw_amount":{"local_denom":"4781957"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"532"},{"claim_amount":"63882268","raw_amount":{"local_denom":"43080701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"533"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"534"},{"claim_amount":"8649659","raw_amount":{"local_denom":"5833127"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"535"},{"claim_amount":"63882268","raw_amount":{"local_denom":"43080701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"536"},{"claim_amount":"63243445","raw_amount":{"local_denom":"42649894"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"537"},{"claim_amount":"63818385","raw_amount":{"local_denom":"43037620"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"538"},{"claim_amount":"18525857","raw_amount":{"local_denom":"12493403"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"539"},{"claim_amount":"638822689","raw_amount":{"local_denom":"430807016"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"540"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"541"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"542"},{"claim_amount":"120098665","raw_amount":{"local_denom":"80991719"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"543"},{"claim_amount":"1149880842","raw_amount":{"local_denom":"775452630"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"544"},{"claim_amount":"12457041","raw_amount":{"local_denom":"8400736"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"545"},{"claim_amount":"25552906","raw_amount":{"local_denom":"17232280"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"546"},{"claim_amount":"8135406967","raw_amount":{"local_denom":"5486327360"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"547"},{"claim_amount":"24914083","raw_amount":{"local_denom":"16801473"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"548"},{"claim_amount":"536611060","raw_amount":{"local_denom":"361877894"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"549"},{"claim_amount":"1149880842","raw_amount":{"local_denom":"775452630"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"550"},{"claim_amount":"63754504","raw_amount":{"local_denom":"42994540"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"551"},{"claim_amount":"159705672","raw_amount":{"local_denom":"107701754"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"552"},{"claim_amount":"5749403","raw_amount":{"local_denom":"3877263"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"553"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"554"},{"claim_amount":"223524059","raw_amount":{"local_denom":"150739375"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"555"},{"claim_amount":"8304694","raw_amount":{"local_denom":"5600491"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"556"},{"claim_amount":"2235878","raw_amount":{"local_denom":"1507824"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"557"},{"claim_amount":"1916467","raw_amount":{"local_denom":"1292421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"558"},{"claim_amount":"702703","raw_amount":{"local_denom":"473887"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"559"},{"claim_amount":"89435176","raw_amount":{"local_denom":"60312982"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"560"},{"claim_amount":"287470209","raw_amount":{"local_denom":"193863157"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"561"},{"claim_amount":"30024665","raw_amount":{"local_denom":"20247929"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"562"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"563"},{"claim_amount":"1852585","raw_amount":{"local_denom":"1249340"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"564"},{"claim_amount":"1213762","raw_amount":{"local_denom":"818533"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"565"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"566"},{"claim_amount":"12712570","raw_amount":{"local_denom":"8573059"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"567"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"568"},{"claim_amount":"82408126","raw_amount":{"local_denom":"55574105"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"569"},{"claim_amount":"12073747","raw_amount":{"local_denom":"8142252"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"570"},{"claim_amount":"19036916","raw_amount":{"local_denom":"12838049"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"571"},{"claim_amount":"19036916","raw_amount":{"local_denom":"12838049"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"572"},{"claim_amount":"18525857","raw_amount":{"local_denom":"12493403"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"573"},{"claim_amount":"19931266","raw_amount":{"local_denom":"13441178"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"574"},{"claim_amount":"45356410","raw_amount":{"local_denom":"30587298"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"575"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"576"}, - {"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"577"},{"claim_amount":"45356410","raw_amount":{"local_denom":"30587298"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"578"},{"claim_amount":"1788702","raw_amount":{"local_denom":"1206259"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"579"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"580"},{"claim_amount":"3832935","raw_amount":{"local_denom":"2584842"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"581"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"582"},{"claim_amount":"670763","raw_amount":{"local_denom":"452347"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"583"},{"claim_amount":"15970566","raw_amount":{"local_denom":"10770175"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"584"},{"claim_amount":"702703","raw_amount":{"local_denom":"473887"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"585"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"586"},{"claim_amount":"4343993","raw_amount":{"local_denom":"2929487"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"587"},{"claim_amount":"3602959975","raw_amount":{"local_denom":"2429751575"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"588"},{"claim_amount":"6068814","raw_amount":{"local_denom":"4092666"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"589"},{"claim_amount":"26830551","raw_amount":{"local_denom":"18093894"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"590"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"591"},{"claim_amount":"2005903","raw_amount":{"local_denom":"1352734"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"592"},{"claim_amount":"3066347","raw_amount":{"local_denom":"2067873"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"593"},{"claim_amount":"3066347","raw_amount":{"local_denom":"2067873"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"594"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"595"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"596"},{"claim_amount":"49828169","raw_amount":{"local_denom":"33602947"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"597"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"598"},{"claim_amount":"1916467","raw_amount":{"local_denom":"1292421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"599"},{"claim_amount":"350713656","raw_amount":{"local_denom":"236513052"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"600"},{"claim_amount":"33186838","raw_amount":{"local_denom":"22380424"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"601"},{"claim_amount":"3769053","raw_amount":{"local_denom":"2541761"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"602"},{"claim_amount":"3769053","raw_amount":{"local_denom":"2541761"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"603"},{"claim_amount":"14692921","raw_amount":{"local_denom":"9908561"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"604"},{"claim_amount":"15331743","raw_amount":{"local_denom":"10339368"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"605"},{"claim_amount":"185258579","raw_amount":{"local_denom":"124934034"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"606"},{"claim_amount":"88796353","raw_amount":{"local_denom":"59882175"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"607"},{"claim_amount":"18525857","raw_amount":{"local_denom":"12493403"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"608"},{"claim_amount":"9262927","raw_amount":{"local_denom":"6246701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"609"},{"claim_amount":"98378693","raw_amount":{"local_denom":"66344280"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"610"},{"claim_amount":"5749403","raw_amount":{"local_denom":"3877263"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"611"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"612"},{"claim_amount":"31621722","raw_amount":{"local_denom":"21324947"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"613"},{"claim_amount":"89435176","raw_amount":{"local_denom":"60312982"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"614"},{"claim_amount":"3833574967","raw_amount":{"local_denom":"2585272908"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"615"},{"claim_amount":"7027049","raw_amount":{"local_denom":"4738877"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"616"},{"claim_amount":"38329361","raw_amount":{"local_denom":"25848421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"617"},{"claim_amount":"1750373","raw_amount":{"local_denom":"1180411"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"618"},{"claim_amount":"905850575","raw_amount":{"local_denom":"610884350"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"619"},{"claim_amount":"4407875","raw_amount":{"local_denom":"2972568"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"620"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"621"},{"claim_amount":"862409","raw_amount":{"local_denom":"581589"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"622"},{"claim_amount":"35135246","raw_amount":{"local_denom":"23694385"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"623"},{"claim_amount":"5110581","raw_amount":{"local_denom":"3446456"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"624"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"625"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"626"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"627"},{"claim_amount":"166093899","raw_amount":{"local_denom":"112009824"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"628"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"629"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"630"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"631"},{"claim_amount":"4471758","raw_amount":{"local_denom":"3015649"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"632"},{"claim_amount":"38329361","raw_amount":{"local_denom":"25848421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"633"},{"claim_amount":"28747019","raw_amount":{"local_denom":"19386315"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"634"},{"claim_amount":"734645","raw_amount":{"local_denom":"495428"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"635"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"636"},{"claim_amount":"1269340685","raw_amount":{"local_denom":"856013542"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"637"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"638"},{"claim_amount":"8815751","raw_amount":{"local_denom":"5945136"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"639"},{"claim_amount":"2874701","raw_amount":{"local_denom":"1938631"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"640"},{"claim_amount":"95184580","raw_amount":{"local_denom":"64190245"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"641"},{"claim_amount":"1852585","raw_amount":{"local_denom":"1249340"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"642"},{"claim_amount":"15970566","raw_amount":{"local_denom":"10770175"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"643"},{"claim_amount":"633073285","raw_amount":{"local_denom":"426929753"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"644"},{"claim_amount":"28747019","raw_amount":{"local_denom":"19386315"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"645"},{"claim_amount":"9262927","raw_amount":{"local_denom":"6246701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"646"},{"claim_amount":"34496423","raw_amount":{"local_denom":"23263578"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"647"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"648"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"649"},{"claim_amount":"766586","raw_amount":{"local_denom":"516968"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"650"},{"claim_amount":"12776453","raw_amount":{"local_denom":"8616140"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"651"},{"claim_amount":"37690537","raw_amount":{"local_denom":"25417613"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"652"},{"claim_amount":"204423260","raw_amount":{"local_denom":"137858245"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"653"},{"claim_amount":"91798820","raw_amount":{"local_denom":"61906968"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"654"},{"claim_amount":"41523474","raw_amount":{"local_denom":"28002456"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"655"},{"claim_amount":"63243445","raw_amount":{"local_denom":"42649894"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"656"},{"claim_amount":"63882268","raw_amount":{"local_denom":"43080701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"657"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"658"},{"claim_amount":"6324343","raw_amount":{"local_denom":"4264989"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"659"},{"claim_amount":"81130481","raw_amount":{"local_denom":"54712491"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"660"},{"claim_amount":"23317028","raw_amount":{"local_denom":"15724456"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"661"},{"claim_amount":"18525857","raw_amount":{"local_denom":"12493403"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"662"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"663"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"664"},{"claim_amount":"1852585","raw_amount":{"local_denom":"1249340"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"665"},{"claim_amount":"22358793","raw_amount":{"local_denom":"15078245"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"666"},{"claim_amount":"241474976","raw_amount":{"local_denom":"162845052"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"667"},{"claim_amount":"1916467","raw_amount":{"local_denom":"1292421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"668"},{"claim_amount":"73464608","raw_amount":{"local_denom":"49542806"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"669"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"670"},{"claim_amount":"1128799693","raw_amount":{"local_denom":"761235998"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"671"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"672"},{"claim_amount":"3609347","raw_amount":{"local_denom":"2434059"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"673"},{"claim_amount":"5704685","raw_amount":{"local_denom":"3847106"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"674"},{"claim_amount":"14692921","raw_amount":{"local_denom":"9908561"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"675"},{"claim_amount":"11179395","raw_amount":{"local_denom":"7539122"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"676"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"677"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"678"},{"claim_amount":"639142101","raw_amount":{"local_denom":"431022420"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"679"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"680"},{"claim_amount":"241794387","raw_amount":{"local_denom":"163060455"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"681"},{"claim_amount":"585800406","raw_amount":{"local_denom":"395050034"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"682"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"683"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"684"},{"claim_amount":"670763","raw_amount":{"local_denom":"452347"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"685"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"686"},{"claim_amount":"5749403","raw_amount":{"local_denom":"3877263"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"687"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"688"},{"claim_amount":"1427129890","raw_amount":{"local_denom":"962422875"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"689"},{"claim_amount":"5781344","raw_amount":{"local_denom":"3898803"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"690"},{"claim_amount":"141818636","raw_amount":{"local_denom":"95639157"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"691"},{"claim_amount":"63243445","raw_amount":{"local_denom":"42649894"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"692"},{"claim_amount":"1149879","raw_amount":{"local_denom":"775452"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"693"},{"claim_amount":"95823402","raw_amount":{"local_denom":"64621052"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"694"},{"claim_amount":"56855218","raw_amount":{"local_denom":"38341824"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"695"},{"claim_amount":"15970566","raw_amount":{"local_denom":"10770175"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"696"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"697"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"698"},{"claim_amount":"65287678","raw_amount":{"local_denom":"44028477"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"699"},{"claim_amount":"15331743","raw_amount":{"local_denom":"10339368"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"700"},{"claim_amount":"14373509","raw_amount":{"local_denom":"9693157"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"701"}, - {"claim_amount":"12648687","raw_amount":{"local_denom":"8529978"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"702"},{"claim_amount":"85921650","raw_amount":{"local_denom":"57943543"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"703"},{"claim_amount":"15331743","raw_amount":{"local_denom":"10339368"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"704"},{"claim_amount":"11498807","raw_amount":{"local_denom":"7754526"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"705"},{"claim_amount":"12137630","raw_amount":{"local_denom":"8185333"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"706"},{"claim_amount":"159066849","raw_amount":{"local_denom":"107270947"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"707"},{"claim_amount":"258723188","raw_amount":{"local_denom":"174476841"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"708"},{"claim_amount":"38329361","raw_amount":{"local_denom":"25848421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"709"},{"claim_amount":"1022115","raw_amount":{"local_denom":"689291"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"710"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"711"},{"claim_amount":"19484092","raw_amount":{"local_denom":"13139614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"712"},{"claim_amount":"11498807","raw_amount":{"local_denom":"7754526"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"713"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"714"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"715"},{"claim_amount":"11498807","raw_amount":{"local_denom":"7754526"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"716"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"717"},{"claim_amount":"159577906","raw_amount":{"local_denom":"107615592"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"718"},{"claim_amount":"18525857","raw_amount":{"local_denom":"12493403"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"719"},{"claim_amount":"479117017","raw_amount":{"local_denom":"323105262"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"720"},{"claim_amount":"3960699","raw_amount":{"local_denom":"2671003"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"721"}, - {"claim_amount":"41204062","raw_amount":{"local_denom":"27787052"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"722"},{"claim_amount":"958234035","raw_amount":{"local_denom":"646210525"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"723"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"724"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"725"},{"claim_amount":"686734392","raw_amount":{"local_denom":"463117543"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"726"},{"claim_amount":"1213762","raw_amount":{"local_denom":"818533"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"727"},{"claim_amount":"245946735","raw_amount":{"local_denom":"165860701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"728"},{"claim_amount":"670763","raw_amount":{"local_denom":"452347"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"729"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"730"},{"claim_amount":"64904384","raw_amount":{"local_denom":"43769992"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"731"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"732"},{"claim_amount":"63818385","raw_amount":{"local_denom":"43037620"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"733"},{"claim_amount":"1469291","raw_amount":{"local_denom":"990856"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"734"},{"claim_amount":"577712911","raw_amount":{"local_denom":"389596017"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"735"},{"claim_amount":"5749403","raw_amount":{"local_denom":"3877263"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"736"},{"claim_amount":"1597055","raw_amount":{"local_denom":"1077017"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"737"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"738"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"739"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"740"},{"claim_amount":"3199224034","raw_amount":{"local_denom":"2157481540"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"741"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"742"},{"claim_amount":"79214013","raw_amount":{"local_denom":"53420070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"743"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"744"},{"claim_amount":"15651154","raw_amount":{"local_denom":"10554771"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"745"},{"claim_amount":"2874701","raw_amount":{"local_denom":"1938631"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"746"},{"claim_amount":"8943517","raw_amount":{"local_denom":"6031298"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"747"},{"claim_amount":"5749403","raw_amount":{"local_denom":"3877263"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"748"},{"claim_amount":"19164679","raw_amount":{"local_denom":"12924210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"749"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"750"},{"claim_amount":"82216480","raw_amount":{"local_denom":"55444863"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"752"},{"claim_amount":"42801120","raw_amount":{"local_denom":"28864070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"753"},{"claim_amount":"40884652","raw_amount":{"local_denom":"27571649"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"754"},{"claim_amount":"12776453","raw_amount":{"local_denom":"8616140"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"755"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"756"},{"claim_amount":"5749403","raw_amount":{"local_denom":"3877263"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"757"},{"claim_amount":"36540657","raw_amount":{"local_denom":"24642161"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"758"},{"claim_amount":"21719970","raw_amount":{"local_denom":"14647438"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"759"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"760"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"761"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"762"},{"claim_amount":"70909317","raw_amount":{"local_denom":"47819578"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"763"},{"claim_amount":"2171995","raw_amount":{"local_denom":"1464743"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"764"},{"claim_amount":"10859985","raw_amount":{"local_denom":"7323719"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"765"},{"claim_amount":"894350","raw_amount":{"local_denom":"603129"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"766"},{"claim_amount":"9582339","raw_amount":{"local_denom":"6462105"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"767"},{"claim_amount":"3832935","raw_amount":{"local_denom":"2584842"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"768"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"769"},{"claim_amount":"6068814","raw_amount":{"local_denom":"4092666"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"770"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"771"},{"claim_amount":"86879885","raw_amount":{"local_denom":"58589754"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"772"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"773"},{"claim_amount":"9582339","raw_amount":{"local_denom":"6462105"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"774"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"775"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"776"},{"claim_amount":"141818636","raw_amount":{"local_denom":"95639157"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"777"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"778"},{"claim_amount":"5110581","raw_amount":{"local_denom":"3446456"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"779"},{"claim_amount":"127125714","raw_amount":{"local_denom":"85730596"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"780"},{"claim_amount":"8943517","raw_amount":{"local_denom":"6031298"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"781"},{"claim_amount":"12137630","raw_amount":{"local_denom":"8185333"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"782"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"783"},{"claim_amount":"31302310","raw_amount":{"local_denom":"21109543"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"784"},{"claim_amount":"318133699","raw_amount":{"local_denom":"214541894"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"785"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"786"},{"claim_amount":"1916467","raw_amount":{"local_denom":"1292421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"787"},{"claim_amount":"2108114","raw_amount":{"local_denom":"1421663"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"788"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"789"},{"claim_amount":"19803502","raw_amount":{"local_denom":"13355017"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"790"},{"claim_amount":"25552906","raw_amount":{"local_denom":"17232280"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"791"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"792"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"793"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"794"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"795"},{"claim_amount":"5749403","raw_amount":{"local_denom":"3877263"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"796"},{"claim_amount":"670763","raw_amount":{"local_denom":"452347"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"797"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"798"},{"claim_amount":"5110581","raw_amount":{"local_denom":"3446456"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"799"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"800"},{"claim_amount":"4471758","raw_amount":{"local_denom":"3015649"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"801"},{"claim_amount":"2874701","raw_amount":{"local_denom":"1938631"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"802"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"803"},{"claim_amount":"3385759","raw_amount":{"local_denom":"2283277"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"804"},{"claim_amount":"63882268","raw_amount":{"local_denom":"43080701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"805"},{"claim_amount":"49828169","raw_amount":{"local_denom":"33602947"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"806"},{"claim_amount":"31941133","raw_amount":{"local_denom":"21540350"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"807"},{"claim_amount":"17887034","raw_amount":{"local_denom":"12062596"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"808"},{"claim_amount":"27469374","raw_amount":{"local_denom":"18524701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"809"},{"claim_amount":"44078765","raw_amount":{"local_denom":"29725684"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"810"},{"claim_amount":"3832935","raw_amount":{"local_denom":"2584842"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"811"},{"claim_amount":"22358793","raw_amount":{"local_denom":"15078245"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"812"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"813"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"814"},{"claim_amount":"98378693","raw_amount":{"local_denom":"66344280"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"815"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"817"},{"claim_amount":"9422633","raw_amount":{"local_denom":"6354403"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"818"},{"claim_amount":"4471758","raw_amount":{"local_denom":"3015649"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"819"},{"claim_amount":"82408126","raw_amount":{"local_denom":"55574105"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"820"},{"claim_amount":"1213762","raw_amount":{"local_denom":"818533"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"821"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"822"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"823"},{"claim_amount":"86241062","raw_amount":{"local_denom":"58158947"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"824"},{"claim_amount":"12712570","raw_amount":{"local_denom":"8573059"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"825"},{"claim_amount":"60688154","raw_amount":{"local_denom":"40926666"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"826"},{"claim_amount":"3832935","raw_amount":{"local_denom":"2584842"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"827"},{"claim_amount":"1916467","raw_amount":{"local_denom":"1292421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"828"},{"claim_amount":"28747019","raw_amount":{"local_denom":"19386315"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"829"},{"claim_amount":"141754755","raw_amount":{"local_denom":"95596077"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"830"},{"claim_amount":"1916467","raw_amount":{"local_denom":"1292421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"831"}, - {"claim_amount":"2491407","raw_amount":{"local_denom":"1680147"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"832"},{"claim_amount":"5749403","raw_amount":{"local_denom":"3877263"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"833"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"834"},{"claim_amount":"3162171","raw_amount":{"local_denom":"2132494"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"835"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"836"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"837"},{"claim_amount":"63243445","raw_amount":{"local_denom":"42649894"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"838"},{"claim_amount":"47911701","raw_amount":{"local_denom":"32310526"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"839"},{"claim_amount":"77297545","raw_amount":{"local_denom":"52127649"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"840"},{"claim_amount":"1277645380","raw_amount":{"local_denom":"861614033"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"841"},{"claim_amount":"5110581","raw_amount":{"local_denom":"3446456"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"842"},{"claim_amount":"62604622","raw_amount":{"local_denom":"42219087"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"843"},{"claim_amount":"27469374","raw_amount":{"local_denom":"18524701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"844"},{"claim_amount":"42149521","raw_amount":{"local_denom":"28424647"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"845"},{"claim_amount":"127125714","raw_amount":{"local_denom":"85730596"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"846"},{"claim_amount":"124570424","raw_amount":{"local_denom":"84007368"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"847"},{"claim_amount":"19739619","raw_amount":{"local_denom":"13311936"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"848"},{"claim_amount":"10859985","raw_amount":{"local_denom":"7323719"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"849"},{"claim_amount":"67076381","raw_amount":{"local_denom":"45234736"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"850"},{"claim_amount":"412040634","raw_amount":{"local_denom":"277870525"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"851"},{"claim_amount":"638183866","raw_amount":{"local_denom":"430376209"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"852"},{"claim_amount":"1916467","raw_amount":{"local_denom":"1292421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"853"},{"claim_amount":"1245703","raw_amount":{"local_denom":"840073"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"854"},{"claim_amount":"12776453","raw_amount":{"local_denom":"8616140"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"855"},{"claim_amount":"40884652","raw_amount":{"local_denom":"27571649"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"856"},{"claim_amount":"2235878","raw_amount":{"local_denom":"1507824"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"857"},{"claim_amount":"1852585","raw_amount":{"local_denom":"1249340"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"858"},{"claim_amount":"8879634","raw_amount":{"local_denom":"5988217"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"859"},{"claim_amount":"63882268","raw_amount":{"local_denom":"43080701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"860"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"861"},{"claim_amount":"6260461","raw_amount":{"local_denom":"4221908"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"862"},{"claim_amount":"20442325","raw_amount":{"local_denom":"13785824"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"863"},{"claim_amount":"3705170","raw_amount":{"local_denom":"2498680"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"864"},{"claim_amount":"41459592","raw_amount":{"local_denom":"27959375"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"865"},{"claim_amount":"3130230","raw_amount":{"local_denom":"2110954"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"866"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"867"},{"claim_amount":"785750","raw_amount":{"local_denom":"529892"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"868"},{"claim_amount":"10221162","raw_amount":{"local_denom":"6892912"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"869"},{"claim_amount":"9582339","raw_amount":{"local_denom":"6462105"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"870"},{"claim_amount":"1248898360","raw_amount":{"local_denom":"842227718"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"871"},{"claim_amount":"15970566","raw_amount":{"local_denom":"10770175"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"872"},{"claim_amount":"63946150","raw_amount":{"local_denom":"43123782"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"873"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"874"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"875"},{"claim_amount":"7027049","raw_amount":{"local_denom":"4738877"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"876"},{"claim_amount":"50466992","raw_amount":{"local_denom":"34033754"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"877"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"878"},{"claim_amount":"1788702","raw_amount":{"local_denom":"1206259"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"879"},{"claim_amount":"319347462","raw_amount":{"local_denom":"215360427"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"880"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"881"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"882"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"883"},{"claim_amount":"31302310","raw_amount":{"local_denom":"21109543"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"884"},{"claim_amount":"1290421","raw_amount":{"local_denom":"870230"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"885"},{"claim_amount":"638183866","raw_amount":{"local_denom":"430376209"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"886"},{"claim_amount":"159386260","raw_amount":{"local_denom":"107486350"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"888"},{"claim_amount":"12137630","raw_amount":{"local_denom":"8185333"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"889"},{"claim_amount":"702703","raw_amount":{"local_denom":"473887"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"890"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"891"},{"claim_amount":"4407875","raw_amount":{"local_denom":"2972568"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"892"},{"claim_amount":"3832935","raw_amount":{"local_denom":"2584842"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"893"},{"claim_amount":"3513523","raw_amount":{"local_denom":"2369438"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"894"},{"claim_amount":"318772522","raw_amount":{"local_denom":"214972701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"895"},{"claim_amount":"159705672","raw_amount":{"local_denom":"107701754"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"896"},{"claim_amount":"43439942","raw_amount":{"local_denom":"29294877"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"897"},{"claim_amount":"13351393","raw_amount":{"local_denom":"9003866"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"898"},{"claim_amount":"13415275","raw_amount":{"local_denom":"9046947"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"899"},{"claim_amount":"320018226","raw_amount":{"local_denom":"215812775"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"900"},{"claim_amount":"2874701","raw_amount":{"local_denom":"1938631"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"901"},{"claim_amount":"37051714","raw_amount":{"local_denom":"24986806"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"902"},{"claim_amount":"17887034","raw_amount":{"local_denom":"12062596"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"903"},{"claim_amount":"4471758","raw_amount":{"local_denom":"3015649"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"904"},{"claim_amount":"753810","raw_amount":{"local_denom":"508352"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"905"},{"claim_amount":"79852836","raw_amount":{"local_denom":"53850877"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"906"},{"claim_amount":"1279561848","raw_amount":{"local_denom":"862906454"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"907"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"908"},{"claim_amount":"7985282","raw_amount":{"local_denom":"5385087"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"909"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"910"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"911"},{"claim_amount":"1277645380","raw_amount":{"local_denom":"861614033"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"912"},{"claim_amount":"8879634","raw_amount":{"local_denom":"5988217"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"913"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"914"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"915"},{"claim_amount":"2235878","raw_amount":{"local_denom":"1507824"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"916"},{"claim_amount":"6324343","raw_amount":{"local_denom":"4264989"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"917"},{"claim_amount":"63818385","raw_amount":{"local_denom":"43037620"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"918"},{"claim_amount":"2407721","raw_amount":{"local_denom":"1623711"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"919"},{"claim_amount":"56216396","raw_amount":{"local_denom":"37911017"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"920"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"921"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"922"},{"claim_amount":"12776453","raw_amount":{"local_denom":"8616140"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"923"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"924"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"925"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"926"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"927"},{"claim_amount":"1852585","raw_amount":{"local_denom":"1249340"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"928"},{"claim_amount":"63882268","raw_amount":{"local_denom":"43080701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"929"},{"claim_amount":"4471758","raw_amount":{"local_denom":"3015649"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"930"},{"claim_amount":"12137630","raw_amount":{"local_denom":"8185333"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"931"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"932"},{"claim_amount":"8943517","raw_amount":{"local_denom":"6031298"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"933"},{"claim_amount":"6068814","raw_amount":{"local_denom":"4092666"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"934"},{"claim_amount":"11434925","raw_amount":{"local_denom":"7711445"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"935"},{"claim_amount":"1252091","raw_amount":{"local_denom":"844381"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"936"},{"claim_amount":"13415275","raw_amount":{"local_denom":"9046947"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"937"},{"claim_amount":"24275261","raw_amount":{"local_denom":"16370666"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"938"},{"claim_amount":"142457458","raw_amount":{"local_denom":"96069964"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"939"},{"claim_amount":"1341526","raw_amount":{"local_denom":"904694"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"940"},{"claim_amount":"2414110948","raw_amount":{"local_denom":"1628019717"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"941"},{"claim_amount":"97420460","raw_amount":{"local_denom":"65698070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"942"},{"claim_amount":"6707637","raw_amount":{"local_denom":"4523473"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"943"},{"claim_amount":"395431244","raw_amount":{"local_denom":"266669543"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"944"},{"claim_amount":"19036916","raw_amount":{"local_denom":"12838049"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"945"},{"claim_amount":"12137630","raw_amount":{"local_denom":"8185333"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"946"},{"claim_amount":"79533424","raw_amount":{"local_denom":"53635473"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"947"},{"claim_amount":"1641774","raw_amount":{"local_denom":"1107174"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"948"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"949"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"950"},{"claim_amount":"4343993","raw_amount":{"local_denom":"2929487"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"951"},{"claim_amount":"6260461","raw_amount":{"local_denom":"4221908"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"952"},{"claim_amount":"76594840","raw_amount":{"local_denom":"51653761"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"953"},{"claim_amount":"3769053","raw_amount":{"local_denom":"2541761"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"954"},{"claim_amount":"205062083","raw_amount":{"local_denom":"138289052"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"955"}, - {"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"956"},{"claim_amount":"38329361","raw_amount":{"local_denom":"25848421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"957"},{"claim_amount":"75381076","raw_amount":{"local_denom":"50835227"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"958"},{"claim_amount":"63882268","raw_amount":{"local_denom":"43080701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"959"},{"claim_amount":"63243445","raw_amount":{"local_denom":"42649894"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"960"},{"claim_amount":"31941133","raw_amount":{"local_denom":"21540350"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"961"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"962"},{"claim_amount":"766586","raw_amount":{"local_denom":"516968"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"963"},{"claim_amount":"15970566","raw_amount":{"local_denom":"10770175"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"964"},{"claim_amount":"63243445","raw_amount":{"local_denom":"42649894"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"965"},{"claim_amount":"6324343","raw_amount":{"local_denom":"4264989"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"966"}, - {"claim_amount":"127125714","raw_amount":{"local_denom":"85730596"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"967"},{"claim_amount":"1533173","raw_amount":{"local_denom":"1033936"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"968"},{"claim_amount":"18525857","raw_amount":{"local_denom":"12493403"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"969"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"970"},{"claim_amount":"109238678","raw_amount":{"local_denom":"73667999"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"971"},{"claim_amount":"31941133","raw_amount":{"local_denom":"21540350"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"972"},{"claim_amount":"44717588","raw_amount":{"local_denom":"30156491"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"973"},{"claim_amount":"1533173","raw_amount":{"local_denom":"1033936"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"974"},{"claim_amount":"59410509","raw_amount":{"local_denom":"40065052"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"975"},{"claim_amount":"683539","raw_amount":{"local_denom":"460963"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"976"},{"claim_amount":"10540573","raw_amount":{"local_denom":"7108315"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"977"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"978"},{"claim_amount":"12137630","raw_amount":{"local_denom":"8185333"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"979"},{"claim_amount":"42801120","raw_amount":{"local_denom":"28864070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"980"},{"claim_amount":"12137630","raw_amount":{"local_denom":"8185333"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"981"},{"claim_amount":"79852836","raw_amount":{"local_denom":"53850877"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"982"},{"claim_amount":"19164679","raw_amount":{"local_denom":"12924210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"983"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"984"},{"claim_amount":"1724819","raw_amount":{"local_denom":"1163178"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"985"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"986"},{"claim_amount":"31302310","raw_amount":{"local_denom":"21109543"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"987"},{"claim_amount":"2491407","raw_amount":{"local_denom":"1680147"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"988"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"989"},{"claim_amount":"89435176","raw_amount":{"local_denom":"60312982"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"990"},{"claim_amount":"8943517","raw_amount":{"local_denom":"6031298"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"991"},{"claim_amount":"8943517","raw_amount":{"local_denom":"6031298"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"992"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"993"},{"claim_amount":"167946484","raw_amount":{"local_denom":"113259164"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"994"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"995"},{"claim_amount":"376266563","raw_amount":{"local_denom":"253745332"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"996"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"997"},{"claim_amount":"35135246","raw_amount":{"local_denom":"23694385"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"998"},{"claim_amount":"175676239","raw_amount":{"local_denom":"118471929"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"999"},{"claim_amount":"12776453","raw_amount":{"local_denom":"8616140"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1000"},{"claim_amount":"702703","raw_amount":{"local_denom":"473887"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1001"},{"claim_amount":"12712570","raw_amount":{"local_denom":"8573059"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1002"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1003"},{"claim_amount":"78575190","raw_amount":{"local_denom":"52989263"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1004"},{"claim_amount":"3513523","raw_amount":{"local_denom":"2369438"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1005"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1006"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1007"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1008"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1009"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1010"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1011"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1013"},{"claim_amount":"12776453","raw_amount":{"local_denom":"8616140"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1014"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1015"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1016"},{"claim_amount":"7346459","raw_amount":{"local_denom":"4954280"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1017"},{"claim_amount":"7985282","raw_amount":{"local_denom":"5385087"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1018"},{"claim_amount":"734645","raw_amount":{"local_denom":"495428"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1019"},{"claim_amount":"5749403","raw_amount":{"local_denom":"3877263"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1020"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1021"},{"claim_amount":"7665871","raw_amount":{"local_denom":"5169684"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1022"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1023"},{"claim_amount":"191646807","raw_amount":{"local_denom":"129242105"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1024"},{"claim_amount":"5493874","raw_amount":{"local_denom":"3704940"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1025"},{"claim_amount":"127445125","raw_amount":{"local_denom":"85945999"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1026"},{"claim_amount":"12137630","raw_amount":{"local_denom":"8185333"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1027"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1028"},{"claim_amount":"7027049","raw_amount":{"local_denom":"4738877"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1029"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1030"},{"claim_amount":"318772522","raw_amount":{"local_denom":"214972701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1031"},{"claim_amount":"114988084","raw_amount":{"local_denom":"77545263"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1032"},{"claim_amount":"56855218","raw_amount":{"local_denom":"38341824"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1033"},{"claim_amount":"1724819","raw_amount":{"local_denom":"1163178"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1034"},{"claim_amount":"641377981","raw_amount":{"local_denom":"432530245"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1035"},{"claim_amount":"11498807","raw_amount":{"local_denom":"7754526"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1036"},{"claim_amount":"66437558","raw_amount":{"local_denom":"44803929"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1037"},{"claim_amount":"2191161","raw_amount":{"local_denom":"1477668"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1038"},{"claim_amount":"28683138","raw_amount":{"local_denom":"19343235"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1039"},{"claim_amount":"2427525","raw_amount":{"local_denom":"1637066"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1040"},{"claim_amount":"10859985","raw_amount":{"local_denom":"7323719"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1041"},{"claim_amount":"12137630","raw_amount":{"local_denom":"8185333"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1042"},{"claim_amount":"12137630","raw_amount":{"local_denom":"8185333"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1043"},{"claim_amount":"2389196863","raw_amount":{"local_denom":"1611218243"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1044"},{"claim_amount":"127125714","raw_amount":{"local_denom":"85730596"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1045"},{"claim_amount":"89435176","raw_amount":{"local_denom":"60312982"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1046"},{"claim_amount":"6260461","raw_amount":{"local_denom":"4221908"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1047"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1048"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1049"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1050"},{"claim_amount":"95823402","raw_amount":{"local_denom":"64621052"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1051"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1052"},{"claim_amount":"12073747","raw_amount":{"local_denom":"8142252"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1053"},{"claim_amount":"5429991","raw_amount":{"local_denom":"3661859"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1054"},{"claim_amount":"33793720","raw_amount":{"local_denom":"22789691"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1055"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1056"},{"claim_amount":"926292","raw_amount":{"local_denom":"624670"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1057"},{"claim_amount":"31877252","raw_amount":{"local_denom":"21497270"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1058"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1059"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1060"},{"claim_amount":"574940421","raw_amount":{"local_denom":"387726315"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1061"},{"claim_amount":"17887034","raw_amount":{"local_denom":"12062596"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1062"},{"claim_amount":"6324343","raw_amount":{"local_denom":"4264989"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1063"},{"claim_amount":"82855302","raw_amount":{"local_denom":"55875670"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1064"},{"claim_amount":"12137630","raw_amount":{"local_denom":"8185333"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1065"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1066"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1067"},{"claim_amount":"25552906","raw_amount":{"local_denom":"17232280"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1068"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1069"},{"claim_amount":"3130230","raw_amount":{"local_denom":"2110954"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1070"},{"claim_amount":"791501312","raw_amount":{"local_denom":"533769893"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1071"},{"claim_amount":"7665871","raw_amount":{"local_denom":"5169684"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1072"},{"claim_amount":"12457041","raw_amount":{"local_denom":"8400736"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1073"},{"claim_amount":"1597055","raw_amount":{"local_denom":"1077017"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1074"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1075"},{"claim_amount":"574940421","raw_amount":{"local_denom":"387726315"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1076"},{"claim_amount":"7665871","raw_amount":{"local_denom":"5169684"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1077"},{"claim_amount":"1022115","raw_amount":{"local_denom":"689291"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1078"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1079"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1080"},{"claim_amount":"89051882","raw_amount":{"local_denom":"60054498"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1081"},{"claim_amount":"7665871","raw_amount":{"local_denom":"5169684"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1082"},{"claim_amount":"125145364","raw_amount":{"local_denom":"84395094"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1083"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1084"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1085"}, - {"claim_amount":"1245703","raw_amount":{"local_denom":"840073"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1086"},{"claim_amount":"5941050","raw_amount":{"local_denom":"4006505"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1087"},{"claim_amount":"6707637","raw_amount":{"local_denom":"4523473"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1088"},{"claim_amount":"44653705","raw_amount":{"local_denom":"30113410"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1089"},{"claim_amount":"7027049","raw_amount":{"local_denom":"4738877"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1090"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1091"},{"claim_amount":"1916467","raw_amount":{"local_denom":"1292421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1092"},{"claim_amount":"76658722","raw_amount":{"local_denom":"51696842"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1093"},{"claim_amount":"7027049","raw_amount":{"local_denom":"4738877"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1094"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1095"},{"claim_amount":"3130230","raw_amount":{"local_denom":"2110954"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1096"},{"claim_amount":"1213762","raw_amount":{"local_denom":"818533"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1097"},{"claim_amount":"7027049","raw_amount":{"local_denom":"4738877"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1098"},{"claim_amount":"22997615","raw_amount":{"local_denom":"15509052"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1099"},{"claim_amount":"127690862","raw_amount":{"local_denom":"86111718"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1100"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1101"},{"claim_amount":"40884652","raw_amount":{"local_denom":"27571649"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1102"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1103"},{"claim_amount":"318772522","raw_amount":{"local_denom":"214972701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1104"},{"claim_amount":"6707637","raw_amount":{"local_denom":"4523473"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1105"},{"claim_amount":"239558508","raw_amount":{"local_denom":"161552631"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1106"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1107"},{"claim_amount":"8304694","raw_amount":{"local_denom":"5600491"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1108"},{"claim_amount":"2317009899","raw_amount":{"local_denom":"1562537050"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1109"},{"claim_amount":"15970566","raw_amount":{"local_denom":"10770175"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1110"}, - {"claim_amount":"7027049","raw_amount":{"local_denom":"4738877"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1111"},{"claim_amount":"63243445","raw_amount":{"local_denom":"42649894"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1112"},{"claim_amount":"6707637","raw_amount":{"local_denom":"4523473"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1113"},{"claim_amount":"1012708","raw_amount":{"local_denom":"682947"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1114"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1115"},{"claim_amount":"3769053","raw_amount":{"local_denom":"2541761"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1116"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1117"},{"claim_amount":"18525857","raw_amount":{"local_denom":"12493403"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1118"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1119"},{"claim_amount":"3513523","raw_amount":{"local_denom":"2369438"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1120"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1121"},{"claim_amount":"14054098","raw_amount":{"local_denom":"9477754"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1122"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1123"},{"claim_amount":"1916467","raw_amount":{"local_denom":"1292421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1124"},{"claim_amount":"702703","raw_amount":{"local_denom":"473887"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1125"},{"claim_amount":"19100797","raw_amount":{"local_denom":"12881129"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1126"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1127"},{"claim_amount":"702703","raw_amount":{"local_denom":"473887"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1128"},{"claim_amount":"5685521","raw_amount":{"local_denom":"3834182"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1129"},{"claim_amount":"3513523","raw_amount":{"local_denom":"2369438"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1130"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1131"},{"claim_amount":"76019900","raw_amount":{"local_denom":"51266035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1132"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1133"},{"claim_amount":"4471758","raw_amount":{"local_denom":"3015649"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1134"},{"claim_amount":"2235878","raw_amount":{"local_denom":"1507824"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1135"},{"claim_amount":"63882269087","raw_amount":{"local_denom":"43080701693"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1136"},{"claim_amount":"16289977","raw_amount":{"local_denom":"10985578"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1137"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1138"},{"claim_amount":"6324343","raw_amount":{"local_denom":"4264989"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1139"},{"claim_amount":"894350","raw_amount":{"local_denom":"603129"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1140"},{"claim_amount":"78575190","raw_amount":{"local_denom":"52989263"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1141"},{"claim_amount":"172482125","raw_amount":{"local_denom":"116317894"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1142"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1143"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1144"},{"claim_amount":"22358793","raw_amount":{"local_denom":"15078245"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1145"},{"claim_amount":"3130230","raw_amount":{"local_denom":"2110954"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1146"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1147"},{"claim_amount":"8304694","raw_amount":{"local_denom":"5600491"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1148"},{"claim_amount":"2491407","raw_amount":{"local_denom":"1680147"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1149"},{"claim_amount":"1916467","raw_amount":{"local_denom":"1292421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1150"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1151"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1152"},{"claim_amount":"4212396822","raw_amount":{"local_denom":"2840741469"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1153"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1154"},{"claim_amount":"4471758","raw_amount":{"local_denom":"3015649"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1155"},{"claim_amount":"18525857","raw_amount":{"local_denom":"12493403"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1156"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1157"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1158"},{"claim_amount":"2529736","raw_amount":{"local_denom":"1705995"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1159"},{"claim_amount":"4791169","raw_amount":{"local_denom":"3231052"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1160"},{"claim_amount":"58771686","raw_amount":{"local_denom":"39634245"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1161"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1162"},{"claim_amount":"47272878","raw_amount":{"local_denom":"31879719"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1163"},{"claim_amount":"37690537","raw_amount":{"local_denom":"25417613"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1164"},{"claim_amount":"3066347","raw_amount":{"local_denom":"2067873"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1165"},{"claim_amount":"734645","raw_amount":{"local_denom":"495428"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1166"},{"claim_amount":"14692921","raw_amount":{"local_denom":"9908561"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1167"},{"claim_amount":"13415275","raw_amount":{"local_denom":"9046947"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1168"},{"claim_amount":"6324343","raw_amount":{"local_denom":"4264989"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1169"},{"claim_amount":"12457041","raw_amount":{"local_denom":"8400736"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1170"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1171"},{"claim_amount":"2491407","raw_amount":{"local_denom":"1680147"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1172"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1173"},{"claim_amount":"14054098","raw_amount":{"local_denom":"9477754"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1174"},{"claim_amount":"11179395","raw_amount":{"local_denom":"7539122"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1175"},{"claim_amount":"16258036","raw_amount":{"local_denom":"10964038"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1176"},{"claim_amount":"1916467","raw_amount":{"local_denom":"1292421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1177"},{"claim_amount":"6356284","raw_amount":{"local_denom":"4286529"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1178"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1179"},{"claim_amount":"97739870","raw_amount":{"local_denom":"65913473"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1180"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1181"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1182"},{"claim_amount":"102658805","raw_amount":{"local_denom":"69230687"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1183"},{"claim_amount":"14692921","raw_amount":{"local_denom":"9908561"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1184"},{"claim_amount":"44717588","raw_amount":{"local_denom":"30156491"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1185"},{"claim_amount":"2427525","raw_amount":{"local_denom":"1637066"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1186"},{"claim_amount":"1149879","raw_amount":{"local_denom":"775452"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1187"},{"claim_amount":"3194113","raw_amount":{"local_denom":"2154035"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1188"},{"claim_amount":"5642720827","raw_amount":{"local_denom":"3805318380"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1189"},{"claim_amount":"30982900","raw_amount":{"local_denom":"20894140"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1190"},{"claim_amount":"38329361","raw_amount":{"local_denom":"25848421"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1191"},{"claim_amount":"9837869","raw_amount":{"local_denom":"6634428"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1192"},{"claim_amount":"72186962","raw_amount":{"local_denom":"48681192"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1193"},{"claim_amount":"929487014","raw_amount":{"local_denom":"626824209"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1194"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1195"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1196"},{"claim_amount":"6068814","raw_amount":{"local_denom":"4092666"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1197"},{"claim_amount":"1213762","raw_amount":{"local_denom":"818533"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1198"},{"claim_amount":"60049332","raw_amount":{"local_denom":"40495859"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1199"},{"claim_amount":"14629038","raw_amount":{"local_denom":"9865480"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1200"},{"claim_amount":"2523348","raw_amount":{"local_denom":"1701687"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1201"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1202"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1203"},{"claim_amount":"2491407","raw_amount":{"local_denom":"1680147"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1204"},{"claim_amount":"82791420","raw_amount":{"local_denom":"55832589"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1205"},{"claim_amount":"7665871","raw_amount":{"local_denom":"5169684"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1206"},{"claim_amount":"32579955","raw_amount":{"local_denom":"21971157"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1207"},{"claim_amount":"4407875","raw_amount":{"local_denom":"2972568"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1208"},{"claim_amount":"14692921","raw_amount":{"local_denom":"9908561"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1209"},{"claim_amount":"1054056","raw_amount":{"local_denom":"710831"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1210"},{"claim_amount":"23317028","raw_amount":{"local_denom":"15724456"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1211"},{"claim_amount":"9422633","raw_amount":{"local_denom":"6354403"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1212"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1213"},{"claim_amount":"436315897","raw_amount":{"local_denom":"294241192"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1214"},{"claim_amount":"31877252","raw_amount":{"local_denom":"21497270"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1215"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1216"},{"claim_amount":"45931350","raw_amount":{"local_denom":"30975024"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1217"},{"claim_amount":"2555290","raw_amount":{"local_denom":"1723228"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1218"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1219"},{"claim_amount":"1277645","raw_amount":{"local_denom":"861614"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1220"},{"claim_amount":"16609389","raw_amount":{"local_denom":"11200982"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1221"},{"claim_amount":"31302310","raw_amount":{"local_denom":"21109543"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1222"},{"claim_amount":"2986304432","raw_amount":{"local_denom":"2013893562"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1223"},{"claim_amount":"57494041","raw_amount":{"local_denom":"38772631"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1224"},{"claim_amount":"70909317","raw_amount":{"local_denom":"47819578"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1225"}, - {"claim_amount":"63882268","raw_amount":{"local_denom":"43080701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1226"},{"claim_amount":"830469","raw_amount":{"local_denom":"560049"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1227"},{"claim_amount":"9582339","raw_amount":{"local_denom":"6462105"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1228"},{"claim_amount":"6324343","raw_amount":{"local_denom":"4264989"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1229"},{"claim_amount":"213366777","raw_amount":{"local_denom":"143889543"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1230"},{"claim_amount":"31941133","raw_amount":{"local_denom":"21540350"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1231"},{"claim_amount":"12137630","raw_amount":{"local_denom":"8185333"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1232"},{"claim_amount":"14054098","raw_amount":{"local_denom":"9477754"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1233"},{"claim_amount":"37371126","raw_amount":{"local_denom":"25202210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1234"},{"claim_amount":"111155146","raw_amount":{"local_denom":"74960420"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1235"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1236"},{"claim_amount":"18525857","raw_amount":{"local_denom":"12493403"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1237"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1238"},{"claim_amount":"6388226","raw_amount":{"local_denom":"4308070"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1239"},{"claim_amount":"16609389","raw_amount":{"local_denom":"11200982"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1240"},{"claim_amount":"6324343","raw_amount":{"local_denom":"4264989"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1241"},{"claim_amount":"32899368","raw_amount":{"local_denom":"22186561"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1242"},{"claim_amount":"17887034","raw_amount":{"local_denom":"12062596"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1243"}, - {"claim_amount":"1852585","raw_amount":{"local_denom":"1249340"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1244"},{"claim_amount":"60688154","raw_amount":{"local_denom":"40926666"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1245"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1246"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1247"},{"claim_amount":"31302310","raw_amount":{"local_denom":"21109543"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1248"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1249"},{"claim_amount":"1597055","raw_amount":{"local_denom":"1077017"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1250"},{"claim_amount":"958233","raw_amount":{"local_denom":"646210"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1251"},{"claim_amount":"93906934","raw_amount":{"local_denom":"63328631"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1252"},{"claim_amount":"1213762","raw_amount":{"local_denom":"818533"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1253"},{"claim_amount":"555775739","raw_amount":{"local_denom":"374802104"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1254"},{"claim_amount":"47911701","raw_amount":{"local_denom":"32310526"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1255"},{"claim_amount":"18845267","raw_amount":{"local_denom":"12708806"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1256"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1257"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1258"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1259"},{"claim_amount":"204103848","raw_amount":{"local_denom":"137642841"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1260"},{"claim_amount":"20059032","raw_amount":{"local_denom":"13527340"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1261"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1262"},{"claim_amount":"63882268","raw_amount":{"local_denom":"43080701"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1263"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1264"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1265"},{"claim_amount":"638822","raw_amount":{"local_denom":"430807"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1266"},{"claim_amount":"14054098","raw_amount":{"local_denom":"9477754"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1267"},{"claim_amount":"753171951","raw_amount":{"local_denom":"507921472"},"owner":"quasar18a2u6az6dzw528rptepfg6n49ak6hdzkf8ewf0n5r0nwju7gtdgqamr7qu","bond_id":"1268"} -] \ No newline at end of file diff --git a/smart-contracts/osmosis/contracts/lp-strategy/proptest-regressions/icq.txt b/smart-contracts/osmosis/contracts/lp-strategy/proptest-regressions/icq.txt deleted file mode 100644 index 59fc9c668..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/proptest-regressions/icq.txt +++ /dev/null @@ -1,8 +0,0 @@ -# Seeds for failure cases proptest has generated in the past. It is -# automatically read and these particular cases re-run before any -# novel cases are generated. -# -# It is recommended to check this file in to source control so that -# everyone who runs the test benefits from these saved cases. -cc 2b300d09f5f1856b156f41658fdd276b8a378e67da0908c4c158b8d7730d81b9 # shrinks to ica_balance = 1, base_amount = 1, quote_amount = 327625762938489495774, spot_price = 1 -cc 6448d8264198d301ef2dbab1d66ec3e06a1e912fa9bcd78bcd518cc776a91c06 # shrinks to ica_balance = 58306639683943076671774772009787831350, base_amount = 281975727236995386788616564490221065989, quote_amount = 193186435338440089613902032006337132281, spot_price = 1 diff --git a/smart-contracts/osmosis/contracts/lp-strategy/proptest-regressions/proptests.txt b/smart-contracts/osmosis/contracts/lp-strategy/proptest-regressions/proptests.txt deleted file mode 100644 index f97783546..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/proptest-regressions/proptests.txt +++ /dev/null @@ -1,10 +0,0 @@ -# Seeds for failure cases proptest has generated in the past. It is -# automatically read and these particular cases re-run before any -# novel cases are generated. -# -# It is recommended to check this file in to source control so that -# everyone who runs the test benefits from these saved cases. -cc 18cce7ef7418ba2e32c0dc974cce3839ed3afdd23c44eeff10c7cd502b664e52 # shrinks to (claim_amount, raw_amount, owner, bond_id) = ([0, 0, 0, 0, 0, 0, 0, 0, 11673845048084930547460394097995684, 38681151403059791997234025049684207626, 275054852874891480362198002940302056891, 248004106615233309637537480600637151515, 12165567352195846609387282639635621409, 260605945814629129877776296832005137162, 332498733738228587954824791897661214933, 220466759761543005181347682445321394851, 276955038478825658651724087112112977189, 327002931333860120384022494611419106764, 147064090442518249001596779703329850197, 39564122707571826747255339378284677948, 17245431880610274106465508916866826111, 26339003119474933218657724991629946399, 82276117695411068556044142010840638542, 295008677415348915376590468404000416830, 151554982649645368787246484795119526682, 180975261261844223519603233989024384027, 168751969190378588139721758050330892898, 131638036830868649013316482547580628713, 220442041130371752953825283455113893493, 65440179817723704813041205675233168717, 302341085874258666373729283438827979429, 59484899708101852479526669734567026566, 167625816230803446734600484639624676839, 223247288265999841840523557165498328548, 271205220985832846755417320083279796734, 188428825985446459356920011698020643141, 55914548276615285259550848482812446934, 102195989075020376935386609445353607918, 121781350805295556121749379931595727418, 49381516483748229739304062049605408686, 329483797478926557991409582759842888536, 25050877269533561643188662513251020289, 184084971816401394199792095345498654512, 314553285012672846226294513049818847532, 118214012495816784551227736304553654376, 331325951165181232595902138691158403353, 229895801368881505703484862144444531032, 217114635620378362608851676834552215349, 159876936511935302157126642045367287856, 135181955021589654255626347995453150573], [227365969172255363624507819951096565724, 301233924106392053527349799565535715008, 215189084033479621725931318636708931919, 269766051904425953113532064070452847870, 61626890029710227563630958944174916236, 121643162891773300986476522338839386555, 157890849575919296860121842741320755656, 318428887894696259998226374343390699537, 150429733389995246829353939053231673536, 1309984178131570546375402682674361514, 288596204117498775793769566300644324747, 221239221463166352388729594400491553656, 148978909380753273521137273830591697007, 275538702349313158916849577919121729747, 163713187093681229043970290243259360890, 151924563830510961661700273092653868361, 148155351515033332222116696976228855043, 23040063650194206500661391710955447418, 13976091836981510172184743689715944696, 185666796634122241421163204703768333325, 31949197993783774617668624979640684182, 44236163852815662641567703574148698704, 47478877286618988935887578375249136659, 70426128900427005954075445206391113470, 140906409014291614293719794814848067102, 107269575146883060322539137018471008849, 296995157204714311405805419068824989229, 161750435276928235606415971917746751228, 130756986618466217781348998859419151606, 105073686713080197476347255702114582082, 129373854793859197217362919487938017347, 285793942650403489640653646346783262704, 176448516216785083528112658599394780511, 200568359946301769234774760267636045793, 27117563929292709236084290248168245624, 132398191031838449046483348706831060869, 57516945481821405412092628600363321519, 197427381802548279413420705792351222501, 222846337535771089032432325957371072165, 79516707543765823940780266761219013517, 332746313329984099982163571762137691017, 297346398595796224323092462643166162137, 337071417935433974216532431467712569575, 41085095792457838701721359791557450918, 702470706316076453758134797401550118, 100190323026370930887276274578630211218, 54132925651950671821377657598243746105, 160055808294617960457289618967505797919, 112719553546396289399469348203275740183, 242448663799866037822921626508547285315], ["pelxgcgce", "eypufagrnmxaahmtwbz", "cksckqcgu", "johwpajdhxxdilpdqrskizhddv", "ecdyivkeowdhfpw", "ueuajshhjamecwhwunb", "ytwfxzlqv", "hbbmwqnofuhemyeziytmbdf", "ygqznvztityjg", "ntflh", "mivqrepqpwb", "gdvidjpuxwlrrqbr", "xjarmvwsgdwvwbblsooirfpcc", "zepkhzdkhvuakrhxsevrtdmbethihul", "ifrbn", "mersihvlcbvl", "ofyoghkarnxgpkorimi", "drdnbwlazpjrnmjpsksn", "mlppcbqtffsanomjokhth", "rxvlxdmxxlhckfklgd", "abf", "xltocqunoj", "ivsethz", "nzxsbgopozgqdxdmjmahmlglzwppoqg", "tkfsavgckbbzenoy", "sszuzdirsyfcwqogzhcnoaku", "rqnynflkonzpf", "kxlbd", "sylpzasamyg", "arbxqxavnptpewgb", "zovsuusznqywcnmqjppgzhwlcl", "pvjqbwupe", "xlnxdjtvxpd", "", "xbdygipeqfbazl", "j", "mdxudvywbvbfxkhnq", "ijrbpwsmvhzksyztnbqctyfqoiwkcy", "dqthiwptekttiplbhttgiuj", "twqkimydeuvacaelnklsxsbavmobuiqk", "qixstkpwyugrfwxsgohjpvtaa", "rvsiw", "ghwgcnggijoyapcaadxjk", "vkjv", "icpqzohhbpiywrdonllmzadjmffin", "ridgzcfb", "xbeunufvpqzytneqr", "cgoazeqiogbsv", "pprfeonetfysfbylr", "w"], [245156543980833747090242765144329311038, 194567896024102730141520967076722737253, 78321590831082444086273322095061857666, 46234274478596011116062623468273869668, 263465158450098859717724367516067905975, 100752541069266103352244815706779144613, 246329798443977159630785364311541203165, 89071066193382807867647247777458784703, 89680324833787488426590379191240982648, 326751368326903363754619742229709084471, 81949509939543357493869595004811069397, 320435057684033242953952509794605645127, 133212895719388354112667050350502201843, 262256541475020683812327300478510544022, 68423813730476634270402958635073934367, 156738226689669755944851331748791005616, 165800725532545879609977936210035055040, 172018455739485735148072644379216290447, 230951541362177602359868002660805912181, 152287027695533706215823402052033189960, 185057933550094735087283806786823061589, 122688000701961209416344853008164886381, 134958802702793640081845590040343954920, 234580960409424622564273814575333116288, 183739892098724792237695657764961499807, 39015994699012683337617007960068058843, 324137172833870104374658339409853260398, 156871594659100780158350555737480367369, 76549132260116470292478935610594931831, 206367556367473756044436720233354369995, 242121355820534272849028361122887112128, 188631998326034004600861498586852938791, 216069174770114465624174207718872444354, 307944890529036529511929496919799050224, 316346783727245703493409702940347462603, 232794123719215275741536507199048918940, 230699271415462493000400243339292494907, 215746109799468672448745109724428767505, 10735964370838476891653111627115651590, 177546680977013945994944742461442930075, 48188853863336799882618830643360361251, 123072710083781474823580256603344815581, 53696605966638634884824007767847163895, 244813750508221881267886759962353770844, 138389416756738828485107907845869178387, 28787808414027521881066099129863372506, 208960876676490369283588392054850944689, 291922470923877549823217120675909351267, 115223001020919711020256788413332787629, 309151305555635829791059270121082705015]) -cc ed5adefd853fbe7386e63b9d852d81fcfb66fd30eb1ee0ac95f4915dc32af26d # shrinks to (claim_amount, raw_amount, owner, bond_id) = ([0, 0], [5567553736818050811761864155695314435, 334714813184120412651612743276072882021], ["", ""], [0, 0]) -cc bb29a873ab429224989abbadbb61fd8492c0c147fcbfdd7c03eb64d853c254db # shrinks to (claim_amount, raw_amount, owner, bond_id) = ([], [], [], []), (amount_pd, owner_pd, bond_id_pd) = ([], [], []), raw_balalance_rq = 0, quote_balance_rq = 0, lp_balance_rq = 0, join_pool_rq = 0, exit_pool_base_rq = 0, exit_pool_quote_rq = 0, spot_price_rq = 0 -cc 34775cc5f6b44bd958d663493908f0e5546434ed9156d1ebe18a1503b4166df0 # shrinks to (claim_amount, raw_amount, owner, bond_id) = ([1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 83553156567498, 11808849592816811342, 8130787376546610459, 14980519244902503230, 14234410997718520343, 10225549382265082313, 14258484066235613374, 16132509078925444533, 12496179773926113447, 5163701959177179557, 6115732151642180644, 16014557328226574517, 17910735935150482215, 15905882731006320604, 6938858134545344839, 311364060211843560, 8464979079766962920, 8688102988532725869, 6950140404087665517, 61793720827537470, 15697647606715928957, 6549806696217709345, 6362990510017777241, 10957519013857898322, 12513697704065533392, 10682377085392477145, 11467648984684906402, 6914635055688605728, 6355149797117565939, 3759358825474970803, 1512101291444601349, 12542255319701031943, 7005223357316672524, 11709985910294196329, 16431675148878993558, 18088527547444258371, 18128405577901023348, 533460672940784407, 6547510708312023640, 12866793111322861531, 8672995590736397231, 14424622615502074446, 2307613407971382985, 17441942398532076118], [15826457340195732357, 3704826055518850654, 7534652765877672533, 17446234302595466983, 6161974760894201947, 14625587653503619871, 6368026972723040756, 9429109580945985830, 2272938497016214323, 3904354027872436016, 11765354634249073970, 5206622489089162589, 6352158132024455001, 10360948853066292607, 16091076371184297087, 10686302784989918058, 522004517581656444, 4357444334841157825, 8752957838480279932, 1966713813215548561, 16954393754388505823, 984025542946232675, 13665913500959464705, 4955554490241156322, 7294307173539117479, 938763324680373732, 492520615853963630, 14849790482556102869, 13517356408982302285, 12639950288329634165, 13393035703502329928, 16572484674960023979, 6573211058722160462, 14412302373624951367, 6188123385003315881, 11715869108671089008, 5648724300247408734, 5413705664772579881, 5998831249611394040, 1220625596522501898, 1377604803059093507, 141220414108167420, 120942024884388456, 15465952658707590747, 14335606127926440686, 16175674822280915491, 8695491072012231849, 1853068128626976492, 11227189535675615227, 4960793297193568795, 7850650787850785584, 2911908877680041078, 9878681945393343004, 7725132537583460349, 5630970625065211239, 15588629860098892381, 9561298833016379036, 17094307055106940528, 6013650033902031667, 8259839850861701315], ["dgrqiyjcovaookyblrg", "ztmagivlipvydahetyhpdwayykzjj", "tfdbxmfojhsbmzgumapeoz", "gxezjehqzrkzutmghdlqaxfduyvdtmmd", "fhsdpym", "wudbifslwjvruodwlokezlrkouqarvt", "zpnihv", "e", "vkyycuyhdzzdeemvpigsb", "plvetqyxmjjhsgtuysxqqymj", "inwzsdwpbdoerg", "zfwpkflklyt", "mnbfxebcyiailqsrfqowr", "mupbksylxtjtgqalnitlqjitlnzks", "qpjyoai", "bzskbtibcesvxnfgbvuzhzgtycy", "opjjotortcawlyttck", "mgftscgljwawrffmgsffsxleujngatbi", "atofntcuqdxmzqyaykhwyuwzy", "bkehgtfwnxphyxfaryescclprn", "moaianym", "hufolvhlfdjbeogauquuv", "dpkoobsneuuktmrxabwhk", "fxkeasrmkldatftoxaxtrow", "ynqjoxyleuoo", "dmjobjvtdxvfaiiicbjjpmhrke", "rpqahmnaflpqteyeyzfbhuab", "qgxytkazzaowoahejafngnnmoyuwq", "mctubnxzqodnwqxlhfhcrmtnoqaav", "nhjqjrktvc", "ysyqcjioyrfhutkaueaoocfxnctqazeq", "clxcviqulqfkm", "rtnwqgagvt", "jmqaltldcjcvfmdj", "nnlnpocqf", "pptjwiqwjbxfetnztectxrainddacm", "onxfua", "jngvifuoigb", "scmknna", "splnseom", "onqlrofyfmycjjdz", "fifjawebeqjtnbvoeioyimxvjjl", "ovqeftnbpqwp", "sk", "zidmgenlzppemiofhewmlsefjzdo", "rweenuisuhqk", "hop", "hpnyurozekfedbxhzidxbfihcypeljuf", "x", "ttpwuzzlnw", "defee", "pszdtgzfkzdxldlcrbr", "hqqnyylxmm", "ytwhdpgbqqluofeimednsepsvkxhaae", "uwlzmmgbadlymz", "ilpfthwgblxrlcwt", "trxqrxeii", "nklgitkbqdgbuiykweimoxxbus", "wsnxygctcnomzgjywfvtu", "bfcglnnizuhprogakd"], [17564701645947559694, 10684717952737285943, 10219761788279915146, 2109710357599859391, 10487310863879864593, 15822884041780719907, 11848105584455790133, 17931449780282279864, 8909590184269171323, 2959034442329770038, 16596193355103071593, 15485371830524286297, 15155403646891693209, 16618642512816226160, 11042324674626444498, 11974407949461512535, 1763186450440707960, 18290373659633014915, 15628661186771988252, 16836233893596365339, 16156643523011845388, 6812953976141217730, 3685585248911984394, 810033298279336584, 9998175313336111059, 8687288352342553603, 3098556134784640099, 3864774701428716149, 15051486419399376922, 2190077543700623599, 16012320347463077713, 16833307165034632153, 2395940186996495483, 4130047742181157138, 398807872624232724, 5593681726988201550, 17694081960095301824, 15985889760645175226, 16376128475035724969, 4026056742303909744, 15014143413475248664, 3995771739911710916, 1941983340358743336, 13206881376468297167, 14576866369417099042, 11358921675486065870, 2026942581626530092, 3182543356739401115, 13832536031591260330, 10929954346159679353, 17226923001376597991, 14359302255407025404, 15604591261117489966, 4652729393292511177, 14178413536313939160, 8962255765512619945, 11954228840537278358, 7081160827464255778, 12740476215904686133, 901276637696619765]), (amount_pd, owner_pd, bond_id_pd) = ([], [], []), raw_balalance_rq = 3906736672086151414, quote_balance_rq = 5260586167652135091, lp_balance_rq = 47453939318407458, join_pool_rq = 17437550474192041343, exit_pool_base_rq = 12672570040552928040, exit_pool_quote_rq = 261663182347622709, spot_price_rq = 10002155156807470437 diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/admin.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/admin.rs deleted file mode 100644 index 23b5972cd..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/admin.rs +++ /dev/null @@ -1,158 +0,0 @@ -use cosmwasm_std::{Addr, Empty, Env, QuerierWrapper, Storage}; - -use crate::{ - error::ContractError, - helpers::is_contract_admin, - state::{DEPOSITOR, LOCK_ADMIN}, -}; - -// check if sender is the admin -pub fn check_depositor(storage: &mut dyn Storage, sender: &Addr) -> Result { - let depositor = DEPOSITOR.load(storage)?; - Ok(depositor == sender) -} - -pub fn add_lock_admin( - storage: &mut dyn Storage, - querier: &QuerierWrapper, - env: &Env, - sender: Addr, - to_add: Addr, -) -> Result<(), ContractError> { - is_contract_admin(querier, env, &sender)?; - LOCK_ADMIN.save(storage, &to_add, &Empty::default())?; - Ok(()) -} - -pub fn remove_lock_admin( - storage: &mut dyn Storage, - querier: &QuerierWrapper, - env: &Env, - sender: Addr, - to_remove: Addr, -) -> Result<(), ContractError> { - is_contract_admin(querier, env, &sender)?; - LOCK_ADMIN.remove(storage, &to_remove); - Ok(()) -} - -pub fn is_lock_admin( - storage: &mut dyn Storage, - querier: &QuerierWrapper, - env: &Env, - sender: &Addr, -) -> Result<(), ContractError> { - // if the may load is none, we check the contract admin status, if that errors, then the sender is neither contract - // admin or a lock admin - let lock = LOCK_ADMIN.may_load(storage, sender)?; - match lock { - Some(_) => Ok(()), - None => is_contract_admin(querier, env, sender), - } -} - -#[cfg(test)] -mod tests { - use super::*; - use cosmwasm_std::{ - testing::{mock_dependencies, mock_env, MockQuerier}, - to_json_binary, ContractInfoResponse, ContractResult, QuerierResult, WasmQuery, - }; - - #[test] - fn test_lock_admin() { - let mut deps = mock_dependencies(); - let admin = "bob"; - let admin1 = "alice"; - let admin2: &str = "eve"; - let env = mock_env(); - - let mut info = ContractInfoResponse::default(); - info.admin = Some(admin.to_string()); - let mut q = MockQuerier::default(); - q.update_wasm(move |q: &WasmQuery| -> QuerierResult { - match q { - WasmQuery::ContractInfo { contract_addr: _ } => { - QuerierResult::Ok(ContractResult::Ok(to_json_binary(&info).unwrap())) - } - _ => unreachable!(), - } - }); - let querier: QuerierWrapper = QuerierWrapper::new(&q); - - // add the extra admin - add_lock_admin( - deps.as_mut().storage, - &querier, - &env, - Addr::unchecked(admin), - Addr::unchecked(admin1), - ) - .unwrap(); - is_lock_admin( - deps.as_mut().storage, - &querier, - &env, - &Addr::unchecked(admin), - ) - .unwrap(); - is_lock_admin( - deps.as_mut().storage, - &querier, - &env, - &Addr::unchecked(admin1), - ) - .unwrap(); - is_lock_admin( - deps.as_mut().storage, - &querier, - &env, - &Addr::unchecked(admin2), - ) - .unwrap_err(); - add_lock_admin( - deps.as_mut().storage, - &querier, - &env, - Addr::unchecked(admin), - Addr::unchecked(admin2), - ) - .unwrap(); - is_lock_admin( - deps.as_mut().storage, - &querier, - &env, - &Addr::unchecked(admin2), - ) - .unwrap(); - - remove_lock_admin( - deps.as_mut().storage, - &querier, - &env, - Addr::unchecked(admin), - Addr::unchecked(admin2), - ) - .unwrap(); - // check that we fail once removed - is_lock_admin( - deps.as_mut().storage, - &querier, - &env, - &Addr::unchecked(admin2), - ) - .unwrap_err(); - } - - #[test] - fn test_admin() { - let mut deps = mock_dependencies(); - let sender1 = Addr::unchecked("alice"); - let sender2 = Addr::unchecked("eve"); - - DEPOSITOR.save(deps.as_mut().storage, &sender1).unwrap(); - assert!(check_depositor(deps.as_mut().storage, &sender1).unwrap()); - assert_eq!(check_depositor(deps.as_mut().storage, &sender1), Ok(true)); - assert_eq!(check_depositor(deps.as_mut().storage, &sender2), Ok(false)); - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/bond.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/bond.rs deleted file mode 100644 index a63a73c68..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/bond.rs +++ /dev/null @@ -1,554 +0,0 @@ -use std::cmp::Ordering; - -use cosmwasm_std::{Addr, Env, QuerierWrapper, StdResult, Storage, SubMsg, Uint128}; -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -use crate::{ - error::ContractError, - helpers::{get_ica_address, get_total_primitive_shares}, - ibc_util::do_transfer, - icq::try_icq, - state::{ - OngoingDeposit, RawAmount, BONDING_CLAIMS, BOND_QUEUE, CONFIG, FAILED_JOIN_QUEUE, - ICA_CHANNEL, PENDING_BOND_QUEUE, REJOIN_QUEUE, SHARES, - }, -}; - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub struct Bond { - pub amount: Uint128, - pub owner: Addr, - pub bond_id: String, -} - -// A deposit starts of by querying the state of the ica counterparty contract -pub fn do_bond( - storage: &mut dyn Storage, - querier: QuerierWrapper, - env: Env, - amount: Uint128, - sender: Addr, - bond_id: String, -) -> Result, ContractError> { - // check that our bond-id is not a duplicate from a pending bond - let pending: StdResult> = PENDING_BOND_QUEUE.iter(storage)?.collect(); - if pending? - .iter() - .any(|bond| bond.owner == sender && bond_id == bond.bond_id) - { - return Err(ContractError::DuplicateKey); - } - - // check that our bond-id is not a duplicate in the actual bond queue - let pending: StdResult> = BOND_QUEUE.iter(storage)?.collect(); - if pending? - .iter() - .any(|bond| bond.owner == sender && bond_id == bond.bond_id) - { - return Err(ContractError::DuplicateKey); - } - - PENDING_BOND_QUEUE.push_back( - storage, - &Bond { - amount, - owner: sender, - bond_id, - }, - )?; - - // TODO: move this to the execute_bond function - try_icq(storage, querier, env) -} - -// after the balance query, we can calculate the amount of the claim we need to create, we update the claims and transfer the funds -pub fn batch_bond( - storage: &mut dyn Storage, - env: &Env, - total_vault_value: Uint128, -) -> Result, ContractError> { - let transfer_chan = CONFIG.load(storage)?.transfer_channel; - let to_address = get_ica_address(storage, ICA_CHANNEL.load(storage)?)?; - - if let Some((amount, deposits)) = fold_bonds(storage, total_vault_value)? { - if amount.is_zero() { - Ok(None) - } else { - Ok(Some(do_transfer( - storage, - env, - amount, - transfer_chan, - to_address, - deposits, - )?)) - } - } else { - Ok(None) - } -} - -/// fold_bonds folds the queue and attributes shares to the depositors according to the given total value -pub fn fold_bonds( - storage: &mut dyn Storage, - total_balance: Uint128, -) -> Result)>, ContractError> { - let mut total = Uint128::zero(); - let mut deposits: Vec = vec![]; - - if BOND_QUEUE.is_empty(storage)? && FAILED_JOIN_QUEUE.is_empty(storage)? { - return Ok(None); - } - - while !BOND_QUEUE.is_empty(storage)? { - let item: Bond = - BOND_QUEUE - .pop_front(storage)? - .ok_or(ContractError::QueueItemNotFound { - queue: "bond".to_string(), - })?; - let claim_amount = create_claim( - storage, - item.amount, - &item.owner, - &item.bond_id, - total_balance, - )?; - total = total - .checked_add(item.amount) - .map_err(|err| ContractError::TracedOverflowError(err, "fold_bonds".to_string()))?; - deposits.push(OngoingDeposit { - claim_amount, - owner: item.owner, - raw_amount: RawAmount::LocalDenom(item.amount), - bond_id: item.bond_id, - }); - } - - while !FAILED_JOIN_QUEUE.is_empty(storage)? { - let item: Bond = - FAILED_JOIN_QUEUE - .pop_front(storage)? - .ok_or(ContractError::QueueItemNotFound { - queue: "failed_join".to_string(), - })?; - let claim_amount = create_claim( - storage, - item.amount, - &item.owner, - &item.bond_id, - total_balance, - )?; - - REJOIN_QUEUE.push_back( - storage, - &OngoingDeposit { - claim_amount, - owner: item.owner, - raw_amount: RawAmount::LocalDenom(item.amount), - bond_id: item.bond_id, - }, - )?; - } - - Ok(Some((total, deposits))) -} - -// create_claim -pub fn create_claim( - storage: &mut dyn Storage, - user_balance: Uint128, - address: &Addr, - bond_id: &str, - total_balance: Uint128, -) -> Result { - let total_shares = get_total_primitive_shares(storage)?; - - // calculate the correct size of the claim - let claim_amount = calculate_claim(user_balance, total_balance, total_shares)?; - BONDING_CLAIMS.save(storage, (address, bond_id), &claim_amount)?; - Ok(claim_amount) -} - -// create a share and remove the amount from the claim -pub fn create_share( - storage: &mut dyn Storage, - owner: &Addr, - bond_id: &str, - amount: Uint128, -) -> Result { - let claim = BONDING_CLAIMS.load(storage, (owner, bond_id))?; - - match claim.cmp(&amount) { - Ordering::Less => return Err(ContractError::InsufficientClaims), - Ordering::Equal => BONDING_CLAIMS.remove(storage, (owner, bond_id)), - Ordering::Greater => BONDING_CLAIMS.save( - storage, - (owner, bond_id), - &claim.checked_sub(amount).map_err(|err| { - ContractError::TracedOverflowError(err, "create_share".to_string()) - })?, - )?, - } - - // TODO do we want to make shares fungible using cw20? if so, call into the minter and mint shares for the according to the claim - SHARES.update( - storage, - owner.clone(), - |old| -> Result { - if let Some(existing) = old { - Ok(existing.checked_add(amount).map_err(|err| { - ContractError::TracedOverflowError( - err, - "create_share_update_shares".to_string(), - ) - })?) - } else { - Ok(amount) - } - }, - )?; - Ok(claim) -} - -/// calculate the amount of for the claim of the user -/// user_shares = (user_balance / vault_balance) * vault_total_shares = (user_balance * vault_total_shares) / vault_balance -/// if the total_shares are zero, independant of the total_balance, the user shoudl get user_balance amount of shares -/// if the total_balance is zero, what do we do?, for now the same as if total_shares is zero -/// ```rust -/// # use cosmwasm_std::Uint128; -/// # use lp_strategy::bond::calculate_claim; -/// -/// # fn calculate_claim_works() { -/// let val = calculate_claim(Uint128::new(10), Uint128::new(100), Uint128::new(10)).unwrap(); -/// assert_eq!(val, Uint128::one()) -/// # } -/// ``` -pub fn calculate_claim( - user_balance: Uint128, - total_balance: Uint128, - total_shares: Uint128, -) -> Result { - if total_shares == Uint128::zero() || total_balance == Uint128::zero() { - Ok(user_balance) - } else { - Ok(user_balance.checked_multiply_ratio(total_shares, total_balance)?) - } -} - -#[cfg(test)] -mod tests { - use cosmwasm_std::{ - testing::{mock_dependencies, mock_env, MockQuerier}, - to_json_binary, CosmosMsg, Empty, IbcMsg, IbcTimeout, - }; - - use crate::{ - ibc_lock::Lock, - icq::prepare_full_query, - state::{LpCache, IBC_LOCK, ICQ_CHANNEL, LP_SHARES}, - test_helpers::default_setup, - }; - - use super::*; - - #[test] - fn calculate_claim_works() { - let val = calculate_claim(Uint128::new(10), Uint128::new(100), Uint128::new(10)).unwrap(); - assert_eq!(val, Uint128::one()) - } - - #[test] - fn do_bond_locked_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let env = mock_env(); - let _config = CONFIG.load(deps.as_ref().storage).unwrap(); - let owner = Addr::unchecked("bob"); - - IBC_LOCK - .save(deps.as_mut().storage, &Lock::new().lock_bond()) - .unwrap(); - let id = "my-id"; - - let qx: MockQuerier = MockQuerier::new(&[]); - let q = QuerierWrapper::new(&qx); - - let res = do_bond( - deps.as_mut().storage, - q, - env, - Uint128::new(1000), - owner, - id.to_string(), - ) - .unwrap(); - assert_eq!(res, None) - } - - #[test] - fn do_bond_unlocked_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let env = mock_env(); - let _config = CONFIG.load(deps.as_ref().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let id = "my-id"; - - LP_SHARES - .save( - deps.as_mut().storage, - &LpCache { - locked_shares: Uint128::new(100), - w_unlocked_shares: Uint128::zero(), - d_unlocked_shares: Uint128::zero(), - }, - ) - .unwrap(); - - IBC_LOCK.save(deps.as_mut().storage, &Lock::new()).unwrap(); - - let qx: MockQuerier = MockQuerier::new(&[]); - let q = QuerierWrapper::new(&qx); - - let res = do_bond( - deps.as_mut().storage, - q, - env.clone(), - Uint128::new(1000), - owner, - id.to_string(), - ) - .unwrap(); - assert!(res.is_some()); - - // mocking the pending bonds is real ugly here - let packet = prepare_full_query( - deps.as_mut().storage, - env.clone(), - Uint128::new(1000), - Uint128::zero(), - ) - .unwrap(); - - let icq_msg = CosmosMsg::Ibc(IbcMsg::SendPacket { - channel_id: ICQ_CHANNEL.load(deps.as_mut().storage).unwrap(), - data: to_json_binary(&packet).unwrap(), - timeout: IbcTimeout::with_timestamp(env.block.time.plus_seconds(7200)), - }); - - assert_eq!(res.unwrap().msg, icq_msg) - } - - #[test] - fn do_bond_duplicate_id_fails() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let env = mock_env(); - let _config = CONFIG.load(deps.as_ref().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let id = "my-id"; - - LP_SHARES - .save( - deps.as_mut().storage, - &LpCache { - locked_shares: Uint128::new(100), - w_unlocked_shares: Uint128::zero(), - d_unlocked_shares: Uint128::zero(), - }, - ) - .unwrap(); - - IBC_LOCK.save(deps.as_mut().storage, &Lock::new()).unwrap(); - - let qx: MockQuerier = MockQuerier::new(&[]); - let q = QuerierWrapper::new(&qx); - - let res = do_bond( - deps.as_mut().storage, - q, - env.clone(), - Uint128::new(1000), - owner.clone(), - id.to_string(), - ) - .unwrap(); - assert!(res.is_some()); - - // bond with a duplicate id, we expect this to error - let err = do_bond( - deps.as_mut().storage, - q, - env, - Uint128::new(1000), - owner, - id.to_string(), - ) - .unwrap_err(); - assert_eq!(err, ContractError::DuplicateKey) - } - - #[test] - fn create_claim_works() { - let mut deps = mock_dependencies(); - let owner = Addr::unchecked("bob"); - let id = "my-id"; - let user_balance = Uint128::new(10); - let total_balance = Uint128::new(100); - SHARES - .save(deps.as_mut().storage, owner.clone(), &Uint128::new(100)) - .unwrap(); - - let claim_amount = create_claim( - deps.as_mut().storage, - user_balance, - &owner, - id, - total_balance, - ) - .unwrap(); - assert_eq!(claim_amount, Uint128::new(10)); - assert_eq!( - BONDING_CLAIMS - .load(deps.as_ref().storage, (&owner, id)) - .unwrap(), - claim_amount - ); - } - - #[test] - fn create_exact_share_works() { - let mut deps = mock_dependencies(); - let owner = Addr::unchecked("bob"); - let id = "my-id"; - let amount = Uint128::new(100); - - BONDING_CLAIMS - .save(deps.as_mut().storage, (&owner, id), &amount) - .unwrap(); - - create_share(deps.as_mut().storage, &owner, id, amount).unwrap(); - // the claim in BONDING_CLAIMS should have been deleted - assert_eq!( - BONDING_CLAIMS - .may_load(deps.as_ref().storage, (&owner, id)) - .unwrap(), - None - ); - - // we should have minted exactly 100 shares by now, - // we should have minted exactly 100 shares by now, - assert_eq!(SHARES.load(deps.as_ref().storage, owner).unwrap(), amount); - } - - #[test] - fn create_less_shares_works() { - let mut deps = mock_dependencies(); - let owner = Addr::unchecked("bob"); - let id = "my-id"; - let amount = Uint128::new(100); - let smaller_amount = Uint128::new(99); - - BONDING_CLAIMS - .save(deps.as_mut().storage, (&owner, id), &amount) - .unwrap(); - - create_share(deps.as_mut().storage, &owner, id, smaller_amount).unwrap(); - // the claim in BONDING_CLAIMS should have been deleted - assert_eq!( - BONDING_CLAIMS - .may_load(deps.as_ref().storage, (&owner, id)) - .unwrap(), - Some(Uint128::one()) - ); - // we should have amount shares by now - assert_eq!( - SHARES.load(deps.as_ref().storage, owner).unwrap(), - smaller_amount - ); - } - - #[test] - fn create_too_many_shares_fails() { - let mut deps = mock_dependencies(); - let owner = Addr::unchecked("bob"); - let id = "my-id"; - let amount = Uint128::new(100); - let incorrect_amount = Uint128::new(101); - - BONDING_CLAIMS - .save(deps.as_mut().storage, (&owner, id), &amount) - .unwrap(); - - let err = create_share(deps.as_mut().storage, &owner, id, incorrect_amount).unwrap_err(); - // we should not have created shares - assert_eq!(err, ContractError::InsufficientClaims); - // our bonding claim should still exist - assert_eq!( - BONDING_CLAIMS - .load(deps.as_ref().storage, (&owner, id)) - .unwrap(), - amount - ) - } - - #[test] - fn empty_fold_bonds_works() { - let mut deps = mock_dependencies(); - let total_balance = Uint128::new(150); - // we don't have any bonds to setup - - let res = fold_bonds(deps.as_mut().storage, total_balance).unwrap(); - assert_eq!(res, None) - } - - #[test] - fn filled_fold_bonds_works() { - let mut deps = mock_dependencies(); - let total_balance = Uint128::new(150); - let bonds = vec![ - Bond { - amount: Uint128::new(34), - owner: Addr::unchecked("person1"), - bond_id: "id1".to_string(), - }, - Bond { - amount: Uint128::new(50), - owner: Addr::unchecked("person2"), - bond_id: "id2".to_string(), - }, - Bond { - amount: Uint128::new(2), - owner: Addr::unchecked("person3"), - bond_id: "id3".to_string(), - }, - Bond { - amount: Uint128::new(7), - owner: Addr::unchecked("person4"), - bond_id: "id4".to_string(), - }, - Bond { - amount: Uint128::new(57), - owner: Addr::unchecked("person5"), - bond_id: "id5".to_string(), - }, - ]; - for b in &bonds { - BOND_QUEUE.push_back(deps.as_mut().storage, b).unwrap(); - } - - let (total, ongoing) = fold_bonds(deps.as_mut().storage, total_balance) - .unwrap() - .unwrap(); - assert_eq!(total_balance, total); - for (i, b) in bonds.iter().enumerate() { - assert_eq!(ongoing[i].bond_id, b.bond_id); - assert_eq!(ongoing[i].owner, b.owner); - - assert_eq!(ongoing[i].claim_amount, b.amount) - } - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/contract.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/contract.rs deleted file mode 100644 index 3466a5ec3..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/contract.rs +++ /dev/null @@ -1,1331 +0,0 @@ -#[cfg(not(feature = "library"))] -use cosmwasm_std::entry_point; -use cosmwasm_std::{ - from_json, Attribute, DepsMut, Env, IbcMsg, IbcPacketAckMsg, MessageInfo, QuerierWrapper, - Reply, Response, Storage, Uint128, -}; -use cw2::set_contract_version; -use cw_utils::{must_pay, nonpayable}; - -use quasar_types::ibc::IcsAck; - -use crate::admin::{add_lock_admin, check_depositor, is_lock_admin, remove_lock_admin}; -use crate::bond::do_bond; -use crate::error::{ContractError, Trap}; -use crate::execute::execute_retry; -use crate::helpers::{create_callback_submsg, is_contract_admin, lock_try_icq, SubMsgKind}; -use crate::ibc::{handle_failing_ack, handle_succesful_ack, on_packet_timeout}; -use crate::ibc_lock::{IbcLock, Lock}; -use crate::ibc_util::{do_ibc_join_pool_swap_extern_amount_in, do_transfer}; -use crate::icq::try_icq; -use crate::msg::{ExecuteMsg, InstantiateMsg, LockOnly, MigrateMsg, UnlockOnly}; -use crate::reply::{handle_ack_reply, handle_callback_reply, handle_ibc_reply}; -use crate::start_unbond::{do_start_unbond, StartUnbond}; -use crate::state::{ - Config, LpCache, OngoingDeposit, PendingBond, RawAmount, ADMIN, BOND_QUEUE, CONFIG, DEPOSITOR, - FAILED_JOIN_QUEUE, IBC_LOCK, ICA_CHANNEL, LP_SHARES, PENDING_ACK, REPLIES, RETURNING, - START_UNBOND_QUEUE, TIMED_OUT, TOTAL_VAULT_BALANCE, TRAPS, UNBOND_QUEUE, -}; -use crate::unbond::{do_unbond, finish_unbond, PendingReturningUnbonds}; - -// version info for migration info -const CONTRACT_NAME: &str = "crates.io:lp-strategy"; -const CONTRACT_VERSION: &str = env!("CARGO_PKG_VERSION"); - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn instantiate( - deps: DepsMut, - _env: Env, - info: MessageInfo, - msg: InstantiateMsg, -) -> Result { - set_contract_version(deps.storage, CONTRACT_NAME, CONTRACT_VERSION)?; - // check valid token info - msg.validate()?; - - // ADMIN here is only used to decide who can deposit - ADMIN.save(deps.storage, &info.sender)?; - - CONFIG.save( - deps.storage, - &Config { - lock_period: msg.lock_period, - pool_id: msg.pool_id, - pool_denom: msg.pool_denom, - base_denom: msg.base_denom, - local_denom: msg.local_denom, - quote_denom: msg.quote_denom, - return_source_channel: msg.return_source_channel, - transfer_channel: msg.transfer_channel, - expected_connection: msg.expected_connection, - }, - )?; - - IBC_LOCK.save(deps.storage, &Lock::new())?; - - // OSMO_LOCK.save(deps.storage, &u64::MAX)?; - - LP_SHARES.save( - deps.storage, - &LpCache { - locked_shares: Uint128::zero(), - w_unlocked_shares: Uint128::zero(), - d_unlocked_shares: Uint128::zero(), - }, - )?; - - TOTAL_VAULT_BALANCE.save(deps.storage, &Uint128::zero())?; - - TIMED_OUT.save(deps.storage, &false)?; - - Ok(Response::default()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn reply(deps: DepsMut, _env: Env, msg: Reply) -> Result { - // Save the ibc message together with the sequence number, to be handled properly later at the ack, we can pass the ibc_kind one to one - // TODO this needs and error check and error handling - let reply = REPLIES.load(deps.storage, msg.id)?; - match reply { - SubMsgKind::Ibc(pending, channel) => handle_ibc_reply(deps, msg, pending, channel), - SubMsgKind::Ack(seq, channel) => handle_ack_reply(deps, msg, seq, channel), - SubMsgKind::Callback(_callback) => handle_callback_reply(deps, msg, _callback), - } -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn execute( - deps: DepsMut, - env: Env, - info: MessageInfo, - msg: ExecuteMsg, -) -> Result { - match msg { - ExecuteMsg::Bond { id } => execute_bond(deps, env, info, id), - ExecuteMsg::StartUnbond { id, share_amount } => { - execute_start_unbond(deps, env, info, id, share_amount) - } - ExecuteMsg::Unbond { id } => execute_unbond(deps, env, info, id), - ExecuteMsg::AcceptReturningFunds { id, pending } => { - execute_accept_returning_funds(deps.storage, deps.querier, info, id, pending) - } - ExecuteMsg::CloseChannel { channel_id } => execute_close_channel(deps, channel_id), - ExecuteMsg::Ack { ack } => execute_ack(deps, env, info, ack), - ExecuteMsg::TryIcq {} => execute_try_icq(deps, env), - ExecuteMsg::SetDepositor { depositor } => execute_set_depositor(deps, info, depositor), - ExecuteMsg::Unlock { unlock_only } => execute_unlock(deps, env, info, unlock_only), - ExecuteMsg::Lock { lock_only } => execute_lock(deps, env, info, lock_only), - ExecuteMsg::ManualTimeout { - seq, - channel, - should_unlock, - } => manual_timeout(deps, env, info, seq, channel, should_unlock), - ExecuteMsg::AddLockAdmin { to_add } => execute_add_lock_admin(deps, env, info, to_add), - ExecuteMsg::RemoveLockAdmin { to_remove } => { - execute_remove_lock_admin(deps, env, info, to_remove) - } - ExecuteMsg::Retry { seq, channel } => execute_retry(deps, env, info, seq, channel), - } -} - -pub fn execute_add_lock_admin( - deps: DepsMut, - env: Env, - info: MessageInfo, - to_add: String, -) -> Result { - add_lock_admin( - deps.storage, - &deps.querier, - &env, - info.sender, - deps.api.addr_validate(to_add.as_str())?, - )?; - Ok(Response::new() - .add_attribute("action", "add_lock_admin") - .add_attribute("lock_admin", to_add)) -} - -pub fn execute_remove_lock_admin( - deps: DepsMut, - env: Env, - info: MessageInfo, - to_add: String, -) -> Result { - remove_lock_admin( - deps.storage, - &deps.querier, - &env, - info.sender, - deps.api.addr_validate(to_add.as_str())?, - )?; - Ok(Response::new() - .add_attribute("action", "remove_lock_admin") - .add_attribute("lock_admin", to_add)) -} - -pub fn execute_lock( - deps: DepsMut, - env: Env, - info: MessageInfo, - lock_only: LockOnly, -) -> Result { - is_lock_admin(deps.storage, &deps.querier, &env, &info.sender)?; - let mut lock = IBC_LOCK.load(deps.storage)?; - - match lock_only { - LockOnly::Bond => lock = lock.lock_bond(), - LockOnly::StartUnbond => lock = lock.lock_start_unbond(), - LockOnly::Unbond => lock = lock.lock_unbond(), - LockOnly::Migration => lock = lock.lock_migration(), - }; - IBC_LOCK.save(deps.storage, &lock)?; - - Ok(Response::new().add_attribute("lock_only", lock_only.to_string())) -} - -pub fn execute_unlock( - deps: DepsMut, - env: Env, - info: MessageInfo, - unlock_only: UnlockOnly, -) -> Result { - is_lock_admin(deps.storage, &deps.querier, &env, &info.sender)?; - let mut lock = IBC_LOCK.load(deps.storage)?; - - match unlock_only { - UnlockOnly::Bond => lock = lock.unlock_bond(), - UnlockOnly::StartUnbond => lock = lock.unlock_start_unbond(), - UnlockOnly::Unbond => lock = lock.unlock_unbond(), - UnlockOnly::Migration => lock = lock.unlock_migration(), - }; - IBC_LOCK.save(deps.storage, &lock)?; - - Ok(Response::new().add_attribute("unlock_only", unlock_only.to_string())) -} - -pub fn manual_timeout( - deps: DepsMut, - env: Env, - info: MessageInfo, - sequence: u64, - channel: String, - should_unlock: bool, -) -> Result { - is_contract_admin(&deps.querier, &env, &info.sender)?; - - let response = on_packet_timeout( - deps, - sequence, - channel, - "timeout".to_string(), - should_unlock, - )?; - - Ok(Response::new() - .add_attributes(response.attributes) - .add_events(response.events) - .add_submessages(response.messages)) -} - -pub fn execute_set_depositor( - deps: DepsMut, - info: MessageInfo, - depositor: String, -) -> Result { - if info.sender == ADMIN.load(deps.storage)? { - let depositor_addr = deps.api.addr_validate(depositor.as_str())?; - DEPOSITOR.save(deps.storage, &depositor_addr)?; - Ok(Response::new().add_attribute("set-depositor", depositor)) - } else { - Err(ContractError::Unauthorized) - } -} - -pub fn execute_try_icq(deps: DepsMut, env: Env) -> Result { - // if we're unlocked, we can empty the queues and send the submessages - let mut lock = IBC_LOCK.load(deps.storage)?; - let sub_msg = try_icq(deps.storage, deps.querier, env)?; - let mut res = Response::new(); - - if let Some(sub_msg) = sub_msg { - if !BOND_QUEUE.is_empty(deps.storage)? { - lock.bond = IbcLock::Locked; - res = res.add_attribute("bond_queue", "locked"); - } else if !START_UNBOND_QUEUE.is_empty(deps.storage)? { - lock = lock.lock_start_unbond(); - res = res.add_attribute("start_unbond_queue", "locked"); - } else if !UNBOND_QUEUE.is_empty(deps.storage)? { - lock = lock.lock_unbond(); - res = res.add_attribute("unbond_queue", "locked"); - } - if lock.is_unlocked() { - res = res.add_attribute("IBC_LOCK", "unlocked"); - } - IBC_LOCK.save(deps.storage, &lock)?; - res = res.add_submessage(sub_msg); - } else { - res = res.add_attribute("IBC_LOCK", "locked"); - } - Ok(res) -} - -pub fn execute_ack( - deps: DepsMut, - env: Env, - info: MessageInfo, - msg: IbcPacketAckMsg, -) -> Result { - if env.contract.address != info.sender { - return Err(ContractError::Unauthorized); - } - - // TODO: trap error like in receive? - // pro's acks happen anyway, cons? - let ack: IcsAck = from_json(&msg.acknowledgement.data)?; - match ack { - IcsAck::Result(val) => handle_succesful_ack(deps, env, msg, val), - IcsAck::Error(err) => handle_failing_ack(deps, env, msg, err), - } -} - -pub fn execute_accept_returning_funds( - storage: &mut dyn Storage, - querier: QuerierWrapper, - info: MessageInfo, - id: u64, - pending: PendingReturningUnbonds, -) -> Result { - let returning_amount = RETURNING - .may_load(storage, id)? - .ok_or(ContractError::ReturningTransferNotFound)?; - - let amount = must_pay(&info, CONFIG.load(storage)?.local_denom.as_str())?; - if amount != returning_amount { - return Err(ContractError::ReturningTransferIncorrectAmount); - } - - let mut callback_submsgs = vec![]; - for unbond in pending.unbonds.iter() { - let cosmos_msg = finish_unbond(storage, querier, unbond)?; - callback_submsgs.push(create_callback_submsg( - storage, - cosmos_msg, - unbond.owner.clone(), - unbond.id.clone(), - )?) - } - - Ok(Response::new() - .add_attribute("callback-submsgs", callback_submsgs.len().to_string()) - .add_attribute("returning-transfer", id.to_string()) - .add_attribute("success", "true") - .add_submessages(callback_submsgs)) -} - -pub fn execute_bond( - deps: DepsMut, - env: Env, - info: MessageInfo, - id: String, -) -> Result { - if !check_depositor(deps.storage, &info.sender)? { - return Err(ContractError::Unauthorized); - } - - let amount = must_pay(&info, &CONFIG.load(deps.storage)?.local_denom)?; - - let msg = do_bond( - deps.storage, - deps.querier, - env, - amount, - info.sender.clone(), - id, - )?; - - // if msg is some, we are dispatching an icq - let attributes = vec![ - Attribute::new("action", "bond"), - Attribute::new("sender", info.sender), - Attribute::new("token_amount", amount), - ]; - - let res = lock_try_icq(deps, msg)?; - Ok(res.add_attributes(attributes)) -} - -pub fn execute_start_unbond( - deps: DepsMut, - env: Env, - info: MessageInfo, - id: String, - share_amount: Uint128, -) -> Result { - nonpayable(&info)?; - - if !check_depositor(deps.storage, &info.sender)? { - return Err(ContractError::Unauthorized); - } - - do_start_unbond( - deps.storage, - StartUnbond { - owner: info.sender.clone(), - id: id.clone(), - primitive_shares: share_amount, - }, - )?; - - let msg = try_icq(deps.storage, deps.querier, env)?; - - let attributes = vec![ - Attribute::new("action", "start-unbond"), - Attribute::new("sender", info.sender), - Attribute::new("prim_share_amount", share_amount), - Attribute::new("unbond_id", id), - ]; - - let res = lock_try_icq(deps, msg)?; - Ok(res.add_attributes(attributes)) -} - -pub fn execute_unbond( - deps: DepsMut, - env: Env, - info: MessageInfo, - id: String, -) -> Result { - nonpayable(&info)?; - - if !check_depositor(deps.storage, &info.sender)? { - return Err(ContractError::Unauthorized); - } - - do_unbond(deps.storage, &env, info.sender.clone(), id.clone())?; - - let msg = try_icq(deps.storage, deps.querier, env)?; - - let attributes = vec![ - Attribute::new("action", "unbond"), - Attribute::new("sender", info.sender), - Attribute::new("unbond_id", id), - ]; - - let res = lock_try_icq(deps, msg)?; - Ok(res.add_attributes(attributes)) -} - -// transfer funds sent to the contract to an address on osmosis, this call ignores the lock system -pub fn execute_transfer( - deps: DepsMut, - env: Env, - info: MessageInfo, - channel: String, - to_address: String, -) -> Result { - let amount = must_pay(&info, &CONFIG.load(deps.storage)?.local_denom)?; - - let transfer = do_transfer( - deps.storage, - &env, - amount, - channel.clone(), - to_address.clone(), - // add a dummy ongoing deposit, actual ongoing deposit should calculate the claim using the total balance - vec![OngoingDeposit { - claim_amount: amount, - owner: info.sender, - raw_amount: RawAmount::LocalDenom(amount), - bond_id: "id".to_string(), - }], - )?; - - Ok(Response::new() - .add_submessage(transfer) - .add_attribute("ibc-tranfer-channel", channel) - .add_attribute("ibc-transfer-receiver", to_address)) -} - -pub fn execute_join_pool( - deps: DepsMut, - env: Env, - info: MessageInfo, - pool_id: u64, - denom: String, - amount: Uint128, - share_out_min_amount: Uint128, -) -> Result { - let join = do_ibc_join_pool_swap_extern_amount_in( - deps.storage, - env, - pool_id, - denom.clone(), - amount, - share_out_min_amount, - // add a dummy ongoing deposit, actual ongoing deposit should calculate the claim using the total balance - vec![OngoingDeposit { - claim_amount: amount, - owner: info.sender, - raw_amount: RawAmount::LocalDenom(amount), - bond_id: "id".to_string(), - }], - )?; - - Ok(Response::new() - .add_submessage(join) - .add_attribute("denom", denom)) -} - -pub fn execute_close_channel(deps: DepsMut, channel_id: String) -> Result { - if TIMED_OUT.load(deps.storage)? && channel_id == ICA_CHANNEL.load(deps.storage)? { - Ok(Response::new().add_message(IbcMsg::CloseChannel { channel_id })) - } else { - Err(ContractError::IcaChannelAlreadySet) - } -} - -// It's recommended to migrate either pending acks or traps, not both at the same time! -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn migrate(deps: DepsMut, _env: Env, msg: MigrateMsg) -> Result { - // remove old traps - for key in msg.delete_traps.clone() { - TRAPS.remove(deps.storage, key) - } - - // remove old pending acks - for key in msg.delete_pending_acks.clone() { - PENDING_ACK.remove(deps.storage, key) - } - - // add a new fresh unlocked ibc lock - IBC_LOCK.save(deps.storage, &Lock::new())?; - - let mut fjq = vec![]; - while !FAILED_JOIN_QUEUE.is_empty(deps.storage)? { - let fj = FAILED_JOIN_QUEUE.pop_front(deps.storage)?; - fjq.push(fj.unwrap()); - } - - if !fjq.is_empty() { - TRAPS.save( - deps.storage, - (1, "undefined-purge".to_string()), - &Trap { - error: "rejoin purge".to_string(), - step: crate::helpers::IbcMsgKind::Ica( - crate::helpers::IcaMessages::JoinSwapExternAmountIn(PendingBond { - bonds: fjq - .into_iter() - .map(|b| OngoingDeposit { - claim_amount: Uint128::new(1), - raw_amount: RawAmount::LocalDenom(b.amount), - owner: b.owner, - bond_id: b.bond_id, - }) - .collect(), - }), - ), - last_succesful: false, - }, - )?; - } - - Ok(Response::new() - .add_attribute("migrate", CONTRACT_NAME) - .add_attribute("success", "true") - .add_attribute("deleted_traps", msg.delete_traps.len().to_string()) - .add_attribute( - "deleted_pending_acks", - msg.delete_pending_acks.len().to_string(), - )) -} - -#[cfg(test)] -mod tests { - use cosmwasm_std::{ - attr, coins, - testing::{mock_dependencies, mock_env, mock_info, MockQuerier}, - to_json_binary, Addr, ContractInfoResponse, ContractResult, QuerierResult, Timestamp, - WasmQuery, - }; - use cw_utils::PaymentError; - - use crate::{ - bond::Bond, - error::Trap, - queries::handle_trapped_errors_query, - state::{PendingBond, Unbond, LOCK_ADMIN, SHARES, UNBONDING_CLAIMS}, - test_helpers::default_setup, - }; - - use super::*; - - #[test] - fn migrate_purges_queue() { - let env = mock_env(); - let mut deps = mock_dependencies(); - - FAILED_JOIN_QUEUE - .push_back( - deps.as_mut().storage, - &Bond { - amount: Uint128::new(101), - owner: Addr::unchecked("vault"), - bond_id: "1".to_string(), - }, - ) - .unwrap(); - FAILED_JOIN_QUEUE - .push_back( - deps.as_mut().storage, - &Bond { - amount: Uint128::new(101), - owner: Addr::unchecked("vault"), - bond_id: "1".to_string(), - }, - ) - .unwrap(); - FAILED_JOIN_QUEUE - .push_back( - deps.as_mut().storage, - &Bond { - amount: Uint128::new(101), - owner: Addr::unchecked("vault"), - bond_id: "1".to_string(), - }, - ) - .unwrap(); - - let msg = MigrateMsg { - delete_traps: vec![], - delete_pending_acks: vec![], - }; - - migrate(deps.as_mut(), env, msg).unwrap(); - assert!(FAILED_JOIN_QUEUE.is_empty(deps.as_mut().storage).unwrap()); - assert_eq!( - handle_trapped_errors_query(deps.as_ref()) - .unwrap() - .errors - .len(), - 1 - ); - } - - #[test] - fn migrate_msg_works_for_pending_acks() { - let mut deps = mock_dependencies(); - let env = mock_env(); - - let entries = vec![ - ( - (1, "channel-1".to_string()), - crate::helpers::IbcMsgKind::Ica(crate::helpers::IcaMessages::ExitPool( - PendingReturningUnbonds { unbonds: vec![] }, - )), - ), - ( - (2, "channel-1".to_string()), - crate::helpers::IbcMsgKind::Ica(crate::helpers::IcaMessages::ExitPool( - PendingReturningUnbonds { unbonds: vec![] }, - )), - ), - ( - (1, "channel-3".to_string()), - crate::helpers::IbcMsgKind::Ica(crate::helpers::IcaMessages::ExitPool( - PendingReturningUnbonds { unbonds: vec![] }, - )), - ), - ( - (1, "channel-1".to_string()), - crate::helpers::IbcMsgKind::Icq, - ), - ( - (1, "channel-2".to_string()), - crate::helpers::IbcMsgKind::Icq, - ), - ( - (1, "channel-4".to_string()), - crate::helpers::IbcMsgKind::Icq, - ), - ]; - - for (key, value) in entries.clone() { - PENDING_ACK - .save(deps.as_mut().storage, key, &value) - .unwrap(); - } - - let msg = MigrateMsg { - delete_traps: vec![], - delete_pending_acks: entries.iter().map(|(key, _)| key.clone()).collect(), - }; - - let res = migrate(deps.as_mut(), env, msg.clone()).unwrap(); - assert!(PENDING_ACK.is_empty(deps.as_ref().storage)); - assert_eq!(res.attributes[2].value, "0".to_string()); - assert_eq!( - res.attributes[3].value, - msg.delete_pending_acks.len().to_string() - ); - } - - #[test] - fn migrate_msg_works_for_traps() { - let mut deps = mock_dependencies(); - let env = mock_env(); - - let entries = vec![ - ( - (1, "channel-1".to_string()), - Trap { - error: "some_error".to_string(), - step: crate::helpers::IbcMsgKind::Ica( - crate::helpers::IcaMessages::JoinSwapExternAmountIn(PendingBond { - bonds: vec![], - }), - ), - last_succesful: true, - }, - ), - ( - (2, "channel-10".to_string()), - Trap { - error: "some_error".to_string(), - step: crate::helpers::IbcMsgKind::Ica( - crate::helpers::IcaMessages::JoinSwapExternAmountIn(PendingBond { - bonds: vec![OngoingDeposit { - claim_amount: Uint128::new(100), - owner: Addr::unchecked("juan".to_string()), - raw_amount: RawAmount::LocalDenom(Uint128::new(100)), - bond_id: "bond_id_1".to_string(), - }], - }), - ), - last_succesful: true, - }, - ), - ( - (3, "channel-100".to_string()), - Trap { - error: "some_error".to_string(), - step: crate::helpers::IbcMsgKind::Ica( - crate::helpers::IcaMessages::JoinSwapExternAmountIn(PendingBond { - bonds: vec![OngoingDeposit { - claim_amount: Uint128::new(100), - owner: Addr::unchecked("juan".to_string()), - raw_amount: RawAmount::LocalDenom(Uint128::new(100)), - bond_id: "bond_id_1".to_string(), - }], - }), - ), - last_succesful: false, - }, - ), - ]; - - for (key, value) in entries.clone() { - TRAPS.save(deps.as_mut().storage, key, &value).unwrap(); - } - - let msg = MigrateMsg { - delete_traps: entries.iter().map(|(key, _)| key.clone()).collect(), - delete_pending_acks: vec![], - }; - - let res = migrate(deps.as_mut(), env, msg.clone()).unwrap(); - assert!(TRAPS.is_empty(deps.as_ref().storage)); - assert_eq!(res.attributes[2].value, msg.delete_traps.len().to_string()); - assert_eq!(res.attributes[3].value, "0".to_string()); - } - - #[test] - fn migrate_msg_works_for_traps_and_pending_acks_combined() { - let mut deps = mock_dependencies(); - let env = mock_env(); - - let pending_acks_entries = vec![ - ( - (1, "channel-1".to_string()), - crate::helpers::IbcMsgKind::Ica(crate::helpers::IcaMessages::ExitPool( - PendingReturningUnbonds { unbonds: vec![] }, - )), - ), - ( - (2, "channel-1".to_string()), - crate::helpers::IbcMsgKind::Ica(crate::helpers::IcaMessages::ExitPool( - PendingReturningUnbonds { unbonds: vec![] }, - )), - ), - ( - (1, "channel-3".to_string()), - crate::helpers::IbcMsgKind::Ica(crate::helpers::IcaMessages::ExitPool( - PendingReturningUnbonds { unbonds: vec![] }, - )), - ), - ( - (1, "channel-1".to_string()), - crate::helpers::IbcMsgKind::Icq, - ), - ( - (1, "channel-2".to_string()), - crate::helpers::IbcMsgKind::Icq, - ), - ( - (1, "channel-4".to_string()), - crate::helpers::IbcMsgKind::Icq, - ), - ]; - - for (key, value) in pending_acks_entries.clone() { - PENDING_ACK - .save(deps.as_mut().storage, key, &value) - .unwrap(); - } - - let trap_entries = vec![ - ( - (1, "channel-1".to_string()), - Trap { - error: "some_error".to_string(), - step: crate::helpers::IbcMsgKind::Ica( - crate::helpers::IcaMessages::JoinSwapExternAmountIn(PendingBond { - bonds: vec![], - }), - ), - last_succesful: true, - }, - ), - ( - (2, "channel-10".to_string()), - Trap { - error: "some_error".to_string(), - step: crate::helpers::IbcMsgKind::Ica( - crate::helpers::IcaMessages::JoinSwapExternAmountIn(PendingBond { - bonds: vec![OngoingDeposit { - claim_amount: Uint128::new(100), - owner: Addr::unchecked("juan".to_string()), - raw_amount: RawAmount::LocalDenom(Uint128::new(100)), - bond_id: "bond_id_1".to_string(), - }], - }), - ), - last_succesful: true, - }, - ), - ( - (3, "channel-100".to_string()), - Trap { - error: "some_error".to_string(), - step: crate::helpers::IbcMsgKind::Ica( - crate::helpers::IcaMessages::JoinSwapExternAmountIn(PendingBond { - bonds: vec![OngoingDeposit { - claim_amount: Uint128::new(100), - owner: Addr::unchecked("juan".to_string()), - raw_amount: RawAmount::LocalDenom(Uint128::new(100)), - bond_id: "bond_id_1".to_string(), - }], - }), - ), - last_succesful: false, - }, - ), - ]; - - for (key, value) in trap_entries.clone() { - TRAPS.save(deps.as_mut().storage, key, &value).unwrap(); - } - - let msg = MigrateMsg { - delete_traps: trap_entries.iter().map(|(key, _)| key.clone()).collect(), - delete_pending_acks: pending_acks_entries - .iter() - .map(|(key, _)| key.clone()) - .collect(), - }; - - let res = migrate(deps.as_mut(), env, msg.clone()).unwrap(); - assert!(TRAPS.is_empty(deps.as_ref().storage)); - assert_eq!(res.attributes[2].value, msg.delete_traps.len().to_string()); - assert_eq!( - res.attributes[3].value, - msg.delete_pending_acks.len().to_string() - ); - } - - #[test] - fn test_execute_try_icq_ibc_locked() { - let mut deps = mock_dependencies(); - let env = mock_env(); - let mock_lock = Lock::new().lock_bond().lock_start_unbond().lock_unbond(); - IBC_LOCK.save(&mut deps.storage, &mock_lock).unwrap(); - let res = execute_try_icq(deps.as_mut(), env).unwrap(); - - assert_eq!(res.attributes[0], attr("IBC_LOCK", "locked")); - assert!(res.messages.is_empty()); - assert!(IBC_LOCK.load(&deps.storage).unwrap().is_locked()); - } - - #[test] - fn test_execute_try_icq_ibc_unlocked_all_queues_empty() { - let mut deps = mock_dependencies(); - let env = mock_env(); - let mock_lock = Lock::new(); - IBC_LOCK.save(&mut deps.storage, &mock_lock).unwrap(); - - default_setup(deps.as_mut().storage).unwrap(); - let lp_cache = LpCache { - locked_shares: Uint128::new(10), - w_unlocked_shares: Uint128::new(10), - d_unlocked_shares: Uint128::new(10), - }; - LP_SHARES.save(deps.as_mut().storage, &lp_cache).unwrap(); - - let res = execute_try_icq(deps.as_mut(), env).unwrap(); - - assert_eq!(res.attributes[0], attr("IBC_LOCK", "unlocked")); - assert!(IBC_LOCK.load(&deps.storage).unwrap().is_unlocked()); - } - - #[test] - fn test_execute_try_icq_ibc_locked_all_queues_filled() { - let mut deps = mock_dependencies(); - let env = mock_env(); - let mock_lock = Lock::new().lock_bond().lock_start_unbond().lock_unbond(); - default_setup(deps.as_mut().storage).unwrap(); - let lp_cache = LpCache { - locked_shares: Uint128::new(10), - w_unlocked_shares: Uint128::new(10), - d_unlocked_shares: Uint128::new(10), - }; - LP_SHARES.save(deps.as_mut().storage, &lp_cache).unwrap(); - IBC_LOCK.save(&mut deps.storage, &mock_lock).unwrap(); - - BOND_QUEUE - .push_back( - &mut deps.storage, - &Bond { - amount: Uint128::one(), - owner: Addr::unchecked("juan".to_string()), - bond_id: "bond_id_1".to_string(), - }, - ) - .unwrap(); - START_UNBOND_QUEUE - .push_back( - &mut deps.storage, - &StartUnbond { - owner: Addr::unchecked("pepe".to_string()), - id: "bond_id_10".to_string(), - primitive_shares: Uint128::new(10), - }, - ) - .unwrap(); - UNBOND_QUEUE - .push_back( - &mut deps.storage, - &Unbond { - owner: Addr::unchecked("pedro".to_string()), - id: "bond_id_100".to_string(), - lp_shares: Uint128::new(1000), - unlock_time: Timestamp::from_seconds(1000), - attempted: true, - }, - ) - .unwrap(); - - let res = execute_try_icq(deps.as_mut(), env).unwrap(); - - assert_eq!(res.attributes[0], attr("IBC_LOCK", "locked")); - assert!(IBC_LOCK.load(&deps.storage).unwrap().is_locked()); - assert!(res.messages.is_empty()); - } - - #[test] - fn test_execute_try_icq_ibc_unlocked_bond_queue_full() { - let mut deps = mock_dependencies(); - let env = mock_env(); - let mock_lock = Lock::new(); - default_setup(deps.as_mut().storage).unwrap(); - let lp_cache = LpCache { - locked_shares: Uint128::new(10), - w_unlocked_shares: Uint128::new(10), - d_unlocked_shares: Uint128::new(10), - }; - LP_SHARES.save(deps.as_mut().storage, &lp_cache).unwrap(); - IBC_LOCK.save(&mut deps.storage, &mock_lock).unwrap(); - - BOND_QUEUE - .push_back( - &mut deps.storage, - &Bond { - amount: Uint128::one(), - owner: Addr::unchecked("juan".to_string()), - bond_id: "bond_id_1".to_string(), - }, - ) - .unwrap(); - - let res = execute_try_icq(deps.as_mut(), env).unwrap(); - assert_eq!(res.attributes[0], attr("bond_queue", "locked")); - let lock = IBC_LOCK.load(&deps.storage).unwrap(); - assert!(lock.bond.is_locked()); - assert!(lock.start_unbond.is_unlocked()); - assert!(lock.unbond.is_unlocked()); - } - - #[test] - fn test_execute_try_icq_ibc_bond_queue_empty() { - let mut deps = mock_dependencies(); - let env = mock_env(); - let mock_lock = Lock::new(); - default_setup(deps.as_mut().storage).unwrap(); - let lp_cache = LpCache { - locked_shares: Uint128::new(10), - w_unlocked_shares: Uint128::new(10), - d_unlocked_shares: Uint128::new(10), - }; - LP_SHARES.save(deps.as_mut().storage, &lp_cache).unwrap(); - IBC_LOCK.save(&mut deps.storage, &mock_lock).unwrap(); - - START_UNBOND_QUEUE - .push_back( - &mut deps.storage, - &StartUnbond { - owner: Addr::unchecked("pepe".to_string()), - id: "bond_id_10".to_string(), - primitive_shares: Uint128::new(10), - }, - ) - .unwrap(); - UNBOND_QUEUE - .push_back( - &mut deps.storage, - &Unbond { - owner: Addr::unchecked("pedro".to_string()), - id: "bond_id_100".to_string(), - lp_shares: Uint128::new(1000), - unlock_time: Timestamp::from_seconds(1000), - attempted: true, - }, - ) - .unwrap(); - - let res = execute_try_icq(deps.as_mut(), env).unwrap(); - assert_eq!(res.attributes[0], attr("start_unbond_queue", "locked")); - let lock = IBC_LOCK.load(&deps.storage).unwrap(); - assert!(lock.bond.is_unlocked()); - assert!(lock.start_unbond.is_locked()); - assert!(lock.unbond.is_unlocked()); - } - - #[test] - fn test_execute_unbond_filled_queues() { - let mut deps = mock_dependencies(); - let env = mock_env(); - let mock_lock = Lock::new(); - default_setup(deps.as_mut().storage).unwrap(); - let lp_cache = LpCache { - locked_shares: Uint128::new(10), - w_unlocked_shares: Uint128::new(10), - d_unlocked_shares: Uint128::new(10), - }; - LP_SHARES.save(deps.as_mut().storage, &lp_cache).unwrap(); - IBC_LOCK.save(&mut deps.storage, &mock_lock).unwrap(); - - BOND_QUEUE - .push_back( - &mut deps.storage, - &Bond { - amount: Uint128::new(1000), - owner: Addr::unchecked("alice"), - bond_id: "3".to_string(), - }, - ) - .unwrap(); - START_UNBOND_QUEUE - .push_back( - &mut deps.storage, - &StartUnbond { - owner: Addr::unchecked("pepe".to_string()), - id: "bond_id_10".to_string(), - primitive_shares: Uint128::new(10), - }, - ) - .unwrap(); - - let id = "2".to_string(); - let owner = Addr::unchecked("bob"); - DEPOSITOR.save(deps.as_mut().storage, &owner).unwrap(); - - // unbond number are abitrary for this test - UNBONDING_CLAIMS - .save( - deps.as_mut().storage, - (owner.clone(), id.clone()), - &Unbond { - lp_shares: Uint128::new(100), - unlock_time: env.block.time, - attempted: true, - owner: owner.clone(), - id: id.clone(), - }, - ) - .unwrap(); - - let res = execute_unbond( - deps.as_mut(), - env, - MessageInfo { - sender: owner, - funds: vec![], - }, - id, - ) - .unwrap(); - assert_eq!(res.attributes[0], attr("bond_queue", "locked")); - let lock = IBC_LOCK.load(&deps.storage).unwrap(); - assert!(lock.bond.is_locked()); - assert!(lock.start_unbond.is_unlocked()); - assert!(lock.unbond.is_unlocked()); - } - - #[test] - fn test_execute_unbond_filled_start_unbond_queue() { - let mut deps = mock_dependencies(); - let env = mock_env(); - let mock_lock = Lock::new(); - default_setup(deps.as_mut().storage).unwrap(); - let lp_cache = LpCache { - locked_shares: Uint128::new(10), - w_unlocked_shares: Uint128::new(10), - d_unlocked_shares: Uint128::new(10), - }; - LP_SHARES.save(deps.as_mut().storage, &lp_cache).unwrap(); - IBC_LOCK.save(&mut deps.storage, &mock_lock).unwrap(); - - START_UNBOND_QUEUE - .push_back( - &mut deps.storage, - &StartUnbond { - owner: Addr::unchecked("pepe".to_string()), - id: "bond_id_10".to_string(), - primitive_shares: Uint128::new(10), - }, - ) - .unwrap(); - - let id = "2".to_string(); - let owner = Addr::unchecked("bob"); - DEPOSITOR.save(deps.as_mut().storage, &owner).unwrap(); - - // unbond number are abitrary for this test - UNBONDING_CLAIMS - .save( - deps.as_mut().storage, - (owner.clone(), id.clone()), - &Unbond { - lp_shares: Uint128::new(100), - unlock_time: env.block.time, - attempted: true, - owner: owner.clone(), - id: id.clone(), - }, - ) - .unwrap(); - - let res = execute_unbond( - deps.as_mut(), - env, - MessageInfo { - sender: owner, - funds: vec![], - }, - id, - ) - .unwrap(); - assert_eq!(res.attributes[0], attr("start_unbond_queue", "locked")); - let lock = IBC_LOCK.load(&deps.storage).unwrap(); - assert!(lock.bond.is_unlocked()); - assert!(lock.start_unbond.is_locked()); - assert!(lock.unbond.is_unlocked()); - } - - #[test] - fn test_execute_start_unbond_filled_queues() { - let mut deps = mock_dependencies(); - let env = mock_env(); - let mock_lock = Lock::new(); - default_setup(deps.as_mut().storage).unwrap(); - let lp_cache = LpCache { - locked_shares: Uint128::new(10), - w_unlocked_shares: Uint128::new(10), - d_unlocked_shares: Uint128::new(10), - }; - LP_SHARES.save(deps.as_mut().storage, &lp_cache).unwrap(); - IBC_LOCK.save(&mut deps.storage, &mock_lock).unwrap(); - - BOND_QUEUE - .push_back( - &mut deps.storage, - &Bond { - amount: Uint128::new(1000), - owner: Addr::unchecked("alice"), - bond_id: "3".to_string(), - }, - ) - .unwrap(); - UNBOND_QUEUE - .push_back( - &mut deps.storage, - &Unbond { - owner: Addr::unchecked("pedro".to_string()), - id: "bond_id_100".to_string(), - lp_shares: Uint128::new(1000), - unlock_time: Timestamp::from_seconds(1000), - attempted: true, - }, - ) - .unwrap(); - - let id = "2".to_string(); - let owner = Addr::unchecked("bob"); - DEPOSITOR.save(deps.as_mut().storage, &owner).unwrap(); - SHARES - .save(deps.as_mut().storage, owner.clone(), &Uint128::new(1000)) - .unwrap(); - - let res = execute_start_unbond( - deps.as_mut(), - env, - MessageInfo { - sender: owner, - funds: vec![], - }, - id, - Uint128::new(100), - ) - .unwrap(); - - assert_eq!(res.attributes[0], attr("bond_queue", "locked")); - let lock = IBC_LOCK.load(&deps.storage).unwrap(); - assert!(lock.bond.is_locked()); - assert!(lock.start_unbond.is_unlocked()); - assert!(lock.unbond.is_unlocked()); - } - - #[test] - fn test_start_unbond_with_funds() { - let mut deps = mock_dependencies(); - let env = mock_env(); - let info = mock_info("pepe", &coins(420, "uqsr")); - let msg = ExecuteMsg::StartUnbond { - id: "bond_id_1".to_string(), - share_amount: Uint128::new(69), - }; - - let res = execute(deps.as_mut(), env, info, msg); - assert_eq!(res.unwrap_err(), PaymentError::NonPayable {}.into()); - } - - #[test] - fn test_unbond_with_funds() { - let mut deps = mock_dependencies(); - let env = mock_env(); - let info = mock_info("addr0000", &coins(420, "uqsr")); - let msg = ExecuteMsg::Unbond { - id: "unbond_id".to_string(), - }; - - let res = execute(deps.as_mut(), env, info, msg); - assert_eq!(res.unwrap_err(), PaymentError::NonPayable {}.into()); - } - - #[test] - fn test_execute_add_lock_admin() { - let admin = "bob"; - - let mut info = ContractInfoResponse::default(); - info.admin = Some(admin.to_string()); - let mut q = MockQuerier::default(); - q.update_wasm(move |q: &WasmQuery| -> QuerierResult { - match q { - WasmQuery::ContractInfo { contract_addr: _ } => { - QuerierResult::Ok(ContractResult::Ok(to_json_binary(&info).unwrap())) - } - _ => unreachable!(), - } - }); - - let mut deps = mock_dependencies(); - deps.querier = q; - - let env = mock_env(); - - let info = MessageInfo { - sender: Addr::unchecked(admin), - funds: vec![], - }; - - let msg = ExecuteMsg::AddLockAdmin { - to_add: "alice".to_string(), - }; - let res = execute(deps.as_mut(), env, info, msg).unwrap(); - let _ = LOCK_ADMIN - .load(deps.as_mut().storage, &Addr::unchecked("alice")) - .unwrap(); - assert_eq!( - res.attributes, - vec![("action", "add_lock_admin"), ("lock_admin", "alice")] - ) - } - - #[test] - fn test_execute_remove_lock_admin() { - let admin = "bob"; - - let mut info = ContractInfoResponse::default(); - info.admin = Some(admin.to_string()); - let mut q = MockQuerier::default(); - q.update_wasm(move |q: &WasmQuery| -> QuerierResult { - match q { - WasmQuery::ContractInfo { contract_addr: _ } => { - QuerierResult::Ok(ContractResult::Ok(to_json_binary(&info).unwrap())) - } - _ => unreachable!(), - } - }); - - let mut deps = mock_dependencies(); - deps.querier = q; - - let env = mock_env(); - - let info = MessageInfo { - sender: Addr::unchecked(admin), - funds: vec![], - }; - - let msg = ExecuteMsg::AddLockAdmin { - to_add: "alice".to_string(), - }; - let res = execute(deps.as_mut(), env.clone(), info.clone(), msg).unwrap(); - let _ = LOCK_ADMIN - .load(deps.as_mut().storage, &Addr::unchecked("alice")) - .unwrap(); - assert_eq!( - res.attributes, - vec![("action", "add_lock_admin"), ("lock_admin", "alice")] - ); - - let msg = ExecuteMsg::RemoveLockAdmin { - to_remove: "alice".to_string(), - }; - let res = execute(deps.as_mut(), env, info, msg).unwrap(); - assert_eq!( - res.attributes, - vec![("action", "remove_lock_admin"), ("lock_admin", "alice")] - ) - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/error.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/error.rs deleted file mode 100644 index 48b2ba4cf..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/error.rs +++ /dev/null @@ -1,141 +0,0 @@ -use cosmwasm_std::{CheckedMultiplyRatioError, DivideByZeroError, OverflowError, StdError}; -use quasar_types::error::Error as QError; -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; -use std::fmt::Debug; - -use crate::helpers::IbcMsgKind; -use std::str::Utf8Error; -use thiserror::Error; - -/// Never is a placeholder to ensure we don't return any errors -#[derive(Error, Debug)] -pub enum Never {} - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub struct Trap { - // A string describing the trapped error - pub error: String, - // the failed step and underlying values - pub step: IbcMsgKind, - // last_succesful notes whether the IbcMsg of step was succesful on the counterparty chain - pub last_succesful: bool, -} - -#[derive(Error, Debug, PartialEq)] -pub enum ContractError { - #[error("{0}")] - Std(#[from] StdError), - - #[error("{0}")] - Base(#[from] cw20_base::ContractError), - - #[error("{0}")] - QError(#[from] QError), - - #[error("map has duplicate key while no key should be present")] - DuplicateKey, - - #[error("caller is unauthorized")] - Unauthorized, - - #[error("{0}")] - PaymentError(#[from] cw_utils::PaymentError), - - #[error("not enough claims")] - InsufficientClaims, - - #[error("not enough funds")] - InsufficientFunds, - - #[error("base denom not found")] - BaseDenomNotFound, - - #[error("quote denom not found")] - QuoteDenomNotFound, - - #[error("No item in the queue {} while an item was expected", queue)] - QueueItemNotFound { queue: String }, - - #[error("no counterpart ica address found")] - NoCounterpartyIcaAddress, - - #[error("ica channel is already set while it should be unset")] - IcaChannelAlreadySet, - - #[error("channel is not an ica channel")] - NoIcaChannel, - - #[error("channel is not an icq channel")] - NoIcqChannel, - - #[error("no connection is found")] - NoConnectionFound, - - #[error("incorrect connection id")] - IncorrectConnection, - - #[error("incorrect channel open type, should be OpenInit")] - IncorrectChannelOpenType, - - #[error("raw ack in recovery could not be handled")] - IncorrectRecoveryAck, - - #[error("no timestamp time found for ibc packets")] - NoTimestampTime, - - #[error("reply data not found")] - NoReplyData, - - #[error("Could not deserialize ack: {err}, payload was {b64_bin}")] - DeserializeIcaAck { b64_bin: String, err: String }, - - #[error("Could not find returning transfer")] - ReturningTransferNotFound, - - #[error("amount of returning transfer is not the same as the expected amount")] - ReturningTransferIncorrectAmount, - - #[error("Shares are still unbonding")] - SharesNotYetUnbonded, - - #[error("found incorrect raw amount type")] - IncorrectRawAmount, - - #[error("{0}")] - DecodeError(#[from] prost::DecodeError), - - #[error("parse int error: {error} caused by {value}")] - ParseIntError { error: String, value: String }, - - #[error("parse int error: {error} caused by {value}")] - ParseDecError { error: StdError, value: String }, - - #[error("{0}")] - OverflowError(#[from] OverflowError), - - #[error("{0}, location {1}")] - TracedOverflowError(OverflowError, String), - - #[error("{0}")] - DivideByZeroError(#[from] DivideByZeroError), - - #[error("{0}")] - CheckedMultiplyRatioError(#[from] CheckedMultiplyRatioError), - - #[error("{0}")] - Utf8Error(#[from] Utf8Error), - - #[error("{0}")] - SerdeJsonDe(#[from] serde_json_wasm::de::Error), - - #[error("could not serialize to json")] - SerdeJsonSer, - - #[error("The Callback has no amount set")] - CallbackHasNoAmount, - - #[error("No pending unbonds found")] - NoPendingUnbonds, -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/error_recovery.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/error_recovery.rs deleted file mode 100644 index fe5e9267b..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/error_recovery.rs +++ /dev/null @@ -1,298 +0,0 @@ -use cosmwasm_std::{ - from_json, Addr, DepsMut, Env, IbcAcknowledgement, Response, Storage, SubMsg, Uint128, -}; -use osmosis_std::types::osmosis::gamm::v1beta1::MsgJoinSwapExternAmountInResponse; -use quasar_types::{ - ibc::IcsAck, - ica::{packet::AckBody, traits::Unpack}, -}; -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -use crate::{ - error::ContractError, - helpers::{create_ibc_ack_submsg, IbcMsgKind, IcaMessages}, - ibc_util::calculate_token_out_min_amount, - state::{ - FundPath, LpCache, PendingBond, RawAmount, CONFIG, ICA_CHANNEL, LP_SHARES, RECOVERY_ACK, - TRAPS, - }, - unbond::do_exit_swap, -}; - -// start_recovery fetches an error from the TRAPPED_ERRORS and start the appropriate recovery from there -pub fn _start_recovery( - deps: DepsMut, - env: &Env, - error_sequence: u64, - channel: String, -) -> Result { - let error = TRAPS.load(deps.storage, (error_sequence, channel.clone()))?; - match error.last_succesful { - true => { - match error.step { - // if the transfer failed. The funds in pending are still located on Quasar, meaning we - crate::helpers::IbcMsgKind::Transfer { pending, amount } => { - // cleanup error state to prevent multiple error recoveries - TRAPS.remove(deps.storage, (error_sequence, channel)); - let msg = handle_transfer_recovery( - deps.storage, - env, - pending, - amount, - error_sequence, - )?; - Ok(Response::new() - .add_submessage(msg) - .add_attribute("error-recover-sequence", error_sequence.to_string()) - .add_attribute("error-recovery-value", error.last_succesful.to_string())) - } - crate::helpers::IbcMsgKind::Ica(_ica) => todo!(), - // if ICQ was the last successful step, all we failed trying to empty our queues and dispatching any following - // IBC messages, meaning we don't have to do anything with a seperate try_icq endpoint - crate::helpers::IbcMsgKind::Icq => unimplemented!(), - } - } - false => { - todo!() - } - } -} - -// amount is the total amount we will try to transfer back to quasar, pending bond is the users the funds should return back to -#[allow(dead_code)] -fn handle_transfer_recovery( - storage: &mut dyn Storage, - _env: &Env, - bonds: PendingBond, - _amount: Uint128, - trapped_id: u64, -) -> Result { - let _config = CONFIG.load(storage)?; - let returning: Result, ContractError> = bonds - .bonds - .iter() - .map(|bond| { - if let RawAmount::LocalDenom(val) = bond.raw_amount { - Ok(ReturningRecovery { - amount: RawAmount::LocalDenom(val), - owner: bond.owner.clone(), - id: FundPath::Bond { - id: bond.bond_id.clone(), - }, - }) - } else { - Err(ContractError::ReturningTransferIncorrectAmount) - } - }) - .collect(); - - let _returning = PendingReturningRecovery { - returning: returning?, - trapped_id, - }; - - // big TODO here, below should be modified for error recovery criteria - // leaving as todo because this code is not live yet. - // we want to call do_transfer_batch_unbond with the returning unbonds but in the context of error recovery - - // let msg = todo!(); - // Ok(create_ibc_ack_submsg( - // storage, - // IbcMsgKind::Ica(IcaMessages::RecoveryReturnTransfer(returning)), - // msg, - // _config.transfer_channel, - // )?) - todo!() -} - -fn _handle_last_succesful_ica_recovery( - storage: &mut dyn Storage, - env: &Env, - ica: IcaMessages, - _last_succesful: bool, - trapped_id: u64, -) -> Result { - match ica { - // in every succesful ica recovery, our last message was succesful, but our pending state is not, - // For a join swap, we need to exit all shares - IcaMessages::JoinSwapExternAmountIn(pending) => { - handle_join_swap_recovery(storage, env, pending, trapped_id) - } - IcaMessages::LockTokens(_, _) => todo!(), - IcaMessages::BeginUnlocking(_, _) => todo!(), - IcaMessages::ExitPool(_) => todo!(), - IcaMessages::ReturnTransfer(_) => todo!(), - IcaMessages::RecoveryExitPool(_) => todo!(), - IcaMessages::RecoveryReturnTransfer(_) => todo!(), - } -} -fn _handle_last_failed_ica_recovery( - _storage: &mut dyn Storage, - _env: &Env, - ica: IcaMessages, - _trapped_id: u64, -) -> Result { - match ica { - // recover by sending funds back - // since the ica failed, we should transfer the funds back, we do not yet expect a raw amount to be set to lp shares - // how do we know how much to return here? - IcaMessages::JoinSwapExternAmountIn(_pending) => { - todo!() - } - IcaMessages::LockTokens(_, _) => todo!(), - // if BeginUnlocking followup failed, our tokens, the amount of tokens in the request was actually succesful, - // so we continue to a recovery exit swap - IcaMessages::BeginUnlocking(_, _) => todo!(), - // if the exit pool was actually succesful, we need to deserialize the saved ack result again to get the amount of tokens - // users should get - // TODO, can they get rejoined to the lp pool here? Maybe???? - // probably gets compounded back again, how do we know here?, do we know at all? - // we let the exit pool get autocompounded by the contract again, to recover from here, we start a start_unlock for all stuck funds - IcaMessages::ExitPool(_pending) => { - todo!() - // let msg = do_begin_unlocking(storage, env, to_unbond)?; - } - // we just retry the transfer here - IcaMessages::ReturnTransfer(_) => todo!(), - // same as regular exit pool recovery - IcaMessages::RecoveryExitPool(_) => todo!(), - // same as regular transfer recovery - IcaMessages::RecoveryReturnTransfer(_) => todo!(), - } -} - -// kinda messed up that we create duplicate code here, should be solved with a single unpacking function that accepts -// a closure for IcsAck::Result and IcsAck::Error -#[allow(dead_code)] -fn de_succcesful_join( - ack_bin: IbcAcknowledgement, -) -> Result { - let ack: IcsAck = from_json(ack_bin.data)?; - if let IcsAck::Result(val) = ack { - let ack_body = AckBody::from_bytes(val.0.as_ref())?.to_any()?; - let ack = MsgJoinSwapExternAmountInResponse::unpack(ack_body)?; - Ok(ack) - } else { - Err(ContractError::IncorrectRecoveryAck) - } -} - -// if the join_swap was succesful, the refund path means we have to -#[allow(dead_code)] -fn handle_join_swap_recovery( - storage: &mut dyn Storage, - env: &Env, - pending: PendingBond, - trapped_id: u64, -) -> Result { - let channel = ICA_CHANNEL.load(storage)?; - let ack_bin = RECOVERY_ACK.load(storage, (trapped_id, channel.clone()))?; - // in this case the recovery ack should contain a joinswapexternamountin response - // we try to deserialize it - let join_result = de_succcesful_join(ack_bin)?; - - // we expect the total amount here to be in local_denom since, although the join was succesful - // the RawAmount cannot yet have been up updated - let total_lp = pending.bonds.iter().try_fold(Uint128::zero(), |acc, val| { - if let RawAmount::LocalDenom(amount) = val.raw_amount { - Ok(acc.checked_add(amount)?) - } else { - Err(ContractError::IncorrectRawAmount) - } - })?; - // now we need to divide up the lp shares amount our users according to the individual local denom amount - let exits_res: Result, ContractError> = pending - .bonds - .iter() - .map(|val| { - // since the ICA followup failed, we need to figure out how to convert - if let RawAmount::LocalDenom(amount) = val.raw_amount { - Ok(ReturningRecovery { - // lp_shares_i = tokens_i * lp_total / tokens_total - amount: RawAmount::LpShares(amount.checked_mul(total_lp)?.checked_div( - Uint128::new(join_result.share_out_amount.parse().map_err(|err| { - ContractError::ParseIntError { - error: format!("join_swap_recovery:{err}"), - value: join_result.share_out_amount.clone(), - } - })?), - )?), - owner: val.owner.clone(), - // since we are recovering from a join swap, we need do save - // as a Bond for bookkeeping on returned - id: FundPath::Bond { - id: val.bond_id.clone(), - }, - }) - } else { - Err(ContractError::IncorrectRawAmount) - } - }) - .collect(); - - let exits = exits_res?; - let total_exit: Uint128 = exits.iter().try_fold( - Uint128::zero(), - |acc, val| -> Result { - match val.amount { - RawAmount::LocalDenom(_) => unimplemented!(), - RawAmount::LpShares(amount) => Ok(amount.checked_add(acc)?), - } - }, - )?; - - LP_SHARES.update(storage, |mut old| -> Result { - // we remove the amount of shares we are are going to unlock from the locked amount - old.d_unlocked_shares = old.d_unlocked_shares.checked_sub(total_exit)?; - // we add the amount of shares we are going to unlock to the total unlocked - old.w_unlocked_shares = old.w_unlocked_shares.checked_add(total_exit)?; - Ok(old) - })?; - - let token_out_min_amount = calculate_token_out_min_amount(storage).unwrap(); - - let exit = do_exit_swap(storage, env, token_out_min_amount, total_exit)?; - Ok(create_ibc_ack_submsg( - storage, - IbcMsgKind::Ica(IcaMessages::RecoveryExitPool(PendingReturningRecovery { - returning: exits, - trapped_id, - })), - exit, - channel, - )?) -} - -// fn create_recovery_submsg( -// msg: IbcMsg, -// kind: IbcMsgKind -// ) -> Result { - -// } - -fn _handle_lock_recovery() {} - -fn _handle_begin_unlocking_recovery() {} - -fn _handle_exit_pool_recovery() {} - -fn _handle_return_transfer_recovery() {} - -// TODO refactor bonds/unbonds to a single struct item that is bidirectional with a direction enum -// because we did not abstract nicely, we need a separate struct here -// we should feel bad -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug, Eq)] -#[serde(rename_all = "snake_case")] -pub struct PendingReturningRecovery { - pub returning: Vec, - pub trapped_id: u64, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug, Eq)] -#[serde(rename_all = "snake_case")] -pub struct ReturningRecovery { - pub amount: RawAmount, - pub owner: Addr, - pub id: FundPath, -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/execute.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/execute.rs deleted file mode 100644 index 2a3fa66cc..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/execute.rs +++ /dev/null @@ -1,1007 +0,0 @@ -use cosmwasm_std::{DepsMut, Env, MessageInfo, Response}; -use cw_utils::nonpayable; - -use crate::{ - bond::Bond, - error::ContractError, - helpers::{IbcMsgKind, IcaMessages}, - state::{PendingBond, RawAmount, FAILED_JOIN_QUEUE, TRAPS}, - unbond::{do_unbond, PendingReturningUnbonds}, -}; - -/// The retry entry point will be used to retry any failed ICA message given the sequence number and the channel. -/// Depending on the type of ICA message, the contract will handle the retry differently. -/// Funds cannot be sent and, for now, only the lock admin can call retry. -pub fn execute_retry( - deps: DepsMut, - env: Env, - info: MessageInfo, - seq: u64, - channel: String, -) -> Result { - nonpayable(&info)?; - // this endpoint is permissionless in order to allow anyone trigger a retry - let traps = TRAPS.load(deps.storage, (seq, channel.clone()))?; - match traps.step { - IbcMsgKind::Ica(ica_kind) => match ica_kind { - IcaMessages::ExitPool(pending) => { - handle_retry_exit_pool(deps, env, pending, seq, channel) - } - IcaMessages::JoinSwapExternAmountIn(pending) => { - handle_retry_join_pool(deps, env, pending, seq, channel) - } - IcaMessages::BeginUnlocking(pending, to_unbond) => { - handle_retry_begin_unlocking(deps, env, pending, to_unbond) - } - _ => todo!() - }, - _ => todo!(), - } -} - -pub fn handle_retry_join_pool( - deps: DepsMut, - _env: Env, - pending: PendingBond, - seq: u64, - channel: String, -) -> Result { - let mut resp = Response::new() - .add_attribute("action", "retry") - .add_attribute("kind", "join_pool"); - - for ongoing_deposit in pending.bonds { - match ongoing_deposit.raw_amount { - RawAmount::LocalDenom(amount) => { - FAILED_JOIN_QUEUE.push_back( - deps.storage, - &Bond { - amount, - owner: ongoing_deposit.owner.clone(), - bond_id: ongoing_deposit.bond_id.clone(), - }, - )?; - resp = resp - .add_attribute("bond_id", ongoing_deposit.bond_id) - .add_attribute("amount", amount); - } - // We should never have LP shares here - RawAmount::LpShares(_) => return Err(ContractError::IncorrectRawAmount), - } - } - - TRAPS.remove(deps.storage, (seq, channel)); - - Ok(resp) -} - -/// The handle retry exit pool checks that pending unbonds is not empty and then iterates over the pending unbonds vector. -/// For each unbond, it will check that unbond time has expired and push it to the front of the pending unbond queue. -/// A manual TryIcq will be needed to dispatch the IBC message. -pub fn handle_retry_exit_pool( - deps: DepsMut, - env: Env, - pending: PendingReturningUnbonds, - seq: u64, - channel: String, -) -> Result { - if pending.unbonds.is_empty() { - return Err(ContractError::NoPendingUnbonds); - } - - let mut resp = Response::new(); - - for pu in pending.unbonds { - do_unbond(deps.storage, &env, pu.owner.clone(), pu.id.clone())?; - resp = resp - .add_attribute("unbond", pu.owner.clone()) - .add_attribute("unbond_id", pu.id); - } - - resp = resp - .add_attribute("action", "retry") - .add_attribute("kind", "exit_pool"); - - // remove the entry from traps so retrying a single failed tx cannot be triggered twice - TRAPS.remove(deps.storage, (seq, channel)); - - Ok(resp) -} - -#[cfg(test)] -mod tests { - // use cosmos_sdk_proto::tendermint::abci::ResponseQuery; - use cosmwasm_std::{Binary, Coin, CosmosMsg, IbcMsg}; - use osmosis_std::types::cosmos::bank::v1beta1::QueryBalanceResponse; - use osmosis_std::types::{ - cosmos::base::v1beta1::Coin as OsmoCoin, osmosis::lockup::LockedResponse, - }; - - use cosmwasm_std::{ - attr, - testing::{mock_dependencies, mock_env}, - to_json_binary, Addr, StdError, Timestamp, Uint128, - }; - use osmosis_std::types::osmosis::gamm::v1beta1::{ - QueryCalcExitPoolCoinsFromSharesResponse, QueryCalcJoinPoolSharesResponse, - QuerySpotPriceResponse, - }; - use prost::Message; - use quasar_types::icq::{CosmosResponse, InterchainQueryPacketAck}; - - use crate::ibc::handle_icq_ack; - use crate::state::{PENDING_BOND_QUEUE, REJOIN_QUEUE, SIMULATED_JOIN_AMOUNT_IN}; - use crate::test_helpers::{create_query_response, pending_bond_to_bond}; - use crate::{ - contract::execute_try_icq, - error::Trap, - ibc_lock::Lock, - state::{ - OngoingDeposit, RawAmount, Unbond, IBC_LOCK, PENDING_UNBOND_QUEUE, UNBONDING_CLAIMS, - }, - test_helpers::default_setup, - unbond::ReturningUnbond, - }; - - use super::*; - - #[test] - fn test_handle_retry_exit_pool_works() { - let mut deps = mock_dependencies(); - let env = mock_env(); - - let pending = PendingReturningUnbonds { - unbonds: vec![ - ReturningUnbond { - amount: RawAmount::LocalDenom(Uint128::new(101)), - owner: Addr::unchecked("owner1"), - id: "1".to_string(), - }, - ReturningUnbond { - amount: RawAmount::LocalDenom(Uint128::new(102)), - owner: Addr::unchecked("owner2"), - id: "2".to_string(), - }, - ReturningUnbond { - amount: RawAmount::LocalDenom(Uint128::new(103)), - owner: Addr::unchecked("owner3"), - id: "3".to_string(), - }, - ], - }; - - TRAPS - .save( - deps.as_mut().storage, - (3539, "channel-35".to_string()), - &Trap { - error: "exit pool failed on osmosis".to_string(), - step: IbcMsgKind::Ica(IcaMessages::ExitPool(pending)), - last_succesful: true, - }, - ) - .unwrap(); - - UNBONDING_CLAIMS - .save( - deps.as_mut().storage, - (Addr::unchecked("owner1"), "1".to_string()), - &Unbond { - lp_shares: Uint128::new(101), - unlock_time: Timestamp::from_seconds(1000), - attempted: true, - owner: Addr::unchecked("owner1"), - id: "1".to_string(), - }, - ) - .unwrap(); - - UNBONDING_CLAIMS - .save( - deps.as_mut().storage, - (Addr::unchecked("owner2"), "2".to_string()), - &Unbond { - lp_shares: Uint128::new(101), - unlock_time: Timestamp::from_seconds(1000), - attempted: true, - owner: Addr::unchecked("owner2"), - id: "2".to_string(), - }, - ) - .unwrap(); - - UNBONDING_CLAIMS - .save( - deps.as_mut().storage, - (Addr::unchecked("owner3"), "3".to_string()), - &Unbond { - lp_shares: Uint128::new(103), - unlock_time: Timestamp::from_seconds(1000), - attempted: true, - owner: Addr::unchecked("owner3"), - id: "3".to_string(), - }, - ) - .unwrap(); - - let res = execute_retry( - deps.as_mut(), - env, - MessageInfo { - sender: Addr::unchecked("not_admin"), - funds: vec![], - }, - 3539, - "channel-35".to_string(), - ) - .unwrap(); - - assert!(!TRAPS.has(&deps.storage, (3539, "channel-35".to_string()))); - - assert_eq!( - res.attributes, - vec![ - attr("unbond", "owner1"), - attr("unbond_id", "1"), - attr("unbond", "owner2"), - attr("unbond_id", "2"), - attr("unbond", "owner3"), - attr("unbond_id", "3"), - attr("action", "retry"), - attr("kind", "exit_pool"), - ] - ); - - assert_eq!(PENDING_UNBOND_QUEUE.len(&deps.storage).unwrap(), 3); - assert_eq!( - PENDING_UNBOND_QUEUE.back(&deps.storage).unwrap().unwrap(), - Unbond { - lp_shares: Uint128::new(103), - unlock_time: Timestamp::from_seconds(1000), - attempted: true, - owner: Addr::unchecked("owner3"), - id: "3".to_string(), - } - ); - } - - #[test] - fn test_handle_retry_exit_pool_empty_pending_unbonds_fails() { - let mut deps = mock_dependencies(); - let env = mock_env(); - - let pending = PendingReturningUnbonds { unbonds: vec![] }; - - TRAPS - .save( - deps.as_mut().storage, - (3539, "channel-35".to_string()), - &Trap { - error: "exit pool failed on osmosis".to_string(), - step: IbcMsgKind::Ica(IcaMessages::ExitPool(pending)), - last_succesful: true, - }, - ) - .unwrap(); - - UNBONDING_CLAIMS - .save( - deps.as_mut().storage, - (Addr::unchecked("owner1"), "1".to_string()), - &Unbond { - lp_shares: Uint128::new(101), - unlock_time: Timestamp::from_seconds(1000), - attempted: true, - owner: Addr::unchecked("owner1"), - id: "1".to_string(), - }, - ) - .unwrap(); - - let res = execute_retry( - deps.as_mut(), - env, - MessageInfo { - sender: Addr::unchecked("not_admin"), - funds: vec![], - }, - 3539, - "channel-35".to_string(), - ) - .unwrap_err(); - - assert_eq!(res, ContractError::NoPendingUnbonds,); - assert!(PENDING_UNBOND_QUEUE.is_empty(&deps.storage).unwrap()); - } - - #[test] - fn test_handle_retry_exit_pool_without_all_unbonding_claims_fails() { - let mut deps = mock_dependencies(); - let env = mock_env(); - - let pending = PendingReturningUnbonds { - unbonds: vec![ - ReturningUnbond { - amount: RawAmount::LocalDenom(Uint128::new(101)), - owner: Addr::unchecked("owner1"), - id: "1".to_string(), - }, - ReturningUnbond { - amount: RawAmount::LocalDenom(Uint128::new(102)), - owner: Addr::unchecked("owner2"), - id: "2".to_string(), - }, - ], - }; - - TRAPS - .save( - deps.as_mut().storage, - (3539, "channel-35".to_string()), - &Trap { - error: "exit pool failed on osmosis".to_string(), - step: IbcMsgKind::Ica(IcaMessages::ExitPool(pending)), - last_succesful: true, - }, - ) - .unwrap(); - - UNBONDING_CLAIMS - .save( - deps.as_mut().storage, - (Addr::unchecked("owner1"), "1".to_string()), - &Unbond { - lp_shares: Uint128::new(101), - unlock_time: Timestamp::from_seconds(1000), - attempted: true, - owner: Addr::unchecked("owner1"), - id: "1".to_string(), - }, - ) - .unwrap(); - - let res = execute_retry( - deps.as_mut(), - env, - MessageInfo { - sender: Addr::unchecked("not_admin"), - funds: vec![], - }, - 3539, - "channel-35".to_string(), - ); - - assert!(res.is_err()); - - // even though the unbonding claim for owner2 is missing, the retry will still add the unbonding claim for owner1 - assert_eq!(PENDING_UNBOND_QUEUE.len(&deps.storage).unwrap(), 1); - assert_eq!( - PENDING_UNBOND_QUEUE.back(&deps.storage).unwrap().unwrap(), - Unbond { - lp_shares: Uint128::new(101), - unlock_time: Timestamp::from_seconds(1000), - attempted: true, - owner: Addr::unchecked("owner1"), - id: "1".to_string(), - } - ); - } - - #[test] - fn test_handle_retry_exit_pool_with_wrong_seq_channel_fails() { - let mut deps = mock_dependencies(); - let env = mock_env(); - - let pending = PendingReturningUnbonds { - unbonds: vec![ - ReturningUnbond { - amount: RawAmount::LocalDenom(Uint128::new(101)), - owner: Addr::unchecked("owner1"), - id: "1".to_string(), - }, - ReturningUnbond { - amount: RawAmount::LocalDenom(Uint128::new(102)), - owner: Addr::unchecked("owner2"), - id: "2".to_string(), - }, - ], - }; - - TRAPS - .save( - deps.as_mut().storage, - (3539, "channel-35".to_string()), - &Trap { - error: "exit pool failed on osmosis".to_string(), - step: IbcMsgKind::Ica(IcaMessages::ExitPool(pending)), - last_succesful: true, - }, - ) - .unwrap(); - - let res = execute_retry( - deps.as_mut(), - env, - MessageInfo { - sender: Addr::unchecked("not_admin"), - funds: vec![], - }, - 0, - "random_channel".to_string(), - ); - - assert!(res.is_err()); - } - - #[test] - fn test_handle_retry_exit_pool_twice_fails() { - let mut deps = mock_dependencies(); - let env = mock_env(); - - let pending = PendingReturningUnbonds { - unbonds: vec![ - ReturningUnbond { - amount: RawAmount::LocalDenom(Uint128::new(101)), - owner: Addr::unchecked("owner1"), - id: "1".to_string(), - }, - ReturningUnbond { - amount: RawAmount::LocalDenom(Uint128::new(102)), - owner: Addr::unchecked("owner2"), - id: "2".to_string(), - }, - ReturningUnbond { - amount: RawAmount::LocalDenom(Uint128::new(103)), - owner: Addr::unchecked("owner3"), - id: "3".to_string(), - }, - ], - }; - - TRAPS - .save( - deps.as_mut().storage, - (3539, "channel-35".to_string()), - &Trap { - error: "exit pool failed on osmosis".to_string(), - step: IbcMsgKind::Ica(IcaMessages::ExitPool(pending)), - last_succesful: true, - }, - ) - .unwrap(); - - UNBONDING_CLAIMS - .save( - deps.as_mut().storage, - (Addr::unchecked("owner1"), "1".to_string()), - &Unbond { - lp_shares: Uint128::new(101), - unlock_time: Timestamp::from_seconds(1000), - attempted: true, - owner: Addr::unchecked("owner1"), - id: "1".to_string(), - }, - ) - .unwrap(); - - UNBONDING_CLAIMS - .save( - deps.as_mut().storage, - (Addr::unchecked("owner2"), "2".to_string()), - &Unbond { - lp_shares: Uint128::new(101), - unlock_time: Timestamp::from_seconds(1000), - attempted: true, - owner: Addr::unchecked("owner2"), - id: "2".to_string(), - }, - ) - .unwrap(); - - UNBONDING_CLAIMS - .save( - deps.as_mut().storage, - (Addr::unchecked("owner3"), "3".to_string()), - &Unbond { - lp_shares: Uint128::new(103), - unlock_time: Timestamp::from_seconds(1000), - attempted: true, - owner: Addr::unchecked("owner3"), - id: "3".to_string(), - }, - ) - .unwrap(); - - let res = execute_retry( - deps.as_mut(), - env.clone(), - MessageInfo { - sender: Addr::unchecked("not_admin"), - funds: vec![], - }, - 3539, - "channel-35".to_string(), - ) - .unwrap(); - - assert!(!TRAPS.has(&deps.storage, (3539, "channel-35".to_string()))); - - assert_eq!( - res.attributes, - vec![ - attr("unbond", "owner1"), - attr("unbond_id", "1"), - attr("unbond", "owner2"), - attr("unbond_id", "2"), - attr("unbond", "owner3"), - attr("unbond_id", "3"), - attr("action", "retry"), - attr("kind", "exit_pool"), - ] - ); - - assert_eq!(PENDING_UNBOND_QUEUE.len(&deps.storage).unwrap(), 3); - assert_eq!( - PENDING_UNBOND_QUEUE.back(&deps.storage).unwrap().unwrap(), - Unbond { - lp_shares: Uint128::new(103), - unlock_time: Timestamp::from_seconds(1000), - attempted: true, - owner: Addr::unchecked("owner3"), - id: "3".to_string(), - } - ); - - // execute retry for same seq & channel should fail - let res = execute_retry( - deps.as_mut(), - env, - MessageInfo { - sender: Addr::unchecked("not_admin"), - funds: vec![], - }, - 3539, - "channel-35".to_string(), - ); - - assert!(res.is_err()); - } - - #[test] - fn test_handle_retry_join_pool_works() { - let mut deps = mock_dependencies(); - let env = mock_env(); - default_setup(deps.as_mut().storage).unwrap(); - - IBC_LOCK.save(deps.as_mut().storage, &Lock::new()).unwrap(); - - // mock the failed join pool trap with 3 bonds - let failed = PendingBond { - bonds: vec![ - OngoingDeposit { - claim_amount: Uint128::new(100), - raw_amount: RawAmount::LocalDenom(Uint128::new(1000)), - owner: Addr::unchecked("address"), - bond_id: "1".to_string(), - }, - OngoingDeposit { - claim_amount: Uint128::new(99), - raw_amount: RawAmount::LocalDenom(Uint128::new(999)), - owner: Addr::unchecked("address"), - bond_id: "2".to_string(), - }, - OngoingDeposit { - claim_amount: Uint128::new(101), - raw_amount: RawAmount::LocalDenom(Uint128::new(1000)), - owner: Addr::unchecked("address"), - bond_id: "3".to_string(), - }, - ], - }; - - TRAPS - .save( - deps.as_mut().storage, - (3539, "channel-35".to_string()), - &Trap { - error: "join pool failed on osmosis".to_string(), - step: IbcMsgKind::Ica(IcaMessages::JoinSwapExternAmountIn(failed)), - last_succesful: true, - }, - ) - .unwrap(); - - // manually trigger retry join pool - let res = execute_retry( - deps.as_mut(), - env.clone(), - MessageInfo { - sender: Addr::unchecked("not_admin"), - funds: vec![], - }, - 3539, - "channel-35".to_string(), - ) - .unwrap(); - - assert!(!TRAPS.has(&deps.storage, (3539, "channel-35".to_string()))); - - assert_eq!( - res.attributes, - vec![ - attr("action", "retry"), - attr("kind", "join_pool"), - attr("bond_id", "1"), - attr("amount", Uint128::new(1000)), - attr("bond_id", "2"), - attr("amount", Uint128::new(999)), - attr("bond_id", "3"), - attr("amount", Uint128::new(1000)), - ] - ); - - // check that the failed join queue has the same mocked bonds - let failed_join_queue: Result, StdError> = - FAILED_JOIN_QUEUE.iter(&deps.storage).unwrap().collect(); - - let failed_bonds = vec![ - Bond { - amount: Uint128::new(1000), - owner: Addr::unchecked("address"), - bond_id: "1".to_string(), - }, - Bond { - amount: Uint128::new(999), - owner: Addr::unchecked("address"), - bond_id: "2".to_string(), - }, - Bond { - amount: Uint128::new(1000), - owner: Addr::unchecked("address"), - bond_id: "3".to_string(), - }, - ]; - - assert_eq!(failed_join_queue.as_ref().unwrap(), &failed_bonds); - - // manually trigger try_icq - let res = execute_try_icq(deps.as_mut(), env.clone()); - assert_eq!(res.unwrap().messages.len(), 1); - - // mocking the ICQ ACK - let raw_balance = create_query_response( - QueryBalanceResponse { - balance: Some(OsmoCoin { - denom: "uatom".to_string(), - amount: "1000".to_string(), - }), - } - .encode_to_vec(), - ); - - let quote_balance = create_query_response( - QueryBalanceResponse { - balance: Some(OsmoCoin { - denom: "uosmo".to_string(), - amount: "1000".to_string(), - }), - } - .encode_to_vec(), - ); - - let lp_balance = create_query_response( - QueryBalanceResponse { - balance: Some(OsmoCoin { - denom: "uosmo".to_string(), - amount: "1000".to_string(), - }), - } - .encode_to_vec(), - ); - - let join_pool = create_query_response( - QueryCalcJoinPoolSharesResponse { - share_out_amount: "123".to_string(), - tokens_out: vec![OsmoCoin { - denom: "uosmo".to_string(), - amount: "123".to_string(), - }], - } - .encode_to_vec(), - ); - - let exit_pool = create_query_response( - QueryCalcExitPoolCoinsFromSharesResponse { - tokens_out: vec![ - OsmoCoin { - // base denom - denom: "uosmo".to_string(), - amount: "123".to_string(), - }, - OsmoCoin { - // quote denom - denom: "uqsr".to_string(), - amount: "123".to_string(), - }, - ], - } - .encode_to_vec(), - ); - - let spot_price = create_query_response( - QuerySpotPriceResponse { - spot_price: "123".to_string(), - } - .encode_to_vec(), - ); - - let lock = create_query_response(LockedResponse { lock: None }.encode_to_vec()); - - let ibc_ack = InterchainQueryPacketAck { - data: Binary::from( - &CosmosResponse { - responses: vec![ - raw_balance, - quote_balance, - lp_balance, - exit_pool, - spot_price, - join_pool, - lock, - ], - } - .encode_to_vec()[..], - ), - }; - - let res = handle_icq_ack( - deps.as_mut().storage, - env, - to_json_binary(&ibc_ack).unwrap(), - ) - .unwrap(); - - // we do NOT transfer any token here, failed bonds were already transferred to the contract before failing and stay there - // as we do not have any bond_queue items, we return None here - assert!(res.messages.is_empty()); - - assert!(TRAPS.is_empty(deps.as_ref().storage)); - - // FAILED_JOIN_QUEUE should be empty and all bonds moved to REJOIN_QUEUE - assert_eq!(FAILED_JOIN_QUEUE.len(&deps.storage).unwrap(), 0); - assert_eq!(REJOIN_QUEUE.len(&deps.storage).unwrap(), 3); - - assert_eq!( - SIMULATED_JOIN_AMOUNT_IN - .load(deps.as_ref().storage) - .unwrap(), - Uint128::new(1000 + 999 + 1000) - ); - } - - #[test] - fn test_handle_retry_join_pool_with_pending_deposits_works() { - let mut deps = mock_dependencies(); - let env = mock_env(); - default_setup(deps.as_mut().storage).unwrap(); - - IBC_LOCK.save(deps.as_mut().storage, &Lock::new()).unwrap(); - - // mock the failed join pool trap with 3 bonds - let failed = PendingBond { - bonds: vec![ - OngoingDeposit { - claim_amount: Uint128::new(1000), - raw_amount: RawAmount::LocalDenom(Uint128::new(1000)), - owner: Addr::unchecked("address"), - bond_id: "1".to_string(), - }, - OngoingDeposit { - claim_amount: Uint128::new(999), - raw_amount: RawAmount::LocalDenom(Uint128::new(999)), - owner: Addr::unchecked("address"), - bond_id: "2".to_string(), - }, - OngoingDeposit { - claim_amount: Uint128::new(1000), - raw_amount: RawAmount::LocalDenom(Uint128::new(1000)), - owner: Addr::unchecked("address"), - bond_id: "3".to_string(), - }, - ], - }; - - TRAPS - .save( - deps.as_mut().storage, - (3539, "channel-35".to_string()), - &Trap { - error: "join pool failed on osmosis".to_string(), - step: IbcMsgKind::Ica(IcaMessages::JoinSwapExternAmountIn(failed.clone())), - last_succesful: true, - }, - ) - .unwrap(); - - // mock pending deposits and add them to the pending queue - let pending_bonds = vec![ - Bond { - amount: Uint128::new(5_000), - owner: Addr::unchecked("address"), - bond_id: "1".to_string(), - }, - Bond { - amount: Uint128::new(10_000), - owner: Addr::unchecked("address"), - bond_id: "2".to_string(), - }, - ]; - - for bond in pending_bonds.iter() { - PENDING_BOND_QUEUE - .push_back(deps.as_mut().storage, bond) - .unwrap(); - } - - // manually trigger retry join pool - let res = execute_retry( - deps.as_mut(), - env.clone(), - MessageInfo { - sender: Addr::unchecked("not_admin"), - funds: vec![], - }, - 3539, - "channel-35".to_string(), - ) - .unwrap(); - - assert!(!TRAPS.has(&deps.storage, (3539, "channel-35".to_string()))); - - // failed bonds amount should not be included in the response as funds are already - // in Osmosis - assert_eq!( - res.attributes, - vec![ - attr("action", "retry"), - attr("kind", "join_pool"), - attr("bond_id", "1"), - attr("amount", Uint128::new(1000)), - attr("bond_id", "2"), - attr("amount", Uint128::new(999)), - attr("bond_id", "3"), - attr("amount", Uint128::new(1000)), - ] - ); - - // check that the failed join queue has the same mocked bonds - let failed_join_queue: Result, StdError> = - FAILED_JOIN_QUEUE.iter(&deps.storage).unwrap().collect(); - assert_eq!(failed_join_queue.unwrap(), pending_bond_to_bond(&failed)); - - // manually trigger try_icq - let res = execute_try_icq(deps.as_mut(), env.clone()); - assert_eq!(res.unwrap().messages.len(), 1); - - // mocking the ICQ ACK - let raw_balance = create_query_response( - QueryBalanceResponse { - balance: Some(OsmoCoin { - denom: "uatom".to_string(), - amount: "1000".to_string(), - }), - } - .encode_to_vec(), - ); - - let quote_balance = create_query_response( - QueryBalanceResponse { - balance: Some(OsmoCoin { - denom: "uosmo".to_string(), - amount: "1000".to_string(), - }), - } - .encode_to_vec(), - ); - - let lp_balance = create_query_response( - QueryBalanceResponse { - balance: Some(OsmoCoin { - denom: "uosmo".to_string(), - amount: "1000".to_string(), - }), - } - .encode_to_vec(), - ); - - let join_pool = create_query_response( - QueryCalcJoinPoolSharesResponse { - share_out_amount: "123".to_string(), - tokens_out: vec![OsmoCoin { - denom: "uosmo".to_string(), - amount: "123".to_string(), - }], - } - .encode_to_vec(), - ); - - let exit_pool = create_query_response( - QueryCalcExitPoolCoinsFromSharesResponse { - tokens_out: vec![ - OsmoCoin { - // base denom - denom: "uosmo".to_string(), - amount: "123".to_string(), - }, - OsmoCoin { - // quote denom - denom: "uqsr".to_string(), - amount: "123".to_string(), - }, - ], - } - .encode_to_vec(), - ); - - let spot_price = create_query_response( - QuerySpotPriceResponse { - spot_price: "123".to_string(), - } - .encode_to_vec(), - ); - - let lock = create_query_response(LockedResponse { lock: None }.encode_to_vec()); - - let ibc_ack = InterchainQueryPacketAck { - data: Binary::from( - &CosmosResponse { - responses: vec![ - raw_balance, - quote_balance, - lp_balance, - exit_pool, - spot_price, - join_pool, - lock, - ], - } - .encode_to_vec()[..], - ), - }; - - let res = handle_icq_ack( - deps.as_mut().storage, - env, - to_json_binary(&ibc_ack).unwrap(), - ) - .unwrap(); - - // get the pending bonds total amount - let pending_total_amount = pending_bonds - .iter() - .fold(Uint128::zero(), |acc, bond| acc + bond.amount); - - // check that the res amount matches the amount in the pending queue ONLY - match &res.messages[0].msg { - CosmosMsg::Ibc(IbcMsg::Transfer { amount, .. }) => { - assert_eq!( - amount, - &Coin { - denom: "ibc/local_osmo".to_string(), - amount: pending_total_amount, - } - ); - } - _ => panic!("unexpected message type"), - } - - // check that the failed join & pending queues are emptied - assert!(FAILED_JOIN_QUEUE.is_empty(&deps.storage).unwrap()); - assert!(PENDING_BOND_QUEUE.is_empty(&deps.storage).unwrap()); - - // failed bonds should be now in the REJOIN_QUEUE - let rejoin_queue: Result, StdError> = - REJOIN_QUEUE.iter(&deps.storage).unwrap().collect(); - assert_eq!(failed.bonds, rejoin_queue.unwrap()); - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/helpers.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/helpers.rs deleted file mode 100644 index 3990b9041..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/helpers.rs +++ /dev/null @@ -1,365 +0,0 @@ -use crate::{ - error::ContractError, - error_recovery::PendingReturningRecovery, - ibc_lock::Lock, - msg::ExecuteMsg, - state::{ - PendingBond, PendingSingleUnbond, RawAmount, BOND_QUEUE, CHANNELS, CONFIG, - FAILED_JOIN_QUEUE, IBC_LOCK, REJOIN_QUEUE, REPLIES, SHARES, START_UNBOND_QUEUE, TRAPS, - UNBOND_QUEUE, - }, - unbond::PendingReturningUnbonds, -}; -use cosmwasm_std::{ - from_json, to_json_binary, Addr, BankMsg, Binary, CosmosMsg, DepsMut, Env, IbcMsg, - IbcPacketAckMsg, Order, QuerierWrapper, Response, StdError, Storage, SubMsg, Uint128, WasmMsg, -}; -use prost::Message; -use quasar_types::{callback::Callback, ibc::MsgTransferResponse}; -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -pub fn get_total_primitive_shares(storage: &dyn Storage) -> Result { - let mut sum = Uint128::zero(); - for val in SHARES.range(storage, None, None, Order::Ascending) { - sum = sum.checked_add(val?.1)?; - } - Ok(sum) -} - -pub fn get_ica_address(store: &dyn Storage, channel_id: String) -> Result { - let chan = CHANNELS.load(store, channel_id)?; - match chan.channel_type { - quasar_types::ibc::ChannelType::Icq { channel_ty: _ } => Err(ContractError::NoIcaChannel), - quasar_types::ibc::ChannelType::Ica { - channel_ty: _, - counter_party_address, - } => { - if let Some(addr) = counter_party_address { - Ok(addr) - } else { - Err(ContractError::NoCounterpartyIcaAddress) - } - } - quasar_types::ibc::ChannelType::Ics20 { channel_ty: _ } => Err(ContractError::NoIcaChannel), - } -} - -pub fn check_icq_channel(storage: &dyn Storage, channel: String) -> Result<(), ContractError> { - let chan = CHANNELS.load(storage, channel)?; - match chan.channel_type { - quasar_types::ibc::ChannelType::Icq { channel_ty: _ } => Ok(()), - quasar_types::ibc::ChannelType::Ica { - channel_ty: _, - counter_party_address: _, - } => Err(ContractError::NoIcqChannel), - quasar_types::ibc::ChannelType::Ics20 { channel_ty: _ } => Err(ContractError::NoIcqChannel), - } -} - -pub fn create_callback_submsg( - storage: &mut dyn Storage, - cosmos_msg: CosmosMsg, - owner: Addr, - callback_id: String, -) -> Result { - let last = REPLIES.range(storage, None, None, Order::Descending).next(); - let mut id: u64 = 0; - if let Some(val) = last { - id = val?.0 + 1; - } - - let local_denom = CONFIG.load(storage)?.local_denom; - let data: SubMsgKind = match &cosmos_msg { - CosmosMsg::Wasm(WasmMsg::Execute { msg, funds, .. }) => { - SubMsgKind::Callback(ContractCallback::Callback { - callback: from_json(msg)?, - // if we send funds, we expect them to be in local denom - amount: funds - .iter() - .find(|c| c.denom == local_denom) - .map(|val| val.amount), - owner, - }) - } - CosmosMsg::Bank(bank_msg) => SubMsgKind::Callback(ContractCallback::Bank { - bank_msg: bank_msg.to_owned(), - unbond_id: callback_id, - }), - _ => return Err(StdError::generic_err("Unsupported WasmMsg")), - }; - - REPLIES.save(storage, id, &data)?; - Ok(SubMsg::reply_always(cosmos_msg, id)) -} - -pub(crate) fn lock_try_icq( - deps: DepsMut, - sub_msg: Option, -) -> Result { - let mut res = Response::new(); - let mut lock = IBC_LOCK.load(deps.storage)?; - - if let Some(sub_msg) = sub_msg { - if !BOND_QUEUE.is_empty(deps.storage)? { - lock = lock.lock_bond(); - res = res.add_attribute("bond_queue", "locked"); - } else if !START_UNBOND_QUEUE.is_empty(deps.storage)? { - lock = lock.lock_start_unbond(); - res = res.add_attribute("start_unbond_queue", "locked"); - } else if !UNBOND_QUEUE.is_empty(deps.storage)? { - lock = lock.lock_unbond(); - res = res.add_attribute("unbond_queue", "locked"); - } - if lock.is_unlocked() { - res = res.add_attribute("IBC_LOCK", "unlocked"); - } - IBC_LOCK.save(deps.storage, &lock)?; - res = res.add_submessage(sub_msg); - res = res.add_attribute("kind", "dispatch"); - } else { - res = res.add_attribute("IBC_LOCK", "locked"); - res = res.add_attribute("kind", "queue"); - } - Ok(res) -} - -pub fn get_usable_compound_balance( - storage: &dyn Storage, - balance: Uint128, -) -> Result { - // two cases where we exclude funds, either transfer succeeded, but not ica, or transfer succeeded and subsequent ica failed - let traps = TRAPS.range(storage, None, None, Order::Ascending); - - let failed_join_queue_amount = FAILED_JOIN_QUEUE.iter(storage)?.try_fold( - Uint128::zero(), - |acc, val| -> Result { Ok(acc + val?.amount) }, - )?; - - let rejoin_queue_amount = REJOIN_QUEUE.iter(storage)?.try_fold( - Uint128::zero(), - |acc, val| -> Result { - match val?.raw_amount { - crate::state::RawAmount::LocalDenom(amount) => Ok(amount + acc), - crate::state::RawAmount::LpShares(_) => Err(ContractError::IncorrectRawAmount), - } - }, - )?; - - let trapped_errors_amount = traps.fold(Uint128::zero(), |acc, wrapped_trap| { - let trap = wrapped_trap.unwrap().1; - if trap.last_succesful { - if let IbcMsgKind::Transfer { pending: _, amount } = trap.step { - acc + amount - } else { - acc - } - // if last msg was not succesful, we did not join the pool, so we have base_denom funds on the - } else if let IbcMsgKind::Ica(IcaMessages::JoinSwapExternAmountIn(pb)) = trap.step { - pb.bonds.iter().fold(acc, |acc2, bond| { - if let RawAmount::LocalDenom(local_denom_amount) = &bond.raw_amount { - acc2 + local_denom_amount - } else { - acc2 - } - }) - } else { - acc - } - }); - - let excluded_funds = failed_join_queue_amount - .checked_add(rejoin_queue_amount)? - .checked_add(trapped_errors_amount)?; - - Ok(balance.saturating_sub(excluded_funds)) -} - -pub fn create_ibc_ack_submsg( - storage: &mut dyn Storage, - pending: IbcMsgKind, - msg: IbcMsg, - channel: String, -) -> Result { - let last = REPLIES.range(storage, None, None, Order::Descending).next(); - let mut id: u64 = 0; - if let Some(val) = last { - id = val?.0 + 1; - } - // register the message in the replies for handling - REPLIES.save(storage, id, &SubMsgKind::Ibc(pending, channel))?; - Ok(SubMsg::reply_always(msg, id)) -} - -pub fn ack_submsg( - storage: &mut dyn Storage, - env: Env, - msg: IbcPacketAckMsg, - channel: String, -) -> Result { - let last = REPLIES.range(storage, None, None, Order::Descending).next(); - let mut id: u64 = 0; - if let Some(val) = last { - id = val?.0 + 1; - } - - // register the message in the replies for handling - // TODO do we need this state item here? or do we just need the reply hook - REPLIES.save( - storage, - id, - &SubMsgKind::Ack(msg.original_packet.sequence, channel), - )?; - - // TODO for an ack, should the reply hook be always or only on error? Probably only on error - // On succeses, we need to cleanup the state item from REPLIES - Ok(SubMsg::reply_always( - WasmMsg::Execute { - contract_addr: env.contract.address.to_string(), - msg: to_json_binary(&ExecuteMsg::Ack { ack: msg })?, - funds: vec![], - }, - id, - )) -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug, Eq)] -#[serde(rename_all = "snake_case")] -pub enum IbcMsgKind { - Transfer { - pending: PendingBond, - amount: Uint128, - }, - Ica(IcaMessages), - Icq, -} - -// All enums supported by this contract -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug, Eq)] -#[serde(rename_all = "snake_case")] -pub enum IcaMessages { - JoinSwapExternAmountIn(PendingBond), - // pending bonds int the lock and total shares to be locked - // should be gotten from the join pool - LockTokens(PendingBond, Uint128), - BeginUnlocking(Vec, Uint128), - ExitPool(PendingReturningUnbonds), - ReturnTransfer(PendingReturningUnbonds), - RecoveryExitPool(PendingReturningRecovery), - RecoveryReturnTransfer(PendingReturningRecovery), -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub enum SubMsgKind { - Ibc(IbcMsgKind, String), - Ack(u64, String), - Callback(ContractCallback), // in reply match for callback variant -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub enum ContractCallback { - Callback { - callback: Callback, - amount: Option, - owner: Addr, - }, - Bank { - bank_msg: BankMsg, - unbond_id: String, - }, -} - -pub fn is_contract_admin( - querier: &QuerierWrapper, - env: &Env, - sus_admin: &Addr, -) -> Result<(), ContractError> { - let contract_admin = querier - .query_wasm_contract_info(&env.contract.address)? - .admin; - if let Some(contract_admin) = contract_admin { - if contract_admin != *sus_admin { - return Err(ContractError::Unauthorized {}); - } - } else { - return Err(ContractError::Unauthorized {}); - } - Ok(()) -} - -pub(crate) fn parse_seq(data: Binary) -> Result { - let resp = MsgTransferResponse::decode(data.0.as_slice())?; - Ok(resp.seq) -} - -pub(crate) fn unlock_on_error( - storage: &mut dyn Storage, - kind: &IbcMsgKind, -) -> Result<(), ContractError> { - match kind { - IbcMsgKind::Transfer { - pending: _, - amount: _, - } => { - IBC_LOCK.update(storage, |lock| { - Ok::(lock.unlock_bond()) - })?; - Ok(()) - } - IbcMsgKind::Ica(ica) => match ica { - IcaMessages::JoinSwapExternAmountIn(_) => { - IBC_LOCK.update(storage, |lock| { - Ok::(lock.unlock_bond()) - })?; - Ok(()) - } - IcaMessages::LockTokens(_, _) => { - IBC_LOCK.update(storage, |lock| { - Ok::(lock.unlock_bond()) - })?; - Ok(()) - } - IcaMessages::BeginUnlocking(_, _) => { - IBC_LOCK.update(storage, |lock| { - Ok::(lock.unlock_start_unbond()) - })?; - Ok(()) - } - IcaMessages::ExitPool(_) => { - IBC_LOCK.update(storage, |lock| { - Ok::(lock.unlock_unbond()) - })?; - Ok(()) - } - IcaMessages::ReturnTransfer(_) => { - IBC_LOCK.update(storage, |lock| { - Ok::(lock.unlock_unbond()) - })?; - Ok(()) - } - IcaMessages::RecoveryExitPool(_) => todo!(), - IcaMessages::RecoveryReturnTransfer(_) => todo!(), - }, - IbcMsgKind::Icq => { - IBC_LOCK.update(storage, |lock| { - Ok::(lock.unlock_bond().unlock_start_unbond().unlock_unbond()) - })?; - Ok(()) - } - } -} - -#[cfg(test)] -mod tests { - use super::*; - - #[test] - fn parse_reply_seq() { - let seq = 35; - let resp = Binary::from(MsgTransferResponse { seq }.encode_to_vec()); - let parsed_seq = parse_seq(resp).unwrap(); - assert_eq!(seq, parsed_seq) - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/ibc.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/ibc.rs deleted file mode 100644 index 4ab7e6a4a..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/ibc.rs +++ /dev/null @@ -1,1179 +0,0 @@ -use crate::bond::{batch_bond, create_share}; -use crate::error::{ContractError, Never, Trap}; - -use crate::helpers::{ - ack_submsg, create_callback_submsg, create_ibc_ack_submsg, get_ica_address, - get_usable_compound_balance, unlock_on_error, IbcMsgKind, IcaMessages, -}; -use crate::ibc_lock::Lock; -use crate::ibc_util::{ - calculate_share_out_min_amount, consolidate_exit_pool_amount_into_local_denom, - do_ibc_join_pool_swap_extern_amount_in, do_ibc_lock_tokens, parse_join_pool, -}; -use crate::icq::calc_total_balance; -use crate::start_unbond::{batch_start_unbond, handle_start_unbond_ack}; -use crate::state::{ - LpCache, PendingBond, CHANNELS, CONFIG, IBC_LOCK, IBC_TIMEOUT_TIME, ICA_CHANNEL, ICQ_CHANNEL, - LP_SHARES, OSMO_LOCK, PENDING_ACK, RECOVERY_ACK, REJOIN_QUEUE, SIMULATED_EXIT_RESULT, - SIMULATED_EXIT_SHARES_IN, SIMULATED_JOIN_AMOUNT_IN, SIMULATED_JOIN_RESULT, TIMED_OUT, - TOTAL_VAULT_BALANCE, TRAPS, USABLE_COMPOUND_BALANCE, -}; -use crate::unbond::{batch_unbond, transfer_batch_unbond, PendingReturningUnbonds}; -use cosmos_sdk_proto::cosmos::bank::v1beta1::QueryBalanceResponse; - -#[cfg(not(feature = "library"))] -use cosmwasm_std::entry_point; -use osmosis_std::types::cosmos::base::v1beta1::Coin as OsmoCoin; -#[allow(deprecated)] -use osmosis_std::types::osmosis::gamm::v1beta1::QuerySpotPriceResponse; - -use osmosis_std::types::osmosis::gamm::v1beta1::{ - MsgExitSwapShareAmountInResponse, MsgJoinSwapExternAmountInResponse, - QueryCalcExitPoolCoinsFromSharesResponse, QueryCalcJoinPoolSharesResponse, -}; -use osmosis_std::types::osmosis::lockup::{LockedResponse, MsgLockTokensResponse}; -use prost::Message; -use quasar_types::callback::{BondResponse, Callback}; -use quasar_types::error::Error as QError; -use quasar_types::ibc::{enforce_order_and_version, ChannelInfo, ChannelType, HandshakeState}; -use quasar_types::ica::handshake::enforce_ica_order_and_metadata; -use quasar_types::ica::packet::{ica_send, AckBody}; -use quasar_types::ica::traits::Unpack; -use quasar_types::icq::{CosmosResponse, InterchainQueryPacketAck, ICQ_ORDERING}; -use quasar_types::{ibc, ica::handshake::IcaMetadata, icq::ICQ_VERSION}; -use std::str::FromStr; - -use cosmwasm_std::{ - from_json, to_json_binary, Attribute, Binary, Coin, CosmosMsg, Decimal, DepsMut, Env, - IbcBasicResponse, IbcChannel, IbcChannelCloseMsg, IbcChannelConnectMsg, IbcChannelOpenMsg, - IbcPacketAckMsg, IbcPacketReceiveMsg, IbcPacketTimeoutMsg, IbcReceiveResponse, IbcTimeout, - QuerierWrapper, Response, StdError, Storage, SubMsg, Uint128, WasmMsg, -}; - -/// enforces ordering and versioning constraints, this combines ChanOpenInit and ChanOpenTry -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn ibc_channel_open( - deps: DepsMut, - _env: Env, - msg: IbcChannelOpenMsg, -) -> Result<(), ContractError> { - // save the channel as an channel in ChanOpenInit, we support inits from icq and ica channels - if msg.channel().version == ICQ_VERSION { - handle_icq_channel(deps, msg.channel().clone())?; - } else { - handle_ica_channel(deps, msg)?; - } - Ok(()) -} - -fn handle_icq_channel(deps: DepsMut, channel: IbcChannel) -> Result<(), ContractError> { - ibc::enforce_order_and_version(&channel, None, &channel.version, channel.order.clone())?; - - // check the connection id vs the expected connection id - let config = CONFIG.load(deps.storage)?; - if config.expected_connection != channel.connection_id { - return Err(ContractError::IncorrectConnection); - } - - // save the channel state here - let info = ChannelInfo { - id: channel.endpoint.channel_id.clone(), - counterparty_endpoint: channel.counterparty_endpoint, - connection_id: channel.connection_id, - channel_type: ChannelType::Icq { - channel_ty: channel.version, - }, - handshake_state: HandshakeState::Init, - }; - - CHANNELS.save(deps.storage, channel.endpoint.channel_id, &info)?; - Ok(()) -} - -fn handle_ica_channel(deps: DepsMut, msg: IbcChannelOpenMsg) -> Result<(), ContractError> { - let channel = msg.channel().clone(); - let metadata: IcaMetadata = serde_json_wasm::from_str(&channel.version).map_err(|error| { - QError::InvalidIcaMetadata { - raw_metadata: channel.version.clone(), - error: error.to_string(), - } - })?; - - enforce_ica_order_and_metadata(&channel, None, &metadata)?; - - // compare the expected connection id to the channel connection-id and the ica metadata connection-id - let config = CONFIG.load(deps.storage)?; - if &config.expected_connection - != metadata - .controller_connection_id() - .as_ref() - .ok_or(ContractError::NoConnectionFound)? - { - return Err(ContractError::IncorrectConnection); - } - if config.expected_connection != channel.connection_id { - return Err(ContractError::IncorrectConnection); - } - // validate that the message is an OpenInit message and not an OpenTry, such that we don't pollute the channel map - // if let IbcChannelOpenMsg::OpenInit(s) = msg { - // return Err(ContractError::InvalidOrder); - // } - match msg { - IbcChannelOpenMsg::OpenInit { channel } => { - // save the current state of the initializing channel - let info = ChannelInfo { - id: channel.endpoint.channel_id.clone(), - counterparty_endpoint: channel.counterparty_endpoint, - connection_id: channel.connection_id, - channel_type: ChannelType::Ica { - channel_ty: metadata, - counter_party_address: None, - }, - handshake_state: HandshakeState::Init, - }; - CHANNELS.save(deps.storage, channel.endpoint.channel_id, &info)?; - Ok(()) - } - IbcChannelOpenMsg::OpenTry { - channel: _, - counterparty_version: _, - } => Err(ContractError::IncorrectChannelOpenType), - } -} - -/// record the channel in CHANNEL_INFO, this combines the ChanOpenAck and ChanOpenConfirm steps -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn ibc_channel_connect( - deps: DepsMut, - _env: Env, - msg: IbcChannelConnectMsg, -) -> Result { - // try to fetch the connecting channel, we should error if it does not exist\ - let mut info: ChannelInfo = CHANNELS - .load(deps.storage, msg.channel().endpoint.channel_id.clone()) - .map_err(|err| StdError::GenericErr { - msg: err.to_string(), - })?; - // we need to check the counter party version in try and ack (sometimes here) - // TODO we can wrap this match in a function in our ibc package - - // TODO think of a better datastructure so we dont have to parse ICA channels like this - match info.channel_type { - ChannelType::Icq { ref channel_ty } => { - enforce_order_and_version( - msg.channel(), - msg.counterparty_version(), - channel_ty.as_str(), - ICQ_ORDERING, - )?; - ICQ_CHANNEL.save(deps.storage, &msg.channel().endpoint.channel_id)?; - // TODO save the updated state of the ICQ channel - } - ChannelType::Ica { - channel_ty, - counter_party_address: _, - } => { - let counter_party_metadata = enforce_ica_order_and_metadata( - msg.channel(), - msg.counterparty_version(), - &channel_ty, - )?; - - if counter_party_metadata.is_none() { - return Err(ContractError::QError(QError::NoCounterpartyIcaAddress)); - } - let counter_party = counter_party_metadata.unwrap(); - // at this point, we expect a counterparty address, if it's none, we have to error - if counter_party.address().is_none() { - return Err(ContractError::NoCounterpartyIcaAddress); - } - let addr = counter_party.address(); - if addr.is_none() { - return Err(ContractError::NoCounterpartyIcaAddress); - } - - // once we have an Open ICA channel, save it under ICA channel, - // if a channel already exists, and that channel is not timed out reject incoming OPENS - // if that channel is timed out, we overwrite the previous ICA channel for the new one - let channel = ICA_CHANNEL.may_load(deps.storage)?; - // to reject the msg here, ica should not be timed out - if channel.is_some() && !TIMED_OUT.load(deps.storage)? { - return Err(ContractError::IcaChannelAlreadySet); - } - - // set timed out to false - TIMED_OUT.save(deps.storage, &false)?; - - ICA_CHANNEL.save(deps.storage, &msg.channel().endpoint.channel_id)?; - - info.channel_type = ChannelType::Ica { - channel_ty, - counter_party_address: addr, - }; - CHANNELS.save(deps.storage, info.id.clone(), &info)? - } - ChannelType::Ics20 { channel_ty: _ } => unimplemented!(), - } - - info.handshake_state = HandshakeState::Open; - - CHANNELS.save( - deps.storage, - msg.channel().endpoint.channel_id.clone(), - &info, - )?; - - Ok(IbcBasicResponse::default()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn ibc_channel_close( - _deps: DepsMut, - _env: Env, - channel: IbcChannelCloseMsg, -) -> Result { - // TODO look up the channel, in channels, and update the state to closed - Ok(IbcBasicResponse::new() - .add_attribute("channel", channel.channel().endpoint.channel_id.clone()) - .add_attribute("connection", channel.channel().connection_id.clone())) -} - -/// The lp-strategy cannot receive any packets -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn ibc_packet_receive( - _deps: DepsMut, - _env: Env, - _msg: IbcPacketReceiveMsg, -) -> Result { - // Contract does not handle packets/queries. - unimplemented!(); -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn ibc_packet_ack( - deps: DepsMut, - env: Env, - msg: IbcPacketAckMsg, -) -> Result { - // We save the ack binary here for error recovery in case of an join pool recovery - // this should be cleaned up from state in the ack submsg Ok case - RECOVERY_ACK.save( - deps.storage, - ( - msg.original_packet.sequence, - msg.original_packet.src.channel_id.clone(), - ), - &msg.acknowledgement, - )?; - let chan = msg.original_packet.src.channel_id.clone(); - Ok(IbcBasicResponse::new().add_submessage(ack_submsg(deps.storage, env, msg, chan)?)) -} - -pub fn handle_succesful_ack( - deps: DepsMut, - env: Env, - pkt: IbcPacketAckMsg, - ack_bin: Binary, -) -> Result { - let kind = PENDING_ACK.load( - deps.storage, - ( - pkt.original_packet.sequence, - pkt.original_packet.src.channel_id.clone(), - ), - )?; - match kind { - // a transfer ack means we have sent funds to the ica address, return transfers are handled by the ICA ack - IbcMsgKind::Transfer { pending, amount } => { - handle_transfer_ack(deps.storage, env, ack_bin, &pkt, pending, amount) - } - IbcMsgKind::Ica(ica_kind) => { - handle_ica_ack(deps.storage, deps.querier, env, ack_bin, &pkt, ica_kind) - } - IbcMsgKind::Icq => handle_icq_ack(deps.storage, env, ack_bin), - } -} - -pub fn handle_transfer_ack( - storage: &mut dyn Storage, - env: Env, - _ack_bin: Binary, - _pkt: &IbcPacketAckMsg, - mut pending: PendingBond, - transferred_amount: Uint128, -) -> Result { - // once the ibc transfer to the ICA account has succeeded, we send the join pool message - // we need to save and fetch - let config = CONFIG.load(storage)?; - - let share_out_min_amount = calculate_share_out_min_amount(storage)?; - - let rejoin_queue_amount = REJOIN_QUEUE.iter(storage)?.try_fold( - Uint128::zero(), - |acc, val| -> Result { - match val?.raw_amount { - crate::state::RawAmount::LocalDenom(amount) => Ok(amount + acc), - crate::state::RawAmount::LpShares(_) => Err(ContractError::IncorrectRawAmount), - } - }, - )?; - - // add in usable base token compound balance to include rewards - // TODO: In an ideal world, I'd also fix share_out_min_amount to adjust for the amount we are compounding - // However, share_out_min_amount is already broken (doens't include failed_bonds_amount), the fix would live in ibc_util.rs L:78 (this would only fix failed_join_amount) - // a better fix would be to get the last state of the pool and do the math on our side to then also be able to include this USABLE_COMPOUND_BALANCE - // howver, if we are going to deprecate this vault, I'd argue it's not worth it - seeing as the compound balance would just be "extra money" for a user anyway - // TODO: remove this comment - let base_token_rewards = USABLE_COMPOUND_BALANCE.load(storage)?; - - let total_amount = transferred_amount + rejoin_queue_amount + base_token_rewards; - - // remove all items from REJOIN_QUEUE & add them to the deposits: Vec - while !REJOIN_QUEUE.is_empty(storage)? { - let rejoin = REJOIN_QUEUE.pop_front(storage)?; - pending.bonds.push(rejoin.unwrap()); - } - - let msg = do_ibc_join_pool_swap_extern_amount_in( - storage, - env, - config.pool_id, - config.base_denom.clone(), - total_amount, - share_out_min_amount, - pending.bonds, - )?; - - // TODO move this update to after the lock - TOTAL_VAULT_BALANCE.update(storage, |old| -> Result { - Ok(old.checked_add(total_amount)?) - })?; - - Ok(Response::new().add_submessage(msg).add_attribute( - "transfer-ack", - format!("{}-{}", &total_amount, config.base_denom), - )) -} - -// TODO move the parsing of the ICQ to it's own function, ideally we'd have a type that is contstructed in create ICQ and is parsed from a proto here -pub fn handle_icq_ack( - storage: &mut dyn Storage, - env: Env, - ack_bin: Binary, -) -> Result { - // todo: query flows should be separated by which flowType we're doing (bond, unbond, startunbond) - - let ack: InterchainQueryPacketAck = from_json(ack_bin)?; - let resp: CosmosResponse = CosmosResponse::decode(ack.data.0.as_ref())?; - - // we have only dispatched on query and a single kind at this point - let raw_balance = QueryBalanceResponse::decode(resp.responses[0].value.as_ref())? - .balance - .ok_or(ContractError::BaseDenomNotFound)? - .amount; - - let base_balance = - Uint128::new( - raw_balance - .parse::() - .map_err(|err| ContractError::ParseIntError { - error: format!("base_balance:{err}"), - value: raw_balance.to_string(), - })?, - ); - - // free base_token osmo side balance but subtracted out anything from trapped_errors, saved for use in transfer ack - let usable_base_token_compound_balance = get_usable_compound_balance(storage, base_balance)?; - USABLE_COMPOUND_BALANCE.save(storage, &usable_base_token_compound_balance)?; - - // TODO the quote balance should be able to be compounded aswell - let _quote_balance = QueryBalanceResponse::decode(resp.responses[1].value.as_ref())? - .balance - .ok_or(ContractError::BaseDenomNotFound)? - .amount; - - // TODO we can make the LP_SHARES cache less error prone here by using the actual state of lp shares - // We then need to query locked shares aswell, since they are not part of balance - let _lp_balance = QueryBalanceResponse::decode(resp.responses[2].value.as_ref())? - .balance - .ok_or(ContractError::BaseDenomNotFound)? - .amount; - - let exit_total_pool = - QueryCalcExitPoolCoinsFromSharesResponse::decode(resp.responses[3].value.as_ref())?; - - #[allow(deprecated)] - let spot_price = QuerySpotPriceResponse::decode(resp.responses[4].value.as_ref())?.spot_price; - - let mut response_idx = 4; - let join_pool = if SIMULATED_JOIN_AMOUNT_IN - .may_load(storage)? - .unwrap_or(0u128.into()) - > 1u128.into() - { - // found, increment response index - response_idx += 1; - // decode result - QueryCalcJoinPoolSharesResponse::decode(resp.responses[response_idx].value.as_ref())? - } else { - QueryCalcJoinPoolSharesResponse { - share_out_amount: "0".to_string(), - tokens_out: vec![], - } - }; - - let config = CONFIG.load(storage)?; - - let locked_lp_shares = match OSMO_LOCK.may_load(storage)? { - Some(_) => { - // found, increment response index - response_idx += 1; - // decode result - let lock = LockedResponse::decode(resp.responses[response_idx].value.as_ref())?.lock; - // parse the locked lp shares on Osmosis, a bit messy - let gamms = if let Some(lock) = lock { - lock.coins - } else { - vec![] - }; - gamms - .into_iter() - .find(|val| val.denom == config.pool_denom) - .unwrap_or(OsmoCoin { - denom: config.pool_denom.clone(), - amount: Uint128::zero().to_string(), - }) - .amount - .parse()? - } - None => Uint128::zero(), - }; - - // update the locked shares in our cache - LP_SHARES.update(storage, |mut cache| -> Result { - cache.locked_shares = locked_lp_shares; - Ok(cache) - })?; - - let exit_pool_unbonds = if SIMULATED_EXIT_SHARES_IN - .may_load(storage)? - .unwrap_or(0u128.into()) - >= 1u128.into() - { - // found, increment response index - response_idx += 1; - - // decode result - QueryCalcExitPoolCoinsFromSharesResponse::decode( - resp.responses[response_idx].value.as_ref(), - )? - } else { - QueryCalcExitPoolCoinsFromSharesResponse { tokens_out: vec![] } - }; - - let spot_price = - Decimal::from_str(spot_price.as_str()).map_err(|err| ContractError::ParseDecError { - error: err, - value: spot_price, - })?; - - let total_balance = calc_total_balance( - storage, - usable_base_token_compound_balance, - &exit_total_pool.tokens_out, - spot_price, - )?; - - TOTAL_VAULT_BALANCE.save(storage, &total_balance)?; - - let parsed_exit_pool_out = consolidate_exit_pool_amount_into_local_denom( - storage, - &exit_pool_unbonds.tokens_out, - spot_price, - )?; - - let parsed_join_pool_out = parse_join_pool(storage, join_pool)?; - - SIMULATED_JOIN_RESULT.save(storage, &parsed_join_pool_out)?; - SIMULATED_EXIT_RESULT.save(storage, &parsed_exit_pool_out)?; - - // todo move this to below into the lock decisions - let bond: Option = batch_bond(storage, &env, total_balance)?; - - let mut msges = Vec::new(); - let mut attrs = Vec::new(); - // if queues had items, msges should be some, so we add the ibc submessage, if there were no items in a queue, we don't have a submsg to add - // if we have a bond, start_unbond or unbond msg, we lock the repsective lock - - // todo rewrite into flat if/else ifs - if let Some(msg) = bond { - msges.push(msg); - attrs.push(Attribute::new("bond-status", "bonding")); - IBC_LOCK.update(storage, |lock| -> Result { - Ok(lock.lock_bond()) - })?; - } else { - attrs.push(Attribute::new("bond-status", "empty")); - if let Some(msg) = batch_start_unbond(storage, &env)? { - msges.push(msg); - attrs.push(Attribute::new("start-unbond-status", "starting-unbond")); - IBC_LOCK.update(storage, |lock| -> Result { - Ok(lock.lock_start_unbond()) - })?; - } else { - attrs.push(Attribute::new("start-unbond-status", "empty")); - if let Some(msg) = batch_unbond(storage, &env)? { - msges.push(msg); - attrs.push(Attribute::new("unbond-status", "unbonding")); - IBC_LOCK.update(storage, |lock| -> Result { - Ok(lock.lock_unbond()) - })?; - } else { - attrs.push(Attribute::new("unbond-status", "empty")); - } - } - } - - Ok(Response::new().add_submessages(msges).add_attributes(attrs)) -} - -pub fn handle_ica_ack( - storage: &mut dyn Storage, - querier: QuerierWrapper, - env: Env, - ack_bin: Binary, - _pkt: &IbcPacketAckMsg, - ica_kind: IcaMessages, -) -> Result { - match ica_kind { - IcaMessages::JoinSwapExternAmountIn(mut data) => { - handle_join_pool(storage, &env, ack_bin, &mut data) - } - IcaMessages::LockTokens(data, lp_shares) => { - handle_lock_tokens_ack(storage, &env, data, lp_shares, ack_bin, querier) - } - IcaMessages::BeginUnlocking(data, total) => { - handle_start_unbond_ack(storage, querier, &env, data, total) - } - IcaMessages::ExitPool(data) => handle_exit_pool_ack(storage, &env, data, ack_bin), - // TODO decide where we unlock the transfer ack unlock, here or in the ibc hooks receive - IcaMessages::ReturnTransfer(data) => handle_return_transfer_ack(storage, querier, data), - // After a RecoveryExitPool, we do a return transfer that should hit RecoveryReturnTransfer - IcaMessages::RecoveryExitPool(_pending) => todo!(), - // After a RecoveryReturnTransfer, we save the funds to a local map, to be claimed by vaults when a users asks - IcaMessages::RecoveryReturnTransfer(_pending) => todo!(), - } -} - -// fn handle_recovery_return_transfer( -// storage: &mut dyn Storage, -// pending: PendingReturningRecovery, - -// ) -> Result { -// // if we have the succesfully received the recovery, we create an entry -// for p in pending.returning { -// if let RawAmount::LocalDenom(val) = p.amount { -// CLAIMABLE_FUNDS.save(storage, (p.owner, p.id), &val)?; -// } else { -// return Err(ContractError::IncorrectRawAmount); -// } -// // remove the error from TRAPS -// TRAPS.remove(storage, (pending.trapped_id, )); -// } -// todo!() -// } - -fn handle_join_pool( - storage: &mut dyn Storage, - env: &Env, - ack_bin: Binary, - data: &mut PendingBond, -) -> Result { - // TODO move the below locking logic to a separate function - // get the ica address of the channel id - let ica_channel = ICA_CHANNEL.load(storage)?; - let ica_addr = get_ica_address(storage, ica_channel.clone())?; - let ack = AckBody::from_bytes(ack_bin.0.as_ref())?.to_any()?; - let resp = MsgJoinSwapExternAmountInResponse::unpack(ack)?; - let shares_out = Uint128::new(resp.share_out_amount.parse::().map_err(|err| { - ContractError::ParseIntError { - error: format!("{err}"), - value: resp.share_out_amount, - } - })?); - - let denom = CONFIG.load(storage)?.pool_denom; - - LP_SHARES.update( - storage, - |mut old: LpCache| -> Result { - old.d_unlocked_shares = old.d_unlocked_shares.checked_add(shares_out)?; - Ok(old) - }, - )?; - - data.update_raw_amount_to_lp(shares_out)?; - - let msg = do_ibc_lock_tokens( - storage, - ica_addr, - vec![Coin { - denom, - amount: shares_out, - }], - )?; - - let channel = ICA_CHANNEL.load(storage)?; - - let outgoing = ica_send( - msg, - ica_channel, - IbcTimeout::with_timestamp(env.block.time.plus_seconds(IBC_TIMEOUT_TIME)), - )?; - - let msg = create_ibc_ack_submsg( - storage, - IbcMsgKind::Ica(IcaMessages::LockTokens(data.clone(), shares_out)), - outgoing, - channel, - )?; - Ok(Response::new().add_submessage(msg)) -} - -fn handle_lock_tokens_ack( - storage: &mut dyn Storage, - _env: &Env, - data: PendingBond, - total_lp_shares: Uint128, - ack_bin: Binary, - querier: QuerierWrapper, -) -> Result { - let ack = AckBody::from_bytes(ack_bin.0.as_ref())?.to_any()?; - let resp = MsgLockTokensResponse::unpack(ack)?; - - // save the lock id in the contract - OSMO_LOCK.save(storage, &resp.id)?; - - LP_SHARES.update(storage, |mut old| -> Result { - old.d_unlocked_shares = - old.d_unlocked_shares - .checked_sub(total_lp_shares) - .map_err(|err| { - ContractError::TracedOverflowError( - err, - "update_unlocked_deposit_shares".to_string(), - ) - })?; - old.locked_shares = old - .locked_shares - .checked_add(total_lp_shares) - .map_err(|err| { - ContractError::TracedOverflowError(err, "update_locked_shares".to_string()) - })?; - Ok(old) - })?; - - let mut callback_submsgs: Vec = vec![]; - for claim in data.bonds { - let share_amount = create_share(storage, &claim.owner, &claim.bond_id, claim.claim_amount)?; - if querier - .query_wasm_contract_info(claim.owner.as_str()) - .is_ok() - { - let wasm_msg = WasmMsg::Execute { - contract_addr: claim.owner.to_string(), - msg: to_json_binary(&Callback::BondResponse(BondResponse { - share_amount, - bond_id: claim.bond_id.clone(), - }))?, - funds: vec![], - }; - // convert wasm_msg into cosmos_msg to be handled in create_callback_submsg - let cosmos_msg = CosmosMsg::Wasm(wasm_msg); - callback_submsgs.push(create_callback_submsg( - storage, - cosmos_msg, - claim.owner, - claim.bond_id, - )?); - } - } - - // set the bond lock state to unlocked - IBC_LOCK.update(storage, |old| -> Result { - Ok(old.unlock_bond()) - })?; - - // TODO, do we want to also check queue state? and see if we can already start a new execution? - Ok(Response::new() - .add_submessages(callback_submsgs) - .add_attribute("locked_tokens", ack_bin.to_base64()) - .add_attribute("lock_id", resp.id.to_string())) -} - -fn handle_exit_pool_ack( - storage: &mut dyn Storage, - env: &Env, - mut data: PendingReturningUnbonds, - ack_bin: Binary, -) -> Result { - let ack = AckBody::from_bytes(ack_bin.0.as_ref())?.to_any()?; - let msg = MsgExitSwapShareAmountInResponse::unpack(ack)?; - let total_exited_tokens = - Uint128::new(msg.token_out_amount.parse::().map_err(|err| { - ContractError::ParseIntError { - error: format!("{err}"), - value: msg.token_out_amount, - } - })?); - - // we don't need the sum of the lp tokens returned by lp_to_local_denom here - let _ = data.lp_to_local_denom(total_exited_tokens)?; - - let sub_msg = transfer_batch_unbond(storage, env, data, total_exited_tokens)?; - Ok(Response::new() - .add_submessage(sub_msg) - .add_attribute("transfer-funds", total_exited_tokens.to_string())) -} - -fn handle_return_transfer_ack( - storage: &mut dyn Storage, - _querier: QuerierWrapper, - _data: PendingReturningUnbonds, -) -> Result { - IBC_LOCK.update(storage, |lock| -> Result { - Ok(lock.unlock_unbond()) - })?; - - Ok(Response::new().add_attribute("return-transfer", "success")) -} - -pub fn handle_failing_ack( - deps: DepsMut, - _env: Env, - pkt: IbcPacketAckMsg, - error: String, -) -> Result { - // TODO we can expand error handling here to fetch the packet by the ack and add easy retries or something - let step = PENDING_ACK.load( - deps.storage, - ( - pkt.original_packet.sequence, - pkt.original_packet.src.channel_id.clone(), - ), - )?; - unlock_on_error(deps.storage, &step)?; - TRAPS.save( - deps.storage, - ( - pkt.original_packet.sequence, - pkt.original_packet.src.channel_id, - ), - &Trap { - error: format!("packet failure: {error}"), - step, - last_succesful: false, - }, - )?; - Ok(Response::new().add_attribute("ibc-error", error.as_str())) -} - -// if an ICA packet is timed out, we need to reject any further packets (only to the ICA channel or in total -> easiest in total until a new ICA channel is created) -// once time out variable is set, a new ICA channel needs to be able to be opened for the contract to function and the ICA channel val and channels map need to be updated -// what do we do with the trapped errors packets, are they able to be recovered over the new ICA channel? -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn ibc_packet_timeout( - deps: DepsMut, - _env: Env, - msg: IbcPacketTimeoutMsg, -) -> Result { - on_packet_timeout( - deps, - msg.packet.sequence, - msg.packet.src.channel_id, - "timeout".to_string(), - true, - ) -} - -pub(crate) fn on_packet_timeout( - deps: DepsMut, - sequence: u64, - channel: String, - error: String, - should_unlock: bool, -) -> Result { - let step = PENDING_ACK.load(deps.storage, (sequence, channel.clone()))?; - if should_unlock { - unlock_on_error(deps.storage, &step)?; - } - if let IbcMsgKind::Ica(_) = &step { - TIMED_OUT.save(deps.storage, &true)? - } - TRAPS.save( - deps.storage, - (sequence, channel), - &Trap { - error: format!("packet failure: {error}"), - step, - last_succesful: false, - }, - )?; - Ok(IbcBasicResponse::default()) -} - -#[cfg(test)] -mod tests { - use super::*; - use crate::{ - state::{Config, LOCK_ADMIN, SIMULATED_JOIN_AMOUNT_IN}, - test_helpers::{create_query_response, default_setup}, - }; - use cosmos_sdk_proto::cosmos::base::v1beta1::Coin as OsmoCoin; - use cosmwasm_std::{ - testing::{mock_dependencies, mock_env}, - Addr, Empty, IbcEndpoint, IbcOrder, - }; - - #[test] - fn handle_icq_ack_works() { - let mut deps = mock_dependencies(); - let env = mock_env(); - default_setup(deps.as_mut().storage).unwrap(); - - CONFIG - .save( - deps.as_mut().storage, - &Config { - lock_period: 100, - pool_id: 1, - pool_denom: "gamm/pool/1".to_string(), - base_denom: - "ibc/27394FB092D2ECCD56123C74F36E4C1F926001CEADA9CA97EA622B25F41E5EB2" - .to_string(), - quote_denom: - "ibc/C140AFD542AE77BD7DCC83F13FDD8C5E5BB8C4929785E6EC2F4C636F98F17901" - .to_string(), - local_denom: - "ibc/FA0006F056DB6719B8C16C551FC392B62F5729978FC0B125AC9A432DBB2AA1A5" - .to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }, - ) - .unwrap(); - LP_SHARES - .save( - deps.as_mut().storage, - &LpCache { - locked_shares: Uint128::new(1000), - w_unlocked_shares: Uint128::zero(), - d_unlocked_shares: Uint128::zero(), - }, - ) - .unwrap(); - SIMULATED_JOIN_AMOUNT_IN - .save(deps.as_mut().storage, &Uint128::zero()) - .unwrap(); - - // base64 of '{"data":"Chs6FAoSCgV1b3NtbxIJMTkyODcwODgySNW/pQQKUjpLCkkKRGliYy8yNzM5NEZCMDkyRDJFQ0NENTYxMjNDNzRGMzZFNEMxRjkyNjAwMUNFQURBOUNBOTdFQTYyMkIyNUY0MUU1RUIyEgEwSNW/pQQKGToSChAKC2dhbW0vcG9vbC8xEgEwSNW/pQQKFjoPCgEwEgoKBXVvc21vEgEwSNW/pQQKcTpqClIKRGliYy8yNzM5NEZCMDkyRDJFQ0NENTYxMjNDNzRGMzZFNEMxRjkyNjAwMUNFQURBOUNBOTdFQTYyMkIyNUY0MUU1RUIyEgoxMDg5ODQ5Nzk5ChQKBXVvc21vEgsxNTQyOTM2Mzg2MEjVv6UECh06FgoUMC4wNzA2MzQ3ODUwMDAwMDAwMDBI1b+lBAqMATqEAQqBAQj7u2ISP29zbW8xd212ZXpscHNrNDB6M3pmc3l5ZXgwY2Q4ZHN1bTdnenVweDJxZzRoMHVhdms3dHh3NHNlcXE3MmZrbRoECIrqSSILCICSuMOY/v///wEqJwoLZ2FtbS9wb29sLzESGDEwODE3NDg0NTgwODQ4MDkyOTUyMDU1MUjVv6UE"}' - let ack_bin = Binary::from_base64("eyJkYXRhIjoiQ2xVNlV3cFJDa1JwWW1Ndk1qY3pPVFJHUWpBNU1rUXlSVU5EUkRVMk1USXpRemMwUmpNMlJUUkRNVVk1TWpZd01ERkRSVUZFUVRsRFFUazNSVUUyTWpKQ01qVkdOREZGTlVWQ01oSUpNVEV5TWpFek1qUTNDbFE2VWdwUUNrUnBZbU12UXpFME1FRkdSRFUwTWtGRk56ZENSRGRFUTBNNE0wWXhNMFpFUkRoRE5VVTFRa0k0UXpRNU1qazNPRFZGTmtWRE1rWTBRell6TmtZNU9FWXhOemt3TVJJSU1qWTROekV6TnpRS0tUb25DaVVLRFdkaGJXMHZjRzl2YkM4NE1ETVNGRFkxTURnM05qazBPREF4TURZeU5EWXdNVEE0Q3FzQk9xZ0JDbElLUkdsaVl5OHlOek01TkVaQ01Ea3lSREpGUTBORU5UWXhNak5ETnpSR016WkZORU14UmpreU5qQXdNVU5GUVVSQk9VTkJPVGRGUVRZeU1rSXlOVVkwTVVVMVJVSXlFZ295TlRZNE56QTROelExQ2xJS1JHbGlZeTlETVRRd1FVWkVOVFF5UVVVM04wSkVOMFJEUXpnelJqRXpSa1JFT0VNMVJUVkNRamhETkRreU9UYzROVVUyUlVNeVJqUkROak0yUmprNFJqRTNPVEF4RWdveU1UY3dOVFkxTXpNNENoZzZGZ29VTUM0NE5EVXdNREkxTVRBd01EQXdNREF3TURBS2h3RTZoQUVLZ1FFSTVQVmtFajl2YzIxdk1YZGxlbkZoZG5ZNE5uQjVOSE5tWXpOM1pubDBiVFY1Ym1ONk5YcG1Oakk1Wm10NE4yb3daM0JsTlhwNWRtTnlPVGwwZUhNNWRuWjNjemNhQkFpQjZra2lDd2lBa3JqRG1QNy8vLzhCS2ljS0RXZGhiVzB2Y0c5dmJDODRNRE1TRmpJek5EQXpOVGs0TWpBd09UTTVOVFV6TkRJNU1EUT0ifQ==").unwrap(); - // queues are empty at this point so we just expect a succesful response without anyhting else - handle_icq_ack(deps.as_mut().storage, env, ack_bin).unwrap(); - } - - #[test] - fn handle_ica_channel_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - - let endpoint = IbcEndpoint { - port_id: "wasm.my_addr".to_string(), - channel_id: "channel-1".to_string(), - }; - let counterparty_endpoint = IbcEndpoint { - port_id: "icahost".to_string(), - channel_id: "channel-2".to_string(), - }; - - let version = r#"{"version":"ics27-1","encoding":"proto3","tx_type":"sdk_multi_msg","controller_connection_id":"connection-0","host_connection_id":"connection-0"}"#.to_string(); - let channel = IbcChannel::new( - endpoint, - counterparty_endpoint.clone(), - IbcOrder::Ordered, - version, - "connection-0".to_string(), - ); - - let msg = IbcChannelOpenMsg::OpenInit { - channel: channel.clone(), - }; - - handle_ica_channel(deps.as_mut(), msg).unwrap(); - - let expected = ChannelInfo { - id: channel.endpoint.channel_id.clone(), - counterparty_endpoint, - connection_id: "connection-0".to_string(), - channel_type: ChannelType::Ica { - channel_ty: IcaMetadata::with_connections( - "connection-0".to_string(), - "connection-0".to_string(), - ), - counter_party_address: None, - }, - handshake_state: HandshakeState::Init, - }; - assert_eq!( - CHANNELS - .load(deps.as_ref().storage, channel.endpoint.channel_id) - .unwrap(), - expected - ) - } - - #[test] - fn handle_ica_channel_open_try_errors() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - - let endpoint = IbcEndpoint { - port_id: "wasm.my_addr".to_string(), - channel_id: "channel-1".to_string(), - }; - let counterparty_endpoint = IbcEndpoint { - port_id: "icahost".to_string(), - channel_id: "channel-2".to_string(), - }; - - let version = r#"{"version":"ics27-1","encoding":"proto3","tx_type":"sdk_multi_msg","controller_connection_id":"connection-0","host_connection_id":"connection-0"}"#.to_string(); - let channel = IbcChannel::new( - endpoint, - counterparty_endpoint, - IbcOrder::Ordered, - version, - "connection-0".to_string(), - ); - - let msg = IbcChannelOpenMsg::OpenTry { - channel, - counterparty_version: "1".to_string(), - }; - - let err = handle_ica_channel(deps.as_mut(), msg).unwrap_err(); - assert_eq!(err, ContractError::IncorrectChannelOpenType); - } - - #[test] - fn test_handle_icq_ack() { - let mut deps = mock_dependencies(); - let env = mock_env(); - default_setup(deps.as_mut().storage).unwrap(); - - IBC_LOCK.save(deps.as_mut().storage, &Lock::new()).unwrap(); - - // no osmo lock as we're not sending lock ICQ for simplicity - // OSMO_LOCK.save(deps.as_mut().storage, &1u64).unwrap(); - - LOCK_ADMIN - .save(deps.as_mut().storage, &Addr::unchecked("admin"), &Empty {}) - .unwrap(); - - // mocking the ICQ ACK (some values don't make sense, but we're not using them in this math calculation) - let raw_balance = create_query_response( - QueryBalanceResponse { - balance: Some(OsmoCoin { - denom: "uatom".to_string(), - amount: "100".to_string(), - }), - } - .encode_to_vec(), - ); - - let quote_balance = create_query_response( - QueryBalanceResponse { - balance: Some(OsmoCoin { - denom: "uqsr".to_string(), - amount: "100".to_string(), - }), - } - .encode_to_vec(), - ); - - let lp_balance = create_query_response( - QueryBalanceResponse { - balance: Some(OsmoCoin { - denom: "uosmo".to_string(), - amount: "100".to_string(), - }), - } - .encode_to_vec(), - ); - - let exit_pool = create_query_response( - QueryCalcExitPoolCoinsFromSharesResponse { - tokens_out: vec![ - Coin { - // base denom - denom: "uosmo".to_string(), - amount: Uint128::new(100), - } - .into(), - Coin { - // quote denom - denom: "uqsr".to_string(), - amount: Uint128::new(100), - } - .into(), - ], - } - .encode_to_vec(), - ); - - let spot_price = create_query_response( - QuerySpotPriceResponse { - spot_price: "1".to_string(), - } - .encode_to_vec(), - ); - - let join_pool = create_query_response( - QueryCalcJoinPoolSharesResponse { - share_out_amount: "123".to_string(), - tokens_out: vec![Coin { - denom: "uosmo".to_string(), - amount: Uint128::new(100), - } - .into()], - } - .encode_to_vec(), - ); - - let exit_pool_unbonds = create_query_response( - QueryCalcExitPoolCoinsFromSharesResponse { - tokens_out: vec![ - Coin { - denom: "uqsr".to_string(), - amount: Uint128::new(100), - } - .into(), - Coin { - denom: "uosmo".to_string(), - amount: Uint128::new(100), - } - .into(), - ], - } - .encode_to_vec(), - ); - - let ibc_ack = InterchainQueryPacketAck { - data: Binary::from( - &CosmosResponse { - responses: vec![ - raw_balance.clone(), - quote_balance.clone(), - lp_balance.clone(), - exit_pool.clone(), - spot_price, - join_pool.clone(), - //lock, we're not sending lock for simplicity and to test indexing logic without one value works - exit_pool_unbonds, - ], - } - .encode_to_vec()[..], - ), - }; - - // mock the value of shares we had before sending the query - SIMULATED_EXIT_SHARES_IN - .save(deps.as_mut().storage, &Uint128::new(200)) - .unwrap(); - - // mock the value of shares we had before sending the query - SIMULATED_JOIN_AMOUNT_IN - .save(deps.as_mut().storage, &Uint128::new(100)) - .unwrap(); - - // simulate that we received the ICQ ACK, shouldn't return any messages - let _res = handle_icq_ack( - deps.as_mut().storage, - env.clone(), - to_json_binary(&ibc_ack).unwrap(), - ) - .unwrap(); - - assert_eq!( - SIMULATED_EXIT_RESULT - .load(deps.as_ref().storage) - .unwrap() - .u128(), - // base_amount + (quote_amount / spot_price) - 100 + 100 - ); - - // changing some ICQ ACK params to create a different test scenario - let spot_price = create_query_response( - QuerySpotPriceResponse { - spot_price: "5".to_string(), - } - .encode_to_vec(), - ); - - let exit_pool_unbonds = create_query_response( - QueryCalcExitPoolCoinsFromSharesResponse { - tokens_out: vec![ - Coin { - // base denom - denom: "uosmo".to_string(), - amount: Uint128::new(1000), - } - .into(), - Coin { - // quote denom - denom: "uqsr".to_string(), - amount: Uint128::new(5000), - } - .into(), - ], - } - .encode_to_vec(), - ); - - let ibc_ack = InterchainQueryPacketAck { - data: Binary::from( - &CosmosResponse { - responses: vec![ - raw_balance, - quote_balance, - lp_balance, - exit_pool, - spot_price, - join_pool, - //lock, we're not sending lock for simplicity and to test indexing logic without one value works - exit_pool_unbonds, - ], - } - .encode_to_vec()[..], - ), - }; - - // simulate that we received another ICQ ACK, shouldn't return any messages - let _res = handle_icq_ack( - deps.as_mut().storage, - env, - to_json_binary(&ibc_ack).unwrap(), - ) - .unwrap(); - - assert_eq!( - SIMULATED_EXIT_RESULT - .load(deps.as_ref().storage) - .unwrap() - .u128(), - // base_amount + (quote_amount / spot_price) - 1000 + (5000 / 5) - ); - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/ibc_lock.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/ibc_lock.rs deleted file mode 100644 index dd72944cd..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/ibc_lock.rs +++ /dev/null @@ -1,192 +0,0 @@ -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub struct Lock { - pub bond: IbcLock, - pub start_unbond: IbcLock, - pub unbond: IbcLock, - pub recovery: IbcLock, - pub migration: IbcLock, -} - -impl Lock { - pub fn new() -> Self { - Lock { - bond: IbcLock::Unlocked, - start_unbond: IbcLock::Unlocked, - unbond: IbcLock::Unlocked, - recovery: IbcLock::Unlocked, - migration: IbcLock::Unlocked, - } - } - - pub fn unlock_bond(mut self) -> Self { - self.bond = IbcLock::Unlocked; - self - } - - pub fn unlock_start_unbond(mut self) -> Self { - self.start_unbond = IbcLock::Unlocked; - self - } - - pub fn unlock_unbond(mut self) -> Self { - self.unbond = IbcLock::Unlocked; - self - } - - pub fn unlock_migration(mut self) -> Self { - self.migration = IbcLock::Unlocked; - self - } - - pub fn lock_bond(mut self) -> Self { - self.bond = IbcLock::Locked; - self - } - - pub fn lock_start_unbond(mut self) -> Self { - self.start_unbond = IbcLock::Locked; - self - } - - pub fn lock_unbond(mut self) -> Self { - self.unbond = IbcLock::Locked; - self - } - - pub fn lock_migration(mut self) -> Self { - self.migration = IbcLock::Locked; - self - } - - // this doesnt take into account the recovery lock - pub fn is_unlocked(&self) -> bool { - self.bond.is_unlocked() - && self.start_unbond.is_unlocked() - && self.unbond.is_unlocked() - && self.migration.is_unlocked() - } - - // this doesnt take into account the recovery lock - pub fn is_locked(&self) -> bool { - self.bond.is_locked() - || self.start_unbond.is_locked() - || self.unbond.is_locked() - || self.migration.is_locked() - } -} - -impl Default for Lock { - fn default() -> Self { - Self::new() - } -} - -/// IbcLock describes the current state of the contract -/// Upon locking the contract, all current deposits and withdraws are going to be handled, Incoming withdraws are gathered once again gathered into a queue. -/// Once the contract unlocks, if the queue has any deposits and/or withdraws, the contract locks and starts handling all current queries -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub enum IbcLock { - Locked, - Unlocked, -} - -impl IbcLock { - pub fn is_unlocked(&self) -> bool { - self == &IbcLock::Unlocked - } - - pub fn is_locked(&self) -> bool { - self == &IbcLock::Locked - } -} - -// write tests -#[cfg(test)] -mod tests { - use super::*; - - #[test] - fn test_lock() { - // start from unlocked - let mut lock = Lock::new(); - assert!(lock.is_unlocked()); - assert!(!lock.is_locked()); - - // lock one by one and check - lock = lock.lock_bond(); - assert!(!lock.is_unlocked()); - assert!(lock.is_locked()); - - lock = lock.lock_start_unbond(); - assert!(!lock.is_unlocked()); - assert!(lock.is_locked()); - - lock = lock.lock_unbond(); - assert!(!lock.is_unlocked()); - assert!(lock.is_locked()); - - // manually lock recovery - lock.recovery = IbcLock::Locked; - assert!(!lock.is_unlocked()); - assert!(lock.is_locked()); - - lock = lock.lock_migration(); - assert!(!lock.is_unlocked()); - assert!(lock.is_locked()); - - // all should be locked - assert!(lock.bond.is_locked()); - assert!(lock.start_unbond.is_locked()); - assert!(lock.unbond.is_locked()); - assert!(lock.recovery.is_locked()); - assert!(lock.migration.is_locked()); - - // none should be unlocked - assert!(!lock.bond.is_unlocked()); - assert!(!lock.start_unbond.is_unlocked()); - assert!(!lock.unbond.is_unlocked()); - assert!(!lock.recovery.is_unlocked()); - assert!(!lock.migration.is_unlocked()); - - // unlock one by one and check - lock = lock.unlock_bond(); - assert!(!lock.is_unlocked()); - assert!(lock.is_locked()); - - lock = lock.unlock_start_unbond(); - assert!(!lock.is_unlocked()); - assert!(lock.is_locked()); - - lock = lock.unlock_unbond(); - assert!(!lock.is_unlocked()); - assert!(lock.is_locked()); - - // manually unlock recovery - lock.recovery = IbcLock::Unlocked; - assert!(!lock.is_unlocked()); - assert!(lock.is_locked()); - - lock = lock.unlock_migration(); - assert!(lock.is_unlocked()); - assert!(!lock.is_locked()); - - // all should be unlocked - assert!(lock.bond.is_unlocked()); - assert!(lock.start_unbond.is_unlocked()); - assert!(lock.unbond.is_unlocked()); - assert!(lock.recovery.is_unlocked()); - assert!(lock.migration.is_unlocked()); - - // none should be locked - assert!(!lock.bond.is_locked()); - assert!(!lock.start_unbond.is_locked()); - assert!(!lock.unbond.is_locked()); - assert!(!lock.recovery.is_locked()); - assert!(!lock.migration.is_locked()); - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/ibc_util.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/ibc_util.rs deleted file mode 100644 index e2d77545f..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/ibc_util.rs +++ /dev/null @@ -1,397 +0,0 @@ -use cosmwasm_std::{ - Coin, ConversionOverflowError, Decimal, Env, Fraction, IbcMsg, IbcTimeout, StdError, Storage, - SubMsg, Uint128, -}; -use osmosis_std::{ - shim::Duration, - types::{ - cosmos::base::v1beta1::Coin as OsmoCoin, - osmosis::{ - gamm::v1beta1::{MsgJoinSwapExternAmountIn, QueryCalcJoinPoolSharesResponse}, - lockup::MsgLockTokens, - }, - }, -}; - -use quasar_types::ica::packet::ica_send; - -use crate::{ - error::ContractError, - helpers::{create_ibc_ack_submsg, get_ica_address, IbcMsgKind, IcaMessages}, - state::{ - OngoingDeposit, PendingBond, CONFIG, IBC_TIMEOUT_TIME, ICA_CHANNEL, SIMULATED_EXIT_RESULT, - SIMULATED_JOIN_RESULT, - }, -}; - -pub fn do_transfer( - storage: &mut dyn Storage, - env: &Env, - amount: Uint128, - channel_id: String, - to_address: String, - deposits: Vec, -) -> Result { - // todo check denom of funds once we have denom mapping done - - let coin = Coin { - denom: CONFIG.load(storage)?.local_denom, - amount, - }; - - let timeout = IbcTimeout::with_timestamp(env.block.time.plus_seconds(IBC_TIMEOUT_TIME)); - let transfer = IbcMsg::Transfer { - channel_id: channel_id.clone(), - to_address, - amount: coin, - timeout, - }; - - Ok(create_ibc_ack_submsg( - storage, - IbcMsgKind::Transfer { - pending: PendingBond { bonds: deposits }, - amount, - }, - transfer, - channel_id, - )?) -} - -pub fn parse_join_pool( - _storage: &dyn Storage, - join: QueryCalcJoinPoolSharesResponse, -) -> Result { - let join = match join.share_out_amount.parse::() { - Ok(val) => Ok(val), - Err(err) => { - match err.kind() { - // if the string is empty, we return 0 shares out - std::num::IntErrorKind::Empty => Ok(0), - _ => Err(ContractError::ParseIntError { - error: format!("scale:{err}"), - value: join.share_out_amount, - }), - } - } - }?; - - Ok(Uint128::new(join)) -} - -pub fn consolidate_exit_pool_amount_into_local_denom( - storage: &mut dyn Storage, - exit_pool_unbonds: &[OsmoCoin], - spot_price: Decimal, -) -> Result { - let config = CONFIG.load(storage)?; - - // if we receive no tokens in the response, we can't exit the pool - // todo: Should this error? - if exit_pool_unbonds.is_empty() { - return Ok(Uint128::zero()); - } - - // consolidate exit_pool.tokens_out into a single Uint128 by using spot price to convert the quote_denom to local_denom - let base = exit_pool_unbonds - .iter() - .find(|coin| coin.denom == config.base_denom) - .ok_or(ContractError::BaseDenomNotFound)?; - let quote = exit_pool_unbonds - .iter() - .find(|coin| coin.denom == config.quote_denom) - .ok_or(ContractError::QuoteDenomNotFound)?; - - Ok(Uint128::new( - base.amount - .parse::() - .map_err(|err| ContractError::ParseIntError { - error: format!("base_amount:{err}"), - value: base.amount.clone(), - })?, - ) - .checked_add( - Uint128::new( - quote - .amount - .parse::() - .map_err(|err| ContractError::ParseIntError { - error: format!("quote_amount:{err}"), - value: quote.amount.clone(), - })?, - ) - .checked_multiply_ratio(spot_price.denominator(), spot_price.numerator())?, - )?) -} - -pub fn calculate_share_out_min_amount(storage: &mut dyn Storage) -> Result { - let last_sim_join_pool_result = SIMULATED_JOIN_RESULT.load(storage)?; - - // todo: better dynamic slippage estimation, especially for volatile tokens - // diminish the share_out_amount by 5 percent to allow for slippage of 5% on the swap - Ok(last_sim_join_pool_result.checked_multiply_ratio(95u128, 100u128)?) -} - -// exit shares should never be more than total shares here -pub fn calculate_token_out_min_amount(storage: &dyn Storage) -> Result { - let last_sim_exit_pool_unbonds_result = SIMULATED_EXIT_RESULT.load(storage)?; - - // todo: better dynamic slippage estimation, especially for volatile tokens - // diminish the share_out_amount by 5 percent to allow for slippage of 5% on the swap - Ok(last_sim_exit_pool_unbonds_result.checked_multiply_ratio(95u128, 100u128)?) -} - -/// prepare the submsg for joining the pool -#[allow(clippy::too_many_arguments)] // allowing this is not ideal, but for now we want to keep channel_id and pool_id in there -pub fn do_ibc_join_pool_swap_extern_amount_in( - storage: &mut dyn Storage, - env: Env, - pool_id: u64, - denom: String, - amount: Uint128, - share_out_min_amount: Uint128, - deposits: Vec, -) -> Result { - let ica_channel = ICA_CHANNEL.load(storage)?; - let ica_address = get_ica_address(storage, ica_channel.clone())?; - - // setup the first IBC message to send, and save the entire sequence so we have acces to it on acks - let msg = MsgJoinSwapExternAmountIn { - sender: ica_address, - pool_id, - token_in: Some(OsmoCoin { - denom, - amount: amount.to_string(), - }), - share_out_min_amount: share_out_min_amount.to_string(), - }; - - let pkt = ica_send::( - msg, - ica_channel.clone(), - IbcTimeout::with_timestamp(env.block.time.plus_seconds(IBC_TIMEOUT_TIME)), - )?; - - Ok(create_ibc_ack_submsg( - storage, - IbcMsgKind::Ica(IcaMessages::JoinSwapExternAmountIn(PendingBond { - bonds: deposits, - })), - pkt, - ica_channel, - )?) -} - -pub fn do_ibc_lock_tokens( - storage: &mut dyn Storage, - owner: String, - coins: Vec, -) -> Result { - let lock_period = CONFIG.load(storage)?.lock_period; - - // TODO move the duration to a package and make it settable - Ok(MsgLockTokens { - owner, - duration: Some(Duration { - // TODO clean up this conversion a bit - seconds: i64::try_from(lock_period).map_err(|_| { - ContractError::Std(StdError::ConversionOverflow { - source: ConversionOverflowError { - source_type: "u64", - target_type: "i64", - value: lock_period.to_string(), - }, - }) - })?, - nanos: 0, - }), - coins: coins - .iter() - .map(|c| OsmoCoin { - denom: c.denom.clone(), - amount: c.amount.to_string(), - }) - .collect(), - }) -} - -#[cfg(test)] -mod tests { - use std::str::FromStr; - - use cosmwasm_std::{ - testing::{mock_dependencies, MockApi, MockQuerier, MockStorage}, - Decimal, Empty, IbcEndpoint, OwnedDeps, Uint128, - }; - - use cw_storage_plus::Map; - - use osmosis_std::types::cosmos::base::v1beta1::Coin; - use quasar_types::{ - ibc::{ChannelInfo, ChannelType, HandshakeState}, - ica::handshake::IcaMetadata, - }; - - use crate::{ - ibc_util::{calculate_token_out_min_amount, consolidate_exit_pool_amount_into_local_denom}, - state::{SIMULATED_EXIT_RESULT, SIMULATED_JOIN_RESULT}, - test_helpers::default_setup, - }; - - use super::calculate_share_out_min_amount; - - fn default_instantiate( - channels: &Map, - ) -> OwnedDeps { - let mut deps = mock_dependencies(); - - // load an ICA channel into the deps - let (chan_id, channel_info) = default_ica_channel(); - // unwrap here since this is a test function - channels - .save(deps.as_mut().storage, chan_id, &channel_info) - .unwrap(); - deps - } - - fn default_ica_channel() -> (String, ChannelInfo) { - let chan_id = String::from("channel-0"); - ( - chan_id.clone(), - ChannelInfo { - id: chan_id, - counterparty_endpoint: IbcEndpoint { - port_id: String::from("ica-host"), - channel_id: String::from("channel-0"), - }, - connection_id: String::from("connection-0"), - channel_type: ChannelType::Ica { - channel_ty: IcaMetadata::with_connections( - String::from("connection-0"), - String::from("connection-0"), - ), - counter_party_address: Some(String::from("osmo")), - }, - handshake_state: HandshakeState::Open, - }, - ) - } - - #[test] - fn default_instantiate_works() { - let channels = Map::new("channels"); - let deps = default_instantiate(&channels); - - let _chan = channels - .load(deps.as_ref().storage, "channel-0".to_string()) - .unwrap(); - } - - #[test] - fn test_calculate_share_out_min_amount() { - let mut deps = mock_dependencies(); - SIMULATED_JOIN_RESULT - .save(deps.as_mut().storage, &Uint128::new(999999)) - .unwrap(); - - let min_amount_out = calculate_share_out_min_amount(deps.as_mut().storage).unwrap(); - - assert_eq!(min_amount_out, Uint128::from(949999u128)); - } - - #[test] - fn test_calculate_token_out_min_amount() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - - let exit_amount_local_denom = consolidate_exit_pool_amount_into_local_denom( - deps.as_mut().storage, - &vec![ - Coin { - denom: "uosmo".to_string(), - amount: "50".to_string(), - }, - Coin { - denom: "uqsr".to_string(), - amount: "50".to_string(), - }, - ], - Decimal::from_str("1.0").unwrap(), - ) - .unwrap(); - - SIMULATED_EXIT_RESULT - .save(deps.as_mut().storage, &exit_amount_local_denom) - .unwrap(); - - let min_amount_out = calculate_token_out_min_amount(deps.as_mut().storage).unwrap(); - - assert_eq!(min_amount_out, Uint128::from(95u128)); - - // now lets test with a different amount of exit shares - let exit_amount_local_denom = consolidate_exit_pool_amount_into_local_denom( - deps.as_mut().storage, - &vec![ - Coin { - denom: "uosmo".to_string(), - amount: "500".to_string(), - }, - Coin { - denom: "uqsr".to_string(), - amount: "50".to_string(), - }, - ], - // 1 uosmos = 10 uqsr -> price = 0.1 osmo for 1 uqsr - Decimal::from_str("0.1").unwrap(), - ) - .unwrap(); - - SIMULATED_EXIT_RESULT - .save(deps.as_mut().storage, &exit_amount_local_denom) - .unwrap(); - - let min_amount_out = calculate_token_out_min_amount(deps.as_mut().storage).unwrap(); - - // (500 uosmo + (50 uqsr / 0.1 osmo / uqsr)) * 0.95 = (500 + 500) * 0.95 uosmo = 950 uosmo - assert_eq!(min_amount_out, Uint128::from(1000u128 * 95u128 / 100u128)); - } - - #[test] - fn test_consolidate_exit_pool_amount_into_local_denom() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - - let exit_pool = vec![ - Coin { - denom: "uosmo".to_string(), // base_denom - amount: "100".to_string(), - }, - Coin { - denom: "uqsr".to_string(), // quote_denom - amount: "100".to_string(), - }, - ]; - - let spot_price = Decimal::from_str("1.0").unwrap(); - - let parsed = consolidate_exit_pool_amount_into_local_denom( - deps.as_mut().storage, - &exit_pool, - spot_price, - ) - .unwrap(); - - assert_eq!(parsed, Uint128::new(200)); - - // this is for when UNBOND_QUEUE is empty - let exit_pool = vec![]; - let parsed = consolidate_exit_pool_amount_into_local_denom( - deps.as_mut().storage, - &exit_pool, - spot_price, - ) - .unwrap(); - - assert_eq!(parsed, Uint128::zero()); - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/icq.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/icq.rs deleted file mode 100644 index 0e4a5fc5d..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/icq.rs +++ /dev/null @@ -1,410 +0,0 @@ -use cosmwasm_std::{ - to_json_binary, Decimal, Env, Fraction, IbcMsg, IbcTimeout, QuerierWrapper, Storage, SubMsg, - Uint128, -}; -#[allow(deprecated)] -use osmosis_std::types::osmosis::gamm::v1beta1::QuerySpotPriceRequest; -use osmosis_std::types::{ - cosmos::bank::v1beta1::QueryBalanceRequest, - cosmos::base::v1beta1::Coin as OsmoCoin, - osmosis::{ - gamm::v1beta1::{QueryCalcExitPoolCoinsFromSharesRequest, QueryCalcJoinPoolSharesRequest}, - lockup::LockedRequest, - }, -}; -use prost::Message; -use quasar_types::icq::{InterchainQueryPacketData, Query}; - -use crate::{ - error::ContractError, - helpers::{check_icq_channel, create_ibc_ack_submsg, get_ica_address, IbcMsgKind}, - state::{ - BOND_QUEUE, CONFIG, FAILED_JOIN_QUEUE, IBC_LOCK, ICA_CHANNEL, ICQ_CHANNEL, LP_SHARES, - OSMO_LOCK, PENDING_BOND_QUEUE, PENDING_UNBOND_QUEUE, SIMULATED_EXIT_SHARES_IN, - SIMULATED_JOIN_AMOUNT_IN, UNBOND_QUEUE, - }, -}; - -/// try_icq only does something if the IBC_LOCK is unlocked. When it is unlocked, -/// all pending bonds are moved into the active bond queue. -/// -/// It then prepares the following queries: -/// - ICA balance in base denom. -/// - ICA balance in quote denom. -/// - ICA balance in LP shares. -/// - SimulateJoinPool with the total pending bonds amount to estimate slippage and saves the total pending bonds amount in state. -/// - SimulateExitPool with the entire ICA locked amount to get the total value in lp tokens. -/// - SpotPrice of base denom and quote denom to convert quote denom from exitpool to the base denom. -/// - LockedByID to get the current lock state. -/// -/// It also moves all pending unbonds into the active unbond queue and returns an IBC send packet with the queries as a submessage. -pub fn try_icq( - storage: &mut dyn Storage, - _querier: QuerierWrapper, - env: Env, -) -> Result, ContractError> { - if IBC_LOCK.load(storage)?.is_unlocked() { - // TODO fetching ICQ channel and confirming vs handshake version can be a single function - let icq_channel = ICQ_CHANNEL.load(storage)?; - check_icq_channel(storage, icq_channel.clone())?; - - let mut pending_bonds_value = Uint128::zero(); - // we dump pending bonds into the active bond queue - while !PENDING_BOND_QUEUE.is_empty(storage)? { - let bond = PENDING_BOND_QUEUE.pop_front(storage)?; - if let Some(bond) = bond { - BOND_QUEUE.push_back(storage, &bond)?; - pending_bonds_value = pending_bonds_value.checked_add(bond.amount)?; - } - } - - let failed_join_queue_amount = FAILED_JOIN_QUEUE.iter(storage)?.try_fold( - Uint128::zero(), - |acc, val| -> Result { - Ok(acc + val?.amount) - // We should never have LP shares here - }, - )?; - - // the bonding amount that we want to calculate the slippage for is the amount of funds in new bonds and the amount of funds that have - // previously failed to join the pool. These funds are already located on Osmosis and should not be part of the transfer to Osmosis. - let bonding_amount = pending_bonds_value + failed_join_queue_amount; - - // we dump pending unbonds into the active unbond queue and save the total amount of shares that will be unbonded - let mut pending_unbonds_shares = Uint128::zero(); - while !PENDING_UNBOND_QUEUE.is_empty(storage)? { - let unbond = PENDING_UNBOND_QUEUE.pop_front(storage)?; - if let Some(unbond) = unbond { - UNBOND_QUEUE.push_back(storage, &unbond)?; - pending_unbonds_shares = pending_unbonds_shares.checked_add(unbond.lp_shares)?; - } - } - - // deposit needs to internally rebuild the amount of funds under the smart contract - let packet = - prepare_full_query(storage, env.clone(), bonding_amount, pending_unbonds_shares)?; - - let send_packet_msg = IbcMsg::SendPacket { - channel_id: icq_channel, - data: to_json_binary(&packet)?, - timeout: IbcTimeout::with_timestamp(env.block.time.plus_seconds(7200)), - }; - - let channel = ICQ_CHANNEL.load(storage)?; - - Ok(Some(create_ibc_ack_submsg( - storage, - IbcMsgKind::Icq, - send_packet_msg, - channel, - )?)) - } else { - Ok(None) - } -} - -pub fn prepare_full_query( - storage: &mut dyn Storage, - _env: Env, - bonding_amount: Uint128, - pending_unbonds_shares: Uint128, -) -> Result { - let ica_channel = ICA_CHANNEL.load(storage)?; - // todo: query flows should be separated by which flowType we're doing (bond, unbond, startunbond) - let address = get_ica_address(storage, ica_channel)?; - let config = CONFIG.load(storage)?; - // we query the current balance on our ica address - let base_balance = QueryBalanceRequest { - address: address.clone(), - denom: config.base_denom.clone(), - }; - let quote_balance = QueryBalanceRequest { - address: address.clone(), - denom: config.quote_denom.clone(), - }; - let lp_balance = QueryBalanceRequest { - address, - denom: config.pool_denom.clone(), - }; - // we simulate the result of a join pool to estimate the slippage we can expect during this deposit - // we use the current balance of local_denom for this query. This is safe because at any point - // a pending deposit will only use the current balance of the vault. QueryCalcJoinPoolSharesRequest - // since we're going to be moving the entire pending bond queue to the bond queue in this icq, we can - // fold the PENDING_BOND_QUEUE - // April 27 2023 - we removed get_usable_bond_balance, but we will have to bring it back when we do error - // recovery for deposits - - // we save the amount to scale the slippage against in the icq ack for other incoming bonds - SIMULATED_JOIN_AMOUNT_IN.save(storage, &bonding_amount)?; - - // we have to check that bonding amount is >= 1u128, because otherwise we get ABCI error code 1 (see osmosis: osmosis/x/gamm/pool-models/stableswap/amm.go:299 for the error we get in stableswap pools) - let checked_join_pool = match bonding_amount >= 1u128.into() { - true => Some(QueryCalcJoinPoolSharesRequest { - pool_id: config.pool_id, - tokens_in: vec![OsmoCoin { - denom: config.base_denom.clone(), - amount: bonding_amount.to_string(), - }], - }), - false => None, - }; - - // we save the amount to scale the slippage against in the icq ack for other incoming unbonds? - SIMULATED_EXIT_SHARES_IN.save(storage, &pending_unbonds_shares)?; - - // we have to check that unbonding amount is >= 1u128, because otherwise we get ABCI error code 1 (see osmosis: osmosis/x/gamm/pool-models/stableswap/amm.go:299 for the error we get in stableswap pools) - let checked_exit_pool_unbonds = match pending_unbonds_shares >= 1u128.into() { - true => Some(QueryCalcExitPoolCoinsFromSharesRequest { - pool_id: config.pool_id, - share_in_amount: pending_unbonds_shares.to_string(), - }), - false => None, - }; - - let shares = LP_SHARES.load(storage)?.locked_shares; - let shares_out = if !shares.is_zero() { - shares - } else { - Uint128::one() - }; - - let exit_total_pool = QueryCalcExitPoolCoinsFromSharesRequest { - pool_id: config.pool_id, - share_in_amount: shares_out.to_string(), - }; - // we query the spot price of our base_denom and quote_denom so we can convert the quote_denom from exitpool to the base_denom - #[allow(deprecated)] - let spot_price = QuerySpotPriceRequest { - pool_id: config.pool_id, - base_asset_denom: config.base_denom, - quote_asset_denom: config.quote_denom, - }; - - // path have to be set manually, should be equal to the proto_queries of osmosis-std types - let mut q = Query::new() - .add_request( - base_balance.encode_to_vec().into(), - "/cosmos.bank.v1beta1.Query/Balance".to_string(), - ) - .add_request( - quote_balance.encode_to_vec().into(), - "/cosmos.bank.v1beta1.Query/Balance".to_string(), - ) - .add_request( - lp_balance.encode_to_vec().into(), - "/cosmos.bank.v1beta1.Query/Balance".to_string(), - ) - .add_request( - exit_total_pool.encode_to_vec().into(), - "/osmosis.gamm.v1beta1.Query/CalcExitPoolCoinsFromShares".to_string(), - ) - .add_request( - spot_price.encode_to_vec().into(), - "/osmosis.gamm.v2.Query/SpotPrice".to_string(), - ); - - if let Some(join_pool) = checked_join_pool { - q = q.add_request( - join_pool.encode_to_vec().into(), - "/osmosis.gamm.v1beta1.Query/CalcJoinPoolShares".to_string(), - ) - } - - // only query LockedByID if we have a lock_id - if let Some(lock_id) = OSMO_LOCK.may_load(storage)? { - let lock_by_id = LockedRequest { lock_id }; - q = q.add_request( - lock_by_id.encode_to_vec().into(), - "/osmosis.lockup.Query/LockedByID".to_string(), - ); - } - - // if there're items in the unbond_queue, we query CalcExitPoolCoinsFromShares for the added share amount - if let Some(exit_pool) = checked_exit_pool_unbonds { - q = q.add_request( - exit_pool.encode_to_vec().into(), - "/osmosis.gamm.v1beta1.Query/CalcExitPoolCoinsFromShares".to_string(), - ) - } - - Ok(q.encode_pkt()) -} - -// TODO add quote denom to base denom conversion -// calculate the total balance of the vault using the query from prepare_total_balance_query() -pub fn calc_total_balance( - storage: &mut dyn Storage, - ica_balance: Uint128, - exit_pool: &[OsmoCoin], - spot_price: Decimal, -) -> Result { - let config = CONFIG.load(storage)?; - // if we receive no tokens in the response, the total balance - if exit_pool.is_empty() { - return Ok(ica_balance); - } - - let base = exit_pool - .iter() - .find(|coin| coin.denom == config.base_denom) - .ok_or(ContractError::BaseDenomNotFound)?; - - let quote = exit_pool - .iter() - .find(|coin| coin.denom == config.quote_denom) - .ok_or(ContractError::QuoteDenomNotFound)?; - // return ica_balance + base_amount + (quote_amount * spot_price) - Ok(ica_balance - .checked_add(Uint128::new(base.amount.parse::().map_err( - |err| ContractError::ParseIntError { - error: format!("ica_balance:{err:?}"), - value: base.amount.clone(), - }, - )?))? - .checked_add( - Uint128::new(quote.amount.parse::().map_err(|err| { - ContractError::ParseIntError { - error: format!("quote_denom:{err:?}"), - value: quote.amount.clone(), - } - })?) - .checked_multiply_ratio(spot_price.denominator(), spot_price.numerator())?, - )?) -} - -#[cfg(test)] -mod tests { - use cosmwasm_std::{ - testing::{mock_dependencies, mock_env, MockQuerier}, - Empty, - }; - - use crate::{ - ibc_lock::Lock, - state::{LpCache, IBC_LOCK}, - test_helpers::default_setup, - }; - - use proptest::prelude::*; - - use super::*; - - proptest! { - #[test] - fn calc_total_balance_works(ica_balance in 1..u64::MAX as u128, base_amount in 1..u64::MAX as u128, quote_amount in 1..u64::MAX as u128, spot_price in 1..u64::MAX as u128) { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let config = CONFIG.load(deps.as_ref().storage).unwrap(); - - let tokens = vec![OsmoCoin{ denom: config.base_denom, amount: base_amount.to_string() }, OsmoCoin{ denom: config.quote_denom, amount: quote_amount.to_string() }]; - let spot = Decimal::raw(spot_price); - let total = calc_total_balance(deps.as_mut().storage, Uint128::new(ica_balance), &tokens, spot).unwrap(); - let expecte_quote = Uint128::new(quote_amount).multiply_ratio(spot.denominator(), spot.numerator()); - prop_assert_eq!(total.u128(), ica_balance + base_amount + expecte_quote.u128()) - } - } - - #[test] - fn try_icq_unlocked_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let env = mock_env(); - - LP_SHARES - .save( - deps.as_mut().storage, - &LpCache { - locked_shares: Uint128::new(100), - w_unlocked_shares: Uint128::zero(), - d_unlocked_shares: Uint128::zero(), - }, - ) - .unwrap(); - - // lock the ibc lock - IBC_LOCK.save(deps.as_mut().storage, &Lock::new()).unwrap(); - - let qx: MockQuerier = MockQuerier::new(&[]); - let q = QuerierWrapper::new(&qx); - - let res = try_icq(deps.as_mut().storage, q, env.clone()).unwrap(); - - let icq_channel = ICQ_CHANNEL.load(deps.as_mut().storage).unwrap(); - - let pkt = IbcMsg::SendPacket { - channel_id: icq_channel.clone(), - data: to_json_binary( - &prepare_full_query( - deps.as_mut().storage, - env.clone(), - Uint128::new(0), - Uint128::zero(), - ) - .unwrap(), - ) - .unwrap(), - timeout: IbcTimeout::with_timestamp(env.block.time.plus_seconds(7200)), - }; - - assert_eq!( - res.unwrap().msg, - create_ibc_ack_submsg(deps.as_mut().storage, IbcMsgKind::Icq, pkt, icq_channel) - .unwrap() - .msg - ) - } - - #[test] - fn try_icq_locked_bond_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let env = mock_env(); - - // lock the ibc lock - IBC_LOCK - .save(deps.as_mut().storage, &Lock::new().lock_bond()) - .unwrap(); - - let qx: MockQuerier = MockQuerier::new(&[]); - let q = QuerierWrapper::new(&qx); - - let res = try_icq(deps.as_mut().storage, q, env).unwrap(); - assert_eq!(res, None) - } - - #[test] - fn try_icq_locked_start_unbond_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let env = mock_env(); - - // lock the ibc lock - IBC_LOCK - .save(deps.as_mut().storage, &Lock::new().lock_start_unbond()) - .unwrap(); - - let qx: MockQuerier = MockQuerier::new(&[]); - let q = QuerierWrapper::new(&qx); - - let res = try_icq(deps.as_mut().storage, q, env).unwrap(); - assert_eq!(res, None) - } - - #[test] - fn try_icq_locked_unbond_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let env = mock_env(); - - // lock the ibc lock - IBC_LOCK - .save(deps.as_mut().storage, &Lock::new().lock_unbond()) - .unwrap(); - - let qx: MockQuerier = MockQuerier::new(&[]); - let q = QuerierWrapper::new(&qx); - - let res = try_icq(deps.as_mut().storage, q, env).unwrap(); - assert_eq!(res, None) - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/integration_tests.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/integration_tests.rs deleted file mode 100644 index 7bb2e11be..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/integration_tests.rs +++ /dev/null @@ -1,420 +0,0 @@ -#[cfg(test)] -mod tests { - use cosmwasm_std::{attr, Addr, Empty}; - use cw_multi_test::{App, Contract, ContractWrapper, Executor}; - - use crate::{ - contract::{execute, instantiate}, - msg::{ExecuteMsg, InstantiateMsg, LockOnly, LockResponse, QueryMsg, UnlockOnly}, - queries::query, - }; - - #[test] - fn test_execute_lock() { - const ADMIN: &str = "admin"; - const CONTRACT_OWNER: &str = "contract_owner"; - - // returns an object that can be used with cw-multi-test - fn contract() -> Box> { - let contract = ContractWrapper::new(execute, instantiate, query); - Box::new(contract) - } - - // an app object is the blockchain simulator. we send initial balance here too if we need - let mut app = App::new(|_router, _api, _storage| {}); - - // upload the contracts to the blockchain and get back code_id to instantiate the contract later - let contract_code_id = app.store_code(contract()); - - // create the instantiate message - let instantiate_msg = InstantiateMsg { - lock_period: 100, - pool_id: 1, - pool_denom: "gamm/pool/1".to_string(), - base_denom: "uosmo".to_string(), - quote_denom: "uqsr".to_string(), - local_denom: "ibc/local_osmo".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }; - - // instantiate the contract - let contract_addr = app - .instantiate_contract( - contract_code_id, - Addr::unchecked(CONTRACT_OWNER), - &instantiate_msg, - &[], - "lp-strategy", - Some(ADMIN.to_owned()), - ) - .unwrap(); - - // lock the contract manually using the migration lock - let execute_msg = ExecuteMsg::Lock { - lock_only: LockOnly::Migration, - }; - let res = app - .execute_contract( - Addr::unchecked(ADMIN), - contract_addr.clone(), - &execute_msg, - &[], - ) - .unwrap(); - - assert_eq!(res.events[1].attributes[1], attr("lock_only", "migration")); - - // check the lock - let query_msg = QueryMsg::Lock {}; - let res: LockResponse = app - .wrap() - .query_wasm_smart(contract_addr.clone(), &query_msg) - .unwrap(); - - // migration lock should be the only one locked - assert!(res.lock.migration.is_locked()); - assert!(res.lock.bond.is_unlocked()); - assert!(res.lock.start_unbond.is_unlocked()); - assert!(res.lock.unbond.is_unlocked()); - assert!(res.lock.recovery.is_unlocked()); - - // lock the contract manually using the bond lock - let execute_msg = ExecuteMsg::Lock { - lock_only: LockOnly::Bond, - }; - let res = app - .execute_contract( - Addr::unchecked(ADMIN), - contract_addr.clone(), - &execute_msg, - &[], - ) - .unwrap(); - - assert_eq!(res.events[1].attributes[1], attr("lock_only", "bond")); - - // check the lock - let query_msg = QueryMsg::Lock {}; - let res: LockResponse = app - .wrap() - .query_wasm_smart(contract_addr.clone(), &query_msg) - .unwrap(); - - // bond & migration lock should be the only ones locked - assert!(res.lock.migration.is_locked()); - assert!(res.lock.bond.is_locked()); - assert!(res.lock.start_unbond.is_unlocked()); - assert!(res.lock.unbond.is_unlocked()); - assert!(res.lock.recovery.is_unlocked()); - - // lock the contract manually using the start unbond lock - let execute_msg = ExecuteMsg::Lock { - lock_only: LockOnly::StartUnbond, - }; - let res = app - .execute_contract( - Addr::unchecked(ADMIN), - contract_addr.clone(), - &execute_msg, - &[], - ) - .unwrap(); - - assert_eq!( - res.events[1].attributes[1], - attr("lock_only", "start_unbond") - ); - - // check the lock - let query_msg = QueryMsg::Lock {}; - let res: LockResponse = app - .wrap() - .query_wasm_smart(contract_addr.clone(), &query_msg) - .unwrap(); - - // bond, start unbond, & migration lock should be the only ones locked - assert!(res.lock.migration.is_locked()); - assert!(res.lock.bond.is_locked()); - assert!(res.lock.start_unbond.is_locked()); - assert!(res.lock.unbond.is_unlocked()); - assert!(res.lock.recovery.is_unlocked()); - - // lock the contract manually using the unbond lock - let execute_msg = ExecuteMsg::Lock { - lock_only: LockOnly::Unbond, - }; - let res = app - .execute_contract( - Addr::unchecked(ADMIN), - contract_addr.clone(), - &execute_msg, - &[], - ) - .unwrap(); - - assert_eq!(res.events[1].attributes[1], attr("lock_only", "unbond")); - - // check the lock - let query_msg = QueryMsg::Lock {}; - let res: LockResponse = app - .wrap() - .query_wasm_smart(contract_addr.clone(), &query_msg) - .unwrap(); - - // bond, start unbond, unbond, & migration lock should be the only ones locked - assert!(res.lock.migration.is_locked()); - assert!(res.lock.bond.is_locked()); - assert!(res.lock.start_unbond.is_locked()); - assert!(res.lock.unbond.is_locked()); - assert!(res.lock.recovery.is_unlocked()); - - // trying to lock the contract as a non-admin should fail - let execute_msg = ExecuteMsg::Lock { - lock_only: LockOnly::Migration, - }; - let res = app.execute_contract( - Addr::unchecked("not-admin"), - contract_addr, - &execute_msg, - &[], - ); - assert!(res.is_err()); - } - - #[test] - fn test_execute_unlock() { - const ADMIN: &str = "admin"; - const CONTRACT_OWNER: &str = "contract_owner"; - - // returns an object that can be used with cw-multi-test - fn contract() -> Box> { - let contract = ContractWrapper::new(execute, instantiate, query); - Box::new(contract) - } - - // an app object is the blockchain simulator. we send initial balance here too if we need - let mut app = App::new(|_router, _api, _storage| {}); - - // upload the contracts to the blockchain and get back code_id to instantiate the contract later - let contract_code_id = app.store_code(contract()); - - // create the instantiate message - let instantiate_msg = InstantiateMsg { - lock_period: 100, - pool_id: 1, - pool_denom: "gamm/pool/1".to_string(), - base_denom: "uosmo".to_string(), - quote_denom: "uqsr".to_string(), - local_denom: "ibc/local_osmo".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }; - - // instantiate the contract - let contract_addr = app - .instantiate_contract( - contract_code_id, - Addr::unchecked(CONTRACT_OWNER), - &instantiate_msg, - &[], - "lp-strategy", - Some(ADMIN.to_owned()), - ) - .unwrap(); - - // lock the contract manually to be able to unlock it later - let execute_msg = ExecuteMsg::Lock { - lock_only: LockOnly::Migration, - }; - app.execute_contract( - Addr::unchecked(ADMIN), - contract_addr.clone(), - &execute_msg, - &[], - ) - .unwrap(); - - // lock the contract manually using the bond lock - let execute_msg = ExecuteMsg::Lock { - lock_only: LockOnly::Bond, - }; - app.execute_contract( - Addr::unchecked(ADMIN), - contract_addr.clone(), - &execute_msg, - &[], - ) - .unwrap(); - - // lock the contract manually using the start unbond lock - let execute_msg = ExecuteMsg::Lock { - lock_only: LockOnly::StartUnbond, - }; - app.execute_contract( - Addr::unchecked(ADMIN), - contract_addr.clone(), - &execute_msg, - &[], - ) - .unwrap(); - - // lock the contract manually using the unbond lock - let execute_msg = ExecuteMsg::Lock { - lock_only: LockOnly::Unbond, - }; - app.execute_contract( - Addr::unchecked(ADMIN), - contract_addr.clone(), - &execute_msg, - &[], - ) - .unwrap(); - - // check the lock - let query_msg = QueryMsg::Lock {}; - let res: LockResponse = app - .wrap() - .query_wasm_smart(contract_addr.clone(), &query_msg) - .unwrap(); - - // bond, start unbond, unbond, & migration lock should be locked - assert!(res.lock.migration.is_locked()); - assert!(res.lock.bond.is_locked()); - assert!(res.lock.start_unbond.is_locked()); - assert!(res.lock.unbond.is_locked()); - assert!(res.lock.recovery.is_unlocked()); - - // unlock the contract manually using the migration lock - let execute_msg = ExecuteMsg::Unlock { - unlock_only: UnlockOnly::Migration, - }; - let res = app - .execute_contract( - Addr::unchecked(ADMIN), - contract_addr.clone(), - &execute_msg, - &[], - ) - .unwrap(); - - assert_eq!( - res.events[1].attributes[1], - attr("unlock_only", "migration") - ); - - // check the lock - let query_msg = QueryMsg::Lock {}; - let res: LockResponse = app - .wrap() - .query_wasm_smart(contract_addr.clone(), &query_msg) - .unwrap(); - - // bond, start unbond, unbond, & migration lock should be the only ones locked - assert!(res.lock.migration.is_unlocked()); - assert!(res.lock.bond.is_locked()); - assert!(res.lock.start_unbond.is_locked()); - assert!(res.lock.unbond.is_locked()); - assert!(res.lock.recovery.is_unlocked()); - - // unlock the contract manually using the bond lock - let execute_msg = ExecuteMsg::Unlock { - unlock_only: UnlockOnly::Bond, - }; - let res = app - .execute_contract( - Addr::unchecked(ADMIN), - contract_addr.clone(), - &execute_msg, - &[], - ) - .unwrap(); - - assert_eq!(res.events[1].attributes[1], attr("unlock_only", "bond")); - - // check the lock - let query_msg = QueryMsg::Lock {}; - let res: LockResponse = app - .wrap() - .query_wasm_smart(contract_addr.clone(), &query_msg) - .unwrap(); - - // bond, start unbond, unbond, & migration lock should be the only ones locked - assert!(res.lock.migration.is_unlocked()); - assert!(res.lock.bond.is_unlocked()); - assert!(res.lock.start_unbond.is_locked()); - assert!(res.lock.unbond.is_locked()); - assert!(res.lock.recovery.is_unlocked()); - - // unlock the contract manually using the start unbond lock - let execute_msg = ExecuteMsg::Unlock { - unlock_only: UnlockOnly::StartUnbond, - }; - let res = app - .execute_contract( - Addr::unchecked(ADMIN), - contract_addr.clone(), - &execute_msg, - &[], - ) - .unwrap(); - - assert_eq!( - res.events[1].attributes[1], - attr("unlock_only", "start_unbond") - ); - - // check the lock - let query_msg = QueryMsg::Lock {}; - let res: LockResponse = app - .wrap() - .query_wasm_smart(contract_addr.clone(), &query_msg) - .unwrap(); - - // bond, start unbond, unbond, & migration lock should be the only ones locked - assert!(res.lock.migration.is_unlocked()); - assert!(res.lock.bond.is_unlocked()); - assert!(res.lock.start_unbond.is_unlocked()); - assert!(res.lock.unbond.is_locked()); - assert!(res.lock.recovery.is_unlocked()); - - // unlock the contract manually using the unbond lock - let execute_msg = ExecuteMsg::Unlock { - unlock_only: UnlockOnly::Unbond, - }; - let res = app - .execute_contract( - Addr::unchecked(ADMIN), - contract_addr.clone(), - &execute_msg, - &[], - ) - .unwrap(); - - assert_eq!(res.events[1].attributes[1], attr("unlock_only", "unbond")); - - // check the lock - let query_msg = QueryMsg::Lock {}; - let res: LockResponse = app - .wrap() - .query_wasm_smart(contract_addr.clone(), &query_msg) - .unwrap(); - - // everything should be unlocked - assert!(res.lock.is_unlocked()); - - // trying to unlock the contract as a non-admin should fail - let execute_msg = ExecuteMsg::Unlock { - unlock_only: UnlockOnly::Migration, - }; - let res = app.execute_contract( - Addr::unchecked("non-admin"), - contract_addr, - &execute_msg, - &[], - ); - assert!(res.is_err()); - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/lib.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/lib.rs deleted file mode 100644 index ea9fccb3f..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/lib.rs +++ /dev/null @@ -1,31 +0,0 @@ -mod admin; -pub mod bond; -pub mod contract; -pub mod error; -mod error_recovery; -pub mod execute; -pub mod helpers; -pub mod ibc; -pub mod ibc_lock; -mod ibc_util; -pub mod icq; -pub mod msg; -pub mod queries; -pub mod reply; -pub mod start_unbond; -pub mod state; -pub mod unbond; - -#[cfg(test)] -pub mod integration_tests; -pub mod proptests; -pub mod test_helpers; - -#[cfg(test)] -mod tests { - #[test] - fn it_works() { - let result = 2 + 2; - assert_eq!(result, 4); - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/msg.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/msg.rs deleted file mode 100644 index 14a82d17b..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/msg.rs +++ /dev/null @@ -1,285 +0,0 @@ -use std::{ - collections::HashMap, - fmt::{Display, Formatter}, -}; - -use cosmwasm_schema::{cw_serde, QueryResponses}; -use cosmwasm_std::{Addr, Coin, IbcPacketAckMsg, StdResult, Uint128}; - -pub use cw20::BalanceResponse; -use quasar_types::ibc::ChannelInfo; - -use crate::{ - bond::Bond, - error::Trap, - helpers::{IbcMsgKind, SubMsgKind}, - ibc_lock, - start_unbond::StartUnbond, - state::{Config, LpCache, OngoingDeposit, Unbond}, - unbond::PendingReturningUnbonds, -}; - -#[cw_serde] -pub struct InstantiateMsg { - pub lock_period: u64, - pub pool_id: u64, // 2 - pub pool_denom: String, // gamm/pool/2 - // if setup correctly, local_denom on quasar == base_denom on osmosis - pub local_denom: String, // ibc/ED07 - pub base_denom: String, // uosmo - pub quote_denom: String, // uatom - // TODO should this be outgoing_transfer_channel? - pub transfer_channel: String, - // TODO rename to return_transfer_channel - pub return_source_channel: String, - pub expected_connection: String, -} - -impl InstantiateMsg { - pub fn validate(&self) -> StdResult<()> { - Ok(()) - } -} - -#[cw_serde] -pub struct MigrateMsg { - pub delete_pending_acks: Vec<(u64, String)>, - pub delete_traps: Vec<(u64, String)>, -} - -#[cw_serde] -#[derive(QueryResponses)] -pub enum QueryMsg { - #[returns(ChannelsResponse)] - Channels {}, - #[returns(ConfigResponse)] - Config {}, - #[returns(IcaAddressResponse)] - Balance { address: String }, - #[returns(BalanceResponse)] - IcaAddress {}, - #[returns(LockResponse)] - Lock {}, - #[returns(LpSharesResponse)] - LpShares {}, - #[returns(PrimitiveSharesResponse)] - PrimitiveShares {}, - #[returns(IcaBalanceResponse)] - IcaBalance {}, - #[returns(IcaChannelResponse)] - IcaChannel {}, - #[returns(TrappedErrorsResponse)] - TrappedErrors {}, - #[returns(UnbondingClaimResponse)] - UnbondingClaim { addr: Addr, id: String }, - #[returns(ListUnbondingClaimsResponse)] - ListUnbondingClaims {}, - #[returns(ListBondingClaimsResponse)] - ListBondingClaims {}, - #[returns(ListPrimitiveSharesResponse)] - ListPrimitiveShares {}, - #[returns(ListPendingAcksResponse)] - ListPendingAcks {}, - #[returns(ListRepliesResponse)] - ListReplies {}, - #[returns(ListClaimableFundsResponse)] - ListClaimableFunds {}, - #[returns(OsmoLockResponse)] - OsmoLock {}, - #[returns(SimulatedJoinResponse)] - SimulatedJoin {}, - #[returns(GetQueuesResponse)] - GetQueues {}, -} - -#[cw_serde] -pub struct GetQueuesResponse { - pub pending_bond_queue: Vec, - pub bond_queue: Vec, - pub start_unbond_queue: Vec, - pub unbond_queue: Vec, - pub failed_join_queue: Vec, - pub rejoin_queue: Vec, - pub pending_unbond_queue: Vec, -} - -#[cw_serde] -pub struct SimulatedJoinResponse { - pub amount: Option, - pub result: Option, -} - -#[cw_serde] -pub struct OsmoLockResponse { - pub lock_id: u64, -} - -#[cw_serde] -pub struct ListBondingClaimsResponse { - pub bonds: HashMap, -} - -#[cw_serde] -pub struct ListRepliesResponse { - pub replies: HashMap, -} - -#[cw_serde] -pub struct ListClaimableFundsResponse { - pub claimable_funds: HashMap, -} - -#[cw_serde] -pub struct ListPrimitiveSharesResponse { - pub shares: HashMap, -} - -#[cw_serde] -pub struct ListPendingAcksResponse { - pub pending: HashMap, -} - -#[cw_serde] -pub struct ListUnbondingClaimsResponse { - pub unbonds: HashMap, - pub pending_unbonds: HashMap, -} - -#[cw_serde] -pub struct UnbondingClaimResponse { - pub unbond: Option, -} - -#[cw_serde] -pub struct ChannelsResponse { - pub channels: Vec, -} - -#[cw_serde] -pub struct TrappedErrorsResponse { - pub errors: HashMap, -} - -#[cw_serde] -pub struct LpSharesResponse { - pub lp_shares: LpCache, -} - -#[cw_serde] -pub struct ConfigResponse { - pub config: Config, -} - -#[cw_serde] -pub struct LockResponse { - pub lock: ibc_lock::Lock, -} - -#[cw_serde] -pub struct IcaAddressResponse { - pub address: String, -} - -#[cw_serde] -pub struct PrimitiveSharesResponse { - pub total: Uint128, -} - -#[cw_serde] -pub struct IcaBalanceResponse { - pub amount: Coin, -} - -#[cw_serde] -pub struct IcaChannelResponse { - pub channel: String, -} - -#[cw_serde] -pub enum UnlockOnly { - Bond, - StartUnbond, - Unbond, - Migration, -} - -impl Display for UnlockOnly { - fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result { - match self { - UnlockOnly::Bond => write!(f, "bond"), - UnlockOnly::StartUnbond => write!(f, "start_unbond"), - UnlockOnly::Unbond => write!(f, "unbond"), - UnlockOnly::Migration => write!(f, "migration"), - } - } -} - -#[cw_serde] -pub enum LockOnly { - Bond, - StartUnbond, - Unbond, - Migration, -} - -impl Display for LockOnly { - fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result { - match self { - LockOnly::Bond => write!(f, "bond"), - LockOnly::StartUnbond => write!(f, "start_unbond"), - LockOnly::Unbond => write!(f, "unbond"), - LockOnly::Migration => write!(f, "migration"), - } - } -} - -#[cw_serde] -pub enum ExecuteMsg { - Bond { - id: String, - }, - StartUnbond { - id: String, - share_amount: Uint128, - }, - Unbond { - id: String, - }, - SetDepositor { - depositor: String, - }, - // accept a dispatched transfer from osmosis - AcceptReturningFunds { - id: u64, - pending: PendingReturningUnbonds, - }, - // try to close a channel where a timout occured - CloseChannel { - channel_id: String, - }, - Ack { - ack: IbcPacketAckMsg, - }, - TryIcq {}, - Unlock { - unlock_only: UnlockOnly, - }, - Lock { - lock_only: LockOnly, - }, - AddLockAdmin { - to_add: String, - }, - RemoveLockAdmin { - to_remove: String, - }, - ManualTimeout { - seq: u64, - channel: String, - should_unlock: bool, - }, - Retry { - seq: u64, - channel: String, - }, -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/proptests.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/proptests.rs deleted file mode 100644 index f52017bde..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/proptests.rs +++ /dev/null @@ -1,291 +0,0 @@ -#[cfg(test)] -mod tests { - use cosmwasm_std::{ - attr, - testing::{mock_dependencies, mock_env}, - to_json_binary, Addr, Binary, Coin, CosmosMsg, Empty, IbcMsg, MessageInfo, StdError, - Uint128, - }; - use proptest::prelude::*; - use prost::Message; - use quasar_types::icq::{CosmosResponse, InterchainQueryPacketAck}; - - use crate::{ - bond::Bond, - contract::execute_try_icq, - error::Trap, - execute::execute_retry, - helpers::{IbcMsgKind, IcaMessages}, - ibc::handle_icq_ack, - ibc_lock::Lock, - state::{ - OngoingDeposit, PendingBond, RawAmount, FAILED_JOIN_QUEUE, IBC_LOCK, LOCK_ADMIN, - PENDING_BOND_QUEUE, REJOIN_QUEUE, TRAPS, - }, - test_helpers::{create_query_response, default_setup, pending_bond_to_bond}, - }; - use osmosis_std::types::cosmos::base::v1beta1::Coin as OsmoCoin; - use osmosis_std::types::{ - cosmos::bank::v1beta1::QueryBalanceResponse, - osmosis::gamm::v1beta1::{ - QueryCalcExitPoolCoinsFromSharesResponse, QueryCalcJoinPoolSharesResponse, - QuerySpotPriceResponse, - }, - }; - use proptest::collection::vec; - - proptest! { - - #[test] - fn test_handle_retry_join_pool_with_pending_deposits_works( - // values to mock failed join pool - (claim_amount, raw_amount, owner, bond_id) in (0usize..100).prop_flat_map(|size| - ( - // to avoid overflows, we limit the amounts to u64. also force amounts & bond_ids to be >= 1 - vec(any::().prop_map(|x| (x as u128).max(1)), size..=size), - vec(any::().prop_map(|x| (x as u128).max(1)), size..=size), - vec("[a-z]+", size..=size), - vec(any::().prop_map(|x| (x as u128).max(1)), size..=size), - ) - ), - // values to mock pending deposits - (amount_pd, owner_pd, bond_id_pd) in (0usize..100).prop_flat_map(|size| - ( - // to avoid overflows, we limit the amounts to u64. also force amounts & bond_ids to be >= 1 - vec(any::().prop_map(|x| (x as u128).max(1)), size..=size), - vec("[a-z]+", size..=size), - vec(any::().prop_map(|x| (x as u128).max(1)), size..=size), - ) - ), - // values to mock ICQ ACK - raw_balalance_rq in any::(), - quote_balance_rq in any::(), - lp_balance_rq in any::(), - join_pool_rq in any::(), - exit_pool_base_rq in any::(), - exit_pool_quote_rq in any::(), - spot_price_rq in any::(), - ) { - let mut deps = mock_dependencies(); - let env = mock_env(); - default_setup(deps.as_mut().storage).unwrap(); - - IBC_LOCK.save(deps.as_mut().storage, &Lock::new()).unwrap(); - - LOCK_ADMIN - .save(deps.as_mut().storage, &Addr::unchecked("admin"), &Empty {}) - .unwrap(); - - // mock the failed join pool trap with 3 bonds - let failed = PendingBond { - bonds: claim_amount.iter().zip(&raw_amount).zip(&owner).zip(&bond_id).map(|(((_claim, raw), owner), id)| { - OngoingDeposit { - claim_amount: Uint128::new(*raw), - raw_amount: RawAmount::LocalDenom(Uint128::new(*raw)), - owner: Addr::unchecked(owner), - bond_id: id.to_string(), - } - }).collect(), - }; - - TRAPS - .save( - deps.as_mut().storage, - (3539, "channel-35".to_string()), - &Trap { - error: "join pool failed on osmosis".to_string(), - step: IbcMsgKind::Ica(IcaMessages::JoinSwapExternAmountIn(failed.clone())), - last_succesful: true, - }, - ) - .unwrap(); - - // mock pending deposits and add them to the pending queue - let pending_bonds: Vec = amount_pd.iter().zip(&owner_pd).zip(&bond_id_pd).map(|((amount, owner), id)| { - Bond { - amount: Uint128::new(*amount), - owner: Addr::unchecked(owner), - bond_id: id.to_string(), - } - }).collect(); - - for bond in pending_bonds.iter() { - PENDING_BOND_QUEUE - .push_back(deps.as_mut().storage, bond) - .unwrap(); - } - - // manually trigger retry join pool - let res = execute_retry( - deps.as_mut(), - env.clone(), - MessageInfo { - sender: Addr::unchecked("admin"), - funds: vec![], - }, - 3539, - "channel-35".to_string(), - ) - .unwrap(); - - prop_assert!(!TRAPS.has(&deps.storage, (3539, "channel-35".to_string()))); - - let mut attributes = vec![ - attr("action", "retry"), - attr("kind", "join_pool"), - ]; - - for bond in &failed.bonds { - if let RawAmount::LocalDenom(amount) = bond.raw_amount { - attributes.push(attr("bond_id", &bond.bond_id)); - attributes.push(attr("amount", amount)); - } - } - - prop_assert_eq!( - res.attributes, - attributes - ); - - // check that the failed join queue has the same mocked bonds - let failed_join_queue: Result, StdError> = - FAILED_JOIN_QUEUE.iter(&deps.storage).unwrap().collect(); - prop_assert_eq!(failed_join_queue.unwrap(), pending_bond_to_bond(&failed)); - - - // manually trigger try_icq - let res = execute_try_icq(deps.as_mut(), env.clone()); - prop_assert_eq!(res.unwrap().messages.len(), 1); - - // mocking the ICQ ACK - let raw_balance = create_query_response( - QueryBalanceResponse { - balance: Some(OsmoCoin { - denom: "uatom".to_string(), - amount: raw_balalance_rq.to_string(), - }), - } - .encode_to_vec(), - ); - - let quote_balance = create_query_response( - QueryBalanceResponse { - balance: Some(OsmoCoin { - denom: "uosmo".to_string(), - amount: quote_balance_rq.to_string(), - }), - } - .encode_to_vec(), - ); - - let lp_balance = create_query_response( - QueryBalanceResponse { - balance: Some(OsmoCoin { - denom: "uosmo".to_string(), - amount: lp_balance_rq.to_string(), - }), - } - .encode_to_vec(), - ); - - let exit_pool = create_query_response( - QueryCalcExitPoolCoinsFromSharesResponse { - tokens_out: vec![ - OsmoCoin { - // base denom - denom: "uosmo".to_string(), - amount: exit_pool_base_rq.to_string(), - }, - OsmoCoin { - // quote denom - denom: "uqsr".to_string(), - amount: exit_pool_quote_rq.to_string(), - }, - ], - } - .encode_to_vec(), - ); - - let spot_price = create_query_response( - QuerySpotPriceResponse { - spot_price: (spot_price_rq+1).to_string(), - } - .encode_to_vec(), - ); - - let join_pool = create_query_response( - QueryCalcJoinPoolSharesResponse { - share_out_amount: "123".to_string(), - tokens_out: vec![OsmoCoin { - denom: "uosmo".to_string(), - amount: join_pool_rq.to_string(), - }], - } - .encode_to_vec(), - ); - - // LockResponse is fixed to None in this test for simplicity - // let lock = create_query_response(LockedResponse { lock: None }.encode_to_vec()); - - let ibc_ack = InterchainQueryPacketAck { - data: Binary::from( - &CosmosResponse { - responses: vec![ - raw_balance, - quote_balance, - lp_balance, - exit_pool, - spot_price, - join_pool, - // lock, - ], - } - .encode_to_vec()[..], - ), - }; - - // simulate that we received the ICQ ACK - let res = handle_icq_ack(deps.as_mut().storage, env, to_json_binary(&ibc_ack).unwrap()).unwrap(); - - // get the pending bonds total amount - let pending_total_amount = pending_bonds.iter().fold(Uint128::zero(), |acc, bond| { - acc + bond.amount - }); - - // check that the res amount matches the amount in the pending queue ONLY - // only if there are messages - if !res.messages.is_empty() { - match &res.messages[0].msg { - CosmosMsg::Ibc(IbcMsg::Transfer { amount, .. }) => { - assert_eq!( - amount, - &Coin { - denom: "ibc/local_osmo".to_string(), - amount: pending_total_amount, - } - ); - } - _ => panic!("unexpected message type"), - }; - } - - - // if BOND_QUEUE & REJOIN_QUEUE are empty FAILED_JOIN_QUEUE items are not moved to REJOIN_QUEUE - if !pending_bonds.is_empty() && !failed.bonds.is_empty() { - prop_assert!(FAILED_JOIN_QUEUE.is_empty(&deps.storage).unwrap()); - } - - // PENDING_BOND_QUEUE should be empty - prop_assert!(PENDING_BOND_QUEUE.is_empty(&deps.storage).unwrap()); - - // failed bonds should be now in the REJOIN_QUEUE - let rejoin_queue: Result, StdError> = - REJOIN_QUEUE.iter(&deps.storage).unwrap().collect(); - - // only check when there's pending bonds & failed bonds - if !pending_bonds.is_empty() && !failed.bonds.is_empty() { - assert_eq!(failed.bonds, rejoin_queue.unwrap()); - } - } - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/queries.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/queries.rs deleted file mode 100644 index f8e1675e9..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/queries.rs +++ /dev/null @@ -1,336 +0,0 @@ -#[cfg(not(feature = "library"))] -use cosmwasm_std::entry_point; -use cw20::BalanceResponse; -use std::collections::HashMap; - -use cosmwasm_std::{ - to_json_binary, Addr, Binary, Coin, Deps, Env, Order, StdError, StdResult, Uint128, -}; -use quasar_types::ibc::ChannelInfo; - -use crate::{ - bond::Bond, - error::Trap, - helpers::{get_ica_address, get_total_primitive_shares, IbcMsgKind, SubMsgKind}, - msg::{ - ChannelsResponse, ConfigResponse, GetQueuesResponse, IcaAddressResponse, - IcaBalanceResponse, IcaChannelResponse, ListBondingClaimsResponse, - ListClaimableFundsResponse, ListPendingAcksResponse, ListPrimitiveSharesResponse, - ListRepliesResponse, ListUnbondingClaimsResponse, LockResponse, LpSharesResponse, - OsmoLockResponse, PrimitiveSharesResponse, QueryMsg, SimulatedJoinResponse, - TrappedErrorsResponse, UnbondingClaimResponse, - }, - start_unbond::StartUnbond, - state::{ - FundPath, OngoingDeposit, Unbond, BONDING_CLAIMS, BOND_QUEUE, CHANNELS, CLAIMABLE_FUNDS, - CONFIG, FAILED_JOIN_QUEUE, IBC_LOCK, ICA_CHANNEL, LP_SHARES, OSMO_LOCK, PENDING_ACK, - PENDING_BOND_QUEUE, PENDING_UNBONDING_CLAIMS, PENDING_UNBOND_QUEUE, REJOIN_QUEUE, REPLIES, - SHARES, SIMULATED_JOIN_AMOUNT_IN, SIMULATED_JOIN_RESULT, START_UNBOND_QUEUE, - TOTAL_VAULT_BALANCE, TRAPS, UNBONDING_CLAIMS, UNBOND_QUEUE, - }, -}; - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn query(deps: Deps, _env: Env, msg: QueryMsg) -> StdResult { - match msg { - QueryMsg::Channels {} => to_json_binary(&handle_channels_query(deps)?), - QueryMsg::Config {} => to_json_binary(&handle_config_query(deps)?), - QueryMsg::IcaAddress {} => to_json_binary(&handle_ica_address_query(deps)?), - QueryMsg::Balance { address } => to_json_binary(&handle_balance_query(deps, &address)?), - QueryMsg::PrimitiveShares {} => to_json_binary(&handle_primitive_shares(deps)?), - QueryMsg::IcaBalance {} => to_json_binary(&handle_ica_balance(deps)?), - QueryMsg::IcaChannel {} => to_json_binary(&handle_ica_channel(deps)?), - QueryMsg::Lock {} => to_json_binary(&handle_lock(deps)?), - QueryMsg::LpShares {} => to_json_binary(&handle_lp_shares_query(deps)?), - QueryMsg::TrappedErrors {} => to_json_binary(&handle_trapped_errors_query(deps)?), - QueryMsg::ListUnbondingClaims {} => to_json_binary(&handle_list_unbonding_claims(deps)?), - QueryMsg::UnbondingClaim { addr, id } => { - to_json_binary(&handle_unbonding_claim_query(deps, addr, id)?) - } - QueryMsg::ListBondingClaims {} => to_json_binary(&handle_list_bonding_claims(deps)?), - QueryMsg::ListPrimitiveShares {} => to_json_binary(&handle_list_primitive_shares(deps)?), - QueryMsg::ListPendingAcks {} => to_json_binary(&handle_list_pending_acks(deps)?), - QueryMsg::ListReplies {} => to_json_binary(&handle_list_replies(deps)?), - QueryMsg::ListClaimableFunds {} => to_json_binary(&handle_list_claimable_funds(deps)?), - QueryMsg::OsmoLock {} => to_json_binary(&handle_osmo_lock(deps)?), - QueryMsg::SimulatedJoin {} => to_json_binary(&handle_simulated_join(deps)?), - QueryMsg::GetQueues {} => to_json_binary(&handle_get_queues(deps)?), - } -} - -pub fn handle_get_queues(deps: Deps) -> StdResult { - let pbq: Result, StdError> = PENDING_BOND_QUEUE.iter(deps.storage)?.collect(); - let bq: Result, StdError> = BOND_QUEUE.iter(deps.storage)?.collect(); - let suq: Result, StdError> = START_UNBOND_QUEUE.iter(deps.storage)?.collect(); - let uq: Result, StdError> = UNBOND_QUEUE.iter(deps.storage)?.collect(); - let fjq: Result, StdError> = FAILED_JOIN_QUEUE.iter(deps.storage)?.collect(); - let rj: Result, StdError> = REJOIN_QUEUE.iter(deps.storage)?.collect(); - let puq: Result, StdError> = PENDING_UNBOND_QUEUE.iter(deps.storage)?.collect(); - Ok(GetQueuesResponse { - pending_bond_queue: pbq?, - bond_queue: bq?, - start_unbond_queue: suq?, - unbond_queue: uq?, - failed_join_queue: fjq?, - rejoin_queue: rj?, - pending_unbond_queue: puq?, - }) -} - -pub fn handle_simulated_join(deps: Deps) -> StdResult { - Ok(SimulatedJoinResponse { - amount: SIMULATED_JOIN_AMOUNT_IN.may_load(deps.storage)?, - result: SIMULATED_JOIN_RESULT.may_load(deps.storage)?, - }) -} - -pub fn handle_osmo_lock(deps: Deps) -> StdResult { - Ok(OsmoLockResponse { - lock_id: OSMO_LOCK.load(deps.storage)?, - }) -} - -pub fn handle_list_unbonding_claims(deps: Deps) -> StdResult { - let unbonds: StdResult> = UNBONDING_CLAIMS - .range(deps.storage, None, None, Order::Ascending) - .map(|res| { - let val = res?; - Ok((val.0 .0, (val.0 .1, val.1))) - }) - .collect(); - let pending_unbonds: StdResult> = PENDING_UNBONDING_CLAIMS - .range(deps.storage, None, None, Order::Ascending) - .map(|res| { - let val = res?; - Ok((val.0 .0, (val.0 .1, val.1))) - }) - .collect(); - Ok(ListUnbondingClaimsResponse { - unbonds: unbonds?, - pending_unbonds: pending_unbonds?, - }) -} - -pub fn handle_unbonding_claim_query( - deps: Deps, - addr: Addr, - id: String, -) -> StdResult { - Ok(UnbondingClaimResponse { - unbond: UNBONDING_CLAIMS.may_load(deps.storage, (addr, id))?, - }) -} - -pub fn handle_trapped_errors_query(deps: Deps) -> StdResult { - let trapped: StdResult> = TRAPS - .range(deps.storage, None, None, Order::Ascending) - .map(|res| { - let ((seq, chan), kind) = res?; - Ok((format!("{seq}-{chan}"), kind)) - }) - .collect(); - Ok(TrappedErrorsResponse { errors: trapped? }) -} - -pub fn handle_channels_query(deps: Deps) -> StdResult { - let channels: Vec = CHANNELS - .range(deps.storage, None, None, Order::Ascending) - .map(|kv| kv.unwrap().1) - .collect(); - Ok(ChannelsResponse { channels }) -} - -pub fn handle_lp_shares_query(deps: Deps) -> StdResult { - Ok(LpSharesResponse { - lp_shares: LP_SHARES.load(deps.storage)?, - }) -} - -pub fn handle_config_query(deps: Deps) -> StdResult { - Ok(ConfigResponse { - config: CONFIG.load(deps.storage)?, - }) -} - -pub fn handle_ica_address_query(deps: Deps) -> StdResult { - Ok(IcaAddressResponse { - address: get_ica_address(deps.storage, ICA_CHANNEL.load(deps.storage)?) - .expect("ica address setup correctly"), - }) -} - -pub fn handle_balance_query(deps: Deps, address: &str) -> StdResult { - Ok(BalanceResponse { - balance: SHARES.load(deps.storage, deps.api.addr_validate(address)?)?, - }) -} - -pub fn handle_ica_channel(deps: Deps) -> StdResult { - Ok(IcaChannelResponse { - channel: ICA_CHANNEL.load(deps.storage)?, - }) -} - -pub fn handle_primitive_shares(deps: Deps) -> StdResult { - let total = get_total_primitive_shares(deps.storage).map_err(|err| StdError::GenericErr { - msg: err.to_string(), - })?; - Ok(PrimitiveSharesResponse { total }) -} - -pub fn handle_ica_balance(deps: Deps) -> StdResult { - let amount = TOTAL_VAULT_BALANCE.load(deps.storage)?; - - Ok(IcaBalanceResponse { - amount: Coin { - denom: CONFIG.load(deps.storage)?.local_denom, - amount, - }, - }) -} - -pub fn handle_lock(deps: Deps) -> StdResult { - Ok(LockResponse { - lock: IBC_LOCK - .load(deps.storage) - .map_err(|err| StdError::GenericErr { - msg: err.to_string(), - })?, - }) -} - -pub fn handle_list_bonding_claims(deps: Deps) -> StdResult { - let bonds: StdResult> = BONDING_CLAIMS - .range(deps.storage, None, None, Order::Ascending) - .map(|res| { - let val = res?; - Ok((val.0 .0, (val.0 .1, val.1))) - }) - .collect(); - Ok(ListBondingClaimsResponse { bonds: bonds? }) -} - -pub fn handle_list_primitive_shares(deps: Deps) -> StdResult { - let shares: StdResult> = SHARES - .range(deps.storage, None, None, Order::Ascending) - .collect(); - Ok(ListPrimitiveSharesResponse { shares: shares? }) -} - -pub fn handle_list_pending_acks(deps: Deps) -> StdResult { - let pending: StdResult> = PENDING_ACK - .range(deps.storage, None, None, Order::Ascending) - .map(|res| { - let ((seq, chan), kind) = res?; - Ok((format!("{seq}-{chan}"), kind)) - }) - .collect(); - Ok(ListPendingAcksResponse { pending: pending? }) -} - -pub fn handle_list_replies(deps: Deps) -> StdResult { - let replies: StdResult> = REPLIES - .range(deps.storage, None, None, Order::Ascending) - .collect(); - Ok(ListRepliesResponse { replies: replies? }) -} - -pub fn handle_list_claimable_funds(deps: Deps) -> StdResult { - let funds = CLAIMABLE_FUNDS.range(deps.storage, None, None, Order::Ascending); - - let mut claimable_funds: HashMap = HashMap::new(); - for fund in funds { - let ((addr, fp), amount) = fund?; - let path; - let seq = match fp { - FundPath::Bond { id } => { - path = "bond"; - id - } - FundPath::Unbond { id } => { - path = "unbond"; - id - } - }; - claimable_funds.insert(format!("{addr}-{seq}-{path}"), amount); - } - - Ok(ListClaimableFundsResponse { claimable_funds }) -} - -#[cfg(test)] -mod tests { - use cosmwasm_std::{ - from_json, - testing::{mock_dependencies, mock_env}, - }; - - use crate::state::FundPath; - - use super::*; - - #[test] - fn get_trapped_errors_works() { - let mut deps = mock_dependencies(); - let env = mock_env(); - - let q = QueryMsg::TrappedErrors {}; - - TRAPS - .save( - deps.as_mut().storage, - (100, "channel-1".to_string()), - &Trap { - error: "failed to do a thing".to_string(), - step: IbcMsgKind::Icq, - last_succesful: true, - }, - ) - .unwrap(); - - let _res = query(deps.as_ref(), env, q).unwrap(); - } - - #[test] - fn get_trapped_errors_when_empty() { - let deps = mock_dependencies(); - let env = mock_env(); - - let q = QueryMsg::TrappedErrors {}; - - let res: TrappedErrorsResponse = from_json(&query(deps.as_ref(), env, q).unwrap()).unwrap(); - - assert!(res.errors.is_empty()); - } - - #[test] - fn proper_get_claimable_funds() { - let mut deps = mock_dependencies(); - let env = mock_env(); - - let q = QueryMsg::ListClaimableFunds {}; - - CLAIMABLE_FUNDS - .save( - deps.as_mut().storage, - ( - Addr::unchecked("somedepositor"), - FundPath::Bond { - id: "channel-1".to_string(), - }, - ), - &Uint128::new(100), - ) - .unwrap(); - - let res = query(deps.as_ref(), env, q).unwrap(); - let claimable_funds: ListClaimableFundsResponse = from_json(&res).unwrap(); - - println!("{claimable_funds:?}"); - assert_eq!(claimable_funds.claimable_funds.len(), 1); - assert_eq!( - claimable_funds.claimable_funds["somedepositor-\u{1}\0channel-1-bond"], - Uint128::new(100) - ); - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/reply.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/reply.rs deleted file mode 100644 index 5b91dd0dd..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/reply.rs +++ /dev/null @@ -1,370 +0,0 @@ -use cosmwasm_std::{Addr, BankMsg, StdError}; -use cosmwasm_std::{DepsMut, Reply, Response}; -use quasar_types::callback::Callback; - -use crate::error::{ContractError, Trap}; -use crate::helpers::{parse_seq, unlock_on_error, ContractCallback, IbcMsgKind}; -use crate::state::{FundPath, CLAIMABLE_FUNDS, PENDING_ACK, REPLIES, TRAPS}; - -pub fn handle_ibc_reply( - deps: DepsMut, - msg: Reply, - pending: IbcMsgKind, - channel: String, -) -> Result { - let data = msg - .result - .into_result() - .map_err(|msg| StdError::GenericErr { - msg: format!("submsg error: {msg:?}"), - })? - .data - .ok_or(ContractError::NoReplyData) - .map_err(|_| StdError::NotFound { - kind: "reply-data".to_string(), - })?; - - let seq = parse_seq(data).map_err(|err| StdError::SerializeErr { - source_type: "protobuf-decode".to_string(), - msg: err.to_string(), - })?; - - PENDING_ACK.save(deps.storage, (seq, channel), &pending)?; - - // cleanup the REPLIES state item - REPLIES.remove(deps.storage, msg.id); - - Ok(Response::default() - .add_attribute("pending-msg", seq.to_string()) - .add_attribute("step", format!("{pending:?}"))) -} - -pub fn handle_ack_reply( - deps: DepsMut, - msg: Reply, - seq: u64, - channel: String, -) -> Result { - let mut resp = Response::new(); - - // if we have an error in our Ack execution, the submsg saves the error in TRAPS and (should) rollback - // the entire state of the ack execution, - if let Err(error) = msg.result.into_result() { - let step = PENDING_ACK.load(deps.storage, (seq, channel.clone()))?; - unlock_on_error(deps.storage, &step)?; - - // reassignment needed since add_attribute - resp = resp.add_attribute("trapped-error", error.as_str()); - - TRAPS.save( - deps.storage, - (seq, channel), - &Trap { - error, - step, - last_succesful: true, - }, - )?; - } - // if we did not error, we can safely remove the ack entry from the contract - else { - PENDING_ACK.remove(deps.storage, (seq, channel)) - } - - // // cleanup the REPLIES state item - REPLIES.remove(deps.storage, msg.id); - Ok(resp.add_attribute("register-ack-seq", seq.to_string())) -} - -pub fn handle_callback_reply( - deps: DepsMut, - msg: Reply, - callback: ContractCallback, -) -> Result { - // TODO: if error, add manual withdraws to lp-strategy - // - // create in claimable_funds map... Addr, unbond_id -> amount - // in Callback contract add callbacl(callback, amount) - let mut res = Response::new(); - - if let Err(error) = msg.result.clone().into_result() { - match callback.clone() { - // if unbond response callback message, add the amount to the claimable funds map - ContractCallback::Callback { - callback, - amount, - owner, - } => { - if let Callback::UnbondResponse(ur) = callback { - let fund_path = FundPath::Unbond { id: ur.unbond_id }; - match amount { - Some(amount) => { - let amt = amount; - CLAIMABLE_FUNDS.save(deps.storage, (owner, fund_path), &amt)?; - res = res.add_attribute("unbond-callback-error", error.as_str()); - Ok(amt) - } - // TODO: final release should not return an error but log - None => Err(ContractError::CallbackHasNoAmount {}), - }?; - } - } - // if bank callback, add the amount to the claimable funds map - ContractCallback::Bank { - bank_msg, - unbond_id, - } => { - if let BankMsg::Send { to_address, amount } = bank_msg { - CLAIMABLE_FUNDS.save( - deps.storage, - ( - Addr::unchecked(to_address), - FundPath::Unbond { id: unbond_id }, - ), - // should we make sure users don't send more than one Coin? or this can't happen ever - &amount[0].amount, - )?; - res = res.add_attribute("bank-callback-error", error.as_str()); - } - } - } - } - - // Q: should we handle the callback bank message? - - // cleanup the REPLIES state item - REPLIES.remove(deps.storage, msg.id); - Ok(res - .add_attribute("reply-msg-id", msg.id.to_string()) - .add_attribute("reply-result", format!("{:?}", msg.result)) - .add_attribute("action", "handle-callback-reply") - .add_attribute("callback-info", format!("{callback:?}"))) -} - -// test handle callback reply - -#[cfg(test)] -mod tests { - use super::*; - use cosmwasm_std::{ - testing::mock_dependencies, Addr, Attribute, Coin, Reply, SubMsgResponse, SubMsgResult, - Uint128, - }; - use quasar_types::callback::Callback; - - use crate::{helpers::ContractCallback, reply::handle_callback_reply, state::REPLIES}; - - #[test] - fn handle_ack_reply_ok_works() { - let mut deps = mock_dependencies(); - let submsg_id = 1; - let reply = Reply { - id: submsg_id, - result: SubMsgResult::Ok(SubMsgResponse { - events: vec![], - data: None, - }), - }; - - let seq = 1; - let channel = "icq-channel".to_string(); - PENDING_ACK - .save( - deps.as_mut().storage, - (seq, channel.clone()), - &IbcMsgKind::Icq, - ) - .unwrap(); - - let res = handle_ack_reply(deps.as_mut(), reply, seq, channel.clone()).unwrap(); - assert_eq!( - res.attributes, - vec![Attribute { - key: "register-ack-seq".to_string(), - value: seq.to_string() - }] - ); - assert!(!PENDING_ACK.has(deps.as_mut().storage, (seq, channel))) - } - - #[test] - fn test_handle_callback_reply_is_unbond_err() { - use cosmwasm_std::{testing::mock_dependencies, SubMsgResult, Uint128}; - use quasar_types::callback::UnbondResponse; - - use crate::{ - helpers::SubMsgKind, - state::{FundPath, CLAIMABLE_FUNDS}, - }; - - let mut deps = mock_dependencies(); - let owner = Addr::unchecked("owner"); - - let contract_callback = ContractCallback::Callback { - callback: Callback::UnbondResponse(UnbondResponse { - unbond_id: "unbond_id".to_string(), - }), - amount: Some(Uint128::new(100)), - owner: owner.clone(), - }; - let msg = Reply { - id: 1, - result: SubMsgResult::Err("error".to_string()), - }; - - // mocking replies - REPLIES - .save( - &mut deps.storage, - msg.id, - &SubMsgKind::Callback(contract_callback.clone()), - ) - .unwrap(); - - let res = handle_callback_reply(deps.as_mut(), msg.clone(), contract_callback).unwrap(); - assert_eq!(res.attributes.len(), 5); - assert_eq!(res.attributes[0].key, "unbond-callback-error"); - assert_eq!(res.attributes[0].value, "error"); - - assert_eq!( - CLAIMABLE_FUNDS - .load( - &deps.storage, - ( - owner, - FundPath::Unbond { - id: "unbond_id".to_string(), - }, - ), - ) - .unwrap(), - Uint128::new(100) - ); - - // after cleanup it should be empty - assert!(REPLIES.load(&deps.storage, msg.id).is_err()); - } - - #[test] - fn test_handle_callback_reply_is_bank_err() { - use cosmwasm_std::{testing::mock_dependencies, SubMsgResult, Uint128}; - - use crate::{ - helpers::SubMsgKind, - state::{FundPath, CLAIMABLE_FUNDS}, - }; - - let mut deps = mock_dependencies(); - let owner = Addr::unchecked("owner"); - - let bank_msg = BankMsg::Send { - to_address: owner.to_string(), - amount: vec![Coin { - denom: "denom".to_string(), - amount: Uint128::new(69), - }], - }; - - let contract_callback = ContractCallback::Bank { - bank_msg, - unbond_id: "unbond_id".to_string(), - }; - - let msg = Reply { - id: 1, - result: SubMsgResult::Err("error".to_string()), - }; - - // mocking replies - REPLIES - .save( - &mut deps.storage, - msg.id, - &SubMsgKind::Callback(contract_callback.clone()), - ) - .unwrap(); - - let res = handle_callback_reply(deps.as_mut(), msg.clone(), contract_callback).unwrap(); - assert_eq!(res.attributes.len(), 5); - assert_eq!(res.attributes[0].key, "bank-callback-error"); - assert_eq!(res.attributes[0].value, "error"); - - let fund_path = FundPath::Unbond { - id: "unbond_id".to_string(), - }; - - assert_eq!( - CLAIMABLE_FUNDS - .load(&deps.storage, (owner, fund_path),) - .unwrap(), - Uint128::new(69) - ); - - // after cleanup it should be empty - assert!(REPLIES.load(&deps.storage, msg.id).is_err()); - } - - #[test] - fn test_handle_callback_reply_is_err_empty_amount() { - use cosmwasm_std::{testing::mock_dependencies, SubMsgResult}; - use quasar_types::callback::UnbondResponse; - - let mut deps = mock_dependencies(); - let owner = Addr::unchecked("owner"); - - let contract_callback = ContractCallback::Callback { - callback: Callback::UnbondResponse(UnbondResponse { - unbond_id: "unbond_id".to_string(), - }), - amount: None, - owner, - }; - let msg = Reply { - id: 1, - result: SubMsgResult::Err("error".to_string()), - }; - - let res = handle_callback_reply(deps.as_mut(), msg, contract_callback).unwrap_err(); - assert_eq!(res, ContractError::CallbackHasNoAmount {}); - } - - #[test] - fn test_handle_callback_reply_is_ok() { - use crate::helpers::SubMsgKind; - use cosmwasm_std::{testing::mock_dependencies, SubMsgResult}; - use quasar_types::callback::UnbondResponse; - - let mut deps = mock_dependencies(); - let owner = Addr::unchecked("owner"); - - let contract_callback = ContractCallback::Callback { - callback: Callback::UnbondResponse(UnbondResponse { - unbond_id: "unbond_id".to_string(), - }), - amount: Some(Uint128::new(100)), - owner, - }; - let msg = Reply { - id: 1, - result: SubMsgResult::Ok(SubMsgResponse { - data: None, - events: vec![], - }), - }; - - // mocking replies - REPLIES - .save( - &mut deps.storage, - msg.id, - &SubMsgKind::Callback(contract_callback.clone()), - ) - .unwrap(); - - let res = handle_callback_reply(deps.as_mut(), msg, contract_callback).unwrap(); - assert_eq!(res.attributes.len(), 4); - - // after cleanup it should be empty - assert!(REPLIES.is_empty(&deps.storage)); - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/start_unbond.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/start_unbond.rs deleted file mode 100644 index 3602a59fe..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/start_unbond.rs +++ /dev/null @@ -1,819 +0,0 @@ -use cosmwasm_std::{ - to_json_binary, Addr, CosmosMsg, Env, IbcMsg, IbcTimeout, QuerierWrapper, Response, Storage, - SubMsg, Uint128, WasmMsg, -}; - -use osmosis_std::types::{cosmos::base::v1beta1::Coin, osmosis::lockup::MsgBeginUnlocking}; -use quasar_types::{ - callback::{Callback, StartUnbondResponse}, - ica::packet::ica_send, -}; -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -use crate::{ - error::ContractError, - helpers::get_total_primitive_shares, - helpers::{ - create_callback_submsg, create_ibc_ack_submsg, get_ica_address, IbcMsgKind, IcaMessages, - }, - ibc_lock::Lock, - state::{ - LpCache, PendingSingleUnbond, Unbond, CONFIG, IBC_LOCK, IBC_TIMEOUT_TIME, ICA_CHANNEL, - LP_SHARES, OSMO_LOCK, SHARES, START_UNBOND_QUEUE, UNBONDING_CLAIMS, - }, -}; - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub struct StartUnbond { - pub owner: Addr, - pub id: String, - pub primitive_shares: Uint128, -} - -/// Checks that the unbond_id is not already in the queue or in unbonding process. -/// If the user has sufficient shares to unbond, adds the StartUnbond to the queue. -pub fn do_start_unbond( - storage: &mut dyn Storage, - unbond: StartUnbond, -) -> Result<(), ContractError> { - if UNBONDING_CLAIMS.has(storage, (unbond.owner.clone(), unbond.id.clone())) { - return Err(ContractError::DuplicateKey); - } - - //verify here against the amount in the queue aswell - let queued_shares = START_UNBOND_QUEUE - .iter(storage)? - .map(|val| { - let v = val?; - if v.id == unbond.id { - Err(ContractError::DuplicateKey) - } else { - Ok(v) - } - }) - .try_fold( - Uint128::zero(), - |acc, val| -> Result { - let v = val?; - if v.owner == unbond.owner { - Ok(acc + v.primitive_shares) - } else { - Ok(Uint128::zero()) - } - }, - )?; - - if SHARES.load(storage, unbond.owner.clone())? < (unbond.primitive_shares + queued_shares) { - return Err(ContractError::InsufficientFunds); - } - - Ok(START_UNBOND_QUEUE.push_back(storage, &unbond)?) -} - -// batch unbond tries to unbond a batch of unbondings, should be called after the icq query has returned for deposits -pub fn batch_start_unbond( - storage: &mut dyn Storage, - env: &Env, -) -> Result, ContractError> { - let mut to_unbond = Uint128::zero(); - let mut unbonds: Vec = vec![]; - - if START_UNBOND_QUEUE.is_empty(storage)? { - return Ok(None); - } - - let total_lp_shares = LP_SHARES.load(storage)?; - - while !START_UNBOND_QUEUE.is_empty(storage)? { - let unbond = - START_UNBOND_QUEUE - .pop_front(storage)? - .ok_or(ContractError::QueueItemNotFound { - queue: "start_unbond".to_string(), - })?; - let lp_shares = single_unbond(storage, &unbond, total_lp_shares.locked_shares)?; - to_unbond = to_unbond.checked_add(lp_shares)?; - unbonds.push(PendingSingleUnbond { - lp_shares, - primitive_shares: unbond.primitive_shares, - owner: unbond.owner, - id: unbond.id, - }) - } - - let pkt = do_begin_unlocking(storage, env, to_unbond)?; - - let channel = ICA_CHANNEL.load(storage)?; - - Ok(Some(create_ibc_ack_submsg( - storage, - IbcMsgKind::Ica(IcaMessages::BeginUnlocking(unbonds, to_unbond)), - pkt, - channel, - )?)) -} - -pub fn do_begin_unlocking( - storage: &mut dyn Storage, - env: &Env, - to_unbond: Uint128, -) -> Result { - let config = CONFIG.load(storage)?; - let ica_address = get_ica_address(storage, ICA_CHANNEL.load(storage)?)?; - - let msg = MsgBeginUnlocking { - owner: ica_address, - id: OSMO_LOCK.load(storage)?, - coins: vec![Coin { - denom: config.pool_denom, - amount: to_unbond.to_string(), - }], - }; - - let pkt = ica_send::( - msg, - ICA_CHANNEL.load(storage)?, - IbcTimeout::with_timestamp(env.block.time.plus_seconds(IBC_TIMEOUT_TIME)), - )?; - - Ok(pkt) -} - -pub fn handle_start_unbond_ack( - storage: &mut dyn Storage, - querier: QuerierWrapper, - env: &Env, - unbonds: Vec, - total_start_unbonding: Uint128, -) -> Result { - let mut callback_submsgs: Vec = vec![]; - for unbond in unbonds { - if let Some(msg) = start_internal_unbond(storage, querier, env, unbond.clone())? { - // convert wasm_msg into cosmos_msg to be handled in create_callback_submsg - callback_submsgs.push(create_callback_submsg( - storage, - CosmosMsg::Wasm(msg), - unbond.owner, - unbond.id, - )?); - } - } - - IBC_LOCK.update(storage, |lock| -> Result { - Ok(lock.unlock_start_unbond()) - })?; - - // TODO, update the actual amount of locked lp shares in the lp cache here aswell - LP_SHARES.update(storage, |mut cache| -> Result { - cache.w_unlocked_shares = cache.w_unlocked_shares.checked_add(total_start_unbonding)?; - cache.locked_shares = cache.locked_shares.checked_sub(total_start_unbonding)?; - Ok(cache) - })?; - - Ok(Response::new() - .add_attribute("start-unbond", "succes") - .add_attribute("callback-submsgs", callback_submsgs.len().to_string()) - .add_messages(callback_submsgs.iter().map(|m| m.msg.clone()))) -} - -// in single_unbond, we change from using internal primitive to an actual amount of lp-shares that we can unbond -fn single_unbond( - storage: &mut dyn Storage, - unbond: &StartUnbond, - total_lp_shares: Uint128, -) -> Result { - let total_primitive_shares = get_total_primitive_shares(storage)?; - - Ok(unbond - .primitive_shares - .checked_multiply_ratio(total_lp_shares, total_primitive_shares)?) -} - -// unbond starts unbonding an amount of lp shares -fn start_internal_unbond( - storage: &mut dyn Storage, - querier: QuerierWrapper, - env: &Env, - unbond: PendingSingleUnbond, -) -> Result, ContractError> { - // check that we can create a new unbond - if UNBONDING_CLAIMS.has(storage, (unbond.owner.clone(), unbond.id.clone())) { - return Err(ContractError::DuplicateKey); - } - - // remove amount of shares - let left = SHARES - .load(storage, unbond.owner.clone())? - .checked_sub(unbond.primitive_shares) - .map_err(|err| { - ContractError::TracedOverflowError(err, "lower_shares_to_unbond".to_string()) - })?; - // subtracting below zero here should trigger an error in check_sub - if left.is_zero() { - SHARES.remove(storage, unbond.owner.clone()); - } else { - SHARES.save(storage, unbond.owner.clone(), &left)?; - } - - // todo verify logic of unlock times - let unlock_time = env - .block - .time - .plus_seconds(CONFIG.load(storage)?.lock_period); - - // add amount of unbonding claims - UNBONDING_CLAIMS.save( - storage, - (unbond.owner.clone(), unbond.id.clone()), - &Unbond { - lp_shares: unbond.lp_shares, - unlock_time, - attempted: false, - id: unbond.id.clone(), - owner: unbond.owner.clone(), - }, - )?; - - let msg = Callback::StartUnbondResponse(StartUnbondResponse { - unbond_id: unbond.id.clone(), - unlock_time, - }); - - if querier - .query_wasm_contract_info(unbond.owner.as_str()) - .is_ok() - { - Ok(Some(WasmMsg::Execute { - contract_addr: unbond.owner.to_string(), - msg: to_json_binary(&msg)?, - funds: vec![], - })) - } else { - Ok(None) - } -} - -#[cfg(test)] -mod tests { - use cosmwasm_std::{ - testing::{mock_dependencies, mock_env}, - Addr, Binary, ContractInfoResponse, ContractResult, CosmosMsg, OverflowError, - OverflowOperation, QuerierResult, StdError, Timestamp, Uint128, WasmMsg, - }; - - use crate::{ - bond::calculate_claim, - state::{LpCache, PendingSingleUnbond, SHARES}, - test_helpers::default_setup, - }; - - use super::*; - use proptest::prelude::*; - - #[test] - fn do_start_unbond_exact_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let id = "my-id".to_string(); - - SHARES - .save(deps.as_mut().storage, owner.clone(), &Uint128::new(1000)) - .unwrap(); - - let unbond = StartUnbond { - owner, - id, - primitive_shares: Uint128::new(1000), - }; - do_start_unbond(deps.as_mut().storage, unbond).unwrap() - } - - #[test] - fn do_start_unbond_multiple_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let id1 = "my-id-1".to_string(); - let id2 = "my-id-2".to_string(); - let id3 = "my-id-3".to_string(); - - SHARES - .save(deps.as_mut().storage, owner.clone(), &Uint128::new(1000)) - .unwrap(); - - START_UNBOND_QUEUE - .push_back( - deps.as_mut().storage, - &StartUnbond { - owner: Addr::unchecked("alice"), - id: "2".to_string(), - primitive_shares: Uint128::new(1500), - }, - ) - .unwrap(); - - let unbond1 = StartUnbond { - owner: owner.clone(), - id: id1, - primitive_shares: Uint128::new(500), - }; - let unbond2 = StartUnbond { - owner: owner.clone(), - id: id2, - primitive_shares: Uint128::new(300), - }; - let unbond3 = StartUnbond { - owner, - id: id3, - primitive_shares: Uint128::new(200), - }; - - do_start_unbond(deps.as_mut().storage, unbond1.clone()).unwrap(); - do_start_unbond(deps.as_mut().storage, unbond2.clone()).unwrap(); - do_start_unbond(deps.as_mut().storage, unbond3.clone()).unwrap(); - assert_eq!(START_UNBOND_QUEUE.len(deps.as_ref().storage).unwrap(), 4); - // pop alice's start_unbond - START_UNBOND_QUEUE.pop_front(deps.as_mut().storage).unwrap(); - - assert_eq!( - START_UNBOND_QUEUE - .pop_front(deps.as_mut().storage) - .unwrap() - .unwrap(), - unbond1 - ); - assert_eq!( - START_UNBOND_QUEUE - .pop_front(deps.as_mut().storage) - .unwrap() - .unwrap(), - unbond2 - ); - assert_eq!( - START_UNBOND_QUEUE - .pop_front(deps.as_mut().storage) - .unwrap() - .unwrap(), - unbond3 - ) - } - - #[test] - fn do_start_unbond_not_enough_shares_fails() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let id = "my-id".to_string(); - - SHARES - .save(deps.as_mut().storage, owner.clone(), &Uint128::new(999)) - .unwrap(); - - let unbond = StartUnbond { - owner, - id, - primitive_shares: Uint128::new(1000), - }; - let err = do_start_unbond(deps.as_mut().storage, unbond).unwrap_err(); - assert_eq!(err, ContractError::InsufficientFunds) - } - - #[test] - fn do_start_unbond_duplicate_key_fails() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let id = "my-id".to_string(); - - SHARES - .save(deps.as_mut().storage, owner.clone(), &Uint128::new(999)) - .unwrap(); - UNBONDING_CLAIMS - .save( - deps.as_mut().storage, - (owner.clone(), id.clone()), - &Unbond { - lp_shares: Uint128::new(420), - unlock_time: Timestamp::from_seconds(100), - attempted: false, - owner: owner.clone(), - id: id.clone(), - }, - ) - .unwrap(); - - let unbond = StartUnbond { - owner, - id, - primitive_shares: Uint128::new(1000), - }; - let err = do_start_unbond(deps.as_mut().storage, unbond).unwrap_err(); - assert_eq!(err, ContractError::DuplicateKey) - } - - #[test] - fn batch_start_unbond_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let env = mock_env(); - let id = "my-id".to_string(); - //test specific setup - OSMO_LOCK.save(deps.as_mut().storage, &1).unwrap(); - SHARES - .save(deps.as_mut().storage, owner.clone(), &Uint128::new(1000)) - .unwrap(); - - LP_SHARES - .save( - deps.as_mut().storage, - &LpCache { - locked_shares: Uint128::new(1000), - w_unlocked_shares: Uint128::zero(), - d_unlocked_shares: Uint128::zero(), - }, - ) - .unwrap(); - - let unbond1 = StartUnbond { - owner, - id, - primitive_shares: Uint128::new(1000), - }; - - do_start_unbond(deps.as_mut().storage, unbond1).unwrap(); - - let res = batch_start_unbond(deps.as_mut().storage, &env).unwrap(); - assert!(res.is_some()); - - // check that the packet is as we expect - let ica = get_ica_address( - deps.as_ref().storage, - ICA_CHANNEL.load(deps.as_ref().storage).unwrap(), - ) - .unwrap(); - let msg = MsgBeginUnlocking { - owner: ica, - id: OSMO_LOCK.load(deps.as_mut().storage).unwrap(), - coins: vec![Coin { - denom: CONFIG.load(deps.as_ref().storage).unwrap().pool_denom, - // integer truncation present here again - amount: Uint128::new(1000).to_string(), - }], - }; - - let pkt = ica_send::( - msg, - ICA_CHANNEL.load(deps.as_ref().storage).unwrap(), - IbcTimeout::with_timestamp(env.block.time.plus_seconds(IBC_TIMEOUT_TIME)), - ) - .unwrap(); - assert_eq!(res.unwrap().msg, CosmosMsg::Ibc(pkt)); - } - - #[test] - fn single_unbond_big_math() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let id = "my-id".to_string(); - - SHARES - .save(deps.as_mut().storage, owner.clone(), &Uint128::new(100)) - .unwrap(); - SHARES - .save( - deps.as_mut().storage, - Addr::unchecked("other_user"), - &Uint128::new(900), - ) - .unwrap(); - - LP_SHARES - .save( - deps.as_mut().storage, - &LpCache { - locked_shares: Uint128::new(10_000_000_000), - w_unlocked_shares: Uint128::zero(), - d_unlocked_shares: Uint128::zero(), - }, - ) - .unwrap(); - - let res = single_unbond( - deps.as_mut().storage, - &StartUnbond { - owner, - id, - primitive_shares: Uint128::new(100), - }, - Uint128::new(10_000_000_000), - ) - .unwrap(); - - assert_eq!( - get_total_primitive_shares(deps.as_mut().storage).unwrap(), - Uint128::new(1000) - ); - assert_eq!(res, Uint128::new(1000000000)) - } - - // this is an excellent first test to write a proptest for - #[test] - fn single_unbond_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let id = "my-id".to_string(); - - SHARES - .save(deps.as_mut().storage, owner.clone(), &Uint128::new(100)) - .unwrap(); - - let res = single_unbond( - deps.as_mut().storage, - &StartUnbond { - owner, - id, - primitive_shares: Uint128::new(100), - }, - Uint128::new(100), - ) - .unwrap(); - // we have a share loss here due to truncation, is this avoidable? - assert_eq!(res, Uint128::new(100)) - } - - #[test] - fn start_internal_unbond_exact_shares_works() { - // general setup - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let id = "my-id"; - let env = mock_env(); - - deps.querier.update_wasm(|q| match q { - cosmwasm_std::WasmQuery::ContractInfo { contract_addr: _ } => QuerierResult::Ok( - ContractResult::Ok(to_json_binary(&ContractInfoResponse::default()).unwrap()), - ), - _ => unimplemented!(), - }); - let w = QuerierWrapper::new(&deps.querier); - - assert!(w.query_wasm_contract_info(owner.clone()).is_ok()); - - // test specific setup - SHARES - .save(&mut deps.storage, owner.clone(), &Uint128::new(100)) - .unwrap(); - let unbond = PendingSingleUnbond { - lp_shares: Uint128::new(100), - primitive_shares: Uint128::new(100), - owner: owner.clone(), - id: id.to_string(), - }; - - let res = start_internal_unbond(&mut deps.storage, w, &env, unbond).unwrap(); - assert_eq!( - res.unwrap(), - WasmMsg::Execute { - contract_addr: owner.to_string(), - msg: to_json_binary(&Callback::StartUnbondResponse(StartUnbondResponse { - unbond_id: id.to_string(), - unlock_time: env - .block - .time - .plus_seconds(CONFIG.load(deps.as_ref().storage).unwrap().lock_period) - })) - .unwrap(), - funds: vec![] - } - ) - } - - #[test] - fn start_internal_unbond_less_shares_works() { - // general setup - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let id = "my-id"; - let env = mock_env(); - - deps.querier.update_wasm(|q| match q { - cosmwasm_std::WasmQuery::ContractInfo { contract_addr: _ } => QuerierResult::Ok( - ContractResult::Ok(to_json_binary(&ContractInfoResponse::default()).unwrap()), - ), - _ => unimplemented!(), - }); - let w = QuerierWrapper::new(&deps.querier); - - // test specific setup - SHARES - .save(&mut deps.storage, owner.clone(), &Uint128::new(101)) - .unwrap(); - let unbond = PendingSingleUnbond { - lp_shares: Uint128::new(100), - primitive_shares: Uint128::new(100), - owner: owner.clone(), - id: id.to_string(), - }; - - let res = start_internal_unbond(&mut deps.storage, w, &env, unbond).unwrap(); - assert_eq!( - res.unwrap(), - WasmMsg::Execute { - contract_addr: owner.to_string(), - msg: to_json_binary(&Callback::StartUnbondResponse(StartUnbondResponse { - unbond_id: id.to_string(), - unlock_time: env - .block - .time - .plus_seconds(CONFIG.load(deps.as_ref().storage).unwrap().lock_period) - })) - .unwrap(), - funds: vec![] - } - ); - assert_eq!( - SHARES.load(deps.as_ref().storage, owner).unwrap(), - Uint128::one() - ) - } - - #[test] - fn start_internal_unbond_duplicate_key_fails() { - // general setup - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let id = "my-id"; - let env = mock_env(); - - deps.querier.update_wasm(|q| match q { - cosmwasm_std::WasmQuery::ContractInfo { contract_addr: _ } => { - QuerierResult::Ok(ContractResult::Ok(Binary::from_base64("deadbeef").unwrap())) - } - _ => unimplemented!(), - }); - let w = QuerierWrapper::new(&deps.querier); - - // test specific setup - SHARES - .save(&mut deps.storage, owner.clone(), &Uint128::new(99)) - .unwrap(); - let unbond = PendingSingleUnbond { - lp_shares: Uint128::new(100), - primitive_shares: Uint128::new(100), - owner: owner.clone(), - id: id.to_string(), - }; - let unlock_time = env - .block - .time - .plus_seconds(CONFIG.load(deps.as_ref().storage).unwrap().lock_period); - UNBONDING_CLAIMS - .save( - &mut deps.storage, - (owner.clone(), id.to_string()), - &Unbond { - lp_shares: Uint128::new(100), - unlock_time, - attempted: false, - owner, - id: id.to_string(), - }, - ) - .unwrap(); - - let res = start_internal_unbond(&mut deps.storage, w, &env, unbond).unwrap_err(); - assert_eq!(res, ContractError::DuplicateKey) - } - - #[test] - fn start_internal_unbond_not_enough_shares_fails() { - // general setup - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let id = "my-id"; - let env = mock_env(); - - deps.querier.update_wasm(|q| match q { - cosmwasm_std::WasmQuery::ContractInfo { contract_addr: _ } => { - QuerierResult::Ok(ContractResult::Ok(Binary::from_base64("deadbeef").unwrap())) - } - _ => unimplemented!(), - }); - let w = QuerierWrapper::new(&deps.querier); - - // test specific setup - SHARES - .save(&mut deps.storage, owner.clone(), &Uint128::new(99)) - .unwrap(); - let unbond = PendingSingleUnbond { - lp_shares: Uint128::new(100), - primitive_shares: Uint128::new(100), - owner, - id: id.to_string(), - }; - - let res = start_internal_unbond(&mut deps.storage, w, &env, unbond).unwrap_err(); - assert_eq!( - res, - ContractError::TracedOverflowError( - OverflowError { - operation: OverflowOperation::Sub, - operand1: "99".to_string(), - operand2: "100".to_string() - }, - "lower_shares_to_unbond".to_string() - ) - ) - } - - #[test] - fn start_internal_unbond_no_shares_fails() { - // general setup - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let id = "my-id"; - let env = mock_env(); - - deps.querier.update_wasm(|q| match q { - cosmwasm_std::WasmQuery::ContractInfo { contract_addr: _ } => { - QuerierResult::Ok(ContractResult::Ok(Binary::from_base64("deadbeef").unwrap())) - } - _ => unimplemented!(), - }); - let w = QuerierWrapper::new(&deps.querier); - - // test specific setup - let unbond = PendingSingleUnbond { - lp_shares: Uint128::new(100), - primitive_shares: Uint128::new(100), - owner, - id: id.to_string(), - }; - - let res = start_internal_unbond(&mut deps.storage, w, &env, unbond).unwrap_err(); - assert_eq!( - res, - ContractError::Std(StdError::NotFound { - kind: "type: cosmwasm_std::math::uint128::Uint128; key: [00, 06, 73, 68, 61, 72, 65, 73, 62, 6F, 62]".to_string() - }) - ) - } - - proptest! { - #[test] - fn test_calculate_claim_and_single_unbond( - (total_balance, user_balance) in (1..4*10_u128.pow(28)).prop_flat_map(|a| (Just(a), 1..a)), - total_primitive_shares in 1u128..4*10_u128.pow(28), - lp_shares in 1u128..4*10_u128.pow(28), - ) { - - let mut deps = mock_dependencies(); - - SHARES.save(deps.as_mut().storage, Addr::unchecked("other-shares"), &Uint128::new(total_primitive_shares)).unwrap(); - - // Calculate the claim using the calculate_claim function - // here bob gets a claim to a certain amount of - let claim = calculate_claim( - Uint128::new(user_balance), - Uint128::new(total_balance), - Uint128::new(total_primitive_shares), - ) - .unwrap(); - - // Calculate the unbond amount using the single_unbond function - let unbond = single_unbond( - deps.as_mut().storage, - &StartUnbond { - primitive_shares: claim, - id: "1".to_string(), - owner: Addr::unchecked("bobberino"), - }, - lp_shares.into(), - ) - .unwrap(); - - // how do we now assert, basically we get an expected amount of returning lp shares that - // we need to simulate a liquidation for. How do we do that? - // in the test setup, we assume that depositing total_balance has let to lp_shares, - // so the price of a single lp shares is total_balance/lp_shares - let ub = Uint128::new(user_balance); - let recv_balance = unbond.multiply_ratio(total_balance, lp_shares); - // for our assertion, since we are working with interger math and 6 decimals or more on tokens - // we're ok with being either 1 off or some micro (10^-10) off - // TODO for ease of coding, we just accept this ratio - let vals = recv_balance.multiply_ratio(9999999999u128, 10000000000u128)..recv_balance.multiply_ratio(10000000001u128, 1000000000u128); - prop_assert!(vals.contains(&ub), "recv_balance: {recv_balance}, user_balance: {user_balance}"); - } - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/state.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/state.rs deleted file mode 100644 index 90044db77..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/state.rs +++ /dev/null @@ -1,330 +0,0 @@ -use quasar_types::ibc::ChannelInfo; -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; -use std::fmt::{Debug, Display}; - -use cosmwasm_std::{Addr, Empty, IbcAcknowledgement, StdError, StdResult, Timestamp, Uint128}; -use cw_storage_plus::{Deque, Item, Key, KeyDeserialize, Map, Prefixer, PrimaryKey}; - -use crate::{ - bond::Bond, - error::{ContractError, Trap}, - helpers::{IbcMsgKind, SubMsgKind}, - ibc_lock::Lock, - start_unbond::StartUnbond, -}; - -pub const RETURN_SOURCE_PORT: &str = "transfer"; -pub const IBC_TIMEOUT_TIME: u64 = 7200; - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug, Default)] -#[serde(rename_all = "snake_case")] -pub struct Config { - // The lock period is the amount of time we lock tokens on Osmosis - pub lock_period: u64, - pub pool_id: u64, - // pool_denom is the denom of the gamm pool on osmosis; eg gamm/pool/1 - pub pool_denom: String, - // the base denom of the pool on osmosis - pub base_denom: String, - // the quote denom is the "other" side of the pool we deposit tokens in - pub quote_denom: String, - // the denom on the Quasar chain - pub local_denom: String, - // the transfer channel to transfer funds to Osmosis - pub transfer_channel: String, - // the channel for sending tokens back from the counterparty chain to quasar chain - pub return_source_channel: String, - // expected_connection id on which the primitive should function - pub expected_connection: String, -} - -// TODO remove the need for ADMIN - -// the ADMIN in this case is the person allowed to deposit into the contract -// this is set to the first depositor -pub(crate) const ADMIN: Item = Item::new("admin"); -pub(crate) const LOCK_ADMIN: Map<&Addr, Empty> = Map::new("lock_admin"); -pub(crate) const DEPOSITOR: Item = Item::new("depositor"); - -pub(crate) const CONFIG: Item = Item::new("config"); -// IBC related state items -pub(crate) const REPLIES: Map = Map::new("replies"); -// RECOVERY_ACK contains ibc acknowledgements, these packets might be needed for recovery from errors -pub(crate) const RECOVERY_ACK: Map<(u64, String), IbcAcknowledgement> = - Map::new("new_recovery_ack"); -pub(crate) const _OLD_RECOVERY_ACK: Map = Map::new("recovery_ack"); - -// true when a packet has timed out and the ica channel needs to be closed and a new channel needs to be opened -pub(crate) const TIMED_OUT: Item = Item::new("timed_out"); -// Currently we only support one ICA channel to a single destination -pub(crate) const ICA_CHANNEL: Item = Item::new("ica_channel"); -// We also support one ICQ channel to Osmosis at the moment -pub(crate) const ICQ_CHANNEL: Item = Item::new("icq_channel"); - -pub(crate) const CHANNELS: Map = Map::new("channels"); -pub(crate) const _OLD_PENDING_ACK: Map = Map::new("pending_acks"); -pub(crate) const PENDING_ACK: Map<(u64, String), IbcMsgKind> = Map::new("new_pending_acks"); -// The map to store trapped errors, -pub(crate) const _OLD_TRAPS: Map = Map::new("traps"); -pub(crate) const TRAPS: Map<(u64, String), Trap> = Map::new("new_traps"); - -// all vault related state items -pub(crate) const IBC_LOCK: Item = Item::new("lock"); -pub(crate) const PENDING_BOND_QUEUE: Deque = Deque::new("pending_bond_queue"); -pub(crate) const BOND_QUEUE: Deque = Deque::new("bond_queue"); -pub(crate) const FAILED_JOIN_QUEUE: Deque = Deque::new("failed_join_queue"); -pub(crate) const REJOIN_QUEUE: Deque = Deque::new("rejoin_queue"); -pub(crate) const START_UNBOND_QUEUE: Deque = Deque::new("start_unbond_queue"); -pub(crate) const PENDING_UNBOND_QUEUE: Deque = Deque::new("pending_unbond_queue"); -pub(crate) const UNBOND_QUEUE: Deque = Deque::new("unbond_queue"); -// the amount of LP shares that the contract has entered into the pool -pub(crate) const LP_SHARES: Item = Item::new("lp_shares"); - -// the latest known ica balance -pub(crate) const TOTAL_VAULT_BALANCE: Item = Item::new("total_vault_balance"); -/// the "free balance" on osmo side, this is the amount of tokens that we can compound into the pool -/// actually, we also subtract out any failed bond attempts & any failed unbond attempts (trapped_errors, and failed join pool) -/// failed join pool is bonded back in via a different mechanism -/// trapped errors are not bonded back in, but they are not lost either -/// the only part that isn't accounted for is the failed returnTransfer attempts, but we never run that path simultaneously due to locks to it's fine -pub(crate) const USABLE_COMPOUND_BALANCE: Item = Item::new("usable_compound_balance"); - -// TODO we probably want to change this to an OngoingDeposit -pub(crate) const BONDING_CLAIMS: Map<(&Addr, &str), Uint128> = Map::new("bonding_claims"); - -// TODO: this is not being used, remove it -pub(crate) const PENDING_UNBONDING_CLAIMS: Map<(Addr, String), Unbond> = - Map::new("pending_unbonding_claims"); -pub(crate) const UNBONDING_CLAIMS: Map<(Addr, String), Unbond> = Map::new("unbonding_claims"); -// TODO make key borrowed -pub(crate) const SHARES: Map = Map::new("shares"); -// the lock id on osmosis, for each combination of denom and lock duration, only one lock id should exist on osmosis -pub(crate) const OSMO_LOCK: Item = Item::new("osmo_lock"); -// the returning transfer we can expect and their exact amount -pub(crate) const RETURNING: Map = Map::new("returning"); -// TODO, do we remove this state item? is it needed? -// whatever the above todo item is, does not apply to the following -// we save the queried simulate join swap during ICQ so we can read it right before bond join -pub(crate) const SIMULATED_JOIN_RESULT: Item = Item::new("simulated_join_result"); -// we save the amount that went into the QueryCalcJoinPool, so we can scale up the slippage amount if more deposits come -pub(crate) const SIMULATED_JOIN_AMOUNT_IN: Item = Item::new("simulated_join_amount"); -// we also save the queried simulate exit swap during ICQ so we can read it right before unbond exit -pub(crate) const SIMULATED_EXIT_RESULT: Item = Item::new("simulated_exit_result"); -// we save the amount that went into the QueryCalcExitPoolCoinsFromSharesRequest, so we can scale up the slippage amount (if more unbonds come) -// need to check if (...) applies here -pub(crate) const SIMULATED_EXIT_SHARES_IN: Item = Item::new("simulated_exit_amount"); -// CLAIMABLE_FUNDS is the amount of funds claimable by a certain address, either -pub(crate) const CLAIMABLE_FUNDS: Map<(Addr, FundPath), Uint128> = Map::new("claimable_funds"); - -impl PrimaryKey<'_> for FundPath { - type Prefix = Addr; - - type SubPrefix = (); - - type Suffix = u8; - - type SuperSuffix = Self; - - fn key(&self) -> Vec { - // this is a bit yikes but fuck it - match self { - FundPath::Bond { id } => vec![Key::Val8([0]), Key::Ref(id.as_bytes())], - FundPath::Unbond { id } => vec![Key::Val8([1]), Key::Ref(id.as_bytes())], - } - } -} - -impl KeyDeserialize for FundPath { - type Output = FundPath; - - #[inline(always)] - fn from_vec(value: Vec) -> StdResult { - if value[0] == 0 { - Ok(FundPath::Bond { - id: String::from_utf8(value[1..].to_vec()).map_err(|err| { - StdError::InvalidUtf8 { - msg: err.to_string(), - } - })?, - }) - } else if value[0] == 1 { - Ok(FundPath::Unbond { - id: String::from_utf8(value[1..].to_vec()).map_err(|err| { - StdError::InvalidUtf8 { - msg: err.to_string(), - } - })?, - }) - } else { - Err(StdError::SerializeErr { - source_type: "key-de".to_string(), - msg: "enum variant not found".to_string(), - }) - } - } -} - -impl Prefixer<'_> for FundPath { - fn prefix(&self) -> Vec { - match self { - FundPath::Bond { id } => vec![Key::Ref(id.as_bytes())], - FundPath::Unbond { id } => vec![Key::Ref(id.as_bytes())], - } - } -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug, Eq)] -#[serde(rename_all = "snake_case")] -pub enum FundPath { - Bond { id: String }, - Unbond { id: String }, -} - -impl Display for FundPath { - fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { - write!(f, "{self:?}") - } -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub struct LpCache { - // the amount of locked shares we currently have - pub locked_shares: Uint128, - // the amount of unlocked share we have for withdrawing - pub w_unlocked_shares: Uint128, - // the amount unlocked shares we have for depositing - pub d_unlocked_shares: Uint128, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub struct Unbond { - pub lp_shares: Uint128, - pub unlock_time: Timestamp, - pub attempted: bool, - pub owner: Addr, - pub id: String, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug, Eq)] -#[serde(rename_all = "snake_case")] -pub struct PendingSingleUnbond { - pub lp_shares: Uint128, - pub primitive_shares: Uint128, - pub owner: Addr, - pub id: String, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub struct PendingBond { - // the bonds of the original calls - pub bonds: Vec, -} - -impl PendingBond { - pub fn update_raw_amount_to_lp(&mut self, total_lp: Uint128) -> Result<(), ContractError> { - let mut total = Uint128::zero(); - for p in self.bonds.iter() { - match p.raw_amount { - crate::state::RawAmount::LocalDenom(val) => total = total.checked_add(val)?, - crate::state::RawAmount::LpShares(_) => unimplemented!(), - } - } - for p in self.bonds.iter_mut() { - match p.raw_amount { - // amount of lp shares = val * total_lp / total - crate::state::RawAmount::LocalDenom(val) => { - p.raw_amount = - RawAmount::LpShares(val.checked_mul(total_lp)?.checked_div(total)?) - } - crate::state::RawAmount::LpShares(_) => unimplemented!(), - } - } - Ok(()) - } -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug, Eq)] -#[serde(rename_all = "snake_case")] -pub struct Claim { - amount: Uint128, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub struct OngoingDeposit { - pub claim_amount: Uint128, // becomes shares later - pub raw_amount: RawAmount, - pub owner: Addr, - pub bond_id: String, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub enum RawAmount { - LocalDenom(Uint128), - LpShares(Uint128), -} - -#[cfg(test)] -mod tests { - - use super::*; - - #[test] - fn keys_work() { - let bond = FundPath::Bond { - id: "our-id-here".to_string(), - }; - let keys: Vec = bond - .key() - .iter() - .flat_map(|k| k.as_ref().iter().copied()) - .collect(); - let value = FundPath::from_vec(keys).unwrap(); - assert_eq!(bond, value) - } - - #[test] - fn test_update_raw_amount_to_lp() { - let mut pending = PendingBond { - bonds: vec![ - OngoingDeposit { - claim_amount: Uint128::new(100), - raw_amount: RawAmount::LocalDenom(Uint128::new(1000)), - owner: Addr::unchecked("address"), - bond_id: "fake".to_string(), - }, - OngoingDeposit { - claim_amount: Uint128::new(99), - raw_amount: RawAmount::LocalDenom(Uint128::new(999)), - owner: Addr::unchecked("address"), - bond_id: "fake".to_string(), - }, - OngoingDeposit { - claim_amount: Uint128::new(101), - raw_amount: RawAmount::LocalDenom(Uint128::new(1000)), - owner: Addr::unchecked("address"), - bond_id: "fake".to_string(), - }, - ], - }; - pending.update_raw_amount_to_lp(Uint128::new(300)).unwrap(); - assert_eq!( - pending.bonds[0].raw_amount, - RawAmount::LpShares(Uint128::new(100)) - ); - assert_eq!( - pending.bonds[1].raw_amount, - RawAmount::LpShares(Uint128::new(99)) - ); - // because we use integer division and relatively low values, this case us 100 - assert_eq!( - pending.bonds[2].raw_amount, - RawAmount::LpShares(Uint128::new(100)) - ) - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/test_helpers.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/test_helpers.rs deleted file mode 100644 index 3414c4a8a..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/test_helpers.rs +++ /dev/null @@ -1,237 +0,0 @@ -use cosmos_sdk_proto::tendermint::abci::ResponseQuery; -use cosmwasm_std::{IbcEndpoint, Storage}; -use prost::bytes::Bytes; -use quasar_types::{ - ibc::{ChannelInfo, ChannelType}, - ica::handshake::IcaMetadata, -}; - -use crate::{ - bond::Bond, - state::{ - Config, PendingBond, RawAmount, CHANNELS, CONFIG, ICA_CHANNEL, ICQ_CHANNEL, LP_SHARES, - }, -}; - -pub fn default_setup(storage: &mut dyn Storage) -> Result<(), cosmwasm_std::StdError> { - setup_default_icq(storage)?; - setup_default_ica(storage)?; - setup_default_config(storage)?; - setup_default_lp_cache(storage) -} - -pub(crate) fn setup_default_icq(storage: &mut dyn Storage) -> Result<(), cosmwasm_std::StdError> { - let chan = "channel-1"; - CHANNELS.save( - storage, - chan.to_string(), - &ChannelInfo { - id: chan.to_string(), - counterparty_endpoint: IbcEndpoint { - port_id: "icqhost".to_string(), - channel_id: chan.to_string(), - }, - connection_id: "connection-0".to_string(), - channel_type: ChannelType::Icq { - channel_ty: "icq-1".to_string(), - }, - handshake_state: quasar_types::ibc::HandshakeState::Open, - }, - )?; - ICQ_CHANNEL.save(storage, &chan.to_string()) -} - -pub(crate) fn setup_default_ica(storage: &mut dyn Storage) -> Result<(), cosmwasm_std::StdError> { - let chan = "channel-2"; - CHANNELS.save( - storage, - chan.to_string(), - &ChannelInfo { - id: chan.to_string(), - counterparty_endpoint: IbcEndpoint { - port_id: "icahost".to_string(), - channel_id: chan.to_string(), - }, - connection_id: "connection-0".to_string(), - channel_type: ChannelType::Ica { - channel_ty: IcaMetadata::with_connections( - "connection-1".to_string(), - "connection-2".to_string(), - ), - counter_party_address: Some("osmo-address".to_string()), - }, - handshake_state: quasar_types::ibc::HandshakeState::Open, - }, - )?; - ICA_CHANNEL.save(storage, &chan.to_string()) -} - -pub(crate) fn setup_default_config( - storage: &mut dyn Storage, -) -> Result<(), cosmwasm_std::StdError> { - CONFIG.save( - storage, - &Config { - lock_period: 100, - pool_id: 1, - pool_denom: "gamm/pool/1".to_string(), - base_denom: "uosmo".to_string(), - quote_denom: "uqsr".to_string(), - local_denom: "ibc/local_osmo".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - }, - ) -} - -pub(crate) fn setup_default_lp_cache( - storage: &mut dyn Storage, -) -> Result<(), cosmwasm_std::StdError> { - LP_SHARES.save( - storage, - &crate::state::LpCache { - locked_shares: 100u128.into(), - w_unlocked_shares: 100u128.into(), - d_unlocked_shares: 100u128.into(), - }, - ) -} - -pub fn pending_bond_to_bond(pending: &PendingBond) -> Vec { - pending - .bonds - .iter() - .map(|bond| Bond { - amount: match &bond.raw_amount { - RawAmount::LocalDenom(amount) => *amount, - RawAmount::LpShares(_) => panic!("unexpected lp shares"), - }, - owner: bond.owner.clone(), - bond_id: bond.bond_id.clone(), - }) - .collect() -} - -pub fn create_query_response(response: Vec) -> ResponseQuery { - ResponseQuery { - code: 1, - log: "".to_string(), - info: "".to_string(), - index: 1, - key: Bytes::from("0"), - value: response.into(), - proof_ops: None, - height: 0, - codespace: "".to_string(), - } -} - -#[cfg(test)] -mod tests { - use cosmwasm_std::{testing::mock_dependencies, Addr, Uint128}; - - use crate::state::OngoingDeposit; - - use super::*; - - #[test] - fn default_setup_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - - assert_eq!( - CONFIG.load(deps.as_ref().storage).unwrap(), - Config { - lock_period: 100, - pool_id: 1, - pool_denom: "gamm/pool/1".to_string(), - base_denom: "uosmo".to_string(), - quote_denom: "uqsr".to_string(), - local_denom: "ibc/local_osmo".to_string(), - transfer_channel: "channel-0".to_string(), - return_source_channel: "channel-0".to_string(), - expected_connection: "connection-0".to_string(), - } - ); - - let ica_chan = ICA_CHANNEL.load(deps.as_ref().storage).unwrap(); - assert_eq!( - CHANNELS - .load(deps.as_ref().storage, ica_chan.clone()) - .unwrap(), - ChannelInfo { - id: ica_chan.clone(), - counterparty_endpoint: IbcEndpoint { - port_id: "icahost".to_string(), - channel_id: ica_chan, - }, - connection_id: "connection-0".to_string(), - channel_type: ChannelType::Ica { - channel_ty: IcaMetadata::with_connections( - "connection-1".to_string(), - "connection-2".to_string() - ), - counter_party_address: Some("osmo-address".to_string()), - }, - handshake_state: quasar_types::ibc::HandshakeState::Open, - } - ); - - let icq_chan = ICQ_CHANNEL.load(deps.as_ref().storage).unwrap(); - assert_eq!( - CHANNELS - .load(deps.as_ref().storage, icq_chan.clone()) - .unwrap(), - ChannelInfo { - id: icq_chan.to_string(), - counterparty_endpoint: IbcEndpoint { - port_id: "icqhost".to_string(), - channel_id: icq_chan, - }, - connection_id: "connection-0".to_string(), - channel_type: ChannelType::Icq { - channel_ty: "icq-1".to_string(), - }, - handshake_state: quasar_types::ibc::HandshakeState::Open, - } - ) - } - - #[test] - fn test_pending_bond_to_bond_works() { - let pb = PendingBond { - bonds: vec![ - OngoingDeposit { - claim_amount: Uint128::new(100), - raw_amount: RawAmount::LocalDenom(Uint128::new(1000)), - owner: Addr::unchecked("address"), - bond_id: "1".to_string(), - }, - OngoingDeposit { - claim_amount: Uint128::new(99), - raw_amount: RawAmount::LocalDenom(Uint128::new(999)), - owner: Addr::unchecked("address"), - bond_id: "2".to_string(), - }, - OngoingDeposit { - claim_amount: Uint128::new(101), - raw_amount: RawAmount::LocalDenom(Uint128::new(1000)), - owner: Addr::unchecked("address"), - bond_id: "3".to_string(), - }, - ], - }; - - let bonds = pending_bond_to_bond(&pb); - assert_eq!(bonds[0].amount, Uint128::new(1000)); - assert_eq!(bonds[1].amount, Uint128::new(999)); - assert_eq!(bonds[2].amount, Uint128::new(1000)); - assert_eq!(bonds[0].owner, Addr::unchecked("address")); - assert_eq!(bonds[1].owner, Addr::unchecked("address")); - assert_eq!(bonds[2].owner, Addr::unchecked("address")); - assert_eq!(bonds[0].bond_id, "1"); - assert_eq!(bonds[1].bond_id, "2"); - assert_eq!(bonds[2].bond_id, "3"); - } -} diff --git a/smart-contracts/osmosis/contracts/lp-strategy/src/unbond.rs b/smart-contracts/osmosis/contracts/lp-strategy/src/unbond.rs deleted file mode 100644 index 351e92b07..000000000 --- a/smart-contracts/osmosis/contracts/lp-strategy/src/unbond.rs +++ /dev/null @@ -1,659 +0,0 @@ -use cosmwasm_std::{ - to_json_binary, Addr, BankMsg, Coin, CosmosMsg, Env, IbcTimeout, Order, QuerierWrapper, - Storage, SubMsg, Timestamp, Uint128, WasmMsg, -}; -use osmosis_std::types::{ - cosmos::base::v1beta1::Coin as OsmoCoin, osmosis::gamm::v1beta1::MsgExitSwapShareAmountIn, -}; -use quasar_types::{ - callback::{Callback, UnbondResponse}, - ibc::MsgTransfer, - ica::packet::ica_send, -}; -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -use crate::{ - error::ContractError, - helpers::{create_ibc_ack_submsg, get_ica_address, IbcMsgKind, IcaMessages}, - ibc_util::calculate_token_out_min_amount, - msg::ExecuteMsg, - state::{ - RawAmount, CONFIG, IBC_TIMEOUT_TIME, ICA_CHANNEL, PENDING_UNBOND_QUEUE, RETURNING, - RETURN_SOURCE_PORT, UNBONDING_CLAIMS, UNBOND_QUEUE, - }, -}; - -/// do_unbond is used to check whether an unbonding claim can be processed, keeps record of unbonding attempts, -/// and manages the workflow of unbonding operations. -/// -/// It first retrieves the corresponding unbonding claim from storage and checks if the unlock time -/// of the unbonding claim has already passed by comparing it to the current block time. If the unlock time -/// has not yet passed, the function returns an error indicating that the shares are not yet available for unbonding. -/// -/// If the unlock time has already passed, the `attempted` field of the unbonding claim is set to `true`. -/// This operation marks that an unbonding attempt has been made for these shares and prevents the execution of -/// multiple unbonding attempts for the same shares. -/// -/// Finally, the unbonding claim is moved into the PENDING_UNBOND_QUEUE for later processing. -pub fn do_unbond( - storage: &mut dyn Storage, - env: &Env, - owner: Addr, - id: String, -) -> Result<(), ContractError> { - let mut unbond = UNBONDING_CLAIMS.load(storage, (owner.clone(), id.clone()))?; - - if unbond.unlock_time.nanos() > env.block.time.nanos() { - return Err(ContractError::SharesNotYetUnbonded); - } - - unbond.attempted = true; - UNBONDING_CLAIMS.save(storage, (owner, id), &unbond)?; - - Ok(PENDING_UNBOND_QUEUE.push_back(storage, &unbond)?) -} - -pub fn batch_unbond(storage: &mut dyn Storage, env: &Env) -> Result, ContractError> { - let mut total_exit = Uint128::zero(); - let mut pending: Vec = vec![]; - - if UNBOND_QUEUE.is_empty(storage)? { - return Ok(None); - } - - // aggregate the current unbond queue, all items in this queue should be able to unbond - while !UNBOND_QUEUE.is_empty(storage)? { - let unbond = UNBOND_QUEUE - .pop_front(storage)? - .ok_or(ContractError::QueueItemNotFound { - queue: "unbond".to_string(), - })?; - total_exit = total_exit - .checked_add(unbond.lp_shares) - .map_err(|err| ContractError::TracedOverflowError(err, "cal_total_exit".to_string()))?; - // add the unbond to the pending unbonds - pending.push(ReturningUnbond { - amount: RawAmount::LpShares(unbond.lp_shares), - owner: unbond.owner, - id: unbond.id, - }); - } - - // important to use lp_shares before it gets updated - let token_out_min_amount = calculate_token_out_min_amount(storage)?; - - let msg = exit_swap( - storage, - env, - total_exit, - token_out_min_amount, - PendingReturningUnbonds { unbonds: pending }, - )?; - Ok(Some(msg)) -} - -pub(crate) fn exit_swap( - storage: &mut dyn Storage, - env: &Env, - total_exit: Uint128, - token_out_min_amount: Uint128, - pending: PendingReturningUnbonds, -) -> Result { - let pkt = do_exit_swap(storage, env, token_out_min_amount, total_exit)?; - - let channel = ICA_CHANNEL.load(storage)?; - - Ok(create_ibc_ack_submsg( - storage, - IbcMsgKind::Ica(IcaMessages::ExitPool(pending)), - pkt, - channel, - )?) -} - -pub(crate) fn do_exit_swap( - storage: &mut dyn Storage, - env: &Env, - token_out_min_amount: Uint128, - total_exit: Uint128, -) -> Result { - let ica_address = get_ica_address(storage, ICA_CHANNEL.load(storage)?)?; - let config = CONFIG.load(storage)?; - - let msg = MsgExitSwapShareAmountIn { - sender: ica_address, - pool_id: config.pool_id, - token_out_denom: config.base_denom, - share_in_amount: total_exit.to_string(), - token_out_min_amount: token_out_min_amount.to_string(), - }; - - let pkt = ica_send::( - msg, - ICA_CHANNEL.load(storage)?, - IbcTimeout::with_timestamp(env.block.time.plus_seconds(IBC_TIMEOUT_TIME)), - )?; - Ok(pkt) -} - -// TODO the total tokens parameter and pending is maybe a little weird, check whether we want to fold pending to get total_tokens (with gas costs etc) -pub fn transfer_batch_unbond( - storage: &mut dyn Storage, - env: &Env, - pending: PendingReturningUnbonds, - total_tokens: Uint128, -) -> Result { - let pkt = do_transfer_batch_unbond(storage, env, total_tokens, pending.clone())?; - - // this is an ica channel in transfer batch unbond which is fine because even though - // we are doing a transfer, its a return transfer which must be triggered by an ICA - let channel = ICA_CHANNEL.load(storage)?; - - Ok(create_ibc_ack_submsg( - storage, - IbcMsgKind::Ica(IcaMessages::ReturnTransfer(pending)), - pkt, - channel, - )?) -} - -pub(crate) fn do_transfer_batch_unbond( - storage: &mut dyn Storage, - env: &Env, - total_tokens: Uint128, - pending: PendingReturningUnbonds, -) -> Result { - // TODO, assert that raw amounts equal amount - let timeout_timestamp = - IbcTimeout::with_timestamp(env.block.time.plus_seconds(IBC_TIMEOUT_TIME)); - let msg = return_transfer( - storage, - env, - total_tokens, - timeout_timestamp.timestamp().unwrap(), - pending, - )?; - let pkt = ica_send::( - msg, - ICA_CHANNEL.load(storage)?, - IbcTimeout::with_timestamp(env.block.time.plus_seconds(IBC_TIMEOUT_TIME)), - )?; - Ok(pkt) -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug, Eq)] -#[serde(rename_all = "snake_case")] -pub struct PendingReturningUnbonds { - pub unbonds: Vec, -} - -impl PendingReturningUnbonds { - /// convert a se of lp shares to a set of local tokens - pub fn lp_to_local_denom(&mut self, total_local: Uint128) -> Result { - let mut total_lp = Uint128::zero(); - for p in self.unbonds.iter() { - match p.amount { - crate::state::RawAmount::LocalDenom(_) => unimplemented!(), - crate::state::RawAmount::LpShares(val) => total_lp = total_lp.checked_add(val)?, - } - } - for p in self.unbonds.iter_mut() { - match p.amount { - // amount of tokens = lp_shares * total / total_lp - crate::state::RawAmount::LpShares(val) => { - p.amount = - RawAmount::LocalDenom(val.checked_mul(total_local)?.checked_div(total_lp)?) - } - crate::state::RawAmount::LocalDenom(_) => unimplemented!(), - } - } - Ok(total_lp) - } -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -pub struct ReturningUnbond { - pub amount: RawAmount, - pub owner: Addr, - pub id: String, -} - -// TODO this only works for the happy path in the receiver -pub fn finish_unbond( - storage: &dyn Storage, - querier: QuerierWrapper, - unbond: &ReturningUnbond, -) -> Result { - let amount = match unbond.amount { - RawAmount::LocalDenom(val) => val, - RawAmount::LpShares(_) => return Err(ContractError::IncorrectRawAmount), - }; - let msg: CosmosMsg = if querier - .query_wasm_contract_info(unbond.owner.as_str()) - .is_ok() - { - CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr: unbond.owner.to_string(), - msg: to_json_binary(&Callback::UnbondResponse(UnbondResponse { - unbond_id: unbond.id.clone(), - }))?, - funds: vec![Coin { - denom: CONFIG.load(storage)?.local_denom, - amount, - }], - }) - } else { - CosmosMsg::Bank(BankMsg::Send { - to_address: unbond.owner.to_string(), - amount: vec![Coin { - denom: CONFIG.load(storage)?.local_denom, - amount, - }], - }) - }; - Ok(msg) -} - -fn return_transfer( - storage: &mut dyn Storage, - env: &Env, - amount: Uint128, - timeout_timestamp: Timestamp, - pending: PendingReturningUnbonds, -) -> Result { - let config = CONFIG.load(storage)?; - let ica_address = get_ica_address(storage, ICA_CHANNEL.load(storage)?)?; - let id = get_next_return_id(storage)?; - - RETURNING.save(storage, id, &amount)?; - - Ok(MsgTransfer { - // TODO do we want to keep the return port a constant? Leaning towards yes since ibc transfer app uses this the same - source_port: RETURN_SOURCE_PORT.to_string(), - source_channel: config.return_source_channel, - token: Some(OsmoCoin { - denom: config.base_denom, - amount: amount.to_string(), - }), - sender: ica_address, - receiver: env.contract.address.clone().to_string(), - // timeout_height is disabled when set to 0 - // since height is kinda difficult to use, we always want to use the timestamp - timeout_height: None, - // timeout_timestamp is disabled when set to 0 - timeout_timestamp: Some(timeout_timestamp.nanos()), - memo: serde_json_wasm::to_string(&IbcHook { - wasm: Wasm { - contract: env.contract.address.clone(), - msg: ExecuteMsg::AcceptReturningFunds { id, pending }, - }, - }) - .map_err(|_| ContractError::SerdeJsonSer)?, - }) -} - -fn get_next_return_id(storage: &dyn Storage) -> Result { - let last = RETURNING - .range(storage, None, None, Order::Descending) - .next(); - let mut id: u64 = 0; - if let Some(val) = last { - id = val?.0; - } - Ok(id) -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -struct IbcHook { - wasm: Wasm, -} - -#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema, Debug)] -#[serde(rename_all = "snake_case")] -struct Wasm { - contract: Addr, - msg: ExecuteMsg, -} - -#[cfg(test)] -mod tests { - - use cosmwasm_std::{ - testing::{mock_dependencies, mock_env}, - CosmosMsg, - }; - - use crate::{ - state::{Unbond, LP_SHARES, SIMULATED_EXIT_RESULT}, - test_helpers::default_setup, - }; - - use super::*; - - #[test] - fn do_unbond_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let mut env = mock_env(); - let id = "my-id".to_string(); - - let unbond = Unbond { - lp_shares: Uint128::new(100), - unlock_time: env.block.time, - attempted: true, - owner: owner.clone(), - id: id.clone(), - }; - UNBONDING_CLAIMS - .save(deps.as_mut().storage, (owner.clone(), id.clone()), &unbond) - .unwrap(); - - let time = mock_env().block.time.plus_seconds(101); - env.block.time = time; - do_unbond(deps.as_mut().storage, &env, owner, id).unwrap(); - assert_eq!( - PENDING_UNBOND_QUEUE - .pop_front(deps.as_mut().storage) - .unwrap() - .unwrap(), - unbond - ) - } - - #[test] - fn do_unbond_early_fails() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let owner = Addr::unchecked("bob"); - let env = mock_env(); - let id = "my-id".to_string(); - - UNBONDING_CLAIMS - .save( - deps.as_mut().storage, - (owner.clone(), id.clone()), - &Unbond { - lp_shares: Uint128::new(100), - unlock_time: env.block.time.plus_nanos(1), - attempted: false, - owner: owner.clone(), - id: id.clone(), - }, - ) - .unwrap(); - - let err = do_unbond(deps.as_mut().storage, &env, owner, id).unwrap_err(); - assert_eq!(err, ContractError::SharesNotYetUnbonded) - } - - #[test] - fn batch_unbond_empty_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let env = mock_env(); - - let res = batch_unbond(deps.as_mut().storage, &env).unwrap(); - assert!(res.is_none()) - } - - #[test] - fn batch_unbond_multiple_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let env = mock_env(); - let owner = Addr::unchecked("bob"); - let id = "my-id".to_string(); - - // test specific setup - LP_SHARES - .save( - deps.as_mut().storage, - &crate::state::LpCache { - locked_shares: Uint128::new(500), - w_unlocked_shares: Uint128::zero(), - d_unlocked_shares: Uint128::zero(), - }, - ) - .unwrap(); - - let unbonds = vec![ - Unbond { - lp_shares: Uint128::new(100), - unlock_time: env.block.time, - attempted: false, - owner: owner.clone(), - id: id.clone(), - }, - Unbond { - lp_shares: Uint128::new(101), - unlock_time: env.block.time, - attempted: false, - owner: owner.clone(), - id: id.clone(), - }, - Unbond { - lp_shares: Uint128::new(102), - unlock_time: env.block.time, - attempted: false, - owner, - id, - }, - ]; - - for unbond in unbonds.iter() { - UNBOND_QUEUE - .push_back(deps.as_mut().storage, unbond) - .unwrap(); - } - - SIMULATED_EXIT_RESULT - .save(deps.as_mut().storage, &Uint128::from(100u128)) - .unwrap(); - - let res = batch_unbond(deps.as_mut().storage, &env).unwrap(); - assert!(res.is_some()); - - // checking above we have total exit amount = 100 + 101 + 102 - // while total shares in lp cache is defined at 500, this is important for calculating token_min_out_amount - // these asserts are just a quality of life improvement since the failure in this test sucks - let expected_exit_amount = Uint128::from(100u128 + 101u128 + 102u128); - let actual_exit_amount = unbonds - .iter() - .fold(Uint128::zero(), |acc, u| acc + u.lp_shares); - assert_eq!(expected_exit_amount, actual_exit_amount); - - let token_out_min_amount = calculate_token_out_min_amount(deps.as_mut().storage).unwrap(); - - // check that the packet is as we expect - let ica_address = get_ica_address( - deps.as_ref().storage, - ICA_CHANNEL.load(deps.as_ref().storage).unwrap(), - ) - .unwrap(); - let config = CONFIG.load(deps.as_ref().storage).unwrap(); - let msg = MsgExitSwapShareAmountIn { - sender: ica_address, - pool_id: config.pool_id, - token_out_denom: config.base_denom, - share_in_amount: Uint128::new(303).to_string(), - token_out_min_amount: token_out_min_amount.to_string(), - }; - - let pkt = ica_send::( - msg, - ICA_CHANNEL.load(deps.as_ref().storage).unwrap(), - IbcTimeout::with_timestamp(env.block.time.plus_seconds(IBC_TIMEOUT_TIME)), - ) - .unwrap(); - - assert_eq!(res.unwrap().msg, CosmosMsg::Ibc(pkt)); - } - - #[test] - fn transfer_batch_unbond_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let env = mock_env(); - let owner = Addr::unchecked("bob"); - let id = "my-id".to_string(); - - let pending = PendingReturningUnbonds { - unbonds: vec![ - ReturningUnbond { - amount: RawAmount::LocalDenom(Uint128::new(101)), - owner: owner.clone(), - id: id.clone(), - }, - ReturningUnbond { - amount: RawAmount::LocalDenom(Uint128::new(102)), - owner: owner.clone(), - id: id.clone(), - }, - ReturningUnbond { - amount: RawAmount::LocalDenom(Uint128::new(103)), - owner, - id, - }, - ], - }; - - let total_tokens = Uint128::new(306); - let timeout_timestamp = - IbcTimeout::with_timestamp(env.block.time.plus_seconds(IBC_TIMEOUT_TIME)); - - let res = transfer_batch_unbond(deps.as_mut().storage, &env, pending.clone(), total_tokens) - .unwrap(); - - let msg = return_transfer( - deps.as_mut().storage, - &env, - total_tokens, - timeout_timestamp.timestamp().unwrap(), - pending, - ) - .unwrap(); - - let pkt = ica_send::( - msg, - ICA_CHANNEL.load(deps.as_ref().storage).unwrap(), - IbcTimeout::with_timestamp(env.block.time.plus_seconds(IBC_TIMEOUT_TIME)), - ) - .unwrap(); - assert_eq!(res.msg, CosmosMsg::Ibc(pkt)); - } - - #[test] - fn test_lp_to_local_denom() { - let mut pending = PendingReturningUnbonds { - unbonds: vec![ - ReturningUnbond { - owner: Addr::unchecked("address"), - id: "bla".to_string(), - amount: RawAmount::LpShares(Uint128::new(100)), - }, - ReturningUnbond { - owner: Addr::unchecked("address"), - id: "bla".to_string(), - amount: RawAmount::LpShares(Uint128::new(50)), - }, - ReturningUnbond { - owner: Addr::unchecked("address"), - id: "bla".to_string(), - amount: RawAmount::LpShares(Uint128::new(150)), - }, - ], - }; - pending.lp_to_local_denom(Uint128::new(3000)).unwrap(); - assert_eq!( - pending.unbonds[0].amount, - RawAmount::LocalDenom(Uint128::new(1000)) - ); - assert_eq!( - pending.unbonds[1].amount, - RawAmount::LocalDenom(Uint128::new(500)) - ); - assert_eq!( - pending.unbonds[2].amount, - RawAmount::LocalDenom(Uint128::new(1500)) - ) - } - - #[test] - fn exit_swap_works() { - let mut deps = mock_dependencies(); - default_setup(deps.as_mut().storage).unwrap(); - let env = mock_env(); - - let pending = PendingReturningUnbonds { - unbonds: vec![ - ReturningUnbond { - owner: Addr::unchecked("address"), - id: "bla".to_string(), - amount: RawAmount::LpShares(Uint128::new(100)), - }, - ReturningUnbond { - owner: Addr::unchecked("address"), - id: "bla".to_string(), - amount: RawAmount::LpShares(Uint128::new(50)), - }, - ReturningUnbond { - owner: Addr::unchecked("address"), - id: "bla".to_string(), - amount: RawAmount::LpShares(Uint128::new(150)), - }, - ], - }; - - SIMULATED_EXIT_RESULT - .save(deps.as_mut().storage, &Uint128::new(3000)) - .unwrap(); - - let total_exit = pending - .unbonds - .iter() - .fold(Uint128::zero(), |acc, u| match u.amount { - RawAmount::LocalDenom(_) => unimplemented!(), - RawAmount::LpShares(val) => acc + val, - }); - - let token_out_min_amount = calculate_token_out_min_amount(deps.as_mut().storage).unwrap(); - - let msg = exit_swap( - deps.as_mut().storage, - &env, - total_exit, - token_out_min_amount, - pending, - ) - .unwrap(); - - let ica_address = get_ica_address( - deps.as_ref().storage, - ICA_CHANNEL.load(deps.as_ref().storage).unwrap(), - ) - .unwrap(); - let config = CONFIG.load(deps.as_ref().storage).unwrap(); - - let expected = MsgExitSwapShareAmountIn { - sender: ica_address, - pool_id: config.pool_id, - token_out_denom: config.base_denom, - share_in_amount: total_exit.to_string(), - // TODO add a more robust estimation - token_out_min_amount: token_out_min_amount.to_string(), - }; - - let pkt = ica_send::( - expected, - ICA_CHANNEL.load(deps.as_ref().storage).unwrap(), - IbcTimeout::with_timestamp(env.block.time.plus_seconds(IBC_TIMEOUT_TIME)), - ) - .unwrap(); - - assert_eq!(msg.msg, CosmosMsg::Ibc(pkt)) - } -} diff --git a/smart-contracts/osmosis/contracts/multihop-router/.cargo/config b/smart-contracts/osmosis/contracts/multihop-router/.cargo/config deleted file mode 100644 index af5698e58..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/.cargo/config +++ /dev/null @@ -1,4 +0,0 @@ -[alias] -wasm = "build --release --lib --target wasm32-unknown-unknown" -unit-test = "test --lib" -schema = "run --bin schema" diff --git a/smart-contracts/osmosis/contracts/multihop-router/.editorconfig b/smart-contracts/osmosis/contracts/multihop-router/.editorconfig deleted file mode 100644 index 3d36f20b1..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/.editorconfig +++ /dev/null @@ -1,11 +0,0 @@ -root = true - -[*] -indent_style = space -indent_size = 2 -charset = utf-8 -trim_trailing_whitespace = true -insert_final_newline = true - -[*.rs] -indent_size = 4 diff --git a/smart-contracts/osmosis/contracts/multihop-router/.gitignore b/smart-contracts/osmosis/contracts/multihop-router/.gitignore deleted file mode 100644 index 9095deaa4..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/.gitignore +++ /dev/null @@ -1,16 +0,0 @@ -# Build results -/target -/schema - -# Cargo+Git helper file (https://github.com/rust-lang/cargo/blob/0.44.1/src/cargo/sources/git/utils.rs#L320-L327) -.cargo-ok - -# Text file backups -**/*.rs.bk - -# macOS -.DS_Store - -# IDEs -*.iml -.idea diff --git a/smart-contracts/osmosis/contracts/multihop-router/Cargo.toml b/smart-contracts/osmosis/contracts/multihop-router/Cargo.toml deleted file mode 100644 index 8cbfd0392..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/Cargo.toml +++ /dev/null @@ -1,62 +0,0 @@ -[package] -name = "multihop-router" -version = "0.1.0" -authors = ["LaurensKubat <32776056+LaurensKubat@users.noreply.github.com>"] -edition = "2021" - -exclude = [ - # Those files are rust-optimizer artifacts. You might want to commit them for convenience but they should not be part of the source code publication. - "contract.wasm", - "hash.txt", -] - -# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html - -[lib] -crate-type = ["cdylib", "rlib"] - -[profile.release] -opt-level = 3 -debug = false -rpath = false -lto = true -debug-assertions = false -codegen-units = 1 -panic = 'abort' -incremental = false -overflow-checks = true - -[features] -default = ["mutable"] -# for more explicit tests, cargo test --features=backtraces -backtraces = ["cosmwasm-std/backtraces"] -# use library feature to disable all instantiate/execute/query exports -library = [] -mutable = [] - -[package.metadata.scripts] -optimize = """docker run --rm -v "$(pwd)":/code \ - --mount type=volume,source="$(basename "$(pwd)")_cache",target=/code/target \ - --mount type=volume,source=registry_cache,target=/usr/local/cargo/registry \ - cosmwasm/rust-optimizer:0.12.10 -""" - -[dependencies] -cosmwasm-std = { workspace = true } -osmosis-std = { workspace = true } -osmosis-std-derive = { workspace = true } -cosmwasm-schema = { workspace = true } -cw-storage-plus ={ workspace = true } -schemars ={ workspace = true } -serde = { workspace = true } -thiserror = { workspace = true } -cw2 = { workspace = true } -cosmwasm-storage = { workspace = true } - -[dev-dependencies] -cw-multi-test = { workspace = true } -serde-json-wasm = { workspace = true } - -anyhow = "1" -proptest = "1.0.0" -derivative = "2.2.0" diff --git a/smart-contracts/osmosis/contracts/multihop-router/LICENSE b/smart-contracts/osmosis/contracts/multihop-router/LICENSE deleted file mode 100644 index d64569567..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/LICENSE +++ /dev/null @@ -1,202 +0,0 @@ - - Apache License - Version 2.0, January 2004 - http://www.apache.org/licenses/ - - TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION - - 1. Definitions. - - "License" shall mean the terms and conditions for use, reproduction, - and distribution as defined by Sections 1 through 9 of this document. - - "Licensor" shall mean the copyright owner or entity authorized by - the copyright owner that is granting the License. - - "Legal Entity" shall mean the union of the acting entity and all - other entities that control, are controlled by, or are under common - control with that entity. For the purposes of this definition, - "control" means (i) the power, direct or indirect, to cause the - direction or management of such entity, whether by contract or - otherwise, or (ii) ownership of fifty percent (50%) or more of the - outstanding shares, or (iii) beneficial ownership of such entity. - - "You" (or "Your") shall mean an individual or Legal Entity - exercising permissions granted by this License. - - "Source" form shall mean the preferred form for making modifications, - including but not limited to software source code, documentation - source, and configuration files. - - "Object" form shall mean any form resulting from mechanical - transformation or translation of a Source form, including but - not limited to compiled object code, generated documentation, - and conversions to other media types. - - "Work" shall mean the work of authorship, whether in Source or - Object form, made available under the License, as indicated by a - copyright notice that is included in or attached to the work - (an example is provided in the Appendix below). - - "Derivative Works" shall mean any work, whether in Source or Object - form, that is based on (or derived from) the Work and for which the - editorial revisions, annotations, elaborations, or other modifications - represent, as a whole, an original work of authorship. For the purposes - of this License, Derivative Works shall not include works that remain - separable from, or merely link (or bind by name) to the interfaces of, - the Work and Derivative Works thereof. - - "Contribution" shall mean any work of authorship, including - the original version of the Work and any modifications or additions - to that Work or Derivative Works thereof, that is intentionally - submitted to Licensor for inclusion in the Work by the copyright owner - or by an individual or Legal Entity authorized to submit on behalf of - the copyright owner. For the purposes of this definition, "submitted" - means any form of electronic, verbal, or written communication sent - to the Licensor or its representatives, including but not limited to - communication on electronic mailing lists, source code control systems, - and issue tracking systems that are managed by, or on behalf of, the - Licensor for the purpose of discussing and improving the Work, but - excluding communication that is conspicuously marked or otherwise - designated in writing by the copyright owner as "Not a Contribution." - - "Contributor" shall mean Licensor and any individual or Legal Entity - on behalf of whom a Contribution has been received by Licensor and - subsequently incorporated within the Work. - - 2. Grant of Copyright License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - copyright license to reproduce, prepare Derivative Works of, - publicly display, publicly perform, sublicense, and distribute the - Work and such Derivative Works in Source or Object form. - - 3. Grant of Patent License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - (except as stated in this section) patent license to make, have made, - use, offer to sell, sell, import, and otherwise transfer the Work, - where such license applies only to those patent claims licensable - by such Contributor that are necessarily infringed by their - Contribution(s) alone or by combination of their Contribution(s) - with the Work to which such Contribution(s) was submitted. If You - institute patent litigation against any entity (including a - cross-claim or counterclaim in a lawsuit) alleging that the Work - or a Contribution incorporated within the Work constitutes direct - or contributory patent infringement, then any patent licenses - granted to You under this License for that Work shall terminate - as of the date such litigation is filed. - - 4. Redistribution. You may reproduce and distribute copies of the - Work or Derivative Works thereof in any medium, with or without - modifications, and in Source or Object form, provided that You - meet the following conditions: - - (a) You must give any other recipients of the Work or - Derivative Works a copy of this License; and - - (b) You must cause any modified files to carry prominent notices - stating that You changed the files; and - - (c) You must retain, in the Source form of any Derivative Works - that You distribute, all copyright, patent, trademark, and - attribution notices from the Source form of the Work, - excluding those notices that do not pertain to any part of - the Derivative Works; and - - (d) If the Work includes a "NOTICE" text file as part of its - distribution, then any Derivative Works that You distribute must - include a readable copy of the attribution notices contained - within such NOTICE file, excluding those notices that do not - pertain to any part of the Derivative Works, in at least one - of the following places: within a NOTICE text file distributed - as part of the Derivative Works; within the Source form or - documentation, if provided along with the Derivative Works; or, - within a display generated by the Derivative Works, if and - wherever such third-party notices normally appear. The contents - of the NOTICE file are for informational purposes only and - do not modify the License. You may add Your own attribution - notices within Derivative Works that You distribute, alongside - or as an addendum to the NOTICE text from the Work, provided - that such additional attribution notices cannot be construed - as modifying the License. - - You may add Your own copyright statement to Your modifications and - may provide additional or different license terms and conditions - for use, reproduction, or distribution of Your modifications, or - for any such Derivative Works as a whole, provided Your use, - reproduction, and distribution of the Work otherwise complies with - the conditions stated in this License. - - 5. Submission of Contributions. Unless You explicitly state otherwise, - any Contribution intentionally submitted for inclusion in the Work - by You to the Licensor shall be under the terms and conditions of - this License, without any additional terms or conditions. - Notwithstanding the above, nothing herein shall supersede or modify - the terms of any separate license agreement you may have executed - with Licensor regarding such Contributions. - - 6. Trademarks. This License does not grant permission to use the trade - names, trademarks, service marks, or product names of the Licensor, - except as required for reasonable and customary use in describing the - origin of the Work and reproducing the content of the NOTICE file. - - 7. Disclaimer of Warranty. Unless required by applicable law or - agreed to in writing, Licensor provides the Work (and each - Contributor provides its Contributions) on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or - implied, including, without limitation, any warranties or conditions - of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A - PARTICULAR PURPOSE. You are solely responsible for determining the - appropriateness of using or redistributing the Work and assume any - risks associated with Your exercise of permissions under this License. - - 8. Limitation of Liability. In no event and under no legal theory, - whether in tort (including negligence), contract, or otherwise, - unless required by applicable law (such as deliberate and grossly - negligent acts) or agreed to in writing, shall any Contributor be - liable to You for damages, including any direct, indirect, special, - incidental, or consequential damages of any character arising as a - result of this License or out of the use or inability to use the - Work (including but not limited to damages for loss of goodwill, - work stoppage, computer failure or malfunction, or any and all - other commercial damages or losses), even if such Contributor - has been advised of the possibility of such damages. - - 9. Accepting Warranty or Additional Liability. While redistributing - the Work or Derivative Works thereof, You may choose to offer, - and charge a fee for, acceptance of support, warranty, indemnity, - or other liability obligations and/or rights consistent with this - License. However, in accepting such obligations, You may act only - on Your own behalf and on Your sole responsibility, not on behalf - of any other Contributor, and only if You agree to indemnify, - defend, and hold each Contributor harmless for any liability - incurred by, or claims asserted against, such Contributor by reason - of your accepting any such warranty or additional liability. - - END OF TERMS AND CONDITIONS - - APPENDIX: How to apply the Apache License to your work. - - To apply the Apache License to your work, attach the following - boilerplate notice, with the fields enclosed by brackets "[]" - replaced with your own identifying information. (Don't include - the brackets!) The text should be enclosed in the appropriate - comment syntax for the file format. We also recommend that a - file or class name and description of purpose be included on the - same "printed page" as the copyright notice for easier - identification within third-party archives. - - Copyright [yyyy] [name of copyright owner] - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. diff --git a/smart-contracts/osmosis/contracts/multihop-router/NOTICE b/smart-contracts/osmosis/contracts/multihop-router/NOTICE deleted file mode 100644 index 1d8984f55..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/NOTICE +++ /dev/null @@ -1,13 +0,0 @@ -Copyright 2023 LaurensKubat <32776056+LaurensKubat@users.noreply.github.com> - -Licensed under the Apache License, Version 2.0 (the "License"); -you may not use this file except in compliance with the License. -You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - -Unless required by applicable law or agreed to in writing, software -distributed under the License is distributed on an "AS IS" BASIS, -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -See the License for the specific language governing permissions and -limitations under the License. diff --git a/smart-contracts/osmosis/contracts/multihop-router/README.md b/smart-contracts/osmosis/contracts/multihop-router/README.md deleted file mode 100644 index 6e2ab8be0..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/README.md +++ /dev/null @@ -1,9 +0,0 @@ -# Multihop IBC router -The multihop IBC router is a contract that can be queried by other contracts for routes to get and format memo fields for IBC transfers. -The routes within the IBC router are decided upon by the admin of the contract. Adding, updating and removing routes is done by the admin. - -Depending on the expected users of the router, the admin should probably be a multisig or a dao contract to prevent abuse in formatting the memo field hops and sending the funds to a bad receiver. -Each intermediate receiver should be an some sort of account or wallet managed by the admin. - -## Use on Osmosis -For almost all tokens on Osmosis, there is no hop between Osmosis and the host chain. The only exception as of writing this is pstake. The easiest way to start using the router is to embed it in a contract or run it as a side car, depending on whether you trust any current running routers diff --git a/smart-contracts/osmosis/contracts/multihop-router/src/bin/schema.rs b/smart-contracts/osmosis/contracts/multihop-router/src/bin/schema.rs deleted file mode 100644 index 2e73096ab..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/src/bin/schema.rs +++ /dev/null @@ -1,11 +0,0 @@ -use cosmwasm_schema::write_api; - -use multihop_router::msg::{ExecuteMsg, InstantiateMsg, QueryMsg}; - -fn main() { - write_api! { - instantiate: InstantiateMsg, - execute: ExecuteMsg, - query: QueryMsg, - } -} diff --git a/smart-contracts/osmosis/contracts/multihop-router/src/contract.rs b/smart-contracts/osmosis/contracts/multihop-router/src/contract.rs deleted file mode 100644 index 51bd14f86..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/src/contract.rs +++ /dev/null @@ -1,261 +0,0 @@ -#[cfg(not(feature = "library"))] -use cosmwasm_std::entry_point; -use cosmwasm_std::{ - to_json_binary, Binary, Deps, DepsMut, Env, MessageInfo, Order, Response, StdResult, -}; -// use cw2::set_contract_version; - -use crate::error::{ContractError, ContractResult}; -use crate::helpers::is_contract_admin; -use crate::msg::{ - ExecuteMsg, GetMemoResponse, GetRouteResponse, InstantiateMsg, ListRoutesResponse, - MemoResponse, QueryMsg, -}; -use crate::route::{Route, RouteId}; -use crate::state::ROUTES; - -// version info for migration info -const CONTRACT_NAME: &str = "crates.io:multihop-router"; -const CONTRACT_VERSION: &str = env!("CARGO_PKG_VERSION"); - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn instantiate( - _deps: DepsMut, - _env: Env, - _info: MessageInfo, - _msg: InstantiateMsg, -) -> Result { - Ok(Response::new() - .add_attribute("action", "instantiate") - .add_attribute("contract", CONTRACT_NAME) - .add_attribute("version", CONTRACT_VERSION)) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn execute( - deps: DepsMut, - env: Env, - info: MessageInfo, - msg: ExecuteMsg, -) -> Result { - match msg { - ExecuteMsg::AddRoute { route_id, route } => { - execute_add_route(deps, env, info, &route_id, route) - } - #[cfg(feature = "mutable")] - ExecuteMsg::MutateRoute { - route_id, - new_route, - } => execute_mutate_route(deps, env, info, &route_id, new_route), - #[cfg(feature = "mutable")] - ExecuteMsg::RemoveRoute { route_id } => execute_remove_route(deps, env, info, &route_id), - } -} - -pub fn execute_add_route( - deps: DepsMut, - env: Env, - info: MessageInfo, - route_id: &RouteId, - route: Route, -) -> ContractResult { - is_contract_admin(&deps.querier, &env, &info.sender)?; - - if ROUTES.has(deps.storage, route_id) { - return Err(ContractError::DestinationAlreadyExists); - } - - ROUTES.save(deps.storage, route_id, &route)?; - - Ok(Response::new() - .add_attribute("action", "add_route") - .add_attribute("route_id", route_id.to_string()) - .add_attribute("route", route.to_string())) -} - -#[cfg(feature = "mutable")] -pub fn execute_mutate_route( - deps: DepsMut, - env: Env, - info: MessageInfo, - route_id: &RouteId, - route: Route, -) -> ContractResult { - is_contract_admin(&deps.querier, &env, &info.sender)?; - - if !ROUTES.has(deps.storage, route_id) { - return Err(ContractError::DestinationNotExists); - } - - ROUTES.save(deps.storage, route_id, &route)?; - - Ok(Response::new() - .add_attribute("action", "mutate_route") - .add_attribute("route_id", route_id.to_string()) - .add_attribute("route", route.to_string())) -} - -#[cfg(feature = "mutable")] -pub fn execute_remove_route( - deps: DepsMut, - env: Env, - info: MessageInfo, - route_id: &RouteId, -) -> ContractResult { - is_contract_admin(&deps.querier, &env, &info.sender)?; - - if !ROUTES.has(deps.storage, route_id) { - return Err(ContractError::DestinationNotExists); - } - - ROUTES.remove(deps.storage, route_id); - - Ok(Response::new() - .add_attribute("action", "remove_route") - .add_attribute("route_id", route_id.to_string())) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn query(deps: Deps, _env: Env, msg: QueryMsg) -> ContractResult { - match msg { - QueryMsg::GetMemo { - route_id, - timeout, - retries, - actual_memo, - } => Ok(to_json_binary(&handle_get_memo( - deps, - route_id, - timeout, - retries, - actual_memo, - )?)?), - QueryMsg::GetRoute { route_id } => Ok(to_json_binary(&handle_get_route(deps, route_id)?)?), - QueryMsg::ListRoutes {} => Ok(to_json_binary(&handle_list_routes(deps)?)?), - } -} - -fn handle_get_memo( - deps: Deps, - route_id: RouteId, - timeout: String, - retries: i64, - actual_memo: Option, -) -> ContractResult { - let route = ROUTES - .may_load(deps.storage, &route_id)? - .ok_or(ContractError::DestinationNotExists)?; - match route.hop { - Some(hop) => Ok(GetMemoResponse { - channel: route.channel, - port: route.port, - memo: MemoResponse::Forward(hop.to_memo(timeout, retries, actual_memo)), - }), - None => Ok(GetMemoResponse { - channel: route.channel, - port: route.port, - memo: MemoResponse::Actual(actual_memo), - }), - } -} - -fn handle_get_route(deps: Deps, route_id: RouteId) -> ContractResult { - let route = ROUTES - .may_load(deps.storage, &route_id)? - .ok_or(ContractError::DestinationNotExists)?; - Ok(GetRouteResponse { route }) -} - -fn handle_list_routes(deps: Deps) -> ContractResult { - let routes: StdResult> = ROUTES - .range(deps.storage, None, None, Order::Descending) - .collect(); - Ok(ListRoutesResponse { - routes: routes? - .iter() - .map(|(dst, val)| (dst.clone(), val.clone())) - .collect(), - }) -} - -#[cfg(test)] -mod tests { - use cosmwasm_std::{ - from_json, - testing::{mock_dependencies, mock_env}, - }; - - use crate::route::{Destination, Hop}; - - use super::*; - - #[test] - fn query_list_routes_works() { - let mut deps = mock_dependencies(); - let env = mock_env(); - - let hop1 = Hop::new( - "channel-1", - "transfer", - "cosmosBob", - Some(Hop::new("channel-2", "transfer", "quasarBob", None)), - ); - - let route1 = Route::new("channel-2", "transfer", Some(hop1)); - - let hop2 = Hop::new( - "channel-866", - "transfer", - "osmoBob", - Some(Hop::new("channel-644", "transfer", "quasarBob", None)), - ); - - let route2 = Route::new("channel-3", "transfer", Some(hop2)); - - ROUTES - .save( - deps.as_mut().storage, - &RouteId { - destination: Destination("osmosis".to_string()), - asset: "ibc/123".to_string(), - }, - &route1, - ) - .unwrap(); - - ROUTES - .save( - deps.as_mut().storage, - &RouteId { - destination: Destination("gaia".to_string()), - asset: "osmo".to_string(), - }, - &route2, - ) - .unwrap(); - - let result = query(deps.as_ref(), env, crate::msg::QueryMsg::ListRoutes {}).unwrap(); - let response: ListRoutesResponse = from_json(&result).unwrap(); - assert_eq!( - response, - ListRoutesResponse { - routes: vec![ - ( - RouteId { - destination: Destination("osmosis".to_string()), - asset: "ibc/123".to_string() - }, - route1 - ), - ( - RouteId { - destination: Destination("gaia".to_string()), - asset: "osmo".to_string() - }, - route2 - ) - ] - } - ) - } -} diff --git a/smart-contracts/osmosis/contracts/multihop-router/src/error.rs b/smart-contracts/osmosis/contracts/multihop-router/src/error.rs deleted file mode 100644 index 602238d1a..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/src/error.rs +++ /dev/null @@ -1,19 +0,0 @@ -use cosmwasm_std::StdError; -use thiserror::Error; - -pub type ContractResult = Result; - -#[derive(Error, Debug)] -pub enum ContractError { - #[error("{0}")] - Std(#[from] StdError), - - #[error("Unauthorized")] - Unauthorized {}, - - #[error("Destination already exists")] - DestinationAlreadyExists, - - #[error("Destination does not exist")] - DestinationNotExists, -} diff --git a/smart-contracts/osmosis/contracts/multihop-router/src/helpers.rs b/smart-contracts/osmosis/contracts/multihop-router/src/helpers.rs deleted file mode 100644 index 84cc5fb42..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/src/helpers.rs +++ /dev/null @@ -1,45 +0,0 @@ -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -use cosmwasm_std::{to_json_binary, Addr, CosmosMsg, Env, QuerierWrapper, StdResult, WasmMsg}; - -use crate::{msg::ExecuteMsg, ContractError}; - -/// CwTemplateContract is a wrapper around Addr that provides a lot of helpers -/// for working with this. -#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)] -pub struct CwTemplateContract(pub Addr); - -impl CwTemplateContract { - pub fn addr(&self) -> Addr { - self.0.clone() - } - - pub fn call>(&self, msg: T) -> StdResult { - let msg = to_json_binary(&msg.into())?; - Ok(WasmMsg::Execute { - contract_addr: self.addr().into(), - msg, - funds: vec![], - } - .into()) - } -} - -pub fn is_contract_admin( - querier: &QuerierWrapper, - env: &Env, - sus_admin: &Addr, -) -> Result<(), ContractError> { - let contract_admin = querier - .query_wasm_contract_info(&env.contract.address)? - .admin; - if let Some(contract_admin) = contract_admin { - if contract_admin != *sus_admin { - return Err(ContractError::Unauthorized {}); - } - } else { - return Err(ContractError::Unauthorized {}); - } - Ok(()) -} diff --git a/smart-contracts/osmosis/contracts/multihop-router/src/lib.rs b/smart-contracts/osmosis/contracts/multihop-router/src/lib.rs deleted file mode 100644 index 4f62f35cb..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/src/lib.rs +++ /dev/null @@ -1,11 +0,0 @@ -pub mod contract; -mod error; -pub mod helpers; -pub mod msg; -#[cfg(test)] -pub mod multitest; - -pub mod route; -pub mod state; - -pub use crate::error::ContractError; diff --git a/smart-contracts/osmosis/contracts/multihop-router/src/msg.rs b/smart-contracts/osmosis/contracts/multihop-router/src/msg.rs deleted file mode 100644 index 00e2c149d..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/src/msg.rs +++ /dev/null @@ -1,65 +0,0 @@ -use cosmwasm_schema::{cw_serde, QueryResponses}; - -use crate::route::{Memo, Route, RouteId}; - -#[cw_serde] -pub struct InstantiateMsg {} - -#[cw_serde] -#[non_exhaustive] -pub enum ExecuteMsg { - AddRoute { - route_id: RouteId, - route: Route, - }, - #[cfg(feature = "mutable")] - MutateRoute { - route_id: RouteId, - new_route: Route, - }, - #[cfg(feature = "mutable")] - RemoveRoute { - route_id: RouteId, - }, -} - -#[cw_serde] -#[derive(QueryResponses)] -pub enum QueryMsg { - // TODO change timeout to either a Timestamp or a Duration, also find a datastructure that is always a json - #[returns(GetMemoResponse)] - GetMemo { - route_id: RouteId, - timeout: String, - retries: i64, - actual_memo: Option, - }, - #[returns(GetRouteResponse)] - GetRoute { route_id: RouteId }, - #[returns(ListRoutesResponse)] - ListRoutes {}, -} - -#[cw_serde] -pub struct GetMemoResponse { - pub channel: String, - pub port: String, - pub memo: MemoResponse, -} - -#[cw_serde] -#[serde(untagged)] -pub enum MemoResponse { - Forward(Memo), - Actual(Option), -} - -#[cw_serde] -pub struct GetRouteResponse { - pub route: Route, -} - -#[cw_serde] -pub struct ListRoutesResponse { - pub routes: Vec<(RouteId, Route)>, -} diff --git a/smart-contracts/osmosis/contracts/multihop-router/src/multitest/common.rs b/smart-contracts/osmosis/contracts/multihop-router/src/multitest/common.rs deleted file mode 100644 index cb5ef320b..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/src/multitest/common.rs +++ /dev/null @@ -1,21 +0,0 @@ -pub use anyhow::Result; -pub use derivative::Derivative; - -pub use crate::contract::{execute, instantiate, query}; -pub use crate::{ - error::ContractError, - msg::{ExecuteMsg, InstantiateMsg, QueryMsg}, -}; -pub use cosmwasm_std::{coin, BlockInfo, Coin, Decimal, Empty, StdResult, Uint128}; -pub use cw_multi_test::{App, AppResponse, Contract, ContractWrapper, Executor}; - -pub const USER: &str = "user"; -pub const DEPLOYER: &str = "deployer"; -pub const EXECUTOR: &str = "executor"; -pub const DENOM: &str = "uosmo"; -pub const LOCAL_DENOM: &str = "ibc/ilovemymom"; - -pub fn contract() -> Box> { - let contract = ContractWrapper::new(execute, instantiate, query); - Box::new(contract) -} diff --git a/smart-contracts/osmosis/contracts/multihop-router/src/multitest/mod.rs b/smart-contracts/osmosis/contracts/multihop-router/src/multitest/mod.rs deleted file mode 100644 index 0fcb84a57..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/src/multitest/mod.rs +++ /dev/null @@ -1,3 +0,0 @@ -pub mod common; -pub mod suite; -pub mod test; diff --git a/smart-contracts/osmosis/contracts/multihop-router/src/multitest/suite.rs b/smart-contracts/osmosis/contracts/multihop-router/src/multitest/suite.rs deleted file mode 100644 index 7e52812d0..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/src/multitest/suite.rs +++ /dev/null @@ -1,251 +0,0 @@ -use anyhow::{Ok, Result as AnyResult}; -use serde::de::DeserializeOwned; - -use crate::{ - msg::{GetMemoResponse, GetRouteResponse, ListRoutesResponse, MemoResponse}, - multitest::common::*, - route::{Route, RouteId}, -}; -use cosmwasm_std::{testing::MockApi, to_json_binary, Addr, CosmosMsg, MemoryStorage, WasmMsg}; -use cw_multi_test::{ - App, AppBuilder, BankKeeper, DistributionKeeper, FailingModule, StakeKeeper, WasmKeeper, -}; - -pub type QuasarMultiHopRouterApp = App< - BankKeeper, - MockApi, - MemoryStorage, - FailingModule, - WasmKeeper, - StakeKeeper, - DistributionKeeper, ->; - -type OnFailFunction = fn(&[(RouteId, Route)], &[(RouteId, Route)]) -> T; - -use crate::msg::{ExecuteMsg, InstantiateMsg}; - -#[derive(Derivative)] -#[derivative(Debug)] -pub struct QuasarVaultSuite { - #[derivative(Debug = "ignore")] - pub app: QuasarMultiHopRouterApp, - // The account that deploys everything - pub deployer: Addr, - // executor address - pub executor: Addr, - // user address - pub user: Addr, - // router address - pub router: Addr, -} - -impl QuasarVaultSuite { - pub fn init(init_msg: InstantiateMsg, funds: Vec) -> Result { - let genesis_funds = vec![coin(150000, DENOM), coin(150000, LOCAL_DENOM)]; - let deployer = Addr::unchecked(DEPLOYER); - let executor = Addr::unchecked(EXECUTOR); - let user = Addr::unchecked(USER); - let mut app = AppBuilder::new().build(|router, _, storage| { - router - .bank - .init_balance(storage, &deployer, genesis_funds) - .unwrap(); - }); - app.send_tokens( - deployer.clone(), - user.clone(), - &[coin(50000, DENOM), coin(50000, LOCAL_DENOM)], - )?; - app.send_tokens( - deployer.clone(), - executor.clone(), - &[coin(50000, DENOM), coin(50000, LOCAL_DENOM)], - )?; - - let router_id = app.store_code(contract()); - - let addr = app.instantiate_contract( - router_id, - deployer.clone(), - &init_msg, - &funds, - "router-contract", - Some(deployer.to_string()), - )?; - Ok(QuasarVaultSuite { - app, - deployer, - executor, - user, - router: addr, - }) - } - - pub fn execute( - &mut self, - sender: Addr, - msg: ExecuteMsg, - funds: Vec, - ) -> AnyResult { - self.app.execute( - sender, - CosmosMsg::Wasm(WasmMsg::Execute { - contract_addr: self.router.to_string(), - msg: to_json_binary(&msg)?, - funds, - }), - ) - } - - pub fn query(&self, msg: QueryMsg) -> AnyResult - where - T: DeserializeOwned, - { - let res = self - .app - .wrap() - .query_wasm_smart::(self.router.clone(), &msg)?; - Ok(res) - } - - /// the same as check_queries but panics if any query gives a different result than expected from the expected routes - pub fn assert_queries(&self, expected: &[(RouteId, Route)]) { - self.assert_get_memo(expected); - self.assert_get_route(expected); - self.assert_list_route(expected); - } - - /// check whether all queries return values expected from the expected routes - pub fn check_queries(&self, expected: &[(RouteId, Route)]) -> AnyResult { - Ok(self.check_get_memo(expected)? - && self.check_get_route(expected)? - && self.check_list_routes(expected)?) - } - - pub fn assert_get_memo(&self, expected: &[(RouteId, Route)]) { - self.verify_get_memo( - expected, - |actual, expected| { - panic!( - "a different memo was produced than expected, memo: {:?} expected {:?}", - actual, expected - ) - }, - (), - ) - .unwrap() - } - - // do all contract queries and check that the values are the same as any of the routes in expected - pub fn check_get_memo(&self, expected: &[(RouteId, Route)]) -> AnyResult { - self.verify_get_memo(expected, |_, _: &[(RouteId, Route)]| false, true) - } - - pub fn verify_get_memo( - &self, - expected: &[(RouteId, Route)], - on_fail: fn(GetMemoResponse, &[(RouteId, Route)]) -> T, - on_succes: T, - ) -> AnyResult { - let timeout = "1000"; - let retries = 3; - let actual_memo = Some("{\"my-json\": \"myval\"}".to_string()); - for (id, route) in expected.iter() { - let res = self.query::(QueryMsg::GetMemo { - route_id: id.clone(), - timeout: timeout.to_string(), - retries, - actual_memo: actual_memo.clone(), - })?; - if res.channel != route.channel { - return Ok(on_fail(res, expected)); - } - if res.port != route.port { - return Ok(on_fail(res, expected)); - } - if let Some(hop) = route.hop.clone() { - if MemoResponse::Forward(hop.to_memo( - timeout.to_string(), - retries, - actual_memo.clone(), - )) != res.memo - { - return Ok(on_fail(res, expected)); - } - } else if MemoResponse::Actual(actual_memo.clone()) != res.memo { - return Ok(on_fail(res, expected)); - } - } - Ok(on_succes) - } - - pub fn assert_get_route(&self, expected: &[(RouteId, Route)]) { - self.verify_get_route( - expected, - |actual, expected| { - panic!( - "a different memo was produced than expected, memo: {:?} expected {:?}", - actual, expected - ) - }, - (), - ) - .unwrap() - } - - // do all contract queries and check that the values are the same as any of the routes in expected - pub fn check_get_route(&self, expected: &[(RouteId, Route)]) -> AnyResult { - self.verify_get_route(expected, |_, _| false, true) - } - - pub fn verify_get_route( - &self, - expected: &[(RouteId, Route)], - on_fail: fn((&RouteId, &Route), (&RouteId, &Route)) -> T, - on_succes: T, - ) -> AnyResult { - for (id, route) in expected.iter() { - let res = self.query::(QueryMsg::GetRoute { - route_id: id.clone(), - })?; - if &res.route != route { - return Ok(on_fail((id, &res.route), (id, route))); - } - } - Ok(on_succes) - } - - pub fn assert_list_route(&self, expected: &[(RouteId, Route)]) { - self.verify_get_memo( - expected, - |actual, expected| { - panic!( - "a different memo was produced than expected, memo: {:?} expected {:?}", - actual, expected - ) - }, - (), - ) - .unwrap() - } - - // do all contract queries and check that the values are the same as any of the routes in expected - pub fn check_list_routes(&self, expected: &[(RouteId, Route)]) -> AnyResult { - self.verify_get_memo(expected, |_, _| false, true) - } - - pub fn verify_list_routes( - &self, - expected: &[(RouteId, Route)], - on_fail: OnFailFunction, - on_succes: T, - ) -> AnyResult { - let res = self.query::(QueryMsg::ListRoutes {})?; - if res.routes.iter().all(|actual| expected.contains(actual)) { - Ok(on_succes) - } else { - Ok(on_fail(res.routes.as_ref(), expected)) - } - } -} diff --git a/smart-contracts/osmosis/contracts/multihop-router/src/multitest/test.rs b/smart-contracts/osmosis/contracts/multihop-router/src/multitest/test.rs deleted file mode 100644 index 5dd8ebc79..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/src/multitest/test.rs +++ /dev/null @@ -1,99 +0,0 @@ -use cosmwasm_std::attr; -use cosmwasm_std::Addr; -use cosmwasm_std::Event; - -use crate::msg::InstantiateMsg; -use crate::multitest::common::*; -use crate::multitest::suite::*; -use crate::route::Destination; -use crate::route::Hop; -use crate::route::Route; -use crate::route::RouteId; - -#[test] -fn route_lifecycle_works() { - // initialize the suite - let mut suite = QuasarVaultSuite::init(InstantiateMsg {}, vec![]).unwrap(); - - // create some mock routes - let mut osmo_routes = vec![( - RouteId::new(Destination::new("osmosis"), "uosmo".to_string()), - Route::new("channel-12", "transfer", None), - )]; - - // add the routes as admin - for route in osmo_routes.iter() { - let res = suite - .execute( - Addr::unchecked(DEPLOYER), - ExecuteMsg::AddRoute { - route_id: route.0.clone(), - route: route.1.clone(), - }, - vec![], - ) - .unwrap(); - - let e = Event::new("wasm").add_attributes(vec![ - attr("action", "add_route"), - attr("route_id", "destination: osmosis, asset: uosmo"), - attr("route", "channel: channel-12, port: transfer"), - ]); - res.assert_event(&e); - } - - suite.assert_queries(&osmo_routes); - // mutate the first route in our vec - let osmo_routes: Vec<(RouteId, Route)> = osmo_routes - .iter_mut() - .map(|val| { - ( - val.0.clone(), - Route::new( - "channel-13", - "transfer", - Some(Hop::new("channel-11", "transfer", "cosmos123", None)), - ), - ) - }) - .collect(); - - // mutate the route in our contract - let res = suite - .execute( - Addr::unchecked(DEPLOYER), - ExecuteMsg::MutateRoute { - route_id: osmo_routes[0].0.clone(), - new_route: osmo_routes[0].1.clone(), - }, - vec![], - ) - .unwrap(); - - let e = Event::new("wasm").add_attributes(vec![ - attr("action", "mutate_route"), - attr("route_id", "destination: osmosis, asset: uosmo"), - attr( - "route", - "channel: channel-13, port: transfer, hop: (channel: channel-11, port: transfer, receiver: cosmos123)", - ), - ]); - res.assert_event(&e); - - suite.assert_queries(&osmo_routes); - let res = suite - .execute( - Addr::unchecked(DEPLOYER), - ExecuteMsg::RemoveRoute { - route_id: osmo_routes[0].0.clone(), - }, - vec![], - ) - .unwrap(); - let e = Event::new("wasm").add_attributes(vec![ - attr("action", "remove_route"), - attr("route_id", "destination: osmosis, asset: uosmo"), - ]); - res.assert_event(&e); - suite.assert_queries(&[]); -} diff --git a/smart-contracts/osmosis/contracts/multihop-router/src/route.rs b/smart-contracts/osmosis/contracts/multihop-router/src/route.rs deleted file mode 100644 index 0d463f22b..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/src/route.rs +++ /dev/null @@ -1,379 +0,0 @@ -use std::fmt::Display; - -use cosmwasm_schema::cw_serde; -use cosmwasm_std::{StdError, StdResult}; -use cw_storage_plus::{Key, KeyDeserialize, Prefixer, PrimaryKey}; -use schemars::JsonSchema; -use serde::{Deserialize, Serialize}; - -/// A Route represents the route to take to a certain chain. Each route might be unique for a certain asset. -/// A complete usable case of a route includes the asset to send to the destination, since a different asset -/// might need to take a different route -/// A complete Route is then the kv-pair of the Map Routes, namely -#[cw_serde] -pub struct Route { - // the channel to use in an ibc transfer from the current chain - pub channel: String, - // the port to use, this is most likely always "transfer" - pub port: String, - // any potential hops needed to get to the current chain. These hops are dependend on the associated assets - pub hop: Option, -} - -impl Display for Route { - fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { - if self.hop.is_some() { - write!( - f, - "channel: {}, port: {}, hop: ({})", - self.channel, - self.port, - self.hop.as_ref().unwrap() - ) - } else { - write!(f, "channel: {}, port: {}", self.channel, self.port) - } - } -} - -impl Route { - pub fn new(channel: impl Into, port: impl Into, hop: Option) -> Route { - Route { - channel: channel.into(), - port: port.into(), - hop, - } - } -} - -#[cw_serde] -pub struct Hop { - // the channel to reach the first destination chain - channel: String, - // port will most likely be "transfer" - port: String, - // receiver is the receiver of the hop. If the chain has packet forward middelware properly integrated - // the receiver is never relevant. If PFM is not properly integrated, the receiver will have the funds. - // The users of the multihop router should ensure that the receiver works as intended - receiver: String, - // the next hop to take to reach the actual destination chain - next: Option>, -} - -impl Hop { - pub fn new( - channel: impl Into, - port: impl Into, - receiver: impl Into, - hop: Option, - ) -> Hop { - Hop { - channel: channel.into(), - port: port.into(), - receiver: receiver.into(), - next: hop.map(Box::new), - } - } - - /// create a packet forwarder memo field from a route of hops - /// receivers of the tokens on the intermediate chains - pub fn to_memo(&self, timeout: String, retries: i64, actual_memo: Option) -> Memo { - Memo::new(self.to_forward(timeout, retries, actual_memo)) - } - - // wtf are these clones even - fn to_forward(&self, timeout: String, retries: i64, actual_memo: Option) -> Forward { - Forward { - receiver: self.receiver.clone(), - port: self.port.clone(), - channel: self.channel.clone(), - timeout: timeout.clone(), - retries, - next: self - .clone() - .next - .map_or(Box::new(Next::Actual(actual_memo.clone())), |val| { - Box::new(Next::NextForward(val.to_forward( - timeout, - retries, - actual_memo, - ))) - }), - } - } -} - -impl Display for Hop { - fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { - if self.next.is_some() { - write!( - f, - "channel: {}, port: {}, receiver: {}, next: ({})", - self.channel, - self.port, - self.receiver, - self.next.as_ref().unwrap() - ) - } else { - write!( - f, - "channel: {}, port: {}, receiver: {}", - self.channel, self.port, self.receiver - ) - } - } -} - -// in the case of our multihop router, a memo is a set forwarding steps with an actual memo field attached for the host chan -#[cw_serde] -pub struct Memo { - pub forward: Forward, -} - -impl Memo { - pub fn new(forward: Forward) -> Memo { - Memo { forward } - } -} - -#[cw_serde] -pub struct Forward { - pub receiver: String, - pub port: String, - pub channel: String, - pub timeout: String, - pub retries: i64, - pub next: Box, -} - -impl Forward { - pub fn new( - receiver: impl Into, - port: impl Into, - channel: impl Into, - timeout: impl Into, - retries: i64, - next: Box, - ) -> Forward { - Forward { - receiver: receiver.into(), - port: port.into(), - channel: channel.into(), - timeout: timeout.into(), - retries, - next, - } - } -} - -#[cw_serde] -#[serde(untagged)] -pub enum Next { - NextForward(Forward), - Actual(Option), -} - -#[cw_serde] -pub struct RouteId { - pub destination: Destination, - pub asset: String, -} - -impl RouteId { - pub fn new(destination: Destination, asset: String) -> RouteId { - RouteId { destination, asset } - } -} - -impl Display for RouteId { - fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { - write!( - f, - "destination: {}, asset: {}", - self.destination, self.asset - ) - } -} - -impl<'a> PrimaryKey<'a> for RouteId { - type Prefix = Destination; - - type SubPrefix = (); - - type Suffix = String; - - type SuperSuffix = (Destination, String); - - fn key(&self) -> Vec { - let mut keys = self.destination.key(); - keys.extend(self.asset.key()); - keys - } -} - -impl KeyDeserialize for RouteId { - type Output = RouteId; - - fn from_vec(mut value: Vec) -> StdResult { - let mut tu = value.split_off(2); - let t_len = parse_length(&value)?; - let u = tu.split_off(t_len); - - Ok(RouteId { - destination: Destination::from_vec(tu)?, - asset: String::from_vec(u)?, - }) - } -} - -impl KeyDeserialize for &RouteId { - type Output = RouteId; - - fn from_vec(mut value: Vec) -> StdResult { - let mut tu = value.split_off(2); - let t_len = parse_length(&value)?; - let u = tu.split_off(t_len); - - Ok(RouteId { - destination: Destination::from_vec(tu)?, - asset: String::from_vec(u)?, - }) - } -} - -fn parse_length(value: &[u8]) -> StdResult { - Ok(u16::from_be_bytes( - value - .try_into() - .map_err(|_| StdError::generic_err("Could not read 2 byte length"))?, - ) - .into()) -} - -// destination uses a special partialEq, so we don't derive it -#[derive(Serialize, Deserialize, Clone, Debug, JsonSchema)] -pub struct Destination(pub String); - -impl Destination { - pub fn new(destination: impl Into) -> Destination { - Destination(destination.into()) - } -} - -impl From for Destination { - fn from(value: String) -> Self { - Self(value) - } -} - -impl Display for Destination { - fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { - write!(f, "{}", self.0) - } -} - -impl PartialEq for Destination { - // Destinination uses a case insensitive eq - fn eq(&self, other: &Self) -> bool { - self.0.to_lowercase() == other.0.to_lowercase() - } -} - -impl<'a> Prefixer<'a> for Destination { - fn prefix(&self) -> Vec { - self.0.prefix() - } -} - -impl<'a> PrimaryKey<'a> for Destination { - type Prefix = (); - type SubPrefix = (); - type Suffix = Self; - type SuperSuffix = Self; - - fn key(&self) -> Vec { - self.0.key() - } -} - -impl KeyDeserialize for Destination { - type Output = Destination; - - #[inline(always)] - fn from_vec(value: Vec) -> StdResult { - Ok(Destination( - String::from_utf8(value).map_err(StdError::invalid_utf8)?, - )) - } -} - -impl KeyDeserialize for &Destination { - type Output = Destination; - - #[inline(always)] - fn from_vec(value: Vec) -> StdResult { - Ok(Destination( - String::from_utf8(value).map_err(StdError::invalid_utf8)?, - )) - } -} - -#[cfg(test)] -mod tests { - use cw_storage_plus::PrimaryKey; - use proptest::prelude::*; - - use super::*; - - #[test] - fn se_json_works() { - // example json from packet forward middleware without any formatting characters so we match for it - let json_str = r#"{"forward":{"receiver":"chain-c-bech32-address","port":"transfer","channel":"channel-123","timeout":"10m","retries":2,"next":{"receiver":"chain-d-bech32-address","port":"transfer","channel":"channel-234","timeout":"10m","retries":2,"next":"{\"my-json\":\"myval\"}"}}}"#; - - let actual: Memo = serde_json_wasm::from_str(json_str).unwrap(); - let expected = Memo::new(Forward::new( - "chain-c-bech32-address", - "transfer", - "channel-123", - "10m", - 2, - Box::new(Next::NextForward(Forward::new( - "chain-d-bech32-address", - "transfer", - "channel-234", - "10m", - 2, - Box::new(Next::Actual(Some("{\"my-json\":\"myval\"}".to_string()))), - ))), - )); - - assert_eq!(actual, expected); - assert_eq!(serde_json_wasm::to_string(&actual).unwrap(), json_str); - assert_eq!(serde_json_wasm::to_string(&expected).unwrap(), json_str) - } - - prop_compose! { - fn route_id()(dst in any::(), asset in any::()) -> RouteId { - RouteId { destination: Destination(dst), asset } - } - } - - proptest! { - #[test] - fn route_id_key_ser_de(id in route_id()) { - let keys = id.joined_key(); - let route_id = RouteId::from_vec(keys).unwrap(); - prop_assert_eq!(id, route_id) - } - } - - proptest! { - #[test] - fn route_id_borrow_key_ser_de(id in route_id()) { - let b_id = &id; - let keys = b_id.joined_key(); - let route_id = &RouteId::from_vec(keys).unwrap(); - prop_assert_eq!(b_id, route_id) - } - } -} diff --git a/smart-contracts/osmosis/contracts/multihop-router/src/state.rs b/smart-contracts/osmosis/contracts/multihop-router/src/state.rs deleted file mode 100644 index 7d9548e91..000000000 --- a/smart-contracts/osmosis/contracts/multihop-router/src/state.rs +++ /dev/null @@ -1,7 +0,0 @@ -use cw_storage_plus::Map; - -use crate::route::{Route, RouteId}; - -/// ROUTES represents complete routes to a destination for a certain asset. -/// The value of Route represents the route to take to -pub const ROUTES: Map<&RouteId, Route> = Map::new("routes"); diff --git a/smart-contracts/osmosis/contracts/vault-rewards/.cargo/config b/smart-contracts/osmosis/contracts/vault-rewards/.cargo/config deleted file mode 100644 index 336b618a1..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/.cargo/config +++ /dev/null @@ -1,4 +0,0 @@ -[alias] -wasm = "build --release --target wasm32-unknown-unknown" -unit-test = "test --lib" -schema = "run --example schema" diff --git a/smart-contracts/osmosis/contracts/vault-rewards/.gitignore b/smart-contracts/osmosis/contracts/vault-rewards/.gitignore deleted file mode 100644 index e08217044..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/.gitignore +++ /dev/null @@ -1 +0,0 @@ -ts diff --git a/smart-contracts/osmosis/contracts/vault-rewards/CHANGELOG.md b/smart-contracts/osmosis/contracts/vault-rewards/CHANGELOG.md deleted file mode 100644 index 14674671c..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/CHANGELOG.md +++ /dev/null @@ -1,83 +0,0 @@ -# CHANGELOG - -## [0.1.1] - 2023-07-19 - -### Added -- Nothing - -### Changed -- Updated the query to fetch the claimable amount for give user address. - -Earlier it was not changing the user reward index, the push of current balance to history while querying fixed it : -```rust -pub fn query_pending_rewards( - deps: Deps, - env: Env, - user: Addr, -) -> Result { - let config = CONFIG.load(deps.storage)?; - let user_reward_index = get_user_reward_index(deps.storage, &user); - get_claim_amount(deps, &env, &config, &user_reward_index) -} -``` -Changed to : -```rust -pub fn query_pending_rewards( - deps: Deps, - env: Env, - user: Addr, - ) -> Result { - let config = CONFIG.load(deps.storage)?; - let cur_block_height = env.block.height; - let mut user_reward_index = get_user_reward_index(deps.storage, &user); - let user_vault_token_balance = - AssetInfo::cw20(config.vault_token.clone()).query_balance(&deps.querier, &user)?; - if let Some(prev_balance) = user_reward_index.balance { - user_reward_index.history.push(DistributionSchedule { - start: prev_balance.reward_index, - end: cur_block_height, - amount: prev_balance.balance, - }); - user_reward_index.balance = if !user_vault_token_balance.is_zero() { - Some(UserBalance { - reward_index: cur_block_height + 1, - balance: user_vault_token_balance, - }) - } else { - None - }; - } - get_claim_amount(deps, &env, &config, &user_reward_index) - } -``` - -### Deprecated -- Nothing - -### Removed -- Nothing - -### Fixed -- Fixed issue in fn `get_claim_amount()` where previously unwrap was happening on a nil vector. - -Old code -```rust -if reward_indexes.last().unwrap().0 != env.block.height && d.end == env.block.height { - reward_indexes.push((env.block.height, RewardIndex { vault_supply })); -} -``` -New code -```rust -if let Some(value) = reward_indexes.last() { - if value.0 != env.block.height & & d.end == env.block.height { - reward_indexes.push((env.block.height, RewardIndex { vault_supply })); - } -} -``` - -### Security -- Nothing - - - - diff --git a/smart-contracts/osmosis/contracts/vault-rewards/Cargo.toml b/smart-contracts/osmosis/contracts/vault-rewards/Cargo.toml deleted file mode 100644 index bb42d496e..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/Cargo.toml +++ /dev/null @@ -1,54 +0,0 @@ -[package] -name = "vault-rewards" -version = "0.1.0" -authors = ["shab "] -edition = "2021" - -exclude = [ - # Those files are rust-optimizer artifacts. You might want to commit them for convenience but they should not be part of the source code publication. - "contract.wasm", - "hash.txt", -] - -# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html - -[lib] -crate-type = ["cdylib", "rlib"] - -[profile.release] -opt-level = 3 -debug = false -rpath = false -lto = true -debug-assertions = false -codegen-units = 1 -panic = 'abort' -incremental = false -overflow-checks = true - -[features] -# for more explicit tests, cargo test --features=backtraces -backtraces = ["cosmwasm-std/backtraces"] -# use library feature to disable all instantiate/execute/query exports -library = [] - -[package.metadata.scripts] -optimize = """docker run --rm -v "$(pwd)":/code \ - --mount type=volume,source="$(basename "$(pwd)")_cache",target=/target \ - --mount type=volume,source=registry_cache,target=/usr/local/cargo/registry \ - cosmwasm/optimizer:0.16.0 -""" - -[dependencies] -cosmwasm-std = { workspace = true } -osmosis-std = { workspace = true } -osmosis-std-derive = { workspace = true } -cosmwasm-schema = { workspace = true } -cw-storage-plus ={ workspace = true } -schemars ={ workspace = true } -serde = { workspace = true } -thiserror = { workspace = true } -cw2 = { workspace = true } -cw20 = { workspace = true } -cw-asset = { workspace = true } -itertools = { workspace = true } diff --git a/smart-contracts/osmosis/contracts/vault-rewards/examples/schema.rs b/smart-contracts/osmosis/contracts/vault-rewards/examples/schema.rs deleted file mode 100644 index 551f8dd30..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/examples/schema.rs +++ /dev/null @@ -1,17 +0,0 @@ -use cosmwasm_schema::{export_schema, remove_schemas, schema_for}; -use std::env::current_dir; -use std::fs::create_dir_all; -use vault_rewards::msg::{ExecuteMsg, InstantiateMsg, QueryMsg}; -use vault_rewards::state::Config; - -fn main() { - let mut out_dir = current_dir().unwrap(); - out_dir.push("schema"); - create_dir_all(&out_dir).unwrap(); - remove_schemas(&out_dir).unwrap(); - - export_schema(&schema_for!(InstantiateMsg), &out_dir); - export_schema(&schema_for!(ExecuteMsg), &out_dir); - export_schema(&schema_for!(QueryMsg), &out_dir); - export_schema(&schema_for!(Config), &out_dir); -} diff --git a/smart-contracts/osmosis/contracts/vault-rewards/schema/config.json b/smart-contracts/osmosis/contracts/vault-rewards/schema/config.json deleted file mode 100644 index 22fdb530b..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/schema/config.json +++ /dev/null @@ -1,92 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "Config", - "type": "object", - "required": [ - "distribution_schedules", - "reward_token", - "total_claimed", - "vault_token" - ], - "properties": { - "distribution_schedules": { - "type": "array", - "items": { - "$ref": "#/definitions/DistributionSchedule" - } - }, - "reward_token": { - "$ref": "#/definitions/AssetInfoBase_for_Addr" - }, - "total_claimed": { - "$ref": "#/definitions/Uint128" - }, - "vault_token": { - "$ref": "#/definitions/Addr" - } - }, - "additionalProperties": false, - "definitions": { - "Addr": { - "description": "A human readable address.\n\nIn Cosmos, this is typically bech32 encoded. But for multi-chain smart contracts no assumptions should be made other than being UTF-8 encoded and of reasonable length.\n\nThis type represents a validated address. It can be created in the following ways 1. Use `Addr::unchecked(input)` 2. Use `let checked: Addr = deps.api.addr_validate(input)?` 3. Use `let checked: Addr = deps.api.addr_humanize(canonical_addr)?` 4. Deserialize from JSON. This must only be done from JSON that was validated before such as a contract's state. `Addr` must not be used in messages sent by the user because this would result in unvalidated instances.\n\nThis type is immutable. If you really need to mutate it (Really? Are you sure?), create a mutable copy using `let mut mutable = Addr::to_string()` and operate on that `String` instance.", - "type": "string" - }, - "AssetInfoBase_for_Addr": { - "description": "Represents the type of an fungible asset.\n\nEach **asset info** instance can be one of three variants:\n\n- Native SDK coins. To create an **asset info** instance of this type, provide the denomination. - CW20 tokens. To create an **asset info** instance of this type, provide the contract address.", - "oneOf": [ - { - "type": "object", - "required": [ - "native" - ], - "properties": { - "native": { - "type": "string" - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "cw20" - ], - "properties": { - "cw20": { - "$ref": "#/definitions/Addr" - } - }, - "additionalProperties": false - } - ] - }, - "DistributionSchedule": { - "type": "object", - "required": [ - "amount", - "end", - "start" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - }, - "end": { - "type": "integer", - "format": "uint64", - "minimum": 0.0 - }, - "start": { - "type": "integer", - "format": "uint64", - "minimum": 0.0 - } - }, - "additionalProperties": false - }, - "Uint128": { - "description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```", - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/vault-rewards/schema/execute_msg.json b/smart-contracts/osmosis/contracts/vault-rewards/schema/execute_msg.json deleted file mode 100644 index f17d66e75..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/schema/execute_msg.json +++ /dev/null @@ -1,250 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "ExecuteMsg", - "oneOf": [ - { - "type": "object", - "required": [ - "claim" - ], - "properties": { - "claim": { - "type": "object", - "additionalProperties": false - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "admin" - ], - "properties": { - "admin": { - "$ref": "#/definitions/AdminExecuteMsg" - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "vault" - ], - "properties": { - "vault": { - "$ref": "#/definitions/VaultExecuteMsg" - } - }, - "additionalProperties": false - } - ], - "definitions": { - "Addr": { - "description": "A human readable address.\n\nIn Cosmos, this is typically bech32 encoded. But for multi-chain smart contracts no assumptions should be made other than being UTF-8 encoded and of reasonable length.\n\nThis type represents a validated address. It can be created in the following ways 1. Use `Addr::unchecked(input)` 2. Use `let checked: Addr = deps.api.addr_validate(input)?` 3. Use `let checked: Addr = deps.api.addr_humanize(canonical_addr)?` 4. Deserialize from JSON. This must only be done from JSON that was validated before such as a contract's state. `Addr` must not be used in messages sent by the user because this would result in unvalidated instances.\n\nThis type is immutable. If you really need to mutate it (Really? Are you sure?), create a mutable copy using `let mut mutable = Addr::to_string()` and operate on that `String` instance.", - "type": "string" - }, - "AdminExecuteMsg": { - "oneOf": [ - { - "type": "object", - "required": [ - "withdraw_funds" - ], - "properties": { - "withdraw_funds": { - "$ref": "#/definitions/AssetBase_for_Addr" - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "add_distribution_schedule" - ], - "properties": { - "add_distribution_schedule": { - "$ref": "#/definitions/DistributionSchedule" - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "update_distribution_schedule" - ], - "properties": { - "update_distribution_schedule": { - "type": "object", - "required": [ - "id", - "update" - ], - "properties": { - "id": { - "type": "integer", - "format": "uint64", - "minimum": 0.0 - }, - "update": { - "$ref": "#/definitions/DistributionScheduleOptions" - } - }, - "additionalProperties": false - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "remove_distribution_schedule" - ], - "properties": { - "remove_distribution_schedule": { - "type": "integer", - "format": "uint64", - "minimum": 0.0 - } - }, - "additionalProperties": false - } - ] - }, - "AssetBase_for_Addr": { - "description": "Represents a fungible asset with a known amount\n\nEach asset instance contains two values: `info`, which specifies the asset's type (CW20 or native), and its `amount`, which specifies the asset's amount.", - "type": "object", - "required": [ - "amount", - "info" - ], - "properties": { - "amount": { - "description": "Specifies the asset's amount", - "allOf": [ - { - "$ref": "#/definitions/Uint128" - } - ] - }, - "info": { - "description": "Specifies the asset's type (CW20 or native)", - "allOf": [ - { - "$ref": "#/definitions/AssetInfoBase_for_Addr" - } - ] - } - }, - "additionalProperties": false - }, - "AssetInfoBase_for_Addr": { - "description": "Represents the type of an fungible asset.\n\nEach **asset info** instance can be one of three variants:\n\n- Native SDK coins. To create an **asset info** instance of this type, provide the denomination. - CW20 tokens. To create an **asset info** instance of this type, provide the contract address.", - "oneOf": [ - { - "type": "object", - "required": [ - "native" - ], - "properties": { - "native": { - "type": "string" - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "cw20" - ], - "properties": { - "cw20": { - "$ref": "#/definitions/Addr" - } - }, - "additionalProperties": false - } - ] - }, - "DistributionSchedule": { - "type": "object", - "required": [ - "amount", - "end", - "start" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - }, - "end": { - "type": "integer", - "format": "uint64", - "minimum": 0.0 - }, - "start": { - "type": "integer", - "format": "uint64", - "minimum": 0.0 - } - }, - "additionalProperties": false - }, - "DistributionScheduleOptions": { - "type": "object", - "properties": { - "amount": { - "anyOf": [ - { - "$ref": "#/definitions/Uint128" - }, - { - "type": "null" - } - ] - }, - "end": { - "type": [ - "integer", - "null" - ], - "format": "uint64", - "minimum": 0.0 - }, - "start": { - "type": [ - "integer", - "null" - ], - "format": "uint64", - "minimum": 0.0 - } - }, - "additionalProperties": false - }, - "Uint128": { - "description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```", - "type": "string" - }, - "VaultExecuteMsg": { - "oneOf": [ - { - "type": "object", - "required": [ - "update_user_reward_index" - ], - "properties": { - "update_user_reward_index": { - "type": "string" - } - }, - "additionalProperties": false - } - ] - } - } -} diff --git a/smart-contracts/osmosis/contracts/vault-rewards/schema/instantiate_msg.json b/smart-contracts/osmosis/contracts/vault-rewards/schema/instantiate_msg.json deleted file mode 100644 index b1ae7f34f..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/schema/instantiate_msg.json +++ /dev/null @@ -1,85 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "InstantiateMsg", - "type": "object", - "required": [ - "distribution_schedule", - "reward_token", - "vault_token" - ], - "properties": { - "distribution_schedule": { - "$ref": "#/definitions/DistributionSchedule" - }, - "reward_token": { - "$ref": "#/definitions/AssetInfoBase_for_Addr" - }, - "vault_token": { - "type": "string" - } - }, - "additionalProperties": false, - "definitions": { - "Addr": { - "description": "A human readable address.\n\nIn Cosmos, this is typically bech32 encoded. But for multi-chain smart contracts no assumptions should be made other than being UTF-8 encoded and of reasonable length.\n\nThis type represents a validated address. It can be created in the following ways 1. Use `Addr::unchecked(input)` 2. Use `let checked: Addr = deps.api.addr_validate(input)?` 3. Use `let checked: Addr = deps.api.addr_humanize(canonical_addr)?` 4. Deserialize from JSON. This must only be done from JSON that was validated before such as a contract's state. `Addr` must not be used in messages sent by the user because this would result in unvalidated instances.\n\nThis type is immutable. If you really need to mutate it (Really? Are you sure?), create a mutable copy using `let mut mutable = Addr::to_string()` and operate on that `String` instance.", - "type": "string" - }, - "AssetInfoBase_for_Addr": { - "description": "Represents the type of an fungible asset.\n\nEach **asset info** instance can be one of three variants:\n\n- Native SDK coins. To create an **asset info** instance of this type, provide the denomination. - CW20 tokens. To create an **asset info** instance of this type, provide the contract address.", - "oneOf": [ - { - "type": "object", - "required": [ - "native" - ], - "properties": { - "native": { - "type": "string" - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "cw20" - ], - "properties": { - "cw20": { - "$ref": "#/definitions/Addr" - } - }, - "additionalProperties": false - } - ] - }, - "DistributionSchedule": { - "type": "object", - "required": [ - "amount", - "end", - "start" - ], - "properties": { - "amount": { - "$ref": "#/definitions/Uint128" - }, - "end": { - "type": "integer", - "format": "uint64", - "minimum": 0.0 - }, - "start": { - "type": "integer", - "format": "uint64", - "minimum": 0.0 - } - }, - "additionalProperties": false - }, - "Uint128": { - "description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```", - "type": "string" - } - } -} diff --git a/smart-contracts/osmosis/contracts/vault-rewards/schema/query_msg.json b/smart-contracts/osmosis/contracts/vault-rewards/schema/query_msg.json deleted file mode 100644 index c570a2a41..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/schema/query_msg.json +++ /dev/null @@ -1,31 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - "title": "QueryMsg", - "oneOf": [ - { - "type": "object", - "required": [ - "config" - ], - "properties": { - "config": { - "type": "object", - "additionalProperties": false - } - }, - "additionalProperties": false - }, - { - "type": "object", - "required": [ - "pending_rewards" - ], - "properties": { - "pending_rewards": { - "type": "string" - } - }, - "additionalProperties": false - } - ] -} diff --git a/smart-contracts/osmosis/contracts/vault-rewards/src/contract.rs b/smart-contracts/osmosis/contracts/vault-rewards/src/contract.rs deleted file mode 100644 index 44cc78d77..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/src/contract.rs +++ /dev/null @@ -1,96 +0,0 @@ -use crate::error::VaultRewardsError; -use crate::execute::admin::{ - execute_add_distribution_schedule, execute_remove_distribution_schedule, - execute_update_distribution_schedule, execute_withdraw_funds, -}; -use crate::execute::user::execute_claim; -use crate::execute::vault::execute_update_user_reward_index; -use crate::helpers::is_contract_admin; -use crate::msg::{ - AdminExecuteMsg, ExecuteMsg, InstantiateMsg, MigrateMsg, QueryMsg, VaultExecuteMsg, -}; -use crate::query::{query_config, query_pending_rewards, query_user_rewards_index}; -use crate::state::{Config, CONFIG}; -#[cfg(not(feature = "library"))] -use cosmwasm_std::entry_point; -use cosmwasm_std::{to_json_binary, Binary, Deps, DepsMut, Env, MessageInfo, Response, Uint128}; - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn instantiate( - deps: DepsMut, - env: Env, - _info: MessageInfo, - msg: InstantiateMsg, -) -> Result { - let mut config = Config { - vault_token: deps.api.addr_validate(&msg.vault_token)?, - reward_token: msg.reward_token, - distribution_schedules: vec![], - total_claimed: Uint128::zero(), - }; - config.add_distribution_schedules(&deps.querier, &env, msg.distribution_schedules)?; - CONFIG.save(deps.storage, &config)?; - Ok(Response::default()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn execute( - deps: DepsMut, - env: Env, - info: MessageInfo, - msg: ExecuteMsg, -) -> Result { - match msg { - ExecuteMsg::Claim {} => execute_claim(deps, &env, info.sender), - ExecuteMsg::Admin(admin_msg) => { - is_contract_admin(&deps.querier, &env, &info.sender)?; - match admin_msg { - AdminExecuteMsg::WithdrawFunds(asset) => { - execute_withdraw_funds(deps, env, info.sender, asset) - } - AdminExecuteMsg::AddDistributionSchedule(schedule) => { - execute_add_distribution_schedule(deps, env, schedule) - } - AdminExecuteMsg::UpdateDistributionSchedule { id, update } => { - execute_update_distribution_schedule(deps, env, id, update) - } - AdminExecuteMsg::RemoveDistributionSchedule(id) => { - execute_remove_distribution_schedule(deps, env, id) - } - } - } - ExecuteMsg::Vault(vault_msg) => { - let vault_token = CONFIG.load(deps.storage)?.vault_token; - if info.sender != vault_token { - return Err(VaultRewardsError::Unauthorized {}); - } - match vault_msg { - VaultExecuteMsg::UpdateUserRewardIndex(user) => { - let user = deps.api.addr_validate(&user)?; - execute_update_user_reward_index(deps, env, user) - } - } - } - } -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn query(deps: Deps, env: Env, msg: QueryMsg) -> Result { - match msg { - QueryMsg::Config {} => to_json_binary(&query_config(deps, env)?), - QueryMsg::PendingRewards(user) => { - let user = deps.api.addr_validate(&user)?; - to_json_binary(&query_pending_rewards(deps, env, user)?) - } - QueryMsg::GetUserRewardsIndex(user) => { - let user = deps.api.addr_validate(&user)?; - to_json_binary(&query_user_rewards_index(deps, user)?) - } - } - .map_err(|e| e.into()) -} - -#[cfg_attr(not(feature = "library"), entry_point)] -pub fn migrate(_deps: DepsMut, _env: Env, _msg: MigrateMsg) -> Result { - Ok(Response::new().add_attribute("success", "true")) -} diff --git a/smart-contracts/osmosis/contracts/vault-rewards/src/error.rs b/smart-contracts/osmosis/contracts/vault-rewards/src/error.rs deleted file mode 100644 index 6bf82452f..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/src/error.rs +++ /dev/null @@ -1,41 +0,0 @@ -use cosmwasm_std::{StdError, Uint128}; -use cw_asset::AssetError; -use thiserror::Error; - -#[derive(Error, Debug, PartialEq)] -pub enum VaultRewardsError { - #[error("{0}")] - Std(#[from] StdError), - - #[error("{0}")] - Asset(#[from] AssetError), - - #[error("Unauthorized")] - Unauthorized {}, - - #[error("Invalid distribution schedule: {reason:?}")] - InvalidDistributionSchedule { reason: String }, - - #[error("No rewards to claim")] - NoRewardsToClaim {}, - - #[error("Insufficient funds in contract to process claim. Contract balance: {contract_balance:?}, Claim amount: {claim_amount:?}")] - InsufficientFunds { - contract_balance: Uint128, - claim_amount: Uint128, - }, - - #[error("Invalid distribution schedule id. Max ID: {max_id:?}")] - InvalidDistributionScheduleId { max_id: u64 }, - - #[error( - "Cannot edit distribution schedule in progress. ID: {id:?}, Start: {start:?}, End: {end:?}" - )] - DistributionScheduleInProgress { id: u64, start: u64, end: u64 }, - - #[error("Cannot remove distribution schedule with funds left to be claimed. ID: {id:?}")] - DistributionScheduleWithUnclaimedFunds { id: u64 }, - - #[error("Cannot edit distribution schedule that has already ended. ID: {id:?}, End: {end:?}")] - DistributionScheduleExpired { id: u64, end: u64 }, -} diff --git a/smart-contracts/osmosis/contracts/vault-rewards/src/execute/admin.rs b/smart-contracts/osmosis/contracts/vault-rewards/src/execute/admin.rs deleted file mode 100644 index e2431fc36..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/src/execute/admin.rs +++ /dev/null @@ -1,111 +0,0 @@ -use crate::msg::DistributionScheduleOptions; -use crate::state::{DistributionSchedule, CONFIG}; -use crate::VaultRewardsError; -use cosmwasm_std::{attr, Addr, DepsMut, Env, Response, Uint128}; -use cw_asset::Asset; - -pub fn execute_withdraw_funds( - deps: DepsMut, - env: Env, - admin: Addr, - mut asset: Asset, -) -> Result { - let config = CONFIG.load(deps.storage)?; - - let reward_token = &config.reward_token; - - if &asset.info == reward_token { - // check if reward balance is sufficient after withdrawal - let contract_reward_balance = config - .reward_token - .query_balance(&deps.querier, &env.contract.address)? - .checked_sub(asset.amount) - .unwrap_or_default(); - let total_reward_needed_to_distribute = config.get_total_distribution_amount(); - if contract_reward_balance < total_reward_needed_to_distribute { - return Err(VaultRewardsError::InsufficientFunds { - contract_balance: contract_reward_balance, - claim_amount: total_reward_needed_to_distribute, - }); - } - } else { - // send to admin if balance > 0 - asset.amount = asset.amount.min( - asset - .info - .query_balance(&deps.querier, &env.contract.address)?, - ); - }; - - if asset.amount.is_zero() { - return Err(VaultRewardsError::InsufficientFunds { - contract_balance: Uint128::zero(), - claim_amount: Uint128::zero(), - }); - } - let transfer = asset.transfer_msg(&admin)?; - - Ok(Response::new().add_message(transfer).add_attributes(vec![ - ("action", "withdraw_funds"), - ("asset", &asset.to_string()), - ("admin", admin.as_ref()), - ])) -} - -pub fn execute_add_distribution_schedule( - deps: DepsMut, - env: Env, - schedule: DistributionSchedule, -) -> Result { - let mut config = CONFIG.load(deps.storage)?; - config.add_distribution_schedule(&deps.querier, &env, schedule.clone())?; - CONFIG.save(deps.storage, &config)?; - Ok(Response::default().add_attributes(vec![ - ("action", "add_distribution_schedule"), - ("schedule start", &schedule.start.to_string()), - ("schedule end", &schedule.end.to_string()), - ("schedule amount", &schedule.amount.to_string()), - ])) -} - -pub fn execute_update_distribution_schedule( - deps: DepsMut, - env: Env, - id: u64, - update: DistributionScheduleOptions, -) -> Result { - let mut config = CONFIG.load(deps.storage)?; - config.update_distribution_schedule(&deps.querier, &env, id, &update)?; - CONFIG.save(deps.storage, &config)?; - let mut attrs = vec![ - attr("action", "update_distribution_schedule"), - attr("id", id.to_string()), - ]; - if let Some(start) = &update.start { - attrs.push(attr("schedule_start_updated_to", start.to_string())); - } - if let Some(end) = &update.end { - attrs.push(attr("schedule_end_updated_to", end.to_string())); - } - if let Some(amount) = &update.amount { - attrs.push(attr("schedule_amount_updated_to", amount.to_string())); - } - Ok(Response::default().add_attributes(attrs)) -} - -pub fn execute_remove_distribution_schedule( - deps: DepsMut, - env: Env, - id: u64, -) -> Result { - let mut config = CONFIG.load(deps.storage)?; - config.remove_distribution_schedule(deps.storage, &env, id)?; - CONFIG.save(deps.storage, &config)?; - Ok(Response::default().add_attributes(vec![ - ("action", "remove_distribution_schedule"), - ("id", &id.to_string()), - ])) -} - -#[cfg(test)] -mod tests {} diff --git a/smart-contracts/osmosis/contracts/vault-rewards/src/execute/mock_querier.rs b/smart-contracts/osmosis/contracts/vault-rewards/src/execute/mock_querier.rs deleted file mode 100644 index bfa4d6b11..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/src/execute/mock_querier.rs +++ /dev/null @@ -1,104 +0,0 @@ -use cosmwasm_std::testing::{MockApi, MockQuerier, MockStorage, MOCK_CONTRACT_ADDR}; -use cosmwasm_std::{ - from_json, to_json_binary, Coin, ContractResult, Empty, OwnedDeps, Querier, QuerierResult, - QueryRequest, SystemError, SystemResult, Uint128, WasmQuery, -}; -use cw20::{BalanceResponse, Cw20QueryMsg, TokenInfoResponse}; -use std::collections::HashMap; - -pub fn mock_dependencies( - contract_balance: &[Coin], -) -> OwnedDeps { - let custom_querier: WasmMockQuerier = - WasmMockQuerier::new(MockQuerier::new(&[(MOCK_CONTRACT_ADDR, contract_balance)])); - - OwnedDeps { - storage: MockStorage::default(), - api: MockApi::default(), - querier: custom_querier, - custom_query_type: Default::default(), - } -} - -pub struct WasmMockQuerier { - base: MockQuerier, - token_querier: TokenQuerier, -} - -impl Querier for WasmMockQuerier { - fn raw_query(&self, bin_request: &[u8]) -> QuerierResult { - let request: QueryRequest = match from_json(bin_request) { - Ok(v) => v, - Err(e) => { - return SystemResult::Err(SystemError::InvalidRequest { - error: format!("Parsing query request: {e:?}"), - request: bin_request.into(), - }) - } - }; - self.handle_query(&request) - } -} - -#[derive(Clone, Default)] -pub struct TokenQuerier { - balance: HashMap, - supply: Uint128, -} - -impl WasmMockQuerier { - pub fn handle_query(&self, request: &QueryRequest) -> QuerierResult { - match &request { - QueryRequest::Wasm(WasmQuery::Smart { - contract_addr: _, - msg, - }) => match from_json(msg).unwrap() { - Cw20QueryMsg::TokenInfo {} => QuerierResult::Ok(ContractResult::Ok( - to_json_binary(&TokenInfoResponse { - total_supply: self.token_querier.supply, - name: "vault_token".to_string(), - symbol: "".to_string(), - decimals: 0, - }) - .unwrap(), - )), - Cw20QueryMsg::Balance { address } => QuerierResult::Ok(ContractResult::Ok( - to_json_binary(&BalanceResponse { - balance: match self.token_querier.balance.get(&address) { - Some(balance) => *balance, - None => Uint128::zero(), - }, - }) - .unwrap(), - )), - _ => SystemResult::Err(SystemError::UnsupportedRequest { - kind: "unimplemented".to_string(), - }), - }, - _ => self.base.handle_query(request), - } - } -} - -impl WasmMockQuerier { - pub fn new(base: MockQuerier) -> Self { - WasmMockQuerier { - base, - token_querier: TokenQuerier::default(), - } - } - - pub fn with_token_balance(&mut self, address: &str, balance: &Uint128) { - if let Some(prev_balance) = self.token_querier.balance.get(address) { - self.token_querier.supply -= prev_balance; - } - self.token_querier.supply += balance; - self.token_querier - .balance - .insert(address.to_string(), *balance); - } - - pub fn with_bank_balance(&mut self, address: &str, balance: Vec) { - self.base.update_balance(address, balance); - } -} diff --git a/smart-contracts/osmosis/contracts/vault-rewards/src/execute/mod.rs b/smart-contracts/osmosis/contracts/vault-rewards/src/execute/mod.rs deleted file mode 100644 index 9c9b05d84..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/src/execute/mod.rs +++ /dev/null @@ -1,6 +0,0 @@ -pub mod admin; -pub mod user; -pub mod vault; - -#[cfg(test)] -mod mock_querier; diff --git a/smart-contracts/osmosis/contracts/vault-rewards/src/execute/user.rs b/smart-contracts/osmosis/contracts/vault-rewards/src/execute/user.rs deleted file mode 100644 index 355d9ae2d..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/src/execute/user.rs +++ /dev/null @@ -1,505 +0,0 @@ -use crate::helpers::{get_user_reward_index, update_reward_index}; -use crate::state::{ - Config, DistributionSchedule, RewardIndex, UserBalance, UserRewardIndex, CONFIG, REWARD_INDEX, - USER_REWARD_INDEX, -}; -use crate::VaultRewardsError; -use cosmwasm_std::{Addr, Deps, DepsMut, Env, Order, Response, StdResult, Uint128}; -use cw20::Cw20Contract; -use cw_asset::{Asset, AssetInfo}; -use cw_storage_plus::Bound; - -pub fn execute_claim(deps: DepsMut, env: &Env, user: Addr) -> Result { - let mut config = CONFIG.load(deps.storage)?; - let cur_block_height = env.block.height; - let mut user_reward_index = get_user_reward_index(deps.storage, &user); - let user_vault_token_balance = - AssetInfo::cw20(config.vault_token.clone()).query_balance(&deps.querier, &user)?; - // if previous balance, move to history and start new balance index - if let Some(prev_balance) = user_reward_index.balance { - user_reward_index.history.push(DistributionSchedule { - start: prev_balance.reward_index, - end: cur_block_height, - amount: prev_balance.balance, - }); - user_reward_index.balance = if !user_vault_token_balance.is_zero() { - Some(UserBalance { - reward_index: cur_block_height + 1, - balance: user_vault_token_balance, - }) - } else { - None - }; - } - // update global reward index before calculating user claim amount - update_reward_index(deps.storage, &deps.querier, env)?; - let claim_amount = get_claim_amount(deps.as_ref(), env, &config, &user_reward_index)?; - - // double check we have enough balance to cover this - let contract_reward_token_balance = config - .reward_token - .query_balance(&deps.querier, &env.contract.address)?; - if contract_reward_token_balance < claim_amount { - return Err(VaultRewardsError::InsufficientFunds { - contract_balance: contract_reward_token_balance, - claim_amount, - }); - } - - let claim = Asset::new(config.reward_token.clone(), claim_amount).transfer_msg(&user)?; - user_reward_index.history = vec![]; - USER_REWARD_INDEX.save(deps.storage, user.clone(), &user_reward_index)?; - config.total_claimed += claim_amount; - CONFIG.save(deps.storage, &config)?; - Ok(Response::new().add_message(claim).add_attributes(vec![ - ("action", "claim"), - ("user", user.as_ref()), - ("amount", &claim_amount.to_string()), - ])) -} - -pub fn get_claim_amount( - deps: Deps, - env: &Env, - config: &Config, - user_reward_index: &UserRewardIndex, -) -> Result { - let mut user_reward_index_history = user_reward_index.history.clone(); - // if we don't have the index history until current block height - if (user_reward_index_history.is_empty() - || user_reward_index_history.last().unwrap().end < env.block.height) - && user_reward_index.balance.is_some() - { - user_reward_index_history.push(DistributionSchedule { - start: user_reward_index.balance.clone().unwrap().reward_index + 1, - end: env.block.height, - amount: user_reward_index.balance.clone().unwrap().balance, - }); - } - - let vault_supply = Cw20Contract(CONFIG.load(deps.storage)?.vault_token) - .meta(&deps.querier)? - .total_supply; - - let mut claim_amount = user_reward_index_history - .iter() - .map(|d| { - let mut cur_height = d.start; - let mut reward_indexes = REWARD_INDEX - .range( - deps.storage, - Some(Bound::inclusive(d.start - 1)), - Some(Bound::inclusive(d.end)), - Order::Ascending, - ) - .collect::>>() - .unwrap(); - - if let Some(value) = reward_indexes.last() { - if value.0 != env.block.height && d.end == env.block.height { - reward_indexes.push((env.block.height, RewardIndex { vault_supply })); - } - } - // iterate over reward indexes 2 at a time to calculate reward for each period - reward_indexes - .iter() - .zip(reward_indexes.iter().skip(1)) - .map(|(start, end)| { - let (_, reward_index_start) = start; - let (height_end, _) = end; - let mut period_claim_amount = Uint128::zero(); - while cur_height <= *height_end { - let block_reward = config.get_distribution_rate_at_height(cur_height); - // calculate reward for user based on their share of the vault supply - period_claim_amount += - block_reward * d.amount / reward_index_start.vault_supply; - cur_height += 1; - } - period_claim_amount - }) - .sum::() - }) - .sum::(); - // this accounts for edge case where final user withdraws their claim (ends up being ~1% less than expected due to rounding) - claim_amount = claim_amount.min(config.get_total_distribution_amount() - config.total_claimed); - if claim_amount.is_zero() { - return Err(VaultRewardsError::NoRewardsToClaim {}); - } - Ok(claim_amount) -} - -#[cfg(test)] -mod tests { - use crate::execute::mock_querier::{mock_dependencies, WasmMockQuerier}; - use crate::execute::user::{execute_claim, get_claim_amount}; - use crate::execute::vault::execute_update_user_reward_index; - use crate::helpers::get_user_reward_index; - use crate::state::{Config, DistributionSchedule, CONFIG}; - use crate::VaultRewardsError; - use cosmwasm_std::testing::{mock_env, MockApi, MockStorage, MOCK_CONTRACT_ADDR}; - use cosmwasm_std::{attr, Addr, Coin, Env, OwnedDeps, Uint128}; - use cw_asset::AssetInfo; - - #[test] - fn test_execute_claim() { - let mut deps = mock_dependencies(&[]); - let mut env = mock_env(); - let mut total_claim_amount = Uint128::zero(); - let config = Config { - vault_token: Addr::unchecked("vault_token"), - reward_token: AssetInfo::native("reward_token"), - distribution_schedules: vec![ - DistributionSchedule { - start: 100, - end: 1000, - amount: Uint128::new(900000000), - }, - DistributionSchedule { - start: 500, - end: 1500, - amount: Uint128::new(1000000000), - }, - ], - total_claimed: Uint128::zero(), - }; - env.block.height = 1; - CONFIG.save(deps.as_mut().storage, &config).unwrap(); - - let user1 = Addr::unchecked("user1"); - let user2 = Addr::unchecked("user2"); - let user3 = Addr::unchecked("user3"); - let user4 = Addr::unchecked("user4"); - let user5 = Addr::unchecked("user5"); - - deps.querier - .with_token_balance(user1.as_ref(), &Uint128::new(100)); - execute_update_user_reward_index(deps.as_mut(), env.clone(), user1.clone()).unwrap(); - - let res = execute_claim(deps.as_mut(), &env, user1.clone()); - assert!(res.is_err()); - assert_eq!(res.err().unwrap(), VaultRewardsError::NoRewardsToClaim {}); - - env.block.height = 100; - - // should error since no funds in contract (shouldn't happen tho since it's checked for when adding/updating distribution schedules) - let res = execute_claim(deps.as_mut(), &env, user1.clone()); - assert!(res.is_err()); - assert_eq!( - res.err().unwrap(), - VaultRewardsError::InsufficientFunds { - contract_balance: Uint128::zero(), - claim_amount: Uint128::new(1000000) - } - ); - - let mut contract_reward_balance = config - .distribution_schedules - .iter() - .fold(Uint128::zero(), |acc, s| acc + s.amount); - - deps.querier.with_bank_balance( - MOCK_CONTRACT_ADDR, - vec![Coin { - denom: "reward_token".to_string(), - amount: contract_reward_balance, - }], - ); - - execute_claim_helper( - &mut deps, - &env, - &user1, - Uint128::new(1000000), - &mut total_claim_amount, - &mut contract_reward_balance, - ); - - // claim in same block should error - let res = execute_claim(deps.as_mut(), &env, user1.clone()); - assert!(res.is_err()); - assert_eq!(res.err().unwrap(), VaultRewardsError::NoRewardsToClaim {}); - - env.block.height = 200; - - execute_claim_helper( - &mut deps, - &env, - &user1, - Uint128::new(100000000), - &mut total_claim_amount, - &mut contract_reward_balance, - ); - - env.block.height = 251; - - deps.querier - .with_token_balance(user2.as_ref(), &Uint128::new(100)); - execute_update_user_reward_index(deps.as_mut(), env.clone(), user2.clone()).unwrap(); - - env.block.height = 300; - - // user1 should have same rate until block 250, then shares reward with user2 - execute_claim_helper( - &mut deps, - &env, - &user1, - Uint128::new(75500000), - &mut total_claim_amount, - &mut contract_reward_balance, - ); - execute_claim_helper( - &mut deps, - &env, - &user2, - Uint128::new(25000000), - &mut total_claim_amount, - &mut contract_reward_balance, - ); - - env.block.height = 1000; - - // transfer balance from user1 to user2 - deps.querier - .with_token_balance(user1.as_ref(), &Uint128::zero()); - deps.querier - .with_token_balance(user2.as_ref(), &Uint128::new(200)); - execute_update_user_reward_index(deps.as_mut(), env.clone(), user1.clone()).unwrap(); - execute_update_user_reward_index(deps.as_mut(), env.clone(), user2.clone()).unwrap(); - - execute_claim_helper( - &mut deps, - &env, - &user1, - Uint128::new(600000000), - &mut total_claim_amount, - &mut contract_reward_balance, - ); - execute_claim_helper( - &mut deps, - &env, - &user2, - Uint128::new(600000000), - &mut total_claim_amount, - &mut contract_reward_balance, - ); - - env.block.height = 1250; - - deps.querier - .with_token_balance(user3.as_ref(), &Uint128::new(200)); - deps.querier - .with_token_balance(user4.as_ref(), &Uint128::new(400)); - deps.querier - .with_token_balance(user5.as_ref(), &Uint128::new(800)); - execute_update_user_reward_index(deps.as_mut(), env.clone(), user3.clone()).unwrap(); - execute_update_user_reward_index(deps.as_mut(), env.clone(), user4.clone()).unwrap(); - execute_update_user_reward_index(deps.as_mut(), env.clone(), user5.clone()).unwrap(); - - env.block.height = 2000; - - let res = execute_claim(deps.as_mut(), &env, user1.clone()); - assert!(res.is_err()); - assert_eq!(res.err().unwrap(), VaultRewardsError::NoRewardsToClaim {}); - execute_claim_helper( - &mut deps, - &env, - &user2, - Uint128::new(281125000), - &mut total_claim_amount, - &mut contract_reward_balance, - ); - execute_claim_helper( - &mut deps, - &env, - &user3, - Uint128::new(31250000), - &mut total_claim_amount, - &mut contract_reward_balance, - ); - execute_claim_helper( - &mut deps, - &env, - &user4, - Uint128::new(62500000), - &mut total_claim_amount, - &mut contract_reward_balance, - ); - execute_claim_helper( - &mut deps, - &env, - &user5, - Uint128::new(123625000), - &mut total_claim_amount, - &mut contract_reward_balance, - ); - - env.block.height = 3000; - - let res = execute_claim(deps.as_mut(), &env, user1.clone()); - assert!(res.is_err()); - assert_eq!(res.err().unwrap(), VaultRewardsError::NoRewardsToClaim {}); - let res = execute_claim(deps.as_mut(), &env, user2.clone()); - assert!(res.is_err()); - assert_eq!(res.err().unwrap(), VaultRewardsError::NoRewardsToClaim {}); - let res = execute_claim(deps.as_mut(), &env, user3.clone()); - assert!(res.is_err()); - assert_eq!(res.err().unwrap(), VaultRewardsError::NoRewardsToClaim {}); - let res = execute_claim(deps.as_mut(), &env, user4.clone()); - assert!(res.is_err()); - assert_eq!(res.err().unwrap(), VaultRewardsError::NoRewardsToClaim {}); - let res = execute_claim(deps.as_mut(), &env, user5.clone()); - assert!(res.is_err()); - assert_eq!(res.err().unwrap(), VaultRewardsError::NoRewardsToClaim {}); - - let expected_claim_amount = config - .distribution_schedules - .iter() - .fold(Uint128::zero(), |acc, s| acc + s.amount); - assert_eq!(total_claim_amount, expected_claim_amount); - } - - fn execute_claim_helper( - deps: &mut OwnedDeps, - env: &Env, - user: &Addr, - expected_claim_amount: Uint128, - total_claim_amount: &mut Uint128, - contract_reward_balance: &mut Uint128, - ) { - let res = execute_claim(deps.as_mut(), env, user.clone()); - if res.is_err() { - println!("res: {res:?}"); - } - assert!(res.is_ok()); - let res = res.unwrap(); - let claim_amount = res - .attributes - .iter() - .find(|a| a.key == "amount") - .unwrap() - .value - .parse::() - .unwrap(); - *total_claim_amount += claim_amount; - *contract_reward_balance -= claim_amount; - deps.querier.with_bank_balance( - MOCK_CONTRACT_ADDR, - vec![Coin { - denom: "reward_token".to_string(), - amount: *contract_reward_balance, - }], - ); - assert_eq!( - res.attributes, - vec![ - attr("action", "claim"), - attr("user", user.to_string()), - attr("amount", expected_claim_amount.to_string()), - ] - ); - } - - #[test] - fn execute_get_claim_amount() { - let mut deps = mock_dependencies(&[]); - let mut env = mock_env(); - let config = Config { - vault_token: Addr::unchecked("vault_token"), - reward_token: AssetInfo::native("reward_token"), - distribution_schedules: vec![DistributionSchedule { - start: 2, - end: 14, - amount: Uint128::new(900000000), - }], - total_claimed: Uint128::zero(), - }; - env.block.height = 1; - CONFIG.save(deps.as_mut().storage, &config).unwrap(); - - let user1 = Addr::unchecked("user1"); - let user2 = Addr::unchecked("user2"); - - deps.querier - .with_token_balance(user1.as_ref(), &Uint128::new(100)); - deps.querier - .with_token_balance(user2.as_ref(), &Uint128::new(200)); - - let user_reward_index = get_user_reward_index(&deps.storage, &user1); - let res = get_claim_amount(deps.as_ref(), &env, &config, &user_reward_index); - - assert!(res.is_err()); - assert_eq!(res.err().unwrap(), VaultRewardsError::NoRewardsToClaim {}); - - let _res = execute_update_user_reward_index(deps.as_mut(), env.clone(), user1.clone()); - let res = execute_update_user_reward_index(deps.as_mut(), env.clone(), user2.clone()); - assert!(res.is_ok()); - - env.block.height += 10; - - let _res = execute_update_user_reward_index(deps.as_mut(), env.clone(), user1.clone()); - - env.block.height += 10; - - let _res = execute_update_user_reward_index(deps.as_mut(), env.clone(), user2.clone()); - - let user1_reward_index = get_user_reward_index(&deps.storage, &user1); - let user2_reward_index = get_user_reward_index(&deps.storage, &user2); - let res1 = get_claim_amount(deps.as_ref(), &env, &config, &user1_reward_index); - let res2 = get_claim_amount(deps.as_ref(), &env, &config, &user2_reward_index); - - assert!(res1.is_ok()); - assert!(res2.is_ok()); - assert_eq!(res1.unwrap(), Uint128::new(300000000)); // user1 holds 1/3 of the vaulth - assert_eq!(res2.unwrap(), Uint128::new(600000000)); // user2 holds 2/3 of the vaulth - } - - #[test] - fn execute_get_claim_amount_without_distribution_schedule() { - let mut deps = mock_dependencies(&[]); - let mut env = mock_env(); - let config = Config { - vault_token: Addr::unchecked("vault_token"), - reward_token: AssetInfo::native("reward_token"), - distribution_schedules: vec![], - total_claimed: Uint128::zero(), - }; - env.block.height = 1; - CONFIG.save(deps.as_mut().storage, &config).unwrap(); - - let user1 = Addr::unchecked("user1"); - let user2 = Addr::unchecked("user2"); - - deps.querier - .with_token_balance(user1.as_ref(), &Uint128::new(100)); - deps.querier - .with_token_balance(user2.as_ref(), &Uint128::new(200)); - - let user_reward_index = get_user_reward_index(&deps.storage, &user1); - let res = get_claim_amount(deps.as_ref(), &env, &config, &user_reward_index); - - assert!(res.is_err()); - assert_eq!(res.err().unwrap(), VaultRewardsError::NoRewardsToClaim {}); - - let _res = execute_update_user_reward_index(deps.as_mut(), env.clone(), user1.clone()); - let res = execute_update_user_reward_index(deps.as_mut(), env.clone(), user2.clone()); - assert!(res.is_ok()); - - env.block.height += 10; - - let _res = execute_update_user_reward_index(deps.as_mut(), env.clone(), user1.clone()); - - env.block.height += 10; - - let _res = execute_update_user_reward_index(deps.as_mut(), env.clone(), user2.clone()); - - let user1_reward_index = get_user_reward_index(&deps.storage, &user1); - let user2_reward_index = get_user_reward_index(&deps.storage, &user2); - let res1 = get_claim_amount(deps.as_ref(), &env, &config, &user1_reward_index); - let res2 = get_claim_amount(deps.as_ref(), &env, &config, &user2_reward_index); - - assert!(res1.is_err()); - assert_eq!(res1.err().unwrap(), VaultRewardsError::NoRewardsToClaim {}); - assert!(res2.is_err()); - assert_eq!(res2.err().unwrap(), VaultRewardsError::NoRewardsToClaim {}); - } -} diff --git a/smart-contracts/osmosis/contracts/vault-rewards/src/execute/vault.rs b/smart-contracts/osmosis/contracts/vault-rewards/src/execute/vault.rs deleted file mode 100644 index a4f980a49..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/src/execute/vault.rs +++ /dev/null @@ -1,42 +0,0 @@ -use crate::helpers::{get_user_reward_index, update_reward_index}; -use crate::state::{DistributionSchedule, UserBalance, CONFIG, USER_REWARD_INDEX}; -use crate::VaultRewardsError; -use cosmwasm_std::{Addr, DepsMut, Env, Response}; -use cw_asset::AssetInfo; - -pub fn execute_update_user_reward_index( - deps: DepsMut, - env: Env, - user: Addr, -) -> Result { - let cur_block_height = env.block.height; - let user_vault_token_balance = AssetInfo::cw20(CONFIG.load(deps.storage)?.vault_token) - .query_balance(&deps.querier, &user)?; - let mut user_reward_index = get_user_reward_index(deps.storage, &user); - // if previous balance, then move to history and record new balance - if let Some(prev_balance) = user_reward_index.balance { - user_reward_index.history.push(DistributionSchedule { - start: prev_balance.reward_index, - end: cur_block_height, - amount: prev_balance.balance, - }) - } - user_reward_index.balance = if !user_vault_token_balance.is_zero() { - Some(UserBalance { - reward_index: cur_block_height, - balance: user_vault_token_balance, - }) - } else { - None - }; - USER_REWARD_INDEX.save(deps.storage, user.clone(), &user_reward_index)?; - update_reward_index(deps.storage, &deps.querier, &env)?; - Ok(Response::default().add_attributes(vec![ - ("action", "update_user_index"), - ("user", user.as_ref()), - ("vault_token_balance", &user_vault_token_balance.to_string()), - ])) -} - -#[cfg(test)] -mod tests {} diff --git a/smart-contracts/osmosis/contracts/vault-rewards/src/helpers.rs b/smart-contracts/osmosis/contracts/vault-rewards/src/helpers.rs deleted file mode 100644 index 29210707d..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/src/helpers.rs +++ /dev/null @@ -1,47 +0,0 @@ -use crate::state::{RewardIndex, UserRewardIndex, CONFIG, REWARD_INDEX, USER_REWARD_INDEX}; -use crate::VaultRewardsError; -use cosmwasm_std::{Addr, Env, QuerierWrapper, Storage}; -use cw20::Cw20Contract; - -pub fn update_reward_index( - storage: &mut dyn Storage, - querier: &QuerierWrapper, - env: &Env, -) -> Result { - let cur_block_height = env.block.height; - let mut reward_index = REWARD_INDEX - .load(storage, cur_block_height) - .unwrap_or_default(); - reward_index.vault_supply = Cw20Contract(CONFIG.load(storage)?.vault_token) - .meta(querier)? - .total_supply; - REWARD_INDEX.save(storage, cur_block_height, &reward_index)?; - Ok(reward_index) -} - -pub fn get_user_reward_index(storage: &dyn Storage, user: &Addr) -> UserRewardIndex { - USER_REWARD_INDEX - .load(storage, user.clone()) - .unwrap_or_else(|_| UserRewardIndex { - balance: None, - history: vec![], - }) -} - -pub fn is_contract_admin( - querier: &QuerierWrapper, - env: &Env, - sus_admin: &Addr, -) -> Result<(), VaultRewardsError> { - let contract_admin = querier - .query_wasm_contract_info(&env.contract.address)? - .admin; - if let Some(contract_admin) = contract_admin { - if contract_admin != *sus_admin { - return Err(VaultRewardsError::Unauthorized {}); - } - } else { - return Err(VaultRewardsError::Unauthorized {}); - } - Ok(()) -} diff --git a/smart-contracts/osmosis/contracts/vault-rewards/src/lib.rs b/smart-contracts/osmosis/contracts/vault-rewards/src/lib.rs deleted file mode 100644 index 6e19e0e6b..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/src/lib.rs +++ /dev/null @@ -1,9 +0,0 @@ -pub mod contract; -pub mod execute; -pub mod helpers; -pub mod msg; -pub mod query; -pub mod state; - -mod error; -pub use crate::error::VaultRewardsError; diff --git a/smart-contracts/osmosis/contracts/vault-rewards/src/msg.rs b/smart-contracts/osmosis/contracts/vault-rewards/src/msg.rs deleted file mode 100644 index 5b2547639..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/src/msg.rs +++ /dev/null @@ -1,70 +0,0 @@ -use crate::state::DistributionSchedule; -use cosmwasm_schema::cw_serde; -use cosmwasm_std::Uint128; -use cw_asset::{Asset, AssetInfo}; - -#[cw_serde] -pub struct InstantiateMsg { - pub vault_token: String, - pub reward_token: AssetInfo, - pub distribution_schedules: Vec, -} - -#[cw_serde] -pub enum ExecuteMsg { - Claim {}, - - Admin(AdminExecuteMsg), - - Vault(VaultExecuteMsg), -} - -#[cw_serde] -pub enum AdminExecuteMsg { - WithdrawFunds(Asset), - AddDistributionSchedule(DistributionSchedule), - UpdateDistributionSchedule { - id: u64, - update: DistributionScheduleOptions, - }, - RemoveDistributionSchedule(u64), -} - -#[cw_serde] -pub enum VaultExecuteMsg { - UpdateUserRewardIndex(String), -} - -#[cw_serde] -pub enum QueryMsg { - Config {}, - PendingRewards(String), - GetUserRewardsIndex(String), -} - -#[cw_serde] -pub struct MigrateMsg {} - -#[cw_serde] -pub struct ConfigResponse { - pub reward_token: AssetInfo, - pub contract_balance: Uint128, - pub total_claimed: Uint128, - pub distribution_schedules: Vec, - pub current_distribution_rate_per_block: Uint128, -} - -#[cw_serde] -pub struct DistributionScheduleResponse { - pub id: u64, - pub start: u64, - pub end: u64, - pub amount: Uint128, -} - -#[cw_serde] -pub struct DistributionScheduleOptions { - pub start: Option, - pub end: Option, - pub amount: Option, -} diff --git a/smart-contracts/osmosis/contracts/vault-rewards/src/query.rs b/smart-contracts/osmosis/contracts/vault-rewards/src/query.rs deleted file mode 100644 index 61b57e8af..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/src/query.rs +++ /dev/null @@ -1,65 +0,0 @@ -use crate::helpers::get_user_reward_index; -use crate::msg::ConfigResponse; -use crate::state::{DistributionSchedule, UserBalance, CONFIG}; -use crate::VaultRewardsError; -use crate::{execute::user::get_claim_amount, state::UserRewardIndex}; -use cosmwasm_std::{Addr, Deps, Env, Uint128}; -use cw_asset::AssetInfo; - -pub fn query_config(deps: Deps, env: Env) -> Result { - let config = CONFIG.load(deps.storage)?; - Ok(ConfigResponse { - reward_token: config.reward_token.clone(), - contract_balance: config - .reward_token - .query_balance(&deps.querier, env.contract.address)?, - total_claimed: config.total_claimed, - distribution_schedules: config - .distribution_schedules - .iter() - .enumerate() - .map(|(idx, s)| s.to_response(idx)) - .collect(), - current_distribution_rate_per_block: config - .get_distribution_rate_at_height(env.block.height), - }) -} - -pub fn query_pending_rewards( - deps: Deps, - env: Env, - user: Addr, -) -> Result { - let config = CONFIG.load(deps.storage)?; - let cur_block_height = env.block.height; - let mut user_reward_index = get_user_reward_index(deps.storage, &user); - let user_vault_token_balance = - AssetInfo::cw20(config.vault_token.clone()).query_balance(&deps.querier, &user)?; - if let Some(prev_balance) = user_reward_index.balance { - user_reward_index.history.push(DistributionSchedule { - start: prev_balance.reward_index, - end: cur_block_height, - amount: prev_balance.balance, - }); - user_reward_index.balance = if !user_vault_token_balance.is_zero() { - Some(UserBalance { - reward_index: cur_block_height + 1, - balance: user_vault_token_balance, - }) - } else { - None - }; - } - get_claim_amount(deps, &env, &config, &user_reward_index) -} - -pub fn query_user_rewards_index( - deps: Deps, - user: Addr, -) -> Result { - let user_reward_index = get_user_reward_index(deps.storage, &user); - Ok(user_reward_index) -} - -#[cfg(test)] -mod tests {} diff --git a/smart-contracts/osmosis/contracts/vault-rewards/src/state.rs b/smart-contracts/osmosis/contracts/vault-rewards/src/state.rs deleted file mode 100644 index 7456adab7..000000000 --- a/smart-contracts/osmosis/contracts/vault-rewards/src/state.rs +++ /dev/null @@ -1,225 +0,0 @@ -use crate::msg::{DistributionScheduleOptions, DistributionScheduleResponse}; -use crate::VaultRewardsError; -use cosmwasm_schema::cw_serde; -use cosmwasm_std::{Addr, Env, Order, QuerierWrapper, Storage, Uint128}; -use cw_asset::AssetInfo; -use cw_storage_plus::{Item, Map}; - -#[cw_serde] -pub struct Config { - pub vault_token: Addr, - pub reward_token: AssetInfo, - pub distribution_schedules: Vec, - pub total_claimed: Uint128, -} - -impl Config { - pub fn add_distribution_schedule( - &mut self, - querier: &QuerierWrapper, - env: &Env, - schedule: DistributionSchedule, - ) -> Result<(), VaultRewardsError> { - self.validate_distribution_schedule(querier, env, &schedule)?; - self.distribution_schedules.push(schedule); - Ok(()) - } - - pub fn add_distribution_schedules( - &mut self, - querier: &QuerierWrapper, - env: &Env, - schedules: Vec, - ) -> Result<(), VaultRewardsError> { - for schedule in schedules { - self.add_distribution_schedule(querier, env, schedule)?; - } - Ok(()) - } - - pub fn update_distribution_schedule( - &mut self, - querier: &QuerierWrapper, - env: &Env, - id: u64, - update: &DistributionScheduleOptions, - ) -> Result<(), VaultRewardsError> { - let cur_block_height = env.block.height; - let idx = id.checked_sub(1).unwrap_or_default() as usize; - let mut schedule = self - .distribution_schedules - .get(idx) - .ok_or(VaultRewardsError::InvalidDistributionScheduleId { - max_id: self.distribution_schedules.len() as u64, - })? - .clone(); - if schedule.start <= cur_block_height && cur_block_height < schedule.end { - return Err(VaultRewardsError::DistributionScheduleInProgress { - id, - start: schedule.start, - end: schedule.end, - }); - } - if cur_block_height >= schedule.end { - return Err(VaultRewardsError::DistributionScheduleExpired { - id, - end: schedule.end, - }); - } - self.distribution_schedules.remove(idx); - if let Some(start) = update.start { - schedule.start = start; - } else if let Some(end) = update.end { - if end <= cur_block_height { - return Err(VaultRewardsError::InvalidDistributionSchedule { - reason: "end must be in the future".to_string(), - }); - } - schedule.end = end; - } else if let Some(amount) = update.amount { - schedule.amount = amount; - } - self.validate_distribution_schedule(querier, env, &schedule)?; - self.distribution_schedules.insert(idx, schedule); - Ok(()) - } - - pub fn remove_distribution_schedule( - &mut self, - storage: &dyn Storage, - env: &Env, - id: u64, - ) -> Result<(), VaultRewardsError> { - let cur_block_height = env.block.height; - let idx = id.checked_sub(1).unwrap_or_default() as usize; - let schedule = self.distribution_schedules.get(idx).ok_or({ - VaultRewardsError::InvalidDistributionScheduleId { - max_id: self.distribution_schedules.len() as u64, - } - })?; - if schedule.start <= cur_block_height && cur_block_height < schedule.end { - return Err(VaultRewardsError::DistributionScheduleInProgress { - id, - start: schedule.start, - end: schedule.end, - }); - } - if cur_block_height >= schedule.end { - // check if all funds from period were claimed - USER_REWARD_INDEX - .range(storage, None, None, Order::Ascending) - .map(|item| { - let (_, reward_index) = item.unwrap(); - if reward_index - .history - .iter() - .any(|h| h.start >= schedule.start && h.end < schedule.end) - { - return Err(VaultRewardsError::DistributionScheduleWithUnclaimedFunds { - id, - }); - } - Ok(()) - }) - .collect::, _>>()?; - } - self.total_claimed -= schedule.amount; - self.distribution_schedules.remove(idx); - Ok(()) - } - - pub fn validate_distribution_schedule( - &self, - querier: &QuerierWrapper, - env: &Env, - schedule: &DistributionSchedule, - ) -> Result<(), VaultRewardsError> { - if schedule.start <= env.block.height { - return Err(VaultRewardsError::InvalidDistributionSchedule { - reason: "start must be in the future".to_string(), - }); - } - if schedule.start >= schedule.end { - return Err(VaultRewardsError::InvalidDistributionSchedule { - reason: "start must be before end".to_string(), - }); - } - if schedule.amount.is_zero() { - return Err(VaultRewardsError::InvalidDistributionSchedule { - reason: "amount must be greater than 0".to_string(), - }); - } - let reward_token_balance = self - .reward_token - .query_balance(querier, &env.contract.address)?; - let total_distribution_amount = self.get_total_distribution_amount() + schedule.amount; - if VALIDATE_FUNDS && reward_token_balance < total_distribution_amount { - return Err(VaultRewardsError::InsufficientFunds { - contract_balance: reward_token_balance, - claim_amount: total_distribution_amount, - }); - } - Ok(()) - } - - pub fn get_distribution_rate_at_height(&self, height: u64) -> Uint128 { - self.distribution_schedules - .iter() - .fold(Uint128::zero(), |acc, schedule| { - if schedule.start <= height && schedule.end > height { - acc + schedule.amount / Uint128::from(schedule.end - schedule.start) - } else { - acc - } - }) - } - - pub fn get_total_distribution_amount(&self) -> Uint128 { - self.distribution_schedules - .iter() - .fold(Uint128::zero(), |acc, schedule| acc + schedule.amount) - } -} - -#[cw_serde] -pub struct DistributionSchedule { - pub start: u64, - pub end: u64, - pub amount: Uint128, -} - -impl DistributionSchedule { - pub fn to_response(&self, idx: usize) -> DistributionScheduleResponse { - DistributionScheduleResponse { - id: (idx as u64) + 1, - start: self.start, - end: self.end, - amount: self.amount, - } - } -} - -#[cw_serde] -#[derive(Default)] -pub struct RewardIndex { - pub vault_supply: Uint128, -} - -#[cw_serde] -pub struct UserRewardIndex { - pub balance: Option, - pub history: Vec, -} - -#[cw_serde] -pub struct UserBalance { - pub reward_index: u64, - pub balance: Uint128, -} - -pub const CONFIG: Item = Item::new("config"); -pub const REWARD_INDEX: Map = Map::new("reward_index"); -pub const USER_REWARD_INDEX: Map = Map::new("user_reward_index"); - -// to be changed in a future migration -pub const VALIDATE_FUNDS: bool = false;