diff --git a/.changeset/config.json b/.changeset/config.json index edba6eadd51..e17c1ce1245 100644 --- a/.changeset/config.json +++ b/.changeset/config.json @@ -14,7 +14,6 @@ "wallet-dashboard", "apps-backend", "@iota/core", - "@iota/apps-ui-kit", "sponsored-transactions", "kiosk-demo", "kiosk-cli" diff --git a/.changeset/mighty-pans-suffer.md b/.changeset/mighty-pans-suffer.md new file mode 100644 index 00000000000..d58fe1593a3 --- /dev/null +++ b/.changeset/mighty-pans-suffer.md @@ -0,0 +1,5 @@ +--- +'@iota/dapp-kit': minor +--- + +Improved the connect-modal style so its easier to read and interact in small screens diff --git a/.dockerignore b/.dockerignore index ca5560030b7..06c579451f0 100644 --- a/.dockerignore +++ b/.dockerignore @@ -7,3 +7,5 @@ target/ build/ storage/ docker/ +dev-tools/ +setups/ \ No newline at end of file diff --git a/.github/CODEOWNERS b/.github/CODEOWNERS index 38bb6687fd2..cb5faa2324d 100644 --- a/.github/CODEOWNERS +++ b/.github/CODEOWNERS @@ -4,41 +4,90 @@ # Everything within the SDK folder /sdk/ @iotaledger/tooling -# Changes to the genesis builder should be approved by Konstantinos or Mirko at least -/crates/iota-genesis-builder/ @kodemartin @miker83z +/consensus/ @iotaledger/consensus -# infrastructure team -/docker/ @iotaledger/infrastructure @iotaledger/node @iotaledger/devops-admin -/crates/iota-json-rpc*/ @iotaledger/infrastructure -/crates/iota-graphql*/ @iotaledger/infrastructure -/crates/iota-indexer*/ @iotaledger/infrastructure -/crates/iota-data-ingestion*/ @iotaledger/infrastructure +# Crates +/crates/bin-version/ @iotaledger/dev-tools +/crates/iota/ @iotaledger/dev-tools +/crates/iota-adapter-transactional-tests/ @iotaledger/vm-language /crates/iota-analytics-indexer/ @iotaledger/infrastructure - -# node team +/crates/iota-analytics-indexer-derive/ @iotaledger/infrastructure /crates/iota-archival/ @iotaledger/node /crates/iota-authority-aggregation/ @iotaledger/node @iotaledger/consensus +/crates/iota-aws-orchestrator/ @iotaledger/consensus +/crates/iota-benchmark/ @iotaledger/node +/crates/iota-cluster-test/ @iotaledger/infrastructure +/crates/iota-common/ @iotaledger/core-protocol /crates/iota-config/ @iotaledger/node /crates/iota-core/ @iotaledger/node @iotaledger/consensus +/crates/iota-cost/ @iotaledger/vm-language +/crates/iota-data-ingestion*/ @iotaledger/infrastructure +/crates/iota-e2e-tests/ @iotaledger/node @iotaledger/vm-language +/crates/iota-enum-compat-util/ @iotaledger/vm-language +/crates/iota-faucet/ @iotaledger/infrastructure +/crates/iota-framework/ @iotaledger/vm-language +/crates/iota-framework-snapshot/ @iotaledger/vm-language +/crates/iota-framework-tests/ @iotaledger/vm-language +/crates/iota-genesis-builder/ @iotaledger/vm-language +/crates/iota-genesis-common/ @iotaledger/vm-language +/crates/iota-graphql*/ @iotaledger/infrastructure +/crates/iota-indexer*/ @iotaledger/infrastructure +/crates/iota-json/ @iotaledger/infrastructure +/crates/iota-json-rpc*/ @iotaledger/infrastructure +/crates/iota-keys/ @iotaledger/dev-tools +/crates/iota-light-client/ @iotaledger/dev-tools +/crates/iota-macros/ @iotaledger/dev-tools +/crates/iota-metric-checker/ @iotaledger/node +/crates/iota-metrics/ @iotaledger/core-protocol +/crates/iota-move/ @iotaledger/vm-language +/crates/iota-move-build/ @iotaledger/vm-language +/crates/iota-move-lsp/ @iotaledger/vm-language /crates/iota-network/ @iotaledger/node /crates/iota-network-stack/ @iotaledger/node /crates/iota-node/ @iotaledger/node -/crates/iota-types/ @iotaledger/node @iotaledger/sc-platform +/crates/iota-open-rpc/ @iotaledger/dev-tools +/crates/iota-open-rpc-macros/ @iotaledger/dev-tools +/crates/iota-package-dump/ @iotaledger/infrastructure +/crates/iota-package-management/ @iotaledger/vm-language +/crates/iota-package-resolver/ @iotaledger/vm-language +/crates/iota-proc-macros/ @iotaledger/dev-tools /crates/iota-protocol-config/ @iotaledger/node @iotaledger/sc-platform /crates/iota-protocol-config-macros/ @iotaledger/node @iotaledger/sc-platform +/crates/iota-proxy/ @iotaledger/node +/crates/iota-replay/ @iotaledger/node /crates/iota-rest-api/ @iotaledger/node @iotaledger/infrastructure +/crates/iota-rpc-loadgen/ @iotaledger/infrastructure +/crates/iota-sdk/ @iotaledger/dev-tools +/crates/iota-simulator/ @iotaledger/core-protocol +/crates/iota-single-node-benchmark/ @iotaledger/node /crates/iota-snapshot/ @iotaledger/node +/crates/iota-source-validation/ @iotaledger/vm-language /crates/iota-storage/ @iotaledger/node - -# consensus team -/consensus/ @iotaledger/consensus - -# dev-tools team -/crates/iota/ @iotaledger/dev-tools -/crates/iota-open-rpc/ @iotaledger/dev-tools -/crates/iota-open-rpc-macros/ @iotaledger/dev-tools -/crates/iota-sdk/ @iotaledger/dev-tools +/crates/iota-swarm/ @iotaledger/node +/crates/iota-swarm-config/ @iotaledger/node +/crates/iota-test-transaction-builder/ @iotaledger/dev-tools +/crates/iota-tls/ @iotaledger/core-protocol +/crates/iota-tool/ @iotaledger/node /crates/iota-transaction-builder/ @iotaledger/dev-tools +/crates/iota-transaction-checks/ @iotaledger/core-protocol +/crates/iota-transactional-test-runner/ @iotaledger/vm-language +/crates/iota-types/ @iotaledger/core-protocol @iotaledger/sc-platform +/crates/iota-upgrade-compatibility-transactional-tests/ @iotaledger/vm-language +/crates/iota-verifier-transactional-tests/ @iotaledger/vm-language +/crates/prometheus-closure-metric/ @iotaledger/node +/crates/shared-crypto/ @iotaledger/core-protocol +/crates/simulacrum/ @iotaledger/vm-language +/crates/telemetry-subscribers/ @iotaledger/core-protocol +/crates/test-cluster/ @iotaledger/infrastructure +/crates/typed-store/ @iotaledger/core-protocol +/crates/typed-store-derive/ @iotaledger/core-protocol +/crates/typed-store-error/ @iotaledger/core-protocol +/crates/typed-store-workspace-hack/ @iotaledger/core-protocol + +# Docker +/docker/ @iotaledger/infrastructure @iotaledger/node @iotaledger/devops-admin +/dev-tools/ @iotaledger/infrastructure @iotaledger/node @iotaledger/devops-admin +/setups/ @iotaledger/infrastructure @iotaledger/node @iotaledger/devops-admin # Frontend apps to be looked after by Boxfish Studio or the tooling team /apps/ @iotaledger/tooling @@ -57,7 +106,6 @@ prettier.config.js @iotaledger/tooling turbo.json @iotaledger/tooling vercel.json @iotaledger/tooling -# vm-language team # Needs to be after package.json ownership definition to override it /iota-execution/ @iotaledger/vm-language /external-crates/ @iotaledger/vm-language @@ -83,7 +131,23 @@ vercel.json @iotaledger/tooling /scripts/generate_files/ @muXxer /scripts/codesearch/ @muXxer /scripts/slipstream/ @muXxer +/scripts/tooling/ @iotaledger/tooling + +/kiosk/ @iotaledger/vm-language +/nre/ @iotaledger/node # Disable code ownership for these auto-generated files /Cargo.lock /pnpm-lock.yaml + +# TODO +# /crates/iota-bridge/ +# /crates/iota-bridge-cli/ +# /crates/iota-bridge-indexer/ +# /crates/iota-rosetta/ +# /crates/iota-source-validation-service/ +# /crates/iota-surfer/ +# /crates/iota-util-mem/ +# /crates/iota-util-mem-derive/ +# /crates/transaction-fuzzer/ +# /bridge/ \ No newline at end of file diff --git a/.github/actions/diffs/action.yml b/.github/actions/diffs/action.yml index f940c9b26d9..adc302f6e60 100644 --- a/.github/actions/diffs/action.yml +++ b/.github/actions/diffs/action.yml @@ -70,6 +70,8 @@ runs: - "dashboards/**" - "doc/**" - "docker/**" + - "dev-tools/**" + - "setups/**" - "external-crates/**" - "kiosk/**" - "nre/**" diff --git a/.github/crates-filters.yml b/.github/crates-filters.yml index 57a47dd7a25..d4afd5812bd 100644 --- a/.github/crates-filters.yml +++ b/.github/crates-filters.yml @@ -112,6 +112,8 @@ iota-protocol-config: - "crates/iota-protocol-config/**" iota-protocol-config-macros: - "crates/iota-protocol-config-macros/**" +iota-proxy: + - "crates/iota-proxy/**" iota-replay: - "crates/iota-replay/**" iota-rest-api: diff --git a/.github/labeler.yml b/.github/labeler.yml index 9c85e15b9f1..8617b1f2474 100644 --- a/.github/labeler.yml +++ b/.github/labeler.yml @@ -4,3 +4,37 @@ documentation: explorer: - changed-files: - any-glob-to-any-file: apps/explorer/**/* +wallet: + - changed-files: + - any-glob-to-any-file: apps/wallet/**/* +wallet-dashboard: + - changed-files: + - any-glob-to-any-file: apps/wallet-dashboard/**/* +ts-sdk: + - changed-files: + - any-glob-to-any-file: sdk/**/* +apps-ui-kit: + - changed-files: + - any-glob-to-any-file: apps/ui-kit/**/* +apps-backend: + - changed-files: + - any-glob-to-any-file: apps/apps-backend/**/* +tooling: + - changed-files: + - any-glob-to-any-file: | + sdk/** + apps/** + dapps/** + linting/** + .husky/** + .changeset/** + .eslintrc.js + .lintstagedrc.json + .npmrc + .prettierignore + graphql.config.ts + package.json + pnpm-workspace.yaml + prettier.config.js + turbo.json + vercel.json diff --git a/.github/scripts/rosetta/setup.sh b/.github/scripts/rosetta/setup.sh index b85acf125eb..b763b79e952 100755 --- a/.github/scripts/rosetta/setup.sh +++ b/.github/scripts/rosetta/setup.sh @@ -8,10 +8,7 @@ cargo install --locked --bin iota --path crates/iota cargo install --locked --bin iota-rosetta --path crates/iota-rosetta echo "run IOTA genesis" -CONFIG_DIR=~/.iota/iota_config -if ! [ -d "$CONFIG_DIR" ]; then - iota genesis -fi +iota genesis --force echo "generate rosetta configuration" iota-rosetta generate-rosetta-cli-config --online-url http://127.0.0.1:9002 --offline-url http://127.0.0.1:9003 diff --git a/.github/workflows/_e2e.yml b/.github/workflows/_e2e.yml index c40ca8da2de..973ccc0eea4 100644 --- a/.github/workflows/_e2e.yml +++ b/.github/workflows/_e2e.yml @@ -1,6 +1,7 @@ name: TS End-to-end Tests on: + workflow_dispatch: workflow_call: inputs: isRust: @@ -79,7 +80,7 @@ jobs: - name: Build apps-backend if: inputs.isAppsBackend || github.ref_name == 'develop' - run: pnpm --filter apps-backend build + run: pnpm turbo --filter apps-backend build - name: Run apps-backend e2e tests if: inputs.isAppsBackend || github.ref_name == 'develop' diff --git a/.github/workflows/_execution_cut.yml b/.github/workflows/_execution_cut.yml index 9c231940a39..b746b5895d9 100644 --- a/.github/workflows/_execution_cut.yml +++ b/.github/workflows/_execution_cut.yml @@ -12,11 +12,6 @@ jobs: runs-on: [self-hosted] steps: - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 - - name: Install cargo-hakari, and cache the binary - uses: baptiste0928/cargo-install@904927dbe77864e0f2281519fe9d5bd097a220b3 # v3.1.1 - with: - crate: cargo-hakari - locked: true - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 - name: Make cut run: ./scripts/execution_layer.py cut for_ci_test diff --git a/.github/workflows/_external_rust_tests.yml b/.github/workflows/_external_rust_tests.yml index cbb6ff968d0..be98867b4c4 100644 --- a/.github/workflows/_external_rust_tests.yml +++ b/.github/workflows/_external_rust_tests.yml @@ -27,9 +27,6 @@ jobs: runs-on: [self-hosted] steps: - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 - - uses: taiki-e/install-action@375e0c7f08a66b8c2ba7e7eef31a6f91043a81b0 # v2.44.38 - with: - tool: nextest - name: Install python dependencies run: pip install pyopenssl --upgrade --break-system-packages - name: cargo test diff --git a/.github/workflows/_move_tests.yml b/.github/workflows/_move_tests.yml index 54c8b002f08..11d685434bd 100644 --- a/.github/workflows/_move_tests.yml +++ b/.github/workflows/_move_tests.yml @@ -17,31 +17,23 @@ jobs: # to Move code but not Rust code (If there are Rust changes, they # will be run as part of a larger test suite). move-test: - timeout-minutes: 10 runs-on: [self-hosted] steps: - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 - - uses: taiki-e/install-action@375e0c7f08a66b8c2ba7e7eef31a6f91043a81b0 # v2.44.38 - with: - tool: nextest - name: Run move tests - run: | - cargo nextest run -E + run: > + cargo nextest run --profile ci -E 'package(iota-framework-tests) or (package(iota-core) and test(quorum_driver::)) or package(iota-benchmark) or test(move_tests::)' move-simtest: - timeout-minutes: 10 runs-on: [self-hosted] steps: - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 - - uses: taiki-e/install-action@375e0c7f08a66b8c2ba7e7eef31a6f91043a81b0 # v2.44.38 - with: - tool: nextest - name: Run move tests - run: | + run: > scripts/simtest/cargo-simtest simtest --profile ci -E 'package(iota-framework-tests) or (package(iota-core) and test(quorum_driver::)) diff --git a/.github/workflows/_rust_tests.yml b/.github/workflows/_rust_tests.yml index e781ff2ea97..84667b5fa68 100644 --- a/.github/workflows/_rust_tests.yml +++ b/.github/workflows/_rust_tests.yml @@ -39,10 +39,7 @@ jobs: runs-on: [self-hosted] steps: - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 - - uses: taiki-e/install-action@375e0c7f08a66b8c2ba7e7eef31a6f91043a81b0 # v2.44.38 - with: - tool: nextest - - name: cargo test + - name: cargo nextest run: | array=(${{ inputs.changedCrates }}) @@ -91,9 +88,6 @@ jobs: fail-fast: false steps: - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 - - uses: taiki-e/install-action@375e0c7f08a66b8c2ba7e7eef31a6f91043a81b0 # v2.44.38 - with: - tool: nextest - name: benchmark (smoke) run: | cargo run --package iota-benchmark --bin stress -- --log-path /tmp/stress.log --num-client-threads 10 --num-server-threads 24 --num-transfer-accounts 2 bench --target-qps 100 --num-workers 10 --transfer-object 50 --shared-counter 50 --run-duration 10s --stress-stat-collection @@ -103,11 +97,6 @@ jobs: - name: rustdoc run: | cargo doc --all-features --workspace --no-deps - - name: Install cargo-hakari, and cache the binary - uses: baptiste0928/cargo-install@904927dbe77864e0f2281519fe9d5bd097a220b3 # v3.1.1 - with: - crate: cargo-hakari - locked: true - name: Install nightly rustfmt run: rustup toolchain install nightly --component rustfmt --allow-downgrade - name: iota-execution @@ -126,9 +115,6 @@ jobs: MSIM_WATCHDOG_TIMEOUT_MS: 60000 steps: - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 - - uses: taiki-e/install-action@375e0c7f08a66b8c2ba7e7eef31a6f91043a81b0 # v2.44.38 - with: - tool: nextest - name: setup filter run: | array=(${{ inputs.changedCrates }}) @@ -185,9 +171,6 @@ jobs: - 5432:5432 steps: - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 - - uses: taiki-e/install-action@375e0c7f08a66b8c2ba7e7eef31a6f91043a81b0 # v2.44.38 - with: - tool: nextest - name: Install postgresql-client run: sudo apt-get install -y postgresql-client - name: Setup db @@ -202,6 +185,6 @@ jobs: cargo nextest run --no-fail-fast --test-threads 8 --package iota-graphql-e2e-tests --features pg_integration cargo nextest run --no-fail-fast --test-threads 1 --package iota-cluster-test --test local_cluster_test --features pg_integration cargo nextest run --no-fail-fast --test-threads 1 --package iota-indexer --test ingestion_tests --features pg_integration - # Iota-indexer's RPC tests, which depend on a shared runtime, are incompatible with nextest due to its process-per-test execution model. + # iota-indexer's RPC tests, which depend on a shared runtime, are incompatible with nextest due to its process-per-test execution model. # cargo test, on the other hand, allows tests to share state and resources by default. cargo test --profile simulator --package iota-indexer --test rpc-tests --features shared_test_runtime diff --git a/.github/workflows/_vercel_deploy.yml b/.github/workflows/_vercel_deploy.yml deleted file mode 100644 index ffdb8fda37f..00000000000 --- a/.github/workflows/_vercel_deploy.yml +++ /dev/null @@ -1,96 +0,0 @@ -name: Vercel Deploys - -on: - workflow_call: - inputs: - isExplorer: - type: boolean - required: true - isTypescriptSDK: - type: boolean - required: true - isAppsBackend: - type: boolean - required: true - isAppsUiKit: - type: boolean - required: true - isWalletDashboard: - type: boolean - required: true - shouldDeployPreview: - type: boolean - required: true - -concurrency: - group: vercel-deploy-${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }} - cancel-in-progress: true - -jobs: - explorer-preview: - name: Vercel Explorer Preview - if: inputs.shouldDeployPreview && inputs.isExplorer - uses: ./.github/workflows/apps_explorer_deploy.yml - secrets: inherit - with: - isProd: false - isStaging: false - - explorer-staging: - name: Vercel Explorer Staging - if: github.ref_name == 'develop' - uses: ./.github/workflows/apps_explorer_deploy.yml - secrets: inherit - with: - isProd: false - isStaging: true - - ui-kit-preview: - name: Vercel UI Kit Preview - if: inputs.shouldDeployPreview && inputs.isAppsUiKit - uses: ./.github/workflows/apps_ui_kit_deploy.yml - secrets: inherit - with: - isProd: false - - ui-kit-prod: - name: Vercel UI Kit Production - if: github.ref_name == 'develop' - uses: ./.github/workflows/apps_ui_kit_deploy.yml - secrets: inherit - with: - isProd: true - - wallet-dashboard-preview: - name: Vercel Wallet Dashboard Preview - if: inputs.shouldDeployPreview && inputs.isWalletDashboard - uses: ./.github/workflows/apps_wallet_dashboard_deploy.yml - secrets: inherit - with: - isProd: false - isStaging: false - - wallet-dashboard-staging: - name: Vercel Wallet Dashboard Staging - if: github.ref_name == 'develop' - uses: ./.github/workflows/apps_wallet_dashboard_deploy.yml - secrets: inherit - with: - isProd: false - isStaging: true - - apps-backend-preview: - name: Vercel apps-backend Preview - if: inputs.shouldDeployPreview && inputs.isAppsBackend - uses: ./.github/workflows/apps_backend_deploy.yml - secrets: inherit - with: - isProd: false - - apps-backend-prod: - name: Vercel apps-backend Production - if: github.ref_name == 'develop' - uses: ./.github/workflows/apps_backend_deploy.yml - secrets: inherit - with: - isProd: true diff --git a/.github/workflows/apps_backend_deploy.yml b/.github/workflows/apps_backend_deploy.yml deleted file mode 100644 index 63c5b01a9e6..00000000000 --- a/.github/workflows/apps_backend_deploy.yml +++ /dev/null @@ -1,78 +0,0 @@ -name: Deploy for Apps Backend - -env: - VERCEL_ORG_ID: ${{ secrets.VERCEL_ORG_ID }} - VERCEL_PROJECT_ID: ${{ secrets.APPS_BACKEND_VERCEL_PROJECT_ID }} - -on: - workflow_dispatch: - workflow_call: - inputs: - isProd: - type: boolean - required: true - -jobs: - deploy: - permissions: - contents: read - pull-requests: write - runs-on: [self-hosted] - steps: - - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 - - uses: pnpm/action-setup@fe02b34f77f8bc703788d5817da081398fad5dd2 # v4.0.0 - - name: Install Nodejs - uses: actions/setup-node@0a44ba7841725637a19e28fa30b79a866c81b0a6 # v4.0.4 - with: - node-version: "20" - cache: "pnpm" - - name: Install dependencies - run: pnpm install --frozen-lockfile - - name: Setup Prod Flag - id: setup_prod_flags - run: | - if [[ "${{ inputs.isProd }}" = "true" ]]; then - echo "PROD_FLAG=--prod" >> $GITHUB_OUTPUT - echo "ENVIRONMENT=production" >> $GITHUB_OUTPUT - echo "VERCEL_OUTPUT=" >> $GITHUB_OUTPUT - else - echo "PROD_FLAG=" >> $GITHUB_OUTPUT - echo "ENVIRONMENT=preview" >> $GITHUB_OUTPUT - echo "VERCEL_OUTPUT=> vercel_output.txt" >> $GITHUB_OUTPUT - fi - - name: Turbo Cache - id: turbo-cache - uses: actions/cache@3624ceb22c1c5a301c8db4169662070a689d9ea8 # v4.1.1 - with: - path: node_modules/.cache/turbo - key: turbo-${{ runner.os }}-${{ github.sha }} - restore-keys: | - turbo-${{ runner.os }}- - - name: Install Vercel CLI - run: pnpm add --global vercel@canary - - name: Pull Vercel Env variables (network configs) - run: vercel pull --yes --environment=${{steps.setup_prod_flags.outputs.ENVIRONMENT}} --token=${{ secrets.VERCEL_TOKEN }} - - name: Build the Apps Backend - run: pnpm apps-backend build - - name: Build Vercel Project Artifacts - run: vercel build ${{steps.setup_prod_flags.outputs.PROD_FLAG}} --token=${{ secrets.VERCEL_TOKEN }} - - name: Deploy Project Artifacts to Vercel - run: vercel deploy ${{steps.setup_prod_flags.outputs.PROD_FLAG}} --prebuilt --token=${{ secrets.VERCEL_TOKEN }} ${{ steps.setup_prod_flags.outputs.VERCEL_OUTPUT }} - - name: Extract Deploy URL - id: deploy_url - if: ${{ inputs.isProd == false }} - run: echo "DEPLOY_URL=$(cat vercel_output.txt | awk 'END{print}')" >> $GITHUB_OUTPUT - - name: Comment on pull request - if: ${{ inputs.isProd == false }} - uses: actions/github-script@60a0d83039c74a4aee543508d2ffcb1c3799cdea # v7.0.1 - with: - github-token: ${{ secrets.GITHUB_TOKEN }} - script: | - const DEPLOY_URL = '${{ steps.deploy_url.outputs.DEPLOY_URL }}' - const COMMIT_SHA = '${{ github.event.pull_request.head.sha }}' - github.rest.issues.createComment({ - issue_number: context.issue.number, - owner: context.repo.owner, - repo: context.repo.repo, - body: `This pull request has been deployed to Vercel.\n\n\nLatest commit: ${COMMIT_SHA}\n\n:white_check_mark: Preview: ${DEPLOY_URL}` - }) diff --git a/.github/workflows/apps_explorer_deploy.yml b/.github/workflows/apps_explorer_deploy.yml deleted file mode 100644 index 0b57bb6b97c..00000000000 --- a/.github/workflows/apps_explorer_deploy.yml +++ /dev/null @@ -1,94 +0,0 @@ -name: Deploy for Explorer - -env: - VERCEL_ORG_ID: ${{ secrets.VERCEL_ORG_ID }} - VERCEL_PROJECT_ID: ${{ secrets.EXPLORER_VERCEL_PROJECT_ID }} - EXPLORER_VERCEL_PROJECT_STAGING_URL: ${{ secrets.EXPLORER_VERCEL_PROJECT_STAGING_URL }} - -on: - workflow_dispatch: - inputs: - isProd: - type: boolean - required: true - isStaging: - type: boolean - required: true - workflow_call: - inputs: - isProd: - type: boolean - required: true - isStaging: - type: boolean - required: true - -jobs: - deploy: - permissions: - contents: read - pull-requests: write - runs-on: [self-hosted] - steps: - - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 - - uses: pnpm/action-setup@fe02b34f77f8bc703788d5817da081398fad5dd2 # v4.0.0 - - name: Install Nodejs - uses: actions/setup-node@0a44ba7841725637a19e28fa30b79a866c81b0a6 # v4.0.4 - with: - node-version: "20" - cache: "pnpm" - - name: Install dependencies - run: pnpm install --frozen-lockfile - - name: Setup Prod Flag - id: setup_prod_flags - run: | - if [[ "${{ inputs.isProd }}" = "true" ]]; then - echo "PROD_FLAG=--prod" >> $GITHUB_OUTPUT - echo "ENVIRONMENT=production" >> $GITHUB_OUTPUT - echo "VERCEL_OUTPUT=" >> $GITHUB_OUTPUT - else - echo "PROD_FLAG=" >> $GITHUB_OUTPUT - echo "ENVIRONMENT=preview" >> $GITHUB_OUTPUT - echo "VERCEL_OUTPUT=> vercel_output.txt" >> $GITHUB_OUTPUT - fi - - name: Turbo Cache - id: turbo-cache - uses: actions/cache@3624ceb22c1c5a301c8db4169662070a689d9ea8 # v4.1.1 - with: - path: node_modules/.cache/turbo - key: turbo-${{ runner.os }}-${{ github.sha }} - restore-keys: | - turbo-${{ runner.os }}- - - name: Install Vercel CLI - run: pnpm add --global vercel@canary - - name: Pull Vercel Env variables (network configs) - run: vercel pull --yes --environment=${{steps.setup_prod_flags.outputs.ENVIRONMENT}} --token=${{ secrets.VERCEL_TOKEN }} - - name: Copy the .env file - run: cp ./.vercel/.env.${{steps.setup_prod_flags.outputs.ENVIRONMENT}}.local ./sdk/.env - - name: Build Explorer - run: pnpm explorer build - - name: Build Vercel Project Artifacts - run: vercel build ${{steps.setup_prod_flags.outputs.PROD_FLAG}} --token=${{ secrets.VERCEL_TOKEN }} - - name: Deploy Project Artifacts to Vercel - run: vercel deploy ${{steps.setup_prod_flags.outputs.PROD_FLAG}} --prebuilt --token=${{ secrets.VERCEL_TOKEN }} ${{ steps.setup_prod_flags.outputs.VERCEL_OUTPUT }} - - name: Extract Deploy URL - id: deploy_url - if: ${{ inputs.isProd == false }} - run: echo "DEPLOY_URL=$(cat vercel_output.txt | awk 'END{print}')" >> $GITHUB_OUTPUT - - name: Alias Staging deploy - if: ${{ inputs.isStaging }} - run: vercel alias ${{ steps.deploy_url.outputs.DEPLOY_URL }} $EXPLORER_VERCEL_PROJECT_STAGING_URL --token=${{ secrets.VERCEL_TOKEN }} --scope=${{ secrets.VERCEL_SCOPE }} - - name: Comment on pull request - if: ${{ inputs.isProd == false && inputs.isStaging == false }} - uses: actions/github-script@60a0d83039c74a4aee543508d2ffcb1c3799cdea # v7.0.1 - with: - github-token: ${{ secrets.GITHUB_TOKEN }} - script: | - const DEPLOY_URL = '${{ steps.deploy_url.outputs.DEPLOY_URL }}' - const COMMIT_SHA = '${{ github.event.pull_request.head.sha }}' - github.rest.issues.createComment({ - issue_number: context.issue.number, - owner: context.repo.owner, - repo: context.repo.repo, - body: `This pull request has been deployed to Vercel.\n\n\nLatest commit: ${COMMIT_SHA}\n\n:white_check_mark: Preview: ${DEPLOY_URL}` - }) diff --git a/.github/workflows/apps_explorer_prod_deploy.yml b/.github/workflows/apps_explorer_prod_deploy.yml new file mode 100644 index 00000000000..a47ed8bf829 --- /dev/null +++ b/.github/workflows/apps_explorer_prod_deploy.yml @@ -0,0 +1,44 @@ +name: Production Deploy for Explorer + +env: + VERCEL_ORG_ID: ${{ secrets.VERCEL_ORG_ID }} + VERCEL_PROJECT_ID: ${{ secrets.EXPLORER_VERCEL_PROJECT_ID }} + +on: workflow_dispatch + +jobs: + deploy: + permissions: + contents: read + pull-requests: write + runs-on: [self-hosted] + steps: + - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 + - uses: pnpm/action-setup@fe02b34f77f8bc703788d5817da081398fad5dd2 # v4.0.0 + - name: Install Nodejs + uses: actions/setup-node@0a44ba7841725637a19e28fa30b79a866c81b0a6 # v4.0.4 + with: + node-version: "20" + cache: "pnpm" + - name: Install dependencies + run: pnpm install --frozen-lockfile + - name: Turbo Cache + id: turbo-cache + uses: actions/cache@3624ceb22c1c5a301c8db4169662070a689d9ea8 # v4.1.1 + with: + path: node_modules/.cache/turbo + key: turbo-${{ runner.os }}-${{ github.sha }} + restore-keys: | + turbo-${{ runner.os }}- + - name: Install Vercel CLI + run: pnpm add --global vercel@canary + - name: Pull Vercel Env variables (network configs) + run: vercel pull --yes --environment=production --token=${{ secrets.VERCEL_TOKEN }} + - name: Copy the .env file + run: cp ./.vercel/.env.production.local ./sdk/.env + - name: Build Explorer + run: pnpm explorer build + - name: Build Vercel Project Artifacts + run: vercel build --prod --token=${{ secrets.VERCEL_TOKEN }} + - name: Deploy Project Artifacts to Vercel + run: vercel deploy --prod --prebuilt --token=${{ secrets.VERCEL_TOKEN }} diff --git a/.github/workflows/apps_ui_kit_deploy.yml b/.github/workflows/apps_ui_kit_deploy.yml deleted file mode 100644 index 3c2f4e753f5..00000000000 --- a/.github/workflows/apps_ui_kit_deploy.yml +++ /dev/null @@ -1,68 +0,0 @@ -name: Deploy for Apps UI Kit Storybook - -env: - VERCEL_ORG_ID: ${{ secrets.VERCEL_ORG_ID }} - VERCEL_PROJECT_ID: ${{ secrets.APPS_UI_KIT_VERCEL_PROJECT_ID }} - -on: - workflow_dispatch: - workflow_call: - inputs: - isProd: - type: boolean - required: true - -jobs: - deploy: - permissions: - contents: read - pull-requests: write - runs-on: [self-hosted] - steps: - - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 - - uses: pnpm/action-setup@fe02b34f77f8bc703788d5817da081398fad5dd2 # v4.0.0 - - name: Install Nodejs - uses: actions/setup-node@0a44ba7841725637a19e28fa30b79a866c81b0a6 # v4.0.4 - with: - node-version: "20" - cache: "pnpm" - - name: Install dependencies - run: pnpm install --frozen-lockfile - - name: Setup Prod Flag - id: setup_prod_flags - run: | - if [[ "${{ inputs.isProd }}" = "true" ]]; then - echo "PROD_FLAG=--prod" >> $GITHUB_OUTPUT - echo "ENVIRONMENT=production" >> $GITHUB_OUTPUT - echo "VERCEL_OUTPUT=" >> $GITHUB_OUTPUT - else - echo "PROD_FLAG=" >> $GITHUB_OUTPUT - echo "ENVIRONMENT=preview" >> $GITHUB_OUTPUT - echo "VERCEL_OUTPUT=> vercel_output.txt" >> $GITHUB_OUTPUT - fi - - name: Install Vercel CLI - run: pnpm add --global vercel@canary - - name: Pull Vercel Environment Information - run: vercel pull --cwd ./apps/ui-kit --yes --environment=${{steps.setup_prod_flags.outputs.ENVIRONMENT}} --token=${{ secrets.VERCEL_TOKEN }} - - name: Build Project Artifacts - run: vercel build ${{steps.setup_prod_flags.outputs.PROD_FLAG}} --cwd ./apps/ui-kit --token=${{ secrets.VERCEL_TOKEN }} - - name: Deploy Project Artifacts to Vercel - run: vercel deploy --cwd ./apps/ui-kit ${{steps.setup_prod_flags.outputs.PROD_FLAG}} --prebuilt --token=${{ secrets.VERCEL_TOKEN }} ${{ steps.setup_prod_flags.outputs.VERCEL_OUTPUT }} - - name: Extract Deploy URL - id: deploy_url - if: ${{ inputs.isProd == false }} - run: echo "DEPLOY_URL=$(cat vercel_output.txt | awk 'END{print}')" >> $GITHUB_OUTPUT - - name: Comment on pull request - if: ${{ inputs.isProd == false }} - uses: actions/github-script@60a0d83039c74a4aee543508d2ffcb1c3799cdea # v7.0.1 - with: - github-token: ${{ secrets.GITHUB_TOKEN }} - script: | - const DEPLOY_URL = '${{ steps.deploy_url.outputs.DEPLOY_URL }}' - const COMMIT_SHA = '${{ github.event.pull_request.head.sha }}' - github.rest.issues.createComment({ - issue_number: context.issue.number, - owner: context.repo.owner, - repo: context.repo.repo, - body: `This pull request has been deployed to Vercel.\n\n\nLatest commit: ${COMMIT_SHA}\n\n:white_check_mark: Preview: ${DEPLOY_URL}` - }) diff --git a/.github/workflows/apps_wallet_dashboard_deploy.yml b/.github/workflows/apps_wallet_dashboard_deploy.yml deleted file mode 100644 index f533776b7bb..00000000000 --- a/.github/workflows/apps_wallet_dashboard_deploy.yml +++ /dev/null @@ -1,94 +0,0 @@ -name: Deploy for Wallet Dashboard - -env: - VERCEL_ORG_ID: ${{ secrets.VERCEL_ORG_ID }} - VERCEL_PROJECT_ID: ${{ secrets.WALLET_DASHBOARD_VERCEL_PROJECT_ID }} - WALLET_DASHBOARD_VERCEL_PROJECT_STAGING_URL: ${{ secrets.WALLET_DASHBOARD_VERCEL_PROJECT_STAGING_URL }} - -on: - workflow_dispatch: - inputs: - isProd: - type: boolean - required: true - isStaging: - type: boolean - required: true - workflow_call: - inputs: - isProd: - type: boolean - required: true - isStaging: - type: boolean - required: true - -jobs: - deploy: - permissions: - contents: read - pull-requests: write - runs-on: [self-hosted] - steps: - - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 - - uses: pnpm/action-setup@fe02b34f77f8bc703788d5817da081398fad5dd2 # v4.0.0 - - name: Install Nodejs - uses: actions/setup-node@0a44ba7841725637a19e28fa30b79a866c81b0a6 # v4.0.4 - with: - node-version: "20" - cache: "pnpm" - - name: Install dependencies - run: pnpm install --frozen-lockfile - - name: Setup Prod Flag - id: setup_prod_flags - run: | - if [[ "${{ inputs.isProd }}" = "true" ]]; then - echo "PROD_FLAG=--prod" >> $GITHUB_OUTPUT - echo "ENVIRONMENT=production" >> $GITHUB_OUTPUT - echo "VERCEL_OUTPUT=" >> $GITHUB_OUTPUT - else - echo "PROD_FLAG=" >> $GITHUB_OUTPUT - echo "ENVIRONMENT=preview" >> $GITHUB_OUTPUT - echo "VERCEL_OUTPUT=> vercel_output.txt" >> $GITHUB_OUTPUT - fi - - name: Turbo Cache - id: turbo-cache - uses: actions/cache@3624ceb22c1c5a301c8db4169662070a689d9ea8 # v4.1.1 - with: - path: node_modules/.cache/turbo - key: turbo-${{ runner.os }}-${{ github.sha }} - restore-keys: | - turbo-${{ runner.os }}- - - name: Install Vercel CLI - run: pnpm add --global vercel@canary - - name: Pull Vercel Env variables (network configs) - run: vercel pull --yes --environment=${{steps.setup_prod_flags.outputs.ENVIRONMENT}} --token=${{ secrets.VERCEL_TOKEN }} - - name: Copy the .env file - run: cp ./.vercel/.env.${{steps.setup_prod_flags.outputs.ENVIRONMENT}}.local ./sdk/.env - - name: Build Wallet Dashboard - run: pnpm wallet-dashboard build - - name: Build Vercel Project Artifacts - run: vercel build ${{steps.setup_prod_flags.outputs.PROD_FLAG}} --token=${{ secrets.VERCEL_TOKEN }} - - name: Deploy Project Artifacts to Vercel - run: vercel deploy ${{steps.setup_prod_flags.outputs.PROD_FLAG}} --prebuilt --token=${{ secrets.VERCEL_TOKEN }} ${{ steps.setup_prod_flags.outputs.VERCEL_OUTPUT }} - - name: Extract Deploy URL - id: deploy_url - if: ${{ inputs.isProd == false }} - run: echo "DEPLOY_URL=$(cat vercel_output.txt | awk 'END{print}')" >> $GITHUB_OUTPUT - - name: Alias Staging deploy - if: ${{ inputs.isStaging }} - run: vercel alias ${{ steps.deploy_url.outputs.DEPLOY_URL }} $WALLET_DASHBOARD_VERCEL_PROJECT_STAGING_URL --token=${{ secrets.VERCEL_TOKEN }} --scope=${{ secrets.VERCEL_SCOPE }} - - name: Comment on pull request - if: ${{ inputs.isProd == false && inputs.isStaging == false }} - uses: actions/github-script@60a0d83039c74a4aee543508d2ffcb1c3799cdea # v7.0.1 - with: - github-token: ${{ secrets.GITHUB_TOKEN }} - script: | - const DEPLOY_URL = '${{ steps.deploy_url.outputs.DEPLOY_URL }}' - const COMMIT_SHA = '${{ github.event.pull_request.head.sha }}' - github.rest.issues.createComment({ - issue_number: context.issue.number, - owner: context.repo.owner, - repo: context.repo.repo, - body: `This pull request has been deployed to Vercel.\n\n\nLatest commit: ${COMMIT_SHA}\n\n:white_check_mark: Preview: ${DEPLOY_URL}` - }) diff --git a/.github/workflows/apps_wallet_dashboard_prod_deploy.yml b/.github/workflows/apps_wallet_dashboard_prod_deploy.yml new file mode 100644 index 00000000000..8662082ab5e --- /dev/null +++ b/.github/workflows/apps_wallet_dashboard_prod_deploy.yml @@ -0,0 +1,44 @@ +name: Production Deploy for Wallet Dashboard + +env: + VERCEL_ORG_ID: ${{ secrets.VERCEL_ORG_ID }} + VERCEL_PROJECT_ID: ${{ secrets.WALLET_DASHBOARD_VERCEL_PROJECT_ID }} + +on: workflow_dispatch + +jobs: + deploy: + permissions: + contents: read + pull-requests: write + runs-on: [self-hosted] + steps: + - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 + - uses: pnpm/action-setup@fe02b34f77f8bc703788d5817da081398fad5dd2 # v4.0.0 + - name: Install Nodejs + uses: actions/setup-node@0a44ba7841725637a19e28fa30b79a866c81b0a6 # v4.0.4 + with: + node-version: "20" + cache: "pnpm" + - name: Install dependencies + run: pnpm install --frozen-lockfile + - name: Turbo Cache + id: turbo-cache + uses: actions/cache@3624ceb22c1c5a301c8db4169662070a689d9ea8 # v4.1.1 + with: + path: node_modules/.cache/turbo + key: turbo-${{ runner.os }}-${{ github.sha }} + restore-keys: | + turbo-${{ runner.os }}- + - name: Install Vercel CLI + run: pnpm add --global vercel@canary + - name: Pull Vercel Env variables (network configs) + run: vercel pull --yes --environment=production --token=${{ secrets.VERCEL_TOKEN }} + - name: Copy the .env file + run: cp ./.vercel/.env.production.local ./sdk/.env + - name: Build Wallet Dashboard + run: pnpm wallet-dashboard build + - name: Build Vercel Project Artifacts + run: vercel build --prod --token=${{ secrets.VERCEL_TOKEN }} + - name: Deploy Project Artifacts to Vercel + run: vercel deploy --prod --prebuilt --token=${{ secrets.VERCEL_TOKEN }} diff --git a/.github/workflows/apps_wallet_prod_build.yml b/.github/workflows/apps_wallet_prod_build.yml index f4714cd53e1..9a8052331d8 100644 --- a/.github/workflows/apps_wallet_prod_build.yml +++ b/.github/workflows/apps_wallet_prod_build.yml @@ -20,6 +20,11 @@ jobs: steps: - name: Checking out the repository uses: actions/checkout@f43a0e5ff2bd294095638e18286ca9a3d1956744 # v3.6.0 + with: + # Number of commits to fetch. 0 indicates all history for all branches and tags. Default: 1 + fetch-depth: 0 + # Whether to fetch tags, even if fetch-depth > 0. + fetch-tags: "true" - uses: pnpm/action-setup@fe02b34f77f8bc703788d5817da081398fad5dd2 # v4.0.0 - name: Install Nodejs uses: actions/setup-node@60edb5dd545a775178f52524783378180af0d1f8 # v4.0.2 @@ -61,8 +66,9 @@ jobs: echo "No previous tag found. Skipping changelog generation." echo "changelog=No previous tag found. Changelog generation skipped." >> $GITHUB_OUTPUT else - echo "## Changelog" >> CHANGELOG.md - git log ${{ env.PREV_TAG }}..${{ env.CURRENT_TAG }} --pretty=format:"- %s in #%h" -- ./apps/wallet > CHANGELOG.md + echo "## Changelog" > CHANGELOG.md + git log ${{ env.PREV_TAG }}..${{ env.CURRENT_TAG }} --pretty=format:"- %s in #%h" -- ./apps/wallet >> CHANGELOG.md + cat CHANGELOG.md fi - name: Get version from tag diff --git a/.github/workflows/cargo_llvm_cov.yml b/.github/workflows/cargo_llvm_cov.yml index 15000ad8e5b..fb9c241060c 100644 --- a/.github/workflows/cargo_llvm_cov.yml +++ b/.github/workflows/cargo_llvm_cov.yml @@ -42,11 +42,6 @@ jobs: - uses: bmwill/rust-cache@cb2cf0cc7c5198d3364b9630e2c3d457f160790c # v1.4.0 - - name: Install nextest and cargo-llvm-cov - uses: taiki-e/install-action@375e0c7f08a66b8c2ba7e7eef31a6f91043a81b0 # v2.44.38 - with: - tool: nextest,cargo-llvm-cov - - name: Set Swap Space uses: actionhippie/swap-space@73376950a0019f8e1f6a3d3d2673fe2aed74ae17 # v1.0.2 with: @@ -139,8 +134,7 @@ jobs: echo "Creating report (lcov.info)." LLVM_PROFILE_FILE="merged.profdata" cargo llvm-cov report \ --lcov \ - --output-path lcov.info \ - --ignore-filename-regex 'external-crates/.*' + --output-path lcov.info echo "Removing absolute path prefix: ${{ github.workspace }} from report." sed --in-place "s#${{ github.workspace }}#.#g" lcov.info diff --git a/.github/workflows/develop_ci_slack_report.yml b/.github/workflows/develop_ci_slack_report.yml index 1dc6c0561e1..0dce4cca323 100644 --- a/.github/workflows/develop_ci_slack_report.yml +++ b/.github/workflows/develop_ci_slack_report.yml @@ -18,4 +18,4 @@ jobs: message_format: ":fire: *${{github.event.workflow_run.name}}* ${{github.event.workflow_run.conclusion}} <${{github.server_url}}/${{github.repository}}/actions/runs/${{github.event.workflow_run.id}}|View Failure> author: ${{ github.event.workflow_run.head_commit.author.name }}" footer: "${{ github.event.workflow_run.display_title }} \n<${{github.server_url}}/${{github.repository}}/commit/${{github.event.workflow_run.head_commit.id}}>" env: - SLACK_WEBHOOK_URL: ${{ secrets.DEV_BROKEN_CI_SLACK_WEBHOOK }} + SLACK_WEBHOOK_URL: ${{ github.event.workflow_run.name == 'Nightly checks' && secrets.NIGHTLY_BROKEN_CI_SLACK_WEBHOOK || secrets.DEV_BROKEN_CI_SLACK_WEBHOOK }} diff --git a/.github/workflows/hierarchy.yml b/.github/workflows/hierarchy.yml index 45efa02e7ca..49658f11787 100644 --- a/.github/workflows/hierarchy.yml +++ b/.github/workflows/hierarchy.yml @@ -29,7 +29,6 @@ jobs: isTypescriptSDK: ${{ (steps.turbo.outputs.packages && contains(fromJson(steps.turbo.outputs.packages), '@iota/iota-sdk')) }} isAppsBackend: ${{ (steps.turbo.outputs.packages && contains(fromJson(steps.turbo.outputs.packages), 'apps-backend')) }} isAppsUiKit: ${{ (steps.turbo.outputs.packages && contains(fromJson(steps.turbo.outputs.packages), '@iota/apps-ui-kit')) }} - isWalletDashboard: ${{ (steps.turbo.outputs.packages && contains(fromJson(steps.turbo.outputs.packages), 'wallet-dashboard')) }} isGraphQlTransport: ${{ (steps.turbo.outputs.packages && contains(fromJson(steps.turbo.outputs.packages), '@iota/graphql-transport')) }} isLedgerjs: ${{ (steps.turbo.outputs.packages && contains(fromJson(steps.turbo.outputs.packages), '@iota/ledgerjs-hw-app-iota')) }} steps: @@ -108,15 +107,12 @@ jobs: needs: - diff - dprint-format - - license-check - typos - if: | - !cancelled() && !failure() && - (needs.diff.outputs.isRosetta == 'true' || needs.diff.outputs.isRust == 'true') + if: needs.diff.outputs.isRosetta == 'true' || needs.diff.outputs.isRust == 'true' uses: ./.github/workflows/_rosetta.yml e2e: - if: (!github.event.pull_request.draft || github.ref_name == 'develop') + if: (!github.event.pull_request.draft && github.ref_name != 'develop') needs: - diff - dprint-format @@ -131,22 +127,6 @@ jobs: isTypescriptSDK: ${{ needs.diff.outputs.isTypescriptSDK == 'true' }} isGraphQlTransport: ${{ needs.diff.outputs.isGraphQlTransport == 'true' }} - vercel-deploy: - needs: - - diff - - dprint-format - - license-check - - typos - uses: ./.github/workflows/_vercel_deploy.yml - secrets: inherit - with: - shouldDeployPreview: ${{ github.event_name == 'pull_request' && github.event.pull_request.draft == false }} - isExplorer: ${{ needs.diff.outputs.isExplorer == 'true' }} - isTypescriptSDK: ${{ needs.diff.outputs.isTypescriptSDK == 'true' }} - isAppsBackend: ${{ needs.diff.outputs.isAppsBackend == 'true' }} - isAppsUiKit: ${{ needs.diff.outputs.isAppsUiKit == 'true' }} - isWalletDashboard: ${{ needs.diff.outputs.isWalletDashboard == 'true' }} - ledgernano: if: needs.diff.outputs.isLedgerjs == 'true' && github.event.pull_request.draft == false needs: @@ -172,15 +152,15 @@ jobs: needs: - diff - dprint-format - - license-check - typos + - license-check - docusaurus - docs-lint - turborepo - move-tests - rust + - rosetta - e2e - - vercel-deploy - ledgernano - move-ide steps: diff --git a/.github/workflows/nightly.yml b/.github/workflows/nightly.yml index 3e02cfc233f..2fcbcddd583 100644 --- a/.github/workflows/nightly.yml +++ b/.github/workflows/nightly.yml @@ -81,9 +81,6 @@ jobs: external-tests: uses: ./.github/workflows/_external_rust_tests.yml - move-tests: - uses: ./.github/workflows/_move_tests.yml - deny: uses: ./.github/workflows/_cargo_deny.yml @@ -109,8 +106,5 @@ jobs: - uses: actions/checkout@eef61447b9ff4aafe5dcd4e0bbf5d482be7e7871 # v4.2.1 with: ref: ${{ env.IOTA_REF }} - - uses: taiki-e/install-action@375e0c7f08a66b8c2ba7e7eef31a6f91043a81b0 # v2.44.38 - with: - tool: nextest - name: Run simtest run: scripts/simtest/simtest-run.sh diff --git a/.github/workflows/release_docker.yml b/.github/workflows/release_docker.yml index 9acbe8dcc40..5f8ef7568de 100644 --- a/.github/workflows/release_docker.yml +++ b/.github/workflows/release_docker.yml @@ -87,7 +87,7 @@ jobs: with: context: . file: docker/iota-node/Dockerfile - platforms: linux/amd64 + platforms: linux/amd64,linux/arm64 tags: ${{ steps.meta-node.outputs.tags }} push: true pull: true @@ -156,7 +156,7 @@ jobs: with: context: . file: docker/iota-indexer/Dockerfile - platforms: linux/amd64 + platforms: linux/amd64,linux/arm64 tags: ${{ steps.meta-indexer.outputs.tags }} push: true pull: true @@ -225,7 +225,7 @@ jobs: with: context: . file: docker/iota-tools/Dockerfile - platforms: linux/amd64 + platforms: linux/amd64,linux/arm64 tags: ${{ steps.meta-tools.outputs.tags }} push: true pull: true @@ -294,7 +294,7 @@ jobs: with: context: . file: docker/iota-graphql-rpc/Dockerfile - platforms: linux/amd64 + platforms: linux/amd64,linux/arm64 tags: ${{ steps.meta-tools.outputs.tags }} push: true pull: true diff --git a/.github/workflows/stale.yml b/.github/workflows/stale.yml index f0c82a67fa7..878a13f48e2 100644 --- a/.github/workflows/stale.yml +++ b/.github/workflows/stale.yml @@ -14,11 +14,13 @@ jobs: - uses: actions/stale@28ca1036281a5e5922ead5184a1bbf96e5fc984e # v9.0.0 with: repo-token: ${{ secrets.GITHUB_TOKEN }} - days-before-stale: 60 - days-before-close: -1 + days-before-issue-stale: 180 + days-before-pr-stale: 60 + days-before-issue-close: 14 + days-before-pr-close: 14 + stale-issue-message: "This issue is stale because it has been open 180 days with no activity. Remove stale label or comment or this will be closed in 14 days." + stale-pr-message: "This PR is stale because it has been open 60 days with no activity. Remove stale label or comment or this will be closed in 14 days." + close-issue-message: "This issue was closed because it has been stalled for 14 days with no activity." + close-pr-message: "This PR was closed because it has been stalled for 14 days with no activity." exempt-all-milestones: true exempt-all-assignees: true - stale-issue-message: "This issue is stale because it has been open 60 days with no activity. Remove stale label or comment or this will be closed in 7 days." - stale-pr-message: "This PR is stale because it has been open 60 days with no activity. Remove stale label or comment or this will be closed in 7 days." - close-issue-message: "This issue was closed because it has been stalled for 7 days with no activity." - close-pr-message: "This PR was closed because it has been stalled for 7 days with no activity." diff --git a/.gitignore b/.gitignore index 642158476ae..e58897c6432 100644 --- a/.gitignore +++ b/.gitignore @@ -79,6 +79,6 @@ lcov.info *storybook.log # iota-private-network -docker/iota-private-network/data -docker/iota-private-network/configs/validators/validator* -docker/iota-private-network/configs/genesis/genesis.blob +dev-tools/iota-private-network/data +dev-tools/iota-private-network/configs/validators/validator* +dev-tools/iota-private-network/configs/genesis/genesis.blob diff --git a/.vscode/extensions.json b/.vscode/extensions.json index f29818dd52d..f9b31562ae4 100644 --- a/.vscode/extensions.json +++ b/.vscode/extensions.json @@ -1,6 +1,6 @@ { "recommendations": [ - "move.move-analyzer", + "iotaledger.iota-move", "rust-lang.rust-analyzer", "esbenp.prettier-vscode", "ms-playwright.playwright", diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 2b5e2c8a2b8..89198dfb4f9 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -4,7 +4,7 @@ Thanks for considering making a contribution to the IOTA network or its document ## Contribute to IOTA -See [IOTA Environment Setup](https://github.com/iotaledger/iota/blob/main/docs/content/guides/developer/getting-started/iota-environment.mdx) for approach to submitting code fixes and enhancements. +See [IOTA Environment Setup](https://github.com/iotaledger/iota/blob/develop/docs/content/developer/getting-started/iota-environment.mdx) for approach to submitting code fixes and enhancements. Found a bug or security vulnerability? Create a [GitHub issue](https://github.com/iotaledger/iota/issues/new/choose). diff --git a/Cargo.lock b/Cargo.lock index 550aa462be0..16c2287d34b 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -173,7 +173,7 @@ dependencies = [ [[package]] name = "anemo" version = "0.0.0" -source = "git+https://github.com/mystenlabs/anemo.git?rev=dbb5a074c2d25660525ab5d36d65ff0cb8051949#dbb5a074c2d25660525ab5d36d65ff0cb8051949" +source = "git+https://github.com/mystenlabs/anemo.git?rev=e609f7697ed6169bf0760882a0b6c032a57e4f3b#e609f7697ed6169bf0760882a0b6c032a57e4f3b" dependencies = [ "anyhow", "async-trait", @@ -208,18 +208,18 @@ dependencies = [ [[package]] name = "anemo-build" version = "0.0.0" -source = "git+https://github.com/mystenlabs/anemo.git?rev=dbb5a074c2d25660525ab5d36d65ff0cb8051949#dbb5a074c2d25660525ab5d36d65ff0cb8051949" +source = "git+https://github.com/mystenlabs/anemo.git?rev=e609f7697ed6169bf0760882a0b6c032a57e4f3b#e609f7697ed6169bf0760882a0b6c032a57e4f3b" dependencies = [ "prettyplease", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] [[package]] name = "anemo-cli" version = "0.0.0" -source = "git+https://github.com/mystenlabs/anemo.git?rev=dbb5a074c2d25660525ab5d36d65ff0cb8051949#dbb5a074c2d25660525ab5d36d65ff0cb8051949" +source = "git+https://github.com/mystenlabs/anemo.git?rev=e609f7697ed6169bf0760882a0b6c032a57e4f3b#e609f7697ed6169bf0760882a0b6c032a57e4f3b" dependencies = [ "anemo", "anemo-tower", @@ -235,7 +235,7 @@ dependencies = [ [[package]] name = "anemo-tower" version = "0.0.0" -source = "git+https://github.com/mystenlabs/anemo.git?rev=dbb5a074c2d25660525ab5d36d65ff0cb8051949#dbb5a074c2d25660525ab5d36d65ff0cb8051949" +source = "git+https://github.com/mystenlabs/anemo.git?rev=e609f7697ed6169bf0760882a0b6c032a57e4f3b#e609f7697ed6169bf0760882a0b6c032a57e4f3b" dependencies = [ "anemo", "bytes", @@ -425,7 +425,7 @@ version = "0.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3ed4aa4fe255d0bc6d79373f7e31d2ea147bcf486cba1be5ba7ea85abdb92348" dependencies = [ - "quote 1.0.37", + "quote", "syn 1.0.109", ] @@ -437,8 +437,8 @@ checksum = "7abe79b0e4288889c4574159ab790824d0033b9fdcb2a112a3182fac2e514565" dependencies = [ "num-bigint 0.4.6", "num-traits", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -510,8 +510,8 @@ version = "0.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ae3281bc6d0fd7e549af32b52511e1302185bd688fd3359fa36423346ff682ea" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -800,8 +800,8 @@ version = "0.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "726535892e8eae7e70657b4c8ea93d26b8553afb1ce617caee529ef96d7dee6c" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", "synstructure 0.12.6", ] @@ -812,8 +812,8 @@ version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2777730b2039ac0f95f093556e61b6d26cebed5393ca6f152717777cec3a42ed" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -917,8 +917,8 @@ dependencies = [ "async-graphql-parser", "darling 0.20.10", "proc-macro-crate 1.3.1", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "strum 0.25.0", "syn 2.0.77", "thiserror", @@ -954,8 +954,8 @@ version = "1.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3b43422f69d8ff38f95f1b2bb76517c91589a924d1559a0e935d7c8ce0274c11" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -976,8 +976,8 @@ version = "0.3.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "16e62a023e7c117e27523144c5d2459f4397fcc3cab0085af8e2224f643a0193" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -993,8 +993,8 @@ version = "0.1.83" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "721cae7de5c34fbb2acd27e21e6d2cf7b886dce0c27388d46c4e6c47ea4318dd" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -1042,8 +1042,8 @@ version = "1.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3c87f3f15e7794432337fc718554eaa4dc8f04c9677a950ffe366f20a162ae42" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -1805,8 +1805,8 @@ version = "0.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3deeecb812ca5300b7d3f66f730cc2ebd3511c3d36c691dd79c165d5b19a26e3" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -1831,7 +1831,7 @@ checksum = "230c5f1ca6a325a32553f8640d31ac9b49f2411e901e427570154868b46da4f7" [[package]] name = "bin-version" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "const-str", "git-version", @@ -1859,8 +1859,8 @@ dependencies = [ "lazycell", "peeking_take_while", "prettyplease", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "regex", "rustc-hash 1.1.0", "shlex", @@ -1881,8 +1881,8 @@ dependencies = [ "lazycell", "log", "prettyplease", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "regex", "rustc-hash 1.1.0", "shlex", @@ -2249,8 +2249,8 @@ checksum = "e10ca87c81aaa3a949dbbe2b5e6c2c45dbc94ba4897e45ea31ff9ec5087be3dc" dependencies = [ "cached_proc_macro_types", "darling 0.14.4", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -2461,9 +2461,9 @@ dependencies = [ [[package]] name = "clap" -version = "4.5.18" +version = "4.5.23" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b0956a43b323ac1afaffc053ed5c4b7c1f1800bacd1683c353aabbb752515dd3" +checksum = "3135e7ec2ef7b10c6ed8950f0f792ed96ee093fa088608f1c76e569722700c84" dependencies = [ "clap_builder", "clap_derive", @@ -2471,15 +2471,24 @@ dependencies = [ [[package]] name = "clap_builder" -version = "4.5.18" +version = "4.5.23" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4d72166dd41634086d5803a47eb71ae740e61d84709c36f3c34110173db3961b" +checksum = "30582fc632330df2bd26877bde0c1f4470d57c582bbc070376afcd04d8cb4838" dependencies = [ "anstream", "anstyle", "clap_lex", "strsim 0.11.1", - "terminal_size", + "terminal_size 0.4.1", +] + +[[package]] +name = "clap_complete" +version = "4.5.40" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ac2e663e3e3bed2d32d065a8404024dad306e699a04263ec59919529f803aee9" +dependencies = [ + "clap", ] [[package]] @@ -2489,16 +2498,16 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "4ac6a0c7b1a9e9a5186361f67dfa1b88213572f427fb9ab038efb2bd8c582dab" dependencies = [ "heck 0.5.0", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] [[package]] name = "clap_lex" -version = "0.7.2" +version = "0.7.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1462739cb27611015575c0c11df5df7601141071f07518d56fcc1be504cbec97" +checksum = "f46ad14479a25103f283c0f10005961cf086d8dc42205bb44c46ac563475dca6" [[package]] name = "clipboard-win" @@ -2666,7 +2675,7 @@ dependencies = [ name = "consensus-config" version = "0.1.0" dependencies = [ - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "insta", "iota-network-stack", "rand 0.8.5", @@ -2688,7 +2697,7 @@ dependencies = [ "consensus-config", "dashmap", "enum_dispatch", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "futures", "http 1.1.0", "hyper 1.4.1", @@ -2843,6 +2852,15 @@ version = "0.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6245d59a3e82a7fc217c5828a6692dbc6dfb63a0c8c90495621f7b9d79704a0e" +[[package]] +name = "convert_case" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ec182b0ca2f35d8fc196cf3404988fd8b8c739a4d270ff118a398feb0cbec1ca" +dependencies = [ + "unicode-segmentation", +] + [[package]] name = "core-foundation" version = "0.9.4" @@ -3161,8 +3179,8 @@ version = "0.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f46882e17999c6cc590af592290432be3bce0428cb0d5f8b6715e4dc7b383eb3" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -3179,6 +3197,19 @@ dependencies = [ "zeroize", ] +[[package]] +name = "custom-indexer" +version = "0.1.0" +dependencies = [ + "anyhow", + "async-trait", + "iota-data-ingestion-core", + "iota-types", + "prometheus", + "tokio", + "tokio-util 0.7.12", +] + [[package]] name = "cynic" version = "3.7.3" @@ -3205,8 +3236,8 @@ dependencies = [ "darling 0.20.10", "once_cell", "ouroboros 0.18.4", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "strsim 0.10.0", "syn 2.0.77", "thiserror", @@ -3231,7 +3262,7 @@ checksum = "25a69ecdf4aa110fed1c0c8de290bc8ccb2835388733cf2f418f0abdf6ff3899" dependencies = [ "cynic-codegen", "darling 0.20.10", - "quote 1.0.37", + "quote", "syn 2.0.77", ] @@ -3263,8 +3294,8 @@ checksum = "109c1ca6e6b7f82cc233a97004ea8ed7ca123a9af07a8230878fcfda9b158bf0" dependencies = [ "fnv", "ident_case", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "strsim 0.10.0", "syn 1.0.109", ] @@ -3277,8 +3308,8 @@ checksum = "95133861a8032aaea082871032f5815eb9e98cef03fa916ab4500513994df9e5" dependencies = [ "fnv", "ident_case", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "strsim 0.11.1", "syn 2.0.77", ] @@ -3290,7 +3321,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a4aab4dbc9f7611d8b55048a3a16d2d010c2c8334e46304b40ac1cc14bf3b48e" dependencies = [ "darling_core 0.14.4", - "quote 1.0.37", + "quote", "syn 1.0.109", ] @@ -3301,7 +3332,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d336a2a514f6ccccaa3e09b02d41d35330c07ddf03a62165fcec10bb561c7806" dependencies = [ "darling_core 0.20.10", - "quote 1.0.37", + "quote", "syn 2.0.77", ] @@ -3421,8 +3452,8 @@ version = "2.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "fcc3dd5e9e9c0b295d6e1e4d811fb6f157d5ffd784b8d202fc62eac8035a770b" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -3432,8 +3463,8 @@ version = "0.1.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e79116f119dd1dba1abf1f3405f03b9b0e79a27a3883864bfebded8a3dc768cd" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -3443,8 +3474,8 @@ version = "1.3.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "67e77553c4162a157adbf834ebae5b415acbecbeafc7a74b0e886657506a7611" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -3454,9 +3485,9 @@ version = "0.99.18" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "5f33878137e4dafd7fa914ad4e259e18a4e8e532b9617a2d0150262bf53abfce" dependencies = [ - "convert_case", - "proc-macro2 1.0.86", - "quote 1.0.37", + "convert_case 0.4.0", + "proc-macro2", + "quote", "rustc_version", "syn 2.0.77", ] @@ -3476,10 +3507,10 @@ version = "1.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "cb7330aeadfbe296029522e6c40f315320aba36fc43a5b3632f3795348f3bd22" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", - "unicode-xid 0.2.6", + "unicode-xid", ] [[package]] @@ -3509,8 +3540,8 @@ checksum = "e7f2c3de51e2ba6bf2a648285696137aaf0f5f487bcbea93972fe8a364e131a4" dependencies = [ "diesel_table_macro_syntax", "dsl_auto_type", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -3650,8 +3681,8 @@ version = "0.2.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "97369cbbc041bc366949bc74d34658d6cda5621039731c6310521892a3a20ae0" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -3669,14 +3700,14 @@ checksum = "fea41bba32d969b513997752735605054bc0dfa92b4c56bf1189f2e174be7a10" [[package]] name = "docs-examples" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "bcs", "bip32", "iota-keys", "iota-move-build", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "move-binary-format", "move-core-types", "serde_json", @@ -3705,8 +3736,8 @@ dependencies = [ "darling 0.20.10", "either", "heck 0.5.0", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -3943,8 +3974,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a1e6a265c649f3f5979b601d26f1d05ada116434c87741c9493cb56218f76cbc" dependencies = [ "heck 0.5.0", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -3962,8 +3993,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "aa18ce2bc66555b3218614519ac839ddb759a7d6720732f979ef8d13be147ecd" dependencies = [ "once_cell", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -3982,8 +4013,8 @@ version = "0.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3bf679796c0322556351f287a51b49e48f7c4986e727b5dd78c972d30e2e16cc" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -4149,8 +4180,8 @@ dependencies = [ "ethers-etherscan", "eyre", "prettyplease", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "regex", "reqwest 0.11.27", "serde", @@ -4170,8 +4201,8 @@ dependencies = [ "const-hex", "ethers-contract-abigen", "ethers-core", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "serde_json", "syn 2.0.77", ] @@ -4203,7 +4234,7 @@ dependencies = [ "tempfile", "thiserror", "tiny-keccak", - "unicode-xid 0.2.6", + "unicode-xid", ] [[package]] @@ -4384,6 +4415,58 @@ dependencies = [ "ascii_utils", ] +[[package]] +name = "fastcrypto" +version = "0.1.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4f9ea2139faa96e4d713803b7777b8fd2d5c245858b1644c08d3220f407149c6" +dependencies = [ + "ark-ec", + "ark-ff", + "ark-secp256r1", + "ark-serialize", + "auto_ops", + "base64ct", + "bech32", + "bincode", + "blake2", + "blst", + "bs58 0.4.0", + "curve25519-dalek-ng", + "derive_more 0.99.18", + "digest 0.10.7", + "ecdsa 0.16.9", + "ed25519-consensus", + "elliptic-curve 0.13.8", + "eyre", + "fastcrypto-derive 0.1.3 (registry+https://github.com/rust-lang/crates.io-index)", + "generic-array", + "hex", + "hex-literal 0.4.1", + "hkdf", + "lazy_static", + "num-bigint 0.4.6", + "once_cell", + "p256 0.13.2", + "rand 0.8.5", + "readonly", + "rfc6979 0.4.0", + "rsa 0.8.2", + "schemars", + "secp256k1", + "serde", + "serde_json", + "serde_with 2.3.3", + "sha2 0.10.8", + "sha3 0.10.8", + "signature 2.2.0", + "static_assertions", + "thiserror", + "tokio", + "typenum", + "zeroize", +] + [[package]] name = "fastcrypto" version = "0.1.8" @@ -4410,7 +4493,7 @@ dependencies = [ "ecdsa 0.16.9", "ed25519-consensus", "elliptic-curve 0.13.8", - "fastcrypto-derive", + "fastcrypto-derive 0.1.3 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "generic-array", "hex", "hex-literal 0.4.1", @@ -4427,7 +4510,7 @@ dependencies = [ "secp256k1", "serde", "serde_json", - "serde_with", + "serde_with 3.9.0", "sha2 0.10.8", "sha3 0.10.8", "signature 2.2.0", @@ -4438,12 +4521,24 @@ dependencies = [ "zeroize", ] +[[package]] +name = "fastcrypto-derive" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4c0c2af2157f416cb885e11d36cd0de2753f6d5384752d364075c835f5f8f891" +dependencies = [ + "convert_case 0.6.0", + "proc-macro2", + "quote", + "syn 1.0.109", +] + [[package]] name = "fastcrypto-derive" version = "0.1.3" source = "git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597#5f2c63266a065996d53f98156f0412782b468597" dependencies = [ - "quote 1.0.37", + "quote", "syn 1.0.109", ] @@ -4454,7 +4549,7 @@ source = "git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f9 dependencies = [ "bcs", "digest 0.10.7", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "hex", "itertools 0.10.5", "rand 0.8.5", @@ -4472,7 +4567,7 @@ version = "0.1.0" source = "git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597#5f2c63266a065996d53f98156f0412782b468597" dependencies = [ "bcs", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "lazy_static", "num-bigint 0.4.6", "num-integer", @@ -4499,7 +4594,7 @@ dependencies = [ "blst", "byte-slice-cast", "derive_more 0.99.18", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "ff 0.13.0", "im", "itertools 0.12.1", @@ -4575,8 +4670,8 @@ dependencies = [ "num-bigint 0.3.3", "num-integer", "num-traits", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -4798,8 +4893,8 @@ version = "0.3.31" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "162ee34ebcb7c64a8abebc059ce0fee27c2262618d7b60ed8faf72fef13c3650" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -4918,8 +5013,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f636605b743120a8d32ed92fc27b6cde1a769f8f936c065151eb66f88ded513c" dependencies = [ "proc-macro-error2", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -4954,8 +5049,8 @@ version = "0.3.9" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "53010ccb100b96a67bc32c0175f0ed1426b31b655d562898e57325f81c023ac0" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -5613,8 +5708,8 @@ version = "1.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1ec89e9337638ecdc08744df490b221a7399bf8d164eb52a665454e60e075ad6" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -5716,8 +5811,8 @@ version = "0.2.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "11d7a9f6330b71fea57921c9b61c47ee6e84f72d394754eff6163ae67e7395eb" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -5843,7 +5938,7 @@ checksum = "8bb03732005da905c88227371639bf1ad885cc712789c011c31c5fb3ab3ccf02" [[package]] name = "iota" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anemo", "anyhow", @@ -5858,12 +5953,13 @@ dependencies = [ "bip32", "camino", "clap", + "clap_complete", "colored", "csv", "datatest-stable", "diesel", "expect-test", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "fastcrypto-zkp", "fs_extra", "futures", @@ -5888,7 +5984,7 @@ dependencies = [ "iota-package-management", "iota-protocol-config", "iota-replay", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-simulator", "iota-source-validation", "iota-swarm", @@ -5967,7 +6063,7 @@ dependencies = [ [[package]] name = "iota-adapter-transactional-tests" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "datatest-stable", "iota-transactional-test-runner", @@ -5975,7 +6071,7 @@ dependencies = [ [[package]] name = "iota-analytics-indexer" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "arrow", @@ -5989,7 +6085,7 @@ dependencies = [ "clap", "csv", "eyre", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "gcp-bigquery-client", "iota-analytics-indexer-derive", "iota-config", @@ -6026,22 +6122,22 @@ dependencies = [ [[package]] name = "iota-analytics-indexer-derive" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] [[package]] name = "iota-archival" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "byteorder", "bytes", "ed25519-consensus", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "futures", "indicatif", "iota-config", @@ -6067,7 +6163,7 @@ dependencies = [ [[package]] name = "iota-authority-aggregation" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "futures", "iota-metrics", @@ -6078,7 +6174,7 @@ dependencies = [ [[package]] name = "iota-aws-orchestrator" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "async-trait", "aws-config", @@ -6109,7 +6205,7 @@ dependencies = [ [[package]] name = "iota-benchmark" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", @@ -6131,7 +6227,7 @@ dependencies = [ "iota-metrics", "iota-network", "iota-protocol-config", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-simulator", "iota-storage", "iota-surfer", @@ -6159,7 +6255,7 @@ dependencies = [ [[package]] name = "iota-bridge" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "arc-swap", @@ -6172,7 +6268,7 @@ dependencies = [ "enum_dispatch", "ethers", "eyre", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "futures", "hex-literal 0.3.4", "iota-authority-aggregation", @@ -6181,7 +6277,7 @@ dependencies = [ "iota-json-rpc-types", "iota-keys", "iota-metrics", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-test-transaction-builder", "iota-types", "lru 0.12.4", @@ -6194,7 +6290,7 @@ dependencies = [ "reqwest 0.12.7", "serde", "serde_json", - "serde_with", + "serde_with 3.9.0", "shared-crypto", "tap", "telemetry-subscribers", @@ -6208,24 +6304,24 @@ dependencies = [ [[package]] name = "iota-bridge-cli" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "clap", "ethers", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "futures", "iota-bridge", "iota-config", "iota-json-rpc-types", "iota-keys", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-types", "move-core-types", "reqwest 0.12.7", "serde", "serde_json", - "serde_with", + "serde_with 3.9.0", "shared-crypto", "telemetry-subscribers", "tokio", @@ -6234,7 +6330,7 @@ dependencies = [ [[package]] name = "iota-bridge-indexer" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", @@ -6250,7 +6346,7 @@ dependencies = [ "iota-indexer-builder", "iota-json-rpc-types", "iota-metrics", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-test-transaction-builder", "iota-types", "prometheus", @@ -6264,14 +6360,14 @@ dependencies = [ [[package]] name = "iota-cluster-test" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", "clap", "derive_more 1.0.0", "diesel", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "futures", "iota-config", "iota-core", @@ -6282,7 +6378,7 @@ dependencies = [ "iota-json", "iota-json-rpc-types", "iota-keys", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-swarm", "iota-swarm-config", "iota-test-transaction-builder", @@ -6304,7 +6400,7 @@ dependencies = [ [[package]] name = "iota-common" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "futures", "parking_lot 0.12.3", @@ -6313,7 +6409,7 @@ dependencies = [ [[package]] name = "iota-config" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anemo", "anyhow", @@ -6322,7 +6418,7 @@ dependencies = [ "consensus-config", "csv", "dirs 5.0.1", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "iota-genesis-common", "iota-keys", "iota-protocol-config", @@ -6333,14 +6429,14 @@ dependencies = [ "rand 0.8.5", "reqwest 0.12.7", "serde", - "serde_with", + "serde_with 3.9.0", "serde_yaml", "tracing", ] [[package]] name = "iota-core" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anemo", "anyhow", @@ -6361,7 +6457,7 @@ dependencies = [ "enum_dispatch", "expect-test", "eyre", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "fastcrypto-tbls", "fastcrypto-zkp", "fs_extra", @@ -6419,7 +6515,7 @@ dependencies = [ "serde", "serde-reflection", "serde_json", - "serde_with", + "serde_with 3.9.0", "serde_yaml", "shared-crypto", "signature 1.6.4", @@ -6442,7 +6538,7 @@ dependencies = [ [[package]] name = "iota-cost" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "bcs", @@ -6497,7 +6593,7 @@ dependencies = [ [[package]] name = "iota-data-ingestion" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", @@ -6525,13 +6621,14 @@ dependencies = [ "telemetry-subscribers", "tempfile", "tokio", + "tokio-util 0.7.12", "tracing", "url", ] [[package]] name = "iota-data-ingestion-core" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", @@ -6552,13 +6649,14 @@ dependencies = [ "tap", "tempfile", "tokio", + "tokio-util 0.7.12", "tracing", "url", ] [[package]] name = "iota-e2e-tests" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", @@ -6566,7 +6664,7 @@ dependencies = [ "bip32", "clap", "coset", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "fastcrypto-zkp", "fs_extra", "futures", @@ -6589,7 +6687,7 @@ dependencies = [ "iota-node", "iota-protocol-config", "iota-rest-api", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-simulator", "iota-storage", "iota-swarm", @@ -6620,7 +6718,7 @@ dependencies = [ [[package]] name = "iota-enum-compat-util" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "serde_yaml", ] @@ -6658,7 +6756,7 @@ dependencies = [ [[package]] name = "iota-faucet" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-recursion", @@ -6672,7 +6770,7 @@ dependencies = [ "iota-json-rpc-types", "iota-keys", "iota-metrics", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-types", "parking_lot 0.12.3", "prometheus", @@ -6696,7 +6794,7 @@ dependencies = [ [[package]] name = "iota-framework" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "bcs", @@ -6717,7 +6815,7 @@ dependencies = [ [[package]] name = "iota-framework-snapshot" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "bcs", @@ -6732,7 +6830,7 @@ dependencies = [ [[package]] name = "iota-framework-tests" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "datatest-stable", "iota-adapter-latest", @@ -6752,7 +6850,7 @@ dependencies = [ [[package]] name = "iota-genesis-builder" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "bcs", @@ -6760,7 +6858,7 @@ dependencies = [ "camino", "clap", "csv", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "flate2", "fs_extra", "iota-adapter-latest", @@ -6793,7 +6891,7 @@ dependencies = [ "schemars", "serde", "serde_json", - "serde_with", + "serde_with 3.9.0", "serde_yaml", "shared-crypto", "tempfile", @@ -6805,7 +6903,7 @@ dependencies = [ [[package]] name = "iota-genesis-common" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "iota-execution", "iota-protocol-config", @@ -6815,15 +6913,15 @@ dependencies = [ [[package]] name = "iota-graphql-config" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ - "quote 1.0.37", + "quote", "syn 1.0.109", ] [[package]] name = "iota-graphql-e2e-tests" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "datatest-stable", "iota-graphql-rpc", @@ -6834,7 +6932,7 @@ dependencies = [ [[package]] name = "iota-graphql-rpc" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-graphql", @@ -6852,7 +6950,7 @@ dependencies = [ "downcast", "either", "expect-test", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "fastcrypto-zkp", "futures", "hex", @@ -6873,7 +6971,7 @@ dependencies = [ "iota-package-resolver", "iota-protocol-config", "iota-rest-api", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-swarm-config", "iota-test-transaction-builder", "iota-types", @@ -6891,7 +6989,7 @@ dependencies = [ "reqwest 0.12.7", "serde", "serde_json", - "serde_with", + "serde_with 3.9.0", "serde_yaml", "serial_test", "shared-crypto", @@ -6913,7 +7011,7 @@ dependencies = [ [[package]] name = "iota-graphql-rpc-client" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-graphql", @@ -6928,14 +7026,14 @@ dependencies = [ [[package]] name = "iota-graphql-rpc-headers" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "axum", ] [[package]] name = "iota-indexer" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", @@ -6948,7 +7046,7 @@ dependencies = [ "diesel", "diesel_migrations", "downcast", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "futures", "iota-config", "iota-data-ingestion-core", @@ -6964,7 +7062,7 @@ dependencies = [ "iota-package-resolver", "iota-protocol-config", "iota-rest-api", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-swarm-config", "iota-test-transaction-builder", "iota-transaction-builder", @@ -6982,7 +7080,7 @@ dependencies = [ "secrecy", "serde", "serde_json", - "serde_with", + "serde_with 3.9.0", "simulacrum", "tap", "telemetry-subscribers", @@ -6997,7 +7095,7 @@ dependencies = [ [[package]] name = "iota-indexer-builder" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", @@ -7007,16 +7105,17 @@ dependencies = [ "prometheus", "telemetry-subscribers", "tokio", + "tokio-util 0.7.12", "tracing", ] [[package]] name = "iota-json" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "bcs", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "iota-framework", "iota-move-build", "iota-types", @@ -7031,7 +7130,7 @@ dependencies = [ [[package]] name = "iota-json-rpc" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "arc-swap", @@ -7041,7 +7140,7 @@ dependencies = [ "cached", "expect-test", "eyre", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "futures", "http-body 1.0.1", "hyper 1.4.1", @@ -7085,10 +7184,10 @@ dependencies = [ [[package]] name = "iota-json-rpc-api" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "iota-json", "iota-json-rpc-types", "iota-metrics", @@ -7104,7 +7203,7 @@ dependencies = [ [[package]] name = "iota-json-rpc-tests" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", @@ -7123,7 +7222,7 @@ dependencies = [ "iota-open-rpc", "iota-open-rpc-macros", "iota-protocol-config", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-simulator", "iota-swarm-config", "iota-test-transaction-builder", @@ -7142,13 +7241,13 @@ dependencies = [ [[package]] name = "iota-json-rpc-types" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "bcs", "colored", "enum_dispatch", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "iota-enum-compat-util", "iota-json", "iota-macros", @@ -7164,18 +7263,18 @@ dependencies = [ "schemars", "serde", "serde_json", - "serde_with", + "serde_with 3.9.0", "tabled", "tracing", ] [[package]] name = "iota-keys" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "bip32", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "iota-types", "rand 0.8.5", "regex", @@ -7190,7 +7289,7 @@ dependencies = [ [[package]] name = "iota-light-client" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", @@ -7202,7 +7301,7 @@ dependencies = [ "iota-json-rpc-types", "iota-package-resolver", "iota-rest-api", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-types", "move-binary-format", "move-core-types", @@ -7214,7 +7313,7 @@ dependencies = [ [[package]] name = "iota-macros" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "futures", "iota-proc-macros", @@ -7224,7 +7323,7 @@ dependencies = [ [[package]] name = "iota-metric-checker" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "backoff", @@ -7245,7 +7344,7 @@ dependencies = [ [[package]] name = "iota-metrics" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anemo", "anemo-tower", @@ -7268,7 +7367,7 @@ dependencies = [ [[package]] name = "iota-move" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "better_any", @@ -7306,10 +7405,10 @@ dependencies = [ [[package]] name = "iota-move-build" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "iota-package-management", "iota-protocol-config", "iota-types", @@ -7329,7 +7428,7 @@ dependencies = [ [[package]] name = "iota-move-lsp" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "bin-version", "clap", @@ -7342,7 +7441,7 @@ version = "0.1.0" dependencies = [ "bcs", "better_any", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "fastcrypto-vdf", "fastcrypto-zkp", "indexmap 2.5.0", @@ -7360,7 +7459,7 @@ dependencies = [ [[package]] name = "iota-network" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anemo", "anemo-build", @@ -7371,7 +7470,7 @@ dependencies = [ "bytes", "dashmap", "ed25519-consensus", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "fastcrypto-tbls", "futures", "governor", @@ -7398,7 +7497,7 @@ dependencies = [ [[package]] name = "iota-network-stack" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anemo", "bcs", @@ -7421,7 +7520,7 @@ dependencies = [ [[package]] name = "iota-node" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anemo", "anemo-tower", @@ -7433,7 +7532,7 @@ dependencies = [ "bin-version", "clap", "const-str", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "fastcrypto-zkp", "futures", "git-version", @@ -7471,12 +7570,12 @@ dependencies = [ [[package]] name = "iota-open-rpc" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "bcs", "clap", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "iota-json", "iota-json-rpc", "iota-json-rpc-api", @@ -7495,25 +7594,25 @@ dependencies = [ [[package]] name = "iota-open-rpc-macros" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "derive-syn-parse", "itertools 0.13.0", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", "unescape", ] [[package]] name = "iota-package-dump" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "bcs", "cynic", "cynic-codegen", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "iota-types", "move-core-types", "reqwest 0.12.7", @@ -7524,11 +7623,11 @@ dependencies = [ [[package]] name = "iota-package-management" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "iota-json-rpc-types", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-types", "move-core-types", "move-package", @@ -7539,7 +7638,7 @@ dependencies = [ [[package]] name = "iota-package-resolver" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "async-trait", "bcs", @@ -7562,17 +7661,17 @@ dependencies = [ [[package]] name = "iota-proc-macros" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "msim-macros", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] [[package]] name = "iota-protocol-config" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "clap", "insta", @@ -7580,22 +7679,66 @@ dependencies = [ "move-vm-config", "schemars", "serde", - "serde_with", + "serde_with 3.9.0", "tracing", ] [[package]] name = "iota-protocol-config-macros" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] +[[package]] +name = "iota-proxy" +version = "0.9.0-alpha" +dependencies = [ + "anyhow", + "axum", + "axum-extra", + "axum-server", + "bytes", + "clap", + "const-str", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", + "futures", + "git-version", + "hex", + "hyper 1.4.1", + "iota-metrics", + "iota-tls", + "iota-types", + "itertools 0.13.0", + "mime", + "multiaddr", + "once_cell", + "prometheus", + "prost", + "prost-build", + "protobuf", + "rand 0.8.5", + "reqwest 0.12.7", + "rustls 0.23.18", + "rustls-pemfile 2.1.3", + "serde", + "serde_json", + "serde_with 3.9.0", + "serde_yaml", + "snap", + "telemetry-subscribers", + "tokio", + "tower 0.4.13", + "tower-http", + "tracing", + "url", +] + [[package]] name = "iota-replay" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-recursion", @@ -7611,7 +7754,7 @@ dependencies = [ "iota-json-rpc-api", "iota-json-rpc-types", "iota-protocol-config", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-storage", "iota-transaction-checks", "iota-types", @@ -7627,7 +7770,7 @@ dependencies = [ "regex", "serde", "serde_json", - "serde_with", + "serde_with 3.9.0", "serde_yaml", "shared-crypto", "shellexpand", @@ -7642,14 +7785,14 @@ dependencies = [ [[package]] name = "iota-rest-api" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", "axum", "bcs", "diffy", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "iota-network-stack", "iota-protocol-config", "iota-rust-sdk", @@ -7663,7 +7806,7 @@ dependencies = [ "schemars", "serde", "serde_json", - "serde_with", + "serde_with 3.9.0", "serde_yaml", "tap", "thiserror", @@ -7673,7 +7816,7 @@ dependencies = [ [[package]] name = "iota-rosetta" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", @@ -7682,7 +7825,7 @@ dependencies = [ "bcs", "clap", "eyre", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "futures", "hyper 1.4.1", "iota-config", @@ -7691,7 +7834,7 @@ dependencies = [ "iota-metrics", "iota-move-build", "iota-node", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-swarm-config", "iota-types", "move-core-types", @@ -7715,7 +7858,7 @@ dependencies = [ [[package]] name = "iota-rpc-loadgen" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", @@ -7725,7 +7868,7 @@ dependencies = [ "futures", "iota-json-rpc-types", "iota-keys", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-types", "itertools 0.13.0", "serde", @@ -7741,7 +7884,7 @@ dependencies = [ [[package]] name = "iota-rust-sdk" version = "0.0.0" -source = "git+https://github.com/iotaledger/iota-rust-sdk.git?rev=d605da95029e74376f0f39a95526bb1a5c0ebd7a#d605da95029e74376f0f39a95526bb1a5c0ebd7a" +source = "git+https://github.com/iotaledger/iota-rust-sdk.git?rev=2ba6b293bdede769a1d9b4d1aecaede2ff7682dd#2ba6b293bdede769a1d9b4d1aecaede2ff7682dd" dependencies = [ "base64ct", "bcs", @@ -7754,13 +7897,13 @@ dependencies = [ "serde", "serde_derive", "serde_json", - "serde_with", + "serde_with 3.9.0", "winnow 0.6.20", ] [[package]] name = "iota-sdk" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", @@ -7769,7 +7912,7 @@ dependencies = [ "clap", "colored", "dirs 5.0.1", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "futures", "futures-core", "getset", @@ -7788,7 +7931,7 @@ dependencies = [ "rustls 0.23.18", "serde", "serde_json", - "serde_with", + "serde_with 3.9.0", "shared-crypto", "tempfile", "thiserror", @@ -7838,12 +7981,12 @@ dependencies = [ [[package]] name = "iota-simulator" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anemo", "anemo-tower", "bcs", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "iota-framework", "iota-move-build", "iota-types", @@ -7860,7 +8003,7 @@ dependencies = [ [[package]] name = "iota-single-node-benchmark" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "async-trait", "bcs", @@ -7895,13 +8038,13 @@ dependencies = [ [[package]] name = "iota-snapshot" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "bcs", "byteorder", "bytes", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "futures", "indicatif", "integer-encoding", @@ -7923,7 +8066,7 @@ dependencies = [ [[package]] name = "iota-source-validation" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "colored", @@ -7933,7 +8076,7 @@ dependencies = [ "iota-json-rpc-types", "iota-move-build", "iota-package-management", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-test-transaction-builder", "iota-types", "move-binary-format", @@ -7955,7 +8098,7 @@ dependencies = [ [[package]] name = "iota-source-validation-service" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "axum", @@ -7971,7 +8114,7 @@ dependencies = [ "iota-metrics", "iota-move", "iota-move-build", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-source-validation", "jsonrpsee", "move-compiler", @@ -7994,7 +8137,7 @@ dependencies = [ [[package]] name = "iota-storage" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", @@ -8007,7 +8150,7 @@ dependencies = [ "chrono", "clap", "eyre", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "futures", "hyper 1.4.1", "hyper-rustls 0.27.3", @@ -8048,7 +8191,7 @@ dependencies = [ [[package]] name = "iota-surfer" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "async-trait", "bcs", @@ -8077,7 +8220,7 @@ dependencies = [ [[package]] name = "iota-swarm" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "futures", @@ -8102,12 +8245,12 @@ dependencies = [ [[package]] name = "iota-swarm-config" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anemo", "anyhow", "bcs", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "insta", "iota-config", "iota-execution", @@ -8120,7 +8263,7 @@ dependencies = [ "prometheus", "rand 0.8.5", "serde", - "serde_with", + "serde_with 3.9.0", "serde_yaml", "shared-crypto", "tempfile", @@ -8129,12 +8272,12 @@ dependencies = [ [[package]] name = "iota-test-transaction-builder" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "bcs", "iota-genesis-builder", "iota-move-build", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-types", "move-core-types", "shared-crypto", @@ -8142,13 +8285,13 @@ dependencies = [ [[package]] name = "iota-tls" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "axum", "axum-server", "ed25519 2.2.3", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "pkcs8 0.10.2", "rand 0.8.5", "rcgen", @@ -8163,7 +8306,7 @@ dependencies = [ [[package]] name = "iota-tool" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anemo", "anemo-cli", @@ -8174,7 +8317,7 @@ dependencies = [ "colored", "comfy-table", "eyre", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "futures", "hex", "indicatif", @@ -8185,7 +8328,7 @@ dependencies = [ "iota-package-dump", "iota-protocol-config", "iota-replay", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-snapshot", "iota-storage", "iota-types", @@ -8208,7 +8351,7 @@ dependencies = [ [[package]] name = "iota-transaction-builder" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", @@ -8224,7 +8367,7 @@ dependencies = [ [[package]] name = "iota-transaction-checks" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "fastcrypto-zkp", "iota-config", @@ -8238,7 +8381,7 @@ dependencies = [ [[package]] name = "iota-transactional-test-runner" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", @@ -8247,7 +8390,7 @@ dependencies = [ "clap", "criterion", "eyre", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "futures", "iota-config", "iota-core", @@ -8287,7 +8430,7 @@ dependencies = [ [[package]] name = "iota-types" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anemo", "anyhow", @@ -8304,7 +8447,7 @@ dependencies = [ "enum_dispatch", "expect-test", "eyre", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "fastcrypto-tbls", "fastcrypto-zkp", "hex", @@ -8350,7 +8493,7 @@ dependencies = [ "serde", "serde-name", "serde_json", - "serde_with", + "serde_with 3.9.0", "serde_yaml", "shared-crypto", "signature 1.6.4", @@ -8368,7 +8511,7 @@ dependencies = [ [[package]] name = "iota-upgrade-compatibility-transactional-tests" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "datatest-stable", @@ -8382,11 +8525,11 @@ dependencies = [ [[package]] name = "iota-util-mem" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "cfg-if", "ed25519-consensus", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "fastcrypto-tbls", "hashbrown 0.12.3", "impl-trait-for-tuples", @@ -8400,9 +8543,9 @@ dependencies = [ [[package]] name = "iota-util-mem-derive" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ - "proc-macro2 1.0.86", + "proc-macro2", "syn 1.0.109", "synstructure 0.12.6", ] @@ -8424,7 +8567,7 @@ dependencies = [ [[package]] name = "iota-verifier-transactional-tests" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "datatest-stable", "iota-transactional-test-runner", @@ -8712,8 +8855,8 @@ checksum = "fc660a9389e2748e794a40673a4155d501f32db667757cdb80edeff0306b489b" dependencies = [ "heck 0.5.0", "proc-macro-crate 3.2.0", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -8870,7 +9013,7 @@ dependencies = [ "string_cache", "term", "tiny-keccak", - "unicode-xid 0.2.6", + "unicode-xid", "walkdir", ] @@ -9005,7 +9148,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "4979f22fdb869068da03c9f7528f8297c6fd2606bc3a4affe42e6a823fdb8da4" dependencies = [ "cfg-if", - "windows-targets 0.48.5", + "windows-targets 0.52.6", ] [[package]] @@ -9150,8 +9293,8 @@ dependencies = [ "beef", "fnv", "lazy_static", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "regex-syntax 0.8.4", "syn 2.0.77", ] @@ -9318,7 +9461,7 @@ dependencies = [ "supports-color", "supports-hyperlinks", "supports-unicode", - "terminal_size", + "terminal_size 0.3.0", "textwrap", "thiserror", "unicode-width", @@ -9330,8 +9473,8 @@ version = "7.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "dcf09caffaac8068c346b6df2a7fc27a177fd20b39421a39ce0a211bde679a6c" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -9352,8 +9495,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ffb161cc72176cb37aa47f1fc520d3ef02263d67d661f44f05d05a079e1237fd" dependencies = [ "migrations_internals", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", ] [[package]] @@ -9454,8 +9597,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "22ce75669015c4f47b289fd4d4f56e894e4c96003ffdf3ac51313126f94c6cbb" dependencies = [ "cfg-if", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -9890,7 +10033,7 @@ dependencies = [ name = "move-proc-macros" version = "0.1.0" dependencies = [ - "quote 1.0.37", + "quote", "syn 2.0.77", ] @@ -10127,8 +10270,8 @@ version = "0.1.0" source = "git+https://github.com/iotaledger/iota-sim.git?rev=f16ef50ba7d874fe1f0960f248f6c651a634d6a5#f16ef50ba7d874fe1f0960f248f6c651a634d6a5" dependencies = [ "darling 0.14.4", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -10198,8 +10341,8 @@ checksum = "fc076939022111618a5026d3be019fd8b366e76314538ff9a1b59ffbcbf98bcd" dependencies = [ "proc-macro-crate 1.3.1", "proc-macro-error", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", "synstructure 0.12.6", ] @@ -10263,8 +10406,8 @@ version = "0.2.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "254a5372af8fc138e36684761d3c0cdb758a4410e938babcff1c860ce14ddbfc" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -10594,8 +10737,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "af1844ef2428cc3e1cb900be36181049ef3d3193c63e43026cfe202983b27a56" dependencies = [ "proc-macro-crate 3.2.0", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -10731,8 +10874,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "003b2be5c6c53c1cfeb0a238b8a1c3915cd410feb684457a36c10038f764bb1c" dependencies = [ "bytes", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -10903,8 +11046,8 @@ checksum = "ec4c6225c69b4ca778c0aea097321a64c421cf4577b331c61b229267edabb6f8" dependencies = [ "heck 0.4.1", "proc-macro-error", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -10916,9 +11059,9 @@ checksum = "39b0deead1528fd0e5947a8546a9642a9777c25f6e1e26f34c97b204bbb465bd" dependencies = [ "heck 0.4.1", "itertools 0.12.1", - "proc-macro2 1.0.86", + "proc-macro2", "proc-macro2-diagnostics", - "quote 1.0.37", + "quote", "syn 2.0.77", ] @@ -11015,8 +11158,8 @@ checksum = "e9567693dd2f9a4339cb0a54adfcc0cb431c0ac88b2e46c6ddfb5f5d11a1cc4f" dependencies = [ "proc-macro-crate 1.3.1", "proc-macro-error", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -11075,8 +11218,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1557010476e0595c9b568d16dcfb81b93cdeb157612726f5170d31aa707bed27" dependencies = [ "proc-macro-crate 1.3.1", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -11087,8 +11230,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d830939c76d294956402033aee57a6da7b438f2294eb94864c37b0569053a42c" dependencies = [ "proc-macro-crate 3.2.0", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -11367,8 +11510,8 @@ checksum = "94429506bde1ca69d1b5601962c73f4172ab4726571a59ea95931218cb0e930e" dependencies = [ "pest", "pest_meta", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -11441,8 +11584,8 @@ checksum = "3444646e286606587e49f3bcf1679b8cef1dc2c5ecc29ddacaffc305180d464b" dependencies = [ "phf_generator", "phf_shared 0.11.2", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -11479,8 +11622,8 @@ version = "1.1.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2f38a4412a78282e09a2cf38d195ea5420d15ba0602cb375210efbc877243965" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -11738,7 +11881,7 @@ version = "0.2.22" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "479cf940fbbb3426c32c5d5176f62ad57549a0bb84773423ba8be9d089f5faba" dependencies = [ - "proc-macro2 1.0.86", + "proc-macro2", "syn 2.0.77", ] @@ -11817,8 +11960,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "da25490ff9892aab3fcf7c36f08cfb902dd3e71ca0f9f9517bea02a73a5ce38c" dependencies = [ "proc-macro-error-attr", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", "version_check", ] @@ -11829,8 +11972,8 @@ version = "1.0.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a1be40180e52ecc98ad80b184934baf3d0d29f979574e439af5a55274b35f869" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "version_check", ] @@ -11840,8 +11983,8 @@ version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "96de42df36bb9bba5542fe9f1a054b8cc87e172759a1868aa05c1f3acc89dfc5" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", ] [[package]] @@ -11851,20 +11994,11 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "11ec05c52be0a07b08061f7dd003e7d7092e0472bc731b4af7bb1ef876109802" dependencies = [ "proc-macro-error-attr2", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] -[[package]] -name = "proc-macro2" -version = "0.4.30" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cf3d2011ab5c909338f7887f4fc896d35932e29146c12c8d01da6b22a80ba759" -dependencies = [ - "unicode-xid 0.1.0", -] - [[package]] name = "proc-macro2" version = "1.0.86" @@ -11880,8 +12014,8 @@ version = "0.10.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "af066a9c399a26e020ada66a034357a868728e72cd426f3adcd35f80d88d88c8" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", "version_check", "yansi 1.0.1", @@ -11904,7 +12038,7 @@ dependencies = [ [[package]] name = "prometheus-closure-metric" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "prometheus", @@ -11959,13 +12093,13 @@ dependencies = [ [[package]] name = "proptest-derive" -version = "0.3.0" +version = "0.5.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "90b46295382dc76166cb7cf2bb4a97952464e4b7ed5a43e6cd34e1fec3349ddc" +checksum = "4ee1c9ac207483d5e7db4940700de86a9aae46ef90c48b57f99fe7edb8345e49" dependencies = [ - "proc-macro2 0.4.30", - "quote 0.6.13", - "syn 0.15.44", + "proc-macro2", + "quote", + "syn 2.0.77", ] [[package]] @@ -12007,8 +12141,8 @@ checksum = "e9552f850d5f0964a4e4d0bf306459ac29323ddfbae05e35a7c0d35cb0803cc5" dependencies = [ "anyhow", "itertools 0.13.0", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -12125,22 +12259,13 @@ dependencies = [ "windows-sys 0.59.0", ] -[[package]] -name = "quote" -version = "0.6.13" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6ce23b6b870e8f94f81fb0a363d65d86675884b34a09043c81e5562f11c1f8e1" -dependencies = [ - "proc-macro2 0.4.30", -] - [[package]] name = "quote" version = "1.0.37" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b5b9d34b8991d19d98081b46eacdd8eb58c6f2b201139f7c5f643cc155a633af" dependencies = [ - "proc-macro2 1.0.86", + "proc-macro2", ] [[package]] @@ -12347,8 +12472,8 @@ version = "0.2.12" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a25d631e41bfb5fdcde1d4e2215f62f7f0afa3ff11e26563765bd6ea1d229aeb" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -12413,8 +12538,8 @@ version = "1.0.23" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "bcc303e793d3734489387d205e9b186fac9c6cfacedd98cbb2e8a5943595f3e6" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -12685,8 +12810,8 @@ version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e33d7b2abe0c340d8797fe2907d3f20d3b5ea5908683618bfe80df7f621f672a" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -12789,8 +12914,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7229b505ae0706e64f37ffc54a9c163e11022a6636d58fe1f3f52018257ff9f7" dependencies = [ "cfg-if", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "rustc_version", "syn 1.0.109", "unicode-ident", @@ -13176,8 +13301,8 @@ version = "0.10.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e5af959c8bf6af1aff6d2b463a57f71aae53d1332da58419e30ad8dc7011d951" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -13233,8 +13358,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2d35494501194174bda522a32605929eefc9ecf7e0a326c26db1fdd85881eb62" dependencies = [ "proc-macro-crate 3.2.0", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -13275,8 +13400,8 @@ version = "0.8.21" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b1eee588578aff73f856ab961cd2f79e36bc45d7ded33a7562adba4667aecc0e" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "serde_derive_internals", "syn 2.0.77", ] @@ -13480,8 +13605,8 @@ version = "1.0.210" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "243902eda00fad750862fc144cea25caca5e20d615af0a81bee94ca738f1df1f" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -13491,8 +13616,8 @@ version = "0.29.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "18d26a20a969b9e3fdf2fc2d9f21eda6c40e2de84c9408bb5d3b05d499aae711" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -13525,8 +13650,8 @@ version = "0.1.19" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6c64451ba24fc7a6a2d60fc75dd9c83c90903b19028d4eff35e88fc1e86564e9" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -13551,6 +13676,22 @@ dependencies = [ "serde", ] +[[package]] +name = "serde_with" +version = "2.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "07ff71d2c147a7b57362cead5e22f772cd52f6ab31cfcd9edcd7f6aeb2a0afbe" +dependencies = [ + "base64 0.13.1", + "chrono", + "hex", + "indexmap 1.9.3", + "serde", + "serde_json", + "serde_with_macros 2.3.3", + "time", +] + [[package]] name = "serde_with" version = "3.9.0" @@ -13565,10 +13706,22 @@ dependencies = [ "serde", "serde_derive", "serde_json", - "serde_with_macros", + "serde_with_macros 3.9.0", "time", ] +[[package]] +name = "serde_with_macros" +version = "2.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "881b6f881b17d13214e5d494c939ebab463d01264ce1811e9d4ac3a882e7695f" +dependencies = [ + "darling 0.20.10", + "proc-macro2", + "quote", + "syn 2.0.77", +] + [[package]] name = "serde_with_macros" version = "3.9.0" @@ -13576,8 +13729,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a8fee4991ef4f274617a51ad4af30519438dacb2f56ac773b08a1922ff743350" dependencies = [ "darling 0.20.10", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -13623,8 +13776,8 @@ version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "91d129178576168c589c9ec973feedf7d3126c01ac2bf08795109aa35b69fb8f" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -13707,11 +13860,11 @@ dependencies = [ [[package]] name = "shared-crypto" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "bcs", "eyre", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "serde", "serde_repr", ] @@ -13843,12 +13996,12 @@ dependencies = [ [[package]] name = "simulacrum" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "async-trait", "bcs", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "futures", "iota-config", "iota-execution", @@ -13943,8 +14096,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "990079665f075b699031e9c08fd3ab99be5029b96f3b78dc0709e8f77e4efebf" dependencies = [ "heck 0.4.1", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -13955,8 +14108,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "03c3c6b7927ffe7ecaa769ee0e3994da3b8cafc8f444578982c83ecb161af917" dependencies = [ "heck 0.5.0", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -14045,7 +14198,7 @@ dependencies = [ "lalrpop-util", "phf", "thiserror", - "unicode-xid 0.2.6", + "unicode-xid", ] [[package]] @@ -14202,8 +14355,8 @@ version = "1.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2835db23c4724c05a2f85b81c4681f4aa8ea158edc8a7f4ad791c916fb766c2e" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -14297,8 +14450,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1e385be0d24f186b4ce2f9982191e7101bb737312ad61c1f2f984f34bcf85d59" dependencies = [ "heck 0.4.1", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "rustversion", "syn 1.0.109", ] @@ -14310,8 +14463,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "23dc1fa9ac9c169a78ba62f0b841814b7abae11bdd047b9c58f893439e309ea0" dependencies = [ "heck 0.4.1", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "rustversion", "syn 2.0.77", ] @@ -14323,8 +14476,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "4c6bee85a5a24955dc440386795aa378cd9cf82acd5f764469152d2270e581be" dependencies = [ "heck 0.5.0", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "rustversion", "syn 2.0.77", ] @@ -14415,25 +14568,14 @@ dependencies = [ "symbolic-common", ] -[[package]] -name = "syn" -version = "0.15.44" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9ca4b3b69a77cbe1ffc9e198781b7acb0c7365a883670e8f1c1bc66fba79a5c5" -dependencies = [ - "proc-macro2 0.4.30", - "quote 0.6.13", - "unicode-xid 0.1.0", -] - [[package]] name = "syn" version = "1.0.109" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "72b64191b275b66ffe2469e8af2c1cfe3bafa67b529ead792a6d0160888b4237" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "unicode-ident", ] @@ -14443,8 +14585,8 @@ version = "2.0.77" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9f35bcdf61fd8e7be6caf75f429fdca8beb3ed76584befb503b1569faee373ed" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "unicode-ident", ] @@ -14469,10 +14611,10 @@ version = "0.12.6" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f36bdaa60a83aca3921b5259d5400cbf5e90fc51931376a9bd4a0eb79aa7210f" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", - "unicode-xid 0.2.6", + "unicode-xid", ] [[package]] @@ -14481,8 +14623,8 @@ version = "0.13.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c8af7666ab7b6390ab78131fb5b0fce11d6b7a6951602017c35fa82800708971" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -14540,8 +14682,8 @@ checksum = "99f688a08b54f4f02f0a3c382aefdb7884d3d69609f785bd253dc033243e3fe4" dependencies = [ "heck 0.4.1", "proc-macro-error", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] @@ -14570,7 +14712,7 @@ dependencies = [ [[package]] name = "telemetry-subscribers" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "atomic_float", "bytes", @@ -14638,6 +14780,16 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "terminal_size" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5352447f921fda68cf61b4101566c0bdb5104eff6804d0678e5227580ab6a4e9" +dependencies = [ + "rustix", + "windows-sys 0.59.0", +] + [[package]] name = "termtree" version = "0.4.1" @@ -14646,11 +14798,11 @@ checksum = "3369f5ac52d5eb6ab48c6b4ffdc8efbcad6b89c765749064ba298f2c68a16a76" [[package]] name = "test-cluster" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "anyhow", "bcs", - "fastcrypto", + "fastcrypto 0.1.8 (git+https://github.com/MystenLabs/fastcrypto?rev=5f2c63266a065996d53f98156f0412782b468597)", "fastcrypto-zkp", "futures", "iota-bridge", @@ -14666,7 +14818,7 @@ dependencies = [ "iota-metrics", "iota-node", "iota-protocol-config", - "iota-sdk 0.8.0-alpha", + "iota-sdk 0.9.0-alpha", "iota-simulator", "iota-swarm", "iota-swarm-config", @@ -14699,8 +14851,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "48db3bbc562408b2111f3a0c96ec416ffa3ab66f8a6ab42579b608b9f74744e1" dependencies = [ "cargo_metadata 0.15.4", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "serde", "strum_macros 0.24.3", ] @@ -14715,8 +14867,8 @@ dependencies = [ "if_chain", "itertools 0.10.5", "lazy_static", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "subprocess", "syn 2.0.77", "test-fuzz-internal", @@ -14763,8 +14915,8 @@ version = "1.0.64" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "08904e7672f5eb876eaaf87e0ce17857500934f4981c4a0ab2b4aa98baac7fc3" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -14798,6 +14950,24 @@ dependencies = [ "ordered-float", ] +[[package]] +name = "tic-tac-toe" +version = "0.1.0" +dependencies = [ + "anyhow", + "bcs", + "clap", + "dirs 5.0.1", + "fastcrypto 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)", + "iota-keys", + "iota-sdk 0.9.0-alpha", + "iota-types", + "move-core-types", + "serde", + "shared-crypto", + "tokio", +] + [[package]] name = "time" version = "0.3.36" @@ -14919,8 +15089,8 @@ version = "2.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "693d596312e88961bc67d7f1f97af8a70227d9f90c31bba5806eec004978d752" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -14929,8 +15099,8 @@ name = "tokio-macros" version = "2.4.0" source = "git+https://github.com/iotaledger/tokio-madsim-fork.git?branch=main#e34a35287024b341db16139a402508aaea8ec955" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -15144,9 +15314,9 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "fe4ee8877250136bd7e3d2331632810a4df4ea5e004656990d8d66d2f5ee8a67" dependencies = [ "prettyplease", - "proc-macro2 1.0.86", + "proc-macro2", "prost-build", - "quote 1.0.37", + "quote", "syn 2.0.77", ] @@ -15286,8 +15456,8 @@ version = "0.1.27" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "34704c8d6ebcbc939824180af020566b01a7c01f80641264eba0999f6c2b6be7" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -15390,14 +15560,14 @@ version = "0.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b79e2e9c9ab44c6d7c20d5976961b47e8f49ac199154daa514b77cd1ab536625" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] [[package]] name = "transaction-fuzzer" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "iota-core", "iota-move-build", @@ -15497,13 +15667,13 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "97fee6b57c6a41524a810daee9286c02d7752c4253064d0b05472833a438f675" dependencies = [ "cfg-if", - "rand 0.7.3", + "rand 0.8.5", "static_assertions", ] [[package]] name = "typed-store" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "async-trait", "bcs", @@ -15534,17 +15704,17 @@ dependencies = [ [[package]] name = "typed-store-derive" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "itertools 0.13.0", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 1.0.109", ] [[package]] name = "typed-store-error" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "serde", "thiserror", @@ -15552,7 +15722,7 @@ dependencies = [ [[package]] name = "typed-store-workspace-hack" -version = "0.8.0-alpha" +version = "0.9.0-alpha" dependencies = [ "libc", "memchr", @@ -15582,7 +15752,7 @@ version = "1.0.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a615d6c2764852a2e88a4f16e9ce1ea49bb776b5872956309e170d63a042a34f" dependencies = [ - "quote 1.0.37", + "quote", "syn 2.0.77", ] @@ -15658,12 +15828,6 @@ version = "0.1.14" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7dd6e30e90baa6f72411720665d41d89b9a3d039dc45b8faea1ddd07f617f6af" -[[package]] -name = "unicode-xid" -version = "0.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fc72304796d0818e357ead4e000d19c9c174ab23dc11093ac919054d20a6a7fc" - [[package]] name = "unicode-xid" version = "0.2.6" @@ -15788,7 +15952,7 @@ version = "1.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "aae2faf80ac463422992abf4de234731279c058aaf33171ca70277c98406b124" dependencies = [ - "quote 1.0.37", + "quote", "syn 1.0.109", ] @@ -15892,8 +16056,8 @@ dependencies = [ "bumpalo", "log", "once_cell", - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", "wasm-bindgen-shared", ] @@ -15916,7 +16080,7 @@ version = "0.2.93" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "585c4c91a46b072c92e908d99cb1dcdf95c5218eeb6f3bf1efa991ee7a68cccf" dependencies = [ - "quote 1.0.37", + "quote", "wasm-bindgen-macro-support", ] @@ -15926,8 +16090,8 @@ version = "0.2.93" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "afc340c74d9005395cf9dd098506f7f44e38f2b4a21c6aaacf9a105ea5e1e836" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", "wasm-bindgen-backend", "wasm-bindgen-shared", @@ -16122,8 +16286,8 @@ version = "0.57.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9107ddc059d5b6fbfbffdfa7a7fe3e22a226def0b2608f72e9d552763d3e1ad7" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -16133,8 +16297,8 @@ version = "0.57.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "29bee4b38ea3cde66011baa44dba677c432a78593e202392d1e9070cf2a7fca7" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -16523,8 +16687,8 @@ version = "0.7.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2380878cad4ac9aac1e2435f3eb4020e8374b5f13c296cb75b4620ff8e229154" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", "synstructure 0.13.1", ] @@ -16572,8 +16736,8 @@ version = "0.7.35" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "fa4f8080344d4671fb4e831a13ad1e68092748387dfc4f55e356242fae12ce3e" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -16592,8 +16756,8 @@ version = "0.1.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "595eed982f7d355beb85837f651fa22e90b3c044842dc7f2c2842c086f295808" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", "synstructure 0.13.1", ] @@ -16614,8 +16778,8 @@ version = "1.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ce36e65b0d2999d2aafac989fb249189a141aee1f53c612c1f37d72631959f69" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] @@ -16636,8 +16800,8 @@ version = "0.10.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6eafa6dfb17584ea3e2bd6e76e0cc15ad7af12b09abdd1ca55961bed9b1063c6" dependencies = [ - "proc-macro2 1.0.86", - "quote 1.0.37", + "proc-macro2", + "quote", "syn 2.0.77", ] diff --git a/Cargo.toml b/Cargo.toml index 9dbfc1385f2..be2b62e55c7 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -116,6 +116,7 @@ members = [ "crates/iota-proc-macros", "crates/iota-protocol-config", "crates/iota-protocol-config-macros", + "crates/iota-proxy", "crates/iota-replay", "crates/iota-rest-api", "crates/iota-rosetta", @@ -152,6 +153,8 @@ members = [ "crates/typed-store-error", "crates/typed-store-workspace-hack", "docs/examples/rust", + "examples/custom-indexer/rust", + "examples/tic-tac-toe/cli", "iota-execution", "iota-execution/cut", "iota-execution/latest/iota-adapter", @@ -161,7 +164,7 @@ members = [ [workspace.package] # This version string will be inherited by iota-core, iota-faucet, iota-node, iota-tools, iota-sdk, iota-move-build, and iota crates. -version = "0.8.0-alpha" +version = "0.9.0-alpha" [profile.release] # debug = 1 means line charts only, which is minimum needed for good stack traces @@ -204,9 +207,9 @@ unexpected_cfgs = { level = "warn", check-cfg = ['cfg(msim)', 'cfg(fail_points)' [workspace.dependencies] # external dependencies # anemo dependencies -anemo = { git = "https://github.com/mystenlabs/anemo.git", rev = "dbb5a074c2d25660525ab5d36d65ff0cb8051949" } -anemo-build = { git = "https://github.com/mystenlabs/anemo.git", rev = "dbb5a074c2d25660525ab5d36d65ff0cb8051949" } -anemo-tower = { git = "https://github.com/mystenlabs/anemo.git", rev = "dbb5a074c2d25660525ab5d36d65ff0cb8051949" } +anemo = { git = "https://github.com/mystenlabs/anemo.git", rev = "e609f7697ed6169bf0760882a0b6c032a57e4f3b" } +anemo-build = { git = "https://github.com/mystenlabs/anemo.git", rev = "e609f7697ed6169bf0760882a0b6c032a57e4f3b" } +anemo-tower = { git = "https://github.com/mystenlabs/anemo.git", rev = "e609f7697ed6169bf0760882a0b6c032a57e4f3b" } anyhow = "1.0.71" arc-swap = { version = "1.5.1", features = ["serde"] } async-graphql = "=7.0.1" @@ -215,6 +218,7 @@ async-trait = "0.1.61" aws-config = "0.56" axum = { version = "0.7", default-features = false, features = ["tokio", "http1", "http2", "json", "matched-path", "original-uri", "form", "query", "ws"] } axum-extra = { version = "0.9", features = ["typed-header"] } +axum-server = { git = "https://github.com/bmwill/axum-server.git", rev = "f44323e271afdd1365fd0c8b0a4c0bbdf4956cb7", version = "0.6", default-features = false, features = ["tls-rustls"] } backoff = { version = "0.4.0", features = ["futures", "futures-core", "pin-project-lite", "tokio", "tokio_1"] } base64 = "0.21.2" base64-url = "2" @@ -291,8 +295,9 @@ pretty_assertions = "1.3.0" proc-macro2 = "1.0.47" prometheus = "0.13.3" proptest = "1.1.0" -proptest-derive = "0.3.0" +proptest-derive = "0.5.1" prost = "0.13" +protobuf = { version = "2.28", features = ["with-bytes"] } quinn-proto = "0.11.6" quote = "1.0.23" rand = "0.8.5" @@ -409,7 +414,7 @@ iota-rosetta = { path = "crates/iota-rosetta" } iota-rpc-loadgen = { path = "crates/iota-rpc-loadgen" } iota-sdk = { path = "crates/iota-sdk" } # core-types with json format for REST API -iota-sdk2 = { package = "iota-rust-sdk", git = "https://github.com/iotaledger/iota-rust-sdk.git", rev = "d605da95029e74376f0f39a95526bb1a5c0ebd7a", features = ["hash", "serde", "schemars"] } +iota-sdk2 = { package = "iota-rust-sdk", git = "https://github.com/iotaledger/iota-rust-sdk.git", rev = "2ba6b293bdede769a1d9b4d1aecaede2ff7682dd", features = ["hash", "serde", "schemars"] } iota-simulator = { path = "crates/iota-simulator" } iota-snapshot = { path = "crates/iota-snapshot" } iota-source-validation = { path = "crates/iota-source-validation" } diff --git a/apps/apps-backend/jest-e2e.json b/apps/apps-backend/jest-e2e.json index 8a02f0e8a41..a3ce13344dd 100644 --- a/apps/apps-backend/jest-e2e.json +++ b/apps/apps-backend/jest-e2e.json @@ -1,13 +1,13 @@ { - "moduleFileExtensions": ["js", "json", "ts"], - "rootDir": "./", - "testRegex": ".e2e-spec.ts$", - "transform": { - "^.+\\.(t|j)s$": "ts-jest" - }, - "testEnvironment": "node", - "moduleNameMapper": { - "^src/(.*)$": "/dist/$1", - "^@iota/core/(.*)$": "/dist/core/src/$1" - } + "moduleFileExtensions": ["js", "json", "ts"], + "rootDir": "./", + "testRegex": ".e2e-spec.ts$", + "transform": { + "^.+\\.(t|j)s$": "ts-jest" + }, + "testEnvironment": "node", + "moduleNameMapper": { + "^src/(.*)$": "/dist/$1", + "^@iota/core/(.*)$": "/dist/core/src/$1" + } } diff --git a/apps/apps-backend/src/app.module.ts b/apps/apps-backend/src/app.module.ts index b7fd88272e7..d9272d12ba0 100644 --- a/apps/apps-backend/src/app.module.ts +++ b/apps/apps-backend/src/app.module.ts @@ -8,7 +8,7 @@ import { ConfigModule } from '@nestjs/config'; import { AnalyticsModule } from './analytics/analytics.module'; import { FeaturesModule } from './features/features.module'; -import { MonitorNetworkModule } from './monitor-network/monitor-network.module'; +import { MonitorNetworkModule } from './monitor-network/monitorNetwork.module'; import { PricesModule } from './prices/prices.module'; import { RestrictedModule } from './restricted/restricted.module'; import { HealthModule } from './health/health.module'; diff --git a/apps/apps-backend/src/features/features.controller.ts b/apps/apps-backend/src/features/features.controller.ts index a3f3dbbdd3f..ce0e127c6ba 100644 --- a/apps/apps-backend/src/features/features.controller.ts +++ b/apps/apps-backend/src/features/features.controller.ts @@ -2,7 +2,8 @@ // SPDX-License-Identifier: Apache-2.0 import { Controller, Get } from '@nestjs/common'; -import { Feature } from '@iota/core/constants/features.enum'; +import { Feature } from '@iota/core/enums/features.enums'; +import { Network } from '@iota/iota-sdk/client'; @Controller('/api/features') export class FeaturesController { @@ -62,7 +63,7 @@ export class FeaturesController { defaultValue: true, }, [Feature.AccountFinder]: { - defaultValue: false, + defaultValue: true, }, [Feature.StardustMigration]: { defaultValue: true, @@ -70,6 +71,15 @@ export class FeaturesController { [Feature.SupplyIncreaseVesting]: { defaultValue: true, }, + [Feature.BurntAndMintedTokensInEndedEpochs]: { + defaultValue: { + [Network.Mainnet]: false, + [Network.Devnet]: true, + [Network.Testnet]: false, + [Network.Localnet]: false, + [Network.Custom]: false, + }, + }, }, dateUpdated: new Date().toISOString(), }; @@ -139,6 +149,15 @@ export class FeaturesController { [Feature.SupplyIncreaseVesting]: { defaultValue: true, }, + [Feature.BurntAndMintedTokensInEndedEpochs]: { + defaultValue: { + [Network.Mainnet]: false, + [Network.Devnet]: true, + [Network.Testnet]: false, + [Network.Localnet]: false, + [Network.Custom]: false, + }, + }, }, dateUpdated: new Date().toISOString(), }; diff --git a/apps/apps-backend/src/monitor-network/monitor-network.controller.ts b/apps/apps-backend/src/monitor-network/monitorNetwork.controller.ts similarity index 100% rename from apps/apps-backend/src/monitor-network/monitor-network.controller.ts rename to apps/apps-backend/src/monitor-network/monitorNetwork.controller.ts diff --git a/apps/apps-backend/src/monitor-network/monitor-network.module.ts b/apps/apps-backend/src/monitor-network/monitorNetwork.module.ts similarity index 74% rename from apps/apps-backend/src/monitor-network/monitor-network.module.ts rename to apps/apps-backend/src/monitor-network/monitorNetwork.module.ts index 0273c17be8b..9897c7598b5 100644 --- a/apps/apps-backend/src/monitor-network/monitor-network.module.ts +++ b/apps/apps-backend/src/monitor-network/monitorNetwork.module.ts @@ -3,7 +3,7 @@ import { Module } from '@nestjs/common'; -import { MonitorNetworkController } from './monitor-network.controller'; +import { MonitorNetworkController } from './monitorNetwork.controller'; @Module({ controllers: [MonitorNetworkController], diff --git a/apps/apps-backend/tsconfig.json b/apps/apps-backend/tsconfig.json index 11f5788c888..1ee903b9da4 100644 --- a/apps/apps-backend/tsconfig.json +++ b/apps/apps-backend/tsconfig.json @@ -18,7 +18,7 @@ "forceConsistentCasingInFileNames": false, "noFallthroughCasesInSwitch": false, "paths": { - "@iota/core/constants/*": ["./../core/src/constants/*"] + "@iota/core/enums/*": ["./../core/src/enums/*"], } - }, + } } diff --git a/apps/apps-backend/types/iota-core.d.ts b/apps/apps-backend/types/iota-core.d.ts index e51933507ea..b5ad75d0244 100644 --- a/apps/apps-backend/types/iota-core.d.ts +++ b/apps/apps-backend/types/iota-core.d.ts @@ -1,4 +1,4 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -declare module '@iota/core/constants/features.enum'; +declare module '@iota/core/enums/features.enums'; diff --git a/apps/core/package.json b/apps/core/package.json index 0a8cd7acfeb..ab6b447365d 100644 --- a/apps/core/package.json +++ b/apps/core/package.json @@ -26,16 +26,17 @@ "@amplitude/analytics-types": "^0.20.0", "@growthbook/growthbook-react": "^1.0.0", "@hookform/resolvers": "^3.9.0", + "@iota/apps-ui-icons": "workspace:*", "@iota/apps-ui-kit": "workspace:*", "@iota/dapp-kit": "workspace:*", "@iota/iota-sdk": "workspace:*", "@iota/kiosk": "workspace:*", - "@iota/ui-icons": "workspace:*", "@sentry/react": "^7.59.2", "@tanstack/react-query": "^5.50.1", "bignumber.js": "^9.1.1", "clsx": "^2.1.1", "formik": "^2.4.2", + "idb-keyval": "^6.2.1", "qrcode.react": "^4.0.1", "react": "^18.3.1", "react-dom": "^18.3.1", diff --git a/apps/core/src/api/SentryHttpTransport.ts b/apps/core/src/api/SentryHttpTransport.ts index 66fa4756c94..913debac24f 100644 --- a/apps/core/src/api/SentryHttpTransport.ts +++ b/apps/core/src/api/SentryHttpTransport.ts @@ -15,27 +15,30 @@ export class SentryHttpTransport extends IotaHTTPTransport { } async withRequest(input: { method: string; params: unknown[] }, handler: () => Promise) { - const transaction = Sentry.startTransaction({ - name: input.method, - op: 'http.rpc-request', - data: input.params, - tags: { - url: this.url, + return Sentry.startSpan( + { + name: input.method, + op: 'http.rpc-request', + data: input.params, + tags: { + url: this.url, + }, }, - }); - - try { - const res = await handler(); - const status: Sentry.SpanStatusType = 'ok'; - transaction.setStatus(status); - return res; - } catch (e) { - const status: Sentry.SpanStatusType = 'internal_error'; - transaction.setStatus(status); - throw e; - } finally { - transaction.finish(); - } + async (span) => { + try { + const res = await handler(); + const status: Sentry.SpanStatusType = 'ok'; + span?.setStatus(status); + return res; + } catch (e) { + const status: Sentry.SpanStatusType = 'internal_error'; + span?.setStatus(status); + throw e; + } finally { + span?.end(); + } + }, + ); } override async request(input: { method: string; params: unknown[] }) { diff --git a/apps/core/src/components/QR.tsx b/apps/core/src/components/QR.tsx index 7ab47b246e3..2c4b806b161 100644 --- a/apps/core/src/components/QR.tsx +++ b/apps/core/src/components/QR.tsx @@ -1,7 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import { QRCodeSVG } from 'qrcode.react'; export enum QRLevel { diff --git a/apps/wallet-dashboard/components/Dialogs/Staking/views/Validator.tsx b/apps/core/src/components/Validator.tsx similarity index 63% rename from apps/wallet-dashboard/components/Dialogs/Staking/views/Validator.tsx rename to apps/core/src/components/Validator.tsx index a76ed05db57..3161fe8a812 100644 --- a/apps/wallet-dashboard/components/Dialogs/Staking/views/Validator.tsx +++ b/apps/core/src/components/Validator.tsx @@ -1,6 +1,7 @@ -// Copyright (c) 2024 IOTA Stiftung +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { ImageIcon, ImageIconSize, formatPercentageDisplay, useValidatorInfo } from '@iota/core'; +import { ImageIcon, ImageIconSize, formatPercentageDisplay, useValidatorInfo } from '../'; import { Card, CardBody, @@ -10,24 +11,28 @@ import { CardType, Badge, BadgeType, + ImageShape, + Skeleton, } from '@iota/apps-ui-kit'; import { formatAddress } from '@iota/iota-sdk/utils'; interface ValidatorProps { - isSelected: boolean; + isSelected?: boolean; address: string; + type?: CardType; showActiveStatus?: boolean; - onClick?: (address: string) => void; - showAction?: boolean; - activeEpoch?: string; + onClick?: () => void; + showApy?: boolean; + activeEpoch?: number; } export function Validator({ address, - showActiveStatus, + type, + showActiveStatus = false, onClick, isSelected, - showAction = true, + showApy = true, activeEpoch, }: ValidatorProps) { const { @@ -38,10 +43,27 @@ export function Validator({ isApyApproxZero, validatorSummary, system, + isPendingValidators, } = useValidatorInfo({ validatorAddress: address, }); + if (isPendingValidators) { + return ( + + + + +
+ + +
+
+ +
+
+ ); + } // for inactive validators, show the epoch number const fallBackText = activeEpoch ? `Staked ${Number(system?.epoch) - Number(activeEpoch)} epochs ago` @@ -58,11 +80,8 @@ export function Validator({ ) : ( formatAddress(address) ); - - const handleClick = onClick ? () => onClick(address) : undefined; - return ( - + - {showAction && ( + {showApy && ( void; } -export function Kiosk({ object }: KioskProps) { - const address = useActiveAddress(); +export function KioskTile({ object, address, onClick }: KioskTileProps) { const { data: kioskData, isPending } = useGetKioskContents(address); const kioskId = getKioskIdFromOwnerCap(object); @@ -25,10 +31,18 @@ export function Kiosk({ object }: KioskProps) { ? null : itemsWithDisplay[0].data?.display?.data?.image_url || null; - if (isPending) return null; + if (isPending) + return ( +
+ +
+ ); return ( -
+
+
{isHoverable && (
)} diff --git a/apps/wallet-dashboard/components/Buttons/index.ts b/apps/core/src/components/nft/index.ts similarity index 62% rename from apps/wallet-dashboard/components/Buttons/index.ts rename to apps/core/src/components/nft/index.ts index ecd1cf7a7ce..ffc837bd846 100644 --- a/apps/wallet-dashboard/components/Buttons/index.ts +++ b/apps/core/src/components/nft/index.ts @@ -1,4 +1,4 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export { default as Button } from './Button'; +export * from './NftImage'; diff --git a/apps/core/src/components/stake/StakedCard.tsx b/apps/core/src/components/stake/StakedCard.tsx index 92f6b0e8184..22d4997f6b5 100644 --- a/apps/core/src/components/stake/StakedCard.tsx +++ b/apps/core/src/components/stake/StakedCard.tsx @@ -9,7 +9,6 @@ import { useIotaClientQuery } from '@iota/dapp-kit'; import { ImageIcon } from '../icon'; import { ExtendedDelegatedStake } from '../../utils'; import { useFormatCoin, useStakeRewardStatus } from '../../hooks'; -import React from 'react'; interface StakedCardProps { extendedStake: ExtendedDelegatedStake; @@ -61,7 +60,10 @@ export function StakedCard({ fallback={validatorMeta?.name || ''} /> - + ); diff --git a/apps/core/src/components/transaction/TransactionIcon.tsx b/apps/core/src/components/transaction/TransactionIcon.tsx new file mode 100644 index 00000000000..416a0cac65a --- /dev/null +++ b/apps/core/src/components/transaction/TransactionIcon.tsx @@ -0,0 +1,48 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { + ArrowBottomLeft, + ArrowTopRight, + Info, + IotaLogoMark, + Migration, + Person, + Stake, + Unstake, + Vesting, +} from '@iota/apps-ui-icons'; +import { TransactionAction } from '../../interfaces'; + +const ICON_COLORS = { + primary: 'text-primary-30', + error: 'text-error-30', +}; + +const icons = { + [TransactionAction.Send]: , + [TransactionAction.Receive]: , + [TransactionAction.Transaction]: , + [TransactionAction.Staked]: , + [TransactionAction.Unstaked]: , + [TransactionAction.Rewards]: , + [TransactionAction.Failed]: , + [TransactionAction.PersonalMessage]: , + [TransactionAction.TimelockedStaked]: , + [TransactionAction.TimelockedUnstaked]: , + [TransactionAction.Migration]: , + [TransactionAction.TimelockedCollect]: , +}; + +interface TransactionIconProps { + txnFailed?: boolean; + variant: TransactionAction; +} + +export function TransactionIcon({ txnFailed, variant }: TransactionIconProps) { + return ( +
+ {icons[txnFailed ? TransactionAction.Failed : variant]} +
+ ); +} diff --git a/apps/core/src/components/transaction/TransactionReceipt.tsx b/apps/core/src/components/transaction/TransactionReceipt.tsx index 99cac6112af..82326d56a30 100644 --- a/apps/core/src/components/transaction/TransactionReceipt.tsx +++ b/apps/core/src/components/transaction/TransactionReceipt.tsx @@ -1,23 +1,22 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import { InfoBox, InfoBoxStyle, InfoBoxType } from '@iota/apps-ui-kit'; -import { formatDate, type useTransactionSummary } from '../../hooks'; -import { CheckmarkFilled } from '@iota/ui-icons'; +import type { useTransactionSummary } from '../../hooks'; +import { CheckmarkFilled } from '@iota/apps-ui-icons'; import { IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; import { STAKING_REQUEST_EVENT, UNSTAKING_REQUEST_EVENT } from '../../constants'; import { StakeTransactionDetails } from './details'; import { UnstakeTransactionInfo } from './info'; import { TransactionSummary } from './summary'; -import { RenderExplorerLink, RenderValidatorLogo } from '../../types'; +import { RenderExplorerLink } from '../../types'; import { GasFees } from '../gas'; +import { formatDate } from '../../utils'; interface TransactionReceiptProps { txn: IotaTransactionBlockResponse; activeAddress: string | null; summary: Exclude, null>; - renderValidatorLogo: RenderValidatorLogo; renderExplorerLink: RenderExplorerLink; } @@ -25,7 +24,6 @@ export function TransactionReceipt({ txn, activeAddress, summary, - renderValidatorLogo, renderExplorerLink, }: TransactionReceiptProps) { const { events } = txn; @@ -49,7 +47,6 @@ export function TransactionReceipt({ activeAddress={activeAddress} event={stakeTypeTransaction} gasSummary={summary?.gas} - renderValidatorLogo={renderValidatorLogo} renderExplorerLink={renderExplorerLink} /> ) : null} @@ -60,7 +57,6 @@ export function TransactionReceipt({ event={unstakeTypeTransaction} gasSummary={summary?.gas} renderExplorerLink={renderExplorerLink} - renderValidatorLogo={renderValidatorLogo} /> ) : null} @@ -87,7 +83,9 @@ interface TransactionStatusProps { } function TransactionStatus({ success, timestamp, isIncoming }: TransactionStatusProps) { - const txnDate = timestamp ? formatDate(Number(timestamp)) : ''; + const txnDate = timestamp + ? formatDate(Number(timestamp), ['day', 'month', 'year', 'hour', 'minute']) + : ''; const successMessage = isIncoming ? 'Successfully received' : 'Successfully sent'; return ( {validatorAddress && ( - )} diff --git a/apps/core/src/components/transaction/index.ts b/apps/core/src/components/transaction/index.ts index bedefbc5d7e..e3b830d27d8 100644 --- a/apps/core/src/components/transaction/index.ts +++ b/apps/core/src/components/transaction/index.ts @@ -5,3 +5,4 @@ export * from './info'; export * from './summary'; export * from './TransactionReceipt'; +export * from './TransactionIcon'; diff --git a/apps/core/src/components/transaction/info/StakeTransactionInfo.tsx b/apps/core/src/components/transaction/info/StakeTransactionInfo.tsx index 47f00c48d4f..f2802ad809a 100644 --- a/apps/core/src/components/transaction/info/StakeTransactionInfo.tsx +++ b/apps/core/src/components/transaction/info/StakeTransactionInfo.tsx @@ -2,7 +2,6 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import { Divider, KeyValueInfo, Panel, TooltipPosition } from '@iota/apps-ui-kit'; import { type GasSummaryType, useStakeTxnInfo, GasSummary } from '../../../'; import { RenderExplorerLink } from '../../../types'; diff --git a/apps/core/src/components/transaction/info/UnstakeTransactionInfo.tsx b/apps/core/src/components/transaction/info/UnstakeTransactionInfo.tsx index 94e120821d3..932fa20d5fc 100644 --- a/apps/core/src/components/transaction/info/UnstakeTransactionInfo.tsx +++ b/apps/core/src/components/transaction/info/UnstakeTransactionInfo.tsx @@ -2,19 +2,17 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import { TransactionAmount } from '../amount'; import type { IotaEvent } from '@iota/iota-sdk/client'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; -import type { GasSummaryType, RenderExplorerLink, RenderValidatorLogo } from '../../../types'; +import type { GasSummaryType, RenderExplorerLink } from '../../../types'; import { useFormatCoin } from '../../../hooks'; -import { Divider, KeyValueInfo, Panel } from '@iota/apps-ui-kit'; -import { GasSummary } from '../../..'; +import { Divider, KeyValueInfo, Panel, CardType } from '@iota/apps-ui-kit'; +import { GasSummary, getUnstakeDetailsFromEvent, Validator } from '../../..'; interface UnstakeTransactionInfoProps { activeAddress: string | null; event: IotaEvent; - renderValidatorLogo: RenderValidatorLogo; renderExplorerLink: RenderExplorerLink; gasSummary?: GasSummaryType; } @@ -23,24 +21,17 @@ export function UnstakeTransactionInfo({ activeAddress, event, gasSummary, - renderValidatorLogo: ValidatorLogo, renderExplorerLink, }: UnstakeTransactionInfoProps) { - const json = event.parsedJson as { - principal_amount?: string; - reward_amount?: string; - validator_address?: string; - }; - const principalAmount = json?.principal_amount || '0'; - const rewardAmount = json?.reward_amount || '0'; - const validatorAddress = json?.validator_address; - const totalAmount = BigInt(principalAmount) + BigInt(rewardAmount); + const { principalAmount, rewardAmount, totalAmount, validatorAddress } = + getUnstakeDetailsFromEvent(event); + const [formatPrinciple, symbol] = useFormatCoin(principalAmount, IOTA_TYPE_ARG); const [formatRewards] = useFormatCoin(rewardAmount || 0, IOTA_TYPE_ARG); return (
- {validatorAddress && } + {validatorAddress && } {totalAmount !== 0n && ( )} diff --git a/apps/core/src/components/transaction/summary/TransactionSummary.tsx b/apps/core/src/components/transaction/summary/TransactionSummary.tsx index 090d84d8ece..cd669d6cc49 100644 --- a/apps/core/src/components/transaction/summary/TransactionSummary.tsx +++ b/apps/core/src/components/transaction/summary/TransactionSummary.tsx @@ -1,7 +1,7 @@ // Copyright (c) Mysten Labs, Inc. // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; + import { type TransactionSummaryType } from '../../..'; import { BalanceChanges, ObjectChanges } from '../../cards'; import { LoadingIndicator, Title, TitleSize } from '@iota/apps-ui-kit'; diff --git a/apps/core/src/constants/coins.constants.ts b/apps/core/src/constants/coins.constants.ts index 11fba0c12a0..b0e38e586cd 100644 --- a/apps/core/src/constants/coins.constants.ts +++ b/apps/core/src/constants/coins.constants.ts @@ -3,3 +3,4 @@ export const COINS_QUERY_REFETCH_INTERVAL = 20_000; export const COINS_QUERY_STALE_TIME = 20_000; +export const COIN_TYPE = '0x2::coin::Coin'; diff --git a/apps/core/src/constants/index.ts b/apps/core/src/constants/index.ts index 834fd50145e..693b46c5973 100644 --- a/apps/core/src/constants/index.ts +++ b/apps/core/src/constants/index.ts @@ -6,6 +6,5 @@ export * from './recognizedPackages.constants'; export * from './coins.constants'; export * from './timelock.constants'; export * from './migration.constants'; -export * from './features.enum'; export * from './gas.constants'; export * from './time.constants'; diff --git a/apps/core/src/constants/timelock.constants.ts b/apps/core/src/constants/timelock.constants.ts index 6b9ab9c2627..21c7999aef0 100644 --- a/apps/core/src/constants/timelock.constants.ts +++ b/apps/core/src/constants/timelock.constants.ts @@ -1,5 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export const TIMELOCK_IOTA_TYPE = '0x2::timelock::TimeLock<0x2::balance::Balance<0x2::iota::IOTA>>'; -export const TIMELOCK_STAKED_TYPE = '0x3::timelocked_staking::TimelockedStakedIota'; +export const TIMELOCK_MODULE = 'timelock'; +export const TIMELOCK_IOTA_TYPE = `0x2::${TIMELOCK_MODULE}::TimeLock<0x2::balance::Balance<0x2::iota::IOTA>>`; +export const TIMELOCK_STAKED_TYPE = `0x3::timelocked_staking::TimelockedStakedIota`; diff --git a/apps/core/src/contexts/HiddenAssetsProvider.tsx b/apps/core/src/contexts/HiddenAssetsProvider.tsx new file mode 100644 index 00000000000..3d03e54669f --- /dev/null +++ b/apps/core/src/contexts/HiddenAssetsProvider.tsx @@ -0,0 +1,112 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { get, set } from 'idb-keyval'; +import { + PropsWithChildren, + createContext, + useCallback, + useContext, + useEffect, + useState, + useRef, +} from 'react'; + +const HIDDEN_ASSET_IDS = 'hidden-asset-ids'; + +export type HiddenAssets = + | { + type: 'loading'; + } + | { + type: 'loaded'; + assetIds: string[]; + }; + +interface HiddenAssetContext { + hiddenAssets: HiddenAssets; + hideAsset: (assetId: string) => string | void; + showAsset: (assetId: string) => string | void; +} + +export const HiddenAssetsContext = createContext({ + hiddenAssets: { + type: 'loading', + }, + hideAsset: () => {}, + showAsset: () => {}, +}); + +export const HiddenAssetsProvider = ({ children }: PropsWithChildren) => { + const [hiddenAssets, setHiddenAssets] = useState({ + type: 'loading', + }); + const hiddenAssetIdsRef = useRef([]); + + useEffect(() => { + (async () => { + try { + const hiddenAssetsFromStorage = (await get(HIDDEN_ASSET_IDS)) ?? []; + hiddenAssetIdsRef.current = hiddenAssetsFromStorage; + setHiddenAssetIds(hiddenAssetsFromStorage); + } catch (error) { + console.error('Failed to load hidden assets from storage:', error); + setHiddenAssetIds([]); + } + })(); + }, []); + + function setHiddenAssetIds(hiddenAssetIds: string[]) { + hiddenAssetIdsRef.current = hiddenAssetIds; + setHiddenAssets({ + type: 'loaded', + assetIds: hiddenAssetIds, + }); + } + + const syncIdb = useCallback(async (nextState: string[], prevState: string[]) => { + try { + await set(HIDDEN_ASSET_IDS, nextState); + } catch (error) { + console.error('Error syncing with IndexedDB:', error); + // Revert to the previous state on failure + setHiddenAssetIds(prevState); + } + }, []); + + const hideAsset = useCallback((assetId: string) => { + const prevIds = [...hiddenAssetIdsRef.current]; + const newHiddenAssetIds = Array.from(new Set([...hiddenAssetIdsRef.current, assetId])); + setHiddenAssetIds(newHiddenAssetIds); + syncIdb(newHiddenAssetIds, prevIds); + return assetId; + }, []); + + const showAsset = useCallback((assetId: string) => { + // Ensure the asset exists in the hidden list + if (!hiddenAssetIdsRef.current.includes(assetId)) return; + + const prevIds = [...hiddenAssetIdsRef.current]; + // Compute the new list of hidden assets + const updatedHiddenAssetIds = hiddenAssetIdsRef.current.filter((id) => id !== assetId); + setHiddenAssetIds(updatedHiddenAssetIds); + syncIdb(updatedHiddenAssetIds, prevIds); + }, []); + + return ( + + {children} + + ); +}; + +export const useHiddenAssets = () => { + return useContext(HiddenAssetsContext); +}; diff --git a/apps/core/src/contexts/index.ts b/apps/core/src/contexts/index.ts index c592171afa8..fe492c59d09 100644 --- a/apps/core/src/contexts/index.ts +++ b/apps/core/src/contexts/index.ts @@ -2,3 +2,4 @@ // SPDX-License-Identifier: Apache-2.0 export * from './ThemeContext'; +export * from './HiddenAssetsProvider'; diff --git a/apps/core/src/enums/explorer-link-type.enums.ts b/apps/core/src/enums/explorerLinkType.enums.ts similarity index 100% rename from apps/core/src/enums/explorer-link-type.enums.ts rename to apps/core/src/enums/explorerLinkType.enums.ts diff --git a/apps/core/src/constants/features.enum.ts b/apps/core/src/enums/features.enums.ts similarity index 92% rename from apps/core/src/constants/features.enum.ts rename to apps/core/src/enums/features.enums.ts index 3670db4edc1..5d260493c5f 100644 --- a/apps/core/src/constants/features.enum.ts +++ b/apps/core/src/enums/features.enums.ts @@ -20,4 +20,5 @@ export enum Feature { WalletEffectsOnlySharedTransaction = 'wallet-effects-only-shared-transaction', StardustMigration = 'migration', SupplyIncreaseVesting = 'supply-increase-vesting', + BurntAndMintedTokensInEndedEpochs = 'burnt-and-minted-tokens-in-ended-epochs', } diff --git a/apps/core/src/enums/index.ts b/apps/core/src/enums/index.ts index 7c01aed335d..e7bd792140c 100644 --- a/apps/core/src/enums/index.ts +++ b/apps/core/src/enums/index.ts @@ -2,4 +2,5 @@ // SPDX-License-Identifier: Apache-2.0 export * from './theme.enums'; -export * from './explorer-link-type.enums'; +export * from './explorerLinkType.enums'; +export * from './features.enums'; diff --git a/apps/core/src/hooks/__tests__/useFormatCoin.test.ts b/apps/core/src/hooks/__tests__/useFormatCoin.test.ts index c55d16574c5..a138a90a8ca 100644 --- a/apps/core/src/hooks/__tests__/useFormatCoin.test.ts +++ b/apps/core/src/hooks/__tests__/useFormatCoin.test.ts @@ -24,6 +24,21 @@ describe('formatBalance', () => { expect(formatBalance('0.000', IOTA_DECIMALS)).toEqual('0'); }); + it('formats decimal amounts with less than 4 leading zeroes, truncated with up to 4 decimals', () => { + expect(formatBalance('512345678', IOTA_DECIMALS)).toEqual('0.5'); + expect(formatBalance('51234567', IOTA_DECIMALS)).toEqual('0.05'); + expect(formatBalance('5123456', IOTA_DECIMALS)).toEqual('0.005'); + expect(formatBalance('523456', IOTA_DECIMALS)).toEqual('0.0005'); + }); + + it('formats decimal amounts with 4 or more leading zeroes (after decimal point) with subscripts', () => { + expect(formatBalance('19723', IOTA_DECIMALS)).toEqual('0.0₄19723'); + expect(formatBalance('1234', IOTA_DECIMALS)).toEqual('0.0₅1234'); + expect(formatBalance('123', IOTA_DECIMALS)).toEqual('0.0₆123'); + expect(formatBalance('12', IOTA_DECIMALS)).toEqual('0.0₇12'); + expect(formatBalance('1', IOTA_DECIMALS)).toEqual('0.0₈1'); + }); + it('formats integer amounts correctly', () => { expect(formatBalance(toNano('1'), IOTA_DECIMALS)).toEqual('1'); expect(formatBalance(toNano('1.0001'), IOTA_DECIMALS)).toEqual('1'); diff --git a/apps/core/src/hooks/index.ts b/apps/core/src/hooks/index.ts index 1602541e57b..aee1fb3e15e 100644 --- a/apps/core/src/hooks/index.ts +++ b/apps/core/src/hooks/index.ts @@ -47,6 +47,10 @@ export * from './useOwnedNFT'; export * from './useNftDetails'; export * from './useCountdownByTimestamp'; export * from './useStakeRewardStatus'; +export * from './useGetNFTs'; export * from './useRecognizedPackages'; +export * from './useTransferAsset'; +export * from './useFeatureEnabledByNetwork'; export * from './stake'; +export * from './ui'; diff --git a/apps/core/src/hooks/stake/useValidatorInfo.tsx b/apps/core/src/hooks/stake/useValidatorInfo.ts similarity index 97% rename from apps/core/src/hooks/stake/useValidatorInfo.tsx rename to apps/core/src/hooks/stake/useValidatorInfo.ts index f8d6dcd5a2f..b3d5b9833f9 100644 --- a/apps/core/src/hooks/stake/useValidatorInfo.tsx +++ b/apps/core/src/hooks/stake/useValidatorInfo.ts @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 import { useIotaClientQuery } from '@iota/dapp-kit'; -import { useGetValidatorsApy } from '../'; +import { useGetValidatorsApy } from '..'; export function useValidatorInfo({ validatorAddress }: { validatorAddress: string }) { const { diff --git a/apps/core/src/hooks/ui/index.ts b/apps/core/src/hooks/ui/index.ts new file mode 100644 index 00000000000..37bbc87557c --- /dev/null +++ b/apps/core/src/hooks/ui/index.ts @@ -0,0 +1,4 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export * from './usePageAssets'; diff --git a/apps/core/src/hooks/ui/usePageAssets.ts b/apps/core/src/hooks/ui/usePageAssets.ts new file mode 100644 index 00000000000..6fbd8f35836 --- /dev/null +++ b/apps/core/src/hooks/ui/usePageAssets.ts @@ -0,0 +1,131 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 +import { useState, useMemo, useRef, useEffect } from 'react'; +import { useGetNFTs, HiddenAssets, useOnScreen } from '../..'; + +export enum AssetCategory { + Visual = 'Visual', + Other = 'Other', + Hidden = 'Hidden', +} + +export function usePageAssets(address: string | null, hiddenAssets?: HiddenAssets) { + const [selectedAssetCategory, setSelectedAssetCategory] = useState(null); + const observerElem = useRef(null); + const { isIntersecting } = useOnScreen(observerElem); + const { + data: ownedAssets, + hasNextPage, + isFetchingNextPage, + fetchNextPage, + error, + isPending, + isError, + isFetching, + refetch, + } = useGetNFTs(address, hiddenAssets); + + const isAssetsLoaded = !!ownedAssets; + const isSpinnerVisible = isFetchingNextPage && hasNextPage; + + const filteredAssets = (() => { + if (!ownedAssets) return []; + switch (selectedAssetCategory) { + case AssetCategory.Visual: + return ownedAssets.visual; + case AssetCategory.Other: + return ownedAssets.other; + default: + return []; + } + })(); + + const filteredHiddenAssets = useMemo(() => { + return ( + ownedAssets?.hidden + .flatMap((data) => { + return { + data: data, + display: data?.display?.data, + }; + }) + .sort((nftA, nftB) => { + const nameA = nftA.display?.name || ''; + const nameB = nftB.display?.name || ''; + + if (nameA < nameB) { + return -1; + } else if (nameA > nameB) { + return 1; + } + return 0; + }) ?? [] + ); + }, [ownedAssets]); + + // Fetch the next page if the user scrolls to the bottom of the page + useEffect(() => { + if (isIntersecting && hasNextPage && !isFetchingNextPage) { + fetchNextPage(); + } + }, [isIntersecting, fetchNextPage, hasNextPage, isFetchingNextPage]); + + // select the default category if no category is selected and assets are loaded + useEffect(() => { + let computeSelectedCategory = false; + if ( + (selectedAssetCategory === AssetCategory.Visual && ownedAssets?.visual.length === 0) || + (selectedAssetCategory === AssetCategory.Other && ownedAssets?.other.length === 0) || + (selectedAssetCategory === AssetCategory.Hidden && ownedAssets?.hidden.length === 0) || + !selectedAssetCategory + ) { + computeSelectedCategory = true; + } + if (computeSelectedCategory && ownedAssets) { + const defaultCategory = + ownedAssets.visual.length > 0 + ? AssetCategory.Visual + : ownedAssets.other.length > 0 + ? AssetCategory.Other + : ownedAssets.hidden.length > 0 + ? AssetCategory.Hidden + : null; + setSelectedAssetCategory(defaultCategory); + } + }, [ownedAssets]); + + // Fetch the next page if there are no visual assets, other + hidden assets are present in multiples of 50, and there are more pages to fetch + useEffect(() => { + if ( + hasNextPage && + ownedAssets?.visual.length === 0 && + ownedAssets?.other.length + ownedAssets?.hidden.length > 0 && + (ownedAssets.other.length + ownedAssets.hidden.length) % 50 === 0 && + !isFetchingNextPage + ) { + fetchNextPage(); + setSelectedAssetCategory(null); + } + }, [hasNextPage, ownedAssets, isFetchingNextPage]); + + return { + // reexport from useGetNFTs + ownedAssets, + hasNextPage, + isFetchingNextPage, + fetchNextPage, + error, + isPending, + isError, + isFetching, + refetch, + + isAssetsLoaded, + filteredAssets, + filteredHiddenAssets, + selectedAssetCategory, + setSelectedAssetCategory, + observerElem, + isSpinnerVisible, + }; +} diff --git a/apps/core/src/hooks/useFeatureEnabledByNetwork.ts b/apps/core/src/hooks/useFeatureEnabledByNetwork.ts new file mode 100644 index 00000000000..2f3c51761af --- /dev/null +++ b/apps/core/src/hooks/useFeatureEnabledByNetwork.ts @@ -0,0 +1,15 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { useFeature } from '@growthbook/growthbook-react'; +import type { Network } from '@iota/iota-sdk/client'; +import { Feature } from '../enums'; + +type NetworkBasedFeature = { + [key in Network]: boolean; +}; + +export function useFeatureEnabledByNetwork(feature: Feature, network: Network): boolean { + const featureFlag = useFeature(feature)?.value; + return featureFlag?.[network] ?? false; +} diff --git a/apps/core/src/hooks/useFileExtensionType.ts b/apps/core/src/hooks/useFileExtensionType.ts index 04a0ce3f604..e21274a22b7 100644 --- a/apps/core/src/hooks/useFileExtensionType.ts +++ b/apps/core/src/hooks/useFileExtensionType.ts @@ -98,7 +98,7 @@ export const extractFileType = async (imgUrl: string) => { }); }; */ -export default function useFileExtensionType(url: string) { +export function useFileExtensionType(url: string) { if (!url) return { name: '', type: '' }; const fileType = url.split('.').pop() || ''; return FILE_EXTENSION_TYPE_MAP[fileType] || { name: '', type: '' }; diff --git a/apps/wallet/src/ui/app/hooks/useGetNFTs.ts b/apps/core/src/hooks/useGetNFTs.ts similarity index 73% rename from apps/wallet/src/ui/app/hooks/useGetNFTs.ts rename to apps/core/src/hooks/useGetNFTs.ts index 8e891e129be..9fedb74b911 100644 --- a/apps/wallet/src/ui/app/hooks/useGetNFTs.ts +++ b/apps/core/src/hooks/useGetNFTs.ts @@ -2,10 +2,16 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { hasDisplayData, isKioskOwnerToken, useGetOwnedObjects, useKioskClient } from '@iota/core'; +import { + hasDisplayData, + isKioskOwnerToken, + useGetOwnedObjects, + useKioskClient, + HiddenAssets, + COIN_TYPE, +} from '../../'; import { type IotaObjectData } from '@iota/iota-sdk/client'; import { useMemo } from 'react'; -import { useHiddenAssets } from '../pages/home/assets/HiddenAssetsProvider'; type OwnedAssets = { visual: IotaObjectData[]; @@ -18,10 +24,13 @@ export enum AssetFilterTypes { Other = 'other', } -export function useGetNFTs(address?: string | null) { +const OBJECTS_PER_REQ = 50; + +export function useGetNFTs(address?: string | null, hiddenAssets?: HiddenAssets) { const kioskClient = useKioskClient(); const { data, + isFetching, isPending, error, isError, @@ -29,14 +38,14 @@ export function useGetNFTs(address?: string | null) { hasNextPage, fetchNextPage, isLoading, + refetch, } = useGetOwnedObjects( address, { - MatchNone: [{ StructType: '0x2::coin::Coin' }], + MatchNone: [{ StructType: COIN_TYPE }], }, - 50, + OBJECTS_PER_REQ, ); - const { hiddenAssets } = useHiddenAssets(); const assets = useMemo(() => { const ownedAssets: OwnedAssets = { @@ -45,13 +54,16 @@ export function useGetNFTs(address?: string | null) { hidden: [], }; - if (hiddenAssets.type === 'loading') { + if (hiddenAssets?.type === 'loading') { return ownedAssets; } else { const groupedAssets = data?.pages .flatMap((page) => page.data) .reduce((acc, curr) => { - if (curr.data?.objectId && hiddenAssets.assetIds.includes(curr.data?.objectId)) + if ( + curr.data?.objectId && + hiddenAssets?.assetIds?.includes(curr.data?.objectId) + ) acc.hidden.push(curr.data as IotaObjectData); else if (hasDisplayData(curr) || isKioskOwnerToken(kioskClient.network, curr)) acc.visual.push(curr.data as IotaObjectData); @@ -64,6 +76,7 @@ export function useGetNFTs(address?: string | null) { return { data: assets, + isFetching, isLoading, hasNextPage, isFetchingNextPage, @@ -71,5 +84,6 @@ export function useGetNFTs(address?: string | null) { isPending: isPending, isError: isError, error, + refetch, }; } diff --git a/apps/core/src/hooks/useLocalStorage.ts b/apps/core/src/hooks/useLocalStorage.ts index 7aa7276228e..5e610c8ceaa 100644 --- a/apps/core/src/hooks/useLocalStorage.ts +++ b/apps/core/src/hooks/useLocalStorage.ts @@ -9,6 +9,12 @@ type SetValue = Dispatch>; export function useLocalStorage(key: string, initialValue: T): [T, SetValue] { const getValue = useCallback(() => { try { + if (typeof window === 'undefined') { + console.warn( + `Tried reading localStorage key "${key}" even though window is not defined`, + ); + return initialValue; + } const item = window.localStorage.getItem(key); return item ? (JSON.parse(item) as T) : initialValue; } catch (error) { @@ -21,12 +27,6 @@ export function useLocalStorage(key: string, initialValue: T): [T, SetValue = useCallback( (value) => { - if (typeof window === 'undefined') { - console.warn( - `Tried setting localStorage key "${key}" even though window is not defined`, - ); - } - try { const newValue = value instanceof Function ? value(storedValue) : value; window.localStorage.setItem(key, JSON.stringify(newValue)); diff --git a/apps/core/src/hooks/useMediaUrl.ts b/apps/core/src/hooks/useMediaUrl.ts index bf4dc6cf4a6..e96c96124f3 100644 --- a/apps/core/src/hooks/useMediaUrl.ts +++ b/apps/core/src/hooks/useMediaUrl.ts @@ -7,7 +7,7 @@ import { useMemo } from 'react'; const parseIpfsUrl = (ipfsUrl: string) => ipfsUrl.replace(/^ipfs:\/\//, 'https://ipfs.io/ipfs/'); -export default function useMediaUrl(objData: IotaParsedData | null) { +export function useMediaUrl(objData: IotaParsedData | null) { const { fields } = ((objData?.dataType === 'moveObject' && objData) as { fields: { url?: string; metadata?: { fields: { url: string } } }; diff --git a/apps/core/src/hooks/useNFTBasicData.ts b/apps/core/src/hooks/useNFTBasicData.ts index 227a9707de1..4911a5594c2 100644 --- a/apps/core/src/hooks/useNFTBasicData.ts +++ b/apps/core/src/hooks/useNFTBasicData.ts @@ -3,9 +3,8 @@ // SPDX-License-Identifier: Apache-2.0 import type { IotaObjectData } from '@iota/iota-sdk/client'; - -import useFileExtensionType from './useFileExtensionType'; -import useMediaUrl from './useMediaUrl'; +import { useMediaUrl } from './useMediaUrl'; +import { useFileExtensionType } from './useFileExtensionType'; export function useNFTBasicData(nftObj: IotaObjectData | null) { const nftObjectID = nftObj?.objectId || null; diff --git a/apps/core/src/hooks/useNftDetails.ts b/apps/core/src/hooks/useNftDetails.ts index f76001b0273..dc65ab58a9a 100644 --- a/apps/core/src/hooks/useNftDetails.ts +++ b/apps/core/src/hooks/useNftDetails.ts @@ -8,7 +8,6 @@ import { useIsAssetTransferable, } from './'; import { formatAddress } from '@iota/iota-sdk/utils'; -import { truncateString } from '../utils'; type NftField = { keys: string[]; values: string[] }; @@ -35,7 +34,7 @@ export function useNftDetails(nftId: string, accountAddress: string | null) { // Extract either the attributes, or use the top-level NFT fields: const { keys: metaKeys, values: metaValues } = - (nftFields as NftFields)?.metadata?.fields?.attributes?.fields || + (nftFields as unknown as NftFields)?.metadata?.fields?.attributes?.fields || Object.entries(nftFields ?? {}) .filter(([key]) => key !== 'id') .reduce( @@ -54,28 +53,6 @@ export function useNftDetails(nftId: string, accountAddress: string | null) { objectData.owner.AddressOwner) || ''; - function formatMetaValue(value: string | object) { - if (typeof value === 'object') { - return { - value: JSON.stringify(value), - valueLink: undefined, - }; - } else { - if (value.includes('http')) { - return { - value: value.startsWith('http') - ? truncateString(value, 20, 8) - : formatAddress(value), - valueLink: value, - }; - } - return { - value: value, - valueLink: undefined, - }; - } - } - const isLoading = isNftLoading || isCheckingAssetTransferability || isPendingNftDislpay; return { @@ -89,7 +66,6 @@ export function useNftDetails(nftId: string, accountAddress: string | null) { isAssetTransferable, metaKeys, metaValues, - formatMetaValue, isContainedInKiosk, kioskItem, nftDisplayData, diff --git a/apps/core/src/hooks/useOnScreen.ts b/apps/core/src/hooks/useOnScreen.ts index 33ae1701b8c..a1532e79818 100644 --- a/apps/core/src/hooks/useOnScreen.ts +++ b/apps/core/src/hooks/useOnScreen.ts @@ -19,7 +19,7 @@ export const useOnScreen = (elementRef: MutableRefObject) => { ); observer.observe(node); return () => observer.disconnect(); - }, [elementRef]); + }, [elementRef.current]); return { isIntersecting }; }; diff --git a/apps/core/src/hooks/useTimeAgo.ts b/apps/core/src/hooks/useTimeAgo.ts index 6e892254b7d..9011034b20a 100644 --- a/apps/core/src/hooks/useTimeAgo.ts +++ b/apps/core/src/hooks/useTimeAgo.ts @@ -123,32 +123,3 @@ export const timeAgo = ( return result ? result : endLabel; }; - -// TODO - Merge with related functions -type Format = 'year' | 'month' | 'day' | 'hour' | 'minute' | 'second' | 'weekday'; - -export function formatDate(date: Date | number, format?: Format[]): string { - const formatOption = format ?? (['month', 'day', 'hour', 'minute'] as Format[]); - const dateTime = new Date(date); - if (!(dateTime instanceof Date)) return ''; - - const options = { - year: 'numeric', - month: 'short', - day: 'numeric', - hour: 'numeric', - weekday: 'short', - minute: 'numeric', - second: 'numeric', - }; - - const formatOptions = formatOption.reduce((accumulator, current: Format) => { - const responseObj = { - ...accumulator, - ...{ [current]: options[current] }, - }; - return responseObj; - }, {}); - - return new Intl.DateTimeFormat('en-US', formatOptions).format(dateTime); -} diff --git a/apps/core/src/hooks/useTransactionSummary.ts b/apps/core/src/hooks/useTransactionSummary.ts index f34bfa4001d..2e79dc69a00 100644 --- a/apps/core/src/hooks/useTransactionSummary.ts +++ b/apps/core/src/hooks/useTransactionSummary.ts @@ -11,7 +11,7 @@ import { IotaObjectChangeWithDisplay } from '../types'; import { getBalanceChangeSummary, getGasSummary, - getLabel, + getTransactionAction, getObjectChangeSummary, getObjectDisplayLookup, } from '../utils'; @@ -57,7 +57,7 @@ export function useTransactionSummary({ sender: transaction.transaction?.data.sender, balanceChanges: balanceChangeSummary, digest: transaction.digest, - label: getLabel(transaction, currentAddress), + label: getTransactionAction(transaction, currentAddress), objectSummary, status: transaction.effects?.status.status, timestamp: transaction.timestampMs, diff --git a/apps/core/src/hooks/useTransferAsset.ts b/apps/core/src/hooks/useTransferAsset.ts new file mode 100644 index 00000000000..e11f4ef1c67 --- /dev/null +++ b/apps/core/src/hooks/useTransferAsset.ts @@ -0,0 +1,64 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 +import { useGetKioskContents, TransferAssetExecuteFn } from '../'; +import { type IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; +import { Transaction } from '@iota/iota-sdk/transactions'; +import { useMutation } from '@tanstack/react-query'; +import { useTransferKioskItem } from './useTransferKioskItem'; + +export function useTransferAsset({ + objectId, + objectType, + activeAddress, + executeFn, + onSuccess, + onError, +}: { + objectId: string; + objectType?: string | null; + activeAddress?: string | null; + executeFn?: TransferAssetExecuteFn; + onSuccess?: (response: IotaTransactionBlockResponse, variables: string) => void; + onError?: (error: Error) => void; +}) { + const { data: kiosk } = useGetKioskContents(activeAddress); + const transferKioskItem = useTransferKioskItem({ + objectId, + objectType, + executeFn, + address: activeAddress, + }); + const isContainedInKiosk = kiosk?.list.some( + (kioskItem) => kioskItem.data?.objectId === objectId, + ); + + const handleKioskTransfer = async (to: string) => { + return transferKioskItem.mutateAsync({ to }); + }; + + const handleDirectTransfer = async (to: string) => { + const tx = new Transaction(); + tx.transferObjects([tx.object(objectId)], to); + + return executeFn!({ + transaction: tx, + options: { + showInput: true, + showEffects: true, + showEvents: true, + }, + }); + }; + + return useMutation({ + mutationFn: async (to: string) => { + if (!to || !executeFn) { + throw new Error('Missing data'); + } + + return isContainedInKiosk ? handleKioskTransfer(to) : handleDirectTransfer(to); + }, + onSuccess, + onError, + }); +} diff --git a/apps/wallet/src/ui/app/pages/home/nft-transfer/useTransferKioskItem.tsx b/apps/core/src/hooks/useTransferKioskItem.tsx similarity index 87% rename from apps/wallet/src/ui/app/pages/home/nft-transfer/useTransferKioskItem.tsx rename to apps/core/src/hooks/useTransferKioskItem.tsx index f4edd7930bc..e04b9c98428 100644 --- a/apps/wallet/src/ui/app/pages/home/nft-transfer/useTransferKioskItem.tsx +++ b/apps/core/src/hooks/useTransferKioskItem.tsx @@ -1,8 +1,7 @@ // Copyright (c) Mysten Labs, Inc. // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useActiveAccount } from '_src/ui/app/hooks/useActiveAccount'; -import { useSigner } from '_src/ui/app/hooks/useSigner'; + import { useFeatureValue } from '@growthbook/growthbook-react'; import { useKioskClient, @@ -12,7 +11,8 @@ import { useGetKioskContents, useGetObject, Feature, -} from '@iota/core'; + TransferAssetExecuteFn, +} from '../../'; import { useIotaClient } from '@iota/dapp-kit'; import { KioskTransaction } from '@iota/kiosk'; import { Transaction } from '@iota/iota-sdk/transactions'; @@ -23,14 +23,15 @@ const ORIGINBYTE_PACKAGE_ID = '0x083b02db943238dcea0ff0938a54a17d7575f5b48034506 export function useTransferKioskItem({ objectId, objectType, + executeFn, + address, }: { objectId: string; objectType?: string | null; + executeFn?: TransferAssetExecuteFn; + address?: string | null; }) { const client = useIotaClient(); - const activeAccount = useActiveAccount(); - const signer = useSigner(activeAccount); - const address = activeAccount?.address; const obPackageId = useFeatureValue(Feature.KioskOriginbytePackageid, ORIGINBYTE_PACKAGE_ID); const { data: kioskData } = useGetKioskContents(address); // show personal kiosks too const objectData = useGetObject(objectId); @@ -38,7 +39,7 @@ export function useTransferKioskItem({ return useMutation({ mutationFn: async ({ to }: { to: string }) => { - if (!to || !signer || !objectType) { + if (!to || !executeFn || !objectType) { throw new Error('Missing data'); } @@ -60,8 +61,8 @@ export function useTransferKioskItem({ }) .finalize(); - return signer.signAndExecuteTransaction({ - transactionBlock: txb, + return executeFn({ + transaction: txb, options: { showInput: true, showEffects: true, @@ -99,8 +100,8 @@ export function useTransferKioskItem({ arguments: [tx.object(kioskId), tx.pure.address(to), tx.pure.id(objectId)], }); } - return signer.signAndExecuteTransaction({ - transactionBlock: tx, + return executeFn({ + transaction: tx, options: { showInput: true, showEffects: true, diff --git a/apps/core/src/interfaces/index.ts b/apps/core/src/interfaces/index.ts index 3194d2ae0a1..226be390797 100644 --- a/apps/core/src/interfaces/index.ts +++ b/apps/core/src/interfaces/index.ts @@ -1,5 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export * from './validatorLogo.interfaces'; export * from './balanceChange.interfaces'; +export * from './transactions.interfaces'; +export * from './stakeEvent.interfaces'; diff --git a/apps/core/src/interfaces/stakeEvent.interfaces.ts b/apps/core/src/interfaces/stakeEvent.interfaces.ts new file mode 100644 index 00000000000..1b7514f0a64 --- /dev/null +++ b/apps/core/src/interfaces/stakeEvent.interfaces.ts @@ -0,0 +1,14 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export interface StakeEventJson { + amount: string; + validator_address: string; + epoch: string; +} + +export interface UnstakeEventJson { + principal_amount?: string; + reward_amount?: string; + validator_address?: string; +} diff --git a/apps/wallet-dashboard/lib/interfaces/transactions.interface.ts b/apps/core/src/interfaces/transactions.interfaces.ts similarity index 77% rename from apps/wallet-dashboard/lib/interfaces/transactions.interface.ts rename to apps/core/src/interfaces/transactions.interfaces.ts index 3a7b9c789d8..75ff392dbfd 100644 --- a/apps/wallet-dashboard/lib/interfaces/transactions.interface.ts +++ b/apps/core/src/interfaces/transactions.interfaces.ts @@ -14,8 +14,13 @@ export enum TransactionAction { Send = 'Send', Receive = 'Receive', Transaction = 'Transaction', + Failed = 'Failed', Staked = 'Staked', Unstaked = 'Unstaked', + TimelockedStaked = 'Timelocked Staked', + TimelockedUnstaked = 'Timelocked Unstaked', + TimelockedCollect = 'Timelocked Collect', + Migration = 'Migration', Rewards = 'Rewards', PersonalMessage = 'PersonalMessage', } diff --git a/apps/core/src/interfaces/validatorLogo.interfaces.ts b/apps/core/src/interfaces/validatorLogo.interfaces.ts deleted file mode 100644 index 26b2d558ff2..00000000000 --- a/apps/core/src/interfaces/validatorLogo.interfaces.ts +++ /dev/null @@ -1,9 +0,0 @@ -// Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -export interface ValidatorLogoProps { - address: string; - isSelected: boolean; - showActiveStatus?: boolean; - activeEpoch?: string; -} diff --git a/apps/core/src/types/index.ts b/apps/core/src/types/index.ts index da698c54d8d..80c31f7bb78 100644 --- a/apps/core/src/types/index.ts +++ b/apps/core/src/types/index.ts @@ -6,3 +6,4 @@ export * from './balanceChange'; export * from './objectChange'; export * from './transactionSummary'; export * from './gasSummary'; +export * from './transactionExecute'; diff --git a/apps/core/src/types/renderComponent.ts b/apps/core/src/types/renderComponent.ts index 3059aa3f24d..b28b83ce439 100644 --- a/apps/core/src/types/renderComponent.ts +++ b/apps/core/src/types/renderComponent.ts @@ -3,8 +3,6 @@ import type { PropsWithChildren, JSX } from 'react'; import type { ExplorerLinkConfig } from '../utils/getExplorerLink'; -import type { ValidatorLogoProps } from '../interfaces'; export type RenderExplorerLinkProps = PropsWithChildren; export type RenderExplorerLink = (props: RenderExplorerLinkProps) => JSX.Element; -export type RenderValidatorLogo = (props: ValidatorLogoProps) => JSX.Element; diff --git a/apps/core/src/types/transactionExecute.ts b/apps/core/src/types/transactionExecute.ts new file mode 100644 index 00000000000..9f60d20e0b3 --- /dev/null +++ b/apps/core/src/types/transactionExecute.ts @@ -0,0 +1,13 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { + IotaTransactionBlockResponse, + IotaTransactionBlockResponseOptions, +} from '@iota/iota-sdk/client'; +import { Transaction } from '@iota/iota-sdk/transactions'; + +export type TransferAssetExecuteFn = (input: { + transaction: Transaction; + options?: IotaTransactionBlockResponseOptions; +}) => Promise; diff --git a/apps/core/src/utils/formatAmount.ts b/apps/core/src/utils/formatAmount.ts index a745d5c7552..b241f16556c 100644 --- a/apps/core/src/utils/formatAmount.ts +++ b/apps/core/src/utils/formatAmount.ts @@ -4,7 +4,7 @@ import BigNumber from 'bignumber.js'; -export function formatAmountParts(amount?: BigNumber | bigint | number | string | null) { +export function formatAmountParts(amount?: BigNumber | bigint | number | string | null): string[] { if (typeof amount === 'undefined' || amount === null) { return ['--']; } @@ -29,6 +29,16 @@ export function formatAmountParts(amount?: BigNumber | bigint | number | string bn = bn.decimalPlaces(2, BigNumber.ROUND_DOWN); } + if (bnAbs.gt(0) && bnAbs.lt(1)) { + const leadingZeros = countDecimalLeadingZeros(bn.toFormat()); + + if (leadingZeros >= 4) { + return [formatWithSubscript(bn.toFormat(), leadingZeros), postfix]; + } else { + return [bn.toFormat(leadingZeros + 1), postfix]; + } + } + return [bn.toFormat(), postfix]; } @@ -37,3 +47,45 @@ export function formatAmount(...args: Parameters) { .filter(Boolean) .join(' '); } + +export const countDecimalLeadingZeros = ( + input: BigNumber | bigint | number | string | null, +): number => { + if (input === null) { + return 0; + } + + const [, decimals] = input.toString().split('.'); + + if (!decimals) { + return 0; + } + + let count = 0; + + for (const digit of decimals) { + if (digit === '0') { + count++; + } else { + break; + } + } + + return count; +}; + +const SUBSCRIPTS = ['₀', '₁', '₂', '₃', '₄', '₅', '₆', '₇', '₈', '₉']; + +export const formatWithSubscript = ( + input: BigNumber | bigint | number | string | null, + zeroCount: number, +): string => { + if (input === null) { + return '0'; + } + + const [, decimals] = input.toString().split('.'); + const remainder = decimals.slice(zeroCount); + + return `0.0${SUBSCRIPTS[zeroCount]}${remainder}`; +}; diff --git a/apps/core/src/utils/formatDate.ts b/apps/core/src/utils/formatDate.ts new file mode 100644 index 00000000000..5d386d7afb5 --- /dev/null +++ b/apps/core/src/utils/formatDate.ts @@ -0,0 +1,32 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +type Format = 'year' | 'month' | 'day' | 'hour' | 'minute' | 'second' | 'weekday'; + +export function formatDate( + date: Date | number, + format: Format[] = ['day', 'month', 'hour', 'minute'], +): string { + const dateTime = new Date(date); + if (!(dateTime instanceof Date)) return ''; + + const options = { + year: 'numeric', + month: 'short', + day: 'numeric', + hour: 'numeric', + weekday: 'short', + minute: 'numeric', + second: 'numeric', + }; + + const formatOptions = format.reduce((accumulator, current: Format) => { + const responseObj = { + ...accumulator, + ...{ [current]: options[current] }, + }; + return responseObj; + }, {}); + + return new Intl.DateTimeFormat('en-GB', formatOptions).format(dateTime); +} diff --git a/apps/core/src/utils/index.ts b/apps/core/src/utils/index.ts index cfbe4e93729..98be7606fdb 100644 --- a/apps/core/src/utils/index.ts +++ b/apps/core/src/utils/index.ts @@ -4,6 +4,7 @@ export * from './calculateStakeShare'; export * from './chunkArray'; export * from './formatAmount'; +export * from './formatDate'; export * from './formatPercentageDisplay'; export * from './getRefGasPrice'; export * from './hasDisplayData'; @@ -22,6 +23,7 @@ export * from './getExplorerPaths'; export * from './getExplorerLink'; export * from './truncateString'; export * from './determineCountDownText'; +export * from './toTitleCase'; export * from './stake'; export * from './transaction'; diff --git a/apps/core/src/utils/stake/checkIfIsTimelockedStaking.ts b/apps/core/src/utils/stake/checkIfIsTimelockedStaking.ts new file mode 100644 index 00000000000..b88e6166068 --- /dev/null +++ b/apps/core/src/utils/stake/checkIfIsTimelockedStaking.ts @@ -0,0 +1,25 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import type { IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; +import { STAKING_REQUEST_EVENT, UNSTAKING_REQUEST_EVENT } from '../../constants'; + +export function checkIfIsTimelockedStaking(events: IotaTransactionBlockResponse['events']) { + const TIMELOCKED_STAKING_EVENT_MODULE = 'timelocked_staking'; + if (!events) return { isTimelockedStaking: false, isTimelockedUnstaking: false }; + + const stakeTypeTransaction = events?.find(({ type }) => type === STAKING_REQUEST_EVENT); + const unstakeTypeTransaction = events?.find(({ type }) => type === UNSTAKING_REQUEST_EVENT); + + const isTimelockedStaking = + stakeTypeTransaction?.transactionModule === TIMELOCKED_STAKING_EVENT_MODULE; + const isTimelockedUnstaking = + unstakeTypeTransaction?.transactionModule === TIMELOCKED_STAKING_EVENT_MODULE; + + return { + isTimelockedStaking, + isTimelockedUnstaking, + stakeTypeTransaction, + unstakeTypeTransaction, + }; +} diff --git a/apps/core/src/utils/stake/getStakeDetailsFromEvent.ts b/apps/core/src/utils/stake/getStakeDetailsFromEvent.ts new file mode 100644 index 00000000000..d1e4f406d77 --- /dev/null +++ b/apps/core/src/utils/stake/getStakeDetailsFromEvent.ts @@ -0,0 +1,18 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { StakeEventJson } from '../../interfaces'; +import type { IotaEvent } from '@iota/iota-sdk/client'; + +export function getStakeDetailsFromEvent(event: IotaEvent): { + stakedAmount: string; + validatorAddress: string; + epoch: number; +} { + const eventJson = event.parsedJson as StakeEventJson; + return { + stakedAmount: eventJson.amount || '0', + validatorAddress: eventJson.validator_address || '', + epoch: Number(eventJson.epoch || '0'), + }; +} diff --git a/apps/core/src/utils/stake/getTransactionAmountForTimelocked.ts b/apps/core/src/utils/stake/getTransactionAmountForTimelocked.ts new file mode 100644 index 00000000000..42af4741347 --- /dev/null +++ b/apps/core/src/utils/stake/getTransactionAmountForTimelocked.ts @@ -0,0 +1,24 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import type { IotaEvent } from '@iota/iota-sdk/client'; +import { + getStakeDetailsFromEvent, + getUnstakeDetailsFromEvent, + checkIfIsTimelockedStaking, +} from '.'; + +export function getTransactionAmountForTimelocked( + events: IotaEvent[], +): bigint | undefined | string { + if (!events) return; + const { isTimelockedStaking, isTimelockedUnstaking } = checkIfIsTimelockedStaking(events); + + if (isTimelockedStaking) { + const { stakedAmount } = getStakeDetailsFromEvent(events[0]); + return stakedAmount; + } else if (isTimelockedUnstaking) { + const { totalAmount } = getUnstakeDetailsFromEvent(events[0]); + return totalAmount; + } +} diff --git a/apps/core/src/utils/stake/getUnstakeDetailsFromEvent.ts b/apps/core/src/utils/stake/getUnstakeDetailsFromEvent.ts new file mode 100644 index 00000000000..5a9a13ddc57 --- /dev/null +++ b/apps/core/src/utils/stake/getUnstakeDetailsFromEvent.ts @@ -0,0 +1,23 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { UnstakeEventJson } from '../../interfaces'; +import type { IotaEvent } from '@iota/iota-sdk/client'; + +export function getUnstakeDetailsFromEvent(event: IotaEvent): { + principalAmount: string; + rewardAmount: string; + totalAmount: bigint; + validatorAddress: string; +} { + const eventJson = event.parsedJson as UnstakeEventJson; + const principalAmount = eventJson.principal_amount || '0'; + const rewardAmount = eventJson.reward_amount || '0'; + const totalAmount = BigInt(principalAmount) + BigInt(rewardAmount); + return { + principalAmount, + rewardAmount, + totalAmount, + validatorAddress: eventJson.validator_address || '', + }; +} diff --git a/apps/core/src/utils/stake/index.ts b/apps/core/src/utils/stake/index.ts index 6ecca7353f6..e742b3377ec 100644 --- a/apps/core/src/utils/stake/index.ts +++ b/apps/core/src/utils/stake/index.ts @@ -7,3 +7,7 @@ export * from './createStakeTransaction'; export * from './createTimelockedUnstakeTransaction'; export * from './createTimelockedStakeTransaction'; export * from './createValidationSchema'; +export * from './getStakeDetailsFromEvent'; +export * from './checkIfIsTimelockedStaking'; +export * from './getUnstakeDetailsFromEvent'; +export * from './getTransactionAmountForTimelocked'; diff --git a/apps/core/src/utils/toTitleCase.ts b/apps/core/src/utils/toTitleCase.ts new file mode 100644 index 00000000000..1081f0c2857 --- /dev/null +++ b/apps/core/src/utils/toTitleCase.ts @@ -0,0 +1,10 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export function toTitleCase(str: string): string { + if (!str) return str; + return str + .split(' ') + .map((word) => word.charAt(0).toUpperCase() + word.slice(1).toLowerCase()) + .join(' '); +} diff --git a/apps/core/src/utils/transaction/getLabel.ts b/apps/core/src/utils/transaction/getLabel.ts deleted file mode 100644 index fbcd0e57649..00000000000 --- a/apps/core/src/utils/transaction/getLabel.ts +++ /dev/null @@ -1,11 +0,0 @@ -// Copyright (c) Mysten Labs, Inc. -// Modifications Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 -import { IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; - -// todo: add more logic for deriving transaction label -export const getLabel = (transaction: IotaTransactionBlockResponse, currentAddress?: string) => { - const isSender = transaction.transaction?.data.sender === currentAddress; - // Rename to "Send" to Transaction - return isSender ? 'Transaction' : 'Receive'; -}; diff --git a/apps/core/src/utils/transaction/getTransactionAction.ts b/apps/core/src/utils/transaction/getTransactionAction.ts new file mode 100644 index 00000000000..990576c9bc3 --- /dev/null +++ b/apps/core/src/utils/transaction/getTransactionAction.ts @@ -0,0 +1,42 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; +import { TransactionAction } from '../../interfaces'; +import { checkIfIsTimelockedStaking } from '../stake'; +import { isMigrationTransaction, isUnlockTimelockedObjectTransaction } from '..'; + +export const getTransactionAction = ( + transaction: IotaTransactionBlockResponse, + currentAddress?: string, +) => { + const sender = transaction.transaction?.data.sender; + const { + isTimelockedStaking, + isTimelockedUnstaking, + stakeTypeTransaction, + unstakeTypeTransaction, + } = checkIfIsTimelockedStaking(transaction?.events); + + const isMigration = isMigrationTransaction(transaction.transaction); + const isSupplyIncreaseVestingCollect = isUnlockTimelockedObjectTransaction( + transaction.transaction, + ); + + if (isMigration) { + return TransactionAction.Migration; + } else if (isSupplyIncreaseVestingCollect) { + return TransactionAction.TimelockedCollect; + } else if (stakeTypeTransaction) { + return isTimelockedStaking ? TransactionAction.TimelockedStaked : TransactionAction.Staked; + } else if (unstakeTypeTransaction) { + return isTimelockedUnstaking + ? TransactionAction.TimelockedUnstaked + : TransactionAction.Unstaked; + } else if (sender) { + return sender === currentAddress ? TransactionAction.Send : TransactionAction.Receive; + } else { + return TransactionAction.Transaction; + } +}; diff --git a/apps/core/src/utils/transaction/index.ts b/apps/core/src/utils/transaction/index.ts index f5ef47663b6..375e87ca890 100644 --- a/apps/core/src/utils/transaction/index.ts +++ b/apps/core/src/utils/transaction/index.ts @@ -4,7 +4,7 @@ export * from './getBalanceChangeSummary'; export * from './getObjectChangeSummary'; -export * from './getLabel'; +export * from './getTransactionAction'; export * from './getGasSummary'; export * from './groupByOwner'; export * from './getOwnerType'; @@ -13,3 +13,5 @@ export * from './createTokenTransferTransaction'; export * from './getObjectDisplayLookup'; export * from './createNftSendValidationSchema'; export * from './createUnlockTimelockedObjectsTransaction'; +export * from './isMigrationTransaction'; +export * from './isUnlockTimelockedObjectTransaction'; diff --git a/apps/core/src/utils/transaction/isMigrationTransaction.ts b/apps/core/src/utils/transaction/isMigrationTransaction.ts new file mode 100644 index 00000000000..b7889f2c747 --- /dev/null +++ b/apps/core/src/utils/transaction/isMigrationTransaction.ts @@ -0,0 +1,29 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import type { + IotaTransaction, + IotaTransactionBlockResponse, + MoveCallIotaTransaction, +} from '@iota/iota-sdk/client'; +import { STARDUST_PACKAGE_ID } from '../../constants'; + +export function isMigrationTransaction( + transaction: IotaTransactionBlockResponse['transaction'], +): boolean { + if (!transaction || transaction.data.transaction.kind !== 'ProgrammableTransaction') + return false; + const moveCallTxs = transaction.data.transaction.transactions.filter(isMoveCall); + const isMigration = moveCallTxs.some( + (tx) => + tx.MoveCall.package === STARDUST_PACKAGE_ID && + tx.MoveCall.function === 'extract_assets', + ); + return isMigration; +} + +function isMoveCall( + transaction: IotaTransaction, +): transaction is { MoveCall: MoveCallIotaTransaction } { + return 'MoveCall' in transaction; +} diff --git a/apps/core/src/utils/transaction/isUnlockTimelockedObjectTransaction.ts b/apps/core/src/utils/transaction/isUnlockTimelockedObjectTransaction.ts new file mode 100644 index 00000000000..e0ced9c249e --- /dev/null +++ b/apps/core/src/utils/transaction/isUnlockTimelockedObjectTransaction.ts @@ -0,0 +1,28 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import type { + IotaTransaction, + IotaTransactionBlockResponse, + MoveCallIotaTransaction, +} from '@iota/iota-sdk/client'; +import { TIMELOCK_MODULE } from '../..'; + +export function isUnlockTimelockedObjectTransaction( + transaction: IotaTransactionBlockResponse['transaction'], +): boolean { + if (!transaction || transaction.data.transaction.kind !== 'ProgrammableTransaction') + return false; + const moveCallTxs = transaction.data.transaction.transactions + .filter(isMoveCall) + .filter((tx) => tx.MoveCall.module === TIMELOCK_MODULE); + const isUnlockTimelockedObject = + moveCallTxs.length > 0 && moveCallTxs.every((tx) => tx.MoveCall.function === 'unlock'); + return isUnlockTimelockedObject; +} + +function isMoveCall( + transaction: IotaTransaction, +): transaction is { MoveCall: MoveCallIotaTransaction } { + return 'MoveCall' in transaction; +} diff --git a/apps/core/tailwind.config.ts b/apps/core/tailwind.config.ts index 6add2c1e8b4..917136bd1c6 100644 --- a/apps/core/tailwind.config.ts +++ b/apps/core/tailwind.config.ts @@ -4,7 +4,7 @@ import { type Config } from 'tailwindcss'; import colors from 'tailwindcss/colors'; -import uiKitResponsivePreset from '../../apps/ui-kit/src/lib/tailwind/responsive.preset'; +import uiKitResponsivePreset from '../../apps/ui-kit/src/lib/tailwind/responsive.presets'; export default { presets: [uiKitResponsivePreset], diff --git a/apps/core/tsconfig.json b/apps/core/tsconfig.json new file mode 100644 index 00000000000..1f2191974c4 --- /dev/null +++ b/apps/core/tsconfig.json @@ -0,0 +1,13 @@ +{ + "compilerOptions": { + "jsx": "react-jsx", + "module": "ESNext", + "target": "ES2020", + "strict": true, + "esModuleInterop": true, + "moduleResolution": "node", + "skipLibCheck": true + }, + "include": ["src"], + "exclude": ["node_modules", "dist"] +} diff --git a/apps/explorer/package.json b/apps/explorer/package.json index 8982b0a9d72..96c9e97af5f 100644 --- a/apps/explorer/package.json +++ b/apps/explorer/package.json @@ -27,11 +27,11 @@ "@growthbook/growthbook": "^1.0.0", "@growthbook/growthbook-react": "^1.0.0", "@headlessui/react": "^1.7.15", + "@iota/apps-ui-icons": "workspace:*", "@iota/apps-ui-kit": "workspace:*", "@iota/core": "workspace:*", "@iota/dapp-kit": "workspace:*", "@iota/iota-sdk": "workspace:*", - "@iota/ui-icons": "workspace:*", "@radix-ui/react-collapsible": "^1.0.3", "@radix-ui/react-dropdown-menu": "^2.1.1", "@radix-ui/react-popover": "^1.1.1", diff --git a/apps/explorer/src/components/ThemedIotaLogo.tsx b/apps/explorer/src/components/ThemedIotaLogo.tsx index 71a42a49a8d..de85d75486d 100644 --- a/apps/explorer/src/components/ThemedIotaLogo.tsx +++ b/apps/explorer/src/components/ThemedIotaLogo.tsx @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { IotaLogoWeb } from '@iota/ui-icons'; +import { IotaLogoWeb } from '@iota/apps-ui-icons'; export function ThemedIotaLogo(): React.JSX.Element { return ; diff --git a/apps/explorer/src/components/activity/EpochsActivityTable.tsx b/apps/explorer/src/components/activity/EpochsActivityTable.tsx index f0db0b602ce..00cb1f54849 100644 --- a/apps/explorer/src/components/activity/EpochsActivityTable.tsx +++ b/apps/explorer/src/components/activity/EpochsActivityTable.tsx @@ -5,7 +5,7 @@ import { InfoBox, InfoBoxStyle, InfoBoxType, Select, SelectSize } from '@iota/apps-ui-kit'; import { useIotaClientQuery, useIotaClient, useIotaClientInfiniteQuery } from '@iota/dapp-kit'; import { useCursorPagination } from '@iota/core'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; import { useQuery } from '@tanstack/react-query'; import { useState } from 'react'; import { PlaceholderTable, TableCard } from '~/components/ui'; diff --git a/apps/explorer/src/components/activity/TransactionsActivityTable.tsx b/apps/explorer/src/components/activity/TransactionsActivityTable.tsx index e5f06e71af0..78bbd1bb25e 100644 --- a/apps/explorer/src/components/activity/TransactionsActivityTable.tsx +++ b/apps/explorer/src/components/activity/TransactionsActivityTable.tsx @@ -14,7 +14,7 @@ import { import { numberSuffix } from '~/lib/utils'; import { InfoBox, InfoBoxStyle, InfoBoxType, Select, SelectSize } from '@iota/apps-ui-kit'; import { generateTransactionsTableColumns } from '~/lib/ui'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; import { PAGE_SIZES_RANGE_20_60 } from '~/lib/constants'; interface TransactionsActivityTableProps { diff --git a/apps/explorer/src/components/checkpoints/CheckpointsTable.tsx b/apps/explorer/src/components/checkpoints/CheckpointsTable.tsx index 6d90547d25e..67b1e5680fe 100644 --- a/apps/explorer/src/components/checkpoints/CheckpointsTable.tsx +++ b/apps/explorer/src/components/checkpoints/CheckpointsTable.tsx @@ -4,7 +4,7 @@ import { InfoBox, InfoBoxStyle, InfoBoxType, Select, SelectSize } from '@iota/apps-ui-kit'; import { useIotaClientQuery } from '@iota/dapp-kit'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; import { useMemo, useState } from 'react'; import { PlaceholderTable, TableCard } from '~/components/ui'; import { DEFAULT_CHECKPOINTS_LIMIT, useGetCheckpoints } from '~/hooks/useGetCheckpoints'; diff --git a/apps/explorer/src/components/error-boundary/ErrorBoundary.tsx b/apps/explorer/src/components/error-boundary/ErrorBoundary.tsx index f62bd7fd95c..02f9114f70f 100644 --- a/apps/explorer/src/components/error-boundary/ErrorBoundary.tsx +++ b/apps/explorer/src/components/error-boundary/ErrorBoundary.tsx @@ -7,7 +7,7 @@ import { useLocation } from 'react-router-dom'; import type { ReactNode } from 'react'; import type { FallbackProps } from 'react-error-boundary'; import { InfoBox, InfoBoxType, InfoBoxStyle } from '@iota/apps-ui-kit'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; function Fallback({ error }: FallbackProps): JSX.Element { return ( diff --git a/apps/explorer/src/components/footer/Footer.tsx b/apps/explorer/src/components/footer/Footer.tsx index 8779d6cc52f..93a2f629ed1 100644 --- a/apps/explorer/src/components/footer/Footer.tsx +++ b/apps/explorer/src/components/footer/Footer.tsx @@ -28,7 +28,7 @@ function FooterLinks(): JSX.Element { ); } -function Footer(): JSX.Element { +export function Footer(): JSX.Element { return (
); } - -export default Footer; diff --git a/apps/explorer/src/components/gas-breakdown/GasBreakdown.tsx b/apps/explorer/src/components/gas-breakdown/GasBreakdown.tsx index a226583b306..34e84d2f645 100644 --- a/apps/explorer/src/components/gas-breakdown/GasBreakdown.tsx +++ b/apps/explorer/src/components/gas-breakdown/GasBreakdown.tsx @@ -10,7 +10,7 @@ import { useFormatCoin, } from '@iota/core'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; -import { Copy } from '@iota/ui-icons'; +import { Copy } from '@iota/apps-ui-icons'; import toast from 'react-hot-toast'; import { AddressLink, CollapsibleCard, ObjectLink } from '~/components/ui'; diff --git a/apps/explorer/src/components/header/Header.tsx b/apps/explorer/src/components/header/Header.tsx index a2f95b165d5..9d1d428f7cd 100644 --- a/apps/explorer/src/components/header/Header.tsx +++ b/apps/explorer/src/components/header/Header.tsx @@ -3,12 +3,12 @@ // SPDX-License-Identifier: Apache-2.0 import { NetworkSelector } from '../network'; -import Search from '../search/Search'; +import { Search } from '../search'; import { LinkWithQuery } from '~/components/ui'; import { ThemedIotaLogo } from '~/components'; import { ThemeSwitcher } from '@iota/core'; -function Header(): JSX.Element { +export function Header(): JSX.Element { return (
@@ -30,5 +30,3 @@ function Header(): JSX.Element {
); } - -export default Header; diff --git a/apps/explorer/src/components/layout/PageLayout.tsx b/apps/explorer/src/components/layout/PageLayout.tsx index 8b8192d799e..da1a7ab00e9 100644 --- a/apps/explorer/src/components/layout/PageLayout.tsx +++ b/apps/explorer/src/components/layout/PageLayout.tsx @@ -7,11 +7,11 @@ import { useAppsBackend, Feature } from '@iota/core'; import { Network } from '@iota/iota-sdk/client'; import { useQuery } from '@tanstack/react-query'; import { type ReactNode, useRef } from 'react'; -import Footer from '../footer/Footer'; -import Header from '../header/Header'; +import { Footer } from '../footer'; +import { Header } from '../header'; import { useNetworkContext } from '~/contexts'; import { InfoBox, InfoBoxStyle, InfoBoxType, LoadingIndicator } from '@iota/apps-ui-kit'; -import { Info } from '@iota/ui-icons'; +import { Info } from '@iota/apps-ui-icons'; type PageLayoutProps = { content: ReactNode; diff --git a/apps/explorer/src/components/module/module-functions-interaction/FunctionExecutionResult.tsx b/apps/explorer/src/components/module/module-functions-interaction/FunctionExecutionResult.tsx index ef2bbb9a238..d81f63a2194 100644 --- a/apps/explorer/src/components/module/module-functions-interaction/FunctionExecutionResult.tsx +++ b/apps/explorer/src/components/module/module-functions-interaction/FunctionExecutionResult.tsx @@ -4,7 +4,7 @@ import { LinkGroup } from './LinkGroup'; import type { IotaTransactionBlockResponse, OwnedObjectRef } from '@iota/iota-sdk/client'; -import { Close } from '@iota/ui-icons'; +import { Close } from '@iota/apps-ui-icons'; interface ToObjectLink { text: string; diff --git a/apps/explorer/src/components/module/module-functions-interaction/ModuleFunctionsInteraction.tsx b/apps/explorer/src/components/module/module-functions-interaction/ModuleFunctionsInteraction.tsx index 2e914e5a6cd..56cd9e3c82b 100644 --- a/apps/explorer/src/components/module/module-functions-interaction/ModuleFunctionsInteraction.tsx +++ b/apps/explorer/src/components/module/module-functions-interaction/ModuleFunctionsInteraction.tsx @@ -6,7 +6,7 @@ import { useMemo } from 'react'; import { ModuleFunction } from './ModuleFunction'; import { useNormalizedMoveModule } from '~/hooks/useNormalizedMoveModule'; import { InfoBox, InfoBoxStyle, InfoBoxType, LoadingIndicator } from '@iota/apps-ui-kit'; -import { Info, Warning } from '@iota/ui-icons'; +import { Info, Warning } from '@iota/apps-ui-icons'; type ModuleFunctionsInteractionProps = { packageId: string; diff --git a/apps/explorer/src/components/network/NetworkSelector.tsx b/apps/explorer/src/components/network/NetworkSelector.tsx index deb20e35e9b..05f56f46e5e 100644 --- a/apps/explorer/src/components/network/NetworkSelector.tsx +++ b/apps/explorer/src/components/network/NetworkSelector.tsx @@ -11,7 +11,7 @@ import { CustomRPCInput } from '~/components/ui'; import { ampli } from '~/lib/utils'; import { type NetworkId, getAllNetworks } from '@iota/iota-sdk/client'; import { Button, ButtonSize, ButtonType, Dropdown, ListItem } from '@iota/apps-ui-kit'; -import { ArrowDown, CheckmarkFilled } from '@iota/ui-icons'; +import { ArrowDown, CheckmarkFilled } from '@iota/apps-ui-icons'; import { Transition } from '@headlessui/react'; interface NetworkOption { diff --git a/apps/explorer/src/components/object/FieldItem.tsx b/apps/explorer/src/components/object/FieldItem.tsx index 50fc0d0f6c3..25fdc86df61 100644 --- a/apps/explorer/src/components/object/FieldItem.tsx +++ b/apps/explorer/src/components/object/FieldItem.tsx @@ -49,7 +49,7 @@ export function FieldItem({ if (normalizedType === TYPE_URL) { return ( -
+
{value} @@ -58,7 +58,7 @@ export function FieldItem({ } return ( -
+
{value === null || value === undefined ? null : String(value)}
); diff --git a/apps/explorer/src/components/object/ObjectFieldsCard.tsx b/apps/explorer/src/components/object/ObjectFieldsCard.tsx index db5390ed143..8f2cdd62087 100644 --- a/apps/explorer/src/components/object/ObjectFieldsCard.tsx +++ b/apps/explorer/src/components/object/ObjectFieldsCard.tsx @@ -24,7 +24,7 @@ import { InfoBoxStyle, InfoBoxType, } from '@iota/apps-ui-kit'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; const DEFAULT_OPEN_FIELDS = 3; const DEFAULT_FIELDS_COUNT_TO_SHOW_SEARCH = 10; diff --git a/apps/explorer/src/components/object/UnderlyingObjectCard.tsx b/apps/explorer/src/components/object/UnderlyingObjectCard.tsx index 7d4ce9061e9..824ae045be1 100644 --- a/apps/explorer/src/components/object/UnderlyingObjectCard.tsx +++ b/apps/explorer/src/components/object/UnderlyingObjectCard.tsx @@ -6,7 +6,7 @@ import { useIotaClientQuery } from '@iota/dapp-kit'; import { FieldItem } from './FieldItem'; import type { DynamicFieldName } from '@iota/iota-sdk/client'; import { InfoBox, InfoBoxStyle, InfoBoxType, LoadingIndicator } from '@iota/apps-ui-kit'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; interface UnderlyingObjectCardProps { parentId: string; diff --git a/apps/explorer/src/components/owned-coins/CoinItem.tsx b/apps/explorer/src/components/owned-coins/CoinItem.tsx index d745b9202c3..c06e2d6def5 100644 --- a/apps/explorer/src/components/owned-coins/CoinItem.tsx +++ b/apps/explorer/src/components/owned-coins/CoinItem.tsx @@ -12,7 +12,7 @@ interface CoinItemProps { coin: CoinStruct; } -export default function CoinItem({ coin }: CoinItemProps): JSX.Element { +export function CoinItem({ coin }: CoinItemProps): JSX.Element { const [formattedBalance, symbol] = useFormatCoin(coin.balance, coin.coinType); return ( (isIotaCoin); const [formattedTotalBalance, symbol] = useFormatCoin(coin.totalBalance, coin.coinType); diff --git a/apps/explorer/src/components/owned-coins/OwnedCoins.tsx b/apps/explorer/src/components/owned-coins/OwnedCoins.tsx index 619fd8e9864..6f25165f27c 100644 --- a/apps/explorer/src/components/owned-coins/OwnedCoins.tsx +++ b/apps/explorer/src/components/owned-coins/OwnedCoins.tsx @@ -7,9 +7,9 @@ import { useIotaClientQuery } from '@iota/dapp-kit'; import { type CoinBalance, type Network } from '@iota/iota-sdk/client'; import { useNetwork } from '~/hooks'; import { normalizeIotaAddress } from '@iota/iota-sdk/utils'; -import { FilterList, Warning } from '@iota/ui-icons'; +import { FilterList, Warning } from '@iota/apps-ui-icons'; import { useMemo, useState } from 'react'; -import OwnedCoinView from './OwnedCoinView'; +import { OwnedCoinView } from './OwnedCoinView'; import { Button, ButtonType, diff --git a/apps/explorer/src/components/owned-coins/OwnedCoinsPanel.tsx b/apps/explorer/src/components/owned-coins/OwnedCoinsPanel.tsx index 2f2895ab885..5d9fd09d144 100644 --- a/apps/explorer/src/components/owned-coins/OwnedCoinsPanel.tsx +++ b/apps/explorer/src/components/owned-coins/OwnedCoinsPanel.tsx @@ -5,7 +5,7 @@ import { useElementDimensions, useGetCoins, useOnScreen } from '@iota/core'; import clsx from 'clsx'; import { useEffect, useRef } from 'react'; -import CoinItem from './CoinItem'; +import { CoinItem } from './CoinItem'; import { LoadingIndicator } from '@iota/apps-ui-kit'; const MIN_CONTAINER_WIDTH_SIZE = 500; @@ -15,7 +15,7 @@ type CoinsPanelProps = { id: string; }; -export default function CoinsPanel({ coinType, id }: CoinsPanelProps): JSX.Element { +export function CoinsPanel({ coinType, id }: CoinsPanelProps): JSX.Element { const containerRef = useRef(null); const coinsSectionRef = useRef(null); const { isIntersecting } = useOnScreen(containerRef); diff --git a/apps/explorer/src/components/owned-objects/OwnedObjects.tsx b/apps/explorer/src/components/owned-objects/OwnedObjects.tsx index 4323c7301c3..32f4559062e 100644 --- a/apps/explorer/src/components/owned-objects/OwnedObjects.tsx +++ b/apps/explorer/src/components/owned-objects/OwnedObjects.tsx @@ -27,7 +27,7 @@ import { InfoBoxStyle, InfoBoxType, } from '@iota/apps-ui-kit'; -import { ListViewLarge, ListViewMedium, ListViewSmall, Warning } from '@iota/ui-icons'; +import { ListViewLarge, ListViewMedium, ListViewSmall, Warning } from '@iota/apps-ui-icons'; import clsx from 'clsx'; import { useEffect, useMemo, useState } from 'react'; import { ListView, NoObjectsOwnedMessage, SmallThumbnailsView, ThumbnailsView } from '~/components'; @@ -68,9 +68,15 @@ const VIEW_MODES = [ function getItemsRangeFromCurrentPage( currentPage: number, itemsPerPage: number, + availableItems?: number, ): ItemsRangeFromCurrentPage { const start = currentPage * itemsPerPage + 1; - const end = start + itemsPerPage - 1; + let end = start + itemsPerPage - 1; + + if (availableItems && availableItems < itemsPerPage) { + end = start + availableItems - 1; + } + return { start, end }; } @@ -143,11 +149,7 @@ export function OwnedObjects({ id }: OwnedObjectsProps): JSX.Element { ); const { start, end } = useMemo( - () => - getItemsRangeFromCurrentPage( - pagination.currentPage, - filteredData?.length || PAGE_SIZES_RANGE_10_50[0], - ), + () => getItemsRangeFromCurrentPage(pagination.currentPage, limit, filteredData?.length), [filteredData?.length, pagination.currentPage], ); diff --git a/apps/explorer/src/components/owned-objects/SmallThumbnailsView.tsx b/apps/explorer/src/components/owned-objects/SmallThumbnailsView.tsx index c9095bc12c1..8c1ef9539b3 100644 --- a/apps/explorer/src/components/owned-objects/SmallThumbnailsView.tsx +++ b/apps/explorer/src/components/owned-objects/SmallThumbnailsView.tsx @@ -5,7 +5,7 @@ import { Tooltip, TooltipPosition } from '@iota/apps-ui-kit'; import { type IotaObjectResponse } from '@iota/iota-sdk/client'; import { formatAddress } from '@iota/iota-sdk/utils'; -import { Info, Loader } from '@iota/ui-icons'; +import { Info, Loader } from '@iota/apps-ui-icons'; import { type ReactNode } from 'react'; import { ObjectLink, ObjectVideoImage } from '~/components/ui'; import { useResolveVideo } from '~/hooks/useResolveVideo'; diff --git a/apps/explorer/src/components/owned-objects/ThumbnailsView.tsx b/apps/explorer/src/components/owned-objects/ThumbnailsView.tsx index 2f397d42f2e..45b1a86a26f 100644 --- a/apps/explorer/src/components/owned-objects/ThumbnailsView.tsx +++ b/apps/explorer/src/components/owned-objects/ThumbnailsView.tsx @@ -4,7 +4,7 @@ import { type IotaObjectResponse } from '@iota/iota-sdk/client'; import { formatAddress } from '@iota/iota-sdk/utils'; -import { Loader } from '@iota/ui-icons'; +import { Loader } from '@iota/apps-ui-icons'; import { ObjectLink, ObjectVideoImage } from '~/components/ui'; import { useResolveVideo } from '~/hooks/useResolveVideo'; import { parseObjectType, trimStdLibPrefix } from '~/lib/utils'; diff --git a/apps/explorer/src/components/search/Search.tsx b/apps/explorer/src/components/search/Search.tsx index d2a0f41eebf..61a5bfaa2d4 100644 --- a/apps/explorer/src/components/search/Search.tsx +++ b/apps/explorer/src/components/search/Search.tsx @@ -10,7 +10,7 @@ import { useDebouncedValue } from '~/hooks/useDebouncedValue'; import { useSearch } from '~/hooks/useSearch'; import { ampli } from '~/lib/utils'; -function Search(): JSX.Element { +export function Search(): JSX.Element { const [query, setQuery] = useState(''); const debouncedQuery = useDebouncedValue(query); const { isPending, data: results } = useSearch(debouncedQuery); @@ -58,5 +58,3 @@ function Search(): JSX.Element { /> ); } - -export default Search; diff --git a/apps/explorer/src/components/top-validators-card/TopValidatorsCard.tsx b/apps/explorer/src/components/top-validators-card/TopValidatorsCard.tsx index 2f4f653a9cc..1d43fadbd8b 100644 --- a/apps/explorer/src/components/top-validators-card/TopValidatorsCard.tsx +++ b/apps/explorer/src/components/top-validators-card/TopValidatorsCard.tsx @@ -3,11 +3,20 @@ // SPDX-License-Identifier: Apache-2.0 import { useIotaClientQuery } from '@iota/dapp-kit'; -import { PlaceholderTable, TableCard } from '~/components/ui'; +import { Link, PlaceholderTable, TableCard } from '~/components/ui'; import { generateValidatorsTableColumns } from '~/lib/ui'; -import { InfoBox, InfoBoxStyle, InfoBoxType, Panel, Title } from '@iota/apps-ui-kit'; +import { + Button, + ButtonSize, + ButtonType, + InfoBox, + InfoBoxStyle, + InfoBoxType, + Panel, + Title, +} from '@iota/apps-ui-kit'; import { ErrorBoundary } from '../error-boundary/ErrorBoundary'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; const NUMBER_OF_VALIDATORS = 10; @@ -19,6 +28,9 @@ type TopValidatorsCardProps = { export function TopValidatorsCard({ limit, showIcon }: TopValidatorsCardProps): JSX.Element { const { data, isPending, isSuccess, isError } = useIotaClientQuery('getLatestIotaSystemState'); + const topActiveValidators = + data?.activeValidators.slice(0, limit || NUMBER_OF_VALIDATORS) ?? []; + const tableColumns = generateValidatorsTableColumns({ atRiskValidators: [], validatorEvents: [], @@ -42,26 +54,33 @@ export function TopValidatorsCard({ limit, showIcon }: TopValidatorsCardProps): return ( - - - <div className="p-md"> - {isPending && ( - <PlaceholderTable - rowCount={limit || NUMBER_OF_VALIDATORS} - rowHeight="13px" - colHeadings={['Name', 'Address', 'Stake']} - /> - )} + <div className="relative"> + <div className="absolute right-0 mr-4 mt-2"> + <Link to="/validators"> + <Button + type={ButtonType.Secondary} + size={ButtonSize.Small} + text="View All" + /> + </Link> + </div> + <Title title="Top Validators" /> - {isSuccess && ( - <ErrorBoundary> - <TableCard - data={data.activeValidators} - columns={tableColumns} - viewAll="/validators" + <div className="p-md"> + {isPending && ( + <PlaceholderTable + rowCount={limit || NUMBER_OF_VALIDATORS} + rowHeight="13px" + colHeadings={['Name', 'Address', 'Stake']} /> - </ErrorBoundary> - )} + )} + + {isSuccess && ( + <ErrorBoundary> + <TableCard data={topActiveValidators} columns={tableColumns} /> + </ErrorBoundary> + )} + </div> </div> </Panel> ); diff --git a/apps/explorer/src/components/transactions/TransactionsForAddress.tsx b/apps/explorer/src/components/transactions/TransactionsForAddress.tsx index e3547919a9b..97de53ef95a 100644 --- a/apps/explorer/src/components/transactions/TransactionsForAddress.tsx +++ b/apps/explorer/src/components/transactions/TransactionsForAddress.tsx @@ -5,7 +5,7 @@ import { InfoBox, InfoBoxStyle, InfoBoxType, LoadingIndicator } from '@iota/apps-ui-kit'; import { useIotaClient } from '@iota/dapp-kit'; import { type IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; import { useQuery } from '@tanstack/react-query'; import { TableCard } from '~/components/ui'; import { generateTransactionsTableColumns } from '~/lib/ui'; diff --git a/apps/explorer/src/components/ui/ExpandableList.tsx b/apps/explorer/src/components/ui/ExpandableList.tsx index 73060d50085..abf14e4ba37 100644 --- a/apps/explorer/src/components/ui/ExpandableList.tsx +++ b/apps/explorer/src/components/ui/ExpandableList.tsx @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import { type ReactNode, useMemo, useState, createContext, useContext } from 'react'; -import { TriangleDown } from '@iota/ui-icons'; +import { TriangleDown } from '@iota/apps-ui-icons'; import { Button, ButtonSize, ButtonType } from '@iota/apps-ui-kit'; import clsx from 'clsx'; diff --git a/apps/explorer/src/components/ui/ObjectVideoImage.tsx b/apps/explorer/src/components/ui/ObjectVideoImage.tsx index 36584243015..454193699b6 100644 --- a/apps/explorer/src/components/ui/ObjectVideoImage.tsx +++ b/apps/explorer/src/components/ui/ObjectVideoImage.tsx @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { Play } from '@iota/ui-icons'; +import { Play } from '@iota/apps-ui-icons'; import { cva, type VariantProps } from 'class-variance-authority'; import clsx from 'clsx'; diff --git a/apps/explorer/src/components/ui/PageHeader.tsx b/apps/explorer/src/components/ui/PageHeader.tsx index ada56b11ffc..cdf02bcf77c 100644 --- a/apps/explorer/src/components/ui/PageHeader.tsx +++ b/apps/explorer/src/components/ui/PageHeader.tsx @@ -11,7 +11,7 @@ import { Panel, Placeholder, } from '@iota/apps-ui-kit'; -import { Copy, Warning } from '@iota/ui-icons'; +import { Copy, Warning } from '@iota/apps-ui-icons'; import { useCopyToClipboard } from '@iota/core'; import toast from 'react-hot-toast'; diff --git a/apps/explorer/src/components/ui/Pagination.tsx b/apps/explorer/src/components/ui/Pagination.tsx index 3dceec84667..8d39b2da1e0 100644 --- a/apps/explorer/src/components/ui/Pagination.tsx +++ b/apps/explorer/src/components/ui/Pagination.tsx @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import { Button, ButtonSize, ButtonType } from '@iota/apps-ui-kit'; -import { ArrowLeft, ArrowRight, DoubleArrowLeft } from '@iota/ui-icons'; +import { ArrowLeft, ArrowRight, DoubleArrowLeft } from '@iota/apps-ui-icons'; import { useState } from 'react'; interface PaginationProps { diff --git a/apps/explorer/src/components/ui/PlayPause.tsx b/apps/explorer/src/components/ui/PlayPause.tsx index 451ae069b42..977ff83223f 100644 --- a/apps/explorer/src/components/ui/PlayPause.tsx +++ b/apps/explorer/src/components/ui/PlayPause.tsx @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import { ButtonUnstyled } from '@iota/apps-ui-kit'; -import { Pause, Play } from '@iota/ui-icons'; +import { Pause, Play } from '@iota/apps-ui-icons'; import { motion } from 'framer-motion'; import { useEffect } from 'react'; diff --git a/apps/explorer/src/components/ui/SplitPanes.tsx b/apps/explorer/src/components/ui/SplitPanes.tsx index be26b5d04f8..53fc5fec872 100644 --- a/apps/explorer/src/components/ui/SplitPanes.tsx +++ b/apps/explorer/src/components/ui/SplitPanes.tsx @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import { Button, ButtonSize, ButtonType } from '@iota/apps-ui-kit'; -import { ArrowRight, ArrowUp } from '@iota/ui-icons'; +import { ArrowRight, ArrowUp } from '@iota/apps-ui-icons'; import { cva, type VariantProps } from 'class-variance-authority'; import clsx from 'clsx'; import { type ReactNode, useRef, useState } from 'react'; diff --git a/apps/explorer/src/components/ui/image/Image.tsx b/apps/explorer/src/components/ui/image/Image.tsx index 63e18775839..39b1131a5c9 100644 --- a/apps/explorer/src/components/ui/image/Image.tsx +++ b/apps/explorer/src/components/ui/image/Image.tsx @@ -3,13 +3,12 @@ // SPDX-License-Identifier: Apache-2.0 import { LoadingIndicator } from '@iota/apps-ui-kit'; -import { PlaceholderReplace, VisibilityOff } from '@iota/ui-icons'; +import { PlaceholderReplace, VisibilityOff } from '@iota/apps-ui-icons'; import { cva, cx, type VariantProps } from 'class-variance-authority'; import clsx from 'clsx'; import { useAnimate } from 'framer-motion'; import { type ImgHTMLAttributes, useEffect, useState } from 'react'; - -import useImage from '~/hooks/useImage'; +import { useImage } from '~/hooks'; import { ImageVisibility } from '~/lib/enums'; const imageStyles = cva(null, { diff --git a/apps/explorer/src/components/ui/modal/Modal.tsx b/apps/explorer/src/components/ui/modal/Modal.tsx index 236a054d7a2..6d66cf820ad 100644 --- a/apps/explorer/src/components/ui/modal/Modal.tsx +++ b/apps/explorer/src/components/ui/modal/Modal.tsx @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import { Dialog, Transition } from '@headlessui/react'; -import { Close } from '@iota/ui-icons'; +import { Close } from '@iota/apps-ui-icons'; import { Fragment, type ReactNode } from 'react'; export interface ModalProps { diff --git a/apps/explorer/src/components/ui/modal/ObjectModal.tsx b/apps/explorer/src/components/ui/modal/ObjectModal.tsx index 3e44d3c0536..26ae2a6d89f 100644 --- a/apps/explorer/src/components/ui/modal/ObjectModal.tsx +++ b/apps/explorer/src/components/ui/modal/ObjectModal.tsx @@ -4,7 +4,7 @@ import { Modal, type ModalProps } from './Modal'; import { Image } from '../image/Image'; -import { Close } from '@iota/ui-icons'; +import { Close } from '@iota/apps-ui-icons'; export interface ObjectModalProps extends Omit<ModalProps, 'children'> { title: string; diff --git a/apps/explorer/src/components/validator/ValidatorMeta.tsx b/apps/explorer/src/components/validator/ValidatorMeta.tsx index 1b04666cf06..5543344c591 100644 --- a/apps/explorer/src/components/validator/ValidatorMeta.tsx +++ b/apps/explorer/src/components/validator/ValidatorMeta.tsx @@ -5,7 +5,7 @@ import { Badge, BadgeType, KeyValueInfo, Panel } from '@iota/apps-ui-kit'; import { type IotaValidatorSummary } from '@iota/iota-sdk/client'; import toast from 'react-hot-toast'; -import { ArrowTopRight } from '@iota/ui-icons'; +import { ArrowTopRight } from '@iota/apps-ui-icons'; import { AddressLink } from '~/components/ui'; import { ImageIcon, ImageIconSize } from '@iota/core'; diff --git a/apps/explorer/src/components/validator/ValidatorStats.tsx b/apps/explorer/src/components/validator/ValidatorStats.tsx index a39b65a7eec..06992a863a1 100644 --- a/apps/explorer/src/components/validator/ValidatorStats.tsx +++ b/apps/explorer/src/components/validator/ValidatorStats.tsx @@ -149,7 +149,7 @@ export function ValidatorStats({ tooltipText={ !tallyingScore ? 'Coming soon' - : 'A score created by validators to assess each other’s performance during Iota’s standard operations.' + : 'A score created by validators to assess each other’s performance during IOTA’s standard operations.' } tooltipPosition={TooltipPosition.Right} /> diff --git a/apps/explorer/src/hooks/useImage.ts b/apps/explorer/src/hooks/useImage.ts index f750e4ac715..50018b6568f 100644 --- a/apps/explorer/src/hooks/useImage.ts +++ b/apps/explorer/src/hooks/useImage.ts @@ -47,5 +47,3 @@ export function useImage({ src = '', moderate = true }: UseImageProps) { return { moderation, url: formatted, status, ref }; } - -export default useImage; diff --git a/apps/explorer/src/index.tsx b/apps/explorer/src/index.tsx index 670a7e6ad93..aad5ea550a5 100644 --- a/apps/explorer/src/index.tsx +++ b/apps/explorer/src/index.tsx @@ -9,7 +9,6 @@ import { QueryClientProvider } from '@tanstack/react-query'; import React from 'react'; import ReactDOM from 'react-dom/client'; import { RouterProvider } from 'react-router-dom'; - import { growthbook, initAmplitude, initSentry, queryClient } from './lib/utils'; import { router } from './pages'; diff --git a/apps/explorer/src/lib/enums/imageVisibility.enum.ts b/apps/explorer/src/lib/enums/imageVisibility.enums.ts similarity index 100% rename from apps/explorer/src/lib/enums/imageVisibility.enum.ts rename to apps/explorer/src/lib/enums/imageVisibility.enums.ts diff --git a/apps/explorer/src/lib/enums/index.ts b/apps/explorer/src/lib/enums/index.ts index 21eccaf2e98..459938152ff 100644 --- a/apps/explorer/src/lib/enums/index.ts +++ b/apps/explorer/src/lib/enums/index.ts @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export * from './objectViewMode.enum'; -export * from './objectFilterValue.enum'; -export * from './imageVisibility.enum'; -export * from './splitPaneKey.enum'; +export * from './objectViewMode.enums'; +export * from './objectFilterValue.enums'; +export * from './imageVisibility.enums'; +export * from './splitPaneKey.enums'; diff --git a/apps/explorer/src/lib/enums/objectFilterValue.enum.ts b/apps/explorer/src/lib/enums/objectFilterValue.enums.ts similarity index 100% rename from apps/explorer/src/lib/enums/objectFilterValue.enum.ts rename to apps/explorer/src/lib/enums/objectFilterValue.enums.ts diff --git a/apps/explorer/src/lib/enums/objectViewMode.enum.ts b/apps/explorer/src/lib/enums/objectViewMode.enums.ts similarity index 100% rename from apps/explorer/src/lib/enums/objectViewMode.enum.ts rename to apps/explorer/src/lib/enums/objectViewMode.enums.ts diff --git a/apps/explorer/src/lib/enums/splitPaneKey.enum.ts b/apps/explorer/src/lib/enums/splitPaneKey.enums.ts similarity index 100% rename from apps/explorer/src/lib/enums/splitPaneKey.enum.ts rename to apps/explorer/src/lib/enums/splitPaneKey.enums.ts diff --git a/apps/explorer/src/lib/ui/utils/generateCheckpointsTableColumns.tsx b/apps/explorer/src/lib/ui/utils/generateCheckpointsTableColumns.tsx index 27db60a3252..091dab73950 100644 --- a/apps/explorer/src/lib/ui/utils/generateCheckpointsTableColumns.tsx +++ b/apps/explorer/src/lib/ui/utils/generateCheckpointsTableColumns.tsx @@ -46,12 +46,12 @@ export function generateCheckpointsTableColumns(): ColumnDef<Checkpoint>[] { }, { header: 'Transactions', - accessorKey: 'networkTotalTransactions', + accessorKey: 'transactions', cell: ({ getValue }) => { - const networkTotalTransactions = getValue<Checkpoint['networkTotalTransactions']>(); + const transactions = getValue<Checkpoint['transactions']>(); return ( <TableCellBase> - <TableCellText>{networkTotalTransactions}</TableCellText> + <TableCellText>{transactions.length}</TableCellText> </TableCellBase> ); }, @@ -70,17 +70,5 @@ export function generateCheckpointsTableColumns(): ColumnDef<Checkpoint>[] { ); }, }, - { - header: 'Transaction Block Count', - accessorKey: 'transactions', - cell: ({ getValue }) => { - const transactions = getValue<Checkpoint['transactions']>(); - return ( - <TableCellBase> - <TableCellText>{transactions.length}</TableCellText> - </TableCellBase> - ); - }, - }, ]; } diff --git a/apps/explorer/src/lib/ui/utils/generateTransactionsTableColumns.tsx b/apps/explorer/src/lib/ui/utils/generateTransactionsTableColumns.tsx index 3cc12a45e2a..ace65b6ba98 100644 --- a/apps/explorer/src/lib/ui/utils/generateTransactionsTableColumns.tsx +++ b/apps/explorer/src/lib/ui/utils/generateTransactionsTableColumns.tsx @@ -89,11 +89,12 @@ export function generateTransactionsTableColumns(): ColumnDef<IotaTransactionBlo accessorKey: 'timestampMs', cell: ({ getValue }) => { const timestampMs = getValue(); + const elapsedTime = timestampMs + ? getElapsedTime(Number(timestampMs), Date.now()) + : '--'; return ( <TableCellBase> - <TableCellText> - {getElapsedTime(Number(timestampMs), Date.now()) || '--'} - </TableCellText> + <TableCellText>{elapsedTime}</TableCellText> </TableCellBase> ); }, diff --git a/apps/explorer/src/lib/utils/sentry.ts b/apps/explorer/src/lib/utils/sentry.ts index e8c3f689aab..d49b9db38a8 100644 --- a/apps/explorer/src/lib/utils/sentry.ts +++ b/apps/explorer/src/lib/utils/sentry.ts @@ -24,14 +24,12 @@ export function initSentry() { : 'https://c8085701fa2650fb2a090ed6aba6bc62@o4508279186718720.ingest.de.sentry.io/4508279963320400', environment: import.meta.env.VITE_VERCEL_ENV, integrations: [ - new Sentry.BrowserTracing({ - routingInstrumentation: Sentry.reactRouterV6Instrumentation( - useEffect, - useLocation, - useNavigationType, - createRoutesFromChildren, - matchRoutes, - ), + Sentry.reactRouterV6BrowserTracingIntegration({ + useEffect, + useLocation, + useNavigationType, + createRoutesFromChildren, + matchRoutes, }), ], tracesSampleRate: SENTRY_SAMPLE_RATE, diff --git a/apps/explorer/src/pages/address-result/AddressResult.tsx b/apps/explorer/src/pages/address-result/AddressResult.tsx index 32808a8d796..7f55716dbaf 100644 --- a/apps/explorer/src/pages/address-result/AddressResult.tsx +++ b/apps/explorer/src/pages/address-result/AddressResult.tsx @@ -47,7 +47,7 @@ function AddressResult({ address }: { address: string }): JSX.Element { ); } -export default function AddressResultPage(): JSX.Element { +export function AddressResultPage(): JSX.Element { const { id } = useParams(); return ( diff --git a/apps/explorer/src/pages/checkpoints/CheckpointDetail.tsx b/apps/explorer/src/pages/checkpoints/CheckpointDetail.tsx index 7b23c9054b1..45fda86c521 100644 --- a/apps/explorer/src/pages/checkpoints/CheckpointDetail.tsx +++ b/apps/explorer/src/pages/checkpoints/CheckpointDetail.tsx @@ -24,7 +24,7 @@ import { import { useState } from 'react'; import { useFormatCoin } from '@iota/core'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; enum FeesTabs { GasAndStorageFees = 'gas-and-storage-fees', @@ -37,7 +37,7 @@ enum NestedTabs { Aggregated = 'aggregated', } -export default function CheckpointDetail(): JSX.Element { +export function CheckpointDetail(): JSX.Element { const [activeFeesTabId, setActiveFeesTabId] = useState(FeesTabs.GasAndStorageFees); const [activeDetailsTabId, setActiveDetailsTabId] = useState(DetailsTabs.Details); const [activeNestedTabId, setActiveNestedTabId] = useState(NestedTabs.Aggregated); diff --git a/apps/explorer/src/pages/epochs/EpochDetail.tsx b/apps/explorer/src/pages/epochs/EpochDetail.tsx index cf6b074838d..468afa3ef5a 100644 --- a/apps/explorer/src/pages/epochs/EpochDetail.tsx +++ b/apps/explorer/src/pages/epochs/EpochDetail.tsx @@ -28,14 +28,14 @@ import cx from 'clsx'; import { TokenStats } from './stats/TokenStats'; import { EpochTopStats } from './stats/EpochTopStats'; import { getEpochStorageFundFlow } from '~/lib/utils'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; enum EpochTabs { Checkpoints = 'checkpoints', Validators = 'validators', } -export default function EpochDetail() { +export function EpochDetail() { const [activeTabId, setActiveTabId] = useState(EpochTabs.Checkpoints); const { id } = useParams(); const enhancedRpc = useEnhancedRpcClient(); diff --git a/apps/explorer/src/pages/epochs/stats/EpochTopStats.tsx b/apps/explorer/src/pages/epochs/stats/EpochTopStats.tsx index c99f5a51c39..93d650863e1 100644 --- a/apps/explorer/src/pages/epochs/stats/EpochTopStats.tsx +++ b/apps/explorer/src/pages/epochs/stats/EpochTopStats.tsx @@ -5,11 +5,12 @@ import { ProgressBar } from '~/components'; import { EpochStatsGrid } from './EpochStats'; import { LabelText, LabelTextSize } from '@iota/apps-ui-kit'; -import { formatDate } from '@iota/core'; +import { Feature, formatDate, useFeatureEnabledByNetwork } from '@iota/core'; import { TokenStats } from './TokenStats'; import { getSupplyChangeAfterEpochEnd } from '~/lib'; import { useEpochProgress } from '../utils'; -import type { EndOfEpochInfo } from '@iota/iota-sdk/src/client'; +import type { Network, EndOfEpochInfo } from '@iota/iota-sdk/client'; +import { useNetworkContext } from '~/contexts'; interface EpochProgressProps { start: number; @@ -25,8 +26,15 @@ export function EpochTopStats({ endOfEpochInfo, }: EpochProgressProps): React.JSX.Element { const { progress, label } = useEpochProgress(); + const [network] = useNetworkContext(); + const endTime = inProgress ? label : end ? formatDate(end) : undefined; + const isBurntAndMintedTokensInEndedEpochsFeatureEnabled = useFeatureEnabledByNetwork( + Feature.BurntAndMintedTokensInEndedEpochs, + network as Network, + ); + return ( <div className="flex w-full flex-col gap-md--rs"> {inProgress ? <ProgressBar progress={progress || 0} /> : null} @@ -35,12 +43,30 @@ export function EpochTopStats({ <LabelText text={formatDate(start)} label="Start" /> {endTime ? <LabelText text={endTime} label="End" /> : null} {endOfEpochInfo && ( - <TokenStats - label="Supply Change" - size={LabelTextSize.Large} - amount={getSupplyChangeAfterEpochEnd(endOfEpochInfo)} - showSign - /> + <> + {isBurntAndMintedTokensInEndedEpochsFeatureEnabled && ( + <> + <TokenStats + label="Burnt Tokens" + size={LabelTextSize.Large} + amount={BigInt(endOfEpochInfo?.burntTokensAmount)} + showSign + /> + <TokenStats + label="Minted Tokens" + size={LabelTextSize.Large} + amount={BigInt(endOfEpochInfo?.mintedTokensAmount)} + showSign + /> + </> + )} + <TokenStats + label="Supply Change" + size={LabelTextSize.Large} + amount={getSupplyChangeAfterEpochEnd(endOfEpochInfo)} + showSign + /> + </> )} </EpochStatsGrid> </div> diff --git a/apps/explorer/src/pages/home/Home.tsx b/apps/explorer/src/pages/home/Home.tsx index 1cee4ba88ab..d89625da7b0 100644 --- a/apps/explorer/src/pages/home/Home.tsx +++ b/apps/explorer/src/pages/home/Home.tsx @@ -21,7 +21,7 @@ import { useNetwork } from '~/hooks'; const TRANSACTIONS_LIMIT = 15; -function Home(): JSX.Element { +export function Home(): JSX.Element { const [network] = useNetwork(); const isIotaTokenCardEnabled = network === Network.Mainnet; return ( @@ -77,5 +77,3 @@ function Home(): JSX.Element { /> ); } - -export default Home; diff --git a/apps/explorer/src/pages/index.tsx b/apps/explorer/src/pages/index.tsx index c1e40edb438..d2bd6722775 100644 --- a/apps/explorer/src/pages/index.tsx +++ b/apps/explorer/src/pages/index.tsx @@ -4,13 +4,13 @@ import { wrapCreateBrowserRouter } from '@sentry/react'; import { createBrowserRouter, Navigate, useLocation, useParams } from 'react-router-dom'; -import AddressResult from './address-result/AddressResult'; -import CheckpointDetail from './checkpoints/CheckpointDetail'; -import EpochDetail from './epochs/EpochDetail'; -import Home from './home/Home'; +import { AddressResultPage } from './address-result/AddressResult'; +import { CheckpointDetail } from './checkpoints/CheckpointDetail'; +import { EpochDetail } from './epochs/EpochDetail'; +import { Home } from './home/Home'; import { ObjectResult } from './object-result/ObjectResult'; import { Recent } from './recent'; -import TransactionResult from './transaction-result/TransactionResult'; +import { TransactionResult } from './transaction-result/TransactionResult'; import { ValidatorDetails } from './validator/ValidatorDetails'; import { ValidatorPageResult } from './validators/Validators'; import { Layout } from '~/components'; @@ -39,7 +39,7 @@ export const router = sentryCreateBrowserRouter([ { path: 'epoch/current', element: <EpochDetail /> }, { path: 'txblock/:id', element: <TransactionResult /> }, { path: 'epoch/:id', element: <EpochDetail /> }, - { path: 'address/:id', element: <AddressResult /> }, + { path: 'address/:id', element: <AddressResultPage /> }, { path: 'validators', element: <ValidatorPageResult /> }, { path: 'validator/:id', element: <ValidatorDetails /> }, ], diff --git a/apps/explorer/src/pages/object-result/ObjectResult.tsx b/apps/explorer/src/pages/object-result/ObjectResult.tsx index 147ba0fc4d5..5cea76ef174 100644 --- a/apps/explorer/src/pages/object-result/ObjectResult.tsx +++ b/apps/explorer/src/pages/object-result/ObjectResult.tsx @@ -8,10 +8,9 @@ import { ErrorBoundary, PageLayout } from '~/components'; import { PageHeader } from '~/components/ui'; import { ObjectView } from '~/pages/object-result/views/ObjectView'; import { translate, type DataType } from './ObjectResultType'; -import PkgView from './views/PkgView'; -import { TokenView } from './views/TokenView'; +import { PkgView, TokenView } from './views'; import { InfoBox, InfoBoxStyle, InfoBoxType, LoadingIndicator } from '@iota/apps-ui-kit'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; const PACKAGE_TYPE_NAME = 'Move Package'; diff --git a/apps/explorer/src/pages/object-result/views/ObjectView.tsx b/apps/explorer/src/pages/object-result/views/ObjectView.tsx index 087498cf6ef..79dead884ce 100644 --- a/apps/explorer/src/pages/object-result/views/ObjectView.tsx +++ b/apps/explorer/src/pages/object-result/views/ObjectView.tsx @@ -13,7 +13,7 @@ import { normalizeStructTag, parseStructTag, } from '@iota/iota-sdk/utils'; -import { ArrowTopRight } from '@iota/ui-icons'; +import { ArrowTopRight } from '@iota/apps-ui-icons'; import { useQuery } from '@tanstack/react-query'; import clsx from 'clsx'; import { type PropsWithChildren, type ReactNode, useEffect, useState } from 'react'; diff --git a/apps/explorer/src/pages/object-result/views/PkgView.tsx b/apps/explorer/src/pages/object-result/views/PkgView.tsx index 7710894489a..56beb84b038 100644 --- a/apps/explorer/src/pages/object-result/views/PkgView.tsx +++ b/apps/explorer/src/pages/object-result/views/PkgView.tsx @@ -39,7 +39,7 @@ interface PkgViewProps { data: DataType; } -function PkgView({ data }: PkgViewProps): JSX.Element { +export function PkgView({ data }: PkgViewProps): JSX.Element { const [selectedSplitPanelOrientation, setSplitPanelOrientation] = useState( SPLIT_PANELS_ORIENTATION[1].value, ); @@ -145,5 +145,3 @@ function PkgView({ data }: PkgViewProps): JSX.Element { </div> ); } - -export default PkgView; diff --git a/apps/explorer/src/pages/object-result/views/index.ts b/apps/explorer/src/pages/object-result/views/index.ts new file mode 100644 index 00000000000..08b740d5988 --- /dev/null +++ b/apps/explorer/src/pages/object-result/views/index.ts @@ -0,0 +1,5 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export * from './PkgView'; +export * from './TokenView'; diff --git a/apps/explorer/src/pages/transaction-result/Events.tsx b/apps/explorer/src/pages/transaction-result/Events.tsx index cc53d32ec83..936a34f962e 100644 --- a/apps/explorer/src/pages/transaction-result/Events.tsx +++ b/apps/explorer/src/pages/transaction-result/Events.tsx @@ -11,7 +11,7 @@ import { } from '@iota/apps-ui-kit'; import { type IotaEvent } from '@iota/iota-sdk/client'; import { formatAddress, parseStructTag } from '@iota/iota-sdk/utils'; -import { TriangleDown } from '@iota/ui-icons'; +import { TriangleDown } from '@iota/apps-ui-icons'; import clsx from 'clsx'; import { useState } from 'react'; import { SyntaxHighlighter } from '~/components'; diff --git a/apps/explorer/src/pages/transaction-result/TransactionResult.tsx b/apps/explorer/src/pages/transaction-result/TransactionResult.tsx index 2abdd1770a9..f98645a6d30 100644 --- a/apps/explorer/src/pages/transaction-result/TransactionResult.tsx +++ b/apps/explorer/src/pages/transaction-result/TransactionResult.tsx @@ -9,7 +9,7 @@ import { PageLayout } from '~/components'; import { PageHeader } from '~/components/ui'; import { TransactionView } from './TransactionView'; import { InfoBox, InfoBoxType, InfoBoxStyle } from '@iota/apps-ui-kit'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; interface TransactionResultPageHeaderProps { transaction?: IotaTransactionBlockResponse; @@ -40,7 +40,7 @@ function TransactionResultPageHeader({ ); } -export default function TransactionResult(): JSX.Element { +export function TransactionResult(): JSX.Element { const { id } = useParams(); const { isPending, diff --git a/apps/explorer/src/pages/transaction-result/programmable-transaction-view/InputsCard.tsx b/apps/explorer/src/pages/transaction-result/programmable-transaction-view/InputsCard.tsx index 8610e071aed..439e6ba6593 100644 --- a/apps/explorer/src/pages/transaction-result/programmable-transaction-view/InputsCard.tsx +++ b/apps/explorer/src/pages/transaction-result/programmable-transaction-view/InputsCard.tsx @@ -4,6 +4,7 @@ import { KeyValueInfo, TitleSize } from '@iota/apps-ui-kit'; import { type IotaCallArg } from '@iota/iota-sdk/client'; +import { toHEX } from '@iota/iota-sdk/utils'; import { ProgrammableTxnBlockCard, AddressLink, ObjectLink, CollapsibleCard } from '~/components'; import { useBreakpoint } from '~/hooks'; @@ -48,6 +49,13 @@ export function InputsCard({ inputs }: InputsCardProps): JSX.Element | null { } else if (REGEX_NUMBER.test(stringValue)) { const bigNumber = BigInt(stringValue); renderValue = bigNumber.toLocaleString(); + } else if ( + 'valueType' in input && + 'value' in input && + input.valueType === 'vector<u8>' && + key === 'value' + ) { + renderValue = toHEX(new Uint8Array(stringValue.split(',').map(Number))); } else { renderValue = stringValue; } diff --git a/apps/explorer/src/pages/transaction-result/transaction-summary/BalanceChanges.tsx b/apps/explorer/src/pages/transaction-result/transaction-summary/BalanceChanges.tsx index 8597bec0a25..27e6081325f 100644 --- a/apps/explorer/src/pages/transaction-result/transaction-summary/BalanceChanges.tsx +++ b/apps/explorer/src/pages/transaction-result/transaction-summary/BalanceChanges.tsx @@ -23,7 +23,7 @@ import { ImageIconSize, CoinIcon, } from '@iota/core'; -import { RecognizedBadge } from '@iota/ui-icons'; +import { RecognizedBadge } from '@iota/apps-ui-icons'; import { useMemo } from 'react'; import { AddressLink, CollapsibleCard } from '~/components/ui'; import { BREAK_POINT, useMediaQuery } from '~/hooks'; diff --git a/apps/explorer/src/pages/transaction-result/transaction-summary/ObjectChanges.tsx b/apps/explorer/src/pages/transaction-result/transaction-summary/ObjectChanges.tsx index 34d4e09ef68..7b7c83838a1 100644 --- a/apps/explorer/src/pages/transaction-result/transaction-summary/ObjectChanges.tsx +++ b/apps/explorer/src/pages/transaction-result/transaction-summary/ObjectChanges.tsx @@ -35,7 +35,7 @@ import { KeyValueInfo, TitleSize, } from '@iota/apps-ui-kit'; -import { TriangleDown } from '@iota/ui-icons'; +import { TriangleDown } from '@iota/apps-ui-icons'; interface ItemProps { label: string; diff --git a/apps/explorer/src/pages/transaction-result/transaction-summary/ObjectDisplay.tsx b/apps/explorer/src/pages/transaction-result/transaction-summary/ObjectDisplay.tsx index 278df5ca5c4..0d351a6f64d 100644 --- a/apps/explorer/src/pages/transaction-result/transaction-summary/ObjectDisplay.tsx +++ b/apps/explorer/src/pages/transaction-result/transaction-summary/ObjectDisplay.tsx @@ -5,7 +5,7 @@ import { Card, CardAction, CardActionType, CardBody, CardImage, CardType } from '@iota/apps-ui-kit'; import { ImageIcon } from '@iota/core'; import { type DisplayFieldsResponse } from '@iota/iota-sdk/client'; -import { ArrowTopRight } from '@iota/ui-icons'; +import { ArrowTopRight } from '@iota/apps-ui-icons'; import { useState } from 'react'; import { ObjectModal } from '~/components/ui'; diff --git a/apps/explorer/src/pages/validator/ValidatorDetails.tsx b/apps/explorer/src/pages/validator/ValidatorDetails.tsx index a8ea8c7ae60..d491edac6da 100644 --- a/apps/explorer/src/pages/validator/ValidatorDetails.tsx +++ b/apps/explorer/src/pages/validator/ValidatorDetails.tsx @@ -5,13 +5,13 @@ import { useGetValidatorsApy, useGetValidatorsEvents } from '@iota/core'; import { useIotaClientQuery } from '@iota/dapp-kit'; import { type IotaSystemStateSummary } from '@iota/iota-sdk/client'; -import React, { useMemo } from 'react'; +import { useMemo } from 'react'; import { useParams } from 'react-router-dom'; import { PageLayout, ValidatorMeta, ValidatorStats } from '~/components'; import { VALIDATOR_LOW_STAKE_GRACE_PERIOD } from '~/lib/constants'; import { getValidatorMoveEvent } from '~/lib/utils'; import { InfoBox, InfoBoxStyle, InfoBoxType, LoadingIndicator } from '@iota/apps-ui-kit'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; const getAtRiskRemainingEpochs = ( data: IotaSystemStateSummary | undefined, diff --git a/apps/explorer/src/pages/validators/Validators.tsx b/apps/explorer/src/pages/validators/Validators.tsx index 78e333cbf90..a11a5b69790 100644 --- a/apps/explorer/src/pages/validators/Validators.tsx +++ b/apps/explorer/src/pages/validators/Validators.tsx @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React, { type JSX, useMemo } from 'react'; +import { type JSX, useMemo } from 'react'; import { roundFloat, useFormatCoin, useGetValidatorsApy, useGetValidatorsEvents } from '@iota/core'; import { DisplayStats, @@ -19,7 +19,7 @@ import { useIotaClientQuery } from '@iota/dapp-kit'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { ErrorBoundary, PageLayout, PlaceholderTable, TableCard } from '~/components'; import { generateValidatorsTableColumns } from '~/lib/ui'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; function ValidatorPageResult(): JSX.Element { const { data, isPending, isSuccess, isError } = useIotaClientQuery('getLatestIotaSystemState'); diff --git a/apps/explorer/tailwind.config.ts b/apps/explorer/tailwind.config.ts index 553727d7246..4d1800c3212 100644 --- a/apps/explorer/tailwind.config.ts +++ b/apps/explorer/tailwind.config.ts @@ -5,7 +5,7 @@ import colors from 'tailwindcss/colors'; import { type Config } from 'tailwindcss'; // Note: exception for the tailwind preset import -import uiKitResponsivePreset from '../../apps/ui-kit/src/lib/tailwind/responsive.preset'; +import uiKitResponsivePreset from '../../apps/ui-kit/src/lib/tailwind/responsive.presets'; export default { presets: [uiKitResponsivePreset], diff --git a/apps/ui-icons/CHANGELOG.md b/apps/ui-icons/CHANGELOG.md new file mode 100644 index 00000000000..63b69e18156 --- /dev/null +++ b/apps/ui-icons/CHANGELOG.md @@ -0,0 +1,13 @@ +# @iota/apps-ui-icons + +## 0.2.1 + +### Patch Changes + +- b772ffa: Add correct prepublish script + +## 0.2.0 + +### Minor Changes + +- c1c5e9b: Initial npm release diff --git a/apps/ui-icons/README.md b/apps/ui-icons/README.md index 058a955d440..e5b6d018eca 100644 --- a/apps/ui-icons/README.md +++ b/apps/ui-icons/README.md @@ -1,4 +1,4 @@ -# `@iota/ui-icons` +# `@iota/apps-ui-icons` ## Exporting Icons diff --git a/apps/ui-icons/package.json b/apps/ui-icons/package.json index eea49bf5aeb..89fe42bc47a 100644 --- a/apps/ui-icons/package.json +++ b/apps/ui-icons/package.json @@ -1,9 +1,13 @@ { - "name": "@iota/ui-icons", - "main": "src/index.ts", - "private": true, + "name": "@iota/apps-ui-icons", + "version": "0.2.1", + "main": "dist/index.js", + "module": "dist/index.mjs", + "types": "dist/index.d.ts", "sideEffects": false, "scripts": { + "build": "tsup src/index.ts --format cjs,esm --dts", + "prepublishOnly": "pnpm run build", "clean:src": "rimraf src/**", "clean:svgs": "rimraf svgs/**", "clean:all": "pnpm clean:src && pnpm clean:svgs", @@ -21,6 +25,7 @@ "@types/react": "^18.3.3", "react": "^18.3.1", "rimraf": "^5.0.1", + "tsup": "^8.3.5", "typescript": "^5.5.3" } } diff --git a/apps/ui-kit/CHANGELOG.md b/apps/ui-kit/CHANGELOG.md new file mode 100644 index 00000000000..e5deb97322c --- /dev/null +++ b/apps/ui-kit/CHANGELOG.md @@ -0,0 +1,20 @@ +# @iota/apps-ui-kit + +## 0.2.1 + +### Patch Changes + +- b772ffa: Add correct prepublish script +- Updated dependencies [b772ffa] + - @iota/apps-ui-icons@0.2.1 + +## 0.2.0 + +### Minor Changes + +- c1c5e9b: Initial npm release + +### Patch Changes + +- Updated dependencies [c1c5e9b] + - @iota/apps-ui-icons@0.2.0 diff --git a/apps/ui-kit/package.json b/apps/ui-kit/package.json index 89e9785c466..92ccab8f83a 100644 --- a/apps/ui-kit/package.json +++ b/apps/ui-kit/package.json @@ -1,16 +1,19 @@ { "name": "@iota/apps-ui-kit", - "private": true, - "version": "0.0.1-rc.1", + "version": "0.2.1", "repository": { "type": "git", "url": "github.com:iotaledger/iota.git" }, "license": "Apache-2.0", "files": [ + "CHANGELOG.md", + "LICENSE", + "src", "dist" ], "scripts": { + "prepublishOnly": "pnpm turbo run build", "prettier:check": "prettier -c --ignore-unknown --ignore-path=../../.prettierignore --ignore-path=.prettierignore .", "prettier:fix": "prettier -w --ignore-unknown --ignore-path=../../.prettierignore --ignore-path=.prettierignore .", "eslint:check": "eslint --max-warnings=0 .", @@ -34,7 +37,7 @@ "types": "./dist/index.d.ts", "dependencies": { "@fontsource/inter": "^5.0.17", - "@iota/ui-icons": "workspace:*", + "@iota/apps-ui-icons": "workspace:*", "@radix-ui/react-dialog": "^1.1.1", "@radix-ui/react-visually-hidden": "^1.1.0", "classnames": "^2.5.1", diff --git a/apps/ui-kit/src/lib/components/atoms/badge/Badge.tsx b/apps/ui-kit/src/lib/components/atoms/badge/Badge.tsx index 390b8078342..c6dcba7dff9 100644 --- a/apps/ui-kit/src/lib/components/atoms/badge/Badge.tsx +++ b/apps/ui-kit/src/lib/components/atoms/badge/Badge.tsx @@ -1,7 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import cx from 'classnames'; import { BadgeType } from './badge.enums'; import { BACKGROUND_COLORS, BADGE_TEXT_CLASS, BORDER_COLORS, TEXT_COLORS } from './badge.classes'; diff --git a/apps/ui-kit/src/lib/components/atoms/button-segment/ButtonSegment.tsx b/apps/ui-kit/src/lib/components/atoms/button-segment/ButtonSegment.tsx index f3ff331c2d8..1024457df90 100644 --- a/apps/ui-kit/src/lib/components/atoms/button-segment/ButtonSegment.tsx +++ b/apps/ui-kit/src/lib/components/atoms/button-segment/ButtonSegment.tsx @@ -1,7 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import { BACKGROUND_COLORS, BACKGROUND_COLORS_SELECTED, @@ -9,9 +8,9 @@ import { TEXT_COLORS_SELECTED, UNDERLINED, UNDERLINED_SELECTED, -} from './button-segment.classes'; +} from './buttonSegment.classes'; import cx from 'classnames'; -import { ButtonSegmentType } from './button-segment.enums'; +import { ButtonSegmentType } from './buttonSegment.enums'; import { ButtonUnstyled } from '../button'; interface ButtonSegmentProps { diff --git a/apps/ui-kit/src/lib/components/atoms/button-segment/button-segment.classes.ts b/apps/ui-kit/src/lib/components/atoms/button-segment/buttonSegment.classes.ts similarity index 100% rename from apps/ui-kit/src/lib/components/atoms/button-segment/button-segment.classes.ts rename to apps/ui-kit/src/lib/components/atoms/button-segment/buttonSegment.classes.ts diff --git a/apps/ui-kit/src/lib/components/atoms/button-segment/button-segment.enums.ts b/apps/ui-kit/src/lib/components/atoms/button-segment/buttonSegment.enums.ts similarity index 100% rename from apps/ui-kit/src/lib/components/atoms/button-segment/button-segment.enums.ts rename to apps/ui-kit/src/lib/components/atoms/button-segment/buttonSegment.enums.ts diff --git a/apps/ui-kit/src/lib/components/atoms/button-segment/index.ts b/apps/ui-kit/src/lib/components/atoms/button-segment/index.ts index f8b4143cc86..0c48b7e0022 100644 --- a/apps/ui-kit/src/lib/components/atoms/button-segment/index.ts +++ b/apps/ui-kit/src/lib/components/atoms/button-segment/index.ts @@ -2,4 +2,4 @@ // SPDX-License-Identifier: Apache-2.0 export * from './ButtonSegment'; -export * from './button-segment.enums'; +export * from './buttonSegment.enums'; diff --git a/apps/ui-kit/src/lib/components/atoms/button/Button.tsx b/apps/ui-kit/src/lib/components/atoms/button/Button.tsx index b35d566c50f..793aa5d8cb1 100644 --- a/apps/ui-kit/src/lib/components/atoms/button/Button.tsx +++ b/apps/ui-kit/src/lib/components/atoms/button/Button.tsx @@ -1,7 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import { ButtonHtmlType, ButtonSize, ButtonType } from './button.enums'; import { PADDINGS, diff --git a/apps/ui-kit/src/lib/components/atoms/button/button-variants/ButtonPill.tsx b/apps/ui-kit/src/lib/components/atoms/button/button-variants/ButtonPill.tsx index b502085574e..53556258c6a 100644 --- a/apps/ui-kit/src/lib/components/atoms/button/button-variants/ButtonPill.tsx +++ b/apps/ui-kit/src/lib/components/atoms/button/button-variants/ButtonPill.tsx @@ -2,7 +2,7 @@ // SPDX-License-Identifier: Apache-2.0 import { ButtonHtmlType } from '../button.enums'; -import { ButtonVariantProps } from './button-variants.types'; +import { ButtonVariantProps } from './buttonVariants.types'; export function ButtonPill({ htmlType = ButtonHtmlType.Button, diff --git a/apps/ui-kit/src/lib/components/atoms/button/button-variants/ButtonUnstyled.tsx b/apps/ui-kit/src/lib/components/atoms/button/button-variants/ButtonUnstyled.tsx index 1b4d64ad861..53400f13513 100644 --- a/apps/ui-kit/src/lib/components/atoms/button/button-variants/ButtonUnstyled.tsx +++ b/apps/ui-kit/src/lib/components/atoms/button/button-variants/ButtonUnstyled.tsx @@ -2,7 +2,7 @@ // SPDX-License-Identifier: Apache-2.0 import { ButtonHtmlType } from '../button.enums'; -import { ButtonVariantProps } from './button-variants.types'; +import { ButtonVariantProps } from './buttonVariants.types'; import cx from 'classnames'; export function ButtonUnstyled({ diff --git a/apps/ui-kit/src/lib/components/atoms/button/button-variants/button-variants.types.ts b/apps/ui-kit/src/lib/components/atoms/button/button-variants/buttonVariants.types.ts similarity index 100% rename from apps/ui-kit/src/lib/components/atoms/button/button-variants/button-variants.types.ts rename to apps/ui-kit/src/lib/components/atoms/button/button-variants/buttonVariants.types.ts diff --git a/apps/ui-kit/src/lib/components/atoms/checkbox/Checkbox.tsx b/apps/ui-kit/src/lib/components/atoms/checkbox/Checkbox.tsx index aec9826a44f..ca2a875e0a6 100644 --- a/apps/ui-kit/src/lib/components/atoms/checkbox/Checkbox.tsx +++ b/apps/ui-kit/src/lib/components/atoms/checkbox/Checkbox.tsx @@ -3,7 +3,7 @@ import { forwardRef, useEffect, useRef } from 'react'; import cx from 'classnames'; -import { Dash, Checkmark } from '@iota/ui-icons'; +import { Dash, Checkmark } from '@iota/apps-ui-icons'; interface CheckboxProps { /** diff --git a/apps/ui-kit/src/lib/components/atoms/divider/Divider.tsx b/apps/ui-kit/src/lib/components/atoms/divider/Divider.tsx index a9391cfe151..996bc0ac005 100644 --- a/apps/ui-kit/src/lib/components/atoms/divider/Divider.tsx +++ b/apps/ui-kit/src/lib/components/atoms/divider/Divider.tsx @@ -1,7 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import { DividerType } from './divider.enums'; import cx from 'classnames'; import { BACKGROUND_COLORS, DIVIDER_FULL_WIDTH } from './divider.classes'; diff --git a/apps/ui-kit/src/lib/components/atoms/header/Header.tsx b/apps/ui-kit/src/lib/components/atoms/header/Header.tsx index 0f9f3ff6eda..e3e43aa6913 100644 --- a/apps/ui-kit/src/lib/components/atoms/header/Header.tsx +++ b/apps/ui-kit/src/lib/components/atoms/header/Header.tsx @@ -3,7 +3,7 @@ import cx from 'classnames'; import { Button, ButtonSize, ButtonType } from '../button'; -import { ArrowBack, Close } from '@iota/ui-icons'; +import { ArrowBack, Close } from '@iota/apps-ui-icons'; interface HeaderProps { /** diff --git a/apps/ui-kit/src/lib/components/atoms/info-box/InfoBox.tsx b/apps/ui-kit/src/lib/components/atoms/info-box/InfoBox.tsx index e8548979d21..bb668cbe24b 100644 --- a/apps/ui-kit/src/lib/components/atoms/info-box/InfoBox.tsx +++ b/apps/ui-kit/src/lib/components/atoms/info-box/InfoBox.tsx @@ -1,10 +1,9 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import cx from 'classnames'; -import { InfoBoxStyle, InfoBoxType } from './info-box.enums'; -import { BACKGROUND_COLORS, ICON_COLORS } from './info-box.classes'; +import { InfoBoxStyle, InfoBoxType } from './infoBox.enums'; +import { BACKGROUND_COLORS, ICON_COLORS } from './infoBox.classes'; export interface InfoBoxProps { /** diff --git a/apps/ui-kit/src/lib/components/atoms/info-box/index.ts b/apps/ui-kit/src/lib/components/atoms/info-box/index.ts index c4f12c2be81..fff4ed5d37a 100644 --- a/apps/ui-kit/src/lib/components/atoms/info-box/index.ts +++ b/apps/ui-kit/src/lib/components/atoms/info-box/index.ts @@ -2,4 +2,4 @@ // SPDX-License-Identifier: Apache-2.0 export * from './InfoBox'; -export * from './info-box.enums'; +export * from './infoBox.enums'; diff --git a/apps/ui-kit/src/lib/components/atoms/info-box/info-box.classes.ts b/apps/ui-kit/src/lib/components/atoms/info-box/infoBox.classes.ts similarity index 91% rename from apps/ui-kit/src/lib/components/atoms/info-box/info-box.classes.ts rename to apps/ui-kit/src/lib/components/atoms/info-box/infoBox.classes.ts index ce7136fa7e8..d78493a4af3 100644 --- a/apps/ui-kit/src/lib/components/atoms/info-box/info-box.classes.ts +++ b/apps/ui-kit/src/lib/components/atoms/info-box/infoBox.classes.ts @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { InfoBoxType } from './info-box.enums'; +import { InfoBoxType } from './infoBox.enums'; export const ICON_COLORS: Record<InfoBoxType, string> = { [InfoBoxType.Default]: 'bg-neutral-96 dark:bg-neutral-12 text-neutral-10 dark:text-neutral-92', diff --git a/apps/ui-kit/src/lib/components/atoms/info-box/info-box.enums.ts b/apps/ui-kit/src/lib/components/atoms/info-box/infoBox.enums.ts similarity index 100% rename from apps/ui-kit/src/lib/components/atoms/info-box/info-box.enums.ts rename to apps/ui-kit/src/lib/components/atoms/info-box/infoBox.enums.ts diff --git a/apps/ui-kit/src/lib/components/atoms/key-value-info/KeyValueInfo.tsx b/apps/ui-kit/src/lib/components/atoms/key-value-info/KeyValueInfo.tsx index d15da2af5de..c05c9884f20 100644 --- a/apps/ui-kit/src/lib/components/atoms/key-value-info/KeyValueInfo.tsx +++ b/apps/ui-kit/src/lib/components/atoms/key-value-info/KeyValueInfo.tsx @@ -1,9 +1,9 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React, { ReactNode } from 'react'; +import { ReactNode } from 'react'; import cx from 'classnames'; -import { Copy, Info } from '@iota/ui-icons'; +import { Copy, Info } from '@iota/apps-ui-icons'; import { ValueSize } from './keyValue.enums'; import { Tooltip, TooltipPosition } from '../tooltip'; import { ButtonUnstyled } from '../button'; @@ -154,5 +154,3 @@ export function KeyValueInfo({ </div> ); } - -export default KeyValueInfo; diff --git a/apps/ui-kit/src/lib/components/atoms/label-text/LabelText.tsx b/apps/ui-kit/src/lib/components/atoms/label-text/LabelText.tsx index 5135219c33c..2cf95723a91 100644 --- a/apps/ui-kit/src/lib/components/atoms/label-text/LabelText.tsx +++ b/apps/ui-kit/src/lib/components/atoms/label-text/LabelText.tsx @@ -1,11 +1,10 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import cx from 'classnames'; import { LabelTextSize } from './labelText.enums'; import { LABEL_TEXT_SIZE, SUPPORTING_TEXT_SIZE, TEXT_SIZE } from './labelText.classes'; -import { Info } from '@iota/ui-icons'; +import { Info } from '@iota/apps-ui-icons'; import { TooltipPosition, Tooltip } from '../tooltip'; interface LabelTextProps { diff --git a/apps/ui-kit/src/lib/components/atoms/list-item/ListItem.tsx b/apps/ui-kit/src/lib/components/atoms/list-item/ListItem.tsx index 9f6475a674b..6eb6cf79b50 100644 --- a/apps/ui-kit/src/lib/components/atoms/list-item/ListItem.tsx +++ b/apps/ui-kit/src/lib/components/atoms/list-item/ListItem.tsx @@ -1,9 +1,9 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React, { PropsWithChildren } from 'react'; +import { PropsWithChildren } from 'react'; import cx from 'classnames'; -import { ArrowRight } from '@iota/ui-icons'; +import { ArrowRight } from '@iota/apps-ui-icons'; import { Button, ButtonSize, ButtonType } from '@/components'; export interface ListItemProps { diff --git a/apps/ui-kit/src/lib/components/atoms/loading-indicator/LoadingIndicator.tsx b/apps/ui-kit/src/lib/components/atoms/loading-indicator/LoadingIndicator.tsx index 86160c55771..53814f9457b 100644 --- a/apps/ui-kit/src/lib/components/atoms/loading-indicator/LoadingIndicator.tsx +++ b/apps/ui-kit/src/lib/components/atoms/loading-indicator/LoadingIndicator.tsx @@ -1,9 +1,8 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import cx from 'classnames'; -import { Loader } from '@iota/ui-icons'; +import { Loader } from '@iota/apps-ui-icons'; export interface LoadingIndicatorProps { /** diff --git a/apps/ui-kit/src/lib/components/atoms/panel/Panel.tsx b/apps/ui-kit/src/lib/components/atoms/panel/Panel.tsx index d5eae32a334..0f87336f524 100644 --- a/apps/ui-kit/src/lib/components/atoms/panel/Panel.tsx +++ b/apps/ui-kit/src/lib/components/atoms/panel/Panel.tsx @@ -1,7 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import cx from 'classnames'; interface PanelProps { diff --git a/apps/ui-kit/src/lib/components/atoms/radio-button/RadioButton.tsx b/apps/ui-kit/src/lib/components/atoms/radio-button/RadioButton.tsx index 54bbb8277de..b1d0177f39f 100644 --- a/apps/ui-kit/src/lib/components/atoms/radio-button/RadioButton.tsx +++ b/apps/ui-kit/src/lib/components/atoms/radio-button/RadioButton.tsx @@ -1,9 +1,8 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import cx from 'classnames'; -import { RadioOn, RadioOff } from '@iota/ui-icons'; +import { RadioOn, RadioOff } from '@iota/apps-ui-icons'; interface RadioButtonProps { /** diff --git a/apps/ui-kit/src/lib/components/atoms/snackbar/Snackbar.tsx b/apps/ui-kit/src/lib/components/atoms/snackbar/Snackbar.tsx index 173ae161c1b..da41a13a024 100644 --- a/apps/ui-kit/src/lib/components/atoms/snackbar/Snackbar.tsx +++ b/apps/ui-kit/src/lib/components/atoms/snackbar/Snackbar.tsx @@ -3,7 +3,7 @@ import { useEffect } from 'react'; import cx from 'classnames'; -import { Close } from '@iota/ui-icons'; +import { Close } from '@iota/apps-ui-icons'; import { SnackbarType } from './snackbar.enums'; import { BACKGROUND_COLOR, TEXT_COLOR } from '@/components/atoms/snackbar/snackbar.classes'; diff --git a/apps/ui-kit/src/lib/components/atoms/tooltip/Tooltip.tsx b/apps/ui-kit/src/lib/components/atoms/tooltip/Tooltip.tsx index 467aa94d316..220f47f7da7 100644 --- a/apps/ui-kit/src/lib/components/atoms/tooltip/Tooltip.tsx +++ b/apps/ui-kit/src/lib/components/atoms/tooltip/Tooltip.tsx @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React, { PropsWithChildren } from 'react'; +import { PropsWithChildren } from 'react'; import cx from 'classnames'; import { TooltipPosition } from './tooltip.enums'; import { TOOLTIP_POSITION } from './tooltip.classes'; diff --git a/apps/ui-kit/src/lib/components/atoms/visual-asset-card/VisualAssetCard.tsx b/apps/ui-kit/src/lib/components/atoms/visual-asset-card/VisualAssetCard.tsx index 1dbcf4731cf..4f35afb1296 100644 --- a/apps/ui-kit/src/lib/components/atoms/visual-asset-card/VisualAssetCard.tsx +++ b/apps/ui-kit/src/lib/components/atoms/visual-asset-card/VisualAssetCard.tsx @@ -1,10 +1,9 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; -import { VisualAssetType } from './visual-asset-card.enums'; +import { VisualAssetType } from './visualAssetCard.enums'; import { ButtonUnstyled } from '../button'; -import { MoreHoriz } from '@iota/ui-icons'; +import { MoreHoriz } from '@iota/apps-ui-icons'; import cx from 'classnames'; export interface VisualAssetCardProps { diff --git a/apps/ui-kit/src/lib/components/atoms/visual-asset-card/index.ts b/apps/ui-kit/src/lib/components/atoms/visual-asset-card/index.ts index 05a2d49f88d..3e758c82d59 100644 --- a/apps/ui-kit/src/lib/components/atoms/visual-asset-card/index.ts +++ b/apps/ui-kit/src/lib/components/atoms/visual-asset-card/index.ts @@ -2,4 +2,4 @@ // SPDX-License-Identifier: Apache-2.0 export * from './VisualAssetCard'; -export * from './visual-asset-card.enums'; +export * from './visualAssetCard.enums'; diff --git a/apps/ui-kit/src/lib/components/atoms/visual-asset-card/visual-asset-card.enums.ts b/apps/ui-kit/src/lib/components/atoms/visual-asset-card/visualAssetCard.enums.ts similarity index 100% rename from apps/ui-kit/src/lib/components/atoms/visual-asset-card/visual-asset-card.enums.ts rename to apps/ui-kit/src/lib/components/atoms/visual-asset-card/visualAssetCard.enums.ts diff --git a/apps/ui-kit/src/lib/components/molecules/account/Account.tsx b/apps/ui-kit/src/lib/components/molecules/account/Account.tsx index e62f598ad1b..37d6c3099ae 100644 --- a/apps/ui-kit/src/lib/components/molecules/account/Account.tsx +++ b/apps/ui-kit/src/lib/components/molecules/account/Account.tsx @@ -1,11 +1,16 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import cx from 'classnames'; import { ButtonUnstyled } from '../../atoms/button'; import { Badge, BadgeType } from '../../atoms'; -import { LockLocked, LockUnlocked, MoreHoriz, CheckmarkFilled, RadioOff } from '@iota/ui-icons'; +import { + LockLocked, + LockUnlocked, + MoreHoriz, + CheckmarkFilled, + RadioOff, +} from '@iota/apps-ui-icons'; import { Address } from '../address'; interface AccountProps { diff --git a/apps/ui-kit/src/lib/components/molecules/address/Address.tsx b/apps/ui-kit/src/lib/components/molecules/address/Address.tsx index 869048a62a2..f05e283e827 100644 --- a/apps/ui-kit/src/lib/components/molecules/address/Address.tsx +++ b/apps/ui-kit/src/lib/components/molecules/address/Address.tsx @@ -1,9 +1,8 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import cx from 'classnames'; -import { Copy, ArrowTopRight } from '@iota/ui-icons'; +import { Copy, ArrowTopRight } from '@iota/apps-ui-icons'; import { ButtonUnstyled } from '../../atoms/button'; interface AddressProps { diff --git a/apps/ui-kit/src/lib/components/molecules/card/Card.tsx b/apps/ui-kit/src/lib/components/molecules/card/Card.tsx index cc1b8368349..34c9ca68ef6 100644 --- a/apps/ui-kit/src/lib/components/molecules/card/Card.tsx +++ b/apps/ui-kit/src/lib/components/molecules/card/Card.tsx @@ -1,7 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import cx from 'classnames'; import { CARD_DISABLED_CLASSES, CARD_TYPE_CLASSES } from './card.classes'; import { CardType } from './card.enums'; diff --git a/apps/ui-kit/src/lib/components/molecules/card/CardAction.tsx b/apps/ui-kit/src/lib/components/molecules/card/CardAction.tsx index 7a4774df310..f12203bc45c 100644 --- a/apps/ui-kit/src/lib/components/molecules/card/CardAction.tsx +++ b/apps/ui-kit/src/lib/components/molecules/card/CardAction.tsx @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { ArrowRight } from '@iota/ui-icons'; +import { ArrowRight } from '@iota/apps-ui-icons'; import { CardActionType } from './card.enums'; import { Button, ButtonSize, ButtonType } from '../../atoms/button'; diff --git a/apps/ui-kit/src/lib/components/molecules/card/CardImage.tsx b/apps/ui-kit/src/lib/components/molecules/card/CardImage.tsx index 6f4efc1e9d7..1a0048dd2fa 100644 --- a/apps/ui-kit/src/lib/components/molecules/card/CardImage.tsx +++ b/apps/ui-kit/src/lib/components/molecules/card/CardImage.tsx @@ -1,7 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import cx from 'classnames'; import { ImageType, ImageShape } from './card.enums'; import { IMAGE_BG_CLASSES, IMAGE_VARIANT_CLASSES } from './card.classes'; diff --git a/apps/ui-kit/src/lib/components/molecules/chip/Chip.tsx b/apps/ui-kit/src/lib/components/molecules/chip/Chip.tsx index 8d98db426aa..f79777b7c73 100644 --- a/apps/ui-kit/src/lib/components/molecules/chip/Chip.tsx +++ b/apps/ui-kit/src/lib/components/molecules/chip/Chip.tsx @@ -12,7 +12,7 @@ import { FOCUS_CLASSES, } from './chip.classes'; import { ButtonUnstyled } from '../../atoms/button'; -import { Close } from '@iota/ui-icons'; +import { Close } from '@iota/apps-ui-icons'; interface ChipProps { /** @@ -47,6 +47,10 @@ interface ChipProps { * Trailing element to show in the chip. */ trailingElement?: React.JSX.Element; + /** + * The button is disabled or not. + */ + disabled?: boolean; } export function Chip({ @@ -58,18 +62,20 @@ export function Chip({ avatar, leadingElement, trailingElement, + disabled, }: ChipProps) { const chipState = selected ? ChipState.Selected : ChipState.Default; return ( <ButtonUnstyled onClick={onClick} className={cx( - 'border transition-all duration-500 ease-in-out', + 'border transition-all duration-500 ease-in-out disabled:opacity-40', ROUNDED_CLASS, BACKGROUND_CLASSES[chipState], BORDER_CLASSES[chipState], FOCUS_CLASSES, )} + disabled={disabled} > <span className={cx( @@ -77,7 +83,7 @@ export function Chip({ avatar ? 'py-xxs' : 'py-[6px]', avatar ? 'pl-xxs' : leadingElement ? 'pl-xs' : 'pl-sm', ROUNDED_CLASS, - STATE_LAYER_CLASSES, + !disabled && STATE_LAYER_CLASSES, showClose ? 'pr-xs' : 'pr-sm', TEXT_COLOR[chipState], )} diff --git a/apps/ui-kit/src/lib/components/molecules/display-stats/DisplayStats.tsx b/apps/ui-kit/src/lib/components/molecules/display-stats/DisplayStats.tsx index 417b2b8427e..c68566f6343 100644 --- a/apps/ui-kit/src/lib/components/molecules/display-stats/DisplayStats.tsx +++ b/apps/ui-kit/src/lib/components/molecules/display-stats/DisplayStats.tsx @@ -1,10 +1,10 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React, { ReactNode } from 'react'; +import { ReactNode } from 'react'; import { Tooltip, TooltipPosition } from '../../atoms'; -import { Info } from '@iota/ui-icons'; -import { DisplayStatsType, DisplayStatsSize } from './display-stats.enums'; +import { Info } from '@iota/apps-ui-icons'; +import { DisplayStatsType, DisplayStatsSize } from './displayStats.enums'; import cx from 'classnames'; import { BACKGROUND_CLASSES, @@ -13,7 +13,7 @@ import { VALUE_TEXT_CLASSES, SUPPORTING_LABEL_TEXT_CLASSES, LABEL_TEXT_CLASSES, -} from './display-stats.classes'; +} from './displayStats.classes'; interface DisplayStatsProps { /** diff --git a/apps/ui-kit/src/lib/components/molecules/display-stats/display-stats.classes.ts b/apps/ui-kit/src/lib/components/molecules/display-stats/displayStats.classes.ts similarity index 94% rename from apps/ui-kit/src/lib/components/molecules/display-stats/display-stats.classes.ts rename to apps/ui-kit/src/lib/components/molecules/display-stats/displayStats.classes.ts index 55a470f8d6d..cd61e36c357 100644 --- a/apps/ui-kit/src/lib/components/molecules/display-stats/display-stats.classes.ts +++ b/apps/ui-kit/src/lib/components/molecules/display-stats/displayStats.classes.ts @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { DisplayStatsType, DisplayStatsSize } from './display-stats.enums'; +import { DisplayStatsType, DisplayStatsSize } from './displayStats.enums'; export const BACKGROUND_CLASSES: Record<DisplayStatsType, string> = { [DisplayStatsType.Default]: 'bg-neutral-96 dark:bg-neutral-10', diff --git a/apps/ui-kit/src/lib/components/molecules/display-stats/display-stats.enums.ts b/apps/ui-kit/src/lib/components/molecules/display-stats/displayStats.enums.ts similarity index 100% rename from apps/ui-kit/src/lib/components/molecules/display-stats/display-stats.enums.ts rename to apps/ui-kit/src/lib/components/molecules/display-stats/displayStats.enums.ts diff --git a/apps/ui-kit/src/lib/components/molecules/display-stats/index.ts b/apps/ui-kit/src/lib/components/molecules/display-stats/index.ts index 9203c88a7db..36d4c1dcf47 100644 --- a/apps/ui-kit/src/lib/components/molecules/display-stats/index.ts +++ b/apps/ui-kit/src/lib/components/molecules/display-stats/index.ts @@ -2,4 +2,4 @@ // SPDX-License-Identifier: Apache-2.0 export * from './DisplayStats'; -export * from './display-stats.enums'; +export * from './displayStats.enums'; diff --git a/apps/ui-kit/src/lib/components/molecules/dropdown/dropdown-position.enum.ts b/apps/ui-kit/src/lib/components/molecules/dropdown/dropdownPosition.enums.ts similarity index 100% rename from apps/ui-kit/src/lib/components/molecules/dropdown/dropdown-position.enum.ts rename to apps/ui-kit/src/lib/components/molecules/dropdown/dropdownPosition.enums.ts diff --git a/apps/ui-kit/src/lib/components/molecules/dropdown/index.ts b/apps/ui-kit/src/lib/components/molecules/dropdown/index.ts index 7661c872580..97c3abbbd0b 100644 --- a/apps/ui-kit/src/lib/components/molecules/dropdown/index.ts +++ b/apps/ui-kit/src/lib/components/molecules/dropdown/index.ts @@ -2,4 +2,4 @@ // SPDX-License-Identifier: Apache-2.0 export * from './Dropdown'; -export * from './dropdown-position.enum'; +export * from './dropdownPosition.enums'; diff --git a/apps/ui-kit/src/lib/components/molecules/input/Input.tsx b/apps/ui-kit/src/lib/components/molecules/input/Input.tsx index c0ff7fe4b4b..86526254985 100644 --- a/apps/ui-kit/src/lib/components/molecules/input/Input.tsx +++ b/apps/ui-kit/src/lib/components/molecules/input/Input.tsx @@ -13,7 +13,7 @@ import { } from './input.classes'; import { InputType } from './input.enums'; import { SecondaryText } from '../../atoms/secondary-text'; -import { Close, VisibilityOff, VisibilityOn } from '@iota/ui-icons'; +import { Close, VisibilityOff, VisibilityOn } from '@iota/apps-ui-icons'; import { ButtonUnstyled } from '../../atoms/button'; import { InputPropsByType, NumericFormatInputProps } from './input.types'; import { NumericFormat } from 'react-number-format'; diff --git a/apps/ui-kit/src/lib/components/molecules/input/TextArea.tsx b/apps/ui-kit/src/lib/components/molecules/input/TextArea.tsx index 003be38944c..4ca28019a2a 100644 --- a/apps/ui-kit/src/lib/components/molecules/input/TextArea.tsx +++ b/apps/ui-kit/src/lib/components/molecules/input/TextArea.tsx @@ -11,7 +11,7 @@ import { } from './input.classes'; import cx from 'classnames'; import { ButtonUnstyled } from '../../atoms/button'; -import { VisibilityOff, VisibilityOn } from '@iota/ui-icons'; +import { VisibilityOff, VisibilityOn } from '@iota/apps-ui-icons'; type TextAreaProps = Omit< React.TextareaHTMLAttributes<HTMLTextAreaElement>, diff --git a/apps/ui-kit/src/lib/components/molecules/navbar-item/NavbarItem.tsx b/apps/ui-kit/src/lib/components/molecules/navbar-item/NavbarItem.tsx index 52eb085081a..a9864a14a17 100644 --- a/apps/ui-kit/src/lib/components/molecules/navbar-item/NavbarItem.tsx +++ b/apps/ui-kit/src/lib/components/molecules/navbar-item/NavbarItem.tsx @@ -1,7 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import { NavbarItemHorizontal } from './NavbarItemHorizontal'; import { NavbarItemVertical } from './NavbarItemVertical'; import { NavbarItemType } from './navbarItem.enums'; diff --git a/apps/ui-kit/src/lib/components/molecules/navbar-item/NavbarItemHorizontal.tsx b/apps/ui-kit/src/lib/components/molecules/navbar-item/NavbarItemHorizontal.tsx index 328f90aa384..ac0b7fceb26 100644 --- a/apps/ui-kit/src/lib/components/molecules/navbar-item/NavbarItemHorizontal.tsx +++ b/apps/ui-kit/src/lib/components/molecules/navbar-item/NavbarItemHorizontal.tsx @@ -1,7 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import cx from 'classnames'; import { BADGE_WITH_TEXT, diff --git a/apps/ui-kit/src/lib/components/molecules/navbar-item/NavbarItemVertical.tsx b/apps/ui-kit/src/lib/components/molecules/navbar-item/NavbarItemVertical.tsx index 0212d273523..039ff6fcaef 100644 --- a/apps/ui-kit/src/lib/components/molecules/navbar-item/NavbarItemVertical.tsx +++ b/apps/ui-kit/src/lib/components/molecules/navbar-item/NavbarItemVertical.tsx @@ -1,7 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import cx from 'classnames'; import { SELECTED_BACKGROUND, diff --git a/apps/ui-kit/src/lib/components/molecules/search/Search.tsx b/apps/ui-kit/src/lib/components/molecules/search/Search.tsx index 35565d8f6af..71dc2499446 100644 --- a/apps/ui-kit/src/lib/components/molecules/search/Search.tsx +++ b/apps/ui-kit/src/lib/components/molecules/search/Search.tsx @@ -1,9 +1,9 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React, { useEffect, useRef, useState } from 'react'; +import { useEffect, useRef, useState } from 'react'; import cx from 'classnames'; -import { Loader, Search as SearchIcon } from '@iota/ui-icons'; +import { Loader, Search as SearchIcon } from '@iota/apps-ui-icons'; import { Divider, SearchBarType } from '@/components'; import { BACKGROUND_COLORS, diff --git a/apps/ui-kit/src/lib/components/molecules/segmented-button/SegmentedButton.tsx b/apps/ui-kit/src/lib/components/molecules/segmented-button/SegmentedButton.tsx index fb09cabcb1f..195606ecec4 100644 --- a/apps/ui-kit/src/lib/components/molecules/segmented-button/SegmentedButton.tsx +++ b/apps/ui-kit/src/lib/components/molecules/segmented-button/SegmentedButton.tsx @@ -1,10 +1,10 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React, { PropsWithChildren } from 'react'; -import { BACKGROUND_COLORS, OUTLINED_BORDER } from './segmented-button.classes'; +import { PropsWithChildren } from 'react'; +import { BACKGROUND_COLORS, OUTLINED_BORDER } from './segmentedButton.classes'; import cx from 'classnames'; -import { SegmentedButtonType } from './segmented-button.enums'; +import { SegmentedButtonType } from './segmentedButton.enums'; import { ButtonSegmentType } from '../../atoms'; interface SegmentedButtonProps { diff --git a/apps/ui-kit/src/lib/components/molecules/segmented-button/index.ts b/apps/ui-kit/src/lib/components/molecules/segmented-button/index.ts index 481c1f46d93..0df9efe8def 100644 --- a/apps/ui-kit/src/lib/components/molecules/segmented-button/index.ts +++ b/apps/ui-kit/src/lib/components/molecules/segmented-button/index.ts @@ -3,4 +3,4 @@ export * from './SegmentedButton'; -export * from './segmented-button.enums'; +export * from './segmentedButton.enums'; diff --git a/apps/ui-kit/src/lib/components/molecules/segmented-button/segmented-button.classes.ts b/apps/ui-kit/src/lib/components/molecules/segmented-button/segmentedButton.classes.ts similarity index 86% rename from apps/ui-kit/src/lib/components/molecules/segmented-button/segmented-button.classes.ts rename to apps/ui-kit/src/lib/components/molecules/segmented-button/segmentedButton.classes.ts index f7160bed6b1..3ed05f9ac8c 100644 --- a/apps/ui-kit/src/lib/components/molecules/segmented-button/segmented-button.classes.ts +++ b/apps/ui-kit/src/lib/components/molecules/segmented-button/segmentedButton.classes.ts @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { SegmentedButtonType } from './segmented-button.enums'; +import { SegmentedButtonType } from './segmentedButton.enums'; export const BACKGROUND_COLORS: Record<SegmentedButtonType, string> = { [SegmentedButtonType.Outlined]: 'bg-transparent', diff --git a/apps/ui-kit/src/lib/components/molecules/segmented-button/segmented-button.enums.ts b/apps/ui-kit/src/lib/components/molecules/segmented-button/segmentedButton.enums.ts similarity index 100% rename from apps/ui-kit/src/lib/components/molecules/segmented-button/segmented-button.enums.ts rename to apps/ui-kit/src/lib/components/molecules/segmented-button/segmentedButton.enums.ts diff --git a/apps/ui-kit/src/lib/components/molecules/select/Select.tsx b/apps/ui-kit/src/lib/components/molecules/select/Select.tsx index d5d0754de25..866eb7bc74a 100644 --- a/apps/ui-kit/src/lib/components/molecules/select/Select.tsx +++ b/apps/ui-kit/src/lib/components/molecules/select/Select.tsx @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { TriangleDown } from '@iota/ui-icons'; +import { TriangleDown } from '@iota/apps-ui-icons'; import cx from 'classnames'; import { forwardRef, useEffect, useState } from 'react'; import { Dropdown } from '../dropdown/Dropdown'; diff --git a/apps/ui-kit/src/lib/components/molecules/table-cell/table-cell.enums.ts b/apps/ui-kit/src/lib/components/molecules/table-cell/table-cell.enums.ts deleted file mode 100644 index c89759317d8..00000000000 --- a/apps/ui-kit/src/lib/components/molecules/table-cell/table-cell.enums.ts +++ /dev/null @@ -1,18 +0,0 @@ -// Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -export enum TableCellType { - Text = 'text', - TextToCopy = 'text-to-copy', - Badge = 'badge', - AvatarText = 'avatar-text', - Checkbox = 'checkbox', - Placeholder = 'placeholder', - Link = 'link', - Children = 'children', -} - -export enum TableCellTextColor { - Default = 'text-neutral-40 dark:text-neutral-60', - Dark = 'text-neutral-10 dark:text-neutral-92', -} diff --git a/apps/ui-kit/src/lib/components/molecules/table-header-cell/TableHeaderCell.tsx b/apps/ui-kit/src/lib/components/molecules/table-header-cell/TableHeaderCell.tsx index 84beb087ecb..08f071e26ae 100644 --- a/apps/ui-kit/src/lib/components/molecules/table-header-cell/TableHeaderCell.tsx +++ b/apps/ui-kit/src/lib/components/molecules/table-header-cell/TableHeaderCell.tsx @@ -1,10 +1,11 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; -import { SortByDown, SortByUp } from '@iota/ui-icons'; + +import { useState } from 'react'; +import { SortByDown, SortByUp } from '@iota/apps-ui-icons'; import cx from 'classnames'; import { Checkbox } from '@/lib'; -import { TableHeaderCellSortOrder } from './table-header-cell.enums'; +import { TableHeaderCellSortOrder } from './tableHeaderCell.enums'; export interface TableHeaderCellProps { /** @@ -64,7 +65,7 @@ export function TableHeaderCell({ onSortClick, onCheckboxChange, }: TableHeaderCellProps): JSX.Element { - const [sortOrder, setSortOrder] = React.useState<TableHeaderCellSortOrder | null>( + const [sortOrder, setSortOrder] = useState<TableHeaderCellSortOrder | null>( TableHeaderCellSortOrder.Asc, ); diff --git a/apps/ui-kit/src/lib/components/molecules/table-header-cell/index.ts b/apps/ui-kit/src/lib/components/molecules/table-header-cell/index.ts index 42f32bb3b76..d4e701c2740 100644 --- a/apps/ui-kit/src/lib/components/molecules/table-header-cell/index.ts +++ b/apps/ui-kit/src/lib/components/molecules/table-header-cell/index.ts @@ -2,4 +2,4 @@ // SPDX-License-Identifier: Apache-2.0 export * from './TableHeaderCell'; -export * from './table-header-cell.enums'; +export * from './tableHeaderCell.enums'; diff --git a/apps/ui-kit/src/lib/components/molecules/table-header-cell/table-header-cell.enums.ts b/apps/ui-kit/src/lib/components/molecules/table-header-cell/tableHeaderCell.enums.ts similarity index 100% rename from apps/ui-kit/src/lib/components/molecules/table-header-cell/table-header-cell.enums.ts rename to apps/ui-kit/src/lib/components/molecules/table-header-cell/tableHeaderCell.enums.ts diff --git a/apps/ui-kit/src/lib/components/molecules/title/Title.tsx b/apps/ui-kit/src/lib/components/molecules/title/Title.tsx index bfaaaf8e37b..1298499ecf3 100644 --- a/apps/ui-kit/src/lib/components/molecules/title/Title.tsx +++ b/apps/ui-kit/src/lib/components/molecules/title/Title.tsx @@ -2,10 +2,10 @@ // SPDX-License-Identifier: Apache-2.0 import { Tooltip, TooltipPosition } from '../../atoms'; -import { Info } from '@iota/ui-icons'; -import { TitleSize } from './title-size.enum'; +import { Info } from '@iota/apps-ui-icons'; +import { TitleSize } from './titleSize.enums'; import cx from 'classnames'; -import { TITLE_PADDINGS, TITLE_SIZE } from './title-classes.constants'; +import { TITLE_PADDINGS, TITLE_SIZE } from './titleClasses.constants'; interface TitleProps { /** diff --git a/apps/ui-kit/src/lib/components/molecules/title/index.ts b/apps/ui-kit/src/lib/components/molecules/title/index.ts index 9a78adf3f24..34efb8414ac 100644 --- a/apps/ui-kit/src/lib/components/molecules/title/index.ts +++ b/apps/ui-kit/src/lib/components/molecules/title/index.ts @@ -2,4 +2,4 @@ // SPDX-License-Identifier: Apache-2.0 export * from './Title'; -export { TitleSize } from './title-size.enum'; +export { TitleSize } from './titleSize.enums'; diff --git a/apps/ui-kit/src/lib/components/molecules/title/title-classes.constants.ts b/apps/ui-kit/src/lib/components/molecules/title/titleClasses.constants.ts similarity index 88% rename from apps/ui-kit/src/lib/components/molecules/title/title-classes.constants.ts rename to apps/ui-kit/src/lib/components/molecules/title/titleClasses.constants.ts index 15640fbb854..eb66d1726b5 100644 --- a/apps/ui-kit/src/lib/components/molecules/title/title-classes.constants.ts +++ b/apps/ui-kit/src/lib/components/molecules/title/titleClasses.constants.ts @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { TitleSize } from './title-size.enum'; +import { TitleSize } from './titleSize.enums'; export const TITLE_PADDINGS: Record<TitleSize, string> = { [TitleSize.Small]: 'px-md py-sm--rs', diff --git a/apps/ui-kit/src/lib/components/molecules/title/title-size.enum.ts b/apps/ui-kit/src/lib/components/molecules/title/titleSize.enums.ts similarity index 100% rename from apps/ui-kit/src/lib/components/molecules/title/title-size.enum.ts rename to apps/ui-kit/src/lib/components/molecules/title/titleSize.enums.ts diff --git a/apps/ui-kit/src/lib/components/organisms/accordion/Accordion.tsx b/apps/ui-kit/src/lib/components/organisms/accordion/Accordion.tsx index 1489c52c1d5..47768f08b73 100644 --- a/apps/ui-kit/src/lib/components/organisms/accordion/Accordion.tsx +++ b/apps/ui-kit/src/lib/components/organisms/accordion/Accordion.tsx @@ -1,9 +1,9 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React, { PropsWithChildren } from 'react'; +import { PropsWithChildren } from 'react'; import cx from 'classnames'; -import { ArrowDown } from '@iota/ui-icons'; +import { ArrowDown } from '@iota/apps-ui-icons'; import { Button, ButtonType } from '@/lib'; import { ICON_STYLE } from './accordion.classes'; diff --git a/apps/ui-kit/src/lib/components/organisms/dialog/Dialog.tsx b/apps/ui-kit/src/lib/components/organisms/dialog/Dialog.tsx index 50fd835d2e3..3f78f5635de 100644 --- a/apps/ui-kit/src/lib/components/organisms/dialog/Dialog.tsx +++ b/apps/ui-kit/src/lib/components/organisms/dialog/Dialog.tsx @@ -5,7 +5,7 @@ import * as RadixDialog from '@radix-ui/react-dialog'; import * as VisuallyHidden from '@radix-ui/react-visually-hidden'; import cx from 'classnames'; import * as React from 'react'; -import { Close } from '@iota/ui-icons'; +import { Close } from '@iota/apps-ui-icons'; import { useEffect, useState } from 'react'; import { DialogPosition } from './dialog.enums'; diff --git a/apps/ui-kit/src/lib/components/organisms/navbar/Navbar.tsx b/apps/ui-kit/src/lib/components/organisms/navbar/Navbar.tsx index 845bb339142..a7af094a0be 100644 --- a/apps/ui-kit/src/lib/components/organisms/navbar/Navbar.tsx +++ b/apps/ui-kit/src/lib/components/organisms/navbar/Navbar.tsx @@ -2,7 +2,7 @@ // SPDX-License-Identifier: Apache-2.0 import cx from 'classnames'; -import { IotaLogoMark, MenuIcon } from '@iota/ui-icons'; +import { IotaLogoMark, MenuIcon } from '@iota/apps-ui-icons'; import { NavbarItem, NavbarItemProps } from '@/components/molecules/navbar-item/NavbarItem'; export type NavbarItemWithId = NavbarItemProps & { id: string }; diff --git a/apps/ui-kit/src/lib/components/organisms/navbar/NavbarSlideout.tsx b/apps/ui-kit/src/lib/components/organisms/navbar/NavbarSlideout.tsx index ad2055b8964..495f1c546ac 100644 --- a/apps/ui-kit/src/lib/components/organisms/navbar/NavbarSlideout.tsx +++ b/apps/ui-kit/src/lib/components/organisms/navbar/NavbarSlideout.tsx @@ -1,8 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; -import { ArrowBack } from '@iota/ui-icons'; +import { ArrowBack } from '@iota/apps-ui-icons'; import cx from 'classnames'; import { Button, ButtonType, NavbarItem, NavbarItemType } from '@/components'; import { NavbarProps } from './Navbar'; diff --git a/apps/ui-kit/src/lib/components/organisms/table/Table.tsx b/apps/ui-kit/src/lib/components/organisms/table/Table.tsx index ac11bc19eff..83375a81b19 100644 --- a/apps/ui-kit/src/lib/components/organisms/table/Table.tsx +++ b/apps/ui-kit/src/lib/components/organisms/table/Table.tsx @@ -13,7 +13,7 @@ import { TableCellBase, TableHeaderCell, } from '@/lib'; -import { ArrowLeft, DoubleArrowLeft, ArrowRight, DoubleArrowRight } from '@iota/ui-icons'; +import { ArrowLeft, DoubleArrowLeft, ArrowRight, DoubleArrowRight } from '@iota/apps-ui-icons'; export interface TablePaginationOptions { /** diff --git a/apps/ui-kit/src/lib/enums/index.ts b/apps/ui-kit/src/lib/enums/index.ts index b85a71aeebb..c15d6246d2c 100644 --- a/apps/ui-kit/src/lib/enums/index.ts +++ b/apps/ui-kit/src/lib/enums/index.ts @@ -1,4 +1,4 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export * from './screenSize.enum'; +export * from './screenSize.enums'; diff --git a/apps/ui-kit/src/lib/enums/screenSize.enum.ts b/apps/ui-kit/src/lib/enums/screenSize.enums.ts similarity index 100% rename from apps/ui-kit/src/lib/enums/screenSize.enum.ts rename to apps/ui-kit/src/lib/enums/screenSize.enums.ts diff --git a/apps/ui-kit/src/lib/tailwind/base.preset.ts b/apps/ui-kit/src/lib/tailwind/base.presets.ts similarity index 100% rename from apps/ui-kit/src/lib/tailwind/base.preset.ts rename to apps/ui-kit/src/lib/tailwind/base.presets.ts diff --git a/apps/ui-kit/src/lib/tailwind/constants/index.ts b/apps/ui-kit/src/lib/tailwind/constants/index.ts index 2adc80fcca0..7be12319899 100644 --- a/apps/ui-kit/src/lib/tailwind/constants/index.ts +++ b/apps/ui-kit/src/lib/tailwind/constants/index.ts @@ -3,4 +3,4 @@ export * from './scaling.constants'; export * from './fontSizes.constants'; -export * from './variable-spacing.constants'; +export * from './variableSpacing.constants'; diff --git a/apps/ui-kit/src/lib/tailwind/constants/variable-spacing.constants.ts b/apps/ui-kit/src/lib/tailwind/constants/variableSpacing.constants.ts similarity index 97% rename from apps/ui-kit/src/lib/tailwind/constants/variable-spacing.constants.ts rename to apps/ui-kit/src/lib/tailwind/constants/variableSpacing.constants.ts index ad6ed83f78d..c091667ccba 100644 --- a/apps/ui-kit/src/lib/tailwind/constants/variable-spacing.constants.ts +++ b/apps/ui-kit/src/lib/tailwind/constants/variableSpacing.constants.ts @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { ScreenSize } from '../../enums/screenSize.enum'; +import { ScreenSize } from '../../enums/screenSize.enums'; enum VariableSpacingClassPrefix { Xs = 'xs--rs', diff --git a/apps/ui-kit/src/lib/tailwind/index.ts b/apps/ui-kit/src/lib/tailwind/index.ts index 5edddfd1578..d0766485b1b 100644 --- a/apps/ui-kit/src/lib/tailwind/index.ts +++ b/apps/ui-kit/src/lib/tailwind/index.ts @@ -1,5 +1,5 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export { default as uiKitResponsivePreset } from './responsive.preset'; -export { default as uiKitStaticPreset } from './static.preset'; +export { default as uiKitResponsivePreset } from './responsive.presets'; +export { default as uiKitStaticPreset } from './static.presets'; diff --git a/apps/ui-kit/src/lib/tailwind/responsive.preset.ts b/apps/ui-kit/src/lib/tailwind/responsive.presets.ts similarity index 95% rename from apps/ui-kit/src/lib/tailwind/responsive.preset.ts rename to apps/ui-kit/src/lib/tailwind/responsive.presets.ts index b86c59eb154..a0c99a45340 100644 --- a/apps/ui-kit/src/lib/tailwind/responsive.preset.ts +++ b/apps/ui-kit/src/lib/tailwind/responsive.presets.ts @@ -5,7 +5,7 @@ import type { Config } from 'tailwindcss'; import type { ScreenBreakpoints } from './types'; import { ScreenSize } from '../enums'; import { getTailwindScreens, pxToRem } from './helpers'; -import { BASE_CONFIG } from './base.preset'; +import { BASE_CONFIG } from './base.presets'; import merge from 'lodash.merge'; const BREAKPOINTS: ScreenBreakpoints = { diff --git a/apps/ui-kit/src/lib/tailwind/static.preset.ts b/apps/ui-kit/src/lib/tailwind/static.presets.ts similarity index 92% rename from apps/ui-kit/src/lib/tailwind/static.preset.ts rename to apps/ui-kit/src/lib/tailwind/static.presets.ts index ad50674fb55..d1a3868ad42 100644 --- a/apps/ui-kit/src/lib/tailwind/static.preset.ts +++ b/apps/ui-kit/src/lib/tailwind/static.presets.ts @@ -4,7 +4,7 @@ import type { Config } from 'tailwindcss'; import type { ScreenBreakpoints } from './types'; import { getTailwindScreens } from './helpers'; -import { BASE_CONFIG } from './base.preset'; +import { BASE_CONFIG } from './base.presets'; import merge from 'lodash.merge'; const BREAKPOINTS: Partial<ScreenBreakpoints> & { default: number } = { diff --git a/apps/ui-kit/src/storybook/constants/index.ts b/apps/ui-kit/src/storybook/constants/index.ts index a259cd8f991..ae109adc501 100644 --- a/apps/ui-kit/src/storybook/constants/index.ts +++ b/apps/ui-kit/src/storybook/constants/index.ts @@ -2,4 +2,4 @@ // SPDX-License-Identifier: Apache-2.0 export * from './typeset.constants'; -export * from './variable-spacing.constants'; +export * from './variableSpacing.constants'; diff --git a/apps/ui-kit/src/storybook/constants/variable-spacing.constants.ts b/apps/ui-kit/src/storybook/constants/variableSpacing.constants.ts similarity index 100% rename from apps/ui-kit/src/storybook/constants/variable-spacing.constants.ts rename to apps/ui-kit/src/storybook/constants/variableSpacing.constants.ts diff --git a/apps/ui-kit/src/storybook/stories/atoms/InfoBox.stories.tsx b/apps/ui-kit/src/storybook/stories/atoms/InfoBox.stories.tsx index 3ae962543f1..33700c4099a 100644 --- a/apps/ui-kit/src/storybook/stories/atoms/InfoBox.stories.tsx +++ b/apps/ui-kit/src/storybook/stories/atoms/InfoBox.stories.tsx @@ -3,7 +3,7 @@ import type { Meta, StoryObj } from '@storybook/react'; import { InfoBox, InfoBoxStyle, InfoBoxType } from '@/components'; -import { Info } from '@iota/ui-icons'; +import { Info } from '@iota/apps-ui-icons'; const meta: Meta<typeof InfoBox> = { component: InfoBox, diff --git a/apps/ui-kit/src/storybook/stories/atoms/ListItem.stories.tsx b/apps/ui-kit/src/storybook/stories/atoms/ListItem.stories.tsx index bcb48ddf751..c3887ab539f 100644 --- a/apps/ui-kit/src/storybook/stories/atoms/ListItem.stories.tsx +++ b/apps/ui-kit/src/storybook/stories/atoms/ListItem.stories.tsx @@ -2,7 +2,7 @@ // SPDX-License-Identifier: Apache-2.0 import { ListItem } from '@/components'; -import { Seed } from '@iota/ui-icons'; +import { Seed } from '@iota/apps-ui-icons'; import type { Meta, StoryObj } from '@storybook/react'; const meta = { diff --git a/apps/ui-kit/src/storybook/stories/atoms/Tooltip.stories.tsx b/apps/ui-kit/src/storybook/stories/atoms/Tooltip.stories.tsx index 3b72332277e..33e71c8e5e2 100644 --- a/apps/ui-kit/src/storybook/stories/atoms/Tooltip.stories.tsx +++ b/apps/ui-kit/src/storybook/stories/atoms/Tooltip.stories.tsx @@ -2,7 +2,7 @@ // SPDX-License-Identifier: Apache-2.0 import { Tooltip, TooltipPosition } from '@/lib/components'; -import { Info } from '@iota/ui-icons'; +import { Info } from '@iota/apps-ui-icons'; import type { Meta, StoryObj } from '@storybook/react'; const meta: Meta<typeof Tooltip> = { diff --git a/apps/ui-kit/src/storybook/stories/components/VariableSpacingShowcase.tsx b/apps/ui-kit/src/storybook/stories/components/VariableSpacingShowcase.tsx index 761e9f883e5..5822fdf8ea2 100644 --- a/apps/ui-kit/src/storybook/stories/components/VariableSpacingShowcase.tsx +++ b/apps/ui-kit/src/storybook/stories/components/VariableSpacingShowcase.tsx @@ -4,7 +4,6 @@ import { Button } from '@/lib'; import { Showcase } from '@/storybook/blocks'; import { VARIABLE_GAP_CLASSES, VARIABLE_PADDING_CLASSES } from '@/storybook/constants'; -import React from 'react'; export function VariableSpacingShowcase() { return ( diff --git a/apps/ui-kit/src/storybook/stories/design-tokens/icons.stories.mdx b/apps/ui-kit/src/storybook/stories/design-tokens/icons.stories.mdx index 3fbab5da9c7..aa4896bba02 100644 --- a/apps/ui-kit/src/storybook/stories/design-tokens/icons.stories.mdx +++ b/apps/ui-kit/src/storybook/stories/design-tokens/icons.stories.mdx @@ -1,6 +1,6 @@ import React from 'react'; import { Meta, Title, IconGallery, IconItem } from '@storybook/blocks'; -import * as Icons from '@iota/ui-icons'; +import * as Icons from '@iota/apps-ui-icons'; <Meta title="Design Tokens/Icon Gallery" /> diff --git a/apps/ui-kit/src/storybook/stories/molecules/Card.stories.tsx b/apps/ui-kit/src/storybook/stories/molecules/Card.stories.tsx index aa1429b7815..8889ddb698e 100644 --- a/apps/ui-kit/src/storybook/stories/molecules/Card.stories.tsx +++ b/apps/ui-kit/src/storybook/stories/molecules/Card.stories.tsx @@ -2,7 +2,7 @@ // SPDX-License-Identifier: Apache-2.0 import type { Meta, StoryObj } from '@storybook/react'; -import { IotaLogoSmall } from '@iota/ui-icons'; +import { IotaLogoSmall } from '@iota/apps-ui-icons'; import { Card, CardProps, diff --git a/apps/ui-kit/src/storybook/stories/molecules/Chip.stories.tsx b/apps/ui-kit/src/storybook/stories/molecules/Chip.stories.tsx index 2c891a3e4c2..33fb28a7355 100644 --- a/apps/ui-kit/src/storybook/stories/molecules/Chip.stories.tsx +++ b/apps/ui-kit/src/storybook/stories/molecules/Chip.stories.tsx @@ -4,7 +4,7 @@ import type { Meta, StoryObj } from '@storybook/react'; import { Chip } from '@/components/molecules/chip/Chip'; -import { PlaceholderReplace } from '@iota/ui-icons'; +import { PlaceholderReplace } from '@iota/apps-ui-icons'; const meta: Meta<typeof Chip> = { component: Chip, diff --git a/apps/ui-kit/src/storybook/stories/molecules/DisplayStats.stories.tsx b/apps/ui-kit/src/storybook/stories/molecules/DisplayStats.stories.tsx index e892659ce9d..9bc46e2f7e3 100644 --- a/apps/ui-kit/src/storybook/stories/molecules/DisplayStats.stories.tsx +++ b/apps/ui-kit/src/storybook/stories/molecules/DisplayStats.stories.tsx @@ -1,7 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import type { Meta, StoryObj } from '@storybook/react'; import { DisplayStats, TooltipPosition, DisplayStatsType, DisplayStatsSize } from '@/components'; diff --git a/apps/ui-kit/src/storybook/stories/molecules/Input.stories.tsx b/apps/ui-kit/src/storybook/stories/molecules/Input.stories.tsx index d9fccd8a736..044e3f23455 100644 --- a/apps/ui-kit/src/storybook/stories/molecules/Input.stories.tsx +++ b/apps/ui-kit/src/storybook/stories/molecules/Input.stories.tsx @@ -4,7 +4,7 @@ import type { Meta, StoryObj } from '@storybook/react'; import { Input, InputType } from '@/lib/components/molecules/input'; -import { PlaceholderReplace } from '@iota/ui-icons'; +import { PlaceholderReplace } from '@iota/apps-ui-icons'; import { ComponentProps, useCallback, useEffect, useState } from 'react'; import { ButtonPill } from '@/lib/components/atoms/button'; diff --git a/apps/ui-kit/src/storybook/stories/molecules/NavbarItem.stories.tsx b/apps/ui-kit/src/storybook/stories/molecules/NavbarItem.stories.tsx index aad83ca81f8..24c93f71b28 100644 --- a/apps/ui-kit/src/storybook/stories/molecules/NavbarItem.stories.tsx +++ b/apps/ui-kit/src/storybook/stories/molecules/NavbarItem.stories.tsx @@ -1,10 +1,9 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import type { Meta, StoryObj } from '@storybook/react'; import { NavbarItem, NavbarItemType } from '@/components'; -import { Home } from '@iota/ui-icons'; +import { Home } from '@iota/apps-ui-icons'; const meta: Meta<typeof NavbarItem> = { component: NavbarItem, diff --git a/apps/ui-kit/src/storybook/stories/molecules/SegmentedButton.stories.tsx b/apps/ui-kit/src/storybook/stories/molecules/SegmentedButton.stories.tsx index a8b1ddc4cfe..07cbdbe8548 100644 --- a/apps/ui-kit/src/storybook/stories/molecules/SegmentedButton.stories.tsx +++ b/apps/ui-kit/src/storybook/stories/molecules/SegmentedButton.stories.tsx @@ -10,7 +10,7 @@ import { SegmentedButtonType, } from '@/components'; import { ComponentProps, useState } from 'react'; -import { PlaceholderReplace } from '@iota/ui-icons'; +import { PlaceholderReplace } from '@iota/apps-ui-icons'; const meta: Meta<typeof SegmentedButton> = { component: SegmentedButton, diff --git a/apps/ui-kit/src/storybook/stories/molecules/Select.stories.tsx b/apps/ui-kit/src/storybook/stories/molecules/Select.stories.tsx index 43ecefbb391..22614d24618 100644 --- a/apps/ui-kit/src/storybook/stories/molecules/Select.stories.tsx +++ b/apps/ui-kit/src/storybook/stories/molecules/Select.stories.tsx @@ -4,7 +4,7 @@ import type { Meta, StoryObj } from '@storybook/react'; import { Select, SelectOption } from '@/components/molecules/select/Select'; import { useState } from 'react'; -import { IotaLogoMark, PlaceholderReplace } from '@iota/ui-icons'; +import { IotaLogoMark, PlaceholderReplace } from '@iota/apps-ui-icons'; const meta: Meta<typeof Select> = { component: Select, diff --git a/apps/ui-kit/src/storybook/stories/organisms/Navbar.stories.tsx b/apps/ui-kit/src/storybook/stories/organisms/Navbar.stories.tsx index 21189796c67..54e2ecccaea 100644 --- a/apps/ui-kit/src/storybook/stories/organisms/Navbar.stories.tsx +++ b/apps/ui-kit/src/storybook/stories/organisms/Navbar.stories.tsx @@ -2,7 +2,7 @@ // SPDX-License-Identifier: Apache-2.0 import type { Meta, StoryObj } from '@storybook/react'; -import { Activity, Apps, Assets, Home } from '@iota/ui-icons'; +import { Activity, Apps, Assets, Home } from '@iota/apps-ui-icons'; import { Navbar, NavbarSlideout, NavbarItemWithId, NavbarProps } from '@/components'; import { useState } from 'react'; diff --git a/apps/ui-kit/tailwind.config.ts b/apps/ui-kit/tailwind.config.ts index 319b7dc1818..1ebb4d52c0f 100644 --- a/apps/ui-kit/tailwind.config.ts +++ b/apps/ui-kit/tailwind.config.ts @@ -1,6 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import preset from './src/lib/tailwind/responsive.preset'; +import preset from './src/lib/tailwind/responsive.presets'; export default preset; diff --git a/apps/ui-kit/turbo.json b/apps/ui-kit/turbo.json new file mode 100644 index 00000000000..ded95d0156a --- /dev/null +++ b/apps/ui-kit/turbo.json @@ -0,0 +1,9 @@ +{ + "extends": ["//"], + "tasks": { + "build-storybook": { + "dependsOn": ["^build"], + "outputs": ["storybook-static/**"] + } + } +} diff --git a/apps/wallet-dashboard/app/(protected)/activity/page.tsx b/apps/wallet-dashboard/app/(protected)/activity/page.tsx index 14f4bc516da..7e89ca90b24 100644 --- a/apps/wallet-dashboard/app/(protected)/activity/page.tsx +++ b/apps/wallet-dashboard/app/(protected)/activity/page.tsx @@ -3,7 +3,6 @@ 'use client'; -import React from 'react'; import { Panel, Title, TitleSize } from '@iota/apps-ui-kit'; import { TransactionsList } from '@/components'; diff --git a/apps/wallet-dashboard/app/(protected)/assets/page.tsx b/apps/wallet-dashboard/app/(protected)/assets/page.tsx index 7007cc86c5c..b091ab2f345 100644 --- a/apps/wallet-dashboard/app/(protected)/assets/page.tsx +++ b/apps/wallet-dashboard/app/(protected)/assets/page.tsx @@ -3,16 +3,25 @@ 'use client'; -import { Panel, Title, Chip, TitleSize } from '@iota/apps-ui-kit'; -import { hasDisplayData, useGetOwnedObjects } from '@iota/core'; +import { useState } from 'react'; +import cl from 'clsx'; +import { usePageAssets, AssetCategory } from '@iota/core'; +import { + Panel, + Title, + TitleSize, + LoadingIndicator, + InfoBox, + InfoBoxType, + InfoBoxStyle, + Chip, +} from '@iota/apps-ui-kit'; import { useCurrentAccount } from '@iota/dapp-kit'; import { IotaObjectData } from '@iota/iota-sdk/client'; -import { useState } from 'react'; -import { AssetCategory } from '@/lib/enums'; -import { AssetList } from '@/components/AssetsList'; -import { AssetDialog } from '@/components/Dialogs/Assets'; +import { Warning } from '@iota/apps-ui-icons'; -const OBJECTS_PER_REQ = 50; +import { AssetTileLink, Loading } from '@/components'; +import { AssetDialog } from '@/components/dialogs/assets'; const ASSET_CATEGORIES: { label: string; value: AssetCategory }[] = [ { @@ -25,31 +34,32 @@ const ASSET_CATEGORIES: { label: string; value: AssetCategory }[] = [ }, ]; +const ASSET_LAYOUT: Record<AssetCategory, string> = { + [AssetCategory.Visual]: + 'grid-template-visual-assets grid max-h-[600px] gap-md overflow-auto py-sm', + [AssetCategory.Other]: 'flex flex-col overflow-auto py-sm', + [AssetCategory.Hidden]: 'flex flex-col overflow-auto py-sm', +}; + export default function AssetsDashboardPage(): React.JSX.Element { const [selectedAsset, setSelectedAsset] = useState<IotaObjectData | null>(null); - const [selectedCategory, setSelectedCategory] = useState<AssetCategory>(AssetCategory.Visual); const account = useCurrentAccount(); - const { data, isFetching, fetchNextPage, hasNextPage, refetch } = useGetOwnedObjects( - account?.address, - undefined, - OBJECTS_PER_REQ, - ); + const accountAddress = account?.address || null; - const assets: IotaObjectData[] = []; + const { + isPending, + refetch, + error, + isError, - for (const page of data?.pages || []) { - for (const asset of page.data) { - if (asset.data && asset.data.objectId) { - if (selectedCategory == AssetCategory.Visual) { - if (hasDisplayData(asset)) { - assets.push(asset.data); - } - } else if (selectedCategory == AssetCategory.Other) { - assets.push(asset.data); - } - } - } - } + ownedAssets, + isAssetsLoaded, + filteredAssets, + selectedAssetCategory, + setSelectedAssetCategory, + observerElem, + isSpinnerVisible, + } = usePageAssets(accountAddress); function onAssetClick(asset: IotaObjectData) { setSelectedAsset(asset); @@ -59,31 +69,69 @@ export default function AssetsDashboardPage(): React.JSX.Element { <Panel> <Title title="Assets" size={TitleSize.Medium} /> <div className="px-lg"> - <div className="flex flex-row items-center justify-start gap-xs py-xs"> - {ASSET_CATEGORIES.map((tab) => ( - <Chip - key={tab.label} - label={tab.label} - onClick={() => setSelectedCategory(tab.value)} - selected={selectedCategory === tab.value} + {isError ? ( + <div className="mb-2 flex h-full w-full items-center justify-center p-2"> + <InfoBox + type={InfoBoxType.Error} + title="Sync error (data might be outdated)" + supportingText={error?.message ?? 'An error occurred'} + icon={<Warning />} + style={InfoBoxStyle.Default} /> - ))} - </div> + </div> + ) : ( + <> + {isAssetsLoaded && + Boolean(ownedAssets?.visual.length || ownedAssets?.other.length) ? ( + <div className="flex flex-row items-center justify-start gap-xs py-xs"> + {ASSET_CATEGORIES.map(({ value, label }) => ( + <Chip + key={value} + onClick={() => setSelectedAssetCategory(value)} + label={label} + selected={selectedAssetCategory === value} + disabled={ + AssetCategory.Visual === value + ? !ownedAssets?.visual.length + : !ownedAssets?.other.length + } + /> + ))} + </div> + ) : null} + <Loading loading={isPending}> + <div + className={cl( + 'max-h-[600px]', + selectedAssetCategory && ASSET_LAYOUT[selectedAssetCategory], + )} + > + {filteredAssets.map((asset) => ( + <AssetTileLink + key={asset.digest} + asset={asset} + type={selectedAssetCategory} + onClick={onAssetClick} + /> + ))} + <div ref={observerElem}> + {isSpinnerVisible ? ( + <div className="mt-1 flex h-full w-full justify-center"> + <LoadingIndicator /> + </div> + ) : null} + </div> + </div> + </Loading> - <AssetList - assets={assets} - selectedCategory={selectedCategory} - onClick={onAssetClick} - hasNextPage={hasNextPage} - isFetchingNextPage={isFetching} - fetchNextPage={fetchNextPage} - /> - {selectedAsset && ( - <AssetDialog - onClose={() => setSelectedAsset(null)} - asset={selectedAsset} - refetchAssets={refetch} - /> + {selectedAsset && ( + <AssetDialog + onClose={() => setSelectedAsset(null)} + asset={selectedAsset} + refetchAssets={refetch} + /> + )} + </> )} </div> </Panel> diff --git a/apps/wallet-dashboard/app/(protected)/components/sidebar/Sidebar.tsx b/apps/wallet-dashboard/app/(protected)/components/sidebar/Sidebar.tsx index 03742f9e8e4..298b2bc5642 100644 --- a/apps/wallet-dashboard/app/(protected)/components/sidebar/Sidebar.tsx +++ b/apps/wallet-dashboard/app/(protected)/components/sidebar/Sidebar.tsx @@ -2,7 +2,7 @@ // SPDX-License-Identifier: Apache-2.0 import { PROTECTED_ROUTES } from '@/lib/constants/routes.constants'; -import { IotaLogoMark } from '@iota/ui-icons'; +import { IotaLogoMark } from '@iota/apps-ui-icons'; import { SidebarItem } from './SidebarItem'; import { Feature } from '@iota/core'; import { useFeature } from '@growthbook/growthbook-react'; diff --git a/apps/wallet-dashboard/app/(protected)/components/top-nav/TopNav.tsx b/apps/wallet-dashboard/app/(protected)/components/top-nav/TopNav.tsx index 3b802387ca8..52337093bc3 100644 --- a/apps/wallet-dashboard/app/(protected)/components/top-nav/TopNav.tsx +++ b/apps/wallet-dashboard/app/(protected)/components/top-nav/TopNav.tsx @@ -3,14 +3,15 @@ import { SettingsDialog, useSettingsDialog } from '@/components'; import { Badge, BadgeType, Button, ButtonType } from '@iota/apps-ui-kit'; -import { ConnectButton, useIotaClientContext } from '@iota/dapp-kit'; -import { getNetwork, Network } from '@iota/iota-sdk/client'; -import { ThemeSwitcher } from '@iota/core'; -import { Settings } from '@iota/ui-icons'; +import { ConnectButton } from '@iota/dapp-kit'; +import { Network } from '@iota/iota-sdk/client'; +import { toTitleCase, ThemeSwitcher } from '@iota/core'; +import { Settings } from '@iota/apps-ui-icons'; +import { usePersistedNetwork } from '@/hooks'; export function TopNav() { - const { network } = useIotaClientContext(); - const { name: networkName } = getNetwork(network); + const { persistedNetwork } = usePersistedNetwork(); + const { isSettingsDialogOpen, settingsDialogView, @@ -22,8 +23,10 @@ export function TopNav() { return ( <div className="flex w-full flex-row items-center justify-end gap-md py-xs--rs"> <Badge - label={networkName} - type={network === Network.Mainnet ? BadgeType.PrimarySoft : BadgeType.Neutral} + label={toTitleCase(persistedNetwork)} + type={ + persistedNetwork === Network.Mainnet ? BadgeType.PrimarySoft : BadgeType.Neutral + } /> <ConnectButton size="md" /> <SettingsDialog diff --git a/apps/wallet-dashboard/app/(protected)/layout.tsx b/apps/wallet-dashboard/app/(protected)/layout.tsx index 72fa80c67cc..620a8a1184b 100644 --- a/apps/wallet-dashboard/app/(protected)/layout.tsx +++ b/apps/wallet-dashboard/app/(protected)/layout.tsx @@ -2,8 +2,7 @@ // SPDX-License-Identifier: Apache-2.0 'use client'; -import { Notifications } from '@/components/index'; -import React, { type PropsWithChildren } from 'react'; +import { type PropsWithChildren } from 'react'; import { Sidebar, TopNav } from './components'; function DashboardLayout({ children }: PropsWithChildren): JSX.Element { @@ -22,7 +21,6 @@ function DashboardLayout({ children }: PropsWithChildren): JSX.Element { <div className="flex-1 py-md--rs">{children}</div> </div> </div> - <Notifications /> </div> ); } diff --git a/apps/wallet-dashboard/app/(protected)/migrations/page.tsx b/apps/wallet-dashboard/app/(protected)/migrations/page.tsx index b042465c49e..e9215b53f13 100644 --- a/apps/wallet-dashboard/app/(protected)/migrations/page.tsx +++ b/apps/wallet-dashboard/app/(protected)/migrations/page.tsx @@ -6,8 +6,8 @@ import { useState, useMemo, useCallback } from 'react'; import { useQueryClient } from '@tanstack/react-query'; import clsx from 'clsx'; -import { useGetStardustMigratableObjects } from '@/hooks'; -import { summarizeMigratableObjectValues, summarizeTimelockedObjectValues } from '@/lib/utils'; +import { useGetStardustMigratableObjects, useGroupedStardustObjects } from '@/hooks'; +import { getStardustObjectsTotals } from '@/lib/utils'; import { Button, ButtonSize, @@ -19,18 +19,20 @@ import { Panel, Title, } from '@iota/apps-ui-kit'; -import { Assets, Clock, IotaLogoMark, Tokens } from '@iota/ui-icons'; +import { Assets, IotaLogoMark, Tokens } from '@iota/apps-ui-icons'; import { useCurrentAccount, useIotaClient } from '@iota/dapp-kit'; import { STARDUST_BASIC_OUTPUT_TYPE, STARDUST_NFT_OUTPUT_TYPE, useFormatCoin } from '@iota/core'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { StardustOutputMigrationStatus } from '@/lib/enums'; import { MigrationObjectsPanel, MigrationDialog } from '@/components'; +import { useRouter } from 'next/navigation'; function MigrationDashboardPage(): JSX.Element { const account = useCurrentAccount(); const address = account?.address || ''; const queryClient = useQueryClient(); const iotaClient = useIotaClient(); + const router = useRouter(); const [isMigrationDialogOpen, setIsMigrationDialogOpen] = useState(false); const [selectedStardustObjectsCategory, setSelectedStardustObjectsCategory] = useState< StardustOutputMigrationStatus | undefined @@ -45,24 +47,49 @@ function MigrationDashboardPage(): JSX.Element { timelockedNftOutputs, } = stardustMigrationObjects || {}; + const { data: resolvedMigrationObjects = [] } = useGroupedStardustObjects( + [...(migratableBasicOutputs || []), ...(migratableNftOutputs || [])], + false, + ); + const { data: resolvedTimelockedObjects = [] } = useGroupedStardustObjects( + [...(timelockedBasicOutputs || []), ...(timelockedNftOutputs || [])], + true, + ); + const { - totalIotaAmount, + totalIotaAmount: migratableIotaAmount, totalNativeTokens: migratableNativeTokens, totalVisualAssets: migratableVisualAssets, - } = summarizeMigratableObjectValues({ + } = getStardustObjectsTotals({ basicOutputs: migratableBasicOutputs, nftOutputs: migratableNftOutputs, address, + resolvedObjects: resolvedMigrationObjects, }); - const { totalTimelockedObjects } = summarizeTimelockedObjectValues({ + const { + totalIotaAmount: timelockedIotaAmount, + totalNativeTokens: timelockedNativeTokens, + totalVisualAssets: timelockedVisualAssets, + } = getStardustObjectsTotals({ basicOutputs: timelockedBasicOutputs, nftOutputs: timelockedNftOutputs, + address, + resolvedObjects: resolvedTimelockedObjects, }); const hasMigratableObjects = - (migratableBasicOutputs?.length || 0) > 0 && (migratableNftOutputs?.length || 0) > 0; + (migratableBasicOutputs?.length || 0) > 0 || (migratableNftOutputs?.length || 0) > 0; + const hasTimelockedObjects = + (timelockedBasicOutputs?.length || 0) > 0 || (timelockedNftOutputs?.length || 0) > 0; - const [timelockedIotaTokens, symbol] = useFormatCoin(totalIotaAmount, IOTA_TYPE_ARG); + const [migratableIotaAmountFormatted, migratableIotaAmountSymbol] = useFormatCoin( + migratableIotaAmount, + IOTA_TYPE_ARG, + ); + const [timelockedIotaAmountFormatted, timelockedIotaAmountSymbol] = useFormatCoin( + timelockedIotaAmount, + IOTA_TYPE_ARG, + ); const handleOnSuccess = useCallback( (digest: string) => { @@ -81,6 +108,9 @@ function MigrationDashboardPage(): JSX.Element { { StructType: STARDUST_NFT_OUTPUT_TYPE }, ], }); + queryClient.invalidateQueries({ + queryKey: ['migration-transaction', address], + }); }); }, [iotaClient, queryClient, address], @@ -88,7 +118,7 @@ function MigrationDashboardPage(): JSX.Element { const MIGRATION_CARDS: MigrationDisplayCardProps[] = [ { - title: `${timelockedIotaTokens} ${symbol}`, + title: `${migratableIotaAmountFormatted} ${migratableIotaAmountSymbol}`, subtitle: 'IOTA Tokens', icon: IotaLogoMark, }, @@ -106,9 +136,19 @@ function MigrationDashboardPage(): JSX.Element { const TIMELOCKED_ASSETS_CARDS: MigrationDisplayCardProps[] = [ { - title: `${totalTimelockedObjects}`, - subtitle: 'Time-locked', - icon: Clock, + title: `${timelockedIotaAmountFormatted} ${timelockedIotaAmountSymbol}`, + subtitle: 'IOTA Tokens', + icon: IotaLogoMark, + }, + { + title: `${timelockedNativeTokens}`, + subtitle: 'Native Tokens', + icon: Tokens, + }, + { + title: `${timelockedVisualAssets}`, + subtitle: 'Visual Assets', + icon: Assets, }, ]; @@ -139,6 +179,11 @@ function MigrationDashboardPage(): JSX.Element { setSelectedStardustObjectsCategory(undefined); } + function handleMigrationDialogClose() { + setIsMigrationDialogOpen(false); + router.replace('/home'); + } + return ( <div className="flex h-full w-full flex-wrap items-center justify-center space-y-4"> <div @@ -159,6 +204,7 @@ function MigrationDashboardPage(): JSX.Element { selectedStardustObjectsCategory === StardustOutputMigrationStatus.TimeLocked } + handleClose={handleMigrationDialogClose} /> )} <Panel> @@ -216,7 +262,7 @@ function MigrationDashboardPage(): JSX.Element { disabled={ selectedStardustObjectsCategory === StardustOutputMigrationStatus.TimeLocked || - !totalTimelockedObjects + !hasTimelockedObjects } onClick={() => setSelectedStardustObjectsCategory( @@ -231,7 +277,7 @@ function MigrationDashboardPage(): JSX.Element { <MigrationObjectsPanel selectedObjects={selectedObjects} onClose={handleCloseDetailsPanel} - isTimelocked={ + groupByTimelockUC={ selectedStardustObjectsCategory === StardustOutputMigrationStatus.TimeLocked } /> diff --git a/apps/wallet-dashboard/app/(protected)/staking/page.tsx b/apps/wallet-dashboard/app/(protected)/staking/page.tsx index 87f69eccffa..05e17b53e0b 100644 --- a/apps/wallet-dashboard/app/(protected)/staking/page.tsx +++ b/apps/wallet-dashboard/app/(protected)/staking/page.tsx @@ -37,7 +37,7 @@ import { } from '@iota/core'; import { useCurrentAccount, useIotaClient, useIotaClientQuery } from '@iota/dapp-kit'; import { IotaSystemStateSummary } from '@iota/iota-sdk/client'; -import { Info } from '@iota/ui-icons'; +import { Info } from '@iota/apps-ui-icons'; import { useMemo } from 'react'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { IotaSignAndExecuteTransactionOutput } from '@iota/wallet-standard'; diff --git a/apps/wallet-dashboard/app/(protected)/vesting/page.tsx b/apps/wallet-dashboard/app/(protected)/vesting/page.tsx index e29d94a7fef..37fb903678f 100644 --- a/apps/wallet-dashboard/app/(protected)/vesting/page.tsx +++ b/apps/wallet-dashboard/app/(protected)/vesting/page.tsx @@ -11,11 +11,10 @@ import { UnstakeDialog, StakeDialogView, } from '@/components'; -import { UnstakeDialogView } from '@/components/Dialogs/unstake/enums'; -import { useUnstakeDialog } from '@/components/Dialogs/unstake/hooks'; -import { useGetSupplyIncreaseVestingObjects, useNotifications } from '@/hooks'; +import { UnstakeDialogView } from '@/components/dialogs/unstake/enums'; +import { useUnstakeDialog } from '@/components/dialogs/unstake/hooks'; +import { useGetSupplyIncreaseVestingObjects } from '@/hooks'; import { groupTimelockedStakedObjects, TimelockedStakedObjectsGrouped } from '@/lib/utils'; -import { NotificationType } from '@/stores/notificationStore'; import { useFeature } from '@growthbook/growthbook-react'; import { Panel, @@ -34,6 +33,8 @@ import { Button, ButtonType, LoadingIndicator, + LabelText, + LabelTextSize, } from '@iota/apps-ui-kit'; import { Theme, @@ -51,11 +52,12 @@ import { } from '@iota/dapp-kit'; import { IotaValidatorSummary } from '@iota/iota-sdk/client'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; -import { Calendar, StarHex } from '@iota/ui-icons'; +import { Calendar, StarHex } from '@iota/apps-ui-icons'; import { useRouter } from 'next/navigation'; import { useEffect, useState } from 'react'; import { StakedTimelockObject } from '@/components'; import { IotaSignAndExecuteTransactionOutput } from '@iota/wallet-standard'; +import toast from 'react-hot-toast'; export default function VestingDashboardPage(): JSX.Element { const [timelockedObjectsToUnstake, setTimelockedObjectsToUnstake] = @@ -66,7 +68,6 @@ export default function VestingDashboardPage(): JSX.Element { const router = useRouter(); const { data: system } = useIotaClientQuery('getLatestIotaSystemState'); const [isVestingScheduleDialogOpen, setIsVestingScheduleDialogOpen] = useState(false); - const { addNotification } = useNotifications(); const { data: activeValidators } = useGetActiveValidatorsInfo(); const { mutateAsync: signAndExecuteTransaction } = useSignAndExecuteTransaction(); const { theme } = useTheme(); @@ -82,11 +83,11 @@ export default function VestingDashboardPage(): JSX.Element { nextPayout, supplyIncreaseVestingPortfolio, supplyIncreaseVestingSchedule, - supplyIncreaseVestingMapped, supplyIncreaseVestingStakedMapped, isTimelockedStakedObjectsLoading, unlockAllSupplyIncreaseVesting, refreshStakeList, + isSupplyIncreaseVestingScheduleEmpty, } = useGetSupplyIncreaseVestingObjects(address); const timelockedStakedObjectsGrouped: TimelockedStakedObjectsGrouped[] = @@ -151,6 +152,11 @@ export default function VestingDashboardPage(): JSX.Element { IOTA_TYPE_ARG, ); + const [formattedAvailableStaking, availableStakingSymbol] = useFormatCoin( + supplyIncreaseVestingSchedule.availableStaking, + IOTA_TYPE_ARG, + ); + function handleOnSuccess(digest: string): void { setTimelockedObjectsToUnstake(null); @@ -163,7 +169,7 @@ export default function VestingDashboardPage(): JSX.Element { const handleCollect = () => { if (!unlockAllSupplyIncreaseVesting?.transactionBlock) { - addNotification('Failed to create a Transaction', NotificationType.Error); + toast.error('Failed to create a Transaction'); return; } signAndExecuteTransaction( @@ -177,10 +183,10 @@ export default function VestingDashboardPage(): JSX.Element { }, ) .then(() => { - addNotification('Collect transaction has been sent'); + toast.success('Collect transaction has been sent'); }) .catch(() => { - addNotification('Collect transaction was not sent', NotificationType.Error); + toast.error('Collect transaction was not sent'); }); }; @@ -203,7 +209,7 @@ export default function VestingDashboardPage(): JSX.Element { useEffect(() => { if (!supplyIncreaseVestingEnabled) { - router.push('/'); + router.push('/home'); } }, [router, supplyIncreaseVestingEnabled]); @@ -222,7 +228,7 @@ export default function VestingDashboardPage(): JSX.Element { <Panel> <Title title="Vesting" size={TitleSize.Medium} /> <div className="flex flex-col gap-md p-lg pt-sm"> - <div className="flex h-24 flex-row gap-4"> + <div className="flex h-24 flex-row gap-md"> <DisplayStats label="Total Vested" value={formattedTotalVested} @@ -291,18 +297,20 @@ export default function VestingDashboardPage(): JSX.Element { </div> </Panel> - {supplyIncreaseVestingMapped.length === 0 ? ( + {isSupplyIncreaseVestingScheduleEmpty ? ( <Banner videoSrc={videoSrc} title="Stake Vested Tokens" subtitle="Earn Rewards" onButtonClick={() => handleNewStake()} buttonText="Stake" + disabled={supplyIncreaseVestingSchedule.availableStaking === 0n} /> ) : null} </div> - {supplyIncreaseVestingMapped.length !== 0 ? ( + {!isSupplyIncreaseVestingScheduleEmpty && + supplyIncreaseVestingSchedule.totalStaked !== 0n ? ( <div className="flex w-full md:w-1/2"> <Panel> <Title @@ -321,8 +329,8 @@ export default function VestingDashboardPage(): JSX.Element { } /> - <div className="flex flex-col px-lg py-sm"> - <div className="flex flex-row gap-md"> + <div className="flex flex-col gap-y-md px-lg py-sm"> + <div className="flex flex-row gap-x-md"> <DisplayStats label="Your stake" value={`${totalStakedFormatted} ${totalStakedSymbol}`} @@ -332,6 +340,21 @@ export default function VestingDashboardPage(): JSX.Element { value={`${totalEarnedFormatted} ${totalEarnedSymbol}`} /> </div> + <div className="flex w-full"> + <Card type={CardType.Filled}> + <CardBody + title="" + subtitle={ + <LabelText + size={LabelTextSize.Large} + label="Available for staking" + text={formattedAvailableStaking} + supportingLabel={availableStakingSymbol} + /> + } + /> + </Card> + </div> </div> <div className="flex flex-col px-lg py-sm"> <div className="flex w-full flex-col items-center justify-center space-y-4 pt-4"> diff --git a/apps/wallet-dashboard/app/page.tsx b/apps/wallet-dashboard/app/page.tsx index 3392c231d31..d6b8fde2f65 100644 --- a/apps/wallet-dashboard/app/page.tsx +++ b/apps/wallet-dashboard/app/page.tsx @@ -5,7 +5,7 @@ import { ConnectButton, useCurrentWallet, useAutoConnectWallet } from '@iota/dapp-kit'; import { redirect } from 'next/navigation'; -import { IotaLogoWeb } from '@iota/ui-icons'; +import { IotaLogoWeb } from '@iota/apps-ui-icons'; import { HOMEPAGE_ROUTE } from '@/lib/constants/routes.constants'; import { Theme, ThemeSwitcher, useTheme } from '@iota/core'; import { LoadingIndicator } from '@iota/apps-ui-kit'; diff --git a/apps/wallet-dashboard/components/AmountBox.tsx b/apps/wallet-dashboard/components/AmountBox.tsx index 8db7bb609d0..a59926b1357 100644 --- a/apps/wallet-dashboard/components/AmountBox.tsx +++ b/apps/wallet-dashboard/components/AmountBox.tsx @@ -8,7 +8,7 @@ interface AmountBoxProps { amount: string; } -function AmountBox({ title, amount }: AmountBoxProps): JSX.Element { +export function AmountBox({ title, amount }: AmountBoxProps): JSX.Element { return ( <div className="flex items-center justify-center gap-4 pt-12"> <Box title={title}> @@ -17,5 +17,3 @@ function AmountBox({ title, amount }: AmountBoxProps): JSX.Element { </div> ); } - -export default AmountBox; diff --git a/apps/wallet-dashboard/components/AssetsList.tsx b/apps/wallet-dashboard/components/AssetsList.tsx deleted file mode 100644 index adb9e251581..00000000000 --- a/apps/wallet-dashboard/components/AssetsList.tsx +++ /dev/null @@ -1,64 +0,0 @@ -// Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -import { AssetCategory } from '@/lib/enums'; -import { IotaObjectData } from '@iota/iota-sdk/client'; -import { AssetTileLink } from '@/components'; -import { LoadingIndicator } from '@iota/apps-ui-kit'; -import { useEffect, useRef } from 'react'; -import { useOnScreen } from '@iota/core'; -import cl from 'clsx'; - -interface AssetListProps { - assets: IotaObjectData[]; - selectedCategory: AssetCategory; - hasNextPage: boolean; - isFetchingNextPage: boolean; - fetchNextPage: () => void; - onClick: (asset: IotaObjectData) => void; -} - -const ASSET_LAYOUT: Record<AssetCategory, string> = { - [AssetCategory.Visual]: - 'grid-template-visual-assets grid max-h-[600px] gap-md overflow-auto py-sm', - [AssetCategory.Other]: 'flex flex-col overflow-auto py-sm', -}; - -export function AssetList({ - assets, - selectedCategory, - hasNextPage, - isFetchingNextPage, - fetchNextPage, - onClick, -}: AssetListProps): React.JSX.Element { - const observerElem = useRef<HTMLDivElement | null>(null); - const { isIntersecting } = useOnScreen(observerElem); - const isSpinnerVisible = isFetchingNextPage && hasNextPage; - - useEffect(() => { - if (isIntersecting && hasNextPage && !isFetchingNextPage && fetchNextPage) { - fetchNextPage(); - } - }, [isIntersecting, fetchNextPage, hasNextPage, isFetchingNextPage]); - - return ( - <div className={cl('max-h-[600px]', ASSET_LAYOUT[selectedCategory])}> - {assets.map((asset) => ( - <AssetTileLink - key={asset.digest} - asset={asset} - type={selectedCategory} - onClick={onClick} - /> - ))} - <div ref={observerElem}> - {isSpinnerVisible ? ( - <div className="mt-1 flex h-full w-full justify-center"> - <LoadingIndicator /> - </div> - ) : null} - </div> - </div> - ); -} diff --git a/apps/wallet-dashboard/components/Banner.tsx b/apps/wallet-dashboard/components/Banner.tsx index aa697f9f7b5..93a828b5732 100644 --- a/apps/wallet-dashboard/components/Banner.tsx +++ b/apps/wallet-dashboard/components/Banner.tsx @@ -2,7 +2,6 @@ // SPDX-License-Identifier: Apache-2.0 import { Button, ButtonSize, ButtonType, Panel } from '@iota/apps-ui-kit'; -import React from 'react'; interface BannerProps { videoSrc: string; @@ -10,8 +9,16 @@ interface BannerProps { subtitle?: string; onButtonClick: () => void; buttonText: string; + disabled?: boolean; } -export function Banner({ videoSrc, title, subtitle, onButtonClick, buttonText }: BannerProps) { +export function Banner({ + videoSrc, + title, + subtitle, + onButtonClick, + buttonText, + disabled, +}: BannerProps) { return ( <Panel bgColor="bg-secondary-90 dark:bg-secondary-10"> <div className="flex h-full w-full justify-between "> @@ -30,6 +37,7 @@ export function Banner({ videoSrc, title, subtitle, onButtonClick, buttonText }: size={ButtonSize.Small} type={ButtonType.Outlined} text={buttonText} + disabled={disabled} /> </div> </div> diff --git a/apps/wallet-dashboard/components/Box.tsx b/apps/wallet-dashboard/components/Box.tsx index 3b9b285ea6e..7ffda69c41c 100644 --- a/apps/wallet-dashboard/components/Box.tsx +++ b/apps/wallet-dashboard/components/Box.tsx @@ -1,14 +1,14 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; +import { ReactNode } from 'react'; interface BoxProps { - children: React.ReactNode; + children: ReactNode; title?: string; } -function Box({ children, title }: BoxProps): JSX.Element { +export function Box({ children, title }: BoxProps): JSX.Element { return ( <div className="flex flex-col items-center gap-2 rounded-lg border border-white p-4"> {title && <h2>{title}</h2>} @@ -16,5 +16,3 @@ function Box({ children, title }: BoxProps): JSX.Element { </div> ); } - -export default Box; diff --git a/apps/wallet-dashboard/components/Dialogs/Assets/AssetDialog.tsx b/apps/wallet-dashboard/components/Dialogs/Assets/AssetDialog.tsx deleted file mode 100644 index c71390a33aa..00000000000 --- a/apps/wallet-dashboard/components/Dialogs/Assets/AssetDialog.tsx +++ /dev/null @@ -1,102 +0,0 @@ -// Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -import React, { useState } from 'react'; -import { Dialog } from '@iota/apps-ui-kit'; -import { FormikProvider, useFormik } from 'formik'; -import { useIotaClient, useCurrentAccount } from '@iota/dapp-kit'; -import { createNftSendValidationSchema } from '@iota/core'; -import { DetailsView, SendView } from './views'; -import { IotaObjectData } from '@iota/iota-sdk/client'; -import { AssetsDialogView } from './constants'; -import { useCreateSendAssetTransaction, useNotifications } from '@/hooks'; -import { NotificationType } from '@/stores/notificationStore'; -import { TransactionDetailsView } from '../SendToken'; -import { DialogLayout } from '../layout'; - -interface AssetsDialogProps { - onClose: () => void; - asset: IotaObjectData; - refetchAssets: () => void; -} - -interface FormValues { - to: string; -} - -const INITIAL_VALUES: FormValues = { - to: '', -}; - -export function AssetDialog({ onClose, asset, refetchAssets }: AssetsDialogProps): JSX.Element { - const [view, setView] = useState<AssetsDialogView>(AssetsDialogView.Details); - const account = useCurrentAccount(); - const [digest, setDigest] = useState<string>(''); - const activeAddress = account?.address ?? ''; - const objectId = asset?.objectId ?? ''; - const { addNotification } = useNotifications(); - const iotaClient = useIotaClient(); - const validationSchema = createNftSendValidationSchema(activeAddress, objectId); - - const { mutation: sendAsset } = useCreateSendAssetTransaction(objectId); - - const formik = useFormik<FormValues>({ - initialValues: INITIAL_VALUES, - validationSchema: validationSchema, - onSubmit: onSubmit, - validateOnChange: true, - }); - - async function onSubmit(values: FormValues) { - try { - const executed = await sendAsset.mutateAsync(values.to); - - const tx = await iotaClient.waitForTransaction({ - digest: executed.digest, - }); - - setDigest(tx.digest); - refetchAssets(); - addNotification('Transfer transaction successful', NotificationType.Success); - setView(AssetsDialogView.TransactionDetails); - } catch { - addNotification('Transfer transaction failed', NotificationType.Error); - } - } - - function onDetailsSend() { - setView(AssetsDialogView.Send); - } - - function onSendViewBack() { - setView(AssetsDialogView.Details); - } - function onOpenChange() { - setView(AssetsDialogView.Details); - onClose(); - } - return ( - <Dialog open onOpenChange={onOpenChange}> - <DialogLayout> - <> - {view === AssetsDialogView.Details && ( - <DetailsView asset={asset} onClose={onOpenChange} onSend={onDetailsSend} /> - )} - {view === AssetsDialogView.Send && ( - <FormikProvider value={formik}> - <SendView - asset={asset} - onClose={onOpenChange} - onBack={onSendViewBack} - /> - </FormikProvider> - )} - - {view === AssetsDialogView.TransactionDetails && !!digest ? ( - <TransactionDetailsView digest={digest} onClose={onOpenChange} /> - ) : null} - </> - </DialogLayout> - </Dialog> - ); -} diff --git a/apps/wallet-dashboard/components/Dialogs/MigrationDialog.tsx b/apps/wallet-dashboard/components/Dialogs/MigrationDialog.tsx deleted file mode 100644 index f1f661c2db4..00000000000 --- a/apps/wallet-dashboard/components/Dialogs/MigrationDialog.tsx +++ /dev/null @@ -1,176 +0,0 @@ -// Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -import React from 'react'; -import { VirtualList } from '@/components'; -import { useCurrentAccount, useSignAndExecuteTransaction } from '@iota/dapp-kit'; -import { IotaObjectData } from '@iota/iota-sdk/client'; -import { useMigrationTransaction } from '@/hooks/useMigrationTransaction'; -import { - Button, - Dialog, - Header, - InfoBox, - InfoBoxStyle, - InfoBoxType, - KeyValueInfo, - LoadingIndicator, - Panel, - Title, - TitleSize, -} from '@iota/apps-ui-kit'; -import { useGroupedMigrationObjectsByExpirationDate } from '@/hooks'; -import { Loader, Warning } from '@iota/ui-icons'; -import { DialogLayout, DialogLayoutBody, DialogLayoutFooter } from './layout'; -import { MigrationObjectDetailsCard } from '../migration/migration-object-details-card'; -import { Collapsible, useFormatCoin } from '@iota/core'; -import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; -import { summarizeMigratableObjectValues } from '@/lib/utils'; -import toast from 'react-hot-toast'; - -interface MigrationDialogProps { - basicOutputObjects: IotaObjectData[] | undefined; - nftOutputObjects: IotaObjectData[] | undefined; - onSuccess: (digest: string) => void; - setOpen: (bool: boolean) => void; - open: boolean; - isTimelocked: boolean; -} - -export function MigrationDialog({ - basicOutputObjects = [], - nftOutputObjects = [], - onSuccess, - open, - setOpen, - isTimelocked, -}: MigrationDialogProps): JSX.Element { - const account = useCurrentAccount(); - const { - data: migrateData, - isPending: isMigrationPending, - isError: isMigrationError, - } = useMigrationTransaction(account?.address || '', basicOutputObjects, nftOutputObjects); - - const { - data: resolvedObjects = [], - isLoading, - error: isGroupedMigrationError, - } = useGroupedMigrationObjectsByExpirationDate( - [...basicOutputObjects, ...nftOutputObjects], - isTimelocked, - ); - - const { mutateAsync: signAndExecuteTransaction, isPending: isSendingTransaction } = - useSignAndExecuteTransaction(); - const { totalNotOwnedStorageDepositReturnAmount } = summarizeMigratableObjectValues({ - basicOutputs: basicOutputObjects, - nftOutputs: nftOutputObjects, - address: account?.address || '', - }); - - const [gasFee, gasFeeSymbol] = useFormatCoin(migrateData?.gasBudget, IOTA_TYPE_ARG); - const [totalStorageDepositReturnAmountFormatted, totalStorageDepositReturnAmountSymbol] = - useFormatCoin(totalNotOwnedStorageDepositReturnAmount.toString(), IOTA_TYPE_ARG); - - async function handleMigrate(): Promise<void> { - if (!migrateData) return; - signAndExecuteTransaction( - { - transaction: migrateData.transaction, - }, - { - onSuccess: (tx) => { - onSuccess(tx.digest); - }, - }, - ) - .then(() => { - toast.success('Migration transaction has been sent'); - }) - .catch(() => { - toast.error('Migration transaction was not sent'); - }); - } - - return ( - <Dialog open={open} onOpenChange={setOpen}> - <DialogLayout> - <Header title="Confirmation" onClose={() => setOpen(false)} titleCentered /> - <DialogLayoutBody> - <div className="flex h-full flex-col gap-y-md overflow-y-auto"> - {isGroupedMigrationError && !isLoading && ( - <InfoBox - title="Error" - supportingText="Failed to load migration objects" - style={InfoBoxStyle.Elevated} - type={InfoBoxType.Error} - icon={<Warning />} - /> - )} - {isLoading ? ( - <LoadingIndicator text="Loading migration objects" /> - ) : ( - <> - <Collapsible - defaultOpen - render={() => ( - <Title size={TitleSize.Small} title="Assets to Migrate" /> - )} - > - <div className="h-[600px] pb-md--rs"> - <VirtualList - heightClassName="h-full" - overflowClassName="overflow-y-auto" - items={resolvedObjects} - estimateSize={() => 58} - render={(migrationObject) => ( - <MigrationObjectDetailsCard - migrationObject={migrationObject} - isTimelocked={isTimelocked} - /> - )} - /> - </div> - </Collapsible> - <Panel hasBorder> - <div className="flex flex-col gap-y-sm p-md"> - <KeyValueInfo - keyText="Legacy storage deposit" - value={totalStorageDepositReturnAmountFormatted || '-'} - supportingLabel={totalStorageDepositReturnAmountSymbol} - fullwidth - /> - <KeyValueInfo - keyText="Gas Fees" - value={gasFee || '-'} - supportingLabel={gasFeeSymbol} - fullwidth - /> - </div> - </Panel> - </> - )} - </div> - </DialogLayoutBody> - <DialogLayoutFooter> - <Button - text="Migrate" - disabled={isMigrationPending || isMigrationError || isSendingTransaction} - onClick={handleMigrate} - icon={ - isMigrationPending || isSendingTransaction ? ( - <Loader - className="h-4 w-4 animate-spin" - data-testid="loading-indicator" - /> - ) : null - } - iconAfterText - fullWidth - /> - </DialogLayoutFooter> - </DialogLayout> - </Dialog> - ); -} diff --git a/apps/wallet-dashboard/components/Dialogs/Staking/views/index.ts b/apps/wallet-dashboard/components/Dialogs/Staking/views/index.ts deleted file mode 100644 index 5a0ffed2be6..00000000000 --- a/apps/wallet-dashboard/components/Dialogs/Staking/views/index.ts +++ /dev/null @@ -1,8 +0,0 @@ -// Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -export { default as EnterAmountView } from './EnterAmountView'; -export { default as EnterTimelockedAmountView } from './EnterTimelockedAmountView'; -export { default as EnterAmountDialogLayout } from './EnterAmountDialogLayout'; -export { default as SelectValidatorView } from './SelectValidatorView'; -export * from './DetailsView'; diff --git a/apps/wallet-dashboard/components/ExternalImage.tsx b/apps/wallet-dashboard/components/ExternalImage.tsx index 8e23c5a8d08..8f582018f15 100644 --- a/apps/wallet-dashboard/components/ExternalImage.tsx +++ b/apps/wallet-dashboard/components/ExternalImage.tsx @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export default function ExternalImage({ +export function ExternalImage({ ...imageProps }: React.ImgHTMLAttributes<HTMLImageElement>): JSX.Element { // eslint-disable-next-line @next/next/no-img-element diff --git a/apps/wallet-dashboard/components/Input.tsx b/apps/wallet-dashboard/components/Input.tsx index 40463e507ce..6216f607aa8 100644 --- a/apps/wallet-dashboard/components/Input.tsx +++ b/apps/wallet-dashboard/components/Input.tsx @@ -1,8 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; - interface InputProps { label?: string; value: string; @@ -12,7 +10,7 @@ interface InputProps { error?: string; } -function Input({ +export function Input({ label, value, onChange, @@ -34,5 +32,3 @@ function Input({ </div> ); } - -export default Input; diff --git a/apps/wallet-dashboard/components/Notifications/Notifications.tsx b/apps/wallet-dashboard/components/Notifications/Notifications.tsx deleted file mode 100644 index 77493b164d9..00000000000 --- a/apps/wallet-dashboard/components/Notifications/Notifications.tsx +++ /dev/null @@ -1,64 +0,0 @@ -// Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -'use client'; - -import { useEffect, useState } from 'react'; - -import { - NotificationData, - NotificationType, - useNotificationStore, -} from '@/stores/notificationStore'; - -const ALERT_FADE_OUT_DURATION = 700; - -const NOTIFICATION_TYPE_TO_COLOR = { - [NotificationType.Info]: 'bg-blue-500', - [NotificationType.Success]: 'bg-green-500', - [NotificationType.Error]: 'bg-red-500', - [NotificationType.Warning]: 'bg-yellow-500', -}; - -function Notification(props: { notification: NotificationData }): JSX.Element { - const clearNotification = useNotificationStore((state) => state.clearNotification); - const [transition, setTransition] = useState('opacity-100'); - - const { notification } = props; - - const bgColor = NOTIFICATION_TYPE_TO_COLOR[notification.type]; - - useEffect(() => { - const fadeOutputTimeout = setTimeout(() => { - setTransition('opacity-0'); - }, notification.duration - ALERT_FADE_OUT_DURATION); - - const removeTimeout = setTimeout(() => { - clearNotification(notification.index); - }, notification.duration); - - return () => { - clearTimeout(fadeOutputTimeout); - clearTimeout(removeTimeout); - }; - }, [notification, clearNotification]); - - return ( - <div - className={`flex items-center justify-center rounded-xl text-center ${bgColor} mt-1 w-[300px] p-2 transition-opacity duration-[${ALERT_FADE_OUT_DURATION}] ease-in ${transition}`} - > - {notification.message} - </div> - ); -} - -export default function Notifications(): JSX.Element { - const notifications = useNotificationStore((state) => state.notifications); - return ( - <div className="fixed right-2 top-1 z-[99999]"> - {notifications.map((notification) => ( - <Notification key={`${notification.index}`} notification={notification} /> - ))} - </div> - ); -} diff --git a/apps/wallet-dashboard/components/PaginationOptions.tsx b/apps/wallet-dashboard/components/PaginationOptions.tsx index eba852565cb..baa17a1f178 100644 --- a/apps/wallet-dashboard/components/PaginationOptions.tsx +++ b/apps/wallet-dashboard/components/PaginationOptions.tsx @@ -2,7 +2,7 @@ // SPDX-License-Identifier: Apache-2.0 import { Button, ButtonType, ButtonSize } from '@iota/apps-ui-kit'; -import { ArrowLeft, ArrowRight, DoubleArrowLeft, DoubleArrowRight } from '@iota/ui-icons'; +import { ArrowLeft, ArrowRight, DoubleArrowLeft, DoubleArrowRight } from '@iota/apps-ui-icons'; interface PaginationOptionsProps { pagination: { diff --git a/apps/wallet-dashboard/components/RouteLink.tsx b/apps/wallet-dashboard/components/RouteLink.tsx index f4395dcd421..73c0d94e954 100644 --- a/apps/wallet-dashboard/components/RouteLink.tsx +++ b/apps/wallet-dashboard/components/RouteLink.tsx @@ -4,7 +4,6 @@ 'use client'; import { usePathname } from 'next/navigation'; -import React from 'react'; import Link from 'next/link'; interface RouteLinkProps { @@ -12,7 +11,7 @@ interface RouteLinkProps { path: string; } -function RouteLink({ title, path }: RouteLinkProps): JSX.Element { +export function RouteLink({ title, path }: RouteLinkProps): JSX.Element { const currentPath = usePathname(); const isActive = currentPath && (path === currentPath || path.startsWith(currentPath)); @@ -29,5 +28,3 @@ function RouteLink({ title, path }: RouteLinkProps): JSX.Element { </Link> ); } - -export default RouteLink; diff --git a/apps/wallet-dashboard/components/SupplyIncreaseVestingOverview.tsx b/apps/wallet-dashboard/components/SupplyIncreaseVestingOverview.tsx index 7608bda1a3a..3f9ba046256 100644 --- a/apps/wallet-dashboard/components/SupplyIncreaseVestingOverview.tsx +++ b/apps/wallet-dashboard/components/SupplyIncreaseVestingOverview.tsx @@ -15,10 +15,10 @@ import { Panel, Title, } from '@iota/apps-ui-kit'; -import { StakeDialog, useStakeDialog } from './Dialogs'; +import { StakeDialog, useStakeDialog } from './dialogs'; import { TIMELOCK_IOTA_TYPE, useCountdownByTimestamp, useFormatCoin } from '@iota/core'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; -import SvgClock from '@iota/ui-icons/src/Clock'; +import { Clock } from '@iota/apps-ui-icons'; import { useQueryClient } from '@tanstack/react-query'; export function SupplyIncreaseVestingOverview() { @@ -29,7 +29,7 @@ export function SupplyIncreaseVestingOverview() { const { nextPayout, supplyIncreaseVestingSchedule, - supplyIncreaseVestingMapped, + isSupplyIncreaseVestingScheduleEmpty, supplyIncreaseVestingStakedMapped, } = useGetSupplyIncreaseVestingObjects(address); @@ -57,9 +57,6 @@ export function SupplyIncreaseVestingOverview() { IOTA_TYPE_ARG, ); - const showSupplyIncreaseVestingOverview = - supplyIncreaseVestingMapped.length > 0 || supplyIncreaseVestingStakedMapped.length > 0; - function handleOnSuccess(digest: string): void { iotaClient .waitForTransaction({ @@ -81,7 +78,7 @@ export function SupplyIncreaseVestingOverview() { }); } - return showSupplyIncreaseVestingOverview ? ( + return !isSupplyIncreaseVestingScheduleEmpty || supplyIncreaseVestingStakedMapped.length > 0 ? ( <div style={{ gridArea: 'vesting' }} className="with-vesting flex grow overflow-hidden"> <Panel> <Title title="Vesting" /> @@ -107,7 +104,7 @@ export function SupplyIncreaseVestingOverview() { type={CardActionType.Button} buttonType={ButtonType.Ghost} title={formattedLastPayoutExpirationTime} - icon={<SvgClock />} + icon={<Clock />} /> </Card> </div> diff --git a/apps/wallet-dashboard/components/VirtualList.tsx b/apps/wallet-dashboard/components/VirtualList.tsx index 58dc598a364..667cef76df3 100644 --- a/apps/wallet-dashboard/components/VirtualList.tsx +++ b/apps/wallet-dashboard/components/VirtualList.tsx @@ -3,7 +3,7 @@ 'use client'; -import React, { ReactNode, useEffect } from 'react'; +import { ReactNode, useEffect, useRef } from 'react'; import { useVirtualizer } from '@tanstack/react-virtual'; import clsx from 'clsx'; @@ -19,7 +19,7 @@ interface VirtualListProps<T> { overflowClassName?: string; } -function VirtualList<T>({ +export function VirtualList<T>({ items, hasNextPage = false, isFetchingNextPage = false, @@ -30,7 +30,7 @@ function VirtualList<T>({ heightClassName = 'h-fit', overflowClassName, }: VirtualListProps<T>): JSX.Element { - const containerRef = React.useRef<HTMLDivElement | null>(null); + const containerRef = useRef<HTMLDivElement | null>(null); const virtualizer = useVirtualizer({ // Render one more item if there is still pages to be fetched count: hasNextPage ? items.length + 1 : items.length, @@ -98,5 +98,3 @@ function VirtualList<T>({ </div> ); } - -export default VirtualList; diff --git a/apps/wallet-dashboard/components/account-balance/AccountBalance.tsx b/apps/wallet-dashboard/components/account-balance/AccountBalance.tsx index 9bb3f297b27..b84dec37910 100644 --- a/apps/wallet-dashboard/components/account-balance/AccountBalance.tsx +++ b/apps/wallet-dashboard/components/account-balance/AccountBalance.tsx @@ -6,7 +6,7 @@ import { formatAddress, IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { useBalance, useFormatCoin } from '@iota/core'; import { Address, Button, ButtonSize, ButtonType, Panel } from '@iota/apps-ui-kit'; import { getNetwork } from '@iota/iota-sdk/client'; -import { ReceiveFundsDialog, SendTokenDialog } from '../Dialogs'; +import { ReceiveFundsDialog, SendTokenDialog } from '../dialogs'; import toast from 'react-hot-toast'; import { useState } from 'react'; diff --git a/apps/wallet-dashboard/components/AppList/AppList.tsx b/apps/wallet-dashboard/components/app-list/AppList.tsx similarity index 95% rename from apps/wallet-dashboard/components/AppList/AppList.tsx rename to apps/wallet-dashboard/components/app-list/AppList.tsx index 0ccd665cf95..a3e1b487d03 100644 --- a/apps/wallet-dashboard/components/AppList/AppList.tsx +++ b/apps/wallet-dashboard/components/app-list/AppList.tsx @@ -1,11 +1,10 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import Image from 'next/image'; import { useAppsBackend } from '@iota/core'; import { useQuery } from '@tanstack/react-query'; -import { type AppListItem } from './AppList.types'; +import { type AppListItem } from './appList.types'; import { getDefaultNetwork } from '@iota/iota-sdk/client'; const AppListItem = (props: AppListItem) => { diff --git a/apps/wallet-dashboard/components/AppList/AppList.types.ts b/apps/wallet-dashboard/components/app-list/appList.types.ts similarity index 100% rename from apps/wallet-dashboard/components/AppList/AppList.types.ts rename to apps/wallet-dashboard/components/app-list/appList.types.ts diff --git a/apps/wallet-dashboard/components/AppList/index.ts b/apps/wallet-dashboard/components/app-list/index.ts similarity index 100% rename from apps/wallet-dashboard/components/AppList/index.ts rename to apps/wallet-dashboard/components/app-list/index.ts diff --git a/apps/wallet-dashboard/components/Buttons/Button.tsx b/apps/wallet-dashboard/components/buttons/Button.tsx similarity index 79% rename from apps/wallet-dashboard/components/Buttons/Button.tsx rename to apps/wallet-dashboard/components/buttons/Button.tsx index b15149e1201..d8bc2efca2d 100644 --- a/apps/wallet-dashboard/components/Buttons/Button.tsx +++ b/apps/wallet-dashboard/components/buttons/Button.tsx @@ -1,15 +1,13 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; - interface ButtonProps { onClick?: (event: React.MouseEvent<HTMLButtonElement, MouseEvent>) => void; children: React.ReactNode; disabled?: boolean; } -function Button({ onClick, children, disabled }: ButtonProps): JSX.Element { +export function Button({ onClick, children, disabled }: ButtonProps): JSX.Element { return ( <button onClick={onClick} @@ -20,5 +18,3 @@ function Button({ onClick, children, disabled }: ButtonProps): JSX.Element { </button> ); } - -export default Button; diff --git a/apps/wallet-dashboard/components/buttons/index.ts b/apps/wallet-dashboard/components/buttons/index.ts new file mode 100644 index 00000000000..0ecf88b8b19 --- /dev/null +++ b/apps/wallet-dashboard/components/buttons/index.ts @@ -0,0 +1,4 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export * from './Button'; diff --git a/apps/wallet-dashboard/components/Cards/VisualAssetDetailsCard.tsx b/apps/wallet-dashboard/components/cards/VisualAssetDetailsCard.tsx similarity index 100% rename from apps/wallet-dashboard/components/Cards/VisualAssetDetailsCard.tsx rename to apps/wallet-dashboard/components/cards/VisualAssetDetailsCard.tsx diff --git a/apps/wallet-dashboard/components/Cards/index.ts b/apps/wallet-dashboard/components/cards/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Cards/index.ts rename to apps/wallet-dashboard/components/cards/index.ts diff --git a/apps/wallet-dashboard/components/coins/MyCoins.tsx b/apps/wallet-dashboard/components/coins/MyCoins.tsx index 411d7cdc702..d9d1c0f6f23 100644 --- a/apps/wallet-dashboard/components/coins/MyCoins.tsx +++ b/apps/wallet-dashboard/components/coins/MyCoins.tsx @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React, { useState } from 'react'; +import { useState } from 'react'; import { useCurrentAccount, useIotaClientQuery } from '@iota/dapp-kit'; import { CoinBalance } from '@iota/iota-sdk/client'; import { @@ -18,7 +18,7 @@ import { SegmentedButtonType, Title, } from '@iota/apps-ui-kit'; -import { RecognizedBadge } from '@iota/ui-icons'; +import { RecognizedBadge } from '@iota/apps-ui-icons'; import { SendTokenDialog } from '@/components'; enum TokenCategory { @@ -42,7 +42,7 @@ const TOKEN_CATEGORIES = [ }, ]; -function MyCoins(): React.JSX.Element { +export function MyCoins(): React.JSX.Element { const [selectedTokenCategory, setSelectedTokenCategory] = useState(TokenCategory.All); const [isSendTokenDialogOpen, setIsSendTokenDialogOpen] = useState(false); const [selectedCoin, setSelectedCoin] = useState<CoinBalance>(); @@ -138,5 +138,3 @@ function MyCoins(): React.JSX.Element { </Panel> ); } - -export default MyCoins; diff --git a/apps/wallet-dashboard/components/coins/index.ts b/apps/wallet-dashboard/components/coins/index.ts index 2ff5894ab77..1bdbadc6356 100644 --- a/apps/wallet-dashboard/components/coins/index.ts +++ b/apps/wallet-dashboard/components/coins/index.ts @@ -1,4 +1,4 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export { default as MyCoins } from './MyCoins'; +export * from './MyCoins'; diff --git a/apps/wallet-dashboard/components/Dialogs/ReceiveFundsDialog.tsx b/apps/wallet-dashboard/components/dialogs/ReceiveFundsDialog.tsx similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/ReceiveFundsDialog.tsx rename to apps/wallet-dashboard/components/dialogs/ReceiveFundsDialog.tsx diff --git a/apps/wallet-dashboard/components/Dialogs/TransactionDialog.tsx b/apps/wallet-dashboard/components/dialogs/TransactionDialog.tsx similarity index 94% rename from apps/wallet-dashboard/components/Dialogs/TransactionDialog.tsx rename to apps/wallet-dashboard/components/dialogs/TransactionDialog.tsx index 8b78f53d190..71418f1b673 100644 --- a/apps/wallet-dashboard/components/Dialogs/TransactionDialog.tsx +++ b/apps/wallet-dashboard/components/dialogs/TransactionDialog.tsx @@ -11,7 +11,6 @@ import { ViewTxnOnExplorerButton, } from '@iota/core'; import { useCurrentAccount } from '@iota/dapp-kit'; -import { Validator } from './Staking/views/Validator'; interface SharedProps { txDigest?: string | null; @@ -43,7 +42,6 @@ export function TransactionDialogView({ activeAddress={activeAddress} summary={summary} renderExplorerLink={ExplorerLink} - renderValidatorLogo={Validator} /> ) : ( <div className="flex h-full w-full justify-center"> diff --git a/apps/wallet-dashboard/components/dialogs/assets/AssetDialog.tsx b/apps/wallet-dashboard/components/dialogs/assets/AssetDialog.tsx new file mode 100644 index 00000000000..f2190f2b713 --- /dev/null +++ b/apps/wallet-dashboard/components/dialogs/assets/AssetDialog.tsx @@ -0,0 +1,151 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { useState } from 'react'; +import { Dialog } from '@iota/apps-ui-kit'; +import { FormikProvider, useFormik } from 'formik'; +import { useIotaClient, useCurrentAccount, useSignAndExecuteTransaction } from '@iota/dapp-kit'; +import { + createNftSendValidationSchema, + useTransferAsset, + isKioskOwnerToken, + useKioskClient, + useNftDetails, +} from '@iota/core'; +import { DetailsView, SendView, KioskDetailsView } from './views'; +import { IotaObjectData, IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; +import { AssetsDialogView } from './constants'; +import { TransactionDetailsView } from '../send-token'; +import { DialogLayout } from '../layout'; +import toast from 'react-hot-toast'; + +interface AssetsDialogProps { + onClose: () => void; + asset: IotaObjectData; + refetchAssets: () => void; +} + +interface FormValues { + to: string; +} + +const INITIAL_VALUES: FormValues = { + to: '', +}; + +export function AssetDialog({ onClose, asset, refetchAssets }: AssetsDialogProps): JSX.Element { + const kioskClient = useKioskClient(); + const account = useCurrentAccount(); + const iotaClient = useIotaClient(); + const { mutateAsync: signAndExecuteTransaction } = + useSignAndExecuteTransaction<IotaTransactionBlockResponse>(); + + const isTokenOwnedByKiosk = isKioskOwnerToken(kioskClient.network, asset); + const activeAddress = account?.address ?? ''; + + const initView = isTokenOwnedByKiosk ? AssetsDialogView.KioskDetails : AssetsDialogView.Details; + + const [view, setView] = useState<AssetsDialogView>(initView); + const [chosenKioskAsset, setChoosenKioskAsset] = useState<IotaObjectData | null>(null); + const [digest, setDigest] = useState<string>(''); + + const activeAsset = chosenKioskAsset || asset; + const objectId = chosenKioskAsset ? chosenKioskAsset.objectId : asset ? asset.objectId : ''; + const validationSchema = createNftSendValidationSchema(activeAddress, objectId); + const { objectData } = useNftDetails(objectId, activeAddress); + + const { mutateAsync: sendAsset } = useTransferAsset({ + objectId, + objectType: objectData?.type, + activeAddress: activeAddress, + executeFn: signAndExecuteTransaction, + }); + + const formik = useFormik<FormValues>({ + initialValues: INITIAL_VALUES, + validationSchema: validationSchema, + onSubmit: onSubmit, + validateOnChange: true, + }); + + async function onSubmit(values: FormValues) { + try { + const executed = await sendAsset(values.to); + + const tx = await iotaClient.waitForTransaction({ + digest: executed.digest, + }); + + setDigest(tx.digest); + refetchAssets(); + toast.success('Transfer transaction successful'); + setView(AssetsDialogView.TransactionDetails); + } catch { + toast.error('Transfer transaction failed'); + } + } + + function onDetailsSend() { + setView(AssetsDialogView.Send); + } + + function onSendViewBack() { + setView(AssetsDialogView.Details); + } + function onOpenChange() { + setView(AssetsDialogView.Details); + setChoosenKioskAsset(null); + onClose(); + } + + function onKioskItemClick(item: IotaObjectData) { + setChoosenKioskAsset(item); + setView(AssetsDialogView.Details); + } + + function onBack() { + if (!chosenKioskAsset) { + return; + } + setChoosenKioskAsset(null); + setView(AssetsDialogView.KioskDetails); + } + + return ( + <Dialog open onOpenChange={onOpenChange}> + <DialogLayout> + <> + {view === AssetsDialogView.KioskDetails && ( + <KioskDetailsView + asset={activeAsset} + onClose={onOpenChange} + onItemClick={onKioskItemClick} + /> + )} + {view === AssetsDialogView.Details && ( + <DetailsView + asset={activeAsset} + onClose={onOpenChange} + onSend={onDetailsSend} + onBack={onBack} + /> + )} + {view === AssetsDialogView.Send && ( + <FormikProvider value={formik}> + <SendView + objectId={objectId} + senderAddress={activeAddress} + onClose={onOpenChange} + onBack={onSendViewBack} + /> + </FormikProvider> + )} + + {view === AssetsDialogView.TransactionDetails && !!digest ? ( + <TransactionDetailsView digest={digest} onClose={onOpenChange} /> + ) : null} + </> + </DialogLayout> + </Dialog> + ); +} diff --git a/apps/wallet-dashboard/components/Dialogs/Assets/constants/AssetsDialogView.ts b/apps/wallet-dashboard/components/dialogs/assets/constants/assetsDialogView.ts similarity index 85% rename from apps/wallet-dashboard/components/Dialogs/Assets/constants/AssetsDialogView.ts rename to apps/wallet-dashboard/components/dialogs/assets/constants/assetsDialogView.ts index 242da33c2c3..8b4221c98a7 100644 --- a/apps/wallet-dashboard/components/Dialogs/Assets/constants/AssetsDialogView.ts +++ b/apps/wallet-dashboard/components/dialogs/assets/constants/assetsDialogView.ts @@ -5,4 +5,5 @@ export enum AssetsDialogView { Details = 'Details', Send = 'Send', TransactionDetails = 'TransactionDetails', + KioskDetails = 'KioskDetails', } diff --git a/apps/wallet-dashboard/components/dialogs/assets/constants/index.ts b/apps/wallet-dashboard/components/dialogs/assets/constants/index.ts new file mode 100644 index 00000000000..84f36d56cf4 --- /dev/null +++ b/apps/wallet-dashboard/components/dialogs/assets/constants/index.ts @@ -0,0 +1,4 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export * from './assetsDialogView'; diff --git a/apps/wallet-dashboard/components/Dialogs/Assets/index.ts b/apps/wallet-dashboard/components/dialogs/assets/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/Assets/index.ts rename to apps/wallet-dashboard/components/dialogs/assets/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/Assets/views/DetailsView.tsx b/apps/wallet-dashboard/components/dialogs/assets/views/DetailsView.tsx similarity index 79% rename from apps/wallet-dashboard/components/Dialogs/Assets/views/DetailsView.tsx rename to apps/wallet-dashboard/components/dialogs/assets/views/DetailsView.tsx index 8520d62a484..9c4575681dc 100644 --- a/apps/wallet-dashboard/components/Dialogs/Assets/views/DetailsView.tsx +++ b/apps/wallet-dashboard/components/dialogs/assets/views/DetailsView.tsx @@ -1,17 +1,14 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; -import { ExplorerLinkType, useNftDetails, Collapsible, useNFTBasicData } from '@iota/core'; import { - Button, - ButtonType, - Header, - KeyValueInfo, - VisualAssetCard, - VisualAssetType, -} from '@iota/apps-ui-kit'; -import Link from 'next/link'; + ExplorerLinkType, + useNftDetails, + Collapsible, + useNFTBasicData, + NftImage, +} from '@iota/core'; +import { Button, ButtonType, Header, KeyValueInfo } from '@iota/apps-ui-kit'; import { formatAddress } from '@iota/iota-sdk/utils'; import { DialogLayoutBody, DialogLayoutFooter } from '../../layout'; import { IotaObjectData } from '@iota/iota-sdk/client'; @@ -22,9 +19,10 @@ interface DetailsViewProps { asset: IotaObjectData; onClose: () => void; onSend: () => void; + onBack?: () => void; } -export function DetailsView({ onClose, asset, onSend }: DetailsViewProps) { +export function DetailsView({ onClose, asset, onSend, onBack }: DetailsViewProps) { const account = useCurrentAccount(); const senderAddress = account?.address ?? ''; @@ -38,7 +36,6 @@ export function DetailsView({ onClose, asset, onSend }: DetailsViewProps) { isAssetTransferable, metaKeys, metaValues, - formatMetaValue, isContainedInKiosk, kioskItem, objectData, @@ -46,27 +43,21 @@ export function DetailsView({ onClose, asset, onSend }: DetailsViewProps) { const { fileExtensionType, filePath } = useNFTBasicData(objectData); function handleMoreAboutKiosk() { - window.open('https://docs.iota.org/references/ts-sdk/kiosk/', '_blank'); + window.open('https://docs.iota.org/ts-sdk/kiosk/', '_blank'); } function handleMarketplace() { // TODO: https://github.com/iotaledger/iota/issues/4024 - window.open('https://docs.iota.org/references/ts-sdk/kiosk/', '_blank'); + window.open('https://docs.iota.org/ts-sdk/kiosk/', '_blank'); } return ( <> - <Header title="Asset" onClose={onClose} titleCentered /> + <Header title="Asset" onClose={onClose} titleCentered onBack={onBack} /> <DialogLayoutBody> <div className="flex w-full flex-col items-center justify-center gap-xs"> <div className="w-[172px]"> - <VisualAssetCard - assetSrc={nftImageUrl} - assetTitle={nftName} - assetType={VisualAssetType.Image} - altText={nftName || 'NFT'} - isHoverable={false} - /> + <NftImage src={nftImageUrl} title={nftName || 'NFT'} isHoverable={false} /> </div> <ExplorerLink type={ExplorerLinkType.Object} objectID={objectId}> <Button type={ButtonType.Ghost} text="View on Explorer" /> @@ -88,11 +79,7 @@ export function DetailsView({ onClose, asset, onSend }: DetailsViewProps) { {nftDisplayData?.projectUrl && ( <KeyValueInfo keyText="Website" - value={ - <Link href={nftDisplayData?.projectUrl}> - {nftDisplayData?.projectUrl} - </Link> - } + value={nftDisplayData?.projectUrl} fullwidth /> )} @@ -144,21 +131,14 @@ export function DetailsView({ onClose, asset, onSend }: DetailsViewProps) { <Collapsible defaultOpen title="Attributes"> <div className="flex flex-col gap-xs px-md pb-xs pt-sm"> {metaKeys.map((aKey, idx) => { - const { value, valueLink } = formatMetaValue( - metaValues[idx], - ); return ( <KeyValueInfo key={idx} keyText={aKey} value={ - valueLink ? ( - <Link key={aKey} href={valueLink || ''}> - {value} - </Link> - ) : ( - value - ) + typeof metaValues[idx] === 'object' + ? JSON.stringify(metaValues[idx]) + : metaValues[idx] } fullwidth /> diff --git a/apps/wallet-dashboard/components/dialogs/assets/views/KioskDetailsView.tsx b/apps/wallet-dashboard/components/dialogs/assets/views/KioskDetailsView.tsx new file mode 100644 index 00000000000..c259b170a39 --- /dev/null +++ b/apps/wallet-dashboard/components/dialogs/assets/views/KioskDetailsView.tsx @@ -0,0 +1,85 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { + useGetKioskContents, + getKioskIdFromOwnerCap, + useNftDetails, + NftImage, + ExplorerLinkType, + ViewTxnOnExplorerButton, +} from '@iota/core'; +import { Badge, BadgeType, Header, LoadingIndicator } from '@iota/apps-ui-kit'; +import { DialogLayoutBody, DialogLayoutFooter } from '../../layout'; +import { IotaObjectData } from '@iota/iota-sdk/client'; +import { useCurrentAccount } from '@iota/dapp-kit'; +import { ExplorerLink } from '@/components/ExplorerLink'; + +interface DetailsViewProps { + asset: IotaObjectData; + onClose: () => void; + onItemClick: (asset: IotaObjectData) => void; +} + +export function KioskDetailsView({ onClose, asset, onItemClick }: DetailsViewProps) { + const account = useCurrentAccount(); + const senderAddress = account?.address ?? ''; + const objectId = getKioskIdFromOwnerCap(asset); + const { data: kioskData, isPending } = useGetKioskContents(account?.address); + const kiosk = kioskData?.kiosks.get(objectId); + const items = kiosk?.items; + + if (isPending) { + return ( + <div className="flex h-full items-center justify-center"> + <LoadingIndicator /> + </div> + ); + } + + return ( + <> + <Header title="Kiosk" onClose={onClose} titleCentered /> + <DialogLayoutBody> + <div className="flex flex-col gap-md"> + <div className="flex flex-row gap-x-sm"> + <span className="text-title-lg text-neutral-10 dark:text-neutral-92"> + Kiosk items + </span> + <Badge type={BadgeType.Neutral} label={items?.length.toString() ?? '0'} /> + </div> + <div className="grid grid-cols-3 items-center justify-center gap-sm"> + {items?.map((item) => { + return item.data?.objectId ? ( + <div + onClick={() => { + item.data && onItemClick(item.data); + }} + key={item.data?.objectId} + > + <KioskItem object={item.data} address={senderAddress} /> + </div> + ) : null; + })} + </div> + </div> + </DialogLayoutBody> + <DialogLayoutFooter> + <ExplorerLink objectID={objectId} type={ExplorerLinkType.Object}> + <ViewTxnOnExplorerButton digest={objectId} /> + </ExplorerLink> + </DialogLayoutFooter> + </> + ); +} + +interface KioskItemProps { + object: IotaObjectData; + address: string; +} + +function KioskItem({ object, address }: KioskItemProps) { + const { nftName, nftImageUrl } = useNftDetails(object.objectId, address); + + return <NftImage title={nftName} src={nftImageUrl} isHoverable />; +} diff --git a/apps/wallet-dashboard/components/Dialogs/Assets/views/SendView.tsx b/apps/wallet-dashboard/components/dialogs/assets/views/SendView.tsx similarity index 63% rename from apps/wallet-dashboard/components/Dialogs/Assets/views/SendView.tsx rename to apps/wallet-dashboard/components/dialogs/assets/views/SendView.tsx index 3d0350478ac..f73f850a118 100644 --- a/apps/wallet-dashboard/components/Dialogs/Assets/views/SendView.tsx +++ b/apps/wallet-dashboard/components/dialogs/assets/views/SendView.tsx @@ -1,50 +1,30 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; -import { AddressInput, useNftDetails } from '@iota/core'; +import { AddressInput, NftImage, useNftDetails } from '@iota/core'; import { useFormikContext } from 'formik'; import { DialogLayoutFooter, DialogLayoutBody } from '../../layout'; -import { - Button, - ButtonHtmlType, - Header, - VisualAssetCard, - VisualAssetType, - Title, -} from '@iota/apps-ui-kit'; -import { Loader } from '@iota/ui-icons'; -import { useCurrentAccount } from '@iota/dapp-kit'; -import { IotaObjectData } from '@iota/iota-sdk/client'; +import { Button, ButtonHtmlType, Header, Title } from '@iota/apps-ui-kit'; +import { Loader } from '@iota/apps-ui-icons'; interface SendViewProps { - asset: IotaObjectData; + objectId: string; + senderAddress: string; onClose: () => void; onBack: () => void; } -export function SendView({ asset, onClose, onBack }: SendViewProps) { +export function SendView({ objectId, senderAddress, onClose, onBack }: SendViewProps) { const { isValid, dirty, isSubmitting, submitForm } = useFormikContext(); - - const account = useCurrentAccount(); - - const senderAddress = account?.address ?? ''; - const objectId = asset?.objectId || ''; - const { nftName, nftImageUrl } = useNftDetails(objectId, senderAddress); + return ( <> <Header title="Send asset" onClose={onClose} titleCentered onBack={onBack} /> <DialogLayoutBody> <div className="flex w-full flex-col items-center justify-center gap-xs"> <div className="w-[172px]"> - <VisualAssetCard - assetSrc={nftImageUrl} - assetTitle={nftName} - assetType={VisualAssetType.Image} - altText={nftName || 'NFT'} - isHoverable={false} - /> + <NftImage src={nftImageUrl} title={nftName || 'NFT'} isHoverable={false} /> </div> <div className="flex w-full flex-col gap-md"> <div className="flex flex-col items-center gap-xxxs"> diff --git a/apps/wallet-dashboard/components/Dialogs/Assets/views/index.ts b/apps/wallet-dashboard/components/dialogs/assets/views/index.ts similarity index 78% rename from apps/wallet-dashboard/components/Dialogs/Assets/views/index.ts rename to apps/wallet-dashboard/components/dialogs/assets/views/index.ts index fcc3856cac3..20ad446cf93 100644 --- a/apps/wallet-dashboard/components/Dialogs/Assets/views/index.ts +++ b/apps/wallet-dashboard/components/dialogs/assets/views/index.ts @@ -3,3 +3,4 @@ export * from './DetailsView'; export * from './SendView'; +export * from './KioskDetailsView'; diff --git a/apps/wallet-dashboard/components/Dialogs/index.ts b/apps/wallet-dashboard/components/dialogs/index.ts similarity index 68% rename from apps/wallet-dashboard/components/Dialogs/index.ts rename to apps/wallet-dashboard/components/dialogs/index.ts index db380d17927..88c9e90c332 100644 --- a/apps/wallet-dashboard/components/Dialogs/index.ts +++ b/apps/wallet-dashboard/components/dialogs/index.ts @@ -1,10 +1,10 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export * from './SendToken'; +export * from './send-token'; export * from './ReceiveFundsDialog'; -export * from './Staking'; +export * from './staking'; export * from './unstake'; export * from './vesting'; export * from './settings'; -export * from './MigrationDialog'; +export * from './migration'; diff --git a/apps/wallet-dashboard/components/Dialogs/layout/DialogLayout.tsx b/apps/wallet-dashboard/components/dialogs/layout/DialogLayout.tsx similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/layout/DialogLayout.tsx rename to apps/wallet-dashboard/components/dialogs/layout/DialogLayout.tsx diff --git a/apps/wallet-dashboard/components/Dialogs/layout/index.ts b/apps/wallet-dashboard/components/dialogs/layout/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/layout/index.ts rename to apps/wallet-dashboard/components/dialogs/layout/index.ts diff --git a/apps/wallet-dashboard/components/dialogs/migration/MigrationDialog.tsx b/apps/wallet-dashboard/components/dialogs/migration/MigrationDialog.tsx new file mode 100644 index 00000000000..e3e9cd19541 --- /dev/null +++ b/apps/wallet-dashboard/components/dialogs/migration/MigrationDialog.tsx @@ -0,0 +1,88 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { useState } from 'react'; +import { useCurrentAccount, useSignAndExecuteTransaction } from '@iota/dapp-kit'; +import { IotaObjectData } from '@iota/iota-sdk/client'; +import { useMigrationTransaction } from '@/hooks/useMigrationTransaction'; +import { Dialog } from '@iota/apps-ui-kit'; +import toast from 'react-hot-toast'; +import { TransactionDialogView } from '../TransactionDialog'; +import { MigrationDialogView } from './enums'; +import { ConfirmMigrationView } from './views'; + +interface MigrationDialogProps { + handleClose: () => void; + basicOutputObjects: IotaObjectData[] | undefined; + nftOutputObjects: IotaObjectData[] | undefined; + onSuccess: (digest: string) => void; + setOpen: (bool: boolean) => void; + open: boolean; + isTimelocked: boolean; +} + +export function MigrationDialog({ + handleClose, + basicOutputObjects = [], + nftOutputObjects = [], + onSuccess, + open, + setOpen, + isTimelocked, +}: MigrationDialogProps): JSX.Element { + const account = useCurrentAccount(); + const [txDigest, setTxDigest] = useState<string>(''); + const [view, setView] = useState<MigrationDialogView>(MigrationDialogView.Confirmation); + + const { + data: migrateData, + isPending: isMigrationPending, + isError: isMigrationError, + } = useMigrationTransaction(account?.address || '', basicOutputObjects, nftOutputObjects); + + const { mutateAsync: signAndExecuteTransaction, isPending: isSendingTransaction } = + useSignAndExecuteTransaction(); + + async function handleMigrate(): Promise<void> { + if (!migrateData) return; + signAndExecuteTransaction( + { + transaction: migrateData.transaction, + }, + { + onSuccess: (tx) => { + onSuccess(tx.digest); + setTxDigest(tx.digest); + setView(MigrationDialogView.TransactionDetails); + }, + }, + ) + .then(() => { + toast.success('Migration transaction has been sent'); + }) + .catch(() => { + toast.error('Migration transaction was not sent'); + }); + } + + return ( + <Dialog open={open} onOpenChange={setOpen}> + {view === MigrationDialogView.Confirmation && ( + <ConfirmMigrationView + basicOutputObjects={basicOutputObjects} + nftOutputObjects={nftOutputObjects} + onSuccess={handleMigrate} + setOpen={setOpen} + groupByTimelockUC={isTimelocked} + migrateData={migrateData} + isMigrationPending={isMigrationPending} + isMigrationError={isMigrationError} + isSendingTransaction={isSendingTransaction} + /> + )} + {view === MigrationDialogView.TransactionDetails && ( + <TransactionDialogView txDigest={txDigest} onClose={handleClose} /> + )} + </Dialog> + ); +} diff --git a/apps/wallet-dashboard/components/Dialogs/Staking/enums/index.ts b/apps/wallet-dashboard/components/dialogs/migration/enums/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/Staking/enums/index.ts rename to apps/wallet-dashboard/components/dialogs/migration/enums/index.ts diff --git a/apps/wallet-dashboard/components/dialogs/migration/enums/view.enums.ts b/apps/wallet-dashboard/components/dialogs/migration/enums/view.enums.ts new file mode 100644 index 00000000000..5b16d31b836 --- /dev/null +++ b/apps/wallet-dashboard/components/dialogs/migration/enums/view.enums.ts @@ -0,0 +1,7 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export enum MigrationDialogView { + Confirmation = 'Confirmation', + TransactionDetails = 'TransactionDetails', +} diff --git a/apps/wallet-dashboard/components/dialogs/migration/index.ts b/apps/wallet-dashboard/components/dialogs/migration/index.ts new file mode 100644 index 00000000000..41dd3ff2b30 --- /dev/null +++ b/apps/wallet-dashboard/components/dialogs/migration/index.ts @@ -0,0 +1,6 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export * from './MigrationDialog'; + +export * from './views'; diff --git a/apps/wallet-dashboard/components/dialogs/migration/views/ConfirmMigrationView.tsx b/apps/wallet-dashboard/components/dialogs/migration/views/ConfirmMigrationView.tsx new file mode 100644 index 00000000000..6e95aa85d46 --- /dev/null +++ b/apps/wallet-dashboard/components/dialogs/migration/views/ConfirmMigrationView.tsx @@ -0,0 +1,215 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { MigrationObjectLoading, VirtualList, MigrationObjectDetailsCard } from '@/components'; +import { useCurrentAccount } from '@iota/dapp-kit'; +import { IotaObjectData } from '@iota/iota-sdk/client'; +import { + Button, + Header, + InfoBox, + InfoBoxStyle, + InfoBoxType, + KeyValueInfo, + Panel, + Skeleton, + Title, + TitleSize, +} from '@iota/apps-ui-kit'; +import { useGroupedStardustObjects } from '@/hooks'; +import { Loader, Warning } from '@iota/apps-ui-icons'; +import { Collapsible, useFormatCoin } from '@iota/core'; +import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; +import { getStardustObjectsTotals, filterMigrationObjects } from '@/lib/utils'; +import { DialogLayout, DialogLayoutBody, DialogLayoutFooter } from '../../layout'; +import { Transaction } from '@iota/iota-sdk/transactions'; +import { StardustOutputDetailsFilter } from '@/lib/enums'; + +interface ConfirmMigrationViewProps { + basicOutputObjects: IotaObjectData[] | undefined; + nftOutputObjects: IotaObjectData[] | undefined; + onSuccess: () => void; + setOpen: (bool: boolean) => void; + groupByTimelockUC: boolean; + migrateData: + | { + transaction: Transaction; + gasBudget: string | number | null; + } + | undefined; + isMigrationPending: boolean; + isMigrationError: boolean; + isSendingTransaction: boolean; +} + +export function ConfirmMigrationView({ + basicOutputObjects = [], + nftOutputObjects = [], + onSuccess, + setOpen, + groupByTimelockUC, + migrateData, + isMigrationPending, + isMigrationError, + isSendingTransaction, +}: ConfirmMigrationViewProps): JSX.Element { + const account = useCurrentAccount(); + + const { + data: resolvedObjects = [], + isLoading, + error: isGroupedMigrationError, + } = useGroupedStardustObjects([...basicOutputObjects, ...nftOutputObjects], groupByTimelockUC); + + const { + totalIotaAmount, + totalNativeTokens: migratableNativeTokens, + totalVisualAssets: migratableVisualAssets, + totalNotOwnedStorageDepositReturnAmount, + } = getStardustObjectsTotals({ + basicOutputs: basicOutputObjects, + nftOutputs: nftOutputObjects, + address: account?.address || '', + resolvedObjects: resolvedObjects, + }); + + const [timelockedIotaTokens, symbol] = useFormatCoin(totalIotaAmount, IOTA_TYPE_ARG); + const [gasFee, gasFeeSymbol] = useFormatCoin(migrateData?.gasBudget, IOTA_TYPE_ARG); + const [totalStorageDepositReturnAmountFormatted, totalStorageDepositReturnAmountSymbol] = + useFormatCoin(totalNotOwnedStorageDepositReturnAmount.toString(), IOTA_TYPE_ARG); + + const filteredIotaObjects = filterMigrationObjects( + resolvedObjects, + StardustOutputDetailsFilter.IOTA, + ); + const filteredNativeTokens = filterMigrationObjects( + resolvedObjects, + StardustOutputDetailsFilter.NativeTokens, + ); + const filteredVisualAssets = filterMigrationObjects( + resolvedObjects, + StardustOutputDetailsFilter.VisualAssets, + ); + + const assetsToMigrateCategories = [ + { + title: 'IOTA Tokens', + subtitle: `${timelockedIotaTokens} ${symbol}`, + filteredObjects: filteredIotaObjects, + }, + { + title: 'Native Tokens', + subtitle: `${migratableNativeTokens} Types`, + filteredObjects: filteredNativeTokens, + }, + { + title: 'Visual Assets', + subtitle: `${migratableVisualAssets} Assets`, + filteredObjects: filteredVisualAssets, + }, + ]; + const filteredAssetsToMigrateCategories = assetsToMigrateCategories.filter( + ({ filteredObjects }) => filteredObjects.length > 0, + ); + return ( + <DialogLayout> + <Header title="Migrate Your Assets" onClose={() => setOpen(false)} titleCentered /> + <DialogLayoutBody> + <div className="flex h-full flex-col gap-y-md"> + {isGroupedMigrationError && !isLoading && ( + <InfoBox + title="Error" + supportingText="Failed to load migration objects" + style={InfoBoxStyle.Elevated} + type={InfoBoxType.Error} + icon={<Warning />} + /> + )} + {isLoading ? ( + <> + <Panel hasBorder> + <div className="flex flex-col gap-y-sm p-md"> + <Skeleton widthClass="w-40" heightClass="h-3.5" /> + <MigrationObjectLoading /> + </div> + </Panel> + <Panel hasBorder> + <div className="flex flex-col gap-y-md p-md"> + <Skeleton widthClass="w-full" heightClass="h-3.5" /> + <Skeleton widthClass="w-full" heightClass="h-3.5" /> + </div> + </Panel> + </> + ) : ( + <> + <div className="flex flex-col gap-y-sm"> + {filteredAssetsToMigrateCategories.map( + ({ title, subtitle, filteredObjects }) => ( + <Collapsible + key={title} + render={() => ( + <Title + size={TitleSize.Small} + title={title} + subtitle={subtitle} + /> + )} + > + <div className="flex h-full max-h-[300px] flex-col gap-y-sm pb-sm"> + <VirtualList + heightClassName="h-full" + overflowClassName="overflow-y-auto" + items={filteredObjects} + estimateSize={() => 58} + render={(migrationObject) => ( + <MigrationObjectDetailsCard + migrationObject={migrationObject} + isTimelocked={groupByTimelockUC} + /> + )} + /> + </div> + </Collapsible> + ), + )} + </div> + <Panel hasBorder> + <div className="flex flex-col gap-y-sm p-md"> + <KeyValueInfo + keyText="Legacy storage deposit" + value={totalStorageDepositReturnAmountFormatted || '-'} + supportingLabel={totalStorageDepositReturnAmountSymbol} + fullwidth + /> + <KeyValueInfo + keyText="Gas Fees" + value={gasFee || '-'} + supportingLabel={gasFeeSymbol} + fullwidth + /> + </div> + </Panel> + </> + )} + </div> + </DialogLayoutBody> + <DialogLayoutFooter> + <Button + text="Migrate" + disabled={isMigrationPending || isMigrationError || isSendingTransaction} + onClick={onSuccess} + icon={ + isMigrationPending || isSendingTransaction ? ( + <Loader + className="h-4 w-4 animate-spin" + data-testid="loading-indicator" + /> + ) : null + } + iconAfterText + fullWidth + /> + </DialogLayoutFooter> + </DialogLayout> + ); +} diff --git a/apps/wallet-dashboard/components/dialogs/migration/views/index.ts b/apps/wallet-dashboard/components/dialogs/migration/views/index.ts new file mode 100644 index 00000000000..b5a03528f1e --- /dev/null +++ b/apps/wallet-dashboard/components/dialogs/migration/views/index.ts @@ -0,0 +1,4 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export * from './ConfirmMigrationView'; diff --git a/apps/wallet-dashboard/components/Dialogs/SendToken/SendTokenDialog.tsx b/apps/wallet-dashboard/components/dialogs/send-token/SendTokenDialog.tsx similarity index 89% rename from apps/wallet-dashboard/components/Dialogs/SendToken/SendTokenDialog.tsx rename to apps/wallet-dashboard/components/dialogs/send-token/SendTokenDialog.tsx index a33f018a09c..217540c0fa6 100644 --- a/apps/wallet-dashboard/components/Dialogs/SendToken/SendTokenDialog.tsx +++ b/apps/wallet-dashboard/components/dialogs/send-token/SendTokenDialog.tsx @@ -1,17 +1,18 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React, { useState } from 'react'; +import { useState } from 'react'; import { EnterValuesFormView, ReviewValuesFormView, TransactionDetailsView } from './views'; import { CoinBalance } from '@iota/iota-sdk/client'; -import { useSendCoinTransaction, useNotifications } from '@/hooks'; -import { NotificationType } from '@/stores/notificationStore'; +import { useSendCoinTransaction } from '@/hooks'; import { CoinFormat, useFormatCoin, useGetAllCoins } from '@iota/core'; import { Dialog, DialogContent, DialogPosition } from '@iota/apps-ui-kit'; import { FormDataValues } from './interfaces'; import { INITIAL_VALUES } from './constants'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { useTransferTransactionMutation } from '@/hooks'; +import toast from 'react-hot-toast'; +import { useQueryClient } from '@tanstack/react-query'; interface SendCoinDialogProps { coin: CoinBalance; @@ -36,8 +37,8 @@ function SendTokenDialogBody({ const [formData, setFormData] = useState<FormDataValues>(INITIAL_VALUES); const [fullAmount] = useFormatCoin(formData.amount, selectedCoin.coinType, CoinFormat.FULL); const { data: coinsData } = useGetAllCoins(selectedCoin.coinType, activeAddress); + const queryClient = useQueryClient(); - const { addNotification } = useNotifications(); const isPayAllIota = selectedCoin.totalBalance === formData.amount && selectedCoin.coinType === IOTA_TYPE_ARG; @@ -58,18 +59,20 @@ function SendTokenDialogBody({ async function handleTransfer() { if (!transaction) { - addNotification('There was an error with the transaction', NotificationType.Error); + toast.error('There was an error with the transaction'); return; } transfer(transaction, { onSuccess: () => { + queryClient.invalidateQueries({ queryKey: [activeAddress] }); + setStep(FormStep.TransactionDetails); - addNotification('Transfer transaction has been sent', NotificationType.Success); + toast.success('Transfer transaction has been sent'); }, onError: () => { setOpen(false); - addNotification('Transfer transaction failed', NotificationType.Error); + toast.error('Transfer transaction failed'); }, }); } diff --git a/apps/wallet-dashboard/components/Dialogs/SendToken/constants/index.ts b/apps/wallet-dashboard/components/dialogs/send-token/constants/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/SendToken/constants/index.ts rename to apps/wallet-dashboard/components/dialogs/send-token/constants/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/SendToken/index.ts b/apps/wallet-dashboard/components/dialogs/send-token/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/SendToken/index.ts rename to apps/wallet-dashboard/components/dialogs/send-token/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/SendToken/interfaces/index.ts b/apps/wallet-dashboard/components/dialogs/send-token/interfaces/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/SendToken/interfaces/index.ts rename to apps/wallet-dashboard/components/dialogs/send-token/interfaces/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/SendToken/views/EnterValuesFormView.tsx b/apps/wallet-dashboard/components/dialogs/send-token/views/EnterValuesFormView.tsx similarity index 99% rename from apps/wallet-dashboard/components/Dialogs/SendToken/views/EnterValuesFormView.tsx rename to apps/wallet-dashboard/components/dialogs/send-token/views/EnterValuesFormView.tsx index f33ddbc8fef..93c3b3711d5 100644 --- a/apps/wallet-dashboard/components/Dialogs/SendToken/views/EnterValuesFormView.tsx +++ b/apps/wallet-dashboard/components/dialogs/send-token/views/EnterValuesFormView.tsx @@ -29,7 +29,7 @@ import { import { useIotaClientQuery } from '@iota/dapp-kit'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { Form, FormikProvider, useFormik, useFormikContext } from 'formik'; -import { Exclamation } from '@iota/ui-icons'; +import { Exclamation } from '@iota/apps-ui-icons'; import { UseQueryResult } from '@tanstack/react-query'; import { FormDataValues } from '../interfaces'; import { INITIAL_VALUES } from '../constants'; diff --git a/apps/wallet-dashboard/components/Dialogs/SendToken/views/ReviewValuesFormView.tsx b/apps/wallet-dashboard/components/dialogs/send-token/views/ReviewValuesFormView.tsx similarity index 98% rename from apps/wallet-dashboard/components/Dialogs/SendToken/views/ReviewValuesFormView.tsx rename to apps/wallet-dashboard/components/dialogs/send-token/views/ReviewValuesFormView.tsx index d4d82d1ccad..874a078cd82 100644 --- a/apps/wallet-dashboard/components/Dialogs/SendToken/views/ReviewValuesFormView.tsx +++ b/apps/wallet-dashboard/components/dialogs/send-token/views/ReviewValuesFormView.tsx @@ -20,7 +20,7 @@ import { } from '@iota/apps-ui-kit'; import { formatAddress, IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { CoinIcon, ImageIconSize, useFormatCoin, ExplorerLinkType, CoinFormat } from '@iota/core'; -import { Loader } from '@iota/ui-icons'; +import { Loader } from '@iota/apps-ui-icons'; import { ExplorerLink } from '@/components'; import { DialogLayoutBody, DialogLayoutFooter } from '../../layout'; diff --git a/apps/wallet-dashboard/components/Dialogs/SendToken/views/TransactionDetailsView.tsx b/apps/wallet-dashboard/components/dialogs/send-token/views/TransactionDetailsView.tsx similarity index 96% rename from apps/wallet-dashboard/components/Dialogs/SendToken/views/TransactionDetailsView.tsx rename to apps/wallet-dashboard/components/dialogs/send-token/views/TransactionDetailsView.tsx index f534f72e387..430bb921c00 100644 --- a/apps/wallet-dashboard/components/Dialogs/SendToken/views/TransactionDetailsView.tsx +++ b/apps/wallet-dashboard/components/dialogs/send-token/views/TransactionDetailsView.tsx @@ -3,7 +3,7 @@ import { useGetTransaction } from '@iota/core'; import { InfoBoxType, InfoBox, InfoBoxStyle } from '@iota/apps-ui-kit'; -import { Warning, Loader } from '@iota/ui-icons'; +import { Warning, Loader } from '@iota/apps-ui-icons'; import { getExtendedTransaction } from '@/lib/utils'; import { useCurrentAccount } from '@iota/dapp-kit'; import { TransactionDetailsLayout } from '../../transaction'; diff --git a/apps/wallet-dashboard/components/Dialogs/SendToken/views/index.ts b/apps/wallet-dashboard/components/dialogs/send-token/views/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/SendToken/views/index.ts rename to apps/wallet-dashboard/components/dialogs/send-token/views/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/settings/SettingsDialog.tsx b/apps/wallet-dashboard/components/dialogs/settings/SettingsDialog.tsx similarity index 97% rename from apps/wallet-dashboard/components/Dialogs/settings/SettingsDialog.tsx rename to apps/wallet-dashboard/components/dialogs/settings/SettingsDialog.tsx index 447c3e3a4e9..75c40efaf9d 100644 --- a/apps/wallet-dashboard/components/Dialogs/settings/SettingsDialog.tsx +++ b/apps/wallet-dashboard/components/dialogs/settings/SettingsDialog.tsx @@ -1,7 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import { Dialog } from '@iota/apps-ui-kit'; import { SettingsDialogView } from './enums'; import { SettingsListView, NetworkSelectorView } from './views'; diff --git a/apps/wallet-dashboard/components/Dialogs/settings/enums/index.ts b/apps/wallet-dashboard/components/dialogs/settings/enums/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/settings/enums/index.ts rename to apps/wallet-dashboard/components/dialogs/settings/enums/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/settings/enums/view.enums.ts b/apps/wallet-dashboard/components/dialogs/settings/enums/view.enums.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/settings/enums/view.enums.ts rename to apps/wallet-dashboard/components/dialogs/settings/enums/view.enums.ts diff --git a/apps/wallet-dashboard/components/Dialogs/settings/hooks/index.ts b/apps/wallet-dashboard/components/dialogs/settings/hooks/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/settings/hooks/index.ts rename to apps/wallet-dashboard/components/dialogs/settings/hooks/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/settings/hooks/useSettingsDialog.ts b/apps/wallet-dashboard/components/dialogs/settings/hooks/useSettingsDialog.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/settings/hooks/useSettingsDialog.ts rename to apps/wallet-dashboard/components/dialogs/settings/hooks/useSettingsDialog.ts diff --git a/apps/wallet-dashboard/components/Dialogs/settings/index.ts b/apps/wallet-dashboard/components/dialogs/settings/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/settings/index.ts rename to apps/wallet-dashboard/components/dialogs/settings/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/settings/views/NetworkSelectorView.tsx b/apps/wallet-dashboard/components/dialogs/settings/views/NetworkSelectorView.tsx similarity index 75% rename from apps/wallet-dashboard/components/Dialogs/settings/views/NetworkSelectorView.tsx rename to apps/wallet-dashboard/components/dialogs/settings/views/NetworkSelectorView.tsx index 9eb2d3a80d4..f3336f8fb01 100644 --- a/apps/wallet-dashboard/components/Dialogs/settings/views/NetworkSelectorView.tsx +++ b/apps/wallet-dashboard/components/dialogs/settings/views/NetworkSelectorView.tsx @@ -1,12 +1,11 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import { Header, RadioButton } from '@iota/apps-ui-kit'; import { DialogLayout, DialogLayoutBody } from '../../layout'; import { NetworkConfiguration } from '@iota/iota-sdk/client'; import { useIotaClientContext } from '@iota/dapp-kit'; -import toast from 'react-hot-toast'; +import { usePersistedNetwork } from '@/hooks'; interface NetworkSelectorViewProps { handleClose: () => void; @@ -18,17 +17,10 @@ export function NetworkSelectorView({ onBack, }: NetworkSelectorViewProps): JSX.Element { const clientContext = useIotaClientContext(); - const activeNetwork = clientContext.network; // Dashboard doesn't support RPCs yet const networks = clientContext.networks as Record<string, NetworkConfiguration>; - async function handleNetworkChange(network: NetworkConfiguration) { - if (activeNetwork === network.id) { - return; - } - clientContext.selectNetwork(network.id); - toast.success(`Switched to ${network.name}`); - } + const { persistedNetwork, handleNetworkChange } = usePersistedNetwork(); return ( <DialogLayout> @@ -39,7 +31,7 @@ export function NetworkSelectorView({ <div className="px-md" key={network.id}> <RadioButton label={network.name} - isChecked={activeNetwork === network.id} + isChecked={persistedNetwork === network.id} onChange={() => handleNetworkChange(network)} /> </div> diff --git a/apps/wallet-dashboard/components/Dialogs/settings/views/SettingsListView.tsx b/apps/wallet-dashboard/components/dialogs/settings/views/SettingsListView.tsx similarity index 86% rename from apps/wallet-dashboard/components/Dialogs/settings/views/SettingsListView.tsx rename to apps/wallet-dashboard/components/dialogs/settings/views/SettingsListView.tsx index 7e9d484de73..0621134fe17 100644 --- a/apps/wallet-dashboard/components/Dialogs/settings/views/SettingsListView.tsx +++ b/apps/wallet-dashboard/components/dialogs/settings/views/SettingsListView.tsx @@ -1,7 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import { Card, CardAction, @@ -14,9 +13,9 @@ import { } from '@iota/apps-ui-kit'; import { DialogLayout, DialogLayoutBody } from '../../layout'; import { SettingsDialogView } from '../enums'; -import { getNetwork } from '@iota/iota-sdk/client'; -import { useIotaClientContext } from '@iota/dapp-kit'; -import { Globe } from '@iota/ui-icons'; +import { Globe } from '@iota/apps-ui-icons'; +import { usePersistedNetwork } from '@/hooks'; +import { toTitleCase } from '@iota/core'; interface SettingsListViewProps { handleClose: () => void; @@ -24,13 +23,11 @@ interface SettingsListViewProps { } export function SettingsListView({ handleClose, setView }: SettingsListViewProps): JSX.Element { - const { network } = useIotaClientContext(); - const { name: networkName } = getNetwork(network); - + const { persistedNetwork } = usePersistedNetwork(); const MENU_ITEMS = [ { title: 'Network', - subtitle: networkName, + subtitle: toTitleCase(persistedNetwork), icon: <Globe />, onClick: () => setView(SettingsDialogView.NetworkSettings), }, diff --git a/apps/wallet-dashboard/components/Dialogs/settings/views/index.ts b/apps/wallet-dashboard/components/dialogs/settings/views/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/settings/views/index.ts rename to apps/wallet-dashboard/components/dialogs/settings/views/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/Staking/StakeDialog.tsx b/apps/wallet-dashboard/components/dialogs/staking/StakeDialog.tsx similarity index 99% rename from apps/wallet-dashboard/components/Dialogs/Staking/StakeDialog.tsx rename to apps/wallet-dashboard/components/dialogs/staking/StakeDialog.tsx index 6829097f6c3..66ded41a559 100644 --- a/apps/wallet-dashboard/components/Dialogs/Staking/StakeDialog.tsx +++ b/apps/wallet-dashboard/components/dialogs/staking/StakeDialog.tsx @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React, { useMemo, useState } from 'react'; +import { useMemo, useState } from 'react'; import { EnterAmountView, EnterTimelockedAmountView, SelectValidatorView } from './views'; import { ExtendedDelegatedStake, diff --git a/apps/wallet-dashboard/components/dialogs/staking/enums/index.ts b/apps/wallet-dashboard/components/dialogs/staking/enums/index.ts new file mode 100644 index 00000000000..6f408e39b8c --- /dev/null +++ b/apps/wallet-dashboard/components/dialogs/staking/enums/index.ts @@ -0,0 +1,4 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export * from './view.enums'; diff --git a/apps/wallet-dashboard/components/Dialogs/Staking/enums/view.enums.ts b/apps/wallet-dashboard/components/dialogs/staking/enums/view.enums.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/Staking/enums/view.enums.ts rename to apps/wallet-dashboard/components/dialogs/staking/enums/view.enums.ts diff --git a/apps/wallet-dashboard/components/Dialogs/Staking/hooks/index.ts b/apps/wallet-dashboard/components/dialogs/staking/hooks/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/Staking/hooks/index.ts rename to apps/wallet-dashboard/components/dialogs/staking/hooks/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/Staking/hooks/useStakeDialog.ts b/apps/wallet-dashboard/components/dialogs/staking/hooks/useStakeDialog.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/Staking/hooks/useStakeDialog.ts rename to apps/wallet-dashboard/components/dialogs/staking/hooks/useStakeDialog.ts diff --git a/apps/wallet-dashboard/components/Dialogs/Staking/index.ts b/apps/wallet-dashboard/components/dialogs/staking/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/Staking/index.ts rename to apps/wallet-dashboard/components/dialogs/staking/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/Staking/views/DetailsView.tsx b/apps/wallet-dashboard/components/dialogs/staking/views/DetailsView.tsx similarity index 88% rename from apps/wallet-dashboard/components/Dialogs/Staking/views/DetailsView.tsx rename to apps/wallet-dashboard/components/dialogs/staking/views/DetailsView.tsx index 48ebf1b4f3c..8feb22b96c8 100644 --- a/apps/wallet-dashboard/components/Dialogs/Staking/views/DetailsView.tsx +++ b/apps/wallet-dashboard/components/dialogs/staking/views/DetailsView.tsx @@ -1,7 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import { ExtendedDelegatedStake, ImageIcon, @@ -23,14 +22,11 @@ import { Badge, BadgeType, Divider, - InfoBox, - InfoBoxStyle, - InfoBoxType, LoadingIndicator, } from '@iota/apps-ui-kit'; -import { Warning } from '@iota/ui-icons'; import { formatAddress, IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { DialogLayout, DialogLayoutFooter, DialogLayoutBody } from '../../layout'; +import toast from 'react-hot-toast'; interface StakeDialogProps { handleClose: () => void; @@ -66,7 +62,7 @@ export function DetailsView({ const [iotaEarnedFormatted, iotaEarnedSymbol] = useFormatCoin(iotaEarned, IOTA_TYPE_ARG); const [totalStakeFormatted, totalStakeSymbol] = useFormatCoin(totalStake, IOTA_TYPE_ARG); - const validatorName = validatorSummary?.name || ''; + const validatorName = validatorSummary?.name || '--'; const subtitle = showActiveStatus ? ( <div className="flex items-center gap-1"> @@ -87,17 +83,7 @@ export function DetailsView({ } if (errorValidators) { - return ( - <div className="mb-2 flex h-full w-full items-center justify-center p-2"> - <InfoBox - title="Something went wrong" - supportingText={'An error occurred'} - style={InfoBoxStyle.Default} - type={InfoBoxType.Error} - icon={<Warning />} - /> - </div> - ); + toast.error('An error occurred fetching validator information'); } return ( @@ -138,7 +124,7 @@ export function DetailsView({ /> <KeyValueInfo keyText="Commission" - value={`${commission.toString()}%`} + value={`${commission ? commission.toString() : '--'}%`} fullwidth /> </div> diff --git a/apps/wallet-dashboard/components/Dialogs/Staking/views/EnterAmountDialogLayout.tsx b/apps/wallet-dashboard/components/dialogs/staking/views/EnterAmountDialogLayout.tsx similarity index 95% rename from apps/wallet-dashboard/components/Dialogs/Staking/views/EnterAmountDialogLayout.tsx rename to apps/wallet-dashboard/components/dialogs/staking/views/EnterAmountDialogLayout.tsx index 9ec233b1838..dc59e18ac07 100644 --- a/apps/wallet-dashboard/components/Dialogs/Staking/views/EnterAmountDialogLayout.tsx +++ b/apps/wallet-dashboard/components/dialogs/staking/views/EnterAmountDialogLayout.tsx @@ -1,8 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; -import { useFormatCoin, useStakeTxnInfo } from '@iota/core'; +import { useFormatCoin, useStakeTxnInfo, Validator } from '@iota/core'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { Button, @@ -18,14 +17,12 @@ import { InfoBox, } from '@iota/apps-ui-kit'; import { Field, type FieldProps, useFormikContext } from 'formik'; -import { Exclamation, Loader } from '@iota/ui-icons'; +import { Exclamation, Loader } from '@iota/apps-ui-icons'; import { useIotaClientQuery } from '@iota/dapp-kit'; - -import { Validator } from './Validator'; import { StakedInfo } from './StakedInfo'; import { DialogLayout, DialogLayoutBody, DialogLayoutFooter } from '../../layout'; -export interface FormValues { +interface FormValues { amount: string; } @@ -43,7 +40,7 @@ interface EnterAmountDialogLayoutProps { gasBudget?: string | number | null; } -function EnterAmountDialogLayout({ +export function EnterAmountDialogLayout({ selectedValidator, gasBudget, senderAddress, @@ -73,7 +70,7 @@ function EnterAmountDialogLayout({ <div className="flex w-full flex-col justify-between"> <div> <div className="mb-md"> - <Validator address={selectedValidator} isSelected showAction={false} /> + <Validator address={selectedValidator} isSelected showApy={false} /> </div> <StakedInfo validatorAddress={selectedValidator} @@ -162,5 +159,3 @@ function EnterAmountDialogLayout({ </DialogLayout> ); } - -export default EnterAmountDialogLayout; diff --git a/apps/wallet-dashboard/components/Dialogs/Staking/views/EnterAmountView.tsx b/apps/wallet-dashboard/components/dialogs/staking/views/EnterAmountView.tsx similarity index 83% rename from apps/wallet-dashboard/components/Dialogs/Staking/views/EnterAmountView.tsx rename to apps/wallet-dashboard/components/dialogs/staking/views/EnterAmountView.tsx index b90de664794..297ecba4e17 100644 --- a/apps/wallet-dashboard/components/Dialogs/Staking/views/EnterAmountView.tsx +++ b/apps/wallet-dashboard/components/dialogs/staking/views/EnterAmountView.tsx @@ -1,14 +1,13 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import { useFormatCoin, useBalance, CoinFormat, parseAmount, useCoinMetadata } from '@iota/core'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { useFormikContext } from 'formik'; import { useSignAndExecuteTransaction } from '@iota/dapp-kit'; -import { useNewStakeTransaction, useNotifications } from '@/hooks'; -import { NotificationType } from '@/stores/notificationStore'; -import EnterAmountDialogLayout from './EnterAmountDialogLayout'; +import { useNewStakeTransaction } from '@/hooks'; +import { EnterAmountDialogLayout } from './EnterAmountDialogLayout'; +import toast from 'react-hot-toast'; export interface FormValues { amount: string; @@ -24,7 +23,7 @@ interface EnterAmountViewProps { onSuccess: (digest: string) => void; } -function EnterAmountView({ +export function EnterAmountView({ selectedValidator, onBack, handleClose, @@ -32,7 +31,6 @@ function EnterAmountView({ senderAddress, onSuccess, }: EnterAmountViewProps): JSX.Element { - const { addNotification } = useNotifications(); const { mutateAsync: signAndExecuteTransaction } = useSignAndExecuteTransaction(); const { values, resetForm } = useFormikContext<FormValues>(); @@ -65,7 +63,7 @@ function EnterAmountView({ function handleStake(): void { if (!newStakeData?.transaction) { - addNotification('Stake transaction was not created', NotificationType.Error); + toast.error('Stake transaction was not created'); return; } signAndExecuteTransaction( @@ -75,11 +73,11 @@ function EnterAmountView({ { onSuccess: (tx) => { onSuccess(tx.digest); - addNotification('Stake transaction has been sent'); + toast.success('Stake transaction has been sent'); resetForm(); }, onError: () => { - addNotification('Stake transaction was not sent', NotificationType.Error); + toast.error('Stake transaction was not sent'); }, }, ); @@ -100,5 +98,3 @@ function EnterAmountView({ /> ); } - -export default EnterAmountView; diff --git a/apps/wallet-dashboard/components/Dialogs/Staking/views/EnterTimelockedAmountView.tsx b/apps/wallet-dashboard/components/dialogs/staking/views/EnterTimelockedAmountView.tsx similarity index 81% rename from apps/wallet-dashboard/components/Dialogs/Staking/views/EnterTimelockedAmountView.tsx rename to apps/wallet-dashboard/components/dialogs/staking/views/EnterTimelockedAmountView.tsx index b21a6ad3bd1..9ef12ac726d 100644 --- a/apps/wallet-dashboard/components/Dialogs/Staking/views/EnterTimelockedAmountView.tsx +++ b/apps/wallet-dashboard/components/dialogs/staking/views/EnterTimelockedAmountView.tsx @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React, { useEffect, useState } from 'react'; +import { useEffect, useState } from 'react'; import { useFormatCoin, CoinFormat, @@ -12,16 +12,12 @@ import { import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { useFormikContext } from 'formik'; import { useSignAndExecuteTransaction } from '@iota/dapp-kit'; -import { - useGetCurrentEpochStartTimestamp, - useNewStakeTimelockedTransaction, - useNotifications, -} from '@/hooks'; -import { NotificationType } from '@/stores/notificationStore'; +import { useGetCurrentEpochStartTimestamp, useNewStakeTimelockedTransaction } from '@/hooks'; import { prepareObjectsForTimelockedStakingTransaction } from '@/lib/utils'; -import EnterAmountDialogLayout from './EnterAmountDialogLayout'; +import { EnterAmountDialogLayout } from './EnterAmountDialogLayout'; +import toast from 'react-hot-toast'; -export interface FormValues { +interface FormValues { amount: string; } @@ -35,7 +31,7 @@ interface EnterTimelockedAmountViewProps { onSuccess: (digest: string) => void; } -function EnterTimelockedAmountView({ +export function EnterTimelockedAmountView({ selectedValidator, maxStakableTimelockedAmount, amountWithoutDecimals, @@ -44,7 +40,6 @@ function EnterTimelockedAmountView({ handleClose, onSuccess, }: EnterTimelockedAmountViewProps): JSX.Element { - const { addNotification } = useNotifications(); const { mutateAsync: signAndExecuteTransaction } = useSignAndExecuteTransaction(); const { resetForm } = useFormikContext<FormValues>(); @@ -84,11 +79,11 @@ function EnterTimelockedAmountView({ function handleStake(): void { if (groupedTimelockObjects.length === 0) { - addNotification('Invalid stake amount. Please try again.', NotificationType.Error); + toast.error('Invalid stake amount. Please try again.'); return; } if (!newStakeData?.transaction) { - addNotification('Stake transaction was not created', NotificationType.Error); + toast.error('Stake transaction was not created'); return; } signAndExecuteTransaction( @@ -98,11 +93,11 @@ function EnterTimelockedAmountView({ { onSuccess: (tx) => { onSuccess?.(tx.digest); - addNotification('Stake transaction has been sent'); + toast.success('Stake transaction has been sent'); resetForm(); }, onError: () => { - addNotification('Stake transaction was not sent', NotificationType.Error); + toast.error('Stake transaction was not sent'); }, }, ); @@ -124,5 +119,3 @@ function EnterTimelockedAmountView({ /> ); } - -export default EnterTimelockedAmountView; diff --git a/apps/wallet-dashboard/components/Dialogs/Staking/views/SelectValidatorView.tsx b/apps/wallet-dashboard/components/dialogs/staking/views/SelectValidatorView.tsx similarity index 89% rename from apps/wallet-dashboard/components/Dialogs/Staking/views/SelectValidatorView.tsx rename to apps/wallet-dashboard/components/dialogs/staking/views/SelectValidatorView.tsx index 626cbf05990..75032c10690 100644 --- a/apps/wallet-dashboard/components/Dialogs/Staking/views/SelectValidatorView.tsx +++ b/apps/wallet-dashboard/components/dialogs/staking/views/SelectValidatorView.tsx @@ -1,10 +1,8 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import { Button, Header } from '@iota/apps-ui-kit'; - -import { Validator } from './Validator'; +import { Validator } from '@iota/core'; import { DialogLayout, DialogLayoutBody, DialogLayoutFooter } from '../../layout'; interface SelectValidatorViewProps { @@ -15,7 +13,7 @@ interface SelectValidatorViewProps { handleClose: () => void; } -function SelectValidatorView({ +export function SelectValidatorView({ validators, onSelect, onNext, @@ -32,7 +30,7 @@ function SelectValidatorView({ <Validator key={validator} address={validator} - onClick={onSelect} + onClick={() => onSelect(validator)} isSelected={selectedValidator === validator} /> ))} @@ -52,5 +50,3 @@ function SelectValidatorView({ </DialogLayout> ); } - -export default SelectValidatorView; diff --git a/apps/wallet-dashboard/components/Dialogs/Staking/views/StakedInfo.tsx b/apps/wallet-dashboard/components/dialogs/staking/views/StakedInfo.tsx similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/Staking/views/StakedInfo.tsx rename to apps/wallet-dashboard/components/dialogs/staking/views/StakedInfo.tsx diff --git a/apps/wallet-dashboard/components/dialogs/staking/views/index.ts b/apps/wallet-dashboard/components/dialogs/staking/views/index.ts new file mode 100644 index 00000000000..fcddce54bca --- /dev/null +++ b/apps/wallet-dashboard/components/dialogs/staking/views/index.ts @@ -0,0 +1,8 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export * from './EnterAmountView'; +export * from './EnterTimelockedAmountView'; +export * from './EnterAmountDialogLayout'; +export * from './SelectValidatorView'; +export * from './DetailsView'; diff --git a/apps/wallet-dashboard/components/Dialogs/transaction/TransactionDetailsLayout.tsx b/apps/wallet-dashboard/components/dialogs/transaction/TransactionDetailsLayout.tsx similarity index 90% rename from apps/wallet-dashboard/components/Dialogs/transaction/TransactionDetailsLayout.tsx rename to apps/wallet-dashboard/components/dialogs/transaction/TransactionDetailsLayout.tsx index 44bc9474a6a..8b7d9cfa6a2 100644 --- a/apps/wallet-dashboard/components/Dialogs/transaction/TransactionDetailsLayout.tsx +++ b/apps/wallet-dashboard/components/dialogs/transaction/TransactionDetailsLayout.tsx @@ -1,8 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; + import { ExplorerLink } from '@/components'; -import { ExtendedTransaction } from '@/lib/interfaces'; import { Header, LoadingIndicator } from '@iota/apps-ui-kit'; import { useTransactionSummary, @@ -10,10 +9,10 @@ import { ExplorerLinkType, TransactionReceipt, useRecognizedPackages, + ExtendedTransaction, } from '@iota/core'; import { useCurrentAccount, useIotaClientContext } from '@iota/dapp-kit'; import { DialogLayoutBody, DialogLayoutFooter } from '../layout'; -import { Validator } from '../Staking/views/Validator'; import { Network } from '@iota/iota-sdk/client'; interface TransactionDialogDetailsProps { @@ -42,7 +41,6 @@ export function TransactionDetailsLayout({ transaction, onClose }: TransactionDi activeAddress={address} summary={summary} renderExplorerLink={ExplorerLink} - renderValidatorLogo={Validator} /> </DialogLayoutBody> <DialogLayoutFooter> diff --git a/apps/wallet-dashboard/components/Dialogs/transaction/index.ts b/apps/wallet-dashboard/components/dialogs/transaction/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/transaction/index.ts rename to apps/wallet-dashboard/components/dialogs/transaction/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/unstake/UnstakeDialog.tsx b/apps/wallet-dashboard/components/dialogs/unstake/UnstakeDialog.tsx similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/unstake/UnstakeDialog.tsx rename to apps/wallet-dashboard/components/dialogs/unstake/UnstakeDialog.tsx diff --git a/apps/wallet-dashboard/components/Dialogs/unstake/enums/index.ts b/apps/wallet-dashboard/components/dialogs/unstake/enums/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/unstake/enums/index.ts rename to apps/wallet-dashboard/components/dialogs/unstake/enums/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/unstake/enums/views.enums.ts b/apps/wallet-dashboard/components/dialogs/unstake/enums/views.enums.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/unstake/enums/views.enums.ts rename to apps/wallet-dashboard/components/dialogs/unstake/enums/views.enums.ts diff --git a/apps/wallet-dashboard/components/Dialogs/unstake/hooks/index.ts b/apps/wallet-dashboard/components/dialogs/unstake/hooks/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/unstake/hooks/index.ts rename to apps/wallet-dashboard/components/dialogs/unstake/hooks/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/unstake/hooks/useUnstakeDialog.ts b/apps/wallet-dashboard/components/dialogs/unstake/hooks/useUnstakeDialog.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/unstake/hooks/useUnstakeDialog.ts rename to apps/wallet-dashboard/components/dialogs/unstake/hooks/useUnstakeDialog.ts diff --git a/apps/wallet-dashboard/components/Dialogs/unstake/index.ts b/apps/wallet-dashboard/components/dialogs/unstake/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/unstake/index.ts rename to apps/wallet-dashboard/components/dialogs/unstake/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/unstake/views/UnstakeTimelockedObjectsView.tsx b/apps/wallet-dashboard/components/dialogs/unstake/views/UnstakeTimelockedObjectsView.tsx similarity index 94% rename from apps/wallet-dashboard/components/Dialogs/unstake/views/UnstakeTimelockedObjectsView.tsx rename to apps/wallet-dashboard/components/dialogs/unstake/views/UnstakeTimelockedObjectsView.tsx index 45e34ee5cf0..d023d8528b2 100644 --- a/apps/wallet-dashboard/components/Dialogs/unstake/views/UnstakeTimelockedObjectsView.tsx +++ b/apps/wallet-dashboard/components/dialogs/unstake/views/UnstakeTimelockedObjectsView.tsx @@ -3,8 +3,8 @@ import { StakeRewardsPanel, ValidatorStakingData } from '@/components'; import { DialogLayout, DialogLayoutBody, DialogLayoutFooter } from '../../layout'; -import { Validator } from '../../Staking/views/Validator'; -import { useNewUnstakeTimelockedTransaction, useNotifications } from '@/hooks'; +import { Validator } from '@iota/core'; +import { useNewUnstakeTimelockedTransaction } from '@/hooks'; import { Collapsible, TimeUnit, @@ -24,7 +24,7 @@ import { } from '@iota/apps-ui-kit'; import { useCurrentAccount, useSignAndExecuteTransaction } from '@iota/dapp-kit'; import { IotaSignAndExecuteTransactionOutput } from '@iota/wallet-standard'; -import { NotificationType } from '@/stores/notificationStore'; +import toast from 'react-hot-toast'; interface UnstakeTimelockedObjectsViewProps { onClose: () => void; @@ -39,7 +39,6 @@ export function UnstakeTimelockedObjectsView({ onBack, onSuccess, }: UnstakeTimelockedObjectsViewProps) { - const { addNotification } = useNotifications(); const activeAddress = useCurrentAccount()?.address ?? ''; const { data: activeValidators } = useGetActiveValidatorsInfo(); @@ -71,7 +70,7 @@ export function UnstakeTimelockedObjectsView({ ); function handleCopySuccess() { - addNotification('Copied to clipboard'); + toast.success('Copied to clipboard'); } async function handleUnstake(): Promise<void> { @@ -83,12 +82,12 @@ export function UnstakeTimelockedObjectsView({ }, { onSuccess: (tx) => { - addNotification('Unstake transaction has been sent'); + toast.success('Unstake transaction has been sent'); onSuccess(tx); }, }, ).catch(() => { - addNotification('Unstake transaction was not sent', NotificationType.Error); + toast.error('Unstake transaction was not sent'); }); } diff --git a/apps/wallet-dashboard/components/Dialogs/unstake/views/UnstakeView.tsx b/apps/wallet-dashboard/components/dialogs/unstake/views/UnstakeView.tsx similarity index 92% rename from apps/wallet-dashboard/components/Dialogs/unstake/views/UnstakeView.tsx rename to apps/wallet-dashboard/components/dialogs/unstake/views/UnstakeView.tsx index 5d07eadcec3..41e5d3b11f4 100644 --- a/apps/wallet-dashboard/components/Dialogs/unstake/views/UnstakeView.tsx +++ b/apps/wallet-dashboard/components/dialogs/unstake/views/UnstakeView.tsx @@ -17,16 +17,17 @@ import { GAS_SYMBOL, useFormatCoin, useGetStakingValidatorDetails, + Validator, } from '@iota/core'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { useCurrentAccount, useSignAndExecuteTransaction } from '@iota/dapp-kit'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; import { StakeRewardsPanel, ValidatorStakingData } from '@/components'; import { DialogLayout, DialogLayoutFooter, DialogLayoutBody } from '../../layout'; -import { Validator } from '../../Staking/views/Validator'; -import { useNewUnstakeTransaction, useNotifications } from '@/hooks'; + +import { useNewUnstakeTransaction } from '@/hooks'; import { IotaSignAndExecuteTransactionOutput } from '@iota/wallet-standard'; -import { NotificationType } from '@/stores/notificationStore'; +import toast from 'react-hot-toast'; interface UnstakeDialogProps { extendedStake: ExtendedDelegatedStake; @@ -44,7 +45,6 @@ export function UnstakeView({ showActiveStatus, }: UnstakeDialogProps): JSX.Element { const activeAddress = useCurrentAccount()?.address ?? ''; - const { addNotification } = useNotifications(); const { data: unstakeData, isPending: isUnstakeTxPending } = useNewUnstakeTransaction( activeAddress, extendedStake.stakedIotaId, @@ -81,12 +81,12 @@ export function UnstakeView({ }, { onSuccess: (tx) => { - addNotification('Unstake transaction has been sent'); + toast.success('Unstake transaction has been sent'); onSuccess(tx); }, }, ).catch(() => { - addNotification('Unstake transaction was not sent', NotificationType.Error); + toast.error('Unstake transaction was not sent'); }); } diff --git a/apps/wallet-dashboard/components/Dialogs/unstake/views/index.ts b/apps/wallet-dashboard/components/dialogs/unstake/views/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/unstake/views/index.ts rename to apps/wallet-dashboard/components/dialogs/unstake/views/index.ts diff --git a/apps/wallet-dashboard/components/Dialogs/vesting/VestingScheduleBox.tsx b/apps/wallet-dashboard/components/dialogs/vesting/VestingScheduleBox.tsx similarity index 76% rename from apps/wallet-dashboard/components/Dialogs/vesting/VestingScheduleBox.tsx rename to apps/wallet-dashboard/components/dialogs/vesting/VestingScheduleBox.tsx index d9a93fc8d7c..5e789b3da45 100644 --- a/apps/wallet-dashboard/components/Dialogs/vesting/VestingScheduleBox.tsx +++ b/apps/wallet-dashboard/components/dialogs/vesting/VestingScheduleBox.tsx @@ -3,9 +3,9 @@ import { useGetCurrentEpochStartTimestamp } from '@/hooks'; import { DisplayStats, DisplayStatsType } from '@iota/apps-ui-kit'; -import { useFormatCoin } from '@iota/core'; +import { formatDate, useFormatCoin } from '@iota/core'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; -import { LockLocked } from '@iota/ui-icons'; +import { LockLocked } from '@iota/apps-ui-icons'; interface VestingScheduleBoxProps { amount: bigint; @@ -20,9 +20,16 @@ export function VestingScheduleBox({ const { data: currentEpochMs } = useGetCurrentEpochStartTimestamp(); const isLocked = expirationTimestampMs > Number(currentEpochMs); + const transactionDate = formatDate(Number(expirationTimestampMs), [ + 'day', + 'month', + 'year', + 'hour', + 'minute', + ]); return ( <DisplayStats - label={new Date(expirationTimestampMs).toLocaleDateString()} + label={transactionDate} value={`${formattedAmountVested} ${amountVestedSymbol}`} type={isLocked ? DisplayStatsType.Default : DisplayStatsType.Secondary} icon={isLocked && <LockLocked className="h-4 w-4" />} diff --git a/apps/wallet-dashboard/components/Dialogs/vesting/VestingScheduleDialog.tsx b/apps/wallet-dashboard/components/dialogs/vesting/VestingScheduleDialog.tsx similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/vesting/VestingScheduleDialog.tsx rename to apps/wallet-dashboard/components/dialogs/vesting/VestingScheduleDialog.tsx diff --git a/apps/wallet-dashboard/components/Dialogs/vesting/index.ts b/apps/wallet-dashboard/components/dialogs/vesting/index.ts similarity index 100% rename from apps/wallet-dashboard/components/Dialogs/vesting/index.ts rename to apps/wallet-dashboard/components/dialogs/vesting/index.ts diff --git a/apps/wallet-dashboard/components/index.ts b/apps/wallet-dashboard/components/index.ts index fcccf7b3a95..0a713ba5fb6 100644 --- a/apps/wallet-dashboard/components/index.ts +++ b/apps/wallet-dashboard/components/index.ts @@ -1,25 +1,24 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export { default as RouteLink } from './RouteLink'; -export { default as Notifications } from './Notifications/Notifications'; -export { default as Box } from './Box'; -export { default as AmountBox } from './AmountBox'; -export { default as Input } from './Input'; -export { default as VirtualList } from './VirtualList'; -export { default as ExternalImage } from './ExternalImage'; +export * from './RouteLink'; +export * from './Box'; +export * from './AmountBox'; +export * from './Input'; +export * from './VirtualList'; +export * from './ExternalImage'; export * from './PageSizeSelector'; export * from './PaginationOptions'; export * from './account-balance/AccountBalance'; export * from './coins'; -export * from './AppList'; -export * from './Cards'; -export * from './Buttons'; +export * from './app-list'; +export * from './cards'; +export * from './buttons'; export * from './transactions'; export * from './staking-overview'; export * from './ExplorerLink'; -export * from './Dialogs'; +export * from './dialogs'; export * from './ValidatorStakingData'; export * from './tiles'; export * from './migration'; @@ -29,3 +28,4 @@ export * from './StakeRewardsPanel'; export * from './MigrationOverview'; export * from './SupplyIncreaseVestingOverview'; export * from './staked-timelock-object'; +export * from './loading'; diff --git a/apps/wallet-dashboard/components/loading/Loading.tsx b/apps/wallet-dashboard/components/loading/Loading.tsx new file mode 100644 index 00000000000..cea927d254c --- /dev/null +++ b/apps/wallet-dashboard/components/loading/Loading.tsx @@ -0,0 +1,20 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { LoadingIndicator, type LoadingIndicatorProps } from '@iota/apps-ui-kit'; +import type { ReactNode } from 'react'; + +interface LoadingProps extends LoadingIndicatorProps { + loading: boolean; + children: ReactNode | ReactNode[]; +} + +export function Loading({ loading, children, ...indicatorProps }: LoadingProps) { + return loading ? ( + <div className="flex h-full w-full items-center justify-center"> + <LoadingIndicator {...indicatorProps} /> + </div> + ) : ( + <>{children}</> + ); +} diff --git a/apps/wallet-dashboard/components/loading/index.ts b/apps/wallet-dashboard/components/loading/index.ts new file mode 100644 index 00000000000..0c903d2dc5e --- /dev/null +++ b/apps/wallet-dashboard/components/loading/index.ts @@ -0,0 +1,4 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +export * from './Loading'; diff --git a/apps/wallet-dashboard/components/migration/MigrationObjectLoading.tsx b/apps/wallet-dashboard/components/migration/MigrationObjectLoading.tsx new file mode 100644 index 00000000000..297d1a8fa81 --- /dev/null +++ b/apps/wallet-dashboard/components/migration/MigrationObjectLoading.tsx @@ -0,0 +1,27 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { Card, CardImage, ImageShape, Skeleton } from '@iota/apps-ui-kit'; + +export function MigrationObjectLoading() { + return ( + <div className="flex h-full max-h-full w-full flex-col overflow-hidden"> + {new Array(10).fill(0).map((_, index) => ( + <Card key={index}> + <CardImage shape={ImageShape.SquareRounded}> + <div className="h-10 w-10 animate-pulse bg-neutral-90 dark:bg-neutral-12" /> + <Skeleton widthClass="w-10" heightClass="h-10" isRounded={false} /> + </CardImage> + <div className="flex flex-col gap-y-xs"> + <Skeleton widthClass="w-40" heightClass="h-3.5" /> + <Skeleton widthClass="w-32" heightClass="h-3" hasSecondaryColors /> + </div> + <div className="ml-auto flex flex-col gap-y-xs"> + <Skeleton widthClass="w-20" heightClass="h-3.5" /> + <Skeleton widthClass="w-16" heightClass="h-3" hasSecondaryColors /> + </div> + </Card> + ))} + </div> + ); +} diff --git a/apps/wallet-dashboard/components/migration/MigrationObjectsPanel.tsx b/apps/wallet-dashboard/components/migration/MigrationObjectsPanel.tsx index 915cc56dda5..05d28f17ae1 100644 --- a/apps/wallet-dashboard/components/migration/MigrationObjectsPanel.tsx +++ b/apps/wallet-dashboard/components/migration/MigrationObjectsPanel.tsx @@ -3,7 +3,7 @@ 'use client'; -import { useGroupedMigrationObjectsByExpirationDate } from '@/hooks'; +import { useGroupedStardustObjects } from '@/hooks'; import { STARDUST_MIGRATABLE_OBJECTS_FILTER_LIST, STARDUST_TIMELOCKED_OBJECTS_FILTER_LIST, @@ -12,24 +12,21 @@ import { StardustOutputDetailsFilter } from '@/lib/enums'; import { Button, ButtonType, - Card, - CardImage, Chip, - ImageShape, InfoBox, InfoBoxStyle, InfoBoxType, Panel, - Skeleton, Title, } from '@iota/apps-ui-kit'; import type { IotaObjectData } from '@iota/iota-sdk/client'; -import { Close, Warning } from '@iota/ui-icons'; +import { Close, Warning } from '@iota/apps-ui-icons'; import clsx from 'clsx'; import { useState } from 'react'; import { MigrationObjectDetailsCard } from './migration-object-details-card'; -import VirtualList from '../VirtualList'; +import { VirtualList } from '../VirtualList'; import { filterMigrationObjects } from '@/lib/utils'; +import { MigrationObjectLoading } from './MigrationObjectLoading'; const FILTERS = { migratable: STARDUST_MIGRATABLE_OBJECTS_FILTER_LIST, @@ -39,13 +36,13 @@ const FILTERS = { interface MigrationObjectsPanelProps { selectedObjects: IotaObjectData[]; onClose: () => void; - isTimelocked: boolean; + groupByTimelockUC: boolean; } export function MigrationObjectsPanel({ selectedObjects, onClose, - isTimelocked, + groupByTimelockUC, }: MigrationObjectsPanelProps): React.JSX.Element { const [stardustOutputDetailsFilter, setStardustOutputDetailsFilter] = useState<StardustOutputDetailsFilter>(StardustOutputDetailsFilter.All); @@ -54,11 +51,11 @@ export function MigrationObjectsPanel({ data: resolvedObjects = [], isLoading, error: isErrored, - } = useGroupedMigrationObjectsByExpirationDate(selectedObjects, isTimelocked); + } = useGroupedStardustObjects(selectedObjects, groupByTimelockUC); const filteredObjects = filterMigrationObjects(resolvedObjects, stardustOutputDetailsFilter); - const filters = isTimelocked ? FILTERS.timelocked : FILTERS.migratable; + const filters = groupByTimelockUC ? FILTERS.timelocked : FILTERS.migratable; const isHidden = selectedObjects.length === 0; return ( @@ -83,7 +80,7 @@ export function MigrationObjectsPanel({ </div> <div className="flex min-h-0 flex-col py-sm"> <div className="h-full flex-1 overflow-auto"> - {isLoading && <LoadingPanel />} + {isLoading && <MigrationObjectLoading />} {isErrored && !isLoading && ( <div className="flex h-full max-h-full w-full flex-col items-center"> <InfoBox @@ -104,7 +101,7 @@ export function MigrationObjectsPanel({ render={(migrationObject) => ( <MigrationObjectDetailsCard migrationObject={migrationObject} - isTimelocked={isTimelocked} + isTimelocked={groupByTimelockUC} /> )} /> @@ -116,26 +113,3 @@ export function MigrationObjectsPanel({ </div> ); } - -function LoadingPanel() { - return ( - <div className="flex h-full max-h-full w-full flex-col overflow-hidden"> - {new Array(10).fill(0).map((_, index) => ( - <Card key={index}> - <CardImage shape={ImageShape.SquareRounded}> - <div className="h-10 w-10 animate-pulse bg-neutral-90 dark:bg-neutral-12" /> - <Skeleton widthClass="w-10" heightClass="h-10" isRounded={false} /> - </CardImage> - <div className="flex flex-col gap-y-xs"> - <Skeleton widthClass="w-40" heightClass="h-3.5" /> - <Skeleton widthClass="w-32" heightClass="h-3" hasSecondaryColors /> - </div> - <div className="ml-auto flex flex-col gap-y-xs"> - <Skeleton widthClass="w-20" heightClass="h-3.5" /> - <Skeleton widthClass="w-16" heightClass="h-3" hasSecondaryColors /> - </div> - </Card> - ))} - </div> - ); -} diff --git a/apps/wallet-dashboard/components/migration/index.ts b/apps/wallet-dashboard/components/migration/index.ts index cf43709989c..be5fbb96d1a 100644 --- a/apps/wallet-dashboard/components/migration/index.ts +++ b/apps/wallet-dashboard/components/migration/index.ts @@ -2,3 +2,5 @@ // SPDX-License-Identifier: Apache-2.0 export * from './MigrationObjectsPanel'; +export * from './MigrationObjectLoading'; +export * from './migration-object-details-card'; diff --git a/apps/wallet-dashboard/components/migration/migration-object-details-card/MigrationObjectDetailsCard.tsx b/apps/wallet-dashboard/components/migration/migration-object-details-card/MigrationObjectDetailsCard.tsx index 599e58dbd21..424ea81df5d 100644 --- a/apps/wallet-dashboard/components/migration/migration-object-details-card/MigrationObjectDetailsCard.tsx +++ b/apps/wallet-dashboard/components/migration/migration-object-details-card/MigrationObjectDetailsCard.tsx @@ -8,9 +8,9 @@ import { MIGRATION_OBJECT_WITHOUT_UC_KEY } from '@/lib/constants'; import { CommonMigrationObjectType } from '@/lib/enums'; import { ResolvedObjectTypes } from '@/lib/types'; import { Card, CardBody, CardImage, ImageShape, LabelText, LabelTextSize } from '@iota/apps-ui-kit'; -import { MILLISECONDS_PER_SECOND, TimeUnit, useFormatCoin, useTimeAgo } from '@iota/core'; +import { MILLISECONDS_PER_SECOND, useCountdownByTimestamp, useFormatCoin } from '@iota/core'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; -import { Assets, DataStack, IotaLogoMark } from '@iota/ui-icons'; +import { Assets, DataStack, IotaLogoMark } from '@iota/apps-ui-icons'; import { useState } from 'react'; interface MigrationObjectDetailsCardProps { @@ -129,27 +129,22 @@ function UnlockConditionLabel({ groupKey, isTimelocked: isTimelocked }: UnlockCo !isLoadingEpochStart && unlockConditionTimestampMs < parseInt(epochStartMs); // TODO: https://github.com/iotaledger/iota/issues/4369 const isInAFutureEpoch = !isLoadingEpochEnd && unlockConditionTimestampMs > epochEndMs; - const outputTimestampMs = isInAFutureEpoch ? unlockConditionTimestampMs : epochEndMs; - const timeLabel = useTimeAgo({ - timeFrom: outputTimestampMs, - shortedTimeLabel: true, - shouldEnd: true, - maxTimeUnit: TimeUnit.ONE_DAY, + const formattedLastPayoutExpirationTime = useCountdownByTimestamp(Number(outputTimestampMs), { + showSeconds: false, }); - const showLabel = !isFromPreviousEpoch && outputTimestampMs > currentDateMs; return ( - <div className="ml-auto h-full whitespace-nowrap"> - {showLabel && ( + showLabel && ( + <div className="h-full w-1/4 whitespace-nowrap"> <LabelText size={LabelTextSize.Small} - text={timeLabel} + text={formattedLastPayoutExpirationTime} label={isTimelocked ? 'Unlocks in' : 'Expires in'} /> - )} - </div> + </div> + ) ); } diff --git a/apps/wallet-dashboard/components/staking-overview/StartStaking.tsx b/apps/wallet-dashboard/components/staking-overview/StartStaking.tsx index 573a254a3d2..ab99da76411 100644 --- a/apps/wallet-dashboard/components/staking-overview/StartStaking.tsx +++ b/apps/wallet-dashboard/components/staking-overview/StartStaking.tsx @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { StakeDialog, useStakeDialog } from '../Dialogs'; +import { StakeDialog, useStakeDialog } from '../dialogs'; import { Theme, useTheme } from '@iota/core'; import { Banner } from '../Banner'; diff --git a/apps/wallet-dashboard/components/tiles/AssetTileLink.tsx b/apps/wallet-dashboard/components/tiles/AssetTileLink.tsx index 64c4d805513..1c5ab567df8 100644 --- a/apps/wallet-dashboard/components/tiles/AssetTileLink.tsx +++ b/apps/wallet-dashboard/components/tiles/AssetTileLink.tsx @@ -3,29 +3,35 @@ 'use client'; -import { AssetCategory } from '@/lib/enums'; -import { VisibilityOff } from '@iota/ui-icons'; +import { isKioskOwnerToken, useKioskClient, KioskTile, AssetCategory } from '@iota/core'; +import { VisibilityOff } from '@iota/apps-ui-icons'; import { VisualAssetTile } from '.'; import { IotaObjectData } from '@iota/iota-sdk/client'; import { NonVisualAssetCard } from './NonVisualAssetTile'; +import { useCurrentAccount } from '@iota/dapp-kit'; interface AssetTileLinkProps { asset: IotaObjectData; - type: AssetCategory; + type: AssetCategory | null; onClick: (asset: IotaObjectData) => void; } export function AssetTileLink({ asset, type, onClick }: AssetTileLinkProps): React.JSX.Element { + const account = useCurrentAccount(); + const kioskClient = useKioskClient(); + const isTokenOwnedByKiosk = isKioskOwnerToken(kioskClient.network, asset); function handleClick() { onClick(asset); } return ( <> - {type === AssetCategory.Visual ? ( + {type === AssetCategory.Visual && isTokenOwnedByKiosk ? ( + <KioskTile object={asset} address={account?.address} onClick={handleClick} /> + ) : type === AssetCategory.Visual ? ( <VisualAssetTile asset={asset} icon={<VisibilityOff />} onClick={handleClick} /> ) : ( - <NonVisualAssetCard asset={asset} onClick={handleClick} /> + <NonVisualAssetCard asset={asset} /> )} </> ); diff --git a/apps/wallet-dashboard/components/tiles/NonVisualAssetTile.tsx b/apps/wallet-dashboard/components/tiles/NonVisualAssetTile.tsx index 0758714510e..d1f52415989 100644 --- a/apps/wallet-dashboard/components/tiles/NonVisualAssetTile.tsx +++ b/apps/wallet-dashboard/components/tiles/NonVisualAssetTile.tsx @@ -4,22 +4,26 @@ import { Card, CardAction, CardActionType, CardBody, CardType } from '@iota/apps-ui-kit'; import { IotaObjectData } from '@iota/iota-sdk/client'; import { formatAddress, parseStructTag } from '@iota/iota-sdk/utils'; -import { ArrowTopRight } from '@iota/ui-icons'; +import { ArrowTopRight } from '@iota/apps-ui-icons'; +import { ExplorerLink } from '../ExplorerLink'; +import { ExplorerLinkType } from '@iota/core'; type NonVisualAssetCardProps = { asset: IotaObjectData; -} & Pick<React.ComponentProps<typeof Card>, 'onClick'>; +} & React.ComponentProps<typeof Card>; -export function NonVisualAssetCard({ asset, onClick }: NonVisualAssetCardProps): React.JSX.Element { +export function NonVisualAssetCard({ asset }: NonVisualAssetCardProps): React.JSX.Element { const { address, module, name } = parseStructTag(asset.type!); return ( - <Card type={CardType.Default} isHoverable onClick={onClick}> - <CardBody - title={formatAddress(asset.objectId!)} - subtitle={`${formatAddress(address)}::${module}::${name}`} - isTextTruncated - /> - <CardAction type={CardActionType.Link} icon={<ArrowTopRight />} /> - </Card> + <ExplorerLink objectID={asset.objectId} type={ExplorerLinkType.Object}> + <Card type={CardType.Default} isHoverable> + <CardBody + title={formatAddress(asset.objectId!)} + subtitle={`${formatAddress(address)}::${module}::${name}`} + isTextTruncated + /> + <CardAction type={CardActionType.Link} icon={<ArrowTopRight />} /> + </Card> + </ExplorerLink> ); } diff --git a/apps/wallet-dashboard/components/tiles/VisualAssetTile.tsx b/apps/wallet-dashboard/components/tiles/VisualAssetTile.tsx index 642d06daff0..8103bc6fdc7 100644 --- a/apps/wallet-dashboard/components/tiles/VisualAssetTile.tsx +++ b/apps/wallet-dashboard/components/tiles/VisualAssetTile.tsx @@ -4,7 +4,6 @@ 'use client'; import { IotaObjectData } from '@iota/iota-sdk/client'; -import React from 'react'; import { useGetNFTDisplay } from '@iota/core'; import { FlexDirection } from '@/lib/ui/enums'; import { VisualAssetCard, VisualAssetType, type VisualAssetCardProps } from '@iota/apps-ui-kit'; diff --git a/apps/wallet-dashboard/components/transactions/TransactionAmount.tsx b/apps/wallet-dashboard/components/transactions/TransactionAmount.tsx index 498ebc19522..b04e409d977 100644 --- a/apps/wallet-dashboard/components/transactions/TransactionAmount.tsx +++ b/apps/wallet-dashboard/components/transactions/TransactionAmount.tsx @@ -11,7 +11,7 @@ interface TransactionAmountProps { approximation?: boolean; } -export default function TransactionAmount({ +export function TransactionAmount({ amount, coinType, label, diff --git a/apps/wallet-dashboard/components/transactions/TransactionIcon.tsx b/apps/wallet-dashboard/components/transactions/TransactionIcon.tsx deleted file mode 100644 index 8e04e154f13..00000000000 --- a/apps/wallet-dashboard/components/transactions/TransactionIcon.tsx +++ /dev/null @@ -1,32 +0,0 @@ -// Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -import { LoadingIndicator } from '@iota/apps-ui-kit'; -import { ArrowBottomLeft, ArrowTopRight, Info, IotaLogoMark, Person, Stake } from '@iota/ui-icons'; - -const ICON_COLORS = { - primary: 'text-primary-30', - error: 'text-error-30', -}; - -const icons = { - Send: <ArrowTopRight className={ICON_COLORS.primary} />, - Receive: <ArrowBottomLeft className={ICON_COLORS.primary} />, - Transaction: <ArrowTopRight className={ICON_COLORS.primary} />, - Staked: <Stake className={ICON_COLORS.primary} />, - Unstaked: <Stake className={ICON_COLORS.primary} />, - Rewards: <IotaLogoMark className={ICON_COLORS.primary} />, - Failed: <Info className={ICON_COLORS.error} />, - Loading: <LoadingIndicator />, - PersonalMessage: <Person className={ICON_COLORS.primary} />, -}; - -interface TransactionIconProps { - txnFailed?: boolean; - variant: keyof typeof icons; -} - -function TransactionIcon({ txnFailed, variant }: TransactionIconProps) { - return <div className="[&_svg]:h-5 [&_svg]:w-5">{icons[txnFailed ? 'Failed' : variant]}</div>; -} -export default TransactionIcon; diff --git a/apps/wallet-dashboard/components/transactions/TransactionSummary.tsx b/apps/wallet-dashboard/components/transactions/TransactionSummary.tsx index 6cca89942ca..c76614db006 100644 --- a/apps/wallet-dashboard/components/transactions/TransactionSummary.tsx +++ b/apps/wallet-dashboard/components/transactions/TransactionSummary.tsx @@ -13,7 +13,7 @@ interface TransactionSummaryProps { showGasSummary?: boolean; } -export default function TransactionSummary({ +export function TransactionSummary({ summary, isLoading, isError, diff --git a/apps/wallet-dashboard/components/transactions/TransactionTile.tsx b/apps/wallet-dashboard/components/transactions/TransactionTile.tsx index 0c76c27ceb1..e845dd0c870 100644 --- a/apps/wallet-dashboard/components/transactions/TransactionTile.tsx +++ b/apps/wallet-dashboard/components/transactions/TransactionTile.tsx @@ -3,10 +3,7 @@ 'use client'; -import React, { useState } from 'react'; -import TransactionIcon from './TransactionIcon'; -import formatTimestamp from '@/lib/utils/time'; -import { ExtendedTransaction, TransactionState } from '@/lib/interfaces'; +import { useState } from 'react'; import { Card, CardType, @@ -18,11 +15,21 @@ import { CardActionType, Dialog, } from '@iota/apps-ui-kit'; -import { useFormatCoin, getLabel, useTransactionSummary } from '@iota/core'; +import { + useFormatCoin, + getTransactionAction, + useTransactionSummary, + ExtendedTransaction, + TransactionState, + TransactionIcon, + checkIfIsTimelockedStaking, + getTransactionAmountForTimelocked, + formatDate, +} from '@iota/core'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { useCurrentAccount } from '@iota/dapp-kit'; -import { TransactionDetailsLayout } from '../Dialogs/transaction/TransactionDetailsLayout'; -import { DialogLayout } from '../Dialogs/layout'; +import { TransactionDetailsLayout } from '../dialogs/transaction/TransactionDetailsLayout'; +import { DialogLayout } from '../dialogs/layout'; interface TransactionTileProps { transaction: ExtendedTransaction; @@ -38,16 +45,33 @@ export function TransactionTile({ transaction }: TransactionTileProps): JSX.Elem currentAddress: account?.address, recognizedPackagesList: [], }); - const [formatAmount, symbol] = useFormatCoin( - Math.abs(Number(address ? transactionSummary?.balanceChanges?.[address]?.[0]?.amount : 0)), - IOTA_TYPE_ARG, + + const { isTimelockedStaking, isTimelockedUnstaking } = checkIfIsTimelockedStaking( + transaction.raw?.events, ); + const balanceChanges = transactionSummary?.balanceChanges; + + function getAmount(tx: ExtendedTransaction) { + if ((isTimelockedStaking || isTimelockedUnstaking) && tx.raw.events) { + return getTransactionAmountForTimelocked(tx.raw.events); + } else { + return address && balanceChanges?.[address]?.[0]?.amount + ? Math.abs(Number(balanceChanges?.[address]?.[0]?.amount)) + : 0; + } + } + + const transactionAmount = getAmount(transaction); + const [formatAmount, symbol] = useFormatCoin(transactionAmount, IOTA_TYPE_ARG); + function openDetailsDialog() { setOpen(true); } - const transactionDate = transaction?.timestamp && formatTimestamp(transaction.timestamp); + const transactionDate = + transaction?.timestamp && + formatDate(Number(transaction?.timestamp), ['day', 'month', 'year', 'hour', 'minute']); return ( <> @@ -55,7 +79,7 @@ export function TransactionTile({ transaction }: TransactionTileProps): JSX.Elem <CardImage type={ImageType.BgSolid} shape={ImageShape.SquareRounded}> <TransactionIcon txnFailed={transaction.state === TransactionState.Failed} - variant={getLabel(transaction?.raw, address)} + variant={getTransactionAction(transaction?.raw, address)} /> </CardImage> <CardBody diff --git a/apps/wallet-dashboard/components/transactions/TransactionsList.tsx b/apps/wallet-dashboard/components/transactions/TransactionsList.tsx index 526329aed43..bc3f0da5fce 100644 --- a/apps/wallet-dashboard/components/transactions/TransactionsList.tsx +++ b/apps/wallet-dashboard/components/transactions/TransactionsList.tsx @@ -1,14 +1,13 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React from 'react'; import { useCurrentAccount } from '@iota/dapp-kit'; import { VirtualList, TransactionTile } from '@/components'; import { useQueryTransactionsByAddress } from '@iota/core'; import { getExtendedTransaction } from '@/lib/utils/transaction'; import { IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; -function TransactionsList() { +export function TransactionsList() { const currentAccount = useCurrentAccount(); const { data: transactions, error } = useQueryTransactionsByAddress(currentAccount?.address); @@ -23,5 +22,3 @@ function TransactionsList() { return <VirtualList items={transactions || []} estimateSize={() => 60} render={virtualItem} />; } - -export default TransactionsList; diff --git a/apps/wallet-dashboard/components/transactions/TransactionsOverview.tsx b/apps/wallet-dashboard/components/transactions/TransactionsOverview.tsx index fcaf74eddaa..e01628377bc 100644 --- a/apps/wallet-dashboard/components/transactions/TransactionsOverview.tsx +++ b/apps/wallet-dashboard/components/transactions/TransactionsOverview.tsx @@ -3,9 +3,9 @@ 'use client'; import { Panel, Title } from '@iota/apps-ui-kit'; -import TransactionsList from './TransactionsList'; +import { TransactionsList } from './TransactionsList'; -function TransactionsOverview() { +export function TransactionsOverview() { return ( <Panel> <Title title="Activity" /> @@ -15,5 +15,3 @@ function TransactionsOverview() { </Panel> ); } - -export default TransactionsOverview; diff --git a/apps/wallet-dashboard/components/transactions/index.ts b/apps/wallet-dashboard/components/transactions/index.ts index 1dc714be773..276bc9ee5f6 100644 --- a/apps/wallet-dashboard/components/transactions/index.ts +++ b/apps/wallet-dashboard/components/transactions/index.ts @@ -1,9 +1,8 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export { default as TransactionAmount } from './TransactionAmount'; -export { default as TransactionSummary } from './TransactionSummary'; +export * from './TransactionAmount'; +export * from './TransactionSummary'; export * from './TransactionTile'; -export { default as TransactionIcon } from './TransactionIcon'; -export { default as TransactionsList } from './TransactionsList'; -export { default as TransactionsOverview } from './TransactionsOverview'; +export * from './TransactionsList'; +export * from './TransactionsOverview'; diff --git a/apps/wallet-dashboard/hooks/index.ts b/apps/wallet-dashboard/hooks/index.ts index 5ad872afeed..2d59a8c3d89 100644 --- a/apps/wallet-dashboard/hooks/index.ts +++ b/apps/wallet-dashboard/hooks/index.ts @@ -3,13 +3,12 @@ export * from './useNewUnstakeTransaction'; export * from './useNewStakeTransaction'; -export * from './useNotifications'; export * from './useSendCoinTransaction'; -export * from './useCreateSendAssetTransaction'; export * from './useGetCurrentEpochStartTimestamp'; export * from './useTimelockedUnstakeTransaction'; export * from './useExplorerLinkGetter'; export * from './useGetStardustMigratableObjects'; export * from './useGetSupplyIncreaseVestingObjects'; -export * from './useGroupedMigrationObjectsByExpirationDate'; +export * from './useGroupedStardustObjects'; export * from './useTransferTransaction'; +export * from './usePersistedNetwork'; diff --git a/apps/wallet-dashboard/hooks/useCreateSendAssetTransaction.ts b/apps/wallet-dashboard/hooks/useCreateSendAssetTransaction.ts deleted file mode 100644 index aa50c4fac7c..00000000000 --- a/apps/wallet-dashboard/hooks/useCreateSendAssetTransaction.ts +++ /dev/null @@ -1,37 +0,0 @@ -// Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -import { useSignAndExecuteTransaction } from '@iota/dapp-kit'; -import { Transaction } from '@iota/iota-sdk/transactions'; -import { useMutation } from '@tanstack/react-query'; - -export function useCreateSendAssetTransaction( - objectId: string, - onSuccess?: () => void, - onError?: (error: unknown) => void, -) { - const { mutateAsync: signAndExecuteTransaction } = useSignAndExecuteTransaction(); - const mutation = useMutation({ - mutationFn: async (to: string) => { - if (!to) { - throw new Error('Missing data'); - } - - const tx = new Transaction(); - tx.transferObjects([tx.object(objectId)], to); - - return signAndExecuteTransaction({ - transaction: tx, - options: { - showEffects: true, - showEvents: true, - showInput: true, - }, - }); - }, - onSuccess, - onError, - }); - - return { mutation }; -} diff --git a/apps/wallet-dashboard/hooks/useGetSupplyIncreaseVestingObjects.ts b/apps/wallet-dashboard/hooks/useGetSupplyIncreaseVestingObjects.ts index 58d19d63641..e086333e4ce 100644 --- a/apps/wallet-dashboard/hooks/useGetSupplyIncreaseVestingObjects.ts +++ b/apps/wallet-dashboard/hooks/useGetSupplyIncreaseVestingObjects.ts @@ -26,7 +26,7 @@ import { } from '@iota/core'; import { Transaction } from '@iota/iota-sdk/transactions'; -export function useGetSupplyIncreaseVestingObjects(address: string): { +interface SupplyIncreaseVestingObject { nextPayout: SupplyIncreaseVestingPayout | undefined; lastPayout: SupplyIncreaseVestingPayout | undefined; supplyIncreaseVestingSchedule: VestingOverview; @@ -40,7 +40,10 @@ export function useGetSupplyIncreaseVestingObjects(address: string): { } | undefined; refreshStakeList: () => void; -} { + isSupplyIncreaseVestingScheduleEmpty: boolean; +} + +export function useGetSupplyIncreaseVestingObjects(address: string): SupplyIncreaseVestingObject { const { data: currentEpochMs } = useGetCurrentEpochStartTimestamp(); const { data: timelockedObjects, refetch: refetchGetAllOwnedObjects } = useGetAllOwnedObjects( @@ -93,6 +96,13 @@ export function useGetSupplyIncreaseVestingObjects(address: string): { supplyIncreaseVestingUnlockedObjectIds, ); + const isSupplyIncreaseVestingScheduleEmpty = + !supplyIncreaseVestingSchedule.totalVested && + !supplyIncreaseVestingSchedule.totalLocked && + !supplyIncreaseVestingSchedule.availableClaiming && + !supplyIncreaseVestingSchedule.totalStaked && + !supplyIncreaseVestingSchedule.totalEarned; + function refreshStakeList() { refetchTimelockedStakedObjects(); refetchGetAllOwnedObjects(); @@ -108,5 +118,6 @@ export function useGetSupplyIncreaseVestingObjects(address: string): { isTimelockedStakedObjectsLoading, unlockAllSupplyIncreaseVesting, refreshStakeList, + isSupplyIncreaseVestingScheduleEmpty, }; } diff --git a/apps/wallet-dashboard/hooks/useGroupedMigrationObjectsByExpirationDate.ts b/apps/wallet-dashboard/hooks/useGroupedStardustObjects.ts similarity index 91% rename from apps/wallet-dashboard/hooks/useGroupedMigrationObjectsByExpirationDate.ts rename to apps/wallet-dashboard/hooks/useGroupedStardustObjects.ts index d9d96816fd2..9bf3b9ba79f 100644 --- a/apps/wallet-dashboard/hooks/useGroupedMigrationObjectsByExpirationDate.ts +++ b/apps/wallet-dashboard/hooks/useGroupedStardustObjects.ts @@ -13,9 +13,9 @@ import { useQuery } from '@tanstack/react-query'; import { useGetCurrentEpochStartTimestamp } from './useGetCurrentEpochStartTimestamp'; import { useGetCurrentEpochEndTimestamp } from './useGetCurrentEpochEndTimestamp'; -export function useGroupedMigrationObjectsByExpirationDate( +export function useGroupedStardustObjects( objects: IotaObjectData[], - isTimelockUnlockCondition: boolean = false, + groupByTimelockUC: boolean = false, ) { const client = useIotaClient(); const address = useCurrentAccount()?.address; @@ -32,7 +32,7 @@ export function useGroupedMigrationObjectsByExpirationDate( 'grouped-migration-objects', objects, address, - isTimelockUnlockCondition, + groupByTimelockUC, epochStartMs, epochEndMs, ], @@ -44,7 +44,8 @@ export function useGroupedMigrationObjectsByExpirationDate( objects, client, address, - isTimelockUnlockCondition, + groupByTimelockUC, + epochStartMs, ); return sortStardustResolvedObjectsByExpiration( diff --git a/apps/wallet-dashboard/hooks/useNotifications.ts b/apps/wallet-dashboard/hooks/useNotifications.ts deleted file mode 100644 index c5ec455b886..00000000000 --- a/apps/wallet-dashboard/hooks/useNotifications.ts +++ /dev/null @@ -1,9 +0,0 @@ -// Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -import { useNotificationStore } from '@/stores/notificationStore'; - -export function useNotifications() { - const addNotification = useNotificationStore((state) => state.addNotification); - return { addNotification }; -} diff --git a/apps/wallet-dashboard/hooks/usePersistedNetwork.ts b/apps/wallet-dashboard/hooks/usePersistedNetwork.ts new file mode 100644 index 00000000000..9b4bbbf3c03 --- /dev/null +++ b/apps/wallet-dashboard/hooks/usePersistedNetwork.ts @@ -0,0 +1,41 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { useIotaClientContext } from '@iota/dapp-kit'; +import { NetworkConfiguration } from '@iota/iota-sdk/client'; +import { useLocalStorage } from '@iota/core'; +import toast from 'react-hot-toast'; +import { useEffect } from 'react'; + +export function usePersistedNetwork() { + const clientContext = useIotaClientContext(); + const activeNetwork = clientContext.network; + + const LOCAL_STORAGE_KEY = 'network_iota-dashboard'; + + const [persistedNetwork, setPersistedNetwork] = useLocalStorage<string>( + LOCAL_STORAGE_KEY, + activeNetwork, + ); + + async function handleNetworkChange(network: NetworkConfiguration) { + if (persistedNetwork === network.id) { + return; + } + + clientContext.selectNetwork(network.id); + setPersistedNetwork(network.id); + toast.success(`Switched to ${network.name}`); + } + + useEffect(() => { + if (activeNetwork !== persistedNetwork) { + setPersistedNetwork(activeNetwork); + } + }, [persistedNetwork, activeNetwork, setPersistedNetwork]); + + return { + persistedNetwork, + handleNetworkChange, + }; +} diff --git a/apps/wallet-dashboard/lib/constants/routes.constants.ts b/apps/wallet-dashboard/lib/constants/routes.constants.ts index 5d8d27d33e3..983f86c4693 100644 --- a/apps/wallet-dashboard/lib/constants/routes.constants.ts +++ b/apps/wallet-dashboard/lib/constants/routes.constants.ts @@ -3,7 +3,7 @@ import type { ProtectedRoute } from '../interfaces'; import { ProtectedRouteTitle } from '../enums'; -import { Activity, Assets, Calendar, Home, Tokens, Vesting } from '@iota/ui-icons'; +import { Activity, Assets, Home, Migration, Stake, Vesting } from '@iota/apps-ui-icons'; export const HOMEPAGE_ROUTE: ProtectedRoute = { title: ProtectedRouteTitle.Home, @@ -20,18 +20,18 @@ export const ASSETS_ROUTE: ProtectedRoute = { export const STAKING_ROUTE: ProtectedRoute = { title: ProtectedRouteTitle.Staking, path: '/staking', - icon: Activity, + icon: Stake, }; export const ACTIVITY_ROUTE: ProtectedRoute = { title: ProtectedRouteTitle.Activity, path: '/activity', - icon: Tokens, + icon: Activity, }; export const MIGRATIONS_ROUTE: ProtectedRoute = { title: ProtectedRouteTitle.Migrations, path: '/migrations', - icon: Calendar, + icon: Migration, }; export const VESTING_ROUTE: ProtectedRoute = { title: ProtectedRouteTitle.Vesting, diff --git a/apps/wallet-dashboard/lib/constants/vesting.constants.ts b/apps/wallet-dashboard/lib/constants/vesting.constants.ts index 51ace2168a7..71937eb7d05 100644 --- a/apps/wallet-dashboard/lib/constants/vesting.constants.ts +++ b/apps/wallet-dashboard/lib/constants/vesting.constants.ts @@ -7,7 +7,10 @@ import { DAYS_PER_YEAR, } from '@iota/core/constants/time.constants'; import { TimelockedObject } from '../interfaces'; -import { getMockedTimelockedStakedObjectsWithDynamicDate } from '../utils/vesting/getMockedTimelockedStakedObjectsWithDynamicDate'; +import { + getMockedSupplyIncreaseVestingTimelockedObjectsWithDynamicDate, + getMockedVestingTimelockedStakedObjectsWithDynamicDate, +} from '../utils/vesting/buildMockedObjectsWithDynamicDateUtils'; import { DelegatedTimelockedStake } from '@iota/iota-sdk/client'; export const SUPPLY_INCREASE_VESTING_PAYOUT_SCHEDULE = 2 * DAYS_PER_WEEK; @@ -667,5 +670,12 @@ export const MOCKED_VESTING_TIMELOCKED_STAKED_OBJECTS: DelegatedTimelockedStake[ }, ]; -export const mockedTimelockedStackedObjectsWithDynamicDate = - getMockedTimelockedStakedObjectsWithDynamicDate(MOCKED_VESTING_TIMELOCKED_STAKED_OBJECTS); +export const mockedSupplyIncreaseVestingTimelockedObjectsWithDynamicDate = + getMockedSupplyIncreaseVestingTimelockedObjectsWithDynamicDate( + MOCKED_SUPPLY_INCREASE_VESTING_TIMELOCKED_OBJECTS, + ); + +export const mockedVestingTimelockedStakedObjectsWithDynamicDate = + getMockedVestingTimelockedStakedObjectsWithDynamicDate( + MOCKED_VESTING_TIMELOCKED_STAKED_OBJECTS, + ); diff --git a/apps/wallet-dashboard/lib/enums/assetCategory.enum.ts b/apps/wallet-dashboard/lib/enums/assetCategory.enums.ts similarity index 70% rename from apps/wallet-dashboard/lib/enums/assetCategory.enum.ts rename to apps/wallet-dashboard/lib/enums/assetCategory.enums.ts index bab9839ceaa..b7d45b16343 100644 --- a/apps/wallet-dashboard/lib/enums/assetCategory.enum.ts +++ b/apps/wallet-dashboard/lib/enums/assetCategory.enums.ts @@ -2,6 +2,6 @@ // SPDX-License-Identifier: Apache-2.0 export enum AssetCategory { - Visual = 'Visual', - Other = 'Other', + Visual = 'visual', + Other = 'other', } diff --git a/apps/wallet-dashboard/lib/enums/index.ts b/apps/wallet-dashboard/lib/enums/index.ts index 6d936d1ec45..8cac53c8275 100644 --- a/apps/wallet-dashboard/lib/enums/index.ts +++ b/apps/wallet-dashboard/lib/enums/index.ts @@ -1,8 +1,8 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export * from './protectedRouteTitle.enum'; -export * from './assetCategory.enum'; +export * from './protectedRouteTitle.enums'; +export * from './assetCategory.enums'; export * from './commonMigrationObjectType.enums'; -export * from './stardustOutputDetailsFilter.enum'; -export * from './stardustOutputMigrationStatus.enum'; +export * from './stardustOutputDetailsFilter.enums'; +export * from './stardustOutputMigrationStatus.enums'; diff --git a/apps/wallet-dashboard/lib/enums/protectedRouteTitle.enum.ts b/apps/wallet-dashboard/lib/enums/protectedRouteTitle.enums.ts similarity index 100% rename from apps/wallet-dashboard/lib/enums/protectedRouteTitle.enum.ts rename to apps/wallet-dashboard/lib/enums/protectedRouteTitle.enums.ts diff --git a/apps/wallet-dashboard/lib/enums/stardustOutputDetailsFilter.enum.ts b/apps/wallet-dashboard/lib/enums/stardustOutputDetailsFilter.enums.ts similarity index 100% rename from apps/wallet-dashboard/lib/enums/stardustOutputDetailsFilter.enum.ts rename to apps/wallet-dashboard/lib/enums/stardustOutputDetailsFilter.enums.ts diff --git a/apps/wallet-dashboard/lib/enums/stardustOutputMigrationStatus.enum.ts b/apps/wallet-dashboard/lib/enums/stardustOutputMigrationStatus.enums.ts similarity index 100% rename from apps/wallet-dashboard/lib/enums/stardustOutputMigrationStatus.enum.ts rename to apps/wallet-dashboard/lib/enums/stardustOutputMigrationStatus.enums.ts diff --git a/apps/wallet-dashboard/lib/interfaces/appRoute.interface.ts b/apps/wallet-dashboard/lib/interfaces/appRoute.interfaces.ts similarity index 100% rename from apps/wallet-dashboard/lib/interfaces/appRoute.interface.ts rename to apps/wallet-dashboard/lib/interfaces/appRoute.interfaces.ts diff --git a/apps/wallet-dashboard/lib/interfaces/dialogView.interface.ts b/apps/wallet-dashboard/lib/interfaces/dialogView.interfaces.ts similarity index 100% rename from apps/wallet-dashboard/lib/interfaces/dialogView.interface.ts rename to apps/wallet-dashboard/lib/interfaces/dialogView.interfaces.ts diff --git a/apps/wallet-dashboard/lib/interfaces/index.ts b/apps/wallet-dashboard/lib/interfaces/index.ts index 6999e34dd61..8d652795abd 100644 --- a/apps/wallet-dashboard/lib/interfaces/index.ts +++ b/apps/wallet-dashboard/lib/interfaces/index.ts @@ -1,8 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export * from './transactions.interface'; -export * from './timelock.interface'; -export * from './vesting.interface'; -export * from './appRoute.interface'; -export * from './dialogView.interface'; +export * from './timelock.interfaces'; +export * from './vesting.interfaces'; +export * from './appRoute.interfaces'; +export * from './dialogView.interfaces'; diff --git a/apps/wallet-dashboard/lib/interfaces/timelock.interface.ts b/apps/wallet-dashboard/lib/interfaces/timelock.interfaces.ts similarity index 100% rename from apps/wallet-dashboard/lib/interfaces/timelock.interface.ts rename to apps/wallet-dashboard/lib/interfaces/timelock.interfaces.ts diff --git a/apps/wallet-dashboard/lib/interfaces/vesting.interface.ts b/apps/wallet-dashboard/lib/interfaces/vesting.interfaces.ts similarity index 100% rename from apps/wallet-dashboard/lib/interfaces/vesting.interface.ts rename to apps/wallet-dashboard/lib/interfaces/vesting.interfaces.ts diff --git a/apps/wallet-dashboard/lib/ui/enums/flex-direction.enums.ts b/apps/wallet-dashboard/lib/ui/enums/flexDirection.enums.ts similarity index 100% rename from apps/wallet-dashboard/lib/ui/enums/flex-direction.enums.ts rename to apps/wallet-dashboard/lib/ui/enums/flexDirection.enums.ts diff --git a/apps/wallet-dashboard/lib/ui/enums/index.ts b/apps/wallet-dashboard/lib/ui/enums/index.ts index 416ee72bd74..ac0d7e3ead5 100644 --- a/apps/wallet-dashboard/lib/ui/enums/index.ts +++ b/apps/wallet-dashboard/lib/ui/enums/index.ts @@ -1,4 +1,4 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export * from './flex-direction.enums'; +export * from './flexDirection.enums'; diff --git a/apps/wallet-dashboard/lib/utils/migration/groupMigrationObjects.ts b/apps/wallet-dashboard/lib/utils/migration/groupMigrationObjects.ts index 73991f4ac09..b0be24e48f9 100644 --- a/apps/wallet-dashboard/lib/utils/migration/groupMigrationObjects.ts +++ b/apps/wallet-dashboard/lib/utils/migration/groupMigrationObjects.ts @@ -16,7 +16,7 @@ import { STARDUST_BASIC_OUTPUT_TYPE, STARDUST_NFT_OUTPUT_TYPE, } from '@iota/core'; -import { extractMigrationOutputFields, extractOwnedStorageDepositReturnAmount } from '.'; +import { extractOutputFields, extractOwnedStorageDepositReturnAmount } from '.'; import { IotaClient, IotaObjectData } from '@iota/iota-sdk/client'; import { MIGRATION_OBJECT_WITHOUT_UC_KEY } from '@/lib/constants'; @@ -24,7 +24,8 @@ export async function groupMigrationObjectsByUnlockCondition( objectsData: IotaObjectData[], client: IotaClient, currentAddress: string = '', - isTimelockUnlockCondition: boolean = false, + groupByTimelockUC: boolean = false, // If true, group by timelock unlock condition, else group by expiration unlock condition + currentEpochStartMs?: number, ): Promise<ResolvedObjectTypes[]> { const flatObjects: ResolvedObjectTypes[] = []; const basicObjectMap: Map<string, ResolvedBasicObject> = new Map(); @@ -37,49 +38,51 @@ export async function groupMigrationObjectsByUnlockCondition( const chunk = objectsData.slice(i, i + PROMISE_CHUNK_SIZE); const promises = chunk.map(async (object) => { - const objectFields = extractMigrationOutputFields(object); + const objectFields = extractOutputFields(object); let groupKey: string | undefined; - if (isTimelockUnlockCondition) { + if (groupByTimelockUC) { const timestamp = objectFields.timelock_uc?.fields.unix_time.toString(); groupKey = timestamp; } else { const timestamp = objectFields.expiration_uc?.fields.unix_time.toString(); // Timestamp can be undefined if the object was timelocked and is now unlocked // and it doesn't have an expiration unlock condition - groupKey = timestamp ?? MIGRATION_OBJECT_WITHOUT_UC_KEY; + groupKey = + timestamp && + currentEpochStartMs !== undefined && + Number(timestamp) >= currentEpochStartMs / MILLISECONDS_PER_SECOND + ? timestamp + : MIGRATION_OBJECT_WITHOUT_UC_KEY; } if (!groupKey) { return; } - if (object.type === STARDUST_BASIC_OUTPUT_TYPE) { - const existing = basicObjectMap.get(groupKey); - const gasReturn = extractOwnedStorageDepositReturnAmount( - objectFields, - currentAddress, - ); - const newBalance = - (existing ? existing.balance : 0n) + - BigInt(objectFields.balance) + - (gasReturn ?? 0n); + const existingBasicObject = basicObjectMap.get(groupKey); + const gasReturn = extractOwnedStorageDepositReturnAmount(objectFields, currentAddress); + const newBalance = + (existingBasicObject ? existingBasicObject.balance : 0n) + + BigInt(objectFields.balance) + + (gasReturn ?? 0n); - if (existing) { - existing.balance = newBalance; - } else { - const newBasicObject: ResolvedBasicObject = { - balance: newBalance, - unlockConditionTimestamp: groupKey, - type: object.type, - commonObjectType: CommonMigrationObjectType.Basic, - output: object, - uniqueId: objectFields.id.id, - }; - basicObjectMap.set(groupKey, newBasicObject); - flatObjects.push(newBasicObject); - } - } else if (object.type === STARDUST_NFT_OUTPUT_TYPE) { + if (existingBasicObject) { + existingBasicObject.balance = newBalance; + } else { + const newBasicObject: ResolvedBasicObject = { + balance: newBalance, + unlockConditionTimestamp: groupKey, + type: STARDUST_BASIC_OUTPUT_TYPE, + commonObjectType: CommonMigrationObjectType.Basic, + output: object, + uniqueId: objectFields.id.id, + }; + basicObjectMap.set(groupKey, newBasicObject); + flatObjects.push(newBasicObject); + } + + if (object.type === STARDUST_NFT_OUTPUT_TYPE) { const nftDetails = await getNftDetails(object, groupKey, client); flatObjects.push(...nftDetails); } @@ -160,7 +163,7 @@ async function getNftDetails( expirationKey: UnlockConditionTimestamp, client: IotaClient, ): Promise<ResolvedNftObject[]> { - const objectFields = extractMigrationOutputFields(object); + const objectFields = extractOutputFields(object); const nftOutputDynamicFields = await client.getDynamicFields({ parentId: objectFields.id.id, }); @@ -195,7 +198,7 @@ async function extractNativeTokensFromObject( client: IotaClient, expirationKey: UnlockConditionTimestamp, ): Promise<ResolvedNativeToken[]> { - const fields = extractMigrationOutputFields(object); + const fields = extractOutputFields(object); const bagId = fields.native_tokens.fields.id.id; const bagSize = Number(fields.native_tokens.fields.size); @@ -222,7 +225,9 @@ async function extractNativeTokensFromObject( result.push({ name: tokenName, balance, - coinType: nativeTokenFields.name, + coinType: nativeTokenFields.name.startsWith('0x') + ? nativeTokenFields.name + : `0x${nativeTokenFields.name}`, unlockConditionTimestamp: expirationKey, commonObjectType: CommonMigrationObjectType.NativeToken, output: object, diff --git a/apps/wallet-dashboard/lib/utils/migration/groupStardustObjectsByMigrationStatus.ts b/apps/wallet-dashboard/lib/utils/migration/groupStardustObjectsByMigrationStatus.ts index b10f1cf4218..5c0a20a824c 100644 --- a/apps/wallet-dashboard/lib/utils/migration/groupStardustObjectsByMigrationStatus.ts +++ b/apps/wallet-dashboard/lib/utils/migration/groupStardustObjectsByMigrationStatus.ts @@ -3,6 +3,9 @@ import { CommonOutputObjectWithUc, MILLISECONDS_PER_SECOND } from '@iota/core'; import { IotaObjectData } from '@iota/iota-sdk/client'; +import { filterMigrationObjects } from './filterMigrationObjectDetails'; +import { CommonMigrationObjectType, StardustOutputDetailsFilter } from '@/lib/enums'; +import { ResolvedObjectTypes } from '@/lib/types'; export type StardustMigrationGroupedObjects = { migratable: IotaObjectData[]; @@ -20,7 +23,7 @@ export function groupStardustObjectsByMigrationStatus( const epochUnix = epochTimestampMs / MILLISECONDS_PER_SECOND; for (const outputObject of stardustOutputObjects) { - const outputObjectFields = extractMigrationOutputFields(outputObject); + const outputObjectFields = extractOutputFields(outputObject); if ( outputObjectFields.timelock_uc && @@ -47,36 +50,46 @@ export function groupStardustObjectsByMigrationStatus( return { migratable, timelocked }; } -interface MigratableObjectsData { +interface StardustObjectsTotals { totalNativeTokens: number; totalVisualAssets: number; totalIotaAmount: bigint; totalNotOwnedStorageDepositReturnAmount: bigint; } -interface SummarizeMigrationObjectParams { +interface StardustObjectsTotalsParams { basicOutputs: IotaObjectData[] | undefined; nftOutputs: IotaObjectData[] | undefined; address: string; + resolvedObjects?: ResolvedObjectTypes[]; } -export function summarizeMigratableObjectValues({ +export function getStardustObjectsTotals({ basicOutputs = [], nftOutputs = [], address, -}: SummarizeMigrationObjectParams): MigratableObjectsData { + resolvedObjects, +}: StardustObjectsTotalsParams): StardustObjectsTotals { let totalNativeTokens = 0; let totalIotaAmount: bigint = 0n; let totalNotOwnedStorageDepositReturnAmount: bigint = 0n; + let filteredObjects: ResolvedObjectTypes[] = []; const totalVisualAssets = nftOutputs.length; const outputObjects = [...basicOutputs, ...nftOutputs]; + if (resolvedObjects) { + filteredObjects = filterMigrationObjects( + resolvedObjects, + StardustOutputDetailsFilter.NativeTokens, + ); + } + totalNativeTokens = getUniqueNativeTokensByCoinType(filteredObjects); + for (const output of outputObjects) { - const outputObjectFields = extractMigrationOutputFields(output); + const outputObjectFields = extractOutputFields(output); totalIotaAmount += BigInt(outputObjectFields.balance); - totalNativeTokens += parseInt(outputObjectFields.native_tokens.fields.size); totalIotaAmount += extractOwnedStorageDepositReturnAmount(outputObjectFields, address) || 0n; totalNotOwnedStorageDepositReturnAmount += @@ -91,29 +104,6 @@ export function summarizeMigratableObjectValues({ }; } -interface UnmmigratableObjectsData { - totalTimelockedObjects: number; -} - -export function summarizeTimelockedObjectValues({ - basicOutputs = [], - nftOutputs = [], -}: Omit<SummarizeMigrationObjectParams, 'address'>): UnmmigratableObjectsData { - const basicObjects = basicOutputs.length; - const nftObjects = nftOutputs.length; - let nativeTokens = 0; - - for (const output of [...basicOutputs, ...nftOutputs]) { - const outputObjectFields = extractMigrationOutputFields(output); - - nativeTokens += parseInt(outputObjectFields.native_tokens.fields.size); - } - - const totalTimelockedObjects = basicObjects + nativeTokens + nftObjects; - - return { totalTimelockedObjects }; -} - export function extractOwnedStorageDepositReturnAmount( { storage_deposit_return_uc }: CommonOutputObjectWithUc, address: string, @@ -127,9 +117,7 @@ export function extractOwnedStorageDepositReturnAmount( return null; } -export function extractMigrationOutputFields( - outputObject: IotaObjectData, -): CommonOutputObjectWithUc { +export function extractOutputFields(outputObject: IotaObjectData): CommonOutputObjectWithUc { return ( outputObject.content as unknown as { fields: CommonOutputObjectWithUc; @@ -149,3 +137,11 @@ export function extractNotOwnedStorageDepositReturnAmount( } return null; } + +export function getUniqueNativeTokensByCoinType(objects: ResolvedObjectTypes[]): number { + return new Set( + objects + .filter((obj) => obj.commonObjectType === CommonMigrationObjectType.NativeToken) + .map((obj) => obj.coinType), + ).size; +} diff --git a/apps/wallet-dashboard/lib/utils/time.ts b/apps/wallet-dashboard/lib/utils/time.ts index 8c66df0c883..9df8d1969f0 100644 --- a/apps/wallet-dashboard/lib/utils/time.ts +++ b/apps/wallet-dashboard/lib/utils/time.ts @@ -1,11 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export default function formatTimestamp(timeStamp: number): string { - const date = new Date(timeStamp); - return new Intl.DateTimeFormat('en-US').format(date); -} - export function parseTimestamp(timestampMs: string): number { const timestamp = parseInt(timestampMs, 10); if (!Number.isFinite(timestamp)) { diff --git a/apps/wallet-dashboard/lib/utils/transaction.ts b/apps/wallet-dashboard/lib/utils/transaction.ts index bc7f08c2756..f89f3281619 100644 --- a/apps/wallet-dashboard/lib/utils/transaction.ts +++ b/apps/wallet-dashboard/lib/utils/transaction.ts @@ -1,9 +1,9 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { ExtendedTransaction, TransactionAction, TransactionState } from '@/lib/interfaces'; import { IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; import { parseTimestamp } from './time'; +import { TransactionState, ExtendedTransaction, getTransactionAction } from '@iota/core'; const getTransactionTransactionState = (tx: IotaTransactionBlockResponse): TransactionState => { const executionStatus = tx.effects?.status.status; @@ -20,14 +20,6 @@ const getTransactionTransactionState = (tx: IotaTransactionBlockResponse): Trans return TransactionState.Pending; }; -export const getTransactionAction = ( - transaction: IotaTransactionBlockResponse, - currentAddress: string, -) => { - const isSender = transaction.transaction?.data.sender === currentAddress; - return isSender ? TransactionAction.Send : TransactionAction.Receive; -}; - export const getExtendedTransaction = ( tx: IotaTransactionBlockResponse, address: string, diff --git a/apps/wallet-dashboard/lib/utils/vesting/getMockedTimelockedStakedObjectsWithDynamicDate.ts b/apps/wallet-dashboard/lib/utils/vesting/buildMockedObjectsWithDynamicDateUtils.ts similarity index 53% rename from apps/wallet-dashboard/lib/utils/vesting/getMockedTimelockedStakedObjectsWithDynamicDate.ts rename to apps/wallet-dashboard/lib/utils/vesting/buildMockedObjectsWithDynamicDateUtils.ts index 6a475e7251e..e51875647f3 100644 --- a/apps/wallet-dashboard/lib/utils/vesting/getMockedTimelockedStakedObjectsWithDynamicDate.ts +++ b/apps/wallet-dashboard/lib/utils/vesting/buildMockedObjectsWithDynamicDateUtils.ts @@ -1,14 +1,35 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { MILLISECONDS_PER_DAY } from '@iota/core/constants/time.constants'; +import { TimelockedObject } from '@/lib/interfaces'; +import { DAYS_PER_WEEK, MILLISECONDS_PER_DAY } from '@iota/core/constants/time.constants'; import { DelegatedTimelockedStake } from '@iota/iota-sdk/client'; +/** + * Maps the passed objects to a set of objects with modified expirationTimestampMs date. + * It spreads the dates from CURRENT datetime PLUS two weeks, + * so that the reference payout we use will always be with expiration of two weeks from now, + * then the rest are spread BACK up to array length with a STEP of two weeks (vesting schedule). + */ +export function getMockedSupplyIncreaseVestingTimelockedObjectsWithDynamicDate( + vestingObjects: TimelockedObject[], +): TimelockedObject[] { + const twoWeeksMs = 2 * DAYS_PER_WEEK * MILLISECONDS_PER_DAY; + const twoWeeksFromNow = Date.now() + twoWeeksMs; + + return structuredClone(vestingObjects) + .map((object, idx) => { + object.expirationTimestampMs = twoWeeksFromNow - idx * twoWeeksMs; + return object; + }) + .reverse(); +} + /** * Gets the objects in a distributed manner with half of the objects * being unlocked and the other half being locked. */ -export function getMockedTimelockedStakedObjectsWithDynamicDate( +export function getMockedVestingTimelockedStakedObjectsWithDynamicDate( delegatedObjects: DelegatedTimelockedStake[], ): DelegatedTimelockedStake[] { const now = Date.now(); diff --git a/apps/wallet-dashboard/lib/utils/vesting/vesting.spec.ts b/apps/wallet-dashboard/lib/utils/vesting/vesting.spec.ts index 3612418d48b..4bb594a8952 100644 --- a/apps/wallet-dashboard/lib/utils/vesting/vesting.spec.ts +++ b/apps/wallet-dashboard/lib/utils/vesting/vesting.spec.ts @@ -3,11 +3,12 @@ import { MILLISECONDS_PER_HOUR } from '@iota/core/constants/time.constants'; import { - mockedTimelockedStackedObjectsWithDynamicDate, + mockedVestingTimelockedStakedObjectsWithDynamicDate, MOCKED_SUPPLY_INCREASE_VESTING_TIMELOCKED_OBJECTS, MOCKED_VESTING_TIMELOCKED_STAKED_OBJECTS, SUPPLY_INCREASE_STAKER_VESTING_DURATION, SUPPLY_INCREASE_VESTING_PAYOUTS_IN_1_YEAR, + mockedSupplyIncreaseVestingTimelockedObjectsWithDynamicDate, } from '../../constants'; import { SupplyIncreaseUserType, SupplyIncreaseVestingPayout } from '../../interfaces'; import { formatDelegatedTimelockedStake, isTimelockedObject } from '../timelock'; @@ -23,12 +24,12 @@ const MOCKED_CURRENT_EPOCH_TIMESTAMP = Date.now() + MILLISECONDS_PER_HOUR * 6; / describe('get last supply increase vesting payout', () => { it('should get the object with highest expirationTimestampMs', () => { - const timelockedObjects = MOCKED_SUPPLY_INCREASE_VESTING_TIMELOCKED_OBJECTS; + const timelockedObjects = mockedSupplyIncreaseVestingTimelockedObjectsWithDynamicDate; // the last in the array is also the one with the latest expiration time const expectedObject = - MOCKED_SUPPLY_INCREASE_VESTING_TIMELOCKED_OBJECTS[ - MOCKED_SUPPLY_INCREASE_VESTING_TIMELOCKED_OBJECTS.length - 1 + mockedSupplyIncreaseVestingTimelockedObjectsWithDynamicDate[ + mockedSupplyIncreaseVestingTimelockedObjectsWithDynamicDate.length - 1 ]; const lastPayout = getLatestOrEarliestSupplyIncreaseVestingPayout( @@ -63,7 +64,7 @@ describe('get supply increase user type', () => { describe('build supply increase staker vesting portfolio', () => { it('should build with mocked timelocked objects', () => { - const timelockedObjects = MOCKED_SUPPLY_INCREASE_VESTING_TIMELOCKED_OBJECTS; + const timelockedObjects = mockedSupplyIncreaseVestingTimelockedObjectsWithDynamicDate; const lastPayout = getLatestOrEarliestSupplyIncreaseVestingPayout( timelockedObjects, @@ -80,7 +81,7 @@ describe('build supply increase staker vesting portfolio', () => { }); it('should build properly with mocked timelocked staked objects', () => { - const timelockedStakedObjects = mockedTimelockedStackedObjectsWithDynamicDate; + const timelockedStakedObjects = mockedVestingTimelockedStakedObjectsWithDynamicDate; const extendedTimelockedStakedObjects = formatDelegatedTimelockedStake(timelockedStakedObjects); const lastPayout = getLatestOrEarliestSupplyIncreaseVestingPayout( @@ -98,8 +99,8 @@ describe('build supply increase staker vesting portfolio', () => { }); it('should build properly with mix of mocked timelocked and timelocked staked objects', () => { - const timelockedObjects = MOCKED_SUPPLY_INCREASE_VESTING_TIMELOCKED_OBJECTS; - const timelockedStakedObjects = MOCKED_VESTING_TIMELOCKED_STAKED_OBJECTS; + const timelockedObjects = mockedSupplyIncreaseVestingTimelockedObjectsWithDynamicDate; + const timelockedStakedObjects = mockedVestingTimelockedStakedObjectsWithDynamicDate; const extendedTimelockedStakedObjects = formatDelegatedTimelockedStake(timelockedStakedObjects); const mixedObjects = [...timelockedObjects, ...extendedTimelockedStakedObjects]; diff --git a/apps/wallet-dashboard/package.json b/apps/wallet-dashboard/package.json index 9cff8865c4d..8f2ff1358dd 100644 --- a/apps/wallet-dashboard/package.json +++ b/apps/wallet-dashboard/package.json @@ -17,11 +17,11 @@ "dependencies": { "@growthbook/growthbook": "^1.0.0", "@growthbook/growthbook-react": "^1.0.0", + "@iota/apps-ui-icons": "workspace:*", "@iota/apps-ui-kit": "workspace:*", "@iota/core": "workspace:*", "@iota/dapp-kit": "workspace:*", "@iota/iota-sdk": "workspace:*", - "@iota/ui-icons": "workspace:*", "@iota/wallet-standard": "workspace:*", "@tanstack/react-query": "^5.50.1", "@tanstack/react-virtual": "^3.5.0", @@ -33,6 +33,7 @@ "zustand": "^4.4.1" }, "devDependencies": { + "@tanstack/react-query-devtools": "^5.58.0", "@types/jest": "^29.5.2", "@types/node": "^20.14.10", "@types/react": "^18.3.3", diff --git a/apps/wallet-dashboard/providers/AppProviders.tsx b/apps/wallet-dashboard/providers/AppProviders.tsx index 26be321a43b..f2af54c8658 100644 --- a/apps/wallet-dashboard/providers/AppProviders.tsx +++ b/apps/wallet-dashboard/providers/AppProviders.tsx @@ -8,8 +8,9 @@ import { GrowthBookProvider } from '@growthbook/growthbook-react'; import { IotaClientProvider, lightTheme, darkTheme, WalletProvider } from '@iota/dapp-kit'; import { getAllNetworks, getDefaultNetwork } from '@iota/iota-sdk/client'; import { QueryClient, QueryClientProvider } from '@tanstack/react-query'; +import { ReactQueryDevtools } from '@tanstack/react-query-devtools'; import { useState } from 'react'; -import { KioskClientProvider } from '@iota/core'; +import { KioskClientProvider, useLocalStorage } from '@iota/core'; import { growthbook } from '@/lib/utils'; import { ThemeProvider } from '@iota/core'; @@ -19,6 +20,8 @@ export function AppProviders({ children }: React.PropsWithChildren) { const [queryClient] = useState(() => new QueryClient()); const allNetworks = getAllNetworks(); const defaultNetwork = getDefaultNetwork(); + const [persistedNetwork] = useLocalStorage<string>('network_iota-dashboard', defaultNetwork); + function handleNetworkChange() { queryClient.resetQueries(); queryClient.clear(); @@ -28,7 +31,7 @@ export function AppProviders({ children }: React.PropsWithChildren) { <QueryClientProvider client={queryClient}> <IotaClientProvider networks={allNetworks} - defaultNetwork={defaultNetwork} + defaultNetwork={persistedNetwork} onNetworkChange={handleNetworkChange} > <KioskClientProvider> @@ -51,6 +54,7 @@ export function AppProviders({ children }: React.PropsWithChildren) { </WalletProvider> </KioskClientProvider> </IotaClientProvider> + <ReactQueryDevtools initialIsOpen={false} /> </QueryClientProvider> </GrowthBookProvider> ); diff --git a/apps/wallet-dashboard/stores/notificationStore.ts b/apps/wallet-dashboard/stores/notificationStore.ts deleted file mode 100644 index 08f48e32d1e..00000000000 --- a/apps/wallet-dashboard/stores/notificationStore.ts +++ /dev/null @@ -1,58 +0,0 @@ -// Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -import { indexGenerator } from '@/lib/utils'; -import { create } from 'zustand'; - -const indexGen = indexGenerator(200); - -export enum NotificationType { - Success = 'success', - Error = 'error', - Warning = 'warning', - Info = 'info', -} - -export type NotificationData = { - index: number; - message: string; - type: NotificationType; - duration: number; -}; - -interface NotificationsState { - notifications: NotificationData[]; - addNotification: (message: string, type?: NotificationType, duration?: number) => void; - clearNotification: (index: number) => void; -} - -export const useNotificationStore = create<NotificationsState>()((set) => ({ - notifications: [], - addNotification: ( - message: string, - type: NotificationType = NotificationType.Success, - duration: number = 3000, - ) => - set((state) => { - const index = indexGen.next().value; - - const newNotification: NotificationData = { - index, - message, - type, - duration, - }; - - return { - notifications: [...state.notifications, newNotification], - }; - }), - clearNotification: (index: number) => - set((state) => { - return { - notifications: [ - ...state.notifications.filter((notification) => notification.index !== index), - ], - }; - }), -})); diff --git a/apps/wallet-dashboard/tailwind.config.ts b/apps/wallet-dashboard/tailwind.config.ts index 2ca1f1c0824..720d34d31fd 100644 --- a/apps/wallet-dashboard/tailwind.config.ts +++ b/apps/wallet-dashboard/tailwind.config.ts @@ -3,7 +3,7 @@ import type { Config } from 'tailwindcss'; // Note: exception for the tailwind preset import -import uiKitResponsivePreset from '../../apps/ui-kit/src/lib/tailwind/responsive.preset'; +import uiKitResponsivePreset from '../../apps/ui-kit/src/lib/tailwind/responsive.presets'; export default { presets: [uiKitResponsivePreset], diff --git a/apps/wallet/package.json b/apps/wallet/package.json index 59ad3cd44ec..fd7d505cae2 100644 --- a/apps/wallet/package.json +++ b/apps/wallet/package.json @@ -94,13 +94,13 @@ "@growthbook/growthbook": "^1.0.0", "@growthbook/growthbook-react": "^1.0.0", "@headlessui/react": "^1.7.15", + "@iota/apps-ui-icons": "workspace:*", "@iota/apps-ui-kit": "workspace:*", "@iota/core": "workspace:*", "@iota/dapp-kit": "workspace:*", "@iota/iota-sdk": "workspace:*", "@iota/kiosk": "workspace:*", "@iota/ledgerjs-hw-app-iota": "workspace:*", - "@iota/ui-icons": "workspace:*", "@iota/wallet-standard": "workspace:*", "@ledgerhq/errors": "^6.12.4", "@ledgerhq/hw-transport": "^6.31.0", diff --git a/apps/wallet/src/background/account-sources/AccountSource.ts b/apps/wallet/src/background/account-sources/accountSource.ts similarity index 97% rename from apps/wallet/src/background/account-sources/AccountSource.ts rename to apps/wallet/src/background/account-sources/accountSource.ts index 47cf674fcb1..54d49e5be40 100644 --- a/apps/wallet/src/background/account-sources/AccountSource.ts +++ b/apps/wallet/src/background/account-sources/accountSource.ts @@ -9,7 +9,7 @@ import { clearEphemeralValue, getEphemeralValue, setEphemeralValue, -} from '../session-ephemeral-values'; +} from '../sessionEphemeralValues'; export enum AccountSourceType { Mnemonic = 'mnemonic', diff --git a/apps/wallet/src/background/account-sources/index.ts b/apps/wallet/src/background/account-sources/index.ts index e22684b6a71..9524d8d7e22 100644 --- a/apps/wallet/src/background/account-sources/index.ts +++ b/apps/wallet/src/background/account-sources/index.ts @@ -6,13 +6,13 @@ import { createMessage, type Message } from '_src/shared/messaging/messages'; import { isMethodPayload, type MethodPayload, -} from '_src/shared/messaging/messages/payloads/MethodPayload'; +} from '_src/shared/messaging/messages/payloads/methodPayload'; -import { type UiConnection } from '../connections/UiConnection'; +import { type UiConnection } from '../connections/uiConnection'; import { getDB } from '../db'; -import { AccountSourceType, type AccountSourceSerialized } from './AccountSource'; -import { MnemonicAccountSource } from './MnemonicAccountSource'; -import { SeedAccountSource } from './SeedAccountSource'; +import { AccountSourceType, type AccountSourceSerialized } from './accountSource'; +import { MnemonicAccountSource } from './mnemonicAccountSource'; +import { SeedAccountSource } from './seedAccountSource'; import { toEntropy } from '_src/shared/utils'; function toAccountSource(accountSource: AccountSourceSerialized) { diff --git a/apps/wallet/src/background/account-sources/MnemonicAccountSource.ts b/apps/wallet/src/background/account-sources/mnemonicAccountSource.ts similarity index 97% rename from apps/wallet/src/background/account-sources/MnemonicAccountSource.ts rename to apps/wallet/src/background/account-sources/mnemonicAccountSource.ts index 72820f4497b..4030b2ead83 100644 --- a/apps/wallet/src/background/account-sources/MnemonicAccountSource.ts +++ b/apps/wallet/src/background/account-sources/mnemonicAccountSource.ts @@ -19,16 +19,16 @@ import Dexie from 'dexie'; import { getAccountSources } from '.'; import { getAllAccounts } from '../accounts'; -import { MnemonicAccount, type MnemonicSerializedAccount } from '../accounts/MnemonicAccount'; -import { setupAutoLockAlarm } from '../auto-lock-accounts'; +import { MnemonicAccount, type MnemonicSerializedAccount } from '../accounts/mnemonicAccount'; +import { setupAutoLockAlarm } from '../autoLockAccounts'; import { backupDB, getDB } from '../db'; -import { makeUniqueKey } from '../storage-utils'; +import { makeUniqueKey } from '../storageUtils'; import { AccountSource, AccountSourceType, type AccountSourceSerialized, type AccountSourceSerializedUI, -} from './AccountSource'; +} from './accountSource'; import { accountSourcesEvents } from './events'; import { type MakeDerivationOptions, makeDerivationPath } from './bip44Path'; diff --git a/apps/wallet/src/background/account-sources/SeedAccountSource.ts b/apps/wallet/src/background/account-sources/seedAccountSource.ts similarity index 97% rename from apps/wallet/src/background/account-sources/SeedAccountSource.ts rename to apps/wallet/src/background/account-sources/seedAccountSource.ts index ff4b3e094b9..69a1a202e1e 100644 --- a/apps/wallet/src/background/account-sources/SeedAccountSource.ts +++ b/apps/wallet/src/background/account-sources/seedAccountSource.ts @@ -9,16 +9,16 @@ import Dexie from 'dexie'; import { getAccountSources } from '.'; import { getAllAccounts } from '../accounts'; -import { SeedAccount, type SeedSerializedAccount } from '../accounts/SeedAccount'; -import { setupAutoLockAlarm } from '../auto-lock-accounts'; +import { SeedAccount, type SeedSerializedAccount } from '../accounts/seedAccount'; +import { setupAutoLockAlarm } from '../autoLockAccounts'; import { backupDB, getDB } from '../db'; -import { makeUniqueKey } from '../storage-utils'; +import { makeUniqueKey } from '../storageUtils'; import { AccountSource, AccountSourceType, type AccountSourceSerialized, type AccountSourceSerializedUI, -} from './AccountSource'; +} from './accountSource'; import { accountSourcesEvents } from './events'; import { type MakeDerivationOptions, makeDerivationPath } from './bip44Path'; diff --git a/apps/wallet/src/background/accounts/Account.ts b/apps/wallet/src/background/accounts/account.ts similarity index 98% rename from apps/wallet/src/background/accounts/Account.ts rename to apps/wallet/src/background/accounts/account.ts index 9ab33a5d47c..59549cb4de6 100644 --- a/apps/wallet/src/background/accounts/Account.ts +++ b/apps/wallet/src/background/accounts/account.ts @@ -6,13 +6,13 @@ import { type Serializable } from '_src/shared/cryptography/keystore'; import { toSerializedSignature, type Keypair } from '@iota/iota-sdk/cryptography'; import { blake2b } from '@noble/hashes/blake2b'; -import { setupAutoLockAlarm } from '../auto-lock-accounts'; +import { setupAutoLockAlarm } from '../autoLockAccounts'; import { getDB } from '../db'; import { clearEphemeralValue, getEphemeralValue, setEphemeralValue, -} from '../session-ephemeral-values'; +} from '../sessionEphemeralValues'; import { accountsEvents } from './events'; export enum AccountType { diff --git a/apps/wallet/src/background/accounts/ImportedAccount.ts b/apps/wallet/src/background/accounts/importedAccount.ts similarity index 99% rename from apps/wallet/src/background/accounts/ImportedAccount.ts rename to apps/wallet/src/background/accounts/importedAccount.ts index ac44a31282f..303dfed2de3 100644 --- a/apps/wallet/src/background/accounts/ImportedAccount.ts +++ b/apps/wallet/src/background/accounts/importedAccount.ts @@ -13,7 +13,7 @@ import { type SerializedAccount, type SerializedUIAccount, type SigningAccount, -} from './Account'; +} from './account'; type SessionStorageData = { keyPair: string }; type EncryptedData = { keyPair: string }; diff --git a/apps/wallet/src/background/accounts/index.ts b/apps/wallet/src/background/accounts/index.ts index 665a7b9eddb..189b6e1f1b4 100644 --- a/apps/wallet/src/background/accounts/index.ts +++ b/apps/wallet/src/background/accounts/index.ts @@ -6,30 +6,30 @@ import { createMessage, type Message } from '_src/shared/messaging/messages'; import { isMethodPayload, type MethodPayload, -} from '_src/shared/messaging/messages/payloads/MethodPayload'; +} from '_src/shared/messaging/messages/payloads/methodPayload'; import { type WalletStatusChange } from '_src/shared/messaging/messages/payloads/wallet-status-change'; import { fromB64 } from '@iota/iota-sdk/utils'; import Dexie from 'dexie'; import { getAccountSourceByID } from '../account-sources'; import { accountSourcesEvents } from '../account-sources/events'; -import { MnemonicAccountSource } from '../account-sources/MnemonicAccountSource'; -import { SeedAccountSource } from '../account-sources/SeedAccountSource'; -import { type UiConnection } from '../connections/UiConnection'; +import { MnemonicAccountSource } from '../account-sources/mnemonicAccountSource'; +import { SeedAccountSource } from '../account-sources/seedAccountSource'; +import { type UiConnection } from '../connections/uiConnection'; import { backupDB, getDB } from '../db'; -import { makeUniqueKey } from '../storage-utils'; +import { makeUniqueKey } from '../storageUtils'; import { AccountType, isKeyPairExportableAccount, isPasswordUnLockable, isSigningAccount, type SerializedAccount, -} from './Account'; +} from './account'; import { accountsEvents } from './events'; -import { ImportedAccount } from './ImportedAccount'; -import { LedgerAccount } from './LedgerAccount'; -import { MnemonicAccount } from './MnemonicAccount'; -import { SeedAccount } from './SeedAccount'; +import { ImportedAccount } from './importedAccount'; +import { LedgerAccount } from './ledgerAccount'; +import { MnemonicAccount } from './mnemonicAccount'; +import { SeedAccount } from './seedAccount'; function toAccount(account: SerializedAccount) { if (MnemonicAccount.isOfType(account)) { diff --git a/apps/wallet/src/background/accounts/isMainAccount.ts b/apps/wallet/src/background/accounts/isMainAccount.ts index 0e385c52cf0..9fd860bdbc5 100644 --- a/apps/wallet/src/background/accounts/isMainAccount.ts +++ b/apps/wallet/src/background/accounts/isMainAccount.ts @@ -1,11 +1,11 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { isLedgerAccountSerializedUI } from '_src/background/accounts/LedgerAccount'; -import { isMnemonicSerializedUiAccount } from '_src/background/accounts/MnemonicAccount'; -import { isSeedSerializedUiAccount } from '_src/background/accounts/SeedAccount'; +import { isLedgerAccountSerializedUI } from '_src/background/accounts/ledgerAccount'; +import { isMnemonicSerializedUiAccount } from '_src/background/accounts/mnemonicAccount'; +import { isSeedSerializedUiAccount } from '_src/background/accounts/seedAccount'; import { parseDerivationPath } from '_src/background/account-sources/bip44Path'; -import type { SerializedUIAccount } from '_src/background/accounts/Account'; +import type { SerializedUIAccount } from '_src/background/accounts/account'; export function isMainAccount(account: SerializedUIAccount | null) { if (!account) { diff --git a/apps/wallet/src/background/accounts/LedgerAccount.ts b/apps/wallet/src/background/accounts/ledgerAccount.ts similarity index 99% rename from apps/wallet/src/background/accounts/LedgerAccount.ts rename to apps/wallet/src/background/accounts/ledgerAccount.ts index fbb8ca111c2..477bf379ba9 100644 --- a/apps/wallet/src/background/accounts/LedgerAccount.ts +++ b/apps/wallet/src/background/accounts/ledgerAccount.ts @@ -10,7 +10,7 @@ import { type PasswordUnlockableAccount, type SerializedAccount, type SerializedUIAccount, -} from './Account'; +} from './account'; export interface LedgerAccountSerialized extends SerializedAccount { type: AccountType.LedgerDerived; diff --git a/apps/wallet/src/background/accounts/MnemonicAccount.ts b/apps/wallet/src/background/accounts/mnemonicAccount.ts similarity index 98% rename from apps/wallet/src/background/accounts/MnemonicAccount.ts rename to apps/wallet/src/background/accounts/mnemonicAccount.ts index 0167be027fd..e3c93c843f3 100644 --- a/apps/wallet/src/background/accounts/MnemonicAccount.ts +++ b/apps/wallet/src/background/accounts/mnemonicAccount.ts @@ -5,7 +5,7 @@ import { fromExportedKeypair } from '_src/shared/utils'; import { type Keypair } from '@iota/iota-sdk/cryptography'; -import { MnemonicAccountSource } from '../account-sources/MnemonicAccountSource'; +import { MnemonicAccountSource } from '../account-sources/mnemonicAccountSource'; import { Account, AccountType, @@ -14,7 +14,7 @@ import { type SerializedAccount, type SerializedUIAccount, type SigningAccount, -} from './Account'; +} from './account'; export interface MnemonicSerializedAccount extends SerializedAccount { type: AccountType.MnemonicDerived; diff --git a/apps/wallet/src/background/accounts/SeedAccount.ts b/apps/wallet/src/background/accounts/seedAccount.ts similarity index 98% rename from apps/wallet/src/background/accounts/SeedAccount.ts rename to apps/wallet/src/background/accounts/seedAccount.ts index e2667871adb..0f88a205117 100644 --- a/apps/wallet/src/background/accounts/SeedAccount.ts +++ b/apps/wallet/src/background/accounts/seedAccount.ts @@ -4,7 +4,7 @@ import { fromExportedKeypair } from '_src/shared/utils'; import { type Keypair } from '@iota/iota-sdk/cryptography'; -import { SeedAccountSource } from '../account-sources/SeedAccountSource'; +import { SeedAccountSource } from '../account-sources/seedAccountSource'; import { Account, AccountType, @@ -13,7 +13,7 @@ import { type SerializedAccount, type SerializedUIAccount, type SigningAccount, -} from './Account'; +} from './account'; export interface SeedSerializedAccount extends SerializedAccount { type: AccountType.SeedDerived; diff --git a/apps/wallet/src/background/Alarms.ts b/apps/wallet/src/background/alarms.ts similarity index 100% rename from apps/wallet/src/background/Alarms.ts rename to apps/wallet/src/background/alarms.ts diff --git a/apps/wallet/src/background/auto-lock-accounts.ts b/apps/wallet/src/background/autoLockAccounts.ts similarity index 97% rename from apps/wallet/src/background/auto-lock-accounts.ts rename to apps/wallet/src/background/autoLockAccounts.ts index 474a6846eed..d7287beb0a5 100644 --- a/apps/wallet/src/background/auto-lock-accounts.ts +++ b/apps/wallet/src/background/autoLockAccounts.ts @@ -4,7 +4,7 @@ import { throttle } from 'throttle-debounce'; -import Alarms from './Alarms'; +import Alarms from './alarms'; import { getDB, SETTINGS_KEYS } from './db'; export async function getAutoLockMinutes() { diff --git a/apps/wallet/src/background/connections/Connection.ts b/apps/wallet/src/background/connections/connection.ts similarity index 93% rename from apps/wallet/src/background/connections/Connection.ts rename to apps/wallet/src/background/connections/connection.ts index d7023c9478e..ad0992537c0 100644 --- a/apps/wallet/src/background/connections/Connection.ts +++ b/apps/wallet/src/background/connections/connection.ts @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import type { Message } from '_messages'; -import { PortStream } from '_messaging/PortStream'; +import { PortStream } from '_src/shared/messaging/portStream'; import { map, take } from 'rxjs'; import type { Runtime } from 'webextension-polyfill'; diff --git a/apps/wallet/src/background/connections/ContentScriptConnection.ts b/apps/wallet/src/background/connections/contentScriptConnection.ts similarity index 94% rename from apps/wallet/src/background/connections/ContentScriptConnection.ts rename to apps/wallet/src/background/connections/contentScriptConnection.ts index 8067c3ee7e5..10f50acc3e2 100644 --- a/apps/wallet/src/background/connections/ContentScriptConnection.ts +++ b/apps/wallet/src/background/connections/contentScriptConnection.ts @@ -4,10 +4,10 @@ import { createMessage } from '_messages'; import type { Message } from '_messages'; -import type { PortChannelName } from '_messaging/PortChannelName'; +import type { PortChannelName } from '_src/shared/messaging/portChannelName'; import { isBasePayload, type ErrorPayload } from '_payloads'; -import { isGetAccount } from '_payloads/account/GetAccount'; -import type { GetAccountResponse } from '_payloads/account/GetAccountResponse'; +import { isGetAccount } from '_src/shared/messaging/messages/payloads/account/getAccount'; +import type { GetAccountResponse } from '_src/shared/messaging/messages/payloads/account/getAccountResponse'; import type { SetNetworkPayload } from '_payloads/network'; import { isAcquirePermissionsRequest, @@ -23,19 +23,19 @@ import { type ExecuteTransactionResponse, type SignTransactionResponse, } from '_payloads/transactions'; -import Permissions from '_src/background/Permissions'; -import Transactions from '_src/background/Transactions'; +import Permissions from '_src/background/permissions'; +import Transactions from '_src/background/transactions'; import { isSignMessageRequest, type SignMessageRequest, -} from '_src/shared/messaging/messages/payloads/transactions/SignMessage'; +} from '_src/shared/messaging/messages/payloads/transactions/signMessage'; import { type IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; import type { Runtime } from 'webextension-polyfill'; import { type SignedTransaction } from '@iota/wallet-standard'; import { getAccountsStatusData } from '../accounts'; -import NetworkEnv from '../NetworkEnv'; -import { Connection } from './Connection'; +import NetworkEnv from '../networkEnv'; +import { Connection } from './connection'; export class ContentScriptConnection extends Connection { public static readonly CHANNEL: PortChannelName = 'iota_content<->background'; diff --git a/apps/wallet/src/background/connections/index.ts b/apps/wallet/src/background/connections/index.ts index d6b5c3c591b..62ee3912a24 100644 --- a/apps/wallet/src/background/connections/index.ts +++ b/apps/wallet/src/background/connections/index.ts @@ -7,12 +7,12 @@ import type { SetNetworkPayload } from '_payloads/network'; import type { Permission } from '_payloads/permissions'; import type { WalletStatusChange, WalletStatusChangePayload } from '_payloads/wallet-status-change'; import type { NetworkEnvType } from '@iota/core'; -import { type UIAccessibleEntityType } from '_src/shared/messaging/messages/payloads/MethodPayload'; +import { type UIAccessibleEntityType } from '_src/shared/messaging/messages/payloads/methodPayload'; import Browser from 'webextension-polyfill'; -import type { Connection } from './Connection'; -import { ContentScriptConnection } from './ContentScriptConnection'; -import { UiConnection } from './UiConnection'; +import type { Connection } from './connection'; +import { ContentScriptConnection } from './contentScriptConnection'; +import { UiConnection } from './uiConnection'; const appOrigin = new URL(Browser.runtime.getURL('')).origin; diff --git a/apps/wallet/src/background/connections/UiConnection.ts b/apps/wallet/src/background/connections/uiConnection.ts similarity index 92% rename from apps/wallet/src/background/connections/UiConnection.ts rename to apps/wallet/src/background/connections/uiConnection.ts index 63cdbf79e19..f5115b1f58f 100644 --- a/apps/wallet/src/background/connections/UiConnection.ts +++ b/apps/wallet/src/background/connections/uiConnection.ts @@ -4,27 +4,27 @@ import { createMessage } from '_messages'; import type { Message } from '_messages'; -import type { PortChannelName } from '_messaging/PortChannelName'; +import type { PortChannelName } from '_src/shared/messaging/portChannelName'; import { isBasePayload, type ErrorPayload } from '_payloads'; import type { LoadedFeaturesPayload } from '_payloads/feature-gating'; import { isSetNetworkPayload, type SetNetworkPayload } from '_payloads/network'; import { isGetPermissionRequests, isPermissionResponse } from '_payloads/permissions'; import type { Permission, PermissionRequests } from '_payloads/permissions'; -import { isDisconnectApp } from '_payloads/permissions/DisconnectApp'; +import { isDisconnectApp } from '_src/shared/messaging/messages/payloads/permissions/disconnectApp'; import type { UpdateActiveOrigin } from '_payloads/tabs/updateActiveOrigin'; -import type { ApprovalRequest } from '_payloads/transactions/ApprovalRequest'; -import { isGetTransactionRequests } from '_payloads/transactions/ui/GetTransactionRequests'; -import type { GetTransactionRequestsResponse } from '_payloads/transactions/ui/GetTransactionRequestsResponse'; -import { isTransactionRequestResponse } from '_payloads/transactions/ui/TransactionRequestResponse'; -import Permissions from '_src/background/Permissions'; -import Tabs from '_src/background/Tabs'; -import Transactions from '_src/background/Transactions'; +import type { ApprovalRequest } from '_src/shared/messaging/messages/payloads/transactions/approvalRequest'; +import { isGetTransactionRequests } from '_src/shared/messaging/messages/payloads/transactions/ui/getTransactionRequests'; +import type { GetTransactionRequestsResponse } from '_src/shared/messaging/messages/payloads/transactions/ui/getTransactionRequestsResponse'; +import { isTransactionRequestResponse } from '_src/shared/messaging/messages/payloads/transactions/ui/transactionRequestResponse'; +import Permissions from '_src/background/permissions'; +import Tabs from '_src/background/tabs'; +import Transactions from '_src/background/transactions'; import { growthbook } from '_src/shared/experimentation/features'; import { isMethodPayload, type MethodPayload, type UIAccessibleEntityType, -} from '_src/shared/messaging/messages/payloads/MethodPayload'; +} from '_src/shared/messaging/messages/payloads/methodPayload'; import { toEntropy } from '_src/shared/utils'; import Dexie from 'dexie'; import { BehaviorSubject, filter, switchMap, takeUntil } from 'rxjs'; @@ -37,7 +37,7 @@ import { getAllSerializedUIAccountSources, } from '../account-sources'; import { accountSourcesEvents } from '../account-sources/events'; -import { MnemonicAccountSource } from '../account-sources/MnemonicAccountSource'; +import { MnemonicAccountSource } from '../account-sources/mnemonicAccountSource'; import { accountsHandleUIMessage, addNewAccounts, @@ -45,16 +45,16 @@ import { getAllSerializedUIAccounts, } from '../accounts'; import { accountsEvents } from '../accounts/events'; -import { getAutoLockMinutes, notifyUserActive, setAutoLockMinutes } from '../auto-lock-accounts'; +import { getAutoLockMinutes, notifyUserActive, setAutoLockMinutes } from '../autoLockAccounts'; import { backupDB, getDB, SETTINGS_KEYS } from '../db'; -import { clearStatus, doMigration, getStatus } from '../storage-migration'; -import NetworkEnv from '../NetworkEnv'; -import { Connection } from './Connection'; -import { SeedAccountSource } from '../account-sources/SeedAccountSource'; -import { AccountSourceType } from '../account-sources/AccountSource'; +import { clearStatus, doMigration, getStatus } from '../storageMigration'; +import NetworkEnv from '../networkEnv'; +import { Connection } from './connection'; +import { SeedAccountSource } from '../account-sources/seedAccountSource'; +import { AccountSourceType } from '../account-sources/accountSource'; import { isDeriveBipPathAccountsFinder, isPersistAccountsFinder } from '_payloads/accounts-finder'; -import type { SerializedAccount } from '../accounts/Account'; -import { LedgerAccount } from '../accounts/LedgerAccount'; +import type { SerializedAccount } from '../accounts/account'; +import { LedgerAccount } from '../accounts/ledgerAccount'; export class UiConnection extends Connection { public static readonly CHANNEL: PortChannelName = 'iota_ui<->background'; diff --git a/apps/wallet/src/background/db.ts b/apps/wallet/src/background/db.ts index 71ecbedcc82..43acb34afed 100644 --- a/apps/wallet/src/background/db.ts +++ b/apps/wallet/src/background/db.ts @@ -5,10 +5,10 @@ import Dexie, { type Table } from 'dexie'; import { exportDB, importDB } from 'dexie-export-import'; -import { type AccountSourceSerialized } from './account-sources/AccountSource'; -import { type SerializedAccount } from './accounts/Account'; +import { type AccountSourceSerialized } from './account-sources/accountSource'; +import { type SerializedAccount } from './accounts/account'; import { captureException } from './sentry'; -import { getFromLocalStorage, setToLocalStorage } from './storage-utils'; +import { getFromLocalStorage, setToLocalStorage } from './storageUtils'; const DB_NAME = 'IotaWallet DB'; const DB_LOCAL_STORAGE_BACKUP_KEY = 'indexed-db-backup'; diff --git a/apps/wallet/src/background/index.ts b/apps/wallet/src/background/index.ts index 338679105f2..4fb42a4ae0a 100644 --- a/apps/wallet/src/background/index.ts +++ b/apps/wallet/src/background/index.ts @@ -11,12 +11,12 @@ import { lockAllAccountSources } from './account-sources'; import { accountSourcesEvents } from './account-sources/events'; import { getAccountsStatusData, getAllAccounts, lockAllAccounts } from './accounts'; import { accountsEvents } from './accounts/events'; -import Alarms, { AUTO_LOCK_ALARM_NAME, CLEAN_UP_ALARM_NAME } from './Alarms'; +import Alarms, { AUTO_LOCK_ALARM_NAME, CLEAN_UP_ALARM_NAME } from './alarms'; import { Connections } from './connections'; -import NetworkEnv from './NetworkEnv'; -import Permissions from './Permissions'; +import NetworkEnv from './networkEnv'; +import Permissions from './permissions'; import { initSentry } from './sentry'; -import Transactions from './Transactions'; +import Transactions from './transactions'; growthbook.loadFeatures().catch(() => { // silence the error diff --git a/apps/wallet/src/background/NetworkEnv.ts b/apps/wallet/src/background/networkEnv.ts similarity index 100% rename from apps/wallet/src/background/NetworkEnv.ts rename to apps/wallet/src/background/networkEnv.ts diff --git a/apps/wallet/src/background/Permissions.ts b/apps/wallet/src/background/permissions.ts similarity index 98% rename from apps/wallet/src/background/Permissions.ts rename to apps/wallet/src/background/permissions.ts index 42c85b8cdf4..6fa604b55a0 100644 --- a/apps/wallet/src/background/Permissions.ts +++ b/apps/wallet/src/background/permissions.ts @@ -10,9 +10,9 @@ import type { Observable } from 'rxjs'; import { v4 as uuidV4 } from 'uuid'; import Browser from 'webextension-polyfill'; -import type { ContentScriptConnection } from './connections/ContentScriptConnection'; -import Tabs from './Tabs'; -import { Window } from './Window'; +import type { ContentScriptConnection } from './connections/contentScriptConnection'; +import Tabs from './tabs'; +import { Window } from './window'; const PERMISSIONS_STORAGE_KEY = 'permissions'; const PERMISSION_UI_URL = `${Browser.runtime.getURL('ui.html')}#/dapp/connect/`; diff --git a/apps/wallet/src/background/sentry.ts b/apps/wallet/src/background/sentry.ts index e95d0f9c70b..b5f70562c84 100644 --- a/apps/wallet/src/background/sentry.ts +++ b/apps/wallet/src/background/sentry.ts @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import { growthbook } from '_src/shared/experimentation/features'; -import { getSentryConfig } from '_src/shared/sentry-config'; +import { getSentryConfig } from '_src/shared/sentryConfig'; import * as Sentry from '@sentry/browser'; import { Feature } from '@iota/core'; diff --git a/apps/wallet/src/background/session-ephemeral-values.ts b/apps/wallet/src/background/sessionEphemeralValues.ts similarity index 94% rename from apps/wallet/src/background/session-ephemeral-values.ts rename to apps/wallet/src/background/sessionEphemeralValues.ts index ae820d67b13..b5a9c328f13 100644 --- a/apps/wallet/src/background/session-ephemeral-values.ts +++ b/apps/wallet/src/background/sessionEphemeralValues.ts @@ -7,7 +7,7 @@ import { getEncryptedFromSessionStorage, removeFromSessionStorage, setToSessionStorageEncrypted, -} from '_src/background/storage-utils'; +} from '_src/background/storageUtils'; export function getEphemeralValue<T extends Serializable>(id: string) { return getEncryptedFromSessionStorage<T>(id); diff --git a/apps/wallet/src/background/storage-migration.ts b/apps/wallet/src/background/storageMigration.ts similarity index 93% rename from apps/wallet/src/background/storage-migration.ts rename to apps/wallet/src/background/storageMigration.ts index a78813e380e..5c11b9bda19 100644 --- a/apps/wallet/src/background/storage-migration.ts +++ b/apps/wallet/src/background/storageMigration.ts @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { getFromLocalStorage } from './storage-utils'; +import { getFromLocalStorage } from './storageUtils'; export type Status = 'required' | 'inProgress' | 'ready'; diff --git a/apps/wallet/src/background/storage-utils.ts b/apps/wallet/src/background/storageUtils.ts similarity index 100% rename from apps/wallet/src/background/storage-utils.ts rename to apps/wallet/src/background/storageUtils.ts diff --git a/apps/wallet/src/background/Tabs.ts b/apps/wallet/src/background/tabs.ts similarity index 100% rename from apps/wallet/src/background/Tabs.ts rename to apps/wallet/src/background/tabs.ts diff --git a/apps/wallet/src/background/Transactions.ts b/apps/wallet/src/background/transactions.ts similarity index 92% rename from apps/wallet/src/background/Transactions.ts rename to apps/wallet/src/background/transactions.ts index 6c696bf0b12..262b546c558 100644 --- a/apps/wallet/src/background/Transactions.ts +++ b/apps/wallet/src/background/transactions.ts @@ -5,18 +5,18 @@ import { type ApprovalRequest, type TransactionDataType, -} from '_payloads/transactions/ApprovalRequest'; -import type { IotaSignTransactionSerialized } from '_payloads/transactions/ExecuteTransactionRequest'; -import { type SignMessageRequest } from '_payloads/transactions/SignMessage'; -import type { TransactionRequestResponse } from '_payloads/transactions/ui/TransactionRequestResponse'; -import type { ContentScriptConnection } from '_src/background/connections/ContentScriptConnection'; +} from '_src/shared/messaging/messages/payloads/transactions/approvalRequest'; +import type { IotaSignTransactionSerialized } from '_src/shared/messaging/messages/payloads/transactions/executeTransactionRequest'; +import { type SignMessageRequest } from '_src/shared/messaging/messages/payloads/transactions/signMessage'; +import type { TransactionRequestResponse } from '_src/shared/messaging/messages/payloads/transactions/ui/transactionRequestResponse'; +import type { ContentScriptConnection } from '_src/background/connections/contentScriptConnection'; import { type IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; import { type SignedTransaction, type IotaSignPersonalMessageOutput } from '@iota/wallet-standard'; import { filter, lastValueFrom, map, race, Subject, take } from 'rxjs'; import { v4 as uuidV4 } from 'uuid'; import Browser from 'webextension-polyfill'; -import { Window } from './Window'; +import { Window } from './window'; const STALE_TRANSACTION_MILLISECONDS = 1000 * 60 * 60 * 3; // 3 hours const TX_STORE_KEY = 'transactions'; diff --git a/apps/wallet/src/background/Window.ts b/apps/wallet/src/background/window.ts similarity index 100% rename from apps/wallet/src/background/Window.ts rename to apps/wallet/src/background/window.ts diff --git a/apps/wallet/src/content-script/index.ts b/apps/wallet/src/content-script/index.ts index 9b3662e815d..11ce586b293 100644 --- a/apps/wallet/src/content-script/index.ts +++ b/apps/wallet/src/content-script/index.ts @@ -2,8 +2,8 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { injectDappInterface } from './interface-inject'; -import { setupMessagesProxy } from './messages-proxy'; +import { injectDappInterface } from './interfaceInject'; +import { setupMessagesProxy } from './messagesProxy'; injectDappInterface(); setupMessagesProxy(); diff --git a/apps/wallet/src/content-script/interface-inject.ts b/apps/wallet/src/content-script/interfaceInject.ts similarity index 100% rename from apps/wallet/src/content-script/interface-inject.ts rename to apps/wallet/src/content-script/interfaceInject.ts diff --git a/apps/wallet/src/content-script/messages-proxy.ts b/apps/wallet/src/content-script/messagesProxy.ts similarity index 89% rename from apps/wallet/src/content-script/messages-proxy.ts rename to apps/wallet/src/content-script/messagesProxy.ts index a366e0b5f64..198a4317f20 100644 --- a/apps/wallet/src/content-script/messages-proxy.ts +++ b/apps/wallet/src/content-script/messagesProxy.ts @@ -2,8 +2,8 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { PortStream } from '_messaging/PortStream'; -import { WindowMessageStream } from '_messaging/WindowMessageStream'; +import { PortStream } from '_src/shared/messaging/portStream'; +import { WindowMessageStream } from '_src/shared/messaging/windowMessageStream'; import type { Message } from '_src/shared/messaging/messages'; import { generateWalletMessageStreamIdentifiers } from '_src/shared/utils/generateWalletMessageStreamIdentifiers'; import { take } from 'rxjs'; diff --git a/apps/wallet/src/dapp-interface/index.ts b/apps/wallet/src/dapp-interface/index.ts index 1c27c20b54e..b79ccc1ed16 100644 --- a/apps/wallet/src/dapp-interface/index.ts +++ b/apps/wallet/src/dapp-interface/index.ts @@ -4,6 +4,6 @@ import { registerWallet } from '@iota/wallet-standard'; -import { IotaWallet } from './WalletStandardInterface'; +import { IotaWallet } from './walletStandardInterface'; registerWallet(new IotaWallet()); diff --git a/apps/wallet/src/dapp-interface/WalletStandardInterface.ts b/apps/wallet/src/dapp-interface/walletStandardInterface.ts similarity index 98% rename from apps/wallet/src/dapp-interface/WalletStandardInterface.ts rename to apps/wallet/src/dapp-interface/walletStandardInterface.ts index 5490718fdbc..03aaf9a10d2 100644 --- a/apps/wallet/src/dapp-interface/WalletStandardInterface.ts +++ b/apps/wallet/src/dapp-interface/walletStandardInterface.ts @@ -3,10 +3,10 @@ // SPDX-License-Identifier: Apache-2.0 import { createMessage } from '_messages'; -import { WindowMessageStream } from '_messaging/WindowMessageStream'; +import { WindowMessageStream } from '_src/shared/messaging/windowMessageStream'; import type { BasePayload, Payload } from '_payloads'; -import type { GetAccount } from '_payloads/account/GetAccount'; -import type { GetAccountResponse } from '_payloads/account/GetAccountResponse'; +import type { GetAccount } from '_src/shared/messaging/messages/payloads/account/getAccount'; +import type { GetAccountResponse } from '_src/shared/messaging/messages/payloads/account/getAccountResponse'; import type { SetNetworkPayload } from '_payloads/network'; import { ALL_PERMISSION_TYPES, @@ -22,7 +22,7 @@ import type { SignTransactionResponse, } from '_payloads/transactions'; import { getCustomNetwork, type NetworkEnvType } from '@iota/core'; -import { type SignMessageRequest } from '_src/shared/messaging/messages/payloads/transactions/SignMessage'; +import { type SignMessageRequest } from '_src/shared/messaging/messages/payloads/transactions/signMessage'; import { isWalletStatusChangePayload } from '_src/shared/messaging/messages/payloads/wallet-status-change'; import { getNetwork, Network, type ChainType } from '@iota/iota-sdk/client'; import { fromB64, toB64 } from '@iota/iota-sdk/utils'; diff --git a/apps/wallet/src/shared/iota-client.ts b/apps/wallet/src/shared/iotaClient.ts similarity index 96% rename from apps/wallet/src/shared/iota-client.ts rename to apps/wallet/src/shared/iotaClient.ts index 20af64d663b..8c0d49d7d8f 100644 --- a/apps/wallet/src/shared/iota-client.ts +++ b/apps/wallet/src/shared/iotaClient.ts @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import networkEnv from '_src/background/NetworkEnv'; +import networkEnv from '_src/background/networkEnv'; import { type NetworkEnvType, SentryHttpTransport } from '@iota/core'; import { getNetwork, Network, IotaClient, IotaHTTPTransport } from '@iota/iota-sdk/client'; diff --git a/apps/wallet/src/shared/messaging/messages/index.ts b/apps/wallet/src/shared/messaging/messages/index.ts index bfb0e742de3..82b4b20ae1e 100644 --- a/apps/wallet/src/shared/messaging/messages/index.ts +++ b/apps/wallet/src/shared/messaging/messages/index.ts @@ -2,4 +2,4 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export * from './Message'; +export * from './message'; diff --git a/apps/wallet/src/shared/messaging/messages/Message.ts b/apps/wallet/src/shared/messaging/messages/message.ts similarity index 88% rename from apps/wallet/src/shared/messaging/messages/Message.ts rename to apps/wallet/src/shared/messaging/messages/message.ts index 369f7e9b9cd..23e65cc6612 100644 --- a/apps/wallet/src/shared/messaging/messages/Message.ts +++ b/apps/wallet/src/shared/messaging/messages/message.ts @@ -4,7 +4,7 @@ import { v4 as uuidV4 } from 'uuid'; -import type { Payload } from './payloads/Payload'; +import type { Payload } from './payloads/payload'; export type Message = { id: string; diff --git a/apps/wallet/src/shared/messaging/messages/payloads/account/GetAccount.ts b/apps/wallet/src/shared/messaging/messages/payloads/account/getAccount.ts similarity index 100% rename from apps/wallet/src/shared/messaging/messages/payloads/account/GetAccount.ts rename to apps/wallet/src/shared/messaging/messages/payloads/account/getAccount.ts diff --git a/apps/wallet/src/shared/messaging/messages/payloads/account/GetAccountResponse.ts b/apps/wallet/src/shared/messaging/messages/payloads/account/getAccountResponse.ts similarity index 100% rename from apps/wallet/src/shared/messaging/messages/payloads/account/GetAccountResponse.ts rename to apps/wallet/src/shared/messaging/messages/payloads/account/getAccountResponse.ts diff --git a/apps/wallet/src/shared/messaging/messages/payloads/accounts-finder/DeriveBipPathAccountsFinder.ts b/apps/wallet/src/shared/messaging/messages/payloads/accounts-finder/deriveBipPathAccountsFinder.ts similarity index 100% rename from apps/wallet/src/shared/messaging/messages/payloads/accounts-finder/DeriveBipPathAccountsFinder.ts rename to apps/wallet/src/shared/messaging/messages/payloads/accounts-finder/deriveBipPathAccountsFinder.ts diff --git a/apps/wallet/src/shared/messaging/messages/payloads/accounts-finder/index.ts b/apps/wallet/src/shared/messaging/messages/payloads/accounts-finder/index.ts index db6a662751c..f3012894048 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/accounts-finder/index.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/accounts-finder/index.ts @@ -1,6 +1,6 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export * from './DeriveBipPathAccountsFinder'; -export * from './PersistAccountsFinder'; +export * from './deriveBipPathAccountsFinder'; +export * from './persistAccountsFinder'; export * from './types'; diff --git a/apps/wallet/src/shared/messaging/messages/payloads/accounts-finder/PersistAccountsFinder.ts b/apps/wallet/src/shared/messaging/messages/payloads/accounts-finder/persistAccountsFinder.ts similarity index 100% rename from apps/wallet/src/shared/messaging/messages/payloads/accounts-finder/PersistAccountsFinder.ts rename to apps/wallet/src/shared/messaging/messages/payloads/accounts-finder/persistAccountsFinder.ts diff --git a/apps/wallet/src/shared/messaging/messages/payloads/BasePayload.ts b/apps/wallet/src/shared/messaging/messages/payloads/basePayload.ts similarity index 96% rename from apps/wallet/src/shared/messaging/messages/payloads/BasePayload.ts rename to apps/wallet/src/shared/messaging/messages/payloads/basePayload.ts index 4de18c06bdc..52ab6d2e254 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/BasePayload.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/basePayload.ts @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import type { Payload } from './Payload'; +import type { Payload } from './payload'; export type PayloadType = | 'permission-request' diff --git a/apps/wallet/src/shared/messaging/messages/payloads/ErrorPayload.ts b/apps/wallet/src/shared/messaging/messages/payloads/errorPayload.ts similarity index 89% rename from apps/wallet/src/shared/messaging/messages/payloads/ErrorPayload.ts rename to apps/wallet/src/shared/messaging/messages/payloads/errorPayload.ts index bfd70d93483..95187f90d56 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/ErrorPayload.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/errorPayload.ts @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import type { Payload } from './Payload'; +import type { Payload } from './payload'; export interface ErrorPayload { error: true; diff --git a/apps/wallet/src/shared/messaging/messages/payloads/index.ts b/apps/wallet/src/shared/messaging/messages/payloads/index.ts index 5a8e3e4bb03..d9095538c30 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/index.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/index.ts @@ -2,6 +2,6 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export * from './BasePayload'; -export * from './ErrorPayload'; -export * from './Payload'; +export * from './basePayload'; +export * from './errorPayload'; +export * from './payload'; diff --git a/apps/wallet/src/shared/messaging/messages/payloads/MethodPayload.ts b/apps/wallet/src/shared/messaging/messages/payloads/methodPayload.ts similarity index 93% rename from apps/wallet/src/shared/messaging/messages/payloads/MethodPayload.ts rename to apps/wallet/src/shared/messaging/messages/payloads/methodPayload.ts index 8672bc8653d..d4de79ecdc7 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/MethodPayload.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/methodPayload.ts @@ -5,12 +5,12 @@ import type { AccountSourceType, AccountSourceSerializedUI, -} from '_src/background/account-sources/AccountSource'; -import type { AccountType, SerializedUIAccount } from '_src/background/accounts/Account'; -import { type Status } from '_src/background/storage-migration'; +} from '_src/background/account-sources/accountSource'; +import type { AccountType, SerializedUIAccount } from '_src/background/accounts/account'; +import { type Status } from '_src/background/storageMigration'; -import { isBasePayload } from './BasePayload'; -import type { Payload } from './Payload'; +import { isBasePayload } from './basePayload'; +import type { Payload } from './payload'; export type UIAccessibleEntityType = 'accountSources' | 'accounts'; export type LedgerAccountsPublicKeys = { diff --git a/apps/wallet/src/shared/messaging/messages/payloads/Payload.ts b/apps/wallet/src/shared/messaging/messages/payloads/payload.ts similarity index 63% rename from apps/wallet/src/shared/messaging/messages/payloads/Payload.ts rename to apps/wallet/src/shared/messaging/messages/payloads/payload.ts index c498bf73bbe..0a852ad8257 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/Payload.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/payload.ts @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import type { BasePayload } from './BasePayload'; -import type { ErrorPayload } from './ErrorPayload'; +import type { BasePayload } from './basePayload'; +import type { ErrorPayload } from './errorPayload'; export type Payload = BasePayload | ErrorPayload; diff --git a/apps/wallet/src/shared/messaging/messages/payloads/permissions/AcquirePermissionsRequest.ts b/apps/wallet/src/shared/messaging/messages/payloads/permissions/acquirePermissionsRequest.ts similarity index 91% rename from apps/wallet/src/shared/messaging/messages/payloads/permissions/AcquirePermissionsRequest.ts rename to apps/wallet/src/shared/messaging/messages/payloads/permissions/acquirePermissionsRequest.ts index e78a952a67f..e34ec0f864b 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/permissions/AcquirePermissionsRequest.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/permissions/acquirePermissionsRequest.ts @@ -5,7 +5,7 @@ import { isBasePayload } from '_payloads'; import type { BasePayload, Payload } from '_payloads'; -import type { PermissionType } from './PermissionType'; +import type { PermissionType } from './permissionType'; export interface AcquirePermissionsRequest extends BasePayload { type: 'acquire-permissions-request'; diff --git a/apps/wallet/src/shared/messaging/messages/payloads/permissions/AcquirePermissionsResponse.ts b/apps/wallet/src/shared/messaging/messages/payloads/permissions/acquirePermissionsResponse.ts similarity index 100% rename from apps/wallet/src/shared/messaging/messages/payloads/permissions/AcquirePermissionsResponse.ts rename to apps/wallet/src/shared/messaging/messages/payloads/permissions/acquirePermissionsResponse.ts diff --git a/apps/wallet/src/shared/messaging/messages/payloads/permissions/DisconnectApp.ts b/apps/wallet/src/shared/messaging/messages/payloads/permissions/disconnectApp.ts similarity index 100% rename from apps/wallet/src/shared/messaging/messages/payloads/permissions/DisconnectApp.ts rename to apps/wallet/src/shared/messaging/messages/payloads/permissions/disconnectApp.ts diff --git a/apps/wallet/src/shared/messaging/messages/payloads/permissions/GetPermissionRequests.ts b/apps/wallet/src/shared/messaging/messages/payloads/permissions/getPermissionRequests.ts similarity index 100% rename from apps/wallet/src/shared/messaging/messages/payloads/permissions/GetPermissionRequests.ts rename to apps/wallet/src/shared/messaging/messages/payloads/permissions/getPermissionRequests.ts diff --git a/apps/wallet/src/shared/messaging/messages/payloads/permissions/HasPermissionsRequest.ts b/apps/wallet/src/shared/messaging/messages/payloads/permissions/hasPermissionsRequest.ts similarity index 90% rename from apps/wallet/src/shared/messaging/messages/payloads/permissions/HasPermissionsRequest.ts rename to apps/wallet/src/shared/messaging/messages/payloads/permissions/hasPermissionsRequest.ts index 507970c5843..103b6812258 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/permissions/HasPermissionsRequest.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/permissions/hasPermissionsRequest.ts @@ -5,7 +5,7 @@ import { isBasePayload } from '_payloads'; import type { BasePayload, Payload } from '_payloads'; -import type { PermissionType } from './PermissionType'; +import type { PermissionType } from './permissionType'; export interface HasPermissionsRequest extends BasePayload { type: 'has-permissions-request'; diff --git a/apps/wallet/src/shared/messaging/messages/payloads/permissions/HasPermissionsResponse.ts b/apps/wallet/src/shared/messaging/messages/payloads/permissions/hasPermissionsResponse.ts similarity index 100% rename from apps/wallet/src/shared/messaging/messages/payloads/permissions/HasPermissionsResponse.ts rename to apps/wallet/src/shared/messaging/messages/payloads/permissions/hasPermissionsResponse.ts diff --git a/apps/wallet/src/shared/messaging/messages/payloads/permissions/index.ts b/apps/wallet/src/shared/messaging/messages/payloads/permissions/index.ts index 1e232093c81..68bca0392c9 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/permissions/index.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/permissions/index.ts @@ -2,12 +2,12 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export * from './GetPermissionRequests'; -export * from './PermissionRequests'; -export * from './PermissionResponse'; -export * from './PermissionType'; -export * from './Permission'; -export * from './HasPermissionsRequest'; -export * from './HasPermissionsResponse'; -export * from './AcquirePermissionsRequest'; -export * from './AcquirePermissionsResponse'; +export * from './getPermissionRequests'; +export * from './permissionRequests'; +export * from './permissionResponse'; +export * from './permissionType'; +export * from './permission'; +export * from './hasPermissionsRequest'; +export * from './hasPermissionsResponse'; +export * from './acquirePermissionsRequest'; +export * from './acquirePermissionsResponse'; diff --git a/apps/wallet/src/shared/messaging/messages/payloads/permissions/Permission.ts b/apps/wallet/src/shared/messaging/messages/payloads/permissions/permission.ts similarity index 91% rename from apps/wallet/src/shared/messaging/messages/payloads/permissions/Permission.ts rename to apps/wallet/src/shared/messaging/messages/payloads/permissions/permission.ts index 61a79756ebd..c9e7fe93dd0 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/permissions/Permission.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/permissions/permission.ts @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import type { PermissionType } from './PermissionType'; +import type { PermissionType } from './permissionType'; //TODO: add description, name, tags //TODO add PageLink for instance where the origin and the wallet landing page are different. diff --git a/apps/wallet/src/shared/messaging/messages/payloads/permissions/PermissionRequests.ts b/apps/wallet/src/shared/messaging/messages/payloads/permissions/permissionRequests.ts similarity index 91% rename from apps/wallet/src/shared/messaging/messages/payloads/permissions/PermissionRequests.ts rename to apps/wallet/src/shared/messaging/messages/payloads/permissions/permissionRequests.ts index 0bae8c97003..52b98ea1177 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/permissions/PermissionRequests.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/permissions/permissionRequests.ts @@ -4,7 +4,7 @@ import { isBasePayload } from '_payloads'; import type { BasePayload, Payload } from '_payloads'; -import type { Permission } from './Permission'; +import type { Permission } from './permission'; export interface PermissionRequests extends BasePayload { type: 'permission-request'; diff --git a/apps/wallet/src/shared/messaging/messages/payloads/permissions/PermissionResponse.ts b/apps/wallet/src/shared/messaging/messages/payloads/permissions/permissionResponse.ts similarity index 100% rename from apps/wallet/src/shared/messaging/messages/payloads/permissions/PermissionResponse.ts rename to apps/wallet/src/shared/messaging/messages/payloads/permissions/permissionResponse.ts diff --git a/apps/wallet/src/shared/messaging/messages/payloads/permissions/PermissionType.ts b/apps/wallet/src/shared/messaging/messages/payloads/permissions/permissionType.ts similarity index 100% rename from apps/wallet/src/shared/messaging/messages/payloads/permissions/PermissionType.ts rename to apps/wallet/src/shared/messaging/messages/payloads/permissions/permissionType.ts diff --git a/apps/wallet/src/shared/messaging/messages/payloads/transactions/ApprovalRequest.ts b/apps/wallet/src/shared/messaging/messages/payloads/transactions/approvalRequest.ts similarity index 96% rename from apps/wallet/src/shared/messaging/messages/payloads/transactions/ApprovalRequest.ts rename to apps/wallet/src/shared/messaging/messages/payloads/transactions/approvalRequest.ts index 80685dd2aa6..1b71203744f 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/transactions/ApprovalRequest.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/transactions/approvalRequest.ts @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type SignedTransaction } from '_src/ui/app/WalletSigner'; +import { type SignedTransaction } from '_src/ui/app/walletSigner'; import type { IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; import { type IotaSignAndExecuteTransactionInput, diff --git a/apps/wallet/src/shared/messaging/messages/payloads/transactions/ExecuteTransactionRequest.ts b/apps/wallet/src/shared/messaging/messages/payloads/transactions/executeTransactionRequest.ts similarity index 94% rename from apps/wallet/src/shared/messaging/messages/payloads/transactions/ExecuteTransactionRequest.ts rename to apps/wallet/src/shared/messaging/messages/payloads/transactions/executeTransactionRequest.ts index b7d636006e2..c363f0bd9c9 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/transactions/ExecuteTransactionRequest.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/transactions/executeTransactionRequest.ts @@ -6,7 +6,7 @@ import { isBasePayload } from '_payloads'; import type { BasePayload, Payload } from '_payloads'; import { type IotaSignTransactionInput } from '@iota/wallet-standard'; -import { type TransactionDataType } from './ApprovalRequest'; +import { type TransactionDataType } from './approvalRequest'; export interface ExecuteTransactionRequest extends BasePayload { type: 'execute-transaction-request'; diff --git a/apps/wallet/src/shared/messaging/messages/payloads/transactions/ExecuteTransactionResponse.ts b/apps/wallet/src/shared/messaging/messages/payloads/transactions/executeTransactionResponse.ts similarity index 100% rename from apps/wallet/src/shared/messaging/messages/payloads/transactions/ExecuteTransactionResponse.ts rename to apps/wallet/src/shared/messaging/messages/payloads/transactions/executeTransactionResponse.ts diff --git a/apps/wallet/src/shared/messaging/messages/payloads/transactions/index.ts b/apps/wallet/src/shared/messaging/messages/payloads/transactions/index.ts index ee98d0e2753..a8f2f2f4e55 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/transactions/index.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/transactions/index.ts @@ -2,5 +2,5 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export * from './ExecuteTransactionRequest'; -export * from './ExecuteTransactionResponse'; +export * from './executeTransactionRequest'; +export * from './executeTransactionResponse'; diff --git a/apps/wallet/src/shared/messaging/messages/payloads/transactions/SignMessage.ts b/apps/wallet/src/shared/messaging/messages/payloads/transactions/signMessage.ts similarity index 85% rename from apps/wallet/src/shared/messaging/messages/payloads/transactions/SignMessage.ts rename to apps/wallet/src/shared/messaging/messages/payloads/transactions/signMessage.ts index 4dd520639b1..1ba70eb17a2 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/transactions/SignMessage.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/transactions/signMessage.ts @@ -4,8 +4,8 @@ import { type IotaSignPersonalMessageOutput } from '@iota/wallet-standard'; -import { isBasePayload, type BasePayload } from '../BasePayload'; -import { type Payload } from '../Payload'; +import { isBasePayload, type BasePayload } from '../basePayload'; +import { type Payload } from '../payload'; export interface SignMessageRequest extends BasePayload { type: 'sign-personal-message-request'; diff --git a/apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/GetTransactionRequests.ts b/apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/getTransactionRequests.ts similarity index 100% rename from apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/GetTransactionRequests.ts rename to apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/getTransactionRequests.ts diff --git a/apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/GetTransactionRequestsResponse.ts b/apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/getTransactionRequestsResponse.ts similarity index 84% rename from apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/GetTransactionRequestsResponse.ts rename to apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/getTransactionRequestsResponse.ts index df6f7df89ab..5bcb3534579 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/GetTransactionRequestsResponse.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/getTransactionRequestsResponse.ts @@ -4,7 +4,7 @@ import { isBasePayload } from '_payloads'; import type { BasePayload, Payload } from '_payloads'; -import type { ApprovalRequest } from '_payloads/transactions/ApprovalRequest'; +import type { ApprovalRequest } from '_src/shared/messaging/messages/payloads/transactions/approvalRequest'; export interface GetTransactionRequestsResponse extends BasePayload { type: 'get-transaction-requests-response'; diff --git a/apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/TransactionRequestResponse.ts b/apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/transactionRequestResponse.ts similarity index 93% rename from apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/TransactionRequestResponse.ts rename to apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/transactionRequestResponse.ts index ccfc53a55c3..427e5e18a02 100644 --- a/apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/TransactionRequestResponse.ts +++ b/apps/wallet/src/shared/messaging/messages/payloads/transactions/ui/transactionRequestResponse.ts @@ -4,7 +4,7 @@ import { isBasePayload } from '_payloads'; import type { BasePayload, Payload } from '_payloads'; -import { type SignedTransaction } from '_src/ui/app/WalletSigner'; +import { type SignedTransaction } from '_src/ui/app/walletSigner'; import type { IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; import { type IotaSignPersonalMessageOutput } from '@iota/wallet-standard'; diff --git a/apps/wallet/src/shared/messaging/PortChannelName.ts b/apps/wallet/src/shared/messaging/portChannelName.ts similarity index 100% rename from apps/wallet/src/shared/messaging/PortChannelName.ts rename to apps/wallet/src/shared/messaging/portChannelName.ts diff --git a/apps/wallet/src/shared/messaging/PortStream.ts b/apps/wallet/src/shared/messaging/portStream.ts similarity index 97% rename from apps/wallet/src/shared/messaging/PortStream.ts rename to apps/wallet/src/shared/messaging/portStream.ts index 728635900a5..d5262182870 100644 --- a/apps/wallet/src/shared/messaging/PortStream.ts +++ b/apps/wallet/src/shared/messaging/portStream.ts @@ -9,7 +9,7 @@ import type { Runtime } from 'webextension-polyfill'; import type { Message } from './messages'; import { isErrorPayload } from './messages/payloads'; -import type { PortChannelName } from './PortChannelName'; +import type { PortChannelName } from './portChannelName'; export class PortStream { private _messagesStream: Observable<Message>; diff --git a/apps/wallet/src/shared/messaging/WindowMessageStream.ts b/apps/wallet/src/shared/messaging/windowMessageStream.ts similarity index 100% rename from apps/wallet/src/shared/messaging/WindowMessageStream.ts rename to apps/wallet/src/shared/messaging/windowMessageStream.ts diff --git a/apps/wallet/src/shared/sentry-config.ts b/apps/wallet/src/shared/sentryConfig.ts similarity index 100% rename from apps/wallet/src/shared/sentry-config.ts rename to apps/wallet/src/shared/sentryConfig.ts diff --git a/apps/wallet/src/types/node-env.d.ts b/apps/wallet/src/types/nodeEnv.d.ts similarity index 100% rename from apps/wallet/src/types/node-env.d.ts rename to apps/wallet/src/types/nodeEnv.d.ts diff --git a/apps/wallet/src/types/scss-modules.d.ts b/apps/wallet/src/types/scssModules.d.ts similarity index 100% rename from apps/wallet/src/types/scss-modules.d.ts rename to apps/wallet/src/types/scssModules.d.ts diff --git a/apps/wallet/src/types/webextension-polyfill.d.ts b/apps/wallet/src/types/webExtensionPolyfill.d.ts similarity index 100% rename from apps/wallet/src/types/webextension-polyfill.d.ts rename to apps/wallet/src/types/webExtensionPolyfill.d.ts diff --git a/apps/wallet/src/ui/app/accounts-finder/accounts-finder.test.ts b/apps/wallet/src/ui/app/accounts-finder/accounts-finder.test.ts index f9cba5ad5f9..b7ffa940f6e 100644 --- a/apps/wallet/src/ui/app/accounts-finder/accounts-finder.test.ts +++ b/apps/wallet/src/ui/app/accounts-finder/accounts-finder.test.ts @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { recoverAccounts, mergeAccounts } from './accounts-finder'; +import { recoverAccounts, mergeAccounts } from './utils'; import { assert, test } from 'vitest'; import { FindBalance } from './types'; diff --git a/apps/wallet/src/ui/app/accounts-finder/AccountsFinder.ts b/apps/wallet/src/ui/app/accounts-finder/accountsFinder.ts similarity index 99% rename from apps/wallet/src/ui/app/accounts-finder/AccountsFinder.ts rename to apps/wallet/src/ui/app/accounts-finder/accountsFinder.ts index fadd3ca1b97..1205d75ae63 100644 --- a/apps/wallet/src/ui/app/accounts-finder/AccountsFinder.ts +++ b/apps/wallet/src/ui/app/accounts-finder/accountsFinder.ts @@ -2,7 +2,7 @@ // SPDX-License-Identifier: Apache-2.0 import type { AccountFromFinder, AddressFromFinder } from '_src/shared/accounts'; -import { diffAddressesBipPaths, mergeAccounts, recoverAccounts } from './accounts-finder'; +import { diffAddressesBipPaths, mergeAccounts, recoverAccounts } from './utils'; import type { IotaClient } from '@iota/iota-sdk/client'; import { getEmptyBalance } from './helpers'; import type { FindBalance } from './types'; diff --git a/apps/wallet/src/ui/app/accounts-finder/index.ts b/apps/wallet/src/ui/app/accounts-finder/index.ts index d29d857a2b7..503eb19b16e 100644 --- a/apps/wallet/src/ui/app/accounts-finder/index.ts +++ b/apps/wallet/src/ui/app/accounts-finder/index.ts @@ -1,4 +1,4 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export * from './accounts-finder'; -export * from './AccountsFinder'; +export * from './utils'; +export * from './accountsFinder'; diff --git a/apps/wallet/src/ui/app/accounts-finder/accounts-finder.ts b/apps/wallet/src/ui/app/accounts-finder/utils.ts similarity index 100% rename from apps/wallet/src/ui/app/accounts-finder/accounts-finder.ts rename to apps/wallet/src/ui/app/accounts-finder/utils.ts diff --git a/apps/wallet/src/ui/app/ApiProvider.ts b/apps/wallet/src/ui/app/apiProvider.ts similarity index 93% rename from apps/wallet/src/ui/app/ApiProvider.ts rename to apps/wallet/src/ui/app/apiProvider.ts index 578a0c9418f..853de3a5d7b 100644 --- a/apps/wallet/src/ui/app/ApiProvider.ts +++ b/apps/wallet/src/ui/app/apiProvider.ts @@ -2,14 +2,14 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { AccountType, type SerializedUIAccount } from '_src/background/accounts/Account'; -import { getIotaClient } from '_src/shared/iota-client'; +import { AccountType, type SerializedUIAccount } from '_src/background/accounts/account'; +import { getIotaClient } from '_src/shared/iotaClient'; import { getDefaultNetwork, Network, type IotaClient } from '@iota/iota-sdk/client'; import type { BackgroundClient } from './background-client'; -import { BackgroundServiceSigner } from './background-client/BackgroundServiceSigner'; +import { BackgroundServiceSigner } from './background-client/backgroundServiceSigner'; import { queryClient } from './helpers/queryClient'; -import { type WalletSigner } from './WalletSigner'; +import { type WalletSigner } from './walletSigner'; const ACCOUNT_TYPES_WITH_BACKGROUND_SIGNER: AccountType[] = [ AccountType.MnemonicDerived, @@ -17,7 +17,7 @@ const ACCOUNT_TYPES_WITH_BACKGROUND_SIGNER: AccountType[] = [ AccountType.PrivateKeyDerived, ]; -export default class ApiProvider { +class ApiProvider { private _apiFullNodeProvider?: IotaClient; private _signerByAddress: Map<string, WalletSigner> = new Map(); network = getDefaultNetwork(); diff --git a/apps/wallet/src/ui/app/background-client/BackgroundServiceSigner.ts b/apps/wallet/src/ui/app/background-client/backgroundServiceSigner.ts similarity index 94% rename from apps/wallet/src/ui/app/background-client/BackgroundServiceSigner.ts rename to apps/wallet/src/ui/app/background-client/backgroundServiceSigner.ts index e7991e7bfc1..a61cfaa60de 100644 --- a/apps/wallet/src/ui/app/background-client/BackgroundServiceSigner.ts +++ b/apps/wallet/src/ui/app/background-client/backgroundServiceSigner.ts @@ -2,11 +2,11 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type SerializedUIAccount } from '_src/background/accounts/Account'; +import { type SerializedUIAccount } from '_src/background/accounts/account'; import { type IotaClient } from '@iota/iota-sdk/client'; import type { BackgroundClient } from '.'; -import { WalletSigner } from '../WalletSigner'; +import { WalletSigner } from '../walletSigner'; export class BackgroundServiceSigner extends WalletSigner { readonly #account: SerializedUIAccount; diff --git a/apps/wallet/src/ui/app/background-client/index.ts b/apps/wallet/src/ui/app/background-client/index.ts index a6da569d7fc..9bd8b8b8c6b 100644 --- a/apps/wallet/src/ui/app/background-client/index.ts +++ b/apps/wallet/src/ui/app/background-client/index.ts @@ -4,40 +4,39 @@ import { createMessage } from '_messages'; import type { Message } from '_messages'; -import { PortStream } from '_messaging/PortStream'; +import { PortStream } from '_src/shared/messaging/portStream'; import { type BasePayload } from '_payloads'; import { isLoadedFeaturesPayload } from '_payloads/feature-gating'; import { isSetNetworkPayload, type SetNetworkPayload } from '_payloads/network'; import { isPermissionRequests } from '_payloads/permissions'; import type { GetPermissionRequests, PermissionResponse } from '_payloads/permissions'; -import type { DisconnectApp } from '_payloads/permissions/DisconnectApp'; +import type { DisconnectApp } from '_src/shared/messaging/messages/payloads/permissions/disconnectApp'; import { isUpdateActiveOrigin } from '_payloads/tabs/updateActiveOrigin'; -import type { GetTransactionRequests } from '_payloads/transactions/ui/GetTransactionRequests'; -import { isGetTransactionRequestsResponse } from '_payloads/transactions/ui/GetTransactionRequestsResponse'; -import type { TransactionRequestResponse } from '_payloads/transactions/ui/TransactionRequestResponse'; +import type { GetTransactionRequests } from '_src/shared/messaging/messages/payloads/transactions/ui/getTransactionRequests'; +import { isGetTransactionRequestsResponse } from '_src/shared/messaging/messages/payloads/transactions/ui/getTransactionRequestsResponse'; +import type { TransactionRequestResponse } from '_src/shared/messaging/messages/payloads/transactions/ui/transactionRequestResponse'; import { changeActiveNetwork, setActiveOrigin } from '_redux/slices/app'; import { setPermissions } from '_redux/slices/permissions'; import { setTransactionRequests } from '_redux/slices/transaction-requests'; -import { type MnemonicSerializedUiAccount } from '_src/background/accounts/MnemonicAccount'; -import { type SeedSerializedUiAccount } from '_src/background/accounts/SeedAccount'; +import { type MnemonicSerializedUiAccount } from '_src/background/accounts/mnemonicAccount'; +import { type SeedSerializedUiAccount } from '_src/background/accounts/seedAccount'; import type { NetworkEnvType } from '@iota/core'; import { isMethodPayload, type MethodPayload, type UIAccessibleEntityType, -} from '_src/shared/messaging/messages/payloads/MethodPayload'; -import { type SignedMessage, type SignedTransaction } from '_src/ui/app/WalletSigner'; +} from '_src/shared/messaging/messages/payloads/methodPayload'; +import { type SignedMessage, type SignedTransaction } from '_src/ui/app/walletSigner'; import type { AppDispatch } from '_store'; import { type IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; import { toB64 } from '@iota/iota-sdk/utils'; import { type QueryKey } from '@tanstack/react-query'; import { lastValueFrom, map, take } from 'rxjs'; - -import { growthbook } from '../experimentation/feature-gating'; -import { ACCOUNTS_QUERY_KEY } from '../helpers/query-client-keys'; +import { growthbook } from '../experimentation/featureGating'; +import { ACCOUNTS_QUERY_KEY } from '../helpers/queryClientKeys'; import { queryClient } from '../helpers/queryClient'; import { ACCOUNT_SOURCES_QUERY_KEY } from '../hooks/useAccountSources'; -import { AccountSourceType } from '_src/background/account-sources/AccountSource'; +import { AccountSourceType } from '_src/background/account-sources/accountSource'; import { type DeriveBipPathAccountsFinder, isDeriveBipPathAccountsFinderResponse, diff --git a/apps/wallet/src/ui/app/components/DAppInfoCard.tsx b/apps/wallet/src/ui/app/components/DAppInfoCard.tsx index 6824f2978c5..fa537494f55 100644 --- a/apps/wallet/src/ui/app/components/DAppInfoCard.tsx +++ b/apps/wallet/src/ui/app/components/DAppInfoCard.tsx @@ -5,7 +5,7 @@ import { type PermissionType } from '_src/shared/messaging/messages/payloads/permissions'; import { getValidDAppUrl } from '_src/shared/utils'; import { Card, CardBody, CardImage, CardType, ImageShape, ImageType } from '@iota/apps-ui-kit'; -import { useAccountByAddress } from '../hooks/useAccountByAddress'; +import { useAccountByAddress } from '_hooks'; import { AccountIcon } from './accounts/AccountIcon'; import { AccountItem } from './accounts/AccountItem'; import { useUnlockAccount } from './accounts/UnlockAccountContext'; diff --git a/apps/wallet/src/ui/app/components/DAppPermissionList.tsx b/apps/wallet/src/ui/app/components/DAppPermissionList.tsx index b565f9aff18..2f110bd1e4d 100644 --- a/apps/wallet/src/ui/app/components/DAppPermissionList.tsx +++ b/apps/wallet/src/ui/app/components/DAppPermissionList.tsx @@ -3,7 +3,7 @@ import { type PermissionType } from '_src/shared/messaging/messages/payloads/permissions'; import { SummaryListItem } from './SummaryListItem'; -import { Checkmark } from '@iota/ui-icons'; +import { Checkmark } from '@iota/apps-ui-icons'; export interface DAppPermissionListProps { permissions: PermissionType[]; diff --git a/apps/wallet/src/ui/app/components/ExplorerLinkHelper.tsx b/apps/wallet/src/ui/app/components/ExplorerLinkHelper.tsx index bbc381a0e44..19b5c696113 100644 --- a/apps/wallet/src/ui/app/components/ExplorerLinkHelper.tsx +++ b/apps/wallet/src/ui/app/components/ExplorerLinkHelper.tsx @@ -2,7 +2,7 @@ // SPDX-License-Identifier: Apache-2.0 import type { RenderExplorerLinkProps } from '@iota/core'; -import ExplorerLink from './explorer-link'; +import { ExplorerLink } from './explorer-link'; export function ExplorerLinkHelper({ children, ...linkProps }: RenderExplorerLinkProps) { return <ExplorerLink {...linkProps}>{children}</ExplorerLink>; diff --git a/apps/wallet/src/ui/app/components/MovedAssetNotification.tsx b/apps/wallet/src/ui/app/components/MovedAssetNotification.tsx new file mode 100644 index 00000000000..f6521663de8 --- /dev/null +++ b/apps/wallet/src/ui/app/components/MovedAssetNotification.tsx @@ -0,0 +1,33 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { type Toast, toast } from 'react-hot-toast'; +import { ButtonUnstyled } from '@iota/apps-ui-kit'; + +interface MovedAssetNotificationProps { + t: Toast; + destination: string; + onUndo: () => void; +} + +export function MovedAssetNotification({ t, destination, onUndo }: MovedAssetNotificationProps) { + return ( + <div + className="flex w-full flex-row items-baseline gap-x-xxs" + onClick={() => toast.dismiss(t.id)} + > + <ButtonUnstyled className="text-body-sm text-neutral-12 dark:text-neutral-92"> + Moved to {destination} + </ButtonUnstyled> + <ButtonUnstyled + onClick={() => { + onUndo(); + toast.dismiss(t.id); + }} + className="ml-auto mr-sm text-body-sm text-neutral-12 dark:text-neutral-92" + > + UNDO + </ButtonUnstyled> + </div> + ); +} diff --git a/apps/wallet/src/ui/app/components/OutsideClickHandler.tsx b/apps/wallet/src/ui/app/components/OutsideClickHandler.tsx index b309c5d642a..b79949f48db 100644 --- a/apps/wallet/src/ui/app/components/OutsideClickHandler.tsx +++ b/apps/wallet/src/ui/app/components/OutsideClickHandler.tsx @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import React, { useEffect, useRef } from 'react'; +import { useEffect, useRef } from 'react'; interface OutsideClickHandlerProps { onOutsideClick: () => void; diff --git a/apps/wallet/src/ui/app/components/PasswordInputDialog.tsx b/apps/wallet/src/ui/app/components/PasswordInputDialog.tsx index 6be03d29d6a..5b9f4f8382d 100644 --- a/apps/wallet/src/ui/app/components/PasswordInputDialog.tsx +++ b/apps/wallet/src/ui/app/components/PasswordInputDialog.tsx @@ -2,13 +2,13 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useBackgroundClient } from '_src/ui/app/hooks/useBackgroundClient'; +import { useBackgroundClient } from '_hooks'; import classNames from 'clsx'; import { Form, Formik } from 'formik'; import { toast } from 'react-hot-toast'; import { useNavigate } from 'react-router-dom'; import { object, string as YupString } from 'yup'; -import { ArrowLeft, ArrowRight, Loader } from '@iota/ui-icons'; +import { ArrowLeft, ArrowRight, Loader } from '@iota/apps-ui-icons'; import { Button, ButtonHtmlType, diff --git a/apps/wallet/src/ui/app/components/WalletListSelect.tsx b/apps/wallet/src/ui/app/components/WalletListSelect.tsx index 67a96dc882c..423574e63f7 100644 --- a/apps/wallet/src/ui/app/components/WalletListSelect.tsx +++ b/apps/wallet/src/ui/app/components/WalletListSelect.tsx @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import { cx } from 'class-variance-authority'; -import { useAccounts } from '../hooks/useAccounts'; +import { useAccounts } from '_hooks'; import { useMemo } from 'react'; import { formatAddress } from '@iota/iota-sdk/utils'; import { Checkbox } from '@iota/apps-ui-kit'; diff --git a/apps/wallet/src/ui/app/components/accounts/AccountBalanceItem.tsx b/apps/wallet/src/ui/app/components/accounts/AccountBalanceItem.tsx index a068e239e8e..9508becfd5d 100644 --- a/apps/wallet/src/ui/app/components/accounts/AccountBalanceItem.tsx +++ b/apps/wallet/src/ui/app/components/accounts/AccountBalanceItem.tsx @@ -2,10 +2,10 @@ // SPDX-License-Identifier: Apache-2.0 import { formatAddress } from '@iota/iota-sdk/utils'; -import { useCopyToClipboard } from '../../hooks/useCopyToClipboard'; -import { type SerializedUIAccount } from '_src/background/accounts/Account'; +import { useCopyToClipboard } from '_hooks'; +import { type SerializedUIAccount } from '_src/background/accounts/account'; import { useBalance, useFormatCoin } from '@iota/core'; -import { Copy } from '@iota/ui-icons'; +import { Copy } from '@iota/apps-ui-icons'; import { Panel, ButtonUnstyled } from '@iota/apps-ui-kit'; interface AccountBalanceItemProps { diff --git a/apps/wallet/src/ui/app/components/accounts/AccountIcon.tsx b/apps/wallet/src/ui/app/components/accounts/AccountIcon.tsx index 5ae5de338f1..d64260b9bf9 100644 --- a/apps/wallet/src/ui/app/components/accounts/AccountIcon.tsx +++ b/apps/wallet/src/ui/app/components/accounts/AccountIcon.tsx @@ -2,8 +2,8 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { AccountType, type SerializedUIAccount } from '_src/background/accounts/Account'; -import { Ledger, IotaLogoMark } from '@iota/ui-icons'; +import { AccountType, type SerializedUIAccount } from '_src/background/accounts/account'; +import { Ledger, IotaLogoMark } from '@iota/apps-ui-icons'; interface AccountIconProps { account: SerializedUIAccount; diff --git a/apps/wallet/src/ui/app/components/accounts/AccountItem.tsx b/apps/wallet/src/ui/app/components/accounts/AccountItem.tsx index 544da756927..af06e461e9e 100644 --- a/apps/wallet/src/ui/app/components/accounts/AccountItem.tsx +++ b/apps/wallet/src/ui/app/components/accounts/AccountItem.tsx @@ -5,9 +5,7 @@ import { formatAddress } from '@iota/iota-sdk/utils'; import cn from 'clsx'; import { type ReactNode } from 'react'; -import { useAccounts } from '../../hooks/useAccounts'; -import { useCopyToClipboard } from '../../hooks/useCopyToClipboard'; -import { useExplorerLink } from '../../hooks/useExplorerLink'; +import { useExplorerLink, useAccounts, useCopyToClipboard } from '_hooks'; import { ExplorerLinkType } from '_components'; import { Account } from '@iota/apps-ui-kit'; import { formatAccountName } from '../../helpers'; diff --git a/apps/wallet/src/ui/app/components/accounts/AccountItemApproveConnection.tsx b/apps/wallet/src/ui/app/components/accounts/AccountItemApproveConnection.tsx index 15a9e219084..754f7b1c5e4 100644 --- a/apps/wallet/src/ui/app/components/accounts/AccountItemApproveConnection.tsx +++ b/apps/wallet/src/ui/app/components/accounts/AccountItemApproveConnection.tsx @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import { AccountIcon, useUnlockAccount } from '_components'; -import { type SerializedUIAccount } from '_src/background/accounts/Account'; +import { type SerializedUIAccount } from '_src/background/accounts/account'; import { formatAddress } from '@iota/iota-sdk/utils'; import { Account } from '@iota/apps-ui-kit'; import { formatAccountName } from '../../helpers'; diff --git a/apps/wallet/src/ui/app/components/accounts/AccountListItem.tsx b/apps/wallet/src/ui/app/components/accounts/AccountListItem.tsx index 12047afaaab..651e4289b37 100644 --- a/apps/wallet/src/ui/app/components/accounts/AccountListItem.tsx +++ b/apps/wallet/src/ui/app/components/accounts/AccountListItem.tsx @@ -2,9 +2,9 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type SerializedUIAccount } from '_src/background/accounts/Account'; +import { type SerializedUIAccount } from '_src/background/accounts/account'; import { AccountItem } from './AccountItem'; -import { IotaLogoMark } from '@iota/ui-icons'; +import { IotaLogoMark } from '@iota/apps-ui-icons'; interface AccountListItemProps { account: SerializedUIAccount; diff --git a/apps/wallet/src/ui/app/components/accounts/AccountMultiSelect.tsx b/apps/wallet/src/ui/app/components/accounts/AccountMultiSelect.tsx index 972be18e5a9..ddaf5344cb8 100644 --- a/apps/wallet/src/ui/app/components/accounts/AccountMultiSelect.tsx +++ b/apps/wallet/src/ui/app/components/accounts/AccountMultiSelect.tsx @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import { AccountItemApproveConnection, SelectAllButton } from '_components'; -import { type SerializedUIAccount } from '_src/background/accounts/Account'; +import { type SerializedUIAccount } from '_src/background/accounts/account'; import * as ToggleGroup from '@radix-ui/react-toggle-group'; import { useState } from 'react'; diff --git a/apps/wallet/src/ui/app/components/accounts/EditableAccountName.tsx b/apps/wallet/src/ui/app/components/accounts/EditableAccountName.tsx index 1a9abff88f5..5133b0f367a 100644 --- a/apps/wallet/src/ui/app/components/accounts/EditableAccountName.tsx +++ b/apps/wallet/src/ui/app/components/accounts/EditableAccountName.tsx @@ -6,8 +6,7 @@ import { useZodForm } from '@iota/core'; import { forwardRef, useRef } from 'react'; import toast from 'react-hot-toast'; import { z } from 'zod'; - -import { useBackgroundClient } from '../../hooks/useBackgroundClient'; +import { useBackgroundClient } from '_hooks'; import { Form } from '../../shared/forms/Form'; interface InputProps extends Omit<React.InputHTMLAttributes<HTMLInputElement>, 'className'> {} diff --git a/apps/wallet/src/ui/app/components/accounts/ImportPrivateKeyForm.tsx b/apps/wallet/src/ui/app/components/accounts/ImportPrivateKeyForm.tsx index 455484086bc..ebeae3d8dd2 100644 --- a/apps/wallet/src/ui/app/components/accounts/ImportPrivateKeyForm.tsx +++ b/apps/wallet/src/ui/app/components/accounts/ImportPrivateKeyForm.tsx @@ -18,7 +18,7 @@ import { InfoBoxType, } from '@iota/apps-ui-kit'; import { TextAreaField } from '../../shared/forms/TextAreaField'; -import { Exclamation } from '@iota/ui-icons'; +import { Exclamation } from '@iota/apps-ui-icons'; const formSchema = z.object({ privateKey: privateKeyValidation, diff --git a/apps/wallet/src/ui/app/components/accounts/ImportRecoveryPhraseForm.tsx b/apps/wallet/src/ui/app/components/accounts/ImportRecoveryPhraseForm.tsx index 4afa2038447..c7d9c2ac3a2 100644 --- a/apps/wallet/src/ui/app/components/accounts/ImportRecoveryPhraseForm.tsx +++ b/apps/wallet/src/ui/app/components/accounts/ImportRecoveryPhraseForm.tsx @@ -18,7 +18,7 @@ import { InfoBoxType, InfoBoxStyle, } from '@iota/apps-ui-kit'; -import { Exclamation } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; const RECOVERY_PHRASE_WORD_COUNT = 24; @@ -28,7 +28,7 @@ const formSchema = z.object({ .length(RECOVERY_PHRASE_WORD_COUNT) .transform((recoveryPhrase) => normalizeMnemonics(recoveryPhrase.join(' ')).split(' ')) .refine((recoveryPhrase) => validateMnemonics(recoveryPhrase.join(' ')), { - message: 'Recovery Passphrase is invalid', + message: 'Mnenonic is invalid', }), }); @@ -125,14 +125,15 @@ export function ImportRecoveryPhraseForm({ </div> <div className="sticky bottom-0 left-0 flex flex-col gap-2.5 bg-neutral-100 pt-sm dark:bg-neutral-6"> - {touchedFields.recoveryPhrase && errors.recoveryPhrase && ( - <InfoBox - type={InfoBoxType.Default} - supportingText={errors.recoveryPhrase.message} - icon={<Exclamation />} - style={InfoBoxStyle.Elevated} - /> - )} + {errors?.recoveryPhrase?.message && + !touchedFields.recoveryPhrase?.every(Boolean) && ( + <InfoBox + type={InfoBoxType.Error} + supportingText={errors?.recoveryPhrase?.message} + icon={<Warning />} + style={InfoBoxStyle.Elevated} + /> + )} <div className="flex flex-row justify-stretch gap-2.5"> {cancelButtonText ? ( <Button diff --git a/apps/wallet/src/ui/app/components/accounts/NicknameDialog.tsx b/apps/wallet/src/ui/app/components/accounts/NicknameDialog.tsx index 1d03795981f..306fe7b3702 100644 --- a/apps/wallet/src/ui/app/components/accounts/NicknameDialog.tsx +++ b/apps/wallet/src/ui/app/components/accounts/NicknameDialog.tsx @@ -16,8 +16,7 @@ import { Input, InputType, } from '@iota/apps-ui-kit'; -import { useAccounts } from '../../hooks/useAccounts'; -import { useBackgroundClient } from '../../hooks/useBackgroundClient'; +import { useAccounts, useBackgroundClient } from '_hooks'; import { Form } from '../../shared/forms/Form'; const formSchema = z.object({ diff --git a/apps/wallet/src/ui/app/components/accounts/PasswordInputDialog.tsx b/apps/wallet/src/ui/app/components/accounts/PasswordInputDialog.tsx index 642449d78c7..6ae49fcb1a1 100644 --- a/apps/wallet/src/ui/app/components/accounts/PasswordInputDialog.tsx +++ b/apps/wallet/src/ui/app/components/accounts/PasswordInputDialog.tsx @@ -7,10 +7,9 @@ import { useState } from 'react'; import { toast } from 'react-hot-toast'; import { v4 as uuidV4 } from 'uuid'; import { z } from 'zod'; -import { useAccountSources } from '../../hooks/useAccountSources'; -import { useBackgroundClient } from '../../hooks/useBackgroundClient'; +import { useAccountSources, useBackgroundClient } from '_hooks'; import { Form } from '../../shared/forms/Form'; -import { AccountSourceType } from '_src/background/account-sources/AccountSource'; +import { AccountSourceType } from '_src/background/account-sources/accountSource'; import { Button, ButtonHtmlType, diff --git a/apps/wallet/src/ui/app/components/accounts/ProtectAccountForm.tsx b/apps/wallet/src/ui/app/components/accounts/ProtectAccountForm.tsx index de8c36e451b..cb581688c76 100644 --- a/apps/wallet/src/ui/app/components/accounts/ProtectAccountForm.tsx +++ b/apps/wallet/src/ui/app/components/accounts/ProtectAccountForm.tsx @@ -9,7 +9,7 @@ import { type SubmitHandler } from 'react-hook-form'; import { useNavigate } from 'react-router-dom'; import { z } from 'zod'; import zxcvbn from 'zxcvbn'; -import { parseAutoLock, useAutoLockMinutes } from '../../hooks/useAutoLockMinutes'; +import { parseAutoLock, useAutoLockMinutes } from '_hooks'; import { CheckboxField } from '../../shared/forms/CheckboxField'; import { Form } from '../../shared/forms/Form'; import { AutoLockSelector, zodSchema } from './AutoLockSelector'; diff --git a/apps/wallet/src/ui/app/components/accounts/RecoverAccountsGroup.tsx b/apps/wallet/src/ui/app/components/accounts/RecoverAccountsGroup.tsx index 6550e25fffa..e9b8f6cf180 100644 --- a/apps/wallet/src/ui/app/components/accounts/RecoverAccountsGroup.tsx +++ b/apps/wallet/src/ui/app/components/accounts/RecoverAccountsGroup.tsx @@ -2,10 +2,10 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type SerializedUIAccount } from '_src/background/accounts/Account'; +import { type SerializedUIAccount } from '_src/background/accounts/account'; import { AccountListItem } from './AccountListItem'; import { Button, ButtonSize, ButtonType, Tooltip, TooltipPosition } from '@iota/apps-ui-kit'; -import { CheckmarkFilled, Key } from '@iota/ui-icons'; +import { CheckmarkFilled, Key } from '@iota/apps-ui-icons'; export interface RecoverAccountsGroupProps { title: string; diff --git a/apps/wallet/src/ui/app/components/accounts/UnlockAccountButton.tsx b/apps/wallet/src/ui/app/components/accounts/UnlockAccountButton.tsx index 025ff894c73..a6a8e4bed00 100644 --- a/apps/wallet/src/ui/app/components/accounts/UnlockAccountButton.tsx +++ b/apps/wallet/src/ui/app/components/accounts/UnlockAccountButton.tsx @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type SerializedUIAccount } from '_src/background/accounts/Account'; +import { type SerializedUIAccount } from '_src/background/accounts/account'; import { useUnlockAccount } from './UnlockAccountContext'; import { Button, ButtonType } from '@iota/apps-ui-kit'; diff --git a/apps/wallet/src/ui/app/components/accounts/UnlockAccountContext.tsx b/apps/wallet/src/ui/app/components/accounts/UnlockAccountContext.tsx index 71e7758d0ae..1f3a2c3444d 100644 --- a/apps/wallet/src/ui/app/components/accounts/UnlockAccountContext.tsx +++ b/apps/wallet/src/ui/app/components/accounts/UnlockAccountContext.tsx @@ -2,18 +2,10 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type SerializedUIAccount } from '_src/background/accounts/Account'; -import React, { - createContext, - useCallback, - useContext, - useState, - type ReactNode, - useRef, -} from 'react'; +import { type SerializedUIAccount } from '_src/background/accounts/account'; +import { createContext, useCallback, useContext, useState, type ReactNode, useRef } from 'react'; import { toast } from 'react-hot-toast'; -import { useBackgroundClient } from '../../hooks/useBackgroundClient'; -import { useUnlockMutation } from '../../hooks/useUnlockMutation'; +import { useUnlockMutation, useBackgroundClient } from '_hooks'; import { UnlockAccountModal } from './UnlockAccountModal'; type OnSuccessCallback = () => void | Promise<void>; diff --git a/apps/wallet/src/ui/app/components/accounts/UnlockAccountModal.tsx b/apps/wallet/src/ui/app/components/accounts/UnlockAccountModal.tsx index 8f778ee6d7f..8eb2971763e 100644 --- a/apps/wallet/src/ui/app/components/accounts/UnlockAccountModal.tsx +++ b/apps/wallet/src/ui/app/components/accounts/UnlockAccountModal.tsx @@ -2,10 +2,9 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type SerializedUIAccount } from '_src/background/accounts/Account'; +import { type SerializedUIAccount } from '_src/background/accounts/account'; import { toast } from 'react-hot-toast'; - -import { useBackgroundClient } from '../../hooks/useBackgroundClient'; +import { useBackgroundClient } from '_hooks'; import { PasswordModalDialog } from './PasswordInputDialog'; interface UnlockAccountModalProps { diff --git a/apps/wallet/src/ui/app/components/active-coins-card/index.tsx b/apps/wallet/src/ui/app/components/active-coins-card/index.tsx index b5c8e64b490..3933130431c 100644 --- a/apps/wallet/src/ui/app/components/active-coins-card/index.tsx +++ b/apps/wallet/src/ui/app/components/active-coins-card/index.tsx @@ -2,9 +2,8 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useActiveAddress } from '_app/hooks/useActiveAddress'; +import { useCoinsReFetchingConfig, useActiveAddress } from '_hooks'; import { Loading } from '_components'; -import { useCoinsReFetchingConfig } from '_hooks'; import { useIotaClientQuery } from '@iota/dapp-kit'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { Link } from 'react-router-dom'; diff --git a/apps/wallet/src/ui/app/components/error-boundary/index.tsx b/apps/wallet/src/ui/app/components/error-boundary/index.tsx index e450622a9cf..7ba0e3a4524 100644 --- a/apps/wallet/src/ui/app/components/error-boundary/index.tsx +++ b/apps/wallet/src/ui/app/components/error-boundary/index.tsx @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import { InfoBox, InfoBoxType, InfoBoxStyle } from '@iota/apps-ui-kit'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; import type { ReactNode } from 'react'; import { ErrorBoundary as ReactErrorBoundary } from 'react-error-boundary'; import type { FallbackProps } from 'react-error-boundary'; diff --git a/apps/wallet/src/ui/app/components/explorer-link/index.tsx b/apps/wallet/src/ui/app/components/explorer-link/index.tsx index 0ce305b1087..4eeb1207db2 100644 --- a/apps/wallet/src/ui/app/components/explorer-link/index.tsx +++ b/apps/wallet/src/ui/app/components/explorer-link/index.tsx @@ -8,7 +8,7 @@ import { type ExplorerLinkConfig, ExplorerLinkType } from '@iota/core'; import { useExplorerLink } from '_hooks'; import st from './ExplorerLink.module.scss'; import clsx from 'clsx'; -import { ArrowTopRight } from '@iota/ui-icons'; +import { ArrowTopRight } from '@iota/apps-ui-icons'; export type ExplorerLinkProps = ExplorerLinkConfig & { track?: boolean; @@ -45,5 +45,3 @@ export function ExplorerLink({ } export { ExplorerLinkType }; - -export default ExplorerLink; diff --git a/apps/wallet/src/ui/app/components/external-link/index.tsx b/apps/wallet/src/ui/app/components/external-link/index.tsx index 52e73eb0ad2..2f09c645521 100644 --- a/apps/wallet/src/ui/app/components/external-link/index.tsx +++ b/apps/wallet/src/ui/app/components/external-link/index.tsx @@ -2,7 +2,6 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { memo } from 'react'; import type { ReactNode } from 'react'; export interface ExternalLinkProps { @@ -27,5 +26,3 @@ export function ExternalLink({ href, className, children, title, onClick }: Exte </a> ); } - -export default memo(ExternalLink); diff --git a/apps/wallet/src/ui/app/components/filters-tags/index.tsx b/apps/wallet/src/ui/app/components/filters-tags/index.tsx index 09fc2b21dbe..d3499467842 100644 --- a/apps/wallet/src/ui/app/components/filters-tags/index.tsx +++ b/apps/wallet/src/ui/app/components/filters-tags/index.tsx @@ -23,7 +23,7 @@ interface FiltersPortalProps { callback?: (tag: Tag) => void; } -function FiltersPortal({ tags, callback, firstLastMargin }: FiltersPortalProps) { +function FiltersPortalComponent({ tags, callback, firstLastMargin }: FiltersPortalProps) { const [element, setElement] = useState<HTMLElement | null>(null); useEffect(() => { @@ -65,4 +65,4 @@ function FiltersPortal({ tags, callback, firstLastMargin }: FiltersPortalProps) ); } -export default memo(FiltersPortal); +export const FiltersPortal = memo(FiltersPortalComponent); diff --git a/apps/wallet/src/ui/app/components/index.ts b/apps/wallet/src/ui/app/components/index.ts index 978c70b11b4..3c404c548bb 100644 --- a/apps/wallet/src/ui/app/components/index.ts +++ b/apps/wallet/src/ui/app/components/index.ts @@ -26,9 +26,9 @@ export * from './menu'; export * from './navigation'; export * from './network-selector'; export * from './nft-display'; -export * from './nft-display/NftImage'; export * from './receipt-card'; export * from './receipt-card/TxnAmount'; export * from './transactions-card'; export * from './user-approve-container'; -export { default as FiltersPortal } from './filters-tags'; +export * from './filters-tags'; +export * from './MovedAssetNotification'; diff --git a/apps/wallet/src/ui/app/components/iota-apps/AppsPlayGround.tsx b/apps/wallet/src/ui/app/components/iota-apps/AppsPlayGround.tsx new file mode 100644 index 00000000000..dafc4379d57 --- /dev/null +++ b/apps/wallet/src/ui/app/components/iota-apps/AppsPlayGround.tsx @@ -0,0 +1,75 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { useAppSelector } from '_hooks'; +import { Feature } from '@iota/core'; +import { prepareLinkToCompare } from '_src/shared/utils'; +import { useFeature } from '@growthbook/growthbook-react'; +import { useMemo } from 'react'; +import { useParams } from 'react-router-dom'; +import { permissionsSelectors } from '../../redux/slices/permissions'; +import { AppsPageBanner } from './Banner'; +import { IotaApp, type DAppEntry } from './IotaApp'; +import { IotaAppEmpty } from './IotaAppEmpty'; +import { InfoBox, InfoBoxStyle, InfoBoxType, Header } from '@iota/apps-ui-kit'; +import { Info } from '@iota/apps-ui-icons'; + +export function AppsPlayGround() { + const ecosystemApps = useFeature<DAppEntry[]>(Feature.WalletDapps).value; + const { tagName } = useParams(); + + const filteredEcosystemApps = useMemo(() => { + if (!ecosystemApps) { + return []; + } else if (tagName) { + return ecosystemApps.filter((app) => app.tags.includes(tagName)); + } + return ecosystemApps; + }, [ecosystemApps, tagName]); + + const allPermissions = useAppSelector(permissionsSelectors.selectAll); + const linkToPermissionID = useMemo(() => { + const map = new Map<string, string>(); + for (const aPermission of allPermissions) { + map.set(prepareLinkToCompare(aPermission.origin), aPermission.id); + if (aPermission.pagelink) { + map.set(prepareLinkToCompare(aPermission.pagelink), aPermission.id); + } + } + return map; + }, [allPermissions]); + + return ( + <> + <Header titleCentered title="IOTA Apps" /> + <AppsPageBanner /> + + {filteredEcosystemApps?.length ? ( + <InfoBox + type={InfoBoxType.Default} + icon={<Info />} + style={InfoBoxStyle.Elevated} + supportingText="Apps below are actively curated but do not indicate any endorsement or + relationship with IOTA Wallet. Please DYOR." + /> + ) : null} + + {filteredEcosystemApps?.length ? ( + <div className="mt-md flex flex-col gap-sm"> + {filteredEcosystemApps.map((app) => ( + <IotaApp + key={app.link} + {...app} + permissionID={linkToPermissionID.get(prepareLinkToCompare(app.link))} + displayType="full" + openAppSite + /> + ))} + </div> + ) : ( + <IotaAppEmpty displayType="full" /> + )} + </> + ); +} diff --git a/apps/wallet/src/ui/app/components/iota-apps/ConnectedAppsCard.tsx b/apps/wallet/src/ui/app/components/iota-apps/ConnectedAppsCard.tsx index efce9ba05ea..71f4c3d7e8c 100644 --- a/apps/wallet/src/ui/app/components/iota-apps/ConnectedAppsCard.tsx +++ b/apps/wallet/src/ui/app/components/iota-apps/ConnectedAppsCard.tsx @@ -3,19 +3,17 @@ // SPDX-License-Identifier: Apache-2.0 import { Title, TitleSize } from '@iota/apps-ui-kit'; -import { useAppSelector } from '_hooks'; +import { useAppSelector, useBackgroundClient } from '_hooks'; import cn from 'clsx'; import { Feature } from '@iota/core'; import { prepareLinkToCompare } from '_src/shared/utils'; import { useFeature } from '@growthbook/growthbook-react'; import { useEffect, useMemo } from 'react'; - -import { useBackgroundClient } from '../../hooks/useBackgroundClient'; import { permissionsSelectors } from '../../redux/slices/permissions'; import { Loading, NoData, PageTemplate } from '_components'; import { type DAppEntry, IotaApp } from './IotaApp'; -function ConnectedDapps() { +export function ConnectedAppsCard() { const backgroundClient = useBackgroundClient(); useEffect(() => { backgroundClient.sendGetPermissionRequests(); @@ -86,5 +84,3 @@ function ConnectedDapps() { </PageTemplate> ); } - -export default ConnectedDapps; diff --git a/apps/wallet/src/ui/app/components/iota-apps/DisconnectApp.tsx b/apps/wallet/src/ui/app/components/iota-apps/DisconnectApp.tsx index a2ba2745704..c60c04e9ce6 100644 --- a/apps/wallet/src/ui/app/components/iota-apps/DisconnectApp.tsx +++ b/apps/wallet/src/ui/app/components/iota-apps/DisconnectApp.tsx @@ -3,18 +3,15 @@ // SPDX-License-Identifier: Apache-2.0 import { Overlay, DAppInfoCard, WalletListSelect } from '_components'; -import { useAppSelector } from '_hooks'; +import { useAppSelector, useBackgroundClient } from '_hooks'; import { permissionsSelectors } from '_redux/slices/permissions'; import { ampli } from '_src/shared/analytics/ampli'; import { formatAddress } from '@iota/iota-sdk/utils'; import { useMutation } from '@tanstack/react-query'; import { useEffect, useMemo, useState } from 'react'; import { toast } from 'react-hot-toast'; - -import { useBackgroundClient } from '../../hooks/useBackgroundClient'; import { type DAppEntry } from './IotaApp'; - -import { CircleEmitter } from '@iota/ui-icons'; +import { CircleEmitter } from '@iota/apps-ui-icons'; import { Button, ButtonType } from '@iota/apps-ui-kit'; import { SummaryPanel } from '../SummaryPanel'; import { SummaryListItem } from '../SummaryListItem'; @@ -25,7 +22,7 @@ export interface DisconnectAppProps extends Omit<DAppEntry, 'description' | 'tag setShowDisconnectApp: (showModal: boolean) => void; } -function DisconnectApp({ +export function DisconnectApp({ name, icon, link, @@ -72,12 +69,7 @@ function DisconnectApp({ return null; } return ( - <Overlay - showBackButton - showModal - setShowModal={setShowDisconnectApp} - title="Active Connection" - > + <Overlay showModal setShowModal={setShowDisconnectApp} title="Active Connection"> <div className="flex max-w-full flex-1 flex-col flex-nowrap items-stretch gap-y-md"> <DAppInfoCard name={name} iconUrl={icon} url={link} /> @@ -137,5 +129,3 @@ function DisconnectApp({ </Overlay> ); } - -export default DisconnectApp; diff --git a/apps/wallet/src/ui/app/components/iota-apps/IotaApp.tsx b/apps/wallet/src/ui/app/components/iota-apps/IotaApp.tsx index d6db449aaa4..0df8593611d 100644 --- a/apps/wallet/src/ui/app/components/iota-apps/IotaApp.tsx +++ b/apps/wallet/src/ui/app/components/iota-apps/IotaApp.tsx @@ -8,8 +8,7 @@ import { ampli } from '_src/shared/analytics/ampli'; import { getDAppUrl } from '_src/shared/utils'; import { useState } from 'react'; import { Card, CardImage, CardBody, ImageShape, Badge, BadgeType } from '@iota/apps-ui-kit'; - -import DisconnectApp from './DisconnectApp'; +import { DisconnectApp } from './DisconnectApp'; export type DAppEntry = { name: string; diff --git a/apps/wallet/src/ui/app/components/iota-apps/index.tsx b/apps/wallet/src/ui/app/components/iota-apps/index.tsx index 8633641c5ed..c61a0d75d35 100644 --- a/apps/wallet/src/ui/app/components/iota-apps/index.tsx +++ b/apps/wallet/src/ui/app/components/iota-apps/index.tsx @@ -1,78 +1,5 @@ -// Copyright (c) Mysten Labs, Inc. -// Modifications Copyright (c) 2024 IOTA Stiftung +// Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useAppSelector } from '_hooks'; -import { Feature } from '@iota/core'; -import { prepareLinkToCompare } from '_src/shared/utils'; -import { useFeature } from '@growthbook/growthbook-react'; -import { useMemo } from 'react'; -import { useParams } from 'react-router-dom'; -import { permissionsSelectors } from '../../redux/slices/permissions'; -import { AppsPageBanner } from './Banner'; -import { IotaApp, type DAppEntry } from './IotaApp'; -import { IotaAppEmpty } from './IotaAppEmpty'; -import { InfoBox, InfoBoxStyle, InfoBoxType, Header } from '@iota/apps-ui-kit'; -import { Info } from '@iota/ui-icons'; - -export function AppsPlayGround() { - const ecosystemApps = useFeature<DAppEntry[]>(Feature.WalletDapps).value; - const { tagName } = useParams(); - - const filteredEcosystemApps = useMemo(() => { - if (!ecosystemApps) { - return []; - } else if (tagName) { - return ecosystemApps.filter((app) => app.tags.includes(tagName)); - } - return ecosystemApps; - }, [ecosystemApps, tagName]); - - const allPermissions = useAppSelector(permissionsSelectors.selectAll); - const linkToPermissionID = useMemo(() => { - const map = new Map<string, string>(); - for (const aPermission of allPermissions) { - map.set(prepareLinkToCompare(aPermission.origin), aPermission.id); - if (aPermission.pagelink) { - map.set(prepareLinkToCompare(aPermission.pagelink), aPermission.id); - } - } - return map; - }, [allPermissions]); - - return ( - <> - <Header titleCentered title="IOTA Apps" /> - <AppsPageBanner /> - - {filteredEcosystemApps?.length ? ( - <InfoBox - type={InfoBoxType.Default} - icon={<Info />} - style={InfoBoxStyle.Elevated} - supportingText="Apps below are actively curated but do not indicate any endorsement or - relationship with IOTA Wallet. Please DYOR." - /> - ) : null} - - {filteredEcosystemApps?.length ? ( - <div className="mt-md flex flex-col gap-sm"> - {filteredEcosystemApps.map((app) => ( - <IotaApp - key={app.link} - {...app} - permissionID={linkToPermissionID.get(prepareLinkToCompare(app.link))} - displayType="full" - openAppSite - /> - ))} - </div> - ) : ( - <IotaAppEmpty displayType="full" /> - )} - </> - ); -} - -export default AppsPlayGround; -export { default as ConnectedAppsCard } from './ConnectedAppsCard'; +export * from './AppsPlayGround'; +export * from './ConnectedAppsCard'; diff --git a/apps/wallet/src/ui/app/components/ledger/useDeriveLedgerAccounts.ts b/apps/wallet/src/ui/app/components/ledger/useDeriveLedgerAccounts.ts index f20776577ae..171f5d39648 100644 --- a/apps/wallet/src/ui/app/components/ledger/useDeriveLedgerAccounts.ts +++ b/apps/wallet/src/ui/app/components/ledger/useDeriveLedgerAccounts.ts @@ -2,13 +2,13 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type LedgerAccountSerializedUI } from '_src/background/accounts/LedgerAccount'; +import { type LedgerAccountSerializedUI } from '_src/background/accounts/ledgerAccount'; import type IotaLedgerClient from '@iota/ledgerjs-hw-app-iota'; import { Ed25519PublicKey } from '@iota/iota-sdk/keypairs/ed25519'; import { useQuery, type UseQueryOptions } from '@tanstack/react-query'; import { useIotaLedgerClient } from './IotaLedgerClientProvider'; -import { AccountType } from '_src/background/accounts/Account'; +import { AccountType } from '_src/background/accounts/account'; type LedgerAccountKeys = 'address' | 'publicKey' | 'type' | 'derivationPath'; diff --git a/apps/wallet/src/ui/app/components/loading/index.tsx b/apps/wallet/src/ui/app/components/loading/index.tsx index 3ca75bb0db4..98e689aec31 100644 --- a/apps/wallet/src/ui/app/components/loading/index.tsx +++ b/apps/wallet/src/ui/app/components/loading/index.tsx @@ -19,5 +19,3 @@ export function Loading({ loading, children, ...indicatorProps }: LoadingProps) <>{children}</> ); } - -export default Loading; diff --git a/apps/wallet/src/ui/app/components/menu/button/WalletSettingsButton.tsx b/apps/wallet/src/ui/app/components/menu/button/WalletSettingsButton.tsx index 9607f08c8dd..7917c1bb752 100644 --- a/apps/wallet/src/ui/app/components/menu/button/WalletSettingsButton.tsx +++ b/apps/wallet/src/ui/app/components/menu/button/WalletSettingsButton.tsx @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import { Button, ButtonSize, ButtonType } from '@iota/apps-ui-kit'; -import { Close, Settings } from '@iota/ui-icons'; +import { Close, Settings } from '@iota/apps-ui-icons'; import { useMenuIsOpen, useNextMenuUrl } from '_components'; import { Link } from 'react-router-dom'; import { cx } from 'class-variance-authority'; diff --git a/apps/wallet/src/ui/app/components/menu/content/AutoLockAccounts.tsx b/apps/wallet/src/ui/app/components/menu/content/AutoLockAccounts.tsx index 7a8b59b7ce0..aaf11adf3ba 100644 --- a/apps/wallet/src/ui/app/components/menu/content/AutoLockAccounts.tsx +++ b/apps/wallet/src/ui/app/components/menu/content/AutoLockAccounts.tsx @@ -7,8 +7,8 @@ import { autoLockDataToMinutes, parseAutoLock, useAutoLockMinutes, -} from '_src/ui/app/hooks/useAutoLockMinutes'; -import { useAutoLockMinutesMutation } from '_src/ui/app/hooks/useAutoLockMinutesMutation'; + useAutoLockMinutesMutation, +} from '_hooks'; import { Form } from '_src/ui/app/shared/forms/Form'; import { useZodForm } from '@iota/core'; import toast from 'react-hot-toast'; diff --git a/apps/wallet/src/ui/app/components/menu/content/WalletSettingsMenuList.tsx b/apps/wallet/src/ui/app/components/menu/content/WalletSettingsMenuList.tsx index d2a5da8f49c..35ca4170bb1 100644 --- a/apps/wallet/src/ui/app/components/menu/content/WalletSettingsMenuList.tsx +++ b/apps/wallet/src/ui/app/components/menu/content/WalletSettingsMenuList.tsx @@ -3,20 +3,23 @@ // SPDX-License-Identifier: Apache-2.0 import { useNextMenuUrl, Overlay } from '_components'; -import { useAppSelector } from '_hooks'; +import { + useAppSelector, + formatAutoLock, + useAutoLockMinutes, + useBackgroundClient, + useActiveAccount, +} from '_hooks'; import { FAQ_LINK, ToS_LINK } from '_src/shared/constants'; -import { formatAutoLock, useAutoLockMinutes } from '_src/ui/app/hooks/useAutoLockMinutes'; -import FaucetRequestButton from '_src/ui/app/shared/faucet/FaucetRequestButton'; +import { FaucetRequestButton } from '_src/ui/app/shared/faucet/FaucetRequestButton'; import { getNetwork, Network } from '@iota/iota-sdk/client'; import Browser from 'webextension-polyfill'; import { Link, useNavigate } from 'react-router-dom'; import { useQueryClient, useMutation } from '@tanstack/react-query'; import { persister } from '_src/ui/app/helpers/queryClient'; -import { useBackgroundClient } from '_src/ui/app/hooks/useBackgroundClient'; import { useState } from 'react'; import { ConfirmationModal } from '_src/ui/app/shared/ConfirmationModal'; -import { DarkMode, Globe, Info, LockLocked, LockUnlocked, Logout } from '@iota/ui-icons'; -import { useActiveAccount } from '_src/ui/app/hooks/useActiveAccount'; +import { DarkMode, Globe, Info, LockLocked, LockUnlocked, Logout } from '@iota/apps-ui-icons'; import { ButtonType, Card, @@ -30,7 +33,7 @@ import { import { ampli } from '_src/shared/analytics/ampli'; import { useTheme, getCustomNetwork } from '@iota/core'; -function MenuList() { +export function MenuList() { const { themePreference } = useTheme(); const navigate = useNavigate(); const activeAccount = useActiveAccount(); @@ -169,5 +172,3 @@ function MenuList() { </Overlay> ); } - -export default MenuList; diff --git a/apps/wallet/src/ui/app/components/menu/content/index.tsx b/apps/wallet/src/ui/app/components/menu/content/index.tsx index ef5b55515fd..beddd857700 100644 --- a/apps/wallet/src/ui/app/components/menu/content/index.tsx +++ b/apps/wallet/src/ui/app/components/menu/content/index.tsx @@ -15,12 +15,12 @@ import type { MouseEvent } from 'react'; import { Navigate, Route, Routes, useLocation, useNavigate } from 'react-router-dom'; import { AutoLockAccounts } from './AutoLockAccounts'; import { NetworkSettings } from './NetworkSettings'; -import WalletSettingsMenuList from './WalletSettingsMenuList'; +import { MenuList } from './WalletSettingsMenuList'; import { ThemeSettings } from './ThemeSettings'; const CLOSE_KEY_CODES: string[] = ['Escape']; -function MenuContent() { +export function MenuContent() { const mainLocation = useLocation(); const isOpen = useMenuIsOpen(); const menuUrl = useMenuUrl(); @@ -47,7 +47,7 @@ function MenuContent() { <ErrorBoundary> <MainLocationContext.Provider value={mainLocation}> <Routes location={menuUrl || ''}> - <Route path="/" element={<WalletSettingsMenuList />} /> + <Route path="/" element={<MenuList />} /> <Route path="/network" element={<NetworkSettings />} /> <Route path="/auto-lock" element={<AutoLockAccounts />} /> <Route path="/theme" element={<ThemeSettings />} /> @@ -58,5 +58,3 @@ function MenuContent() { </div> ); } - -export default MenuContent; diff --git a/apps/wallet/src/ui/app/components/menu/index.tsx b/apps/wallet/src/ui/app/components/menu/index.tsx index 1daa5025de8..2e2817233f6 100644 --- a/apps/wallet/src/ui/app/components/menu/index.tsx +++ b/apps/wallet/src/ui/app/components/menu/index.tsx @@ -2,6 +2,6 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export { default as MenuContent } from './content'; +export * from './content'; export * from './hooks'; export * from './button/WalletSettingsButton'; diff --git a/apps/wallet/src/ui/app/components/navigation/index.tsx b/apps/wallet/src/ui/app/components/navigation/index.tsx index 17abcb0d65e..4ce35c7325d 100644 --- a/apps/wallet/src/ui/app/components/navigation/index.tsx +++ b/apps/wallet/src/ui/app/components/navigation/index.tsx @@ -3,9 +3,9 @@ // SPDX-License-Identifier: Apache-2.0 import { useNavigate, useLocation } from 'react-router-dom'; -import { useActiveAccount } from '../../hooks/useActiveAccount'; +import { useActiveAccount } from '_hooks'; import { Navbar, type NavbarItemWithId } from '@iota/apps-ui-kit'; -import { Activity, Apps, Assets, Home } from '@iota/ui-icons'; +import { Activity, Apps, Assets, Home } from '@iota/apps-ui-icons'; type NavbarItemWithPath = NavbarItemWithId & { path: string; diff --git a/apps/wallet/src/ui/app/components/network-selector/index.tsx b/apps/wallet/src/ui/app/components/network-selector/index.tsx index 3bbeb7a7b8c..bc4c1cbda1d 100644 --- a/apps/wallet/src/ui/app/components/network-selector/index.tsx +++ b/apps/wallet/src/ui/app/components/network-selector/index.tsx @@ -90,5 +90,3 @@ export function NetworkSelector() { </div> ); } - -export default NetworkSelector; diff --git a/apps/wallet/src/ui/app/components/nft-display/index.tsx b/apps/wallet/src/ui/app/components/nft-display/index.tsx index 779afa87e49..3e279b72935 100644 --- a/apps/wallet/src/ui/app/components/nft-display/index.tsx +++ b/apps/wallet/src/ui/app/components/nft-display/index.tsx @@ -2,13 +2,19 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { Loading, NftImage } from '_components'; -import { isKioskOwnerToken, useGetNFTDisplay, useGetObject, useKioskClient } from '@iota/core'; +import { Loading } from '_components'; +import { + NftImage, + isKioskOwnerToken, + useGetNFTDisplay, + useGetObject, + useKioskClient, + KioskTile, +} from '@iota/core'; import { formatAddress } from '@iota/iota-sdk/utils'; import { cva } from 'class-variance-authority'; import type { VariantProps } from 'class-variance-authority'; -import { useResolveVideo } from '../../hooks/useResolveVideo'; -import { Kiosk } from './Kiosk'; +import { useResolveVideo, useActiveAddress } from '_hooks'; const nftDisplayCardStyles = cva('flex flex-nowrap items-center h-full relative', { variants: { @@ -48,13 +54,14 @@ export function NFTDisplayCard({ const video = useResolveVideo(objectData); const kioskClient = useKioskClient(); const isOwnerToken = isKioskOwnerToken(kioskClient.network, objectData); + const address = useActiveAddress(); return ( <div className={nftDisplayCardStyles({ isHoverable, wideView })}> <Loading loading={isPending}> <div className="flex w-full flex-col justify-center gap-sm text-center"> {objectData?.data && isOwnerToken ? ( - <Kiosk object={objectData} /> + <KioskTile object={objectData} address={address} /> ) : ( <NftImage title={nftName} diff --git a/apps/wallet/src/ui/app/components/receipt-card/TxnAmount.tsx b/apps/wallet/src/ui/app/components/receipt-card/TxnAmount.tsx index 85d4ff481e0..cb17eab7d23 100644 --- a/apps/wallet/src/ui/app/components/receipt-card/TxnAmount.tsx +++ b/apps/wallet/src/ui/app/components/receipt-card/TxnAmount.tsx @@ -28,7 +28,7 @@ export function TxnAmount({ amount, coinType, subtitle, approximation }: TxnAmou return Number(amount) !== 0 ? ( <Card type={CardType.Filled}> <CardImage type={ImageType.BgSolid}> - <CoinIcon coinType={coinType} rounded size={ImageIconSize.Large} hasCoinWrapper /> + <CoinIcon coinType={coinType} rounded size={ImageIconSize.Small} /> </CardImage> <CardBody title={`${approximation ? '~' : ''}${formatAmount} ${symbol}`} diff --git a/apps/wallet/src/ui/app/components/receipt-card/index.tsx b/apps/wallet/src/ui/app/components/receipt-card/index.tsx index 3bba1a45a7f..2c8e6a82c51 100644 --- a/apps/wallet/src/ui/app/components/receipt-card/index.tsx +++ b/apps/wallet/src/ui/app/components/receipt-card/index.tsx @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useRecognizedPackages } from '_src/ui/app/hooks/useRecognizedPackages'; +import { useRecognizedPackages } from '_hooks'; import { useTransactionSummary, TransactionReceipt, @@ -11,10 +11,8 @@ import { } from '@iota/core'; import { type IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; -import { CardType } from '@iota/apps-ui-kit'; -import { ValidatorLogo } from '../../staking/validators/ValidatorLogo'; import { ExplorerLinkHelper } from '../ExplorerLinkHelper'; -import ExplorerLink from '../explorer-link'; +import { ExplorerLink } from '../explorer-link'; interface ReceiptCardProps { txn: IotaTransactionBlockResponse; @@ -40,14 +38,6 @@ export function ReceiptCard({ txn, activeAddress }: ReceiptCardProps) { summary={summary} activeAddress={activeAddress} renderExplorerLink={ExplorerLinkHelper} - renderValidatorLogo={({ address, showActiveStatus, activeEpoch, isSelected }) => ( - <ValidatorLogo - validatorAddress={address} - showActiveStatus={showActiveStatus} - activeEpoch={activeEpoch} - type={isSelected ? CardType.Filled : CardType.Outlined} - /> - )} /> <div className="pt-sm"> <ExplorerLink transactionID={digest ?? ''} type={ExplorerLinkType.Transaction}> diff --git a/apps/wallet/src/ui/app/components/transactions-card/TxnIcon.tsx b/apps/wallet/src/ui/app/components/transactions-card/TxnIcon.tsx deleted file mode 100644 index 598b68a32e5..00000000000 --- a/apps/wallet/src/ui/app/components/transactions-card/TxnIcon.tsx +++ /dev/null @@ -1,32 +0,0 @@ -// Copyright (c) Mysten Labs, Inc. -// Modifications Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -import { LoadingIndicator } from '@iota/apps-ui-kit'; -import { ArrowBottomLeft, ArrowTopRight, Info, IotaLogoMark, Person, Stake } from '@iota/ui-icons'; - -const ICON_COLORS = { - primary: 'text-primary-30', - error: 'text-error-30', -}; - -const icons = { - Send: <ArrowTopRight className={ICON_COLORS.primary} />, - Receive: <ArrowBottomLeft className={ICON_COLORS.primary} />, - Transaction: <ArrowTopRight className={ICON_COLORS.primary} />, - Staked: <Stake className={ICON_COLORS.primary} />, - Unstaked: <Stake className={ICON_COLORS.primary} />, - Rewards: <IotaLogoMark className={ICON_COLORS.primary} />, - Failed: <Info className={ICON_COLORS.error} />, - Loading: <LoadingIndicator />, - PersonalMessage: <Person className={ICON_COLORS.primary} />, -}; - -interface TxnItemIconProps { - txnFailed?: boolean; - variant: keyof typeof icons; -} - -export function TxnIcon({ txnFailed, variant }: TxnItemIconProps) { - return <div className="[&_svg]:h-5 [&_svg]:w-5">{icons[txnFailed ? 'Failed' : variant]}</div>; -} diff --git a/apps/wallet/src/ui/app/components/transactions-card/index.tsx b/apps/wallet/src/ui/app/components/transactions-card/index.tsx index ab15c187695..1c1d8f47226 100644 --- a/apps/wallet/src/ui/app/components/transactions-card/index.tsx +++ b/apps/wallet/src/ui/app/components/transactions-card/index.tsx @@ -2,17 +2,17 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useRecognizedPackages } from '_src/ui/app/hooks/useRecognizedPackages'; +import { useRecognizedPackages } from '_hooks'; import { formatDate, getBalanceChangeSummary, - getLabel, + getTransactionAction, useFormatCoin, useTransactionSummary, + TransactionIcon, } from '@iota/core'; import type { IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; import { Link } from 'react-router-dom'; -import { TxnIcon } from './TxnIcon'; import { Card, CardType, @@ -40,7 +40,7 @@ export function TransactionCard({ txn, address }: TransactionCardProps) { recognizedPackagesList, }); - // we only show IOTA Transfer amount or the first non-Iota transfer amount + // we only show IOTA Transfer amount or the first non-IOTA transfer amount // Get the balance changes for the transaction and the amount const balanceChanges = getBalanceChangeSummary(txn, recognizedPackagesList); const [formatAmount, symbol] = useFormatCoin( @@ -52,7 +52,7 @@ export function TransactionCard({ txn, address }: TransactionCardProps) { const transactionDate = !txn.timestampMs ? '--' - : formatDate(Number(txn.timestampMs), ['month', 'day', 'hour', 'minute']); + : formatDate(Number(txn.timestampMs), ['day', 'month', 'year', 'hour', 'minute']); return ( <Link @@ -64,9 +64,9 @@ export function TransactionCard({ txn, address }: TransactionCardProps) { > <Card type={CardType.Default} isHoverable> <CardImage type={ImageType.BgSolid} shape={ImageShape.SquareRounded}> - <TxnIcon + <TransactionIcon txnFailed={executionStatus !== 'success' || !!error} - variant={getLabel(txn, address)} + variant={getTransactionAction(txn, address)} /> </CardImage> <CardBody diff --git a/apps/wallet/src/ui/app/components/user-approve-container/index.tsx b/apps/wallet/src/ui/app/components/user-approve-container/index.tsx index 3a994d63f6b..190da9dc546 100644 --- a/apps/wallet/src/ui/app/components/user-approve-container/index.tsx +++ b/apps/wallet/src/ui/app/components/user-approve-container/index.tsx @@ -7,7 +7,7 @@ import cn from 'clsx'; import type { ReactNode } from 'react'; import { useCallback, useMemo, useState } from 'react'; import { Button, ButtonType, Header, LoadingIndicator } from '@iota/apps-ui-kit'; -import { useAccountByAddress } from '../../hooks/useAccountByAddress'; +import { useAccountByAddress } from '_hooks'; import { DAppInfoCard, UnlockAccountButton } from '_components'; interface UserApproveContainerProps { diff --git a/apps/wallet/src/ui/app/experimentation/feature-gating.ts b/apps/wallet/src/ui/app/experimentation/featureGating.ts similarity index 100% rename from apps/wallet/src/ui/app/experimentation/feature-gating.ts rename to apps/wallet/src/ui/app/experimentation/featureGating.ts diff --git a/apps/wallet/src/ui/app/helpers/accounts.ts b/apps/wallet/src/ui/app/helpers/accounts.ts index 55f3d9a5d87..215afff629e 100644 --- a/apps/wallet/src/ui/app/helpers/accounts.ts +++ b/apps/wallet/src/ui/app/helpers/accounts.ts @@ -2,9 +2,9 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { AccountType, type SerializedUIAccount } from '_src/background/accounts/Account'; -import { isMnemonicSerializedUiAccount } from '_src/background/accounts/MnemonicAccount'; -import { isSeedSerializedUiAccount } from '_src/background/accounts/SeedAccount'; +import { AccountType, type SerializedUIAccount } from '_src/background/accounts/account'; +import { isMnemonicSerializedUiAccount } from '_src/background/accounts/mnemonicAccount'; +import { isSeedSerializedUiAccount } from '_src/background/accounts/seedAccount'; export function getKey(account: SerializedUIAccount): string { if (isMnemonicSerializedUiAccount(account)) return account.sourceID; diff --git a/apps/wallet/src/ui/app/helpers/checkStakingTxn.ts b/apps/wallet/src/ui/app/helpers/checkStakingTxn.ts deleted file mode 100644 index a0022b8a10b..00000000000 --- a/apps/wallet/src/ui/app/helpers/checkStakingTxn.ts +++ /dev/null @@ -1,10 +0,0 @@ -// Copyright (c) Mysten Labs, Inc. -// Modifications Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -import type { IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; - -// TODO: Support programmable transactions: -export function checkStakingTxn(_txn: IotaTransactionBlockResponse) { - return false; -} diff --git a/apps/wallet/src/ui/app/helpers/formatDate.ts b/apps/wallet/src/ui/app/helpers/formatDate.ts deleted file mode 100644 index d6aaab10be8..00000000000 --- a/apps/wallet/src/ui/app/helpers/formatDate.ts +++ /dev/null @@ -1,32 +0,0 @@ -// Copyright (c) Mysten Labs, Inc. -// Modifications Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -// TODO - handle multiple date formats -// Wed Aug 05 -// -type Show = 'year' | 'month' | 'day' | 'hour' | 'minute' | 'second' | 'weekday'; -export default function formatDate(timeStamp: number, show: Show[]): string { - const date = new Date(timeStamp); - if (!(date instanceof Date) || !show.length) return ''; - - const OPTIONS = { - year: 'numeric', - month: 'short', - day: 'numeric', - hour: 'numeric', - weekday: 'short', - minute: 'numeric', - second: 'numeric', - }; - - const formatOptions = show.reduce((accumulator, current: Show) => { - const responseObj = { - ...accumulator, - ...{ [current]: OPTIONS[current] }, - }; - return responseObj; - }, {}); - - return new Intl.DateTimeFormat('en-US', formatOptions).format(date); -} diff --git a/apps/wallet/src/ui/app/helpers/getEventsSummary.ts b/apps/wallet/src/ui/app/helpers/getEventsSummary.ts deleted file mode 100644 index e6baefa2c33..00000000000 --- a/apps/wallet/src/ui/app/helpers/getEventsSummary.ts +++ /dev/null @@ -1,80 +0,0 @@ -// Copyright (c) Mysten Labs, Inc. -// Modifications Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -export {}; - -// import { -// getCoinBalanceChangeEvent, -// getTransferObjectEvent, -// isEventType, -// type TransactionEvents, -// } from '@iota/iota-sdk'; - -// export type CoinsMetaProps = { -// amount: number; -// coinType: string; -// receiverAddress: string; -// }; - -// export type TxnMetaResponse = { -// objectIDs: string[]; -// coins: CoinsMetaProps[]; -// }; - -// export function getEventsSummary( -// events: TransactionEvents, -// address: string -// ): TxnMetaResponse { -// const coinsMeta = {} as { [coinType: string]: CoinsMetaProps }; -// const objectIDs: string[] = []; - -// events.forEach((event) => { -// // Aggregate coinBalanceChange by coinType and address -// // A net positive amount means the user received coins -// // A net negative amount means the user sent coins -// if ( -// event.type === 'coinBalanceChange' && -// event?.content?.changeType && -// ['Receive', 'Pay'].includes(event?.content?.changeType) -// ) { -// const coinBalanceChange = getCoinBalanceChangeEvent(event)!; -// const { coinType, amount, owner, sender } = coinBalanceChange; - -// const AddressOwner = -// owner !== 'Immutable' && 'AddressOwner' in owner -// ? owner.AddressOwner -// : null; - -// // ChangeEpoch txn includes coinBalanceChange event for other addresses -// if ( -// AddressOwner === address || -// (address === sender && AddressOwner) -// ) { -// coinsMeta[`${AddressOwner}${coinType}`] = { -// amount: -// (coinsMeta[`${AddressOwner}${coinType}`]?.amount || 0) + -// +amount, -// coinType: coinType, -// receiverAddress: AddressOwner, -// }; -// } -// } - -// // return objectIDs of the transfer objects -// if (isEventType(event, 'transferObject')) { -// const transferObject = getTransferObjectEvent(event)!; -// const { AddressOwner } = transferObject.recipient as { -// AddressOwner: string; -// }; -// if (AddressOwner === address) { -// objectIDs.push(transferObject?.objectId); -// } -// } -// }); - -// return { -// objectIDs, -// coins: Object.values(coinsMeta), -// }; -// } diff --git a/apps/wallet/src/ui/app/helpers/index.ts b/apps/wallet/src/ui/app/helpers/index.ts index fde8b2c4f27..2c7c2b77137 100644 --- a/apps/wallet/src/ui/app/helpers/index.ts +++ b/apps/wallet/src/ui/app/helpers/index.ts @@ -2,9 +2,9 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export { default as formatDate } from './formatDate'; -export { default as notEmpty } from './notEmptyCheck'; -// export { getEventsSummary } from './getEventsSummary'; -export { getAmount } from './getAmount'; -export { checkStakingTxn } from './checkStakingTxn'; -export { formatAccountName } from './formatAccountName'; +export * from './getAmount'; +export * from './formatAccountName'; +export * from './accounts'; +export * from './errorMessages'; +export * from './sentry'; +export * from './queryClient'; diff --git a/apps/wallet/src/ui/app/helpers/notEmptyCheck.ts b/apps/wallet/src/ui/app/helpers/notEmptyCheck.ts deleted file mode 100644 index 1156113349a..00000000000 --- a/apps/wallet/src/ui/app/helpers/notEmptyCheck.ts +++ /dev/null @@ -1,8 +0,0 @@ -// Copyright (c) Mysten Labs, Inc. -// Modifications Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -export default function notEmpty<TValue>(value: TValue | null | undefined): value is TValue { - if (value === null || value === undefined) return false; - return true; -} diff --git a/apps/wallet/src/ui/app/helpers/queryClient.ts b/apps/wallet/src/ui/app/helpers/queryClient.ts index 7b7ccaf349e..54b56da5298 100644 --- a/apps/wallet/src/ui/app/helpers/queryClient.ts +++ b/apps/wallet/src/ui/app/helpers/queryClient.ts @@ -6,6 +6,8 @@ import { QueryClient } from '@tanstack/react-query'; import { type PersistedClient, type Persister } from '@tanstack/react-query-persist-client'; import { del, get, set } from 'idb-keyval'; +export const ACCOUNTS_QUERY_KEY = ['background', 'client', 'accounts']; + export const queryClient = new QueryClient({ defaultOptions: { queries: { diff --git a/apps/wallet/src/ui/app/helpers/query-client-keys.ts b/apps/wallet/src/ui/app/helpers/queryClientKeys.ts similarity index 100% rename from apps/wallet/src/ui/app/helpers/query-client-keys.ts rename to apps/wallet/src/ui/app/helpers/queryClientKeys.ts diff --git a/apps/wallet/src/ui/app/helpers/sentry.ts b/apps/wallet/src/ui/app/helpers/sentry.ts index 7be751f15d2..3204e834c82 100644 --- a/apps/wallet/src/ui/app/helpers/sentry.ts +++ b/apps/wallet/src/ui/app/helpers/sentry.ts @@ -2,16 +2,15 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { growthbook } from '_src/ui/app/experimentation/feature-gating'; +import { growthbook } from '_src/ui/app/experimentation/featureGating'; import * as Sentry from '@sentry/react'; import { Feature } from '@iota/core'; +import { getSentryConfig } from '../../../shared/sentryConfig'; -import { getSentryConfig } from '../../../shared/sentry-config'; - -export default function initSentry() { +export function initSentry() { Sentry.init( getSentryConfig({ - integrations: [new Sentry.BrowserTracing()], + integrations: [Sentry.browserTracingIntegration()], tracesSampler: () => { return growthbook.getFeatureValue(Feature.WalletSentryTracing, 0); }, diff --git a/apps/wallet/src/ui/app/hooks/index.ts b/apps/wallet/src/ui/app/hooks/index.ts index f788f5dce44..6ac1ac8378d 100644 --- a/apps/wallet/src/ui/app/hooks/index.ts +++ b/apps/wallet/src/ui/app/hooks/index.ts @@ -2,19 +2,38 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -export { default as useAppDispatch } from './useAppDispatch'; -export { default as useAppSelector } from './useAppSelector'; -export { default as useInitializedGuard } from './useInitializedGuard'; -export { default as useFullscreenGuard } from './useFullscreenGuard'; -export { default as useOnClickOutside } from './useOnClickOutside'; -export { default as useOnKeyboardEvent } from './useOnKeyboardEvent'; -export { useTransactionDryRun } from './useTransactionDryRun'; -export { useGetTxnRecipientAddress } from './useGetTxnRecipientAddress'; -export { useGetTransferAmount } from './useGetTransferAmount'; -export { useCopyToClipboard } from './useCopyToClipboard'; +export * from './useAppDispatch'; +export * from './useAppSelector'; +export * from './useInitializedGuard'; +export * from './useFullscreenGuard'; +export * from './useOnKeyboardEvent'; +export * from './useTransactionDryRun'; +export * from './useCopyToClipboard'; export * from './useExplorerLink'; - export * from './useTransactionData'; export * from './useActiveAddress'; export * from './useCoinsReFetchingConfig'; export * from './useSetGrowthbookAttributes'; +export * from './useAccountByAddress'; +export * from './useAccountGroups'; +export * from './useAccountSources'; +export * from './useAccounts'; +export * from './useAccountsFinder'; +export * from './useActiveAccount'; +export * from './useAutoLockMinutes'; +export * from './useAutoLockMinutesMutation'; +export * from './useBackgroundClient'; +export * from './useCreateAccountsMutation'; +export * from './useExportPassphraseMutation'; +export * from './useExportSeedMutation'; +export * from './useInitialPageView'; +export * from './usePinnedCoinTypes'; +export * from './useRecognizedPackages'; +export * from './useRecoveryDataMutation'; +export * from './useResetPasswordMutation'; +export * from './useResolveVideo'; +export * from './useRestrictedGuard'; +export * from './useSigner'; +export * from './useStorageMigrationStatus'; +export * from './useUnlockMutation'; +export * from './useUnlockedGuard'; diff --git a/apps/wallet/src/ui/app/hooks/useAccountSources.ts b/apps/wallet/src/ui/app/hooks/useAccountSources.ts index f5828295431..0059e441303 100644 --- a/apps/wallet/src/ui/app/hooks/useAccountSources.ts +++ b/apps/wallet/src/ui/app/hooks/useAccountSources.ts @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type AccountSourceSerializedUI } from '_src/background/account-sources/AccountSource'; +import { type AccountSourceSerializedUI } from '_src/background/account-sources/accountSource'; import { useQuery } from '@tanstack/react-query'; import { useBackgroundClient } from './useBackgroundClient'; diff --git a/apps/wallet/src/ui/app/hooks/useAccounts.ts b/apps/wallet/src/ui/app/hooks/useAccounts.ts index a548c0fab97..280315ff6a5 100644 --- a/apps/wallet/src/ui/app/hooks/useAccounts.ts +++ b/apps/wallet/src/ui/app/hooks/useAccounts.ts @@ -2,10 +2,9 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type SerializedUIAccount } from '_src/background/accounts/Account'; +import { type SerializedUIAccount } from '_src/background/accounts/account'; import { useQuery } from '@tanstack/react-query'; - -import { ACCOUNTS_QUERY_KEY } from '../helpers/query-client-keys'; +import { ACCOUNTS_QUERY_KEY } from '../helpers/queryClientKeys'; import { useBackgroundClient } from './useBackgroundClient'; export function useAccounts() { diff --git a/apps/wallet/src/ui/app/hooks/useAccountsFinder.ts b/apps/wallet/src/ui/app/hooks/useAccountsFinder.ts index ae85a99612b..7ba9534970f 100644 --- a/apps/wallet/src/ui/app/hooks/useAccountsFinder.ts +++ b/apps/wallet/src/ui/app/hooks/useAccountsFinder.ts @@ -12,7 +12,7 @@ import type { } from '_src/shared/messaging/messages/payloads/accounts-finder'; import { makeDerivationPath } from '_src/background/account-sources/bip44Path'; import { Ed25519PublicKey } from '@iota/iota-sdk/keypairs/ed25519'; -import { IOTA_BIP44_COIN_TYPE } from '../redux/slices/iota-objects/Coin'; +import { IOTA_BIP44_COIN_TYPE } from '../redux/slices/iota-objects/coin'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; export interface UseAccountFinderOptions { diff --git a/apps/wallet/src/ui/app/hooks/useAddressLink.ts b/apps/wallet/src/ui/app/hooks/useAddressLink.ts deleted file mode 100644 index 86d0956d087..00000000000 --- a/apps/wallet/src/ui/app/hooks/useAddressLink.ts +++ /dev/null @@ -1,20 +0,0 @@ -// Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -import { useExplorerLink } from '_app/hooks/useExplorerLink'; -import { ExplorerLinkType } from '_components'; -import { formatAddress } from '@iota/iota-sdk/utils'; - -export function useAddressLink(inputAddress: string | null) { - const outputAddress = inputAddress || ''; - const explorerHref = useExplorerLink({ - type: ExplorerLinkType.Address, - address: outputAddress || undefined, - }); - - return { - explorerHref: explorerHref || '', - addressFull: inputAddress || '', - address: formatAddress(outputAddress), - }; -} diff --git a/apps/wallet/src/ui/app/hooks/useAppDispatch.ts b/apps/wallet/src/ui/app/hooks/useAppDispatch.ts index b02feed0d22..4e475ac3606 100644 --- a/apps/wallet/src/ui/app/hooks/useAppDispatch.ts +++ b/apps/wallet/src/ui/app/hooks/useAppDispatch.ts @@ -5,6 +5,6 @@ import type { AppDispatch } from '_store'; import { useDispatch } from 'react-redux'; -export default function useAppDispatch() { +export function useAppDispatch() { return useDispatch<AppDispatch>(); } diff --git a/apps/wallet/src/ui/app/hooks/useAppSelector.ts b/apps/wallet/src/ui/app/hooks/useAppSelector.ts index 73eef316546..f2eca930bdf 100644 --- a/apps/wallet/src/ui/app/hooks/useAppSelector.ts +++ b/apps/wallet/src/ui/app/hooks/useAppSelector.ts @@ -2,10 +2,8 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import type { RootState } from '_redux/RootReducer'; +import type { RootState } from '_src/ui/app/redux/rootReducer'; import { useSelector } from 'react-redux'; import type { TypedUseSelectorHook } from 'react-redux'; -const useAppSelector: TypedUseSelectorHook<RootState> = useSelector; - -export default useAppSelector; +export const useAppSelector: TypedUseSelectorHook<RootState> = useSelector; diff --git a/apps/wallet/src/ui/app/hooks/useAutoLockMinutesMutation.ts b/apps/wallet/src/ui/app/hooks/useAutoLockMinutesMutation.ts index 07887bae398..30e679719a0 100644 --- a/apps/wallet/src/ui/app/hooks/useAutoLockMinutesMutation.ts +++ b/apps/wallet/src/ui/app/hooks/useAutoLockMinutesMutation.ts @@ -3,7 +3,6 @@ // SPDX-License-Identifier: Apache-2.0 import { useMutation, useQueryClient } from '@tanstack/react-query'; - import { AUTO_LOCK_MINUTES_QUERY_KEY } from './useAutoLockMinutes'; import { useBackgroundClient } from './useBackgroundClient'; diff --git a/apps/wallet/src/ui/app/hooks/useBackgroundClient.ts b/apps/wallet/src/ui/app/hooks/useBackgroundClient.ts index ef19242b4d2..5d28ea623aa 100644 --- a/apps/wallet/src/ui/app/hooks/useBackgroundClient.ts +++ b/apps/wallet/src/ui/app/hooks/useBackgroundClient.ts @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { thunkExtras } from '../redux/store/thunk-extras'; +import { thunkExtras } from '../redux/store/thunkExtras'; export function useBackgroundClient() { return thunkExtras.background; diff --git a/apps/wallet/src/ui/app/hooks/useCountAccountByType.ts b/apps/wallet/src/ui/app/hooks/useCountAccountByType.ts deleted file mode 100644 index 0f722f25c72..00000000000 --- a/apps/wallet/src/ui/app/hooks/useCountAccountByType.ts +++ /dev/null @@ -1,22 +0,0 @@ -// Copyright (c) Mysten Labs, Inc. -// Modifications Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -import { type AccountType } from '_src/background/accounts/Account'; -import { useMemo } from 'react'; - -import { useAccounts } from './useAccounts'; - -export function useCountAccountsByType() { - const { data: accounts, isPending } = useAccounts(); - const countPerType = useMemo( - () => - accounts?.reduce<Partial<Record<AccountType, { total: number }>>>((acc, anAccount) => { - acc[anAccount.type] = acc[anAccount.type] || { total: 0 }; - acc[anAccount.type]!.total++; - return acc; - }, {}) || {}, - [accounts], - ); - return { data: countPerType, isPending }; -} diff --git a/apps/wallet/src/ui/app/hooks/useCreateAccountMutation.ts b/apps/wallet/src/ui/app/hooks/useCreateAccountsMutation.ts similarity index 90% rename from apps/wallet/src/ui/app/hooks/useCreateAccountMutation.ts rename to apps/wallet/src/ui/app/hooks/useCreateAccountsMutation.ts index 843f4462a53..65ce909edcc 100644 --- a/apps/wallet/src/ui/app/hooks/useCreateAccountMutation.ts +++ b/apps/wallet/src/ui/app/hooks/useCreateAccountsMutation.ts @@ -7,7 +7,7 @@ import { useMutation } from '@tanstack/react-query'; import { useAccountsFormContext, AccountsFormType, type AccountsFormValues } from '_components'; import { useBackgroundClient } from './useBackgroundClient'; -import { AccountType } from '_src/background/accounts/Account'; +import { AccountType } from '_src/background/accounts/account'; function validateAccountFormValues<T extends AccountsFormType>( createType: T, @@ -36,22 +36,21 @@ enum AmpliAccountType { Ledger = 'Ledger', } -const CREATE_TYPE_TO_AMPLI_ACCOUNT: Record< - AccountsFormType, - AddedAccountsProperties['accountType'] -> = { - [AccountsFormType.NewMnemonic]: AmpliAccountType.Derived, - [AccountsFormType.ImportMnemonic]: AmpliAccountType.Derived, - [AccountsFormType.ImportSeed]: AmpliAccountType.Derived, - [AccountsFormType.MnemonicSource]: AmpliAccountType.Derived, - [AccountsFormType.SeedSource]: AmpliAccountType.Derived, - [AccountsFormType.ImportPrivateKey]: AmpliAccountType.ImportPrivateKey, - [AccountsFormType.ImportLedger]: AmpliAccountType.Ledger, -}; - export function useCreateAccountsMutation() { const backgroundClient = useBackgroundClient(); const [accountsFormValuesRef, setAccountFormValues] = useAccountsFormContext(); + const CREATE_TYPE_TO_AMPLI_ACCOUNT: Record< + AccountsFormType, + AddedAccountsProperties['accountType'] + > = { + [AccountsFormType.NewMnemonic]: AmpliAccountType.Derived, + [AccountsFormType.ImportMnemonic]: AmpliAccountType.Derived, + [AccountsFormType.ImportSeed]: AmpliAccountType.Derived, + [AccountsFormType.MnemonicSource]: AmpliAccountType.Derived, + [AccountsFormType.SeedSource]: AmpliAccountType.Derived, + [AccountsFormType.ImportPrivateKey]: AmpliAccountType.ImportPrivateKey, + [AccountsFormType.ImportLedger]: AmpliAccountType.Ledger, + }; return useMutation({ mutationKey: ['create accounts'], mutationFn: async ({ type, password }: { type: AccountsFormType; password?: string }) => { diff --git a/apps/wallet/src/ui/app/hooks/useExplorerLink.ts b/apps/wallet/src/ui/app/hooks/useExplorerLink.ts index 33bfb54dd1b..5320791655a 100644 --- a/apps/wallet/src/ui/app/hooks/useExplorerLink.ts +++ b/apps/wallet/src/ui/app/hooks/useExplorerLink.ts @@ -4,7 +4,7 @@ import { type ExplorerLinkConfig, getExplorerLink as useGetExplorerLink } from '@iota/core'; import { useActiveAddress } from './useActiveAddress'; -import useAppSelector from './useAppSelector'; +import { useAppSelector } from './useAppSelector'; export function useExplorerLink(linkConfig: ExplorerLinkConfig) { const app = useAppSelector(({ app }) => app); diff --git a/apps/wallet/src/ui/app/hooks/useExportPassphraseMutation.ts b/apps/wallet/src/ui/app/hooks/useExportPassphraseMutation.ts index a55ff565788..4fcbc57cdba 100644 --- a/apps/wallet/src/ui/app/hooks/useExportPassphraseMutation.ts +++ b/apps/wallet/src/ui/app/hooks/useExportPassphraseMutation.ts @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type MethodPayload } from '_src/shared/messaging/messages/payloads/MethodPayload'; +import { type MethodPayload } from '_src/shared/messaging/messages/payloads/methodPayload'; import { entropyToMnemonic, toEntropy } from '_src/shared/utils'; import { useMutation } from '@tanstack/react-query'; diff --git a/apps/wallet/src/ui/app/hooks/useExportSeedMutation.ts b/apps/wallet/src/ui/app/hooks/useExportSeedMutation.ts index ba08ea658e1..fb53230b57e 100644 --- a/apps/wallet/src/ui/app/hooks/useExportSeedMutation.ts +++ b/apps/wallet/src/ui/app/hooks/useExportSeedMutation.ts @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type MethodPayload } from '_src/shared/messaging/messages/payloads/MethodPayload'; +import { type MethodPayload } from '_src/shared/messaging/messages/payloads/methodPayload'; import { useMutation } from '@tanstack/react-query'; import { useBackgroundClient } from './useBackgroundClient'; diff --git a/apps/wallet/src/ui/app/hooks/useFullscreenGuard.ts b/apps/wallet/src/ui/app/hooks/useFullscreenGuard.ts index 6e38ea7e66a..6c09bedfaca 100644 --- a/apps/wallet/src/ui/app/hooks/useFullscreenGuard.ts +++ b/apps/wallet/src/ui/app/hooks/useFullscreenGuard.ts @@ -2,13 +2,12 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { AppType } from '_redux/slices/app/AppType'; +import { AppType } from '_src/ui/app/redux/slices/app/appType'; import { openInNewTab } from '_shared/utils'; import { useEffect, useRef } from 'react'; +import { useAppSelector } from './useAppSelector'; -import useAppSelector from './useAppSelector'; - -export default function useFullscreenGuard(enabled: boolean) { +export function useFullscreenGuard(enabled: boolean) { const appType = useAppSelector((state) => state.app.appType); const isOpenTabInProgressRef = useRef(false); useEffect(() => { diff --git a/apps/wallet/src/ui/app/hooks/useGetTransferAmount.ts b/apps/wallet/src/ui/app/hooks/useGetTransferAmount.ts deleted file mode 100644 index 24dc6c0d62a..00000000000 --- a/apps/wallet/src/ui/app/hooks/useGetTransferAmount.ts +++ /dev/null @@ -1,46 +0,0 @@ -// Copyright (c) Mysten Labs, Inc. -// Modifications Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -import { getAmount } from '_helpers'; -import type { IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; -import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; -import { useMemo } from 'react'; - -export function useGetTransferAmount({ - txn, - activeAddress, -}: { - txn: IotaTransactionBlockResponse; - activeAddress: string; -}) { - const { effects, events } = txn; - // const { coins } = getEventsSummary(events!, activeAddress); - - const iotaTransfer = useMemo(() => { - const txdetails = txn.transaction?.data.transaction; - return !txdetails - ? [] - : getAmount(txdetails, effects!, events!)?.map( - ({ amount, coinType, recipientAddress }) => { - return { - amount: amount || 0, - coinType: coinType || IOTA_TYPE_ARG, - receiverAddress: recipientAddress, - }; - }, - ); - }, [txn, effects, events]); - - // MUSTFIX(chris) - // const transferAmount = useMemo(() => { - // return iotaTransfer?.length - // ? iotaTransfer - // : coins.filter( - // ({ receiverAddress }) => receiverAddress === activeAddress - // ); - // }, [iotaTransfer, coins, activeAddress]); - - // return iotaTransfer ?? transferAmount; - return iotaTransfer; -} diff --git a/apps/wallet/src/ui/app/hooks/useGetTxnRecipientAddress.ts b/apps/wallet/src/ui/app/hooks/useGetTxnRecipientAddress.ts deleted file mode 100644 index 35196877b94..00000000000 --- a/apps/wallet/src/ui/app/hooks/useGetTxnRecipientAddress.ts +++ /dev/null @@ -1,44 +0,0 @@ -// Copyright (c) Mysten Labs, Inc. -// Modifications Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -import { getAmount } from '_helpers'; -import { type IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; -import { useMemo } from 'react'; - -type Props = { - txn: IotaTransactionBlockResponse; - address: string; -}; - -export function useGetTxnRecipientAddress({ txn, address }: Props) { - const events = txn.events!; - - // const eventsSummary = useMemo(() => { - // const { coins } = getEventsSummary(events, address); - // return coins; - // }, [events, address]); - - const transaction = txn.transaction?.data.transaction; - const amountByRecipient = !transaction ? null : getAmount(transaction, txn.effects!, events); - - const recipientAddress = useMemo(() => { - const transferObjectRecipientAddress = - amountByRecipient && - amountByRecipient?.find(({ recipientAddress }) => recipientAddress !== address) - ?.recipientAddress; - // MUSTFIX(chris) - // const receiverAddr = - // eventsSummary && - // eventsSummary.find( - // ({ receiverAddress }) => receiverAddress !== address - // )?.receiverAddress; - - // eslint-disable-next-line @typescript-eslint/ban-ts-comment - // @ts-ignore - return null ?? transferObjectRecipientAddress ?? txn.transaction?.data.sender; - }, [address, amountByRecipient, txn]); - // }, [address, amountByRecipient, eventsSummary, txn]); - - return recipientAddress; -} diff --git a/apps/wallet/src/ui/app/hooks/useInitialPageView.ts b/apps/wallet/src/ui/app/hooks/useInitialPageView.ts index 1fde2375182..3a163c4a23e 100644 --- a/apps/wallet/src/ui/app/hooks/useInitialPageView.ts +++ b/apps/wallet/src/ui/app/hooks/useInitialPageView.ts @@ -8,10 +8,9 @@ import { getNetwork } from '@iota/iota-sdk/client'; import { useEffect } from 'react'; import { useLocation } from 'react-router-dom'; import Browser from 'webextension-polyfill'; - -import { AppType } from '../redux/slices/app/AppType'; +import { AppType } from '../redux/slices/app/appType'; import { useActiveAccount } from './useActiveAccount'; -import useAppSelector from './useAppSelector'; +import { useAppSelector } from './useAppSelector'; export function useInitialPageView() { const activeAccount = useActiveAccount(); diff --git a/apps/wallet/src/ui/app/hooks/useInitializedGuard.ts b/apps/wallet/src/ui/app/hooks/useInitializedGuard.ts index 16a735611c9..7ff822116d0 100644 --- a/apps/wallet/src/ui/app/hooks/useInitializedGuard.ts +++ b/apps/wallet/src/ui/app/hooks/useInitializedGuard.ts @@ -4,11 +4,10 @@ import { useEffect } from 'react'; import { useNavigate } from 'react-router-dom'; - import { useAccounts } from './useAccounts'; import { useRestrictedGuard } from './useRestrictedGuard'; -export default function useInitializedGuard(initializedRequired: boolean, enabled = true) { +export function useInitializedGuard(initializedRequired: boolean, enabled = true) { const restricted = useRestrictedGuard(); const { data: allAccounts, isPending } = useAccounts(); const isInitialized = !!allAccounts?.length; diff --git a/apps/wallet/src/ui/app/hooks/useOnClickOutside.ts b/apps/wallet/src/ui/app/hooks/useOnClickOutside.ts deleted file mode 100644 index 4cd2e61541b..00000000000 --- a/apps/wallet/src/ui/app/hooks/useOnClickOutside.ts +++ /dev/null @@ -1,34 +0,0 @@ -// Copyright (c) Mysten Labs, Inc. -// Modifications Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -import { useEffect } from 'react'; -import type { RefObject } from 'react'; - -type Event = MouseEvent | TouchEvent; - -const useOnClickOutside = <T extends HTMLElement = HTMLElement>( - ref: RefObject<T>, - handler: (event: Event) => void, -) => { - useEffect(() => { - const listener = (event: Event) => { - const el = ref?.current; - if (!el || el.contains(event?.target as Node)) { - return; - } - - handler(event); // Call the handler only if the click is outside of the element passed. - }; - - document.addEventListener('click', listener, true); - document.addEventListener('touchstart', listener, true); - - return () => { - document.removeEventListener('click', listener, true); - document.removeEventListener('touchstart', listener, true); - }; - }, [ref, handler]); // Reload only if ref or handler changes -}; - -export default useOnClickOutside; diff --git a/apps/wallet/src/ui/app/hooks/useOnKeyboardEvent.ts b/apps/wallet/src/ui/app/hooks/useOnKeyboardEvent.ts index 160745ce239..34dc3e886ad 100644 --- a/apps/wallet/src/ui/app/hooks/useOnKeyboardEvent.ts +++ b/apps/wallet/src/ui/app/hooks/useOnKeyboardEvent.ts @@ -4,7 +4,7 @@ import { useEffect } from 'react'; -function useOnKeyboardEvent<K extends 'keydown' | 'keyup' | 'keypress'>( +export function useOnKeyboardEvent<K extends 'keydown' | 'keyup' | 'keypress'>( eventType: K, keys: string[], handler: (e: KeyboardEvent) => void, @@ -26,5 +26,3 @@ function useOnKeyboardEvent<K extends 'keydown' | 'keyup' | 'keypress'>( } }, [eventType, keys, handler, enabled]); } - -export default useOnKeyboardEvent; diff --git a/apps/wallet/src/ui/app/hooks/useRecognizedPackages.ts b/apps/wallet/src/ui/app/hooks/useRecognizedPackages.ts index 7c9eb1d0f55..6e7a1be92c6 100644 --- a/apps/wallet/src/ui/app/hooks/useRecognizedPackages.ts +++ b/apps/wallet/src/ui/app/hooks/useRecognizedPackages.ts @@ -4,7 +4,7 @@ import { useFeatureValue } from '@growthbook/growthbook-react'; import { Network } from '@iota/iota-sdk/client'; -import useAppSelector from './useAppSelector'; +import { useAppSelector } from './useAppSelector'; import { DEFAULT_RECOGNIZED_PACKAGES, Feature } from '@iota/core'; export function useRecognizedPackages() { diff --git a/apps/wallet/src/ui/app/hooks/useRecoveryDataMutation.ts b/apps/wallet/src/ui/app/hooks/useRecoveryDataMutation.ts index 0567bef0896..8db1f1d483d 100644 --- a/apps/wallet/src/ui/app/hooks/useRecoveryDataMutation.ts +++ b/apps/wallet/src/ui/app/hooks/useRecoveryDataMutation.ts @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type PasswordRecoveryData } from '_src/shared/messaging/messages/payloads/MethodPayload'; +import { type PasswordRecoveryData } from '_src/shared/messaging/messages/payloads/methodPayload'; import { useMutation } from '@tanstack/react-query'; import { useForgotPasswordContext } from '../pages/accounts/forgot-password/ForgotPasswordPage'; diff --git a/apps/wallet/src/ui/app/hooks/useSigner.ts b/apps/wallet/src/ui/app/hooks/useSigner.ts index a7e6398a071..3fa8ba95b2c 100644 --- a/apps/wallet/src/ui/app/hooks/useSigner.ts +++ b/apps/wallet/src/ui/app/hooks/useSigner.ts @@ -2,14 +2,14 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type SerializedUIAccount } from '_src/background/accounts/Account'; -import { isLedgerAccountSerializedUI } from '_src/background/accounts/LedgerAccount'; +import { type SerializedUIAccount } from '_src/background/accounts/account'; +import { isLedgerAccountSerializedUI } from '_src/background/accounts/ledgerAccount'; import { useIotaClient } from '@iota/dapp-kit'; -import { walletApiProvider } from '../ApiProvider'; +import { walletApiProvider } from '../apiProvider'; import { useIotaLedgerClient } from '_components'; -import { LedgerSigner } from '../LedgerSigner'; -import { type WalletSigner } from '../WalletSigner'; +import { LedgerSigner } from '../ledgerSigner'; +import { type WalletSigner } from '../walletSigner'; import { useBackgroundClient } from './useBackgroundClient'; export function useSigner(account: SerializedUIAccount | null): WalletSigner | null { diff --git a/apps/wallet/src/ui/app/hooks/useUnlockedGuard.ts b/apps/wallet/src/ui/app/hooks/useUnlockedGuard.ts index fb30507a6c3..ccbef72b3e3 100644 --- a/apps/wallet/src/ui/app/hooks/useUnlockedGuard.ts +++ b/apps/wallet/src/ui/app/hooks/useUnlockedGuard.ts @@ -4,9 +4,7 @@ import { useEffect } from 'react'; import { useNavigate } from 'react-router-dom'; - -import { useAccounts } from '../hooks/useAccounts'; -import { useActiveAccount } from '../hooks/useActiveAccount'; +import { useAccounts, useActiveAccount } from '_hooks'; export function useUnlockedGuard() { const navigate = useNavigate(); diff --git a/apps/wallet/src/ui/app/index.tsx b/apps/wallet/src/ui/app/index.tsx index 77f06c3f03d..b821e43fb8f 100644 --- a/apps/wallet/src/ui/app/index.tsx +++ b/apps/wallet/src/ui/app/index.tsx @@ -2,21 +2,24 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useAppDispatch, useAppSelector } from '_hooks'; +import { + useAutoLockMinutes, + useBackgroundClient, + useInitialPageView, + useStorageMigrationStatus, + useAccounts, + useAppDispatch, + useAppSelector, +} from './hooks'; import { setNavVisibility } from '_redux/slices/app'; -import { isLedgerAccountSerializedUI } from '_src/background/accounts/LedgerAccount'; +import { isLedgerAccountSerializedUI } from '_src/background/accounts/ledgerAccount'; import { persistableStorage } from '_src/shared/analytics/amplitude'; -import { type LedgerAccountsPublicKeys } from '_src/shared/messaging/messages/payloads/MethodPayload'; +import { type LedgerAccountsPublicKeys } from '_src/shared/messaging/messages/payloads/methodPayload'; import { toB64 } from '@iota/iota-sdk/utils'; import { useEffect, useMemo } from 'react'; import { Navigate, Route, Routes, useLocation } from 'react-router-dom'; import { throttle } from 'throttle-debounce'; - import { useIotaLedgerClient } from './components/ledger/IotaLedgerClientProvider'; -import { useAccounts } from './hooks/useAccounts'; -import { useAutoLockMinutes } from './hooks/useAutoLockMinutes'; -import { useBackgroundClient } from './hooks/useBackgroundClient'; -import { useInitialPageView } from './hooks/useInitialPageView'; import { AccountsPage } from './pages/accounts/AccountsPage'; import { AddAccountPage } from './pages/accounts/AddAccountPage'; import { BackupMnemonicPage } from './pages/accounts/BackupMnemonicPage'; @@ -37,7 +40,8 @@ import { ManageAccountsPage } from './pages/accounts/manage/ManageAccountsPage'; import { ProtectAccountPage } from './pages/accounts/ProtectAccountPage'; import { WelcomePage } from './pages/accounts/WelcomePage'; import { ApprovalRequestPage } from './pages/approval-request'; -import HomePage, { +import { + HomePage, AppsPage, AssetsPage, CoinsSelectorPage, @@ -48,14 +52,14 @@ import HomePage, { TransactionBlocksPage, TransferCoinPage, } from './pages/home'; -import TokenDetailsPage from './pages/home/tokens/TokenDetailsPage'; +import { TokenDetailsPage } from './pages/home/tokens/TokenDetailsPage'; import { RestrictedPage } from './pages/restricted'; -import SiteConnectPage from './pages/site-connect'; -import { AppType } from './redux/slices/app/AppType'; +import { SiteConnectPage } from './pages/site-connect'; +import { AppType } from './redux/slices/app/appType'; import { StakingPage } from './staking/home'; import { StorageMigrationPage } from './pages/StorageMigrationPage'; -import { useStorageMigrationStatus } from './hooks/useStorageMigrationStatus'; import { AccountsFinderPage } from './pages/accounts/manage/accounts-finder/AccountsFinderPage'; +import { AccountsFinderIntroPage } from './pages/accounts/manage/accounts-finder/AccountsFinderIntroPage'; const HIDDEN_MENU_PATHS = [ '/nft-details', @@ -68,7 +72,7 @@ const HIDDEN_MENU_PATHS = [ const NOTIFY_USER_ACTIVE_INTERVAL = 5 * 1000; // 5 seconds -const App = () => { +export function App() { const dispatch = useAppDispatch(); const isPopup = useAppSelector((state) => state.app.appType === AppType.Popup); useEffect(() => { @@ -189,6 +193,7 @@ const App = () => { <Route path="import-private-key" element={<ImportPrivateKeyPage />} /> <Route path="import-seed" element={<ImportSeedPage />} /> <Route path="manage" element={<ManageAccountsPage />} /> + <Route path="manage/accounts-finder/intro" element={<AccountsFinderIntroPage />} /> <Route path="manage/accounts-finder/:accountSourceId" element={<AccountsFinderPage />} @@ -215,6 +220,4 @@ const App = () => { </Route> </Routes> ); -}; - -export default App; +} diff --git a/apps/wallet/src/ui/app/LedgerSigner.ts b/apps/wallet/src/ui/app/ledgerSigner.ts similarity index 98% rename from apps/wallet/src/ui/app/LedgerSigner.ts rename to apps/wallet/src/ui/app/ledgerSigner.ts index 2d824f45324..2ac65e1ca5c 100644 --- a/apps/wallet/src/ui/app/LedgerSigner.ts +++ b/apps/wallet/src/ui/app/ledgerSigner.ts @@ -7,7 +7,7 @@ import { type IotaClient } from '@iota/iota-sdk/client'; import { toSerializedSignature, type SignatureScheme } from '@iota/iota-sdk/cryptography'; import { Ed25519PublicKey } from '@iota/iota-sdk/keypairs/ed25519'; -import { WalletSigner } from './WalletSigner'; +import { WalletSigner } from './walletSigner'; export class LedgerSigner extends WalletSigner { #iotaLedgerClient: IotaLedgerClient | null; diff --git a/apps/wallet/src/ui/app/pages/StorageMigrationPage.tsx b/apps/wallet/src/ui/app/pages/StorageMigrationPage.tsx index b3f37cb6e3c..71381895abb 100644 --- a/apps/wallet/src/ui/app/pages/StorageMigrationPage.tsx +++ b/apps/wallet/src/ui/app/pages/StorageMigrationPage.tsx @@ -5,8 +5,7 @@ import { useMutation } from '@tanstack/react-query'; import { toast } from 'react-hot-toast'; import { PasswordInputDialog } from '_components'; -import { useBackgroundClient } from '../hooks/useBackgroundClient'; -import { useStorageMigrationStatus } from '../hooks/useStorageMigrationStatus'; +import { useBackgroundClient, useStorageMigrationStatus } from '_hooks'; import { CardLayout } from '../shared/card-layout'; import { Toaster } from '../shared/toaster'; import { LoadingIndicator } from '@iota/apps-ui-kit'; diff --git a/apps/wallet/src/ui/app/pages/accounts/AddAccountPage.tsx b/apps/wallet/src/ui/app/pages/accounts/AddAccountPage.tsx index 6f352beb796..17f9725d778 100644 --- a/apps/wallet/src/ui/app/pages/accounts/AddAccountPage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/AddAccountPage.tsx @@ -23,10 +23,9 @@ import { PageTemplate, } from '_components'; import { getLedgerConnectionErrorMessage } from '../../helpers/errorMessages'; -import { useAppSelector } from '../../hooks'; -import { useCreateAccountsMutation } from '../../hooks/useCreateAccountMutation'; -import { AppType } from '../../redux/slices/app/AppType'; -import { Create, ImportPass, Key, Seed, Ledger } from '@iota/ui-icons'; +import { useAppSelector, useCreateAccountsMutation } from '_hooks'; +import { AppType } from '../../redux/slices/app/appType'; +import { Create, ImportPass, Key, Seed, Ledger } from '@iota/apps-ui-icons'; async function openTabWithSearchParam(searchParam: string, searchParamValue: string) { const currentURL = new URL(window.location.href); @@ -86,7 +85,7 @@ export function AddAccountPage() { ], }, { - title: 'Import from Legder', + title: 'Import from Ledger', cards: [ { title: 'Ledger', diff --git a/apps/wallet/src/ui/app/pages/accounts/BackupMnemonicPage.tsx b/apps/wallet/src/ui/app/pages/accounts/BackupMnemonicPage.tsx index 3c0d667ddc0..ca1e7dab321 100644 --- a/apps/wallet/src/ui/app/pages/accounts/BackupMnemonicPage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/BackupMnemonicPage.tsx @@ -10,13 +10,12 @@ import { InfoBoxStyle, InfoBoxType, } from '@iota/apps-ui-kit'; -import { Exclamation, Info } from '@iota/ui-icons'; +import { Exclamation, Info } from '@iota/apps-ui-icons'; import { HideShowDisplayBox, Loading, PageTemplate } from '_components'; -import { AccountSourceType } from '_src/background/account-sources/AccountSource'; +import { AccountSourceType } from '_src/background/account-sources/accountSource'; import { useEffect, useMemo, useState } from 'react'; import { Navigate, useNavigate, useParams } from 'react-router-dom'; -import { useAccountSources } from '../../hooks/useAccountSources'; -import { useExportPassphraseMutation } from '../../hooks/useExportPassphraseMutation'; +import { useAccountSources, useExportPassphraseMutation } from '_hooks'; export function BackupMnemonicPage() { const [mnemonicBackedUp, setMnemonicBackedUp] = useState(false); diff --git a/apps/wallet/src/ui/app/pages/accounts/ExportAccountPage.tsx b/apps/wallet/src/ui/app/pages/accounts/ExportAccountPage.tsx index 6307e2e7761..7160d7d33d8 100644 --- a/apps/wallet/src/ui/app/pages/accounts/ExportAccountPage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/ExportAccountPage.tsx @@ -2,14 +2,12 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useBackgroundClient } from '_src/ui/app/hooks/useBackgroundClient'; +import { useBackgroundClient, useAccounts } from '_hooks'; import { useMutation } from '@tanstack/react-query'; import { Navigate, useNavigate, useParams } from 'react-router-dom'; - import { VerifyPasswordModal, HideShowDisplayBox, Loading, Overlay } from '_components'; -import { useAccounts } from '../../hooks/useAccounts'; import { InfoBox, InfoBoxStyle, InfoBoxType } from '@iota/apps-ui-kit'; -import { Info } from '@iota/ui-icons'; +import { Info } from '@iota/apps-ui-icons'; export function ExportAccountPage() { const { accountID } = useParams(); diff --git a/apps/wallet/src/ui/app/pages/accounts/ExportPassphrasePage.tsx b/apps/wallet/src/ui/app/pages/accounts/ExportPassphrasePage.tsx index 53287018e22..a8024277819 100644 --- a/apps/wallet/src/ui/app/pages/accounts/ExportPassphrasePage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/ExportPassphrasePage.tsx @@ -4,11 +4,10 @@ import { HideShowDisplayBox, VerifyPasswordModal, Loading, Overlay } from '_components'; import { Navigate, useNavigate, useParams } from 'react-router-dom'; -import { useAccountSources } from '../../hooks/useAccountSources'; -import { useExportPassphraseMutation } from '../../hooks/useExportPassphraseMutation'; -import { AccountSourceType } from '_src/background/account-sources/AccountSource'; +import { useAccountSources, useExportPassphraseMutation } from '_hooks'; +import { AccountSourceType } from '_src/background/account-sources/accountSource'; import { InfoBox, InfoBoxType, InfoBoxStyle } from '@iota/apps-ui-kit'; -import { Info } from '@iota/ui-icons'; +import { Info } from '@iota/apps-ui-icons'; export function ExportPassphrasePage() { const { accountSourceID } = useParams(); diff --git a/apps/wallet/src/ui/app/pages/accounts/ExportSeedPage.tsx b/apps/wallet/src/ui/app/pages/accounts/ExportSeedPage.tsx index a14ab150a52..a9c6018f241 100644 --- a/apps/wallet/src/ui/app/pages/accounts/ExportSeedPage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/ExportSeedPage.tsx @@ -3,11 +3,10 @@ import { HideShowDisplayBox, VerifyPasswordModal, Loading, Overlay } from '_components'; import { Navigate, useNavigate, useParams } from 'react-router-dom'; -import { useAccountSources } from '../../hooks/useAccountSources'; -import { useExportSeedMutation } from '../../hooks/useExportSeedMutation'; -import { AccountSourceType } from '_src/background/account-sources/AccountSource'; +import { useAccountSources, useExportSeedMutation } from '_hooks'; +import { AccountSourceType } from '_src/background/account-sources/accountSource'; import { InfoBox, InfoBoxType, InfoBoxStyle } from '@iota/apps-ui-kit'; -import { Info } from '@iota/ui-icons'; +import { Info } from '@iota/apps-ui-icons'; export function ExportSeedPage() { const { accountSourceID } = useParams(); diff --git a/apps/wallet/src/ui/app/pages/accounts/ImportLedgerAccountsPage.tsx b/apps/wallet/src/ui/app/pages/accounts/ImportLedgerAccountsPage.tsx index 92b8dcdce92..84a992cbee6 100644 --- a/apps/wallet/src/ui/app/pages/accounts/ImportLedgerAccountsPage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/ImportLedgerAccountsPage.tsx @@ -14,9 +14,9 @@ import { Overlay, } from '_components'; import { getIotaApplicationErrorMessage } from '../../helpers/errorMessages'; -import { useAccounts } from '../../hooks/useAccounts'; +import { useAccounts } from '_hooks'; import { Button, LoadingIndicator } from '@iota/apps-ui-kit'; -import { CheckmarkFilled } from '@iota/ui-icons'; +import { CheckmarkFilled } from '@iota/apps-ui-icons'; const NUM_LEDGER_ACCOUNTS_TO_DERIVE_BY_DEFAULT = 10; diff --git a/apps/wallet/src/ui/app/pages/accounts/ImportPassphrasePage.tsx b/apps/wallet/src/ui/app/pages/accounts/ImportPassphrasePage.tsx index 6a3a60a2f6e..ffee5f84fe5 100644 --- a/apps/wallet/src/ui/app/pages/accounts/ImportPassphrasePage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/ImportPassphrasePage.tsx @@ -13,7 +13,7 @@ import { } from '_components'; import { Button, ButtonType } from '@iota/apps-ui-kit'; import { useState } from 'react'; -import { VisibilityOff, VisibilityOn } from '@iota/ui-icons'; +import { VisibilityOff, VisibilityOn } from '@iota/apps-ui-icons'; export function ImportPassphrasePage() { const navigate = useNavigate(); diff --git a/apps/wallet/src/ui/app/pages/accounts/ProtectAccountPage.tsx b/apps/wallet/src/ui/app/pages/accounts/ProtectAccountPage.tsx index 6cd3d880426..33d2cfa4099 100644 --- a/apps/wallet/src/ui/app/pages/accounts/ProtectAccountPage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/ProtectAccountPage.tsx @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { isMnemonicSerializedUiAccount } from '_src/background/accounts/MnemonicAccount'; +import { isMnemonicSerializedUiAccount } from '_src/background/accounts/mnemonicAccount'; import { useCallback, useEffect, useMemo, useState } from 'react'; import { toast } from 'react-hot-toast'; import { Navigate, useNavigate, useSearchParams } from 'react-router-dom'; @@ -14,13 +14,14 @@ import { PageTemplate, type ProtectAccountFormValues, } from '_components'; -import { useAccounts } from '../../hooks/useAccounts'; -import { autoLockDataToMinutes } from '../../hooks/useAutoLockMinutes'; -import { useAutoLockMinutesMutation } from '../../hooks/useAutoLockMinutesMutation'; -import { useCreateAccountsMutation } from '../../hooks/useCreateAccountMutation'; -import { isSeedSerializedUiAccount } from '_src/background/accounts/SeedAccount'; -import { isLedgerAccountSerializedUI } from '_src/background/accounts/LedgerAccount'; -import { AllowedAccountSourceTypes } from '../../accounts-finder'; +import { + useAccounts, + autoLockDataToMinutes, + useAutoLockMinutesMutation, + useCreateAccountsMutation, +} from '_hooks'; +import { isSeedSerializedUiAccount } from '_src/background/accounts/seedAccount'; +import { isLedgerAccountSerializedUI } from '_src/background/accounts/ledgerAccount'; import { useFeature } from '@growthbook/growthbook-react'; import { Feature } from '@iota/core'; @@ -92,7 +93,7 @@ export function ProtectAccountPage() { (isMnemonicSerializedUiAccount(createdAccounts[0]) || isSeedSerializedUiAccount(createdAccounts[0])) ) { - const path = `/accounts/manage/accounts-finder/${createdAccounts[0].sourceID}`; + const path = '/accounts/manage/accounts-finder/intro'; navigate(path, { replace: true, state: { @@ -103,7 +104,7 @@ export function ProtectAccountPage() { featureAccountFinderEnabled && isLedgerAccountSerializedUI(createdAccounts[0]) ) { - const path = `/accounts/manage/accounts-finder/${AllowedAccountSourceTypes.LedgerDerived}`; + const path = '/accounts/manage/accounts-finder/intro'; navigate(path, { replace: true, state: { diff --git a/apps/wallet/src/ui/app/pages/accounts/WelcomePage.tsx b/apps/wallet/src/ui/app/pages/accounts/WelcomePage.tsx index c9f6d36ddd7..eae3b5bde77 100644 --- a/apps/wallet/src/ui/app/pages/accounts/WelcomePage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/WelcomePage.tsx @@ -4,11 +4,9 @@ import { Loading } from '_components'; import { useNavigate } from 'react-router-dom'; -import { useFullscreenGuard, useInitializedGuard } from '_hooks'; +import { useFullscreenGuard, useInitializedGuard, useCreateAccountsMutation } from '_hooks'; import { Button, ButtonType } from '@iota/apps-ui-kit'; -import { IotaLogoWeb } from '@iota/ui-icons'; - -import { useCreateAccountsMutation } from '../../hooks/useCreateAccountMutation'; +import { IotaLogoWeb } from '@iota/apps-ui-icons'; export function WelcomePage() { const createAccountsMutation = useCreateAccountsMutation(); diff --git a/apps/wallet/src/ui/app/pages/accounts/forgot-password/ForgotPasswordIndexPage.tsx b/apps/wallet/src/ui/app/pages/accounts/forgot-password/ForgotPasswordIndexPage.tsx index 5c994dd90c3..b4fa19b98fe 100644 --- a/apps/wallet/src/ui/app/pages/accounts/forgot-password/ForgotPasswordIndexPage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/forgot-password/ForgotPasswordIndexPage.tsx @@ -4,9 +4,8 @@ import { useEffect } from 'react'; import { useNavigate } from 'react-router-dom'; - -import { useAccountSources } from '../../../hooks/useAccountSources'; -import { AccountSourceType } from '_src/background/account-sources/AccountSource'; +import { useAccountSources } from '_hooks'; +import { AccountSourceType } from '_src/background/account-sources/accountSource'; export function ForgotPasswordIndexPage() { const allAccountSources = useAccountSources(); diff --git a/apps/wallet/src/ui/app/pages/accounts/forgot-password/ForgotPasswordPage.tsx b/apps/wallet/src/ui/app/pages/accounts/forgot-password/ForgotPasswordPage.tsx index ebc1e4e60c1..a13e93f6448 100644 --- a/apps/wallet/src/ui/app/pages/accounts/forgot-password/ForgotPasswordPage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/forgot-password/ForgotPasswordPage.tsx @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type PasswordRecoveryData } from '_src/shared/messaging/messages/payloads/MethodPayload'; +import { type PasswordRecoveryData } from '_src/shared/messaging/messages/payloads/methodPayload'; import { createContext, useCallback, useContext, useState } from 'react'; import { Outlet } from 'react-router-dom'; diff --git a/apps/wallet/src/ui/app/pages/accounts/forgot-password/RecoverManyPage.tsx b/apps/wallet/src/ui/app/pages/accounts/forgot-password/RecoverManyPage.tsx index 61ae8475f46..65d2e20c990 100644 --- a/apps/wallet/src/ui/app/pages/accounts/forgot-password/RecoverManyPage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/forgot-password/RecoverManyPage.tsx @@ -10,15 +10,13 @@ import { ImportSeedForm, PageTemplate, } from '_components'; -import { useRecoveryDataMutation } from '_src/ui/app/hooks/useRecoveryDataMutation'; +import { useRecoveryDataMutation, useAccountGroups, useAccountSources } from '_hooks'; import { useEffect, useState } from 'react'; import toast from 'react-hot-toast'; import { useNavigate } from 'react-router-dom'; -import { useAccountGroups } from '../../../hooks/useAccountGroups'; -import { useAccountSources } from '../../../hooks/useAccountSources'; import { useForgotPasswordContext } from './ForgotPasswordPage'; -import { AccountSourceType } from '_src/background/account-sources/AccountSource'; -import { AccountType } from '_src/background/accounts/Account'; +import { AccountSourceType } from '_src/background/account-sources/accountSource'; +import { AccountType } from '_src/background/accounts/account'; import { Button, ButtonType } from '@iota/apps-ui-kit'; export function RecoverManyPage() { diff --git a/apps/wallet/src/ui/app/pages/accounts/forgot-password/RecoverPage.tsx b/apps/wallet/src/ui/app/pages/accounts/forgot-password/RecoverPage.tsx index ae04304d2de..9b0d21bab8f 100644 --- a/apps/wallet/src/ui/app/pages/accounts/forgot-password/RecoverPage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/forgot-password/RecoverPage.tsx @@ -4,12 +4,11 @@ import { entropyToSerialized, mnemonicToEntropy } from '_src/shared/utils'; import { ImportRecoveryPhraseForm, ImportSeedForm } from '_components'; -import { useRecoveryDataMutation } from '_src/ui/app/hooks/useRecoveryDataMutation'; +import { useRecoveryDataMutation, useAccountSources } from '_hooks'; import { useEffect } from 'react'; import toast from 'react-hot-toast'; import { useNavigate } from 'react-router-dom'; -import { useAccountSources } from '../../../hooks/useAccountSources'; -import { AccountSourceType } from '_src/background/account-sources/AccountSource'; +import { AccountSourceType } from '_src/background/account-sources/accountSource'; import { PageTemplate } from '_src/ui/app/components/PageTemplate'; export function RecoverPage() { diff --git a/apps/wallet/src/ui/app/pages/accounts/forgot-password/ResetPasswordPage.tsx b/apps/wallet/src/ui/app/pages/accounts/forgot-password/ResetPasswordPage.tsx index ec3a0e5ec46..4ea1edcfa41 100644 --- a/apps/wallet/src/ui/app/pages/accounts/forgot-password/ResetPasswordPage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/forgot-password/ResetPasswordPage.tsx @@ -2,12 +2,14 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useAutoLockMinutesMutation } from '_src/ui/app/hooks/useAutoLockMinutesMutation'; -import { useResetPasswordMutation } from '_src/ui/app/hooks/useResetPasswordMutation'; +import { + useAutoLockMinutesMutation, + useResetPasswordMutation, + autoLockDataToMinutes, +} from '_hooks'; import { toast } from 'react-hot-toast'; import { Navigate, useNavigate } from 'react-router-dom'; import { ProtectAccountForm, type ProtectAccountFormValues } from '_components'; -import { autoLockDataToMinutes } from '../../../hooks/useAutoLockMinutes'; import { useForgotPasswordContext } from './ForgotPasswordPage'; import { PageTemplate } from '_src/ui/app/components/PageTemplate'; diff --git a/apps/wallet/src/ui/app/pages/accounts/forgot-password/ResetWarningPage.tsx b/apps/wallet/src/ui/app/pages/accounts/forgot-password/ResetWarningPage.tsx index e8c1b2ba88a..face4ddbfe2 100644 --- a/apps/wallet/src/ui/app/pages/accounts/forgot-password/ResetWarningPage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/forgot-password/ResetWarningPage.tsx @@ -2,10 +2,9 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useAccounts } from '_src/ui/app/hooks/useAccounts'; +import { useAccounts, useAccountGroups } from '_hooks'; import { Navigate, useNavigate } from 'react-router-dom'; import { PageTemplate, RecoverAccountsGroup } from '_components'; -import { useAccountGroups } from '../../../hooks/useAccountGroups'; import { getGroupTitle } from '../manage/AccountGroup'; import { useForgotPasswordContext } from './ForgotPasswordPage'; import { Button, ButtonHtmlType } from '@iota/apps-ui-kit'; diff --git a/apps/wallet/src/ui/app/pages/accounts/manage/AccountGroup.tsx b/apps/wallet/src/ui/app/pages/accounts/manage/AccountGroup.tsx index c1a63ae97cd..5d8e96bb7e1 100644 --- a/apps/wallet/src/ui/app/pages/accounts/manage/AccountGroup.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/manage/AccountGroup.tsx @@ -2,21 +2,18 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { AccountType, type SerializedUIAccount } from '_src/background/accounts/Account'; +import { AccountType, type SerializedUIAccount } from '_src/background/accounts/account'; import { AccountsFormType, useAccountsFormContext, VerifyPasswordModal } from '_components'; -import { useAccountSources } from '_src/ui/app/hooks/useAccountSources'; -import { useCreateAccountsMutation } from '_src/ui/app/hooks/useCreateAccountMutation'; -import React, { useState } from 'react'; +import { useAccountSources, useCreateAccountsMutation, useActiveAccount } from '_hooks'; +import { useState } from 'react'; import { useNavigate } from 'react-router-dom'; import clsx from 'clsx'; - import { Button, ButtonSize, ButtonType, Dropdown, ListItem } from '@iota/apps-ui-kit'; -import { Add, MoreHoriz, TriangleDown } from '@iota/ui-icons'; +import { Add, MoreHoriz, TriangleDown } from '@iota/apps-ui-icons'; import { OutsideClickHandler } from '_components/OutsideClickHandler'; import { AccountGroupItem } from '_pages/accounts/manage/AccountGroupItem'; import { useFeature } from '@growthbook/growthbook-react'; import { Feature, Collapsible } from '@iota/core'; -import { useActiveAccount } from '_app/hooks/useActiveAccount'; const ACCOUNT_TYPE_TO_LABEL: Record<AccountType, string> = { [AccountType.MnemonicDerived]: 'Mnemonic', diff --git a/apps/wallet/src/ui/app/pages/accounts/manage/AccountGroupItem.tsx b/apps/wallet/src/ui/app/pages/accounts/manage/AccountGroupItem.tsx index 6762b389280..dce1c35d9a1 100644 --- a/apps/wallet/src/ui/app/pages/accounts/manage/AccountGroupItem.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/manage/AccountGroupItem.tsx @@ -1,20 +1,18 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { AccountType, type SerializedUIAccount } from '_src/background/accounts/Account'; +import { AccountType, type SerializedUIAccount } from '_src/background/accounts/account'; import { useState, useRef } from 'react'; import clsx from 'clsx'; import { formatAddress } from '@iota/iota-sdk/utils'; import { ExplorerLinkType, NicknameDialog, useUnlockAccount } from '_components'; import { useNavigate } from 'react-router-dom'; -import { useAccounts } from '_app/hooks/useAccounts'; -import { useExplorerLink } from '_app/hooks/useExplorerLink'; +import { useAccounts, useExplorerLink, useBackgroundClient } from '_hooks'; import toast from 'react-hot-toast'; import { Account, BadgeType, Dropdown, ListItem } from '@iota/apps-ui-kit'; import { OutsideClickHandler } from '_components/OutsideClickHandler'; -import { IotaLogoMark, Ledger } from '@iota/ui-icons'; +import { IotaLogoMark, Ledger } from '@iota/apps-ui-icons'; import { RemoveDialog } from './RemoveDialog'; -import { useBackgroundClient } from '_app/hooks/useBackgroundClient'; import { isMainAccount } from '_src/background/accounts/isMainAccount'; import { Portal } from '_app/shared/Portal'; import { formatAccountName } from '_src/ui/app/helpers'; diff --git a/apps/wallet/src/ui/app/pages/accounts/manage/ManageAccountsPage.tsx b/apps/wallet/src/ui/app/pages/accounts/manage/ManageAccountsPage.tsx index 73859789962..65b8189b757 100644 --- a/apps/wallet/src/ui/app/pages/accounts/manage/ManageAccountsPage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/manage/ManageAccountsPage.tsx @@ -3,11 +3,9 @@ // SPDX-License-Identifier: Apache-2.0 import { useRef } from 'react'; import { Button, ButtonType } from '@iota/apps-ui-kit'; -import { type AccountType } from '_src/background/accounts/Account'; -import { useInitializedGuard } from '_src/ui/app/hooks'; -import { useAccountGroups } from '_src/ui/app/hooks/useAccountGroups'; +import { type AccountType } from '_src/background/accounts/account'; +import { useInitializedGuard, useAccountGroups } from '_hooks'; import { useNavigate } from 'react-router-dom'; - import { Overlay } from '_components'; import { AccountGroup } from './AccountGroup'; diff --git a/apps/wallet/src/ui/app/pages/accounts/manage/RemoveDialog.tsx b/apps/wallet/src/ui/app/pages/accounts/manage/RemoveDialog.tsx index 807ef298573..f058bbf8550 100644 --- a/apps/wallet/src/ui/app/pages/accounts/manage/RemoveDialog.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/manage/RemoveDialog.tsx @@ -1,11 +1,21 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useAccounts } from '_app/hooks/useAccounts'; -import { useBackgroundClient } from '_app/hooks/useBackgroundClient'; +import { useAccounts, useBackgroundClient } from '_hooks'; import { useMutation } from '@tanstack/react-query'; -import { Button, ButtonType, Dialog, DialogBody, DialogContent, Header } from '@iota/apps-ui-kit'; +import { + Button, + ButtonType, + Dialog, + DialogBody, + DialogContent, + Header, + InfoBox, + InfoBoxStyle, + InfoBoxType, +} from '@iota/apps-ui-kit'; import toast from 'react-hot-toast'; +import { Info } from '@iota/apps-ui-icons'; interface RemoveDialogProps { accountID: string; @@ -42,27 +52,32 @@ export function RemoveDialog({ isOpen, setOpen, accountID }: RemoveDialogProps) <DialogContent containerId="overlay-portal-container"> <Header title="Remove account" onClose={() => setOpen(false)} /> <DialogBody> - <div className="mb-md text-body-md"> - Are you sure you want to remove this account? - </div> - {totalAccounts === 1 ? ( - <div className="text-center"> - Removing this account will require you to set up your IOTA wallet again. + <div className="flex flex-col gap-y-md"> + <div className="text-body-md"> + Are you sure you want to remove this account? + </div> + {totalAccounts === 1 ? ( + <InfoBox + type={InfoBoxType.Default} + supportingText="Removing this account will require you to set up your IOTA wallet again." + icon={<Info />} + style={InfoBoxStyle.Elevated} + /> + ) : null} + <div className="flex gap-xs"> + <Button + fullWidth + type={ButtonType.Secondary} + text="Cancel" + onClick={handleCancel} + /> + <Button + fullWidth + type={ButtonType.Destructive} + text="Remove" + onClick={handleRemove} + /> </div> - ) : null} - <div className="flex gap-xs"> - <Button - fullWidth - type={ButtonType.Secondary} - text="Cancel" - onClick={handleCancel} - /> - <Button - fullWidth - type={ButtonType.Destructive} - text="Remove" - onClick={handleRemove} - /> </div> </DialogBody> </DialogContent> diff --git a/apps/wallet/src/ui/app/pages/accounts/manage/accounts-finder/AccountsFinderIntroPage.tsx b/apps/wallet/src/ui/app/pages/accounts/manage/accounts-finder/AccountsFinderIntroPage.tsx new file mode 100644 index 00000000000..e575d0e12f2 --- /dev/null +++ b/apps/wallet/src/ui/app/pages/accounts/manage/accounts-finder/AccountsFinderIntroPage.tsx @@ -0,0 +1,68 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +import { useNavigate } from 'react-router-dom'; +import { Overlay } from '_components'; +import { useActiveAccount } from '_hooks'; +import { getKey } from '_helpers'; +import { Theme, useTheme } from '@iota/core'; +import { Button, ButtonType } from '@iota/apps-ui-kit'; +import BalanceFinderIntroImage from '_assets/images/balance_finder_intro.svg'; +import BalanceFinderIntroDarkImage from '_assets/images/balance_finder_intro_darkmode.svg'; +import { isLedgerAccountSerializedUI } from '_src/background/accounts/ledgerAccount'; +import { AllowedAccountSourceTypes } from '_src/ui/app/accounts-finder'; + +export function AccountsFinderIntroPage() { + const { theme } = useTheme(); + const navigate = useNavigate(); + const activeAccount = useActiveAccount(); + + const isLedgerAccount = activeAccount && isLedgerAccountSerializedUI(activeAccount); + const accountSourceId = activeAccount && getKey(activeAccount); + + const ledgerPath = `/accounts/manage/accounts-finder/${AllowedAccountSourceTypes.LedgerDerived}`; + const accountPath = isLedgerAccount + ? ledgerPath + : `/accounts/manage/accounts-finder/${accountSourceId}`; + + return ( + <Overlay showModal> + <div className="flex h-full flex-col items-center justify-between"> + <div> + {theme === Theme.Dark ? ( + <BalanceFinderIntroDarkImage /> + ) : ( + <BalanceFinderIntroImage /> + )} + </div> + <div className="flex h-full flex-col items-center justify-between"> + <div className="flex flex-col gap-y-sm p-md text-center"> + <span className="text-label-lg text-neutral-40 dark:text-neutral-60"> + Wallet Setup + </span> + <span className="text-headline-md text-neutral-10 dark:text-neutral-92"> + Balance Finder + </span> + <span className="text-body-md text-neutral-40 dark:text-neutral-60"> + Easily find and import all your accounts with balances, in one place. + </span> + </div> + <div className="flex w-full flex-row gap-x-xs"> + <Button + type={ButtonType.Secondary} + text="Skip" + onClick={() => navigate('/')} + fullWidth + /> + <Button + type={ButtonType.Primary} + text="Start" + fullWidth + onClick={() => navigate(accountPath)} + /> + </div> + </div> + </div> + </Overlay> + ); +} diff --git a/apps/wallet/src/ui/app/pages/accounts/manage/accounts-finder/AccountsFinderPage.tsx b/apps/wallet/src/ui/app/pages/accounts/manage/accounts-finder/AccountsFinderPage.tsx index 337147753e6..b49e66ba0d9 100644 --- a/apps/wallet/src/ui/app/pages/accounts/manage/accounts-finder/AccountsFinderPage.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/manage/accounts-finder/AccountsFinderPage.tsx @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useInitializedGuard } from '_src/ui/app/hooks'; +import { useInitializedGuard } from '_hooks'; import { useNavigate } from 'react-router-dom'; import { Overlay } from '_components'; import { AccountsFinderView } from './AccountsFinderView'; diff --git a/apps/wallet/src/ui/app/pages/accounts/manage/accounts-finder/AccountsFinderView.tsx b/apps/wallet/src/ui/app/pages/accounts/manage/accounts-finder/AccountsFinderView.tsx index 97353451383..87a74dea45d 100644 --- a/apps/wallet/src/ui/app/pages/accounts/manage/accounts-finder/AccountsFinderView.tsx +++ b/apps/wallet/src/ui/app/pages/accounts/manage/accounts-finder/AccountsFinderView.tsx @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { Search } from '@iota/ui-icons'; +import { Search } from '@iota/apps-ui-icons'; import { Button, ButtonType, ButtonSize, LoadingIndicator } from '@iota/apps-ui-kit'; import { AccountBalanceItem, @@ -12,16 +12,12 @@ import { import { AccountSourceType, type AccountSourceSerializedUI, -} from '_src/background/account-sources/AccountSource'; -import { AccountType } from '_src/background/accounts/Account'; +} from '_src/background/account-sources/accountSource'; +import { AccountType } from '_src/background/accounts/account'; import { type SourceStrategyToFind } from '_src/shared/messaging/messages/payloads/accounts-finder'; import { AllowedAccountSourceTypes } from '_src/ui/app/accounts-finder'; -import { getKey } from '_src/ui/app/helpers/accounts'; -import { getLedgerConnectionErrorMessage } from '_src/ui/app/helpers/errorMessages'; -import { useAccountSources } from '_src/ui/app/hooks/useAccountSources'; -import { useAccounts } from '_src/ui/app/hooks/useAccounts'; -import { useAccountsFinder } from '_src/ui/app/hooks/useAccountsFinder'; -import { useUnlockMutation } from '_src/ui/app/hooks/useUnlockMutation'; +import { getKey, getLedgerConnectionErrorMessage } from '_src/ui/app/helpers'; +import { useAccountSources, useAccounts, useUnlockMutation, useAccountsFinder } from '_hooks'; import { useMemo, useState } from 'react'; import toast from 'react-hot-toast'; import { useParams } from 'react-router-dom'; diff --git a/apps/wallet/src/ui/app/pages/approval-request/SignMessageRequest.tsx b/apps/wallet/src/ui/app/pages/approval-request/SignMessageRequest.tsx index 7f708d31f9f..4ad17a04eb3 100644 --- a/apps/wallet/src/ui/app/pages/approval-request/SignMessageRequest.tsx +++ b/apps/wallet/src/ui/app/pages/approval-request/SignMessageRequest.tsx @@ -2,13 +2,11 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type SignPersonalMessageApprovalRequest } from '_payloads/transactions/ApprovalRequest'; +import { type SignPersonalMessageApprovalRequest } from '_src/shared/messaging/messages/payloads/transactions/approvalRequest'; import { toUtf8OrB64 } from '_src/shared/utils'; import { useMemo } from 'react'; import { UserApproveContainer } from '_components'; -import { useAppDispatch } from '../../hooks'; -import { useAccountByAddress } from '../../hooks/useAccountByAddress'; -import { useSigner } from '../../hooks/useSigner'; +import { useAppDispatch, useAccountByAddress, useSigner } from '_hooks'; import { respondToTransactionRequest } from '../../redux/slices/transaction-requests'; import { PageMainLayoutTitle } from '../../shared/page-main-layout/PageMainLayoutTitle'; import { Panel } from '@iota/apps-ui-kit'; diff --git a/apps/wallet/src/ui/app/pages/approval-request/index.tsx b/apps/wallet/src/ui/app/pages/approval-request/index.tsx index 14c7dafa767..3faf86263a6 100644 --- a/apps/wallet/src/ui/app/pages/approval-request/index.tsx +++ b/apps/wallet/src/ui/app/pages/approval-request/index.tsx @@ -5,13 +5,12 @@ import { isSignPersonalMessageApprovalRequest, isTransactionApprovalRequest, -} from '_payloads/transactions/ApprovalRequest'; +} from '_src/shared/messaging/messages/payloads/transactions/approvalRequest'; import { useEffect, useMemo } from 'react'; import { useParams } from 'react-router-dom'; - import { Loading } from '_components'; -import { useAppSelector } from '../../hooks'; -import { type RootState } from '../../redux/RootReducer'; +import { useAppSelector } from '_hooks'; +import { type RootState } from '../../redux/rootReducer'; import { txRequestsSelectors } from '../../redux/slices/transaction-requests'; import { SignMessageRequest } from './SignMessageRequest'; import { TransactionRequest } from './transaction-request'; diff --git a/apps/wallet/src/ui/app/pages/approval-request/transaction-request/index.tsx b/apps/wallet/src/ui/app/pages/approval-request/transaction-request/index.tsx index e3cf45db226..1f1cb711369 100644 --- a/apps/wallet/src/ui/app/pages/approval-request/transaction-request/index.tsx +++ b/apps/wallet/src/ui/app/pages/approval-request/transaction-request/index.tsx @@ -3,20 +3,24 @@ // SPDX-License-Identifier: Apache-2.0 import { ExplorerLinkHelper, UserApproveContainer } from '_components'; -import { useActiveAddress, useAppDispatch, useTransactionData, useTransactionDryRun } from '_hooks'; -import { type TransactionApprovalRequest } from '_payloads/transactions/ApprovalRequest'; +import { + useActiveAddress, + useAppDispatch, + useTransactionData, + useTransactionDryRun, + useAccountByAddress, + useRecognizedPackages, + useSigner, +} from '_hooks'; +import { type TransactionApprovalRequest } from '_src/shared/messaging/messages/payloads/transactions/approvalRequest'; import { respondToTransactionRequest } from '_redux/slices/transaction-requests'; import { ampli } from '_src/shared/analytics/ampli'; -import { useAccountByAddress } from '_src/ui/app/hooks/useAccountByAddress'; -import { useRecognizedPackages } from '_src/ui/app/hooks/useRecognizedPackages'; -import { useSigner } from '_src/ui/app/hooks/useSigner'; import { PageMainLayoutTitle } from '_src/ui/app/shared/page-main-layout/PageMainLayoutTitle'; import { useTransactionSummary, TransactionSummary, GasFees } from '@iota/core'; import { Transaction } from '@iota/iota-sdk/transactions'; import { useMemo, useState } from 'react'; - import { ConfirmationModal } from '../../../shared/ConfirmationModal'; -import { TransactionDetails } from './TransactionDetails'; +import { TransactionDetails } from './transaction-details'; export interface TransactionRequestProps { txRequest: TransactionApprovalRequest; diff --git a/apps/wallet/src/ui/app/pages/approval-request/transaction-request/TransactionDetails/Command.tsx b/apps/wallet/src/ui/app/pages/approval-request/transaction-request/transaction-details/Command.tsx similarity index 100% rename from apps/wallet/src/ui/app/pages/approval-request/transaction-request/TransactionDetails/Command.tsx rename to apps/wallet/src/ui/app/pages/approval-request/transaction-request/transaction-details/Command.tsx diff --git a/apps/wallet/src/ui/app/pages/approval-request/transaction-request/TransactionDetails/Input.tsx b/apps/wallet/src/ui/app/pages/approval-request/transaction-request/transaction-details/Input.tsx similarity index 100% rename from apps/wallet/src/ui/app/pages/approval-request/transaction-request/TransactionDetails/Input.tsx rename to apps/wallet/src/ui/app/pages/approval-request/transaction-request/transaction-details/Input.tsx diff --git a/apps/wallet/src/ui/app/pages/approval-request/transaction-request/TransactionDetails/index.tsx b/apps/wallet/src/ui/app/pages/approval-request/transaction-request/transaction-details/index.tsx similarity index 98% rename from apps/wallet/src/ui/app/pages/approval-request/transaction-request/TransactionDetails/index.tsx rename to apps/wallet/src/ui/app/pages/approval-request/transaction-request/transaction-details/index.tsx index 518a0929061..9526de837d6 100644 --- a/apps/wallet/src/ui/app/pages/approval-request/transaction-request/TransactionDetails/index.tsx +++ b/apps/wallet/src/ui/app/pages/approval-request/transaction-request/transaction-details/index.tsx @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useTransactionData } from '_src/ui/app/hooks'; +import { useTransactionData } from '_hooks'; import { type Transaction } from '@iota/iota-sdk/transactions'; import { Command } from './Command'; import { Input } from './Input'; @@ -21,7 +21,7 @@ import { } from '@iota/apps-ui-kit'; import { useEffect, useState } from 'react'; import { Loading } from '_src/ui/app/components'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; interface TransactionDetailsProps { sender?: string; diff --git a/apps/wallet/src/ui/app/pages/home/apps/index.tsx b/apps/wallet/src/ui/app/pages/home/apps/index.tsx index fb20b27115f..84d7a35a197 100644 --- a/apps/wallet/src/ui/app/pages/home/apps/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/apps/index.tsx @@ -4,9 +4,9 @@ import { useFeature } from '@growthbook/growthbook-react'; import { FiltersPortal, ConnectedAppsCard, type DAppEntry } from '_components'; -import { getFromSessionStorage, setToSessionStorage } from '_src/background/storage-utils'; +import { getFromSessionStorage, setToSessionStorage } from '_src/background/storageUtils'; import { Feature } from '@iota/core'; -import { useUnlockedGuard } from '_src/ui/app/hooks/useUnlockedGuard'; +import { useUnlockedGuard } from '_hooks'; import { useEffect } from 'react'; import { Navigate, Route, Routes, useNavigate } from 'react-router-dom'; @@ -19,7 +19,7 @@ type FilterTag = { link: string; }; -function AppsPage() { +export function AppsPage() { const navigate = useNavigate(); const DEFAULT_FILTER_TAGS: FilterTag[] = [ @@ -83,5 +83,3 @@ function AppsPage() { </div> ); } - -export default AppsPage; diff --git a/apps/wallet/src/ui/app/pages/home/assets/HiddenAssetsProvider.tsx b/apps/wallet/src/ui/app/pages/home/assets/HiddenAssetsProvider.tsx deleted file mode 100644 index ff94f428a74..00000000000 --- a/apps/wallet/src/ui/app/pages/home/assets/HiddenAssetsProvider.tsx +++ /dev/null @@ -1,205 +0,0 @@ -// Copyright (c) Mysten Labs, Inc. -// Modifications Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 - -import { get, set } from 'idb-keyval'; -import { createContext, useCallback, useContext, useEffect, useState, type ReactNode } from 'react'; -import { type Toast, toast } from 'react-hot-toast'; -import { ButtonUnstyled } from '@iota/apps-ui-kit'; - -const HIDDEN_ASSET_IDS = 'hidden-asset-ids'; - -type HiddenAssets = - | { - type: 'loading'; - } - | { - type: 'loaded'; - assetIds: string[]; - }; - -interface HiddenAssetContext { - hiddenAssets: HiddenAssets; - setHiddenAssetIds: (hiddenAssetIds: string[]) => void; - hideAsset: (assetId: string) => void; - showAsset: (assetId: string) => void; -} - -export const HiddenAssetsContext = createContext<HiddenAssetContext>({ - hiddenAssets: { - type: 'loading', - }, - setHiddenAssetIds: () => {}, - hideAsset: () => {}, - showAsset: () => {}, -}); - -export const HiddenAssetsProvider = ({ children }: { children: ReactNode }) => { - const [hiddenAssets, setHiddenAssets] = useState<HiddenAssets>({ - type: 'loading', - }); - - const hiddenAssetIds = hiddenAssets.type === 'loaded' ? hiddenAssets.assetIds : []; - - useEffect(() => { - (async () => { - const hiddenAssets = (await get<string[]>(HIDDEN_ASSET_IDS)) ?? []; - setHiddenAssetIds(hiddenAssets); - })(); - }, []); - - function setHiddenAssetIds(hiddenAssetIds: string[]) { - setHiddenAssets({ - type: 'loaded', - assetIds: hiddenAssetIds, - }); - } - - const hideAssetId = useCallback( - async (newAssetId: string) => { - if (hiddenAssetIds.includes(newAssetId)) return; - - const newHiddenAssetIds = [...hiddenAssetIds, newAssetId]; - setHiddenAssetIds(newHiddenAssetIds); - await set(HIDDEN_ASSET_IDS, newHiddenAssetIds); - - const undoHideAsset = async (assetId: string) => { - try { - let updatedHiddenAssetIds; - setHiddenAssets((previous) => { - const previousIds = previous.type === 'loaded' ? previous.assetIds : []; - updatedHiddenAssetIds = previousIds.filter((id) => id !== assetId); - return { - type: 'loaded', - assetIds: updatedHiddenAssetIds, - }; - }); - await set(HIDDEN_ASSET_IDS, updatedHiddenAssetIds); - } catch (error) { - // Handle any error that occurred during the unhide process - toast.error('Failed to unhide asset.'); - // Restore the asset ID back to the hidden asset IDs list - setHiddenAssetIds([...hiddenAssetIds, assetId]); - await set(HIDDEN_ASSET_IDS, hiddenAssetIds); - } - }; - - const showAssetHiddenToast = async (objectId: string) => { - toast.success( - (t) => ( - <MovedAssetNotification - t={t} - destination="Hidden Assets" - onUndo={() => undoHideAsset(objectId)} - /> - ), - { - duration: 4000, - }, - ); - }; - showAssetHiddenToast(newAssetId); - }, - [hiddenAssetIds], - ); - - const showAssetId = useCallback( - async (newAssetId: string) => { - if (!hiddenAssetIds.includes(newAssetId)) return; - - try { - const updatedHiddenAssetIds = hiddenAssetIds.filter((id) => id !== newAssetId); - setHiddenAssetIds(updatedHiddenAssetIds); - await set(HIDDEN_ASSET_IDS, updatedHiddenAssetIds); - } catch (error) { - // Handle any error that occurred during the unhide process - toast.error('Failed to show asset.'); - // Restore the asset ID back to the hidden asset IDs list - setHiddenAssetIds([...hiddenAssetIds, newAssetId]); - await set(HIDDEN_ASSET_IDS, hiddenAssetIds); - } - - const undoShowAsset = async (assetId: string) => { - let newHiddenAssetIds; - setHiddenAssets((previous) => { - const previousIds = previous.type === 'loaded' ? previous.assetIds : []; - newHiddenAssetIds = [...previousIds, assetId]; - return { - type: 'loaded', - assetIds: newHiddenAssetIds, - }; - }); - await set(HIDDEN_ASSET_IDS, newHiddenAssetIds); - }; - - const assetShownToast = async (objectId: string) => { - toast.success( - (t) => ( - <MovedAssetNotification - t={t} - destination="Visual Assets" - onUndo={() => undoShowAsset(objectId)} - /> - ), - { - duration: 4000, - }, - ); - }; - - assetShownToast(newAssetId); - }, - [hiddenAssetIds], - ); - - const showAsset = (objectId: string) => { - showAssetId(objectId); - }; - - return ( - <HiddenAssetsContext.Provider - value={{ - hiddenAssets: - hiddenAssets.type === 'loaded' - ? { ...hiddenAssets, assetIds: Array.from(new Set(hiddenAssetIds)) } - : { type: 'loading' }, - setHiddenAssetIds, - hideAsset: hideAssetId, - showAsset, - }} - > - {children} - </HiddenAssetsContext.Provider> - ); -}; - -export const useHiddenAssets = () => { - return useContext(HiddenAssetsContext); -}; - -interface MovedAssetNotificationProps { - t: Toast; - destination: string; - onUndo: () => void; -} -function MovedAssetNotification({ t, destination, onUndo }: MovedAssetNotificationProps) { - return ( - <div - className="flex w-full flex-row items-baseline gap-x-xxs" - onClick={() => toast.dismiss(t.id)} - > - <ButtonUnstyled className="text-body-sm text-neutral-12 dark:text-neutral-92"> - Moved to {destination} - </ButtonUnstyled> - <ButtonUnstyled - onClick={() => { - onUndo(); - toast.dismiss(t.id); - }} - className="ml-auto mr-sm text-body-sm text-neutral-12 dark:text-neutral-92" - > - UNDO - </ButtonUnstyled> - </div> - ); -} diff --git a/apps/wallet/src/ui/app/pages/home/assets/index.tsx b/apps/wallet/src/ui/app/pages/home/assets/index.tsx index 9f224ecd1ec..b14a5de1b90 100644 --- a/apps/wallet/src/ui/app/pages/home/assets/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/assets/index.tsx @@ -2,12 +2,12 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useUnlockedGuard } from '_src/ui/app/hooks/useUnlockedGuard'; +import { useUnlockedGuard } from '_hooks'; import { Route, Routes } from 'react-router-dom'; import { NftsPage } from '..'; -import { HiddenAssetsProvider } from './HiddenAssetsProvider'; +import { HiddenAssetsProvider } from '@iota/core'; -function AssetsPage() { +export function AssetsPage() { if (useUnlockedGuard()) { return null; } @@ -19,5 +19,3 @@ function AssetsPage() { </HiddenAssetsProvider> ); } - -export default AssetsPage; diff --git a/apps/wallet/src/ui/app/pages/home/index.tsx b/apps/wallet/src/ui/app/pages/home/index.tsx index 4ce600b17c8..f43925680c8 100644 --- a/apps/wallet/src/ui/app/pages/home/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/index.tsx @@ -11,7 +11,7 @@ interface HomePageProps { disableNavigation?: boolean; } -const HomePage = ({ disableNavigation }: HomePageProps) => { +export function HomePage({ disableNavigation }: HomePageProps) { const initChecking = useInitializedGuard(true); const guardChecking = initChecking; @@ -27,17 +27,16 @@ const HomePage = ({ disableNavigation }: HomePageProps) => { </PageMainLayout> </Loading> ); -}; +} -export default HomePage; -export { default as NftsPage } from './nfts'; -export { default as AssetsPage } from './assets'; -export { default as TokensPage } from './tokens'; -export { default as TransactionBlocksPage } from './transactions'; -export { default as TransferCoinPage } from './transfer-coin'; -export { default as NFTDetailsPage } from './nft-details'; -export { default as KioskDetailsPage } from './kiosk-details'; -export { default as NftTransferPage } from './nft-transfer'; -export { default as ReceiptPage } from './receipt'; -export { default as CoinsSelectorPage } from './transfer-coin/CoinSelector'; -export { default as AppsPage } from './apps'; +export * from './nfts'; +export * from './assets'; +export * from './tokens'; +export * from './transactions'; +export * from './transfer-coin'; +export * from './nft-details'; +export * from './kiosk-details'; +export * from './nft-transfer'; +export * from './receipt'; +export * from './transfer-coin/CoinSelector'; +export * from './apps'; diff --git a/apps/wallet/src/ui/app/pages/home/interstitial/index.tsx b/apps/wallet/src/ui/app/pages/home/interstitial/index.tsx index b435f4d650e..d4c23dd22bb 100644 --- a/apps/wallet/src/ui/app/pages/home/interstitial/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/interstitial/index.tsx @@ -8,7 +8,7 @@ import { useEffect } from 'react'; import { useNavigate } from 'react-router-dom'; import { Portal } from '../../../shared/Portal'; -import { Close } from '@iota/ui-icons'; +import { Close } from '@iota/apps-ui-icons'; export type InterstitialConfig = { enabled: boolean; @@ -23,7 +23,13 @@ interface InterstitialProps extends InterstitialConfig { const setInterstitialDismissed = (dismissKey: string) => localStorage.setItem(dismissKey, 'true'); -function Interstitial({ enabled, dismissKey, imageUrl, bannerUrl, onClose }: InterstitialProps) { +export function Interstitial({ + enabled, + dismissKey, + imageUrl, + bannerUrl, + onClose, +}: InterstitialProps) { const navigate = useNavigate(); useEffect(() => { @@ -68,5 +74,3 @@ function Interstitial({ enabled, dismissKey, imageUrl, bannerUrl, onClose }: Int </Portal> ); } - -export default Interstitial; diff --git a/apps/wallet/src/ui/app/pages/home/kiosk-details/index.tsx b/apps/wallet/src/ui/app/pages/home/kiosk-details/index.tsx index 8d02c1ea8da..66df3b88a55 100644 --- a/apps/wallet/src/ui/app/pages/home/kiosk-details/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/kiosk-details/index.tsx @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useActiveAddress } from '_app/hooks/useActiveAddress'; +import { useActiveAddress, useUnlockedGuard } from '_hooks'; import { ErrorBoundary, ExplorerLink, @@ -11,14 +11,13 @@ import { NFTDisplayCard, PageTemplate, } from '_components'; -import { useUnlockedGuard } from '_src/ui/app/hooks/useUnlockedGuard'; import { useGetKioskContents, Collapsible } from '@iota/core'; import { formatAddress } from '@iota/iota-sdk/utils'; import { Link, useSearchParams, useNavigate } from 'react-router-dom'; import cl from 'clsx'; import { KeyValueInfo } from '@iota/apps-ui-kit'; -function KioskDetailsPage() { +export function KioskDetailsPage() { const navigate = useNavigate(); const [searchParams] = useSearchParams(); const kioskId = searchParams.get('kioskId'); @@ -92,5 +91,3 @@ function KioskDetailsPage() { </PageTemplate> ); } - -export default KioskDetailsPage; diff --git a/apps/wallet/src/ui/app/pages/home/nft-details/index.tsx b/apps/wallet/src/ui/app/pages/home/nft-details/index.tsx index 062577a10b1..9956867955c 100644 --- a/apps/wallet/src/ui/app/pages/home/nft-details/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/nft-details/index.tsx @@ -2,16 +2,15 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useActiveAddress } from '_app/hooks/useActiveAddress'; +import { useActiveAddress, useUnlockedGuard } from '_hooks'; import { ExplorerLink, ExplorerLinkType, Loading, NFTDisplayCard, PageTemplate } from '_components'; -import { useUnlockedGuard } from '_src/ui/app/hooks/useUnlockedGuard'; import { useNFTBasicData, useNftDetails, Collapsible } from '@iota/core'; import { formatAddress } from '@iota/iota-sdk/utils'; import cl from 'clsx'; -import { Link, Navigate, useNavigate, useSearchParams } from 'react-router-dom'; +import { Navigate, useNavigate, useSearchParams } from 'react-router-dom'; import { Button, ButtonType, KeyValueInfo } from '@iota/apps-ui-kit'; -function NFTDetailsPage() { +export function NFTDetailsPage() { const navigate = useNavigate(); const [searchParams] = useSearchParams(); const nftId = searchParams.get('objectId'); @@ -23,7 +22,6 @@ function NFTDetailsPage() { objectData, metaKeys, metaValues, - formatMetaValue, isContainedInKiosk, kioskItem, isAssetTransferable, @@ -34,12 +32,12 @@ function NFTDetailsPage() { const isPending = isLoading || isGuardLoading; function handleMoreAboutKiosk() { - window.open('https://docs.iota.org/references/ts-sdk/kiosk/', '_blank'); + window.open('https://docs.iota.org/ts-sdk/kiosk/', '_blank'); } function handleMarketplace() { // TODO: https://github.com/iotaledger/iota/issues/4024 - window.open('https://docs.iota.org/references/ts-sdk/kiosk/', '_blank'); + window.open('https://docs.iota.org/ts-sdk/kiosk/', '_blank'); } function handleSend() { @@ -94,11 +92,7 @@ function NFTDetailsPage() { {nftDisplayData?.projectUrl && ( <KeyValueInfo keyText="Website" - value={ - <Link to={nftDisplayData?.projectUrl}> - {nftDisplayData?.projectUrl} - </Link> - } + value={nftDisplayData?.projectUrl} fullwidth /> )} @@ -153,19 +147,17 @@ function NFTDetailsPage() { <Collapsible defaultOpen title="Attributes"> <div className="flex flex-col gap-xs px-md pb-xs pt-sm"> {metaKeys.map((aKey, idx) => { - const { value, valueLink } = - formatMetaValue(metaValues[idx]); return ( <KeyValueInfo key={idx} keyText={aKey} value={ - <Link - key={aKey} - to={valueLink || ''} - > - {value} - </Link> + typeof metaValues[idx] === + 'object' + ? JSON.stringify( + metaValues[idx], + ) + : metaValues[idx] } fullwidth /> @@ -211,5 +203,3 @@ function NFTDetailsPage() { </PageTemplate> ); } - -export default NFTDetailsPage; diff --git a/apps/wallet/src/ui/app/pages/home/nft-transfer/TransferNFTForm.tsx b/apps/wallet/src/ui/app/pages/home/nft-transfer/TransferNFTForm.tsx index 9e4801aa3cb..35d4181f6e8 100644 --- a/apps/wallet/src/ui/app/pages/home/nft-transfer/TransferNFTForm.tsx +++ b/apps/wallet/src/ui/app/pages/home/nft-transfer/TransferNFTForm.tsx @@ -4,25 +4,35 @@ import { ampli } from '_src/shared/analytics/ampli'; import { getSignerOperationErrorMessage } from '_src/ui/app/helpers/errorMessages'; -import { useActiveAddress } from '_src/ui/app/hooks'; -import { useActiveAccount } from '_src/ui/app/hooks/useActiveAccount'; -import { useSigner } from '_src/ui/app/hooks/useSigner'; -import { createNftSendValidationSchema, useGetKioskContents, AddressInput } from '@iota/core'; -import { Transaction } from '@iota/iota-sdk/transactions'; -import { useMutation, useQueryClient } from '@tanstack/react-query'; +import { useActiveAccount, useSigner, useActiveAddress } from '_hooks'; +import { + createNftSendValidationSchema, + AddressInput, + useTransferAsset, + type TransferAssetExecuteFn, +} from '@iota/core'; +import { useQueryClient } from '@tanstack/react-query'; import { Form, Formik } from 'formik'; import { toast } from 'react-hot-toast'; import { useNavigate } from 'react-router-dom'; - -import { useTransferKioskItem } from './useTransferKioskItem'; import { Button, ButtonHtmlType } from '@iota/apps-ui-kit'; -import { Loader } from '@iota/ui-icons'; +import { Loader } from '@iota/apps-ui-icons'; +import { type WalletSigner } from '_src/ui/app/walletSigner'; interface TransferNFTFormProps { objectId: string; objectType?: string | null; } +function normalizeWalletSignAndExecute( + signer: WalletSigner | null, +): TransferAssetExecuteFn | undefined { + if (!signer || !signer) return; + + const executeFn = signer.signAndExecuteTransaction.bind(signer); + return ({ transaction, ...rest }) => executeFn({ transactionBlock: transaction, ...rest }); +} + export function TransferNFTForm({ objectId, objectType }: TransferNFTFormProps) { const activeAddress = useActiveAddress(); const validationSchema = createNftSendValidationSchema(activeAddress || '', objectId); @@ -30,34 +40,12 @@ export function TransferNFTForm({ objectId, objectType }: TransferNFTFormProps) const signer = useSigner(activeAccount); const queryClient = useQueryClient(); const navigate = useNavigate(); - const { data: kiosk } = useGetKioskContents(activeAddress); - const transferKioskItem = useTransferKioskItem({ objectId, objectType }); - const isContainedInKiosk = kiosk?.list.some( - (kioskItem) => kioskItem.data?.objectId === objectId, - ); - const transferNFT = useMutation({ - mutationFn: async (to: string) => { - if (!to || !signer) { - throw new Error('Missing data'); - } - - if (isContainedInKiosk) { - return transferKioskItem.mutateAsync({ to }); - } - - const tx = new Transaction(); - tx.transferObjects([tx.object(objectId)], to); - - return signer.signAndExecuteTransaction({ - transactionBlock: tx, - options: { - showInput: true, - showEffects: true, - showEvents: true, - }, - }); - }, + const transferNFT = useTransferAsset({ + activeAddress, + objectId, + objectType, + executeFn: normalizeWalletSignAndExecute(signer), onSuccess: (response) => { queryClient.invalidateQueries({ queryKey: ['object', objectId] }); queryClient.invalidateQueries({ queryKey: ['get-kiosk-contents'] }); diff --git a/apps/wallet/src/ui/app/pages/home/nft-transfer/index.tsx b/apps/wallet/src/ui/app/pages/home/nft-transfer/index.tsx index 8d4c920ec38..8ee9730e53c 100644 --- a/apps/wallet/src/ui/app/pages/home/nft-transfer/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/nft-transfer/index.tsx @@ -2,14 +2,13 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useActiveAddress } from '_app/hooks/useActiveAddress'; +import { useActiveAddress, useUnlockedGuard } from '_hooks'; import { Loading, NFTDisplayCard, Overlay } from '_components'; -import { useUnlockedGuard } from '_src/ui/app/hooks/useUnlockedGuard'; import { Navigate, useNavigate, useParams } from 'react-router-dom'; import { TransferNFTForm } from './TransferNFTForm'; import { useOwnedNFT, useIsAssetTransferable } from '@iota/core'; -function NftTransferPage() { +export function NftTransferPage() { const { nftId } = useParams(); const address = useActiveAddress(); // verify that the nft is owned by the user and is transferable @@ -38,5 +37,3 @@ function NftTransferPage() { </Overlay> ); } - -export default NftTransferPage; diff --git a/apps/wallet/src/ui/app/pages/home/nfts/HiddenAsset.tsx b/apps/wallet/src/ui/app/pages/home/nfts/HiddenAsset.tsx index 92ac60bc862..6e0a8bd477f 100644 --- a/apps/wallet/src/ui/app/pages/home/nfts/HiddenAsset.tsx +++ b/apps/wallet/src/ui/app/pages/home/nfts/HiddenAsset.tsx @@ -2,17 +2,18 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { ErrorBoundary } from '_components'; +import { ErrorBoundary, MovedAssetNotification } from '_components'; import { ampli } from '_src/shared/analytics/ampli'; import { type IotaObjectData } from '@iota/iota-sdk/client'; import { useNavigate } from 'react-router-dom'; -import { useHiddenAssets } from '../assets/HiddenAssetsProvider'; +import { toast } from 'react-hot-toast'; import { getKioskIdFromOwnerCap, isKioskOwnerToken, useGetNFTDisplay, useGetObject, useKioskClient, + useHiddenAssets, } from '@iota/core'; import { Card, @@ -25,8 +26,8 @@ import { ImageType, } from '@iota/apps-ui-kit'; import { formatAddress } from '@iota/iota-sdk/utils'; -import { useResolveVideo } from '_src/ui/app/hooks/useResolveVideo'; -import { VisibilityOff } from '@iota/ui-icons'; +import { useResolveVideo } from '_hooks'; +import { VisibilityOff } from '@iota/apps-ui-icons'; export interface HiddenAssetProps { data: IotaObjectData | null | undefined; @@ -38,8 +39,8 @@ export interface HiddenAssetProps { | undefined; } -export default function HiddenAsset(item: HiddenAssetProps) { - const { showAsset } = useHiddenAssets(); +export function HiddenAsset(item: HiddenAssetProps) { + const { showAsset, hideAsset } = useHiddenAssets(); const kioskClient = useKioskClient(); const navigate = useNavigate(); const { objectId, type } = item.data!; @@ -65,6 +66,23 @@ export default function HiddenAsset(item: HiddenAssetProps) { collectibleType: type!, }); } + + function handleShowAsset() { + showAsset(objectId); + toast.success( + (t) => ( + <MovedAssetNotification + t={t} + destination="Visual Assets" + onUndo={() => hideAsset(objectId)} + /> + ), + { + duration: 4000, + }, + ); + } + return ( <ErrorBoundary> <Card type={CardType.Default} onClick={handleHiddenAssetClick}> @@ -88,9 +106,7 @@ export default function HiddenAsset(item: HiddenAssetProps) { <CardBody title={nftMeta?.name ?? 'Asset'} subtitle={formatAddress(objectId)} /> <CardAction type={CardActionType.Link} - onClick={() => { - showAsset(objectId); - }} + onClick={handleShowAsset} icon={<VisibilityOff />} /> </Card> diff --git a/apps/wallet/src/ui/app/pages/home/nfts/HiddenAssets.tsx b/apps/wallet/src/ui/app/pages/home/nfts/HiddenAssets.tsx index abab72eadb2..d01346c3c9a 100644 --- a/apps/wallet/src/ui/app/pages/home/nfts/HiddenAssets.tsx +++ b/apps/wallet/src/ui/app/pages/home/nfts/HiddenAssets.tsx @@ -2,13 +2,13 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import HiddenAsset, { type HiddenAssetProps } from './HiddenAsset'; +import { HiddenAsset, type HiddenAssetProps } from './HiddenAsset'; interface HiddenAssetsProps { items: HiddenAssetProps[]; } -export default function HiddenAssets({ items }: HiddenAssetsProps) { +export function HiddenAssets({ items }: HiddenAssetsProps) { return ( <div className="flex w-full flex-col overflow-y-auto"> {items?.map((object) => <HiddenAsset key={object.data!.objectId} {...object} />)} diff --git a/apps/wallet/src/ui/app/pages/home/nfts/NonVisualAssets.tsx b/apps/wallet/src/ui/app/pages/home/nfts/NonVisualAssets.tsx index 025f2000a98..b603f60e14e 100644 --- a/apps/wallet/src/ui/app/pages/home/nfts/NonVisualAssets.tsx +++ b/apps/wallet/src/ui/app/pages/home/nfts/NonVisualAssets.tsx @@ -6,13 +6,13 @@ import { ExplorerLink, ExplorerLinkType, NoData } from '_components'; import { type IotaObjectData } from '@iota/iota-sdk/client'; import { formatAddress, parseStructTag } from '@iota/iota-sdk/utils'; import { Card, CardAction, CardActionType, CardBody, CardType } from '@iota/apps-ui-kit'; -import { ArrowTopRight } from '@iota/ui-icons'; +import { ArrowTopRight } from '@iota/apps-ui-icons'; interface NonVisualAssetsProps { items: IotaObjectData[]; } -export default function NonVisualAssets({ items }: NonVisualAssetsProps) { +export function NonVisualAssets({ items }: NonVisualAssetsProps) { return ( <div className="flex w-full flex-1 flex-col items-center gap-4"> {items?.length ? ( diff --git a/apps/wallet/src/ui/app/pages/home/nfts/VisualAssets.tsx b/apps/wallet/src/ui/app/pages/home/nfts/VisualAssets.tsx index ce05c7e594a..9962ce5e60a 100644 --- a/apps/wallet/src/ui/app/pages/home/nfts/VisualAssets.tsx +++ b/apps/wallet/src/ui/app/pages/home/nfts/VisualAssets.tsx @@ -2,31 +2,52 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { ErrorBoundary, NFTDisplayCard } from '_components'; +import { ErrorBoundary, NFTDisplayCard, MovedAssetNotification } from '_components'; import { ampli } from '_src/shared/analytics/ampli'; import { type IotaObjectData } from '@iota/iota-sdk/client'; import { Link } from 'react-router-dom'; - -import { useHiddenAssets } from '../assets/HiddenAssetsProvider'; -import { getKioskIdFromOwnerCap, isKioskOwnerToken, useKioskClient } from '@iota/core'; -import { VisibilityOff } from '@iota/ui-icons'; +import { toast } from 'react-hot-toast'; +import { + useHiddenAssets, + getKioskIdFromOwnerCap, + isKioskOwnerToken, + useKioskClient, +} from '@iota/core'; +import { VisibilityOff } from '@iota/apps-ui-icons'; interface VisualAssetsProps { items: IotaObjectData[]; } -export default function VisualAssets({ items }: VisualAssetsProps) { - const { hideAsset } = useHiddenAssets(); +export function VisualAssets({ items }: VisualAssetsProps) { + const { hideAsset, showAsset } = useHiddenAssets(); const kioskClient = useKioskClient(); - function handleHideAsset(event: React.MouseEvent<HTMLButtonElement>, object: IotaObjectData) { + async function handleHideAsset( + event: React.MouseEvent<HTMLButtonElement>, + object: IotaObjectData, + ) { event.preventDefault(); event.stopPropagation(); ampli.clickedHideAsset({ objectId: object.objectId, collectibleType: object.type!, }); - hideAsset(object.objectId); + + await hideAsset(object.objectId); + + toast.success( + (t) => ( + <MovedAssetNotification + t={t} + destination="Hidden Assets" + onUndo={() => showAsset(object.objectId)} + /> + ), + { + duration: 4000, + }, + ); } return ( diff --git a/apps/wallet/src/ui/app/pages/home/nfts/index.tsx b/apps/wallet/src/ui/app/pages/home/nfts/index.tsx index 52c62bf96ae..4cd94599f64 100644 --- a/apps/wallet/src/ui/app/pages/home/nfts/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/nfts/index.tsx @@ -11,21 +11,13 @@ import { SegmentedButton, SegmentedButtonType, } from '@iota/apps-ui-kit'; -import { useActiveAddress } from '_app/hooks/useActiveAddress'; +import { useActiveAddress } from '_hooks'; import { Loading, NoData, PageTemplate } from '_components'; -import { useGetNFTs } from '_src/ui/app/hooks/useGetNFTs'; -import { useEffect, useMemo, useRef, useState } from 'react'; -import HiddenAssets from './HiddenAssets'; -import NonVisualAssets from './NonVisualAssets'; -import VisualAssets from './VisualAssets'; -import { Warning } from '@iota/ui-icons'; -import { useOnScreen } from '@iota/core'; - -enum AssetCategory { - Visual = 'Visual', - Other = 'Other', - Hidden = 'Hidden', -} +import { HiddenAssets } from './HiddenAssets'; +import { NonVisualAssets } from './NonVisualAssets'; +import { VisualAssets } from './VisualAssets'; +import { Warning } from '@iota/apps-ui-icons'; +import { useHiddenAssets, usePageAssets, AssetCategory } from '@iota/core'; const ASSET_CATEGORIES = [ { @@ -42,98 +34,23 @@ const ASSET_CATEGORIES = [ }, ]; -function NftsPage() { - const [selectedAssetCategory, setSelectedAssetCategory] = useState<AssetCategory | null>(null); - const observerElem = useRef<HTMLDivElement | null>(null); - const { isIntersecting } = useOnScreen(observerElem); - +export function NftsPage() { const accountAddress = useActiveAddress(); + const { hiddenAssets } = useHiddenAssets(); + const { - data: ownedAssets, - hasNextPage, - isLoading, - isFetchingNextPage, - fetchNextPage, - error, isPending, + isAssetsLoaded, isError, - } = useGetNFTs(accountAddress); - - const isAssetsLoaded = !!ownedAssets; - - const isSpinnerVisible = isFetchingNextPage && hasNextPage; - - const filteredAssets = (() => { - if (!ownedAssets) return []; - switch (selectedAssetCategory) { - case AssetCategory.Visual: - return ownedAssets.visual; - case AssetCategory.Other: - return ownedAssets.other; - default: - return []; - } - })(); - - const filteredHiddenAssets = useMemo(() => { - return ( - ownedAssets?.hidden - .flatMap((data) => { - return { - data: data, - display: data?.display?.data, - }; - }) - .sort((nftA, nftB) => { - const nameA = nftA.display?.name || ''; - const nameB = nftB.display?.name || ''; - - if (nameA < nameB) { - return -1; - } else if (nameA > nameB) { - return 1; - } - return 0; - }) ?? [] - ); - }, [ownedAssets]); - - useEffect(() => { - if (isIntersecting && hasNextPage && !isFetchingNextPage) { - fetchNextPage(); - } - }, [isIntersecting, fetchNextPage, hasNextPage, isFetchingNextPage]); - - useEffect(() => { - let computeSelectedCategory = false; - if ( - (selectedAssetCategory === AssetCategory.Visual && ownedAssets?.visual.length === 0) || - (selectedAssetCategory === AssetCategory.Other && ownedAssets?.other.length === 0) || - (selectedAssetCategory === AssetCategory.Hidden && ownedAssets?.hidden.length === 0) || - !selectedAssetCategory - ) { - computeSelectedCategory = true; - } - if (computeSelectedCategory && ownedAssets) { - const defaultCategory = - ownedAssets.visual.length > 0 - ? AssetCategory.Visual - : ownedAssets.other.length > 0 - ? AssetCategory.Other - : ownedAssets.hidden.length > 0 - ? AssetCategory.Hidden - : null; - setSelectedAssetCategory(defaultCategory); - } - }, [ownedAssets]); - - if (isLoading) { - return ( - <div className="mt-1 flex w-full justify-center"> - <LoadingIndicator /> - </div> - ); - } + error, + ownedAssets, + filteredAssets, + filteredHiddenAssets, + selectedAssetCategory, + setSelectedAssetCategory, + isSpinnerVisible, + observerElem, + } = usePageAssets(accountAddress, hiddenAssets); return ( <PageTemplate title="Assets" isTitleCentered> @@ -196,5 +113,3 @@ function NftsPage() { </PageTemplate> ); } - -export default NftsPage; diff --git a/apps/wallet/src/ui/app/pages/home/receipt/index.tsx b/apps/wallet/src/ui/app/pages/home/receipt/index.tsx index 6afd778f2a9..0d01df4bf70 100644 --- a/apps/wallet/src/ui/app/pages/home/receipt/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/receipt/index.tsx @@ -3,16 +3,15 @@ // SPDX-License-Identifier: Apache-2.0 import { Loading, Overlay, ReceiptCard } from '_components'; -import { useActiveAddress } from '_src/ui/app/hooks/useActiveAddress'; -import { useUnlockedGuard } from '_src/ui/app/hooks/useUnlockedGuard'; +import { useActiveAddress, useUnlockedGuard } from '_hooks'; import { useCallback, useMemo, useState } from 'react'; import { Navigate, useLocation, useNavigate, useSearchParams } from 'react-router-dom'; -import { Checkmark, Warning } from '@iota/ui-icons'; +import { Checkmark, Warning } from '@iota/apps-ui-icons'; import { InfoBox, InfoBoxType, InfoBoxStyle } from '@iota/apps-ui-kit'; import type { IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; import { useGetTransaction } from '@iota/core'; -function ReceiptPage() { +export function ReceiptPage() { const location = useLocation(); const [searchParams] = useSearchParams(); const [showModal, setShowModal] = useState(true); @@ -64,7 +63,6 @@ function ReceiptPage() { title={pageTitle} closeOverlay={closeReceipt} closeIcon={<Checkmark fill="currentColor" className="text-iota-light h-8 w-8" />} - showBackButton > {isError ? ( <div className="mb-2 flex h-full w-full items-center justify-center p-2"> @@ -83,5 +81,3 @@ function ReceiptPage() { </Loading> ); } - -export default ReceiptPage; diff --git a/apps/wallet/src/ui/app/pages/home/tokens/MyTokens.tsx b/apps/wallet/src/ui/app/pages/home/tokens/MyTokens.tsx index 3477e205ff2..483bb0ecb99 100644 --- a/apps/wallet/src/ui/app/pages/home/tokens/MyTokens.tsx +++ b/apps/wallet/src/ui/app/pages/home/tokens/MyTokens.tsx @@ -3,10 +3,10 @@ import { SegmentedButton, SegmentedButtonType, ButtonSegment, Title } from '@iota/apps-ui-kit'; import { useSortedCoinsByCategories } from '@iota/core'; -import { RecognizedBadge } from '@iota/ui-icons'; +import { RecognizedBadge } from '@iota/apps-ui-icons'; import { ampli } from '_src/shared/analytics/ampli'; import { Loading } from '_src/ui/app/components'; -import { usePinnedCoinTypes } from '_src/ui/app/hooks/usePinnedCoinTypes'; +import { usePinnedCoinTypes } from '_hooks'; import { useState } from 'react'; import { PinButton } from './PinButton'; import { TokenLink } from './TokenLink'; diff --git a/apps/wallet/src/ui/app/pages/home/tokens/PinButton.tsx b/apps/wallet/src/ui/app/pages/home/tokens/PinButton.tsx index 86f81e4af95..fa7adc5d929 100644 --- a/apps/wallet/src/ui/app/pages/home/tokens/PinButton.tsx +++ b/apps/wallet/src/ui/app/pages/home/tokens/PinButton.tsx @@ -1,7 +1,7 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { Pined, Unpined } from '@iota/ui-icons'; +import { Pined, Unpined } from '@iota/apps-ui-icons'; interface PinButtonProps { isPinned?: boolean; diff --git a/apps/wallet/src/ui/app/pages/home/tokens/ReceiveTokensDialog.tsx b/apps/wallet/src/ui/app/pages/home/tokens/ReceiveTokensDialog.tsx index d790d723b1f..b4f10fd99f3 100644 --- a/apps/wallet/src/ui/app/pages/home/tokens/ReceiveTokensDialog.tsx +++ b/apps/wallet/src/ui/app/pages/home/tokens/ReceiveTokensDialog.tsx @@ -3,15 +3,14 @@ import { useCallback } from 'react'; import { Button, Address, Dialog, DialogContent, DialogBody, Header } from '@iota/apps-ui-kit'; -import { useCopyToClipboard } from '_hooks'; +import { useCopyToClipboard, useActiveAccount } from '_hooks'; import { QR } from '@iota/core'; import { toast } from 'react-hot-toast'; import { useIotaLedgerClient } from '_src/ui/app/components'; import { isLedgerAccountSerializedUI, type LedgerAccountSerializedUI, -} from '_src/background/accounts/LedgerAccount'; -import { useActiveAccount } from '_src/ui/app/hooks/useActiveAccount'; +} from '_src/background/accounts/ledgerAccount'; interface ReceiveTokensDialogProps { address: string; diff --git a/apps/wallet/src/ui/app/pages/home/tokens/TokenDetailsPage.tsx b/apps/wallet/src/ui/app/pages/home/tokens/TokenDetailsPage.tsx index 66083ed5272..cbf8ff70e02 100644 --- a/apps/wallet/src/ui/app/pages/home/tokens/TokenDetailsPage.tsx +++ b/apps/wallet/src/ui/app/pages/home/tokens/TokenDetailsPage.tsx @@ -2,10 +2,8 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import TokenDetails from './TokensDetails'; +import { TokenDetails } from './TokensDetails'; -function TokenDetailsPage() { +export function TokenDetailsPage() { return <TokenDetails />; } - -export default TokenDetailsPage; diff --git a/apps/wallet/src/ui/app/pages/home/tokens/TokenStakingOverview.tsx b/apps/wallet/src/ui/app/pages/home/tokens/TokenStakingOverview.tsx index 1c164c4a9d6..882c3818736 100644 --- a/apps/wallet/src/ui/app/pages/home/tokens/TokenStakingOverview.tsx +++ b/apps/wallet/src/ui/app/pages/home/tokens/TokenStakingOverview.tsx @@ -22,7 +22,7 @@ import { ImageShape, } from '@iota/apps-ui-kit'; import { useNavigate } from 'react-router-dom'; -import { Stake } from '@iota/ui-icons'; +import { Stake } from '@iota/apps-ui-icons'; export function TokenStakingOverview({ accountAddress, diff --git a/apps/wallet/src/ui/app/pages/home/tokens/TokensDetails.tsx b/apps/wallet/src/ui/app/pages/home/tokens/TokensDetails.tsx index 064184e9f31..0b16d15cf1f 100644 --- a/apps/wallet/src/ui/app/pages/home/tokens/TokensDetails.tsx +++ b/apps/wallet/src/ui/app/pages/home/tokens/TokensDetails.tsx @@ -3,9 +3,13 @@ // SPDX-License-Identifier: Apache-2.0 import { ExplorerLinkType, Loading, UnlockAccountButton } from '_components'; -import { useAppSelector, useCoinsReFetchingConfig } from '_hooks'; -import { useActiveAccount } from '_src/ui/app/hooks/useActiveAccount'; -import FaucetRequestButton from '_src/ui/app/shared/faucet/FaucetRequestButton'; +import { + useActiveAccount, + useAppSelector, + useCoinsReFetchingConfig, + useExplorerLink, +} from '_hooks'; +import { FaucetRequestButton } from '_src/ui/app/shared/faucet/FaucetRequestButton'; import { useFeature } from '@growthbook/growthbook-react'; import { toast } from 'react-hot-toast'; import { @@ -31,12 +35,11 @@ import { Network } from '@iota/iota-sdk/client'; import { formatAddress, IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { useQuery } from '@tanstack/react-query'; import { useEffect, useState } from 'react'; -import { ArrowBottomLeft, Info, Send } from '@iota/ui-icons'; -import Interstitial, { type InterstitialConfig } from '../interstitial'; +import { ArrowBottomLeft, Info, Send } from '@iota/apps-ui-icons'; +import { Interstitial, type InterstitialConfig } from '../interstitial'; import { CoinBalance } from './coin-balance'; import { TokenStakingOverview } from './TokenStakingOverview'; import { useNavigate } from 'react-router-dom'; -import { useExplorerLink } from '_app/hooks/useExplorerLink'; import { MyTokens } from './MyTokens'; import { ReceiveTokensDialog } from './ReceiveTokensDialog'; @@ -44,7 +47,7 @@ interface TokenDetailsProps { coinType?: string; } -function TokenDetails({ coinType }: TokenDetailsProps) { +export function TokenDetails({ coinType }: TokenDetailsProps) { const [dialogReceiveOpen, setDialogReceiveOpen] = useState(false); const navigate = useNavigate(); const [interstitialDismissed, setInterstitialDismissed] = useState<boolean>(false); @@ -234,5 +237,3 @@ function TokenDetails({ coinType }: TokenDetailsProps) { </> ); } - -export default TokenDetails; diff --git a/apps/wallet/src/ui/app/pages/home/tokens/TokensStackIcon.tsx b/apps/wallet/src/ui/app/pages/home/tokens/TokensStackIcon.tsx index 836b8fd3cbe..e140d51c64c 100644 --- a/apps/wallet/src/ui/app/pages/home/tokens/TokensStackIcon.tsx +++ b/apps/wallet/src/ui/app/pages/home/tokens/TokensStackIcon.tsx @@ -37,4 +37,3 @@ export const SvgIotaTokensStack = (props: SVGProps<SVGSVGElement>) => ( </defs> </svg> ); -export default SvgIotaTokensStack; diff --git a/apps/wallet/src/ui/app/pages/home/tokens/coin-balance/index.tsx b/apps/wallet/src/ui/app/pages/home/tokens/coin-balance/index.tsx index 75aecdf86ca..7660cf9bcd7 100644 --- a/apps/wallet/src/ui/app/pages/home/tokens/coin-balance/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/tokens/coin-balance/index.tsx @@ -2,10 +2,12 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 import { useAppSelector } from '_hooks'; -import { useBalanceInUSD, useFormatCoin } from '@iota/core'; +import { CoinFormat, formatBalance, useBalanceInUSD, useFormatCoin } from '@iota/core'; import { Network } from '@iota/iota-sdk/client'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { useMemo } from 'react'; +import { Tooltip, TooltipPosition } from '@iota/apps-ui-kit'; +import BigNumber from 'bignumber.js'; export interface CoinProps { type: string; @@ -37,17 +39,39 @@ function WalletBalanceUsd({ amount: walletBalance }: WalletBalanceUsdProps) { export function CoinBalance({ amount: walletBalance, type }: CoinProps) { const network = useAppSelector((state) => state.app.network); - const [formatted, symbol] = useFormatCoin(walletBalance, type); + const [formatted, symbol, { data: coinMetadata }] = useFormatCoin(walletBalance, type); + + const iotaDecimals = coinMetadata?.decimals ?? 9; + const bnBalance = new BigNumber(walletBalance.toString()).shiftedBy(-1 * iotaDecimals); + const shouldShowTooltip = bnBalance.gt(0) && bnBalance.lt(1); return ( <> <div className="flex items-baseline gap-0.5"> - <div - className="text-headline-lg text-neutral-10 dark:text-neutral-92" - data-testid="coin-balance" - > - {formatted} - </div> + {shouldShowTooltip ? ( + <Tooltip + text={formatBalance( + walletBalance, + coinMetadata?.decimals ?? 9, + CoinFormat.FULL, + )} + position={TooltipPosition.Bottom} + > + <div + className="text-headline-lg text-neutral-10 dark:text-neutral-92" + data-testid="coin-balance" + > + {formatted} + </div> + </Tooltip> + ) : ( + <div + className="text-headline-lg text-neutral-10 dark:text-neutral-92" + data-testid="coin-balance" + > + {formatted} + </div> + )} <div className="text-label-md text-neutral-40">{symbol}</div> </div> {network === Network.Mainnet ? <WalletBalanceUsd amount={walletBalance} /> : null} diff --git a/apps/wallet/src/ui/app/pages/home/tokens/icon-link/index.tsx b/apps/wallet/src/ui/app/pages/home/tokens/icon-link/index.tsx index 2b3d21fca68..e2ea9b847e6 100644 --- a/apps/wallet/src/ui/app/pages/home/tokens/icon-link/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/tokens/icon-link/index.tsx @@ -15,7 +15,7 @@ export interface IconLinkProps { text: string; } -function IconLink({ to, icon, disabled = false, text }: IconLinkProps) { +function IconLinkComponent({ to, icon, disabled = false, text }: IconLinkProps) { return ( <Link to={to} @@ -35,4 +35,4 @@ function IconLink({ to, icon, disabled = false, text }: IconLinkProps) { ); } -export default memo(IconLink); +export const IconLink = memo(IconLinkComponent); diff --git a/apps/wallet/src/ui/app/pages/home/tokens/index.tsx b/apps/wallet/src/ui/app/pages/home/tokens/index.tsx index 98dd57694ab..f7dd11c7d51 100644 --- a/apps/wallet/src/ui/app/pages/home/tokens/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/tokens/index.tsx @@ -3,11 +3,10 @@ // SPDX-License-Identifier: Apache-2.0 import { Route, Routes } from 'react-router-dom'; +import { TokenDetailsPage } from './TokenDetailsPage'; +import { TokenDetails } from './TokensDetails'; -import TokenDetailsPage from './TokenDetailsPage'; -import TokenDetails from './TokensDetails'; - -function TokensPage() { +export function TokensPage() { return ( <Routes> <Route path="/" element={<TokenDetails />} /> @@ -15,5 +14,3 @@ function TokensPage() { </Routes> ); } - -export default TokensPage; diff --git a/apps/wallet/src/ui/app/pages/home/transactions/CompletedTransactions.tsx b/apps/wallet/src/ui/app/pages/home/transactions/CompletedTransactions.tsx index 562144c0886..37c0ad440f3 100644 --- a/apps/wallet/src/ui/app/pages/home/transactions/CompletedTransactions.tsx +++ b/apps/wallet/src/ui/app/pages/home/transactions/CompletedTransactions.tsx @@ -4,9 +4,9 @@ import { ErrorBoundary, Loading, TransactionCard, NoData } from '_components'; import { useQueryTransactionsByAddress } from '@iota/core'; -import { useActiveAddress } from '_src/ui/app/hooks/useActiveAddress'; +import { useActiveAddress } from '_hooks'; import { InfoBox, InfoBoxStyle, InfoBoxType } from '@iota/apps-ui-kit'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; export function CompletedTransactions() { const activeAddress = useActiveAddress(); diff --git a/apps/wallet/src/ui/app/pages/home/transactions/index.tsx b/apps/wallet/src/ui/app/pages/home/transactions/index.tsx index 409d1db7824..3dcbd1deab1 100644 --- a/apps/wallet/src/ui/app/pages/home/transactions/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/transactions/index.tsx @@ -2,13 +2,12 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useActiveAccount } from '_src/ui/app/hooks/useActiveAccount'; -import { useUnlockedGuard } from '_src/ui/app/hooks/useUnlockedGuard'; +import { useActiveAccount, useUnlockedGuard } from '_hooks'; import { Navigate, useParams } from 'react-router-dom'; import { CompletedTransactions } from './CompletedTransactions'; import { PageTemplate } from '_src/ui/app/components'; -function TransactionBlocksPage() { +export function TransactionBlocksPage() { const activeAccount = useActiveAccount(); const { status } = useParams(); const isPendingTransactions = status === 'pending'; @@ -26,5 +25,3 @@ function TransactionBlocksPage() { </PageTemplate> ); } - -export default TransactionBlocksPage; diff --git a/apps/wallet/src/ui/app/pages/home/transfer-coin/CoinSelector.tsx b/apps/wallet/src/ui/app/pages/home/transfer-coin/CoinSelector.tsx index d94b31d6634..50fbb66dee6 100644 --- a/apps/wallet/src/ui/app/pages/home/transfer-coin/CoinSelector.tsx +++ b/apps/wallet/src/ui/app/pages/home/transfer-coin/CoinSelector.tsx @@ -3,11 +3,11 @@ // SPDX-License-Identifier: Apache-2.0 import { ActiveCoinsCard, Overlay } from '_components'; -import { useUnlockedGuard } from '_src/ui/app/hooks/useUnlockedGuard'; +import { useUnlockedGuard } from '_hooks'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { useNavigate, useSearchParams } from 'react-router-dom'; -function CoinsSelectorPage() { +export function CoinsSelectorPage() { const [searchParams] = useSearchParams(); const coinType = searchParams.get('type') || IOTA_TYPE_ARG; const navigate = useNavigate(); @@ -32,5 +32,3 @@ function CoinsSelectorPage() { </Overlay> ); } - -export default CoinsSelectorPage; diff --git a/apps/wallet/src/ui/app/pages/home/transfer-coin/PreviewTransfer.tsx b/apps/wallet/src/ui/app/pages/home/transfer-coin/PreviewTransfer.tsx index faed355c243..8589527279f 100644 --- a/apps/wallet/src/ui/app/pages/home/transfer-coin/PreviewTransfer.tsx +++ b/apps/wallet/src/ui/app/pages/home/transfer-coin/PreviewTransfer.tsx @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import { ExplorerLink, ExplorerLinkType, TxnAmount } from '_components'; -import { useActiveAddress } from '_src/ui/app/hooks/useActiveAddress'; +import { useActiveAddress } from '_hooks'; import { parseAmount, useCoinMetadata, useFormatCoin } from '@iota/core'; import { Divider, KeyValueInfo } from '@iota/apps-ui-kit'; import { formatAddress, IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; diff --git a/apps/wallet/src/ui/app/pages/home/transfer-coin/SendTokenForm.tsx b/apps/wallet/src/ui/app/pages/home/transfer-coin/SendTokenForm.tsx index e1cf2696b6f..a18a8732edd 100644 --- a/apps/wallet/src/ui/app/pages/home/transfer-coin/SendTokenForm.tsx +++ b/apps/wallet/src/ui/app/pages/home/transfer-coin/SendTokenForm.tsx @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useActiveAddress } from '_app/hooks/useActiveAddress'; +import { useActiveAddress } from '_hooks'; import { Loading } from '_components'; import { useGetAllCoins, @@ -27,7 +27,7 @@ import { ButtonType, ButtonHtmlType, } from '@iota/apps-ui-kit'; -import { Exclamation } from '@iota/ui-icons'; +import { Exclamation } from '@iota/apps-ui-icons'; const INITIAL_VALUES = { to: '', diff --git a/apps/wallet/src/ui/app/pages/home/transfer-coin/index.tsx b/apps/wallet/src/ui/app/pages/home/transfer-coin/index.tsx index f4e3a2555c1..26ed1e1299d 100644 --- a/apps/wallet/src/ui/app/pages/home/transfer-coin/index.tsx +++ b/apps/wallet/src/ui/app/pages/home/transfer-coin/index.tsx @@ -5,9 +5,7 @@ import { Overlay } from '_components'; import { ampli } from '_src/shared/analytics/ampli'; import { getSignerOperationErrorMessage } from '_src/ui/app/helpers/errorMessages'; -import { useActiveAccount } from '_src/ui/app/hooks/useActiveAccount'; -import { useSigner } from '_src/ui/app/hooks/useSigner'; -import { useUnlockedGuard } from '_src/ui/app/hooks/useUnlockedGuard'; +import { useSigner, useActiveAccount, useUnlockedGuard } from '_hooks'; import { COINS_QUERY_REFETCH_INTERVAL, COINS_QUERY_STALE_TIME, @@ -25,11 +23,11 @@ import { Navigate, useNavigate, useSearchParams } from 'react-router-dom'; import { PreviewTransfer } from './PreviewTransfer'; import { SendTokenForm, type SubmitProps } from './SendTokenForm'; import { Button, ButtonType, LoadingIndicator } from '@iota/apps-ui-kit'; -import { Loader } from '@iota/ui-icons'; +import { Loader } from '@iota/apps-ui-icons'; import { useIotaClientQuery } from '@iota/dapp-kit'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; -function TransferCoinPage() { +export function TransferCoinPage() { const [searchParams] = useSearchParams(); const selectedCoinType = searchParams.get('type'); const [showTransactionPreview, setShowTransactionPreview] = useState<boolean>(false); @@ -89,22 +87,25 @@ function TransferCoinPage() { if (!transaction || !signer) { throw new Error('Missing data'); } - const sentryTransaction = Sentry.startTransaction({ - name: 'send-tokens', - }); - - try { - return signer.signAndExecuteTransaction({ - transactionBlock: transaction, - options: { - showInput: true, - showEffects: true, - showEvents: true, - }, - }); - } finally { - sentryTransaction.finish(); - } + return Sentry.startSpan( + { + name: 'send-tokens', + }, + (span) => { + try { + return signer.signAndExecuteTransaction({ + transactionBlock: transaction, + options: { + showInput: true, + showEffects: true, + showEvents: true, + }, + }); + } finally { + span?.end(); + } + }, + ); }, onSuccess: (response) => { queryClient.invalidateQueries({ queryKey: ['get-coins'] }); @@ -206,5 +207,3 @@ function TransferCoinPage() { </Overlay> ); } - -export default TransferCoinPage; diff --git a/apps/wallet/src/ui/app/pages/restricted/index.tsx b/apps/wallet/src/ui/app/pages/restricted/index.tsx index 8571e51a6c5..0d43c36ff98 100644 --- a/apps/wallet/src/ui/app/pages/restricted/index.tsx +++ b/apps/wallet/src/ui/app/pages/restricted/index.tsx @@ -2,9 +2,9 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { IotaLogoWeb } from '@iota/ui-icons'; +import { IotaLogoWeb } from '@iota/apps-ui-icons'; import { PageMainLayout } from '_src/ui/app/shared/page-main-layout/PageMainLayout'; -import { useInitializedGuard } from '../../hooks'; +import { useInitializedGuard } from '_hooks'; export function RestrictedPage() { useInitializedGuard(true); diff --git a/apps/wallet/src/ui/app/pages/site-connect/index.tsx b/apps/wallet/src/ui/app/pages/site-connect/index.tsx index 74238e2ab93..356a02bceac 100644 --- a/apps/wallet/src/ui/app/pages/site-connect/index.tsx +++ b/apps/wallet/src/ui/app/pages/site-connect/index.tsx @@ -9,20 +9,18 @@ import { SectionHeader, UserApproveContainer, } from '_components'; -import { useAppDispatch, useAppSelector } from '_hooks'; -import type { RootState } from '_redux/RootReducer'; +import { useAppDispatch, useAppSelector, useAccountGroups, useActiveAccount } from '_hooks'; +import type { RootState } from '_src/ui/app/redux/rootReducer'; import { permissionsSelectors, respondToPermissionRequest } from '_redux/slices/permissions'; -import { type SerializedUIAccount } from '_src/background/accounts/Account'; +import { type SerializedUIAccount } from '_src/background/accounts/account'; import { ampli } from '_src/shared/analytics/ampli'; import { useCallback, useEffect, useMemo, useState } from 'react'; import { useParams } from 'react-router-dom'; -import { useAccountGroups } from '../../hooks/useAccountGroups'; -import { useActiveAccount } from '../../hooks/useActiveAccount'; import { PageMainLayoutTitle } from '../../shared/page-main-layout/PageMainLayoutTitle'; import { InfoBox, InfoBoxStyle, InfoBoxType } from '@iota/apps-ui-kit'; -import { Warning, Info } from '@iota/ui-icons'; +import { Warning, Info } from '@iota/apps-ui-icons'; -function SiteConnectPage() { +export function SiteConnectPage() { const { requestID } = useParams(); const permissionsInitialized = useAppSelector(({ permissions }) => permissions.initialized); const loading = !permissionsInitialized; @@ -177,5 +175,3 @@ function SiteConnectPage() { </Loading> ); } - -export default SiteConnectPage; diff --git a/apps/wallet/src/ui/app/redux/RootReducer.ts b/apps/wallet/src/ui/app/redux/rootReducer.ts similarity index 100% rename from apps/wallet/src/ui/app/redux/RootReducer.ts rename to apps/wallet/src/ui/app/redux/rootReducer.ts diff --git a/apps/wallet/src/ui/app/redux/slices/app/AppType.ts b/apps/wallet/src/ui/app/redux/slices/app/appType.ts similarity index 100% rename from apps/wallet/src/ui/app/redux/slices/app/AppType.ts rename to apps/wallet/src/ui/app/redux/slices/app/appType.ts diff --git a/apps/wallet/src/ui/app/redux/slices/app/index.ts b/apps/wallet/src/ui/app/redux/slices/app/index.ts index 35600347357..0a6189c6ada 100644 --- a/apps/wallet/src/ui/app/redux/slices/app/index.ts +++ b/apps/wallet/src/ui/app/redux/slices/app/index.ts @@ -2,15 +2,15 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { walletApiProvider } from '_app/ApiProvider'; -import type { RootState } from '_redux/RootReducer'; +import { walletApiProvider } from '_src/ui/app/apiProvider'; +import type { RootState } from '_src/ui/app/redux/rootReducer'; import type { NetworkEnvType } from '@iota/core'; -import type { AppThunkConfig } from '_store/thunk-extras'; +import type { AppThunkConfig } from '_src/ui/app/redux/store/thunkExtras'; import { getDefaultNetwork, type Network } from '@iota/iota-sdk/client'; import { createAsyncThunk, createSlice } from '@reduxjs/toolkit'; import type { PayloadAction } from '@reduxjs/toolkit'; -import { AppType } from './AppType'; +import { AppType } from './appType'; type AppState = { appType: AppType; diff --git a/apps/wallet/src/ui/app/redux/slices/iota-objects/Coin.ts b/apps/wallet/src/ui/app/redux/slices/iota-objects/coin.ts similarity index 96% rename from apps/wallet/src/ui/app/redux/slices/iota-objects/Coin.ts rename to apps/wallet/src/ui/app/redux/slices/iota-objects/coin.ts index d624c690069..267b4f573ee 100644 --- a/apps/wallet/src/ui/app/redux/slices/iota-objects/Coin.ts +++ b/apps/wallet/src/ui/app/redux/slices/iota-objects/coin.ts @@ -4,7 +4,7 @@ import type { IotaMoveObject, IotaObjectData } from '@iota/iota-sdk/client'; -const COIN_TYPE = '0x2::coin::Coin'; +export const COIN_TYPE = '0x2::coin::Coin'; const COIN_TYPE_ARG_REGEX = /^0x2::coin::Coin<(.+)>$/; export const IOTA_BIP44_COIN_TYPE = 4218; diff --git a/apps/wallet/src/ui/app/redux/slices/permissions/index.ts b/apps/wallet/src/ui/app/redux/slices/permissions/index.ts index 4912cd8720b..349d54bd8bf 100644 --- a/apps/wallet/src/ui/app/redux/slices/permissions/index.ts +++ b/apps/wallet/src/ui/app/redux/slices/permissions/index.ts @@ -3,8 +3,8 @@ // SPDX-License-Identifier: Apache-2.0 import type { Permission } from '_messages/payloads/permissions'; -import type { RootState } from '_redux/RootReducer'; -import type { AppThunkConfig } from '_store/thunk-extras'; +import type { RootState } from '_src/ui/app/redux/rootReducer'; +import type { AppThunkConfig } from '_src/ui/app/redux/store/thunkExtras'; import { createAsyncThunk, createEntityAdapter, diff --git a/apps/wallet/src/ui/app/redux/slices/transaction-requests/index.ts b/apps/wallet/src/ui/app/redux/slices/transaction-requests/index.ts index 3b5e0ee7fdc..7c7d1ca8b6e 100644 --- a/apps/wallet/src/ui/app/redux/slices/transaction-requests/index.ts +++ b/apps/wallet/src/ui/app/redux/slices/transaction-requests/index.ts @@ -2,15 +2,15 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import type { ApprovalRequest } from '_payloads/transactions/ApprovalRequest'; -import type { RootState } from '_redux/RootReducer'; +import type { ApprovalRequest } from '_src/shared/messaging/messages/payloads/transactions/approvalRequest'; +import type { RootState } from '_src/ui/app/redux/rootReducer'; import { getSignerOperationErrorMessage } from '_src/ui/app/helpers/errorMessages'; import { type SignedMessage, type SignedTransaction, type WalletSigner, -} from '_src/ui/app/WalletSigner'; -import type { AppThunkConfig } from '_store/thunk-extras'; +} from '_src/ui/app/walletSigner'; +import type { AppThunkConfig } from '_src/ui/app/redux/store/thunkExtras'; import { type IotaTransactionBlockResponse } from '@iota/iota-sdk/client'; import { Transaction } from '@iota/iota-sdk/transactions'; import { fromB64 } from '@iota/iota-sdk/utils'; diff --git a/apps/wallet/src/ui/app/redux/store/index.ts b/apps/wallet/src/ui/app/redux/store/index.ts index 3e5d2305941..e21ef95d800 100644 --- a/apps/wallet/src/ui/app/redux/store/index.ts +++ b/apps/wallet/src/ui/app/redux/store/index.ts @@ -2,10 +2,10 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import rootReducer from '_redux/RootReducer'; +import rootReducer from '_src/ui/app/redux/rootReducer'; import { configureStore } from '@reduxjs/toolkit'; -import { thunkExtras } from './thunk-extras'; +import { thunkExtras } from './thunkExtras'; const store = configureStore({ reducer: rootReducer, diff --git a/apps/wallet/src/ui/app/redux/store/thunk-extras.ts b/apps/wallet/src/ui/app/redux/store/thunkExtras.ts similarity index 77% rename from apps/wallet/src/ui/app/redux/store/thunk-extras.ts rename to apps/wallet/src/ui/app/redux/store/thunkExtras.ts index 583943762f3..3e7becadb78 100644 --- a/apps/wallet/src/ui/app/redux/store/thunk-extras.ts +++ b/apps/wallet/src/ui/app/redux/store/thunkExtras.ts @@ -3,8 +3,8 @@ // SPDX-License-Identifier: Apache-2.0 import { BackgroundClient } from '_app/background-client'; -import { growthbook } from '_app/experimentation/feature-gating'; -import type { RootState } from '_redux/RootReducer'; +import { growthbook } from '_src/ui/app/experimentation/featureGating'; +import type { RootState } from '_src/ui/app/redux/rootReducer'; import type { AppDispatch } from '_store'; export const thunkExtras = { diff --git a/apps/wallet/src/ui/app/shared/card-layout/index.tsx b/apps/wallet/src/ui/app/shared/card-layout/index.tsx index c77416b4082..51462be3475 100644 --- a/apps/wallet/src/ui/app/shared/card-layout/index.tsx +++ b/apps/wallet/src/ui/app/shared/card-layout/index.tsx @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import type { ReactNode } from 'react'; -import { CheckmarkFilled, IotaLogoMark } from '@iota/ui-icons'; +import { CheckmarkFilled, IotaLogoMark } from '@iota/apps-ui-icons'; import { Header } from '@iota/apps-ui-kit'; export interface CardLayoutProps { diff --git a/apps/wallet/src/ui/app/shared/dapp-status/actions/index.ts b/apps/wallet/src/ui/app/shared/dapp-status/actions/index.ts index 17015037fb0..81e6989d859 100644 --- a/apps/wallet/src/ui/app/shared/dapp-status/actions/index.ts +++ b/apps/wallet/src/ui/app/shared/dapp-status/actions/index.ts @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import type { AppThunkConfig } from '_redux/store/thunk-extras'; +import type { AppThunkConfig } from '_src/ui/app/redux/store/thunkExtras'; import { createAsyncThunk } from '@reduxjs/toolkit'; export const appDisconnect = createAsyncThunk< diff --git a/apps/wallet/src/ui/app/shared/dapp-status/index.tsx b/apps/wallet/src/ui/app/shared/dapp-status/index.tsx index 42cb18e8598..794a8166abf 100644 --- a/apps/wallet/src/ui/app/shared/dapp-status/index.tsx +++ b/apps/wallet/src/ui/app/shared/dapp-status/index.tsx @@ -3,19 +3,18 @@ // SPDX-License-Identifier: Apache-2.0 import { Loading } from '_components'; -import { useAppDispatch, useAppSelector } from '_hooks'; +import { useAppDispatch, useAppSelector, useActiveAddress } from '_hooks'; import { createDappStatusSelector } from '_redux/slices/permissions'; import { ampli } from '_src/shared/analytics/ampli'; import { useClick, useDismiss, useFloating, useInteractions } from '@floating-ui/react'; import { AnimatePresence, motion } from 'framer-motion'; import { memo, useCallback, useMemo, useState } from 'react'; -import { useActiveAddress } from '../../hooks/useActiveAddress'; import { ButtonConnectedTo } from '../ButtonConnectedTo'; import { appDisconnect } from './actions'; -import { Link } from '@iota/ui-icons'; +import { Link } from '@iota/apps-ui-icons'; import { Button, ButtonSize, ButtonType } from '@iota/apps-ui-kit'; -function DappStatus() { +function DappStatusComponent() { const dispatch = useAppDispatch(); const activeOriginUrl = useAppSelector(({ app }) => app.activeOrigin); const activeOrigin = useMemo(() => { @@ -147,4 +146,4 @@ function DappStatus() { ); } -export default memo(DappStatus); +export const DappStatus = memo(DappStatusComponent); diff --git a/apps/wallet/src/ui/app/shared/faucet/FaucetMessageInfo.tsx b/apps/wallet/src/ui/app/shared/faucet/FaucetMessageInfo.tsx index ffdd417b5b3..460f56b5340 100644 --- a/apps/wallet/src/ui/app/shared/faucet/FaucetMessageInfo.tsx +++ b/apps/wallet/src/ui/app/shared/faucet/FaucetMessageInfo.tsx @@ -11,7 +11,7 @@ export interface FaucetMessageInfoProps { totalReceived?: number | null; } -function FaucetMessageInfo({ +export function FaucetMessageInfo({ error = null, loading = false, totalReceived = null, @@ -30,5 +30,3 @@ function FaucetMessageInfo({ <>{`${totalReceived ? `${coinsReceivedFormatted} ` : ''}${coinsReceivedSymbol} requested`}</> ); } - -export default FaucetMessageInfo; diff --git a/apps/wallet/src/ui/app/shared/faucet/FaucetRequestButton.tsx b/apps/wallet/src/ui/app/shared/faucet/FaucetRequestButton.tsx index fd8326069b2..0cd90bf29c0 100644 --- a/apps/wallet/src/ui/app/shared/faucet/FaucetRequestButton.tsx +++ b/apps/wallet/src/ui/app/shared/faucet/FaucetRequestButton.tsx @@ -10,9 +10,9 @@ import { toast } from 'react-hot-toast'; import { useFaucetMutation } from './useFaucetMutation'; import { useFaucetRateLimiter } from './useFaucetRateLimiter'; import { Button, ButtonType } from '@iota/apps-ui-kit'; -import FaucetMessageInfo from './FaucetMessageInfo'; +import { FaucetMessageInfo } from './FaucetMessageInfo'; -function FaucetRequestButton(): JSX.Element | null { +export function FaucetRequestButton(): JSX.Element | null { const network = useAppSelector(({ app }) => app.network); const customRpc = useAppSelector(({ app }) => app.customRpc); const networkConfig = customRpc ? getCustomNetwork(customRpc) : getNetwork(network); @@ -50,5 +50,3 @@ function FaucetRequestButton(): JSX.Element | null { /> ) : null; } - -export default FaucetRequestButton; diff --git a/apps/wallet/src/ui/app/shared/faucet/useFaucetMutation.ts b/apps/wallet/src/ui/app/shared/faucet/useFaucetMutation.ts index 527c780e200..dca924bb7cc 100644 --- a/apps/wallet/src/ui/app/shared/faucet/useFaucetMutation.ts +++ b/apps/wallet/src/ui/app/shared/faucet/useFaucetMutation.ts @@ -4,8 +4,7 @@ import { getFaucetRequestStatus, requestIotaFromFaucetV1 } from '@iota/iota-sdk/faucet'; import { useIsMutating, useMutation, type UseMutationOptions } from '@tanstack/react-query'; - -import { useActiveAccount } from '../../hooks/useActiveAccount'; +import { useActiveAccount } from '_hooks'; type UseFaucetMutationOptions = Pick<UseMutationOptions, 'onError'> & { host?: string; diff --git a/apps/wallet/src/ui/app/shared/page-main-layout/PageMainLayout.tsx b/apps/wallet/src/ui/app/shared/page-main-layout/PageMainLayout.tsx index 9896be915fb..07944a44a57 100644 --- a/apps/wallet/src/ui/app/shared/page-main-layout/PageMainLayout.tsx +++ b/apps/wallet/src/ui/app/shared/page-main-layout/PageMainLayout.tsx @@ -5,16 +5,15 @@ import { ErrorBoundary, MenuContent, Navigation, WalletSettingsButton } from '_components'; import cn from 'clsx'; import { createContext, type ReactNode, useState } from 'react'; -import { useAppSelector } from '../../hooks'; -import { AppType } from '../../redux/slices/app/AppType'; -import DappStatus from '../dapp-status'; +import { useAppSelector, useActiveAccount } from '_hooks'; +import { AppType } from '../../redux/slices/app/appType'; +import { DappStatus } from '../dapp-status'; import { Header } from '../header/Header'; import { Toaster } from '../toaster'; -import { IotaLogoMark, Ledger } from '@iota/ui-icons'; -import { useActiveAccount } from '../../hooks/useActiveAccount'; +import { IotaLogoMark, Ledger } from '@iota/apps-ui-icons'; import { Link } from 'react-router-dom'; -import { isLedgerAccountSerializedUI } from '_src/background/accounts/LedgerAccount'; -import { type SerializedUIAccount } from '_src/background/accounts/Account'; +import { isLedgerAccountSerializedUI } from '_src/background/accounts/ledgerAccount'; +import { type SerializedUIAccount } from '_src/background/accounts/account'; import { formatAccountName } from '../../helpers'; export const PageMainLayoutContext = createContext<HTMLDivElement | null>(null); diff --git a/apps/wallet/src/ui/app/staking/delegation-detail/DelegationDetailCard.tsx b/apps/wallet/src/ui/app/staking/delegation-detail/DelegationDetailCard.tsx index d55abd8b0b7..4ed1e6df5fa 100644 --- a/apps/wallet/src/ui/app/staking/delegation-detail/DelegationDetailCard.tsx +++ b/apps/wallet/src/ui/app/staking/delegation-detail/DelegationDetailCard.tsx @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { useAppSelector } from '_hooks'; +import { useAppSelector, useActiveAddress } from '_hooks'; import { ampli } from '_src/shared/analytics/ampli'; import { useBalance, @@ -14,13 +14,13 @@ import { useFormatCoin, formatPercentageDisplay, MIN_NUMBER_IOTA_TO_STAKE, + Validator, } from '@iota/core'; import { useIotaClientQuery } from '@iota/dapp-kit'; import { Network, type StakeObject } from '@iota/iota-sdk/client'; import { NANOS_PER_IOTA, IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import BigNumber from 'bignumber.js'; import { useMemo } from 'react'; -import { useActiveAddress } from '../../hooks/useActiveAddress'; import { getDelegationDataByStakeId } from '../getDelegationByStakeId'; import { CardType, @@ -35,8 +35,8 @@ import { LoadingIndicator, } from '@iota/apps-ui-kit'; import { useNavigate } from 'react-router-dom'; -import { ValidatorLogo } from '../validators/ValidatorLogo'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; +import toast from 'react-hot-toast'; interface DelegationDetailCardProps { validatorAddress: string; @@ -122,17 +122,7 @@ export function DelegationDetailCard({ validatorAddress, stakedId }: DelegationD } if (isError || errorValidators) { - return ( - <div className="mb-2 flex h-full w-full items-center justify-center p-2"> - <InfoBox - title="Something went wrong" - supportingText={error?.message ?? 'An error occurred'} - style={InfoBoxStyle.Default} - type={InfoBoxType.Error} - icon={<Warning />} - /> - </div> - ); + toast.error(error?.message ?? 'An error occurred fetching validator information'); } function handleAddNewStake() { navigate(stakeByValidatorAddress); @@ -153,7 +143,7 @@ export function DelegationDetailCard({ validatorAddress, stakedId }: DelegationD return ( <div className="flex h-full w-full flex-col justify-between"> <div className="flex flex-col gap-y-md"> - <ValidatorLogo validatorAddress={validatorAddress} type={CardType.Filled} /> + <Validator address={validatorAddress} type={CardType.Filled} /> {hasInactiveValidatorDelegation ? ( <InfoBox type={InfoBoxType.Error} diff --git a/apps/wallet/src/ui/app/staking/delegation-detail/index.tsx b/apps/wallet/src/ui/app/staking/delegation-detail/index.tsx index eb0854b601f..c24a35ffc66 100644 --- a/apps/wallet/src/ui/app/staking/delegation-detail/index.tsx +++ b/apps/wallet/src/ui/app/staking/delegation-detail/index.tsx @@ -5,7 +5,7 @@ import { Overlay } from '_components'; import { useGetDelegatedStake } from '@iota/core'; import { Navigate, useNavigate, useSearchParams } from 'react-router-dom'; -import { useActiveAddress } from '../../hooks/useActiveAddress'; +import { useActiveAddress } from '_hooks'; import { DelegationDetailCard } from './DelegationDetailCard'; import { LoadingIndicator } from '@iota/apps-ui-kit'; diff --git a/apps/wallet/src/ui/app/staking/home/index.tsx b/apps/wallet/src/ui/app/staking/home/index.tsx index 1dd9407de84..10add7f16c3 100644 --- a/apps/wallet/src/ui/app/staking/home/index.tsx +++ b/apps/wallet/src/ui/app/staking/home/index.tsx @@ -3,10 +3,9 @@ // SPDX-License-Identifier: Apache-2.0 import { Route, Routes } from 'react-router-dom'; - -import { useUnlockedGuard } from '../../hooks/useUnlockedGuard'; +import { useUnlockedGuard } from '_hooks'; import { DelegationDetail } from '../delegation-detail'; -import StakePage from '../stake'; +import { StakePage } from '../stake'; import { Validators } from '../validators'; export function StakingPage() { diff --git a/apps/wallet/src/ui/app/staking/stake/StakeForm.tsx b/apps/wallet/src/ui/app/staking/stake/StakeForm.tsx index 093991e6f33..59045e9a01e 100644 --- a/apps/wallet/src/ui/app/staking/stake/StakeForm.tsx +++ b/apps/wallet/src/ui/app/staking/stake/StakeForm.tsx @@ -14,11 +14,11 @@ import { } from '@iota/core'; import { Field, type FieldProps, Form, useFormikContext } from 'formik'; import { memo, useEffect, useMemo } from 'react'; -import { useActiveAddress, useTransactionDryRun } from '../../hooks'; +import { useActiveAddress, useTransactionDryRun } from '_hooks'; import { type FormValues } from './StakingCard'; import { InfoBox, InfoBoxStyle, InfoBoxType, Input, InputType } from '@iota/apps-ui-kit'; import { Transaction } from '@iota/iota-sdk/transactions'; -import { Exclamation } from '@iota/ui-icons'; +import { Exclamation } from '@iota/apps-ui-icons'; import { ExplorerLinkHelper } from '../../components'; export interface StakeFromProps { @@ -28,7 +28,12 @@ export interface StakeFromProps { epoch?: string | number; } -function StakeForm({ validatorAddress, coinBalance, coinType, epoch }: StakeFromProps) { +export function StakeFormComponent({ + validatorAddress, + coinBalance, + coinType, + epoch, +}: StakeFromProps) { const { values, setFieldValue } = useFormikContext<FormValues>(); const activeAddress = useActiveAddress(); const { data: metadata } = useCoinMetadata(coinType); @@ -117,4 +122,4 @@ function StakeForm({ validatorAddress, coinBalance, coinType, epoch }: StakeFrom ); } -export default memo(StakeForm); +export const StakeForm = memo(StakeFormComponent); diff --git a/apps/wallet/src/ui/app/staking/stake/StakingCard.tsx b/apps/wallet/src/ui/app/staking/stake/StakingCard.tsx index a4f94a5de47..0a3510a5ba7 100644 --- a/apps/wallet/src/ui/app/staking/stake/StakingCard.tsx +++ b/apps/wallet/src/ui/app/staking/stake/StakingCard.tsx @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 import { Loading } from '_components'; -import { Coin } from '_redux/slices/iota-objects/Coin'; +import { Coin } from '_src/ui/app/redux/slices/iota-objects/coin'; import { ampli } from '_src/shared/analytics/ampli'; import { createStakeTransaction, @@ -17,6 +17,7 @@ import { getStakeIotaByIotaId, createValidationSchema, MIN_NUMBER_IOTA_TO_STAKE, + Validator, } from '@iota/core'; import { useIotaClientQuery } from '@iota/dapp-kit'; import type { StakeObject } from '@iota/iota-sdk/client'; @@ -29,10 +30,9 @@ import { useCallback, useMemo } from 'react'; import { toast } from 'react-hot-toast'; import { Navigate, useNavigate, useSearchParams } from 'react-router-dom'; import { getSignerOperationErrorMessage } from '../../helpers/errorMessages'; -import { useActiveAccount } from '../../hooks/useActiveAccount'; -import { useSigner } from '../../hooks/useSigner'; +import { useActiveAccount, useSigner } from '_hooks'; import { getDelegationDataByStakeId } from '../getDelegationByStakeId'; -import StakeForm from './StakeForm'; +import { StakeForm } from './StakeForm'; import { UnStakeForm } from './UnstakeForm'; import { ValidatorFormDetail } from './ValidatorFormDetail'; import { @@ -43,8 +43,7 @@ import { InfoBoxStyle, InfoBoxType, } from '@iota/apps-ui-kit'; -import { ValidatorLogo } from '../validators/ValidatorLogo'; -import { Info, Loader } from '@iota/ui-icons'; +import { Info, Loader } from '@iota/apps-ui-icons'; const INITIAL_VALUES = { amount: '', @@ -52,7 +51,7 @@ const INITIAL_VALUES = { export type FormValues = typeof INITIAL_VALUES; -function StakingCard() { +export function StakingCard() { const coinType = IOTA_TYPE_ARG; const activeAccount = useActiveAccount(); const accountAddress = activeAccount?.address; @@ -121,26 +120,33 @@ function StakingCard() { throw new Error('Failed, missing required field'); } - const sentryTransaction = Sentry.startTransaction({ - name: 'stake', - }); - try { - const transactionBlock = createStakeTransaction(amount, validatorAddress); - const tx = await signer.signAndExecuteTransaction({ - transactionBlock, - options: { - showInput: true, - showEffects: true, - showEvents: true, - }, - }); - await signer.client.waitForTransaction({ - digest: tx.digest, - }); - return tx; - } finally { - sentryTransaction.finish(); - } + return Sentry.startSpan( + { + name: 'stake', + }, + async (span) => { + try { + const transactionBlock = createStakeTransaction( + amount, + validatorAddress, + ); + const tx = await signer.signAndExecuteTransaction({ + transactionBlock, + options: { + showInput: true, + showEffects: true, + showEvents: true, + }, + }); + await signer.client.waitForTransaction({ + digest: tx.digest, + }); + return tx; + } finally { + span?.end(); + } + }, + ); }, onSuccess: (_, { amount, validatorAddress }) => { ampli.stakedIota({ @@ -157,26 +163,30 @@ function StakingCard() { throw new Error('Failed, missing required field.'); } - const sentryTransaction = Sentry.startTransaction({ - name: 'stake', - }); - try { - const transactionBlock = createUnstakeTransaction(stakedIotaId); - const tx = await signer.signAndExecuteTransaction({ - transactionBlock, - options: { - showInput: true, - showEffects: true, - showEvents: true, - }, - }); - await signer.client.waitForTransaction({ - digest: tx.digest, - }); - return tx; - } finally { - sentryTransaction.finish(); - } + return Sentry.startSpan( + { + name: 'stake', + }, + async (span) => { + try { + const transactionBlock = createUnstakeTransaction(stakedIotaId); + const tx = await signer.signAndExecuteTransaction({ + transactionBlock, + options: { + showInput: true, + showEffects: true, + showEvents: true, + }, + }); + await signer.client.waitForTransaction({ + digest: tx.digest, + }); + return tx; + } finally { + span?.end(); + } + }, + ); }, onSuccess: () => { ampli.unstakedIota({ @@ -285,10 +295,7 @@ function StakingCard() { {({ isSubmitting, isValid, submitForm }) => ( <> <div className="flex h-full flex-col gap-md overflow-auto"> - <ValidatorLogo - validatorAddress={validatorAddress} - type={CardType.Filled} - /> + <Validator address={validatorAddress} type={CardType.Filled} /> <ValidatorFormDetail validatorAddress={validatorAddress} unstake={unstake} @@ -347,5 +354,3 @@ function StakingCard() { </div> ); } - -export default StakingCard; diff --git a/apps/wallet/src/ui/app/staking/stake/UnstakeForm.tsx b/apps/wallet/src/ui/app/staking/stake/UnstakeForm.tsx index 51067f5f966..82ed60683b8 100644 --- a/apps/wallet/src/ui/app/staking/stake/UnstakeForm.tsx +++ b/apps/wallet/src/ui/app/staking/stake/UnstakeForm.tsx @@ -14,7 +14,7 @@ import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; import { Form } from 'formik'; import { useMemo } from 'react'; -import { useActiveAddress, useTransactionGasBudget } from '../../hooks'; +import { useActiveAddress, useTransactionGasBudget } from '_hooks'; import { Divider, KeyValueInfo, Panel } from '@iota/apps-ui-kit'; export interface StakeFromProps { diff --git a/apps/wallet/src/ui/app/staking/stake/ValidatorFormDetail.tsx b/apps/wallet/src/ui/app/staking/stake/ValidatorFormDetail.tsx index 5d3caf7314a..4772c32d487 100644 --- a/apps/wallet/src/ui/app/staking/stake/ValidatorFormDetail.tsx +++ b/apps/wallet/src/ui/app/staking/stake/ValidatorFormDetail.tsx @@ -4,7 +4,7 @@ import { formatPercentageDisplay, useGetStakingValidatorDetails } from '@iota/core'; import { useSearchParams } from 'react-router-dom'; -import { useActiveAddress } from '../../hooks/useActiveAddress'; +import { useActiveAddress } from '_hooks'; import { InfoBox, InfoBoxStyle, @@ -14,7 +14,7 @@ import { TooltipPosition, LoadingIndicator, } from '@iota/apps-ui-kit'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; interface ValidatorFormDetailProps { validatorAddress: string; diff --git a/apps/wallet/src/ui/app/staking/stake/index.tsx b/apps/wallet/src/ui/app/staking/stake/index.tsx index 4add0478403..60770068945 100644 --- a/apps/wallet/src/ui/app/staking/stake/index.tsx +++ b/apps/wallet/src/ui/app/staking/stake/index.tsx @@ -6,9 +6,9 @@ import { Overlay } from '_components'; import { useNavigate, useSearchParams } from 'react-router-dom'; import { SelectValidatorCard } from '../validators/SelectValidatorCard'; -import StakingCard from './StakingCard'; +import { StakingCard } from './StakingCard'; -function StakePage() { +export function StakePage() { const [searchParams] = useSearchParams(); const validatorAddress = searchParams.get('address'); const unstake = searchParams.get('unstake') === 'true'; @@ -27,5 +27,3 @@ function StakePage() { </Overlay> ); } - -export default StakePage; diff --git a/apps/wallet/src/ui/app/staking/validators/SelectValidatorCard.tsx b/apps/wallet/src/ui/app/staking/validators/SelectValidatorCard.tsx index 015463bc23d..ed14f2a16db 100644 --- a/apps/wallet/src/ui/app/staking/validators/SelectValidatorCard.tsx +++ b/apps/wallet/src/ui/app/staking/validators/SelectValidatorCard.tsx @@ -3,14 +3,13 @@ // SPDX-License-Identifier: Apache-2.0 import { ampli } from '_src/shared/analytics/ampli'; -import { calculateStakeShare, useGetValidatorsApy } from '@iota/core'; +import { calculateStakeShare, useGetValidatorsApy, Validator } from '@iota/core'; import { useIotaClientQuery } from '@iota/dapp-kit'; import cl from 'clsx'; import { useMemo, useState } from 'react'; import { Button, InfoBox, InfoBoxStyle, InfoBoxType, LoadingIndicator } from '@iota/apps-ui-kit'; import { useNavigate } from 'react-router-dom'; -import { ValidatorLogo } from './ValidatorLogo'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; type Validator = { name: string; @@ -96,9 +95,8 @@ export function SelectValidatorCard() { })} key={validator.address} > - <ValidatorLogo - validatorAddress={validator.address} - showApy + <Validator + address={validator.address} onClick={() => { selectValidator(validator); }} diff --git a/apps/wallet/src/ui/app/staking/validators/ValidatorLogo.tsx b/apps/wallet/src/ui/app/staking/validators/ValidatorLogo.tsx deleted file mode 100644 index bf8d05a97b6..00000000000 --- a/apps/wallet/src/ui/app/staking/validators/ValidatorLogo.tsx +++ /dev/null @@ -1,101 +0,0 @@ -// Copyright (c) Mysten Labs, Inc. -// Modifications Copyright (c) 2024 IOTA Stiftung -// SPDX-License-Identifier: Apache-2.0 -import { - Badge, - BadgeType, - Card, - CardAction, - CardActionType, - CardBody, - CardImage, - type CardType, -} from '@iota/apps-ui-kit'; -import { useIotaClientQuery } from '@iota/dapp-kit'; -import { formatAddress } from '@iota/iota-sdk/utils'; -import { useMemo } from 'react'; -import { formatPercentageDisplay, useGetValidatorsApy, ImageIcon, ImageIconSize } from '@iota/core'; - -interface ValidatorLogoProps { - validatorAddress: string; - type?: CardType; - showApy?: boolean; - showActiveStatus?: boolean; - activeEpoch?: string; - onClick?(): void; -} - -export function ValidatorLogo({ - validatorAddress, - type, - showApy, - showActiveStatus = false, - activeEpoch, - onClick, -}: ValidatorLogoProps) { - const { data: system, isPending } = useIotaClientQuery('getLatestIotaSystemState'); - const { data: rollingAverageApys } = useGetValidatorsApy(); - const { apy, isApyApproxZero } = rollingAverageApys?.[validatorAddress] ?? { - apy: null, - }; - - const validatorMeta = useMemo(() => { - if (!system) return null; - - return ( - system.activeValidators.find( - (validator) => validator.iotaAddress === validatorAddress, - ) || null - ); - }, [validatorAddress, system]); - - const stakingPoolActivationEpoch = Number(validatorMeta?.stakingPoolActivationEpoch || 0); - const currentEpoch = Number(system?.epoch || 0); - - // flag as new validator if the validator was activated in the last epoch - // for genesis validators, this will be false - const newValidator = currentEpoch - stakingPoolActivationEpoch <= 1 && currentEpoch !== 0; - - // flag if the validator is at risk of being removed from the active set - const isAtRisk = system?.atRiskValidators.some((item) => item[0] === validatorAddress); - - if (isPending) { - return <div className="flex items-center justify-center">...</div>; - } - // for inactive validators, show the epoch number - const fallBackText = activeEpoch - ? `Staked ${Number(system?.epoch) - Number(activeEpoch)} epochs ago` - : ''; - const validatorName = validatorMeta?.name || fallBackText; - - const subtitle = showActiveStatus ? ( - <div className="flex items-center gap-1"> - {formatAddress(validatorAddress)} - {newValidator && <Badge label="New" type={BadgeType.PrimarySoft} />} - {isAtRisk && <Badge label="At Risk" type={BadgeType.PrimarySolid} />} - </div> - ) : ( - formatAddress(validatorAddress) - ); - return ( - <> - <Card type={type} onClick={onClick}> - <CardImage> - <ImageIcon - src={validatorMeta?.imageUrl ?? null} - label={validatorName} - fallback={validatorName} - size={ImageIconSize.Large} - /> - </CardImage> - <CardBody title={validatorName} subtitle={subtitle} isTextTruncated /> - {showApy && ( - <CardAction - type={CardActionType.SupportingText} - title={formatPercentageDisplay(apy, '-', isApyApproxZero)} - /> - )} - </Card> - </> - ); -} diff --git a/apps/wallet/src/ui/app/staking/validators/ValidatorsCard.tsx b/apps/wallet/src/ui/app/staking/validators/ValidatorsCard.tsx index 606c51ec947..7c5516650ad 100644 --- a/apps/wallet/src/ui/app/staking/validators/ValidatorsCard.tsx +++ b/apps/wallet/src/ui/app/staking/validators/ValidatorsCard.tsx @@ -15,7 +15,7 @@ import { } from '@iota/core'; import { useIotaClientQuery } from '@iota/dapp-kit'; import { useMemo } from 'react'; -import { useActiveAddress } from '../../hooks/useActiveAddress'; +import { useActiveAddress } from '_hooks'; import { Title, TitleSize, @@ -28,7 +28,7 @@ import { DisplayStats, } from '@iota/apps-ui-kit'; import { useNavigate } from 'react-router-dom'; -import { Info, Warning } from '@iota/ui-icons'; +import { Info, Warning } from '@iota/apps-ui-icons'; import { IOTA_TYPE_ARG } from '@iota/iota-sdk/utils'; export function ValidatorsCard() { diff --git a/apps/wallet/src/ui/app/staking/validators/index.tsx b/apps/wallet/src/ui/app/staking/validators/index.tsx index a5ad53c1b61..e24784424ce 100644 --- a/apps/wallet/src/ui/app/staking/validators/index.tsx +++ b/apps/wallet/src/ui/app/staking/validators/index.tsx @@ -9,12 +9,11 @@ import { DELEGATED_STAKES_QUERY_STALE_TIME, } from '@iota/core'; import { useNavigate } from 'react-router-dom'; - -import { useActiveAddress } from '../../hooks/useActiveAddress'; +import { useActiveAddress } from '_hooks'; import { SelectValidatorCard } from './SelectValidatorCard'; import { ValidatorsCard } from './ValidatorsCard'; import { InfoBox, InfoBoxType, InfoBoxStyle } from '@iota/apps-ui-kit'; -import { Warning } from '@iota/ui-icons'; +import { Warning } from '@iota/apps-ui-icons'; export function Validators() { const accountAddress = useActiveAddress(); diff --git a/apps/wallet/src/ui/app/WalletSigner.ts b/apps/wallet/src/ui/app/walletSigner.ts similarity index 100% rename from apps/wallet/src/ui/app/WalletSigner.ts rename to apps/wallet/src/ui/app/walletSigner.ts diff --git a/apps/wallet/src/ui/assets/images/balance_finder_intro.svg b/apps/wallet/src/ui/assets/images/balance_finder_intro.svg new file mode 100644 index 00000000000..15d0910299f --- /dev/null +++ b/apps/wallet/src/ui/assets/images/balance_finder_intro.svg @@ -0,0 +1,9 @@ +<svg width="360" height="270" viewBox="0 0 360 270" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> +<path d="M360 0H0V270H360V0Z" fill="url(#pattern0_1_8)"/> +<defs> +<pattern id="pattern0_1_8" patternContentUnits="objectBoundingBox" width="1" height="1"> +<use xlink:href="#image0_1_8" transform="scale(0.000706215 0.00094162)"/> +</pattern> +<image id="image0_1_8" width="1416" height="1062" xlink:href=""/> +</defs> +</svg> diff --git a/apps/wallet/src/ui/assets/images/balance_finder_intro_darkmode.svg b/apps/wallet/src/ui/assets/images/balance_finder_intro_darkmode.svg new file mode 100644 index 00000000000..606b35b3640 --- /dev/null +++ b/apps/wallet/src/ui/assets/images/balance_finder_intro_darkmode.svg @@ -0,0 +1,9 @@ +<svg width="360" height="270" viewBox="0 0 360 270" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> +<path d="M0 0H360V270H0V0Z" fill="url(#pattern0_7485_6388)"/> +<defs> +<pattern id="pattern0_7485_6388" patternContentUnits="objectBoundingBox" width="1" height="1"> +<use xlink:href="#image0_7485_6388" transform="scale(0.000706215 0.00094162)"/> +</pattern> +<image id="image0_7485_6388" width="1416" height="1062" xlink:href=""/> +</defs> +</svg> diff --git a/apps/wallet/src/ui/index.tsx b/apps/wallet/src/ui/index.tsx index f3ef312b1c3..d3328c97491 100644 --- a/apps/wallet/src/ui/index.tsx +++ b/apps/wallet/src/ui/index.tsx @@ -4,15 +4,14 @@ import '@fontsource-variable/inter'; import '@fontsource-variable/red-hat-mono'; - import { ErrorBoundary } from '_components'; import { initAppType } from '_redux/slices/app'; -import { AppType, getFromLocationSearch } from '_redux/slices/app/AppType'; +import { AppType, getFromLocationSearch } from '_src/ui/app/redux/slices/app/appType'; import { initAmplitude } from '_src/shared/analytics/amplitude'; import { setAttributes } from '_src/shared/experimentation/features'; -import initSentry from '_src/ui/app/helpers/sentry'; +import { initSentry } from '_src/ui/app/helpers'; import store from '_store'; -import { thunkExtras } from '_store/thunk-extras'; +import { thunkExtras } from '_src/ui/app/redux/store/thunkExtras'; import { KioskClientProvider, ThemeProvider } from '@iota/core'; import { GrowthBookProvider } from '@growthbook/growthbook-react'; import { IotaClientProvider } from '@iota/dapp-kit'; @@ -22,15 +21,14 @@ import { Fragment, StrictMode } from 'react'; import { createRoot } from 'react-dom/client'; import { Provider } from 'react-redux'; import { HashRouter } from 'react-router-dom'; - -import App from './app'; -import { walletApiProvider } from './app/ApiProvider'; +import { App } from './app'; +import { walletApiProvider } from './app/apiProvider'; import { AccountsFormProvider } from './app/components/accounts/AccountsFormContext'; import { UnlockAccountProvider } from './app/components/accounts/UnlockAccountContext'; import { IotaLedgerClientProvider } from './app/components/ledger/IotaLedgerClientProvider'; -import { growthbook } from './app/experimentation/feature-gating'; +import { growthbook } from './app/experimentation/featureGating'; import { persister, queryClient } from './app/helpers/queryClient'; -import { useAppSelector } from './app/hooks'; +import { useAppSelector } from '_hooks'; import './styles/global.scss'; import 'bootstrap-icons/font/bootstrap-icons.scss'; diff --git a/apps/wallet/tailwind.config.ts b/apps/wallet/tailwind.config.ts index 088b09b22c6..e837e56436f 100644 --- a/apps/wallet/tailwind.config.ts +++ b/apps/wallet/tailwind.config.ts @@ -5,7 +5,7 @@ import { type Config } from 'tailwindcss'; import animatePlugin from 'tailwindcss-animate'; // Note: exception for the tailwind preset import -import uiKitStaticPreset from '../../apps/ui-kit/src/lib/tailwind/static.preset'; +import uiKitStaticPreset from '../../apps/ui-kit/src/lib/tailwind/static.presets'; export default { presets: [uiKitStaticPreset], diff --git a/apps/wallet/tests/balanceChanges.spec.ts b/apps/wallet/tests/balance-changes.spec.ts similarity index 100% rename from apps/wallet/tests/balanceChanges.spec.ts rename to apps/wallet/tests/balance-changes.spec.ts diff --git a/apps/wallet/tests/demo-app/src/index.tsx b/apps/wallet/tests/demo-app/src/index.tsx index 8ee934abd37..09f32b3a042 100644 --- a/apps/wallet/tests/demo-app/src/index.tsx +++ b/apps/wallet/tests/demo-app/src/index.tsx @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -import { type IotaWallet } from '_src/dapp-interface/WalletStandardInterface'; +import { type IotaWallet } from '_src/dapp-interface/walletStandardInterface'; import { Transaction } from '@iota/iota-sdk/transactions'; import { getWallets, ReadonlyWalletAccount, type Wallet } from '@iota/wallet-standard'; import { useEffect, useState } from 'react'; diff --git a/apps/wallet/tests/sites-to-cs-api.spec.ts b/apps/wallet/tests/sites-to-cs-api.spec.ts index 3ed5774e19f..b4e2ec028e5 100644 --- a/apps/wallet/tests/sites-to-cs-api.spec.ts +++ b/apps/wallet/tests/sites-to-cs-api.spec.ts @@ -5,7 +5,7 @@ import { type Page } from '@playwright/test'; import { expect, test } from './fixtures'; import { createWallet } from './utils/auth'; -import { demoDappConnect } from './utils/dapp-connect'; +import { demoDappConnect } from './utils/dappConnect'; import dotenv from 'dotenv'; dotenv.config(); diff --git a/apps/wallet/tests/sites-to-cs-messaging.spec.ts b/apps/wallet/tests/sites-to-cs-messaging.spec.ts index aac667175db..eca803c5263 100644 --- a/apps/wallet/tests/sites-to-cs-messaging.spec.ts +++ b/apps/wallet/tests/sites-to-cs-messaging.spec.ts @@ -5,7 +5,7 @@ import { type Page } from '@playwright/test'; import { expect, test } from './fixtures'; import { createWallet } from './utils/auth'; -import { demoDappConnect } from './utils/dapp-connect'; +import { demoDappConnect } from './utils/dappConnect'; import { generateWalletMessageStreamIdentifiers } from '../src/shared/utils/generateWalletMessageStreamIdentifiers'; import dotenv from 'dotenv'; diff --git a/apps/wallet/tests/utils/dapp-connect.ts b/apps/wallet/tests/utils/dappConnect.ts similarity index 100% rename from apps/wallet/tests/utils/dapp-connect.ts rename to apps/wallet/tests/utils/dappConnect.ts diff --git a/bridge/evm/contracts/BridgeConfig.sol b/bridge/evm/contracts/BridgeConfig.sol index d9a9069976b..4a10ee5e3c5 100644 --- a/bridge/evm/contracts/BridgeConfig.sol +++ b/bridge/evm/contracts/BridgeConfig.sol @@ -8,7 +8,7 @@ import "./interfaces/IBridgeConfig.sol"; /// @title BridgeConfig /// @notice This contract manages a registry of supported tokens and supported chain IDs for the IotaBridge. -/// It also provides functions to convert token amounts to Iota decimal adjusted amounts and vice versa. +/// It also provides functions to convert token amounts to IOTA decimal adjusted amounts and vice versa. contract BridgeConfig is IBridgeConfig, CommitteeUpgradeable { /* ========== STATE VARIABLES ========== */ @@ -166,11 +166,11 @@ contract BridgeConfig is IBridgeConfig, CommitteeUpgradeable { bool native ) private { require(tokenAddress != address(0), "BridgeConfig: Invalid token address"); - require(iotaDecimal > 0, "BridgeConfig: Invalid Iota decimal"); + require(iotaDecimal > 0, "BridgeConfig: Invalid IOTA decimal"); require(tokenPrice > 0, "BridgeConfig: Invalid token price"); uint8 erc20Decimals = IERC20Metadata(tokenAddress).decimals(); - require(erc20Decimals >= iotaDecimal, "BridgeConfig: Invalid Iota decimal"); + require(erc20Decimals >= iotaDecimal, "BridgeConfig: Invalid IOTA decimal"); supportedTokens[tokenID] = Token(tokenAddress, iotaDecimal, native); tokenPrices[tokenID] = tokenPrice; diff --git a/bridge/evm/contracts/IotaBridge.sol b/bridge/evm/contracts/IotaBridge.sol index cd4362d0765..dc1fe6351e0 100644 --- a/bridge/evm/contracts/IotaBridge.sol +++ b/bridge/evm/contracts/IotaBridge.sol @@ -16,7 +16,7 @@ import "./interfaces/IWETH9.sol"; /// @title IotaBridge /// @notice This contract implements a token bridge that enables users to deposit and withdraw /// supported tokens to and from other chains. The bridge supports the transfer of Ethereum and ERC20 -/// tokens. Bridge operations are managed by a committee of Iota validators that are responsible +/// tokens. Bridge operations are managed by a committee of IOTA validators that are responsible /// for verifying and processing bridge messages. The bridge is designed to be upgradeable and /// can be paused in case of an emergency. The bridge also enforces limits on the amount of /// assets that can be withdrawn to prevent abuse. @@ -131,7 +131,7 @@ contract IotaBridge is IIotaBridge, CommitteeUpgradeable, PausableUpgradeable { /// have approved this contract to transfer the given token. /// @param tokenID The ID of the token to be bridged. /// @param amount The amount of tokens to be bridged. - /// @param recipientAddress The address on the Iota chain where the tokens will be sent. + /// @param recipientAddress The address on the IOTA chain where the tokens will be sent. /// @param destinationChainID The ID of the destination chain. function bridgeERC20( uint8 tokenID, diff --git a/bridge/evm/contracts/interfaces/IIotaBridge.sol b/bridge/evm/contracts/interfaces/IIotaBridge.sol index f7f70a307f6..ad874d85ee7 100644 --- a/bridge/evm/contracts/interfaces/IIotaBridge.sol +++ b/bridge/evm/contracts/interfaces/IIotaBridge.sol @@ -3,14 +3,14 @@ pragma solidity ^0.8.20; /// @title IIotaBridge -/// @dev Interface for the Iota Bridge contract. +/// @dev Interface for the IOTA Bridge contract. interface IIotaBridge { /// @notice Emitted when tokens are deposited to be bridged. /// @param sourceChainID The ID of the source chain (this chain). /// @param nonce The nonce of the transaction on source chain. /// @param destinationChainID The ID of the destination chain. /// @param tokenID The code of the token. - /// @param iotaAdjustedAmount The amount of tokens to transfer, adjusted for Iota decimals. + /// @param iotaAdjustedAmount The amount of tokens to transfer, adjusted for IOTA decimals. /// @param senderAddress The address of the sender. /// @param recipientAddress The address of the sender. event TokensDeposited( diff --git a/bridge/evm/contracts/utils/BridgeUtils.sol b/bridge/evm/contracts/utils/BridgeUtils.sol index 2b094549f6b..9db8093cc02 100644 --- a/bridge/evm/contracts/utils/BridgeUtils.sol +++ b/bridge/evm/contracts/utils/BridgeUtils.sol @@ -3,7 +3,7 @@ pragma solidity ^0.8.20; /// @title BridgeUtils -/// @notice This library defines the message format and constants for the Iota native bridge. It also +/// @notice This library defines the message format and constants for the IOTA native bridge. It also /// provides functions to encode and decode bridge messages and their payloads. /// @dev This library only utilizes internal functions to enable upgradeability via the OpenZeppelin /// UUPS proxy pattern (external libraries are not supported). @@ -118,11 +118,11 @@ library BridgeUtils { } } - /// @notice Converts the provided token amount to the Iota decimal adjusted amount. + /// @notice Converts the provided token amount to the IOTA decimal adjusted amount. /// @param erc20Decimal The erc20 decimal value for the token. /// @param iotaDecimal The iota decimal value for the token. - /// @param amount The ERC20 amount of the tokens to convert to Iota. - /// @return Iota converted amount. + /// @param amount The ERC20 amount of the tokens to convert to IOTA. + /// @return IOTA converted amount. function convertERC20ToIotaDecimal(uint8 erc20Decimal, uint8 iotaDecimal, uint256 amount) internal pure @@ -136,7 +136,7 @@ library BridgeUtils { return uint64(amount); } - require(erc20Decimal > iotaDecimal, "BridgeUtils: Invalid Iota decimal"); + require(erc20Decimal > iotaDecimal, "BridgeUtils: Invalid IOTA decimal"); // Difference in decimal places uint256 factor = 10 ** (erc20Decimal - iotaDecimal); @@ -151,10 +151,10 @@ library BridgeUtils { return uint64(amount); } - /// @notice Converts the provided Iota decimal adjusted amount to the ERC20 token amount. + /// @notice Converts the provided IOTA decimal adjusted amount to the ERC20 token amount. /// @param erc20Decimal The erc20 decimal value for the token. /// @param iotaDecimal The iota decimal value for the token. - /// @param amount The Iota amount of the tokens to convert to ERC20. + /// @param amount The IOTA amount of the tokens to convert to ERC20. /// @return ERC20 converted amount. function convertIotaToERC20Decimal(uint8 erc20Decimal, uint8 iotaDecimal, uint64 amount) internal @@ -165,7 +165,7 @@ library BridgeUtils { return uint256(amount); } - require(erc20Decimal > iotaDecimal, "BridgeUtils: Invalid Iota decimal"); + require(erc20Decimal > iotaDecimal, "BridgeUtils: Invalid IOTA decimal"); // Difference in decimal places uint256 factor = 10 ** (erc20Decimal - iotaDecimal); @@ -176,7 +176,7 @@ library BridgeUtils { /// @dev The function will revert if the payload length is invalid. /// TokenTransfer payload is 64 bytes. /// byte 0 : sender address length - /// bytes 1-32 : sender address (as we only support Iota now, it has to be 32 bytes long) + /// bytes 1-32 : sender address (as we only support IOTA now, it has to be 32 bytes long) /// bytes 33 : target chain id /// byte 34 : target address length /// bytes 35-54 : target address @@ -195,7 +195,7 @@ library BridgeUtils { require( senderAddressLength == 32, - "BridgeUtils: Invalid sender address length, Iota address must be 32 bytes" + "BridgeUtils: Invalid sender address length, IOTA address must be 32 bytes" ); // used to offset already read bytes @@ -225,9 +225,9 @@ library BridgeUtils { // why `add(recipientAddressLength, offset)`? // At this point, offset = 35, recipientAddressLength = 20. `mload(add(payload, 55))` - // reads the next 32 bytes from bytes 23 in paylod, because the first 32 bytes + // reads the next 32 bytes from bytes 23 in payload, because the first 32 bytes // of payload stores its length. So in reality, bytes 23 - 54 is loaded. During - // casting to address (20 bytes), the least sigificiant bytes are retained, namely + // casting to address (20 bytes), the least significant bytes are retained, namely // `recipientAddress` is bytes 35-54 assembly { recipientAddress := mload(add(_payload, add(recipientAddressLength, offset))) @@ -245,9 +245,9 @@ library BridgeUtils { // Why `add(amountLength, offset)`? // At this point, offset = 56, amountLength = 8. `mload(add(payload, 64))` - // reads the next 32 bytes from bytes 32 in paylod, because the first 32 bytes + // reads the next 32 bytes from bytes 32 in payload, because the first 32 bytes // of payload stores its length. So in reality, bytes 32 - 63 is loaded. During - // casting to uint64 (8 bytes), the least sigificiant bytes are retained, namely + // casting to uint64 (8 bytes), the least significant bytes are retained, namely // `recipientAddress` is bytes 56-63 assembly { amount := mload(add(_payload, add(amountLength, offset))) @@ -391,7 +391,7 @@ library BridgeUtils { /// @return native whether the token is native to the chain. /// @return tokenIDs the token ID to be added. /// @return tokenAddresses the address of the token to be added. - /// @return iotaDecimals the Iota decimal places of the tokens to be added. + /// @return iotaDecimals the IOTA decimal places of the tokens to be added. /// @return tokenPrices the price of the tokens to be added. function decodeAddTokensPayload(bytes memory _payload) internal diff --git a/bridge/evm/script/deploy_bridge.s.sol b/bridge/evm/script/deploy_bridge.s.sol index 5f15f006d00..5397073c3e8 100644 --- a/bridge/evm/script/deploy_bridge.s.sol +++ b/bridge/evm/script/deploy_bridge.s.sol @@ -68,7 +68,7 @@ contract DeployBridge is Script { } // deploy bridge config - // price of Iota (id = 0) should not be included in tokenPrices + // price of IOTA (id = 0) should not be included in tokenPrices require( deployConfig.supportedTokens.length == deployConfig.tokenPrices.length, "supportedTokens.length + 1 != tokenPrices.length" @@ -149,7 +149,7 @@ contract DeployBridge is Script { uint8[] memory _destinationChains = new uint8[](1); _destinationChains[0] = 1; - // deploy Iota Bridge ======================================================================== + // deploy IOTA Bridge ======================================================================== address iotaBridge = Upgrades.deployUUPSProxy( "IotaBridge.sol", diff --git a/bridge/evm/test/BridgeConfigTest.t.sol b/bridge/evm/test/BridgeConfigTest.t.sol index 32eceea61fb..f87d64c5c76 100644 --- a/bridge/evm/test/BridgeConfigTest.t.sol +++ b/bridge/evm/test/BridgeConfigTest.t.sol @@ -195,7 +195,7 @@ contract BridgeConfigTest is BridgeBaseTest { signatures[3] = getSignature(messageHash, committeeMemberPkD); // add token should fail because the iota decimal is greater than the eth decimal - vm.expectRevert(bytes("BridgeConfig: Invalid Iota decimal")); + vm.expectRevert(bytes("BridgeConfig: Invalid IOTA decimal")); config.addTokensWithSignatures(signatures, message); } diff --git a/bridge/evm/test/BridgeUtilsTest.t.sol b/bridge/evm/test/BridgeUtilsTest.t.sol index c9e93074d32..8a16fd71d46 100644 --- a/bridge/evm/test/BridgeUtilsTest.t.sol +++ b/bridge/evm/test/BridgeUtilsTest.t.sol @@ -17,12 +17,12 @@ contract BridgeUtilsTest is BridgeBaseTest { } function testConvertERC20ToIotaDecimalInvalidIotaDecimal() public { - vm.expectRevert(bytes("BridgeUtils: Invalid Iota decimal")); + vm.expectRevert(bytes("BridgeUtils: Invalid IOTA decimal")); BridgeUtils.convertERC20ToIotaDecimal(10, 11, 100); } function testconvertIotaToERC20DecimalInvalidIotaDecimal() public { - vm.expectRevert(bytes("BridgeUtils: Invalid Iota decimal")); + vm.expectRevert(bytes("BridgeUtils: Invalid IOTA decimal")); BridgeUtils.convertIotaToERC20Decimal(10, 11, 100); } diff --git a/bridge/evm/test/IotaBridgeTest.t.sol b/bridge/evm/test/IotaBridgeTest.t.sol index f21a8ff7969..11e26000e37 100644 --- a/bridge/evm/test/IotaBridgeTest.t.sol +++ b/bridge/evm/test/IotaBridgeTest.t.sol @@ -103,7 +103,7 @@ contract IotaBridgeTest is BridgeBaseTest, IIotaBridge { recipientAddressLength: 0, recipientAddress: bridgerA, tokenID: BridgeUtils.ETH, - // This is Iota amount (eth decimal 8) + // This is IOTA amount (eth decimal 8) amount: 100_000_000 }); BridgeUtils.Message memory message = BridgeUtils.Message({ @@ -131,7 +131,7 @@ contract IotaBridgeTest is BridgeBaseTest, IIotaBridge { recipientAddressLength: 0, recipientAddress: bridgerA, tokenID: BridgeUtils.ETH, - // This is Iota amount (eth decimal 8) + // This is IOTA amount (eth decimal 8) amount: 100_000_000 }); BridgeUtils.Message memory message = BridgeUtils.Message({ diff --git a/chocolatey/iota.nuspec b/chocolatey/iota.nuspec index 173e11fe35d..c547358e932 100644 --- a/chocolatey/iota.nuspec +++ b/chocolatey/iota.nuspec @@ -6,17 +6,17 @@ enclosed in quotation marks, you should use an editor that supports UTF-8, not t <id>iota</id> <version>$version$</version> <owners>iota</owners> - <title>Main Iota Binary + Main IOTA Binary iota https://iota.org/ - https://github.com/iotaledger/iota/blob/main/LICENSE + https://github.com/iotaledger/iota/blob/develop/LICENSE https://assets-global.website-files.com/6425f546844727ce5fb9e5ab/643775f4a15c9a9e10426daa_Iota_Favicon_256.png https://github.com/iotaledger/iota/ https://github.com/iotaledger/iota/issues iota https://community.chocolatey.org/packages/iota.portable Run a local iota binary - Iota is the first internet-scale programmable blockchain platform + IOTA is the first internet-scale programmable blockchain platform https://github.com/iotaledger/iota/releases/tag/mainnet-v$version$ diff --git a/consensus/config/src/committee.rs b/consensus/config/src/committee.rs index 9b809a614cc..0793d9d8b8b 100644 --- a/consensus/config/src/committee.rs +++ b/consensus/config/src/committee.rs @@ -16,7 +16,7 @@ use crate::{AuthorityPublicKey, NetworkPublicKey, ProtocolPublicKey}; pub type Epoch = u64; /// Voting power of an authority, roughly proportional to the actual amount of -/// Iota staked by the authority. +/// IOTA staked by the authority. /// Total stake / voting power of all authorities should sum to 10,000. pub type Stake = u64; @@ -58,8 +58,8 @@ impl Committee { } } - /// ----------------------------------------------------------------------- - /// Accessors to Committee fields. + // ----------------------------------------------------------------------- + // Accessors to Committee fields. pub fn epoch(&self) -> Epoch { self.epoch @@ -92,8 +92,8 @@ impl Committee { .map(|(i, a)| (AuthorityIndex(i as u32), a)) } - /// ----------------------------------------------------------------------- - /// Helpers for Committee properties. + // ----------------------------------------------------------------------- + // Helpers for Committee properties. /// Returns true if the provided stake has reached quorum (2f+1). pub fn reached_quorum(&self, stake: Stake) -> bool { diff --git a/consensus/config/src/parameters.rs b/consensus/config/src/parameters.rs index 61d2b3eba7f..be777fbbd27 100644 --- a/consensus/config/src/parameters.rs +++ b/consensus/config/src/parameters.rs @@ -9,7 +9,7 @@ use serde::{Deserialize, Serialize}; /// Operational configurations of a consensus authority. /// /// All fields should tolerate inconsistencies among authorities, without -/// affecting safety of the protocol. Otherwise, they need to be part of Iota +/// affecting safety of the protocol. Otherwise, they need to be part of IOTA /// protocol config or epoch state on-chain. /// /// NOTE: fields with default values are specified in the serde default diff --git a/consensus/core/src/authority_node.rs b/consensus/core/src/authority_node.rs index 70f81c271fb..7c66745bc4e 100644 --- a/consensus/core/src/authority_node.rs +++ b/consensus/core/src/authority_node.rs @@ -11,7 +11,7 @@ use prometheus::Registry; use tracing::{info, warn}; use crate::{ - CommitConsumer, + CommitConsumer, CommitConsumerMonitor, authority_service::AuthorityService, block_manager::BlockManager, block_verifier::SignedBlockVerifier, @@ -33,7 +33,7 @@ use crate::{ transaction::{TransactionClient, TransactionConsumer, TransactionVerifier}, }; -/// ConsensusAuthority is used by Iota to manage the lifetime of AuthorityNode. +/// ConsensusAuthority is used by IOTA to manage the lifetime of AuthorityNode. /// It hides the details of the implementation from the caller, /// MysticetiManager. pub enum ConsensusAuthority { @@ -93,6 +93,12 @@ impl ConsensusAuthority { } } + pub async fn replay_complete(&self) { + match self { + Self::WithTonic(authority) => authority.replay_complete().await, + } + } + #[cfg(test)] fn context(&self) -> &Arc { match self { @@ -116,6 +122,8 @@ where start_time: Instant, transaction_client: Arc, synchronizer: Arc, + commit_consumer_monitor: Arc, + commit_syncer_handle: CommitSyncerHandle, leader_timeout_handle: LeaderTimeoutTaskHandle, core_thread_handle: CoreThreadHandle, @@ -187,6 +195,9 @@ where let store_path = context.parameters.db_path.as_path().to_str().unwrap(); let store = Arc::new(RocksDBStore::new(store_path)); let dag_state = Arc::new(RwLock::new(DagState::new(context.clone(), store.clone()))); + + let highest_known_commit_at_startup = dag_state.read().last_commit_index(); + let sync_last_known_own_block = boot_counter == 0 && dag_state.read().highest_accepted_round() == 0 && !context @@ -209,6 +220,8 @@ where )); let commit_consumer_monitor = commit_consumer.monitor(); + commit_consumer_monitor + .set_highest_observed_commit_at_startup(highest_known_commit_at_startup); let commit_observer = CommitObserver::new( context.clone(), commit_consumer, @@ -254,7 +267,7 @@ where context.clone(), core_dispatcher.clone(), commit_vote_monitor.clone(), - commit_consumer_monitor, + commit_consumer_monitor.clone(), network_client.clone(), block_verifier.clone(), dag_state.clone(), @@ -302,6 +315,7 @@ where transaction_client: Arc::new(tx_client), synchronizer, commit_syncer_handle, + commit_consumer_monitor, leader_timeout_handle, core_thread_handle, broadcaster, @@ -352,6 +366,10 @@ where pub(crate) fn transaction_client(&self) -> Arc { self.transaction_client.clone() } + + pub(crate) async fn replay_complete(&self) { + self.commit_consumer_monitor.replay_complete().await; + } } #[cfg(test)] diff --git a/consensus/core/src/block.rs b/consensus/core/src/block.rs index c08bcce1f46..fbe1ae2fb9c 100644 --- a/consensus/core/src/block.rs +++ b/consensus/core/src/block.rs @@ -34,7 +34,7 @@ pub(crate) const GENESIS_ROUND: Round = 0; /// Block proposal as epoch UNIX timestamp in milliseconds. pub type BlockTimestampMs = u64; -/// Iota transaction in serialised bytes +/// IOTA transaction in serialised bytes #[derive(Clone, Eq, PartialEq, Serialize, Deserialize, Default, Debug)] pub struct Transaction { data: Bytes, diff --git a/consensus/core/src/commit.rs b/consensus/core/src/commit.rs index fc2221a84b0..36735cfd771 100644 --- a/consensus/core/src/commit.rs +++ b/consensus/core/src/commit.rs @@ -308,7 +308,7 @@ pub struct CommittedSubDag { /// a index incremented by 1. pub commit_ref: CommitRef, /// Optional scores that are provided as part of the consensus output to - /// Iota that can then be used by Iota for future submission to + /// IOTA that can then be used by IOTA for future submission to /// consensus. pub reputation_scores_desc: Vec<(AuthorityIndex, u64)>, } diff --git a/consensus/core/src/commit_consumer.rs b/consensus/core/src/commit_consumer.rs index aca8df61bba..6147f7d88f8 100644 --- a/consensus/core/src/commit_consumer.rs +++ b/consensus/core/src/commit_consumer.rs @@ -2,9 +2,10 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -use std::sync::{Arc, atomic::AtomicU32}; +use std::sync::{Arc, RwLock}; use iota_metrics::monitored_mpsc::UnboundedSender; +use tokio::sync::watch; use crate::{CommitIndex, CommittedSubDag}; @@ -39,24 +40,65 @@ impl CommitConsumer { } pub struct CommitConsumerMonitor { - highest_handled_commit: AtomicU32, + // highest commit that has been handled by IOTA + highest_handled_commit: watch::Sender, + + // the highest commit found in local storage at startup + highest_observed_commit_at_startup: RwLock, } impl CommitConsumerMonitor { pub(crate) fn new(last_handled_commit: CommitIndex) -> Self { Self { - highest_handled_commit: AtomicU32::new(last_handled_commit), + highest_handled_commit: watch::Sender::new(last_handled_commit), + highest_observed_commit_at_startup: RwLock::new(0), } } pub(crate) fn highest_handled_commit(&self) -> CommitIndex { - self.highest_handled_commit - .load(std::sync::atomic::Ordering::Acquire) + *self.highest_handled_commit.borrow() } pub fn set_highest_handled_commit(&self, highest_handled_commit: CommitIndex) { self.highest_handled_commit - .store(highest_handled_commit, std::sync::atomic::Ordering::Release); + .send_replace(highest_handled_commit); + } + + pub fn highest_observed_commit_at_startup(&self) -> CommitIndex { + *self.highest_observed_commit_at_startup.read().unwrap() + } + + pub fn set_highest_observed_commit_at_startup( + &self, + highest_observed_commit_at_startup: CommitIndex, + ) { + let highest_handled_commit = self.highest_handled_commit(); + assert!( + highest_observed_commit_at_startup >= highest_handled_commit, + "we cannot have handled a commit that we do not know about: {} < {}", + highest_observed_commit_at_startup, + highest_handled_commit, + ); + + let mut commit = self.highest_observed_commit_at_startup.write().unwrap(); + + assert!( + *commit == 0, + "highest_known_commit_at_startup can only be set once" + ); + *commit = highest_observed_commit_at_startup; + } + + pub(crate) async fn replay_complete(&self) { + let highest_observed_commit_at_startup = self.highest_observed_commit_at_startup(); + let mut rx = self.highest_handled_commit.subscribe(); + loop { + let highest_handled = *rx.borrow_and_update(); + if highest_handled >= highest_observed_commit_at_startup { + return; + } + rx.changed().await.unwrap(); + } } } diff --git a/consensus/core/src/commit_syncer.rs b/consensus/core/src/commit_syncer.rs index ef6ce101cb3..8dec50c2a83 100644 --- a/consensus/core/src/commit_syncer.rs +++ b/consensus/core/src/commit_syncer.rs @@ -15,7 +15,7 @@ //! when blocks are included in commits with >= 2f+1 certifiers by stake, these //! blocks must have passed verifications on some honest validators, so //! re-verifying them is unnecessary. In fact, the quorum certified commits -//! themselves can be trusted to be sent to Iota directly, but for simplicity +//! themselves can be trusted to be sent to IOTA directly, but for simplicity //! this is not done. Blocks from trusted commits still go through Core and //! committer. //! diff --git a/consensus/core/src/linearizer.rs b/consensus/core/src/linearizer.rs index 54f20e7f537..7ebfba9cc8d 100644 --- a/consensus/core/src/linearizer.rs +++ b/consensus/core/src/linearizer.rs @@ -158,7 +158,7 @@ impl Linearizer { committed_sub_dags.push(sub_dag); } - // Committed blocks must be persisted to storage before sending them to Iota and + // Committed blocks must be persisted to storage before sending them to IOTA and // executing their transactions. // Commit metadata can be persisted more lazily because they are recoverable. // Uncommitted blocks can wait to persist too. diff --git a/consensus/core/src/network/metrics_layer.rs b/consensus/core/src/network/metrics_layer.rs index 01c35cc31c5..c8d4d962102 100644 --- a/consensus/core/src/network/metrics_layer.rs +++ b/consensus/core/src/network/metrics_layer.rs @@ -2,15 +2,15 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 +//! Tower layer adapters that allow specifying callbacks for request and +//! response handling can be implemented for different networking stacks. + use std::sync::Arc; use prometheus::HistogramTimer; use super::metrics::NetworkRouteMetrics; -/// Tower layer adapters that allow specifying callbacks for request and -/// response handling can be implemented for different networking stacks. - pub(crate) trait SizedRequest { fn size(&self) -> usize; fn route(&self) -> String; diff --git a/consensus/core/src/test_dag_parser.rs b/consensus/core/src/test_dag_parser.rs index 486db4ef36e..f00c50a6f75 100644 --- a/consensus/core/src/test_dag_parser.rs +++ b/consensus/core/src/test_dag_parser.rs @@ -52,7 +52,6 @@ use crate::{ /// dag_builder.print(); // print the parsed DAG /// dag_builder.persist_all_blocks(dag_state.clone()); // persist all blocks to DagState /// ``` - pub(crate) fn parse_dag(dag_string: &str) -> IResult<&str, DagBuilder> { let (input, _) = tuple((tag("DAG"), multispace0, char('{')))(dag_string)?; diff --git a/consensus/core/src/transaction.rs b/consensus/core/src/transaction.rs index f22eeeba635..fb8dbb8c99f 100644 --- a/consensus/core/src/transaction.rs +++ b/consensus/core/src/transaction.rs @@ -226,7 +226,7 @@ impl TransactionClient { } } -/// `TransactionVerifier` implementation is supplied by Iota to validate +/// `TransactionVerifier` implementation is supplied by IOTA to validate /// transactions in a block, before acceptance of the block. pub trait TransactionVerifier: Send + Sync + 'static { /// Determines if this batch can be voted on diff --git a/crates/iota-adapter-transactional-tests/Cargo.toml b/crates/iota-adapter-transactional-tests/Cargo.toml index cbbf533c426..1112a3474de 100644 --- a/crates/iota-adapter-transactional-tests/Cargo.toml +++ b/crates/iota-adapter-transactional-tests/Cargo.toml @@ -5,7 +5,7 @@ authors = ["IOTA Foundation "] edition = "2021" license = "Apache-2.0" publish = false -description = "Transactional tests for Iota Adapter" +description = "Transactional tests for IOTA Adapter" [dev-dependencies] # external dependencies diff --git a/crates/iota-adapter-transactional-tests/tests/entry_points/generic_by_ref_invalid.exp b/crates/iota-adapter-transactional-tests/tests/entry_points/generic_by_ref_invalid.exp index 82813a5d23d..af712cd7028 100644 --- a/crates/iota-adapter-transactional-tests/tests/entry_points/generic_by_ref_invalid.exp +++ b/crates/iota-adapter-transactional-tests/tests/entry_points/generic_by_ref_invalid.exp @@ -5,20 +5,20 @@ A: object(0,0) task 1, lines 9-12: //# publish -Error: Transaction Effects Status: Iota Move Bytecode Verification Error. Please run the Iota Move Verifier for more information. +Error: Transaction Effects Status: IOTA Move Bytecode Verification Error. Please run the IOTA Move Verifier for more information. Execution Error: ExecutionError: ExecutionError { inner: ExecutionErrorInner { kind: IotaMoveVerificationError, source: Some("Invalid entry point parameter type. Expected primitive or object type. Got: &T0"), command: Some(0) } } task 2, lines 14-17: //# publish -Error: Transaction Effects Status: Iota Move Bytecode Verification Error. Please run the Iota Move Verifier for more information. +Error: Transaction Effects Status: IOTA Move Bytecode Verification Error. Please run the IOTA Move Verifier for more information. Execution Error: ExecutionError: ExecutionError { inner: ExecutionErrorInner { kind: IotaMoveVerificationError, source: Some("Invalid entry point parameter type. Expected primitive or object type. Got: &mut T0"), command: Some(0) } } task 3, lines 19-22: //# publish -Error: Transaction Effects Status: Iota Move Bytecode Verification Error. Please run the Iota Move Verifier for more information. +Error: Transaction Effects Status: IOTA Move Bytecode Verification Error. Please run the IOTA Move Verifier for more information. Execution Error: ExecutionError: ExecutionError { inner: ExecutionErrorInner { kind: IotaMoveVerificationError, source: Some("Invalid entry point parameter type. Expected primitive or object type. Got: &T0"), command: Some(0) } } task 4, lines 24-27: //# publish -Error: Transaction Effects Status: Iota Move Bytecode Verification Error. Please run the Iota Move Verifier for more information. +Error: Transaction Effects Status: IOTA Move Bytecode Verification Error. Please run the IOTA Move Verifier for more information. Execution Error: ExecutionError: ExecutionError { inner: ExecutionErrorInner { kind: IotaMoveVerificationError, source: Some("Invalid entry point parameter type. Expected primitive or object type. Got: &mut T0"), command: Some(0) } } diff --git a/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_only.exp b/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_only.exp index 10c7961e704..e676ae72cd3 100644 --- a/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_only.exp +++ b/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_only.exp @@ -2,7 +2,7 @@ processed 6 tasks task 1, lines 7-38: //# publish -Error: Transaction Effects Status: Iota Move Bytecode Verification Error. Please run the Iota Move Verifier for more information. +Error: Transaction Effects Status: IOTA Move Bytecode Verification Error. Please run the IOTA Move Verifier for more information. Execution Error: ExecutionError: ExecutionError { inner: ExecutionErrorInner { kind: IotaMoveVerificationError, source: Some("Enum X cannot have the 'key' ability. Enums cannot have the 'key' ability."), command: Some(0) } } task 2, line 40: diff --git a/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_only_uid_field.exp b/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_only_uid_field.exp index 1bf0ae5713c..3289f2895a1 100644 --- a/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_only_uid_field.exp +++ b/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_only_uid_field.exp @@ -2,7 +2,7 @@ processed 6 tasks task 1, lines 7-46: //# publish -Error: Transaction Effects Status: Iota Move Bytecode Verification Error. Please run the Iota Move Verifier for more information. +Error: Transaction Effects Status: IOTA Move Bytecode Verification Error. Please run the IOTA Move Verifier for more information. Execution Error: ExecutionError: ExecutionError { inner: ExecutionErrorInner { kind: IotaMoveVerificationError, source: Some("Enum X cannot have the 'key' ability. Enums cannot have the 'key' ability."), command: Some(0) } } task 2, line 48: diff --git a/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_store.exp b/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_store.exp index d35bae25d13..94035a4ef9d 100644 --- a/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_store.exp +++ b/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_store.exp @@ -2,7 +2,7 @@ processed 11 tasks task 1, lines 7-89: //# publish -Error: Transaction Effects Status: Iota Move Bytecode Verification Error. Please run the Iota Move Verifier for more information. +Error: Transaction Effects Status: IOTA Move Bytecode Verification Error. Please run the IOTA Move Verifier for more information. Execution Error: ExecutionError: ExecutionError { inner: ExecutionErrorInner { kind: IotaMoveVerificationError, source: Some("Enum X cannot have the 'key' ability. Enums cannot have the 'key' ability."), command: Some(0) } } task 2, line 91: diff --git a/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_store_uid_field.exp b/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_store_uid_field.exp index 01175813bf2..c8fc8b16efb 100644 --- a/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_store_uid_field.exp +++ b/crates/iota-adapter-transactional-tests/tests/enums/enum_with_key_store_uid_field.exp @@ -2,7 +2,7 @@ processed 11 tasks task 1, lines 7-105: //# publish -Error: Transaction Effects Status: Iota Move Bytecode Verification Error. Please run the Iota Move Verifier for more information. +Error: Transaction Effects Status: IOTA Move Bytecode Verification Error. Please run the IOTA Move Verifier for more information. Execution Error: ExecutionError: ExecutionError { inner: ExecutionErrorInner { kind: IotaMoveVerificationError, source: Some("Enum X cannot have the 'key' ability. Enums cannot have the 'key' ability."), command: Some(0) } } task 2, line 107: diff --git a/crates/iota-adapter-transactional-tests/tests/init/entry_new.exp b/crates/iota-adapter-transactional-tests/tests/init/entry_new.exp index 7a6336cfa87..e15ba170bd5 100644 --- a/crates/iota-adapter-transactional-tests/tests/init/entry_new.exp +++ b/crates/iota-adapter-transactional-tests/tests/init/entry_new.exp @@ -2,7 +2,7 @@ processed 3 tasks task 1, lines 9-18: //# publish -Error: Transaction Effects Status: Iota Move Bytecode Verification Error. Please run the Iota Move Verifier for more information. +Error: Transaction Effects Status: IOTA Move Bytecode Verification Error. Please run the IOTA Move Verifier for more information. Execution Error: ExecutionError: ExecutionError { inner: ExecutionErrorInner { kind: IotaMoveVerificationError, source: Some("_::m. 'init' cannot be 'entry'"), command: Some(0) } } task 2, line 19: diff --git a/crates/iota-adapter-transactional-tests/tests/publish/init_param.exp b/crates/iota-adapter-transactional-tests/tests/publish/init_param.exp index 3e7498ef22e..056f62ab500 100644 --- a/crates/iota-adapter-transactional-tests/tests/publish/init_param.exp +++ b/crates/iota-adapter-transactional-tests/tests/publish/init_param.exp @@ -2,5 +2,5 @@ processed 2 tasks task 1, lines 6-28: //# publish -Error: Transaction Effects Status: Iota Move Bytecode Verification Error. Please run the Iota Move Verifier for more information. +Error: Transaction Effects Status: IOTA Move Bytecode Verification Error. Please run the IOTA Move Verifier for more information. Execution Error: ExecutionError: ExecutionError { inner: ExecutionErrorInner { kind: IotaMoveVerificationError, source: Some("Expected last parameter for _::M1::init to be &mut iota::tx_context::TxContext or &iota::tx_context::TxContext, but found &mut iota::tx_context::TxContext"), command: Some(0) } } diff --git a/crates/iota-adapter-transactional-tests/tests/publish/init_public.exp b/crates/iota-adapter-transactional-tests/tests/publish/init_public.exp index bf37defa5ed..2d9e59a6aee 100644 --- a/crates/iota-adapter-transactional-tests/tests/publish/init_public.exp +++ b/crates/iota-adapter-transactional-tests/tests/publish/init_public.exp @@ -2,5 +2,5 @@ processed 2 tasks task 1, lines 6-28: //# publish -Error: Transaction Effects Status: Iota Move Bytecode Verification Error. Please run the Iota Move Verifier for more information. +Error: Transaction Effects Status: IOTA Move Bytecode Verification Error. Please run the IOTA Move Verifier for more information. Execution Error: ExecutionError: ExecutionError { inner: ExecutionErrorInner { kind: IotaMoveVerificationError, source: Some("_::M1. 'init' function must be private"), command: Some(0) } } diff --git a/crates/iota-adapter-transactional-tests/tests/publish/init_ret.exp b/crates/iota-adapter-transactional-tests/tests/publish/init_ret.exp index 76c2847072f..efea232dc2f 100644 --- a/crates/iota-adapter-transactional-tests/tests/publish/init_ret.exp +++ b/crates/iota-adapter-transactional-tests/tests/publish/init_ret.exp @@ -2,5 +2,5 @@ processed 2 tasks task 1, lines 6-28: //# publish -Error: Transaction Effects Status: Iota Move Bytecode Verification Error. Please run the Iota Move Verifier for more information. +Error: Transaction Effects Status: IOTA Move Bytecode Verification Error. Please run the IOTA Move Verifier for more information. Execution Error: ExecutionError: ExecutionError { inner: ExecutionErrorInner { kind: IotaMoveVerificationError, source: Some("_::M1, 'init' function cannot have return values"), command: Some(0) } } diff --git a/crates/iota-adapter-transactional-tests/tests/transfer_object/transfer_coin.move b/crates/iota-adapter-transactional-tests/tests/transfer_object/transfer_coin.move index 75b2057f9a8..3a5b6316760 100644 --- a/crates/iota-adapter-transactional-tests/tests/transfer_object/transfer_coin.move +++ b/crates/iota-adapter-transactional-tests/tests/transfer_object/transfer_coin.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -// tests TransferObject with a IOTA coin +// tests TransferObject with an IOTA coin //# init --accounts A B C diff --git a/crates/iota-adapter-transactional-tests/tests/upgrade/add_ability_during_upgrade.exp b/crates/iota-adapter-transactional-tests/tests/upgrade/add_ability_during_upgrade.exp index 95e29002e95..72f21648323 100644 --- a/crates/iota-adapter-transactional-tests/tests/upgrade/add_ability_during_upgrade.exp +++ b/crates/iota-adapter-transactional-tests/tests/upgrade/add_ability_during_upgrade.exp @@ -41,7 +41,7 @@ Execution Error: ExecutionError: ExecutionError { inner: ExecutionErrorInner { k task 8, lines 46-49: //# upgrade --package Test_V0 --upgrade-capability 1,1 --sender A -Error: Transaction Effects Status: Iota Move Bytecode Verification Error. Please run the Iota Move Verifier for more information. +Error: Transaction Effects Status: IOTA Move Bytecode Verification Error. Please run the IOTA Move Verifier for more information. Execution Error: ExecutionError: ExecutionError { inner: ExecutionErrorInner { kind: IotaMoveVerificationError, source: Some("First field of struct Foo must be 'id', dummy_field found"), command: Some(1) } } task 9, lines 51-57: diff --git a/crates/iota-analytics-indexer/README.md b/crates/iota-analytics-indexer/README.md new file mode 100644 index 00000000000..7018b1323fb --- /dev/null +++ b/crates/iota-analytics-indexer/README.md @@ -0,0 +1,119 @@ +# IOTA Analytics Indexer + +The IOTA Analytics Indexer is a service that exports data from the main IOTA network to a remote big object store (S3/GCS/Azure) for further analytical processing. It does not perform any analysis on its own. + +## **Key Features** + +- Exports data from the IOTA network to a remote big object store +- Provides BigQuery and Snowflake schemas for the exported data + +> [!NOTE] +> BigQuery and Snowflake are cloud-based data warehousing solutions. +> After getting data there one can analyse it in the cloud using SQL queries. +> +> BigQuery is part of Google Cloud Platform: [https://cloud.google.com/bigquery] +> +> Snowflake isn't part of any large cloud provider: [https://snowflake.com] + +## **Relation to iota-indexer** + +### iota-indexer + +Currently iota-indexer is computing and storing analytical metrics about: + +- network statistics (amount of transactions, transactions per second) +- (active) addresses (transactions senders/recipients) +- move calls + +Those metrics are computed by a separate analytical worker instance of the indexer, but it uses the main DB as the main indexer instance. + +It seems that some of the values stored in main indexer tables by iota-indexer's `fullnode_sync_worker` are only stored there for analytical purposes (move calls, tx recipients) and could potentially be excluded from further processing if it was not for analytical reasons. + +### iota-analytics-indexer + +The `iota-analytics-indexer` is not computing any analytical metrics directly. +It is only exporting data for further processing via external tools (BigQuery/SnowFlake). + +On this premise, the functionality in `iota-indexer` that is currently used for extracting analytics (and thus unrelated to the JSON-RPC/GraphQL service) could be moved out and delegated to another tool that processes data exported by `iota-analytics-indexer`. +Then the sync logic in `iota-indexer` could be simplified as well to store only data that is needed for the purposes of the RPC APIs. + +## **Schemas** + +The crate provides: + +- [BigQuery Schemas](src/store/bq/schemas/) +- [SnowFlake Schemas](src/store/snowflake/schemas/) +- [Rust struct representations](src/tables.rs) + +for the data that it is exporting. + +The tables covered by the schemas: + +- CHECKPOINT +- EVENT +- MOVE_CALL +- OBJECT +- MOVE_PACKAGE +- TRANSACTION_OBJECT - input and output objects for given transactions +- TRANSACTION + +> [!NOTE] +> The following rust structs currently do not have DB schemas prepared: +> +> - DynamicFieldEntry +> - WrappedObjectEntry + +## **Architecture** + +When running the indexer, one needs to specify object type that would be extracted from checkpoints and uploaded to the cloud. + +The following object types are supported: + +- Checkpoint +- Object +- Transaction +- TransactionObjects +- Event +- MoveCall +- MovePackage +- DynamicField +- WrappedObject + +Only one object type can be passed in given run, to process multiple object types it is needed to run multiple analytics indexer instances. + +In general, the data flow is as follows: + +- Checkpoints are read via JSON RPC using reused code from `iota_data_ingestion_core`. +- Checkpoints are processed by an appropriate handler (e.g. `EventHandler`), which extracts relevant objects from each transaction of the checkpoint. +- Objects are passed to the Writer, which writes the objects to a local temporary store in CSV or Parquet format. +- The `AnalyticsProcessor` syncs the objects from the local store to the remote store (S3/GCS/Azure, or also local, for testing purposes). +- Every 5 minutes the last processed checkpoint ID is fetched from BigQuery/Snowflake and reported as a metric. + +> [!NOTE] +> It is assumed that data from the big object store will be readable from BigQuery/Snowflake automatically, the indexer is not putting the data in BigQuery/Snowflake tables explicitly. + +Here is a graph summarizing the data flow: + +```mermaid +flowchart TD + FNODE["Fullnode/Indexer"] <-->|JSON RPC| CPREADER["`IndexerExecutor/CheckpointReader from the **iota_data_ingestion_core** package`"]; + subgraph "`**iota-analytics-indexer**`" + CPREADER -->|"`Executor calls **AnalyticsProcessor** for each checkpoint, which in turn passes the checkpoint to appropriate Handler`"| HANDLER["CheckpointHandler/EventHandler etc., depending on indexer configuration"] + HANDLER -->|"`**AnalyticsProcessor** reads object data extracted from the checkpoint by the Handler and passes it to the Writer`"| WRITER["CSVWriter/ParquetWriter"] + WRITER -->|Writes objects to temporary local storage| DISK[Temporary Local Storage] + DISK --> REMOTESYNC["`Task inside of **AnalyticsProcessor** that removes files from Local Storage and uploads them to Remote Storage(S3/GCS/Azure)`"] + WRITER -->|"`Once every few checkpoints, **AnalyticsProcessor** calls cut() to prepare file to be sent, FileMetadata is sent to the Remote Sync Task which triggers the sync`"| REMOTESYNC + REMOTESYNC -->|Some process outside of analytics indexer makes the newly uploaded data available via BigQuery/Snowflake tables| BQSF["BigQuery/Snowflake"] + BQSF -->|"Every 5 minutes max processed checkpoint number is read from the tables"| METRICS[Analytics Indexer Prometheus Metrics] + end + +linkStyle 6 stroke:red,stroke-width:2px,stroke-dasharray:3; +``` + +## **Metrics** + +The following Prometheus metrics are served by `iota-analytics-indexer` to monitor the indexer execution: + +- **total_received**: count of checkpoints processed in given run +- **last_uploaded_checkpoint**: id of last checkpoint uploaded to the big object store +- **max_checkpoint_on_store**: id of last checkpoint available via BigQuery/Snowflake tables diff --git a/crates/iota-analytics-indexer/src/lib.rs b/crates/iota-analytics-indexer/src/lib.rs index a2d4b25b894..a778e4da270 100644 --- a/crates/iota-analytics-indexer/src/lib.rs +++ b/crates/iota-analytics-indexer/src/lib.rs @@ -65,56 +65,56 @@ const DYNAMIC_FIELD_PREFIX: &str = "dynamic_field"; const WRAPPED_OBJECT_PREFIX: &str = "wrapped_object"; #[derive(Parser, Clone, Debug)] -#[clap( - name = "Iota Analytics Indexer", +#[command( + name = "IOTA Analytics Indexer", about = "Indexer service to upload data for the analytics pipeline.", rename_all = "kebab-case" )] pub struct AnalyticsIndexerConfig { /// The url of the checkpoint client to connect to. - #[clap(long)] + #[arg(long)] pub rest_url: String, /// The url of the metrics client to connect to. - #[clap(long, default_value = "127.0.0.1", global = true)] + #[arg(long, default_value = "127.0.0.1", global = true)] pub client_metric_host: String, /// The port of the metrics client to connect to. - #[clap(long, default_value = "8081", global = true)] + #[arg(long, default_value = "8081", global = true)] pub client_metric_port: u16, /// Directory to contain the temporary files for checkpoint entries. - #[clap(long, global = true, default_value = "/tmp")] + #[arg(long, global = true, default_value = "/tmp")] pub checkpoint_dir: PathBuf, /// Number of checkpoints to process before uploading to the datastore. - #[clap(long, default_value = "10000", global = true)] + #[arg(long, default_value = "10000", global = true)] pub checkpoint_interval: u64, /// Maximum file size in mb before uploading to the datastore. - #[clap(long, default_value = "100", global = true)] + #[arg(long, default_value = "100", global = true)] pub max_file_size_mb: u64, /// Checkpoint sequence number to start the download from - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub starting_checkpoint_seq_num: Option, /// Time to process in seconds before uploading to the datastore. - #[clap(long, default_value = "600", global = true)] + #[arg(long, default_value = "600", global = true)] pub time_interval_s: u64, // Remote object store where data gets written to #[command(flatten)] pub remote_store_config: ObjectStoreConfig, // Remote object store path prefix to use while writing - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub remote_store_path_prefix: Option, // File format to store data in i.e. csv, parquet, etc - #[clap(long, value_enum, default_value = "csv", global = true)] + #[arg(long, value_enum, default_value = "csv", global = true)] pub file_format: FileFormat, // Type of data to write i.e. checkpoint, object, transaction, etc - #[clap(long, value_enum, long, global = true)] + #[arg(long, value_enum, long, global = true)] pub file_type: FileType, - #[clap( + #[arg( long, - default_value = "https://checkpoints.mainnet.iota.io", + default_value = "https://checkpoints.mainnet.iota.cafe", global = true )] pub remote_store_url: String, // Directory to contain the package cache for pipelines - #[clap( + #[arg( long, value_enum, long, @@ -122,37 +122,37 @@ pub struct AnalyticsIndexerConfig { default_value = "/opt/iota/db/package_cache" )] pub package_cache_path: PathBuf, - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub bq_service_account_key_file: Option, - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub bq_project_id: Option, - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub bq_dataset_id: Option, - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub bq_table_id: Option, - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub bq_checkpoint_col_id: Option, - #[clap(long, global = true)] + #[arg(long, global = true)] pub report_bq_max_table_checkpoint: bool, - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub sf_account_identifier: Option, - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub sf_warehouse: Option, - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub sf_database: Option, - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub sf_schema: Option, - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub sf_username: Option, - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub sf_role: Option, - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub sf_password: Option, - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub sf_table_id: Option, - #[clap(long, default_value = None, global = true)] + #[arg(long, default_value = None, global = true)] pub sf_checkpoint_col_id: Option, - #[clap(long, global = true)] + #[arg(long, global = true)] pub report_sf_max_table_checkpoint: bool, } diff --git a/crates/iota-analytics-indexer/src/main.rs b/crates/iota-analytics-indexer/src/main.rs index 3e08ce16efc..281f6ade02e 100644 --- a/crates/iota-analytics-indexer/src/main.rs +++ b/crates/iota-analytics-indexer/src/main.rs @@ -42,7 +42,7 @@ async fn main() -> Result<()> { batch_size: 10, ..Default::default() }; - let (executor, exit_sender) = setup_single_workflow( + let (executor, token) = setup_single_workflow( processor, remote_store_url, watermark, @@ -51,13 +51,11 @@ async fn main() -> Result<()> { ) .await?; - tokio::spawn(async { + tokio::spawn(async move { signal::ctrl_c() .await .expect("Failed to install Ctrl+C handler"); - exit_sender - .send(()) - .expect("Failed to gracefully process shutdown"); + token.cancel(); }); executor.await?; Ok(()) diff --git a/crates/iota-aws-orchestrator/assets/plot.py b/crates/iota-aws-orchestrator/assets/plot.py index a30f6f08511..17817417b5f 100644 --- a/crates/iota-aws-orchestrator/assets/plot.py +++ b/crates/iota-aws-orchestrator/assets/plot.py @@ -419,7 +419,7 @@ def plot_duration(self, file, precision): if __name__ == "__main__": parser = argparse.ArgumentParser( - prog='Iota Plotter', + prog='IOTA Plotter', description='Simple script to plot measurement data' ) parser.add_argument( diff --git a/crates/iota-aws-orchestrator/src/main.rs b/crates/iota-aws-orchestrator/src/main.rs index 7e1696c3ed9..91ff63896a1 100644 --- a/crates/iota-aws-orchestrator/src/main.rs +++ b/crates/iota-aws-orchestrator/src/main.rs @@ -39,7 +39,7 @@ pub struct Opts { /// The path to the settings file. This file contains basic information to /// deploy testbeds and run benchmarks such as the url of the git repo, /// the commit to deploy, etc. - #[clap( + #[arg( long, value_name = "FILE", default_value = "crates/iota-aws-orchestrator/assets/settings.json", @@ -48,7 +48,7 @@ pub struct Opts { settings_path: String, /// The type of operation to run. - #[clap(subcommand)] + #[command(subcommand)] operation: Operation, } @@ -56,7 +56,7 @@ pub struct Opts { pub enum Operation { /// Get or modify the status of the testbed. Testbed { - #[clap(subcommand)] + #[command(subcommand)] action: TestbedAction, }, @@ -64,79 +64,79 @@ pub enum Operation { Benchmark { /// Percentage of shared vs owned objects; 0 means only owned objects /// and 100 means only shared objects. - #[clap(long, default_value = "0", global = true)] + #[arg(long, default_value = "0", global = true)] benchmark_type: String, /// The committee size to deploy. - #[clap(long, value_name = "INT")] + #[arg(long, value_name = "INT")] committee: usize, /// Number of faulty nodes. - #[clap(long, value_name = "INT", default_value = "0", global = true)] + #[arg(long, value_name = "INT", default_value = "0", global = true)] faults: usize, /// Whether the faulty nodes recover. - #[clap(long, action, default_value = "false", global = true)] + #[arg(long, action, default_value = "false", global = true)] crash_recovery: bool, /// The interval to crash nodes in seconds. - #[clap(long, value_parser = parse_duration, default_value = "60", global = true)] + #[arg(long, value_parser = parse_duration, default_value = "60", global = true)] crash_interval: Duration, /// The minimum duration of the benchmark in seconds. - #[clap(long, value_parser = parse_duration, default_value = "600", global = true)] + #[arg(long, value_parser = parse_duration, default_value = "600", global = true)] duration: Duration, /// The interval between measurements collection in seconds. - #[clap(long, value_parser = parse_duration, default_value = "15", global = true)] + #[arg(long, value_parser = parse_duration, default_value = "15", global = true)] scrape_interval: Duration, /// Whether to skip testbed updates before running benchmarks. - #[clap(long, action, default_value = "false", global = true)] + #[arg(long, action, default_value = "false", global = true)] skip_testbed_update: bool, /// Whether to skip testbed configuration before running benchmarks. - #[clap(long, action, default_value = "false", global = true)] + #[arg(long, action, default_value = "false", global = true)] skip_testbed_configuration: bool, /// Whether to download and analyze the client and node log files. - #[clap(long, action, default_value = "false", global = true)] + #[arg(long, action, default_value = "false", global = true)] log_processing: bool, /// The number of instances running exclusively load generators. If set /// to zero the orchestrator collocates one load generator with /// each node. - #[clap(long, value_name = "INT", default_value = "0", global = true)] + #[arg(long, value_name = "INT", default_value = "0", global = true)] dedicated_clients: usize, /// Whether to forgo a grafana and prometheus instance and leave the /// testbed unmonitored. - #[clap(long, action, default_value = "false", global = true)] + #[arg(long, action, default_value = "false", global = true)] skip_monitoring: bool, /// The timeout duration for ssh commands (in seconds). - #[clap(long, action, value_parser = parse_duration, default_value = "30", global = true)] + #[arg(long, action, value_parser = parse_duration, default_value = "30", global = true)] timeout: Duration, /// The number of times the orchestrator should retry an ssh command. - #[clap(long, value_name = "INT", default_value = "5", global = true)] + #[arg(long, value_name = "INT", default_value = "5", global = true)] retries: usize, /// The load to submit to the system. - #[clap(subcommand)] + #[command(subcommand)] load_type: Load, }, /// Print a summary of the specified measurements collection. Summarize { /// The path to the settings file. - #[clap(long, value_name = "FILE")] + #[arg(long, value_name = "FILE")] path: String, }, } #[derive(Parser)] -#[clap(rename_all = "kebab-case")] +#[command(rename_all = "kebab-case")] pub enum TestbedAction { /// Display the testbed status. Status, @@ -145,13 +145,13 @@ pub enum TestbedAction { /// the setting file. Deploy { /// Number of instances to deploy. - #[clap(long)] + #[arg(long)] instances: usize, /// The region where to deploy the instances. If this parameter is not /// specified, the command deploys the specified number of /// instances in all regions listed in the setting file. - #[clap(long)] + #[arg(long)] region: Option, }, @@ -159,7 +159,7 @@ pub enum TestbedAction { /// existing testbed. Start { /// Number of instances to deploy. - #[clap(long, default_value = "200")] + #[arg(long, default_value = "200")] instances: usize, }, @@ -175,7 +175,7 @@ pub enum Load { /// The fixed loads (in tx/s) to submit to the nodes. FixedLoad { /// A list of fixed load (tx/s). - #[clap( + #[arg( long, value_name = "INT", num_args(1..), @@ -187,11 +187,11 @@ pub enum Load { /// Search for the maximum load that the system can sustainably handle. Search { /// The initial load (in tx/s) to test and use a baseline. - #[clap(long, value_name = "INT", default_value = "250")] + #[arg(long, value_name = "INT", default_value = "250")] starting_load: usize, /// The maximum number of iterations before converging on a breaking /// point. - #[clap(long, value_name = "INT", default_value = "5")] + #[arg(long, value_name = "INT", default_value = "5")] max_iterations: usize, }, } diff --git a/crates/iota-aws-orchestrator/src/protocol/iota.rs b/crates/iota-aws-orchestrator/src/protocol/iota.rs index 8eaf3931f43..ac3d3e88c3d 100644 --- a/crates/iota-aws-orchestrator/src/protocol/iota.rs +++ b/crates/iota-aws-orchestrator/src/protocol/iota.rs @@ -50,7 +50,7 @@ impl FromStr for IotaBenchmarkType { impl BenchmarkType for IotaBenchmarkType {} -/// All configurations information to run a Iota client or validator. +/// All configurations information to run an IOTA client or validator. pub struct IotaProtocol { working_dir: PathBuf, } @@ -60,7 +60,7 @@ impl ProtocolCommands for IotaProtocol { vec![ // Install typical iota dependencies. "sudo apt-get -y install curl git-all clang cmake gcc libssl-dev pkg-config libclang-dev", - // This dependency is missing from the Iota docs. + // This dependency is missing from the IOTA docs. "sudo apt-get -y install libpq-dev", ] } @@ -214,7 +214,7 @@ impl ProtocolCommands for IotaProtocol { impl IotaProtocol { const CLIENT_METRICS_PORT: u16 = GenesisConfig::BENCHMARKS_PORT_OFFSET + 2000; - /// Make a new instance of the Iota protocol commands generator. + /// Make a new instance of the IOTA protocol commands generator. pub fn new(settings: &Settings) -> Self { Self { working_dir: [&settings.working_dir, &iota_config::IOTA_CONFIG_DIR.into()] diff --git a/crates/iota-benchmark/src/in_memory_wallet.rs b/crates/iota-benchmark/src/in_memory_wallet.rs index 62cffdfb7b7..40273a31e14 100644 --- a/crates/iota-benchmark/src/in_memory_wallet.rs +++ b/crates/iota-benchmark/src/in_memory_wallet.rs @@ -18,7 +18,7 @@ use crate::{ workloads::Gas, }; -/// A Iota account and all of the objects it owns +/// An IOTA account and all of the objects it owns #[derive(Debug)] pub struct IotaAccount { key: Arc, diff --git a/crates/iota-benchmark/src/options.rs b/crates/iota-benchmark/src/options.rs index 7b7c36c92b3..922a3afa47e 100644 --- a/crates/iota-benchmark/src/options.rs +++ b/crates/iota-benchmark/src/options.rs @@ -10,69 +10,69 @@ use strum_macros::EnumString; use crate::drivers::Interval; #[derive(Parser)] -#[clap(name = "Stress Testing Framework")] +#[command(name = "Stress Testing Framework")] pub struct Opts { - /// Size of the Iota committee. - #[clap(long, default_value = "4", global = true)] + /// Size of the IOTA committee. + #[arg(long, default_value = "4", global = true)] pub committee_size: u64, /// Num of accounts to use for transfer objects - #[clap(long, default_value = "5", global = true)] + #[arg(long, default_value = "5", global = true)] pub num_transfer_accounts: u64, /// Num server threads - #[clap(long, default_value = "24", global = true)] + #[arg(long, default_value = "24", global = true)] pub num_server_threads: u64, /// Num client threads /// ideally same as number of workers - #[clap(long, default_value = "3", global = true)] + #[arg(long, default_value = "3", global = true)] pub num_client_threads: u64, - #[clap(long, default_value = "", global = true)] + #[arg(long, default_value = "", global = true)] pub log_path: String, /// [Required for remote benchmark] /// Path where genesis.blob is stored when running remote benchmark - #[clap(long, default_value = "/tmp/genesis.blob", global = true)] + #[arg(long, default_value = "/tmp/genesis.blob", global = true)] pub genesis_blob_path: String, /// [Required for remote benchmark] /// Path where keypair for primary gas account is stored. The format of /// this file is same as what `iota keytool generate` outputs - #[clap(long, default_value = "", global = true)] + #[arg(long, default_value = "", global = true)] pub keystore_path: String, /// [Required for remote benchmark] /// Object id of the primary gas coin used for benchmark /// NOTE: THe remote network should have this coin in its genesis config /// with large enough gas. - #[clap(long, default_value = "", global = true)] + #[arg(long, default_value = "", global = true)] pub primary_gas_owner_id: String, - #[clap(long, default_value = "500", global = true)] + #[arg(long, default_value = "500", global = true)] pub gas_request_chunk_size: u64, /// Whether to run local or remote benchmark /// NOTE: For running remote benchmark we must have the following /// genesis_blob_path, keypair_path and primary_gas_id - #[clap(long, action = clap::ArgAction::Set, default_value = "true", global = true)] + #[arg(long, action = clap::ArgAction::Set, default_value = "true", global = true)] pub local: bool, /// Required in remote benchmark, namely when local = false /// Multiple fullnodes can be specified. - #[clap(long, num_args(1..), value_delimiter = ',', global = true)] + #[arg(long, num_args(1..), value_delimiter = ',', global = true)] pub fullnode_rpc_addresses: Vec, /// Whether to submit transactions to a fullnode. /// If true, use FullNodeProxy. /// Otherwise, use LocalValidatorAggregatorProxy. /// This param only matters when local = false, namely local runs always /// use a LocalValidatorAggregatorProxy. - #[clap(long, action = clap::ArgAction::Set, default_value = "false", global = true)] + #[arg(long, action = clap::ArgAction::Set, default_value = "false", global = true)] pub use_fullnode_for_execution: bool, /// True to use FullNodeReconfigObserver, /// Otherwise use EmbeddedReconfigObserver, - #[clap(long, action = clap::ArgAction::Set, default_value = "false", global = true)] + #[arg(long, action = clap::ArgAction::Set, default_value = "false", global = true)] pub use_fullnode_for_reconfig: bool, /// Default workload is 100% transfer object - #[clap(subcommand)] + #[command(subcommand)] pub run_spec: RunSpec, - #[clap(long, default_value = "127.0.0.1", global = true)] + #[arg(long, default_value = "127.0.0.1", global = true)] pub client_metric_host: String, - #[clap(long, default_value = "8081", global = true)] + #[arg(long, default_value = "8081", global = true)] pub client_metric_port: u16, /// Whether or no to download TXes during follow - #[clap(long, global = true)] + #[arg(long, global = true)] pub download_txes: bool, /// Number of transactions or duration to /// run the benchmark for. Default set to @@ -83,38 +83,37 @@ pub struct Opts { /// And if we wanted to run the test for /// 10,000 transactions we could set it to /// "10000" - #[clap(long, global = true, default_value = "unbounded")] + #[arg(long, global = true, default_value = "unbounded")] pub run_duration: Interval, /// Path where benchmark stats is stored - #[clap(long, default_value = "/tmp/bench_result", global = true)] + #[arg(long, default_value = "/tmp/bench_result", global = true)] pub benchmark_stats_path: String, /// Path where previous benchmark stats is stored to use for comparison - #[clap(long, default_value = "", global = true)] + #[arg(long, default_value = "", global = true)] pub compare_with: String, // Stat collection interval seconds - #[clap(long, default_value = "10", global = true)] + #[arg(long, default_value = "10", global = true)] pub stat_collection_interval: u64, // Enable stress stat collection. When enabled the sysinfo crate will be used // to gather system information. For example cpu usage will be polled every // 1 second and the P50/P99 usage statistics will be outputted either at // the end of the benchmark or periodically during a continuous run. - #[clap(long, action, global = true)] + #[arg(long, action, global = true)] pub stress_stat_collection: bool, // When starting multiple stress clients, stagger the start time by a random multiplier // between 0 and this value, times initialization time which is 1min. This helps to avoid // transaction conflicts between clients. - #[clap(long, default_value = "0", global = true)] + #[arg(long, default_value = "0", global = true)] pub staggered_start_max_multiplier: u32, - /// Start the stress test at a given protocol version. (Usually unnecessary /// if stress test is built at the same commit as the validators. - #[clap(long, global = true)] + #[arg(long, global = true)] pub protocol_version: Option, } #[derive(Debug, Clone, Parser, Eq, PartialEq, EnumString)] #[non_exhaustive] -#[clap(rename_all = "kebab-case")] +#[command(rename_all = "kebab-case")] pub enum RunSpec { // Allow the ability to mix shared object and // single owner transactions in the benchmarking @@ -163,30 +162,30 @@ pub enum RunSpec { // `num_of_benchmark_groups = 2`, then we expect all the arguments under this // subcommand to contain two values on their vectors - one for each benchmark set. // If an argument doesn't contain the right number of values then it will panic. - #[clap(long, default_value = "1")] + #[arg(long, default_value = "1")] num_of_benchmark_groups: u32, // relative weight of shared counter // transaction in the benchmark workload - #[clap(long, num_args(1..), value_delimiter = ',', default_values_t = [0])] + #[arg(long, num_args(1..), value_delimiter = ',', default_values_t = [0])] shared_counter: Vec, // relative weight of transfer object // transactions in the benchmark workload - #[clap(long, num_args(1..), value_delimiter = ',', default_values_t = [1])] + #[arg(long, num_args(1..), value_delimiter = ',', default_values_t = [1])] transfer_object: Vec, // relative weight of delegation transactions in the benchmark workload - #[clap(long, num_args(1..), value_delimiter = ',', default_values_t = [0])] + #[arg(long, num_args(1..), value_delimiter = ',', default_values_t = [0])] delegation: Vec, // relative weight of batch payment transactions in the benchmark workload - #[clap(long, num_args(1..), value_delimiter = ',', default_values_t = [0])] + #[arg(long, num_args(1..), value_delimiter = ',', default_values_t = [0])] batch_payment: Vec, // relative weight of adversarial transactions in the benchmark workload - #[clap(long, num_args(1..), value_delimiter = ',', default_values_t = [0])] + #[arg(long, num_args(1..), value_delimiter = ',', default_values_t = [0])] adversarial: Vec, // relative weight of shared deletion transactions in the benchmark workload - #[clap(long, num_args(1..), value_delimiter = ',', default_values_t = [0])] + #[arg(long, num_args(1..), value_delimiter = ',', default_values_t = [0])] shared_deletion: Vec, // relative weight of randomness transactions in the benchmark workload - #[clap(long, num_args(1..), value_delimiter = ',', default_values_t = [0])] + #[arg(long, num_args(1..), value_delimiter = ',', default_values_t = [0])] randomness: Vec, // --- workload-specific options --- (TODO: use subcommands or similar) @@ -196,43 +195,43 @@ pub enum RunSpec { // counter. The way total number of counters to // create is computed roughly as: // total_shared_counters = max(1, qps * (1.0 - hotness/100.0)) - #[clap(long, num_args(1..), value_delimiter = ',', default_values_t = [50])] + #[arg(long, num_args(1..), value_delimiter = ',', default_values_t = [50])] shared_counter_hotness_factor: Vec, // The number of shared counters this stress client will create and use. // This parameter takes precedence over `shared_counter_hotness_factor`, meaning that when // this parameter is specified, `shared_counter_hotness_factor` is ignored when // deciding the number of shared counters to create. - #[clap(long, num_args(1..), value_delimiter = ',')] + #[arg(long, num_args(1..), value_delimiter = ',')] num_shared_counters: Option>, // Maximum gas price increment over the RGP for shared counter transactions. // The actual increment for each transaction is chosen at random a value between 0 and this // value. - #[clap(long, num_args(1..), value_delimiter = ',', default_values_t = [0])] + #[arg(long, num_args(1..), value_delimiter = ',', default_values_t = [0])] shared_counter_max_tip: Vec, // batch size use for batch payment workload - #[clap(long, num_args(1..), value_delimiter = ',', default_values_t = [15])] + #[arg(long, num_args(1..), value_delimiter = ',', default_values_t = [15])] batch_payment_size: Vec, // type and load % of adversarial transactions in the benchmark workload. // Format is "{adversarial_type}-{load_factor}". // `load_factor` is a number between 0.0 and 1.0 which dictates how much load per tx // Default is (0-0.5) implying random load at 50% load. See `AdversarialPayloadType` enum // for `adversarial_type` - #[clap(long, num_args(1..), value_delimiter = ',', default_values_t = ["0-1.0".to_string()])] + #[arg(long, num_args(1..), value_delimiter = ',', default_values_t = ["0-1.0".to_string()])] adversarial_cfg: Vec, // --- generic options --- // Target qps - #[clap(long, num_args(1..), value_delimiter = ',', default_values_t = [1000])] + #[arg(long, num_args(1..), value_delimiter = ',', default_values_t = [1000])] target_qps: Vec, // Number of workers - #[clap(long, num_args(1..), value_delimiter = ',', default_values_t = [12])] + #[arg(long, num_args(1..), value_delimiter = ',', default_values_t = [12])] num_workers: Vec, // Max in-flight ratio - #[clap(long, num_args(1..), value_delimiter = ',', default_values_t = [5])] + #[arg(long, num_args(1..), value_delimiter = ',', default_values_t = [5])] in_flight_ratio: Vec, // Setting the duration of each benchmark. Benchmarks will run in sequence. - #[clap(long, num_args(1..), value_delimiter = ',', default_values_t = [Interval::from_str("unbounded").unwrap()])] + #[arg(long, num_args(1..), value_delimiter = ',', default_values_t = [Interval::from_str("unbounded").unwrap()])] duration: Vec, }, } diff --git a/crates/iota-bridge-cli/src/lib.rs b/crates/iota-bridge-cli/src/lib.rs index 74392f8977c..dc7062f0dfa 100644 --- a/crates/iota-bridge-cli/src/lib.rs +++ b/crates/iota-bridge-cli/src/lib.rs @@ -47,9 +47,9 @@ use tracing::info; pub const SEPOLIA_BRIDGE_PROXY_ADDR: &str = "0xAE68F87938439afEEDd6552B0E83D2CbC2473623"; #[derive(Parser)] -#[clap(rename_all = "kebab-case")] +#[command(rename_all = "kebab-case")] pub struct Args { - #[clap(subcommand)] + #[command(subcommand)] pub command: BridgeCommand, } @@ -59,156 +59,156 @@ pub enum Network { } #[derive(Parser)] -#[clap(rename_all = "kebab-case")] +#[command(rename_all = "kebab-case")] pub enum BridgeCommand { - #[clap(name = "create-bridge-validator-key")] + #[command(name = "create-bridge-validator-key")] CreateBridgeValidatorKey { path: PathBuf }, - #[clap(name = "create-bridge-client-key")] + #[command(name = "create-bridge-client-key")] CreateBridgeClientKey { path: PathBuf, - #[clap(long = "use-ecdsa", default_value = "false")] + #[arg(long = "use-ecdsa", default_value = "false")] use_ecdsa: bool, }, /// Read bridge key from a file and print related information /// If `is-validator-key` is true, the key must be a secp256k1 key - #[clap(name = "examine-key")] + #[command(name = "examine-key")] ExamineKey { path: PathBuf, - #[clap(long = "is-validator-key")] + #[arg(long = "is-validator-key")] is_validator_key: bool, }, - #[clap(name = "create-bridge-node-config-template")] + #[command(name = "create-bridge-node-config-template")] CreateBridgeNodeConfigTemplate { path: PathBuf, - #[clap(long = "run-client")] + #[arg(long = "run-client")] run_client: bool, }, /// Governance client to facilitate and execute Bridge governance actions - #[clap(name = "governance")] + #[command(name = "governance")] Governance { /// Path of BridgeCliConfig - #[clap(long = "config-path")] + #[arg(long = "config-path")] config_path: PathBuf, - #[clap(long = "chain-id")] + #[arg(long = "chain-id")] chain_id: u8, - #[clap(subcommand)] + #[command(subcommand)] cmd: GovernanceClientCommands, /// If true, only collect signatures but not execute on chain - #[clap(long = "dry-run")] + #[arg(long = "dry-run")] dry_run: bool, }, /// View current status of Eth bridge - #[clap(name = "view-eth-bridge")] + #[command(name = "view-eth-bridge")] ViewEthBridge { - #[clap(long = "network")] + #[arg(long = "network")] network: Option, - #[clap(long = "bridge-proxy")] + #[arg(long = "bridge-proxy")] bridge_proxy: Option, - #[clap(long = "eth-rpc-url")] + #[arg(long = "eth-rpc-url")] eth_rpc_url: String, }, /// View current list of registered validators - #[clap(name = "view-bridge-registration")] + #[command(name = "view-bridge-registration")] ViewBridgeRegistration { - #[clap(long = "iota-rpc-url")] + #[arg(long = "iota-rpc-url")] iota_rpc_url: String, }, - /// View current status of Iota bridge - #[clap(name = "view-iota-bridge")] + /// View current status of IOTA bridge + #[command(name = "view-iota-bridge")] ViewIotaBridge { - #[clap(long = "iota-rpc-url")] + #[arg(long = "iota-rpc-url")] iota_rpc_url: String, - #[clap(long, default_value = "false")] + #[arg(long, default_value = "false")] hex: bool, - #[clap(long, default_value = "false")] + #[arg(long, default_value = "false")] ping: bool, }, /// Client to facilitate and execute Bridge actions - #[clap(name = "client")] + #[command(name = "client")] Client { /// Path of BridgeCliConfig - #[clap(long = "config-path")] + #[arg(long = "config-path")] config_path: PathBuf, - #[clap(subcommand)] + #[command(subcommand)] cmd: BridgeClientCommands, }, } #[derive(Parser)] -#[clap(rename_all = "kebab-case")] +#[command(rename_all = "kebab-case")] pub enum GovernanceClientCommands { - #[clap(name = "emergency-button")] + #[command(name = "emergency-button")] EmergencyButton { - #[clap(name = "nonce", long)] + #[arg(name = "nonce", long)] nonce: u64, - #[clap(name = "action-type", long)] + #[arg(name = "action-type", long)] action_type: EmergencyActionType, }, - #[clap(name = "update-committee-blocklist")] + #[command(name = "update-committee-blocklist")] UpdateCommitteeBlocklist { - #[clap(name = "nonce", long)] + #[arg(name = "nonce", long)] nonce: u64, - #[clap(name = "blocklist-type", long)] + #[arg(name = "blocklist-type", long)] blocklist_type: BlocklistType, - #[clap(name = "pubkey-hex", use_value_delimiter = true, long)] + #[arg(name = "pubkey-hex", use_value_delimiter = true, long)] pubkeys_hex: Vec, }, - #[clap(name = "update-limit")] + #[command(name = "update-limit")] UpdateLimit { - #[clap(name = "nonce", long)] + #[arg(name = "nonce", long)] nonce: u64, - #[clap(name = "sending-chain", long)] + #[arg(name = "sending-chain", long)] sending_chain: u8, - #[clap(name = "new-usd-limit", long)] + #[arg(name = "new-usd-limit", long)] new_usd_limit: u64, }, - #[clap(name = "update-asset-price")] + #[command(name = "update-asset-price")] UpdateAssetPrice { - #[clap(name = "nonce", long)] + #[arg(name = "nonce", long)] nonce: u64, - #[clap(name = "token-id", long)] + #[arg(name = "token-id", long)] token_id: u8, - #[clap(name = "new-usd-price", long)] + #[arg(name = "new-usd-price", long)] new_usd_price: u64, }, - #[clap(name = "add-tokens-on-iota")] + #[command(name = "add-tokens-on-iota")] AddTokensOnIota { - #[clap(name = "nonce", long)] + #[arg(name = "nonce", long)] nonce: u64, - #[clap(name = "token-ids", use_value_delimiter = true, long)] + #[arg(name = "token-ids", use_value_delimiter = true, long)] token_ids: Vec, - #[clap(name = "token-type-names", use_value_delimiter = true, long)] + #[arg(name = "token-type-names", use_value_delimiter = true, long)] token_type_names: Vec, - #[clap(name = "token-prices", use_value_delimiter = true, long)] + #[arg(name = "token-prices", use_value_delimiter = true, long)] token_prices: Vec, }, - #[clap(name = "add-tokens-on-evm")] + #[command(name = "add-tokens-on-evm")] AddTokensOnEvm { - #[clap(name = "nonce", long)] + #[arg(name = "nonce", long)] nonce: u64, - #[clap(name = "token-ids", use_value_delimiter = true, long)] + #[arg(name = "token-ids", use_value_delimiter = true, long)] token_ids: Vec, - #[clap(name = "token-type-names", use_value_delimiter = true, long)] + #[arg(name = "token-type-names", use_value_delimiter = true, long)] token_addresses: Vec, - #[clap(name = "token-prices", use_value_delimiter = true, long)] + #[arg(name = "token-prices", use_value_delimiter = true, long)] token_prices: Vec, - #[clap(name = "token-iota-decimals", use_value_delimiter = true, long)] + #[arg(name = "token-iota-decimals", use_value_delimiter = true, long)] token_iota_decimals: Vec, }, - #[clap(name = "upgrade-evm-contract")] + #[command(name = "upgrade-evm-contract")] UpgradeEVMContract { - #[clap(name = "nonce", long)] + #[arg(name = "nonce", long)] nonce: u64, - #[clap(name = "proxy-address", long)] + #[arg(name = "proxy-address", long)] proxy_address: EthAddress, /// The address of the new implementation contract - #[clap(name = "implementation-address", long)] + #[arg(name = "implementation-address", long)] implementation_address: EthAddress, /// Function selector with params types, e.g. `foo(uint256,bool,string)` - #[clap(name = "function-selector", long)] + #[arg(name = "function-selector", long)] function_selector: Option, /// Params to be passed to the function, e.g. `420,false,hello` - #[clap(name = "params", use_value_delimiter = true, long)] + #[arg(name = "params", use_value_delimiter = true, long)] params: Vec, }, } @@ -379,7 +379,7 @@ pub fn select_contract_address( #[derive(Clone, Debug, Deserialize, Serialize)] #[serde(rename_all = "kebab-case")] pub struct BridgeCliConfig { - /// Rpc url for Iota fullnode, used for query stuff and submit transactions. + /// Rpc url for IOTA fullnode, used for query stuff and submit transactions. pub iota_rpc_url: String, /// Rpc url for Eth fullnode, used for query stuff. pub eth_rpc_url: String, @@ -391,7 +391,7 @@ pub struct BridgeCliConfig { /// - Base64 encoded `privkey` for Raw key /// - Hex encoded `privkey` for Raw key /// At least one of `iota_key_path` or `eth_key_path` must be provided. - /// If only one is provided, it will be used for both Iota and Eth. + /// If only one is provided, it will be used for both IOTA and Eth. pub iota_key_path: Option, /// See `iota_key_path`. Must be Secp256k1 key. pub eth_key_path: Option, @@ -400,7 +400,7 @@ pub struct BridgeCliConfig { impl Config for BridgeCliConfig {} pub struct LoadedBridgeCliConfig { - /// Rpc url for Iota fullnode, used for query stuff and submit transactions. + /// Rpc url for IOTA fullnode, used for query stuff and submit transactions. pub iota_rpc_url: String, /// Rpc url for Eth fullnode, used for query stuff. pub eth_rpc_url: String, @@ -412,7 +412,7 @@ pub struct LoadedBridgeCliConfig { pub eth_bridge_config_proxy_address: EthAddress, /// Proxy address for BridgeLimiter deployed on Eth pub eth_bridge_limiter_proxy_address: EthAddress, - /// Key pair for Iota operations + /// Key pair for IOTA operations iota_key: IotaKeyPair, /// Key pair for Eth operations, must be Secp256k1 key eth_signer: EthSigner, @@ -469,7 +469,7 @@ impl LoadedBridgeCliConfig { let eth_address = eth_signer.address(); let eth_chain_id = provider.get_chainid().await?; let iota_address = IotaAddress::from(&iota_key.public()); - println!("Using Iota address: {:?}", iota_address); + println!("Using IOTA address: {:?}", iota_address); println!("Using Eth address: {:?}", eth_address); println!("Using Eth chain: {:?}", eth_chain_id); @@ -504,7 +504,7 @@ impl LoadedBridgeCliConfig { .get_coins(iota_client_address, None, None, None) .await? .data; - // TODO: is 5 Iota a good number? + // TODO: is 5 IOTA a good number? let gas = gases .into_iter() .find(|coin| coin.balance >= 5_000_000_000) @@ -517,31 +517,31 @@ impl LoadedBridgeCliConfig { } } #[derive(Parser)] -#[clap(rename_all = "kebab-case")] +#[command(rename_all = "kebab-case")] pub enum BridgeClientCommands { - #[clap(name = "deposit-native-ether-on-eth")] + #[command(name = "deposit-native-ether-on-eth")] DepositNativeEtherOnEth { - #[clap(long)] + #[arg(long)] ether_amount: f64, - #[clap(long)] + #[arg(long)] target_chain: u8, - #[clap(long)] + #[arg(long)] iota_recipient_address: IotaAddress, }, - #[clap(name = "deposit-on-iota")] + #[command(name = "deposit-on-iota")] DepositOnIota { - #[clap(long)] + #[arg(long)] coin_object_id: ObjectID, - #[clap(long)] + #[arg(long)] coin_type: String, - #[clap(long)] + #[arg(long)] target_chain: u8, - #[clap(long)] + #[arg(long)] recipient_address: EthAddress, }, - #[clap(name = "claim-on-eth")] + #[command(name = "claim-on-eth")] ClaimOnEth { - #[clap(long)] + #[arg(long)] seq_num: u64, }, } @@ -665,7 +665,7 @@ async fn deposit_on_iota( ); let signed_tx = Transaction::from_data(tx_data, vec![sig]); let tx_digest = *signed_tx.digest(); - info!(?tx_digest, "Sending deposit transction to Iota."); + info!(?tx_digest, "Sending deposit transction to IOTA."); let resp = iota_bridge_client .execute_transaction_block_with_effects(signed_tx) .await @@ -715,7 +715,7 @@ async fn claim_on_eth( let message = eth_iota_bridge::Message::from(parsed_message); let tx = eth_iota_bridge.transfer_bridged_tokens_with_signatures(signatures, message); let _eth_claim_tx_receipt = tx.send().await.unwrap().await.unwrap().unwrap(); - info!("Iota to Eth bridge transfer claimed"); + info!("IOTA to Eth bridge transfer claimed"); Ok(()) } diff --git a/crates/iota-bridge-cli/src/main.rs b/crates/iota-bridge-cli/src/main.rs index ed90c1d7c2e..0a4f1b44304 100644 --- a/crates/iota-bridge-cli/src/main.rs +++ b/crates/iota-bridge-cli/src/main.rs @@ -99,7 +99,7 @@ async fn main() -> anyhow::Result<()> { ); let agg = BridgeAuthorityAggregator::new(bridge_committee); - // Handle Iota Side + // Handle IOTA Side if chain_id.is_iota_chain() { let iota_chain_id = BridgeChainId::try_from(bridge_summary.chain_id).unwrap(); assert_eq!( @@ -109,7 +109,7 @@ async fn main() -> anyhow::Result<()> { ); // Create BridgeAction let iota_action = make_action(iota_chain_id, &cmd); - println!("Action to execute on Iota: {:?}", iota_action); + println!("Action to execute on IOTA: {:?}", iota_action); let certified_action = agg .request_committee_signatures(iota_action) .await @@ -144,10 +144,10 @@ async fn main() -> anyhow::Result<()> { .await .expect("Failed to execute transaction block with effects"); if resp.status_ok().unwrap() { - println!("Iota Transaction succeeded: {:?}", resp.digest); + println!("IOTA Transaction succeeded: {:?}", resp.digest); } else { println!( - "Iota Transaction failed: {:?}. Effects: {:?}", + "IOTA Transaction failed: {:?}. Effects: {:?}", resp.digest, resp.effects ); } diff --git a/crates/iota-bridge-indexer/config.yaml b/crates/iota-bridge-indexer/config.yaml index 5ce78b55a94..2363925284f 100644 --- a/crates/iota-bridge-indexer/config.yaml +++ b/crates/iota-bridge-indexer/config.yaml @@ -12,7 +12,7 @@ # concurrency: 1 # Bridge genesis checkpoint in IOTA, for testnet = 43917829 # bridge_genesis_checkpoint: -# Ethereum to Iota bridge contract address +# Ethereum to IOTA bridge contract address # eth_iota_bridge_contract_address: # Starting block number # start_block: @@ -23,4 +23,4 @@ # checkpoint size of each backfill worker, use 432000 for 1 worker per day, assume 5 checkpoint per second # back_fill_lot_size: # Optional starting checkpoint for realtime ingestion task -# resume_from_checkpoint: +# resume_from_checkpoint: diff --git a/crates/iota-bridge-indexer/src/iota_bridge_indexer.rs b/crates/iota-bridge-indexer/src/iota_bridge_indexer.rs index a1f6be3d743..82673332753 100644 --- a/crates/iota-bridge-indexer/src/iota_bridge_indexer.rs +++ b/crates/iota-bridge-indexer/src/iota_bridge_indexer.rs @@ -234,7 +234,7 @@ fn process_iota_event( Ok(if ev.type_.address == BRIDGE_ADDRESS { match ev.type_.name.as_str() { "TokenDepositedEvent" => { - info!("Observed Iota Deposit {:?}", ev); + info!("Observed IOTA Deposit {:?}", ev); // todo: metrics.total_iota_token_deposited.inc(); let move_event: MoveTokenDepositedEvent = bcs::from_bytes(&ev.contents)?; Some(ProcessedTxnData::TokenTransfer(TokenTransfer { @@ -257,7 +257,7 @@ fn process_iota_event( })) } "TokenTransferApproved" => { - info!("Observed Iota Approval {:?}", ev); + info!("Observed IOTA Approval {:?}", ev); // todo: metrics.total_iota_token_transfer_approved.inc(); let event: MoveTokenTransferApproved = bcs::from_bytes(&ev.contents)?; Some(ProcessedTxnData::TokenTransfer(TokenTransfer { @@ -274,7 +274,7 @@ fn process_iota_event( })) } "TokenTransferClaimed" => { - info!("Observed Iota Claim {:?}", ev); + info!("Observed IOTA Claim {:?}", ev); // todo: metrics.total_iota_token_transfer_claimed.inc(); let event: MoveTokenTransferClaimed = bcs::from_bytes(&ev.contents)?; Some(ProcessedTxnData::TokenTransfer(TokenTransfer { diff --git a/crates/iota-bridge-indexer/src/iota_transaction_handler.rs b/crates/iota-bridge-indexer/src/iota_transaction_handler.rs index 5317677a84c..30d88b05c17 100644 --- a/crates/iota-bridge-indexer/src/iota_transaction_handler.rs +++ b/crates/iota-bridge-indexer/src/iota_transaction_handler.rs @@ -92,7 +92,7 @@ pub fn into_token_transfers( } match ev.type_.name.as_str() { "TokenDepositedEvent" => { - info!("Observed Iota Deposit {:?}", ev); + info!("Observed IOTA Deposit {:?}", ev); metrics.total_iota_token_deposited.inc(); let move_event: MoveTokenDepositedEvent = bcs::from_bytes(&ev.bcs)?; transfers.push(ProcessedTxnData::TokenTransfer(TokenTransfer { @@ -115,7 +115,7 @@ pub fn into_token_transfers( })); } "TokenTransferApproved" => { - info!("Observed Iota Approval {:?}", ev); + info!("Observed IOTA Approval {:?}", ev); metrics.total_iota_token_transfer_approved.inc(); let event: MoveTokenTransferApproved = bcs::from_bytes(&ev.bcs)?; transfers.push(ProcessedTxnData::TokenTransfer(TokenTransfer { @@ -132,7 +132,7 @@ pub fn into_token_transfers( })); } "TokenTransferClaimed" => { - info!("Observed Iota Claim {:?}", ev); + info!("Observed IOTA Claim {:?}", ev); metrics.total_iota_token_transfer_claimed.inc(); let event: MoveTokenTransferClaimed = bcs::from_bytes(&ev.bcs)?; transfers.push(ProcessedTxnData::TokenTransfer(TokenTransfer { diff --git a/crates/iota-bridge-indexer/src/main.rs b/crates/iota-bridge-indexer/src/main.rs index 4d96bad8a90..15055fd00b5 100644 --- a/crates/iota-bridge-indexer/src/main.rs +++ b/crates/iota-bridge-indexer/src/main.rs @@ -33,7 +33,7 @@ use tracing::info; #[derive(Parser, Clone, Debug)] struct Args { /// Path to a yaml config - #[clap(long, short)] + #[arg(long, short)] config_path: Option, } diff --git a/crates/iota-bridge/src/action_executor.rs b/crates/iota-bridge/src/action_executor.rs index a1070ae3aab..75b6e1a268d 100644 --- a/crates/iota-bridge/src/action_executor.rs +++ b/crates/iota-bridge/src/action_executor.rs @@ -485,7 +485,7 @@ where return; } - info!("Building Iota transaction"); + info!("Building IOTA transaction"); let rgp = iota_client.get_reference_gas_price_until_success().await; let tx_data = match build_iota_transaction( *iota_address, @@ -527,7 +527,7 @@ where return; } - info!(?tx_digest, ?gas_object_ref, "Sending transaction to Iota"); + info!(?tx_digest, ?gas_object_ref, "Sending transaction to IOTA"); match iota_client .execute_transaction_block_with_effects(signed_tx) .await @@ -541,7 +541,7 @@ where error!( ?action_key, ?tx_digest, - "Iota transaction failed at signing: {err:?}" + "IOTA transaction failed at signing: {err:?}" ); metrics.err_iota_transaction_submission.inc(); let metrics_clone = metrics.clone(); @@ -599,7 +599,7 @@ where "Expected TokenTransferAlreadyClaimed, TokenTransferClaimed, TokenTransferApproved or TokenTransferAlreadyApproved event but got: {:?}", events, ); - info!(?tx_digest, "Iota transaction executed successfully"); + info!(?tx_digest, "IOTA transaction executed successfully"); store .remove_pending_actions(&[action.digest()]) .unwrap_or_else(|e| { @@ -617,7 +617,7 @@ where metrics.err_iota_transaction_execution.inc(); error!( ?tx_digest, - "Manual intervention is needed. Iota transaction executed and failed with error: {error:?}" + "Manual intervention is needed. IOTA transaction executed and failed with error: {error:?}" ); } } diff --git a/crates/iota-bridge/src/config.rs b/crates/iota-bridge/src/config.rs index 1714fe4270a..f1f37a81419 100644 --- a/crates/iota-bridge/src/config.rs +++ b/crates/iota-bridge/src/config.rs @@ -67,9 +67,9 @@ pub struct EthConfig { #[derive(Clone, Debug, Deserialize, Serialize)] #[serde(rename_all = "kebab-case")] pub struct IotaConfig { - /// Rpc url for Iota fullnode, used for query stuff and submit transactions. + /// Rpc url for IOTA fullnode, used for query stuff and submit transactions. pub iota_rpc_url: String, - /// The expected BridgeChainId on Iota side. + /// The expected BridgeChainId on IOTA side. pub iota_bridge_chain_id: u8, /// Path of the file where bridge client key (any IotaKeyPair) is stored. /// If `run_client` is true, and this is None, then use @@ -114,7 +114,7 @@ pub struct BridgeNodeConfig { /// A list of approved governance actions. Action in this list will be /// signed when requested by client. pub approved_governance_actions: Vec, - /// Iota configuration + /// IOTA configuration pub iota: IotaConfig, /// Eth configuration pub eth: EthConfig, @@ -132,7 +132,7 @@ impl BridgeNodeConfig { BridgeChainId::try_from(self.eth.eth_bridge_chain_id)?, ) { return Err(anyhow!( - "Route between Iota chain id {} and Eth chain id {} is not valid", + "Route between IOTA chain id {} and Eth chain id {} is not valid", self.iota.iota_bridge_chain_id, self.eth.eth_bridge_chain_id, )); @@ -308,12 +308,12 @@ impl BridgeNodeConfig { Some(path) => read_key(path, false), }?; - // If bridge chain id is Iota Mainent or Testnet, we expect to see chain + // If bridge chain id is IOTA Mainent or Testnet, we expect to see chain // identifier to match accordingly. let iota_identifier = iota_client .get_chain_identifier() .await - .map_err(|e| anyhow!("Error getting chain identifier from Iota: {:?}", e))?; + .map_err(|e| anyhow!("Error getting chain identifier from IOTA: {:?}", e))?; if self.iota.iota_bridge_chain_id == BridgeChainId::IotaMainnet as u8 && iota_identifier != get_mainnet_chain_identifier().to_string() { @@ -333,7 +333,7 @@ impl BridgeNodeConfig { ); } info!( - "Connected to Iota chain: {}, Bridge chain id: {}", + "Connected to IOTA chain: {}, Bridge chain id: {}", iota_identifier, self.iota.iota_bridge_chain_id, ); @@ -428,11 +428,11 @@ pub async fn pick_highest_balance_coin( }) .await; if highest_balance_coin.is_none() { - return Err(anyhow!("No Iota coins found for address {:?}", address)); + return Err(anyhow!("No IOTA coins found for address {:?}", address)); } if highest_balance < minimal_amount { return Err(anyhow!( - "Found no single coin that has >= {} balance Iota for address {:?}", + "Found no single coin that has >= {} balance IOTA for address {:?}", minimal_amount, address, )); diff --git a/crates/iota-bridge/src/e2e_tests/basic.rs b/crates/iota-bridge/src/e2e_tests/basic.rs index d752a8322f1..3b4e3ebe738 100644 --- a/crates/iota-bridge/src/e2e_tests/basic.rs +++ b/crates/iota-bridge/src/e2e_tests/basic.rs @@ -152,7 +152,7 @@ async fn test_bridge_from_eth_to_iota_to_eth() { let call = eth_iota_bridge.transfer_bridged_tokens_with_signatures(signatures, message); let eth_claim_tx_receipt = send_eth_tx_and_get_tx_receipt(call).await; assert_eq!(eth_claim_tx_receipt.status.unwrap().as_u64(), 1); - info!("Iota to Eth bridge transfer claimed"); + info!("IOTA to Eth bridge transfer claimed"); // Assert eth_address_1 has received ETH assert_eq!( eth_signer.get_balance(eth_address_1, None).await.unwrap(), @@ -160,7 +160,7 @@ async fn test_bridge_from_eth_to_iota_to_eth() { ); } -// Test add new coins on both Iota and Eth +// Test add new coins on both IOTA and Eth // Also test bridge node handling `NewTokenEvent`` #[tokio::test(flavor = "multi_thread", worker_threads = 8)] #[ignore = "https://github.com/iotaledger/iota/issues/3224"] @@ -175,7 +175,7 @@ async fn test_add_new_coins_on_iota_and_eth() { let bridge_arg = bridge_test_cluster.get_mut_bridge_arg().await.unwrap(); - // Register tokens on Iota + // Register tokens on IOTA let token_id = 5; let token_iota_decimal = 9; // this needs to match ka.move let token_price = 10000; @@ -258,7 +258,7 @@ async fn test_add_new_coins_on_iota_and_eth() { _ => false, } })); - info!("Approved new token on Iota"); + info!("Approved new token on IOTA"); // Assert new token is correctly added let treasury_summary = bridge_test_cluster @@ -456,7 +456,7 @@ pub async fn initiate_bridge_erc20_to_iota( .tap_ok(|_| { info!( nonce, - token_id, amount_u64, "Eth to Iota bridge transfer claimed" + token_id, amount_u64, "Eth to IOTA bridge transfer claimed" ); }) } @@ -477,7 +477,7 @@ pub async fn initiate_bridge_eth_to_iota( let eth_chain_id = bridge_test_cluster.eth_chain_id(); let token_id = TOKEN_ID_ETH; - let iota_amount = (U256::from(amount) * U256::exp10(8)).as_u64(); // DP for Ether on Iota + let iota_amount = (U256::from(amount) * U256::exp10(8)).as_u64(); // DP for Ether on IOTA let eth_tx = deposit_native_eth_to_sol_contract( ð_signer, @@ -517,7 +517,7 @@ pub async fn initiate_bridge_eth_to_iota( ) .await .tap_ok(|_| { - info!("Eth to Iota bridge transfer claimed"); + info!("Eth to IOTA bridge transfer claimed"); }) } @@ -554,7 +554,7 @@ pub async fn initiate_bridge_iota_to_eth( { Ok(resp) => { if !resp.status_ok().unwrap() { - return Err(anyhow!("Iota TX error")); + return Err(anyhow!("IOTA TX error")); } else { resp } @@ -604,7 +604,7 @@ pub async fn initiate_bridge_iota_to_eth( ) .await .unwrap(); - info!("Iota to Eth bridge transfer approved."); + info!("IOTA to Eth bridge transfer approved."); Ok(bridge_event) } diff --git a/crates/iota-bridge/src/e2e_tests/complex.rs b/crates/iota-bridge/src/e2e_tests/complex.rs index c258002c824..76ab551688c 100644 --- a/crates/iota-bridge/src/e2e_tests/complex.rs +++ b/crates/iota-bridge/src/e2e_tests/complex.rs @@ -62,7 +62,7 @@ async fn test_iota_bridge_paused() { initiate_bridge_eth_to_iota(&bridge_test_cluster, 10, 0) .await .unwrap(); - // verify Eth was transferred to Iota address + // verify Eth was transferred to IOTA address let eth_coin_type = iota_token_type_tags.get(&TOKEN_ID_ETH).unwrap(); let eth_coin = bridge_client .iota_client() @@ -111,10 +111,10 @@ async fn test_iota_bridge_paused() { // verify bridge paused assert!(bridge_client.get_bridge_summary().await.unwrap().is_frozen); - // Transfer from eth to iota should fail on Iota + // Transfer from eth to iota should fail on IOTA let eth_to_iota_bridge_action = initiate_bridge_eth_to_iota(&bridge_test_cluster, 10, 1).await; assert!(eth_to_iota_bridge_action.is_err()); - // message should not be recorded on Iota when the bridge is paused + // message should not be recorded on IOTA when the bridge is paused let res = bridge_test_cluster .bridge_client() .get_token_transfer_action_onchain_status_until_success( @@ -123,7 +123,7 @@ async fn test_iota_bridge_paused() { ) .await; assert_eq!(BridgeActionStatus::NotFound, res); - // Transfer from Iota to eth should fail + // Transfer from IOTA to eth should fail let iota_to_eth_bridge_action = initiate_bridge_iota_to_eth( &bridge_test_cluster, EthAddress::random(), diff --git a/crates/iota-bridge/src/error.rs b/crates/iota-bridge/src/error.rs index de290334a58..cd344de337c 100644 --- a/crates/iota-bridge/src/error.rs +++ b/crates/iota-bridge/src/error.rs @@ -18,7 +18,7 @@ pub enum BridgeError { NoBridgeEventsInTxPosition, // Found a bridge event but not in a recognized Eth bridge contract BridgeEventInUnrecognizedEthContract, - // Found a bridge event but not in a recognized Iota bridge package + // Found a bridge event but not in a recognized IOTA bridge package BridgeEventInUnrecognizedIotaPackage, // Found BridgeEvent but not BridgeAction BridgeEventNotActionable, @@ -60,7 +60,7 @@ pub enum BridgeError { AuthorityUrlInvalid, // Action is not token transfer ActionIsNotTokenTransferAction, - // Iota transaction failure due to generic error + // IOTA transaction failure due to generic error IotaTxFailureGeneric(String), // Zero value bridge transfer should not be allowed ZeroValueBridgeTransfer(String), diff --git a/crates/iota-bridge/src/events.rs b/crates/iota-bridge/src/events.rs index 101db3aca1c..7d6ae66f936 100644 --- a/crates/iota-bridge/src/events.rs +++ b/crates/iota-bridge/src/events.rs @@ -221,7 +221,7 @@ pub struct EmittedIotaToEthTokenBridgeV1 { pub iota_address: IotaAddress, pub eth_address: EthAddress, pub token_id: u8, - // The amount of tokens deposited with decimal points on Iota side + // The amount of tokens deposited with decimal points on IOTA side pub amount_iota_adjusted: u64, } diff --git a/crates/iota-bridge/src/iota_client.rs b/crates/iota-bridge/src/iota_client.rs index 63da0c725c8..3d7ed87baf3 100644 --- a/crates/iota-bridge/src/iota_client.rs +++ b/crates/iota-bridge/src/iota_client.rs @@ -58,7 +58,7 @@ impl IotaBridgeClient { .build(rpc_url) .await .map_err(|e| { - anyhow!("Can't establish connection with Iota Rpc {rpc_url}. Error: {e}") + anyhow!("Can't establish connection with IOTA Rpc {rpc_url}. Error: {e}") })?; let self_ = Self { inner }; self_.describe().await?; @@ -132,7 +132,7 @@ where Ok(events) } - /// Returns BridgeAction from a Iota Transaction with transaction hash + /// Returns BridgeAction from an IOTA Transaction with transaction hash /// and the event index. If event is declared in an unrecognized /// package, return error. pub async fn get_bridge_action_by_tx_digest_and_event_idx_maybe( @@ -792,7 +792,7 @@ mod tests { .await; let id_token_map = iota_client.get_token_id_map().await.unwrap(); - // 1. Create a Eth -> Iota Transfer (recipient is sender address), approve with + // 1. Create a Eth -> IOTA Transfer (recipient is sender address), approve with // validator secrets and assert its status to be Claimed let action = get_test_eth_to_iota_bridge_action(None, Some(usdc_amount), Some(sender), None); @@ -818,7 +818,7 @@ mod tests { .unwrap(); assert_eq!(status, BridgeActionStatus::Claimed); - // 2. Create a Iota -> Eth Transfer, approve with validator secrets and assert + // 2. Create an IOTA -> Eth Transfer, approve with validator secrets and assert // its status to be Approved // We need to actually send tokens to bridge to initialize the record. let eth_recv_address = EthAddress::random(); diff --git a/crates/iota-bridge/src/iota_mock_client.rs b/crates/iota-bridge/src/iota_mock_client.rs index acf4fa4fd27..c7738276cda 100644 --- a/crates/iota-bridge/src/iota_mock_client.rs +++ b/crates/iota-bridge/src/iota_mock_client.rs @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//! A mock implementation of Iota JSON-RPC client. +//! A mock implementation of IOTA JSON-RPC client. use std::{ collections::{HashMap, VecDeque}, diff --git a/crates/iota-bridge/src/iota_syncer.rs b/crates/iota-bridge/src/iota_syncer.rs index dfc7a40cb85..fbfaa506cdf 100644 --- a/crates/iota-bridge/src/iota_syncer.rs +++ b/crates/iota-bridge/src/iota_syncer.rs @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 //! The IotaSyncer module is responsible for synchronizing Events emitted -//! on Iota blockchain from concerned modules of bridge package 0x9. +//! on IOTA blockchain from concerned modules of bridge package 0x9. use std::{collections::HashMap, sync::Arc}; @@ -106,11 +106,11 @@ where events_sender .send((module.clone(), events.data)) .await - .expect("All Iota event channel receivers are closed"); + .expect("All IOTA event channel receivers are closed"); if let Some(next) = events.next_cursor { cursor = Some(next); } - tracing::info!(?module, ?cursor, "Observed {len} new Iota events"); + tracing::info!(?module, ?cursor, "Observed {len} new IOTA events"); } } } diff --git a/crates/iota-bridge/src/iota_transaction_builder.rs b/crates/iota-bridge/src/iota_transaction_builder.rs index 550f5d83d74..c39891ab2fc 100644 --- a/crates/iota-bridge/src/iota_transaction_builder.rs +++ b/crates/iota-bridge/src/iota_transaction_builder.rs @@ -80,7 +80,7 @@ pub fn build_iota_transaction( rgp, ), BridgeAction::EvmContractUpgradeAction(_) => { - // It does not need a Iota tranaction to execute EVM contract upgrade + // It does not need an IOTA tranaction to execute EVM contract upgrade unreachable!() } BridgeAction::AddTokensOnIotaAction(_) => build_add_tokens_on_iota_transaction( @@ -91,7 +91,7 @@ pub fn build_iota_transaction( rgp, ), BridgeAction::AddTokensOnEvmAction(_) => { - // It does not need a Iota tranaction to add tokens on EVM + // It does not need an IOTA tranaction to add tokens on EVM unreachable!() } } @@ -672,7 +672,7 @@ mod tests { .await; let id_token_map = iota_client.get_token_id_map().await.unwrap(); - // 1. Test Eth -> Iota Transfer approval + // 1. Test Eth -> IOTA Transfer approval let action = get_test_eth_to_iota_bridge_action(None, Some(usdc_amount), Some(sender), None); // `approve_action_with_validator_secrets` covers transaction building @@ -687,7 +687,7 @@ mod tests { .await .unwrap(); - // 2. Test Iota -> Eth Transfer approval + // 2. Test IOTA -> Eth Transfer approval let bridge_event = bridge_token( context, EthAddress::random(), diff --git a/crates/iota-bridge/src/main.rs b/crates/iota-bridge/src/main.rs index ee238ba4165..81ebf611b97 100644 --- a/crates/iota-bridge/src/main.rs +++ b/crates/iota-bridge/src/main.rs @@ -19,11 +19,9 @@ use tracing::info; bin_version::bin_version!(); #[derive(Parser)] -#[clap(rename_all = "kebab-case")] -#[clap(name = env!("CARGO_BIN_NAME"))] -#[clap(version = VERSION)] +#[command(rename_all = "kebab-case", name = env!("CARGO_BIN_NAME"), version = VERSION)] struct Args { - #[clap(long)] + #[arg(long)] pub config_path: PathBuf, } diff --git a/crates/iota-bridge/src/orchestrator.rs b/crates/iota-bridge/src/orchestrator.rs index 8604d0fe28d..c44e162f5bb 100644 --- a/crates/iota-bridge/src/orchestrator.rs +++ b/crates/iota-bridge/src/orchestrator.rs @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 //! `BridgeOrchestrator` is the component that: -//! 1. monitors Iota and Ethereum events with the help of `IotaSyncer` and +//! 1. monitors IOTA and Ethereum events with the help of `IotaSyncer` and //! `EthSyncer` //! 2. updates WAL table and cursor tables //! 2. hands actions to `BridgeExecutor` for execution @@ -117,7 +117,7 @@ where if events.is_empty() { continue; } - info!("Received {} Iota events: {:?}", events.len(), events); + info!("Received {} IOTA events: {:?}", events.len(), events); metrics .iota_watcher_received_events .inc_by(events.len() as u64); @@ -134,7 +134,7 @@ where } Err(e) => { panic!( - "Iota Event could not be deserialzed to IotaBridgeEvent: {:?}", + "IOTA Event could not be deserialzed to IotaBridgeEvent: {:?}", e ); } @@ -147,12 +147,12 @@ where if opt_bridge_event.is_none() { // TODO: we probably should not miss any events, log for now. metrics.iota_watcher_unrecognized_events.inc(); - error!("Iota event not recognized: {:?}", iota_event); + error!("IOTA event not recognized: {:?}", iota_event); continue; } // Unwrap safe: checked above let bridge_event: IotaBridgeEvent = opt_bridge_event.unwrap(); - info!("Observed Iota bridge event: {:?}", bridge_event); + info!("Observed IOTA bridge event: {:?}", bridge_event); // Send event to monitor monitor_tx @@ -169,7 +169,7 @@ where if !actions.is_empty() { info!( - "Received {} actions from Iota: {:?}", + "Received {} actions from IOTA: {:?}", actions.len(), actions ); @@ -193,7 +193,7 @@ where .update_iota_event_cursor(identifier, cursor) .expect("Store operation should not fail"); } - panic!("Iota event channel was closed unexpectedly"); + panic!("IOTA event channel was closed unexpectedly"); } async fn run_eth_watcher( diff --git a/crates/iota-bridge/src/server/handler.rs b/crates/iota-bridge/src/server/handler.rs index 951a6869b73..b1292bf3e44 100644 --- a/crates/iota-bridge/src/server/handler.rs +++ b/crates/iota-bridge/src/server/handler.rs @@ -28,7 +28,7 @@ use crate::{ #[async_trait] pub trait BridgeRequestHandlerTrait { /// Handles a request to sign a BridgeAction that bridges assets - /// from Ethereum to Iota. The inputs are a transaction hash on Ethereum + /// from Ethereum to IOTA. The inputs are a transaction hash on Ethereum /// that emitted the bridge event and the Event index in that transaction async fn handle_eth_tx_hash( &self, @@ -36,7 +36,7 @@ pub trait BridgeRequestHandlerTrait { event_idx: u16, ) -> Result, BridgeError>; /// Handles a request to sign a BridgeAction that bridges assets - /// from Iota to Ethereum. The inputs are a transaction digest on Iota + /// from IOTA to Ethereum. The inputs are a transaction digest on IOTA /// that emitted the bridge event and the Event index in that transaction async fn handle_iota_tx_digest( &self, @@ -80,7 +80,7 @@ where self.iota_client .get_bridge_action_by_tx_digest_and_event_idx_maybe(&tx_digest, event_idx) .await - .tap_ok(|action| info!("Iota action found: {:?}", action)) + .tap_ok(|action| info!("IOTA action found: {:?}", action)) } } diff --git a/crates/iota-bridge/src/server/mod.rs b/crates/iota-bridge/src/server/mod.rs index 2085b4cc19e..a6affca3caf 100644 --- a/crates/iota-bridge/src/server/mod.rs +++ b/crates/iota-bridge/src/server/mod.rs @@ -424,7 +424,7 @@ async fn handle_add_tokens_on_iota( if !chain_id.is_iota_chain() { return Err(BridgeError::InvalidBridgeClientRequest( - "handle_add_tokens_on_iota only expects Iota chain id".to_string(), + "handle_add_tokens_on_iota only expects IOTA chain id".to_string(), )); } @@ -505,7 +505,7 @@ async fn handle_add_tokens_on_evm( })?; if chain_id.is_iota_chain() { return Err(BridgeError::InvalidBridgeClientRequest( - "handle_add_tokens_on_evm does not expect Iota chain id".to_string(), + "handle_add_tokens_on_evm does not expect IOTA chain id".to_string(), )); } diff --git a/crates/iota-bridge/src/test_utils.rs b/crates/iota-bridge/src/test_utils.rs index 5a4bda4f769..00a02d95d64 100644 --- a/crates/iota-bridge/src/test_utils.rs +++ b/crates/iota-bridge/src/test_utils.rs @@ -336,7 +336,7 @@ pub fn get_certified_action_with_validator_secrets( /// Approve a bridge action with the given validator secrets. Return the /// newly created token object reference if `expected_token_receiver` is Some -/// (only relevant when the action is eth -> Iota transfer), +/// (only relevant when the action is eth -> IOTA transfer), /// Otherwise return None. /// Note: for iota -> eth transfers, the actual deposit needs to be recorded. /// Use `bridge_token` to do it. diff --git a/crates/iota-bridge/src/types.rs b/crates/iota-bridge/src/types.rs index 334a679ef02..2b1e669a1b1 100644 --- a/crates/iota-bridge/src/types.rs +++ b/crates/iota-bridge/src/types.rs @@ -343,7 +343,7 @@ pub struct AddTokensOnEvmAction { #[derive(Debug, Clone, Serialize, Deserialize, PartialEq, Eq, Hash)] #[enum_dispatch(BridgeMessageEncoding)] pub enum BridgeAction { - /// Iota to Eth bridge action + /// IOTA to Eth bridge action IotaToEthBridgeAction(IotaToEthBridgeAction), /// Eth to iota bridge action EthToIotaBridgeAction(EthToIotaBridgeAction), diff --git a/crates/iota-bridge/src/utils.rs b/crates/iota-bridge/src/utils.rs index 91b9549ddfe..4f342e6c4ff 100644 --- a/crates/iota-bridge/src/utils.rs +++ b/crates/iota-bridge/src/utils.rs @@ -67,7 +67,7 @@ pub fn generate_bridge_authority_key_and_write_to_file( ); let iota_address = IotaAddress::from(&kp.public); println!( - "Corresponding Iota address by this ecdsa key: {:?}", + "Corresponding IOTA address by this ecdsa key: {:?}", iota_address ); let base64_encoded = kp.encode_base64(); @@ -94,7 +94,7 @@ pub fn generate_bridge_client_key_and_write_to_file( IotaKeyPair::from(kp) }; let iota_address = IotaAddress::from(&kp.public()); - println!("Corresponding Iota address by this key: {:?}", iota_address); + println!("Corresponding IOTA address by this key: {:?}", iota_address); let contents = kp.encode_base64(); std::fs::write(path, contents) @@ -168,7 +168,7 @@ pub fn examine_key(path: &PathBuf, is_validator_key: bool) -> Result<(), anyhow: kp.public().as_bytes().to_vec() } }; - println!("Corresponding Iota address: {:?}", iota_address); + println!("Corresponding IOTA address: {:?}", iota_address); println!("Corresponding PublicKey: {:?}", Hex::encode(pubkey)); Ok(()) } diff --git a/crates/iota-cluster-test/src/cluster.rs b/crates/iota-cluster-test/src/cluster.rs index 3c1d993f992..14375a8ae6c 100644 --- a/crates/iota-cluster-test/src/cluster.rs +++ b/crates/iota-cluster-test/src/cluster.rs @@ -34,16 +34,10 @@ use tracing::info; use super::config::{ClusterTestOpt, Env}; -const DEVNET_FAUCET_ADDR: &str = "https://faucet.devnet.iota.io:443"; -const STAGING_FAUCET_ADDR: &str = "https://faucet.staging.iota.io:443"; -const CONTINUOUS_FAUCET_ADDR: &str = "https://faucet.ci.iota.io:443"; -const CONTINUOUS_NOMAD_FAUCET_ADDR: &str = "https://faucet.nomad.ci.iota.io:443"; -const TESTNET_FAUCET_ADDR: &str = "https://faucet.testnet.iota.io:443"; -const DEVNET_FULLNODE_ADDR: &str = "https://rpc.devnet.iota.io:443"; -const STAGING_FULLNODE_ADDR: &str = "https://fullnode.staging.iota.io:443"; -const CONTINUOUS_FULLNODE_ADDR: &str = "https://fullnode.ci.iota.io:443"; -const CONTINUOUS_NOMAD_FULLNODE_ADDR: &str = "https://fullnode.nomad.ci.iota.io:443"; -const TESTNET_FULLNODE_ADDR: &str = "https://fullnode.testnet.iota.io:443"; +const DEVNET_FAUCET_ADDR: &str = "https://faucet.devnet.iota.cafe:443"; +const TESTNET_FAUCET_ADDR: &str = "https://faucet.testnet.iota.cafe:443"; +const DEVNET_FULLNODE_ADDR: &str = "https://api.devnet.iota.cafe:443"; +const TESTNET_FULLNODE_ADDR: &str = "https://api.testnet.iota.cafe:443"; pub struct ClusterFactory; @@ -94,18 +88,6 @@ impl Cluster for RemoteRunningCluster { String::from(DEVNET_FULLNODE_ADDR), String::from(DEVNET_FAUCET_ADDR), ), - Env::Staging => ( - String::from(STAGING_FULLNODE_ADDR), - String::from(STAGING_FAUCET_ADDR), - ), - Env::Ci => ( - String::from(CONTINUOUS_FULLNODE_ADDR), - String::from(CONTINUOUS_FAUCET_ADDR), - ), - Env::CiNomad => ( - String::from(CONTINUOUS_NOMAD_FULLNODE_ADDR), - String::from(CONTINUOUS_NOMAD_FAUCET_ADDR), - ), Env::Testnet => ( String::from(TESTNET_FULLNODE_ADDR), String::from(TESTNET_FAUCET_ADDR), @@ -195,7 +177,7 @@ impl Cluster for LocalNewCluster { // Check if we already have a config directory that is passed if let Some(config_dir) = options.config_dir.clone() { assert!(options.epoch_duration_ms.is_none()); - // Load the config of the Iota authority. + // Load the config of the IOTA authority. let network_config_path = config_dir.join(IOTA_NETWORK_CONFIG); let NetworkConfigLight { validator_configs, @@ -203,7 +185,7 @@ impl Cluster for LocalNewCluster { committee_with_network: _, } = PersistedConfig::read(&network_config_path).map_err(|err| { err.context(format!( - "Cannot open Iota network config file at {:?}", + "Cannot open IOTA network config file at {:?}", network_config_path )) })?; diff --git a/crates/iota-cluster-test/src/config.rs b/crates/iota-cluster-test/src/config.rs index 85814590fab..1877c5999e4 100644 --- a/crates/iota-cluster-test/src/config.rs +++ b/crates/iota-cluster-test/src/config.rs @@ -11,44 +11,39 @@ use regex::Regex; #[derive(Parser, Clone, ValueEnum, Debug)] pub enum Env { Devnet, - Staging, - Ci, - CiNomad, Testnet, CustomRemote, NewLocal, } #[derive(derive_more::Debug, Parser)] -#[clap(name = "", rename_all = "kebab-case")] +#[command(name = "", rename_all = "kebab-case")] pub struct ClusterTestOpt { - #[clap(value_enum)] + #[arg(value_enum)] pub env: Env, - #[clap(long)] + #[arg(long)] pub faucet_address: Option, - #[clap(long)] + #[arg(long)] pub fullnode_address: Option, - #[clap(long)] + #[arg(long)] pub epoch_duration_ms: Option, /// URL for the indexer RPC server - #[clap(long)] + #[arg(long)] pub indexer_address: Option, /// URL for the Indexer Postgres DB - #[clap(long)] + #[arg(long)] #[debug("{}", ObfuscatedPgAddress(pg_address))] pub pg_address: Option, - #[clap(long)] + #[arg(long)] pub config_dir: Option, /// URL for the indexer RPC server - #[clap(long)] + #[arg(long)] pub graphql_address: Option, /// Locations for local migration snapshots. - #[clap(long, name = "path")] - #[arg(num_args(0..))] + #[arg(long, name = "path", num_args(0..))] pub local_migration_snapshots: Vec, /// Remote migration snapshots. - #[clap(long, name = "iota|")] - #[arg(num_args(0..))] + #[arg(long, name = "iota|", num_args(0..))] pub remote_migration_snapshots: Vec, } diff --git a/crates/iota-cluster-test/src/helper.rs b/crates/iota-cluster-test/src/helper.rs index c25a6854e8e..2d60b4683f3 100644 --- a/crates/iota-cluster-test/src/helper.rs +++ b/crates/iota-cluster-test/src/helper.rs @@ -12,7 +12,7 @@ use iota_types::{ use move_core_types::language_storage::TypeTag; use tracing::{debug, trace}; -/// A util struct that helps verify Iota Object. +/// A util struct that helps verify IOTA Object. /// Use builder style to construct the conditions. /// When optionals fields are not set, related checks are omitted. /// Consuming functions such as `check` perform the check and panics if diff --git a/crates/iota-cluster-test/src/lib.rs b/crates/iota-cluster-test/src/lib.rs index 855ab846157..89a55318af0 100644 --- a/crates/iota-cluster-test/src/lib.rs +++ b/crates/iota-cluster-test/src/lib.rs @@ -79,7 +79,7 @@ impl TestContext { if gas_coins.len() < minimum_coins { panic!( - "Expect to get at least {minimum_coins} Iota Coins for address {addr}, but only got {}", + "Expect to get at least {minimum_coins} IOTA Coins for address {addr}, but only got {}", gas_coins.len() ) } diff --git a/crates/iota-common/src/sync/mod.rs b/crates/iota-common/src/sync/mod.rs index b474955f390..a5a1ba10fb2 100644 --- a/crates/iota-common/src/sync/mod.rs +++ b/crates/iota-common/src/sync/mod.rs @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -/// Low level utilities shared across Iota. +/// Low level utilities shared across IOTA. pub mod async_once_cell; pub mod notify_once; pub mod notify_read; diff --git a/crates/iota-config/data/fullnode-template-with-path.yaml b/crates/iota-config/data/fullnode-template-with-path.yaml index 12854a33135..dc95921b7c9 100644 --- a/crates/iota-config/data/fullnode-template-with-path.yaml +++ b/crates/iota-config/data/fullnode-template-with-path.yaml @@ -1,4 +1,4 @@ -# Update this value to the location you want Iota to store its database +# Update this value to the location you want IOTA to store its database db-path: "iotadb" # For ipv4, update this to "/ipv4/X.X.X.X/tcp/8080/http" diff --git a/crates/iota-config/data/fullnode-template.yaml b/crates/iota-config/data/fullnode-template.yaml index b161d62372b..9aa48c4e7ba 100644 --- a/crates/iota-config/data/fullnode-template.yaml +++ b/crates/iota-config/data/fullnode-template.yaml @@ -1,4 +1,4 @@ -# Update this value to the location you want Iota to store its database +# Update this value to the location you want IOTA to store its database db-path: "/opt/iota/db" # For ipv4, update this to "/ipv4/X.X.X.X/tcp/8080/http" diff --git a/crates/iota-config/src/genesis.rs b/crates/iota-config/src/genesis.rs index d7c850ca148..5be816f64ce 100644 --- a/crates/iota-config/src/genesis.rs +++ b/crates/iota-config/src/genesis.rs @@ -159,7 +159,7 @@ impl Genesis { pub fn iota_system_wrapper_object(&self) -> IotaSystemStateWrapper { get_iota_system_state_wrapper(&self.objects()) - .expect("Iota System State Wrapper object must always exist") + .expect("IOTA System State Wrapper object must always exist") } pub fn contains_migrations(&self) -> bool { @@ -167,7 +167,7 @@ impl Genesis { } pub fn iota_system_object(&self) -> IotaSystemState { - get_iota_system_state(&self.objects()).expect("Iota System State object must always exist") + get_iota_system_state(&self.objects()).expect("IOTA System State object must always exist") } pub fn clock(&self) -> Clock { @@ -321,11 +321,11 @@ impl UnsignedGenesis { pub fn iota_system_wrapper_object(&self) -> IotaSystemStateWrapper { get_iota_system_state_wrapper(&self.objects()) - .expect("Iota System State Wrapper object must always exist") + .expect("IOTA System State Wrapper object must always exist") } pub fn iota_system_object(&self) -> IotaSystemState { - get_iota_system_state(&self.objects()).expect("Iota System State object must always exist") + get_iota_system_state(&self.objects()).expect("IOTA System State object must always exist") } pub fn authenticator_state_object(&self) -> Option { @@ -454,7 +454,7 @@ impl TokenDistributionSchedule { for allocation in &self.allocations { total_nanos = total_nanos .checked_add(allocation.amount_nanos) - .expect("TokenDistributionSchedule allocates more than the maximum supply which equals u64::MAX", ); + .expect("TokenDistributionSchedule allocates more than the maximum supply which equals u64::MAX"); } } @@ -516,12 +516,7 @@ impl TokenDistributionSchedule { /// Helper to read a TokenDistributionSchedule from a csv file. /// /// The file is encoded such that the final entry in the CSV file is used to - /// denote the allocation to the stake subsidy fund. It must be in the - /// following format: - /// `0x0000000000000000000000000000000000000000000000000000000000000000, - ///
minted supply
,` - /// - /// All entries in a token distribution schedule must add up to 10B Iota. + /// denote the allocation to the stake subsidy fund. pub fn from_csv(reader: R) -> Result { let mut reader = csv::Reader::from_reader(reader); let mut allocations: Vec = @@ -568,7 +563,17 @@ impl TokenDistributionSchedule { #[derive(Clone, Debug, PartialEq, Eq, Serialize, Deserialize)] #[serde(rename_all = "kebab-case")] pub struct TokenAllocation { + /// Indicates the address that owns the tokens. It means that this + /// `TokenAllocation` can serve to stake some funds to the + /// `staked_with_validator` during genesis, but it's the `recipient_address` + /// which will receive the associated StakedIota (or TimelockedStakedIota) + /// object. pub recipient_address: IotaAddress, + /// Indicates an amount of nanos that is: + /// - minted for the `recipient_address` and staked to a validator, only in + /// the case `staked_with_validator` is Some + /// - minted for the `recipient_address` and transferred that address, + /// otherwise. pub amount_nanos: u64, /// Indicates if this allocation should be staked at genesis and with which @@ -628,3 +633,117 @@ impl TokenDistributionScheduleBuilder { schedule } } + +/// Represents the allocation of stake and gas payment to a validator. +#[derive(Clone, Copy, Debug, PartialEq, Eq, Serialize, Deserialize)] +#[serde(rename_all = "kebab-case")] +pub struct ValidatorAllocation { + /// The validator address receiving the stake and/or gas payment + pub validator: IotaAddress, + /// The amount of nanos to stake to the validator + pub amount_nanos_to_stake: u64, + /// The amount of nanos to transfer as gas payment to the validator + pub amount_nanos_to_pay_gas: u64, +} + +/// Represents a delegation of stake and gas payment to a validator, +/// coming from a delegator. This struct is used to serialize and deserialize +/// delegations to and from a csv file. +#[derive(Clone, Debug, PartialEq, Eq, Serialize, Deserialize)] +#[serde(rename_all = "kebab-case")] +pub struct Delegation { + /// The address from which to take the nanos for staking/gas + pub delegator: IotaAddress, + /// The allocation to a validator receiving a stake and/or a gas payment + #[serde(flatten)] + pub validator_allocation: ValidatorAllocation, +} + +/// Represents genesis delegations to validators. +/// +/// This struct maps a delegator address to a list of validators and their +/// stake and gas allocations. Each ValidatorAllocation contains the address of +/// a validator that will receive an amount of nanos to stake and an amount as +/// gas payment. +#[derive(Clone, Debug, Default, PartialEq, Eq, Serialize, Deserialize)] +#[serde(rename_all = "kebab-case")] +pub struct Delegations { + pub allocations: HashMap>, +} + +impl Delegations { + pub fn new_for_validators_with_default_allocation( + validators: impl IntoIterator, + delegator: IotaAddress, + ) -> Self { + let validator_allocations = validators + .into_iter() + .map(|address| ValidatorAllocation { + validator: address, + amount_nanos_to_stake: iota_types::governance::MIN_VALIDATOR_JOINING_STAKE_NANOS, + amount_nanos_to_pay_gas: 0, + }) + .collect(); + + let mut allocations = HashMap::new(); + allocations.insert(delegator, validator_allocations); + + Self { allocations } + } + + /// Helper to read a Delegations struct from a csv file. + /// + /// The file is encoded such that the final entry in the CSV file is used to + /// denote the allocation coming from a delegator. It must be in the + /// following format: + /// `delegator,validator,amount-nanos-to-stake,amount-nanos-to-pay-gas + /// ,,2000000000000000,5000000000 + /// ,,3000000000000000,5000000000 + /// ,,4500000000000000,5000000000` + pub fn from_csv(reader: R) -> Result { + let mut reader = csv::Reader::from_reader(reader); + + let mut delegations = Self::default(); + for delegation in reader.deserialize::() { + let delegation = delegation?; + delegations + .allocations + .entry(delegation.delegator) + .or_default() + .push(delegation.validator_allocation); + } + + Ok(delegations) + } + + /// Helper to write a Delegations struct into a csv file. + /// + /// It writes in the following format: + /// `delegator,validator,amount-nanos-to-stake,amount-nanos-to-pay-gas + /// ,,2000000000000000,5000000000 + /// ,,3000000000000000,5000000000 + /// ,,4500000000000000,5000000000` + pub fn to_csv(&self, writer: W) -> Result<()> { + let mut writer = csv::Writer::from_writer(writer); + + writer.write_record([ + "delegator", + "validator", + "amount-nanos-to-stake", + "amount-nanos-to-pay-gas", + ])?; + + for (&delegator, validator_allocations) in &self.allocations { + for validator_allocation in validator_allocations { + writer.write_record(&[ + delegator.to_string(), + validator_allocation.validator.to_string(), + validator_allocation.amount_nanos_to_stake.to_string(), + validator_allocation.amount_nanos_to_pay_gas.to_string(), + ])?; + } + } + + Ok(()) + } +} diff --git a/crates/iota-config/src/lib.rs b/crates/iota-config/src/lib.rs index bdb16c8865f..b098549321a 100644 --- a/crates/iota-config/src/lib.rs +++ b/crates/iota-config/src/lib.rs @@ -35,7 +35,7 @@ pub const IOTA_KEYSTORE_ALIASES_FILENAME: &str = "iota.aliases"; pub const IOTA_BENCHMARK_GENESIS_GAS_KEYSTORE_FILENAME: &str = "benchmark.keystore"; pub const IOTA_GENESIS_FILENAME: &str = "genesis.blob"; pub const IOTA_GENESIS_MIGRATION_TX_DATA_FILENAME: &str = "migration.blob"; -pub const IOTA_DEV_NET_URL: &str = "https://fullnode.devnet.iota.io:443"; +pub const IOTA_DEV_NET_URL: &str = "https://api.devnet.iota.cafe:443"; pub const AUTHORITIES_DB_NAME: &str = "authorities_db"; pub const CONSENSUS_DB_NAME: &str = "consensus_db"; diff --git a/crates/iota-config/src/node.rs b/crates/iota-config/src/node.rs index ddf63329504..441daddea72 100644 --- a/crates/iota-config/src/node.rs +++ b/crates/iota-config/src/node.rs @@ -151,6 +151,9 @@ pub struct NodeConfig { #[serde(default)] pub checkpoint_executor_config: CheckpointExecutorConfig, + #[serde(skip_serializing_if = "Option::is_none")] + pub metrics: Option, + /// In a `iota-node` binary, this is set to /// SupportedProtocolVersions::SYSTEM_DEFAULT in iota-node/src/main.rs. /// It is present in the config so that it can be changed by tests in @@ -308,7 +311,7 @@ pub fn default_zklogin_oauth_providers() -> BTreeMap> { fn default_transaction_kv_store_config() -> TransactionKeyValueStoreReadConfig { TransactionKeyValueStoreReadConfig { - base_url: "https://transactions.iota.io/".to_string(), + base_url: "https://transactions.iota.cafe/".to_string(), } } @@ -757,6 +760,15 @@ impl AuthorityStorePruningConfig { } } +#[derive(Debug, Clone, Deserialize, Serialize)] +#[serde(rename_all = "kebab-case")] +pub struct MetricsConfig { + #[serde(skip_serializing_if = "Option::is_none")] + pub push_interval_seconds: Option, + #[serde(skip_serializing_if = "Option::is_none")] + pub push_url: Option, +} + #[derive(Default, Debug, Clone, Deserialize, Serialize)] #[serde(rename_all = "kebab-case")] pub struct DBCheckpointConfig { diff --git a/crates/iota-config/src/p2p.rs b/crates/iota-config/src/p2p.rs index 58a43c224b0..21fa661c274 100644 --- a/crates/iota-config/src/p2p.rs +++ b/crates/iota-config/src/p2p.rs @@ -321,7 +321,7 @@ pub struct DiscoveryConfig { #[serde(skip_serializing_if = "Option::is_none")] pub access_type: Option, - /// Like `seed_peers` in `P2pConfig`, allowlisted peers will awlays be + /// Like `seed_peers` in `P2pConfig`, allowlisted peers will always be /// allowed to establish connection with this node regardless of the /// concurrency limit. Unlike `seed_peers`, a node does not reach out to /// `allowlisted_peers` preferentially. It is also used to determine if diff --git a/crates/iota-core/src/authority.rs b/crates/iota-core/src/authority.rs index d85c217974f..462adc8ca5a 100644 --- a/crates/iota-core/src/authority.rs +++ b/crates/iota-core/src/authority.rs @@ -847,6 +847,9 @@ impl AuthorityState { transaction: VerifiedTransaction, epoch_store: &Arc, ) -> IotaResult { + // Ensure that validator cannot reconfigure while we are signing the tx + let _execution_lock = self.execution_lock_for_signing().await; + let tx_digest = transaction.digest(); let tx_data = transaction.data().transaction_data(); @@ -930,16 +933,6 @@ impl AuthorityState { .start_timer(); self.metrics.tx_orders.inc(); - // The should_accept_user_certs check here is best effort, because - // between a validator signs a tx and a cert is formed, the validator - // could close the window. - if !epoch_store - .get_reconfig_state_read_lock_guard() - .should_accept_user_certs() - { - return Err(IotaError::ValidatorHaltedAtEpochEnd); - } - let signed = self.handle_transaction_impl(transaction, epoch_store).await; match signed { Ok(s) => { @@ -1593,7 +1586,7 @@ impl AuthorityState { let transaction_data = &certificate.data().intent_message().value; let (kind, signer, gas) = transaction_data.execution_parts(); - #[expect(unused_mut)] + #[cfg_attr(not(any(msim, fail_points)), expect(unused_mut))] let (inner_temp_store, _, mut effects, execution_error_opt) = epoch_store.executor().execute_transaction_to_effects( self.get_backing_store().as_ref(), @@ -2862,6 +2855,15 @@ impl AuthorityState { } } + /// Acquires the execution lock for the duration of a transaction signing + /// request. This prevents reconfiguration from starting until we are + /// finished handling the signing request. Otherwise, in-memory lock + /// state could be cleared (by `ObjectLocks::clear_cached_locks`) + /// while we are attempting to acquire locks for the transaction. + pub async fn execution_lock_for_signing(&self) -> ExecutionLockReadGuard { + self.execution_lock.read().await + } + pub async fn execution_lock_for_reconfiguration(&self) -> ExecutionLockWriteGuard { self.execution_lock.write().await } diff --git a/crates/iota-core/src/authority/authority_per_epoch_store.rs b/crates/iota-core/src/authority/authority_per_epoch_store.rs index cf080a649d9..01879a00b6d 100644 --- a/crates/iota-core/src/authority/authority_per_epoch_store.rs +++ b/crates/iota-core/src/authority/authority_per_epoch_store.rs @@ -572,8 +572,9 @@ pub struct AuthorityEpochTables { /// Transactions that are being deferred until some future time deferred_transactions: DBMap>, - /// Tables for recording state for RandomnessManager. + // Tables for recording state for RandomnessManager. + // /// Records messages processed from other nodes. Updated when receiving a /// new dkg::Message via consensus. pub(crate) dkg_processed_messages: DBMap, @@ -1358,8 +1359,8 @@ impl AuthorityPerEpochStore { Ok(result) } - /// `pending_certificates` table related methods. Should only be used from - /// TransactionManager. + // `pending_certificates` table related methods. + // Should only be used from TransactionManager. /// Gets all pending certificates. Used during recovery. pub fn all_pending_execution(&self) -> IotaResult> { diff --git a/crates/iota-core/src/authority/authority_store.rs b/crates/iota-core/src/authority/authority_store.rs index 2ae0de3fb71..9e0f9d31a2f 100644 --- a/crates/iota-core/src/authority/authority_store.rs +++ b/crates/iota-core/src/authority/authority_store.rs @@ -163,7 +163,7 @@ impl AuthorityStore { let epoch_start_configuration = if perpetual_tables.database_is_empty()? { info!("Creating new epoch start config from genesis"); - #[expect(unused_mut)] + #[cfg_attr(not(any(msim, fail_points)), expect(unused_mut))] let mut initial_epoch_flags = EpochFlag::default_flags_for_new_epoch(config); fail_point_arg!("initial_epoch_flags", |flags: Vec| { info!("Setting initial epoch flags to {:?}", flags); diff --git a/crates/iota-core/src/authority/shared_object_congestion_tracker.rs b/crates/iota-core/src/authority/shared_object_congestion_tracker.rs index 75b9a4c7053..6117b8eaefd 100644 --- a/crates/iota-core/src/authority/shared_object_congestion_tracker.rs +++ b/crates/iota-core/src/authority/shared_object_congestion_tracker.rs @@ -93,7 +93,8 @@ impl SharedObjectCongestionTracker { } let start_cost = self.compute_tx_start_at_cost(&shared_input_objects); - if start_cost + tx_cost <= max_accumulated_txn_cost_per_object_in_commit { + let (end_cost, cost_overflow) = start_cost.overflowing_add(tx_cost); + if !cost_overflow && end_cost <= max_accumulated_txn_cost_per_object_in_commit { return None; } @@ -148,12 +149,12 @@ impl SharedObjectCongestionTracker { let shared_input_objects: Vec<_> = cert.shared_input_objects().collect(); let start_cost = self.compute_tx_start_at_cost(&shared_input_objects); - let end_cost = start_cost + tx_cost; + let end_cost = start_cost.saturating_add(tx_cost); for obj in shared_input_objects { if obj.mutable { let old_end_cost = self.object_execution_cost.insert(obj.id, end_cost); - assert!(old_end_cost.is_none() || old_end_cost.unwrap() < end_cost); + assert!(old_end_cost.is_none() || old_end_cost.unwrap() <= end_cost); } } } @@ -556,4 +557,158 @@ mod object_cost_tests { expected_object_cost ); } + + #[test] + fn test_cost_overflow() { + let object_id_0 = ObjectID::random(); + let object_id_1 = ObjectID::random(); + let object_id_2 = ObjectID::random(); + // edge case: max value is saturated + let max_accumulated_txn_cost_per_object_in_commit = u64::MAX; + + // case 1: large initial cost, small tx cost + let mut shared_object_congestion_tracker = + SharedObjectCongestionTracker::new_with_initial_value_for_test( + &[(object_id_0, u64::MAX - 1), (object_id_1, u64::MAX - 1)], + PerObjectCongestionControlMode::TotalGasBudget, + ); + + let tx = build_transaction(&[(object_id_0, true)], 1); + assert!( + shared_object_congestion_tracker + .should_defer_due_to_object_congestion( + &tx, + max_accumulated_txn_cost_per_object_in_commit, + &HashMap::new(), + 0, + ) + .is_none(), + "objects are not yet congested" + ); + shared_object_congestion_tracker.bump_object_execution_cost(&tx); + assert_eq!( + shared_object_congestion_tracker, + SharedObjectCongestionTracker::new_with_initial_value_for_test( + &[(object_id_0, u64::MAX), (object_id_1, u64::MAX - 1),], + PerObjectCongestionControlMode::TotalGasBudget + ) + ); + + let tx = build_transaction(&[(object_id_0, true), (object_id_1, true)], 1); + if let Some((_, congested_objects)) = shared_object_congestion_tracker + .should_defer_due_to_object_congestion( + &tx, + max_accumulated_txn_cost_per_object_in_commit, + &HashMap::new(), + 0, + ) + { + assert_eq!(congested_objects.len(), 1); + assert_eq!(congested_objects[0], object_id_0); + } else { + panic!("object 0 is congesting, should defer"); + } + shared_object_congestion_tracker.bump_object_execution_cost(&tx); + assert_eq!( + shared_object_congestion_tracker, + SharedObjectCongestionTracker::new_with_initial_value_for_test( + &[(object_id_0, u64::MAX), (object_id_1, u64::MAX),], + PerObjectCongestionControlMode::TotalGasBudget + ) + ); + + if let Some((_, congested_objects)) = shared_object_congestion_tracker + .should_defer_due_to_object_congestion( + &tx, + max_accumulated_txn_cost_per_object_in_commit, + &HashMap::new(), + 0, + ) + { + assert_eq!(congested_objects.len(), 2); + assert_eq!(congested_objects[0], object_id_0); + assert_eq!(congested_objects[1], object_id_1); + } else { + panic!("objects 0 and 1 are congesting, should defer"); + } + shared_object_congestion_tracker.bump_object_execution_cost(&tx); + assert_eq!( + shared_object_congestion_tracker, + SharedObjectCongestionTracker::new_with_initial_value_for_test( + &[(object_id_0, u64::MAX), (object_id_1, u64::MAX),], + PerObjectCongestionControlMode::TotalGasBudget + ) + ); + + // case 2: small initial cost, large tx cost + let mut shared_object_congestion_tracker = + SharedObjectCongestionTracker::new_with_initial_value_for_test( + &[(object_id_0, 0), (object_id_1, 1), (object_id_2, 2)], + PerObjectCongestionControlMode::TotalGasBudget, + ); + + let tx = build_transaction( + &[ + (object_id_0, true), + (object_id_1, true), + (object_id_2, true), + ], + u64::MAX - 1, + ); + if let Some((_, congested_objects)) = shared_object_congestion_tracker + .should_defer_due_to_object_congestion( + &tx, + max_accumulated_txn_cost_per_object_in_commit, + &HashMap::new(), + 0, + ) + { + assert_eq!(congested_objects.len(), 1); + assert_eq!(congested_objects[0], object_id_2); + } else { + panic!("case 2: object 2 is congested, should defer"); + } + shared_object_congestion_tracker.bump_object_execution_cost(&tx); + assert_eq!( + shared_object_congestion_tracker, + SharedObjectCongestionTracker::new_with_initial_value_for_test( + &[ + (object_id_0, u64::MAX), + (object_id_1, u64::MAX), + (object_id_2, u64::MAX), + ], + PerObjectCongestionControlMode::TotalGasBudget + ) + ); + + // case 3: max initial cost, max tx cost + let mut shared_object_congestion_tracker = + SharedObjectCongestionTracker::new_with_initial_value_for_test( + &[(object_id_0, u64::MAX)], + PerObjectCongestionControlMode::TotalGasBudget, + ); + + let tx = build_transaction(&[(object_id_0, true)], u64::MAX); + if let Some((_, congested_objects)) = shared_object_congestion_tracker + .should_defer_due_to_object_congestion( + &tx, + max_accumulated_txn_cost_per_object_in_commit, + &HashMap::new(), + 0, + ) + { + assert_eq!(congested_objects.len(), 1); + assert_eq!(congested_objects[0], object_id_0); + } else { + panic!("case 3: object 0 is congested, should defer"); + } + shared_object_congestion_tracker.bump_object_execution_cost(&tx); + assert_eq!( + shared_object_congestion_tracker, + SharedObjectCongestionTracker::new_with_initial_value_for_test( + &[(object_id_0, u64::MAX),], + PerObjectCongestionControlMode::TotalGasBudget + ) + ); + } } diff --git a/crates/iota-core/src/authority_aggregator.rs b/crates/iota-core/src/authority_aggregator.rs index f42945e4990..42934a53e1b 100644 --- a/crates/iota-core/src/authority_aggregator.rs +++ b/crates/iota-core/src/authority_aggregator.rs @@ -516,7 +516,7 @@ impl ProcessTransactionResult { /// the validators and determining the final state of the transaction. #[derive(Clone)] pub struct AuthorityAggregator { - /// Our Iota committee. + /// Our IOTA committee. pub committee: Arc, /// For more human readable metrics reporting. /// It's OK for this map to be empty or missing validators, it then defaults @@ -1216,11 +1216,7 @@ where } /// Records the rpc error if it is. - fn record_rpc_error_maybe( - metrics: Arc, - display_name: &String, - error: &IotaError, - ) { + fn record_rpc_error_maybe(metrics: Arc, display_name: &str, error: &IotaError) { if let IotaError::Rpc(_message, code) = error { metrics .total_rpc_err @@ -1905,7 +1901,7 @@ where } /// `AuthorityAggregatorBuilder` is used to build an `AuthorityAggregator` with -/// customizable configurations for the Iota network. +/// customizable configurations for the IOTA network. #[derive(Default)] pub struct AuthorityAggregatorBuilder<'a> { network_config: Option<&'a NetworkConfig>, diff --git a/crates/iota-core/src/authority_server.rs b/crates/iota-core/src/authority_server.rs index 75ef3f41997..98f11887069 100644 --- a/crates/iota-core/src/authority_server.rs +++ b/crates/iota-core/src/authority_server.rs @@ -819,8 +819,8 @@ impl ValidatorService { }; let request = request.into_inner(); - let certificates = NonEmpty::from_vec(request.certificates) - .ok_or_else(|| IotaError::NoCertificateProvided)?; + let certificates = + NonEmpty::from_vec(request.certificates).ok_or(IotaError::NoCertificateProvided)?; for certificate in &certificates { // We need to check this first because we haven't verified the cert signature. certificate.validity_check(epoch_store.protocol_config(), epoch_store.epoch())?; diff --git a/crates/iota-core/src/checkpoints/checkpoint_executor/mod.rs b/crates/iota-core/src/checkpoints/checkpoint_executor/mod.rs index d1266c66400..b97e5644c1f 100644 --- a/crates/iota-core/src/checkpoints/checkpoint_executor/mod.rs +++ b/crates/iota-core/src/checkpoints/checkpoint_executor/mod.rs @@ -201,7 +201,7 @@ impl CheckpointExecutor { // check if we want to run this epoch based on RunWithRange condition value // we want to be inclusive of the defined RunWithRangeEpoch::Epoch // i.e Epoch(N) means we will execute epoch N and stop when reaching N+1 - if run_with_range.map_or(false, |rwr| rwr.is_epoch_gt(epoch_store.epoch())) { + if run_with_range.is_some_and(|rwr| rwr.is_epoch_gt(epoch_store.epoch())) { info!( "RunWithRange condition satisfied at {:?}, run_epoch={:?}", run_with_range, @@ -315,7 +315,7 @@ impl CheckpointExecutor { now_transaction_num = current_transaction_num; } // we want to be inclusive of checkpoints in RunWithRange::Checkpoint type - if run_with_range.map_or(false, |rwr| rwr.matches_checkpoint(checkpoint.sequence_number)) { + if run_with_range.is_some_and(|rwr| rwr.matches_checkpoint(checkpoint.sequence_number)) { info!( "RunWithRange condition satisfied after checkpoint sequence number {:?}", checkpoint.sequence_number diff --git a/crates/iota-core/src/consensus_adapter.rs b/crates/iota-core/src/consensus_adapter.rs index eb25d7b5137..5e68b3db330 100644 --- a/crates/iota-core/src/consensus_adapter.rs +++ b/crates/iota-core/src/consensus_adapter.rs @@ -186,7 +186,7 @@ pub trait SubmitToConsensus: Sync + Send + 'static { ) -> IotaResult; } -/// Submit Iota certificates to the consensus. +/// Submit IOTA certificates to the consensus. pub struct ConsensusAdapter { /// The network client connecting to the consensus node of this authority. consensus_client: Arc, diff --git a/crates/iota-core/src/consensus_manager/mod.rs b/crates/iota-core/src/consensus_manager/mod.rs index 2e2c7a7d521..d77f2c6eafa 100644 --- a/crates/iota-core/src/consensus_manager/mod.rs +++ b/crates/iota-core/src/consensus_manager/mod.rs @@ -81,7 +81,7 @@ impl ProtocolManager { } } -/// Used by Iota validator to start consensus protocol for each epoch. +/// Used by IOTA validator to start consensus protocol for each epoch. pub struct ConsensusManager { consensus_config: ConsensusConfig, mysticeti_manager: ProtocolManager, diff --git a/crates/iota-core/src/consensus_manager/mysticeti_manager.rs b/crates/iota-core/src/consensus_manager/mysticeti_manager.rs index 795051a325b..b2f82e4f6d9 100644 --- a/crates/iota-core/src/consensus_manager/mysticeti_manager.rs +++ b/crates/iota-core/src/consensus_manager/mysticeti_manager.rs @@ -183,8 +183,8 @@ impl ConsensusManagerTrait for MysticetiManager { let registry_id = self.registry_service.add(registry.clone()); - self.authority - .swap(Some(Arc::new((authority, registry_id)))); + let registered_authority = Arc::new((authority, registry_id)); + self.authority.swap(Some(registered_authority.clone())); // Initialize the client to send transactions to this Mysticeti instance. self.client.set(client); @@ -193,6 +193,9 @@ impl ConsensusManagerTrait for MysticetiManager { let handler = MysticetiConsensusHandler::new(consensus_handler, commit_receiver, monitor); let mut consensus_handler = self.consensus_handler.lock().await; *consensus_handler = Some(handler); + + // Wait until all locally available commits have been processed + registered_authority.0.replay_complete().await; } async fn shutdown(&self) { diff --git a/crates/iota-core/src/epoch/randomness.rs b/crates/iota-core/src/epoch/randomness.rs index be323fa4747..919aed058f3 100644 --- a/crates/iota-core/src/epoch/randomness.rs +++ b/crates/iota-core/src/epoch/randomness.rs @@ -426,7 +426,7 @@ impl RandomnessManager { info!("random beacon: created {msg:?} with dkg version {dkg_version}"); let transaction = ConsensusTransaction::new_randomness_dkg_message(epoch_store.name, &msg); - #[expect(unused_mut)] + #[cfg_attr(not(any(msim, fail_points)), expect(unused_mut))] let mut fail_point_skip_sending = false; fail_point_if!("rb-dkg", || { // maybe skip sending in simtests @@ -498,7 +498,7 @@ impl RandomnessManager { &conf, ); - #[expect(unused_mut)] + #[cfg_attr(not(any(msim, fail_points)), expect(unused_mut))] let mut fail_point_skip_sending = false; fail_point_if!("rb-dkg", || { // maybe skip sending in simtests diff --git a/crates/iota-core/src/execution_cache/object_locks.rs b/crates/iota-core/src/execution_cache/object_locks.rs index 8f360f66e7a..2e119133c90 100644 --- a/crates/iota-core/src/execution_cache/object_locks.rs +++ b/crates/iota-core/src/execution_cache/object_locks.rs @@ -10,11 +10,13 @@ use iota_types::{ storage::ObjectStore, transaction::VerifiedSignedTransaction, }; -use tracing::{debug, info, instrument, trace}; +use tracing::{debug, error, info, instrument, trace}; use super::writeback_cache::WritebackCache; use crate::authority::authority_per_epoch_store::{AuthorityPerEpochStore, LockDetails}; +type RefCount = usize; + pub(super) struct ObjectLocks { // When acquire transaction locks, lock entries are briefly inserted into this map. The map // exists to provide atomic test-and-set operations on the locks. After all locks have been @@ -26,7 +28,7 @@ pub(super) struct ObjectLocks { // for those objects. Therefore we do a db read for each object we are locking. // // TODO: find a strategy to allow us to avoid db reads for each object. - locked_transactions: DashMap, + locked_transactions: DashMap, } impl ObjectLocks { @@ -41,29 +43,10 @@ impl ObjectLocks { obj_ref: &ObjectRef, epoch_store: &AuthorityPerEpochStore, ) -> IotaResult> { - match self.locked_transactions.entry(*obj_ref) { - DashMapEntry::Vacant(vacant) => { - let tables = epoch_store.tables()?; - let lock = tables.get_locked_transaction(obj_ref)?; - if let Some(lock_details) = lock { - vacant.insert(lock_details); - } - Ok(lock) - } - DashMapEntry::Occupied(occupied) => { - if cfg!(debug_assertions) { - if let Some(lock_details) = epoch_store - .tables() - .unwrap() - .get_locked_transaction(obj_ref) - .unwrap() - { - assert_eq!(*occupied.get(), lock_details); - } - } - Ok(Some(*occupied.get())) - } - } + // We don't consult the in-memory state here. We are only interested in state + // that has been committed to the db. This is because in memory state is + // reverted if the transaction is not successfully locked. + epoch_store.tables()?.get_locked_transaction(obj_ref) } /// Attempts to atomically test-and-set a transaction lock on an object. @@ -101,15 +84,18 @@ impl ObjectLocks { let tables = epoch_store.tables()?; if let Some(lock_details) = tables.get_locked_transaction(obj_ref)? { trace!("read lock from db: {:?}", lock_details); - vacant.insert(lock_details); + vacant.insert((1, lock_details)); lock_details } else { trace!("set lock: {:?}", new_lock); - vacant.insert(new_lock); + vacant.insert((1, new_lock)); new_lock } } - DashMapEntry::Occupied(occupied) => *occupied.get(), + DashMapEntry::Occupied(mut occupied) => { + occupied.get_mut().0 += 1; + occupied.get().1 + } }; if prev_lock != new_lock { @@ -161,14 +147,24 @@ impl ObjectLocks { fn clear_cached_locks(&self, locks: &[(ObjectRef, LockDetails)]) { for (obj_ref, lock) in locks { let entry = self.locked_transactions.entry(*obj_ref); - let occupied = match entry { - DashMapEntry::Vacant(_) => panic!("lock must exist"), + let mut occupied = match entry { + DashMapEntry::Vacant(_) => { + if cfg!(debug_assertions) { + panic!("lock must exist"); + } else { + error!(?obj_ref, "lock should exist"); + } + continue; + } DashMapEntry::Occupied(occupied) => occupied, }; - if occupied.get() == lock { - trace!("clearing lock: {:?}", lock); - occupied.remove(); + if occupied.get().1 == *lock { + occupied.get_mut().0 -= 1; + if occupied.get().0 == 0 { + trace!("clearing lock: {:?}", lock); + occupied.remove(); + } } else { // this is impossible because the only case in which we overwrite a // lock is when the lock is from a previous epoch. but we are holding diff --git a/crates/iota-core/src/execution_cache/unit_tests/writeback_cache_tests.rs b/crates/iota-core/src/execution_cache/unit_tests/writeback_cache_tests.rs index 585c53d2061..24d14676e83 100644 --- a/crates/iota-core/src/execution_cache/unit_tests/writeback_cache_tests.rs +++ b/crates/iota-core/src/execution_cache/unit_tests/writeback_cache_tests.rs @@ -1150,6 +1150,79 @@ async fn test_concurrent_lockers() { } } +#[tokio::test(flavor = "multi_thread", worker_threads = 8)] +async fn test_concurrent_lockers_same_tx() { + telemetry_subscribers::init_for_testing(); + + let mut s = Scenario::new(None, Arc::new(AtomicU32::new(0))).await; + let cache = s.cache.clone(); + let mut txns = Vec::new(); + + for i in 0..1000 { + let a = i * 4; + let b = i * 4 + 1; + s.with_created(&[a, b]); + s.do_tx().await; + + let a_ref = s.obj_ref(a); + let b_ref = s.obj_ref(b); + + let tx1 = s.take_outputs(); + + let tx1 = s.make_signed_transaction(&tx1.transaction); + + txns.push((tx1, a_ref, b_ref)); + } + + let barrier = Arc::new(tokio::sync::Barrier::new(2)); + + let t1 = { + let txns = txns.clone(); + let cache = cache.clone(); + let barrier = barrier.clone(); + let epoch_store = s.epoch_store.clone(); + tokio::task::spawn(async move { + let mut results = Vec::new(); + for (tx1, a_ref, b_ref) in txns { + results.push( + cache + .acquire_transaction_locks(&epoch_store, &[a_ref, b_ref], tx1) + .await, + ); + barrier.wait().await; + } + results + }) + }; + + let t2 = { + let txns = txns.clone(); + let cache = cache.clone(); + let barrier = barrier.clone(); + let epoch_store = s.epoch_store.clone(); + tokio::task::spawn(async move { + let mut results = Vec::new(); + for (tx1, a_ref, b_ref) in txns { + results.push( + cache + .acquire_transaction_locks(&epoch_store, &[a_ref, b_ref], tx1) + .await, + ); + barrier.wait().await; + } + results + }) + }; + + let results1 = t1.await.unwrap(); + let results2 = t2.await.unwrap(); + + for (r1, r2) in results1.into_iter().zip(results2) { + assert!(r1.is_ok()); + assert!(r2.is_ok()); + } +} + #[tokio::test] async fn latest_object_cache_race_test() { let authority = TestAuthorityBuilder::new().build().await; diff --git a/crates/iota-core/src/execution_cache/writeback_cache.rs b/crates/iota-core/src/execution_cache/writeback_cache.rs index 4307f7485da..eb33845674e 100644 --- a/crates/iota-core/src/execution_cache/writeback_cache.rs +++ b/crates/iota-core/src/execution_cache/writeback_cache.rs @@ -480,15 +480,41 @@ impl WritebackCache { debug!(?object_id, ?version, ?object, "inserting object entry"); fail_point_async!("write_object_entry"); self.metrics.record_cache_write("object"); - self.dirty - .objects - .entry(*object_id) - .or_default() - .insert(version, object.clone()); + + // We must hold the lock for the object entry while inserting to the + // object_by_id_cache. Otherwise, a surprising bug can occur: + // + // 1. A thread executing TX1 can write object (O,1) to the dirty set and then + // pause. + // 2. TX2, which reads (O,1) can begin executing, because TransactionManager + // immediately schedules transactions if their inputs are available. It does + // not matter that TX1 hasn't finished executing yet. + // 3. TX2 can write (O,2) to both the dirty set and the object_by_id_cache. + // 4. The thread executing TX1 can resume and write (O,1) to the + // object_by_id_cache. + // + // Now, any subsequent attempt to get the latest version of O will return (O,1) + // instead of (O,2). + // + // This seems very unlikely, but it may be possible under the following + // circumstances: + // - While a thread is unlikely to pause for so long, moka cache uses optimistic + // lock-free algorithms that have retry loops. Possibly, under high + // contention, this code might spin for a surprisingly long time. + // - Additionally, many concurrent re-executions of the same tx could happen due + // to the tx finalizer, plus checkpoint executor, consensus, and RPCs from + // fullnodes. + let mut entry = self.dirty.objects.entry(*object_id).or_default(); + self.cached.object_by_id_cache.insert( *object_id, - Arc::new(Mutex::new(LatestObjectCacheEntry::Object(version, object))), + Arc::new(Mutex::new(LatestObjectCacheEntry::Object( + version, + object.clone(), + ))), ); + + entry.insert(version, object); } async fn write_marker_value( diff --git a/crates/iota-core/src/execution_driver.rs b/crates/iota-core/src/execution_driver.rs index ea3bf455fba..147b9279228 100644 --- a/crates/iota-core/src/execution_driver.rs +++ b/crates/iota-core/src/execution_driver.rs @@ -109,7 +109,8 @@ pub async fn execution_process( // Certificate execution can take significant time, so run it in a separate // task. - spawn_monitored_task!(async move { + let epoch_store_clone = epoch_store.clone(); + spawn_monitored_task!(epoch_store.within_alive_epoch(async move { let _scope = monitored_scope("ExecutionDriver::task"); let _guard = permit; if let Ok(true) = authority.is_tx_already_executed(&digest) { @@ -120,7 +121,7 @@ pub async fn execution_process( fail_point_async!("transaction_execution_delay"); attempts += 1; let res = authority - .try_execute_immediately(&certificate, expected_effects_digest, &epoch_store) + .try_execute_immediately(&certificate, expected_effects_digest, &epoch_store_clone) .await; if let Err(e) = res { if attempts == EXECUTION_MAX_ATTEMPTS { @@ -138,6 +139,6 @@ pub async fn execution_process( .metrics .execution_driver_executed_transactions .inc(); - }.instrument(error_span!("execution_driver", tx_digest = ?digest))); + }.instrument(error_span!("execution_driver", tx_digest = ?digest)))); } } diff --git a/crates/iota-core/src/generate_format.rs b/crates/iota-core/src/generate_format.rs index 1f0ff6e1ee5..306d3165bf1 100644 --- a/crates/iota-core/src/generate_format.rs +++ b/crates/iota-core/src/generate_format.rs @@ -196,12 +196,12 @@ enum Action { } #[derive(Debug, Parser)] -#[clap( - name = "Iota format generator", - about = "Trace serde (de)serialization to generate format descriptions for Iota types" +#[command( + name = "IOTA format generator", + about = "Trace serde (de)serialization to generate format descriptions for IOTA types" )] struct Options { - #[clap(value_enum, default_value = "Print", ignore_case = true)] + #[arg(value_enum, default_value = "Print", ignore_case = true)] action: Action, } diff --git a/crates/iota-core/src/overload_monitor.rs b/crates/iota-core/src/overload_monitor.rs index c4e921d15cf..47c686a28e6 100644 --- a/crates/iota-core/src/overload_monitor.rs +++ b/crates/iota-core/src/overload_monitor.rs @@ -255,7 +255,7 @@ pub fn overload_monitor_accept_tx( // all validators makes the same decision. let temporal_seed = SystemTime::now() .duration_since(UNIX_EPOCH) - .expect("Iota did not exist prior to 1970") + .expect("IOTA did not exist prior to 1970") .as_secs() / SEED_UPDATE_DURATION_SECS; diff --git a/crates/iota-core/src/quorum_driver/tests.rs b/crates/iota-core/src/quorum_driver/tests.rs index 7f3f94659c4..6237691cbe4 100644 --- a/crates/iota-core/src/quorum_driver/tests.rs +++ b/crates/iota-core/src/quorum_driver/tests.rs @@ -138,8 +138,8 @@ async fn test_quorum_driver_submit_transaction_no_ticket() { handle.await.unwrap(); } -async fn verify_ticket_response<'a>( - ticket: Registration<'a, TransactionDigest, QuorumDriverResult>, +async fn verify_ticket_response( + ticket: Registration<'_, TransactionDigest, QuorumDriverResult>, tx_digest: &TransactionDigest, ) { let QuorumDriverResponse { effects_cert, .. } = ticket.await.unwrap(); diff --git a/crates/iota-core/src/safe_client.rs b/crates/iota-core/src/safe_client.rs index 1929097797d..85744b91bc1 100644 --- a/crates/iota-core/src/safe_client.rs +++ b/crates/iota-core/src/safe_client.rs @@ -305,7 +305,7 @@ impl SafeClient where C: AuthorityAPI + Send + Sync + Clone + 'static, { - /// Initiate a new transfer to a Iota or Primary account. + /// Initiate a new transfer to an IOTA or Primary account. pub async fn handle_transaction( &self, transaction: Transaction, @@ -372,9 +372,9 @@ where for object in input_objects { let object_ref = object.compute_object_reference(); - if !expected + if expected .get(&object_ref.0) - .is_some_and(|expect| &object_ref == expect) + .is_none_or(|expect| &object_ref != expect) { return Err(IotaError::ByzantineAuthoritySuspicion { authority: self.address, @@ -395,9 +395,9 @@ where for object in output_objects { let object_ref = object.compute_object_reference(); - if !expected + if expected .get(&object_ref.0) - .is_some_and(|expect| &object_ref == expect) + .is_none_or(|expect| &object_ref != expect) { return Err(IotaError::ByzantineAuthoritySuspicion { authority: self.address, diff --git a/crates/iota-core/src/scoring_decision.rs b/crates/iota-core/src/scoring_decision.rs index 631c8238e3a..e3ae21cede5 100644 --- a/crates/iota-core/src/scoring_decision.rs +++ b/crates/iota-core/src/scoring_decision.rs @@ -174,7 +174,7 @@ mod tests { ); } - /// Generate a pair of Iota and consensus committees for the given size. + /// Generate a pair of IOTA and consensus committees for the given size. fn generate_committees(committee_size: usize) -> (Committee, ConsensusCommittee) { let (consensus_committee, _) = local_committee_and_keys(0, vec![1; committee_size]); diff --git a/crates/iota-core/src/test_authority_clients.rs b/crates/iota-core/src/test_authority_clients.rs index bc17d9d2754..91e7d533cc2 100644 --- a/crates/iota-core/src/test_authority_clients.rs +++ b/crates/iota-core/src/test_authority_clients.rs @@ -250,7 +250,7 @@ impl MockAuthorityApi { #[async_trait] impl AuthorityAPI for MockAuthorityApi { - /// Initiate a new transaction to a Iota or Primary account. + /// Initiate a new transaction to an IOTA or Primary account. async fn handle_transaction( &self, _transaction: Transaction, diff --git a/crates/iota-core/src/transaction_orchestrator.rs b/crates/iota-core/src/transaction_orchestrator.rs index 9aee380abea..cf6ef27c734 100644 --- a/crates/iota-core/src/transaction_orchestrator.rs +++ b/crates/iota-core/src/transaction_orchestrator.rs @@ -673,7 +673,7 @@ impl TransactionOrchestratorMetrics { ) .unwrap(), wait_for_finality_finished: register_int_counter_with_registry!( - "tx_orchestrator_wait_for_finality_fnished", + "tx_orchestrator_wait_for_finality_finished", "Total number of txns Transaction Orchestrator gets responses from Quorum Driver before timeout, either success or failure", registry, ) diff --git a/crates/iota-core/src/unit_tests/authority_aggregator_tests.rs b/crates/iota-core/src/unit_tests/authority_aggregator_tests.rs index ac734a40e8f..c34f11d80dd 100644 --- a/crates/iota-core/src/unit_tests/authority_aggregator_tests.rs +++ b/crates/iota-core/src/unit_tests/authority_aggregator_tests.rs @@ -1380,6 +1380,50 @@ async fn test_handle_transaction_response() { ) .await; + println!("Case 8.3 - Retryable Transaction (EpochEnded Error)"); + + set_tx_info_response_with_signed_tx(&mut clients, &authority_keys, &tx, 0); + + // 2 out of 4 validators return epoch ended error + for (name, _) in authority_keys.iter().skip(2) { + clients + .get_mut(name) + .unwrap() + .set_tx_info_response_error(IotaError::EpochEnded(0)); + } + let agg = get_genesis_agg(authorities.clone(), clients.clone()); + assert_resp_err( + &agg, + tx.clone().into(), + |e| { + matches!( + e, + AggregatorProcessTransactionError::RetryableTransaction { .. } + ) + }, + |e| matches!(e, IotaError::EpochEnded(0)), + ) + .await; + + println!("Case 8.4 - Retryable Transaction (EpochEnded Error) eventually succeeds"); + + set_tx_info_response_with_signed_tx(&mut clients, &authority_keys, &tx, 0); + + // 1 out of 4 validators return epoch ended error + for (name, _) in authority_keys.iter().take(1) { + clients + .get_mut(name) + .unwrap() + .set_tx_info_response_error(IotaError::EpochEnded(0)); + } + + let agg = get_genesis_agg(authorities.clone(), clients.clone()); + let cert = agg + .process_transaction(tx.clone().into(), Some(client_ip)) + .await + .unwrap(); + matches!(cert, ProcessTransactionResult::Certified { .. }); + println!("Case 9 - Non-Retryable Transaction (>=2f+1 ObjectNotFound Error)"); // >= 2f+1 object not found errors set_retryable_tx_info_response_error(&mut clients, &authority_keys); diff --git a/crates/iota-core/src/unit_tests/congestion_control_tests.rs b/crates/iota-core/src/unit_tests/congestion_control_tests.rs index 6dc1763f5b9..0141bd51582 100644 --- a/crates/iota-core/src/unit_tests/congestion_control_tests.rs +++ b/crates/iota-core/src/unit_tests/congestion_control_tests.rs @@ -206,7 +206,7 @@ impl TestSetup { } } -// Creates a transaction that touchs the shared objects `shared_object_1` and +// Creates a transaction that touches the shared objects `shared_object_1` and // `shared_object_2`, and `owned_object`, and executes the transaction in // `authority_state`. Returns the transaction and the effects of the execution. async fn update_objects( diff --git a/crates/iota-cost/tests/empirical_transaction_cost.rs b/crates/iota-cost/tests/empirical_transaction_cost.rs index da6183eadc6..650908f40a2 100644 --- a/crates/iota-cost/tests/empirical_transaction_cost.rs +++ b/crates/iota-cost/tests/empirical_transaction_cost.rs @@ -115,7 +115,7 @@ async fn create_txes( .build(); ret.insert(CommonTransactionCosts::Publish, publish_tx); - // Transfer Whole Iota Coin and Transfer Portion of Iota Coin + // Transfer Whole IOTA Coin and Transfer Portion of IOTA Coin // let whole_iota_coin_tx = TestTransactionBuilder::new(sender, gas_objects.pop().unwrap(), gas_price) diff --git a/crates/iota-cost/troubleshooting.md b/crates/iota-cost/troubleshooting.md index 9d37cba8362..5e739ee78a9 100644 --- a/crates/iota-cost/troubleshooting.md +++ b/crates/iota-cost/troubleshooting.md @@ -1,8 +1,8 @@ # Troubleshooting -## Iota Framework change +## IOTA Framework change -If Iota framework code got updated, the expectations need to be changed. Follow these steps: +If IOTA framework code got updated, the expectations need to be changed. Follow these steps: ```bash # required; can be omitted if cargo-insta is installed diff --git a/crates/iota-data-ingestion-core/Cargo.toml b/crates/iota-data-ingestion-core/Cargo.toml index 7c81d572656..bd0d00b3099 100644 --- a/crates/iota-data-ingestion-core/Cargo.toml +++ b/crates/iota-data-ingestion-core/Cargo.toml @@ -21,6 +21,7 @@ serde_json.workspace = true tap.workspace = true tempfile.workspace = true tokio = { workspace = true, features = ["full"] } +tokio-util.workspace = true tracing.workspace = true url.workspace = true diff --git a/crates/iota-data-ingestion-core/README.md b/crates/iota-data-ingestion-core/README.md new file mode 100644 index 00000000000..814c17d7534 --- /dev/null +++ b/crates/iota-data-ingestion-core/README.md @@ -0,0 +1,379 @@ +# Custom indexer + +The `iota-data-ingestion-core` crate provides an easy solution to create custom indexers. To create an indexer, you subscribe to a checkpoint stream with full checkpoint content. This stream can be one of the publicly available streams from IOTA, one that you set up in your local environment, or a combination of the two. + +Establishing a custom indexer helps improve latency, allows pruning of the data of your IOTA Full node, and provides an efficient assemblage of checkpoint data. + +## Overview + +To start implementing a custom Indexer, the [Worker](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/lib.rs#L27) trait must be implemented: + +```rust +#[async_trait] +pub trait Worker: Send + Sync { + async fn process_checkpoint(&self, checkpoint: CheckpointData) -> Result<()>; + + async fn save_progress( + &self, + sequence_number: CheckpointSequenceNumber, + ) -> Option { + Some(sequence_number) + } + + fn preprocess_hook(&self, _: CheckpointData) -> Result<()> { + Ok(()) + } +} +``` + +The `process_checkpoint` method is the central component of this trait. It handles the processing of data extracted from the `CheckpointData` struct and its subsequent storage. The `CheckpointData` struct encapsulates the complete content of a checkpoint, containing a summary, the checkpoint data itself, and detailed information about each transaction, including associated events and input/output objects. The framework offers three modes for sourcing checkpoint data: **Local**, **Remote**, and **Hybrid** (Local + Remote), allowing for flexible data ingestion strategies. + +#### Local + +The local checkpoint fetcher is ideal when the indexer and a Full Node run on the same machine. By configuring the Full Node to save generated checkpoints to a local directory, `iota-data-ingestion-core` can monitor this directory for changes using an inotify-based mechanism. This minimizes ingestion latency, as checkpoints are processed immediately after they are generated by the Full Node. + +#### Remote + +The remote checkpoint fetcher is useful when the indexer and Full Node reside on separate machines. The Full Node can be configured to expose a REST API for checkpoint retrieval. + +#### Hybrid + +A hybrid configuration combines local and remote stores, providing a fallback mechanism for continuous data ingestion. The framework prioritizes locally available checkpoint data. This mode is beneficial when transitioning to using a local Full Node for data ingestion while needing to backfill historical data or provide failover capabilities. + +## Detailed API Functionality + +To fully understand the inner workings and configuration options of `iota-data-ingestion-core`, a closer look at its API is necessary. While the `Worker` trait dictates the processing and storage of `CheckpointData`, the `IndexerExecutor` is responsible for orchestrating the workers and implementing the core ingestion logic. Before diving into the details of the `IndexerExecutor`, we will first analyze the components it relies on. + +### Checkpoint Reader + +The `CheckpointReader` actor is responsible for reading and managing `CheckpointData` from multiple sources, including local files and remote storage. In addition to fetching new checkpoints, it tracks which checkpoints have been processed, performs garbage collection by deleting processed checkpoint files from local storage, and enforces a memory limit during batch processing to prevent out-of-memory (OOM) conditions. The `CheckpointReader` is instantiated using the [initialize](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/reader.rs#L313) method. + +```rust +pub fn initialize( + // The directory on which checkpoints will be read + path: PathBuf, + // From which checkpoint sequence number to start fetching new checkpoints + starting_checkpoint_number: CheckpointSequenceNumber, + // Provide a remote URL to download checkpoints + remote_store_url: Option, + // Any config key-value pair for the Remote fetcher + remote_store_options: Vec<(String, String)>, + // Customize and optimize the Checkpoint fetch behavior + options: ReaderOptions, +) -> ( + Self, + // Channel the Workers will listen and process & store checkpoints + mpsc::Receiver, + // Channel on which workers will notify GC of the checkpoint + mpsc::Sender, + // Graceful shutdown + oneshot::Sender<()>, +) +``` + +The `ReaderOptions` struct provides options for configuring how the checkpoint reader fetches new checkpoints. + +```rust +pub struct ReaderOptions { + // How often to check for new checkpoints + // Lower values mean faster detection but more CPU usage + // Default: 100ms + pub tick_interval_ms: u64, + // Network request timeout + // Applies to remote store operations + // Default: 5 seconds + pub timeout_secs: u64, + // Maximum concurrent remote requests + // Higher values increase throughput but use more resources + // Default: 10 + pub batch_size: usize, + // Memory limit for processing batch checkpoints + // 0 means no limit + // Helps prevent OOM issues + pub data_limit: usize, +} +``` + +The [run](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/reader.rs#L344) method is responsible for starting the actor. It creates an [inotify](https://man7.org/linux/man-pages/man7/inotify.7.html) listener to monitor the local directory for new files. The method then enters a loop with a `tokio::select!` statement to handle three execution branches: + +```rust +loop { + tokio::select! { + // Graceful shutdown + _ = &mut self.exit_receiver => break, + // Receive from Workers the processed checkpoint sequence numbers, this means that the + // files are no longer needed and a GC can be safely performed + Some(gc_checkpoint_number) = self.processed_receiver.recv() => { + self.gc_processed_files(gc_checkpoint_number).expect("Failed to clean the directory"); + } + // `self.sync()` is invoked either after waiting `self.options.tick_interval_ms` + // without a new file notification, or immediately upon receiving such a notification. + Ok(Some(_)) | Err(_) = timeout(Duration::from_millis(self.options.tick_interval_ms), inotify_recv.recv()) => { + self.sync().await.expect("Failed to read checkpoint files"); + } + } +} +``` + +The [sync](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/reader.rs#L243) method manages the retrieval of checkpoints, coordinating between local and (optionally) remote sources. It prioritizes local checkpoints, using the [read_local_files](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/reader.rs#L75) method to fetch them from the local directory. Remote fetching, performed by the [remote_fetch](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/reader.rs#L216) method, is only activated if the local checkpoints are missing or lag behind the expected sequence. This design ensures that local storage is the primary source of truth. + +- [read_local_files](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/reader.rs#L75) method iterates through all entries in the configured directory, attempting to extract a sequence number from each filename. It filters out entries with sequence numbers lower than the current checkpoint number (`current_checkpoint_number`). The remaining entries are then sorted in ascending order by sequence number. The function then attempts to deserialize each remaining file as a `CheckpointData` struct but only adds the deserialized checkpoint to the result if adding it does not exceed the configured capacity (`data_limit`). +- [remote_fetch](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/reader.rs#L216) method fetches checkpoints from a remote store. It supports the Full Node REST API and various object-store interfaces, including [Amazon S3](https://docs.rs/object_store/latest/object_store/aws/index.html), [Google Cloud Storage](https://docs.rs/object_store/latest/object_store/gcp/index.html), and [WebDAV](https://docs.rs/object_store/latest/object_store/http/index.html). Based on the configuration, it can also create a hybrid client combining an object store and a REST API client. It fetches checkpoints in batches, starting from the `current_checkpoint_number`, and retrieves data until the `batch_size` is reached. While iterating through the fetched checkpoints, it checks for capacity limitations. If the capacity is exceeded, it stops fetching and returns the collected checkpoints. + +### Progress Store + +The `ProgressStore` plays a crucial role in tracking the progress of checkpoint synchronization for each task. This ensures that the Worker can resume synchronization from the last successfully processed checkpoint after an Indexer restart. The framework offers two built-in implementations: + +- `ShimProgressStore`: A simple, in-memory progress store primarily used for unit testing. It does not persist progress across restarts. +- `FileProgressStore`: A persistent progress store that uses a file to track each task's synchronization state. + +Custom progress stores can be created by implementing the `ProgressStore` trait. + +```rust +#[async_trait] +pub trait ProgressStore: Send { + async fn load(&mut self, task_name: String) -> Result; + async fn save( + &mut self, + task_name: String, + checkpoint_number: CheckpointSequenceNumber, + ) -> Result<()>; +} +``` + +Instead of directly using the custom `ProgressStore` implementation, the `IndexerExecutor` interacts with it through a wrapper: + +```rust +pub type ExecutorProgress = HashMap; + +pub struct ProgressStoreWrapper

{ + // Our custom implemented ProgressStore + progress_store: P, + // An internal Map holding the TaskName and the CheckpointSequenceNumber + pending_state: ExecutorProgress, +} +``` + +This wrapper implements the `ProgressStore` trait. Internally, it retrieves progress data from the underlying `ProgressStore` implementation and populates its internal cache. This cached data is then used to calculate the minimum watermark efficiently across all tracked tasks. + +> [!NOTE] +> A watermark represents the checkpoint sequence number up to which checkpoints have been successfully processed across all workers. It serves multiple important purposes: +> +> 1. **Progress Tracking** +> +> - Keeps track of which checkpoints have been fully processed across all workers +> - Ensures no checkpoints are skipped during synchronization +> - It provides a reliable way to resume processing after the system restarts +> +> 2. **Garbage Collection** +> +> - Used to determine which checkpoint files can be safely removed from local storage +> - Only checkpoints with sequence numbers below the watermark are eligible for deletion +> +> The system maintains two types of watermarks: +> +> 1. **Individual Worker Watermarks** +> +> - Each worker tracks its progress through individual watermarks +> - Workers report their progress after successfully processing each checkpoint +> +> 2. **Global Minimum Watermark** +> +> - Represents the minimum watermark across all workers +> - Used to ensure consistency and data safety +> - Critical for garbage collection decisions + +### WorkerPool + +The `WorkerPool` actor has the responsibility to coordinate and manage `Workers` + +```rust +pub struct WorkerPool { + // An unique name of the Worker task + pub task_name: String, + // How many instances of the current Worker to create, more workers are created + // more checkpoints they can process concurrently + concurrency: usize, + // The actual Worker instance itself + worker: Arc, +} +``` + +The [run](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/worker_pool.rs#L34) method implements the core worker behavior and management logic. It maintains a set of `worker_id`s (ranging from `0` to `concurrency - 1`) to ensure each worker processes a distinct checkpoint without race conditions. A `VecDeque` cache of `CheckpointData` is used to buffer incoming checkpoints from the `CheckpointReader` when no workers are immediately available. + +The worker pool's size is determined by the `concurrency` value. The pool is implemented as a vector of tasks, one for each `Worker` instance. `mpsc` channels are used for communication, allowing the pool to receive `CheckpointData`. After initializing the worker pool, the actor enters a loop with a `tokio::select!` statement to handle two distinct execution branches: + +```rust + loop { + tokio::select! { + // Receives a synced checkpoint notification from the Worker + Some((worker_id, status_update, progress_watermark)) = progress_receiver.recv() => { + ... + } + // Receive a `CheckpointData` from CheckpointReader + Some(checkpoint) = checkpoint_receiver.recv() => { + ... + } + } +} +``` + +- The first branch first marks the `worker_id` as idle, updates the watermark, and enforces ordered checkpoint execution. It then checks the cache for pending checkpoints and assigns any found to the idle `worker_id`. +- The second branch handles incoming `CheckpointData` from the `CheckpointReader`. It discards checkpoints with sequence numbers less than the `current_checkpoint_number`, assuming they have already been processed. It then invokes the optional [preprocess_hook](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/lib.rs#L41) method (defined in the `Worker` trait) to allow for validation, preparation, or caching of checkpoint data. If the hook executes successfully and an idle worker is available, the checkpoint is sent for processing otherwise, it is added to the cache for later processing. + +### Indexer Executor + +The `IndexerExecutor` actor is the coordinator of the entire framework logic. + +```rust +pub struct IndexerExecutor

{ + // Holds the registered WorkerPools actors + pools: Vec + Send>>>, + // Store the Sender half of the channel to notofy Worker Pool of new CheckpointData + pool_senders: Vec>, + // A wrapper around the implemented ProgressStore by having an internal cache + progress_store: ProgressStoreWrapper

, + // Worker Pools will send on this channel and notify the Executor that + // the Checkpoint was synced and a GC operation can be performed + pool_progress_sender: mpsc::Sender<(String, CheckpointSequenceNumber)>, + // Listens on synced checkpoints from Worker Pools and performs GC operation + pool_progress_receiver: mpsc::Receiver<(String, CheckpointSequenceNumber)>, + metrics: DataIngestionMetrics, +} +``` + +```rust +pub fn new(progress_store: P, number_of_jobs: usize, metrics: DataIngestionMetrics) -> Self {..} +``` + +Instantiating an `IndexerExecutor` using the [new](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/executor.rs#L35) method requires careful consideration of the `number_of_jobs` parameter. This parameter determines the capacity of a buffered `mpsc` channel used for communication, calculated as `number_of_jobs * MAX_CHECKPOINTS_IN_PROGRESS` (where `MAX_CHECKPOINTS_IN_PROGRESS` is defined as 10000). The sender end of this channel is cloned and provided to each registered `WorkerPool`. Each `WorkerPool` also maintains its internal buffered channel with a capacity of `MAX_CHECKPOINTS_IN_PROGRESS` to track its internal progress. Therefore, as a best practice, the `number_of_jobs` should be set equal to the number of registered worker pools to ensure efficient and reliable communication of progress updates. + +```rust +pub async fn register(&mut self, pool: WorkerPool) -> Result<()> {..} +``` + +After instantiation, the `IndexerExecutor` requires registration of one or more `WorkerPool` instances via the [register](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/executor.rs#L49) method to function. During registration, the executor retrieves the last recorded watermark from its `ProgressStoreWrapper`, using it as the initial checkpoint sequence number for synchronization. The `register` method also creates a buffered `mpsc` channel with a capacity of `MAX_CHECKPOINTS_IN_PROGRESS` and stores the `WorkerPool` in its internal `pools` vector. While the[pool.run()](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/executor.rs#L52-L56) method is called, it does not begin execution until awaited. The sender half of the created channel is stored in `pool_senders` vector for later distribution of `CheckpointData`. + +```rust +pub async fn run( + mut self, + // The directory on which checkpoints will be read + path: PathBuf, + // Provide a remote URL to download checkpoints + remote_store_url: Option, + // Any config key-value pair for the Remote fetcher + remote_store_options: Vec<(String, String)>, + // Customize and optimize the Checkpoint fetch behavior + reader_options: ReaderOptions, + // Graceful shutdown signal + mut exit_receiver: oneshot::Receiver<()>, +) -> Result {..} +``` + +The [run](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/executor.rs#L62) method executes the actor's main logic; it's the main orchestrator of the checkpoint processing pipeline. It coordinates between: + +- Checkpoint reading (local/remote) +- Worker pool management +- Progress tracking +- Garbage collection +- Metric reporting + +The following diagram shows on a high level the overall flow: + +```mermaid +sequenceDiagram + participant CR as CheckpointReader + participant IE as IndexerExecutor + participant WP as WorkerPool + participant PS as ProgressStore + + IE->>CR: Initialize + IE->>WP: Spawn Workers + + loop Until Exit + CR->>IE: New Checkpoint (checkpoint_recv) + IE->>WP: Distribute (pool_senders) + + Note over WP: Process Checkpoint + + WP->>IE: Progress Update (pool_progress_sender) + IE->>PS: Save Progress + PS->>IE: Return min_watermark + + alt Watermark Advanced + IE->>CR: GC Signal (gc_sender) + end + end +``` + +To initiate checkpoint fetching, the `CheckpointReader` requires an initial checkpoint sequence number. This number is obtained by calling the `ProgressStoreWrapper`'s [min_watermark](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/progress_store/mod.rs#L59) method, which determines the minimum processed checkpoint sequence number across all worker pools. This ensures that no checkpoints are skipped during synchronization and serves as the garbage collection starting point. Subsequently, the `CheckpointReader` is started in a separate task by invoking its [run](https://github.com/iotaledger/iota/blob/develop/crates/iota-data-ingestion-core/src/reader.rs#L344) method. + +> [!NOTE] +> While each `WorkerPool` retrieves its own watermark during registration, the executor also calculates a global minimum watermark at startup which is needed. This is crucial because different `WorkerPool` instances might be targeting different storage backends. The global minimum watermark ensures that each pool has the opportunity to synchronize the necessary checkpoints for its specific storage, even if other pools have already registered those checkpoints to different targets. Subsequently, each `WorkerPool` uses its `current_checkpoint_number` (set during registration) to discard any redundant checkpoints, preventing duplicate processing. + +The next step is to spawn the worker pools in separate tasks: + +```rust +for pool in std::mem::take(&mut self.pools) { + spawn_monitored_task!(pool); +} +``` + +Finally, the actor enters a loop with a `tokio::select!` statement to handle three distinct execution branches: + +```rust +tokio::select! { + // Handle graceful shutdown + _ = &mut exit_receiver => break, + // Receive processed checkpoints from Worker Pools, update the ProgressStore, + // calculate its minimum watermark and perform GC operation + Some((task_name, sequence_number)) = self.pool_progress_receiver.recv() => { + self.progress_store.save(task_name.clone(), sequence_number).await?; + let seq_number = self.progress_store.min_watermark()?; + if seq_number > reader_checkpoint_number { + gc_sender.send(seq_number).await?; + reader_checkpoint_number = seq_number; + } + self.metrics.data_ingestion_checkpoint.with_label_values(&[&task_name]).set(sequence_number as i64); + } + // Distribution of checkpoints across Worker pools + Some(checkpoint) = checkpoint_recv.recv() => { + for sender in &self.pool_senders { + sender.send(checkpoint.clone()).await?; + } + } +} +``` + +- The first branch handles the graceful shutdown of the executor. +- The second branch is responsible for receiving the progress of processed checkpoints from worker pools, and it records the current processing sequence number for each task in the progress store. Calculates the lowest processed sequence number across all tasks. If the minimum watermark exceeds the checkpoint reader's current watermark, it signals the reader to initiate garbage collection. Updates the checkpoint reader's last pruned watermark to reflect the new minimum. +- The third branch is responsible for distributing the incoming checkpoints from `CheckpointReader` across all Workers Pools. + +## Use cases + +The use cases for the `iota-data-ingestion-core` can be endless. It not only could be used to create Custom Indexers but also for analyzing the data and integrate it with external systems + +#### 1. **Building Custom Indexers for Specific Data:** + +- **Indexing Only the Necessary Data:** Depending on the specific use case, a custom indexer can be developed to store only the relevant data. For example, businesses can register transactions based on the sender and recipient, allowing them to retain essential transactional data for the parties they need without the concern that a full node may delete this information. +- **Indexer as a service**: Provide users with the tools to efficiently index and query specific data, unlocking the potential of the IOTA Tangle for various applications. + +#### 2. **Data Analysis and Visualization:** + +- **Historical Analysis:** Analyzing historical data to identify trends, patterns, and anomalies. +- **Real-time Analytics:** Processing data in real-time to generate insights and alerts. +- **Data Visualization:** Creating visualizations to represent data insights in a user-friendly way. + +#### 3. **Integration with External Systems:** + +- **Data Pipelines:** Integrating with other data processing systems (e.g., Apache Kafka, Apache Spark) for large-scale data processing. +- **Machine Learning:** Feeding indexed data into machine learning models for predictive analytics. +- **Alerting Systems:** Triggering alerts based on specific events or conditions identified in the data. + +## Usage in existing crates + +As an example of a custom indexer the `iota-data-ingestion-core` is used in the `iota-indexer` and `iota-analytics-indexer` both persist data in different storages with completely different use cases: + +- **iota-indexer**: The primary function of the indexer is to provide historical data from the ledger, which is crucial for both Explorer and developers who want to utilize ledger data in their applications. By pre-processing and storing this data, the indexer significantly reduces the load on the full node, preventing performance degradation from frequent queries. The data is stored in `PostgreSQL`. + +- **iota-analytics-indexer**: This specialized indexer is designed to extract and export relevant blockchain data from the IOTA network. It collects and stores data related to network statistics (such as transaction volume and throughput), user activity (including the number of active addresses and transaction patterns), and smart contract interactions (e.g., Move call executions). The extracted data is efficiently stored in Object Storage, such as **AWS S3**. Subsequently, this raw data is transferred to specialized data warehouses like **BigQuery** or **Snowflake**. These columnar databases are optimized for handling large datasets and performing complex analytical queries, enabling in-depth analysis of IOTA network activity. diff --git a/crates/iota-data-ingestion-core/src/executor.rs b/crates/iota-data-ingestion-core/src/executor.rs index d9e4906fb47..fd0490494a3 100644 --- a/crates/iota-data-ingestion-core/src/executor.rs +++ b/crates/iota-data-ingestion-core/src/executor.rs @@ -11,7 +11,8 @@ use iota_types::{ full_checkpoint_content::CheckpointData, messages_checkpoint::CheckpointSequenceNumber, }; use prometheus::Registry; -use tokio::sync::{mpsc, oneshot}; +use tokio::sync::mpsc; +use tokio_util::sync::CancellationToken; use crate::{ DataIngestionMetrics, ReaderOptions, Worker, @@ -65,7 +66,7 @@ impl IndexerExecutor

{ remote_store_url: Option, remote_store_options: Vec<(String, String)>, reader_options: ReaderOptions, - mut exit_receiver: oneshot::Receiver<()>, + token: CancellationToken, ) -> Result { let mut reader_checkpoint_number = self.progress_store.min_watermark()?; let (checkpoint_reader, mut checkpoint_recv, gc_sender, _exit_sender) = @@ -83,7 +84,7 @@ impl IndexerExecutor

{ } loop { tokio::select! { - _ = &mut exit_receiver => break, + _ = token.cancelled() => break, Some((task_name, sequence_number)) = self.pool_progress_receiver.recv() => { self.progress_store.save(task_name.clone(), sequence_number).await?; let seq_number = self.progress_store.min_watermark()?; @@ -112,9 +113,9 @@ pub async fn setup_single_workflow( reader_options: Option, ) -> Result<( impl Future>, - oneshot::Sender<()>, + CancellationToken, )> { - let (exit_sender, exit_receiver) = oneshot::channel(); + let token = CancellationToken::new(); let metrics = DataIngestionMetrics::new(&Registry::new()); let progress_store = ShimProgressStore(initial_checkpoint_number); let mut executor = IndexerExecutor::new(progress_store, 1, metrics); @@ -126,8 +127,8 @@ pub async fn setup_single_workflow( Some(remote_store_url), vec![], reader_options.unwrap_or_default(), - exit_receiver, + token.clone(), ), - exit_sender, + token, )) } diff --git a/crates/iota-data-ingestion-core/src/tests.rs b/crates/iota-data-ingestion-core/src/tests.rs index a04fa8c8ccc..e271f69be93 100644 --- a/crates/iota-data-ingestion-core/src/tests.rs +++ b/crates/iota-data-ingestion-core/src/tests.rs @@ -21,7 +21,7 @@ use iota_types::{ use prometheus::Registry; use rand::{SeedableRng, prelude::StdRng}; use tempfile::NamedTempFile; -use tokio::sync::oneshot; +use tokio_util::sync::CancellationToken; use crate::{ DataIngestionMetrics, FileProgressStore, IndexerExecutor, ReaderOptions, Worker, WorkerPool, @@ -48,21 +48,31 @@ async fn run( batch_size: 1, ..Default::default() }; - let (sender, recv) = oneshot::channel(); + match duration { None => { indexer - .run(path.unwrap_or_else(temp_dir), None, vec![], options, recv) + .run( + path.unwrap_or_else(temp_dir), + None, + vec![], + options, + CancellationToken::new(), + ) .await } Some(duration) => { - let handle = tokio::task::spawn(async move { - indexer - .run(path.unwrap_or_else(temp_dir), None, vec![], options, recv) - .await - }); + let token = CancellationToken::new(); + let token_child = token.child_token(); + let handle = tokio::task::spawn(indexer.run( + path.unwrap_or_else(temp_dir), + None, + vec![], + options, + token_child, + )); tokio::time::sleep(duration).await; - drop(sender); + token.cancel(); handle.await? } } diff --git a/crates/iota-data-ingestion/Cargo.toml b/crates/iota-data-ingestion/Cargo.toml index 2f998e650cc..e15a6b00ea1 100644 --- a/crates/iota-data-ingestion/Cargo.toml +++ b/crates/iota-data-ingestion/Cargo.toml @@ -26,6 +26,7 @@ serde.workspace = true serde_json.workspace = true serde_yaml.workspace = true tokio = { workspace = true, features = ["full"] } +tokio-util.workspace = true tracing.workspace = true url.workspace = true diff --git a/crates/iota-data-ingestion/src/main.rs b/crates/iota-data-ingestion/src/main.rs index 1d87bd36c0b..14f947509f4 100644 --- a/crates/iota-data-ingestion/src/main.rs +++ b/crates/iota-data-ingestion/src/main.rs @@ -12,7 +12,8 @@ use iota_data_ingestion::{ use iota_data_ingestion_core::{DataIngestionMetrics, IndexerExecutor, ReaderOptions, WorkerPool}; use prometheus::Registry; use serde::{Deserialize, Serialize}; -use tokio::{signal, sync::oneshot}; +use tokio::signal; +use tokio_util::sync::CancellationToken; #[derive(Serialize, Deserialize, Clone, Debug)] #[serde(rename_all = "lowercase")] @@ -68,7 +69,7 @@ fn default_remote_read_batch_size() -> usize { 100 } -fn setup_env(exit_sender: oneshot::Sender<()>) { +fn setup_env(token: CancellationToken) { let default_hook = std::panic::take_hook(); std::panic::set_hook(Box::new(move |panic| { @@ -76,20 +77,19 @@ fn setup_env(exit_sender: oneshot::Sender<()>) { std::process::exit(12); })); - tokio::spawn(async { + tokio::spawn(async move { signal::ctrl_c() .await .expect("Failed to install Ctrl+C handler"); - exit_sender - .send(()) - .expect("Failed to gracefully process shutdown"); + token.cancel(); }); } #[tokio::main] async fn main() -> Result<()> { - let (exit_sender, exit_receiver) = oneshot::channel(); - setup_env(exit_sender); + let token = CancellationToken::new(); + let token_child = token.child_token(); + setup_env(token); let args: Vec = env::args().collect(); assert_eq!(args.len(), 2, "configuration yaml file is required"); @@ -152,7 +152,7 @@ async fn main() -> Result<()> { config.remote_store_url, config.remote_store_options, reader_options, - exit_receiver, + token_child, ) .await?; Ok(()) diff --git a/crates/iota-e2e-tests/tests/framework_upgrades/mock_iota_systems/base/README.txt b/crates/iota-e2e-tests/tests/framework_upgrades/mock_iota_systems/base/README.txt index c40cfc5d190..db148aa9e7d 100644 --- a/crates/iota-e2e-tests/tests/framework_upgrades/mock_iota_systems/base/README.txt +++ b/crates/iota-e2e-tests/tests/framework_upgrades/mock_iota_systems/base/README.txt @@ -1,5 +1,5 @@ This directory contains a mock version of the 0x3 package (iota-system). -The idea is to introduce a minimum version of the iota-system that we can use to start Iota. +The idea is to introduce a minimum version of the iota-system that we can use to start IOTA. We can then use this mock version as the base package to test various things such as iota system state upgrades. This allows us to decouple from the complicated code in the original iota-system under iota-framework. We only need to update code here and in other mock versions when the core protocol changes. This includes: diff --git a/crates/iota-e2e-tests/tests/full_node_migration_tests.rs b/crates/iota-e2e-tests/tests/full_node_migration_tests.rs index e6c79e17779..45f52d2e844 100644 --- a/crates/iota-e2e-tests/tests/full_node_migration_tests.rs +++ b/crates/iota-e2e-tests/tests/full_node_migration_tests.rs @@ -1,13 +1,26 @@ // Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -use std::{path::PathBuf, str::FromStr}; +use std::{ + fs::File, + io::{BufWriter, Write}, + path::PathBuf, + str::FromStr, +}; use anyhow::anyhow; use bip32::DerivationPath; -use iota_genesis_builder::SnapshotSource; +use iota_genesis_builder::{ + SnapshotSource, + stardust::{ + migration::{Migration, MigrationTargetNetwork}, + parse::HornetSnapshotParser, + process_outputs::scale_amount_for_iota, + types::{address_swap_map::AddressSwapMap, address_swap_split_map::AddressSwapSplitMap}, + }, +}; use iota_json_rpc_types::{ - IotaObjectDataFilter, IotaObjectDataOptions, IotaObjectResponseQuery, + IotaData, IotaObjectDataFilter, IotaObjectDataOptions, IotaObjectResponseQuery, IotaTransactionBlockResponse, IotaTransactionBlockResponseOptions, }; use iota_keys::keystore::{AccountKeystore, FileBasedKeystore}; @@ -15,13 +28,15 @@ use iota_macros::sim_test; use iota_sdk::IotaClient; use iota_types::{ IOTA_FRAMEWORK_ADDRESS, STARDUST_ADDRESS, TypeTag, - base_types::{IotaAddress, ObjectID}, + balance::Balance, + base_types::{IotaAddress, MoveObjectType, ObjectID}, crypto::SignatureScheme::ED25519, dynamic_field::DynamicFieldName, gas_coin::GAS, programmable_transaction_builder::ProgrammableTransactionBuilder, quorum_driver_types::ExecuteTransactionRequestType, - stardust::output::NftOutput, + stardust::{coin_type::CoinType, output::NftOutput}, + timelock::timelock::TimeLock, transaction::{Argument, ObjectArg, Transaction, TransactionData}, }; use move_core_types::ident_str; @@ -29,7 +44,12 @@ use shared_crypto::intent::Intent; use tempfile::tempdir; use test_cluster::TestClusterBuilder; -const MIGRATION_DATA_PATH: &str = "tests/migration/stardust_object_snapshot.bin"; +const HORNET_SNAPSHOT_PATH: &str = "tests/migration/test_hornet_full_snapshot.bin"; +const ADDRESS_SWAP_MAP_PATH: &str = "tests/migration/address_swap.csv"; +const ADDRESS_SWAP_SPLIT_MAP_PATH: &str = "tests/migration/swap_split.csv"; +const TEST_TARGET_NETWORK: &str = "alphanet-test"; +const MIGRATION_DATA_FILE_NAME: &str = "stardust_object_snapshot.bin"; +const DELEGATOR: &str = "0x4f72f788cdf4bb478cf9809e878e6163d5b351c82c11f1ea28750430752e7892"; /// Got from iota-genesis-builder/src/stardust/test_outputs/alias_ownership.rs const MAIN_ADDRESS_MNEMONIC: &str = "few hood high omit camp keep burger give happy iron evolve draft few dawn pulp jazz box dash load snake gown bag draft car"; @@ -37,28 +57,117 @@ const MAIN_ADDRESS_MNEMONIC: &str = "few hood high omit camp keep burger give ha const SPONSOR_ADDRESS_MNEMONIC: &str = "okay pottery arch air egg very cave cash poem gown sorry mind poem crack dawn wet car pink extra crane hen bar boring salt"; #[sim_test] -async fn test_full_node_load_migration_data() -> Result<(), anyhow::Error> { +async fn test_full_node_load_migration_data_with_address_swap() -> Result<(), anyhow::Error> { telemetry_subscribers::init_for_testing(); - let snapshot_source = SnapshotSource::Local(PathBuf::from_str(MIGRATION_DATA_PATH).unwrap()); + + // Setup the temporary dir and create the writer for the stardust object + // snapshot + let dir = tempdir()?; + let stardudst_object_snapshot_file_path = dir.path().join(MIGRATION_DATA_FILE_NAME); + let object_snapshot_writer = + BufWriter::new(File::create(&stardudst_object_snapshot_file_path)?); + + // Get the address swap map + let address_swap_map = AddressSwapMap::from_csv(ADDRESS_SWAP_MAP_PATH)?; + + // Generate the stardust object snapshot + genesis_builder_snapshot_generation( + object_snapshot_writer, + address_swap_map, + AddressSwapSplitMap::default(), + )?; + // Then load it + let snapshot_source = SnapshotSource::Local(stardudst_object_snapshot_file_path); + + // A new test cluster can be spawn with the stardust object snapshot let test_cluster = TestClusterBuilder::new() .with_migration_data(vec![snapshot_source]) + .with_delegator(IotaAddress::from_str(DELEGATOR).unwrap()) .build() .await; + // Use a client to issue a test transaction let client = test_cluster.wallet.get_client().await.unwrap(); - let tx_response = address_unlock_condition(client).await?; - let IotaTransactionBlockResponse { confirmed_local_execution, errors, .. } = tx_response; + + // The transaction must be successful assert!(confirmed_local_execution.unwrap()); assert!(errors.is_empty()); Ok(()) } +#[sim_test] +async fn test_full_node_load_migration_data_with_address_swap_split() -> Result<(), anyhow::Error> { + telemetry_subscribers::init_for_testing(); + + // Setup the temporary dir and create the writer for the stardust object + // snapshot + let dir = tempdir()?; + let stardudst_object_snapshot_file_path = dir.path().join(MIGRATION_DATA_FILE_NAME); + let object_snapshot_writer = + BufWriter::new(File::create(&stardudst_object_snapshot_file_path)?); + + // Get the address swap split map + let address_swap_split_map = AddressSwapSplitMap::from_csv(ADDRESS_SWAP_SPLIT_MAP_PATH)?; + + // Generate the stardust object snapshot + genesis_builder_snapshot_generation( + object_snapshot_writer, + AddressSwapMap::default(), + address_swap_split_map.clone(), + )?; + // Then load it + let snapshot_source = SnapshotSource::Local(stardudst_object_snapshot_file_path); + + // A new test cluster can be spawn with the stardust object snapshot + let test_cluster = TestClusterBuilder::new() + .with_migration_data(vec![snapshot_source]) + .with_delegator(IotaAddress::from_str(DELEGATOR).unwrap()) + .build() + .await; + + // Use a client to issue a test transaction + let client = test_cluster.wallet.get_client().await.unwrap(); + + check_address_swap_split_map_after_migration(client, address_swap_split_map).await?; + + Ok(()) +} + +fn genesis_builder_snapshot_generation( + object_snapshot_writer: impl Write, + address_swap_map: AddressSwapMap, + address_swap_split_map: AddressSwapSplitMap, +) -> Result<(), anyhow::Error> { + let mut snapshot_parser = + HornetSnapshotParser::new::(File::open(HORNET_SNAPSHOT_PATH)?)?; + let total_supply = scale_amount_for_iota(snapshot_parser.total_supply()?)?; + let target_network = MigrationTargetNetwork::from_str(TEST_TARGET_NETWORK)?; + let coin_type = CoinType::Iota; + + // Migrate using the parser output stream + Migration::new( + snapshot_parser.target_milestone_timestamp(), + total_supply, + target_network, + coin_type, + address_swap_map, + )? + .run_for_iota( + snapshot_parser.target_milestone_timestamp(), + address_swap_split_map, + snapshot_parser.outputs(), + object_snapshot_writer, + )?; + + Ok(()) +} + async fn address_unlock_condition( iota_client: IotaClient, ) -> Result { @@ -347,3 +456,55 @@ pub async fn fund_address( Ok(()) } + +async fn check_address_swap_split_map_after_migration( + iota_client: IotaClient, + address_swap_split_map: AddressSwapSplitMap, +) -> Result<(), anyhow::Error> { + for destinations in address_swap_split_map.map().values() { + for (destination, tokens, tokens_timelocked) in destinations { + if *tokens > 0 { + let balance = iota_client + .coin_read_api() + .get_balance(*destination, None) + .await?; + assert_eq!(balance.total_balance, (*tokens as u128)); + } + if *tokens_timelocked > 0 { + let mut total = 0; + let owned_timelocks = iota_client + .read_api() + .get_owned_objects( + *destination, + Some(IotaObjectResponseQuery::new( + Some(IotaObjectDataFilter::StructType( + MoveObjectType::timelocked_iota_balance().into(), + )), + Some(IotaObjectDataOptions::new().with_bcs()), + )), + None, + None, + ) + .await? + .data; + for response in owned_timelocks { + total += bcs::from_bytes::>( + &response + .data + .expect("missing response data") + .bcs + .expect("missing BCS data") + .try_as_move() + .expect("failed to convert to Move object") + .bcs_bytes, + ) + .expect("should be a timelock balance") + .locked() + .value(); + } + assert_eq!(total, *tokens_timelocked); + } + } + } + Ok(()) +} diff --git a/crates/iota-e2e-tests/tests/migration/address_swap.csv b/crates/iota-e2e-tests/tests/migration/address_swap.csv new file mode 100644 index 00000000000..7ece8894744 --- /dev/null +++ b/crates/iota-e2e-tests/tests/migration/address_swap.csv @@ -0,0 +1,2 @@ +Origin,Destination +iota1qp8h9augeh6tk3uvlxqfapuwv93atv63eqkpru029p6sgvr49eufyz7katr,0x4f72f788cdf4bb478cf9809e878e6163d5b351c82c11f1ea28750430752e7892 \ No newline at end of file diff --git a/crates/iota-e2e-tests/tests/migration/stardust_object_snapshot.bin b/crates/iota-e2e-tests/tests/migration/stardust_object_snapshot.bin deleted file mode 100644 index c7fa51debb7..00000000000 Binary files a/crates/iota-e2e-tests/tests/migration/stardust_object_snapshot.bin and /dev/null differ diff --git a/crates/iota-e2e-tests/tests/migration/swap_split.csv b/crates/iota-e2e-tests/tests/migration/swap_split.csv new file mode 100644 index 00000000000..9cf9413ab9a --- /dev/null +++ b/crates/iota-e2e-tests/tests/migration/swap_split.csv @@ -0,0 +1,3 @@ +Origin,Destination,Tokens,TokensTimelocked +iota1qp8h9augeh6tk3uvlxqfapuwv93atv63eqkpru029p6sgvr49eufyz7katr,0x1336d143de5eb55bcb069f55da5fc9f0c84e368022fd2bbe0125b1093b446313,107667149000,107667149000 +iota1qp8h9augeh6tk3uvlxqfapuwv93atv63eqkpru029p6sgvr49eufyz7katr,0x83b5ed87bac715ecb09017a72d531ccc3c43bcb58edeb1ce383f1c46cfd79bec,388647312000,0 \ No newline at end of file diff --git a/crates/iota-e2e-tests/tests/migration/test_hornet_full_snapshot.bin b/crates/iota-e2e-tests/tests/migration/test_hornet_full_snapshot.bin new file mode 100644 index 00000000000..bac42b91117 Binary files /dev/null and b/crates/iota-e2e-tests/tests/migration/test_hornet_full_snapshot.bin differ diff --git a/crates/iota-e2e-tests/tests/passkey_e2e_tests.rs b/crates/iota-e2e-tests/tests/passkey_e2e_tests.rs index 82c9c4a9d91..5a2053a136a 100644 --- a/crates/iota-e2e-tests/tests/passkey_e2e_tests.rs +++ b/crates/iota-e2e-tests/tests/passkey_e2e_tests.rs @@ -97,7 +97,7 @@ async fn create_credential_and_sign_test_tx( let store: Option = None; let my_authenticator = Authenticator::new(my_aaguid, store, user_validation_method); let mut my_client = Client::new(my_authenticator); - let origin = Url::parse("https://www.iota.io").unwrap(); + let origin = Url::parse("https://www.iota.org").unwrap(); // Create credential. let challenge_bytes_from_rp: Bytes = random_vec(32).into(); diff --git a/crates/iota-e2e-tests/tests/protocol_version_tests.rs b/crates/iota-e2e-tests/tests/protocol_version_tests.rs index 4d34b9a39bc..773504aeaee 100644 --- a/crates/iota-e2e-tests/tests/protocol_version_tests.rs +++ b/crates/iota-e2e-tests/tests/protocol_version_tests.rs @@ -996,7 +996,7 @@ mod sim_only_tests { framework_injection::set_override_cb(IOTA_SYSTEM_PACKAGE_ID, f) } - /// Get compiled modules for Iota System, built from fixture `fixture` in + /// Get compiled modules for IOTA System, built from fixture `fixture` in /// the `framework_upgrades` directory. fn iota_system_modules(fixture: &str) -> Vec { fixture_package(fixture) diff --git a/crates/iota-e2e-tests/tests/shared_objects_tests.rs b/crates/iota-e2e-tests/tests/shared_objects_tests.rs index cbabd65cf64..663e81dabd0 100644 --- a/crates/iota-e2e-tests/tests/shared_objects_tests.rs +++ b/crates/iota-e2e-tests/tests/shared_objects_tests.rs @@ -28,7 +28,7 @@ use rand::distributions::Distribution; use test_cluster::TestClusterBuilder; use tokio::time::sleep; -/// Send a simple shared object transaction to Iota and ensures the client gets +/// Send a simple shared object transaction to IOTA and ensures the client gets /// back a response. #[sim_test] async fn shared_object_transaction() { @@ -319,8 +319,8 @@ async fn shared_object_deletion_multi_certs() { .unwrap(); } -/// End-to-end shared transaction test for a Iota validator. It does not test -/// the client or wallet, but tests the end-to-end flow from Iota to consensus. +/// End-to-end shared transaction test for an IOTA validator. It does not test +/// the client or wallet, but tests the end-to-end flow from IOTA to consensus. #[sim_test] async fn call_shared_object_contract() { let test_cluster = TestClusterBuilder::new().build().await; @@ -635,7 +635,7 @@ async fn shared_object_sync() { assert!(effects.status().is_ok()); } -/// Send a simple shared object transaction to Iota and ensures the client gets +/// Send a simple shared object transaction to IOTA and ensures the client gets /// back a response. #[sim_test] async fn replay_shared_object_transaction() { diff --git a/crates/iota-e2e-tests/tests/snapshot_tests.rs b/crates/iota-e2e-tests/tests/snapshot_tests.rs index 65cde1c5ae8..ecf9cb91eda 100644 --- a/crates/iota-e2e-tests/tests/snapshot_tests.rs +++ b/crates/iota-e2e-tests/tests/snapshot_tests.rs @@ -63,10 +63,10 @@ async fn basic_read_cmd_snapshot_tests() -> Result<(), anyhow::Error> { "iota client objects 0x0000000000000000000000000000000000000000000000000000000000000000", /* empty addr */ "iota client object 0x5", // valid object "iota client object 0x5 --bcs", // valid object BCS - "iota client object 0x9135cb3b5aca99a1555b742bd11ddc45fba33343be182bdc161be69da2c41be1", /* valid object */ - "iota client object 0x9135cb3b5aca99a1555b742bd11ddc45fba33343be182bdc161be69da2c41be1 --bcs", /* valid object BCS */ + "iota client object 0x4d03f39deb5e27a76a568adb591da553688e6df6fb053bc9ac069f2bd9495ae3", /* valid object */ + "iota client object 0x4d03f39deb5e27a76a568adb591da553688e6df6fb053bc9ac069f2bd9495ae3 --bcs", /* valid object BCS */ "iota client object 0x0000000000000000000000000000000000000000000000000000000000000000", /* non-existent object */ - "iota client tx-block E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", // valid tx digest + "iota client tx-block 88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", // valid tx digest "iota client tx-block 11111111111111111111111111111111", /* non-existent tx * digest */ ]; diff --git a/crates/iota-e2e-tests/tests/snapshots/snapshot_tests__body_fn.snap b/crates/iota-e2e-tests/tests/snapshots/snapshot_tests__body_fn.snap index 8a61ca65af0..397de4bf8f4 100644 --- a/crates/iota-e2e-tests/tests/snapshots/snapshot_tests__body_fn.snap +++ b/crates/iota-e2e-tests/tests/snapshots/snapshot_tests__body_fn.snap @@ -7,14 +7,14 @@ expression: "run_one(cmds, context).await?" [ { "data": { - "objectId": "0x495af1962cebf47d1eb0cdabb02508114dc8a05c8bb4e6c18b8e81af78c8c4b5", + "objectId": "0x02ebee2236a4cf262c571124e9a0931ae4c9e236085457e9483698d041a93586", "version": "1", - "digest": "A7cf1SRMJYAqceQb78geXHujpHQuiqEggZnWu24enHu1", + "digest": "H9URfhPraVYY3S1LDcGkzwMxCJop5bRJUtSYKrNQGFHe", "type": "0x2::coin::Coin<0x2::iota::IOTA>", "owner": { "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, - "previousTransaction": "E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "previousTransaction": "88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", "storageRebate": "0", "content": { "dataType": "moveObject", @@ -22,7 +22,7 @@ expression: "run_one(cmds, context).await?" "fields": { "balance": "30000000000000000", "id": { - "id": "0x495af1962cebf47d1eb0cdabb02508114dc8a05c8bb4e6c18b8e81af78c8c4b5" + "id": "0x02ebee2236a4cf262c571124e9a0931ae4c9e236085457e9483698d041a93586" } } } @@ -30,14 +30,14 @@ expression: "run_one(cmds, context).await?" }, { "data": { - "objectId": "0x6689ff8b1b4ce68bde6f39ebd1031e0339a4036a66b7d0a85c366ac0e6817748", + "objectId": "0x23dfee0adf69baea358751a2a081bcfa4768ea2b1c7d6a882575076e0791cd2f", "version": "1", - "digest": "GYs13Dvcx4XAj76Q4fYfXEgFTyUvVfZYorzvFv8KBRrc", + "digest": "7b2jvLUFXF5J8Ue2tPoPUvtbtTkhcPps6Xaa77NywDS1", "type": "0x2::coin::Coin<0x2::iota::IOTA>", "owner": { "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, - "previousTransaction": "E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "previousTransaction": "88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", "storageRebate": "0", "content": { "dataType": "moveObject", @@ -45,7 +45,7 @@ expression: "run_one(cmds, context).await?" "fields": { "balance": "30000000000000000", "id": { - "id": "0x6689ff8b1b4ce68bde6f39ebd1031e0339a4036a66b7d0a85c366ac0e6817748" + "id": "0x23dfee0adf69baea358751a2a081bcfa4768ea2b1c7d6a882575076e0791cd2f" } } } @@ -53,14 +53,14 @@ expression: "run_one(cmds, context).await?" }, { "data": { - "objectId": "0x9135cb3b5aca99a1555b742bd11ddc45fba33343be182bdc161be69da2c41be1", + "objectId": "0x4d03f39deb5e27a76a568adb591da553688e6df6fb053bc9ac069f2bd9495ae3", "version": "1", - "digest": "5B8YV4P7Q93phJVPgm8MVZzsoXdSxFnKLAjAkMnLYcQb", + "digest": "6cjAEhKn8mmgQC6TPSrFASBNUBcSkTscuYpVdBEBbbfz", "type": "0x2::coin::Coin<0x2::iota::IOTA>", "owner": { "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, - "previousTransaction": "E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "previousTransaction": "88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", "storageRebate": "0", "content": { "dataType": "moveObject", @@ -68,7 +68,7 @@ expression: "run_one(cmds, context).await?" "fields": { "balance": "30000000000000000", "id": { - "id": "0x9135cb3b5aca99a1555b742bd11ddc45fba33343be182bdc161be69da2c41be1" + "id": "0x4d03f39deb5e27a76a568adb591da553688e6df6fb053bc9ac069f2bd9495ae3" } } } @@ -76,14 +76,14 @@ expression: "run_one(cmds, context).await?" }, { "data": { - "objectId": "0x94b18ab08b2ddd60e7db0756a605bd31589136eb861b76f49ff3b5ae688f65bc", + "objectId": "0x59fbac7e900adfc10b42456542f93bb42ddbcb5bbf56b8427e32ef09d990b184", "version": "1", - "digest": "DXGesd5mWPPaLtwza28GDfumr9NqswcfpbcHp2LaL287", + "digest": "GCGVjasFfqD2HKisfq4RWi9FaDRVxwMSf2M7oyT6p9uG", "type": "0x2::coin::Coin<0x2::iota::IOTA>", "owner": { "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, - "previousTransaction": "E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "previousTransaction": "88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", "storageRebate": "0", "content": { "dataType": "moveObject", @@ -91,7 +91,7 @@ expression: "run_one(cmds, context).await?" "fields": { "balance": "30000000000000000", "id": { - "id": "0x94b18ab08b2ddd60e7db0756a605bd31589136eb861b76f49ff3b5ae688f65bc" + "id": "0x59fbac7e900adfc10b42456542f93bb42ddbcb5bbf56b8427e32ef09d990b184" } } } @@ -99,14 +99,14 @@ expression: "run_one(cmds, context).await?" }, { "data": { - "objectId": "0x9ee19bb4564baa0449b5160e4dccd02d5d8a54ca794aaba2130db4feb809c1f5", + "objectId": "0xc6e00266120ad997beadd6b8d3ebc6d9f68460ede3e249d5885e108f39898ed6", "version": "1", - "digest": "zoUU8mDtdVPRCqfRKWmd5VpMNfgBHd4jVjq4gu8i9vn", + "digest": "5UUM6FvaxCuzJnnWoSQph4mPRZo7SHxM6ydAemxRcuY9", "type": "0x2::coin::Coin<0x2::iota::IOTA>", "owner": { "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, - "previousTransaction": "E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "previousTransaction": "88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", "storageRebate": "0", "content": { "dataType": "moveObject", @@ -114,7 +114,7 @@ expression: "run_one(cmds, context).await?" "fields": { "balance": "30000000000000000", "id": { - "id": "0x9ee19bb4564baa0449b5160e4dccd02d5d8a54ca794aaba2130db4feb809c1f5" + "id": "0xc6e00266120ad997beadd6b8d3ebc6d9f68460ede3e249d5885e108f39898ed6" } } } @@ -129,14 +129,14 @@ expression: "run_one(cmds, context).await?" "data": { "objectId": "0x0000000000000000000000000000000000000000000000000000000000000005", "version": "1", - "digest": "8CWv4f5Nov5aRXTQa2EJk7zsNhfKVbX1fEG8VP6Wsa6d", + "digest": "UT7gYyRZBUj7sgGKbuyMvxTS32HZZvr98vXruhZ9buM", "type": "0x3::iota_system::IotaSystemState", "owner": { "Shared": { "initial_shared_version": 1 } }, - "previousTransaction": "E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "previousTransaction": "88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", "storageRebate": "0", "content": { "dataType": "moveObject", @@ -157,14 +157,14 @@ expression: "run_one(cmds, context).await?" "data": { "objectId": "0x0000000000000000000000000000000000000000000000000000000000000005", "version": "1", - "digest": "8CWv4f5Nov5aRXTQa2EJk7zsNhfKVbX1fEG8VP6Wsa6d", + "digest": "UT7gYyRZBUj7sgGKbuyMvxTS32HZZvr98vXruhZ9buM", "type": "0x3::iota_system::IotaSystemState", "owner": { "Shared": { "initial_shared_version": 1 } }, - "previousTransaction": "E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "previousTransaction": "88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", "storageRebate": "0", "bcs": { "dataType": "moveObject", @@ -175,18 +175,18 @@ expression: "run_one(cmds, context).await?" } } ], - "iota client object 0x9135cb3b5aca99a1555b742bd11ddc45fba33343be182bdc161be69da2c41be1", + "iota client object 0x4d03f39deb5e27a76a568adb591da553688e6df6fb053bc9ac069f2bd9495ae3", [ { "data": { - "objectId": "0x9135cb3b5aca99a1555b742bd11ddc45fba33343be182bdc161be69da2c41be1", + "objectId": "0x4d03f39deb5e27a76a568adb591da553688e6df6fb053bc9ac069f2bd9495ae3", "version": "1", - "digest": "5B8YV4P7Q93phJVPgm8MVZzsoXdSxFnKLAjAkMnLYcQb", + "digest": "6cjAEhKn8mmgQC6TPSrFASBNUBcSkTscuYpVdBEBbbfz", "type": "0x2::coin::Coin<0x2::iota::IOTA>", "owner": { "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, - "previousTransaction": "E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "previousTransaction": "88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", "storageRebate": "0", "content": { "dataType": "moveObject", @@ -194,31 +194,31 @@ expression: "run_one(cmds, context).await?" "fields": { "balance": "30000000000000000", "id": { - "id": "0x9135cb3b5aca99a1555b742bd11ddc45fba33343be182bdc161be69da2c41be1" + "id": "0x4d03f39deb5e27a76a568adb591da553688e6df6fb053bc9ac069f2bd9495ae3" } } } } } ], - "iota client object 0x9135cb3b5aca99a1555b742bd11ddc45fba33343be182bdc161be69da2c41be1 --bcs", + "iota client object 0x4d03f39deb5e27a76a568adb591da553688e6df6fb053bc9ac069f2bd9495ae3 --bcs", [ { "data": { - "objectId": "0x9135cb3b5aca99a1555b742bd11ddc45fba33343be182bdc161be69da2c41be1", + "objectId": "0x4d03f39deb5e27a76a568adb591da553688e6df6fb053bc9ac069f2bd9495ae3", "version": "1", - "digest": "5B8YV4P7Q93phJVPgm8MVZzsoXdSxFnKLAjAkMnLYcQb", + "digest": "6cjAEhKn8mmgQC6TPSrFASBNUBcSkTscuYpVdBEBbbfz", "type": "0x2::coin::Coin<0x2::iota::IOTA>", "owner": { "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, - "previousTransaction": "E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "previousTransaction": "88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", "storageRebate": "0", "bcs": { "dataType": "moveObject", "type": "0x2::coin::Coin<0x2::iota::IOTA>", "version": 1, - "bcsBytes": "kTXLO1rKmaFVW3Qr0R3cRfujM0O+GCvcFhvmnaLEG+EAAENP15RqAA==" + "bcsBytes": "TQPzneteJ6dqVorbWR2lU2iObfb7BTvJrAafK9lJWuMAAENP15RqAA==" } } } @@ -232,9 +232,9 @@ expression: "run_one(cmds, context).await?" } } ], - "iota client tx-block E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "iota client tx-block 88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", { - "digest": "E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "digest": "88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", "transaction": { "data": { "messageVersion": "v1", @@ -250,64 +250,64 @@ expression: "run_one(cmds, context).await?" "0x000000000000000000000000000000000000000000000000000000000000000b", "0x0000000000000000000000000000000000000000000000000000000000000403", "0x000000000000000000000000000000000000000000000000000000000000107a", - "0x00d74df673e00106b1a45136cf8cfae1ca792c3de53232e06d09fbf5d4f60b66", - "0x0214b29e560e3fa6b37836ed7167db0da40a1823cf8b01f0690e1b3bfb4ebdc5", - "0x04942308c8384041a5af2c13c807418d069d3a6e53519dadd1236f0344385a14", - "0x137cb52b1bbef8e93857fd3dc67f7f70f2aabc1d7d6eb4afb28e7dea87b9ccf1", - "0x2a74fa6e7b6f90225cb36fcdc3e416d1d3d9e209ab14aafff3afd2c3587c9aec", - "0x2ad996aa647ab628cf294b4d98b24c9debb351e72b4a72ee3425b7d9a41d08a8", - "0x30f537424026a4259959d4c28f4b474558ba881bb517267f9dac6e7bc7518a5d", - "0x3ad5e2039f28a1d146d5f7e2094b7e55f1cc35f34a53aa207250d51e76febbb4", - "0x3e14d7d3f210d79f29a18955f4e98676ebb40e51119ddd6d26a8add99daceb95", - "0x3e5742903bb62d61adea88aa26bad2fad57984e3eefc58a2f1810b82228e5117", - "0x42cfd74b065c48da45fe6048796ce9cdd3a380c182e7d89573fabc229e3ce4ab", - "0x4728ee263c6a7de9fcd0d4d58db500e0ceed99977f44ca323e9a521970efd085", - "0x495af1962cebf47d1eb0cdabb02508114dc8a05c8bb4e6c18b8e81af78c8c4b5", - "0x608863c1d4738cb9829f9bd7c5b3ab15123b6e6eab762dedcfacc23ec23d37c9", - "0x660e677bad9d848d1f131d14faef42025eedf0f0e14ede894760e0ebb51a80da", - "0x6689ff8b1b4ce68bde6f39ebd1031e0339a4036a66b7d0a85c366ac0e6817748", - "0x69cd03a3cc6b2e4a1eae6bef4892f642d8f87ed3d83b758cc5242034bd36f565", + "0x00f22f34d2616c7f8ba4495bb560e3cf43ebf18cd2ccac12e495f57d1bc6f25c", + "0x020f19ebc937d1d40077d2abcde980c8b048ca8dc3ccbac358c7c94b82b20fed", + "0x0260aaaecfef6cce63c55d8f470a06dcaf445151a2785888011a8e2194cb4648", + "0x02ebee2236a4cf262c571124e9a0931ae4c9e236085457e9483698d041a93586", + "0x18f627179e18aed322f828570d83454d3b312e171982c3527b0f41764a367a1d", + "0x23dfee0adf69baea358751a2a081bcfa4768ea2b1c7d6a882575076e0791cd2f", + "0x24a640dee174b0ed74322070be09e3bb9d96c646858f47de7c1b663496cd6672", + "0x2508c2e30a9718a98cef20f0cb58fa74d79bf743a5c73441a70640f96bbf68f0", + "0x29723a1c473b6f6d2c10a9ccddcd8062a3dbb24b31091470c9341ca36d004c7c", + "0x299f315d1f944855d2fb3f3868d18a32cf2e7f243d3a057b2724ce4395106cca", + "0x2b7c5961cbac364bb9711f2791019b594c9110be9c4ed246cab9fedf9bd159e7", + "0x333a99382fa83b4d0e6ed4f25018e848af2f0a48ca06e94948f1aec691011080", + "0x45f36b9e92ea7ce24fcf76b9a147f91d6329d19d2b79ba0252bd1be4fa0b3114", + "0x47be074115c6bd8b19dabd11020c8b59bb7c8a1f64c78ec2a1a3ff366e524b18", + "0x4d03f39deb5e27a76a568adb591da553688e6df6fb053bc9ac069f2bd9495ae3", + "0x50cb4102d8bc479b9697d6e9e7327e23cb1f00400261f919c8745e49c7732e54", + "0x566ad3b404129b7740255d9eb77d8f649816cbf8f4fa4f0e0b3088eb40f4be7d", + "0x59fbac7e900adfc10b42456542f93bb42ddbcb5bbf56b8427e32ef09d990b184", + "0x63de3e53d6a19a6d42db458e025917b19340a94c638039dc6f771618dee1c6f9", + "0x65b17c1361e97dd96998ec36253fe21441400d111d790d7b312dfe9364687338", + "0x6665f50841dbe82152d70d15b852a1d2471a83c162830438afb06c3e1717d2c1", + "0x6a70af3f08b53d7bd91be9ace2da765e9f7ab0f9046eecd0a92c41897eb742d4", "0x6af2a2b7ca60bf76174adfd3e9c4957f8e937759603182f9b46c7f6c5f19c6d2", - "0x6b2a75a39565fe93f6f2694e79274205ec65dac6aecff0a38b3c3947e2c9e561", - "0x82054bc6567488b4924203907f0bc62f3f789e1f5edc901798c5a58042dfa1a8", - "0x8a23c86abe72190400e37fd3affc876000ad1302712cbdaf76077f7d541b6400", - "0x9045719b475148ea5e0547b2a0dd8f8aaa19a658deac6bfca7a184d1dfa13d87", - "0x9135cb3b5aca99a1555b742bd11ddc45fba33343be182bdc161be69da2c41be1", - "0x9254a40663ad7e72d73dd1f04b1eca4e0486cc9f54cdd9a7c3d4e479a8131aa0", - "0x939337494e5c4fe00e78f120459bf108367938719f717e32c845ab83e762f22c", - "0x94b18ab08b2ddd60e7db0756a605bd31589136eb861b76f49ff3b5ae688f65bc", - "0x98c27dcdc3cb082f2ae47756648c92ee054721273663cf0ad25d7f2d4fd20128", - "0x9bafea6cfb0a89e0eee6e66ec0d598848067196ed07857012e00e609e8f3ca17", - "0x9df5ecf6c7c2ed3f31b6ff471aaea04af7d331647c34ab319c8eef45dd043620", - "0x9ee19bb4564baa0449b5160e4dccd02d5d8a54ca794aaba2130db4feb809c1f5", - "0xa01451ee51efc5ef58d07e5d373b68187ee3a042553670da8a4dc613adf3ed61", - "0xa181feb7b518f4e06794c678eace3633c6b1d7222736e9d059e75ca3023b95a0", - "0xa629b978cb416fdeac008d532329429114c70e49909d97bede0f794b376eafe8", - "0xaca6bf6562b4cbbd909e6f9310c23c9517deca84e82d22dfa781c3dd980bbc95", - "0xb12dd88f42ce5db0ba12181deae915d795de5a1e4dfe87deb44a51c8dd8ced7a", - "0xc3647488b7f826d5e2602102964e93859bab9374389c1cca8799f91c2b8b1589", - "0xcc36edb95e82c14b7175ae99e58c3a27b43fb1ba7d7f1454fbda1ce66e0e5a8f", - "0xcf326c0a4f56e81beb81e1ee93cda5bd5bc5e88fa4b730a1909679b2ffb3f0d3", - "0xd3260d5e56e16920f270b3f92a40782447e360ea000aba2af08f0d1f8ad6b5f0", - "0xd43afda73e2bb89e67d8049618176ba09456438f9fcea5f8a3e91f8a1b17ecd8", - "0xd8fa8491b91a220af49eb0b54e1d5de725ca40cad8ecca989f27a2f713402490", - "0xdde2baa8f84257d47c8051b5abd214afb850211d7ee74a99c21698e8bd66f81d", - "0xe0ebce954f5f7a4f32e3fa49fa1b76bc1e56008535860b3fe78387db50f95dbd", - "0xe360ccacf592e9a2d80c7314d882d2adb850db51fd18572a6baf11a85b233e5d", - "0xe5395355afde6c5624ce706e91738cb766a8324c86ea933f38c3f8efbba38507", - "0xe8bcfc0181e426b9dfcad3785fb4db1a0d0a2b1fa40c99a36b5fe988d2a2f561", - "0xed9ffc59b7e841e0de23ac1b72c4f04a784e2fef9c46f93eed0afdd14f8d047c", - "0xeed5eebe436d3b4dedb5d25eff6275c5212fb408ab6e2d1756fd9c56b2af892d", - "0xf69faa3aa3015380b78c11aae2ee8eed497bb5ee2555e686b53e375ab90fe8cd", - "0xfe1d0d9ef3118023510cdf338ca59ae1d3e88e1ca98a3e1904d61a8620a1ad98" + "0x6f651492979fb7f04a2eb053ec8559074c2fa562d8ee922cf2c55277978d5c89", + "0x71654ced3581c229f5f2743a808263effc4e63090835f77ca9e8ec4bf667ce5f", + "0x74f9e1e67cc981675746f2811572b4ea8ce3e6835f49398313b97ad19322f7c5", + "0x7e87e1c8ff4d9ac72fd6af8e55bc23404dbada8ac7ed2e234524e76904f362cb", + "0x813dd1c1277072ec5f24cc2628513883618b6987fc5435b872cab8f27102329a", + "0x9756463d0f40b3b3ac6de7026fb38690ab2b998dfeb72a51cdd563ae0c373f0a", + "0xa4a778c9213d3035bca59c69f380e80fd7196a72a3e2b51cb22dea1411d91401", + "0xa51b7309e9d47b1ba39cddc19a2e5308b1cbaf73112697fbee332a4a66014a85", + "0xa93e761264922c29a33a97b7330bebdfc7eceb438260efd44ff08a7b407d118e", + "0xa9e70f2bac16567977470169378c3496ac727b9489add6399ba7a568d9d40aa6", + "0xa9fa3dae3e8fa5eaabe1b29d5af2f0a25490d41677b58c4bedbe1920b03ac7ef", + "0xbd8eaa142a631472ac39ba6172ad0dd11af7cdb633acdc2e62adbbecf2469578", + "0xc040e5614e1e604d3bc663b9f612a818ebb918aeab40a6ba25caa7a2a6a237b6", + "0xc31493f7fd0cbfaad8106574193f492d2727bf48db0602f78b155da5a51402ba", + "0xc325849c128782699f66a3fea12d317b5486309cae90a5cce5c9a013d504f000", + "0xc6e00266120ad997beadd6b8d3ebc6d9f68460ede3e249d5885e108f39898ed6", + "0xcd7e3cb7274220eddb40315c39018c145a4bdc29ae274897178a00e9e5df7719", + "0xd813cf320b2cbac0e64c8983484130518715c01961a0bbd2667b055117967dd3", + "0xd9677e045ceed60ae258d0fc79f2e04b3b54cd1e37d2bb4ea5364651d81a354c", + "0xdf7bf4613216b40876db6e0e41da95724ed3521fdc36a031ba70a66f4fed33d4", + "0xe2d939c957ebcc0c9a9c0dcad88df409d63291e2260dd3f1a0e2b967558e5745", + "0xe378d2cec927e9c247bab671285b8f7b23dc02abe786866b2503b3764c9d2ac7", + "0xe72634ff60a76ad75299b6681e5ac68ae16cd5052f55ada644c69f6619ca8f96", + "0xe7445ae60304857a2efd0c4767583987a3de4f78a94e893aca46e957a7743c02", + "0xf1fa0fb0d8312d048e280ed73cf0fd19db5a666470ce8ef13ccafbf37f922b6f", + "0xf87126f9f001bdf5a14efe58a703fd97d2e2aa6c8ee108243cfcc03f70bec36b", + "0xfc3a360f44f32104c49dcf159ba04e021a42b04c969e2ecf3b7c3f8a5d4a4329" ], "events": [ { - "txDigest": "E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "txDigest": "88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", "eventSeq": "0" }, { - "txDigest": "E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "txDigest": "88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", "eventSeq": "1" } ] @@ -343,14 +343,14 @@ expression: "run_one(cmds, context).await?" "storageRebate": "0", "nonRefundableStorageFee": "0" }, - "transactionDigest": "E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "transactionDigest": "88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", "created": [ { "owner": "Immutable", "reference": { "objectId": "0x0000000000000000000000000000000000000000000000000000000000000001", "version": 1, - "digest": "7XoN6TGQbADiyEaVmrD6gmXJDJkX21MxGMc4pYCo9mKN" + "digest": "85UCZYd34oqKh3x3baQuV7giuUzyNLQvY1qL5UNFQa6t" } }, { @@ -358,7 +358,7 @@ expression: "run_one(cmds, context).await?" "reference": { "objectId": "0x0000000000000000000000000000000000000000000000000000000000000002", "version": 1, - "digest": "5DVdPqNcu7yg8VSoQy9GwmoUprvHyK87jxnbRoac1r1S" + "digest": "9BM59kZjX2BAq8vFvKDuuZM5eKhtoGRxxXuS1kzFyVR9" } }, { @@ -366,7 +366,7 @@ expression: "run_one(cmds, context).await?" "reference": { "objectId": "0x0000000000000000000000000000000000000000000000000000000000000003", "version": 1, - "digest": "B7r86UhqxCjrBYVNGnSf7QBNX7EWWm7kLHZ5ht1MLAre" + "digest": "75Le8fZznGd5Bvzxuve3UPQKX2fej1avUJYnA3J1rBjk" } }, { @@ -378,7 +378,7 @@ expression: "run_one(cmds, context).await?" "reference": { "objectId": "0x0000000000000000000000000000000000000000000000000000000000000005", "version": 1, - "digest": "8CWv4f5Nov5aRXTQa2EJk7zsNhfKVbX1fEG8VP6Wsa6d" + "digest": "UT7gYyRZBUj7sgGKbuyMvxTS32HZZvr98vXruhZ9buM" } }, { @@ -390,7 +390,7 @@ expression: "run_one(cmds, context).await?" "reference": { "objectId": "0x0000000000000000000000000000000000000000000000000000000000000006", "version": 1, - "digest": "DjXTupBThwgJcFY9YDL2Xan9bwTTA3BYrW72Dgh1AMJA" + "digest": "9BNNarX2ktYyq83ySt9hkq5Nr95e3NWdLDaUtFxPv2Lj" } }, { @@ -402,7 +402,7 @@ expression: "run_one(cmds, context).await?" "reference": { "objectId": "0x0000000000000000000000000000000000000000000000000000000000000008", "version": 1, - "digest": "aCZvLjShGf5G1AEq3jb9c3y3jxi8NhX17kFUxKwt1q1" + "digest": "7UbMmfwwtbb9RxAtRAuqND8MieggKR6rHn5nvqdqx1HZ" } }, { @@ -410,7 +410,7 @@ expression: "run_one(cmds, context).await?" "reference": { "objectId": "0x000000000000000000000000000000000000000000000000000000000000000b", "version": 1, - "digest": "7rF3B3o4ZHxju9pxz2eGmYhigdg5MBx4WLkyYcBaMMs9" + "digest": "5wsBmq7vkfrRoEqcDxgtJCmLn3YfYRJHP8i6U2EnBgLn" } }, { @@ -422,7 +422,7 @@ expression: "run_one(cmds, context).await?" "reference": { "objectId": "0x0000000000000000000000000000000000000000000000000000000000000403", "version": 1, - "digest": "57nJzNa4jJBPDpbVVKcntb8rEZLhwhAsLAH7o6r6YLAb" + "digest": "GYMKk7f1KhowX6cq3QBKPZHuosPTMfEXpXDqNDW47zQp" } }, { @@ -430,165 +430,163 @@ expression: "run_one(cmds, context).await?" "reference": { "objectId": "0x000000000000000000000000000000000000000000000000000000000000107a", "version": 1, - "digest": "5Ad5TaF7cAZwf8U3rTD19oxAEfrhptBrSg9qGfYf36nY" + "digest": "EZXQcnmtgSTVm437D9yzcoQ3ToKAeoqD4eJ2TLE8KJxo" } }, { "owner": { - "AddressOwner": "0x4b2a0b010344ffda7202ecd5f76b742b78516cfcdb208e3314d65a4157654c4b" + "ObjectOwner": "0xc2f868e67e57dae016192c5e39e5da2fc5b091badcf0a2e943f859c736187657" }, "reference": { - "objectId": "0x00d74df673e00106b1a45136cf8cfae1ca792c3de53232e06d09fbf5d4f60b66", + "objectId": "0x00f22f34d2616c7f8ba4495bb560e3cf43ebf18cd2ccac12e495f57d1bc6f25c", "version": 1, - "digest": "57LAitmwQMZgioGaRVybfVmZ2Yc57HrJ2Hkm7iGmX5aD" + "digest": "9r9ScS86fEETEc2esEQDGgrGdkC8JSDA1HjAtBA7KGDF" } }, { "owner": { - "ObjectOwner": "0x1d3c44ed95179381aec577194b10aaaccddc14ceedf533b9b6b4eef5535476e0" + "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" }, "reference": { - "objectId": "0x0214b29e560e3fa6b37836ed7167db0da40a1823cf8b01f0690e1b3bfb4ebdc5", + "objectId": "0x020f19ebc937d1d40077d2abcde980c8b048ca8dc3ccbac358c7c94b82b20fed", "version": 1, - "digest": "3w6HwgnTmaGDCnAciokuEpZHj2NjFWv6i9KiCHqH7Nct" + "digest": "Ann24wsSiJBWNtRJnib87r4B9XU5eJgRuBPAKDr15X1e" } }, { "owner": { - "AddressOwner": "0x20227cfbc6699debf187d2d7bca3d253cd50f87c165516d3dc53a246a9af7182" + "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" }, "reference": { - "objectId": "0x04942308c8384041a5af2c13c807418d069d3a6e53519dadd1236f0344385a14", + "objectId": "0x0260aaaecfef6cce63c55d8f470a06dcaf445151a2785888011a8e2194cb4648", "version": 1, - "digest": "811mkHeY9LoDN16a3CSnr22AZK8R2rJa9DbDwuvDp9kd" + "digest": "DNPTGrZEvnFwQBc9SKeTWQfnQy8SouFbUkXooEpdRL8P" } }, { "owner": { - "ObjectOwner": "0xee64ad330164aaa798846df2f2df3525410779fb120765a1c5fb1f931bed784b" + "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, "reference": { - "objectId": "0x137cb52b1bbef8e93857fd3dc67f7f70f2aabc1d7d6eb4afb28e7dea87b9ccf1", + "objectId": "0x02ebee2236a4cf262c571124e9a0931ae4c9e236085457e9483698d041a93586", "version": 1, - "digest": "9fG99ZbX8SPLLkLnRcah82PyLY3BSbgbLD7HKD6XntZ4" + "digest": "H9URfhPraVYY3S1LDcGkzwMxCJop5bRJUtSYKrNQGFHe" } }, { "owner": { - "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" + "ObjectOwner": "0x85c7bca7f3fb7ebfe0d076cdb331ac21abafbcda69065c1abb1056278e0156db" }, "reference": { - "objectId": "0x2a74fa6e7b6f90225cb36fcdc3e416d1d3d9e209ab14aafff3afd2c3587c9aec", + "objectId": "0x18f627179e18aed322f828570d83454d3b312e171982c3527b0f41764a367a1d", "version": 1, - "digest": "E37WvCF8d2asqAJH7MtR9rnyiWeHqjgsQqL4gbjAwgLk" + "digest": "2idkUZoaajnS1S1sJtGSszNBzGv45q47sTv5jruCM78S" } }, { "owner": { - "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" + "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, "reference": { - "objectId": "0x2ad996aa647ab628cf294b4d98b24c9debb351e72b4a72ee3425b7d9a41d08a8", + "objectId": "0x23dfee0adf69baea358751a2a081bcfa4768ea2b1c7d6a882575076e0791cd2f", "version": 1, - "digest": "F5fPj6gyhcWnNMJTnxis1goTvTNUXYr69zJudZaqMt2t" + "digest": "7b2jvLUFXF5J8Ue2tPoPUvtbtTkhcPps6Xaa77NywDS1" } }, { - "owner": "Immutable", + "owner": { + "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" + }, "reference": { - "objectId": "0x30f537424026a4259959d4c28f4b474558ba881bb517267f9dac6e7bc7518a5d", + "objectId": "0x24a640dee174b0ed74322070be09e3bb9d96c646858f47de7c1b663496cd6672", "version": 1, - "digest": "CF2vkHV4X8YVrhgVKRafUDX7HmSbxSkvmwFYFPkEE4PM" + "digest": "2Q2aeKK2fLQBQX83ihYjpkyEoTDyhEz1QicN2sH1XqKL" } }, { "owner": { - "AddressOwner": "0x7a9730f69d7c9eed4058b7e3ad9019adffdf63a94ef73cd2fb0217f3b9eb6eb1" + "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" }, "reference": { - "objectId": "0x3ad5e2039f28a1d146d5f7e2094b7e55f1cc35f34a53aa207250d51e76febbb4", + "objectId": "0x2508c2e30a9718a98cef20f0cb58fa74d79bf743a5c73441a70640f96bbf68f0", "version": 1, - "digest": "2B9M7nEmFp1jcPM72pd7qXNNenQZwdp5dGu4xoykGeXZ" + "digest": "3uEZ8L6yNm61aZDW9TU53Gw5ecTBceJHjdFsbbN3PnSc" } }, { "owner": { - "AddressOwner": "0x7a9730f69d7c9eed4058b7e3ad9019adffdf63a94ef73cd2fb0217f3b9eb6eb1" + "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" }, "reference": { - "objectId": "0x3e14d7d3f210d79f29a18955f4e98676ebb40e51119ddd6d26a8add99daceb95", + "objectId": "0x29723a1c473b6f6d2c10a9ccddcd8062a3dbb24b31091470c9341ca36d004c7c", "version": 1, - "digest": "A6ZSdzLj5Bvv27zaQoFm5rUkS91sK8nJKfntqKbHepUb" + "digest": "EKpry9R3MSw89yYRN9dd2KbVwVVzXrNWtwgDPw1qbDpu" } }, { "owner": { - "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" + "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" }, "reference": { - "objectId": "0x3e5742903bb62d61adea88aa26bad2fad57984e3eefc58a2f1810b82228e5117", + "objectId": "0x299f315d1f944855d2fb3f3868d18a32cf2e7f243d3a057b2724ce4395106cca", "version": 1, - "digest": "3MVJSV5RKkXLxsTdHqPQERjnmAQAoxkkiRv1Q6G8zHaU" + "digest": "BLfSDqvn8aNUfAxrWVfdMQSjGsiDqM6Zi4ELFgfRjxTL" } }, { "owner": { - "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" + "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" }, "reference": { - "objectId": "0x42cfd74b065c48da45fe6048796ce9cdd3a380c182e7d89573fabc229e3ce4ab", + "objectId": "0x2b7c5961cbac364bb9711f2791019b594c9110be9c4ed246cab9fedf9bd159e7", "version": 1, - "digest": "5AzKkrWAReMDgNF2bbiveSDDWvJpQzSX7xHRHNJu9Phy" + "digest": "DR61KQATeDiSkpFWft6ckEC5Cv4YA8ws9W8YoGQDZTGA" } }, { "owner": { - "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" + "ObjectOwner": "0xc7cf938bf2621982fdbe983a16f7e1ae01358aab0dee88eed62ccf03b0538b35" }, "reference": { - "objectId": "0x4728ee263c6a7de9fcd0d4d58db500e0ceed99977f44ca323e9a521970efd085", + "objectId": "0x333a99382fa83b4d0e6ed4f25018e848af2f0a48ca06e94948f1aec691011080", "version": 1, - "digest": "781mLWh1jZYNUxtypVJ4XWvHvvga91AZyRfrtXSG53RZ" + "digest": "5EzJs5Q9tzmn73soqnrghEWHZsZRmsC1MoXzsUDLjZje" } }, { "owner": { - "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" + "AddressOwner": "0x20227cfbc6699debf187d2d7bca3d253cd50f87c165516d3dc53a246a9af7182" }, "reference": { - "objectId": "0x495af1962cebf47d1eb0cdabb02508114dc8a05c8bb4e6c18b8e81af78c8c4b5", + "objectId": "0x45f36b9e92ea7ce24fcf76b9a147f91d6329d19d2b79ba0252bd1be4fa0b3114", "version": 1, - "digest": "A7cf1SRMJYAqceQb78geXHujpHQuiqEggZnWu24enHu1" + "digest": "ku4ZwBs3e3wxPf1jfQ4RDFYhW5WhvGvwURSZNCEBkkd" } }, { - "owner": { - "AddressOwner": "0x7dbda9c2efa8255eea64cf28b64294ffed6af432f53b661d24f3807895ee828d" - }, + "owner": "Immutable", "reference": { - "objectId": "0x608863c1d4738cb9829f9bd7c5b3ab15123b6e6eab762dedcfacc23ec23d37c9", + "objectId": "0x47be074115c6bd8b19dabd11020c8b59bb7c8a1f64c78ec2a1a3ff366e524b18", "version": 1, - "digest": "6NnLvzoYdm9qBrrsgCj3x2vtrQeMceUt1QuBSvxvgXq5" + "digest": "ACEM96A7XicvdzrRuD4LRm6KyocHATyRKDfKkx9ZaL21" } }, { "owner": { - "ObjectOwner": "0xaca630a6c6d4a9e7a7b033174b3952dfd7c2edeea9fa3a356e97d1a5bc31dae0" + "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, "reference": { - "objectId": "0x660e677bad9d848d1f131d14faef42025eedf0f0e14ede894760e0ebb51a80da", + "objectId": "0x4d03f39deb5e27a76a568adb591da553688e6df6fb053bc9ac069f2bd9495ae3", "version": 1, - "digest": "EJiD9QzGf9afkBrdYyCfsg9f7Cx64oSLDTVMuvxc2jb2" + "digest": "6cjAEhKn8mmgQC6TPSrFASBNUBcSkTscuYpVdBEBbbfz" } }, { - "owner": { - "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" - }, + "owner": "Immutable", "reference": { - "objectId": "0x6689ff8b1b4ce68bde6f39ebd1031e0339a4036a66b7d0a85c366ac0e6817748", + "objectId": "0x50cb4102d8bc479b9697d6e9e7327e23cb1f00400261f919c8745e49c7732e54", "version": 1, - "digest": "GYs13Dvcx4XAj76Q4fYfXEgFTyUvVfZYorzvFv8KBRrc" + "digest": "J9f9eKbfyHs6ks9tGxQPCp41pRfgKy8oRqU4gZb4nMha" } }, { @@ -596,199 +594,199 @@ expression: "run_one(cmds, context).await?" "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" }, "reference": { - "objectId": "0x69cd03a3cc6b2e4a1eae6bef4892f642d8f87ed3d83b758cc5242034bd36f565", + "objectId": "0x566ad3b404129b7740255d9eb77d8f649816cbf8f4fa4f0e0b3088eb40f4be7d", "version": 1, - "digest": "FSfz1Dtc26vvg9ASRGoVHGg1WVbVqLcBtvrcYtTt6bLe" + "digest": "Hg7fVwRrTquxfD5nN6X8jNLbmdE9xiFbw6frCPPmdXWu" } }, { "owner": { - "ObjectOwner": "0x0000000000000000000000000000000000000000000000000000000000000005" + "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, "reference": { - "objectId": "0x6af2a2b7ca60bf76174adfd3e9c4957f8e937759603182f9b46c7f6c5f19c6d2", + "objectId": "0x59fbac7e900adfc10b42456542f93bb42ddbcb5bbf56b8427e32ef09d990b184", "version": 1, - "digest": "Er8zLC9VrCrgZZcnhjLkA73mZ3J63rxiboYV6EsqmhN2" + "digest": "GCGVjasFfqD2HKisfq4RWi9FaDRVxwMSf2M7oyT6p9uG" } }, { "owner": { - "AddressOwner": "0x0b37c2167535af8a5729c44690f7be89eb248a6eb3cda5fd37ee1029949f0c29" + "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" }, "reference": { - "objectId": "0x6b2a75a39565fe93f6f2694e79274205ec65dac6aecff0a38b3c3947e2c9e561", + "objectId": "0x63de3e53d6a19a6d42db458e025917b19340a94c638039dc6f771618dee1c6f9", "version": 1, - "digest": "Hy3XSG83B625ywGh161TYQ2VqBJzKxFccgp1fL3Zjbob" + "digest": "Bv4sCpQQgssCQroeqq5e9VctzQiXbefNVPxi9DKwU7Jq" } }, { "owner": { - "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" + "AddressOwner": "0x7dbda9c2efa8255eea64cf28b64294ffed6af432f53b661d24f3807895ee828d" }, "reference": { - "objectId": "0x82054bc6567488b4924203907f0bc62f3f789e1f5edc901798c5a58042dfa1a8", + "objectId": "0x65b17c1361e97dd96998ec36253fe21441400d111d790d7b312dfe9364687338", "version": 1, - "digest": "DRYTDtLfuoxayHMXVqxuuWvv4DJnjgZQvoZebdCoGszX" + "digest": "CEMFxNJKgsJoZS5p6djiBcoas2yjWYGe93ib4fmq7mU" } }, { "owner": { - "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" + "AddressOwner": "0x4b2a0b010344ffda7202ecd5f76b742b78516cfcdb208e3314d65a4157654c4b" }, "reference": { - "objectId": "0x8a23c86abe72190400e37fd3affc876000ad1302712cbdaf76077f7d541b6400", + "objectId": "0x6665f50841dbe82152d70d15b852a1d2471a83c162830438afb06c3e1717d2c1", "version": 1, - "digest": "2fqjooDKXPoqAFBjzh6cGr53zuTfngNLuHzJoPoR7uvk" + "digest": "5FQAwHKd9BSMAqByENw4AjBPxtu1dUFdvnwFnP5ssefc" } }, { "owner": { - "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" + "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" }, "reference": { - "objectId": "0x9045719b475148ea5e0547b2a0dd8f8aaa19a658deac6bfca7a184d1dfa13d87", + "objectId": "0x6a70af3f08b53d7bd91be9ace2da765e9f7ab0f9046eecd0a92c41897eb742d4", "version": 1, - "digest": "9puPcXBGsDq9hTfkFMUHpmruAkuqsqQmBt6b2ChxRU2f" + "digest": "F244scn8ByNwXxG7Sd2zCkn4TTrGPHy8c1ixYdK1VHyP" } }, { "owner": { - "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" + "ObjectOwner": "0x0000000000000000000000000000000000000000000000000000000000000005" }, "reference": { - "objectId": "0x9135cb3b5aca99a1555b742bd11ddc45fba33343be182bdc161be69da2c41be1", + "objectId": "0x6af2a2b7ca60bf76174adfd3e9c4957f8e937759603182f9b46c7f6c5f19c6d2", "version": 1, - "digest": "5B8YV4P7Q93phJVPgm8MVZzsoXdSxFnKLAjAkMnLYcQb" + "digest": "6AyGKpbs2yy8j6nt6mXVuE7yvhYQvZ1quTi1nhZm67nQ" } }, { "owner": { - "ObjectOwner": "0xaca630a6c6d4a9e7a7b033174b3952dfd7c2edeea9fa3a356e97d1a5bc31dae0" + "AddressOwner": "0x7a9730f69d7c9eed4058b7e3ad9019adffdf63a94ef73cd2fb0217f3b9eb6eb1" }, "reference": { - "objectId": "0x9254a40663ad7e72d73dd1f04b1eca4e0486cc9f54cdd9a7c3d4e479a8131aa0", + "objectId": "0x6f651492979fb7f04a2eb053ec8559074c2fa562d8ee922cf2c55277978d5c89", "version": 1, - "digest": "Du8gWCJQvqcsNBEpSYTDSB2aaRyVEG8h7niCPhKQtZFf" + "digest": "21JXcJtash7RUymcy5MvDHHHsWewg6H7Ck36RbcPruKp" } }, { "owner": { - "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" + "ObjectOwner": "0xd6aea7b7b28562ef76634df18c8c84e7f0d456d0053c9f3264e748aeaaf61ec2" }, "reference": { - "objectId": "0x939337494e5c4fe00e78f120459bf108367938719f717e32c845ab83e762f22c", + "objectId": "0x71654ced3581c229f5f2743a808263effc4e63090835f77ca9e8ec4bf667ce5f", "version": 1, - "digest": "5RBuu4Mz4xV3bQAdU8DQnXLppZtpjGToNcoVujUfpxwG" + "digest": "AJXAfxWsnsNRLrdxeARMMXMs7bAw93r1Qu2VG9YB9jH8" } }, { "owner": { - "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" + "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" }, "reference": { - "objectId": "0x94b18ab08b2ddd60e7db0756a605bd31589136eb861b76f49ff3b5ae688f65bc", + "objectId": "0x74f9e1e67cc981675746f2811572b4ea8ce3e6835f49398313b97ad19322f7c5", "version": 1, - "digest": "DXGesd5mWPPaLtwza28GDfumr9NqswcfpbcHp2LaL287" + "digest": "CmYJTpbWV1sJKXaRXA6FnXcaRpaW2eMufwpi1YV3Haet" } }, { "owner": { - "ObjectOwner": "0xaca630a6c6d4a9e7a7b033174b3952dfd7c2edeea9fa3a356e97d1a5bc31dae0" + "ObjectOwner": "0xd6aea7b7b28562ef76634df18c8c84e7f0d456d0053c9f3264e748aeaaf61ec2" }, "reference": { - "objectId": "0x98c27dcdc3cb082f2ae47756648c92ee054721273663cf0ad25d7f2d4fd20128", + "objectId": "0x7e87e1c8ff4d9ac72fd6af8e55bc23404dbada8ac7ed2e234524e76904f362cb", "version": 1, - "digest": "h6NdgxuL5GUffJJjUicoXB1yDDv3qytM1PJFXQQvxdi" + "digest": "8oJHcJ3vePe99k1VKjrbLxYwmk6LXcSuwvethwzzFreX" } }, { "owner": { - "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" + "AddressOwner": "0x0b37c2167535af8a5729c44690f7be89eb248a6eb3cda5fd37ee1029949f0c29" }, "reference": { - "objectId": "0x9bafea6cfb0a89e0eee6e66ec0d598848067196ed07857012e00e609e8f3ca17", + "objectId": "0x813dd1c1277072ec5f24cc2628513883618b6987fc5435b872cab8f27102329a", "version": 1, - "digest": "3yh2VHtBQACirSUeoFC6YGUimULgQEAeXVHQYzC8423e" + "digest": "4b8z6dtD5Ztw2Ln26qmnrLdgnUcUpJLnBV1Jzxf3Dkqs" } }, { "owner": { - "ObjectOwner": "0x7a85e205f86b500a8597a0e3258cfd36cb5e39c04580f5598e501dc31aa55979" + "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" }, "reference": { - "objectId": "0x9df5ecf6c7c2ed3f31b6ff471aaea04af7d331647c34ab319c8eef45dd043620", + "objectId": "0x9756463d0f40b3b3ac6de7026fb38690ab2b998dfeb72a51cdd563ae0c373f0a", "version": 1, - "digest": "Fnr1EGFNzyCZgEjLJAYYzHfWrxdLdD1epz2ToST4bTQY" + "digest": "GGiQhnbFhpMsyoL61fW7rcA4AYtBDW3QXciHDfCjYdfa" } }, { "owner": { - "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" + "AddressOwner": "0x20227cfbc6699debf187d2d7bca3d253cd50f87c165516d3dc53a246a9af7182" }, "reference": { - "objectId": "0x9ee19bb4564baa0449b5160e4dccd02d5d8a54ca794aaba2130db4feb809c1f5", + "objectId": "0xa4a778c9213d3035bca59c69f380e80fd7196a72a3e2b51cb22dea1411d91401", "version": 1, - "digest": "zoUU8mDtdVPRCqfRKWmd5VpMNfgBHd4jVjq4gu8i9vn" + "digest": "5rfqFb512rV1VZtQ7GGS86hfKe1tR4BAD8FikiPgfHLc" } }, { "owner": { - "ObjectOwner": "0xaca630a6c6d4a9e7a7b033174b3952dfd7c2edeea9fa3a356e97d1a5bc31dae0" + "AddressOwner": "0x7a9730f69d7c9eed4058b7e3ad9019adffdf63a94ef73cd2fb0217f3b9eb6eb1" }, "reference": { - "objectId": "0xa01451ee51efc5ef58d07e5d373b68187ee3a042553670da8a4dc613adf3ed61", + "objectId": "0xa51b7309e9d47b1ba39cddc19a2e5308b1cbaf73112697fbee332a4a66014a85", "version": 1, - "digest": "BUizCaTGo3YEdsuDzodmKs9njztBmrQbpNA5DXiLcXZt" + "digest": "J9Rn2JkDpMZu43WdiN4RtR4VATkLt5a3223N1g8nUKW1" } }, { "owner": { - "ObjectOwner": "0x337b119cf80fabd2c403926df427294e0d1c5444a317a42b1c9dbdbc07375f1b" + "AddressOwner": "0x0b37c2167535af8a5729c44690f7be89eb248a6eb3cda5fd37ee1029949f0c29" }, "reference": { - "objectId": "0xa181feb7b518f4e06794c678eace3633c6b1d7222736e9d059e75ca3023b95a0", + "objectId": "0xa93e761264922c29a33a97b7330bebdfc7eceb438260efd44ff08a7b407d118e", "version": 1, - "digest": "CYWkB4L1JxphwGcDyzdFZQfNcfbhV33jWxNkRxgcquSH" + "digest": "647NGY7b4MXwdWi5krZj2Uhk6tz9MnZSJNu2ivRdfBrf" } }, { "owner": { - "AddressOwner": "0x7a9730f69d7c9eed4058b7e3ad9019adffdf63a94ef73cd2fb0217f3b9eb6eb1" + "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" }, "reference": { - "objectId": "0xa629b978cb416fdeac008d532329429114c70e49909d97bede0f794b376eafe8", + "objectId": "0xa9e70f2bac16567977470169378c3496ac727b9489add6399ba7a568d9d40aa6", "version": 1, - "digest": "6Mtsb7P5EAb696MW6fHanVamZe9zSR8kbnAf7kfZd4n8" + "digest": "2MHZVnXihhSBVxMaw73iiDXR8nTCysGtivdzyAqE3D8f" } }, { "owner": { - "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" + "AddressOwner": "0x0b37c2167535af8a5729c44690f7be89eb248a6eb3cda5fd37ee1029949f0c29" }, "reference": { - "objectId": "0xaca6bf6562b4cbbd909e6f9310c23c9517deca84e82d22dfa781c3dd980bbc95", + "objectId": "0xa9fa3dae3e8fa5eaabe1b29d5af2f0a25490d41677b58c4bedbe1920b03ac7ef", "version": 1, - "digest": "Akr7cJiBjxTYvEm5GttimcB7NVBu8qEU3JEMggT2NkVS" + "digest": "GvqzaN32ixA3ES5RosrYLox3YiYp6UwCxhZzTCwH1Nvv" } }, { "owner": { - "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" + "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" }, "reference": { - "objectId": "0xb12dd88f42ce5db0ba12181deae915d795de5a1e4dfe87deb44a51c8dd8ced7a", + "objectId": "0xbd8eaa142a631472ac39ba6172ad0dd11af7cdb633acdc2e62adbbecf2469578", "version": 1, - "digest": "AqgNX8aLoZyCfmq3N3F3kaMTRWpQCQnZ8uzxJvfckbFM" + "digest": "HcAw9ucPpCnJDk34um1JaYfrtmRB2xy7uUM2WjWHQFH5" } }, { "owner": { - "AddressOwner": "0x0b37c2167535af8a5729c44690f7be89eb248a6eb3cda5fd37ee1029949f0c29" + "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" }, "reference": { - "objectId": "0xc3647488b7f826d5e2602102964e93859bab9374389c1cca8799f91c2b8b1589", + "objectId": "0xc040e5614e1e604d3bc663b9f612a818ebb918aeab40a6ba25caa7a2a6a237b6", "version": 1, - "digest": "65ZbHUaDjGRnqwUVGHiSVeHV1Koxpcmk2pPKv9Uj9wYR" + "digest": "BfpGVgZcxazX2hhxe18NkCiTotN38HcXGxe94N4Lb65v" } }, { @@ -796,79 +794,79 @@ expression: "run_one(cmds, context).await?" "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" }, "reference": { - "objectId": "0xcc36edb95e82c14b7175ae99e58c3a27b43fb1ba7d7f1454fbda1ce66e0e5a8f", + "objectId": "0xc31493f7fd0cbfaad8106574193f492d2727bf48db0602f78b155da5a51402ba", "version": 1, - "digest": "4X4Mmi3CynwtZsuonDK26HvV4SyQ4d4SykRdn1KLu22a" + "digest": "9mbG13HJkdZ1iqLcNYc1UFe2v6haXh3kemomT77LQS2o" } }, { "owner": { - "AddressOwner": "0x0b37c2167535af8a5729c44690f7be89eb248a6eb3cda5fd37ee1029949f0c29" + "AddressOwner": "0x7dbda9c2efa8255eea64cf28b64294ffed6af432f53b661d24f3807895ee828d" }, "reference": { - "objectId": "0xcf326c0a4f56e81beb81e1ee93cda5bd5bc5e88fa4b730a1909679b2ffb3f0d3", + "objectId": "0xc325849c128782699f66a3fea12d317b5486309cae90a5cce5c9a013d504f000", "version": 1, - "digest": "DcBPndnKeykAEYA7cmTz7PW7u2pg46nUB7JDEG2gEaQj" + "digest": "3ubyCZKjxR6RoBSxz5rp11eiWuKXgPU9iVkmccZSL735" } }, { "owner": { - "AddressOwner": "0x20227cfbc6699debf187d2d7bca3d253cd50f87c165516d3dc53a246a9af7182" + "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, "reference": { - "objectId": "0xd3260d5e56e16920f270b3f92a40782447e360ea000aba2af08f0d1f8ad6b5f0", + "objectId": "0xc6e00266120ad997beadd6b8d3ebc6d9f68460ede3e249d5885e108f39898ed6", "version": 1, - "digest": "2W8RQHXqNzdkhKPUCmawicG7iHgvR6BXYMCiktweVruJ" + "digest": "5UUM6FvaxCuzJnnWoSQph4mPRZo7SHxM6ydAemxRcuY9" } }, { "owner": { - "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" + "AddressOwner": "0x20227cfbc6699debf187d2d7bca3d253cd50f87c165516d3dc53a246a9af7182" }, "reference": { - "objectId": "0xd43afda73e2bb89e67d8049618176ba09456438f9fcea5f8a3e91f8a1b17ecd8", + "objectId": "0xcd7e3cb7274220eddb40315c39018c145a4bdc29ae274897178a00e9e5df7719", "version": 1, - "digest": "8PrSccVLMdHmUji4HbGeUvxh8WTHDnzqk75tWz67Tf8d" + "digest": "Hfot87knyUxj4uLhajMUW2Eh83vb6xguhE8q5GqzWwbw" } }, { "owner": { - "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" + "ObjectOwner": "0xb419c5eec03155cddf033685be7ab8bef9fc1e9134d1108f86aab0c0e94148a1" }, "reference": { - "objectId": "0xd8fa8491b91a220af49eb0b54e1d5de725ca40cad8ecca989f27a2f713402490", + "objectId": "0xd813cf320b2cbac0e64c8983484130518715c01961a0bbd2667b055117967dd3", "version": 1, - "digest": "F7NGNkMwvuPkKEuvnYgErENvp1HqQfWkMNySiKssSy6b" + "digest": "4j1Z5GvY1q5qRR2ArCEThV4cFT4BPN7DYgk1hZcwXqE7" } }, { "owner": { - "AddressOwner": "0x20227cfbc6699debf187d2d7bca3d253cd50f87c165516d3dc53a246a9af7182" + "ObjectOwner": "0xd6aea7b7b28562ef76634df18c8c84e7f0d456d0053c9f3264e748aeaaf61ec2" }, "reference": { - "objectId": "0xdde2baa8f84257d47c8051b5abd214afb850211d7ee74a99c21698e8bd66f81d", + "objectId": "0xd9677e045ceed60ae258d0fc79f2e04b3b54cd1e37d2bb4ea5364651d81a354c", "version": 1, - "digest": "3ue34riJ3GWEcmegp2kzVEpnHgWzamPaCZTx7kpjc5ZL" + "digest": "D6o7Zm3PfJ6X92p2m6ZyjAAWVCNg1CQRdSpWRtJAnetM" } }, { "owner": { - "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" + "ObjectOwner": "0xdf12e15860adaae1cd6c238227a585c224ca47ed05239e88f3230bd08ed9b698" }, "reference": { - "objectId": "0xe0ebce954f5f7a4f32e3fa49fa1b76bc1e56008535860b3fe78387db50f95dbd", + "objectId": "0xdf7bf4613216b40876db6e0e41da95724ed3521fdc36a031ba70a66f4fed33d4", "version": 1, - "digest": "kNSMhUBQszVBjveLRYdxX9JEUiLzdV3vCzz9ikaYL1n" + "digest": "66Tjav6hornHGzx5A4HWLPDoDwwkTUEocbML6pVRxkJo" } }, { "owner": { - "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" + "ObjectOwner": "0xd6aea7b7b28562ef76634df18c8c84e7f0d456d0053c9f3264e748aeaaf61ec2" }, "reference": { - "objectId": "0xe360ccacf592e9a2d80c7314d882d2adb850db51fd18572a6baf11a85b233e5d", + "objectId": "0xe2d939c957ebcc0c9a9c0dcad88df409d63291e2260dd3f1a0e2b967558e5745", "version": 1, - "digest": "A2VPHWvMEQHPWw1m5oX6vVZcAwsmkLyCvotCx4MrNjzG" + "digest": "4U4nX8h1bM5BRNzduEDshyRxirtan6VadienujiQyy6n" } }, { @@ -876,19 +874,19 @@ expression: "run_one(cmds, context).await?" "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" }, "reference": { - "objectId": "0xe5395355afde6c5624ce706e91738cb766a8324c86ea933f38c3f8efbba38507", + "objectId": "0xe378d2cec927e9c247bab671285b8f7b23dc02abe786866b2503b3764c9d2ac7", "version": 1, - "digest": "DLCXf4UxpB6CMKQ7tSvdpMLZzAj5z2Qfuox7qHNefKqR" + "digest": "3rVPyctvmQRdKEmDRzvCSaNgAFrU86XPGnmcpDL4TSy7" } }, { "owner": { - "AddressOwner": "0x7dbda9c2efa8255eea64cf28b64294ffed6af432f53b661d24f3807895ee828d" + "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" }, "reference": { - "objectId": "0xe8bcfc0181e426b9dfcad3785fb4db1a0d0a2b1fa40c99a36b5fe988d2a2f561", + "objectId": "0xe72634ff60a76ad75299b6681e5ac68ae16cd5052f55ada644c69f6619ca8f96", "version": 1, - "digest": "2eMPqCumhC9U4G5rJmSr1ZKxftCS264Z8csh9zBbe4aQ" + "digest": "D4VEzzdqbxkjnuLkYKPPgzpiJnMiJMgehjMoTqStigiB" } }, { @@ -896,37 +894,39 @@ expression: "run_one(cmds, context).await?" "AddressOwner": "0x7dbda9c2efa8255eea64cf28b64294ffed6af432f53b661d24f3807895ee828d" }, "reference": { - "objectId": "0xed9ffc59b7e841e0de23ac1b72c4f04a784e2fef9c46f93eed0afdd14f8d047c", + "objectId": "0xe7445ae60304857a2efd0c4767583987a3de4f78a94e893aca46e957a7743c02", "version": 1, - "digest": "69xyHdtHM8S6Voaqv4VHX5p7kxehGDVEAosn236u3EYH" + "digest": "2yTbCF94GoG8oAZ2iL9RGP9gGMur4AKBhG2prqy3iFX5" } }, { - "owner": "Immutable", + "owner": { + "AddressOwner": "0x7a9730f69d7c9eed4058b7e3ad9019adffdf63a94ef73cd2fb0217f3b9eb6eb1" + }, "reference": { - "objectId": "0xeed5eebe436d3b4dedb5d25eff6275c5212fb408ab6e2d1756fd9c56b2af892d", + "objectId": "0xf1fa0fb0d8312d048e280ed73cf0fd19db5a666470ce8ef13ccafbf37f922b6f", "version": 1, - "digest": "HyUxFE4PT5azA17c56oxTLkT81weUpxi5ZkoCxefEbyC" + "digest": "FRv2kzMRCcA85oJAi1MuDrpCUT8DnNHsrZpcsrxEgJG7" } }, { "owner": { - "ObjectOwner": "0x723bc96ed7aa917e131832dd0c60b9b54bb74f5d6c7c5edb89251db0b2881d58" + "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" }, "reference": { - "objectId": "0xf69faa3aa3015380b78c11aae2ee8eed497bb5ee2555e686b53e375ab90fe8cd", + "objectId": "0xf87126f9f001bdf5a14efe58a703fd97d2e2aa6c8ee108243cfcc03f70bec36b", "version": 1, - "digest": "6e9719u3ETKKnmysHMKE5kJoPcV6qGc8gLkoecxbibcm" + "digest": "FQ1iAi2kLuFgs2NEDpYkMGeuB6JJHimhmFWdsvrevRzP" } }, { "owner": { - "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" + "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" }, "reference": { - "objectId": "0xfe1d0d9ef3118023510cdf338ca59ae1d3e88e1ca98a3e1904d61a8620a1ad98", + "objectId": "0xfc3a360f44f32104c49dcf159ba04e021a42b04c969e2ecf3b7c3f8a5d4a4329", "version": 1, - "digest": "7iwW12UgvrywkA79tKwWSZ5AviKqcgKytqz9dWwZLDoh" + "digest": "4jsmns1bceQhaH1iN4mykXv6t4Uz7TELLsHQZDaG9NE4" } } ], @@ -940,12 +940,12 @@ expression: "run_one(cmds, context).await?" "digest": "11111111111111111111111111111111" } }, - "eventsDigest": "652jxdjvsGZDUHS3GoacF4VaCqx614uQNBBNuHSFZsWL" + "eventsDigest": "35Rz6E8uPDfRhMjHh6KWjbms7jhk7Msp6dJNqWgiT9up" }, "events": [ { "id": { - "txDigest": "E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "txDigest": "88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", "eventSeq": "0" }, "packageId": "0x000000000000000000000000000000000000000000000000000000000000107a", @@ -953,13 +953,13 @@ expression: "run_one(cmds, context).await?" "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "type": "0x2::display::DisplayCreated<0x107a::nft::Nft>", "parsedJson": { - "id": "0xeed5eebe436d3b4dedb5d25eff6275c5212fb408ab6e2d1756fd9c56b2af892d" + "id": "0x47be074115c6bd8b19dabd11020c8b59bb7c8a1f64c78ec2a1a3ff366e524b18" }, - "bcs": "H5KEdze24AHvBWWazyGUYQRmhE3zFDzH85xKo3PGYFYg" + "bcs": "5q41VpETxpsuLJmvgz8orZJtx13dupKtKVZKVgKBNm11" }, { "id": { - "txDigest": "E5Zp4QQ84PQEceSw4JRi4VTScSAQweKSgdwp9XH4aVPd", + "txDigest": "88FqW2hyUgShTyLcGzbh6scZB45XZYmXpXSxydkBVTPu", "eventSeq": "1" }, "packageId": "0x000000000000000000000000000000000000000000000000000000000000107a", @@ -1003,10 +1003,10 @@ expression: "run_one(cmds, context).await?" } ] }, - "id": "0xeed5eebe436d3b4dedb5d25eff6275c5212fb408ab6e2d1756fd9c56b2af892d", + "id": "0x47be074115c6bd8b19dabd11020c8b59bb7c8a1f64c78ec2a1a3ff366e524b18", "version": 1 }, - "bcs": "2tpynDmB6yz6PBwHubkQHtuoUCpW2Ko34tEUKCUYU4hDbWkG2JELHHSrtenbXE8EcA9CHviCfpsdnvYbUpg58KW4EJBsb4tzfqEgPzNRmtWxM29JmEfgYygghy3oRsnjudFohMUgoKCCZbka4UHMvpVqKvEz9CKM9KrSvZFMEz6n2vyCUnuD6WZ7nKuTWmZD9SDKC7sx7VNCcE3iPnfN7gTJELP3UT7zNxWF1HnitZPdVdvgQwFmNsxpX881CXboKPSij3yRiLZauDknV6TDvus5QxB9sP7SioVbYSWMZoTdha1FAqRHWBT9r5buiKVj7WxKZiAm3ijfsVS47GbQ1xV3GSTvngJ4yUTp5TxFcdxRU18LsuvkTdMNwo9rwzyn1aKM3rHxCJ3fQgzBPVZ1Q5ENmHHfHpCxwprrYsktWCKM5RHErbTNog6PyB9D1wchgkNqaGvhFNmRP6bittnM33wBgoJ9MhE9CcADVeU" + "bcs": "ZzSKrm3oceLmwZC33JPMzNzgNauuSbGogjTrEYob4dec6NKXhiekQvdweVnVhiG2BXXSYkqFxoKX6n7rZRMfMn2ocXPfNaaVmAYu41kWspKsRkGqB55vC388z4DQaNssaDJn75a1rmfBNebTcB72UFVEMUQt5LGGXuTGmjJirWaNUnaDNmSocfcGY7Zg3KhccL49CGnGefDCW2H9Lb487RtrcJb8xnkDRozsFrw8uVCDGWXcRtXpfrZbkZQr9DVCUw1W3F2bysMki3euUKQdXsxs7agVmZjeNYryrpHTA9fgHDLBtzsMRqQFtScUU3wZ6bYCiqtmh9hsPtStGaoTmznXe6q9jtEpcWfqJq89VV67vyo6Wtd9Mk7VpsL65ib6ArsEhtG9JuJ6a7eCAN1HTtRTuL7zUiVyJm7kW4EdD5jo9RnPXaSzavxzDco3ZrfTVBe9sAfyrKkJ9oZeEu1exUfttgpWjdzbdXWsjA" } ], "objectChanges": [ @@ -1014,7 +1014,7 @@ expression: "run_one(cmds, context).await?" "type": "published", "packageId": "0x0000000000000000000000000000000000000000000000000000000000000001", "version": "1", - "digest": "7XoN6TGQbADiyEaVmrD6gmXJDJkX21MxGMc4pYCo9mKN", + "digest": "85UCZYd34oqKh3x3baQuV7giuUzyNLQvY1qL5UNFQa6t", "modules": [ "address", "ascii", @@ -1040,7 +1040,7 @@ expression: "run_one(cmds, context).await?" "type": "published", "packageId": "0x0000000000000000000000000000000000000000000000000000000000000002", "version": "1", - "digest": "5DVdPqNcu7yg8VSoQy9GwmoUprvHyK87jxnbRoac1r1S", + "digest": "9BM59kZjX2BAq8vFvKDuuZM5eKhtoGRxxXuS1kzFyVR9", "modules": [ "address", "authenticator_state", @@ -1103,7 +1103,7 @@ expression: "run_one(cmds, context).await?" "type": "published", "packageId": "0x0000000000000000000000000000000000000000000000000000000000000003", "version": "1", - "digest": "B7r86UhqxCjrBYVNGnSf7QBNX7EWWm7kLHZ5ht1MLAre", + "digest": "75Le8fZznGd5Bvzxuve3UPQKX2fej1avUJYnA3J1rBjk", "modules": [ "genesis", "iota_system", @@ -1129,7 +1129,7 @@ expression: "run_one(cmds, context).await?" "objectType": "0x3::iota_system::IotaSystemState", "objectId": "0x0000000000000000000000000000000000000000000000000000000000000005", "version": "1", - "digest": "8CWv4f5Nov5aRXTQa2EJk7zsNhfKVbX1fEG8VP6Wsa6d" + "digest": "UT7gYyRZBUj7sgGKbuyMvxTS32HZZvr98vXruhZ9buM" }, { "type": "created", @@ -1142,7 +1142,7 @@ expression: "run_one(cmds, context).await?" "objectType": "0x2::clock::Clock", "objectId": "0x0000000000000000000000000000000000000000000000000000000000000006", "version": "1", - "digest": "DjXTupBThwgJcFY9YDL2Xan9bwTTA3BYrW72Dgh1AMJA" + "digest": "9BNNarX2ktYyq83ySt9hkq5Nr95e3NWdLDaUtFxPv2Lj" }, { "type": "created", @@ -1155,13 +1155,13 @@ expression: "run_one(cmds, context).await?" "objectType": "0x2::random::Random", "objectId": "0x0000000000000000000000000000000000000000000000000000000000000008", "version": "1", - "digest": "aCZvLjShGf5G1AEq3jb9c3y3jxi8NhX17kFUxKwt1q1" + "digest": "7UbMmfwwtbb9RxAtRAuqND8MieggKR6rHn5nvqdqx1HZ" }, { "type": "published", "packageId": "0x000000000000000000000000000000000000000000000000000000000000000b", "version": "1", - "digest": "7rF3B3o4ZHxju9pxz2eGmYhigdg5MBx4WLkyYcBaMMs9", + "digest": "5wsBmq7vkfrRoEqcDxgtJCmLn3YfYRJHP8i6U2EnBgLn", "modules": [ "bridge", "chain_ids", @@ -1184,13 +1184,13 @@ expression: "run_one(cmds, context).await?" "objectType": "0x2::deny_list::DenyList", "objectId": "0x0000000000000000000000000000000000000000000000000000000000000403", "version": "1", - "digest": "57nJzNa4jJBPDpbVVKcntb8rEZLhwhAsLAH7o6r6YLAb" + "digest": "GYMKk7f1KhowX6cq3QBKPZHuosPTMfEXpXDqNDW47zQp" }, { "type": "published", "packageId": "0x000000000000000000000000000000000000000000000000000000000000107a", "version": "1", - "digest": "5Ad5TaF7cAZwf8U3rTD19oxAEfrhptBrSg9qGfYf36nY", + "digest": "EZXQcnmtgSTVm437D9yzcoQ3ToKAeoqD4eJ2TLE8KJxo", "modules": [ "address_unlock_condition", "alias", @@ -1210,120 +1210,111 @@ expression: "run_one(cmds, context).await?" "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x4b2a0b010344ffda7202ecd5f76b742b78516cfcdb208e3314d65a4157654c4b" + "ObjectOwner": "0xc2f868e67e57dae016192c5e39e5da2fc5b091badcf0a2e943f859c736187657" }, - "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x00d74df673e00106b1a45136cf8cfae1ca792c3de53232e06d09fbf5d4f60b66", + "objectType": "0x2::dynamic_field::Field", + "objectId": "0x00f22f34d2616c7f8ba4495bb560e3cf43ebf18cd2ccac12e495f57d1bc6f25c", "version": "1", - "digest": "57LAitmwQMZgioGaRVybfVmZ2Yc57HrJ2Hkm7iGmX5aD" + "digest": "9r9ScS86fEETEc2esEQDGgrGdkC8JSDA1HjAtBA7KGDF" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "ObjectOwner": "0x1d3c44ed95179381aec577194b10aaaccddc14ceedf533b9b6b4eef5535476e0" + "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" }, - "objectType": "0x2::dynamic_field::Field", - "objectId": "0x0214b29e560e3fa6b37836ed7167db0da40a1823cf8b01f0690e1b3bfb4ebdc5", + "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", + "objectId": "0x020f19ebc937d1d40077d2abcde980c8b048ca8dc3ccbac358c7c94b82b20fed", "version": "1", - "digest": "3w6HwgnTmaGDCnAciokuEpZHj2NjFWv6i9KiCHqH7Nct" + "digest": "Ann24wsSiJBWNtRJnib87r4B9XU5eJgRuBPAKDr15X1e" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x20227cfbc6699debf187d2d7bca3d253cd50f87c165516d3dc53a246a9af7182" + "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x04942308c8384041a5af2c13c807418d069d3a6e53519dadd1236f0344385a14", + "objectId": "0x0260aaaecfef6cce63c55d8f470a06dcaf445151a2785888011a8e2194cb4648", "version": "1", - "digest": "811mkHeY9LoDN16a3CSnr22AZK8R2rJa9DbDwuvDp9kd" + "digest": "DNPTGrZEvnFwQBc9SKeTWQfnQy8SouFbUkXooEpdRL8P" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "ObjectOwner": "0xee64ad330164aaa798846df2f2df3525410779fb120765a1c5fb1f931bed784b" + "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, - "objectType": "0x2::dynamic_field::Field", - "objectId": "0x137cb52b1bbef8e93857fd3dc67f7f70f2aabc1d7d6eb4afb28e7dea87b9ccf1", + "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", + "objectId": "0x02ebee2236a4cf262c571124e9a0931ae4c9e236085457e9483698d041a93586", "version": "1", - "digest": "9fG99ZbX8SPLLkLnRcah82PyLY3BSbgbLD7HKD6XntZ4" + "digest": "H9URfhPraVYY3S1LDcGkzwMxCJop5bRJUtSYKrNQGFHe" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" + "ObjectOwner": "0x85c7bca7f3fb7ebfe0d076cdb331ac21abafbcda69065c1abb1056278e0156db" }, - "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x2a74fa6e7b6f90225cb36fcdc3e416d1d3d9e209ab14aafff3afd2c3587c9aec", + "objectType": "0x2::dynamic_field::Field", + "objectId": "0x18f627179e18aed322f828570d83454d3b312e171982c3527b0f41764a367a1d", "version": "1", - "digest": "E37WvCF8d2asqAJH7MtR9rnyiWeHqjgsQqL4gbjAwgLk" + "digest": "2idkUZoaajnS1S1sJtGSszNBzGv45q47sTv5jruCM78S" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" + "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x2ad996aa647ab628cf294b4d98b24c9debb351e72b4a72ee3425b7d9a41d08a8", + "objectId": "0x23dfee0adf69baea358751a2a081bcfa4768ea2b1c7d6a882575076e0791cd2f", "version": "1", - "digest": "F5fPj6gyhcWnNMJTnxis1goTvTNUXYr69zJudZaqMt2t" - }, - { - "type": "created", - "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", - "owner": "Immutable", - "objectType": "0x2::coin::CoinMetadata<0x2::iota::IOTA>", - "objectId": "0x30f537424026a4259959d4c28f4b474558ba881bb517267f9dac6e7bc7518a5d", - "version": "1", - "digest": "CF2vkHV4X8YVrhgVKRafUDX7HmSbxSkvmwFYFPkEE4PM" + "digest": "7b2jvLUFXF5J8Ue2tPoPUvtbtTkhcPps6Xaa77NywDS1" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x7a9730f69d7c9eed4058b7e3ad9019adffdf63a94ef73cd2fb0217f3b9eb6eb1" + "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" }, - "objectType": "0x3::staking_pool::StakedIota", - "objectId": "0x3ad5e2039f28a1d146d5f7e2094b7e55f1cc35f34a53aa207250d51e76febbb4", + "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", + "objectId": "0x24a640dee174b0ed74322070be09e3bb9d96c646858f47de7c1b663496cd6672", "version": "1", - "digest": "2B9M7nEmFp1jcPM72pd7qXNNenQZwdp5dGu4xoykGeXZ" + "digest": "2Q2aeKK2fLQBQX83ihYjpkyEoTDyhEz1QicN2sH1XqKL" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x7a9730f69d7c9eed4058b7e3ad9019adffdf63a94ef73cd2fb0217f3b9eb6eb1" + "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x3e14d7d3f210d79f29a18955f4e98676ebb40e51119ddd6d26a8add99daceb95", + "objectId": "0x2508c2e30a9718a98cef20f0cb58fa74d79bf743a5c73441a70640f96bbf68f0", "version": "1", - "digest": "A6ZSdzLj5Bvv27zaQoFm5rUkS91sK8nJKfntqKbHepUb" + "digest": "3uEZ8L6yNm61aZDW9TU53Gw5ecTBceJHjdFsbbN3PnSc" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" + "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x3e5742903bb62d61adea88aa26bad2fad57984e3eefc58a2f1810b82228e5117", + "objectId": "0x29723a1c473b6f6d2c10a9ccddcd8062a3dbb24b31091470c9341ca36d004c7c", "version": "1", - "digest": "3MVJSV5RKkXLxsTdHqPQERjnmAQAoxkkiRv1Q6G8zHaU" + "digest": "EKpry9R3MSw89yYRN9dd2KbVwVVzXrNWtwgDPw1qbDpu" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" + "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x42cfd74b065c48da45fe6048796ce9cdd3a380c182e7d89573fabc229e3ce4ab", + "objectId": "0x299f315d1f944855d2fb3f3868d18a32cf2e7f243d3a057b2724ce4395106cca", "version": "1", - "digest": "5AzKkrWAReMDgNF2bbiveSDDWvJpQzSX7xHRHNJu9Phy" + "digest": "BLfSDqvn8aNUfAxrWVfdMQSjGsiDqM6Zi4ELFgfRjxTL" }, { "type": "created", @@ -1332,42 +1323,40 @@ expression: "run_one(cmds, context).await?" "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x4728ee263c6a7de9fcd0d4d58db500e0ceed99977f44ca323e9a521970efd085", + "objectId": "0x2b7c5961cbac364bb9711f2791019b594c9110be9c4ed246cab9fedf9bd159e7", "version": "1", - "digest": "781mLWh1jZYNUxtypVJ4XWvHvvga91AZyRfrtXSG53RZ" + "digest": "DR61KQATeDiSkpFWft6ckEC5Cv4YA8ws9W8YoGQDZTGA" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" + "ObjectOwner": "0xc7cf938bf2621982fdbe983a16f7e1ae01358aab0dee88eed62ccf03b0538b35" }, - "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x495af1962cebf47d1eb0cdabb02508114dc8a05c8bb4e6c18b8e81af78c8c4b5", + "objectType": "0x2::dynamic_field::Field", + "objectId": "0x333a99382fa83b4d0e6ed4f25018e848af2f0a48ca06e94948f1aec691011080", "version": "1", - "digest": "A7cf1SRMJYAqceQb78geXHujpHQuiqEggZnWu24enHu1" + "digest": "5EzJs5Q9tzmn73soqnrghEWHZsZRmsC1MoXzsUDLjZje" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x7dbda9c2efa8255eea64cf28b64294ffed6af432f53b661d24f3807895ee828d" + "AddressOwner": "0x20227cfbc6699debf187d2d7bca3d253cd50f87c165516d3dc53a246a9af7182" }, "objectType": "0x3::staking_pool::StakedIota", - "objectId": "0x608863c1d4738cb9829f9bd7c5b3ab15123b6e6eab762dedcfacc23ec23d37c9", + "objectId": "0x45f36b9e92ea7ce24fcf76b9a147f91d6329d19d2b79ba0252bd1be4fa0b3114", "version": "1", - "digest": "6NnLvzoYdm9qBrrsgCj3x2vtrQeMceUt1QuBSvxvgXq5" + "digest": "ku4ZwBs3e3wxPf1jfQ4RDFYhW5WhvGvwURSZNCEBkkd" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", - "owner": { - "ObjectOwner": "0xaca630a6c6d4a9e7a7b033174b3952dfd7c2edeea9fa3a356e97d1a5bc31dae0" - }, - "objectType": "0x2::dynamic_field::Field<0x2::object::ID, address>", - "objectId": "0x660e677bad9d848d1f131d14faef42025eedf0f0e14ede894760e0ebb51a80da", + "owner": "Immutable", + "objectType": "0x2::display::Display<0x107a::nft::Nft>", + "objectId": "0x47be074115c6bd8b19dabd11020c8b59bb7c8a1f64c78ec2a1a3ff366e524b18", "version": "1", - "digest": "EJiD9QzGf9afkBrdYyCfsg9f7Cx64oSLDTVMuvxc2jb2" + "digest": "ACEM96A7XicvdzrRuD4LRm6KyocHATyRKDfKkx9ZaL21" }, { "type": "created", @@ -1376,9 +1365,18 @@ expression: "run_one(cmds, context).await?" "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x6689ff8b1b4ce68bde6f39ebd1031e0339a4036a66b7d0a85c366ac0e6817748", + "objectId": "0x4d03f39deb5e27a76a568adb591da553688e6df6fb053bc9ac069f2bd9495ae3", "version": "1", - "digest": "GYs13Dvcx4XAj76Q4fYfXEgFTyUvVfZYorzvFv8KBRrc" + "digest": "6cjAEhKn8mmgQC6TPSrFASBNUBcSkTscuYpVdBEBbbfz" + }, + { + "type": "created", + "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", + "owner": "Immutable", + "objectType": "0x2::coin::CoinMetadata<0x2::iota::IOTA>", + "objectId": "0x50cb4102d8bc479b9697d6e9e7327e23cb1f00400261f919c8745e49c7732e54", + "version": "1", + "digest": "J9f9eKbfyHs6ks9tGxQPCp41pRfgKy8oRqU4gZb4nMha" }, { "type": "created", @@ -1387,218 +1385,218 @@ expression: "run_one(cmds, context).await?" "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x69cd03a3cc6b2e4a1eae6bef4892f642d8f87ed3d83b758cc5242034bd36f565", + "objectId": "0x566ad3b404129b7740255d9eb77d8f649816cbf8f4fa4f0e0b3088eb40f4be7d", "version": "1", - "digest": "FSfz1Dtc26vvg9ASRGoVHGg1WVbVqLcBtvrcYtTt6bLe" + "digest": "Hg7fVwRrTquxfD5nN6X8jNLbmdE9xiFbw6frCPPmdXWu" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "ObjectOwner": "0x0000000000000000000000000000000000000000000000000000000000000005" + "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, - "objectType": "0x2::dynamic_field::Field", - "objectId": "0x6af2a2b7ca60bf76174adfd3e9c4957f8e937759603182f9b46c7f6c5f19c6d2", + "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", + "objectId": "0x59fbac7e900adfc10b42456542f93bb42ddbcb5bbf56b8427e32ef09d990b184", "version": "1", - "digest": "Er8zLC9VrCrgZZcnhjLkA73mZ3J63rxiboYV6EsqmhN2" + "digest": "GCGVjasFfqD2HKisfq4RWi9FaDRVxwMSf2M7oyT6p9uG" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x0b37c2167535af8a5729c44690f7be89eb248a6eb3cda5fd37ee1029949f0c29" + "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" }, - "objectType": "0x3::staking_pool::StakedIota", - "objectId": "0x6b2a75a39565fe93f6f2694e79274205ec65dac6aecff0a38b3c3947e2c9e561", + "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", + "objectId": "0x63de3e53d6a19a6d42db458e025917b19340a94c638039dc6f771618dee1c6f9", "version": "1", - "digest": "Hy3XSG83B625ywGh161TYQ2VqBJzKxFccgp1fL3Zjbob" + "digest": "Bv4sCpQQgssCQroeqq5e9VctzQiXbefNVPxi9DKwU7Jq" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" + "AddressOwner": "0x7dbda9c2efa8255eea64cf28b64294ffed6af432f53b661d24f3807895ee828d" }, - "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x82054bc6567488b4924203907f0bc62f3f789e1f5edc901798c5a58042dfa1a8", + "objectType": "0x3::staking_pool::StakedIota", + "objectId": "0x65b17c1361e97dd96998ec36253fe21441400d111d790d7b312dfe9364687338", "version": "1", - "digest": "DRYTDtLfuoxayHMXVqxuuWvv4DJnjgZQvoZebdCoGszX" + "digest": "CEMFxNJKgsJoZS5p6djiBcoas2yjWYGe93ib4fmq7mU" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" + "AddressOwner": "0x4b2a0b010344ffda7202ecd5f76b742b78516cfcdb208e3314d65a4157654c4b" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x8a23c86abe72190400e37fd3affc876000ad1302712cbdaf76077f7d541b6400", + "objectId": "0x6665f50841dbe82152d70d15b852a1d2471a83c162830438afb06c3e1717d2c1", "version": "1", - "digest": "2fqjooDKXPoqAFBjzh6cGr53zuTfngNLuHzJoPoR7uvk" + "digest": "5FQAwHKd9BSMAqByENw4AjBPxtu1dUFdvnwFnP5ssefc" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" + "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x9045719b475148ea5e0547b2a0dd8f8aaa19a658deac6bfca7a184d1dfa13d87", + "objectId": "0x6a70af3f08b53d7bd91be9ace2da765e9f7ab0f9046eecd0a92c41897eb742d4", "version": "1", - "digest": "9puPcXBGsDq9hTfkFMUHpmruAkuqsqQmBt6b2ChxRU2f" + "digest": "F244scn8ByNwXxG7Sd2zCkn4TTrGPHy8c1ixYdK1VHyP" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" + "ObjectOwner": "0x0000000000000000000000000000000000000000000000000000000000000005" }, - "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x9135cb3b5aca99a1555b742bd11ddc45fba33343be182bdc161be69da2c41be1", + "objectType": "0x2::dynamic_field::Field", + "objectId": "0x6af2a2b7ca60bf76174adfd3e9c4957f8e937759603182f9b46c7f6c5f19c6d2", "version": "1", - "digest": "5B8YV4P7Q93phJVPgm8MVZzsoXdSxFnKLAjAkMnLYcQb" + "digest": "6AyGKpbs2yy8j6nt6mXVuE7yvhYQvZ1quTi1nhZm67nQ" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "ObjectOwner": "0xaca630a6c6d4a9e7a7b033174b3952dfd7c2edeea9fa3a356e97d1a5bc31dae0" + "AddressOwner": "0x7a9730f69d7c9eed4058b7e3ad9019adffdf63a94ef73cd2fb0217f3b9eb6eb1" }, - "objectType": "0x2::dynamic_field::Field<0x2::object::ID, address>", - "objectId": "0x9254a40663ad7e72d73dd1f04b1eca4e0486cc9f54cdd9a7c3d4e479a8131aa0", + "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", + "objectId": "0x6f651492979fb7f04a2eb053ec8559074c2fa562d8ee922cf2c55277978d5c89", "version": "1", - "digest": "Du8gWCJQvqcsNBEpSYTDSB2aaRyVEG8h7niCPhKQtZFf" + "digest": "21JXcJtash7RUymcy5MvDHHHsWewg6H7Ck36RbcPruKp" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" + "ObjectOwner": "0xd6aea7b7b28562ef76634df18c8c84e7f0d456d0053c9f3264e748aeaaf61ec2" }, - "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x939337494e5c4fe00e78f120459bf108367938719f717e32c845ab83e762f22c", + "objectType": "0x2::dynamic_field::Field<0x2::object::ID, address>", + "objectId": "0x71654ced3581c229f5f2743a808263effc4e63090835f77ca9e8ec4bf667ce5f", "version": "1", - "digest": "5RBuu4Mz4xV3bQAdU8DQnXLppZtpjGToNcoVujUfpxwG" + "digest": "AJXAfxWsnsNRLrdxeARMMXMs7bAw93r1Qu2VG9YB9jH8" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" + "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x94b18ab08b2ddd60e7db0756a605bd31589136eb861b76f49ff3b5ae688f65bc", + "objectId": "0x74f9e1e67cc981675746f2811572b4ea8ce3e6835f49398313b97ad19322f7c5", "version": "1", - "digest": "DXGesd5mWPPaLtwza28GDfumr9NqswcfpbcHp2LaL287" + "digest": "CmYJTpbWV1sJKXaRXA6FnXcaRpaW2eMufwpi1YV3Haet" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "ObjectOwner": "0xaca630a6c6d4a9e7a7b033174b3952dfd7c2edeea9fa3a356e97d1a5bc31dae0" + "ObjectOwner": "0xd6aea7b7b28562ef76634df18c8c84e7f0d456d0053c9f3264e748aeaaf61ec2" }, "objectType": "0x2::dynamic_field::Field<0x2::object::ID, address>", - "objectId": "0x98c27dcdc3cb082f2ae47756648c92ee054721273663cf0ad25d7f2d4fd20128", + "objectId": "0x7e87e1c8ff4d9ac72fd6af8e55bc23404dbada8ac7ed2e234524e76904f362cb", "version": "1", - "digest": "h6NdgxuL5GUffJJjUicoXB1yDDv3qytM1PJFXQQvxdi" + "digest": "8oJHcJ3vePe99k1VKjrbLxYwmk6LXcSuwvethwzzFreX" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" + "AddressOwner": "0x0b37c2167535af8a5729c44690f7be89eb248a6eb3cda5fd37ee1029949f0c29" }, - "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x9bafea6cfb0a89e0eee6e66ec0d598848067196ed07857012e00e609e8f3ca17", + "objectType": "0x3::validator_cap::UnverifiedValidatorOperationCap", + "objectId": "0x813dd1c1277072ec5f24cc2628513883618b6987fc5435b872cab8f27102329a", "version": "1", - "digest": "3yh2VHtBQACirSUeoFC6YGUimULgQEAeXVHQYzC8423e" + "digest": "4b8z6dtD5Ztw2Ln26qmnrLdgnUcUpJLnBV1Jzxf3Dkqs" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "ObjectOwner": "0x7a85e205f86b500a8597a0e3258cfd36cb5e39c04580f5598e501dc31aa55979" + "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" }, - "objectType": "0x2::dynamic_field::Field", - "objectId": "0x9df5ecf6c7c2ed3f31b6ff471aaea04af7d331647c34ab319c8eef45dd043620", + "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", + "objectId": "0x9756463d0f40b3b3ac6de7026fb38690ab2b998dfeb72a51cdd563ae0c373f0a", "version": "1", - "digest": "Fnr1EGFNzyCZgEjLJAYYzHfWrxdLdD1epz2ToST4bTQY" + "digest": "GGiQhnbFhpMsyoL61fW7rcA4AYtBDW3QXciHDfCjYdfa" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" + "AddressOwner": "0x20227cfbc6699debf187d2d7bca3d253cd50f87c165516d3dc53a246a9af7182" }, - "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0x9ee19bb4564baa0449b5160e4dccd02d5d8a54ca794aaba2130db4feb809c1f5", + "objectType": "0x3::validator_cap::UnverifiedValidatorOperationCap", + "objectId": "0xa4a778c9213d3035bca59c69f380e80fd7196a72a3e2b51cb22dea1411d91401", "version": "1", - "digest": "zoUU8mDtdVPRCqfRKWmd5VpMNfgBHd4jVjq4gu8i9vn" + "digest": "5rfqFb512rV1VZtQ7GGS86hfKe1tR4BAD8FikiPgfHLc" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "ObjectOwner": "0xaca630a6c6d4a9e7a7b033174b3952dfd7c2edeea9fa3a356e97d1a5bc31dae0" + "AddressOwner": "0x7a9730f69d7c9eed4058b7e3ad9019adffdf63a94ef73cd2fb0217f3b9eb6eb1" }, - "objectType": "0x2::dynamic_field::Field<0x2::object::ID, address>", - "objectId": "0xa01451ee51efc5ef58d07e5d373b68187ee3a042553670da8a4dc613adf3ed61", + "objectType": "0x3::staking_pool::StakedIota", + "objectId": "0xa51b7309e9d47b1ba39cddc19a2e5308b1cbaf73112697fbee332a4a66014a85", "version": "1", - "digest": "BUizCaTGo3YEdsuDzodmKs9njztBmrQbpNA5DXiLcXZt" + "digest": "J9Rn2JkDpMZu43WdiN4RtR4VATkLt5a3223N1g8nUKW1" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "ObjectOwner": "0x337b119cf80fabd2c403926df427294e0d1c5444a317a42b1c9dbdbc07375f1b" + "AddressOwner": "0x0b37c2167535af8a5729c44690f7be89eb248a6eb3cda5fd37ee1029949f0c29" }, - "objectType": "0x2::dynamic_field::Field", - "objectId": "0xa181feb7b518f4e06794c678eace3633c6b1d7222736e9d059e75ca3023b95a0", + "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", + "objectId": "0xa93e761264922c29a33a97b7330bebdfc7eceb438260efd44ff08a7b407d118e", "version": "1", - "digest": "CYWkB4L1JxphwGcDyzdFZQfNcfbhV33jWxNkRxgcquSH" + "digest": "647NGY7b4MXwdWi5krZj2Uhk6tz9MnZSJNu2ivRdfBrf" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x7a9730f69d7c9eed4058b7e3ad9019adffdf63a94ef73cd2fb0217f3b9eb6eb1" + "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" }, - "objectType": "0x3::validator_cap::UnverifiedValidatorOperationCap", - "objectId": "0xa629b978cb416fdeac008d532329429114c70e49909d97bede0f794b376eafe8", + "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", + "objectId": "0xa9e70f2bac16567977470169378c3496ac727b9489add6399ba7a568d9d40aa6", "version": "1", - "digest": "6Mtsb7P5EAb696MW6fHanVamZe9zSR8kbnAf7kfZd4n8" + "digest": "2MHZVnXihhSBVxMaw73iiDXR8nTCysGtivdzyAqE3D8f" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" + "AddressOwner": "0x0b37c2167535af8a5729c44690f7be89eb248a6eb3cda5fd37ee1029949f0c29" }, - "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0xaca6bf6562b4cbbd909e6f9310c23c9517deca84e82d22dfa781c3dd980bbc95", + "objectType": "0x3::staking_pool::StakedIota", + "objectId": "0xa9fa3dae3e8fa5eaabe1b29d5af2f0a25490d41677b58c4bedbe1920b03ac7ef", "version": "1", - "digest": "Akr7cJiBjxTYvEm5GttimcB7NVBu8qEU3JEMggT2NkVS" + "digest": "GvqzaN32ixA3ES5RosrYLox3YiYp6UwCxhZzTCwH1Nvv" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" + "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0xb12dd88f42ce5db0ba12181deae915d795de5a1e4dfe87deb44a51c8dd8ced7a", + "objectId": "0xbd8eaa142a631472ac39ba6172ad0dd11af7cdb633acdc2e62adbbecf2469578", "version": "1", - "digest": "AqgNX8aLoZyCfmq3N3F3kaMTRWpQCQnZ8uzxJvfckbFM" + "digest": "HcAw9ucPpCnJDk34um1JaYfrtmRB2xy7uUM2WjWHQFH5" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x0b37c2167535af8a5729c44690f7be89eb248a6eb3cda5fd37ee1029949f0c29" + "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" }, - "objectType": "0x3::validator_cap::UnverifiedValidatorOperationCap", - "objectId": "0xc3647488b7f826d5e2602102964e93859bab9374389c1cca8799f91c2b8b1589", + "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", + "objectId": "0xc040e5614e1e604d3bc663b9f612a818ebb918aeab40a6ba25caa7a2a6a237b6", "version": "1", - "digest": "65ZbHUaDjGRnqwUVGHiSVeHV1Koxpcmk2pPKv9Uj9wYR" + "digest": "BfpGVgZcxazX2hhxe18NkCiTotN38HcXGxe94N4Lb65v" }, { "type": "created", @@ -1607,86 +1605,86 @@ expression: "run_one(cmds, context).await?" "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0xcc36edb95e82c14b7175ae99e58c3a27b43fb1ba7d7f1454fbda1ce66e0e5a8f", + "objectId": "0xc31493f7fd0cbfaad8106574193f492d2727bf48db0602f78b155da5a51402ba", "version": "1", - "digest": "4X4Mmi3CynwtZsuonDK26HvV4SyQ4d4SykRdn1KLu22a" + "digest": "9mbG13HJkdZ1iqLcNYc1UFe2v6haXh3kemomT77LQS2o" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x0b37c2167535af8a5729c44690f7be89eb248a6eb3cda5fd37ee1029949f0c29" + "AddressOwner": "0x7dbda9c2efa8255eea64cf28b64294ffed6af432f53b661d24f3807895ee828d" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0xcf326c0a4f56e81beb81e1ee93cda5bd5bc5e88fa4b730a1909679b2ffb3f0d3", + "objectId": "0xc325849c128782699f66a3fea12d317b5486309cae90a5cce5c9a013d504f000", "version": "1", - "digest": "DcBPndnKeykAEYA7cmTz7PW7u2pg46nUB7JDEG2gEaQj" + "digest": "3ubyCZKjxR6RoBSxz5rp11eiWuKXgPU9iVkmccZSL735" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x20227cfbc6699debf187d2d7bca3d253cd50f87c165516d3dc53a246a9af7182" + "AddressOwner": "0x2868fed4dbeb23d2ace3ee3ad6e39061423c5692a2b289b39c643c0baf2d8d85" }, - "objectType": "0x3::validator_cap::UnverifiedValidatorOperationCap", - "objectId": "0xd3260d5e56e16920f270b3f92a40782447e360ea000aba2af08f0d1f8ad6b5f0", + "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", + "objectId": "0xc6e00266120ad997beadd6b8d3ebc6d9f68460ede3e249d5885e108f39898ed6", "version": "1", - "digest": "2W8RQHXqNzdkhKPUCmawicG7iHgvR6BXYMCiktweVruJ" + "digest": "5UUM6FvaxCuzJnnWoSQph4mPRZo7SHxM6ydAemxRcuY9" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" + "AddressOwner": "0x20227cfbc6699debf187d2d7bca3d253cd50f87c165516d3dc53a246a9af7182" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0xd43afda73e2bb89e67d8049618176ba09456438f9fcea5f8a3e91f8a1b17ecd8", + "objectId": "0xcd7e3cb7274220eddb40315c39018c145a4bdc29ae274897178a00e9e5df7719", "version": "1", - "digest": "8PrSccVLMdHmUji4HbGeUvxh8WTHDnzqk75tWz67Tf8d" + "digest": "Hfot87knyUxj4uLhajMUW2Eh83vb6xguhE8q5GqzWwbw" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" + "ObjectOwner": "0xb419c5eec03155cddf033685be7ab8bef9fc1e9134d1108f86aab0c0e94148a1" }, - "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0xd8fa8491b91a220af49eb0b54e1d5de725ca40cad8ecca989f27a2f713402490", + "objectType": "0x2::dynamic_field::Field", + "objectId": "0xd813cf320b2cbac0e64c8983484130518715c01961a0bbd2667b055117967dd3", "version": "1", - "digest": "F7NGNkMwvuPkKEuvnYgErENvp1HqQfWkMNySiKssSy6b" + "digest": "4j1Z5GvY1q5qRR2ArCEThV4cFT4BPN7DYgk1hZcwXqE7" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x20227cfbc6699debf187d2d7bca3d253cd50f87c165516d3dc53a246a9af7182" + "ObjectOwner": "0xd6aea7b7b28562ef76634df18c8c84e7f0d456d0053c9f3264e748aeaaf61ec2" }, - "objectType": "0x3::staking_pool::StakedIota", - "objectId": "0xdde2baa8f84257d47c8051b5abd214afb850211d7ee74a99c21698e8bd66f81d", + "objectType": "0x2::dynamic_field::Field<0x2::object::ID, address>", + "objectId": "0xd9677e045ceed60ae258d0fc79f2e04b3b54cd1e37d2bb4ea5364651d81a354c", "version": "1", - "digest": "3ue34riJ3GWEcmegp2kzVEpnHgWzamPaCZTx7kpjc5ZL" + "digest": "D6o7Zm3PfJ6X92p2m6ZyjAAWVCNg1CQRdSpWRtJAnetM" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x2f1c55f14bcab32b91bccbf9909aa2818d644d4a1f6401c37f97383e4d8275ca" + "ObjectOwner": "0xdf12e15860adaae1cd6c238227a585c224ca47ed05239e88f3230bd08ed9b698" }, - "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0xe0ebce954f5f7a4f32e3fa49fa1b76bc1e56008535860b3fe78387db50f95dbd", + "objectType": "0x2::dynamic_field::Field", + "objectId": "0xdf7bf4613216b40876db6e0e41da95724ed3521fdc36a031ba70a66f4fed33d4", "version": "1", - "digest": "kNSMhUBQszVBjveLRYdxX9JEUiLzdV3vCzz9ikaYL1n" + "digest": "66Tjav6hornHGzx5A4HWLPDoDwwkTUEocbML6pVRxkJo" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" + "ObjectOwner": "0xd6aea7b7b28562ef76634df18c8c84e7f0d456d0053c9f3264e748aeaaf61ec2" }, - "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0xe360ccacf592e9a2d80c7314d882d2adb850db51fd18572a6baf11a85b233e5d", + "objectType": "0x2::dynamic_field::Field<0x2::object::ID, address>", + "objectId": "0xe2d939c957ebcc0c9a9c0dcad88df409d63291e2260dd3f1a0e2b967558e5745", "version": "1", - "digest": "A2VPHWvMEQHPWw1m5oX6vVZcAwsmkLyCvotCx4MrNjzG" + "digest": "4U4nX8h1bM5BRNzduEDshyRxirtan6VadienujiQyy6n" }, { "type": "created", @@ -1695,20 +1693,20 @@ expression: "run_one(cmds, context).await?" "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0xe5395355afde6c5624ce706e91738cb766a8324c86ea933f38c3f8efbba38507", + "objectId": "0xe378d2cec927e9c247bab671285b8f7b23dc02abe786866b2503b3764c9d2ac7", "version": "1", - "digest": "DLCXf4UxpB6CMKQ7tSvdpMLZzAj5z2Qfuox7qHNefKqR" + "digest": "3rVPyctvmQRdKEmDRzvCSaNgAFrU86XPGnmcpDL4TSy7" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0x7dbda9c2efa8255eea64cf28b64294ffed6af432f53b661d24f3807895ee828d" + "AddressOwner": "0x44f42943b20151d6a0c3cb0f84006b3441bb3378c668a6df71fa36cb441538c8" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0xe8bcfc0181e426b9dfcad3785fb4db1a0d0a2b1fa40c99a36b5fe988d2a2f561", + "objectId": "0xe72634ff60a76ad75299b6681e5ac68ae16cd5052f55ada644c69f6619ca8f96", "version": "1", - "digest": "2eMPqCumhC9U4G5rJmSr1ZKxftCS264Z8csh9zBbe4aQ" + "digest": "D4VEzzdqbxkjnuLkYKPPgzpiJnMiJMgehjMoTqStigiB" }, { "type": "created", @@ -1717,40 +1715,42 @@ expression: "run_one(cmds, context).await?" "AddressOwner": "0x7dbda9c2efa8255eea64cf28b64294ffed6af432f53b661d24f3807895ee828d" }, "objectType": "0x3::validator_cap::UnverifiedValidatorOperationCap", - "objectId": "0xed9ffc59b7e841e0de23ac1b72c4f04a784e2fef9c46f93eed0afdd14f8d047c", + "objectId": "0xe7445ae60304857a2efd0c4767583987a3de4f78a94e893aca46e957a7743c02", "version": "1", - "digest": "69xyHdtHM8S6Voaqv4VHX5p7kxehGDVEAosn236u3EYH" + "digest": "2yTbCF94GoG8oAZ2iL9RGP9gGMur4AKBhG2prqy3iFX5" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", - "owner": "Immutable", - "objectType": "0x2::display::Display<0x107a::nft::Nft>", - "objectId": "0xeed5eebe436d3b4dedb5d25eff6275c5212fb408ab6e2d1756fd9c56b2af892d", + "owner": { + "AddressOwner": "0x7a9730f69d7c9eed4058b7e3ad9019adffdf63a94ef73cd2fb0217f3b9eb6eb1" + }, + "objectType": "0x3::validator_cap::UnverifiedValidatorOperationCap", + "objectId": "0xf1fa0fb0d8312d048e280ed73cf0fd19db5a666470ce8ef13ccafbf37f922b6f", "version": "1", - "digest": "HyUxFE4PT5azA17c56oxTLkT81weUpxi5ZkoCxefEbyC" + "digest": "FRv2kzMRCcA85oJAi1MuDrpCUT8DnNHsrZpcsrxEgJG7" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "ObjectOwner": "0x723bc96ed7aa917e131832dd0c60b9b54bb74f5d6c7c5edb89251db0b2881d58" + "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" }, - "objectType": "0x2::dynamic_field::Field", - "objectId": "0xf69faa3aa3015380b78c11aae2ee8eed497bb5ee2555e686b53e375ab90fe8cd", + "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", + "objectId": "0xf87126f9f001bdf5a14efe58a703fd97d2e2aa6c8ee108243cfcc03f70bec36b", "version": "1", - "digest": "6e9719u3ETKKnmysHMKE5kJoPcV6qGc8gLkoecxbibcm" + "digest": "FQ1iAi2kLuFgs2NEDpYkMGeuB6JJHimhmFWdsvrevRzP" }, { "type": "created", "sender": "0x0000000000000000000000000000000000000000000000000000000000000000", "owner": { - "AddressOwner": "0xb4b32db0fb2bc9170f796efb2439aeaf46fd0dc0b79af4e12d14bb87c3e9aa65" + "AddressOwner": "0x4b578d914e64de1ca47bead61bcfa0932646e16de6063089810cfa904eb8b06f" }, "objectType": "0x2::coin::Coin<0x2::iota::IOTA>", - "objectId": "0xfe1d0d9ef3118023510cdf338ca59ae1d3e88e1ca98a3e1904d61a8620a1ad98", + "objectId": "0xfc3a360f44f32104c49dcf159ba04e021a42b04c969e2ecf3b7c3f8a5d4a4329", "version": "1", - "digest": "7iwW12UgvrywkA79tKwWSZ5AviKqcgKytqz9dWwZLDoh" + "digest": "4jsmns1bceQhaH1iN4mykXv6t4Uz7TELLsHQZDaG9NE4" } ], "timestampMs": "1641175496000", diff --git a/crates/iota-faucet/src/faucet/mod.rs b/crates/iota-faucet/src/faucet/mod.rs index 0461473841b..4421ed524a4 100644 --- a/crates/iota-faucet/src/faucet/mod.rs +++ b/crates/iota-faucet/src/faucet/mod.rs @@ -86,49 +86,49 @@ pub const DEFAULT_AMOUNT: u64 = 1_000_000_000; pub const DEFAULT_NUM_OF_COINS: usize = 1; #[derive(Parser, Clone)] -#[clap( - name = "Iota Faucet", - about = "Faucet for requesting test tokens on Iota", +#[command( + name = "IOTA Faucet", + about = "Faucet for requesting test tokens on IOTA", rename_all = "kebab-case" )] pub struct FaucetConfig { - #[clap(long, default_value_t = 5003)] + #[arg(long, default_value_t = 5003)] pub port: u16, - #[clap(long, default_value = "127.0.0.1")] + #[arg(long, default_value = "127.0.0.1")] pub host_ip: Ipv4Addr, - #[clap(long, default_value_t = DEFAULT_AMOUNT)] + #[arg(long, default_value_t = DEFAULT_AMOUNT)] pub amount: u64, - #[clap(long, default_value_t = DEFAULT_NUM_OF_COINS)] + #[arg(long, default_value_t = DEFAULT_NUM_OF_COINS)] pub num_coins: usize, - #[clap(long, default_value_t = 10)] + #[arg(long, default_value_t = 10)] pub request_buffer_size: usize, - #[clap(long, default_value_t = 10)] + #[arg(long, default_value_t = 10)] pub max_request_per_second: u64, - #[clap(long, default_value_t = 60)] + #[arg(long, default_value_t = 60)] pub wallet_client_timeout_secs: u64, - #[clap(long)] + #[arg(long)] pub write_ahead_log: PathBuf, - #[clap(long, default_value_t = 300)] + #[arg(long, default_value_t = 300)] pub wal_retry_interval: u64, - #[clap(long, default_value_t = 10000)] + #[arg(long, default_value_t = 10000)] pub max_request_queue_length: u64, - #[clap(long, default_value_t = 500)] + #[arg(long, default_value_t = 500)] pub batch_request_size: u64, - #[clap(long, default_value_t = 300)] + #[arg(long, default_value_t = 300)] pub ttl_expiration: u64, - #[clap(long, action = clap::ArgAction::Set, default_value_t = false)] + #[arg(long, action = clap::ArgAction::Set, default_value_t = false)] pub batch_enabled: bool, } diff --git a/crates/iota-faucet/src/metrics.rs b/crates/iota-faucet/src/metrics.rs index bc073027669..6ea6085c852 100644 --- a/crates/iota-faucet/src/metrics.rs +++ b/crates/iota-faucet/src/metrics.rs @@ -3,13 +3,14 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 +//! Prometheus metrics which can be displayed in Grafana, queried and alerted +//! on. + use prometheus::{ Histogram, IntCounter, IntGauge, Registry, register_histogram_with_registry, register_int_counter_with_registry, register_int_gauge_with_registry, }; -/// Prometheus metrics which can be displayed in Grafana, queried and alerted on - /// Metrics relevant to the requests coming into the service #[derive(Clone, Debug)] pub struct RequestMetrics { diff --git a/crates/iota-framework-snapshot/bytecode_snapshot/3/0x0000000000000000000000000000000000000000000000000000000000000001 b/crates/iota-framework-snapshot/bytecode_snapshot/3/0x0000000000000000000000000000000000000000000000000000000000000001 new file mode 100644 index 00000000000..6aae6016489 Binary files /dev/null and b/crates/iota-framework-snapshot/bytecode_snapshot/3/0x0000000000000000000000000000000000000000000000000000000000000001 differ diff --git a/crates/iota-framework-snapshot/bytecode_snapshot/3/0x0000000000000000000000000000000000000000000000000000000000000002 b/crates/iota-framework-snapshot/bytecode_snapshot/3/0x0000000000000000000000000000000000000000000000000000000000000002 new file mode 100644 index 00000000000..ea9d0986490 Binary files /dev/null and b/crates/iota-framework-snapshot/bytecode_snapshot/3/0x0000000000000000000000000000000000000000000000000000000000000002 differ diff --git a/crates/iota-framework-snapshot/bytecode_snapshot/3/0x0000000000000000000000000000000000000000000000000000000000000003 b/crates/iota-framework-snapshot/bytecode_snapshot/3/0x0000000000000000000000000000000000000000000000000000000000000003 new file mode 100644 index 00000000000..ddf88e95212 Binary files /dev/null and b/crates/iota-framework-snapshot/bytecode_snapshot/3/0x0000000000000000000000000000000000000000000000000000000000000003 differ diff --git a/crates/iota-framework-snapshot/bytecode_snapshot/3/0x000000000000000000000000000000000000000000000000000000000000000b b/crates/iota-framework-snapshot/bytecode_snapshot/3/0x000000000000000000000000000000000000000000000000000000000000000b new file mode 100644 index 00000000000..a28b3206877 Binary files /dev/null and b/crates/iota-framework-snapshot/bytecode_snapshot/3/0x000000000000000000000000000000000000000000000000000000000000000b differ diff --git a/crates/iota-framework-snapshot/bytecode_snapshot/3/0x000000000000000000000000000000000000000000000000000000000000107a b/crates/iota-framework-snapshot/bytecode_snapshot/3/0x000000000000000000000000000000000000000000000000000000000000107a new file mode 100644 index 00000000000..56d912f6c89 Binary files /dev/null and b/crates/iota-framework-snapshot/bytecode_snapshot/3/0x000000000000000000000000000000000000000000000000000000000000107a differ diff --git a/crates/iota-framework-snapshot/manifest.json b/crates/iota-framework-snapshot/manifest.json index 261265acff2..2f92c9d8340 100644 --- a/crates/iota-framework-snapshot/manifest.json +++ b/crates/iota-framework-snapshot/manifest.json @@ -18,5 +18,15 @@ "0x000000000000000000000000000000000000000000000000000000000000000b", "0x000000000000000000000000000000000000000000000000000000000000107a" ] + }, + "3": { + "git_revision": "378e09e7aa88", + "package_ids": [ + "0x0000000000000000000000000000000000000000000000000000000000000001", + "0x0000000000000000000000000000000000000000000000000000000000000002", + "0x0000000000000000000000000000000000000000000000000000000000000003", + "0x000000000000000000000000000000000000000000000000000000000000000b", + "0x000000000000000000000000000000000000000000000000000000000000107a" + ] } } \ No newline at end of file diff --git a/crates/iota-framework-snapshot/tests/compatibility_tests.rs b/crates/iota-framework-snapshot/tests/compatibility_tests.rs index 9fe13cf62da..01cfcc51bad 100644 --- a/crates/iota-framework-snapshot/tests/compatibility_tests.rs +++ b/crates/iota-framework-snapshot/tests/compatibility_tests.rs @@ -35,7 +35,7 @@ mod compatibility_tests { .is_none() { panic!( - "The current Iota framework {:?} is not compatible with version {:?}", + "The current IOTA framework {:?} is not compatible with version {:?}", cur_package.id(), version ); diff --git a/crates/iota-framework/CONTRIBUTING.md b/crates/iota-framework/CONTRIBUTING.md index 511dff52b67..ab52bea50e6 100644 --- a/crates/iota-framework/CONTRIBUTING.md +++ b/crates/iota-framework/CONTRIBUTING.md @@ -6,11 +6,11 @@ If changes need to be made to the framework's Move code, additional actions need ### Snapshot tests update -Run the following commands in Iota's [root directory](../../) and accept the changes, if any (if you do not have `cargo-insta` command installed, please run the `cargo install cargo-insta` command first): +Run the following commands in IOTA's [root directory](../../) and accept the changes, if any (if you do not have `cargo-insta` command installed, please run the `cargo install cargo-insta` command first): ```bash cargo insta test -p iota-cost --review cargo insta test -p iota-config --review ``` -Please use your best judgment to decide if the changes between old and new versions of the snapshots look "reasonable" (e.g., a minor change in gas costs). When in doubt, please reach out to a member of Iota core team. +Please use your best judgment to decide if the changes between old and new versions of the snapshots look "reasonable" (e.g., a minor change in gas costs). When in doubt, please reach out to a member of IOTA core team. diff --git a/crates/iota-framework/packages/bridge/sources/bridge.move b/crates/iota-framework/packages/bridge/sources/bridge.move index 84c6e794068..dedc2870cc6 100644 --- a/crates/iota-framework/packages/bridge/sources/bridge.move +++ b/crates/iota-framework/packages/bridge/sources/bridge.move @@ -254,7 +254,7 @@ module bridge::bridge { ); } - // Record bridge message approvals in Iota, called by the bridge client + // Record bridge message approvals in IOTA, called by the bridge client // If already approved, return early instead of aborting. public fun approve_token_transfer( bridge: &mut Bridge, @@ -276,7 +276,7 @@ module bridge::bridge { ); let message_key = message.key(); - // retrieve pending message if source chain is Iota, the initial message + // retrieve pending message if source chain is IOTA, the initial message // must exist on chain if (message.source_chain() == inner.chain_id) { let record = &mut inner.token_transfer_records[message_key]; diff --git a/crates/iota-framework/packages/bridge/sources/committee.move b/crates/iota-framework/packages/bridge/sources/committee.move index 1b0e49b14b0..c1383fcfbc5 100644 --- a/crates/iota-framework/packages/bridge/sources/committee.move +++ b/crates/iota-framework/packages/bridge/sources/committee.move @@ -60,7 +60,7 @@ module bridge::committee { } public struct CommitteeMember has copy, drop, store { - /// The Iota Address of the validator + /// The IOTA Address of the validator iota_address: address, /// The public key bytes of the bridge key bridge_pubkey_bytes: vector, @@ -74,7 +74,7 @@ module bridge::committee { } public struct CommitteeMemberRegistration has copy, drop, store { - /// The Iota Address of the validator + /// The IOTA Address of the validator iota_address: address, /// The public key bytes of the bridge key bridge_pubkey_bytes: vector, diff --git a/crates/iota-framework/packages/bridge/sources/limiter.move b/crates/iota-framework/packages/bridge/sources/limiter.move index b4d11d4d5f8..3063226ca34 100644 --- a/crates/iota-framework/packages/bridge/sources/limiter.move +++ b/crates/iota-framework/packages/bridge/sources/limiter.move @@ -177,10 +177,10 @@ module bridge::limiter { // It's tedious to list every pair, but it's safer to do so so we don't // accidentally turn off limiter for a new production route in the future. // Note limiter only takes effects on the receiving chain, so we only need to - // specify routes from Ethereum to Iota. + // specify routes from Ethereum to IOTA. fun initial_transfer_limits(): VecMap { let mut transfer_limits = vec_map::empty(); - // 5M limit on Iota -> Ethereum mainnet + // 5M limit on IOTA -> Ethereum mainnet transfer_limits.insert( chain_ids::get_route(chain_ids::eth_mainnet(), chain_ids::iota_mainnet()), 5_000_000 * USD_VALUE_MULTIPLIER diff --git a/crates/iota-framework/packages/bridge/sources/message.move b/crates/iota-framework/packages/bridge/sources/message.move index f3b2ade6055..c4a04a8b947 100644 --- a/crates/iota-framework/packages/bridge/sources/message.move +++ b/crates/iota-framework/packages/bridge/sources/message.move @@ -385,7 +385,7 @@ module bridge::message { } } - /// Update Iota token message + /// Update IOTA token message /// [message_type:u8] /// [version:u8] /// [nonce:u64] diff --git a/crates/iota-framework/packages/iota-framework/sources/address.move b/crates/iota-framework/packages/iota-framework/sources/address.move index 7dc2aed0873..b9354e96df3 100644 --- a/crates/iota-framework/packages/iota-framework/sources/address.move +++ b/crates/iota-framework/packages/iota-framework/sources/address.move @@ -75,7 +75,7 @@ module iota::address { else abort EAddressParseError } - /// Length of a Iota address in bytes + /// Length of an IOTA address in bytes public fun length(): u64 { LENGTH } diff --git a/crates/iota-framework/packages/iota-framework/sources/bag.move b/crates/iota-framework/packages/iota-framework/sources/bag.move index e672c30b4ab..4adadc8f110 100644 --- a/crates/iota-framework/packages/iota-framework/sources/bag.move +++ b/crates/iota-framework/packages/iota-framework/sources/bag.move @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 /// A bag is a heterogeneous map-like collection. The collection is similar to `iota::table` in that -/// its keys and values are not stored within the `Bag` value, but instead are stored using Iota's +/// its keys and values are not stored within the `Bag` value, but instead are stored using IOTA's /// object system. The `Bag` struct acts only as a handle into the object system to retrieve those /// keys and values. /// Note that this means that `Bag` values with exactly the same key-value mapping will not be diff --git a/crates/iota-framework/packages/iota-framework/sources/coin_manager.move b/crates/iota-framework/packages/iota-framework/sources/coin_manager.move index 6b8ce449c39..348b1378794 100644 --- a/crates/iota-framework/packages/iota-framework/sources/coin_manager.move +++ b/crates/iota-framework/packages/iota-framework/sources/coin_manager.move @@ -26,14 +26,14 @@ module iota::coin_manager { /// The error returned if a attempt is made to change the maximum supply that is lower than the total supply const EMaximumSupplyLowerThanTotalSupply: u64 = 2; - /// The error returned if additional metadata already exists and you try to overwrite - const EAdditionalMetadataAlreadyExists: u64 = 3; + /// The error returned if a attempt is made to change the maximum supply that is higher than the maximum possible supply + const EMaximumSupplyHigherThanPossible: u64 = 3; /// The error returned if you try to edit nonexisting additional metadata const EAdditionalMetadataDoesNotExist: u64 = 4; - /// The error returned if you try to edit immutable metadata - const ENoMutableMetadata: u64 = 5; + /// The maximum supply supported by `CoinManager` + const MAX_SUPPLY: u64 = 18_446_744_073_709_551_614u64; /// Holds all related objects to a Coin in a convenient shared function public struct CoinManager has key, store { @@ -196,7 +196,6 @@ module iota::coin_manager { manager: &mut CoinManager, value: Value ) { - assert!(!df::exists_(&manager.id, b"additional_metadata"), EAdditionalMetadataAlreadyExists); df::add(&mut manager.id, b"additional_metadata", value); } @@ -230,6 +229,7 @@ module iota::coin_manager { maximum_supply: u64 ) { assert!(option::is_none(&manager.maximum_supply), EMaximumSupplyAlreadySet); + assert!(maximum_supply <= MAX_SUPPLY, EMaximumSupplyHigherThanPossible); assert!(total_supply(manager) <= maximum_supply, EMaximumSupplyLowerThanTotalSupply); option::fill(&mut manager.maximum_supply, maximum_supply); } @@ -311,7 +311,7 @@ module iota::coin_manager { /// Get the maximum supply possible as a number. /// If no maximum set it's the maximum u64 possible public fun maximum_supply(manager: &CoinManager): u64 { - option::get_with_default(&manager.maximum_supply, 18_446_744_073_709_551_615u64) + option::get_with_default(&manager.maximum_supply, MAX_SUPPLY) } /// Convenience function returning the remaining supply that can be minted still @@ -383,7 +383,6 @@ module iota::coin_manager { manager: &mut CoinManager, name: string::String ) { - assert!(manager.metadata_is_immutable(), ENoMutableMetadata); coin::update_name(&manager.treasury_cap, option::borrow_mut(&mut manager.metadata), name) } @@ -393,7 +392,6 @@ module iota::coin_manager { manager: &mut CoinManager, symbol: ascii::String ) { - assert!(manager.metadata_is_immutable(), ENoMutableMetadata); coin::update_symbol(&manager.treasury_cap, option::borrow_mut(&mut manager.metadata), symbol) } @@ -403,7 +401,6 @@ module iota::coin_manager { manager: &mut CoinManager, description: string::String ) { - assert!(manager.metadata_is_immutable(), ENoMutableMetadata); coin::update_description(&manager.treasury_cap, option::borrow_mut(&mut manager.metadata), description) } @@ -413,7 +410,6 @@ module iota::coin_manager { manager: &mut CoinManager, url: ascii::String ) { - assert!(manager.metadata_is_immutable(), ENoMutableMetadata); coin::update_icon_url(&manager.treasury_cap, option::borrow_mut(&mut manager.metadata), url) } diff --git a/crates/iota-framework/packages/iota-framework/sources/display.move b/crates/iota-framework/packages/iota-framework/sources/display.move index bf6cc92765a..c557f469f7b 100644 --- a/crates/iota-framework/packages/iota-framework/sources/display.move +++ b/crates/iota-framework/packages/iota-framework/sources/display.move @@ -58,7 +58,7 @@ module iota::display { /// Type signature of the event corresponds to the type while id serves for /// the discovery. /// - /// Since Iota RPC supports querying events by type, finding a Display for the T + /// Since IOTA RPC supports querying events by type, finding a Display for the T /// would be as simple as looking for the first event with `Display`. public struct DisplayCreated has copy, drop { id: ID diff --git a/crates/iota-framework/packages/iota-framework/sources/dynamic_field.move b/crates/iota-framework/packages/iota-framework/sources/dynamic_field.move index 4daaba11071..5a71552837e 100644 --- a/crates/iota-framework/packages/iota-framework/sources/dynamic_field.move +++ b/crates/iota-framework/packages/iota-framework/sources/dynamic_field.move @@ -3,11 +3,11 @@ // SPDX-License-Identifier: Apache-2.0 #[allow(unused_const)] -/// In addition to the fields declared in its type definition, a Iota object can have dynamic fields +/// In addition to the fields declared in its type definition, an IOTA object can have dynamic fields /// that can be added after the object has been constructed. Unlike ordinary field names /// (which are always statically declared identifiers) a dynamic field name can be any value with /// the `copy`, `drop`, and `store` abilities, e.g. an integer, a boolean, or a string. -/// This gives Iota programmers the flexibility to extend objects on-the-fly, and it also serves as a +/// This gives IOTA programmers the flexibility to extend objects on-the-fly, and it also serves as a /// building block for core collection types module iota::dynamic_field { diff --git a/crates/iota-framework/packages/iota-framework/sources/object.move b/crates/iota-framework/packages/iota-framework/sources/object.move index aff3fbbf9f0..ba66ccb9f3c 100644 --- a/crates/iota-framework/packages/iota-framework/sources/object.move +++ b/crates/iota-framework/packages/iota-framework/sources/object.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -/// Iota object identifiers +/// IOTA object identifiers module iota::object { use std::bcs; use iota::address; @@ -25,7 +25,7 @@ module iota::object { /// Allows calling `.to_bytes` on a `UID` to get a `vector`. public use fun uid_to_bytes as UID.to_bytes; - /// The hardcoded ID for the singleton Iota System State Object. + /// The hardcoded ID for the singleton IOTA System State Object. const IOTA_SYSTEM_STATE_OBJECT_ID: address = @0x5; /// The hardcoded ID for the singleton Clock Object. @@ -46,7 +46,7 @@ module iota::object { /// Sender is not @0x0 the system address. const ENotSystemAddress: u64 = 0; - /// An object ID. This is used to reference Iota Objects. + /// An object ID. This is used to reference IOTA Objects. /// This is *not* guaranteed to be globally unique--anyone can create an `ID` from a `UID` or /// from an object, and ID's can be freely copied and dropped. /// Here, the values are not globally unique because there can be multiple values of type `ID` @@ -60,7 +60,7 @@ module iota::object { bytes: address } - /// Globally unique IDs that define an object's ID in storage. Any Iota Object, that is a struct + /// Globally unique IDs that define an object's ID in storage. Any IOTA Object, that is a struct /// with the `key` ability, must have `id: UID` as its first field. /// These are globally unique in the sense that no two values of type `UID` are ever equal, in /// other words for any two values `id1: UID` and `id2: UID`, `id1` != `id2`. @@ -167,7 +167,7 @@ module iota::object { // === any object === - /// Create a new object. Returns the `UID` that must be stored in a Iota object. + /// Create a new object. Returns the `UID` that must be stored in an IOTA object. /// This is the only way to create `UID`s. public fun new(ctx: &mut TxContext): UID { UID { @@ -176,10 +176,10 @@ module iota::object { } /// Delete the object and it's `UID`. This is the only way to eliminate a `UID`. - // This exists to inform Iota of object deletions. When an object + // This exists to inform IOTA of object deletions. When an object // gets unpacked, the programmer will have to do something with its // `UID`. The implementation of this function emits a deleted - // system event so Iota knows to process the object deletion + // system event so IOTA knows to process the object deletion public fun delete(id: UID) { let UID { id: ID { bytes } } = id; delete_impl(bytes) @@ -206,7 +206,7 @@ module iota::object { } /// Get the `UID` for `obj`. - /// Safe because Iota has an extra bytecode verifier pass that forces every struct with + /// Safe because IOTA has an extra bytecode verifier pass that forces every struct with /// the `key` ability to have a distinguished `UID` field. /// Cannot be made public as the access to `UID` for a given object must be privileged, and /// restrictable in the object's module. diff --git a/crates/iota-framework/packages/iota-framework/sources/table.move b/crates/iota-framework/packages/iota-framework/sources/table.move index 944da32a0e4..15a26dbce92 100644 --- a/crates/iota-framework/packages/iota-framework/sources/table.move +++ b/crates/iota-framework/packages/iota-framework/sources/table.move @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 /// A table is a map-like collection. But unlike a traditional collection, it's keys and values are -/// not stored within the `Table` value, but instead are stored using Iota's object system. The +/// not stored within the `Table` value, but instead are stored using IOTA's object system. The /// `Table` struct acts only as a handle into the object system to retrieve those keys and values. /// Note that this means that `Table` values with exactly the same key-value mapping will not be /// equal, with `==`, at runtime. For example diff --git a/crates/iota-framework/packages/iota-framework/sources/test/test_scenario.move b/crates/iota-framework/packages/iota-framework/sources/test/test_scenario.move index da316004a5d..ea484de915e 100644 --- a/crates/iota-framework/packages/iota-framework/sources/test/test_scenario.move +++ b/crates/iota-framework/packages/iota-framework/sources/test/test_scenario.move @@ -41,7 +41,7 @@ module iota::test_scenario { /// Unable to deallocate the receiving ticket const EUnableToDeallocateReceivingTicket: u64 = 7; - /// Utility for mocking a multi-transaction Iota execution in a single Move procedure. + /// Utility for mocking a multi-transaction IOTA execution in a single Move procedure. /// A `Scenario` maintains a view of the global object pool built up by the execution. /// These objects can be accessed via functions like `take_from_sender`, which gives the /// transaction sender access to objects in (only) their inventory. diff --git a/crates/iota-framework/packages/iota-framework/sources/timelock.move b/crates/iota-framework/packages/iota-framework/sources/timelock.move index 2651e782f0a..be530780f3e 100644 --- a/crates/iota-framework/packages/iota-framework/sources/timelock.move +++ b/crates/iota-framework/packages/iota-framework/sources/timelock.move @@ -7,11 +7,10 @@ module iota::timelock { use std::string::{Self, String}; use iota::balance::Balance; + use iota::clock::Clock; use iota::labeler::LabelerCap; use iota::system_admin_cap::IotaSystemAdminCap; - - /// Expiration timestamp of the lock is in the past. - const EExpireEpochIsPast: u64 = 0; + /// The lock has not expired yet. const ENotExpiredYet: u64 = 1; /// For when trying to join two time-locked balances with different expiration time. @@ -34,9 +33,6 @@ module iota::timelock { /// Function to lock an object till a unix timestamp in milliseconds. public fun lock(locked: T, expiration_timestamp_ms: u64, ctx: &mut TxContext): TimeLock { - // Check that `expiration_timestamp_ms` is valid. - check_expiration_timestamp_ms(expiration_timestamp_ms, ctx); - // Create a timelock. pack(locked, expiration_timestamp_ms, option::none(), ctx) } @@ -48,9 +44,6 @@ module iota::timelock { expiration_timestamp_ms: u64, ctx: &mut TxContext ): TimeLock { - // Check that `expiration_timestamp_ms` is valid. - check_expiration_timestamp_ms(expiration_timestamp_ms, ctx); - // Calculate a label value. let label = type_name(); @@ -81,7 +74,7 @@ module iota::timelock { transfer(lock_with_label(labeler, obj, expiration_timestamp_ms, ctx), to); } - /// Function to unlock the object from a `TimeLock`. + /// Function to unlock the object from a `TimeLock` based on the epoch start time. public fun unlock(self: TimeLock, ctx: &TxContext): T { // Unpack the timelock. let (locked, expiration_timestamp_ms, _) = unpack(self); @@ -92,6 +85,17 @@ module iota::timelock { locked } + /// Function to unlock the object from a `TimeLock` based on the `Clock` object. + public fun unlock_with_clock(self: TimeLock, clock: &Clock): T { + // Unpack the timelock. + let (locked, expiration_timestamp_ms, _) = unpack(self); + + // Check if the lock has expired. + assert!(expiration_timestamp_ms <= clock.timestamp_ms(), ENotExpiredYet); + + locked + } + // === TimeLock balance functions === /// Join two `TimeLock>` together. @@ -174,24 +178,32 @@ module iota::timelock { self.expiration_timestamp_ms } - /// Function to check if a `TimeLock` is locked. + /// Function to check if a `TimeLock` is locked based on the epoch start time. public fun is_locked(self: &TimeLock, ctx: &TxContext): bool { self.remaining_time(ctx) > 0 } - /// Function to get the remaining time of a `TimeLock`. + /// Function to get the remaining time of a `TimeLock` based on the epoch start time. /// Returns 0 if the lock has expired. public fun remaining_time(self: &TimeLock, ctx: &TxContext): u64 { // Get the epoch timestamp. let current_timestamp_ms = ctx.epoch_timestamp_ms(); - // Check if the lock has expired. - if (self.expiration_timestamp_ms < current_timestamp_ms) { - return 0 - }; + self.remaining_time_with_timestamp(current_timestamp_ms) + } - // Calculate the remaining time. - self.expiration_timestamp_ms - current_timestamp_ms + /// Function to check if a `TimeLock` is locked based on the `Clock` object. + public fun is_locked_with_clock(self: &TimeLock, clock: &Clock): bool { + self.remaining_time_with_clock(clock) > 0 + } + + /// Function to get the remaining time of a `TimeLock` based on the `Clock` object. + /// Returns 0 if the lock has expired. + public fun remaining_time_with_clock(self: &TimeLock, clock: &Clock): u64 { + // Get the clock's timestamp. + let current_timestamp_ms = clock.timestamp_ms(); + + self.remaining_time_with_timestamp(current_timestamp_ms) } /// Function to get the locked object of a `TimeLock`. @@ -253,12 +265,14 @@ module iota::timelock { transfer::transfer(lock, receiver); } - /// An utility function to check that the `expiration_timestamp_ms` value is valid. - fun check_expiration_timestamp_ms(expiration_timestamp_ms: u64, ctx: &TxContext) { - // Get the epoch timestamp. - let epoch_timestamp_ms = ctx.epoch_timestamp_ms(); + /// An utility function to get the remaining time of a `TimeLock`. + fun remaining_time_with_timestamp(self: &TimeLock, current_timestamp_ms: u64): u64 { + // Check if the lock has expired. + if (self.expiration_timestamp_ms < current_timestamp_ms) { + return 0 + }; - // Check that `expiration_timestamp_ms` is valid. - assert!(expiration_timestamp_ms > epoch_timestamp_ms, EExpireEpochIsPast); + // Calculate the remaining time. + self.expiration_timestamp_ms - current_timestamp_ms } } diff --git a/crates/iota-framework/packages/iota-framework/sources/transfer.move b/crates/iota-framework/packages/iota-framework/sources/transfer.move index 0a941f651a2..884e38ca845 100644 --- a/crates/iota-framework/packages/iota-framework/sources/transfer.move +++ b/crates/iota-framework/packages/iota-framework/sources/transfer.move @@ -43,7 +43,7 @@ module iota::transfer { /// which (in turn) ensures that `obj` has a globally unique ID. Note that if the recipient /// address represents an object ID, the `obj` sent will be inaccessible after the transfer /// (though they will be retrievable at a future date once new features are added). - /// This function has custom rules performed by the Iota Move bytecode verifier that ensures + /// This function has custom rules performed by the IOTA Move bytecode verifier that ensures /// that `T` is an object defined in the module where `transfer` is invoked. Use /// `public_transfer` to transfer an object with `store` outside of its module. public fun transfer(obj: T, recipient: address) { @@ -61,7 +61,7 @@ module iota::transfer { /// Freeze `obj`. After freezing `obj` becomes immutable and can no longer be transferred or /// mutated. - /// This function has custom rules performed by the Iota Move bytecode verifier that ensures + /// This function has custom rules performed by the IOTA Move bytecode verifier that ensures /// that `T` is an object defined in the module where `freeze_object` is invoked. Use /// `public_freeze_object` to freeze an object with `store` outside of its module. public fun freeze_object(obj: T) { @@ -79,7 +79,7 @@ module iota::transfer { /// This is irreversible, i.e. once an object is shared, it will stay shared forever. /// Aborts with `ESharedNonNewObject` of the object being shared was not created in this /// transaction. This restriction may be relaxed in the future. - /// This function has custom rules performed by the Iota Move bytecode verifier that ensures + /// This function has custom rules performed by the IOTA Move bytecode verifier that ensures /// that `T` is an object defined in the module where `share_object` is invoked. Use /// `public_share_object` to share an object with `store` outside of its module. public fun share_object(obj: T) { @@ -98,7 +98,7 @@ module iota::transfer { /// Given mutable (i.e., locked) access to the `parent` and a `Receiving` argument /// referencing an object of type `T` owned by `parent` use the `to_receive` /// argument to receive and return the referenced owned object of type `T`. - /// This function has custom rules performed by the Iota Move bytecode verifier that ensures + /// This function has custom rules performed by the IOTA Move bytecode verifier that ensures /// that `T` is an object defined in the module where `receive` is invoked. Use /// `public_receive` to receivne an object with `store` outside of its module. public fun receive(parent: &mut UID, to_receive: Receiving): T { diff --git a/crates/iota-framework/packages/iota-framework/sources/types.move b/crates/iota-framework/packages/iota-framework/sources/types.move index 4edc5dd3909..dd37f1865e9 100644 --- a/crates/iota-framework/packages/iota-framework/sources/types.move +++ b/crates/iota-framework/packages/iota-framework/sources/types.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -/// Iota types helpers and utilities +/// IOTA types helpers and utilities module iota::types { // === one-time witness === diff --git a/crates/iota-framework/packages/iota-framework/tests/coin_manager_tests.move b/crates/iota-framework/packages/iota-framework/tests/coin_manager_tests.move index 320c347a57c..77d01bdc9e7 100644 --- a/crates/iota-framework/packages/iota-framework/tests/coin_manager_tests.move +++ b/crates/iota-framework/packages/iota-framework/tests/coin_manager_tests.move @@ -4,11 +4,14 @@ #[test_only] module iota::coin_manager_tests { + use std::ascii; + use std::string; + use iota::coin_manager; use iota::coin::{Self, CoinMetadata}; use iota::test_scenario; + use iota::test_utils::assert_eq; use iota::url::{Self, Url}; - use std::ascii::{string}; public struct COIN_MANAGER_TESTS has drop {} @@ -221,6 +224,40 @@ module iota::coin_manager_tests { scenario.end(); } + #[test] + #[expected_failure(abort_code = coin_manager::EMaximumSupplyHigherThanPossible)] + fun test_max_supply_higher_than_maximum() { + let sender = @0xA; + let mut scenario = test_scenario::begin(sender); + let witness = COIN_MANAGER_TESTS{}; + + // Create a `Coin`. + let (cap, meta) = coin::create_currency( + witness, + 0, + b"TEST", + b"TEST", + b"TEST", + option::none(), + scenario.ctx(), + ); + + let (cmcap, metacap, mut wrapper) = coin_manager::new(cap, meta, scenario.ctx()); + + // Check the default maximum supply. + assert_eq(wrapper.maximum_supply(), 18_446_744_073_709_551_614u64); + + // Update the maximum supply to be higher than is maximum possible. + cmcap.enforce_maximum_supply(&mut wrapper, 18_446_744_073_709_551_615u64); + + transfer::public_transfer(cmcap, scenario.ctx().sender()); + metacap.renounce_metadata_ownership(&mut wrapper); + + transfer::public_share_object(wrapper); + + scenario.end(); + } + #[test] #[expected_failure(abort_code = coin_manager::EMaximumSupplyAlreadySet)] fun test_max_supply_once() { @@ -331,7 +368,7 @@ module iota::coin_manager_tests { let (cmcap, metacap, mut wrapper) = coin_manager::new(cap, meta, scenario.ctx()); let bonus = BonusMetadata { - website: url::new_unsafe(string(b"https://example.com")), + website: url::new_unsafe(ascii::string(b"https://example.com")), is_amazing: false }; @@ -340,7 +377,7 @@ module iota::coin_manager_tests { assert!(!wrapper.additional_metadata().is_amazing); let bonus2 = BonusMetadata { - website: url::new_unsafe(string(b"https://iota.org")), + website: url::new_unsafe(ascii::string(b"https://iota.org")), is_amazing: true }; @@ -356,6 +393,49 @@ module iota::coin_manager_tests { scenario.end(); } + + #[test] + #[expected_failure(abort_code = iota::dynamic_field::EFieldAlreadyExists)] + fun test_double_adding_additional_metadata() { + let sender = @0xA; + let mut scenario = test_scenario::begin(sender); + let witness = COIN_MANAGER_TESTS{}; + + // Create a `Coin`. + let (cap, meta) = coin::create_currency( + witness, + 0, + b"TEST", + b"TEST", + b"TEST", + option::none(), + scenario.ctx(), + ); + + let (cmcap, metacap, mut wrapper) = coin_manager::new(cap, meta, scenario.ctx()); + + // Add an additional metadata. + let bonus1 = BonusMetadata { + website: url::new_unsafe(ascii::string(b"https://example1.com")), + is_amazing: false + }; + + metacap.add_additional_metadata(&mut wrapper, bonus1); + + // Add an additional metadata one more time. + let bonus2 = BonusMetadata { + website: url::new_unsafe(ascii::string(b"https://example2.com")), + is_amazing: false + }; + + metacap.add_additional_metadata(&mut wrapper, bonus2); + + cmcap.renounce_treasury_ownership(&mut wrapper); + metacap.renounce_metadata_ownership(&mut wrapper); + transfer::public_share_object(wrapper); + + scenario.end(); + } #[test] fun test_coin_manager_immutable() { @@ -396,4 +476,48 @@ module iota::coin_manager_tests { scenario.end(); } + + #[test] + fun test_coin_manager_update_metadata() { + let sender = @0xA; + let mut scenario = test_scenario::begin(sender); + let witness = COIN_MANAGER_TESTS{}; + + // Create a `Coin`. + let (cap, meta) = coin::create_currency( + witness, + 0, + b"SYMBOL1", + b"NAME1", + b"DESCRIPTION1", + option::some(url::new_unsafe(ascii::string(b"https://url1.com"))), + scenario.ctx(), + ); + + let (cmcap, metacap, mut wrapper) = coin_manager::new(cap, meta, scenario.ctx()); + + // Check the original metadata. + assert_eq(wrapper.name(), string::utf8(b"NAME1")); + assert_eq(wrapper.symbol(), ascii::string(b"SYMBOL1")); + assert_eq(wrapper.description(), string::utf8(b"DESCRIPTION1")); + assert_eq(wrapper.icon_url(), option::some(url::new_unsafe(ascii::string(b"https://url1.com")))); + + // Update the metadata. + coin_manager::update_name(&metacap, &mut wrapper, string::utf8(b"NAME2")); + coin_manager::update_symbol(&metacap, &mut wrapper, ascii::string(b"SYMBOL2")); + coin_manager::update_description(&metacap, &mut wrapper, string::utf8(b"DESCRIPTION2")); + coin_manager::update_icon_url(&metacap, &mut wrapper, ascii::string(b"https://url2.com")); + + // Check the metadata again. + assert_eq(wrapper.name(), string::utf8(b"NAME2")); + assert_eq(wrapper.symbol(), ascii::string(b"SYMBOL2")); + assert_eq(wrapper.description(), string::utf8(b"DESCRIPTION2")); + assert_eq(wrapper.icon_url(), option::some(url::new_unsafe(ascii::string(b"https://url2.com")))); + + transfer::public_transfer(cmcap, scenario.ctx().sender()); + metacap.renounce_metadata_ownership(&mut wrapper); + transfer::public_share_object(wrapper); + + scenario.end(); + } } diff --git a/crates/iota-framework/packages/iota-framework/tests/kiosk/kiosk_test_utils.move b/crates/iota-framework/packages/iota-framework/tests/kiosk/kiosk_test_utils.move index 9db936952b8..2ceeb57f23b 100644 --- a/crates/iota-framework/packages/iota-framework/tests/kiosk/kiosk_test_utils.move +++ b/crates/iota-framework/packages/iota-framework/tests/kiosk/kiosk_test_utils.move @@ -33,7 +33,7 @@ module iota::kiosk_test_utils { (policy, cap) } - /// Prepare: Get Iota + /// Prepare: Get IOTA public fun get_iota(amount: u64, ctx: &mut TxContext): Coin { coin::mint_for_testing(amount, ctx) } diff --git a/crates/iota-framework/packages/iota-framework/tests/timelock/timelock_tests.move b/crates/iota-framework/packages/iota-framework/tests/timelock/timelock_tests.move index 7de604ca427..f417ed90772 100644 --- a/crates/iota-framework/packages/iota-framework/tests/timelock/timelock_tests.move +++ b/crates/iota-framework/packages/iota-framework/tests/timelock/timelock_tests.move @@ -7,9 +7,10 @@ module iota::timelock_tests { use std::string; use iota::balance::{Self, Balance}; + use iota::clock; use iota::iota::IOTA; use iota::test_scenario; - use iota::test_utils::{Self, assert_eq}; + use iota::test_utils::assert_eq; use iota::labeler::LabelerCap; use iota::timelock::{Self, TimeLock}; @@ -128,6 +129,124 @@ module iota::timelock_tests { scenario.end(); } + #[test] + fun test_lock_unlock_flow_with_clock() { + // Set up a test environment. + let sender = @0xA; + let mut scenario = test_scenario::begin(sender); + + // Minting some IOTA. + let iota = balance::create_for_testing(10); + + // Lock the IOTA balance. + let timelock = timelock::lock(iota, 100, scenario.ctx()); + + // Check the locked IOTA. + assert_eq(timelock.locked().value(), 10); + + // Create a clock object. + let mut clock = clock::create_for_testing(scenario.ctx()); + + // Check if the timelock is locked. + assert_eq(timelock.is_locked_with_clock(&clock), true); + assert_eq(timelock.remaining_time_with_clock(&clock), 100); + + // Increment the clock timestamp. + clock.increment_for_testing(10); + + // Check if the timelock is still locked. + assert_eq(timelock.is_locked_with_clock(&clock), true); + assert_eq(timelock.remaining_time_with_clock(&clock), 90); + + // Increment the clock timestamp again. + clock.increment_for_testing(90); + + // Check if the timelock is unlocked. + assert_eq(timelock.is_locked_with_clock(&clock), false); + assert_eq(timelock.remaining_time_with_clock(&clock), 0); + + // Unlock the IOTA balance. + let balance = timelock::unlock_with_clock(timelock, &clock); + + // Check the unlocked IOTA balance. + assert_eq(balance.value(), 10); + + // Cleanup. + balance::destroy_for_testing(balance); + clock ::destroy_for_testing(clock); + + scenario.end(); + } + + #[test] + fun test_lock_unlock_mixed_flow() { + // Set up a test environment. + let sender = @0xA; + let mut scenario = test_scenario::begin(sender); + + // Minting some IOTA. + let iota = balance::create_for_testing(10); + + // Lock the IOTA balance. + let timelock = timelock::lock(iota, 100, scenario.ctx()); + + // Check the locked IOTA. + assert_eq(timelock.locked().value(), 10); + + // Create a clock object. + let mut clock = clock::create_for_testing(scenario.ctx()); + + // Check if the timelock is locked. + assert_eq(timelock.is_locked(scenario.ctx()), true); + assert_eq(timelock.remaining_time(scenario.ctx()), 100); + + assert_eq(timelock.is_locked_with_clock(&clock), true); + assert_eq(timelock.remaining_time_with_clock(&clock), 100); + + // Increment the timestamps. + clock.increment_for_testing(10); + scenario.ctx().increment_epoch_timestamp(10); + + // Check if the timelock is still locked. + assert_eq(timelock.is_locked(scenario.ctx()), true); + assert_eq(timelock.remaining_time(scenario.ctx()), 90); + + assert_eq(timelock.is_locked_with_clock(&clock), true); + assert_eq(timelock.remaining_time_with_clock(&clock), 90); + + // Increment the clock timestamp. + clock.increment_for_testing(90); + + // Check if the timelock is locked according to the epoch time but unlocked by the clock. + assert_eq(timelock.is_locked(scenario.ctx()), true); + assert_eq(timelock.remaining_time(scenario.ctx()), 90); + + assert_eq(timelock.is_locked_with_clock(&clock), false); + assert_eq(timelock.remaining_time_with_clock(&clock), 0); + + // Increment the epoch timestamp. + scenario.ctx().increment_epoch_timestamp(90); + + // Check if the timelock is unlocked. + assert_eq(timelock.is_locked(scenario.ctx()), false); + assert_eq(timelock.remaining_time(scenario.ctx()), 0); + + assert_eq(timelock.is_locked_with_clock(&clock), false); + assert_eq(timelock.remaining_time_with_clock(&clock), 0); + + // Unlock the IOTA balance. + let balance = timelock::unlock(timelock, scenario.ctx()); + + // Check the unlocked IOTA balance. + assert_eq(balance.value(), 10); + + // Cleanup. + balance::destroy_for_testing(balance); + clock ::destroy_for_testing(clock); + + scenario.end(); + } + #[test] fun test_lock_to_unlock_flow() { // Set up a test environment. @@ -263,8 +382,7 @@ module iota::timelock_tests { } #[test] - #[expected_failure(abort_code = timelock::EExpireEpochIsPast)] - fun test_expiration_time_is_passed() { + fun test_unlock_expiration_time_is_passed() { // Set up a test environment. let sender = @0xA; let mut scenario = test_scenario::begin(sender); @@ -275,11 +393,48 @@ module iota::timelock_tests { // Minting some IOTA. let iota = balance::create_for_testing(10); - // Lock the IOTA balance with a wrong expiration time. + // Lock the IOTA balance with an expiration time which is passed. let timelock = timelock::lock(iota, 10, scenario.ctx()); + // Check if the timelock is unlocked. + assert_eq(timelock.is_locked(scenario.ctx()), false); + assert_eq(timelock.remaining_time(scenario.ctx()), 0); + + // Unlock the IOTA balance. + let balance = timelock::unlock(timelock, scenario.ctx()); + // Cleanup. - test_utils::destroy(timelock); + balance::destroy_for_testing(balance); + + scenario.end(); + } + + #[test] + fun test_unlock_with_clock_expiration_time_is_passed() { + // Set up a test environment. + let sender = @0xA; + let mut scenario = test_scenario::begin(sender); + + // Create a clock object. + let mut clock = clock::create_for_testing(scenario.ctx()); + clock.increment_for_testing(100); + + // Minting some IOTA. + let iota = balance::create_for_testing(10); + + // Lock the IOTA balance with an expiration time which is passed. + let timelock = timelock::lock(iota, 10, scenario.ctx()); + + // Check if the timelock is unlocked. + assert_eq(timelock.is_locked_with_clock(&clock), false); + assert_eq(timelock.remaining_time_with_clock(&clock), 0); + + // Unlock the IOTA balance. + let balance = timelock::unlock_with_clock(timelock, &clock); + + // Cleanup. + balance::destroy_for_testing(balance); + clock::destroy_for_testing(clock); scenario.end(); } @@ -308,4 +463,31 @@ module iota::timelock_tests { scenario.end(); } + + #[test] + #[expected_failure(abort_code = timelock::ENotExpiredYet)] + fun test_unlock_not_expired_object_with_clock() { + // Set up a test environment. + let sender = @0xA; + let mut scenario = test_scenario::begin(sender); + + // Minting some IOTA. + let iota = balance::create_for_testing(10); + + // Lock the IOTA balance. + let timelock = timelock::lock(iota, 100, scenario.ctx()); + + // Create a clock object. + let mut clock = clock::create_for_testing(scenario.ctx()); + clock.increment_for_testing(10); + + // Unlock the IOTA balance which is not expired. + let balance = timelock::unlock_with_clock(timelock, &clock); + + // Cleanup. + balance::destroy_for_testing(balance); + clock::destroy_for_testing(clock); + + scenario.end(); + } } diff --git a/crates/iota-framework/packages/iota-framework/tests/verifier_tests.move b/crates/iota-framework/packages/iota-framework/tests/verifier_tests.move index 9200fad14b1..10eb0fc2b91 100644 --- a/crates/iota-framework/packages/iota-framework/tests/verifier_tests.move +++ b/crates/iota-framework/packages/iota-framework/tests/verifier_tests.move @@ -3,7 +3,7 @@ // SPDX-License-Identifier: Apache-2.0 #[test_only] -/// Tests if normally illegal (in terms of Iota bytecode verification) code is allowed in tests. +/// Tests if normally illegal (in terms of IOTA bytecode verification) code is allowed in tests. module iota::verifier_tests { public struct VERIFIER_TESTS has drop {} diff --git a/crates/iota-framework/packages/iota-system/sources/iota_system.move b/crates/iota-framework/packages/iota-system/sources/iota_system.move index 573ffddb165..81876d03abd 100644 --- a/crates/iota-framework/packages/iota-system/sources/iota_system.move +++ b/crates/iota-framework/packages/iota-system/sources/iota_system.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -/// Iota System State Type Upgrade Guide +/// IOTA System State Type Upgrade Guide /// `IotaSystemState` is a thin wrapper around `IotaSystemStateV1` that provides a versioned interface. /// The `IotaSystemState` object has a fixed ID 0x5, and the `IotaSystemStateV1` object is stored as a dynamic field. /// There are a few different ways to upgrade the `IotaSystemStateV1` type: diff --git a/crates/iota-framework/packages/iota-system/sources/iota_system_state_inner.move b/crates/iota-framework/packages/iota-system/sources/iota_system_state_inner.move index 2097871e915..b4840789b09 100644 --- a/crates/iota-framework/packages/iota-system/sources/iota_system_state_inner.move +++ b/crates/iota-framework/packages/iota-system/sources/iota_system_state_inner.move @@ -58,7 +58,7 @@ module iota_system::iota_system_state_inner { extra_fields: Bag, } - /// The top-level object containing all information of the Iota system. + /// The top-level object containing all information of the IOTA system. public struct IotaSystemStateV1 has store { /// The current epoch ID, starting from 0. epoch: u64, diff --git a/crates/iota-framework/packages/iota-system/sources/validator.move b/crates/iota-framework/packages/iota-system/sources/validator.move index 3e19f8fddf8..cbba81c6c0a 100644 --- a/crates/iota-framework/packages/iota-system/sources/validator.move +++ b/crates/iota-framework/packages/iota-system/sources/validator.move @@ -72,7 +72,7 @@ module iota_system::validator { const MAX_VALIDATOR_GAS_PRICE: u64 = 100_000; public struct ValidatorMetadataV1 has store { - /// The Iota Address of the validator. This is the sender that created the ValidatorV1 object, + /// The IOTA Address of the validator. This is the sender that created the ValidatorV1 object, /// and also the address to send validator/coins to during withdraws. iota_address: address, /// The public key bytes corresponding to the private key that the validator diff --git a/crates/iota-framework/packages/move-stdlib/sources/address.move b/crates/iota-framework/packages/move-stdlib/sources/address.move index 924a1d939dd..79e0c960c42 100644 --- a/crates/iota-framework/packages/move-stdlib/sources/address.move +++ b/crates/iota-framework/packages/move-stdlib/sources/address.move @@ -6,7 +6,7 @@ /// platform-specific parameter. module std::address { /// Should be converted to a native function. - /// Current implementation only works for Iota. + /// Current implementation only works for IOTA. public fun length(): u64 { 32 } diff --git a/crates/iota-framework/packages/stardust/sources/basic/basic_output.move b/crates/iota-framework/packages/stardust/sources/basic/basic_output.move index 10e3dad75d7..e322baadf41 100644 --- a/crates/iota-framework/packages/stardust/sources/basic/basic_output.move +++ b/crates/iota-framework/packages/stardust/sources/basic/basic_output.move @@ -4,7 +4,7 @@ module stardust::basic_output { // === Imports === - // Iota imports. + // IOTA imports. use iota::bag::Bag; use iota::balance::Balance; use iota::transfer::Receiving; diff --git a/crates/iota-framework/packages/stardust/sources/nft/nft.move b/crates/iota-framework/packages/stardust/sources/nft/nft.move index 5ffbd7677d1..a257dd7dcd8 100644 --- a/crates/iota-framework/packages/stardust/sources/nft/nft.move +++ b/crates/iota-framework/packages/stardust/sources/nft/nft.move @@ -39,7 +39,7 @@ module stardust::nft { // Build a `Display` object. let keys = vector[ - // The Iota standard fields. + // The IOTA standard fields. string::utf8(b"name"), string::utf8(b"image_url"), string::utf8(b"description"), @@ -55,7 +55,7 @@ module stardust::nft { ]; let values = vector[ - // The Iota standard fields. + // The IOTA standard fields. string::utf8(b"{immutable_metadata.name}"), string::utf8(b"{immutable_metadata.uri}"), string::utf8(b"{immutable_metadata.description}"), diff --git a/crates/iota-framework/packages_compiled/iota-framework b/crates/iota-framework/packages_compiled/iota-framework index b1dfb3584d4..502259e0382 100644 Binary files a/crates/iota-framework/packages_compiled/iota-framework and b/crates/iota-framework/packages_compiled/iota-framework differ diff --git a/crates/iota-framework/published_api.txt b/crates/iota-framework/published_api.txt index 72299612773..b26c1ea49ef 100644 --- a/crates/iota-framework/published_api.txt +++ b/crates/iota-framework/published_api.txt @@ -3088,6 +3088,9 @@ lock_with_label_and_transfer unlock public fun 0x2::timelock +unlock_with_clock + public fun + 0x2::timelock join public fun 0x2::timelock @@ -3121,6 +3124,12 @@ is_locked remaining_time public fun 0x2::timelock +is_locked_with_clock + public fun + 0x2::timelock +remaining_time_with_clock + public fun + 0x2::timelock locked public fun 0x2::timelock @@ -3139,7 +3148,7 @@ unpack transfer fun 0x2::timelock -check_expiration_timestamp_ms +remaining_time_with_timestamp fun 0x2::timelock Token diff --git a/crates/iota-framework/tests/build-system-packages.rs b/crates/iota-framework/tests/build-system-packages.rs index 8e343a0e33f..aefc06ab5d7 100644 --- a/crates/iota-framework/tests/build-system-packages.rs +++ b/crates/iota-framework/tests/build-system-packages.rs @@ -198,12 +198,17 @@ fn build_packages_with_move_config( &mut files_to_write, ); create_category_file(framework_dir); - create_category_file(stdlib_dir); relocate_docs( framework_dir, &framework_pkg.package.compiled_docs.unwrap(), &mut files_to_write, ); + create_category_file(stdlib_dir); + relocate_docs( + stdlib_dir, + &stdlib_pkg.package.compiled_docs.unwrap(), + &mut files_to_write, + ); create_category_file(bridge_dir); relocate_docs( bridge_dir, diff --git a/crates/iota-genesis-builder/examples/build_stardust_genesis.rs b/crates/iota-genesis-builder/examples/build_stardust_genesis.rs index 2eb01e6e3df..79717c36516 100644 --- a/crates/iota-genesis-builder/examples/build_stardust_genesis.rs +++ b/crates/iota-genesis-builder/examples/build_stardust_genesis.rs @@ -14,11 +14,11 @@ use iota_swarm_config::genesis_config::ValidatorGenesisConfigBuilder; use rand::rngs::OsRng; #[derive(Parser, Debug)] -#[clap( +#[command( about = "Example Tool for generating a genesis file from a Stardust Migration Objects snapshot" )] struct Cli { - #[clap(long, default_value_t = OBJECT_SNAPSHOT_FILE_PATH.to_string(), help = "Path to the Stardust Migration Objects snapshot file")] + #[arg(long, default_value_t = OBJECT_SNAPSHOT_FILE_PATH.to_string(), help = "Path to the Stardust Migration Objects snapshot file")] snapshot_path: String, } diff --git a/crates/iota-genesis-builder/examples/build_stardust_genesis_from_s3.rs b/crates/iota-genesis-builder/examples/build_stardust_genesis_from_s3.rs index 8c12bb1798b..231979ccb9f 100644 --- a/crates/iota-genesis-builder/examples/build_stardust_genesis_from_s3.rs +++ b/crates/iota-genesis-builder/examples/build_stardust_genesis_from_s3.rs @@ -11,10 +11,10 @@ use tracing::{Level, info}; use tracing_subscriber::FmtSubscriber; #[derive(Parser, Debug)] -#[clap(about = "Example that builds genesis in a temp folder with migration snapshots")] +#[command(about = "Example that builds genesis in a temp folder with migration snapshots")] struct Args { /// Remotely stored migration snapshots. - #[clap( + #[arg( long, name = "iota|", help = "Remote migration snapshots.", diff --git a/crates/iota-genesis-builder/examples/snapshot_add_test_outputs.rs b/crates/iota-genesis-builder/examples/snapshot_add_test_outputs.rs index 3600e31afea..0913782ed32 100644 --- a/crates/iota-genesis-builder/examples/snapshot_add_test_outputs.rs +++ b/crates/iota-genesis-builder/examples/snapshot_add_test_outputs.rs @@ -13,17 +13,17 @@ use iota_genesis_builder::stardust::{ use iota_types::{gas_coin::STARDUST_TOTAL_SUPPLY_IOTA, stardust::coin_type::CoinType}; #[derive(Parser, Debug)] -#[clap(about = "Tool for adding test data to Iota Hornet full-snapshot")] +#[command(about = "Tool for adding test data to IOTA Hornet full-snapshot")] struct Cli { - #[clap(subcommand)] + #[command(subcommand)] snapshot: Snapshot, } #[derive(Subcommand, Debug)] enum Snapshot { - #[clap(about = "Parse an Iota Hornet full-snapshot file")] + #[command(about = "Parse an IOTA Hornet full-snapshot file")] Iota { - #[clap(long, help = "Path to the Iota Hornet full-snapshot file")] + #[arg(long, help = "Path to the IOTA Hornet full-snapshot file")] snapshot_path: String, }, } diff --git a/crates/iota-genesis-builder/examples/snapshot_only_test_outputs.rs b/crates/iota-genesis-builder/examples/snapshot_only_test_outputs.rs index 55b25187a01..eae82e48d02 100644 --- a/crates/iota-genesis-builder/examples/snapshot_only_test_outputs.rs +++ b/crates/iota-genesis-builder/examples/snapshot_only_test_outputs.rs @@ -7,31 +7,32 @@ use std::{fs::File, path::Path}; use clap::{Parser, Subcommand}; -use iota_genesis_builder::{ - IF_STARDUST_ADDRESS, - stardust::{ - parse::HornetSnapshotParser, - test_outputs::{add_snapshot_test_outputs, to_nanos}, - }, +use iota_genesis_builder::stardust::{ + parse::HornetSnapshotParser, + test_outputs::{add_snapshot_test_outputs, to_nanos}, +}; +use iota_sdk::types::block::address::Ed25519Address; +use iota_types::{ + base_types::IotaAddress, gas_coin::STARDUST_TOTAL_SUPPLY_IOTA, stardust::coin_type::CoinType, }; -use iota_sdk::types::block::address::Address; -use iota_types::{gas_coin::STARDUST_TOTAL_SUPPLY_IOTA, stardust::coin_type::CoinType}; const WITH_SAMPLING: bool = false; #[derive(Parser, Debug)] -#[clap(about = "Tool for generating Iota Hornet full-snapshot file with test data")] +#[command(about = "Tool for generating IOTA Hornet full-snapshot file with test data")] struct Cli { - #[clap(subcommand)] + #[command(subcommand)] snapshot: Snapshot, } #[derive(Subcommand, Debug)] enum Snapshot { - #[clap(about = "Parse an Iota Hornet full-snapshot file")] + #[command(about = "Parse an IOTA Hornet full-snapshot file")] Iota { - #[clap(long, help = "Path to the Iota Hornet full-snapshot file")] + #[arg(long, help = "Path to the IOTA Hornet full-snapshot file")] snapshot_path: String, + #[arg(long, help = "Specify the delegator address")] + delegator: IotaAddress, }, } @@ -62,8 +63,11 @@ fn parse_snapshot( #[tokio::main] async fn main() -> anyhow::Result<()> { let cli = Cli::parse(); - let (current_path, coin_type) = match cli.snapshot { - Snapshot::Iota { snapshot_path } => (snapshot_path, CoinType::Iota), + let (current_path, delegator, coin_type) = match cli.snapshot { + Snapshot::Iota { + snapshot_path, + delegator, + } => (snapshot_path, delegator, CoinType::Iota), }; let mut new_path = String::from("test-"); // prepend "test-" before the file name @@ -80,7 +84,7 @@ async fn main() -> anyhow::Result<()> { let (randomness_seed, delegator_address) = match coin_type { CoinType::Iota => { // IOTA coin type values - (0, IF_STARDUST_ADDRESS) + (0, delegator) } }; @@ -89,7 +93,7 @@ async fn main() -> anyhow::Result<()> { &new_path, coin_type, randomness_seed, - *Address::try_from_bech32(delegator_address)?.as_ed25519(), + Ed25519Address::from(delegator_address.to_inner()), WITH_SAMPLING, ) .await?; diff --git a/crates/iota-genesis-builder/src/lib.rs b/crates/iota-genesis-builder/src/lib.rs index ac464c309ce..69de4c1e0be 100644 --- a/crates/iota-genesis-builder/src/lib.rs +++ b/crates/iota-genesis-builder/src/lib.rs @@ -19,8 +19,8 @@ use genesis_build_effects::GenesisBuildEffects; use iota_config::{ IOTA_GENESIS_MIGRATION_TX_DATA_FILENAME, genesis::{ - Genesis, GenesisCeremonyParameters, GenesisChainParameters, TokenDistributionSchedule, - UnsignedGenesis, + Delegations, Genesis, GenesisCeremonyParameters, GenesisChainParameters, + TokenDistributionSchedule, UnsignedGenesis, }, migration_tx_data::{MigrationTxData, TransactionsData}, }; @@ -28,7 +28,7 @@ use iota_execution::{self, Executor}; use iota_framework::{BuiltInFramework, SystemPackage}; use iota_genesis_common::{execute_genesis_transaction, get_genesis_protocol_config}; use iota_protocol_config::{Chain, ProtocolConfig, ProtocolVersion}; -use iota_sdk::{Url, types::block::address::Address}; +use iota_sdk::Url; use iota_types::{ BRIDGE_ADDRESS, IOTA_BRIDGE_OBJECT_ID, IOTA_FRAMEWORK_PACKAGE_ID, IOTA_SYSTEM_ADDRESS, balance::{BALANCE_MODULE_NAME, Balance}, @@ -63,7 +63,6 @@ use iota_types::{ object::{Object, Owner}, programmable_transaction_builder::ProgrammableTransactionBuilder, randomness_state::{RANDOMNESS_MODULE_NAME, RANDOMNESS_STATE_CREATE_FUNCTION_NAME}, - stardust::stardust_to_iota_address, system_admin_cap::IOTA_SYSTEM_ADMIN_CAP_MODULE_NAME, timelock::{ stardust_upgrade_label::STARDUST_UPGRADE_LABEL_VALUE, @@ -78,7 +77,7 @@ use move_binary_format::CompiledModule; use move_core_types::ident_str; use serde::{Deserialize, Serialize}; use shared_crypto::intent::{Intent, IntentMessage, IntentScope}; -use stake::{GenesisStake, delegate_genesis_stake}; +use stake::GenesisStake; use stardust::migration::MigrationObjects; use tracing::trace; use validator_info::{GenesisValidatorInfo, GenesisValidatorMetadata, ValidatorInfo}; @@ -88,16 +87,14 @@ mod stake; pub mod stardust; pub mod validator_info; -// TODO: Lazy static `stardust_to_iota_address` -pub const IF_STARDUST_ADDRESS: &str = - "iota1qp8h9augeh6tk3uvlxqfapuwv93atv63eqkpru029p6sgvr49eufyz7katr"; - const GENESIS_BUILDER_COMMITTEE_DIR: &str = "committee"; pub const GENESIS_BUILDER_PARAMETERS_FILE: &str = "parameters"; const GENESIS_BUILDER_TOKEN_DISTRIBUTION_SCHEDULE_FILE: &str = "token-distribution-schedule"; const GENESIS_BUILDER_SIGNATURE_DIR: &str = "signatures"; const GENESIS_BUILDER_UNSIGNED_GENESIS_FILE: &str = "unsigned-genesis"; const GENESIS_BUILDER_MIGRATION_SOURCES_FILE: &str = "migration-sources"; +const GENESIS_BUILDER_DELEGATOR_FILE: &str = "delegator"; +const GENESIS_BUILDER_DELEGATOR_MAP_FILE: &str = "delegator-map"; pub const OBJECT_SNAPSHOT_FILE_PATH: &str = "stardust_object_snapshot.bin"; pub const IOTA_OBJECT_SNAPSHOT_URL: &str = "https://stardust-objects.s3.eu-central-1.amazonaws.com/iota/alphanet/latest/stardust_object_snapshot.bin.gz"; @@ -118,6 +115,15 @@ pub struct Builder { genesis_stake: GenesisStake, migration_sources: Vec, migration_tx_data: Option, + delegation: Option, +} + +enum GenesisDelegation { + /// Represents a single delegator address that applies to all validators. + OneToAll(IotaAddress), + /// Represents a map of delegator addresses to validator addresses and + /// a specified stake and gas allocation. + ManyToMany(Delegations), } impl Default for Builder { @@ -139,9 +145,20 @@ impl Builder { genesis_stake: Default::default(), migration_sources: Default::default(), migration_tx_data: Default::default(), + delegation: None, } } + pub fn with_delegator(mut self, delegator: IotaAddress) -> Self { + self.delegation = Some(GenesisDelegation::OneToAll(delegator)); + self + } + + pub fn with_delegations(mut self, delegations: Delegations) -> Self { + self.delegation = Some(GenesisDelegation::ManyToMany(delegations)); + self + } + /// Checks if the genesis to be built has no migration or if it includes /// Stardust migration stakes pub fn contains_migrations(&self) -> bool { @@ -253,19 +270,32 @@ impl Builder { /// Create and cache the [`GenesisStake`] if the builder /// contains migrated objects. + /// + /// Two cases can happen here: + /// 1. if a delegator map is given as input -> then use the map input to + /// create and cache the genesis stake. + /// 2. if a delegator map is NOT given as input -> then use one default + /// delegator passed as input and delegate the minimum required stake to + /// all validators to create and cache the genesis stake. fn create_and_cache_genesis_stake(&mut self) -> anyhow::Result<()> { if !self.migration_objects.is_empty() { - let delegator = - stardust_to_iota_address(Address::try_from_bech32(IF_STARDUST_ADDRESS).unwrap()) - .unwrap(); - // TODO: check whether we need to start with - // VALIDATOR_LOW_STAKE_THRESHOLD_NANOS - let minimum_stake = iota_types::governance::MIN_VALIDATOR_JOINING_STAKE_NANOS; - self.genesis_stake = delegate_genesis_stake( - self.validators.values(), - delegator, + self.genesis_stake = GenesisStake::new_with_delegations( + match &self.delegation { + Some(GenesisDelegation::ManyToMany(delegations)) => { + // Case 1 -> use the delegations input to create and cache the genesis stake + delegations.clone() + } + Some(GenesisDelegation::OneToAll(delegator)) => { + // Case 2 -> use one default delegator passed as input and delegate the + // minimum required stake to all validators to create the genesis stake + Delegations::new_for_validators_with_default_allocation( + self.validators.values().map(|v| v.info.iota_address()), + *delegator, + ) + } + None => bail!("no delegator/s assigned with a migration"), + }, &self.migration_objects, - minimum_stake, )?; } Ok(()) @@ -273,55 +303,76 @@ impl Builder { /// Evaluate the genesis [`TokenDistributionSchedule`]. /// - /// This merges conditionally the cached token distribution - /// (i.e. `self.token_distribution_schedule`) with the genesis stake - /// resulting from the migrated state. - /// - /// If the cached token distribution schedule contains timelocked stake, it - /// is assumed that the genesis stake is already merged and no operation - /// is performed. This is the case where we load a [`Builder`] from disk - /// that has already built genesis with the migrated state. + /// There are 6 cases for evaluating this: + /// 1. The genesis is built WITHOUT migration + /// 1. and a schedule is given as input -> then just use the input + /// schedule; + /// 2. and the schedule is NOT given as input -> then instantiate a + /// default token distribution schedule for a genesis without + /// migration. + /// 2. The genesis is built with migration, + /// 1. and token distribution schedule is given as input + /// 1. if the token distribution schedule contains a timelocked stake + /// -> then just use the input schedule, because it was initialized + /// for migration before this execution (this is the case where we + /// load a [`Builder`] from disk that has already built genesis + /// with the migrated state.); + /// 2. if the token distribution schedule does NOT contain any + /// timelocked stake -> then fetch the cached the genesis stake and + /// merge it to the token distribution schedule; + /// 2. and token distribution schedule is NOT given as input -> then + /// fetch the cached genesis stake and initialize a new token + /// distribution schedule with it. fn resolve_token_distribution_schedule(&mut self) -> TokenDistributionSchedule { - let validator_addresses = self.validators.values().map(|v| v.info.iota_address()); - let token_distribution_schedule = self.token_distribution_schedule.take(); + let is_genesis_with_migration = !self.migration_objects.is_empty(); let stardust_total_supply_nanos = self.migration_sources.len() as u64 * STARDUST_TOTAL_SUPPLY_NANOS; - if self.genesis_stake.is_empty() { - token_distribution_schedule.unwrap_or_else(|| { - TokenDistributionSchedule::new_for_validators_with_default_allocation( - validator_addresses, - ) - }) - } else if let Some(schedule) = token_distribution_schedule { - if schedule.contains_timelocked_stake() { - // Genesis stake is already included + + if let Some(schedule) = self.token_distribution_schedule.take() { + if !is_genesis_with_migration || schedule.contains_timelocked_stake() { + // Case 1.1 and 2.1.1 schedule } else { + // Case 2.1.2 self.genesis_stake .extend_token_distribution_schedule_without_migration( schedule, stardust_total_supply_nanos, ) } + } else if !is_genesis_with_migration { + // Case 1.2 + TokenDistributionSchedule::new_for_validators_with_default_allocation( + self.validators.values().map(|v| v.info.iota_address()), + ) } else { + // Case 2.2 self.genesis_stake .to_token_distribution_schedule(stardust_total_supply_nanos) } } fn build_and_cache_unsigned_genesis(&mut self) { - // Verify that all input data is valid + // Verify that all input data is valid. + // Check that if extra objects are present then it is allowed by the paramenters + // to add extra objects and it also validates the validator info self.validate_inputs().unwrap(); + // If migration sources are present, then load them into memory. + // Otherwise do nothing. self.load_migration_sources() .expect("migration sources should be loaded without errors"); + // If migration objects are present, then create and cache the genesis stake; + // this also prepares the data needed to resolve the token distribution + // schedule. Otherwise do nothing. self.create_and_cache_genesis_stake() .expect("genesis stake should be created without errors"); - // Get the token distribution schedule without migration or merge it with + // Resolve the token distribution schedule based on inputs and a possible // genesis stake let token_distribution_schedule = self.resolve_token_distribution_schedule(); + // Verify that token distribution schedule is valid token_distribution_schedule.validate(); token_distribution_schedule @@ -330,17 +381,17 @@ impl Builder { ) .expect("all validators should have the required stake"); - let objects = self.objects.clone().into_values().collect::>(); - // Finally build the genesis and migration data let (unsigned_genesis, migration_tx_data) = build_unsigned_genesis_data( &self.parameters, &token_distribution_schedule, self.validators.values(), - objects, + self.objects.clone().into_values().collect::>(), &mut self.genesis_stake, &mut self.migration_objects, ); + + // Store built data self.migration_tx_data = (!migration_tx_data.is_empty()).then_some(migration_tx_data); self.built_genesis = Some(unsigned_genesis); self.token_distribution_schedule = Some(token_distribution_schedule); @@ -357,7 +408,7 @@ impl Builder { fn committee(objects: &[Object]) -> Committee { let iota_system_object = - get_iota_system_state(&objects).expect("Iota System State object must always exist"); + get_iota_system_state(&objects).expect("IOTA System State object must always exist"); iota_system_object .get_current_epoch_committee() .committee() @@ -809,6 +860,26 @@ impl Builder { None }; + // Load delegator + let delegator_file = path.join(GENESIS_BUILDER_DELEGATOR_FILE); + let delegator = if delegator_file.exists() { + Some(serde_json::from_slice(&fs::read(delegator_file)?)?) + } else { + None + }; + + // Load delegator map + let delegator_map_file = path.join(GENESIS_BUILDER_DELEGATOR_MAP_FILE); + let delegator_map = if delegator_map_file.exists() { + Some(Delegations::from_csv(fs::File::open(delegator_map_file)?)?) + } else { + None + }; + + let delegation = delegator + .map(GenesisDelegation::OneToAll) + .or(delegator_map.map(GenesisDelegation::ManyToMany)); + let mut builder = Self { parameters, token_distribution_schedule, @@ -820,6 +891,7 @@ impl Builder { genesis_stake: Default::default(), migration_sources, migration_tx_data, + delegation, }; let unsigned_genesis_file = path.join(GENESIS_BUILDER_UNSIGNED_GENESIS_FILE); @@ -907,6 +979,23 @@ impl Builder { .save(file)?; } + if let Some(delegation) = &self.delegation { + match delegation { + GenesisDelegation::OneToAll(delegator) => { + // Write delegator to file + let file = path.join(GENESIS_BUILDER_DELEGATOR_FILE); + let delegator_json = serde_json::to_string(delegator)?; + fs::write(file, delegator_json)?; + } + GenesisDelegation::ManyToMany(delegator_map) => { + // Write delegator map to CSV file + delegator_map.to_csv(fs::File::create( + path.join(GENESIS_BUILDER_DELEGATOR_MAP_FILE), + )?)?; + } + } + } + Ok(()) } } @@ -1009,9 +1098,9 @@ fn build_unsigned_genesis_data<'info>( // migration data. These are either timelocked coins or gas coins. The token // distribution schedule logic assumes that these assets are indeed distributed // to some addresses and this happens above during the creation of the genesis - // objects. Here then we need to burn those assets from the original set of + // objects. Here then we need to destroy those assets from the original set of // migration objects. - let migration_objects = burn_staked_migration_objects( + let migration_objects = destroy_staked_migration_objects( &mut genesis_ctx, migration_objects.take_objects(), &genesis_objects, @@ -1538,9 +1627,9 @@ pub fn generate_genesis_system_object( // Migration objects as input to this function were previously used to create a // genesis stake, that in turn helps to create a token distribution schedule for -// the genesis. In this function the objects needed for the stake are burned +// the genesis. In this function the objects needed for the stake are destroyed // (and, if needed, split) to provide a new set of migration object as output. -fn burn_staked_migration_objects( +fn destroy_staked_migration_objects( genesis_ctx: &mut TxContext, migration_objects: Vec, genesis_objects: &[Object], @@ -1577,15 +1666,15 @@ fn burn_staked_migration_objects( // Extract objects from the store let mut intermediate_store = store.into_inner(); - // Second operation: burn gas and timelocks objects. - // If the genesis stake was created, then burn gas and timelock objects that + // Second operation: destroy gas and timelocks objects. + // If the genesis stake was created, then destroy gas and timelock objects that // were added to the token distribution schedule, because they will be // created on the Move side during genesis. That means we need to prevent // cloning value by evicting these here. - for (id, _, _) in genesis_stake.take_gas_coins_to_burn() { + for (id, _, _) in genesis_stake.take_gas_coins_to_destroy() { intermediate_store.remove(&id); } - for (id, _, _) in genesis_stake.take_timelocks_to_burn() { + for (id, _, _) in genesis_stake.take_timelocks_to_destroy() { intermediate_store.remove(&id); } @@ -1681,7 +1770,7 @@ impl From for SnapshotSource { } } -/// The URLs to download Iota object snapshot. +/// The URLs to download IOTA object snapshot. #[derive(Debug, Clone, Deserialize, Serialize)] pub enum SnapshotUrl { Iota, @@ -1713,7 +1802,7 @@ impl FromStr for SnapshotUrl { } impl SnapshotUrl { - /// Returns the Iota object snapshot download URL. + /// Returns the IOTA object snapshot download URL. pub fn to_url(&self) -> Url { match self { Self::Iota => Url::parse(IOTA_OBJECT_SNAPSHOT_URL).expect("should be valid URL"), diff --git a/crates/iota-genesis-builder/src/main.rs b/crates/iota-genesis-builder/src/main.rs index e0adc0db630..d654d0f6c39 100644 --- a/crates/iota-genesis-builder/src/main.rs +++ b/crates/iota-genesis-builder/src/main.rs @@ -4,50 +4,49 @@ //! Creating a stardust objects snapshot out of a Hornet snapshot. //! TIP that defines the Hornet snapshot file format: //! https://github.com/iotaledger/tips/blob/main/tips/TIP-0035/tip-0035.md -use std::{collections::BTreeMap, fs::File, io::BufWriter}; +use std::{fs::File, io::BufWriter}; -use anyhow::{Result, anyhow}; +use anyhow::Result; use clap::{Parser, Subcommand}; use iota_genesis_builder::{ OBJECT_SNAPSHOT_FILE_PATH, stardust::{ migration::{Migration, MigrationTargetNetwork}, parse::HornetSnapshotParser, - types::{address_swap_map::AddressSwapMap, output_header::OutputHeader}, + process_outputs::scale_amount_for_iota, + types::{address_swap_map::AddressSwapMap, address_swap_split_map::AddressSwapSplitMap}, }, }; -use iota_sdk::types::block::{ - address::Address, - output::{ - AliasOutputBuilder, BasicOutputBuilder, FoundryOutputBuilder, NftOutputBuilder, Output, - unlock_condition::{AddressUnlockCondition, StorageDepositReturnUnlockCondition}, - }, -}; -use iota_types::{stardust::coin_type::CoinType, timelock::timelock::is_vested_reward}; +use iota_types::stardust::coin_type::CoinType; use tracing::Level; use tracing_subscriber::FmtSubscriber; #[derive(Parser, Debug)] -#[clap(about = "Tool for migrating Iota Hornet full-snapshot file")] +#[command(about = "Tool for migrating IOTA Hornet full-snapshot file")] struct Cli { - #[clap(subcommand)] + #[command(subcommand)] snapshot: Snapshot, - #[clap(long, help = "Disable global snapshot verification")] + #[arg(long, help = "Disable global snapshot verification")] disable_global_snapshot_verification: bool, } #[derive(Subcommand, Debug)] enum Snapshot { - #[clap(about = "Migrate an Iota Hornet full-snapshot file")] + #[command(about = "Migrate an IOTA Hornet full-snapshot file")] Iota { - #[clap(long, help = "Path to the Iota Hornet full-snapshot file")] + #[arg(long, help = "Path to the IOTA Hornet full-snapshot file")] snapshot_path: String, - #[clap( + #[arg( long, - help = "Path to the address swap map file. This must be a CSV file with two columns, where an entry contains in the first column an IotaAddress present in the Hornet full-snapshot and in the second column an IotaAddress that will be used for the swap." + help = "Path to the address swap map file. This must be a CSV file with two columns, where an entry contains in the first column an IotaAddress present in the Hornet full-snapshot and in the second column an (ed25519 hex) IOTA Address that will be used for the swap." )] address_swap_map_path: Option, - #[clap(long, value_parser = clap::value_parser!(MigrationTargetNetwork), help = "Target network for migration")] + #[arg( + long, + help = "Path to the address swap split map file. This must be a CSV file with four columns, where an entry contains in the first column a (bech32) Address present in the Hornet full-snapshot, in the second column an (ed25519 hex) IOTA Address that will be used for the swap, in the third column a target amount of iota tokens to be split from the origin address to the destination address and in the fourth column the amount of timelocked iota tokens used for the same scope." + )] + address_swap_split_map_path: Option, + #[arg(long, value_parser = clap::value_parser!(MigrationTargetNetwork), help = "Target network for migration")] target_network: MigrationTargetNetwork, }, } @@ -55,21 +54,29 @@ enum Snapshot { fn main() -> Result<()> { // Initialize tracing let subscriber = FmtSubscriber::builder() - .with_max_level(Level::INFO) + .with_max_level(Level::DEBUG) .finish(); tracing::subscriber::set_global_default(subscriber).expect("setting default subscriber failed"); // Parse the CLI arguments let cli = Cli::parse(); - let (snapshot_path, address_swap_map_path, target_network, coin_type) = match cli.snapshot { + let ( + snapshot_path, + address_swap_map_path, + target_network, + address_swap_split_map_path, + coin_type, + ) = match cli.snapshot { Snapshot::Iota { snapshot_path, address_swap_map_path, + address_swap_split_map_path, target_network, } => ( snapshot_path, address_swap_map_path, target_network, + address_swap_split_map_path, CoinType::Iota, ), }; @@ -89,6 +96,13 @@ fn main() -> Result<()> { } else { AddressSwapMap::default() }; + + let address_swap_split_map = + if let Some(address_swap_split_map_path) = address_swap_split_map_path { + AddressSwapSplitMap::from_csv(&address_swap_split_map_path)? + } else { + AddressSwapSplitMap::default() + }; // Prepare the migration using the parser output stream let migration = Migration::new( snapshot_parser.target_milestone_timestamp(), @@ -104,182 +118,14 @@ fn main() -> Result<()> { match coin_type { CoinType::Iota => { - struct MergingIterator { - unlocked_address_balances: BTreeMap, - snapshot_timestamp_s: u32, - outputs: I, - } - - impl MergingIterator { - fn new(snapshot_timestamp_s: u32, outputs: I) -> Self { - Self { - unlocked_address_balances: Default::default(), - snapshot_timestamp_s, - outputs, - } - } - } - - impl>> Iterator for MergingIterator { - type Item = I::Item; - - fn next(&mut self) -> Option { - // First process all the outputs, building the unlocked_address_balances map as - // we go. - for res in self.outputs.by_ref() { - if let Ok((header, output)) = res { - fn mergeable_address( - header: &OutputHeader, - output: &Output, - snapshot_timestamp_s: u32, - ) -> Option
{ - // ignore all non-basic outputs and non vesting outputs - if !output.is_basic() - || !is_vested_reward(header.output_id(), output.as_basic()) - { - return None; - } - - if let Some(unlock_conditions) = output.unlock_conditions() { - // check if vesting unlock period is already done - if unlock_conditions.is_time_locked(snapshot_timestamp_s) { - return None; - } - unlock_conditions.address().map(|uc| *uc.address()) - } else { - None - } - } - - if let Some(address) = - mergeable_address(&header, &output, self.snapshot_timestamp_s) - { - // collect the unlocked vesting balances - self.unlocked_address_balances - .entry(address) - .and_modify(|x| x.balance += output.amount()) - .or_insert(OutputHeaderWithBalance { - output_header: header, - balance: output.amount(), - }); - continue; - } else { - return Some(Ok((header, output))); - } - } else { - return Some(res); - } - } - - // Now that we are out - self.unlocked_address_balances.pop_first().map( - |(address, output_header_with_balance)| { - // create a new basic output which holds the aggregated balance from - // unlocked vesting outputs for this address - let basic = BasicOutputBuilder::new_with_amount( - output_header_with_balance.balance, - ) - .add_unlock_condition(AddressUnlockCondition::new(address)) - .finish() - .expect("should be able to create a basic output"); - - Ok((output_header_with_balance.output_header, basic.into())) - }, - ) - } - } - - let merged_outputs = MergingIterator::new( + migration.run_for_iota( snapshot_parser.target_milestone_timestamp(), + address_swap_split_map, snapshot_parser.outputs(), - ) - .map(|res| { - let (header, mut output) = res?; - scale_output_amount_for_iota(&mut output)?; - - Ok::<_, anyhow::Error>((header, output)) - }); - itertools::process_results(merged_outputs, |outputs| { - migration.run(outputs, object_snapshot_writer) - })??; + object_snapshot_writer, + )?; } } Ok(()) } - -struct OutputHeaderWithBalance { - output_header: OutputHeader, - balance: u64, -} - -fn scale_output_amount_for_iota(output: &mut Output) -> Result<()> { - *output = match output { - Output::Basic(ref basic_output) => { - // Update amount - let mut builder = BasicOutputBuilder::from(basic_output) - .with_amount(scale_amount_for_iota(basic_output.amount())?); - - // Update amount in potential storage deposit return unlock condition - if let Some(sdr_uc) = basic_output - .unlock_conditions() - .get(StorageDepositReturnUnlockCondition::KIND) - { - let sdr_uc = sdr_uc.as_storage_deposit_return(); - builder = builder.replace_unlock_condition( - StorageDepositReturnUnlockCondition::new( - sdr_uc.return_address(), - scale_amount_for_iota(sdr_uc.amount())?, - u64::MAX, - ) - .unwrap(), - ); - }; - - Output::from(builder.finish()?) - } - Output::Alias(ref alias_output) => Output::from( - AliasOutputBuilder::from(alias_output) - .with_amount(scale_amount_for_iota(alias_output.amount())?) - .finish()?, - ), - Output::Foundry(ref foundry_output) => Output::from( - FoundryOutputBuilder::from(foundry_output) - .with_amount(scale_amount_for_iota(foundry_output.amount())?) - .finish()?, - ), - Output::Nft(ref nft_output) => { - // Update amount - let mut builder = NftOutputBuilder::from(nft_output) - .with_amount(scale_amount_for_iota(nft_output.amount())?); - - // Update amount in potential storage deposit return unlock condition - if let Some(sdr_uc) = nft_output - .unlock_conditions() - .get(StorageDepositReturnUnlockCondition::KIND) - { - let sdr_uc = sdr_uc.as_storage_deposit_return(); - builder = builder.replace_unlock_condition( - StorageDepositReturnUnlockCondition::new( - sdr_uc.return_address(), - scale_amount_for_iota(sdr_uc.amount())?, - u64::MAX, - ) - .unwrap(), - ); - }; - - Output::from(builder.finish()?) - } - Output::Treasury(_) => return Ok(()), - }; - Ok(()) -} - -fn scale_amount_for_iota(amount: u64) -> Result { - const IOTA_MULTIPLIER: u64 = 1000; - - amount - .checked_mul(IOTA_MULTIPLIER) - .ok_or_else(|| anyhow!("overflow multiplying amount {amount} by {IOTA_MULTIPLIER}")) -} diff --git a/crates/iota-genesis-builder/src/stake.rs b/crates/iota-genesis-builder/src/stake.rs index bdbceac5ae7..ef856fd7bd8 100644 --- a/crates/iota-genesis-builder/src/stake.rs +++ b/crates/iota-genesis-builder/src/stake.rs @@ -3,7 +3,8 @@ //! Logic and types to account for stake delegation during genesis. use iota_config::genesis::{ - TokenAllocation, TokenDistributionSchedule, TokenDistributionScheduleBuilder, + Delegations, TokenAllocation, TokenDistributionSchedule, TokenDistributionScheduleBuilder, + ValidatorAllocation, }; use iota_types::{ base_types::{IotaAddress, ObjectRef}, @@ -11,32 +12,29 @@ use iota_types::{ stardust::coin_kind::get_gas_balance_maybe, }; -use crate::{ - stardust::migration::{ExpirationTimestamp, MigrationObjects}, - validator_info::GenesisValidatorInfo, -}; +use crate::stardust::migration::{ExpirationTimestamp, MigrationObjects}; #[derive(Default, Debug, Clone)] pub struct GenesisStake { token_allocation: Vec, - gas_coins_to_burn: Vec, - timelocks_to_burn: Vec, + gas_coins_to_destroy: Vec, + timelocks_to_destroy: Vec, timelocks_to_split: Vec<(ObjectRef, u64, IotaAddress)>, } impl GenesisStake { - /// Take the inner gas-coin objects that must be burned. + /// Take the inner gas-coin objects that must be destroyed. /// /// This follows the semantics of [`std::mem::take`]. - pub fn take_gas_coins_to_burn(&mut self) -> Vec { - std::mem::take(&mut self.gas_coins_to_burn) + pub fn take_gas_coins_to_destroy(&mut self) -> Vec { + std::mem::take(&mut self.gas_coins_to_destroy) } - /// Take the inner timelock objects that must be burned. + /// Take the inner timelock objects that must be destroyed. /// /// This follows the semantics of [`std::mem::take`]. - pub fn take_timelocks_to_burn(&mut self) -> Vec { - std::mem::take(&mut self.timelocks_to_burn) + pub fn take_timelocks_to_destroy(&mut self) -> Vec { + std::mem::take(&mut self.timelocks_to_destroy) } /// Take the inner timelock objects that must be split. @@ -48,8 +46,9 @@ impl GenesisStake { pub fn is_empty(&self) -> bool { self.token_allocation.is_empty() - && self.gas_coins_to_burn.is_empty() - && self.timelocks_to_burn.is_empty() + && self.gas_coins_to_destroy.is_empty() + && self.timelocks_to_destroy.is_empty() + && self.timelocks_to_split.is_empty() } /// Calculate the total amount of token allocations. @@ -82,7 +81,7 @@ impl GenesisStake { /// inner token allocations. /// /// The resulting schedule is guaranteed to contain allocations - /// that sum up the initial total supply of Iota in nanos. + /// that sum up the initial total supply of IOTA in nanos. /// /// ## Errors /// @@ -105,168 +104,332 @@ impl GenesisStake { fn calculate_pre_minted_supply(&self, total_supply_nanos: u64) -> u64 { total_supply_nanos - self.sum_token_allocation() } + + /// Creates a `GenesisStake` using a `Delegations` containing the necessary + /// allocations for validators by some delegators. + /// + /// This function invokes `delegate_genesis_stake` for each delegator found + /// in `Delegations`. + pub fn new_with_delegations( + delegations: Delegations, + migration_objects: &MigrationObjects, + ) -> anyhow::Result { + let mut stake = GenesisStake::default(); + + for (delegator, validators_allocations) in delegations.allocations { + // Fetch all timelock and gas objects owned by the delegator + let timelocks_pool = + migration_objects.get_sorted_timelocks_and_expiration_by_owner(delegator); + let gas_coins_pool = migration_objects.get_gas_coins_by_owner(delegator); + if timelocks_pool.is_none() && gas_coins_pool.is_none() { + anyhow::bail!("no timelocks or gas-coin objects found for delegator {delegator:?}"); + } + stake.delegate_genesis_stake( + &validators_allocations, + delegator, + &mut timelocks_pool.unwrap_or_default().into_iter(), + &mut gas_coins_pool + .unwrap_or_default() + .into_iter() + .map(|object| (object, 0)), + )?; + } + + Ok(stake) + } + + fn create_token_allocation( + &mut self, + recipient_address: IotaAddress, + amount_nanos: u64, + staked_with_validator: Option, + staked_with_timelock_expiration: Option, + ) { + self.token_allocation.push(TokenAllocation { + recipient_address, + amount_nanos, + staked_with_validator, + staked_with_timelock_expiration, + }); + } + + /// Create the necessary allocations for `validators_allocations` using the + /// assets of the `delegator`. + /// + /// This function iterates in turn over [`TimeLock`] and + /// [`GasCoin`][iota_types::gas_coin::GasCoin] objects created + /// during stardust migration that are owned by the `delegator`. + pub fn delegate_genesis_stake<'obj>( + &mut self, + validators_allocations: &[ValidatorAllocation], + delegator: IotaAddress, + timelocks_pool: &mut impl Iterator, + gas_coins_pool: &mut impl Iterator, + ) -> anyhow::Result<()> { + // Temp stores for holding the surplus + let mut timelock_surplus = SurplusCoin::default(); + let mut gas_surplus = SurplusCoin::default(); + + // Then, try to create new token allocations for each validator using the + // objects fetched above + for validator_allocation in validators_allocations { + // The validator address + let validator = validator_allocation.validator; + // The target amount of nanos to be staked, either with timelock or gas objects + let mut target_stake_nanos = validator_allocation.amount_nanos_to_stake; + // The gas to pay to the validator + let gas_to_pay_nanos = validator_allocation.amount_nanos_to_pay_gas; + + // Start filling allocations with timelocks + + // Pick fresh timelock objects (if present) and possibly reuse the surplus + // coming from the previous iteration. + // The method `pick_objects_for_allocation` firstly checks if the + // `timelock_surplus` can be used to reach or reduce the `target_stake_nanos`. + // Then it iterates over the `timelocks_pool`. For each timelock object, its + // balance is used to reduce the `target_stake_nanos` while its the object + // reference is placed into a vector `to_destroy`. At the end, the + // `pick_objects_for_allocation` method returns an `AllocationObjects` including + // the list of objects to destroy, the list `staked_with_timelock` containing + // the information for creating token allocations with timestamps + // and a CoinSurplus (even empty). + let mut timelock_allocation_objects = pick_objects_for_allocation( + timelocks_pool, + target_stake_nanos, + &mut timelock_surplus, + ); + if !timelock_allocation_objects.staked_with_timelock.is_empty() { + // Inside this block some timelock objects were picked from the pool; so we can + // save all the references to timelocks to destroy, if there are any + self.timelocks_to_destroy + .append(&mut timelock_allocation_objects.to_destroy); + // Finally we create some token allocations based on timelock_allocation_objects + timelock_allocation_objects + .staked_with_timelock + .iter() + .for_each(|&(timelocked_amount, expiration_timestamp)| { + // For timelocks we create a `TokenAllocation` object with + // `staked_with_timelock` filled with entries + self.create_token_allocation( + delegator, + timelocked_amount, + Some(validator), + Some(expiration_timestamp), + ); + }); + } + // The remainder of the target stake after timelock objects were used. + target_stake_nanos -= timelock_allocation_objects.amount_nanos; + + // After allocating timelocked stakes, then + // 1. allocate gas coin stakes (if timelocked funds were not enough) + // 2. and/or allocate gas coin payments (if indicated in the validator + // allocation). + + // The target amount of gas coin nanos to be allocated, either with staking or + // to pay + let target_gas_nanos = target_stake_nanos + gas_to_pay_nanos; + // Pick fresh gas coin objects (if present) and possibly reuse the surplus + // coming from the previous iteration. The logic is the same as above with + // timelocks. + let mut gas_coin_objects = + pick_objects_for_allocation(gas_coins_pool, target_gas_nanos, &mut gas_surplus); + if gas_coin_objects.amount_nanos >= target_gas_nanos { + // Inside this block some gas coin objects were picked from the pool; so we can + // save all the references to gas coins to destroy + self.gas_coins_to_destroy + .append(&mut gas_coin_objects.to_destroy); + // Then + // Case 1. allocate gas stakes + if target_stake_nanos > 0 { + // For staking gas coins we create a `TokenAllocation` object with + // an empty `staked_with_timelock` + self.create_token_allocation( + delegator, + target_stake_nanos, + Some(validator), + None, + ); + } + // Case 2. allocate gas payments + if gas_to_pay_nanos > 0 { + // For gas coins payments we create a `TokenAllocation` object with + // `recipient_address` being the validator and no stake + self.create_token_allocation(validator, gas_to_pay_nanos, None, None); + } + } else { + // It means the delegator finished all the timelock or gas funds + return Err(anyhow::anyhow!( + "Not enough funds for delegator {:?}", + delegator + )); + } + } + + // If some surplus amount is left, then return it to the delegator + // In the case of a timelock object, it must be split during the `genesis` PTB + // execution + if let (Some(surplus_timelock), surplus_nanos) = timelock_surplus.take() { + self.timelocks_to_split + .push((surplus_timelock, surplus_nanos, delegator)); + } + // In the case of a gas coin, it must be destroyed and the surplus re-allocated + // to the delegator (no split) + if let (Some(surplus_gas_coin), surplus_nanos) = gas_surplus.take() { + self.gas_coins_to_destroy.push(surplus_gas_coin); + self.create_token_allocation(delegator, surplus_nanos, None, None); + } + + Ok(()) + } } /// The objects picked for token allocation during genesis #[derive(Default, Debug, Clone)] -pub struct AllocationObjects { - inner: Vec, +struct AllocationObjects { + /// The list of objects to destroy for the allocations + to_destroy: Vec, /// The total amount of nanos to be allocated from this /// collection of objects. amount_nanos: u64, - /// The surplus amount that is not be allocated from this - /// collection of objects. - surplus_nanos: u64, /// A (possible empty) vector of (amount, timelock_expiration) pairs /// indicating the amount to timelock stake and its expiration staked_with_timelock: Vec<(u64, u64)>, } +/// The surplus object that should be split for this allocation. Only part +/// of its balance will be used for this collection of this +/// `AllocationObjects`, the surplus might be used later. +#[derive(Default, Debug, Clone)] +struct SurplusCoin { + // The reference of the coin to possibly split to get the surplus. + coin_object_ref: Option, + /// The surplus amount for that coin object. + surplus_nanos: u64, + /// Possibly indicate a timelock stake expiration. + timestamp: u64, +} + +impl SurplusCoin { + // Check if the current surplus can be reused. + // The surplus coin_object_ref is returned to be included in a `to_destroy` list + // when surplus_nanos <= target_amount_nanos. Otherwise it means the + // target_amount_nanos is completely reached, so we can still keep + // coin_object_ref as surplus coin and only reduce the surplus_nanos value. + pub fn maybe_reuse_surplus( + &mut self, + target_amount_nanos: u64, + ) -> (Option, Option, u64) { + if self.coin_object_ref.is_some() { + if self.surplus_nanos <= target_amount_nanos { + let surplus = self.surplus_nanos; + self.surplus_nanos = 0; + (self.coin_object_ref.take(), Some(surplus), self.timestamp) + } else { + self.surplus_nanos -= target_amount_nanos; + (None, Some(target_amount_nanos), self.timestamp) + } + } else { + (None, None, 0) + } + } + + // Destroy the `CoinSurplus` and take the fields. + pub fn take(self) -> (Option, u64) { + (self.coin_object_ref, self.surplus_nanos) + } +} + /// Pick gas-coin like objects from a pool to cover -/// the `target_amount`. +/// the `target_amount_nanos`. It might also make use of a previous coin +/// surplus. /// /// This does not split any surplus balance, but delegates /// splitting to the caller. -pub fn pick_objects_for_allocation<'obj>( +fn pick_objects_for_allocation<'obj>( pool: &mut impl Iterator, - target_amount: u64, + target_amount_nanos: u64, + previous_surplus_coin: &mut SurplusCoin, ) -> AllocationObjects { - let mut amount_nanos = 0; - let mut surplus_nanos = 0; + let mut allocation_tot_amount_nanos = 0; + let mut surplus_coin = SurplusCoin::default(); // Will be left empty in the case of gas coins let mut staked_with_timelock = vec![]; + let mut to_destroy = vec![]; - let objects = pool - .by_ref() - .map_while(|(object, timestamp)| { - if amount_nanos < target_amount { - let mut object_balance = get_gas_balance_maybe(object)?.value(); - // Check if remaining is needed to be handled - let remaining_needed = target_amount - amount_nanos; - if object_balance > remaining_needed { - surplus_nanos = object_balance - remaining_needed; - object_balance = remaining_needed; - } - // Finally update amount - amount_nanos += object_balance; - // Store timestamp if it is a Timelock - if timestamp > 0 { - staked_with_timelock.push((object_balance, timestamp)); - } - Some(object.compute_object_reference()) - } else { - None - } - }) - .collect(); - - AllocationObjects { - inner: objects, - amount_nanos, - surplus_nanos, - staked_with_timelock, + if let (surplus_object_option, Some(surplus_nanos), timestamp) = + previous_surplus_coin.maybe_reuse_surplus(target_amount_nanos) + { + // In here it means there are some surplus nanos that can be used. + // `maybe_reuse_surplus` already deducted the `surplus_nanos` from the + // `surplus_object`. So these can be counted in the + // `allocation_tot_amount_nanos`. + allocation_tot_amount_nanos += surplus_nanos; + // If the ´surplus_object´ is a timelock then store also its timestamp. + if timestamp > 0 { + staked_with_timelock.push((surplus_nanos, timestamp)); + } + // If the `surplus_object` is returned by `maybe_reuse_surplus`, then it means + // it used all its `surplus_nanos` and it can be destroyed. + if let Some(surplus_object) = surplus_object_option { + to_destroy.push(surplus_object); + } + // Else, if the `surplus_object` was not completely drained, then we + // don't need to continue. In this case `allocation_tot_amount_nanos == + // target_amount_nanos`. } -} -/// Create the necessary allocations to cover `amount_nanos` for all -/// `validators`. -/// -/// This function iterates in turn over [`TimeLock`] and -/// [`GasCoin`][iota_types::gas_coin::GasCoin] objects created -/// during stardust migration that are owned by the `delegator`. -pub fn delegate_genesis_stake<'info>( - validators: impl Iterator, - delegator: IotaAddress, - migration_objects: &MigrationObjects, - amount_nanos: u64, -) -> anyhow::Result { - let timelocks_pool = migration_objects.get_sorted_timelocks_and_expiration_by_owner(delegator); - let gas_coins_pool = migration_objects.get_gas_coins_by_owner(delegator); - if timelocks_pool.is_none() && gas_coins_pool.is_none() { - anyhow::bail!("no timelocks or gas-coin objects found for delegator {delegator:?}"); + // We need this check to not consume the first element of the pool in the case + // `allocation_tot_amount_nanos == target_amount_nanos`; this case can only + // happen if the `surplus_coin` contained enough balance to cover for + // `target_amount_nanos`. + if allocation_tot_amount_nanos < target_amount_nanos { + to_destroy.append( + &mut pool + .by_ref() + .map_while(|(object, timestamp)| { + if allocation_tot_amount_nanos < target_amount_nanos { + let difference_from_target = + target_amount_nanos - allocation_tot_amount_nanos; + let obj_ref = object.compute_object_reference(); + let object_balance = get_gas_balance_maybe(object)?.value(); + + if object_balance <= difference_from_target { + if timestamp > 0 { + staked_with_timelock.push((object_balance, timestamp)); + } + allocation_tot_amount_nanos += object_balance; + // Place `obj_ref` in `to_destroy` and continue + Some(obj_ref) + } else { + surplus_coin = SurplusCoin { + coin_object_ref: Some(obj_ref), + surplus_nanos: object_balance - difference_from_target, + timestamp, + }; + if timestamp > 0 { + staked_with_timelock.push((difference_from_target, timestamp)); + } + allocation_tot_amount_nanos += difference_from_target; + // Do NOT place `obj_ref` in `to_destroy` because it is reused in the + // CoinSurplus and then break the map_while + None + } + } else { + // Break the map_while + None + } + }) + .collect::>(), + ); } - let mut timelocks_pool = timelocks_pool.unwrap_or_default().into_iter(); - let mut gas_coins_pool = gas_coins_pool - .unwrap_or_default() - .into_iter() - .map(|object| (object, 0)); - let mut genesis_stake = GenesisStake::default(); - - // For each validator we try to fill their allocation up to - // total_amount_to_stake_per_validator - for validator in validators { - let target_stake = amount_nanos; - - // Start filling allocations with timelocks - let mut timelock_objects = pick_objects_for_allocation(&mut timelocks_pool, target_stake); - // TODO: This is not an optimal solution because the last timelock - // might have a surplus amount, which cannot be used without splitting. - if !timelock_objects.inner.is_empty() { - timelock_objects.staked_with_timelock.iter().for_each( - |&(timelocked_amount, expiration_timestamp)| { - // For timelocks we create a `TokenAllocation` object with - // `staked_with_timelock` filled with entries - genesis_stake.token_allocation.push(TokenAllocation { - recipient_address: delegator, - amount_nanos: timelocked_amount, - staked_with_validator: Some(validator.info.iota_address()), - staked_with_timelock_expiration: Some(expiration_timestamp), - }); - }, - ); - // Get the reference to the timelock to split needed to get exactly - // `amount_nanos` - let timelock_to_split = *timelock_objects - .inner - .last() - .expect("there should be at least two objects"); - // Save all the references to timelocks to burn - genesis_stake - .timelocks_to_burn - .append(&mut timelock_objects.inner); - // Save the reference for the token to split (and then burn) - genesis_stake.timelocks_to_split.push(( - timelock_to_split, - timelock_objects.surplus_nanos, - delegator, - )) - } - // Then cover any remaining target stake with gas coins - let remainder_target_stake = target_stake - timelock_objects.amount_nanos; - let mut gas_coin_objects = - pick_objects_for_allocation(&mut gas_coins_pool, remainder_target_stake); - genesis_stake - .gas_coins_to_burn - .append(&mut gas_coin_objects.inner); - // TODO: also here, this is not an optimal solution because the last gas object - // might have a surplus amount, which cannot be used without splitting. - if gas_coin_objects.amount_nanos < remainder_target_stake { - return Err(anyhow::anyhow!( - "Not enough funds for delegator {:?}", - delegator - )); - } else if gas_coin_objects.amount_nanos > 0 { - // For gas coins we create a `TokenAllocation` object with - // an empty`staked_with_timelock` - genesis_stake.token_allocation.push(TokenAllocation { - recipient_address: delegator, - amount_nanos: gas_coin_objects.amount_nanos, - staked_with_validator: Some(validator.info.iota_address()), - staked_with_timelock_expiration: None, - }); - if gas_coin_objects.surplus_nanos > 0 { - // This essentially schedules returning any surplus amount - // from the last coin in `gas_coin_objects` to the delegator - // as a new coin, so that the split is not needed - genesis_stake.token_allocation.push(TokenAllocation { - recipient_address: delegator, - amount_nanos: gas_coin_objects.surplus_nanos, - staked_with_validator: None, - staked_with_timelock_expiration: None, - }); - } - } + // Update the surplus coin passed from the caller + *previous_surplus_coin = surplus_coin; + + AllocationObjects { + to_destroy, + amount_nanos: allocation_tot_amount_nanos, + staked_with_timelock, } - Ok(genesis_stake) } diff --git a/crates/iota-genesis-builder/src/stardust/migration/migration.rs b/crates/iota-genesis-builder/src/stardust/migration/migration.rs index 96ef8c7fd2a..d80096d99d5 100644 --- a/crates/iota-genesis-builder/src/stardust/migration/migration.rs +++ b/crates/iota-genesis-builder/src/stardust/migration/migration.rs @@ -32,7 +32,11 @@ use crate::stardust::{ verification::{created_objects::CreatedObjects, verify_outputs}, }, native_token::package_data::NativeTokenPackageData, - types::{address_swap_map::AddressSwapMap, output_header::OutputHeader}, + process_outputs::process_outputs_for_iota, + types::{ + address_swap_map::AddressSwapMap, address_swap_split_map::AddressSwapSplitMap, + output_header::OutputHeader, + }, }; /// We fix the protocol version used in the migration. @@ -163,6 +167,21 @@ impl Migration { Ok(()) } + /// Run all stages of the migration coming from a Hornet snapshot with IOTA + /// coin type. + pub fn run_for_iota<'a>( + self, + target_milestone_timestamp: u32, + swap_split_map: AddressSwapSplitMap, + outputs: impl Iterator> + 'a, + writer: impl Write, + ) -> Result<()> { + itertools::process_results( + process_outputs_for_iota(target_milestone_timestamp, swap_split_map, outputs), + |outputs| self.run(outputs, writer), + )? + } + /// The migration objects. /// /// The system packages and underlying `init` objects diff --git a/crates/iota-genesis-builder/src/stardust/migration/tests/basic.rs b/crates/iota-genesis-builder/src/stardust/migration/tests/basic.rs index b503be9862c..7dcddce5933 100644 --- a/crates/iota-genesis-builder/src/stardust/migration/tests/basic.rs +++ b/crates/iota-genesis-builder/src/stardust/migration/tests/basic.rs @@ -76,7 +76,7 @@ fn basic_simple_coin_id() { fn basic_simple_coin_id_with_expired_timelock() { for header in [ random_output_header(), - OutputHeader::new_testing( + OutputHeader::new( // A potential vesting reward output transaction ID. *TransactionId::from_str( "0xb191c4bc825ac6983789e50545d5ef07a1d293a98ad974fc9498cb1812345678", diff --git a/crates/iota-genesis-builder/src/stardust/migration/tests/mod.rs b/crates/iota-genesis-builder/src/stardust/migration/tests/mod.rs index f694f1b8758..97615e4f0f1 100644 --- a/crates/iota-genesis-builder/src/stardust/migration/tests/mod.rs +++ b/crates/iota-genesis-builder/src/stardust/migration/tests/mod.rs @@ -52,7 +52,7 @@ mod foundry; mod nft; fn random_output_header() -> OutputHeader { - OutputHeader::new_testing( + OutputHeader::new( random(), random_output_index(), random(), diff --git a/crates/iota-genesis-builder/src/stardust/mod.rs b/crates/iota-genesis-builder/src/stardust/mod.rs index 56a24edab30..b4e832eea96 100644 --- a/crates/iota-genesis-builder/src/stardust/mod.rs +++ b/crates/iota-genesis-builder/src/stardust/mod.rs @@ -8,6 +8,7 @@ pub mod migration; pub mod native_token; pub mod parse; +pub mod process_outputs; #[cfg(feature = "test-outputs")] pub mod test_outputs; pub mod types; diff --git a/crates/iota-genesis-builder/src/stardust/native_token/package_builder.rs b/crates/iota-genesis-builder/src/stardust/native_token/package_builder.rs index b6ad5d813e7..38765a8a672 100644 --- a/crates/iota-genesis-builder/src/stardust/native_token/package_builder.rs +++ b/crates/iota-genesis-builder/src/stardust/native_token/package_builder.rs @@ -147,7 +147,7 @@ fn format_string_as_move_vector(string: &str) -> String { for (idx, byte) in string.as_bytes().iter().enumerate() { byte_string.push_str(&format!("{byte:#x}")); - if idx != string.as_bytes().len() - 1 { + if idx != string.len() - 1 { byte_string.push_str(", "); } } diff --git a/crates/iota-genesis-builder/src/stardust/process_outputs.rs b/crates/iota-genesis-builder/src/stardust/process_outputs.rs new file mode 100644 index 00000000000..86460c89161 --- /dev/null +++ b/crates/iota-genesis-builder/src/stardust/process_outputs.rs @@ -0,0 +1,739 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +use std::{ + cmp::Ordering, + collections::{BTreeMap, BTreeSet}, +}; + +use anyhow::{Result, anyhow}; +use fastcrypto::{encoding::Hex, hash::HashFunction}; +use iota_sdk::types::{ + api::plugins::participation::types::PARTICIPATION_TAG, + block::{ + address::{Address, Ed25519Address}, + output::{ + AliasOutputBuilder, BasicOutput, BasicOutputBuilder, FoundryOutputBuilder, + NftOutputBuilder, OUTPUT_INDEX_MAX, Output, OutputId, + feature::SenderFeature, + unlock_condition::{AddressUnlockCondition, StorageDepositReturnUnlockCondition}, + }, + payload::transaction::TransactionId, + }, +}; +use iota_types::{ + base_types::IotaAddress, + crypto::DefaultHash, + timelock::timelock::{VESTED_REWARD_ID_PREFIX, is_vested_reward}, +}; +use tracing::debug; + +use super::types::{ + address_swap_split_map::AddressSwapSplitMap, output_header::OutputHeader, + output_index::OutputIndex, +}; + +/// Processes an iterator of outputs coming from a Hornet snapshot chaining some +/// filters: +/// - the `ScaleIotaAmountIterator` scales balances of IOTA Tokens from micro to +/// nano. +/// - the `UnlockedVestingIterator` takes vesting outputs that can be unlocked +/// and merges them into a unique basic output. +/// - the `ParticipationOutputFilter` removes all features from the basic +/// outputs with a participation tag. +/// - the `SwapSplitIterator` performs the operation of SwapSplit given a map as +/// input, i.e., for certain origin addresses it swaps the addressUC to a +/// destination address and splits some amounts of tokens and/or timelocked +/// tokens (this operation can be done for several destinations). +pub fn process_outputs_for_iota<'a>( + target_milestone_timestamp: u32, + swap_split_map: AddressSwapSplitMap, + outputs: impl Iterator> + 'a, +) -> impl Iterator> + 'a { + // Create the iterator with the filters needed for an IOTA snapshot + outputs + .scale_iota_amount() + .filter_unlocked_vesting_outputs(target_milestone_timestamp) + .filter_participation_outputs() + .perform_swap_split(swap_split_map) + .map(|res| { + let (header, output) = res?; + Ok((header, output)) + }) +} + +/// Take an `amount` and scale it by a multiplier defined for the IOTA token. +pub fn scale_amount_for_iota(amount: u64) -> Result { + const IOTA_MULTIPLIER: u64 = 1000; + + amount + .checked_mul(IOTA_MULTIPLIER) + .ok_or_else(|| anyhow!("overflow multiplying amount {amount} by {IOTA_MULTIPLIER}")) +} + +// Check if the output is basic and has a feature Tag using the Participation +// Tag: https://github.com/iota-community/treasury/blob/main/specifications/hornet-participation-plugin.md +pub fn is_participation_output(output: &Output) -> bool { + if let Some(feat) = output.features() { + if output.is_basic() && !feat.is_empty() { + if let Some(tag) = feat.tag() { + return tag.to_string() == Hex::encode_with_format(PARTICIPATION_TAG); + }; + } + }; + false +} + +/// Iterator that modifies some outputs address unlocked by certain origin +/// addresses found in the `swap_split_map`. For each origin address there can +/// be a set of destination addresses. Each destination address as either a +/// tokens target, a tokens timelocked target or both. So, each output found by +/// this filter with an address unlock condition being the origin address, a +/// SwapSplit operation is performed. This operation consists in splitting the +/// output in different outputs given the targets indicated for the destinations +/// and swapping the address unlock condition to be the destination address one. +/// This operation is performed on all basic outputs and (vesting) timelocked +/// basic outputs until the targets are reached. +struct SwapSplitIterator { + /// Iterator over `(OutputHeader, Output)` pairs. + outputs: I, + /// Map used for the SwapSplit operation. It associates an origin address to + /// a vector of destinations. A destination is a tuple containing a + /// destination address, a tokens target and a timelocked tokens target. + swap_split_map: AddressSwapSplitMap, + /// Basic outputs with timelock unlock conditions. These are candidate + /// outputs that are kept in ascending order of timestamp and, when the + /// iteration over all outputs has finished, some of them will be popped + /// to be picked for the SwapSplit operation. + timelock_candidates: BTreeSet, + /// Basic outputs that have been split during the processing. These can be + /// either basic outputs or (vesting) timelocked basic outputs that will + /// be added as new in the ledger, before the migration. + split_basic_outputs: Vec<(OutputHeader, Output)>, + num_swapped_basic: u64, + num_swapped_timelocks: u64, + num_splits: u64, +} + +impl SwapSplitIterator { + fn new(outputs: I, swap_split_map: AddressSwapSplitMap) -> Self { + Self { + outputs, + swap_split_map, + timelock_candidates: Default::default(), + split_basic_outputs: Default::default(), + num_swapped_basic: 0, + num_swapped_timelocks: 0, + num_splits: 0, + } + } + + /// Pop an output from `split_basic_outputs`. Since this contains newly + /// created outputs, there is the need to create a new OutputHeader that + /// is not in conflict with any other one in the ledger. Use some data + /// coming from the original output header plus some unique information + /// about the new output. + fn get_split_output(&mut self) -> Option<(OutputHeader, Output)> { + let (original_header, output) = self.split_basic_outputs.pop()?; + self.num_splits += 1; + let pos = self.split_basic_outputs.len(); + + let (transaction_id, output_index) = if original_header + .output_id() + .to_string() + .starts_with(VESTED_REWARD_ID_PREFIX) + { + // If the original basic output is a vesting output, generate the new OutputId + // as: original-transaction-id|index + // where index is a unique input + // index being a number in the range 1 to OUTPUT_INDEX_MAX is safe because + // vesting output indexes are always 0 + // https://github.com/iotaledger/snapshot-tool-new-supply + if original_header.output_id().index() != 0 { + debug!( + "Found a vesting output with output index different than 0: {}", + original_header.output_id() + ); + } + let index = 1 + (pos as u16 % (OUTPUT_INDEX_MAX - 1)); + ( + *original_header.output_id().transaction_id(), + OutputIndex::new(index).unwrap(), + ) + } else { + // Otherwise, generate the new OutputId as: + // DefaultHash("iota-genesis-outputs"|original-output-id|pos)|index + // where pos is a unique input + let index = pos as u16 % OUTPUT_INDEX_MAX; + let mut hasher = DefaultHash::default(); + hasher.update(b"iota-genesis-outputs"); + hasher.update(original_header.output_id().hash()); + hasher.update(pos.to_le_bytes()); + let hash = hasher.finalize(); + ( + TransactionId::new(hash.into()), + OutputIndex::new(index).unwrap(), + ) + }; + + Some(( + OutputHeader::new( + *transaction_id, + output_index, + *original_header.block_id(), + *original_header.ms_index(), + original_header.ms_timestamp(), + ), + output, + )) + } +} + +impl Iterator for SwapSplitIterator +where + I: Iterator>, +{ + type Item = I::Item; + + /// Get the next from the chained self.outputs iterator and apply the + /// SwapSplit filter if that's the case. + fn next(&mut self) -> Option { + for mut output in self.outputs.by_ref() { + if let Ok((header, inner)) = &mut output { + if let Output::Basic(ref basic_output) = inner { + let uc = basic_output.unlock_conditions(); + // Only for outputs with timelock and/or address unlock conditions the SwapSplit + // operation can be performed + if uc.storage_deposit_return().is_none() && uc.expiration().is_none() { + // Now check if the addressUC's address is to swap + if let Some(destinations) = self + .swap_split_map + .get_destination_maybe_mut(uc.address().unwrap().address()) + { + if uc.timelock().is_some() { + // If the output has a timelock UC (and it is a vested reward) and + // at least one destination requires some timelocked tokens, then + // store it as a candidate and continue with the iterator + if is_vested_reward(header.output_id(), basic_output) + && destinations.contains_tokens_timelocked_target() + { + // Here we store all the timelocked basic outputs we find, + // because we need all the ones owned by the origin address + // sorted by the unlocking timestamp; outside this loop, + // i.e., once all have been collected, we'll start the + // SwapSplit operation in order, starting from the one that + // unlocks later in time. + self.timelock_candidates.insert(TimelockOrderedOutput { + header: header.clone(), + output: inner.clone(), + }); + continue; + } + } else { + // If it is just a basic output, try to perform the SwapSplit + // operation for several destinations once all tokens targets are + // meet. + let (original_output_opt, split_outputs) = swap_split_operation( + destinations.iter_by_tokens_target_mut_filtered(), + basic_output, + ); + // If some SwapSplit were performed, their result are basic inputs + // stored in split_outputs; so, we save them in + // split_basic_outputs to return them later + if !split_outputs.is_empty() { + self.num_swapped_basic += 1; + } + self.split_basic_outputs.extend( + split_outputs + .into_iter() + .map(|output| (header.clone(), output)), + ); + // If there was a remainder, the original output is returned for the + // iterator, possibly with a modified amount; else, continue the + // loop + if let Some(original_output) = original_output_opt { + *inner = original_output; + } else { + continue; + } + }; + } + } + } + } + return Some(output); + } + // Now that we are out of the loop we collect the processed outputs from the + // timelock filter and try to fulfill the target. + // First, resolve timelocks SwapSplit operations, taking those out from + // timelocks; the ordered_timelock_candidates is ordered by timestamp + // and we want to take the latest ones first. + while let Some(TimelockOrderedOutput { header, output }) = + self.timelock_candidates.pop_last() + { + // We know that all of them are timelocked basic outputs + let timelocked_basic_output = output.as_basic(); + let uc = timelocked_basic_output.unlock_conditions(); + // Get destination address and mutable timelocked tokens target + let destinations = self + .swap_split_map + .get_destination_maybe_mut(uc.address().unwrap().address()) + .expect("ordered timelock candidates should be part of the swap map"); + + // Try to perform the SwapSplit operation for several destinations once all + // tokens timelocked targets are met + let (original_output_opt, split_outputs) = swap_split_operation( + destinations.iter_by_tokens_timelocked_target_mut_filtered(), + timelocked_basic_output, + ); + // If some SwapSplit were performed, their result are timelocked basic inputs + // stored in split_outputs; so, we save them in + // split_basic_outputs to return them later + if !split_outputs.is_empty() { + self.num_swapped_timelocks += 1; + } + self.split_basic_outputs.extend( + split_outputs + .into_iter() + .map(|output| (header.clone(), output)), + ); + // If there was a remainder, the original output is returned for the + // iterator, possibly with a modified amount; otherwise, continue the + // loop + if let Some(original_output) = original_output_opt { + return Some(Ok((header, original_output))); + } else { + continue; + } + } + // Second, return all the remaining split outputs generated suring SwapSplit + // operations + Some(Ok(self.get_split_output()?)) + } +} + +impl Drop for SwapSplitIterator { + fn drop(&mut self) { + if let Some((origin, destination, tokens_target, tokens_timelocked_target)) = + self.swap_split_map.validate_successful_swap_split() + { + panic!( + "For at least one address, the SwapSplit operation was not fully performed. Origin: {}, destination: {}, tokens left: {}, timelocked tokens left: {}", + origin, destination, tokens_target, tokens_timelocked_target + ) + } + debug!( + "Number of basic outputs used for a SwapSplit (no timelock): {}", + self.num_swapped_basic + ); + debug!( + "Number of timelocked basic outputs used for a SwapSplit: {}", + self.num_swapped_timelocks + ); + debug!("Number of outputs created with splits: {}", self.num_splits); + } +} + +/// Iterator that modifies the amount of IOTA tokens for any output, scaling the +/// amount from micros to nanos. +struct ScaleIotaAmountIterator { + /// Iterator over `(OutputHeader, Output)` pairs. + outputs: I, + num_scaled_outputs: u64, +} + +impl ScaleIotaAmountIterator { + fn new(outputs: I) -> Self { + Self { + outputs, + num_scaled_outputs: 0, + } + } +} + +impl Iterator for ScaleIotaAmountIterator +where + I: Iterator>, +{ + type Item = I::Item; + + /// Get the next from the chained self.outputs iterator and always apply the + /// scaling (only an Output::Treasury kind is left out) + fn next(&mut self) -> Option { + let mut output = self.outputs.next()?; + if let Ok((_, inner)) = &mut output { + self.num_scaled_outputs += 1; + match inner { + Output::Basic(ref basic_output) => { + // Update amount + let mut builder = BasicOutputBuilder::from(basic_output).with_amount( + scale_amount_for_iota(basic_output.amount()) + .expect("should scale the amount for iota"), + ); + // Update amount in potential storage deposit return unlock condition + if let Some(sdr_uc) = basic_output + .unlock_conditions() + .get(StorageDepositReturnUnlockCondition::KIND) + { + let sdr_uc = sdr_uc.as_storage_deposit_return(); + builder = builder.replace_unlock_condition( + StorageDepositReturnUnlockCondition::new( + sdr_uc.return_address(), + scale_amount_for_iota(sdr_uc.amount()) + .expect("should scale the amount for iota"), + u64::MAX, + ) + .unwrap(), + ); + }; + *inner = builder + .finish() + .expect("failed to create basic output") + .into() + } + Output::Alias(ref alias_output) => { + *inner = AliasOutputBuilder::from(alias_output) + .with_amount( + scale_amount_for_iota(alias_output.amount()) + .expect("should scale the amount for iota"), + ) + .finish() + .expect("should be able to create an alias output") + .into() + } + Output::Foundry(ref foundry_output) => { + *inner = FoundryOutputBuilder::from(foundry_output) + .with_amount( + scale_amount_for_iota(foundry_output.amount()) + .expect("should scale the amount for iota"), + ) + .finish() + .expect("should be able to create a foundry output") + .into() + } + Output::Nft(ref nft_output) => { + // Update amount + let mut builder = NftOutputBuilder::from(nft_output).with_amount( + scale_amount_for_iota(nft_output.amount()) + .expect("should scale the amount for iota"), + ); + // Update amount in potential storage deposit return unlock condition + if let Some(sdr_uc) = nft_output + .unlock_conditions() + .get(StorageDepositReturnUnlockCondition::KIND) + { + let sdr_uc = sdr_uc.as_storage_deposit_return(); + builder = builder.replace_unlock_condition( + StorageDepositReturnUnlockCondition::new( + sdr_uc.return_address(), + scale_amount_for_iota(sdr_uc.amount()) + .expect("should scale the amount for iota"), + u64::MAX, + ) + .unwrap(), + ); + }; + *inner = builder + .finish() + .expect("should be able to create an nft output") + .into(); + } + Output::Treasury(_) => (), + } + } + Some(output) + } +} + +impl Drop for ScaleIotaAmountIterator { + fn drop(&mut self) { + debug!("Number of scaled outputs: {}", self.num_scaled_outputs); + } +} + +struct OutputHeaderWithBalance { + output_header: OutputHeader, + balance: u64, +} + +/// Filtering iterator that looks for vesting outputs that can be unlocked and +/// stores them during the iteration. At the end of the iteration it merges all +/// vesting outputs owned by a single address into a unique basic output. +struct UnlockedVestingIterator { + /// Iterator over `(OutputHeader, Output)` pairs. + outputs: I, + /// Stores aggregated balances for eligible addresses. + unlocked_address_balances: BTreeMap, + /// Timestamp used to evaluate timelock conditions. + snapshot_timestamp_s: u32, + /// Output picked to be merged + vesting_outputs: Vec, + num_vesting_outputs: u64, +} + +impl UnlockedVestingIterator { + fn new(outputs: I, snapshot_timestamp_s: u32) -> Self { + Self { + outputs, + unlocked_address_balances: Default::default(), + snapshot_timestamp_s, + vesting_outputs: Default::default(), + num_vesting_outputs: Default::default(), + } + } +} + +impl Iterator for UnlockedVestingIterator +where + I: Iterator>, +{ + type Item = I::Item; + + /// Get the next from the chained self.outputs iterator and apply the + /// processing only if the output is an unlocked vesting one + fn next(&mut self) -> Option { + for output in self.outputs.by_ref() { + if let Ok((header, inner)) = &output { + if let Some(address) = + get_address_if_vesting_output(header, inner, self.snapshot_timestamp_s) + { + self.vesting_outputs.push(header.output_id()); + self.unlocked_address_balances + .entry(address) + .and_modify(|x| x.balance += inner.amount()) + .or_insert(OutputHeaderWithBalance { + output_header: header.clone(), + balance: inner.amount(), + }); + continue; + } + } + return Some(output); + } + // Now that we are out of the loop we collect the processed outputs from the + // filters + let (address, output_header_with_balance) = self.unlocked_address_balances.pop_first()?; + self.num_vesting_outputs += 1; + // create a new basic output which holds the aggregated balance from + // unlocked vesting outputs for this address + let basic = BasicOutputBuilder::new_with_amount(output_header_with_balance.balance) + .add_unlock_condition(AddressUnlockCondition::new(address)) + .finish() + .expect("failed to create basic output"); + + Some(Ok((output_header_with_balance.output_header, basic.into()))) + } +} + +impl Drop for UnlockedVestingIterator { + fn drop(&mut self) { + debug!( + "Number of vesting outputs before merge: {}", + self.vesting_outputs.len() + ); + debug!( + "Number of vesting outputs after merging: {}", + self.num_vesting_outputs + ); + } +} + +/// Iterator that looks for basic outputs having a tag being the Participation +/// Tag and removes all features from the basic output. +struct ParticipationOutputIterator { + /// Iterator over `(OutputHeader, Output)` pairs. + outputs: I, + participation_outputs: Vec, +} + +impl ParticipationOutputIterator { + fn new(outputs: I) -> Self { + Self { + outputs, + participation_outputs: Default::default(), + } + } +} + +impl Iterator for ParticipationOutputIterator +where + I: Iterator>, +{ + type Item = I::Item; + + /// Get the next from the chained self.outputs iterator and apply the + /// processing only if the output has a participation tag + fn next(&mut self) -> Option { + let mut output = self.outputs.next()?; + if let Ok((header, inner)) = &mut output { + if is_participation_output(inner) { + self.participation_outputs.push(header.output_id()); + let basic_output = inner.as_basic(); + // replace the inner output + *inner = BasicOutputBuilder::from(basic_output) + .with_features( + vec![basic_output.features().get(SenderFeature::KIND).cloned()] + .into_iter() + .flatten(), + ) + .finish() + .expect("failed to create basic output") + .into() + } + } + Some(output) + } +} + +impl Drop for ParticipationOutputIterator { + fn drop(&mut self) { + debug!( + "Number of participation outputs: {}", + self.participation_outputs.len() + ); + debug!("Participation outputs: {:?}", self.participation_outputs); + } +} + +/// Extension trait that provides convenient methods for chaining and filtering +/// iterator operations. +/// +/// The iterators produced by this trait are designed to chain such that, +/// calling `next()` on the last iterator will recursively invoke `next()` on +/// the preceding iterators, maintaining the expected behavior. +trait IteratorExt: Iterator> + Sized { + fn perform_swap_split(self, swap_split_map: AddressSwapSplitMap) -> SwapSplitIterator { + SwapSplitIterator::new(self, swap_split_map) + } + + fn scale_iota_amount(self) -> ScaleIotaAmountIterator { + ScaleIotaAmountIterator::new(self) + } + + fn filter_unlocked_vesting_outputs( + self, + snapshot_timestamp_s: u32, + ) -> UnlockedVestingIterator { + UnlockedVestingIterator::new(self, snapshot_timestamp_s) + } + + fn filter_participation_outputs(self) -> ParticipationOutputIterator { + ParticipationOutputIterator::new(self) + } +} +impl>> IteratorExt for T {} + +/// Skip all outputs that are not basic or not vesting. For vesting (basic) +/// outputs, extract and return the address from their address unlock condition. +fn get_address_if_vesting_output( + header: &OutputHeader, + output: &Output, + snapshot_timestamp_s: u32, +) -> Option
{ + if !output.is_basic() || !is_vested_reward(header.output_id(), output.as_basic()) { + // if the output is not basic and a vested reward then skip + return None; + } + + output.unlock_conditions().and_then(|uc| { + if uc.is_time_locked(snapshot_timestamp_s) { + // if the output would still be time locked at snapshot_timestamp_s then skip + None + } else { + // return the address of a vested output that is or can be unlocked + uc.address().map(|a| *a.address()) + } + }) +} + +/// SwapSplit operation. Take a `basic_output` and split it until all targets +/// found in the `destinations` are meet. In the meantime, swap the address +/// unlock condition origin address with the destination address. Finally, if +/// the original `basic_output` has some remainder amount, then return it +/// (without swapping its address unlock condition). +fn swap_split_operation<'a>( + destinations: impl Iterator, + basic_output: &BasicOutput, +) -> (Option, Vec) { + let mut original_output_opt = None; + let mut split_outputs = vec![]; + let mut original_basic_output_remainder = basic_output.amount(); + + // if the addressUC's address is to swap, then it can have several + // destinations + for (destination, target) in destinations { + // break if the basic output was drained already + if original_basic_output_remainder == 0 { + break; + } + // we need to make sure that we split at most OUTPUT_INDEX_MAX - 1 times + debug_assert!( + split_outputs.len() < OUTPUT_INDEX_MAX as usize, + "Too many swap split operations to perform for a single output" + ); + // if the target for this destination is less than the basic output remainder, + // then use it to split the basic output and swap address; + // otherwise split and swap using the original_basic_output_remainder, and then + // break the loop. + let swap_split_amount = original_basic_output_remainder.min(*target); + split_outputs.push( + BasicOutputBuilder::from(basic_output) + .with_amount(swap_split_amount) + .replace_unlock_condition(AddressUnlockCondition::new(Ed25519Address::new( + destination.to_inner(), + ))) + .finish() + .expect("failed to create basic output during split") + .into(), + ); + *target -= swap_split_amount; + original_basic_output_remainder -= swap_split_amount; + } + + // if the basic output remainder is some, it means that all destinations are + // already covered; so the original basic output can be just kept with (maybe) + // an adjusted amount. + if original_basic_output_remainder > 0 { + original_output_opt = Some( + BasicOutputBuilder::from(basic_output) + .with_amount(original_basic_output_remainder) + .finish() + .expect("failed to create basic output") + .into(), + ); + } + (original_output_opt, split_outputs) +} + +/// Utility struct that defines the ordering between timelocked basic outputs. +/// It is required that the output is a basic outputs with timelock unlock +/// condition. +#[derive(PartialEq, Eq)] +struct TimelockOrderedOutput { + header: OutputHeader, + output: Output, +} + +impl TimelockOrderedOutput { + fn get_timestamp(&self) -> u32 { + self.output + .as_basic() + .unlock_conditions() + .timelock() + .unwrap() + .timestamp() + } +} + +impl Ord for TimelockOrderedOutput { + fn cmp(&self, other: &Self) -> Ordering { + self.get_timestamp() + .cmp(&other.get_timestamp()) + .then_with(|| self.header.output_id().cmp(&other.header.output_id())) + } +} +impl PartialOrd for TimelockOrderedOutput { + fn partial_cmp(&self, other: &Self) -> Option { + Some(self.cmp(other)) + } +} diff --git a/crates/iota-genesis-builder/src/stardust/test_outputs/alias_ownership.rs b/crates/iota-genesis-builder/src/stardust/test_outputs/alias_ownership.rs index 1f1f0620401..89168937e3b 100644 --- a/crates/iota-genesis-builder/src/stardust/test_outputs/alias_ownership.rs +++ b/crates/iota-genesis-builder/src/stardust/test_outputs/alias_ownership.rs @@ -180,7 +180,7 @@ fn random_foundry_output( } fn random_output_header(rng: &mut StdRng) -> OutputHeader { - OutputHeader::new_testing( + OutputHeader::new( rng.gen(), OutputIndex::new(rng.gen_range(OUTPUT_INDEX_RANGE)) .expect("range is guaranteed to be valid"), diff --git a/crates/iota-genesis-builder/src/stardust/test_outputs/delegator_outputs.rs b/crates/iota-genesis-builder/src/stardust/test_outputs/delegator_outputs.rs index ac9714fd80a..1084a4d33c3 100644 --- a/crates/iota-genesis-builder/src/stardust/test_outputs/delegator_outputs.rs +++ b/crates/iota-genesis-builder/src/stardust/test_outputs/delegator_outputs.rs @@ -30,7 +30,7 @@ pub(crate) fn new_simple_basic_output( address: Ed25519Address, rng: &mut StdRng, ) -> anyhow::Result<(OutputHeader, Output)> { - let output_header = OutputHeader::new_testing( + let output_header = OutputHeader::new( rng.gen::<[u8; 32]>(), random_output_index_with_rng(rng), [0; 32], @@ -59,7 +59,7 @@ pub(crate) fn new_vested_output( .copy_from_slice(&prefix_hex::decode::<[u8; 28]>(VESTED_REWARD_ID_PREFIX)?); transaction_id[28..32].copy_from_slice(&vested_index.to_le_bytes()); - let output_header = OutputHeader::new_testing( + let output_header = OutputHeader::new( transaction_id, random_output_index_with_rng(rng), [0; 32], diff --git a/crates/iota-genesis-builder/src/stardust/test_outputs/stardust_mix.rs b/crates/iota-genesis-builder/src/stardust/test_outputs/stardust_mix.rs index 1b55cfd2c57..63fb2e347e7 100644 --- a/crates/iota-genesis-builder/src/stardust/test_outputs/stardust_mix.rs +++ b/crates/iota-genesis-builder/src/stardust/test_outputs/stardust_mix.rs @@ -329,7 +329,7 @@ fn random_alias_foundry_native_token( fn finish_with_header(builder: impl Into, rng: &mut StdRng) -> (OutputHeader, Output) { ( - OutputHeader::new_testing( + OutputHeader::new( rng.gen::<[u8; 32]>(), random_output_index_with_rng(rng), [0; 32], diff --git a/crates/iota-genesis-builder/src/stardust/test_outputs/vesting_schedule_portfolio_mix.rs b/crates/iota-genesis-builder/src/stardust/test_outputs/vesting_schedule_portfolio_mix.rs index 3766af98425..05779e1ae93 100644 --- a/crates/iota-genesis-builder/src/stardust/test_outputs/vesting_schedule_portfolio_mix.rs +++ b/crates/iota-genesis-builder/src/stardust/test_outputs/vesting_schedule_portfolio_mix.rs @@ -73,7 +73,7 @@ pub(crate) async fn outputs( } fn random_output_header(rng: &mut StdRng) -> OutputHeader { - OutputHeader::new_testing( + OutputHeader::new( rng.gen(), OutputIndex::new(rng.gen_range(OUTPUT_INDEX_RANGE)) .expect("range is guaranteed to be valid"), diff --git a/crates/iota-genesis-builder/src/stardust/types/address_swap_split_map.rs b/crates/iota-genesis-builder/src/stardust/types/address_swap_split_map.rs new file mode 100644 index 00000000000..3ba7f80ca5c --- /dev/null +++ b/crates/iota-genesis-builder/src/stardust/types/address_swap_split_map.rs @@ -0,0 +1,171 @@ +// Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +use std::collections::HashMap; + +use iota_sdk::types::block::address::Address; +use iota_types::base_types::IotaAddress; + +type OriginAddress = Address; +type Destination = (IotaAddress, u64, u64); + +#[derive(Clone, Debug, Default)] +pub struct AddressSwapSplitDestinations { + destinations: Vec, +} + +impl AddressSwapSplitDestinations { + /// Iterate over mutable destinations filtered by `tokens_target > 0`. + pub fn iter_by_tokens_target_mut_filtered( + &mut self, + ) -> impl Iterator { + self.destinations + .iter_mut() + .filter_map(|(destination, tokens_target, _)| { + if *tokens_target > 0 { + Some((destination, tokens_target)) + } else { + None + } + }) + } + + /// Iterate over mutable destinations filtered by `tokens_timelocked_target + /// > 0`. + pub fn iter_by_tokens_timelocked_target_mut_filtered( + &mut self, + ) -> impl Iterator { + self.destinations + .iter_mut() + .filter_map(|(destination, _, tokens_timelocked_target)| { + if *tokens_timelocked_target > 0 { + Some((destination, tokens_timelocked_target)) + } else { + None + } + }) + } + + /// Returns true only if the destinations contains at least one + /// tokens_timelocked_target that is greater than 0. + pub fn contains_tokens_timelocked_target(&self) -> bool { + self.destinations + .iter() + .any(|&(_, _, tokens_timelocked_target)| tokens_timelocked_target > 0) + } +} + +impl<'a> IntoIterator for &'a AddressSwapSplitDestinations { + type Item = &'a Destination; + type IntoIter = std::slice::Iter<'a, Destination>; + + fn into_iter(self) -> Self::IntoIter { + self.destinations.iter() + } +} + +#[derive(Clone, Debug, Default)] +pub struct AddressSwapSplitMap { + map: HashMap, +} + +impl AddressSwapSplitMap { + /// If the `address` passed as input is present in the map, then return + /// a mutable reference to the destination, i.e., a tuple containing a + /// destination address, a tokens target and a timelocked tokens target. + pub fn get_destination_maybe_mut( + &mut self, + address: &OriginAddress, + ) -> Option<&mut AddressSwapSplitDestinations> { + self.map.get_mut(address) + } + + /// Check whether the map has all targets set to 0. Return the first + /// occurrence of an entry where one or both the two targets are greater + /// than zero. If none is found, then return None. + pub fn validate_successful_swap_split( + &self, + ) -> Option<(&OriginAddress, &IotaAddress, u64, u64)> { + for (origin, destinations) in self.map.iter() { + for (destination, tokens_target, tokens_timelocked_target) in destinations { + if *tokens_target > 0 || *tokens_timelocked_target > 0 { + return Some(( + origin, + destination, + *tokens_target, + *tokens_timelocked_target, + )); + } + } + } + None + } + + /// Get the map. + pub fn map(&self) -> &HashMap { + &self.map + } + + /// Initializes an [`AddressSwapSplitMap`] by reading address pairs from a + /// CSV file. + /// + /// The function expects the file to contain four columns: the origin + /// address (first column), the destination address (second column), the + /// tokens target (third column) and the timelocked tokens target + /// (fourth column). These are parsed into a [`HashMap`] that maps + /// origin addresses to tuples containing the destination address and + /// the two targets. + /// + /// # Example CSV File + /// ```csv + /// Origin,Destination,Tokens,TokensTimelocked + /// iota1qrukjnd6jhgwc0ls6dgt574sxuulcsmq5lnzhtv4jmlwkydhe2zvy69t7jj,0x1336d143de5eb55bcb069f55da5fc9f0c84e368022fd2bbe0125b1093b446313,107667149000,107667149000 + /// iota1qr4chj9jwhauvegqy40sdhj93mzmvc3mg9cmzlv2y6j8vpyxpvug2y6h5jd,0x83b5ed87bac715ecb09017a72d531ccc3c43bcb58edeb1ce383f1c46cfd79bec,388647312000,0 + /// ``` + /// + /// # Parameters + /// - `file_path`: The relative path to the CSV file containing the address + /// mappings. + /// + /// # Returns + /// - An [`AddressSwapSplitMap`] containing the parsed mappings. + /// + /// # Errors + /// - Returns an error if the file cannot be found, read, or parsed + /// correctly. + /// - Returns an error if the origin, destination addresses, or targets + /// cannot be parsed into. + pub fn from_csv(file_path: &str) -> Result { + let current_dir = std::env::current_dir()?; + let file_path = current_dir.join(file_path); + let mut reader = csv::ReaderBuilder::new().from_path(file_path)?; + let mut address_swap_split_map: AddressSwapSplitMap = Default::default(); + + let headers = reader.headers()?; + anyhow::ensure!( + headers.len() == 4 + && &headers[0] == "Origin" + && &headers[1] == "Destination" + && &headers[2] == "Tokens" + && &headers[3] == "TokensTimelocked", + "Invalid CSV headers" + ); + + for result in reader.records() { + let record = result?; + let origin = OriginAddress::try_from_bech32(&record[0])?; + let destination_address = record[1].parse()?; + let tokens_target = record[2].parse()?; + let tokens_timelocked_target = record[3].parse()?; + + address_swap_split_map + .map + .entry(origin) + .or_default() + .destinations + .push((destination_address, tokens_target, tokens_timelocked_target)); + } + + Ok(address_swap_split_map) + } +} diff --git a/crates/iota-genesis-builder/src/stardust/types/mod.rs b/crates/iota-genesis-builder/src/stardust/types/mod.rs index 49ea9368cf5..89cf2b9264a 100644 --- a/crates/iota-genesis-builder/src/stardust/types/mod.rs +++ b/crates/iota-genesis-builder/src/stardust/types/mod.rs @@ -2,6 +2,7 @@ // SPDX-License-Identifier: Apache-2.0 pub mod address_swap_map; +pub mod address_swap_split_map; pub mod output_header; pub mod output_index; pub mod snapshot; diff --git a/crates/iota-genesis-builder/src/stardust/types/output_header.rs b/crates/iota-genesis-builder/src/stardust/types/output_header.rs index ae6de6eb270..3c0689af997 100644 --- a/crates/iota-genesis-builder/src/stardust/types/output_header.rs +++ b/crates/iota-genesis-builder/src/stardust/types/output_header.rs @@ -13,7 +13,7 @@ use crate::stardust::types::output_index::OutputIndex; /// The header of an [`Output`](iota_sdk::types::block::output::Output) in the /// snapshot -#[derive(Debug, Clone, Packable)] +#[derive(Debug, Clone, Packable, PartialEq, Eq)] pub struct OutputHeader { output_id: OutputId, block_id: BlockId, @@ -56,8 +56,8 @@ impl OutputHeader { self.length } - /// Creates a new OutputHeader for testing. - pub fn new_testing( + /// Creates a new OutputHeader + pub fn new( transaction_id_bytes: [u8; 32], output_index: OutputIndex, block_id_bytes: [u8; 32], diff --git a/crates/iota-graphql-e2e-tests/Cargo.toml b/crates/iota-graphql-e2e-tests/Cargo.toml index de5796645cb..e912872c3ec 100644 --- a/crates/iota-graphql-e2e-tests/Cargo.toml +++ b/crates/iota-graphql-e2e-tests/Cargo.toml @@ -5,7 +5,7 @@ authors = ["IOTA Foundation "] edition = "2021" license = "Apache-2.0" publish = false -description = "End to end tests for Iota GraphQL" +description = "End to end tests for IOTA GraphQL" [lints] workspace = true diff --git a/crates/iota-graphql-e2e-tests/README.md b/crates/iota-graphql-e2e-tests/README.md index a0001d1c739..71d3660615b 100644 --- a/crates/iota-graphql-e2e-tests/README.md +++ b/crates/iota-graphql-e2e-tests/README.md @@ -19,7 +19,7 @@ It is recommended that the database server is started in a docker container. ## Using `docker compose` ```sh -$ POSTGRES_USER=postgres POSTGRES_DB=postgres POSTGRES_PASSWORD=postgrespw POSTGRES_INITDB_ARGS="-U postgres" docker compose -f docker/pg-services-local/docker-compose.yaml up -d postgres +$ POSTGRES_USER=postgres POSTGRES_DB=postgres POSTGRES_PASSWORD=postgrespw POSTGRES_INITDB_ARGS="-U postgres" docker compose -f dev-tools/pg-services-local/docker-compose.yaml up -d postgres ``` ## Using `docker` diff --git a/crates/iota-graphql-e2e-tests/tests/available_range/available_range.exp b/crates/iota-graphql-e2e-tests/tests/available_range/available_range.exp index 6dfbb9b815e..9b7c31b39b0 100644 --- a/crates/iota-graphql-e2e-tests/tests/available_range/available_range.exp +++ b/crates/iota-graphql-e2e-tests/tests/available_range/available_range.exp @@ -6,20 +6,20 @@ Response: { "data": { "availableRange": { "first": { - "digest": "4ELBAJ4AW3LXSgsEoXn525rFUdvtAPNhdavRcGaFqNBr", + "digest": "Cg9T9D1hd7CfrbmtSnPNTtiaa4tXEmafknyZrSoLRqwM", "sequenceNumber": 0 }, "last": { - "digest": "4ELBAJ4AW3LXSgsEoXn525rFUdvtAPNhdavRcGaFqNBr", + "digest": "Cg9T9D1hd7CfrbmtSnPNTtiaa4tXEmafknyZrSoLRqwM", "sequenceNumber": 0 } }, "first": { - "digest": "4ELBAJ4AW3LXSgsEoXn525rFUdvtAPNhdavRcGaFqNBr", + "digest": "Cg9T9D1hd7CfrbmtSnPNTtiaa4tXEmafknyZrSoLRqwM", "sequenceNumber": 0 }, "last": { - "digest": "4ELBAJ4AW3LXSgsEoXn525rFUdvtAPNhdavRcGaFqNBr", + "digest": "Cg9T9D1hd7CfrbmtSnPNTtiaa4tXEmafknyZrSoLRqwM", "sequenceNumber": 0 } } @@ -39,20 +39,20 @@ Response: { "data": { "availableRange": { "first": { - "digest": "4ELBAJ4AW3LXSgsEoXn525rFUdvtAPNhdavRcGaFqNBr", + "digest": "Cg9T9D1hd7CfrbmtSnPNTtiaa4tXEmafknyZrSoLRqwM", "sequenceNumber": 0 }, "last": { - "digest": "6rszwvzRELvaUnWaphcdzP4fd7Q3uhbGYa2sUHkxw6p3", + "digest": "2um6TfiJxKGVxPVdDuvuN1ifqNtTgaP7n6mCSpQnBpbX", "sequenceNumber": 2 } }, "first": { - "digest": "4ELBAJ4AW3LXSgsEoXn525rFUdvtAPNhdavRcGaFqNBr", + "digest": "Cg9T9D1hd7CfrbmtSnPNTtiaa4tXEmafknyZrSoLRqwM", "sequenceNumber": 0 }, "last": { - "digest": "6rszwvzRELvaUnWaphcdzP4fd7Q3uhbGYa2sUHkxw6p3", + "digest": "2um6TfiJxKGVxPVdDuvuN1ifqNtTgaP7n6mCSpQnBpbX", "sequenceNumber": 2 } } diff --git a/crates/iota-graphql-e2e-tests/tests/available_range/available_range.move b/crates/iota-graphql-e2e-tests/tests/available_range/available_range.move index a88e1a9dde9..59a5ee4f030 100644 --- a/crates/iota-graphql-e2e-tests/tests/available_range/available_range.move +++ b/crates/iota-graphql-e2e-tests/tests/available_range/available_range.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --simulator +//# init --protocol-version 3 --simulator //# run-graphql { diff --git a/crates/iota-graphql-e2e-tests/tests/call/checkpoint_connection_pagination.move b/crates/iota-graphql-e2e-tests/tests/call/checkpoint_connection_pagination.move index ef951e1de3a..558a404c3cd 100644 --- a/crates/iota-graphql-e2e-tests/tests/call/checkpoint_connection_pagination.move +++ b/crates/iota-graphql-e2e-tests/tests/call/checkpoint_connection_pagination.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 --simulator +//# init --protocol-version 3 --addresses Test=0x0 --simulator // Test cursor connection pagination logic // The implementation privileges `after`, `before`, `first`, and `last` in that order. diff --git a/crates/iota-graphql-e2e-tests/tests/call/coin_metadata.move b/crates/iota-graphql-e2e-tests/tests/call/coin_metadata.move index 9564caafec2..251b2a67c16 100644 --- a/crates/iota-graphql-e2e-tests/tests/call/coin_metadata.move +++ b/crates/iota-graphql-e2e-tests/tests/call/coin_metadata.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses test=0x0 --accounts A --simulator //# publish --sender A module test::fake { diff --git a/crates/iota-graphql-e2e-tests/tests/call/dynamic_fields.exp b/crates/iota-graphql-e2e-tests/tests/call/dynamic_fields.exp index 4ea401d7a3f..d1187a2d134 100644 --- a/crates/iota-graphql-e2e-tests/tests/call/dynamic_fields.exp +++ b/crates/iota-graphql-e2e-tests/tests/call/dynamic_fields.exp @@ -41,12 +41,12 @@ Response: { { "name": { "type": { - "repr": "u64" + "repr": "vector" }, "data": { - "Number": "0" + "Vector": [] }, - "bcs": "AAAAAAAAAAA=" + "bcs": "AA==" }, "value": { "__typename": "MoveValue" @@ -55,15 +55,15 @@ Response: { { "name": { "type": { - "repr": "vector" + "repr": "u64" }, "data": { - "Vector": [] + "Number": "0" }, - "bcs": "AA==" + "bcs": "AAAAAAAAAAA=" }, "value": { - "__typename": "MoveValue" + "__typename": "MoveObject" } }, { @@ -91,7 +91,7 @@ Response: { "bcs": "AAAAAAAAAAA=" }, "value": { - "__typename": "MoveObject" + "__typename": "MoveValue" } } ] @@ -121,12 +121,12 @@ Response: { { "name": { "type": { - "repr": "u64" + "repr": "vector" }, "data": { - "Number": "0" + "Vector": [] }, - "bcs": "AAAAAAAAAAA=" + "bcs": "AA==" }, "value": { "__typename": "MoveValue" @@ -135,15 +135,15 @@ Response: { { "name": { "type": { - "repr": "vector" + "repr": "u64" }, "data": { - "Vector": [] + "Number": "0" }, - "bcs": "AA==" + "bcs": "AAAAAAAAAAA=" }, "value": { - "__typename": "MoveValue" + "__typename": "MoveObject" } }, { @@ -171,7 +171,7 @@ Response: { "bcs": "AAAAAAAAAAA=" }, "value": { - "__typename": "MoveObject" + "__typename": "MoveValue" } } ] @@ -190,17 +190,17 @@ Response: { { "name": { "type": { - "repr": "u64" + "repr": "vector" }, "data": { - "Number": "0" + "Vector": [] }, - "bcs": "AAAAAAAAAAA=" + "bcs": "AA==" }, "value": { - "bcs": "AAAAAAAAAAA=", + "bcs": "AQAAAAAAAAA=", "data": { - "Number": "0" + "Number": "1" }, "__typename": "MoveValue" } @@ -208,19 +208,15 @@ Response: { { "name": { "type": { - "repr": "vector" + "repr": "u64" }, "data": { - "Vector": [] + "Number": "0" }, - "bcs": "AA==" + "bcs": "AAAAAAAAAAA=" }, "value": { - "bcs": "AQAAAAAAAAA=", - "data": { - "Number": "1" - }, - "__typename": "MoveValue" + "__typename": "MoveObject" } }, { @@ -252,7 +248,11 @@ Response: { "bcs": "AAAAAAAAAAA=" }, "value": { - "__typename": "MoveObject" + "bcs": "AAAAAAAAAAA=", + "data": { + "Number": "0" + }, + "__typename": "MoveValue" } } ] diff --git a/crates/iota-graphql-e2e-tests/tests/call/dynamic_fields.move b/crates/iota-graphql-e2e-tests/tests/call/dynamic_fields.move index 8942d4af172..45db0a290b7 100644 --- a/crates/iota-graphql-e2e-tests/tests/call/dynamic_fields.move +++ b/crates/iota-graphql-e2e-tests/tests/call/dynamic_fields.move @@ -9,7 +9,7 @@ // This test also demonstrates why we need separate dynamicField and dynamicObjectField APIs. // It is possible for a dynamic field and a dynamic object field to share the same name lookup. -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::m { diff --git a/crates/iota-graphql-e2e-tests/tests/call/owned_objects.move b/crates/iota-graphql-e2e-tests/tests/call/owned_objects.move index 98090a2a635..c48cfd43b8a 100644 --- a/crates/iota-graphql-e2e-tests/tests/call/owned_objects.move +++ b/crates/iota-graphql-e2e-tests/tests/call/owned_objects.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 A=0x42 --simulator +//# init --protocol-version 3 --addresses Test=0x0 A=0x42 --simulator // Tests objects on address, object, and owner. // diff --git a/crates/iota-graphql-e2e-tests/tests/call/simple.exp b/crates/iota-graphql-e2e-tests/tests/call/simple.exp index 732d92b4bd4..4f13efa33a6 100644 --- a/crates/iota-graphql-e2e-tests/tests/call/simple.exp +++ b/crates/iota-graphql-e2e-tests/tests/call/simple.exp @@ -25,14 +25,18 @@ task 4, line 32: //# view-object 0,0 Owner: Account Address ( validator_0 ) Version: 1 -Contents: iota::coin::Coin { +Contents: iota_system::staking_pool::StakedIota { id: iota::object::UID { id: iota::object::ID { bytes: fake(0,0), }, }, - balance: iota::balance::Balance { - value: 300000000000000u64, + pool_id: iota::object::ID { + bytes: _, + }, + stake_activation_epoch: 0u64, + principal: iota::balance::Balance { + value: 1500000000000000u64, }, } @@ -77,7 +81,7 @@ Epoch advanced: 5 task 10, line 44: //# view-checkpoint -CheckpointSummary { epoch: 5, seq: 10, content_digest: Ee6ZUSAsovYRwYFDNkWyii9fVgYaAmL9kcdgDwnDQ5if, +CheckpointSummary { epoch: 5, seq: 10, content_digest: CL93M1EZvQpyaKNGyzktdDemHqDsryHhnB2ccsBEGbNN, epoch_rolling_gas_cost_summary: GasCostSummary { computation_cost: 0, computation_cost_burned: 0, storage_cost: 0, storage_rebate: 0, non_refundable_storage_fee: 0 }} task 11, lines 46-51: @@ -114,11 +118,11 @@ task 15, lines 64-69: Headers: { "content-type": "application/json", "content-length": "157", - "x-iota-rpc-version": "0.8.0-testing-no-sha", + "x-iota-rpc-version": "0.9.0-testing-no-sha", "vary": "origin, access-control-request-method, access-control-request-headers", "access-control-allow-origin": "*", } -Service version: 0.8.0-testing-no-sha +Service version: 0.9.0-testing-no-sha Response: { "data": { "checkpoint": { @@ -155,8 +159,8 @@ Response: { "edges": [ { "node": { - "address": "0xeb65a9b4c043abdf3e0fefa3047fb5ea8ecb9d0239477624d1fd55d67d229891", - "digest": "GpfNenBbHNc3zUKfsAGEkiDsryUtvSSDN3k55eAAKsVv", + "address": "0xf5b2015775703e52d984852ad314d54d05b91c04caaa4202da5e36da30e54612", + "digest": "ELFhMCtdYyZbchZ1imeFxcZux4ZDkF5uGtJaXxs23epn", "owner": { "__typename": "AddressOwner" } @@ -182,8 +186,8 @@ Response: { "edges": [ { "node": { - "address": "0xeb65a9b4c043abdf3e0fefa3047fb5ea8ecb9d0239477624d1fd55d67d229891", - "digest": "GpfNenBbHNc3zUKfsAGEkiDsryUtvSSDN3k55eAAKsVv", + "address": "0xf5b2015775703e52d984852ad314d54d05b91c04caaa4202da5e36da30e54612", + "digest": "ELFhMCtdYyZbchZ1imeFxcZux4ZDkF5uGtJaXxs23epn", "owner": { "__typename": "AddressOwner" } @@ -197,8 +201,8 @@ Response: { "edges": [ { "node": { - "address": "0x0133ca00247f984c7280ea2aa1dac2630055ced0d5080825a0d9cdddf523f4f6", - "digest": "7Ye4pTNg8MB1KWaRyz1cT4WdiNKB3MxaMJ1tgxgDbkfS", + "address": "0x1e1bfe85b16cc0699b1d953bb3e9a1ff8c729e0fab4a4fecf9a642ba1ff76473", + "digest": "CFRoehNgzop1Hy7eauzeyXht6dK3GNJ7gCrzp76pHxGx", "owner": { "__typename": "AddressOwner" } @@ -206,8 +210,8 @@ Response: { }, { "node": { - "address": "0x0ef04a70a5215989129b276c3ada1b43a2772169b1db5e157b40743e1ecac0ad", - "digest": "A1Gp7sqj3AeNUYBq8SScjGZJv2Hp9cuvJ1W4V7KZswDy", + "address": "0x5897aac1e95a981e9bd3a7a9bbf0ebbd2cc78cbb0bc594458715e679c84097dc", + "digest": "BedWH4HtkS9dWjFpAsLBM4dcH8mLwKq9xQBMUqxcE23x", "owner": { "__typename": "AddressOwner" } @@ -215,8 +219,8 @@ Response: { }, { "node": { - "address": "0x26bf0f03edd09bb3f862e13cec014d17c26dd739d2563d9c704aa8d206301df2", - "digest": "9LoSY4zTcsU3iZkaXK3vreVgQnNFBmuuTmxhbHXGoTju", + "address": "0x72e3c78b1a7949e55e291745f3e44c423fd6dcddeb00012dfa81b994647c552d", + "digest": "Do5wuaRiJWP2UyqtCXZX1xMaeu6fPCrDAFCyxfpJTWvw", "owner": { "__typename": "AddressOwner" } @@ -224,8 +228,8 @@ Response: { }, { "node": { - "address": "0x3442446f02377d1025b211e51564e4f1c9a46845ccad733beeb06180d3f66c31", - "digest": "BAdieXDoeWexinDjhx69GEu3cKKiXc7XeEZg21cvymTp", + "address": "0x8b58506a0c77bbc2d7ddf7183c02b88318972bdce3e1f0e122e4511e4b4b7696", + "digest": "A2s8G7x2CK6p3Pt5NU29amt5DEP7xL4hpukWnr4jsfSa", "owner": { "__typename": "AddressOwner" } @@ -233,8 +237,8 @@ Response: { }, { "node": { - "address": "0x3c88cad5799a8da0467b8456a7429a97e7c141cfcf25f02faf51aa8cb4840461", - "digest": "Bg4vTzDaxxRmzHqKjRfv895zdc85Hv27xVu9ErR1Utsi", + "address": "0x9448ced828b28e01e6ecea8827a5476c3ea65dda6384c2d2bad2427ea3848b5d", + "digest": "4dh8DYxwoqstXF4nrsUqW6jKgourMiaUxqNHtAWWVL8H", "owner": { "__typename": "AddressOwner" } @@ -242,8 +246,8 @@ Response: { }, { "node": { - "address": "0x4bf4927b5491f5611d4cb83ada18d95c1bf6cc73320a5aa10b521e74a62016d2", - "digest": "mVvbS2vkuQrnRW1FsBjXjpNSdR92fJiRXXyUNdnmb67", + "address": "0x954705871e318ba8de121943084b47a25d42930d7471841971d73402e1e067a5", + "digest": "HHbxAHGH5MApVmZ5howtXkxhEbb7mruqZftcUZBCYX5G", "owner": { "__typename": "AddressOwner" } @@ -251,8 +255,8 @@ Response: { }, { "node": { - "address": "0x51e8ad8999fd29749b8afed87b965adbd781deb08b2f5325e4cf4e5528e17a7f", - "digest": "GgbrysrMW9XnKr2MvsxenrnShDgZ47J4adpwcnL23bdG", + "address": "0xad4386af8fc645ceeb78057b77b2bdc398fde74ba9f31458b3e4c0a381cd9ba4", + "digest": "5WmaWbEeGMqRuMf9icBdjMwdLk59huNsDXREixDNhgmm", "owner": { "__typename": "AddressOwner" } @@ -260,8 +264,8 @@ Response: { }, { "node": { - "address": "0x6139a8d91e9662934040407da8832dabd0f1b4e8ed9a79a6f0075fff7477e1f3", - "digest": "BtL1usVLMKgswweuB6wv7bddgVb4pV1ASmNEEvgiPcB1", + "address": "0xb0027d9b634e7d91f66eff316a097c92ae54ccc9a4f5081b8d5c076c6c0de10b", + "digest": "49fJ1Prd9C7WkLQXKW8C7catXr1pzvvX4NavA4utgcT2", "owner": { "__typename": "AddressOwner" } @@ -269,8 +273,8 @@ Response: { }, { "node": { - "address": "0x7d569ebee60b485b12152a96d76cc7ed6ad521146ad4bd3fd62d12e4c2cae76b", - "digest": "P1Me6uRFcWqjHeNx41oAa4nVQYmYpYpwUk67vd8osjZ", + "address": "0xbe31de2cc7c8fa48810819cdb3f5373451716cb81c16455974c8c2f6a8ef709a", + "digest": "BDyCwG2RwYmw4oM8WNdPoG8f8mD49cwuFx3Qy1HRd3qY", "owner": { "__typename": "AddressOwner" } @@ -278,8 +282,8 @@ Response: { }, { "node": { - "address": "0x847c914d647e7387d7cea4db5ad8fefeffe66773635804a991dfd4385fab6a97", - "digest": "J5mrwkf2RT8qVNjyg9abkJMQWXZEu8PKphBAkPSG3REF", + "address": "0xd70ac75a9489aa0a673a8f8ccd5d4e3e84ade6cf9d2c44d8ba1ce5acb042e02b", + "digest": "FXu6jYwudcV3k6eJa2gwZi5merEWdV2qmgrUKasvrJ5k", "owner": { "__typename": "AddressOwner" } @@ -287,8 +291,8 @@ Response: { }, { "node": { - "address": "0xa79cc544021a721176cdc4497728ec35a5311b8ff0aa293a5d29c0d32cb56056", - "digest": "DU399Yt1piD6iDhMYJ4kywMaorf3rTdW1zUsE7bsB7nX", + "address": "0xd98228affe398ab3bc22286d6b7b11da1442f113d0f2c298a7548c72c8483657", + "digest": "ESfkPLfitLBVAB9u6TjaNuRZfrGHrJvxtkouuJNuZtoL", "owner": { "__typename": "AddressOwner" } @@ -296,8 +300,8 @@ Response: { }, { "node": { - "address": "0xb21c0b03871456040643740e238d03bde38de240d7b17055f4670777a19f07b8", - "digest": "H6FaahQNYfKavippZbge7xHzox29vRPqAaCMuUWnYmCz", + "address": "0xfba59abaf5940f30567b53f21a580203b3e5766f51cbeeab8f15de918bf14fb1", + "digest": "HeM2fifov6dm5EBGLSnBmvwXQZGb8dfwJr8fiWQNNRxH", "owner": { "__typename": "AddressOwner" } @@ -375,7 +379,7 @@ Response: { "data": { "serviceConfig": { "availableVersions": [ - "0.8" + "0.9" ] } } diff --git a/crates/iota-graphql-e2e-tests/tests/call/simple.move b/crates/iota-graphql-e2e-tests/tests/call/simple.move index 91dee3fe9a5..ef6aa861856 100644 --- a/crates/iota-graphql-e2e-tests/tests/call/simple.move +++ b/crates/iota-graphql-e2e-tests/tests/call/simple.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 A=0x42 --simulator --custom-validator-account --reference-gas-price 234 --default-gas-price 1000 +//# init --protocol-version 3 --addresses Test=0x0 A=0x42 --simulator --custom-validator-account --reference-gas-price 234 --default-gas-price 1000 //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/balances.exp b/crates/iota-graphql-e2e-tests/tests/consistency/balances.exp index 98f98b2c9c6..2175bbadbdc 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/balances.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/balances.exp @@ -80,7 +80,7 @@ Response: { }, { "coinType": { - "repr": "0xbdf6cc52e18ba611bf235bf856f16c4c30f609167401546f52e08ba168429473::fake::FAKE" + "repr": "0xd083d89dc732b2e50099892c964ded83f62e4b19d28e1ca62f99ea8a003fd5cf::fake::FAKE" }, "coinObjectCount": 3, "totalBalance": "700" @@ -103,7 +103,7 @@ Response: { { "sender": { "fakeCoinBalance": { - "totalBalance": "500" + "totalBalance": "600" }, "allBalances": { "nodes": [ @@ -116,10 +116,10 @@ Response: { }, { "coinType": { - "repr": "0xbdf6cc52e18ba611bf235bf856f16c4c30f609167401546f52e08ba168429473::fake::FAKE" + "repr": "0xd083d89dc732b2e50099892c964ded83f62e4b19d28e1ca62f99ea8a003fd5cf::fake::FAKE" }, "coinObjectCount": 2, - "totalBalance": "500" + "totalBalance": "600" } ] } @@ -139,7 +139,7 @@ Response: { { "sender": { "fakeCoinBalance": { - "totalBalance": "200" + "totalBalance": "300" }, "allBalances": { "nodes": [ @@ -152,10 +152,10 @@ Response: { }, { "coinType": { - "repr": "0xbdf6cc52e18ba611bf235bf856f16c4c30f609167401546f52e08ba168429473::fake::FAKE" + "repr": "0xd083d89dc732b2e50099892c964ded83f62e4b19d28e1ca62f99ea8a003fd5cf::fake::FAKE" }, "coinObjectCount": 1, - "totalBalance": "200" + "totalBalance": "300" } ] } @@ -196,7 +196,7 @@ Response: { }, { "coinType": { - "repr": "0xbdf6cc52e18ba611bf235bf856f16c4c30f609167401546f52e08ba168429473::fake::FAKE" + "repr": "0xd083d89dc732b2e50099892c964ded83f62e4b19d28e1ca62f99ea8a003fd5cf::fake::FAKE" }, "coinObjectCount": 3, "totalBalance": "700" @@ -219,7 +219,7 @@ Response: { { "sender": { "fakeCoinBalance": { - "totalBalance": "500" + "totalBalance": "600" }, "allBalances": { "nodes": [ @@ -232,10 +232,10 @@ Response: { }, { "coinType": { - "repr": "0xbdf6cc52e18ba611bf235bf856f16c4c30f609167401546f52e08ba168429473::fake::FAKE" + "repr": "0xd083d89dc732b2e50099892c964ded83f62e4b19d28e1ca62f99ea8a003fd5cf::fake::FAKE" }, "coinObjectCount": 2, - "totalBalance": "500" + "totalBalance": "600" } ] } @@ -255,7 +255,7 @@ Response: { { "sender": { "fakeCoinBalance": { - "totalBalance": "200" + "totalBalance": "300" }, "allBalances": { "nodes": [ @@ -268,10 +268,10 @@ Response: { }, { "coinType": { - "repr": "0xbdf6cc52e18ba611bf235bf856f16c4c30f609167401546f52e08ba168429473::fake::FAKE" + "repr": "0xd083d89dc732b2e50099892c964ded83f62e4b19d28e1ca62f99ea8a003fd5cf::fake::FAKE" }, "coinObjectCount": 1, - "totalBalance": "200" + "totalBalance": "300" } ] } @@ -334,7 +334,7 @@ Response: { { "sender": { "fakeCoinBalance": { - "totalBalance": "500" + "totalBalance": "600" }, "allBalances": { "nodes": [ @@ -347,10 +347,10 @@ Response: { }, { "coinType": { - "repr": "0xbdf6cc52e18ba611bf235bf856f16c4c30f609167401546f52e08ba168429473::fake::FAKE" + "repr": "0xd083d89dc732b2e50099892c964ded83f62e4b19d28e1ca62f99ea8a003fd5cf::fake::FAKE" }, "coinObjectCount": 2, - "totalBalance": "500" + "totalBalance": "600" } ] } @@ -370,7 +370,7 @@ Response: { { "sender": { "fakeCoinBalance": { - "totalBalance": "200" + "totalBalance": "300" }, "allBalances": { "nodes": [ @@ -383,10 +383,10 @@ Response: { }, { "coinType": { - "repr": "0xbdf6cc52e18ba611bf235bf856f16c4c30f609167401546f52e08ba168429473::fake::FAKE" + "repr": "0xd083d89dc732b2e50099892c964ded83f62e4b19d28e1ca62f99ea8a003fd5cf::fake::FAKE" }, "coinObjectCount": 1, - "totalBalance": "200" + "totalBalance": "300" } ] } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/balances.move b/crates/iota-graphql-e2e-tests/tests/consistency/balances.move index a4d66327955..5cb4bc54786 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/balances.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/balances.move @@ -15,7 +15,7 @@ // snapshot@[0, 4), first two transaction blocks are out of available range. // snapshot@[0, 6), all transaction blocks are out of available range. -//# init --protocol-version 1 --addresses P0=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses P0=0x0 --accounts A B --simulator //# publish --sender A module P0::fake { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/checkpoints/transaction_blocks.exp b/crates/iota-graphql-e2e-tests/tests/consistency/checkpoints/transaction_blocks.exp index 674bc426868..c01d16c5d2c 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/checkpoints/transaction_blocks.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/checkpoints/transaction_blocks.exp @@ -94,12 +94,12 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "ywf9ak8rV9kDDx73zGH5vtTJkHmfxXHvLbSGhwAyxdM", + "digest": "7BPdUc3LpRmchBjPiSZ5Uju5yDZMbVhTQNYjDjmTXP5D", "sender": { "objects": { "edges": [ { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilAAwAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGAwAAAAAAAAA=" } ] } @@ -109,12 +109,12 @@ Response: { { "cursor": "eyJjIjozLCJ0IjozLCJpIjpmYWxzZX0", "node": { - "digest": "25MhkStEmB44afcLcmtdA426Qbo6a5z6sJrURSfn4wSM", + "digest": "3s48BKV1NGNPV5VRKFsAJKWiyUiWDLVxguSERpo8qRTV", "sender": { "objects": { "edges": [ { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilAAwAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGAwAAAAAAAAA=" } ] } @@ -124,12 +124,12 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo0LCJpIjpmYWxzZX0", "node": { - "digest": "Hutiw6GLNGCuRwMTZhxFW2CrYF6PYt2DomNMwtN1hLUo", + "digest": "6adEehTM2gxGKdumJXeKFwcaYhSB6RwGHbqg7Fv7JBmk", "sender": { "objects": { "edges": [ { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilAAwAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGAwAAAAAAAAA=" } ] } @@ -139,12 +139,12 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo1LCJpIjpmYWxzZX0", "node": { - "digest": "4YeKsZh5nDLNfbNVT2GRwenv1gvaVdDde9BASgiBqkv8", + "digest": "E4Lt2ME1UNXjXnuVzF9CMyrHnhdQWbWRXj3myqpzZMrP", "sender": { "objects": { "edges": [ { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilAAwAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGAwAAAAAAAAA=" } ] } @@ -161,12 +161,12 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo2LCJpIjpmYWxzZX0", "node": { - "digest": "EnghHbuhpcGuNpC1L2iqikieXAyHJUnanzg5urDmwgdz", + "digest": "DEYu7BBRqZyroncK1on7suMYzHy7GLJMoSsC6K8d14BR", "sender": { "objects": { "edges": [ { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilAAwAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGAwAAAAAAAAA=" } ] } @@ -176,12 +176,12 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo3LCJpIjpmYWxzZX0", "node": { - "digest": "9G4gNmS2XYetMxBty65fKfx9iDRBetdpdNZuyK7oRd6c", + "digest": "32S1CoAgaGv7diFo2Bs7StnPB7Aakoma1JUT4k4cUpA7", "sender": { "objects": { "edges": [ { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilAAwAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGAwAAAAAAAAA=" } ] } @@ -191,12 +191,12 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo4LCJpIjpmYWxzZX0", "node": { - "digest": "3ivB1ebWRpc4Si1vd8tsSY7XwaY7DrumxdS3KdXyZLVr", + "digest": "3gEc6E6ypnwUqRtVBcF3ZwXPEQJaisXDDaK21p4sZyM8", "sender": { "objects": { "edges": [ { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilAAwAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGAwAAAAAAAAA=" } ] } @@ -213,12 +213,12 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo5LCJpIjpmYWxzZX0", "node": { - "digest": "6i5QZFKJgWGMArwRBVmd124V2mWv4BaiVVuoBxxfe3Mu", + "digest": "6RpeYVhE2pRoKHE4yP99YreciLexUN7WtKDG3qrRpTpE", "sender": { "objects": { "edges": [ { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilAAwAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGAwAAAAAAAAA=" } ] } @@ -228,12 +228,12 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoxMCwiaSI6ZmFsc2V9", "node": { - "digest": "ATX5C4Vryx6k3utG6b2ASv32hdXZ3dVkxVMHq8E6JbTE", + "digest": "HFhCBUpVFLKMXW3GT7of6iyZJCnMerKdLNRsZE9KVR5C", "sender": { "objects": { "edges": [ { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilAAwAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGAwAAAAAAAAA=" } ] } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/checkpoints/transaction_blocks.move b/crates/iota-graphql-e2e-tests/tests/consistency/checkpoints/transaction_blocks.move index db7c4e7aa68..234a112659b 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/checkpoints/transaction_blocks.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/checkpoints/transaction_blocks.move @@ -9,7 +9,7 @@ // 2 | 3 // 3 | 2 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/coins.exp b/crates/iota-graphql-e2e-tests/tests/consistency/coins.exp index f4477a71240..30a12ed6f1f 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/coins.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/coins.exp @@ -33,7 +33,7 @@ Response: { "queryCoinsAtLatest": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAgAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAgAAAAAAAAA=", "node": { "consistentStateForEachCoin": { "owner": { @@ -41,37 +41,37 @@ Response: { "coins": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAgAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAgAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100100" + "value": "100200" } } } } }, { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAgAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAgAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } } }, { - "cursor": "INqo/pVaj/aZbQ0uWk9SX8bQdKsDqH22Yewchi34U2E0AgAAAAAAAAA=", + "cursor": "IPSGBTtl6PcfooWqNntlxy0bgSKaP1vrFtTbW1TbPtLtAgAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0xdaa8fe955a8ff6996d0d2e5a4f525fc6d074ab03a87db661ec1c862df8536134", + "id": "0xf486053b65e8f71fa285aa367b65c72d1b81229a3f5beb16d4db5b54db3ed2ed", "balance": { "value": "300" } @@ -85,16 +85,16 @@ Response: { }, "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100100" + "value": "100200" } } } } }, { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAgAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAgAAAAAAAAA=", "node": { "consistentStateForEachCoin": { "owner": { @@ -102,37 +102,37 @@ Response: { "coins": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAgAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAgAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100100" + "value": "100200" } } } } }, { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAgAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAgAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } } }, { - "cursor": "INqo/pVaj/aZbQ0uWk9SX8bQdKsDqH22Yewchi34U2E0AgAAAAAAAAA=", + "cursor": "IPSGBTtl6PcfooWqNntlxy0bgSKaP1vrFtTbW1TbPtLtAgAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0xdaa8fe955a8ff6996d0d2e5a4f525fc6d074ab03a87db661ec1c862df8536134", + "id": "0xf486053b65e8f71fa285aa367b65c72d1b81229a3f5beb16d4db5b54db3ed2ed", "balance": { "value": "300" } @@ -146,16 +146,16 @@ Response: { }, "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } } }, { - "cursor": "INqo/pVaj/aZbQ0uWk9SX8bQdKsDqH22Yewchi34U2E0AgAAAAAAAAA=", + "cursor": "IPSGBTtl6PcfooWqNntlxy0bgSKaP1vrFtTbW1TbPtLtAgAAAAAAAAA=", "node": { "consistentStateForEachCoin": { "owner": { @@ -163,37 +163,37 @@ Response: { "coins": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAgAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAgAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100100" + "value": "100200" } } } } }, { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAgAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAgAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } } }, { - "cursor": "INqo/pVaj/aZbQ0uWk9SX8bQdKsDqH22Yewchi34U2E0AgAAAAAAAAA=", + "cursor": "IPSGBTtl6PcfooWqNntlxy0bgSKaP1vrFtTbW1TbPtLtAgAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0xdaa8fe955a8ff6996d0d2e5a4f525fc6d074ab03a87db661ec1c862df8536134", + "id": "0xf486053b65e8f71fa285aa367b65c72d1b81229a3f5beb16d4db5b54db3ed2ed", "balance": { "value": "300" } @@ -207,7 +207,7 @@ Response: { }, "contents": { "json": { - "id": "0xdaa8fe955a8ff6996d0d2e5a4f525fc6d074ab03a87db661ec1c862df8536134", + "id": "0xf486053b65e8f71fa285aa367b65c72d1b81229a3f5beb16d4db5b54db3ed2ed", "balance": { "value": "300" } @@ -221,37 +221,37 @@ Response: { "coins": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAgAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAgAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100100" + "value": "100200" } } } } }, { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAgAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAgAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } } }, { - "cursor": "INqo/pVaj/aZbQ0uWk9SX8bQdKsDqH22Yewchi34U2E0AgAAAAAAAAA=", + "cursor": "IPSGBTtl6PcfooWqNntlxy0bgSKaP1vrFtTbW1TbPtLtAgAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0xdaa8fe955a8ff6996d0d2e5a4f525fc6d074ab03a87db661ec1c862df8536134", + "id": "0xf486053b65e8f71fa285aa367b65c72d1b81229a3f5beb16d4db5b54db3ed2ed", "balance": { "value": "300" } @@ -272,7 +272,7 @@ Response: { "queryCoinsAtChkpt1": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAQAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAQAAAAAAAAA=", "node": { "consistentStateForEachCoin": { "owner": { @@ -280,37 +280,37 @@ Response: { "coins": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAQAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100" + "value": "200" } } } } }, { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAQAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } } }, { - "cursor": "INqo/pVaj/aZbQ0uWk9SX8bQdKsDqH22Yewchi34U2E0AQAAAAAAAAA=", + "cursor": "IPSGBTtl6PcfooWqNntlxy0bgSKaP1vrFtTbW1TbPtLtAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0xdaa8fe955a8ff6996d0d2e5a4f525fc6d074ab03a87db661ec1c862df8536134", + "id": "0xf486053b65e8f71fa285aa367b65c72d1b81229a3f5beb16d4db5b54db3ed2ed", "balance": { "value": "300" } @@ -324,16 +324,16 @@ Response: { }, "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100" + "value": "200" } } } } }, { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAQAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAQAAAAAAAAA=", "node": { "consistentStateForEachCoin": { "owner": { @@ -341,37 +341,37 @@ Response: { "coins": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAQAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100" + "value": "200" } } } } }, { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAQAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } } }, { - "cursor": "INqo/pVaj/aZbQ0uWk9SX8bQdKsDqH22Yewchi34U2E0AQAAAAAAAAA=", + "cursor": "IPSGBTtl6PcfooWqNntlxy0bgSKaP1vrFtTbW1TbPtLtAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0xdaa8fe955a8ff6996d0d2e5a4f525fc6d074ab03a87db661ec1c862df8536134", + "id": "0xf486053b65e8f71fa285aa367b65c72d1b81229a3f5beb16d4db5b54db3ed2ed", "balance": { "value": "300" } @@ -385,9 +385,9 @@ Response: { }, "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } @@ -399,26 +399,26 @@ Response: { "coins": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAQAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100" + "value": "200" } } } } }, { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAQAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } @@ -453,7 +453,7 @@ Response: { "queryCoins": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAwAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAwAAAAAAAAA=", "node": { "owner": { "owner": { @@ -461,13 +461,13 @@ Response: { "coins": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAwAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAwAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100100" + "value": "100200" } } } @@ -479,16 +479,16 @@ Response: { }, "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100100" + "value": "100200" } } } } }, { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAwAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAwAAAAAAAAA=", "node": { "owner": { "owner": { @@ -496,24 +496,24 @@ Response: { "coins": { "edges": [ { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAwAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAwAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } } }, { - "cursor": "INqo/pVaj/aZbQ0uWk9SX8bQdKsDqH22Yewchi34U2E0AwAAAAAAAAA=", + "cursor": "IPSGBTtl6PcfooWqNntlxy0bgSKaP1vrFtTbW1TbPtLtAwAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0xdaa8fe955a8ff6996d0d2e5a4f525fc6d074ab03a87db661ec1c862df8536134", + "id": "0xf486053b65e8f71fa285aa367b65c72d1b81229a3f5beb16d4db5b54db3ed2ed", "balance": { "value": "300" } @@ -527,16 +527,16 @@ Response: { }, "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } } }, { - "cursor": "INqo/pVaj/aZbQ0uWk9SX8bQdKsDqH22Yewchi34U2E0AwAAAAAAAAA=", + "cursor": "IPSGBTtl6PcfooWqNntlxy0bgSKaP1vrFtTbW1TbPtLtAwAAAAAAAAA=", "node": { "owner": { "owner": { @@ -544,24 +544,24 @@ Response: { "coins": { "edges": [ { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAwAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAwAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } } }, { - "cursor": "INqo/pVaj/aZbQ0uWk9SX8bQdKsDqH22Yewchi34U2E0AwAAAAAAAAA=", + "cursor": "IPSGBTtl6PcfooWqNntlxy0bgSKaP1vrFtTbW1TbPtLtAwAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0xdaa8fe955a8ff6996d0d2e5a4f525fc6d074ab03a87db661ec1c862df8536134", + "id": "0xf486053b65e8f71fa285aa367b65c72d1b81229a3f5beb16d4db5b54db3ed2ed", "balance": { "value": "300" } @@ -575,7 +575,7 @@ Response: { }, "contents": { "json": { - "id": "0xdaa8fe955a8ff6996d0d2e5a4f525fc6d074ab03a87db661ec1c862df8536134", + "id": "0xf486053b65e8f71fa285aa367b65c72d1b81229a3f5beb16d4db5b54db3ed2ed", "balance": { "value": "300" } @@ -589,13 +589,13 @@ Response: { "coins": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAwAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAwAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100100" + "value": "100200" } } } @@ -608,24 +608,24 @@ Response: { "coins": { "edges": [ { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAwAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAwAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } } }, { - "cursor": "INqo/pVaj/aZbQ0uWk9SX8bQdKsDqH22Yewchi34U2E0AwAAAAAAAAA=", + "cursor": "IPSGBTtl6PcfooWqNntlxy0bgSKaP1vrFtTbW1TbPtLtAwAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0xdaa8fe955a8ff6996d0d2e5a4f525fc6d074ab03a87db661ec1c862df8536134", + "id": "0xf486053b65e8f71fa285aa367b65c72d1b81229a3f5beb16d4db5b54db3ed2ed", "balance": { "value": "300" } @@ -658,7 +658,7 @@ Response: { "queryCoinsAtChkpt1BeforeSnapshotCatchup": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAQAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAQAAAAAAAAA=", "node": { "consistentStateForEachCoin": { "owner": { @@ -666,37 +666,37 @@ Response: { "coins": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAQAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100" + "value": "200" } } } } }, { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAQAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } } }, { - "cursor": "INqo/pVaj/aZbQ0uWk9SX8bQdKsDqH22Yewchi34U2E0AQAAAAAAAAA=", + "cursor": "IPSGBTtl6PcfooWqNntlxy0bgSKaP1vrFtTbW1TbPtLtAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0xdaa8fe955a8ff6996d0d2e5a4f525fc6d074ab03a87db661ec1c862df8536134", + "id": "0xf486053b65e8f71fa285aa367b65c72d1b81229a3f5beb16d4db5b54db3ed2ed", "balance": { "value": "300" } @@ -710,16 +710,16 @@ Response: { }, "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100" + "value": "200" } } } } }, { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAQAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAQAAAAAAAAA=", "node": { "consistentStateForEachCoin": { "owner": { @@ -727,37 +727,37 @@ Response: { "coins": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAQAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100" + "value": "200" } } } } }, { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAQAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } } }, { - "cursor": "INqo/pVaj/aZbQ0uWk9SX8bQdKsDqH22Yewchi34U2E0AQAAAAAAAAA=", + "cursor": "IPSGBTtl6PcfooWqNntlxy0bgSKaP1vrFtTbW1TbPtLtAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0xdaa8fe955a8ff6996d0d2e5a4f525fc6d074ab03a87db661ec1c862df8536134", + "id": "0xf486053b65e8f71fa285aa367b65c72d1b81229a3f5beb16d4db5b54db3ed2ed", "balance": { "value": "300" } @@ -771,9 +771,9 @@ Response: { }, "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } @@ -785,26 +785,26 @@ Response: { "coins": { "edges": [ { - "cursor": "IFZ/ZyCVFYjCf7cignHQQS2MNn/eq+6kji+rGr7Di8pEAQAAAAAAAAA=", + "cursor": "IApniwe7Z2bjgyFQCb0B9CXz4dET8IjzA12i9bRMR7/CAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x567f6720951588c27fb7228271d0412d8c367fdeabeea48e2fab1abec38bca44", + "id": "0x0a678b07bb6766e383215009bd01f425f3e1d113f088f3035da2f5b44c47bfc2", "balance": { - "value": "100" + "value": "200" } } } } }, { - "cursor": "IJ3CINFJKYLv+Em6UHIRANMjkBe7f+s1nKjlNzVkxaJsAQAAAAAAAAA=", + "cursor": "IKvInbcjeUbNXxlsZmlaw02PXhlQ+UuIUGdLXk9hi5gJAQAAAAAAAAA=", "node": { "contents": { "json": { - "id": "0x9dc220d1492982eff849ba50721100d3239017bb7feb359ca8e5373564c5a26c", + "id": "0xabc89db7237946cd5f196c66695ac34d8f5e1950f94b8850674b5e4f618b9809", "balance": { - "value": "200" + "value": "100" } } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/coins.move b/crates/iota-graphql-e2e-tests/tests/consistency/coins.move index 48557666ed0..65a0ef359c5 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/coins.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/coins.move @@ -8,7 +8,7 @@ // coin2@A | coin2@B | coin2@B | coin2@B | coin2@B | coin2@B // coin3@A | coin3@A | coin3@A | coin3@A | coin3@A | coin3@A -//# init --protocol-version 1 --addresses P0=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses P0=0x0 --accounts A B --simulator //# publish --sender A module P0::fake { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_df.exp b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_df.exp index 7797f60515a..fdda38195c5 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_df.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_df.exp @@ -90,7 +90,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IDlNUKtDqd9vWwIVLOcN6o7pIjYlCnVi16y5TbsLXWPyAQAAAAAAAAA=", + "cursor": "IAOkY3jrEBbDW3xbrO0c1Mx8MP9pP+atJlOUX17hx2rEAQAAAAAAAAA=", "node": { "name": { "bcs": "A2RmNA==" @@ -101,24 +101,24 @@ Response: { } }, { - "cursor": "IJ+efA+QWUV9kAUqmpMdM3i4HNzB+PLztem56WantRmIAQAAAAAAAAA=", + "cursor": "IFv0f9anLK64q0vDAtSGl/jl4D7Og9M2gXu8ei77LkyJAQAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNQ==" + "bcs": "A2RmNg==" }, "value": { - "json": "df5" + "json": "df6" } } }, { - "cursor": "INReOXMS6qJ8gt1WilNgHeSOMmhr2RmX7DmC5kEEwJW8AQAAAAAAAAA=", + "cursor": "INcXbq/S1eJ+TG+iUTbJQB5R7pjHOtoI/CsDpGFVdlpwAQAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNg==" + "bcs": "A2RmNQ==" }, "value": { - "json": "df6" + "json": "df5" } } } @@ -139,7 +139,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IDlNUKtDqd9vWwIVLOcN6o7pIjYlCnVi16y5TbsLXWPyAQAAAAAAAAA=", + "cursor": "IAOkY3jrEBbDW3xbrO0c1Mx8MP9pP+atJlOUX17hx2rEAQAAAAAAAAA=", "node": { "name": { "bcs": "A2RmNA==" @@ -150,24 +150,24 @@ Response: { } }, { - "cursor": "IJ+efA+QWUV9kAUqmpMdM3i4HNzB+PLztem56WantRmIAQAAAAAAAAA=", + "cursor": "IFv0f9anLK64q0vDAtSGl/jl4D7Og9M2gXu8ei77LkyJAQAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNQ==" + "bcs": "A2RmNg==" }, "value": { - "json": "df5" + "json": "df6" } } }, { - "cursor": "INReOXMS6qJ8gt1WilNgHeSOMmhr2RmX7DmC5kEEwJW8AQAAAAAAAAA=", + "cursor": "INcXbq/S1eJ+TG+iUTbJQB5R7pjHOtoI/CsDpGFVdlpwAQAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNg==" + "bcs": "A2RmNQ==" }, "value": { - "json": "df6" + "json": "df5" } } } @@ -188,7 +188,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IDlNUKtDqd9vWwIVLOcN6o7pIjYlCnVi16y5TbsLXWPyAQAAAAAAAAA=", + "cursor": "IAOkY3jrEBbDW3xbrO0c1Mx8MP9pP+atJlOUX17hx2rEAQAAAAAAAAA=", "node": { "name": { "bcs": "A2RmNA==" @@ -199,7 +199,7 @@ Response: { } }, { - "cursor": "IJOw9qCrVkGI8COYq8tW/sbkKDI7w1ag9Pu3bm8smSxgAQAAAAAAAAA=", + "cursor": "IDnc1DUmdy8ysuS4Uqd9eWxiQASc5PGAtHosmFU57ivkAQAAAAAAAAA=", "node": { "name": { "bcs": "A2RmMg==" @@ -210,46 +210,46 @@ Response: { } }, { - "cursor": "IJR01eHq28HFpZPEQAEIn11wvGH3QhEIUdKBy/U4+HTTAQAAAAAAAAA=", + "cursor": "IFv0f9anLK64q0vDAtSGl/jl4D7Og9M2gXu8ei77LkyJAQAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMw==" + "bcs": "A2RmNg==" }, "value": { - "json": "df3" + "json": "df6" } } }, { - "cursor": "IJ+efA+QWUV9kAUqmpMdM3i4HNzB+PLztem56WantRmIAQAAAAAAAAA=", + "cursor": "IH/f6fG8OIen97QWjQixM07rBmDXJS6IpYwlTx/70YsFAQAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNQ==" + "bcs": "A2RmMQ==" }, "value": { - "json": "df5" + "json": "df1" } } }, { - "cursor": "INReOXMS6qJ8gt1WilNgHeSOMmhr2RmX7DmC5kEEwJW8AQAAAAAAAAA=", + "cursor": "IK14TshYNK60jZChixb1mT91DCUiBE+d1Dp56IWxnjN9AQAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNg==" + "bcs": "A2RmMw==" }, "value": { - "json": "df6" + "json": "df3" } } }, { - "cursor": "IPAzcJ3N3Kd+REjvmwJLLDHNOedOaN/m/SpGEatoJZlnAQAAAAAAAAA=", + "cursor": "INcXbq/S1eJ+TG+iUTbJQB5R7pjHOtoI/CsDpGFVdlpwAQAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMQ==" + "bcs": "A2RmNQ==" }, "value": { - "json": "df1" + "json": "df5" } } } @@ -283,7 +283,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IDlNUKtDqd9vWwIVLOcN6o7pIjYlCnVi16y5TbsLXWPyAQAAAAAAAAA=", + "cursor": "IAOkY3jrEBbDW3xbrO0c1Mx8MP9pP+atJlOUX17hx2rEAQAAAAAAAAA=", "node": { "name": { "bcs": "A2RmNA==" @@ -294,24 +294,24 @@ Response: { } }, { - "cursor": "IJ+efA+QWUV9kAUqmpMdM3i4HNzB+PLztem56WantRmIAQAAAAAAAAA=", + "cursor": "IFv0f9anLK64q0vDAtSGl/jl4D7Og9M2gXu8ei77LkyJAQAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNQ==" + "bcs": "A2RmNg==" }, "value": { - "json": "df5" + "json": "df6" } } }, { - "cursor": "INReOXMS6qJ8gt1WilNgHeSOMmhr2RmX7DmC5kEEwJW8AQAAAAAAAAA=", + "cursor": "INcXbq/S1eJ+TG+iUTbJQB5R7pjHOtoI/CsDpGFVdlpwAQAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNg==" + "bcs": "A2RmNQ==" }, "value": { - "json": "df6" + "json": "df5" } } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_df.move b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_df.move index 18f046df98f..c42279e9c60 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_df.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_df.move @@ -10,7 +10,7 @@ // 5 | removed df1, df2, df3 // 6 | mutated parent -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_dof.exp b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_dof.exp index 64fbcfc0f80..3c09e5e36cb 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_dof.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_dof.exp @@ -41,7 +41,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IClB4T7AHWEOVWKrXS5TxSolCdMgz1kvtKBpJwRFEQ4OAQAAAAAAAAA=", + "cursor": "IJ5a72V7783CzJV8LuLqMbXqtMsjLn36fDAtHjr10BjOAQAAAAAAAAA=", "node": { "name": { "bcs": "KgAAAAAAAAA=" @@ -49,7 +49,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x253884793f1f9a266382615636a3407f3e624c02d0a41627cdf04ec9a9a8e26e", + "id": "0x63f53752157292d413cd6ca4b92ebcab016a084f74bfd143f555f57f1fa412bd", "count": "0" } } @@ -65,7 +65,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x253884793f1f9a266382615636a3407f3e624c02d0a41627cdf04ec9a9a8e26e", + "id": "0x63f53752157292d413cd6ca4b92ebcab016a084f74bfd143f555f57f1fa412bd", "count": "0" } } @@ -77,7 +77,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IClB4T7AHWEOVWKrXS5TxSolCdMgz1kvtKBpJwRFEQ4OAQAAAAAAAAA=", + "cursor": "IJ5a72V7783CzJV8LuLqMbXqtMsjLn36fDAtHjr10BjOAQAAAAAAAAA=", "node": { "name": { "bcs": "KgAAAAAAAAA=" @@ -85,7 +85,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x253884793f1f9a266382615636a3407f3e624c02d0a41627cdf04ec9a9a8e26e", + "id": "0x63f53752157292d413cd6ca4b92ebcab016a084f74bfd143f555f57f1fa412bd", "count": "0" } } @@ -101,7 +101,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x253884793f1f9a266382615636a3407f3e624c02d0a41627cdf04ec9a9a8e26e", + "id": "0x63f53752157292d413cd6ca4b92ebcab016a084f74bfd143f555f57f1fa412bd", "count": "0" } } @@ -117,7 +117,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x253884793f1f9a266382615636a3407f3e624c02d0a41627cdf04ec9a9a8e26e", + "id": "0x63f53752157292d413cd6ca4b92ebcab016a084f74bfd143f555f57f1fa412bd", "count": "0" } } @@ -168,7 +168,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x253884793f1f9a266382615636a3407f3e624c02d0a41627cdf04ec9a9a8e26e", + "id": "0x63f53752157292d413cd6ca4b92ebcab016a084f74bfd143f555f57f1fa412bd", "count": "0" } } @@ -222,7 +222,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x253884793f1f9a266382615636a3407f3e624c02d0a41627cdf04ec9a9a8e26e", + "id": "0x63f53752157292d413cd6ca4b92ebcab016a084f74bfd143f555f57f1fa412bd", "count": "0" } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_dof.move b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_dof.move index e50ebdb2ceb..b4e49ecbee3 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_dof.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/deleted_dof.move @@ -10,7 +10,7 @@ // 5 | | deleted child // 6 | | mutated parent -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer.exp b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer.exp index 404738532fc..03d7d92966b 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer.exp @@ -36,7 +36,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IK1Tx9SLMlBMaGqVyH107e/Uq5o56RGyC5wdCvulOCUYAQAAAAAAAAA=", + "cursor": "IIxlYMuyiBfpICRFw7EmSRs0LsIbcKIVngvYcbpZQhJWAQAAAAAAAAA=", "node": { "name": { "bcs": "KgAAAAAAAAA=" @@ -44,7 +44,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xcff88bb86f25644cf26a3943b3d5ea96fdda9a5c65c49bb80443da8137504e73", + "id": "0x7f1515cdcb6f867821d2559cfe754ea0d2345a4c7ac4726451d5548e78b81339", "count": "0" } } @@ -60,7 +60,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xcff88bb86f25644cf26a3943b3d5ea96fdda9a5c65c49bb80443da8137504e73", + "id": "0x7f1515cdcb6f867821d2559cfe754ea0d2345a4c7ac4726451d5548e78b81339", "count": "0" } } @@ -71,7 +71,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IK1Tx9SLMlBMaGqVyH107e/Uq5o56RGyC5wdCvulOCUYAQAAAAAAAAA=", + "cursor": "IIxlYMuyiBfpICRFw7EmSRs0LsIbcKIVngvYcbpZQhJWAQAAAAAAAAA=", "node": { "name": { "bcs": "KgAAAAAAAAA=" @@ -79,7 +79,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xcff88bb86f25644cf26a3943b3d5ea96fdda9a5c65c49bb80443da8137504e73", + "id": "0x7f1515cdcb6f867821d2559cfe754ea0d2345a4c7ac4726451d5548e78b81339", "count": "0" } } @@ -95,7 +95,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xcff88bb86f25644cf26a3943b3d5ea96fdda9a5c65c49bb80443da8137504e73", + "id": "0x7f1515cdcb6f867821d2559cfe754ea0d2345a4c7ac4726451d5548e78b81339", "count": "0" } } @@ -107,7 +107,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IK1Tx9SLMlBMaGqVyH107e/Uq5o56RGyC5wdCvulOCUYAQAAAAAAAAA=", + "cursor": "IIxlYMuyiBfpICRFw7EmSRs0LsIbcKIVngvYcbpZQhJWAQAAAAAAAAA=", "node": { "name": { "bcs": "KgAAAAAAAAA=" @@ -115,7 +115,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xcff88bb86f25644cf26a3943b3d5ea96fdda9a5c65c49bb80443da8137504e73", + "id": "0x7f1515cdcb6f867821d2559cfe754ea0d2345a4c7ac4726451d5548e78b81339", "count": "0" } } @@ -131,7 +131,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xcff88bb86f25644cf26a3943b3d5ea96fdda9a5c65c49bb80443da8137504e73", + "id": "0x7f1515cdcb6f867821d2559cfe754ea0d2345a4c7ac4726451d5548e78b81339", "count": "0" } } @@ -184,7 +184,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xcff88bb86f25644cf26a3943b3d5ea96fdda9a5c65c49bb80443da8137504e73", + "id": "0x7f1515cdcb6f867821d2559cfe754ea0d2345a4c7ac4726451d5548e78b81339", "count": "0" } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer.move b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer.move index 660730c5df2..5b8767227fa 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer.move @@ -15,7 +15,7 @@ // 3 | 3 | added child to parent // 4 | 4 | reclaimed child from parent -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer_reclaim_add.exp b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer_reclaim_add.exp index a941516ec37..6c42a4a84a6 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer_reclaim_add.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer_reclaim_add.exp @@ -61,7 +61,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IBgLCA95qc1evvZsIBNXsfk2tH1khevoO737VjBjaRq1AQAAAAAAAAA=", + "cursor": "IB4FVYgg9V6CDa+eXirLoV3kwHlX7ANZ+jwGvF/CVCj2AQAAAAAAAAA=", "node": { "name": { "bcs": "KgAAAAAAAAA=" @@ -69,7 +69,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xc656ed2c790f5f9e2a5bab26c47613796636b6b54956fe9c92a6d6a191a277e9", + "id": "0x3938cfdf027a1be71937b133b59fb1dd29196ffeed24699c71ce25575a48cb5a", "count": "0" } } @@ -85,7 +85,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xc656ed2c790f5f9e2a5bab26c47613796636b6b54956fe9c92a6d6a191a277e9", + "id": "0x3938cfdf027a1be71937b133b59fb1dd29196ffeed24699c71ce25575a48cb5a", "count": "0" } } @@ -96,7 +96,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IBgLCA95qc1evvZsIBNXsfk2tH1khevoO737VjBjaRq1AQAAAAAAAAA=", + "cursor": "IB4FVYgg9V6CDa+eXirLoV3kwHlX7ANZ+jwGvF/CVCj2AQAAAAAAAAA=", "node": { "name": { "bcs": "KgAAAAAAAAA=" @@ -104,7 +104,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xc656ed2c790f5f9e2a5bab26c47613796636b6b54956fe9c92a6d6a191a277e9", + "id": "0x3938cfdf027a1be71937b133b59fb1dd29196ffeed24699c71ce25575a48cb5a", "count": "0" } } @@ -120,7 +120,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xc656ed2c790f5f9e2a5bab26c47613796636b6b54956fe9c92a6d6a191a277e9", + "id": "0x3938cfdf027a1be71937b133b59fb1dd29196ffeed24699c71ce25575a48cb5a", "count": "0" } } @@ -139,7 +139,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IBgLCA95qc1evvZsIBNXsfk2tH1khevoO737VjBjaRq1AQAAAAAAAAA=", + "cursor": "IB4FVYgg9V6CDa+eXirLoV3kwHlX7ANZ+jwGvF/CVCj2AQAAAAAAAAA=", "node": { "name": { "bcs": "KgAAAAAAAAA=" @@ -147,7 +147,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xc656ed2c790f5f9e2a5bab26c47613796636b6b54956fe9c92a6d6a191a277e9", + "id": "0x3938cfdf027a1be71937b133b59fb1dd29196ffeed24699c71ce25575a48cb5a", "count": "0" } } @@ -163,7 +163,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xc656ed2c790f5f9e2a5bab26c47613796636b6b54956fe9c92a6d6a191a277e9", + "id": "0x3938cfdf027a1be71937b133b59fb1dd29196ffeed24699c71ce25575a48cb5a", "count": "0" } } @@ -182,7 +182,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IBgLCA95qc1evvZsIBNXsfk2tH1khevoO737VjBjaRq1AQAAAAAAAAA=", + "cursor": "IB4FVYgg9V6CDa+eXirLoV3kwHlX7ANZ+jwGvF/CVCj2AQAAAAAAAAA=", "node": { "name": { "bcs": "KgAAAAAAAAA=" @@ -190,7 +190,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xc656ed2c790f5f9e2a5bab26c47613796636b6b54956fe9c92a6d6a191a277e9", + "id": "0x3938cfdf027a1be71937b133b59fb1dd29196ffeed24699c71ce25575a48cb5a", "count": "0" } } @@ -206,7 +206,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xc656ed2c790f5f9e2a5bab26c47613796636b6b54956fe9c92a6d6a191a277e9", + "id": "0x3938cfdf027a1be71937b133b59fb1dd29196ffeed24699c71ce25575a48cb5a", "count": "0" } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer_reclaim_add.move b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer_reclaim_add.move index e64864ade62..c4a87e21c46 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer_reclaim_add.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dof_add_reclaim_transfer_reclaim_add.move @@ -14,7 +14,7 @@ // 4 | 6 | reclaim child from another parent // 7 | 7 | add child to original parent -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dynamic_fields.exp b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dynamic_fields.exp index f553d2641e1..8ccdd3fec00 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dynamic_fields.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dynamic_fields.exp @@ -84,7 +84,7 @@ task 9, lines 103-165: Response: { "data": { "parent_version_2_no_dof": { - "address": "0x76e8be7a8a2c5e59b18ae4123b6c400a060e54acac33ebaef35c6359008a39b7", + "address": "0x8bd5b2580dad76f5c91b04a5c8abfa81dcaf7791e55adf97e758394f3a3ecd64", "dynamicFields": { "edges": [] } @@ -93,7 +93,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IMltckgisEII5yOBiLuvjsH2EULYkwu6lEaHTZDlMMLrAQAAAAAAAAA=", + "cursor": "INM02HGoE6QSb10FrovecYwBvBd26gG4dfgiIucXNRBsAQAAAAAAAAA=", "node": { "name": { "bcs": "pAEAAAAAAAA=", @@ -104,7 +104,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xd9e0a11bb56f6fa383c493fd8aa805417f09e50daf84af662f71b46effdc4c76", + "id": "0x5a2e05a43a7478d7d980b862b1cb771767f79d659c3b4b869d652d5d0ba0aae1", "count": "1" } } @@ -115,13 +115,13 @@ Response: { } }, "child_version_2_no_parent": { - "address": "0xd9e0a11bb56f6fa383c493fd8aa805417f09e50daf84af662f71b46effdc4c76", + "address": "0x5a2e05a43a7478d7d980b862b1cb771767f79d659c3b4b869d652d5d0ba0aae1", "owner": {} }, "child_version_3_has_parent": { "owner": { "parent": { - "address": "0xc96d724822b04208e7238188bbaf8ec1f61142d8930bba9446874d90e530c2eb" + "address": "0xd334d871a813a4126f5d05ae8bde718c01bc1776ea01b875f82222e71735106c" } } } @@ -173,63 +173,63 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IBHqo8yUZkJxCchyKB6tSCkMBIp3j3HtyLiOI5qQ3YCFAgAAAAAAAAA=", + "cursor": "IH5T1ST1RgLPlUZ+IdyKz+sLLiMj/akGfLZJEohOMLR6AgAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMg==", + "bcs": "A2RmMw==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df2" + "json": "df3" } } }, { - "cursor": "ICtSJBD2mvBMSbYw7lALjwqJIu1qj71gDeP4G0ZF2IdPAgAAAAAAAAA=", + "cursor": "INM02HGoE6QSb10FrovecYwBvBd26gG4dfgiIucXNRBsAgAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMQ==", + "bcs": "pAEAAAAAAAA=", "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" + "repr": "u64" } }, "value": { - "json": "df1" + "contents": { + "json": { + "id": "0x5a2e05a43a7478d7d980b862b1cb771767f79d659c3b4b869d652d5d0ba0aae1", + "count": "2" + } + } } } }, { - "cursor": "IMltckgisEII5yOBiLuvjsH2EULYkwu6lEaHTZDlMMLrAgAAAAAAAAA=", + "cursor": "IOIvp81rGEne+kFVHgmiVU8RU3CeKUDAUb7xOhPAwSlQAgAAAAAAAAA=", "node": { "name": { - "bcs": "pAEAAAAAAAA=", + "bcs": "A2RmMg==", "type": { - "repr": "u64" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "contents": { - "json": { - "id": "0xd9e0a11bb56f6fa383c493fd8aa805417f09e50daf84af662f71b46effdc4c76", - "count": "2" - } - } + "json": "df2" } } }, { - "cursor": "IP0sJsB7Ip8J9WsIbqXbgDAcIeih8L+/hTYxca+GYwLFAgAAAAAAAAA=", + "cursor": "IPGhGiz28/F9NiuEIwlViuWrRaNdiBxHKbVKaGY7mykIAgAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMw==", + "bcs": "A2RmMQ==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df3" + "json": "df1" } } } @@ -240,7 +240,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IMltckgisEII5yOBiLuvjsH2EULYkwu6lEaHTZDlMMLrAgAAAAAAAAA=", + "cursor": "INM02HGoE6QSb10FrovecYwBvBd26gG4dfgiIucXNRBsAgAAAAAAAAA=", "node": { "name": { "bcs": "pAEAAAAAAAA=", @@ -251,7 +251,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xd9e0a11bb56f6fa383c493fd8aa805417f09e50daf84af662f71b46effdc4c76", + "id": "0x5a2e05a43a7478d7d980b862b1cb771767f79d659c3b4b869d652d5d0ba0aae1", "count": "1" } } @@ -270,16 +270,30 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IP0sJsB7Ip8J9WsIbqXbgDAcIeih8L+/hTYxca+GYwLFAgAAAAAAAAA=", + "cursor": "IOIvp81rGEne+kFVHgmiVU8RU3CeKUDAUb7xOhPAwSlQAgAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMw==", + "bcs": "A2RmMg==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df3" + "json": "df2" + } + } + }, + { + "cursor": "IPGhGiz28/F9NiuEIwlViuWrRaNdiBxHKbVKaGY7mykIAgAAAAAAAAA=", + "node": { + "name": { + "bcs": "A2RmMQ==", + "type": { + "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" + } + }, + "value": { + "json": "df1" } } } @@ -314,7 +328,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xd9e0a11bb56f6fa383c493fd8aa805417f09e50daf84af662f71b46effdc4c76", + "id": "0x5a2e05a43a7478d7d980b862b1cb771767f79d659c3b4b869d652d5d0ba0aae1", "count": "1" } } @@ -333,7 +347,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xd9e0a11bb56f6fa383c493fd8aa805417f09e50daf84af662f71b46effdc4c76", + "id": "0x5a2e05a43a7478d7d980b862b1cb771767f79d659c3b4b869d652d5d0ba0aae1", "count": "2" } } @@ -398,35 +412,21 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IBHqo8yUZkJxCchyKB6tSCkMBIp3j3HtyLiOI5qQ3YCFAwAAAAAAAAA=", - "node": { - "name": { - "bcs": "A2RmMg==", - "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" - } - }, - "value": { - "json": "df2" - } - } - }, - { - "cursor": "ICtSJBD2mvBMSbYw7lALjwqJIu1qj71gDeP4G0ZF2IdPAwAAAAAAAAA=", + "cursor": "IH5T1ST1RgLPlUZ+IdyKz+sLLiMj/akGfLZJEohOMLR6AwAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMQ==", + "bcs": "A2RmMw==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df1" + "json": "df3" } } }, { - "cursor": "IMltckgisEII5yOBiLuvjsH2EULYkwu6lEaHTZDlMMLrAwAAAAAAAAA=", + "cursor": "INM02HGoE6QSb10FrovecYwBvBd26gG4dfgiIucXNRBsAwAAAAAAAAA=", "node": { "name": { "bcs": "pAEAAAAAAAA=", @@ -437,7 +437,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xd9e0a11bb56f6fa383c493fd8aa805417f09e50daf84af662f71b46effdc4c76", + "id": "0x5a2e05a43a7478d7d980b862b1cb771767f79d659c3b4b869d652d5d0ba0aae1", "count": "2" } } @@ -445,47 +445,41 @@ Response: { } }, { - "cursor": "IP0sJsB7Ip8J9WsIbqXbgDAcIeih8L+/hTYxca+GYwLFAwAAAAAAAAA=", + "cursor": "IOIvp81rGEne+kFVHgmiVU8RU3CeKUDAUb7xOhPAwSlQAwAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMw==", + "bcs": "A2RmMg==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df3" + "json": "df2" } } - } - ] - } - }, - "parent_version_4_paginated_on_dof_consistent": { - "dynamicFields": { - "edges": [ + }, { - "cursor": "IP0sJsB7Ip8J9WsIbqXbgDAcIeih8L+/hTYxca+GYwLFAgAAAAAAAAA=", + "cursor": "IPGhGiz28/F9NiuEIwlViuWrRaNdiBxHKbVKaGY7mykIAwAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMw==", + "bcs": "A2RmMQ==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df3" + "json": "df1" } } } ] } }, - "parent_version_5_has_7_children": { + "parent_version_4_paginated_on_dof_consistent": { "dynamicFields": { "edges": [ { - "cursor": "IBHqo8yUZkJxCchyKB6tSCkMBIp3j3HtyLiOI5qQ3YCFAwAAAAAAAAA=", + "cursor": "IOIvp81rGEne+kFVHgmiVU8RU3CeKUDAUb7xOhPAwSlQAgAAAAAAAAA=", "node": { "name": { "bcs": "A2RmMg==", @@ -499,7 +493,7 @@ Response: { } }, { - "cursor": "ICtSJBD2mvBMSbYw7lALjwqJIu1qj71gDeP4G0ZF2IdPAwAAAAAAAAA=", + "cursor": "IPGhGiz28/F9NiuEIwlViuWrRaNdiBxHKbVKaGY7mykIAgAAAAAAAAA=", "node": { "name": { "bcs": "A2RmMQ==", @@ -511,9 +505,15 @@ Response: { "json": "df1" } } - }, + } + ] + } + }, + "parent_version_5_has_7_children": { + "dynamicFields": { + "edges": [ { - "cursor": "IE+66B4qhmJHvgLRXCzBkJcKtIq9/3t9zYS98b6bMLhDAwAAAAAAAAA=", + "cursor": "IDhNv3675G3XXlexv/qRsZ+/KmiXpzE6611Dg+P5PCpCAwAAAAAAAAA=", "node": { "name": { "bcs": "A2RmNg==", @@ -527,21 +527,35 @@ Response: { } }, { - "cursor": "IFqQu0yJoDJda7AeIZmvhIW3x34EsmfTInmLo9ZXTVExAwAAAAAAAAA=", + "cursor": "IH5T1ST1RgLPlUZ+IdyKz+sLLiMj/akGfLZJEohOMLR6AwAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNQ==", + "bcs": "A2RmMw==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df5" + "json": "df3" } } }, { - "cursor": "IMltckgisEII5yOBiLuvjsH2EULYkwu6lEaHTZDlMMLrAwAAAAAAAAA=", + "cursor": "IJffeetDkRORzdANh97AfytTmdNRS7khUerDMiSvg5OMAwAAAAAAAAA=", + "node": { + "name": { + "bcs": "A2RmNA==", + "type": { + "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" + } + }, + "value": { + "json": "df4" + } + } + }, + { + "cursor": "INM02HGoE6QSb10FrovecYwBvBd26gG4dfgiIucXNRBsAwAAAAAAAAA=", "node": { "name": { "bcs": "pAEAAAAAAAA=", @@ -552,7 +566,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xd9e0a11bb56f6fa383c493fd8aa805417f09e50daf84af662f71b46effdc4c76", + "id": "0x5a2e05a43a7478d7d980b862b1cb771767f79d659c3b4b869d652d5d0ba0aae1", "count": "2" } } @@ -560,30 +574,44 @@ Response: { } }, { - "cursor": "INLCyu+uz92dAqQqrMubrTVBken7AGvouhvfBABDIiOuAwAAAAAAAAA=", + "cursor": "IOIvp81rGEne+kFVHgmiVU8RU3CeKUDAUb7xOhPAwSlQAwAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNA==", + "bcs": "A2RmMg==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df4" + "json": "df2" } } }, { - "cursor": "IP0sJsB7Ip8J9WsIbqXbgDAcIeih8L+/hTYxca+GYwLFAwAAAAAAAAA=", + "cursor": "IOstAacpirk81QX749dCGqXUsLsT0eq1llfo5vUncdfQAwAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMw==", + "bcs": "A2RmNQ==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df3" + "json": "df5" + } + } + }, + { + "cursor": "IPGhGiz28/F9NiuEIwlViuWrRaNdiBxHKbVKaGY7mykIAwAAAAAAAAA=", + "node": { + "name": { + "bcs": "A2RmMQ==", + "type": { + "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" + } + }, + "value": { + "json": "df1" } } } @@ -594,30 +622,44 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "INLCyu+uz92dAqQqrMubrTVBken7AGvouhvfBABDIiOuAwAAAAAAAAA=", + "cursor": "IOIvp81rGEne+kFVHgmiVU8RU3CeKUDAUb7xOhPAwSlQAwAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNA==", + "bcs": "A2RmMg==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df4" + "json": "df2" } } }, { - "cursor": "IP0sJsB7Ip8J9WsIbqXbgDAcIeih8L+/hTYxca+GYwLFAwAAAAAAAAA=", + "cursor": "IOstAacpirk81QX749dCGqXUsLsT0eq1llfo5vUncdfQAwAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMw==", + "bcs": "A2RmNQ==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df3" + "json": "df5" + } + } + }, + { + "cursor": "IPGhGiz28/F9NiuEIwlViuWrRaNdiBxHKbVKaGY7mykIAwAAAAAAAAA=", + "node": { + "name": { + "bcs": "A2RmMQ==", + "type": { + "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" + } + }, + "value": { + "json": "df1" } } } @@ -671,63 +713,63 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IBHqo8yUZkJxCchyKB6tSCkMBIp3j3HtyLiOI5qQ3YCFBAAAAAAAAAA=", + "cursor": "IH5T1ST1RgLPlUZ+IdyKz+sLLiMj/akGfLZJEohOMLR6BAAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMg==", + "bcs": "A2RmMw==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df2" + "json": "df3" } } }, { - "cursor": "ICtSJBD2mvBMSbYw7lALjwqJIu1qj71gDeP4G0ZF2IdPBAAAAAAAAAA=", + "cursor": "INM02HGoE6QSb10FrovecYwBvBd26gG4dfgiIucXNRBsBAAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMQ==", + "bcs": "pAEAAAAAAAA=", "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" + "repr": "u64" } }, "value": { - "json": "df1" + "contents": { + "json": { + "id": "0x5a2e05a43a7478d7d980b862b1cb771767f79d659c3b4b869d652d5d0ba0aae1", + "count": "2" + } + } } } }, { - "cursor": "IMltckgisEII5yOBiLuvjsH2EULYkwu6lEaHTZDlMMLrBAAAAAAAAAA=", + "cursor": "IOIvp81rGEne+kFVHgmiVU8RU3CeKUDAUb7xOhPAwSlQBAAAAAAAAAA=", "node": { "name": { - "bcs": "pAEAAAAAAAA=", + "bcs": "A2RmMg==", "type": { - "repr": "u64" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "contents": { - "json": { - "id": "0xd9e0a11bb56f6fa383c493fd8aa805417f09e50daf84af662f71b46effdc4c76", - "count": "2" - } - } + "json": "df2" } } }, { - "cursor": "IP0sJsB7Ip8J9WsIbqXbgDAcIeih8L+/hTYxca+GYwLFBAAAAAAAAAA=", + "cursor": "IPGhGiz28/F9NiuEIwlViuWrRaNdiBxHKbVKaGY7mykIBAAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMw==", + "bcs": "A2RmMQ==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df3" + "json": "df1" } } } @@ -738,16 +780,30 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IP0sJsB7Ip8J9WsIbqXbgDAcIeih8L+/hTYxca+GYwLFAgAAAAAAAAA=", + "cursor": "IOIvp81rGEne+kFVHgmiVU8RU3CeKUDAUb7xOhPAwSlQAgAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMw==", + "bcs": "A2RmMg==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df3" + "json": "df2" + } + } + }, + { + "cursor": "IPGhGiz28/F9NiuEIwlViuWrRaNdiBxHKbVKaGY7mykIAgAAAAAAAAA=", + "node": { + "name": { + "bcs": "A2RmMQ==", + "type": { + "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" + } + }, + "value": { + "json": "df1" } } } @@ -758,7 +814,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IE+66B4qhmJHvgLRXCzBkJcKtIq9/3t9zYS98b6bMLhDBAAAAAAAAAA=", + "cursor": "IDhNv3675G3XXlexv/qRsZ+/KmiXpzE6611Dg+P5PCpCBAAAAAAAAAA=", "node": { "name": { "bcs": "A2RmNg==", @@ -772,21 +828,21 @@ Response: { } }, { - "cursor": "IFqQu0yJoDJda7AeIZmvhIW3x34EsmfTInmLo9ZXTVExBAAAAAAAAAA=", + "cursor": "IJffeetDkRORzdANh97AfytTmdNRS7khUerDMiSvg5OMBAAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNQ==", + "bcs": "A2RmNA==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df5" + "json": "df4" } } }, { - "cursor": "IMltckgisEII5yOBiLuvjsH2EULYkwu6lEaHTZDlMMLrBAAAAAAAAAA=", + "cursor": "INM02HGoE6QSb10FrovecYwBvBd26gG4dfgiIucXNRBsBAAAAAAAAAA=", "node": { "name": { "bcs": "pAEAAAAAAAA=", @@ -797,7 +853,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xd9e0a11bb56f6fa383c493fd8aa805417f09e50daf84af662f71b46effdc4c76", + "id": "0x5a2e05a43a7478d7d980b862b1cb771767f79d659c3b4b869d652d5d0ba0aae1", "count": "2" } } @@ -805,16 +861,16 @@ Response: { } }, { - "cursor": "INLCyu+uz92dAqQqrMubrTVBken7AGvouhvfBABDIiOuBAAAAAAAAAA=", + "cursor": "IOstAacpirk81QX749dCGqXUsLsT0eq1llfo5vUncdfQBAAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNA==", + "bcs": "A2RmNQ==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df4" + "json": "df5" } } } @@ -825,16 +881,16 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "INLCyu+uz92dAqQqrMubrTVBken7AGvouhvfBABDIiOuBAAAAAAAAAA=", + "cursor": "IOstAacpirk81QX749dCGqXUsLsT0eq1llfo5vUncdfQBAAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNA==", + "bcs": "A2RmNQ==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df4" + "json": "df5" } } } @@ -891,7 +947,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IMltckgisEII5yOBiLuvjsH2EULYkwu6lEaHTZDlMMLrBwAAAAAAAAA=", + "cursor": "INM02HGoE6QSb10FrovecYwBvBd26gG4dfgiIucXNRBsBwAAAAAAAAA=", "node": { "name": { "bcs": "pAEAAAAAAAA=", @@ -902,7 +958,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xd9e0a11bb56f6fa383c493fd8aa805417f09e50daf84af662f71b46effdc4c76", + "id": "0x5a2e05a43a7478d7d980b862b1cb771767f79d659c3b4b869d652d5d0ba0aae1", "count": "2" } } @@ -917,7 +973,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IE+66B4qhmJHvgLRXCzBkJcKtIq9/3t9zYS98b6bMLhDBwAAAAAAAAA=", + "cursor": "IDhNv3675G3XXlexv/qRsZ+/KmiXpzE6611Dg+P5PCpCBwAAAAAAAAA=", "node": { "name": { "bcs": "A2RmNg==", @@ -931,21 +987,21 @@ Response: { } }, { - "cursor": "IFqQu0yJoDJda7AeIZmvhIW3x34EsmfTInmLo9ZXTVExBwAAAAAAAAA=", + "cursor": "IJffeetDkRORzdANh97AfytTmdNRS7khUerDMiSvg5OMBwAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNQ==", + "bcs": "A2RmNA==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df5" + "json": "df4" } } }, { - "cursor": "IMltckgisEII5yOBiLuvjsH2EULYkwu6lEaHTZDlMMLrBwAAAAAAAAA=", + "cursor": "INM02HGoE6QSb10FrovecYwBvBd26gG4dfgiIucXNRBsBwAAAAAAAAA=", "node": { "name": { "bcs": "pAEAAAAAAAA=", @@ -956,7 +1012,7 @@ Response: { "value": { "contents": { "json": { - "id": "0xd9e0a11bb56f6fa383c493fd8aa805417f09e50daf84af662f71b46effdc4c76", + "id": "0x5a2e05a43a7478d7d980b862b1cb771767f79d659c3b4b869d652d5d0ba0aae1", "count": "2" } } @@ -964,16 +1020,16 @@ Response: { } }, { - "cursor": "INLCyu+uz92dAqQqrMubrTVBken7AGvouhvfBABDIiOuBwAAAAAAAAA=", + "cursor": "IOstAacpirk81QX749dCGqXUsLsT0eq1llfo5vUncdfQBwAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNA==", + "bcs": "A2RmNQ==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df4" + "json": "df5" } } } @@ -984,16 +1040,16 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "INLCyu+uz92dAqQqrMubrTVBken7AGvouhvfBABDIiOuBAAAAAAAAAA=", + "cursor": "IOstAacpirk81QX749dCGqXUsLsT0eq1llfo5vUncdfQBAAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmNA==", + "bcs": "A2RmNQ==", "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000001::string::String" } }, "value": { - "json": "df4" + "json": "df5" } } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dynamic_fields.move b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dynamic_fields.move index fece3d3b265..4ada17a2ab1 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dynamic_fields.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/dynamic_fields.move @@ -8,7 +8,7 @@ // chkpt3: add df4, 5, 6 parent @ version 5, child @ version 4 // chkpt4: remove df1, df2, df3 parent @ version 6, child @ version 4 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/immutable_dof.exp b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/immutable_dof.exp index 67c09b48c94..13cd9403ee7 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/immutable_dof.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/immutable_dof.exp @@ -58,11 +58,11 @@ Response: { "nodes": [ { "value": { - "address": "0xbbc2cee5f76e38917236d2524b3ef426edbbc25ef9ffbd6e9778dcffafe96f96", + "address": "0x103f1f568d076c169d121e62d70acd1898c08efea8d0e6506bdfe27a75174702", "version": 5, "contents": { "json": { - "id": "0xbbc2cee5f76e38917236d2524b3ef426edbbc25ef9ffbd6e9778dcffafe96f96", + "id": "0x103f1f568d076c169d121e62d70acd1898c08efea8d0e6506bdfe27a75174702", "count": "0" } }, @@ -86,11 +86,11 @@ Response: { "nodes": [ { "value": { - "address": "0xbbc2cee5f76e38917236d2524b3ef426edbbc25ef9ffbd6e9778dcffafe96f96", + "address": "0x103f1f568d076c169d121e62d70acd1898c08efea8d0e6506bdfe27a75174702", "version": 5, "contents": { "json": { - "id": "0xbbc2cee5f76e38917236d2524b3ef426edbbc25ef9ffbd6e9778dcffafe96f96", + "id": "0x103f1f568d076c169d121e62d70acd1898c08efea8d0e6506bdfe27a75174702", "count": "0" } }, @@ -98,11 +98,11 @@ Response: { "nodes": [ { "value": { - "address": "0x1981177a65ae90d094e5512584c97af55b294aef0b77794b1c6757b6ce2a5897", + "address": "0x9aca42c68d82ee924a8fdf7ccd813cc1e40d723b69e0c1c1b7e8dd04c6c3e0b5", "version": 6, "contents": { "json": { - "id": "0x1981177a65ae90d094e5512584c97af55b294aef0b77794b1c6757b6ce2a5897", + "id": "0x9aca42c68d82ee924a8fdf7ccd813cc1e40d723b69e0c1c1b7e8dd04c6c3e0b5", "count": "0" } } @@ -145,7 +145,7 @@ Response: { "object": { "owner": { "parent": { - "address": "0x261b91955dcf2f276c60d8f04a8c5bfcad92a423111b88925f9df1300715e318" + "address": "0x6c9c3e5963618ad872d20456745f8052d8cc2b06ff312bc41755be4c31c0d85a" } }, "dynamicFields": { @@ -175,11 +175,11 @@ Response: { "nodes": [ { "value": { - "address": "0x1981177a65ae90d094e5512584c97af55b294aef0b77794b1c6757b6ce2a5897", + "address": "0x9aca42c68d82ee924a8fdf7ccd813cc1e40d723b69e0c1c1b7e8dd04c6c3e0b5", "version": 6, "contents": { "json": { - "id": "0x1981177a65ae90d094e5512584c97af55b294aef0b77794b1c6757b6ce2a5897", + "id": "0x9aca42c68d82ee924a8fdf7ccd813cc1e40d723b69e0c1c1b7e8dd04c6c3e0b5", "count": "0" } } @@ -203,11 +203,11 @@ Response: { "nodes": [ { "value": { - "address": "0x1981177a65ae90d094e5512584c97af55b294aef0b77794b1c6757b6ce2a5897", + "address": "0x9aca42c68d82ee924a8fdf7ccd813cc1e40d723b69e0c1c1b7e8dd04c6c3e0b5", "version": 6, "contents": { "json": { - "id": "0x1981177a65ae90d094e5512584c97af55b294aef0b77794b1c6757b6ce2a5897", + "id": "0x9aca42c68d82ee924a8fdf7ccd813cc1e40d723b69e0c1c1b7e8dd04c6c3e0b5", "count": "0" } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/immutable_dof.move b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/immutable_dof.move index 54a7d4a289a..d5ccc2ee8ff 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/immutable_dof.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/immutable_dof.move @@ -16,7 +16,7 @@ // Verify that Parent1 (6) -> Child1 (5) -> Child2 (6) -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_df.exp b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_df.exp index 6d506e4c4b5..6db279e5373 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_df.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_df.exp @@ -78,18 +78,18 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IAbc6C6Kc64uFzGgwo7ZbNDAvqJCXM7FCORnt+FWoN4yAQAAAAAAAAA=", + "cursor": "IGfo1G9O2vQ5AUUfGCaMCW0qCfdIbThO1lc/YW2ZVNDHAQAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMQ==" + "bcs": "A2RmMw==" }, "value": { - "json": "df1" + "json": "df3" } } }, { - "cursor": "IFTGRZmDK4S0lztr3Y3wG1eww8SpX12G7JGr845f5P5sAQAAAAAAAAA=", + "cursor": "IHZ5crzZF1ky2W07ZM4hIIOlfv+idVFTxPgn4ropElLSAQAAAAAAAAA=", "node": { "name": { "bcs": "A2RmMg==" @@ -100,13 +100,13 @@ Response: { } }, { - "cursor": "IMknIBul3WNc9Xq/FWOan+uMZOaR67vleBT738wHRB1dAQAAAAAAAAA=", + "cursor": "IOP6Q1MMTGdwdYPL8ADKAldD+ZlqRKZmTH7qM3KP8ozIAQAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMw==" + "bcs": "A2RmMQ==" }, "value": { - "json": "df3" + "json": "df1" } } } @@ -154,7 +154,7 @@ Contents: Test::M1::Parent { task 11, line 114: //# run Test::M1::mutate_df1 --sender A --args object(2,0) -mutated: object(0,0), object(2,0), object(4,0) +mutated: object(0,0), object(2,0), object(4,2) gas summary: computation_cost: 1000000, computation_cost_burned: 1000000, storage_cost: 4461200, storage_rebate: 4400400, non_refundable_storage_fee: 0 task 12, line 116: @@ -201,18 +201,18 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IAbc6C6Kc64uFzGgwo7ZbNDAvqJCXM7FCORnt+FWoN4yAgAAAAAAAAA=", + "cursor": "IGfo1G9O2vQ5AUUfGCaMCW0qCfdIbThO1lc/YW2ZVNDHAgAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMQ==" + "bcs": "A2RmMw==" }, "value": { - "json": "df1_mutated" + "json": "df3" } } }, { - "cursor": "IFTGRZmDK4S0lztr3Y3wG1eww8SpX12G7JGr845f5P5sAgAAAAAAAAA=", + "cursor": "IHZ5crzZF1ky2W07ZM4hIIOlfv+idVFTxPgn4ropElLSAgAAAAAAAAA=", "node": { "name": { "bcs": "A2RmMg==" @@ -223,13 +223,13 @@ Response: { } }, { - "cursor": "IMknIBul3WNc9Xq/FWOan+uMZOaR67vleBT738wHRB1dAgAAAAAAAAA=", + "cursor": "IOP6Q1MMTGdwdYPL8ADKAldD+ZlqRKZmTH7qM3KP8ozIAgAAAAAAAAA=", "node": { "name": { - "bcs": "A2RmMw==" + "bcs": "A2RmMQ==" }, "value": { - "json": "df3" + "json": "df1_mutated" } } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_df.move b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_df.move index d593c4151f4..e729eb10a6e 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_df.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_df.move @@ -10,7 +10,7 @@ // 5 | mutated df1 // 6 | mutated parent again -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_dof.exp b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_dof.exp index 812f886b276..c939c4f05a0 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_dof.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_dof.exp @@ -41,7 +41,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IE3qhyyw5Ukl1PAavIndl4WVL8BugBR44AsX6MD2XkhfAQAAAAAAAAA=", + "cursor": "ILQtpw3tzVt6lpL15VIqBp3YsW2t1MxgQsR5fGSghVHUAQAAAAAAAAA=", "node": { "name": { "bcs": "KgAAAAAAAAA=" @@ -49,7 +49,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x3df577f7f845c5a18148aac68a2dfc09c674ea1f848c6153ed08a96c46dc53e4", + "id": "0x12cfc028b1cf6c5827a1596b43c6b6e1e2f3695d5a17570ae973916ad5e6f14b", "count": "0" } } @@ -65,7 +65,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x3df577f7f845c5a18148aac68a2dfc09c674ea1f848c6153ed08a96c46dc53e4", + "id": "0x12cfc028b1cf6c5827a1596b43c6b6e1e2f3695d5a17570ae973916ad5e6f14b", "count": "0" } } @@ -77,7 +77,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IE3qhyyw5Ukl1PAavIndl4WVL8BugBR44AsX6MD2XkhfAQAAAAAAAAA=", + "cursor": "ILQtpw3tzVt6lpL15VIqBp3YsW2t1MxgQsR5fGSghVHUAQAAAAAAAAA=", "node": { "name": { "bcs": "KgAAAAAAAAA=" @@ -85,7 +85,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x3df577f7f845c5a18148aac68a2dfc09c674ea1f848c6153ed08a96c46dc53e4", + "id": "0x12cfc028b1cf6c5827a1596b43c6b6e1e2f3695d5a17570ae973916ad5e6f14b", "count": "0" } } @@ -101,7 +101,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x3df577f7f845c5a18148aac68a2dfc09c674ea1f848c6153ed08a96c46dc53e4", + "id": "0x12cfc028b1cf6c5827a1596b43c6b6e1e2f3695d5a17570ae973916ad5e6f14b", "count": "0" } } @@ -117,7 +117,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x3df577f7f845c5a18148aac68a2dfc09c674ea1f848c6153ed08a96c46dc53e4", + "id": "0x12cfc028b1cf6c5827a1596b43c6b6e1e2f3695d5a17570ae973916ad5e6f14b", "count": "0" } } @@ -168,7 +168,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x3df577f7f845c5a18148aac68a2dfc09c674ea1f848c6153ed08a96c46dc53e4", + "id": "0x12cfc028b1cf6c5827a1596b43c6b6e1e2f3695d5a17570ae973916ad5e6f14b", "count": "0" } } @@ -202,7 +202,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IE3qhyyw5Ukl1PAavIndl4WVL8BugBR44AsX6MD2XkhfAwAAAAAAAAA=", + "cursor": "ILQtpw3tzVt6lpL15VIqBp3YsW2t1MxgQsR5fGSghVHUAwAAAAAAAAA=", "node": { "name": { "bcs": "KgAAAAAAAAA=" @@ -210,7 +210,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x3df577f7f845c5a18148aac68a2dfc09c674ea1f848c6153ed08a96c46dc53e4", + "id": "0x12cfc028b1cf6c5827a1596b43c6b6e1e2f3695d5a17570ae973916ad5e6f14b", "count": "1" } } @@ -226,7 +226,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x3df577f7f845c5a18148aac68a2dfc09c674ea1f848c6153ed08a96c46dc53e4", + "id": "0x12cfc028b1cf6c5827a1596b43c6b6e1e2f3695d5a17570ae973916ad5e6f14b", "count": "1" } } @@ -238,7 +238,7 @@ Response: { "dynamicFields": { "edges": [ { - "cursor": "IE3qhyyw5Ukl1PAavIndl4WVL8BugBR44AsX6MD2XkhfAwAAAAAAAAA=", + "cursor": "ILQtpw3tzVt6lpL15VIqBp3YsW2t1MxgQsR5fGSghVHUAwAAAAAAAAA=", "node": { "name": { "bcs": "KgAAAAAAAAA=" @@ -246,7 +246,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x3df577f7f845c5a18148aac68a2dfc09c674ea1f848c6153ed08a96c46dc53e4", + "id": "0x12cfc028b1cf6c5827a1596b43c6b6e1e2f3695d5a17570ae973916ad5e6f14b", "count": "1" } } @@ -262,7 +262,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x3df577f7f845c5a18148aac68a2dfc09c674ea1f848c6153ed08a96c46dc53e4", + "id": "0x12cfc028b1cf6c5827a1596b43c6b6e1e2f3695d5a17570ae973916ad5e6f14b", "count": "1" } } @@ -283,7 +283,7 @@ Response: { "value": { "contents": { "json": { - "id": "0x3df577f7f845c5a18148aac68a2dfc09c674ea1f848c6153ed08a96c46dc53e4", + "id": "0x12cfc028b1cf6c5827a1596b43c6b6e1e2f3695d5a17570ae973916ad5e6f14b", "count": "0" } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_dof.move b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_dof.move index d314ae5a709..be67c5402f3 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_dof.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/mutated_dof.move @@ -11,7 +11,7 @@ // 5 | 6 | mutated child // 7 | 7 | add child back to parent -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/nested_dof.exp b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/nested_dof.exp index 8f0b5aedb4b..8c5ccbdb0ec 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/nested_dof.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/nested_dof.exp @@ -62,11 +62,11 @@ Response: { "nodes": [ { "value": { - "address": "0xc1e75c3f54d6c6bae388d42640d529f080b34becbaabc5c920293f6ab3b8fbfb", + "address": "0x5a9ff7a0b7bbe7fcaa815c5ae574cee72a67636a9672b4334a40bb65ccceac28", "version": 5, "contents": { "json": { - "id": "0xc1e75c3f54d6c6bae388d42640d529f080b34becbaabc5c920293f6ab3b8fbfb", + "id": "0x5a9ff7a0b7bbe7fcaa815c5ae574cee72a67636a9672b4334a40bb65ccceac28", "count": "0" } }, @@ -90,11 +90,11 @@ Response: { "nodes": [ { "value": { - "address": "0xc1e75c3f54d6c6bae388d42640d529f080b34becbaabc5c920293f6ab3b8fbfb", + "address": "0x5a9ff7a0b7bbe7fcaa815c5ae574cee72a67636a9672b4334a40bb65ccceac28", "version": 5, "contents": { "json": { - "id": "0xc1e75c3f54d6c6bae388d42640d529f080b34becbaabc5c920293f6ab3b8fbfb", + "id": "0x5a9ff7a0b7bbe7fcaa815c5ae574cee72a67636a9672b4334a40bb65ccceac28", "count": "0" } }, @@ -102,11 +102,11 @@ Response: { "nodes": [ { "value": { - "address": "0x7bc6cd7c621f45de49d28adf67705526d7455422bcdbf8111a7eb80a2a9a7c67", + "address": "0xc87c76c91ed902d02e6de92c4c5a7aea80f2c1123361775a1615e102e2f37d85", "version": 6, "contents": { "json": { - "id": "0x7bc6cd7c621f45de49d28adf67705526d7455422bcdbf8111a7eb80a2a9a7c67", + "id": "0xc87c76c91ed902d02e6de92c4c5a7aea80f2c1123361775a1615e102e2f37d85", "count": "0" } } @@ -131,11 +131,11 @@ Response: { "nodes": [ { "value": { - "address": "0xc1e75c3f54d6c6bae388d42640d529f080b34becbaabc5c920293f6ab3b8fbfb", + "address": "0x5a9ff7a0b7bbe7fcaa815c5ae574cee72a67636a9672b4334a40bb65ccceac28", "version": 7, "contents": { "json": { - "id": "0xc1e75c3f54d6c6bae388d42640d529f080b34becbaabc5c920293f6ab3b8fbfb", + "id": "0x5a9ff7a0b7bbe7fcaa815c5ae574cee72a67636a9672b4334a40bb65ccceac28", "count": "1" } }, @@ -143,11 +143,11 @@ Response: { "nodes": [ { "value": { - "address": "0x7bc6cd7c621f45de49d28adf67705526d7455422bcdbf8111a7eb80a2a9a7c67", + "address": "0xc87c76c91ed902d02e6de92c4c5a7aea80f2c1123361775a1615e102e2f37d85", "version": 6, "contents": { "json": { - "id": "0x7bc6cd7c621f45de49d28adf67705526d7455422bcdbf8111a7eb80a2a9a7c67", + "id": "0xc87c76c91ed902d02e6de92c4c5a7aea80f2c1123361775a1615e102e2f37d85", "count": "0" } } @@ -172,11 +172,11 @@ Response: { "nodes": [ { "value": { - "address": "0xc1e75c3f54d6c6bae388d42640d529f080b34becbaabc5c920293f6ab3b8fbfb", + "address": "0x5a9ff7a0b7bbe7fcaa815c5ae574cee72a67636a9672b4334a40bb65ccceac28", "version": 7, "contents": { "json": { - "id": "0xc1e75c3f54d6c6bae388d42640d529f080b34becbaabc5c920293f6ab3b8fbfb", + "id": "0x5a9ff7a0b7bbe7fcaa815c5ae574cee72a67636a9672b4334a40bb65ccceac28", "count": "1" } }, @@ -184,11 +184,11 @@ Response: { "nodes": [ { "value": { - "address": "0x7bc6cd7c621f45de49d28adf67705526d7455422bcdbf8111a7eb80a2a9a7c67", + "address": "0xc87c76c91ed902d02e6de92c4c5a7aea80f2c1123361775a1615e102e2f37d85", "version": 8, "contents": { "json": { - "id": "0x7bc6cd7c621f45de49d28adf67705526d7455422bcdbf8111a7eb80a2a9a7c67", + "id": "0xc87c76c91ed902d02e6de92c4c5a7aea80f2c1123361775a1615e102e2f37d85", "count": "1" } } @@ -233,11 +233,11 @@ Response: { "nodes": [ { "value": { - "address": "0x7bc6cd7c621f45de49d28adf67705526d7455422bcdbf8111a7eb80a2a9a7c67", + "address": "0xc87c76c91ed902d02e6de92c4c5a7aea80f2c1123361775a1615e102e2f37d85", "version": 6, "contents": { "json": { - "id": "0x7bc6cd7c621f45de49d28adf67705526d7455422bcdbf8111a7eb80a2a9a7c67", + "id": "0xc87c76c91ed902d02e6de92c4c5a7aea80f2c1123361775a1615e102e2f37d85", "count": "0" } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/nested_dof.move b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/nested_dof.move index af604fe07ae..3625aaa457d 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/nested_dof.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/dynamic_fields/nested_dof.move @@ -20,7 +20,7 @@ // Child1 (5) -> None // Child1 (7) -> Child2 (6) -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/epochs/checkpoints.move b/crates/iota-graphql-e2e-tests/tests/consistency/epochs/checkpoints.move index d83379e9822..40d316a172b 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/epochs/checkpoints.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/epochs/checkpoints.move @@ -9,7 +9,7 @@ // 2 | 2 // An additional checkpoint is created at the end. -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# create-checkpoint diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/epochs/transaction_blocks.exp b/crates/iota-graphql-e2e-tests/tests/consistency/epochs/transaction_blocks.exp index b8ef5a023ea..6fd59e48abd 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/epochs/transaction_blocks.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/epochs/transaction_blocks.exp @@ -41,25 +41,25 @@ Response: { { "cursor": "eyJjIjozLCJ0IjowLCJpIjpmYWxzZX0", "node": { - "digest": "CmPeQ2JKFnEKeZsYz9cYU8q57Wb1fgZxpDqYq9YinuQA" + "digest": "5XJr4MXW36JcHWpqf2PF2oAD2cfWg4bk1jQWVp6Josvj" } }, { "cursor": "eyJjIjozLCJ0IjoxLCJpIjpmYWxzZX0", "node": { - "digest": "Hm38QqFQFCkzedsHCXsxJt7j5VCo9KUn5ZL9zSAZVKWh" + "digest": "DU5Lhorja2vD4cPn5Eqv6HvKA1ZY8mfGC3SHhGccQ9qF" } }, { "cursor": "eyJjIjozLCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "ywf9ak8rV9kDDx73zGH5vtTJkHmfxXHvLbSGhwAyxdM" + "digest": "7BPdUc3LpRmchBjPiSZ5Uju5yDZMbVhTQNYjDjmTXP5D" } }, { "cursor": "eyJjIjozLCJ0IjozLCJpIjpmYWxzZX0", "node": { - "digest": "Fsv6DwsUEPUqRyFP7wkd7mMDq93PXMmSXqQAgrrnsKb3" + "digest": "6ox6bgoqFRZbKftfQPM29MNpo876eW6PdsHxaNLEYiDj" } } ] @@ -154,25 +154,25 @@ Response: { { "cursor": "eyJjIjoxMiwidCI6MCwiaSI6ZmFsc2V9", "node": { - "digest": "CmPeQ2JKFnEKeZsYz9cYU8q57Wb1fgZxpDqYq9YinuQA" + "digest": "5XJr4MXW36JcHWpqf2PF2oAD2cfWg4bk1jQWVp6Josvj" } }, { "cursor": "eyJjIjoxMiwidCI6MSwiaSI6ZmFsc2V9", "node": { - "digest": "Hm38QqFQFCkzedsHCXsxJt7j5VCo9KUn5ZL9zSAZVKWh" + "digest": "DU5Lhorja2vD4cPn5Eqv6HvKA1ZY8mfGC3SHhGccQ9qF" } }, { "cursor": "eyJjIjoxMiwidCI6MiwiaSI6ZmFsc2V9", "node": { - "digest": "ywf9ak8rV9kDDx73zGH5vtTJkHmfxXHvLbSGhwAyxdM" + "digest": "7BPdUc3LpRmchBjPiSZ5Uju5yDZMbVhTQNYjDjmTXP5D" } }, { "cursor": "eyJjIjoxMiwidCI6MywiaSI6ZmFsc2V9", "node": { - "digest": "Fsv6DwsUEPUqRyFP7wkd7mMDq93PXMmSXqQAgrrnsKb3" + "digest": "6ox6bgoqFRZbKftfQPM29MNpo876eW6PdsHxaNLEYiDj" } } ] @@ -183,19 +183,19 @@ Response: { { "cursor": "eyJjIjo0LCJ0IjowLCJpIjpmYWxzZX0", "node": { - "digest": "CmPeQ2JKFnEKeZsYz9cYU8q57Wb1fgZxpDqYq9YinuQA" + "digest": "5XJr4MXW36JcHWpqf2PF2oAD2cfWg4bk1jQWVp6Josvj" } }, { "cursor": "eyJjIjo0LCJ0IjoxLCJpIjpmYWxzZX0", "node": { - "digest": "Hm38QqFQFCkzedsHCXsxJt7j5VCo9KUn5ZL9zSAZVKWh" + "digest": "DU5Lhorja2vD4cPn5Eqv6HvKA1ZY8mfGC3SHhGccQ9qF" } }, { "cursor": "eyJjIjo0LCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "ywf9ak8rV9kDDx73zGH5vtTJkHmfxXHvLbSGhwAyxdM" + "digest": "7BPdUc3LpRmchBjPiSZ5Uju5yDZMbVhTQNYjDjmTXP5D" } } ] @@ -207,25 +207,25 @@ Response: { { "cursor": "eyJjIjoxMiwidCI6NCwiaSI6ZmFsc2V9", "node": { - "digest": "25MhkStEmB44afcLcmtdA426Qbo6a5z6sJrURSfn4wSM" + "digest": "3s48BKV1NGNPV5VRKFsAJKWiyUiWDLVxguSERpo8qRTV" } }, { "cursor": "eyJjIjoxMiwidCI6NSwiaSI6ZmFsc2V9", "node": { - "digest": "Hutiw6GLNGCuRwMTZhxFW2CrYF6PYt2DomNMwtN1hLUo" + "digest": "6adEehTM2gxGKdumJXeKFwcaYhSB6RwGHbqg7Fv7JBmk" } }, { "cursor": "eyJjIjoxMiwidCI6NiwiaSI6ZmFsc2V9", "node": { - "digest": "4YeKsZh5nDLNfbNVT2GRwenv1gvaVdDde9BASgiBqkv8" + "digest": "E4Lt2ME1UNXjXnuVzF9CMyrHnhdQWbWRXj3myqpzZMrP" } }, { "cursor": "eyJjIjoxMiwidCI6NywiaSI6ZmFsc2V9", "node": { - "digest": "3K4FAYTRRVNDcR16Z3ScNDP5W4dgvAdMorFux1xnGVvd" + "digest": "HCaSSfE7rxz2tEA5uEiZ4KDozVm1uSKJGsv8nqwi9YDv" } } ] @@ -236,43 +236,43 @@ Response: { { "cursor": "eyJjIjo4LCJ0IjowLCJpIjpmYWxzZX0", "node": { - "digest": "CmPeQ2JKFnEKeZsYz9cYU8q57Wb1fgZxpDqYq9YinuQA" + "digest": "5XJr4MXW36JcHWpqf2PF2oAD2cfWg4bk1jQWVp6Josvj" } }, { "cursor": "eyJjIjo4LCJ0IjoxLCJpIjpmYWxzZX0", "node": { - "digest": "Hm38QqFQFCkzedsHCXsxJt7j5VCo9KUn5ZL9zSAZVKWh" + "digest": "DU5Lhorja2vD4cPn5Eqv6HvKA1ZY8mfGC3SHhGccQ9qF" } }, { "cursor": "eyJjIjo4LCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "ywf9ak8rV9kDDx73zGH5vtTJkHmfxXHvLbSGhwAyxdM" + "digest": "7BPdUc3LpRmchBjPiSZ5Uju5yDZMbVhTQNYjDjmTXP5D" } }, { "cursor": "eyJjIjo4LCJ0IjozLCJpIjpmYWxzZX0", "node": { - "digest": "Fsv6DwsUEPUqRyFP7wkd7mMDq93PXMmSXqQAgrrnsKb3" + "digest": "6ox6bgoqFRZbKftfQPM29MNpo876eW6PdsHxaNLEYiDj" } }, { "cursor": "eyJjIjo4LCJ0Ijo0LCJpIjpmYWxzZX0", "node": { - "digest": "25MhkStEmB44afcLcmtdA426Qbo6a5z6sJrURSfn4wSM" + "digest": "3s48BKV1NGNPV5VRKFsAJKWiyUiWDLVxguSERpo8qRTV" } }, { "cursor": "eyJjIjo4LCJ0Ijo1LCJpIjpmYWxzZX0", "node": { - "digest": "Hutiw6GLNGCuRwMTZhxFW2CrYF6PYt2DomNMwtN1hLUo" + "digest": "6adEehTM2gxGKdumJXeKFwcaYhSB6RwGHbqg7Fv7JBmk" } }, { "cursor": "eyJjIjo4LCJ0Ijo2LCJpIjpmYWxzZX0", "node": { - "digest": "4YeKsZh5nDLNfbNVT2GRwenv1gvaVdDde9BASgiBqkv8" + "digest": "E4Lt2ME1UNXjXnuVzF9CMyrHnhdQWbWRXj3myqpzZMrP" } } ] @@ -284,25 +284,25 @@ Response: { { "cursor": "eyJjIjoxMiwidCI6OCwiaSI6ZmFsc2V9", "node": { - "digest": "EnghHbuhpcGuNpC1L2iqikieXAyHJUnanzg5urDmwgdz" + "digest": "DEYu7BBRqZyroncK1on7suMYzHy7GLJMoSsC6K8d14BR" } }, { "cursor": "eyJjIjoxMiwidCI6OSwiaSI6ZmFsc2V9", "node": { - "digest": "9G4gNmS2XYetMxBty65fKfx9iDRBetdpdNZuyK7oRd6c" + "digest": "32S1CoAgaGv7diFo2Bs7StnPB7Aakoma1JUT4k4cUpA7" } }, { "cursor": "eyJjIjoxMiwidCI6MTAsImkiOmZhbHNlfQ", "node": { - "digest": "3ivB1ebWRpc4Si1vd8tsSY7XwaY7DrumxdS3KdXyZLVr" + "digest": "3gEc6E6ypnwUqRtVBcF3ZwXPEQJaisXDDaK21p4sZyM8" } }, { "cursor": "eyJjIjoxMiwidCI6MTEsImkiOmZhbHNlfQ", "node": { - "digest": "CpC1zrGsmDThe1sPdhRPmfQCpaaTYJanFTtyr5BBNNNZ" + "digest": "3r7RAxvcfWTz7vCtCtExmfQkr5LdSvNF4V6F1JVzNzGW" } } ] @@ -313,67 +313,67 @@ Response: { { "cursor": "eyJjIjoxMiwidCI6MCwiaSI6ZmFsc2V9", "node": { - "digest": "CmPeQ2JKFnEKeZsYz9cYU8q57Wb1fgZxpDqYq9YinuQA" + "digest": "5XJr4MXW36JcHWpqf2PF2oAD2cfWg4bk1jQWVp6Josvj" } }, { "cursor": "eyJjIjoxMiwidCI6MSwiaSI6ZmFsc2V9", "node": { - "digest": "Hm38QqFQFCkzedsHCXsxJt7j5VCo9KUn5ZL9zSAZVKWh" + "digest": "DU5Lhorja2vD4cPn5Eqv6HvKA1ZY8mfGC3SHhGccQ9qF" } }, { "cursor": "eyJjIjoxMiwidCI6MiwiaSI6ZmFsc2V9", "node": { - "digest": "ywf9ak8rV9kDDx73zGH5vtTJkHmfxXHvLbSGhwAyxdM" + "digest": "7BPdUc3LpRmchBjPiSZ5Uju5yDZMbVhTQNYjDjmTXP5D" } }, { "cursor": "eyJjIjoxMiwidCI6MywiaSI6ZmFsc2V9", "node": { - "digest": "Fsv6DwsUEPUqRyFP7wkd7mMDq93PXMmSXqQAgrrnsKb3" + "digest": "6ox6bgoqFRZbKftfQPM29MNpo876eW6PdsHxaNLEYiDj" } }, { "cursor": "eyJjIjoxMiwidCI6NCwiaSI6ZmFsc2V9", "node": { - "digest": "25MhkStEmB44afcLcmtdA426Qbo6a5z6sJrURSfn4wSM" + "digest": "3s48BKV1NGNPV5VRKFsAJKWiyUiWDLVxguSERpo8qRTV" } }, { "cursor": "eyJjIjoxMiwidCI6NSwiaSI6ZmFsc2V9", "node": { - "digest": "Hutiw6GLNGCuRwMTZhxFW2CrYF6PYt2DomNMwtN1hLUo" + "digest": "6adEehTM2gxGKdumJXeKFwcaYhSB6RwGHbqg7Fv7JBmk" } }, { "cursor": "eyJjIjoxMiwidCI6NiwiaSI6ZmFsc2V9", "node": { - "digest": "4YeKsZh5nDLNfbNVT2GRwenv1gvaVdDde9BASgiBqkv8" + "digest": "E4Lt2ME1UNXjXnuVzF9CMyrHnhdQWbWRXj3myqpzZMrP" } }, { "cursor": "eyJjIjoxMiwidCI6NywiaSI6ZmFsc2V9", "node": { - "digest": "3K4FAYTRRVNDcR16Z3ScNDP5W4dgvAdMorFux1xnGVvd" + "digest": "HCaSSfE7rxz2tEA5uEiZ4KDozVm1uSKJGsv8nqwi9YDv" } }, { "cursor": "eyJjIjoxMiwidCI6OCwiaSI6ZmFsc2V9", "node": { - "digest": "EnghHbuhpcGuNpC1L2iqikieXAyHJUnanzg5urDmwgdz" + "digest": "DEYu7BBRqZyroncK1on7suMYzHy7GLJMoSsC6K8d14BR" } }, { "cursor": "eyJjIjoxMiwidCI6OSwiaSI6ZmFsc2V9", "node": { - "digest": "9G4gNmS2XYetMxBty65fKfx9iDRBetdpdNZuyK7oRd6c" + "digest": "32S1CoAgaGv7diFo2Bs7StnPB7Aakoma1JUT4k4cUpA7" } }, { "cursor": "eyJjIjoxMiwidCI6MTAsImkiOmZhbHNlfQ", "node": { - "digest": "3ivB1ebWRpc4Si1vd8tsSY7XwaY7DrumxdS3KdXyZLVr" + "digest": "3gEc6E6ypnwUqRtVBcF3ZwXPEQJaisXDDaK21p4sZyM8" } } ] @@ -395,19 +395,19 @@ Response: { { "cursor": "eyJjIjo3LCJ0IjoxLCJpIjpmYWxzZX0", "node": { - "digest": "Hm38QqFQFCkzedsHCXsxJt7j5VCo9KUn5ZL9zSAZVKWh" + "digest": "DU5Lhorja2vD4cPn5Eqv6HvKA1ZY8mfGC3SHhGccQ9qF" } }, { "cursor": "eyJjIjo3LCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "ywf9ak8rV9kDDx73zGH5vtTJkHmfxXHvLbSGhwAyxdM" + "digest": "7BPdUc3LpRmchBjPiSZ5Uju5yDZMbVhTQNYjDjmTXP5D" } }, { "cursor": "eyJjIjo3LCJ0IjozLCJpIjpmYWxzZX0", "node": { - "digest": "Fsv6DwsUEPUqRyFP7wkd7mMDq93PXMmSXqQAgrrnsKb3" + "digest": "6ox6bgoqFRZbKftfQPM29MNpo876eW6PdsHxaNLEYiDj" } } ] @@ -420,19 +420,19 @@ Response: { { "cursor": "eyJjIjoxMSwidCI6NSwiaSI6ZmFsc2V9", "node": { - "digest": "Hutiw6GLNGCuRwMTZhxFW2CrYF6PYt2DomNMwtN1hLUo" + "digest": "6adEehTM2gxGKdumJXeKFwcaYhSB6RwGHbqg7Fv7JBmk" } }, { "cursor": "eyJjIjoxMSwidCI6NiwiaSI6ZmFsc2V9", "node": { - "digest": "4YeKsZh5nDLNfbNVT2GRwenv1gvaVdDde9BASgiBqkv8" + "digest": "E4Lt2ME1UNXjXnuVzF9CMyrHnhdQWbWRXj3myqpzZMrP" } }, { "cursor": "eyJjIjoxMSwidCI6NywiaSI6ZmFsc2V9", "node": { - "digest": "3K4FAYTRRVNDcR16Z3ScNDP5W4dgvAdMorFux1xnGVvd" + "digest": "HCaSSfE7rxz2tEA5uEiZ4KDozVm1uSKJGsv8nqwi9YDv" } } ] @@ -445,19 +445,19 @@ Response: { { "cursor": "eyJjIjoxMiwidCI6OSwiaSI6ZmFsc2V9", "node": { - "digest": "9G4gNmS2XYetMxBty65fKfx9iDRBetdpdNZuyK7oRd6c" + "digest": "32S1CoAgaGv7diFo2Bs7StnPB7Aakoma1JUT4k4cUpA7" } }, { "cursor": "eyJjIjoxMiwidCI6MTAsImkiOmZhbHNlfQ", "node": { - "digest": "3ivB1ebWRpc4Si1vd8tsSY7XwaY7DrumxdS3KdXyZLVr" + "digest": "3gEc6E6ypnwUqRtVBcF3ZwXPEQJaisXDDaK21p4sZyM8" } }, { "cursor": "eyJjIjoxMiwidCI6MTEsImkiOmZhbHNlfQ", "node": { - "digest": "CpC1zrGsmDThe1sPdhRPmfQCpaaTYJanFTtyr5BBNNNZ" + "digest": "3r7RAxvcfWTz7vCtCtExmfQkr5LdSvNF4V6F1JVzNzGW" } } ] @@ -480,7 +480,7 @@ Response: { { "cursor": "eyJjIjoyLCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "ywf9ak8rV9kDDx73zGH5vtTJkHmfxXHvLbSGhwAyxdM" + "digest": "7BPdUc3LpRmchBjPiSZ5Uju5yDZMbVhTQNYjDjmTXP5D" } } ] @@ -493,7 +493,7 @@ Response: { { "cursor": "eyJjIjo2LCJ0Ijo2LCJpIjpmYWxzZX0", "node": { - "digest": "4YeKsZh5nDLNfbNVT2GRwenv1gvaVdDde9BASgiBqkv8" + "digest": "E4Lt2ME1UNXjXnuVzF9CMyrHnhdQWbWRXj3myqpzZMrP" } } ] @@ -506,7 +506,7 @@ Response: { { "cursor": "eyJjIjoxMCwidCI6MTAsImkiOmZhbHNlfQ", "node": { - "digest": "3ivB1ebWRpc4Si1vd8tsSY7XwaY7DrumxdS3KdXyZLVr" + "digest": "3gEc6E6ypnwUqRtVBcF3ZwXPEQJaisXDDaK21p4sZyM8" } } ] @@ -527,24 +527,24 @@ Response: { { "cursor": "eyJjIjo2LCJ0Ijo2LCJpIjpmYWxzZX0", "node": { - "digest": "4YeKsZh5nDLNfbNVT2GRwenv1gvaVdDde9BASgiBqkv8", + "digest": "E4Lt2ME1UNXjXnuVzF9CMyrHnhdQWbWRXj3myqpzZMrP", "sender": { "objects": { "edges": [ { - "cursor": "IAxPzPNh6DBZ+lqchUWSL6NuS4y4vgz6pHjStN6JWpF/BgAAAAAAAAA=" + "cursor": "IAliWgwcr8n37dkM7mZzjkkSDkpfTgEPI6/C9SsEvD4wBgAAAAAAAAA=" }, { - "cursor": "IBH2WwM6ABxADOrEJ86TGtH880uA4Ce8kx9mXnSZj2DYBgAAAAAAAAA=" + "cursor": "IBGzpWsWAJqwTtHNtSzMlcNw/lp0Q7mxGubPXWYv7SVDBgAAAAAAAAA=" }, { - "cursor": "IBpSURhn9I1wJERwRFANEXqI98bagpRsksYuuwsrFFtQBgAAAAAAAAA=" + "cursor": "IBOIPAp4MKzHgkh72/O/iMqkg160NT2TOG5zk0R8EOecBgAAAAAAAAA=" }, { - "cursor": "IElmtyPGm1GRSMWeyj+lZFiOkrmPZ2oKnWeSNwmsAXf/BgAAAAAAAAA=" + "cursor": "IJ9s1BXAi+CbOC283CRx1p7dE11yaTraKXAGeCk6JJ22BgAAAAAAAAA=" }, { - "cursor": "IH3CTzQfYFptQFnBjnyGfpphmbw0uuCt4JTcq1tQsIZRBgAAAAAAAAA=" + "cursor": "IMQLpfEhnQHF4QCZEHo91G+rv+6yBXtQFiTVqRnjt4bFBgAAAAAAAAA=" } ] } @@ -558,33 +558,33 @@ Response: { { "cursor": "eyJjIjoxMiwidCI6MiwiaSI6ZmFsc2V9", "node": { - "digest": "ywf9ak8rV9kDDx73zGH5vtTJkHmfxXHvLbSGhwAyxdM", + "digest": "7BPdUc3LpRmchBjPiSZ5Uju5yDZMbVhTQNYjDjmTXP5D", "sender": { "objects": { "edges": [ { - "cursor": "IAxPzPNh6DBZ+lqchUWSL6NuS4y4vgz6pHjStN6JWpF/DAAAAAAAAAA=" + "cursor": "IAliWgwcr8n37dkM7mZzjkkSDkpfTgEPI6/C9SsEvD4wDAAAAAAAAAA=" }, { - "cursor": "IBH2WwM6ABxADOrEJ86TGtH880uA4Ce8kx9mXnSZj2DYDAAAAAAAAAA=" + "cursor": "IBGzpWsWAJqwTtHNtSzMlcNw/lp0Q7mxGubPXWYv7SVDDAAAAAAAAAA=" }, { - "cursor": "IBoQDKT5WktB7OuMUXNBKsoqg3oDa0b1Dc7AIuqcPPDDDAAAAAAAAAA=" + "cursor": "IBOIPAp4MKzHgkh72/O/iMqkg160NT2TOG5zk0R8EOecDAAAAAAAAAA=" }, { - "cursor": "IBpSURhn9I1wJERwRFANEXqI98bagpRsksYuuwsrFFtQDAAAAAAAAAA=" + "cursor": "IH3dM0/kdjLFfKxIeE26rvsOqbgLFXHoXv2N1Adx93zqDAAAAAAAAAA=" }, { - "cursor": "IElmtyPGm1GRSMWeyj+lZFiOkrmPZ2oKnWeSNwmsAXf/DAAAAAAAAAA=" + "cursor": "IJwgTtRy13vdcto1Li65De5RV6AKz5dX/YGwgyYOZeC2DAAAAAAAAAA=" }, { - "cursor": "IGjIsZmGLOFhLkGROat6+WyaYp65xLf0mDp34jCyG3KsDAAAAAAAAAA=" + "cursor": "IJ9s1BXAi+CbOC283CRx1p7dE11yaTraKXAGeCk6JJ22DAAAAAAAAAA=" }, { - "cursor": "IH3CTzQfYFptQFnBjnyGfpphmbw0uuCt4JTcq1tQsIZRDAAAAAAAAAA=" + "cursor": "IMQLpfEhnQHF4QCZEHo91G+rv+6yBXtQFiTVqRnjt4bFDAAAAAAAAAA=" }, { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilADAAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGDAAAAAAAAAA=" } ] } @@ -594,33 +594,33 @@ Response: { { "cursor": "eyJjIjoxMiwidCI6NCwiaSI6ZmFsc2V9", "node": { - "digest": "25MhkStEmB44afcLcmtdA426Qbo6a5z6sJrURSfn4wSM", + "digest": "3s48BKV1NGNPV5VRKFsAJKWiyUiWDLVxguSERpo8qRTV", "sender": { "objects": { "edges": [ { - "cursor": "IAxPzPNh6DBZ+lqchUWSL6NuS4y4vgz6pHjStN6JWpF/DAAAAAAAAAA=" + "cursor": "IAliWgwcr8n37dkM7mZzjkkSDkpfTgEPI6/C9SsEvD4wDAAAAAAAAAA=" }, { - "cursor": "IBH2WwM6ABxADOrEJ86TGtH880uA4Ce8kx9mXnSZj2DYDAAAAAAAAAA=" + "cursor": "IBGzpWsWAJqwTtHNtSzMlcNw/lp0Q7mxGubPXWYv7SVDDAAAAAAAAAA=" }, { - "cursor": "IBoQDKT5WktB7OuMUXNBKsoqg3oDa0b1Dc7AIuqcPPDDDAAAAAAAAAA=" + "cursor": "IBOIPAp4MKzHgkh72/O/iMqkg160NT2TOG5zk0R8EOecDAAAAAAAAAA=" }, { - "cursor": "IBpSURhn9I1wJERwRFANEXqI98bagpRsksYuuwsrFFtQDAAAAAAAAAA=" + "cursor": "IH3dM0/kdjLFfKxIeE26rvsOqbgLFXHoXv2N1Adx93zqDAAAAAAAAAA=" }, { - "cursor": "IElmtyPGm1GRSMWeyj+lZFiOkrmPZ2oKnWeSNwmsAXf/DAAAAAAAAAA=" + "cursor": "IJwgTtRy13vdcto1Li65De5RV6AKz5dX/YGwgyYOZeC2DAAAAAAAAAA=" }, { - "cursor": "IGjIsZmGLOFhLkGROat6+WyaYp65xLf0mDp34jCyG3KsDAAAAAAAAAA=" + "cursor": "IJ9s1BXAi+CbOC283CRx1p7dE11yaTraKXAGeCk6JJ22DAAAAAAAAAA=" }, { - "cursor": "IH3CTzQfYFptQFnBjnyGfpphmbw0uuCt4JTcq1tQsIZRDAAAAAAAAAA=" + "cursor": "IMQLpfEhnQHF4QCZEHo91G+rv+6yBXtQFiTVqRnjt4bFDAAAAAAAAAA=" }, { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilADAAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGDAAAAAAAAAA=" } ] } @@ -630,33 +630,33 @@ Response: { { "cursor": "eyJjIjoxMiwidCI6NSwiaSI6ZmFsc2V9", "node": { - "digest": "Hutiw6GLNGCuRwMTZhxFW2CrYF6PYt2DomNMwtN1hLUo", + "digest": "6adEehTM2gxGKdumJXeKFwcaYhSB6RwGHbqg7Fv7JBmk", "sender": { "objects": { "edges": [ { - "cursor": "IAxPzPNh6DBZ+lqchUWSL6NuS4y4vgz6pHjStN6JWpF/DAAAAAAAAAA=" + "cursor": "IAliWgwcr8n37dkM7mZzjkkSDkpfTgEPI6/C9SsEvD4wDAAAAAAAAAA=" }, { - "cursor": "IBH2WwM6ABxADOrEJ86TGtH880uA4Ce8kx9mXnSZj2DYDAAAAAAAAAA=" + "cursor": "IBGzpWsWAJqwTtHNtSzMlcNw/lp0Q7mxGubPXWYv7SVDDAAAAAAAAAA=" }, { - "cursor": "IBoQDKT5WktB7OuMUXNBKsoqg3oDa0b1Dc7AIuqcPPDDDAAAAAAAAAA=" + "cursor": "IBOIPAp4MKzHgkh72/O/iMqkg160NT2TOG5zk0R8EOecDAAAAAAAAAA=" }, { - "cursor": "IBpSURhn9I1wJERwRFANEXqI98bagpRsksYuuwsrFFtQDAAAAAAAAAA=" + "cursor": "IH3dM0/kdjLFfKxIeE26rvsOqbgLFXHoXv2N1Adx93zqDAAAAAAAAAA=" }, { - "cursor": "IElmtyPGm1GRSMWeyj+lZFiOkrmPZ2oKnWeSNwmsAXf/DAAAAAAAAAA=" + "cursor": "IJwgTtRy13vdcto1Li65De5RV6AKz5dX/YGwgyYOZeC2DAAAAAAAAAA=" }, { - "cursor": "IGjIsZmGLOFhLkGROat6+WyaYp65xLf0mDp34jCyG3KsDAAAAAAAAAA=" + "cursor": "IJ9s1BXAi+CbOC283CRx1p7dE11yaTraKXAGeCk6JJ22DAAAAAAAAAA=" }, { - "cursor": "IH3CTzQfYFptQFnBjnyGfpphmbw0uuCt4JTcq1tQsIZRDAAAAAAAAAA=" + "cursor": "IMQLpfEhnQHF4QCZEHo91G+rv+6yBXtQFiTVqRnjt4bFDAAAAAAAAAA=" }, { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilADAAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGDAAAAAAAAAA=" } ] } @@ -666,33 +666,33 @@ Response: { { "cursor": "eyJjIjoxMiwidCI6NiwiaSI6ZmFsc2V9", "node": { - "digest": "4YeKsZh5nDLNfbNVT2GRwenv1gvaVdDde9BASgiBqkv8", + "digest": "E4Lt2ME1UNXjXnuVzF9CMyrHnhdQWbWRXj3myqpzZMrP", "sender": { "objects": { "edges": [ { - "cursor": "IAxPzPNh6DBZ+lqchUWSL6NuS4y4vgz6pHjStN6JWpF/DAAAAAAAAAA=" + "cursor": "IAliWgwcr8n37dkM7mZzjkkSDkpfTgEPI6/C9SsEvD4wDAAAAAAAAAA=" }, { - "cursor": "IBH2WwM6ABxADOrEJ86TGtH880uA4Ce8kx9mXnSZj2DYDAAAAAAAAAA=" + "cursor": "IBGzpWsWAJqwTtHNtSzMlcNw/lp0Q7mxGubPXWYv7SVDDAAAAAAAAAA=" }, { - "cursor": "IBoQDKT5WktB7OuMUXNBKsoqg3oDa0b1Dc7AIuqcPPDDDAAAAAAAAAA=" + "cursor": "IBOIPAp4MKzHgkh72/O/iMqkg160NT2TOG5zk0R8EOecDAAAAAAAAAA=" }, { - "cursor": "IBpSURhn9I1wJERwRFANEXqI98bagpRsksYuuwsrFFtQDAAAAAAAAAA=" + "cursor": "IH3dM0/kdjLFfKxIeE26rvsOqbgLFXHoXv2N1Adx93zqDAAAAAAAAAA=" }, { - "cursor": "IElmtyPGm1GRSMWeyj+lZFiOkrmPZ2oKnWeSNwmsAXf/DAAAAAAAAAA=" + "cursor": "IJwgTtRy13vdcto1Li65De5RV6AKz5dX/YGwgyYOZeC2DAAAAAAAAAA=" }, { - "cursor": "IGjIsZmGLOFhLkGROat6+WyaYp65xLf0mDp34jCyG3KsDAAAAAAAAAA=" + "cursor": "IJ9s1BXAi+CbOC283CRx1p7dE11yaTraKXAGeCk6JJ22DAAAAAAAAAA=" }, { - "cursor": "IH3CTzQfYFptQFnBjnyGfpphmbw0uuCt4JTcq1tQsIZRDAAAAAAAAAA=" + "cursor": "IMQLpfEhnQHF4QCZEHo91G+rv+6yBXtQFiTVqRnjt4bFDAAAAAAAAAA=" }, { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilADAAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGDAAAAAAAAAA=" } ] } @@ -702,33 +702,33 @@ Response: { { "cursor": "eyJjIjoxMiwidCI6OCwiaSI6ZmFsc2V9", "node": { - "digest": "EnghHbuhpcGuNpC1L2iqikieXAyHJUnanzg5urDmwgdz", + "digest": "DEYu7BBRqZyroncK1on7suMYzHy7GLJMoSsC6K8d14BR", "sender": { "objects": { "edges": [ { - "cursor": "IAxPzPNh6DBZ+lqchUWSL6NuS4y4vgz6pHjStN6JWpF/DAAAAAAAAAA=" + "cursor": "IAliWgwcr8n37dkM7mZzjkkSDkpfTgEPI6/C9SsEvD4wDAAAAAAAAAA=" }, { - "cursor": "IBH2WwM6ABxADOrEJ86TGtH880uA4Ce8kx9mXnSZj2DYDAAAAAAAAAA=" + "cursor": "IBGzpWsWAJqwTtHNtSzMlcNw/lp0Q7mxGubPXWYv7SVDDAAAAAAAAAA=" }, { - "cursor": "IBoQDKT5WktB7OuMUXNBKsoqg3oDa0b1Dc7AIuqcPPDDDAAAAAAAAAA=" + "cursor": "IBOIPAp4MKzHgkh72/O/iMqkg160NT2TOG5zk0R8EOecDAAAAAAAAAA=" }, { - "cursor": "IBpSURhn9I1wJERwRFANEXqI98bagpRsksYuuwsrFFtQDAAAAAAAAAA=" + "cursor": "IH3dM0/kdjLFfKxIeE26rvsOqbgLFXHoXv2N1Adx93zqDAAAAAAAAAA=" }, { - "cursor": "IElmtyPGm1GRSMWeyj+lZFiOkrmPZ2oKnWeSNwmsAXf/DAAAAAAAAAA=" + "cursor": "IJwgTtRy13vdcto1Li65De5RV6AKz5dX/YGwgyYOZeC2DAAAAAAAAAA=" }, { - "cursor": "IGjIsZmGLOFhLkGROat6+WyaYp65xLf0mDp34jCyG3KsDAAAAAAAAAA=" + "cursor": "IJ9s1BXAi+CbOC283CRx1p7dE11yaTraKXAGeCk6JJ22DAAAAAAAAAA=" }, { - "cursor": "IH3CTzQfYFptQFnBjnyGfpphmbw0uuCt4JTcq1tQsIZRDAAAAAAAAAA=" + "cursor": "IMQLpfEhnQHF4QCZEHo91G+rv+6yBXtQFiTVqRnjt4bFDAAAAAAAAAA=" }, { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilADAAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGDAAAAAAAAAA=" } ] } @@ -738,33 +738,33 @@ Response: { { "cursor": "eyJjIjoxMiwidCI6OSwiaSI6ZmFsc2V9", "node": { - "digest": "9G4gNmS2XYetMxBty65fKfx9iDRBetdpdNZuyK7oRd6c", + "digest": "32S1CoAgaGv7diFo2Bs7StnPB7Aakoma1JUT4k4cUpA7", "sender": { "objects": { "edges": [ { - "cursor": "IAxPzPNh6DBZ+lqchUWSL6NuS4y4vgz6pHjStN6JWpF/DAAAAAAAAAA=" + "cursor": "IAliWgwcr8n37dkM7mZzjkkSDkpfTgEPI6/C9SsEvD4wDAAAAAAAAAA=" }, { - "cursor": "IBH2WwM6ABxADOrEJ86TGtH880uA4Ce8kx9mXnSZj2DYDAAAAAAAAAA=" + "cursor": "IBGzpWsWAJqwTtHNtSzMlcNw/lp0Q7mxGubPXWYv7SVDDAAAAAAAAAA=" }, { - "cursor": "IBoQDKT5WktB7OuMUXNBKsoqg3oDa0b1Dc7AIuqcPPDDDAAAAAAAAAA=" + "cursor": "IBOIPAp4MKzHgkh72/O/iMqkg160NT2TOG5zk0R8EOecDAAAAAAAAAA=" }, { - "cursor": "IBpSURhn9I1wJERwRFANEXqI98bagpRsksYuuwsrFFtQDAAAAAAAAAA=" + "cursor": "IH3dM0/kdjLFfKxIeE26rvsOqbgLFXHoXv2N1Adx93zqDAAAAAAAAAA=" }, { - "cursor": "IElmtyPGm1GRSMWeyj+lZFiOkrmPZ2oKnWeSNwmsAXf/DAAAAAAAAAA=" + "cursor": "IJwgTtRy13vdcto1Li65De5RV6AKz5dX/YGwgyYOZeC2DAAAAAAAAAA=" }, { - "cursor": "IGjIsZmGLOFhLkGROat6+WyaYp65xLf0mDp34jCyG3KsDAAAAAAAAAA=" + "cursor": "IJ9s1BXAi+CbOC283CRx1p7dE11yaTraKXAGeCk6JJ22DAAAAAAAAAA=" }, { - "cursor": "IH3CTzQfYFptQFnBjnyGfpphmbw0uuCt4JTcq1tQsIZRDAAAAAAAAAA=" + "cursor": "IMQLpfEhnQHF4QCZEHo91G+rv+6yBXtQFiTVqRnjt4bFDAAAAAAAAAA=" }, { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilADAAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGDAAAAAAAAAA=" } ] } @@ -774,33 +774,33 @@ Response: { { "cursor": "eyJjIjoxMiwidCI6MTAsImkiOmZhbHNlfQ", "node": { - "digest": "3ivB1ebWRpc4Si1vd8tsSY7XwaY7DrumxdS3KdXyZLVr", + "digest": "3gEc6E6ypnwUqRtVBcF3ZwXPEQJaisXDDaK21p4sZyM8", "sender": { "objects": { "edges": [ { - "cursor": "IAxPzPNh6DBZ+lqchUWSL6NuS4y4vgz6pHjStN6JWpF/DAAAAAAAAAA=" + "cursor": "IAliWgwcr8n37dkM7mZzjkkSDkpfTgEPI6/C9SsEvD4wDAAAAAAAAAA=" }, { - "cursor": "IBH2WwM6ABxADOrEJ86TGtH880uA4Ce8kx9mXnSZj2DYDAAAAAAAAAA=" + "cursor": "IBGzpWsWAJqwTtHNtSzMlcNw/lp0Q7mxGubPXWYv7SVDDAAAAAAAAAA=" }, { - "cursor": "IBoQDKT5WktB7OuMUXNBKsoqg3oDa0b1Dc7AIuqcPPDDDAAAAAAAAAA=" + "cursor": "IBOIPAp4MKzHgkh72/O/iMqkg160NT2TOG5zk0R8EOecDAAAAAAAAAA=" }, { - "cursor": "IBpSURhn9I1wJERwRFANEXqI98bagpRsksYuuwsrFFtQDAAAAAAAAAA=" + "cursor": "IH3dM0/kdjLFfKxIeE26rvsOqbgLFXHoXv2N1Adx93zqDAAAAAAAAAA=" }, { - "cursor": "IElmtyPGm1GRSMWeyj+lZFiOkrmPZ2oKnWeSNwmsAXf/DAAAAAAAAAA=" + "cursor": "IJwgTtRy13vdcto1Li65De5RV6AKz5dX/YGwgyYOZeC2DAAAAAAAAAA=" }, { - "cursor": "IGjIsZmGLOFhLkGROat6+WyaYp65xLf0mDp34jCyG3KsDAAAAAAAAAA=" + "cursor": "IJ9s1BXAi+CbOC283CRx1p7dE11yaTraKXAGeCk6JJ22DAAAAAAAAAA=" }, { - "cursor": "IH3CTzQfYFptQFnBjnyGfpphmbw0uuCt4JTcq1tQsIZRDAAAAAAAAAA=" + "cursor": "IMQLpfEhnQHF4QCZEHo91G+rv+6yBXtQFiTVqRnjt4bFDAAAAAAAAAA=" }, { - "cursor": "IOZ8ocIl02ZSSkdsdSrpt8BzDKmSCK2CZJpYvoPS+ilADAAAAAAAAAA=" + "cursor": "IOgqlyHReHTY5weyEZOqIFQo+bdy3dX6B6ZXq8EnpHkGDAAAAAAAAAA=" } ] } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/epochs/transaction_blocks.move b/crates/iota-graphql-e2e-tests/tests/consistency/epochs/transaction_blocks.move index c380d3c57d7..630fc53f989 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/epochs/transaction_blocks.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/epochs/transaction_blocks.move @@ -18,7 +18,7 @@ // 11 | epoch | 11 | 2 // 12 | epoch | 12 | 3 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/object_at_version.exp b/crates/iota-graphql-e2e-tests/tests/consistency/object_at_version.exp index 6e02483be2f..36ce838def2 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/object_at_version.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/object_at_version.exp @@ -29,7 +29,7 @@ Response: { "asMoveObject": { "contents": { "json": { - "id": "0xc9cc2d86cf85f0448da63718fd5e7d0e47f4a707561d2f1ddcb46eabc0dcf382", + "id": "0xf10a93d3a78211ff333d9be01524d9c2275cba2d6b6334d94a6a7505e0e95a9e", "value": "0" } } @@ -57,7 +57,7 @@ Response: { "asMoveObject": { "contents": { "json": { - "id": "0xc9cc2d86cf85f0448da63718fd5e7d0e47f4a707561d2f1ddcb46eabc0dcf382", + "id": "0xf10a93d3a78211ff333d9be01524d9c2275cba2d6b6334d94a6a7505e0e95a9e", "value": "1" } } @@ -69,7 +69,7 @@ Response: { "asMoveObject": { "contents": { "json": { - "id": "0xc9cc2d86cf85f0448da63718fd5e7d0e47f4a707561d2f1ddcb46eabc0dcf382", + "id": "0xf10a93d3a78211ff333d9be01524d9c2275cba2d6b6334d94a6a7505e0e95a9e", "value": "0" } } @@ -104,7 +104,7 @@ Response: { "asMoveObject": { "contents": { "json": { - "id": "0xc9cc2d86cf85f0448da63718fd5e7d0e47f4a707561d2f1ddcb46eabc0dcf382", + "id": "0xf10a93d3a78211ff333d9be01524d9c2275cba2d6b6334d94a6a7505e0e95a9e", "value": "1" } } @@ -134,7 +134,7 @@ Response: { "asMoveObject": { "contents": { "json": { - "id": "0xc9cc2d86cf85f0448da63718fd5e7d0e47f4a707561d2f1ddcb46eabc0dcf382", + "id": "0xf10a93d3a78211ff333d9be01524d9c2275cba2d6b6334d94a6a7505e0e95a9e", "value": "1" } } @@ -151,7 +151,7 @@ Response: { "asMoveObject": { "contents": { "json": { - "id": "0xc9cc2d86cf85f0448da63718fd5e7d0e47f4a707561d2f1ddcb46eabc0dcf382", + "id": "0xf10a93d3a78211ff333d9be01524d9c2275cba2d6b6334d94a6a7505e0e95a9e", "value": "0" } } @@ -205,7 +205,7 @@ Response: { "asMoveObject": { "contents": { "json": { - "id": "0xc9cc2d86cf85f0448da63718fd5e7d0e47f4a707561d2f1ddcb46eabc0dcf382", + "id": "0xf10a93d3a78211ff333d9be01524d9c2275cba2d6b6334d94a6a7505e0e95a9e", "value": "1" } } @@ -222,7 +222,7 @@ Response: { "asMoveObject": { "contents": { "json": { - "id": "0xc9cc2d86cf85f0448da63718fd5e7d0e47f4a707561d2f1ddcb46eabc0dcf382", + "id": "0xf10a93d3a78211ff333d9be01524d9c2275cba2d6b6334d94a6a7505e0e95a9e", "value": "0" } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/object_at_version.move b/crates/iota-graphql-e2e-tests/tests/consistency/object_at_version.move index a41d3e6364b..6094a6d0c4e 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/object_at_version.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/object_at_version.move @@ -15,7 +15,7 @@ // checkpoint 4. The object would only be visible at version 6 from objects_snapshot, and at version // 7 from objects_history. -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination.exp b/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination.exp index 0de0dc4a041..016b20d8dab 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination.exp @@ -36,10 +36,10 @@ Response: { "version": 4, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x99897a446d53989defb6e160415609c939386acbbf3a5064839e13bdc9910da7", + "id": "0xf8a39541f76820d0a241fad3ce1ad92b9e0aaba58725c8f656fc2bd76e94ba55", "value": "1" } } @@ -80,10 +80,10 @@ Response: { "version": 4, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x99897a446d53989defb6e160415609c939386acbbf3a5064839e13bdc9910da7", + "id": "0xf8a39541f76820d0a241fad3ce1ad92b9e0aaba58725c8f656fc2bd76e94ba55", "value": "1" } } @@ -108,10 +108,10 @@ Response: { "version": 3, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x0196134211788ab305d30e0f558ba451f6c65681c965e4ae1d1fbf4ce138aafa", + "id": "0x6f0747e131cf59b36487d82b0c890fd1075e09966a2befda7c37131cb12db6c1", "value": "0" } } @@ -120,35 +120,35 @@ Response: { "version": 5, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x594943312691aa8916bbfc715d0c40044122de94f608d2a1cbb0c6dc49486075", + "id": "0x7c56aa41384c3229d05ac087c1434fbcbe2885285e8a3e167a71f65fff383377", "value": "2" } } }, { - "version": 4, + "version": 6, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x99897a446d53989defb6e160415609c939386acbbf3a5064839e13bdc9910da7", - "value": "1" + "id": "0x940bb12ed38c05a8a88d940203d0986bedb550f050dfe610fadebd8446e96409", + "value": "3" } } }, { - "version": 6, + "version": 4, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0xc41ad14789a46af2b7dd52aa001acfca2193147e06751a9d324c9aec011f5b54", - "value": "3" + "id": "0xf8a39541f76820d0a241fad3ce1ad92b9e0aaba58725c8f656fc2bd76e94ba55", + "value": "1" } } } @@ -169,10 +169,10 @@ Response: { "version": 3, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x0196134211788ab305d30e0f558ba451f6c65681c965e4ae1d1fbf4ce138aafa", + "id": "0x6f0747e131cf59b36487d82b0c890fd1075e09966a2befda7c37131cb12db6c1", "value": "0" } } @@ -181,35 +181,35 @@ Response: { "version": 5, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x594943312691aa8916bbfc715d0c40044122de94f608d2a1cbb0c6dc49486075", + "id": "0x7c56aa41384c3229d05ac087c1434fbcbe2885285e8a3e167a71f65fff383377", "value": "2" } } }, { - "version": 4, + "version": 6, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x99897a446d53989defb6e160415609c939386acbbf3a5064839e13bdc9910da7", - "value": "1" + "id": "0x940bb12ed38c05a8a88d940203d0986bedb550f050dfe610fadebd8446e96409", + "value": "3" } } }, { - "version": 6, + "version": 4, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0xc41ad14789a46af2b7dd52aa001acfca2193147e06751a9d324c9aec011f5b54", - "value": "3" + "id": "0xf8a39541f76820d0a241fad3ce1ad92b9e0aaba58725c8f656fc2bd76e94ba55", + "value": "1" } } } @@ -237,14 +237,14 @@ Response: { "objects": { "nodes": [ { - "version": 4, + "version": 6, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x99897a446d53989defb6e160415609c939386acbbf3a5064839e13bdc9910da7", - "value": "1" + "id": "0x940bb12ed38c05a8a88d940203d0986bedb550f050dfe610fadebd8446e96409", + "value": "3" } }, "owner_at_latest_state_has_iota_only": { @@ -255,61 +255,61 @@ Response: { "version": 3, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x0196134211788ab305d30e0f558ba451f6c65681c965e4ae1d1fbf4ce138aafa", + "id": "0x6f0747e131cf59b36487d82b0c890fd1075e09966a2befda7c37131cb12db6c1", "value": "0" } } }, { - "version": 1, + "version": 5, "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x1a52511867f48d7024447044500d117a88f7c6da82946c92c62ebb0b2b145b50", - "balance": { - "value": "300000000000000" - } + "id": "0x7c56aa41384c3229d05ac087c1434fbcbe2885285e8a3e167a71f65fff383377", + "value": "2" } } }, { - "version": 5, + "version": 6, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x594943312691aa8916bbfc715d0c40044122de94f608d2a1cbb0c6dc49486075", - "value": "2" + "id": "0x940bb12ed38c05a8a88d940203d0986bedb550f050dfe610fadebd8446e96409", + "value": "3" } } }, { - "version": 4, + "version": 1, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" }, "json": { - "id": "0x99897a446d53989defb6e160415609c939386acbbf3a5064839e13bdc9910da7", - "value": "1" + "id": "0x9f6cd415c08be09b382dbcdc2471d69edd135d72693ada29700678293a249db6", + "balance": { + "value": "300000000000000" + } } } }, { - "version": 6, + "version": 4, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0xc41ad14789a46af2b7dd52aa001acfca2193147e06751a9d324c9aec011f5b54", - "value": "3" + "id": "0xf8a39541f76820d0a241fad3ce1ad92b9e0aaba58725c8f656fc2bd76e94ba55", + "value": "1" } } } @@ -319,14 +319,14 @@ Response: { } }, { - "version": 6, + "version": 4, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0xc41ad14789a46af2b7dd52aa001acfca2193147e06751a9d324c9aec011f5b54", - "value": "3" + "id": "0xf8a39541f76820d0a241fad3ce1ad92b9e0aaba58725c8f656fc2bd76e94ba55", + "value": "1" } }, "owner_at_latest_state_has_iota_only": { @@ -337,61 +337,61 @@ Response: { "version": 3, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x0196134211788ab305d30e0f558ba451f6c65681c965e4ae1d1fbf4ce138aafa", + "id": "0x6f0747e131cf59b36487d82b0c890fd1075e09966a2befda7c37131cb12db6c1", "value": "0" } } }, { - "version": 1, + "version": 5, "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x1a52511867f48d7024447044500d117a88f7c6da82946c92c62ebb0b2b145b50", - "balance": { - "value": "300000000000000" - } + "id": "0x7c56aa41384c3229d05ac087c1434fbcbe2885285e8a3e167a71f65fff383377", + "value": "2" } } }, { - "version": 5, + "version": 6, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x594943312691aa8916bbfc715d0c40044122de94f608d2a1cbb0c6dc49486075", - "value": "2" + "id": "0x940bb12ed38c05a8a88d940203d0986bedb550f050dfe610fadebd8446e96409", + "value": "3" } } }, { - "version": 4, + "version": 1, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" }, "json": { - "id": "0x99897a446d53989defb6e160415609c939386acbbf3a5064839e13bdc9910da7", - "value": "1" + "id": "0x9f6cd415c08be09b382dbcdc2471d69edd135d72693ada29700678293a249db6", + "balance": { + "value": "300000000000000" + } } } }, { - "version": 6, + "version": 4, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0xc41ad14789a46af2b7dd52aa001acfca2193147e06751a9d324c9aec011f5b54", - "value": "3" + "id": "0xf8a39541f76820d0a241fad3ce1ad92b9e0aaba58725c8f656fc2bd76e94ba55", + "value": "1" } } } @@ -410,10 +410,10 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x0196134211788ab305d30e0f558ba451f6c65681c965e4ae1d1fbf4ce138aafa", + "id": "0x6f0747e131cf59b36487d82b0c890fd1075e09966a2befda7c37131cb12db6c1", "value": "0" } }, @@ -425,61 +425,61 @@ Response: { "version": 3, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x0196134211788ab305d30e0f558ba451f6c65681c965e4ae1d1fbf4ce138aafa", + "id": "0x6f0747e131cf59b36487d82b0c890fd1075e09966a2befda7c37131cb12db6c1", "value": "0" } } }, { - "version": 1, + "version": 5, "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x1a52511867f48d7024447044500d117a88f7c6da82946c92c62ebb0b2b145b50", - "balance": { - "value": "300000000000000" - } + "id": "0x7c56aa41384c3229d05ac087c1434fbcbe2885285e8a3e167a71f65fff383377", + "value": "2" } } }, { - "version": 5, + "version": 6, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x594943312691aa8916bbfc715d0c40044122de94f608d2a1cbb0c6dc49486075", - "value": "2" + "id": "0x940bb12ed38c05a8a88d940203d0986bedb550f050dfe610fadebd8446e96409", + "value": "3" } } }, { - "version": 4, + "version": 1, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" }, "json": { - "id": "0x99897a446d53989defb6e160415609c939386acbbf3a5064839e13bdc9910da7", - "value": "1" + "id": "0x9f6cd415c08be09b382dbcdc2471d69edd135d72693ada29700678293a249db6", + "balance": { + "value": "300000000000000" + } } } }, { - "version": 6, + "version": 4, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0xc41ad14789a46af2b7dd52aa001acfca2193147e06751a9d324c9aec011f5b54", - "value": "3" + "id": "0xf8a39541f76820d0a241fad3ce1ad92b9e0aaba58725c8f656fc2bd76e94ba55", + "value": "1" } } } @@ -544,10 +544,10 @@ Response: { "version": 7, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x0196134211788ab305d30e0f558ba451f6c65681c965e4ae1d1fbf4ce138aafa", + "id": "0x6f0747e131cf59b36487d82b0c890fd1075e09966a2befda7c37131cb12db6c1", "value": "0" } } @@ -556,10 +556,10 @@ Response: { "version": 7, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x594943312691aa8916bbfc715d0c40044122de94f608d2a1cbb0c6dc49486075", + "id": "0x7c56aa41384c3229d05ac087c1434fbcbe2885285e8a3e167a71f65fff383377", "value": "2" } } @@ -568,11 +568,11 @@ Response: { "version": 7, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x99897a446d53989defb6e160415609c939386acbbf3a5064839e13bdc9910da7", - "value": "1" + "id": "0x940bb12ed38c05a8a88d940203d0986bedb550f050dfe610fadebd8446e96409", + "value": "3" } } }, @@ -580,11 +580,11 @@ Response: { "version": 7, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0xc41ad14789a46af2b7dd52aa001acfca2193147e06751a9d324c9aec011f5b54", - "value": "3" + "id": "0xf8a39541f76820d0a241fad3ce1ad92b9e0aaba58725c8f656fc2bd76e94ba55", + "value": "1" } } } @@ -605,10 +605,10 @@ Response: { "version": 3, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x0196134211788ab305d30e0f558ba451f6c65681c965e4ae1d1fbf4ce138aafa", + "id": "0x6f0747e131cf59b36487d82b0c890fd1075e09966a2befda7c37131cb12db6c1", "value": "0" } } @@ -617,35 +617,35 @@ Response: { "version": 5, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x594943312691aa8916bbfc715d0c40044122de94f608d2a1cbb0c6dc49486075", + "id": "0x7c56aa41384c3229d05ac087c1434fbcbe2885285e8a3e167a71f65fff383377", "value": "2" } } }, { - "version": 4, + "version": 6, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0x99897a446d53989defb6e160415609c939386acbbf3a5064839e13bdc9910da7", - "value": "1" + "id": "0x940bb12ed38c05a8a88d940203d0986bedb550f050dfe610fadebd8446e96409", + "value": "3" } } }, { - "version": 6, + "version": 4, "contents": { "type": { - "repr": "0xf497316682c0ad2d0b757399e9db83e3ed3bceadaedbaad12029111c331445d4::M1::Object" + "repr": "0x6b62cc078a121fab1252edd6ab185f82952cc9fbb4d48995701f1967395137b7::M1::Object" }, "json": { - "id": "0xc41ad14789a46af2b7dd52aa001acfca2193147e06751a9d324c9aec011f5b54", - "value": "3" + "id": "0xf8a39541f76820d0a241fad3ce1ad92b9e0aaba58725c8f656fc2bd76e94ba55", + "value": "1" } } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination.move b/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination.move index 67c12677c4d..ada74bf5e32 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator // cp | object_id | owner diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination_single.exp b/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination_single.exp index fef91b92f80..93b8cd65ec4 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination_single.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination_single.exp @@ -46,10 +46,10 @@ Response: { "version": 4, "contents": { "type": { - "repr": "0x8ba0645055ca9407f1fda27d9c5a12697215f596238dc5db7eaf434471bd1e0a::M1::Object" + "repr": "0x8bbe5284437551822da4df75dc89e391d31121c83485877f32b60fbeeeceaaa3::M1::Object" }, "json": { - "id": "0x0ba09e699b7a2c69fcf9e5c3b84610345a7f92bc8aad27d06a82aa10a94f0cf5", + "id": "0x1bb439946e688b4f5c76a5068b2183eed7ed771aaf63cb803cea55c2b67edfb7", "value": "100" } } @@ -85,10 +85,10 @@ Response: { "version": 4, "contents": { "type": { - "repr": "0x8ba0645055ca9407f1fda27d9c5a12697215f596238dc5db7eaf434471bd1e0a::M1::Object" + "repr": "0x8bbe5284437551822da4df75dc89e391d31121c83485877f32b60fbeeeceaaa3::M1::Object" }, "json": { - "id": "0x0ba09e699b7a2c69fcf9e5c3b84610345a7f92bc8aad27d06a82aa10a94f0cf5", + "id": "0x1bb439946e688b4f5c76a5068b2183eed7ed771aaf63cb803cea55c2b67edfb7", "value": "100" } } @@ -110,10 +110,10 @@ Response: { "version": 5, "contents": { "type": { - "repr": "0x8ba0645055ca9407f1fda27d9c5a12697215f596238dc5db7eaf434471bd1e0a::M1::Object" + "repr": "0x8bbe5284437551822da4df75dc89e391d31121c83485877f32b60fbeeeceaaa3::M1::Object" }, "json": { - "id": "0x0ba09e699b7a2c69fcf9e5c3b84610345a7f92bc8aad27d06a82aa10a94f0cf5", + "id": "0x1bb439946e688b4f5c76a5068b2183eed7ed771aaf63cb803cea55c2b67edfb7", "value": "200" } } @@ -122,10 +122,10 @@ Response: { "version": 4, "contents": { "type": { - "repr": "0x8ba0645055ca9407f1fda27d9c5a12697215f596238dc5db7eaf434471bd1e0a::M1::Object" + "repr": "0x8bbe5284437551822da4df75dc89e391d31121c83485877f32b60fbeeeceaaa3::M1::Object" }, "json": { - "id": "0x1064f0e2c1402df49da99967e53770ca27892458ce8a9ea983db9d1f13d8ce61", + "id": "0xedcb20ee342ef450d49808752f158a4fb8cc4d685f1f894f76c3f646f9755806", "value": "1" } } @@ -145,10 +145,10 @@ Response: { "version": 5, "contents": { "type": { - "repr": "0x8ba0645055ca9407f1fda27d9c5a12697215f596238dc5db7eaf434471bd1e0a::M1::Object" + "repr": "0x8bbe5284437551822da4df75dc89e391d31121c83485877f32b60fbeeeceaaa3::M1::Object" }, "json": { - "id": "0x0ba09e699b7a2c69fcf9e5c3b84610345a7f92bc8aad27d06a82aa10a94f0cf5", + "id": "0x1bb439946e688b4f5c76a5068b2183eed7ed771aaf63cb803cea55c2b67edfb7", "value": "200" } }, @@ -160,10 +160,10 @@ Response: { "version": 5, "contents": { "type": { - "repr": "0x8ba0645055ca9407f1fda27d9c5a12697215f596238dc5db7eaf434471bd1e0a::M1::Object" + "repr": "0x8bbe5284437551822da4df75dc89e391d31121c83485877f32b60fbeeeceaaa3::M1::Object" }, "json": { - "id": "0x0ba09e699b7a2c69fcf9e5c3b84610345a7f92bc8aad27d06a82aa10a94f0cf5", + "id": "0x1bb439946e688b4f5c76a5068b2183eed7ed771aaf63cb803cea55c2b67edfb7", "value": "200" } } @@ -172,10 +172,10 @@ Response: { "version": 4, "contents": { "type": { - "repr": "0x8ba0645055ca9407f1fda27d9c5a12697215f596238dc5db7eaf434471bd1e0a::M1::Object" + "repr": "0x8bbe5284437551822da4df75dc89e391d31121c83485877f32b60fbeeeceaaa3::M1::Object" }, "json": { - "id": "0x1064f0e2c1402df49da99967e53770ca27892458ce8a9ea983db9d1f13d8ce61", + "id": "0xedcb20ee342ef450d49808752f158a4fb8cc4d685f1f894f76c3f646f9755806", "value": "1" } } @@ -216,10 +216,10 @@ Response: { "version": 5, "contents": { "type": { - "repr": "0x8ba0645055ca9407f1fda27d9c5a12697215f596238dc5db7eaf434471bd1e0a::M1::Object" + "repr": "0x8bbe5284437551822da4df75dc89e391d31121c83485877f32b60fbeeeceaaa3::M1::Object" }, "json": { - "id": "0x0ba09e699b7a2c69fcf9e5c3b84610345a7f92bc8aad27d06a82aa10a94f0cf5", + "id": "0x1bb439946e688b4f5c76a5068b2183eed7ed771aaf63cb803cea55c2b67edfb7", "value": "200" } }, @@ -231,10 +231,10 @@ Response: { "version": 5, "contents": { "type": { - "repr": "0x8ba0645055ca9407f1fda27d9c5a12697215f596238dc5db7eaf434471bd1e0a::M1::Object" + "repr": "0x8bbe5284437551822da4df75dc89e391d31121c83485877f32b60fbeeeceaaa3::M1::Object" }, "json": { - "id": "0x0ba09e699b7a2c69fcf9e5c3b84610345a7f92bc8aad27d06a82aa10a94f0cf5", + "id": "0x1bb439946e688b4f5c76a5068b2183eed7ed771aaf63cb803cea55c2b67edfb7", "value": "200" } } @@ -243,10 +243,10 @@ Response: { "version": 4, "contents": { "type": { - "repr": "0x8ba0645055ca9407f1fda27d9c5a12697215f596238dc5db7eaf434471bd1e0a::M1::Object" + "repr": "0x8bbe5284437551822da4df75dc89e391d31121c83485877f32b60fbeeeceaaa3::M1::Object" }, "json": { - "id": "0x1064f0e2c1402df49da99967e53770ca27892458ce8a9ea983db9d1f13d8ce61", + "id": "0xedcb20ee342ef450d49808752f158a4fb8cc4d685f1f894f76c3f646f9755806", "value": "1" } } @@ -273,10 +273,10 @@ Response: { "version": 5, "contents": { "type": { - "repr": "0x8ba0645055ca9407f1fda27d9c5a12697215f596238dc5db7eaf434471bd1e0a::M1::Object" + "repr": "0x8bbe5284437551822da4df75dc89e391d31121c83485877f32b60fbeeeceaaa3::M1::Object" }, "json": { - "id": "0x0ba09e699b7a2c69fcf9e5c3b84610345a7f92bc8aad27d06a82aa10a94f0cf5", + "id": "0x1bb439946e688b4f5c76a5068b2183eed7ed771aaf63cb803cea55c2b67edfb7", "value": "200" } } @@ -285,10 +285,10 @@ Response: { "version": 6, "contents": { "type": { - "repr": "0x8ba0645055ca9407f1fda27d9c5a12697215f596238dc5db7eaf434471bd1e0a::M1::Object" + "repr": "0x8bbe5284437551822da4df75dc89e391d31121c83485877f32b60fbeeeceaaa3::M1::Object" }, "json": { - "id": "0x1064f0e2c1402df49da99967e53770ca27892458ce8a9ea983db9d1f13d8ce61", + "id": "0xedcb20ee342ef450d49808752f158a4fb8cc4d685f1f894f76c3f646f9755806", "value": "300" } } @@ -308,10 +308,10 @@ Response: { "version": 5, "contents": { "type": { - "repr": "0x8ba0645055ca9407f1fda27d9c5a12697215f596238dc5db7eaf434471bd1e0a::M1::Object" + "repr": "0x8bbe5284437551822da4df75dc89e391d31121c83485877f32b60fbeeeceaaa3::M1::Object" }, "json": { - "id": "0x0ba09e699b7a2c69fcf9e5c3b84610345a7f92bc8aad27d06a82aa10a94f0cf5", + "id": "0x1bb439946e688b4f5c76a5068b2183eed7ed771aaf63cb803cea55c2b67edfb7", "value": "200" } }, @@ -323,10 +323,10 @@ Response: { "version": 5, "contents": { "type": { - "repr": "0x8ba0645055ca9407f1fda27d9c5a12697215f596238dc5db7eaf434471bd1e0a::M1::Object" + "repr": "0x8bbe5284437551822da4df75dc89e391d31121c83485877f32b60fbeeeceaaa3::M1::Object" }, "json": { - "id": "0x0ba09e699b7a2c69fcf9e5c3b84610345a7f92bc8aad27d06a82aa10a94f0cf5", + "id": "0x1bb439946e688b4f5c76a5068b2183eed7ed771aaf63cb803cea55c2b67edfb7", "value": "200" } } @@ -335,10 +335,10 @@ Response: { "version": 6, "contents": { "type": { - "repr": "0x8ba0645055ca9407f1fda27d9c5a12697215f596238dc5db7eaf434471bd1e0a::M1::Object" + "repr": "0x8bbe5284437551822da4df75dc89e391d31121c83485877f32b60fbeeeceaaa3::M1::Object" }, "json": { - "id": "0x1064f0e2c1402df49da99967e53770ca27892458ce8a9ea983db9d1f13d8ce61", + "id": "0xedcb20ee342ef450d49808752f158a4fb8cc4d685f1f894f76c3f646f9755806", "value": "300" } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination_single.move b/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination_single.move index dff516f9251..2c83462732e 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination_single.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/objects_pagination_single.move @@ -6,7 +6,7 @@ // one object as a cursor to view the other object that gets mutated per checkpoint. The ordering is // consistent even if the cursor object is mutated. -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/performance/many_objects.exp b/crates/iota-graphql-e2e-tests/tests/consistency/performance/many_objects.exp index 51eaea008d6..1f34b961a62 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/performance/many_objects.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/performance/many_objects.exp @@ -35,11 +35,11 @@ Response: { }, "contents": { "json": { - "id": "0xfdfdbe2471965f393fb0f1e2a849804c2951742cef2833aa5630a982927e65e5", - "value": "459" + "id": "0xfd8751b345372f858c4f0934135b80b285926005c0cf4fad0509ca05ab7a3d89", + "value": "149" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } @@ -54,11 +54,11 @@ Response: { }, "contents": { "json": { - "id": "0xff793ecbad3b0cb648a7f8a676d41abf9479e51e8a70f661fd8a01f1053fb117", - "value": "338" + "id": "0xff549f934fc6cc753ee00f3b02aec8c28ec21a0ec4f96028088425ab58e55fd3", + "value": "32" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } @@ -76,11 +76,11 @@ Response: { }, "contents": { "json": { - "id": "0xfc58a4a75f805dda8348e39cbabdfd88a9358a37207dc08086618c8539f2c863", - "value": "314" + "id": "0xfd1c5ba117f89c7a1505ad527c0dbcd53e4c84b9799ef138ee609a51a462302d", + "value": "475" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } }, @@ -92,11 +92,11 @@ Response: { }, "contents": { "json": { - "id": "0xfca8a7528a05b7a8386c8a976be57d6ddcc1a4e9697aa4c207d3853e1f7a42d8", - "value": "296" + "id": "0xfd66a0ccbb382e17d62697b438f4a1d782775fae030dadffd30f7e0a20e807e9", + "value": "379" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } }, @@ -108,11 +108,11 @@ Response: { }, "contents": { "json": { - "id": "0xfdfdbe2471965f393fb0f1e2a849804c2951742cef2833aa5630a982927e65e5", - "value": "459" + "id": "0xfd8751b345372f858c4f0934135b80b285926005c0cf4fad0509ca05ab7a3d89", + "value": "149" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } }, @@ -124,11 +124,11 @@ Response: { }, "contents": { "json": { - "id": "0xff793ecbad3b0cb648a7f8a676d41abf9479e51e8a70f661fd8a01f1053fb117", - "value": "338" + "id": "0xff549f934fc6cc753ee00f3b02aec8c28ec21a0ec4f96028088425ab58e55fd3", + "value": "32" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } @@ -163,7 +163,7 @@ Contents: Test::M1::Object { bytes: fake(2,498), }, }, - value: 459u64, + value: 149u64, } task 9, line 93: @@ -176,7 +176,7 @@ Contents: Test::M1::Object { bytes: fake(2,497), }, }, - value: 296u64, + value: 379u64, } task 10, line 95: @@ -199,11 +199,11 @@ Response: { }, "contents": { "json": { - "id": "0xfca8a7528a05b7a8386c8a976be57d6ddcc1a4e9697aa4c207d3853e1f7a42d8", - "value": "296" + "id": "0xfd66a0ccbb382e17d62697b438f4a1d782775fae030dadffd30f7e0a20e807e9", + "value": "379" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } @@ -218,11 +218,11 @@ Response: { }, "contents": { "json": { - "id": "0xfdfdbe2471965f393fb0f1e2a849804c2951742cef2833aa5630a982927e65e5", - "value": "459" + "id": "0xfd8751b345372f858c4f0934135b80b285926005c0cf4fad0509ca05ab7a3d89", + "value": "149" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } @@ -237,11 +237,11 @@ Response: { }, "contents": { "json": { - "id": "0xff793ecbad3b0cb648a7f8a676d41abf9479e51e8a70f661fd8a01f1053fb117", - "value": "338" + "id": "0xff549f934fc6cc753ee00f3b02aec8c28ec21a0ec4f96028088425ab58e55fd3", + "value": "32" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } @@ -260,11 +260,11 @@ Response: { }, "contents": { "json": { - "id": "0xfc58a4a75f805dda8348e39cbabdfd88a9358a37207dc08086618c8539f2c863", - "value": "314" + "id": "0xfd1c5ba117f89c7a1505ad527c0dbcd53e4c84b9799ef138ee609a51a462302d", + "value": "475" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } @@ -287,11 +287,11 @@ Response: { }, "contents": { "json": { - "id": "0xff793ecbad3b0cb648a7f8a676d41abf9479e51e8a70f661fd8a01f1053fb117", - "value": "338" + "id": "0xff549f934fc6cc753ee00f3b02aec8c28ec21a0ec4f96028088425ab58e55fd3", + "value": "32" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } @@ -305,11 +305,11 @@ Response: { }, "contents": { "json": { - "id": "0xff793ecbad3b0cb648a7f8a676d41abf9479e51e8a70f661fd8a01f1053fb117", - "value": "338" + "id": "0xff549f934fc6cc753ee00f3b02aec8c28ec21a0ec4f96028088425ab58e55fd3", + "value": "32" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } @@ -325,11 +325,11 @@ Response: { }, "contents": { "json": { - "id": "0xfca8a7528a05b7a8386c8a976be57d6ddcc1a4e9697aa4c207d3853e1f7a42d8", - "value": "296" + "id": "0xfd66a0ccbb382e17d62697b438f4a1d782775fae030dadffd30f7e0a20e807e9", + "value": "379" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } @@ -343,11 +343,11 @@ Response: { }, "contents": { "json": { - "id": "0xfdfdbe2471965f393fb0f1e2a849804c2951742cef2833aa5630a982927e65e5", - "value": "459" + "id": "0xfd8751b345372f858c4f0934135b80b285926005c0cf4fad0509ca05ab7a3d89", + "value": "149" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } @@ -361,11 +361,11 @@ Response: { }, "contents": { "json": { - "id": "0xff793ecbad3b0cb648a7f8a676d41abf9479e51e8a70f661fd8a01f1053fb117", - "value": "338" + "id": "0xff549f934fc6cc753ee00f3b02aec8c28ec21a0ec4f96028088425ab58e55fd3", + "value": "32" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } @@ -383,11 +383,11 @@ Response: { }, "contents": { "json": { - "id": "0xfca8a7528a05b7a8386c8a976be57d6ddcc1a4e9697aa4c207d3853e1f7a42d8", - "value": "296" + "id": "0xfd66a0ccbb382e17d62697b438f4a1d782775fae030dadffd30f7e0a20e807e9", + "value": "379" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } @@ -401,11 +401,11 @@ Response: { }, "contents": { "json": { - "id": "0xfdfdbe2471965f393fb0f1e2a849804c2951742cef2833aa5630a982927e65e5", - "value": "459" + "id": "0xfd8751b345372f858c4f0934135b80b285926005c0cf4fad0509ca05ab7a3d89", + "value": "149" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } @@ -419,11 +419,11 @@ Response: { }, "contents": { "json": { - "id": "0xff793ecbad3b0cb648a7f8a676d41abf9479e51e8a70f661fd8a01f1053fb117", - "value": "338" + "id": "0xff549f934fc6cc753ee00f3b02aec8c28ec21a0ec4f96028088425ab58e55fd3", + "value": "32" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } @@ -442,7 +442,7 @@ Response: { }, "contents": { "json": { - "id": "0xd1bed9d8032538f51c93732600d7002478b40e2036b90c576fb76d589c03693d", + "id": "0xf9c7908a3d09a86f7bddc51e15266a9d8fa570b98b2bc57179cee2410b2f9576", "balance": { "value": "300000000000000" } @@ -461,11 +461,11 @@ Response: { }, "contents": { "json": { - "id": "0xfca8a7528a05b7a8386c8a976be57d6ddcc1a4e9697aa4c207d3853e1f7a42d8", - "value": "296" + "id": "0xfd66a0ccbb382e17d62697b438f4a1d782775fae030dadffd30f7e0a20e807e9", + "value": "379" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } }, @@ -478,11 +478,11 @@ Response: { }, "contents": { "json": { - "id": "0xfdfdbe2471965f393fb0f1e2a849804c2951742cef2833aa5630a982927e65e5", - "value": "459" + "id": "0xfd8751b345372f858c4f0934135b80b285926005c0cf4fad0509ca05ab7a3d89", + "value": "149" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } }, @@ -495,11 +495,11 @@ Response: { }, "contents": { "json": { - "id": "0xff793ecbad3b0cb648a7f8a676d41abf9479e51e8a70f661fd8a01f1053fb117", - "value": "338" + "id": "0xff549f934fc6cc753ee00f3b02aec8c28ec21a0ec4f96028088425ab58e55fd3", + "value": "32" }, "type": { - "repr": "0x7a9f66657f48f8cac66483bb31b0e47a4ad4331237664b0250794d720bff607c::M1::Object" + "repr": "0xa139341a6424399931eacdae9e94c01156651410c82be788100eea2d15f5a71c::M1::Object" } } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/performance/many_objects.move b/crates/iota-graphql-e2e-tests/tests/consistency/performance/many_objects.move index 75146a0c888..6f8bef7f758 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/performance/many_objects.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/performance/many_objects.move @@ -8,7 +8,7 @@ // criteria are then invalidated by a newer version of the matched object. We set `last: 1` but // transfer the last 3 objects because we increase the limit by 2 behind the scenes. -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/staked_iota.exp b/crates/iota-graphql-e2e-tests/tests/consistency/staked_iota.exp index ad831cbb06f..69dc8c0b26a 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/staked_iota.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/staked_iota.exp @@ -25,7 +25,7 @@ gas summary: computation_cost: 1000000, computation_cost_burned: 1000000, storag task 3, line 25: //# run 0x3::iota_system::request_add_stake --args object(0x5) object(2,0) @validator_0 --sender C -events: Event { package_id: iota_system, transaction_module: Identifier("iota_system"), sender: C, type_: StructTag { address: iota_system, module: Identifier("validator"), name: Identifier("StakingRequestEvent"), type_params: [] }, contents: [145, 201, 73, 32, 135, 23, 121, 37, 7, 101, 40, 103, 216, 99, 155, 51, 118, 111, 115, 220, 44, 253, 144, 143, 80, 103, 133, 240, 58, 235, 137, 202, 175, 163, 158, 79, 0, 218, 226, 120, 249, 119, 199, 198, 147, 10, 94, 44, 118, 232, 93, 23, 165, 38, 215, 36, 187, 206, 15, 184, 31, 176, 125, 76, 140, 202, 78, 28, 224, 186, 89, 4, 206, 166, 29, 249, 36, 45, 162, 247, 210, 158, 62, 243, 40, 251, 126, 192, 124, 8, 107, 59, 244, 124, 166, 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 228, 11, 84, 2, 0, 0, 0] } +events: Event { package_id: iota_system, transaction_module: Identifier("iota_system"), sender: C, type_: StructTag { address: iota_system, module: Identifier("validator"), name: Identifier("StakingRequestEvent"), type_params: [] }, contents: [217, 25, 175, 239, 57, 244, 176, 32, 209, 232, 109, 171, 147, 246, 3, 15, 205, 197, 172, 21, 36, 143, 47, 100, 240, 226, 221, 254, 42, 3, 157, 155, 175, 163, 158, 79, 0, 218, 226, 120, 249, 119, 199, 198, 147, 10, 94, 44, 118, 232, 93, 23, 165, 38, 215, 36, 187, 206, 15, 184, 31, 176, 125, 76, 140, 202, 78, 28, 224, 186, 89, 4, 206, 166, 29, 249, 36, 45, 162, 247, 210, 158, 62, 243, 40, 251, 126, 192, 124, 8, 107, 59, 244, 124, 166, 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 228, 11, 84, 2, 0, 0, 0] } created: object(3,0) mutated: object(_), 0x0000000000000000000000000000000000000000000000000000000000000005, object(0,0) deleted: object(2,0) @@ -49,7 +49,7 @@ gas summary: computation_cost: 1000000, computation_cost_burned: 1000000, storag task 7, line 35: //# run 0x3::iota_system::request_add_stake --args object(0x5) object(6,0) @validator_0 --sender C -events: Event { package_id: iota_system, transaction_module: Identifier("iota_system"), sender: C, type_: StructTag { address: iota_system, module: Identifier("validator"), name: Identifier("StakingRequestEvent"), type_params: [] }, contents: [145, 201, 73, 32, 135, 23, 121, 37, 7, 101, 40, 103, 216, 99, 155, 51, 118, 111, 115, 220, 44, 253, 144, 143, 80, 103, 133, 240, 58, 235, 137, 202, 175, 163, 158, 79, 0, 218, 226, 120, 249, 119, 199, 198, 147, 10, 94, 44, 118, 232, 93, 23, 165, 38, 215, 36, 187, 206, 15, 184, 31, 176, 125, 76, 140, 202, 78, 28, 224, 186, 89, 4, 206, 166, 29, 249, 36, 45, 162, 247, 210, 158, 62, 243, 40, 251, 126, 192, 124, 8, 107, 59, 244, 124, 166, 26, 1, 0, 0, 0, 0, 0, 0, 0, 0, 228, 11, 84, 2, 0, 0, 0] } +events: Event { package_id: iota_system, transaction_module: Identifier("iota_system"), sender: C, type_: StructTag { address: iota_system, module: Identifier("validator"), name: Identifier("StakingRequestEvent"), type_params: [] }, contents: [217, 25, 175, 239, 57, 244, 176, 32, 209, 232, 109, 171, 147, 246, 3, 15, 205, 197, 172, 21, 36, 143, 47, 100, 240, 226, 221, 254, 42, 3, 157, 155, 175, 163, 158, 79, 0, 218, 226, 120, 249, 119, 199, 198, 147, 10, 94, 44, 118, 232, 93, 23, 165, 38, 215, 36, 187, 206, 15, 184, 31, 176, 125, 76, 140, 202, 78, 28, 224, 186, 89, 4, 206, 166, 29, 249, 36, 45, 162, 247, 210, 158, 62, 243, 40, 251, 126, 192, 124, 8, 107, 59, 244, 124, 166, 26, 1, 0, 0, 0, 0, 0, 0, 0, 0, 228, 11, 84, 2, 0, 0, 0] } created: object(7,0) mutated: object(_), 0x0000000000000000000000000000000000000000000000000000000000000005, object(0,0) deleted: object(6,0) @@ -109,13 +109,13 @@ Response: { "stakedIotas": { "edges": [ { - "cursor": "ICY/+FbG+Hqd34UX7O+7VLiMXKC/bqtTKAZwQ4YiyetzBAAAAAAAAAA=", + "cursor": "ICx+i9jLtLxeXhS/A0ahYRLGYYlMzFleNwu1meRx2FBtBAAAAAAAAAA=", "node": { "principal": "10000000000" } }, { - "cursor": "IPEp6dbnEicrkwynNujLy5/Voas64Cc5Ta7ruT5oQNbTBAAAAAAAAAA=", + "cursor": "IJ/jh2RIAAFvbLTf7RAKDlv0BCV3CAEBy5bUCfTnTOIgBAAAAAAAAAA=", "node": { "principal": "10000000000" } @@ -158,10 +158,15 @@ task 14, lines 105-148: Response: { "data": { "coins_after_obj_3_0_chkpt_3": { + "stakedIotas": { + "edges": [] + } + }, + "coins_before_obj_3_0_chkpt_3": { "stakedIotas": { "edges": [ { - "cursor": "IPEp6dbnEicrkwynNujLy5/Voas64Cc5Ta7ruT5oQNbTAwAAAAAAAAA=", + "cursor": "ICx+i9jLtLxeXhS/A0ahYRLGYYlMzFleNwu1meRx2FBtAwAAAAAAAAA=", "node": { "principal": "10000000000" } @@ -169,27 +174,22 @@ Response: { ] } }, - "coins_before_obj_3_0_chkpt_3": { - "stakedIotas": { - "edges": [] - } - }, "coins_after_obj_7_0_chkpt_3": { - "stakedIotas": { - "edges": [] - } - }, - "coins_before_obj_7_0_chkpt_3": { "stakedIotas": { "edges": [ { - "cursor": "ICY/+FbG+Hqd34UX7O+7VLiMXKC/bqtTKAZwQ4YiyetzAwAAAAAAAAA=", + "cursor": "IJ/jh2RIAAFvbLTf7RAKDlv0BCV3CAEBy5bUCfTnTOIgAwAAAAAAAAA=", "node": { "principal": "10000000000" } } ] } + }, + "coins_before_obj_7_0_chkpt_3": { + "stakedIotas": { + "edges": [] + } } } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/staked_iota.move b/crates/iota-graphql-e2e-tests/tests/consistency/staked_iota.move index 1b3366f414b..8814cb2e415 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/staked_iota.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/staked_iota.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --simulator --accounts C +//# init --protocol-version 3 --simulator --accounts C //# run-graphql { diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/tx_address_objects.exp b/crates/iota-graphql-e2e-tests/tests/consistency/tx_address_objects.exp index 0b07261f1ff..81a411d29eb 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/tx_address_objects.exp +++ b/crates/iota-graphql-e2e-tests/tests/consistency/tx_address_objects.exp @@ -61,66 +61,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -136,66 +136,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -209,66 +209,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -284,7 +284,7 @@ Response: { "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" }, "json": { - "id": "0x1a52511867f48d7024447044500d117a88f7c6da82946c92c62ebb0b2b145b50", + "id": "0x9f6cd415c08be09b382dbcdc2471d69edd135d72693ada29700678293a249db6", "balance": { "value": "299999993098000" } @@ -306,66 +306,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -390,66 +390,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -463,66 +463,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -538,7 +538,7 @@ Response: { "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" }, "json": { - "id": "0x1a52511867f48d7024447044500d117a88f7c6da82946c92c62ebb0b2b145b50", + "id": "0x9f6cd415c08be09b382dbcdc2471d69edd135d72693ada29700678293a249db6", "balance": { "value": "300000000000000" } @@ -557,66 +557,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -630,66 +630,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -705,7 +705,7 @@ Response: { "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" }, "json": { - "id": "0x1a52511867f48d7024447044500d117a88f7c6da82946c92c62ebb0b2b145b50", + "id": "0x9f6cd415c08be09b382dbcdc2471d69edd135d72693ada29700678293a249db6", "balance": { "value": "299999996712400" } @@ -724,66 +724,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -797,66 +797,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -872,7 +872,7 @@ Response: { "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" }, "json": { - "id": "0x1a52511867f48d7024447044500d117a88f7c6da82946c92c62ebb0b2b145b50", + "id": "0x9f6cd415c08be09b382dbcdc2471d69edd135d72693ada29700678293a249db6", "balance": { "value": "299999993098000" } @@ -907,66 +907,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -982,66 +982,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -1055,66 +1055,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -1130,7 +1130,7 @@ Response: { "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" }, "json": { - "id": "0x1a52511867f48d7024447044500d117a88f7c6da82946c92c62ebb0b2b145b50", + "id": "0x9f6cd415c08be09b382dbcdc2471d69edd135d72693ada29700678293a249db6", "balance": { "value": "299999993098000" } @@ -1152,66 +1152,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -1240,66 +1240,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -1313,66 +1313,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -1388,66 +1388,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -1461,66 +1461,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -1536,66 +1536,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } @@ -1609,66 +1609,66 @@ Response: { { "contents": { "json": { - "id": "0x06d5d2e9af70892fdde85874daf2c3dc5f1adc9cb3f22baabfd56e173fc8c39f", - "value": "2" + "id": "0x1457a25e59dd95037988d1623acefa4ea3ed3aeaaf5233566b32c6f20419a028", + "value": "5" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x1985e10335572563abedc5e04e975dea930a1ef4e4d87685dfc9a9256b293f46", - "value": "3" + "id": "0x20e682e9bd64d57e2b9184ef69c11af67e1b92cb4295c51bf23ee9533f128cf7", + "value": "200" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x2f3706f4e91edbbd4a2f48a4c52581b16acacf01cbe49d0d789a522e3128bf30", - "value": "4" + "id": "0x2d8288b42428e037f480d70ac52b163a6431bc84858823cf18a40d05047cddb0", + "value": "6" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x7137b2b53c2a2acb9a845c9c9b68de4fae6509bbff3e63ebc7b3c7c601a24f56", - "value": "200" + "id": "0xd4b20eb98f5c3714c708e35473bcee3e75ffc863d17fb0042915191624d76776", + "value": "4" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0x93205e5ec2238371d751ff89a8efb12b2436c1faf8152049b02fdfd579face26", - "value": "5" + "id": "0xdc7a601e37bef86cb0009a0d5d2928aad82195410365760ce3af99ba796ccf93", + "value": "2" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } }, { "contents": { "json": { - "id": "0xb6e1ac11a0567db347db4c5ce1d1c609d66d1a799045352697d1ae195ed7e57c", - "value": "6" + "id": "0xffe2c100374e1a53f93cd6d37087ec6db3d95628912104be12dc2b63b95e31ea", + "value": "3" }, "type": { - "repr": "0x87557ebcf30086c52715468f621fd094af7efdc79a0ed647ede304eee0e2006d::M1::Object" + "repr": "0x89f4878c607cda08a8c7e8364507a8800574457a6101a928e5f731bdc4952962::M1::Object" } } } diff --git a/crates/iota-graphql-e2e-tests/tests/consistency/tx_address_objects.move b/crates/iota-graphql-e2e-tests/tests/consistency/tx_address_objects.move index 634d449bf17..d9566e43800 100644 --- a/crates/iota-graphql-e2e-tests/tests/consistency/tx_address_objects.move +++ b/crates/iota-graphql-e2e-tests/tests/consistency/tx_address_objects.move @@ -13,7 +13,7 @@ // 2 | (3, 3, 3) // 3 | (4, 4, 4, 4) -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/datetime/datetime.move b/crates/iota-graphql-e2e-tests/tests/datetime/datetime.move index abd0f649abe..9a00cdce3d9 100644 --- a/crates/iota-graphql-e2e-tests/tests/datetime/datetime.move +++ b/crates/iota-graphql-e2e-tests/tests/datetime/datetime.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --simulator +//# init --protocol-version 3 --simulator //# create-checkpoint diff --git a/crates/iota-graphql-e2e-tests/tests/epoch/chain_identifier.exp b/crates/iota-graphql-e2e-tests/tests/epoch/chain_identifier.exp index 7bbd7811f87..8ec6b0f5856 100644 --- a/crates/iota-graphql-e2e-tests/tests/epoch/chain_identifier.exp +++ b/crates/iota-graphql-e2e-tests/tests/epoch/chain_identifier.exp @@ -11,6 +11,6 @@ task 2, lines 10-13: //# run-graphql Response: { "data": { - "chainIdentifier": "e2a97689" + "chainIdentifier": "924c496c" } } diff --git a/crates/iota-graphql-e2e-tests/tests/epoch/chain_identifier.move b/crates/iota-graphql-e2e-tests/tests/epoch/chain_identifier.move index 88638c14ebc..2e0f4e5badc 100644 --- a/crates/iota-graphql-e2e-tests/tests/epoch/chain_identifier.move +++ b/crates/iota-graphql-e2e-tests/tests/epoch/chain_identifier.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --simulator --accounts C +//# init --protocol-version 3 --simulator --accounts C //# create-checkpoint diff --git a/crates/iota-graphql-e2e-tests/tests/epoch/epoch.exp b/crates/iota-graphql-e2e-tests/tests/epoch/epoch.exp index 30eef67e10a..32ba7fb5d95 100644 --- a/crates/iota-graphql-e2e-tests/tests/epoch/epoch.exp +++ b/crates/iota-graphql-e2e-tests/tests/epoch/epoch.exp @@ -21,7 +21,7 @@ gas summary: computation_cost: 1000000, computation_cost_burned: 1000000, storag task 4, lines 17-19: //# run 0x3::iota_system::request_add_stake --args object(0x5) object(3,0) @validator_0 --sender C -events: Event { package_id: iota_system, transaction_module: Identifier("iota_system"), sender: C, type_: StructTag { address: iota_system, module: Identifier("validator"), name: Identifier("StakingRequestEvent"), type_params: [] }, contents: [145, 201, 73, 32, 135, 23, 121, 37, 7, 101, 40, 103, 216, 99, 155, 51, 118, 111, 115, 220, 44, 253, 144, 143, 80, 103, 133, 240, 58, 235, 137, 202, 175, 163, 158, 79, 0, 218, 226, 120, 249, 119, 199, 198, 147, 10, 94, 44, 118, 232, 93, 23, 165, 38, 215, 36, 187, 206, 15, 184, 31, 176, 125, 76, 140, 202, 78, 28, 224, 186, 89, 4, 206, 166, 29, 249, 36, 45, 162, 247, 210, 158, 62, 243, 40, 251, 126, 192, 124, 8, 107, 59, 244, 124, 166, 26, 1, 0, 0, 0, 0, 0, 0, 0, 0, 228, 11, 84, 2, 0, 0, 0] } +events: Event { package_id: iota_system, transaction_module: Identifier("iota_system"), sender: C, type_: StructTag { address: iota_system, module: Identifier("validator"), name: Identifier("StakingRequestEvent"), type_params: [] }, contents: [217, 25, 175, 239, 57, 244, 176, 32, 209, 232, 109, 171, 147, 246, 3, 15, 205, 197, 172, 21, 36, 143, 47, 100, 240, 226, 221, 254, 42, 3, 157, 155, 175, 163, 158, 79, 0, 218, 226, 120, 249, 119, 199, 198, 147, 10, 94, 44, 118, 232, 93, 23, 165, 38, 215, 36, 187, 206, 15, 184, 31, 176, 125, 76, 140, 202, 78, 28, 224, 186, 89, 4, 206, 166, 29, 249, 36, 45, 162, 247, 210, 158, 62, 243, 40, 251, 126, 192, 124, 8, 107, 59, 244, 124, 166, 26, 1, 0, 0, 0, 0, 0, 0, 0, 0, 228, 11, 84, 2, 0, 0, 0] } created: object(4,0) mutated: object(_), 0x0000000000000000000000000000000000000000000000000000000000000005, object(0,0) deleted: object(3,0) @@ -67,7 +67,7 @@ Response: { ] }, "validatorCandidatesSize": 0, - "inactivePoolsId": "0x48b495a9f2c67e307895dec692c61488d4a82e2a1219fc6d8a2f56b5adb66dce" + "inactivePoolsId": "0x697325617c22344bd6c7921686e670bcbf688bbb32f71908316e918f28b34df4" }, "totalGasFees": "1000000", "totalStakeRewards": "767000000000000", @@ -81,13 +81,13 @@ Response: { "kind": { "__typename": "ProgrammableTransactionBlock" }, - "digest": "7SJtdFybBPY6aTpoRw4y1Z7vShCUUGqqJUDmL5o64CDu" + "digest": "8RUGZcfVT7KR2y4mF8eJ645r1sMkwNhFiAjiAzZ5ZQXW" }, { "kind": { "__typename": "EndOfEpochTransaction" }, - "digest": "9P5t7GdBVgLofELfhVpgeTJ4JoRP3TWYWZfBWrZwrcyV" + "digest": "HxmCwBJxYq7XYbLvt2MQKQAyvwGwue9yyDNSZryxF58m" } ] } diff --git a/crates/iota-graphql-e2e-tests/tests/epoch/epoch.move b/crates/iota-graphql-e2e-tests/tests/epoch/epoch.move index 07950a86ece..4ff9857f984 100644 --- a/crates/iota-graphql-e2e-tests/tests/epoch/epoch.move +++ b/crates/iota-graphql-e2e-tests/tests/epoch/epoch.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --simulator --accounts C +//# init --protocol-version 3 --simulator --accounts C // TODO: Short term hack to get around indexer epoch issue diff --git a/crates/iota-graphql-e2e-tests/tests/epoch/protocol_configs.exp b/crates/iota-graphql-e2e-tests/tests/epoch/protocol_configs.exp index def241be0f3..4328ecd9028 100644 --- a/crates/iota-graphql-e2e-tests/tests/epoch/protocol_configs.exp +++ b/crates/iota-graphql-e2e-tests/tests/epoch/protocol_configs.exp @@ -12,7 +12,7 @@ task 2, lines 9-20: Response: { "data": { "protocolConfig": { - "protocolVersion": 1, + "protocolVersion": 3, "config": { "value": "128" }, diff --git a/crates/iota-graphql-e2e-tests/tests/epoch/protocol_configs.move b/crates/iota-graphql-e2e-tests/tests/epoch/protocol_configs.move index bd095f947e5..774e8bbbcc9 100644 --- a/crates/iota-graphql-e2e-tests/tests/epoch/protocol_configs.move +++ b/crates/iota-graphql-e2e-tests/tests/epoch/protocol_configs.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --simulator --accounts C +//# init --protocol-version 3 --simulator --accounts C //# create-checkpoint diff --git a/crates/iota-graphql-e2e-tests/tests/epoch/system_state.exp b/crates/iota-graphql-e2e-tests/tests/epoch/system_state.exp index 02077205f22..44275610639 100644 --- a/crates/iota-graphql-e2e-tests/tests/epoch/system_state.exp +++ b/crates/iota-graphql-e2e-tests/tests/epoch/system_state.exp @@ -21,7 +21,7 @@ gas summary: computation_cost: 1000000, computation_cost_burned: 1000000, storag task 4, line 19: //# run 0x3::iota_system::request_add_stake --args object(0x5) object(3,0) @validator_0 --sender C -events: Event { package_id: iota_system, transaction_module: Identifier("iota_system"), sender: C, type_: StructTag { address: iota_system, module: Identifier("validator"), name: Identifier("StakingRequestEvent"), type_params: [] }, contents: [82, 156, 67, 66, 179, 182, 139, 22, 41, 95, 18, 92, 229, 180, 139, 208, 171, 130, 171, 230, 62, 173, 24, 194, 70, 211, 255, 135, 134, 155, 120, 225, 175, 163, 158, 79, 0, 218, 226, 120, 249, 119, 199, 198, 147, 10, 94, 44, 118, 232, 93, 23, 165, 38, 215, 36, 187, 206, 15, 184, 31, 176, 125, 76, 140, 202, 78, 28, 224, 186, 89, 4, 206, 166, 29, 249, 36, 45, 162, 247, 210, 158, 62, 243, 40, 251, 126, 192, 124, 8, 107, 59, 244, 124, 166, 26, 1, 0, 0, 0, 0, 0, 0, 0, 0, 228, 11, 84, 2, 0, 0, 0] } +events: Event { package_id: iota_system, transaction_module: Identifier("iota_system"), sender: C, type_: StructTag { address: iota_system, module: Identifier("validator"), name: Identifier("StakingRequestEvent"), type_params: [] }, contents: [92, 255, 40, 237, 198, 73, 205, 67, 209, 145, 49, 106, 217, 73, 88, 110, 184, 125, 246, 24, 38, 172, 47, 25, 61, 71, 153, 80, 215, 118, 139, 200, 175, 163, 158, 79, 0, 218, 226, 120, 249, 119, 199, 198, 147, 10, 94, 44, 118, 232, 93, 23, 165, 38, 215, 36, 187, 206, 15, 184, 31, 176, 125, 76, 140, 202, 78, 28, 224, 186, 89, 4, 206, 166, 29, 249, 36, 45, 162, 247, 210, 158, 62, 243, 40, 251, 126, 192, 124, 8, 107, 59, 244, 124, 166, 26, 1, 0, 0, 0, 0, 0, 0, 0, 0, 228, 11, 84, 2, 0, 0, 0] } created: object(4,0) mutated: object(_), 0x0000000000000000000000000000000000000000000000000000000000000005, object(0,0) deleted: object(3,0) @@ -61,7 +61,7 @@ Epoch advanced: 3 task 12, line 37: //# run 0x3::iota_system::request_withdraw_stake --args object(0x5) object(4,0) --sender C -events: Event { package_id: iota_system, transaction_module: Identifier("iota_system"), sender: C, type_: StructTag { address: iota_system, module: Identifier("validator"), name: Identifier("UnstakingRequestEvent"), type_params: [] }, contents: [82, 156, 67, 66, 179, 182, 139, 22, 41, 95, 18, 92, 229, 180, 139, 208, 171, 130, 171, 230, 62, 173, 24, 194, 70, 211, 255, 135, 134, 155, 120, 225, 175, 163, 158, 79, 0, 218, 226, 120, 249, 119, 199, 198, 147, 10, 94, 44, 118, 232, 93, 23, 165, 38, 215, 36, 187, 206, 15, 184, 31, 176, 125, 76, 140, 202, 78, 28, 224, 186, 89, 4, 206, 166, 29, 249, 36, 45, 162, 247, 210, 158, 62, 243, 40, 251, 126, 192, 124, 8, 107, 59, 244, 124, 166, 26, 2, 0, 0, 0, 0, 0, 0, 0, 4, 0, 0, 0, 0, 0, 0, 0, 0, 228, 11, 84, 2, 0, 0, 0, 12, 33, 189, 38, 1, 0, 0, 0] } +events: Event { package_id: iota_system, transaction_module: Identifier("iota_system"), sender: C, type_: StructTag { address: iota_system, module: Identifier("validator"), name: Identifier("UnstakingRequestEvent"), type_params: [] }, contents: [92, 255, 40, 237, 198, 73, 205, 67, 209, 145, 49, 106, 217, 73, 88, 110, 184, 125, 246, 24, 38, 172, 47, 25, 61, 71, 153, 80, 215, 118, 139, 200, 175, 163, 158, 79, 0, 218, 226, 120, 249, 119, 199, 198, 147, 10, 94, 44, 118, 232, 93, 23, 165, 38, 215, 36, 187, 206, 15, 184, 31, 176, 125, 76, 140, 202, 78, 28, 224, 186, 89, 4, 206, 166, 29, 249, 36, 45, 162, 247, 210, 158, 62, 243, 40, 251, 126, 192, 124, 8, 107, 59, 244, 124, 166, 26, 2, 0, 0, 0, 0, 0, 0, 0, 4, 0, 0, 0, 0, 0, 0, 0, 0, 228, 11, 84, 2, 0, 0, 0, 12, 33, 189, 38, 1, 0, 0, 0] } created: object(12,0) mutated: object(_), 0x0000000000000000000000000000000000000000000000000000000000000005, object(0,0) deleted: object(4,0) diff --git a/crates/iota-graphql-e2e-tests/tests/epoch/system_state.move b/crates/iota-graphql-e2e-tests/tests/epoch/system_state.move index a97d93100ae..b76ac55ef7e 100644 --- a/crates/iota-graphql-e2e-tests/tests/epoch/system_state.move +++ b/crates/iota-graphql-e2e-tests/tests/epoch/system_state.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --simulator --accounts C --custom-validator-account +//# init --protocol-version 3 --simulator --accounts C --custom-validator-account // Run a few transactions and check that the system state storage fund is correctly reported // for historical epochs diff --git a/crates/iota-graphql-e2e-tests/tests/errors/clever_errors.exp b/crates/iota-graphql-e2e-tests/tests/errors/clever_errors.exp index d56278abb77..95dcb6214a5 100644 --- a/crates/iota-graphql-e2e-tests/tests/errors/clever_errors.exp +++ b/crates/iota-graphql-e2e-tests/tests/errors/clever_errors.exp @@ -77,67 +77,67 @@ Response: { { "effects": { "status": "FAILURE", - "errors": "Error in 1st command, from '0x322ea0fab60fef0ce07a128ba221506400816291c3e6352c547b7af4cafa9371::m::callU8' (line 30), abort 'ImAU8': 0" + "errors": "Error in 1st command, from '0x03d41831cb41d89bd851fc0408c68cbfd8eae0504f5e7872efde32591e06f7fb::m::callU8' (line 30), abort 'ImAU8': 0" } }, { "effects": { "status": "FAILURE", - "errors": "Error in 1st command, from '0x322ea0fab60fef0ce07a128ba221506400816291c3e6352c547b7af4cafa9371::m::callU16' (line 33), abort 'ImAU16': 1" + "errors": "Error in 1st command, from '0x03d41831cb41d89bd851fc0408c68cbfd8eae0504f5e7872efde32591e06f7fb::m::callU16' (line 33), abort 'ImAU16': 1" } }, { "effects": { "status": "FAILURE", - "errors": "Error in 1st command, from '0x322ea0fab60fef0ce07a128ba221506400816291c3e6352c547b7af4cafa9371::m::callU32' (line 36), abort 'ImAU32': 2" + "errors": "Error in 1st command, from '0x03d41831cb41d89bd851fc0408c68cbfd8eae0504f5e7872efde32591e06f7fb::m::callU32' (line 36), abort 'ImAU32': 2" } }, { "effects": { "status": "FAILURE", - "errors": "Error in 1st command, from '0x322ea0fab60fef0ce07a128ba221506400816291c3e6352c547b7af4cafa9371::m::callU64' (line 39), abort 'ImAU64': 3" + "errors": "Error in 1st command, from '0x03d41831cb41d89bd851fc0408c68cbfd8eae0504f5e7872efde32591e06f7fb::m::callU64' (line 39), abort 'ImAU64': 3" } }, { "effects": { "status": "FAILURE", - "errors": "Error in 1st command, from '0x322ea0fab60fef0ce07a128ba221506400816291c3e6352c547b7af4cafa9371::m::callU128' (line 42), abort 'ImAU128': 4" + "errors": "Error in 1st command, from '0x03d41831cb41d89bd851fc0408c68cbfd8eae0504f5e7872efde32591e06f7fb::m::callU128' (line 42), abort 'ImAU128': 4" } }, { "effects": { "status": "FAILURE", - "errors": "Error in 1st command, from '0x322ea0fab60fef0ce07a128ba221506400816291c3e6352c547b7af4cafa9371::m::callU256' (line 45), abort 'ImAU256': 5" + "errors": "Error in 1st command, from '0x03d41831cb41d89bd851fc0408c68cbfd8eae0504f5e7872efde32591e06f7fb::m::callU256' (line 45), abort 'ImAU256': 5" } }, { "effects": { "status": "FAILURE", - "errors": "Error in 1st command, from '0x322ea0fab60fef0ce07a128ba221506400816291c3e6352c547b7af4cafa9371::m::callAddress' (line 48), abort 'ImAnAddress': 0x0000000000000000000000000000000000000000000000000000000000000006" + "errors": "Error in 1st command, from '0x03d41831cb41d89bd851fc0408c68cbfd8eae0504f5e7872efde32591e06f7fb::m::callAddress' (line 48), abort 'ImAnAddress': 0x0000000000000000000000000000000000000000000000000000000000000006" } }, { "effects": { "status": "FAILURE", - "errors": "Error in 1st command, from '0x322ea0fab60fef0ce07a128ba221506400816291c3e6352c547b7af4cafa9371::m::callString' (line 51), abort 'ImAString': This is a string" + "errors": "Error in 1st command, from '0x03d41831cb41d89bd851fc0408c68cbfd8eae0504f5e7872efde32591e06f7fb::m::callString' (line 51), abort 'ImAString': This is a string" } }, { "effects": { "status": "FAILURE", - "errors": "Error in 1st command, from '0x322ea0fab60fef0ce07a128ba221506400816291c3e6352c547b7af4cafa9371::m::callU64vec' (line 54), abort 'ImNotAString': BQEAAAAAAAAAAgAAAAAAAAADAAAAAAAAAAQAAAAAAAAABQAAAAAAAAA=" + "errors": "Error in 1st command, from '0x03d41831cb41d89bd851fc0408c68cbfd8eae0504f5e7872efde32591e06f7fb::m::callU64vec' (line 54), abort 'ImNotAString': BQEAAAAAAAAAAgAAAAAAAAADAAAAAAAAAAQAAAAAAAAABQAAAAAAAAA=" } }, { "effects": { "status": "FAILURE", - "errors": "Error in 1st command, from '0x322ea0fab60fef0ce07a128ba221506400816291c3e6352c547b7af4cafa9371::m::normalAbort' (instruction 1), abort code: 0" + "errors": "Error in 1st command, from '0x03d41831cb41d89bd851fc0408c68cbfd8eae0504f5e7872efde32591e06f7fb::m::normalAbort' (instruction 1), abort code: 0" } }, { "effects": { "status": "FAILURE", - "errors": "Error in 1st command, from '0x322ea0fab60fef0ce07a128ba221506400816291c3e6352c547b7af4cafa9371::m::assertLineNo' (line 60)" + "errors": "Error in 1st command, from '0x03d41831cb41d89bd851fc0408c68cbfd8eae0504f5e7872efde32591e06f7fb::m::assertLineNo' (line 60)" } } ] @@ -248,7 +248,7 @@ Response: { }, "errors": [ { - "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: fae1ca9e0539e57d226f63569906b6c6aaf1b87bc59de378ddabbdff9d90792f", + "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: ed305684cc73241a5b1625abb0a973baa0e7dacdd7759308a83a98a916f758b0", "locations": [ { "line": 6, @@ -264,7 +264,7 @@ Response: { ] }, { - "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: fae1ca9e0539e57d226f63569906b6c6aaf1b87bc59de378ddabbdff9d90792f", + "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: ed305684cc73241a5b1625abb0a973baa0e7dacdd7759308a83a98a916f758b0", "locations": [ { "line": 6, @@ -280,7 +280,7 @@ Response: { ] }, { - "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: fae1ca9e0539e57d226f63569906b6c6aaf1b87bc59de378ddabbdff9d90792f", + "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: ed305684cc73241a5b1625abb0a973baa0e7dacdd7759308a83a98a916f758b0", "locations": [ { "line": 6, @@ -296,7 +296,7 @@ Response: { ] }, { - "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: fae1ca9e0539e57d226f63569906b6c6aaf1b87bc59de378ddabbdff9d90792f", + "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: ed305684cc73241a5b1625abb0a973baa0e7dacdd7759308a83a98a916f758b0", "locations": [ { "line": 6, @@ -312,7 +312,7 @@ Response: { ] }, { - "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: fae1ca9e0539e57d226f63569906b6c6aaf1b87bc59de378ddabbdff9d90792f", + "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: ed305684cc73241a5b1625abb0a973baa0e7dacdd7759308a83a98a916f758b0", "locations": [ { "line": 6, @@ -328,7 +328,7 @@ Response: { ] }, { - "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: fae1ca9e0539e57d226f63569906b6c6aaf1b87bc59de378ddabbdff9d90792f", + "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: ed305684cc73241a5b1625abb0a973baa0e7dacdd7759308a83a98a916f758b0", "locations": [ { "line": 6, @@ -344,7 +344,7 @@ Response: { ] }, { - "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: fae1ca9e0539e57d226f63569906b6c6aaf1b87bc59de378ddabbdff9d90792f", + "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: ed305684cc73241a5b1625abb0a973baa0e7dacdd7759308a83a98a916f758b0", "locations": [ { "line": 6, @@ -360,7 +360,7 @@ Response: { ] }, { - "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: fae1ca9e0539e57d226f63569906b6c6aaf1b87bc59de378ddabbdff9d90792f", + "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: ed305684cc73241a5b1625abb0a973baa0e7dacdd7759308a83a98a916f758b0", "locations": [ { "line": 6, @@ -376,7 +376,7 @@ Response: { ] }, { - "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: fae1ca9e0539e57d226f63569906b6c6aaf1b87bc59de378ddabbdff9d90792f", + "message": "Internal error occurred while processing request: Error resolving Move location: Linkage not found for package: ed305684cc73241a5b1625abb0a973baa0e7dacdd7759308a83a98a916f758b0", "locations": [ { "line": 6, diff --git a/crates/iota-graphql-e2e-tests/tests/errors/clever_errors.move b/crates/iota-graphql-e2e-tests/tests/errors/clever_errors.move index 80f4b884b04..cf601d53604 100644 --- a/crates/iota-graphql-e2e-tests/tests/errors/clever_errors.move +++ b/crates/iota-graphql-e2e-tests/tests/errors/clever_errors.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 P1=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses P0=0x0 P1=0x0 --accounts A --simulator //# publish --upgradeable --sender A module P0::m { diff --git a/crates/iota-graphql-e2e-tests/tests/errors/clever_errors_in_macros.exp b/crates/iota-graphql-e2e-tests/tests/errors/clever_errors_in_macros.exp index a36d804579a..becf8efed3e 100644 --- a/crates/iota-graphql-e2e-tests/tests/errors/clever_errors_in_macros.exp +++ b/crates/iota-graphql-e2e-tests/tests/errors/clever_errors_in_macros.exp @@ -37,19 +37,19 @@ Response: { { "effects": { "status": "FAILURE", - "errors": "Error in 1st command, from '0x11dc3ad690a3827513133bf2891ebfe4f51fc934651fb9128b788ca564d998e1::m::t_a' (line 21)" + "errors": "Error in 1st command, from '0x44a9d67e77d63fda2134bb70abc66305c2efdb7fa390636d62e695a7ba6cb1a0::m::t_a' (line 21)" } }, { "effects": { "status": "FAILURE", - "errors": "Error in 1st command, from '0x11dc3ad690a3827513133bf2891ebfe4f51fc934651fb9128b788ca564d998e1::m::t_calls_a' (line 24)" + "errors": "Error in 1st command, from '0x44a9d67e77d63fda2134bb70abc66305c2efdb7fa390636d62e695a7ba6cb1a0::m::t_calls_a' (line 24)" } }, { "effects": { "status": "FAILURE", - "errors": "Error in 1st command, from '0x11dc3ad690a3827513133bf2891ebfe4f51fc934651fb9128b788ca564d998e1::m::t_const_assert' (line 10), abort 'EMsg': This is a string" + "errors": "Error in 1st command, from '0x44a9d67e77d63fda2134bb70abc66305c2efdb7fa390636d62e695a7ba6cb1a0::m::t_const_assert' (line 10), abort 'EMsg': This is a string" } } ] diff --git a/crates/iota-graphql-e2e-tests/tests/errors/clever_errors_in_macros.move b/crates/iota-graphql-e2e-tests/tests/errors/clever_errors_in_macros.move index 54c0c991230..e74f2272b01 100644 --- a/crates/iota-graphql-e2e-tests/tests/errors/clever_errors_in_macros.move +++ b/crates/iota-graphql-e2e-tests/tests/errors/clever_errors_in_macros.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses P0=0x0 --accounts A --simulator //# publish --sender A module P0::m { diff --git a/crates/iota-graphql-e2e-tests/tests/event_connection/combo_filter_error.move b/crates/iota-graphql-e2e-tests/tests/event_connection/combo_filter_error.move index 63e5f295316..26e79074e97 100644 --- a/crates/iota-graphql-e2e-tests/tests/event_connection/combo_filter_error.move +++ b/crates/iota-graphql-e2e-tests/tests/event_connection/combo_filter_error.move @@ -5,7 +5,7 @@ // Tests that fetching events filtered on both emitting module and event would result // in an error. -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/event_connection/event_connection.exp b/crates/iota-graphql-e2e-tests/tests/event_connection/event_connection.exp index 20ea2fdff0d..6e2735a0628 100644 --- a/crates/iota-graphql-e2e-tests/tests/event_connection/event_connection.exp +++ b/crates/iota-graphql-e2e-tests/tests/event_connection/event_connection.exp @@ -62,7 +62,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M1::EventA" + "repr": "0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -80,7 +80,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M1::EventB<0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M1::Object>" + "repr": "0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M1::EventB<0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M1::Object>" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -98,7 +98,7 @@ Response: { "name": "M2" }, "type": { - "repr": "0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M2::EventA" + "repr": "0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M2::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -116,7 +116,7 @@ Response: { "name": "M2" }, "type": { - "repr": "0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M2::EventB<0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M2::Object>" + "repr": "0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M2::EventB<0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M2::Object>" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -145,7 +145,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M1::EventA" + "repr": "0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -163,7 +163,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M1::EventB<0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M1::Object>" + "repr": "0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M1::EventB<0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M1::Object>" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -181,7 +181,7 @@ Response: { "name": "M2" }, "type": { - "repr": "0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M2::EventA" + "repr": "0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M2::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -199,7 +199,7 @@ Response: { "name": "M2" }, "type": { - "repr": "0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M2::EventB<0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M2::Object>" + "repr": "0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M2::EventB<0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M2::Object>" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -228,7 +228,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M1::EventA" + "repr": "0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -246,7 +246,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M1::EventB<0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M1::Object>" + "repr": "0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M1::EventB<0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M1::Object>" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -275,7 +275,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M1::EventA" + "repr": "0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -304,7 +304,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M1::EventB<0x355db42904a277506aa6b010760f2493218951e292423698a0f1cda4fca6626c::M1::Object>" + "repr": "0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M1::EventB<0xf79e299296162171b715da0d4813d40390ab12099666db8022a3eeab8b4c0896::M1::Object>" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" diff --git a/crates/iota-graphql-e2e-tests/tests/event_connection/event_connection.move b/crates/iota-graphql-e2e-tests/tests/event_connection/event_connection.move index e75def35ee1..475662bf2fe 100644 --- a/crates/iota-graphql-e2e-tests/tests/event_connection/event_connection.move +++ b/crates/iota-graphql-e2e-tests/tests/event_connection/event_connection.move @@ -10,7 +10,7 @@ // Verifies correct event when filtered for Test::M1::EventB // Verifies error when filtered on sender, package, module and event type with generics and < -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/event_connection/nested_emit_event.exp b/crates/iota-graphql-e2e-tests/tests/event_connection/nested_emit_event.exp index 10fe278d63e..5048dae8740 100644 --- a/crates/iota-graphql-e2e-tests/tests/event_connection/nested_emit_event.exp +++ b/crates/iota-graphql-e2e-tests/tests/event_connection/nested_emit_event.exp @@ -30,7 +30,7 @@ Response: { "name": "M3" }, "type": { - "repr": "0x4ce60f0572bedc153aaa9b5515c1f8364b76fead9bd0c01dc9e5c8c39846d156::M1::EventA" + "repr": "0x3317437a9fe9c78bda693c85b8a7227ae260153ddaa28e58a7906237bf227d84::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -56,7 +56,7 @@ Response: { "name": "M3" }, "type": { - "repr": "0x4ce60f0572bedc153aaa9b5515c1f8364b76fead9bd0c01dc9e5c8c39846d156::M1::EventA" + "repr": "0x3317437a9fe9c78bda693c85b8a7227ae260153ddaa28e58a7906237bf227d84::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -102,7 +102,7 @@ Response: { "name": "M3" }, "type": { - "repr": "0x4ce60f0572bedc153aaa9b5515c1f8364b76fead9bd0c01dc9e5c8c39846d156::M1::EventA" + "repr": "0x3317437a9fe9c78bda693c85b8a7227ae260153ddaa28e58a7906237bf227d84::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" diff --git a/crates/iota-graphql-e2e-tests/tests/event_connection/nested_emit_event.move b/crates/iota-graphql-e2e-tests/tests/event_connection/nested_emit_event.move index 4f495dd769c..60b882f1fe9 100644 --- a/crates/iota-graphql-e2e-tests/tests/event_connection/nested_emit_event.move +++ b/crates/iota-graphql-e2e-tests/tests/event_connection/nested_emit_event.move @@ -6,7 +6,7 @@ // The emitting module is where the entrypoint function is defined - // in other words, the function called by a programmable transaction block. -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/event_connection/no_filter.exp b/crates/iota-graphql-e2e-tests/tests/event_connection/no_filter.exp index 8a1728c5601..5b58043da43 100644 --- a/crates/iota-graphql-e2e-tests/tests/event_connection/no_filter.exp +++ b/crates/iota-graphql-e2e-tests/tests/event_connection/no_filter.exp @@ -33,12 +33,12 @@ Response: { "nodes": [ { "json": { - "id": "0x7d4c1fe3f73cdd9531f795b56fb6ad1cbbc4a14efff6af465256be7c52c0d404" + "id": "0x55a6b566aa7d7e5b03deef0dd47a9c124b037655f160ba09dcff14cf032a6c90" } }, { "json": { - "id": "0x7d4c1fe3f73cdd9531f795b56fb6ad1cbbc4a14efff6af465256be7c52c0d404", + "id": "0x55a6b566aa7d7e5b03deef0dd47a9c124b037655f160ba09dcff14cf032a6c90", "version": 1, "fields": { "contents": [ diff --git a/crates/iota-graphql-e2e-tests/tests/event_connection/no_filter.move b/crates/iota-graphql-e2e-tests/tests/event_connection/no_filter.move index 2c50f654d39..25349c2de0c 100644 --- a/crates/iota-graphql-e2e-tests/tests/event_connection/no_filter.move +++ b/crates/iota-graphql-e2e-tests/tests/event_connection/no_filter.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/event_connection/pagination.exp b/crates/iota-graphql-e2e-tests/tests/event_connection/pagination.exp index 6015776f3fe..de49d1769ec 100644 --- a/crates/iota-graphql-e2e-tests/tests/event_connection/pagination.exp +++ b/crates/iota-graphql-e2e-tests/tests/event_connection/pagination.exp @@ -42,7 +42,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x999de5aa7620209919ea1dc088028fffa75c36282190e063c3df58534d2842b1::M1::EventA" + "repr": "0x9ad464974f1d2c4ca76338261ac674df3da7f19d51fda13d7fad14c23d1ab9cc::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -60,7 +60,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x999de5aa7620209919ea1dc088028fffa75c36282190e063c3df58534d2842b1::M1::EventA" + "repr": "0x9ad464974f1d2c4ca76338261ac674df3da7f19d51fda13d7fad14c23d1ab9cc::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -78,7 +78,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x999de5aa7620209919ea1dc088028fffa75c36282190e063c3df58534d2842b1::M1::EventA" + "repr": "0x9ad464974f1d2c4ca76338261ac674df3da7f19d51fda13d7fad14c23d1ab9cc::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -111,7 +111,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x999de5aa7620209919ea1dc088028fffa75c36282190e063c3df58534d2842b1::M1::EventA" + "repr": "0x9ad464974f1d2c4ca76338261ac674df3da7f19d51fda13d7fad14c23d1ab9cc::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -129,7 +129,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x999de5aa7620209919ea1dc088028fffa75c36282190e063c3df58534d2842b1::M1::EventA" + "repr": "0x9ad464974f1d2c4ca76338261ac674df3da7f19d51fda13d7fad14c23d1ab9cc::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -162,7 +162,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x999de5aa7620209919ea1dc088028fffa75c36282190e063c3df58534d2842b1::M1::EventA" + "repr": "0x9ad464974f1d2c4ca76338261ac674df3da7f19d51fda13d7fad14c23d1ab9cc::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -180,7 +180,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x999de5aa7620209919ea1dc088028fffa75c36282190e063c3df58534d2842b1::M1::EventA" + "repr": "0x9ad464974f1d2c4ca76338261ac674df3da7f19d51fda13d7fad14c23d1ab9cc::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -213,7 +213,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x999de5aa7620209919ea1dc088028fffa75c36282190e063c3df58534d2842b1::M1::EventA" + "repr": "0x9ad464974f1d2c4ca76338261ac674df3da7f19d51fda13d7fad14c23d1ab9cc::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -231,7 +231,7 @@ Response: { "name": "M1" }, "type": { - "repr": "0x999de5aa7620209919ea1dc088028fffa75c36282190e063c3df58534d2842b1::M1::EventA" + "repr": "0x9ad464974f1d2c4ca76338261ac674df3da7f19d51fda13d7fad14c23d1ab9cc::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" diff --git a/crates/iota-graphql-e2e-tests/tests/event_connection/pagination.move b/crates/iota-graphql-e2e-tests/tests/event_connection/pagination.move index 514f6b954ef..005b6f6c5f4 100644 --- a/crates/iota-graphql-e2e-tests/tests/event_connection/pagination.move +++ b/crates/iota-graphql-e2e-tests/tests/event_connection/pagination.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/event_connection/tx_digest.exp b/crates/iota-graphql-e2e-tests/tests/event_connection/tx_digest.exp index 2b1087ce2ef..47701a5d734 100644 --- a/crates/iota-graphql-e2e-tests/tests/event_connection/tx_digest.exp +++ b/crates/iota-graphql-e2e-tests/tests/event_connection/tx_digest.exp @@ -37,19 +37,19 @@ Response: { "transactionBlocks": { "nodes": [ { - "digest": "CmPeQ2JKFnEKeZsYz9cYU8q57Wb1fgZxpDqYq9YinuQA" + "digest": "5XJr4MXW36JcHWpqf2PF2oAD2cfWg4bk1jQWVp6Josvj" }, { - "digest": "Cr67aCWwfbpbrQvsdhniTfA6v6wEbyTA5cwKr7MeVo6h" + "digest": "6gj6p9W2h9EVhCRKeDp6oqs24b6wKhTD4XzxqUp68gmX" }, { - "digest": "F9dhDrsiFxBWatoDNmExL7PQQW6xBzpAj8iyqbzAagXj" + "digest": "7s5GAykr3qnQKCbAbNT8peqP5Yu5bA1sJgj4bdzXrmqq" }, { - "digest": "EkH8TQ1xRv4Lj4wtD4kRuZrsPYP2vGKSk2mtbeENDGsn" + "digest": "DvfnX27xkS1oE9XuTiQbacrnMhsXshdbV9oYWx6jzkro" }, { - "digest": "9oZ3VdEUep1tbAAHbrACiQEPuRAwvXYU4m3TrViQSB5v" + "digest": "H5Y3w7yiV7N4VFNXKTNjHE6UetWxZ1dFR6uguKKmtVMh" } ] } diff --git a/crates/iota-graphql-e2e-tests/tests/event_connection/tx_digest.move b/crates/iota-graphql-e2e-tests/tests/event_connection/tx_digest.move index e6f9ea2833d..515512a008b 100644 --- a/crates/iota-graphql-e2e-tests/tests/event_connection/tx_digest.move +++ b/crates/iota-graphql-e2e-tests/tests/event_connection/tx_digest.move @@ -6,7 +6,7 @@ // Also tests that fetching events filtered on a tx digest that has events returns the correct // number of page-limit-bound nodes. -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { @@ -47,7 +47,7 @@ module Test::M1 { { # `transactionDigest` is the digest of the 4th transaction returned by # task 6 (see `tx_digest.exp`) - events(filter: {transactionDigest: "EkH8TQ1xRv4Lj4wtD4kRuZrsPYP2vGKSk2mtbeENDGsn"}) { + events(filter: {transactionDigest: "DvfnX27xkS1oE9XuTiQbacrnMhsXshdbV9oYWx6jzkro"}) { edges { cursor node { @@ -62,7 +62,7 @@ module Test::M1 { { # `transactionDigest` is the digest of the 4th transaction returned by # task 6 (see `tx_digest.exp`) - events(after: "@{cursor_0}" filter: {transactionDigest: "EkH8TQ1xRv4Lj4wtD4kRuZrsPYP2vGKSk2mtbeENDGsn"}) { + events(after: "@{cursor_0}" filter: {transactionDigest: "DvfnX27xkS1oE9XuTiQbacrnMhsXshdbV9oYWx6jzkro"}) { edges { cursor node { @@ -79,7 +79,7 @@ module Test::M1 { { # `transactionDigest` is the digest of the 4th transaction returned by # task 6 (see `tx_digest.exp`) - events(after: "@{cursor_0}" filter: {transactionDigest: "EkH8TQ1xRv4Lj4wtD4kRuZrsPYP2vGKSk2mtbeENDGsn"}) { + events(after: "@{cursor_0}" filter: {transactionDigest: "DvfnX27xkS1oE9XuTiQbacrnMhsXshdbV9oYWx6jzkro"}) { edges { cursor node { @@ -94,7 +94,7 @@ module Test::M1 { { # `transactionDigest` is the digest of the 5th transaction returned by # task 6 (see `tx_digest.exp`) - events(filter: {transactionDigest: "9oZ3VdEUep1tbAAHbrACiQEPuRAwvXYU4m3TrViQSB5v"}) { + events(filter: {transactionDigest: "H5Y3w7yiV7N4VFNXKTNjHE6UetWxZ1dFR6uguKKmtVMh"}) { edges { cursor node { @@ -108,7 +108,7 @@ module Test::M1 { { # `transactionDigest` is the digest of the 5th transaction returned by # task 6 (see `tx_digest.exp`) - events(after: "@{cursor_0}" filter: {transactionDigest: "9oZ3VdEUep1tbAAHbrACiQEPuRAwvXYU4m3TrViQSB5v"}) { + events(after: "@{cursor_0}" filter: {transactionDigest: "H5Y3w7yiV7N4VFNXKTNjHE6UetWxZ1dFR6uguKKmtVMh"}) { edges { cursor node { @@ -123,7 +123,7 @@ module Test::M1 { { # `transactionDigest` is the digest of the 4th transaction returned by # task 6 (see `tx_digest.exp`) - events(last: 10 filter: {transactionDigest: "EkH8TQ1xRv4Lj4wtD4kRuZrsPYP2vGKSk2mtbeENDGsn"}) { + events(last: 10 filter: {transactionDigest: "DvfnX27xkS1oE9XuTiQbacrnMhsXshdbV9oYWx6jzkro"}) { edges { cursor node { @@ -138,7 +138,7 @@ module Test::M1 { { # `transactionDigest` is the digest of the 4th transaction returned by # task 6 (see `tx_digest.exp`) - events(last: 10 before: "@{cursor_0}" filter: {transactionDigest: "EkH8TQ1xRv4Lj4wtD4kRuZrsPYP2vGKSk2mtbeENDGsn"}) { + events(last: 10 before: "@{cursor_0}" filter: {transactionDigest: "DvfnX27xkS1oE9XuTiQbacrnMhsXshdbV9oYWx6jzkro"}) { edges { cursor node { @@ -155,7 +155,7 @@ module Test::M1 { { # `transactionDigest` is the digest of the 4th transaction returned by # task 6 (see `tx_digest.exp`) - events(last: 10 before: "@{cursor_0}" filter: {transactionDigest: "EkH8TQ1xRv4Lj4wtD4kRuZrsPYP2vGKSk2mtbeENDGsn"}) { + events(last: 10 before: "@{cursor_0}" filter: {transactionDigest: "DvfnX27xkS1oE9XuTiQbacrnMhsXshdbV9oYWx6jzkro"}) { edges { cursor node { @@ -170,7 +170,7 @@ module Test::M1 { { # `transactionDigest` is the digest of the 5th transaction returned by # task 6 (see `tx_digest.exp`) - events(last: 10 filter: {transactionDigest: "9oZ3VdEUep1tbAAHbrACiQEPuRAwvXYU4m3TrViQSB5v"}) { + events(last: 10 filter: {transactionDigest: "H5Y3w7yiV7N4VFNXKTNjHE6UetWxZ1dFR6uguKKmtVMh"}) { edges { cursor node { @@ -184,7 +184,7 @@ module Test::M1 { { # `transactionDigest` is the digest of the 5th transaction returned by # task 6 (see `tx_digest.exp`) - events(last: 10 before: "@{cursor_0}" filter: {transactionDigest: "9oZ3VdEUep1tbAAHbrACiQEPuRAwvXYU4m3TrViQSB5v"}) { + events(last: 10 before: "@{cursor_0}" filter: {transactionDigest: "H5Y3w7yiV7N4VFNXKTNjHE6UetWxZ1dFR6uguKKmtVMh"}) { edges { cursor node { @@ -199,7 +199,7 @@ module Test::M1 { { # `transactionDigest` is the digest of the 4th transaction returned by # task 6 (see `tx_digest.exp`) - events(filter: {sender: "@{A}" transactionDigest: "EkH8TQ1xRv4Lj4wtD4kRuZrsPYP2vGKSk2mtbeENDGsn"}) { + events(filter: {sender: "@{A}" transactionDigest: "DvfnX27xkS1oE9XuTiQbacrnMhsXshdbV9oYWx6jzkro"}) { edges { cursor node { @@ -214,7 +214,7 @@ module Test::M1 { { # `transactionDigest` is the digest of the 5th transaction returned by # task 6 (see `tx_digest.exp`) - events(filter: {sender: "@{B}" transactionDigest: "9oZ3VdEUep1tbAAHbrACiQEPuRAwvXYU4m3TrViQSB5v"}) { + events(filter: {sender: "@{B}" transactionDigest: "H5Y3w7yiV7N4VFNXKTNjHE6UetWxZ1dFR6uguKKmtVMh"}) { edges { cursor node { @@ -229,7 +229,7 @@ module Test::M1 { { # `transactionDigest` is the digest of the 4th transaction returned by # task 6 (see `tx_digest.exp`) - events(filter: {sender: "@{B}" transactionDigest: "EkH8TQ1xRv4Lj4wtD4kRuZrsPYP2vGKSk2mtbeENDGsn"}) { + events(filter: {sender: "@{B}" transactionDigest: "DvfnX27xkS1oE9XuTiQbacrnMhsXshdbV9oYWx6jzkro"}) { edges { cursor node { @@ -244,7 +244,7 @@ module Test::M1 { { # `transactionDigest` is the digest of the 5th transaction returned by # task 6 (see `tx_digest.exp`) - events(filter: {sender: "@{A}" transactionDigest: "9oZ3VdEUep1tbAAHbrACiQEPuRAwvXYU4m3TrViQSB5v"}) { + events(filter: {sender: "@{A}" transactionDigest: "H5Y3w7yiV7N4VFNXKTNjHE6UetWxZ1dFR6uguKKmtVMh"}) { edges { cursor node { diff --git a/crates/iota-graphql-e2e-tests/tests/event_connection/type_filter.exp b/crates/iota-graphql-e2e-tests/tests/event_connection/type_filter.exp index 3b1903593bf..0d3a212357a 100644 --- a/crates/iota-graphql-e2e-tests/tests/event_connection/type_filter.exp +++ b/crates/iota-graphql-e2e-tests/tests/event_connection/type_filter.exp @@ -30,7 +30,7 @@ Response: { "name": "M2" }, "type": { - "repr": "0x1978d81fd93431a2b3ce6565adff1d6b3608ebbc3f81bca49672392044fa0eca::M1::EventA" + "repr": "0x45fe4a1c1c0acc7fbdd60355112b0227271f9ecb15969de0861f7322ad174def::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -72,7 +72,7 @@ Response: { "name": "M2" }, "type": { - "repr": "0x1978d81fd93431a2b3ce6565adff1d6b3608ebbc3f81bca49672392044fa0eca::M1::EventA" + "repr": "0x45fe4a1c1c0acc7fbdd60355112b0227271f9ecb15969de0861f7322ad174def::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -98,7 +98,7 @@ Response: { "name": "M2" }, "type": { - "repr": "0x1978d81fd93431a2b3ce6565adff1d6b3608ebbc3f81bca49672392044fa0eca::M2::EventB" + "repr": "0x45fe4a1c1c0acc7fbdd60355112b0227271f9ecb15969de0861f7322ad174def::M2::EventB" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -124,7 +124,7 @@ Response: { "name": "M2" }, "type": { - "repr": "0x1978d81fd93431a2b3ce6565adff1d6b3608ebbc3f81bca49672392044fa0eca::M1::EventA" + "repr": "0x45fe4a1c1c0acc7fbdd60355112b0227271f9ecb15969de0861f7322ad174def::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -139,7 +139,7 @@ Response: { "name": "M2" }, "type": { - "repr": "0x1978d81fd93431a2b3ce6565adff1d6b3608ebbc3f81bca49672392044fa0eca::M2::EventB" + "repr": "0x45fe4a1c1c0acc7fbdd60355112b0227271f9ecb15969de0861f7322ad174def::M2::EventB" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -165,7 +165,7 @@ Response: { "name": "M2" }, "type": { - "repr": "0x1978d81fd93431a2b3ce6565adff1d6b3608ebbc3f81bca49672392044fa0eca::M1::EventA" + "repr": "0x45fe4a1c1c0acc7fbdd60355112b0227271f9ecb15969de0861f7322ad174def::M1::EventA" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -180,7 +180,7 @@ Response: { "name": "M2" }, "type": { - "repr": "0x1978d81fd93431a2b3ce6565adff1d6b3608ebbc3f81bca49672392044fa0eca::M2::EventB" + "repr": "0x45fe4a1c1c0acc7fbdd60355112b0227271f9ecb15969de0861f7322ad174def::M2::EventB" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -195,7 +195,7 @@ Response: { "name": "M2" }, "type": { - "repr": "0x1978d81fd93431a2b3ce6565adff1d6b3608ebbc3f81bca49672392044fa0eca::M2::EventB" + "repr": "0x45fe4a1c1c0acc7fbdd60355112b0227271f9ecb15969de0861f7322ad174def::M2::EventB" }, "sender": { "address": "0x28f02a953f3553f51a9365593c7d4bd0643d2085f004b18c6ca9de51682b2c80" diff --git a/crates/iota-graphql-e2e-tests/tests/event_connection/type_filter.move b/crates/iota-graphql-e2e-tests/tests/event_connection/type_filter.move index c3b5d6bf01f..33de4e0cbcd 100644 --- a/crates/iota-graphql-e2e-tests/tests/event_connection/type_filter.move +++ b/crates/iota-graphql-e2e-tests/tests/event_connection/type_filter.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/event_connection/type_param_filter.exp b/crates/iota-graphql-e2e-tests/tests/event_connection/type_param_filter.exp index 034c36b518a..cf82d55558f 100644 --- a/crates/iota-graphql-e2e-tests/tests/event_connection/type_param_filter.exp +++ b/crates/iota-graphql-e2e-tests/tests/event_connection/type_param_filter.exp @@ -38,19 +38,19 @@ Response: { "transactionBlocks": { "nodes": [ { - "digest": "CmPeQ2JKFnEKeZsYz9cYU8q57Wb1fgZxpDqYq9YinuQA" + "digest": "5XJr4MXW36JcHWpqf2PF2oAD2cfWg4bk1jQWVp6Josvj" }, { - "digest": "ChGw1FpdLESefvbz4sAoni52P9jWpTAubgxkNsvV5kSk" + "digest": "2pBMcHK9WnYkufQTf4UisKPSg9Ub1dsocywVuPDqVzmD" }, { - "digest": "6muLxHj13tG23qV2eyNud8ciUrKno4NUpDFFFoGMRTNW" + "digest": "8enbCf5TacFaL4FyrZTL3mybyPUNGNzg8mLuYH5D5WgG" }, { - "digest": "5b7G3sXWboXZSKB6zQqzH3rq15Q4UqbaFRrfS9HpQR9B" + "digest": "AX1vKAJ59ZRqqswuP1XiWULhvUn8VAepSoapmtpYVpkS" }, { - "digest": "mnGPbLwe3rhbRZMokPmZXJtCtrpYzBQ6wFwJ7mrg56w" + "digest": "66oELEdw4Aj9VA751HaSEzwyVccggCW7TcgigqiSyZab" } ] } @@ -65,7 +65,7 @@ Response: { "nodes": [ { "type": { - "repr": "0x0743c995c75b4bdcf02b9585730ea00567a502031977468fdcc80bd0e1449270::M1::EventA<0x0743c995c75b4bdcf02b9585730ea00567a502031977468fdcc80bd0e1449270::M1::T1>" + "repr": "0x9a25c1cfdc5a40871cb1315bd9b7c869b2780e66f539307c2599e97d8048ae49::M1::EventA<0x9a25c1cfdc5a40871cb1315bd9b7c869b2780e66f539307c2599e97d8048ae49::M1::T1>" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -78,7 +78,7 @@ Response: { }, { "type": { - "repr": "0x0743c995c75b4bdcf02b9585730ea00567a502031977468fdcc80bd0e1449270::M1::EventA<0x0743c995c75b4bdcf02b9585730ea00567a502031977468fdcc80bd0e1449270::M1::T2>" + "repr": "0x9a25c1cfdc5a40871cb1315bd9b7c869b2780e66f539307c2599e97d8048ae49::M1::EventA<0x9a25c1cfdc5a40871cb1315bd9b7c869b2780e66f539307c2599e97d8048ae49::M1::T2>" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -91,7 +91,7 @@ Response: { }, { "type": { - "repr": "0x0743c995c75b4bdcf02b9585730ea00567a502031977468fdcc80bd0e1449270::M1::EventA<0x0743c995c75b4bdcf02b9585730ea00567a502031977468fdcc80bd0e1449270::M1::T1>" + "repr": "0x9a25c1cfdc5a40871cb1315bd9b7c869b2780e66f539307c2599e97d8048ae49::M1::EventA<0x9a25c1cfdc5a40871cb1315bd9b7c869b2780e66f539307c2599e97d8048ae49::M1::T1>" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -104,7 +104,7 @@ Response: { }, { "type": { - "repr": "0x0743c995c75b4bdcf02b9585730ea00567a502031977468fdcc80bd0e1449270::M1::EventA<0x0743c995c75b4bdcf02b9585730ea00567a502031977468fdcc80bd0e1449270::M1::T2>" + "repr": "0x9a25c1cfdc5a40871cb1315bd9b7c869b2780e66f539307c2599e97d8048ae49::M1::EventA<0x9a25c1cfdc5a40871cb1315bd9b7c869b2780e66f539307c2599e97d8048ae49::M1::T2>" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -128,7 +128,7 @@ Response: { "nodes": [ { "type": { - "repr": "0x0743c995c75b4bdcf02b9585730ea00567a502031977468fdcc80bd0e1449270::M1::EventA<0x0743c995c75b4bdcf02b9585730ea00567a502031977468fdcc80bd0e1449270::M1::T1>" + "repr": "0x9a25c1cfdc5a40871cb1315bd9b7c869b2780e66f539307c2599e97d8048ae49::M1::EventA<0x9a25c1cfdc5a40871cb1315bd9b7c869b2780e66f539307c2599e97d8048ae49::M1::T1>" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -141,7 +141,7 @@ Response: { }, { "type": { - "repr": "0x0743c995c75b4bdcf02b9585730ea00567a502031977468fdcc80bd0e1449270::M1::EventA<0x0743c995c75b4bdcf02b9585730ea00567a502031977468fdcc80bd0e1449270::M1::T1>" + "repr": "0x9a25c1cfdc5a40871cb1315bd9b7c869b2780e66f539307c2599e97d8048ae49::M1::EventA<0x9a25c1cfdc5a40871cb1315bd9b7c869b2780e66f539307c2599e97d8048ae49::M1::T1>" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" @@ -165,7 +165,7 @@ Response: { "nodes": [ { "type": { - "repr": "0x0743c995c75b4bdcf02b9585730ea00567a502031977468fdcc80bd0e1449270::M1::EventA<0x0743c995c75b4bdcf02b9585730ea00567a502031977468fdcc80bd0e1449270::M1::T2>" + "repr": "0x9a25c1cfdc5a40871cb1315bd9b7c869b2780e66f539307c2599e97d8048ae49::M1::EventA<0x9a25c1cfdc5a40871cb1315bd9b7c869b2780e66f539307c2599e97d8048ae49::M1::T2>" }, "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" diff --git a/crates/iota-graphql-e2e-tests/tests/event_connection/type_param_filter.move b/crates/iota-graphql-e2e-tests/tests/event_connection/type_param_filter.move index a8df0d44e52..ba967f74244 100644 --- a/crates/iota-graphql-e2e-tests/tests/event_connection/type_param_filter.move +++ b/crates/iota-graphql-e2e-tests/tests/event_connection/type_param_filter.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { @@ -81,7 +81,7 @@ module Test::M1 { { # `transactionDigest` is the digest of the 5th transaction returned from # task 6 (see `type_param_filter.exp`) - events(filter: {eventType: "@{Test}::M1::EventA<@{Test}::M1::T2>", transactionDigest: "mnGPbLwe3rhbRZMokPmZXJtCtrpYzBQ6wFwJ7mrg56w"}) { + events(filter: {eventType: "@{Test}::M1::EventA<@{Test}::M1::T2>", transactionDigest: "66oELEdw4Aj9VA751HaSEzwyVccggCW7TcgigqiSyZab"}) { nodes { type { repr diff --git a/crates/iota-graphql-e2e-tests/tests/events/sending_module.exp b/crates/iota-graphql-e2e-tests/tests/events/sending_module.exp new file mode 100644 index 00000000000..ff198aa7f33 --- /dev/null +++ b/crates/iota-graphql-e2e-tests/tests/events/sending_module.exp @@ -0,0 +1,70 @@ +processed 6 tasks + +init: +A: object(0,0) + +task 1, lines 8-13: +//# publish --upgradeable --sender A +created: object(1,0), object(1,1) +mutated: object(0,0) +gas summary: computation_cost: 1000000, computation_cost_burned: 1000000, storage_cost: 5312400, storage_rebate: 0, non_refundable_storage_fee: 0 + +task 2, lines 14-27: +//# upgrade --package Test --upgrade-capability 1,1 --sender A +created: object(2,0) +mutated: object(0,0), object(1,1) +gas summary: computation_cost: 1000000, computation_cost_burned: 1000000, storage_cost: 6786800, storage_rebate: 2606800, non_refundable_storage_fee: 0 + +task 3, line 28: +//# run Test::M1::emit --sender A +events: Event { package_id: Test, transaction_module: Identifier("M1"), sender: A, type_: StructTag { address: fake(1,0), module: Identifier("M0"), name: Identifier("Event"), type_params: [] }, contents: [42, 0, 0, 0, 0, 0, 0, 0] } +mutated: object(0,0) +gas summary: computation_cost: 1000000, computation_cost_burned: 1000000, storage_cost: 980400, storage_rebate: 980400, non_refundable_storage_fee: 0 + +task 4, line 30: +//# create-checkpoint +Checkpoint created: 1 + +task 5, lines 32-47: +//# run-graphql +Response: { + "data": { + "events": { + "nodes": [ + { + "sendingModule": { + "package": { + "address": "0x000000000000000000000000000000000000000000000000000000000000107a" + }, + "name": "nft" + }, + "type": { + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::display::DisplayCreated<0x000000000000000000000000000000000000000000000000000000000000107a::nft::Nft>" + } + }, + { + "sendingModule": { + "package": { + "address": "0x000000000000000000000000000000000000000000000000000000000000107a" + }, + "name": "nft" + }, + "type": { + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::display::VersionUpdated<0x000000000000000000000000000000000000000000000000000000000000107a::nft::Nft>" + } + }, + { + "sendingModule": { + "package": { + "address": "0xbf4a1cd7a1f49231d9ac48596d16f6de48758108e9f73400c9b7e5af7abff125" + }, + "name": "M1" + }, + "type": { + "repr": "0x569dde10dc72ddb81506bd86b753393967f4b76a7eb56677d6fc27aaabb1bfdb::M0::Event" + } + } + ] + } + } +} diff --git a/crates/iota-graphql-e2e-tests/tests/events/sending_module.move b/crates/iota-graphql-e2e-tests/tests/events/sending_module.move new file mode 100644 index 00000000000..db2c9b87bcc --- /dev/null +++ b/crates/iota-graphql-e2e-tests/tests/events/sending_module.move @@ -0,0 +1,47 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + + +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator + +//# publish --upgradeable --sender A +module Test::M0 { + public struct Event has copy, drop { + value: u64 + } +} +//# upgrade --package Test --upgrade-capability 1,1 --sender A +module Test::M0 { + public struct Event has copy, drop { + value: u64 + } + public fun emit() { + iota::event::emit(Event { value: 42 }) + } +} +module Test::M1 { + public fun emit() { + Test::M0::emit() + } +} +//# run Test::M1::emit --sender A + +//# create-checkpoint + +//# run-graphql +{ + events { + nodes { + sendingModule { + package { + address + } + name + } + type { + repr + } + } + } +} \ No newline at end of file diff --git a/crates/iota-graphql-e2e-tests/tests/limits/directives.exp b/crates/iota-graphql-e2e-tests/tests/limits/directives.exp index fbc8a46327c..b250f1e6ba1 100644 --- a/crates/iota-graphql-e2e-tests/tests/limits/directives.exp +++ b/crates/iota-graphql-e2e-tests/tests/limits/directives.exp @@ -73,7 +73,7 @@ task 5, lines 59-63: //# run-graphql Response: { "data": { - "chainIdentifier": "e2a97689" + "chainIdentifier": "924c496c" } } @@ -81,7 +81,7 @@ task 6, lines 65-69: //# run-graphql Response: { "data": { - "chainIdentifier": "e2a97689" + "chainIdentifier": "924c496c" } } diff --git a/crates/iota-graphql-e2e-tests/tests/limits/directives.move b/crates/iota-graphql-e2e-tests/tests/limits/directives.move index a9158fc6484..b9835e21e11 100644 --- a/crates/iota-graphql-e2e-tests/tests/limits/directives.move +++ b/crates/iota-graphql-e2e-tests/tests/limits/directives.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# run-graphql diff --git a/crates/iota-graphql-e2e-tests/tests/limits/output_node_estimation.exp b/crates/iota-graphql-e2e-tests/tests/limits/output_node_estimation.exp index b7660421b69..fd16f6bebd7 100644 --- a/crates/iota-graphql-e2e-tests/tests/limits/output_node_estimation.exp +++ b/crates/iota-graphql-e2e-tests/tests/limits/output_node_estimation.exp @@ -30,7 +30,7 @@ Response: { "edges": [ { "node": { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83" + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ" } } ] @@ -59,7 +59,7 @@ Response: { "edges": [ { "txns": { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83" + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ" } } ] @@ -91,7 +91,7 @@ Response: { "edges": [ { "txns": { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83" + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ" } } ] @@ -100,7 +100,7 @@ Response: { "edges": [ { "txns": { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83" + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ" } } ] @@ -132,7 +132,7 @@ Response: { "edges": [ { "txns": { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83" + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ" } } ] @@ -164,7 +164,7 @@ Response: { "edges": [ { "txns": { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83" + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ" } } ] @@ -190,7 +190,7 @@ Response: { "edges": [ { "txns": { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83" + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ" } } ] @@ -216,7 +216,7 @@ Response: { "edges": [ { "txns": { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83", + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ", "first": null, "last": null } @@ -243,7 +243,7 @@ Response: { "transactionBlocks": { "nodes": [ { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83", + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ", "first": null, "last": null } @@ -270,7 +270,7 @@ Response: { "edges": [ { "txns": { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83", + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ", "a": null, "b": null } @@ -324,7 +324,7 @@ Response: { "edges": [ { "node": { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83", + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ", "a": null } } @@ -350,14 +350,14 @@ Response: { "fragmentSpread": { "nodes": [ { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83" + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ" } ] }, "inlineFragment": { "nodes": [ { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83" + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ" } ] } diff --git a/crates/iota-graphql-e2e-tests/tests/limits/output_node_estimation.move b/crates/iota-graphql-e2e-tests/tests/limits/output_node_estimation.move index d8b769b881e..02f2236e85c 100644 --- a/crates/iota-graphql-e2e-tests/tests/limits/output_node_estimation.move +++ b/crates/iota-graphql-e2e-tests/tests/limits/output_node_estimation.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses A=0x42 --simulator +//# init --protocol-version 3 --addresses A=0x42 --simulator //# run-graphql --show-usage # pageInfo does not inherit connection's weights diff --git a/crates/iota-graphql-e2e-tests/tests/objects/coin.exp b/crates/iota-graphql-e2e-tests/tests/objects/coin.exp index 13c0fed33e1..ebad5352508 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/coin.exp +++ b/crates/iota-graphql-e2e-tests/tests/objects/coin.exp @@ -21,9 +21,9 @@ Response: { "iotaCoins": { "edges": [ { - "cursor": "IAUfdSfIJ+BUduck+TPi0OqPD7kIxW/IadU7pI8JDO2hAQAAAAAAAAA=", + "cursor": "IBily2iGhVVMSzcMP1UUZtum+yU5CSD6WDaBKp+2ZQDEAQAAAAAAAAA=", "node": { - "coinBalance": "30000000000000000", + "coinBalance": "299999983382000", "contents": { "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" @@ -32,7 +32,7 @@ Response: { } }, { - "cursor": "IGQSAAScq/jz2PWkB8n/2Zq/NyWuskIjmIfWJ6p2xcvnAQAAAAAAAAA=", + "cursor": "IG9h5K/B6fAjwRaUUkX/n8scYKtc6LIOKyDq49g+b0/gAQAAAAAAAAA=", "node": { "coinBalance": "300000000000000", "contents": { @@ -43,9 +43,9 @@ Response: { } }, { - "cursor": "IMafGVTG0T42MAdfRilm4G2fBf7CmEbT14ECYYflQRQxAQAAAAAAAAA=", + "cursor": "IJqDZP1WO7o1JivvOMvXGj/0Rnafmg8FbUgL7kjTUGy7AQAAAAAAAAA=", "node": { - "coinBalance": "299999983382000", + "coinBalance": "30000000000000000", "contents": { "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" @@ -58,34 +58,34 @@ Response: { "fakeCoins": { "edges": [ { - "cursor": "IDA83btPrJYvdZJSYxHTMwTntnb1td8qEwJabuS53VYOAQAAAAAAAAA=", + "cursor": "IG0zqMxTwpxTJFpyeAIwRaelE3oOFs8N8Vou2O21YZsOAQAAAAAAAAA=", "node": { - "coinBalance": "2", + "coinBalance": "3", "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0xff92cd43d27e0af392d761a8e253e684dfcc6707854e46d94774a2751877eef5::fake::FAKE>" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x6d8bf133dd0c61c8c4a18781e62eefa7a190afc4e65d5f7d2fc143086c5469bd::fake::FAKE>" } } } }, { - "cursor": "IJa3gXgkZqu5tgHRpzZI9hvBIfX+0OV/h4KlHe/fkQXjAQAAAAAAAAA=", + "cursor": "ILpFYtIteN4ELc+I7UuXbnbFlnG/ZOYvYB6x5BZgI2ESAQAAAAAAAAA=", "node": { - "coinBalance": "1", + "coinBalance": "2", "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0xff92cd43d27e0af392d761a8e253e684dfcc6707854e46d94774a2751877eef5::fake::FAKE>" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x6d8bf133dd0c61c8c4a18781e62eefa7a190afc4e65d5f7d2fc143086c5469bd::fake::FAKE>" } } } }, { - "cursor": "IPclUYS8wwbSrJwJqrO8vj9k3Cks56yvflmE3zZQNZWSAQAAAAAAAAA=", + "cursor": "IN2Z2LcN9DNNVlPNZnZraym86lnlUiNFr7nVycVJxP+nAQAAAAAAAAA=", "node": { - "coinBalance": "3", + "coinBalance": "1", "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0xff92cd43d27e0af392d761a8e253e684dfcc6707854e46d94774a2751877eef5::fake::FAKE>" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x6d8bf133dd0c61c8c4a18781e62eefa7a190afc4e65d5f7d2fc143086c5469bd::fake::FAKE>" } } } @@ -96,7 +96,7 @@ Response: { "coins": { "edges": [ { - "cursor": "IMafGVTG0T42MAdfRilm4G2fBf7CmEbT14ECYYflQRQxAQAAAAAAAAA=", + "cursor": "IBily2iGhVVMSzcMP1UUZtum+yU5CSD6WDaBKp+2ZQDEAQAAAAAAAAA=", "node": { "coinBalance": "299999983382000", "contents": { @@ -121,10 +121,10 @@ Response: { } }, { - "cursor": "eyJ0IjoiMHhmZjkyY2Q0M2QyN2UwYWYzOTJkNzYxYThlMjUzZTY4NGRmY2M2NzA3ODU0ZTQ2ZDk0Nzc0YTI3NTE4NzdlZWY1OjpmYWtlOjpGQUtFIiwiYyI6MX0", + "cursor": "eyJ0IjoiMHg2ZDhiZjEzM2RkMGM2MWM4YzRhMTg3ODFlNjJlZWZhN2ExOTBhZmM0ZTY1ZDVmN2QyZmMxNDMwODZjNTQ2OWJkOjpmYWtlOjpGQUtFIiwiYyI6MX0", "node": { "coinType": { - "repr": "0xff92cd43d27e0af392d761a8e253e684dfcc6707854e46d94774a2751877eef5::fake::FAKE" + "repr": "0x6d8bf133dd0c61c8c4a18781e62eefa7a190afc4e65d5f7d2fc143086c5469bd::fake::FAKE" }, "coinObjectCount": 3, "totalBalance": "6" @@ -142,7 +142,7 @@ Response: { "lastBalance": { "edges": [ { - "cursor": "eyJ0IjoiMHhmZjkyY2Q0M2QyN2UwYWYzOTJkNzYxYThlMjUzZTY4NGRmY2M2NzA3ODU0ZTQ2ZDk0Nzc0YTI3NTE4NzdlZWY1OjpmYWtlOjpGQUtFIiwiYyI6MX0" + "cursor": "eyJ0IjoiMHg2ZDhiZjEzM2RkMGM2MWM4YzRhMTg3ODFlNjJlZWZhN2ExOTBhZmM0ZTY1ZDVmN2QyZmMxNDMwODZjNTQ2OWJkOjpmYWtlOjpGQUtFIiwiYyI6MX0" } ] } diff --git a/crates/iota-graphql-e2e-tests/tests/objects/coin.move b/crates/iota-graphql-e2e-tests/tests/objects/coin.move index 020dd87a4ab..ed5c0d50ab1 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/coin.move +++ b/crates/iota-graphql-e2e-tests/tests/objects/coin.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses P0=0x0 --accounts A --simulator //# publish --sender A module P0::fake { diff --git a/crates/iota-graphql-e2e-tests/tests/objects/data.exp b/crates/iota-graphql-e2e-tests/tests/objects/data.exp index 2fcc77702b3..6d79d516cf0 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/data.exp +++ b/crates/iota-graphql-e2e-tests/tests/objects/data.exp @@ -36,7 +36,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0xe0ef7a7ba01308537537496e046369f6664d6a66766d3ec40418a768c98ba1e8::m::Foo" + "repr": "0xd8858c3f9c4f658f4e48b16aa2b7c51d6c2ea85100535fe1d8cea8126acbfd87::m::Foo" }, "data": { "Struct": [ @@ -44,38 +44,38 @@ Response: { "name": "id", "value": { "UID": [ - 55, - 140, - 182, - 180, + 20, + 122, 109, - 180, - 10, - 32, - 206, - 27, - 49, - 182, + 73, + 199, + 198, + 156, + 23, 57, - 108, + 244, + 61, + 52, 252, - 219, - 129, - 213, - 146, - 143, - 101, - 13, - 152, - 99, - 213, - 8, - 123, - 107, - 48, - 204, - 87, - 150 + 62, + 224, + 62, + 244, + 137, + 119, + 169, + 178, + 144, + 177, + 4, + 6, + 244, + 222, + 150, + 242, + 14, + 53, + 12 ] } }, @@ -83,38 +83,38 @@ Response: { "name": "f0", "value": { "ID": [ - 55, - 140, - 182, - 180, + 20, + 122, 109, - 180, - 10, - 32, - 206, - 27, - 49, - 182, + 73, + 199, + 198, + 156, + 23, 57, - 108, + 244, + 61, + 52, 252, - 219, - 129, - 213, - 146, - 143, - 101, - 13, - 152, - 99, - 213, - 8, - 123, - 107, - 48, - 204, - 87, - 150 + 62, + 224, + 62, + 244, + 137, + 119, + 169, + 178, + 144, + 177, + 4, + 6, + 244, + 222, + 150, + 242, + 14, + 53, + 12 ] } }, @@ -154,38 +154,38 @@ Response: { "Vector": [ { "Address": [ - 55, - 140, - 182, - 180, + 20, + 122, 109, - 180, - 10, - 32, - 206, - 27, - 49, - 182, + 73, + 199, + 198, + 156, + 23, 57, - 108, + 244, + 61, + 52, 252, - 219, - 129, - 213, - 146, - 143, - 101, - 13, - 152, - 99, - 213, - 8, - 123, - 107, - 48, - 204, - 87, - 150 + 62, + 224, + 62, + 244, + 137, + 119, + 169, + 178, + 144, + 177, + 4, + 6, + 244, + 222, + 150, + 242, + 14, + 53, + 12 ] } ] @@ -202,15 +202,15 @@ Response: { ] }, "json": { - "id": "0x378cb6b46db40a20ce1b31b6396cfcdb81d5928f650d9863d5087b6b30cc5796", - "f0": "0x378cb6b46db40a20ce1b31b6396cfcdb81d5928f650d9863d5087b6b30cc5796", + "id": "0x147a6d49c7c69c1739f43d34fc3ee03ef48977a9b290b10406f4de96f20e350c", + "f0": "0x147a6d49c7c69c1739f43d34fc3ee03ef48977a9b290b10406f4de96f20e350c", "f1": true, "f2": 42, "f3": "43", "f4": "hello", "f5": "world", "f6": [ - "0x378cb6b46db40a20ce1b31b6396cfcdb81d5928f650d9863d5087b6b30cc5796" + "0x147a6d49c7c69c1739f43d34fc3ee03ef48977a9b290b10406f4de96f20e350c" ], "f7": 44 } @@ -231,38 +231,38 @@ Response: { "name": "id", "value": { "UID": [ - 100, - 18, - 0, - 4, - 156, - 171, - 248, - 243, - 216, - 245, - 164, - 7, - 201, - 255, - 217, - 154, - 191, - 55, - 37, - 174, - 178, - 66, + 111, + 97, + 228, + 175, + 193, + 233, + 240, 35, - 152, - 135, - 214, - 39, - 170, - 118, - 197, + 193, + 22, + 148, + 82, + 69, + 255, + 159, 203, - 231 + 28, + 96, + 171, + 92, + 232, + 178, + 14, + 43, + 32, + 234, + 227, + 216, + 62, + 111, + 79, + 224 ] } }, @@ -282,7 +282,7 @@ Response: { ] }, "json": { - "id": "0x641200049cabf8f3d8f5a407c9ffd99abf3725aeb242239887d627aa76c5cbe7", + "id": "0x6f61e4afc1e9f023c116945245ff9fcb1c60ab5ce8b20e2b20eae3d83e6f4fe0", "balance": { "value": "299999988469600" } diff --git a/crates/iota-graphql-e2e-tests/tests/objects/data.move b/crates/iota-graphql-e2e-tests/tests/objects/data.move index fe7985f1b9c..1595eb88554 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/data.move +++ b/crates/iota-graphql-e2e-tests/tests/objects/data.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses P0=0x0 --accounts A --simulator //# publish module P0::m { diff --git a/crates/iota-graphql-e2e-tests/tests/objects/display.exp b/crates/iota-graphql-e2e-tests/tests/objects/display.exp index 8d17c8e4fd9..bf50eda9156 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/display.exp +++ b/crates/iota-graphql-e2e-tests/tests/objects/display.exp @@ -5,7 +5,7 @@ A: object(0,0) task 1, lines 7-131: //# publish --sender A -events: Event { package_id: Test, transaction_module: Identifier("boars"), sender: A, type_: StructTag { address: iota, module: Identifier("display"), name: Identifier("DisplayCreated"), type_params: [Struct(StructTag { address: Test, module: Identifier("boars"), name: Identifier("Boar"), type_params: [] })] }, contents: [155, 85, 136, 164, 81, 63, 109, 148, 75, 198, 60, 36, 25, 208, 95, 122, 157, 232, 18, 97, 53, 211, 101, 208, 13, 29, 31, 165, 226, 122, 237, 179] } +events: Event { package_id: Test, transaction_module: Identifier("boars"), sender: A, type_: StructTag { address: iota, module: Identifier("display"), name: Identifier("DisplayCreated"), type_params: [Struct(StructTag { address: Test, module: Identifier("boars"), name: Identifier("Boar"), type_params: [] })] }, contents: [187, 140, 105, 181, 122, 15, 118, 201, 161, 89, 13, 177, 14, 92, 210, 68, 126, 138, 208, 165, 206, 111, 45, 120, 187, 220, 213, 233, 218, 60, 254, 64] } created: object(1,0), object(1,1), object(1,2) mutated: object(0,0) gas summary: computation_cost: 1000000, computation_cost_burned: 1000000, storage_cost: 21447200, storage_rebate: 0, non_refundable_storage_fee: 0 @@ -16,7 +16,7 @@ Checkpoint created: 1 task 3, line 135: //# view-checkpoint -CheckpointSummary { epoch: 0, seq: 1, content_digest: 4zAfAVwFXhmFipWuRjudEjJYvRVd67NfPr9qBW7F9WfZ, +CheckpointSummary { epoch: 0, seq: 1, content_digest: 8KL1ykbGuB4FxZCpQFtTwojA9dAGCUhW1jqPn7Bzbdr5, epoch_rolling_gas_cost_summary: GasCostSummary { computation_cost: 1000000, computation_cost_burned: 1000000, storage_cost: 21447200, storage_rebate: 0, non_refundable_storage_fee: 0 }} task 4, line 137: @@ -27,7 +27,7 @@ gas summary: computation_cost: 1000000, computation_cost_burned: 1000000, storag task 5, line 139: //# run Test::boars::update_display_faulty --sender A --args object(1,1) -events: Event { package_id: Test, transaction_module: Identifier("boars"), sender: A, type_: StructTag { address: iota, module: Identifier("display"), name: Identifier("VersionUpdated"), type_params: [Struct(StructTag { address: Test, module: Identifier("boars"), name: Identifier("Boar"), type_params: [] })] }, contents: [155, 85, 136, 164, 81, 63, 109, 148, 75, 198, 60, 36, 25, 208, 95, 122, 157, 232, 18, 97, 53, 211, 101, 208, 13, 29, 31, 165, 226, 122, 237, 179, 1, 0, 3, 7, 118, 101, 99, 116, 111, 114, 115, 5, 123, 118, 101, 99, 125, 3, 105, 100, 100, 5, 123, 105, 100, 100, 125, 5, 110, 97, 109, 101, 101, 7, 123, 110, 97, 109, 101, 101, 125] } +events: Event { package_id: Test, transaction_module: Identifier("boars"), sender: A, type_: StructTag { address: iota, module: Identifier("display"), name: Identifier("VersionUpdated"), type_params: [Struct(StructTag { address: Test, module: Identifier("boars"), name: Identifier("Boar"), type_params: [] })] }, contents: [187, 140, 105, 181, 122, 15, 118, 201, 161, 89, 13, 177, 14, 92, 210, 68, 126, 138, 208, 165, 206, 111, 45, 120, 187, 220, 213, 233, 218, 60, 254, 64, 1, 0, 3, 7, 118, 101, 99, 116, 111, 114, 115, 5, 123, 118, 101, 99, 125, 3, 105, 100, 100, 5, 123, 105, 100, 100, 125, 5, 110, 97, 109, 101, 101, 7, 123, 110, 97, 109, 101, 101, 125] } mutated: object(0,0), object(1,1) gas summary: computation_cost: 1000000, computation_cost_burned: 1000000, storage_cost: 2926000, storage_rebate: 2637200, non_refundable_storage_fee: 0 @@ -37,7 +37,7 @@ Checkpoint created: 2 task 7, line 143: //# view-checkpoint -CheckpointSummary { epoch: 0, seq: 2, content_digest: GbyYdBVTebRQFzWA9B8nox5yAL4quujoQeZphrp9EguP, +CheckpointSummary { epoch: 0, seq: 2, content_digest: ET9fgY1A1CQcNRxWNYKBYWtvrvQgC7p3DXJhffWhGPzL, epoch_rolling_gas_cost_summary: GasCostSummary { computation_cost: 3000000, computation_cost_burned: 3000000, storage_cost: 27914800, storage_rebate: 3617600, non_refundable_storage_fee: 0 }} task 8, lines 145-158: @@ -74,7 +74,7 @@ Response: { task 9, line 160: //# run Test::boars::single_add --sender A --args object(1,1) -events: Event { package_id: Test, transaction_module: Identifier("boars"), sender: A, type_: StructTag { address: iota, module: Identifier("display"), name: Identifier("VersionUpdated"), type_params: [Struct(StructTag { address: Test, module: Identifier("boars"), name: Identifier("Boar"), type_params: [] })] }, contents: [155, 85, 136, 164, 81, 63, 109, 148, 75, 198, 60, 36, 25, 208, 95, 122, 157, 232, 18, 97, 53, 211, 101, 208, 13, 29, 31, 165, 226, 122, 237, 179, 2, 0, 4, 7, 118, 101, 99, 116, 111, 114, 115, 5, 123, 118, 101, 99, 125, 3, 105, 100, 100, 5, 123, 105, 100, 100, 125, 5, 110, 97, 109, 101, 101, 7, 123, 110, 97, 109, 101, 101, 125, 4, 110, 117, 109, 115, 6, 123, 110, 117, 109, 115, 125] } +events: Event { package_id: Test, transaction_module: Identifier("boars"), sender: A, type_: StructTag { address: iota, module: Identifier("display"), name: Identifier("VersionUpdated"), type_params: [Struct(StructTag { address: Test, module: Identifier("boars"), name: Identifier("Boar"), type_params: [] })] }, contents: [187, 140, 105, 181, 122, 15, 118, 201, 161, 89, 13, 177, 14, 92, 210, 68, 126, 138, 208, 165, 206, 111, 45, 120, 187, 220, 213, 233, 218, 60, 254, 64, 2, 0, 4, 7, 118, 101, 99, 116, 111, 114, 115, 5, 123, 118, 101, 99, 125, 3, 105, 100, 100, 5, 123, 105, 100, 100, 125, 5, 110, 97, 109, 101, 101, 7, 123, 110, 97, 109, 101, 101, 125, 4, 110, 117, 109, 115, 6, 123, 110, 117, 109, 115, 125] } mutated: object(0,0), object(1,1) gas summary: computation_cost: 1000000, computation_cost_burned: 1000000, storage_cost: 3017200, storage_rebate: 2926000, non_refundable_storage_fee: 0 @@ -84,7 +84,7 @@ Checkpoint created: 3 task 11, line 164: //# view-checkpoint -CheckpointSummary { epoch: 0, seq: 3, content_digest: 2KBZj5iLNbskohG5KL5VtdGE6onV6YEZQigAzeXZUwUc, +CheckpointSummary { epoch: 0, seq: 3, content_digest: CaK9KjzwCdVatLFbEkaQKMWaBoyY86XVeRgToHrScNFh, epoch_rolling_gas_cost_summary: GasCostSummary { computation_cost: 4000000, computation_cost_burned: 4000000, storage_cost: 30932000, storage_rebate: 6543600, non_refundable_storage_fee: 0 }} task 12, lines 166-179: @@ -126,7 +126,7 @@ Response: { task 13, line 181: //# run Test::boars::multi_add --sender A --args object(1,1) -events: Event { package_id: Test, transaction_module: Identifier("boars"), sender: A, type_: StructTag { address: iota, module: Identifier("display"), name: Identifier("VersionUpdated"), type_params: [Struct(StructTag { address: Test, module: Identifier("boars"), name: Identifier("Boar"), type_params: [] })] }, contents: [155, 85, 136, 164, 81, 63, 109, 148, 75, 198, 60, 36, 25, 208, 95, 122, 157, 232, 18, 97, 53, 211, 101, 208, 13, 29, 31, 165, 226, 122, 237, 179, 3, 0, 15, 7, 118, 101, 99, 116, 111, 114, 115, 5, 123, 118, 101, 99, 125, 3, 105, 100, 100, 5, 123, 105, 100, 100, 125, 5, 110, 97, 109, 101, 101, 7, 123, 110, 97, 109, 101, 101, 125, 4, 110, 117, 109, 115, 6, 123, 110, 117, 109, 115, 125, 5, 98, 111, 111, 108, 115, 7, 123, 98, 111, 111, 108, 115, 125, 5, 98, 117, 121, 101, 114, 7, 123, 98, 117, 121, 101, 114, 125, 4, 110, 97, 109, 101, 6, 123, 110, 97, 109, 101, 125, 7, 99, 114, 101, 97, 116, 111, 114, 9, 123, 99, 114, 101, 97, 116, 111, 114, 125, 5, 112, 114, 105, 99, 101, 7, 123, 112, 114, 105, 99, 101, 125, 11, 112, 114, 111, 106, 101, 99, 116, 95, 117, 114, 108, 58, 85, 110, 105, 113, 117, 101, 32, 66, 111, 97, 114, 32, 102, 114, 111, 109, 32, 116, 104, 101, 32, 66, 111, 97, 114, 115, 32, 99, 111, 108, 108, 101, 99, 116, 105, 111, 110, 32, 119, 105, 116, 104, 32, 123, 110, 97, 109, 101, 125, 32, 97, 110, 100, 32, 123, 105, 100, 125, 8, 98, 97, 115, 101, 95, 117, 114, 108, 32, 104, 116, 116, 112, 115, 58, 47, 47, 103, 101, 116, 45, 97, 45, 98, 111, 97, 114, 46, 99, 111, 109, 47, 123, 105, 109, 103, 95, 117, 114, 108, 125, 11, 110, 111, 95, 116, 101, 109, 112, 108, 97, 116, 101, 23, 104, 116, 116, 112, 115, 58, 47, 47, 103, 101, 116, 45, 97, 45, 98, 111, 97, 114, 46, 99, 111, 109, 47, 3, 97, 103, 101, 21, 123, 109, 101, 116, 97, 100, 97, 116, 97, 46, 110, 101, 115, 116, 101, 100, 46, 97, 103, 101, 125, 8, 102, 117, 108, 108, 95, 117, 114, 108, 10, 123, 102, 117, 108, 108, 95, 117, 114, 108, 125, 13, 101, 115, 99, 97, 112, 101, 95, 115, 121, 110, 116, 97, 120, 8, 92, 123, 110, 97, 109, 101, 92, 125] } +events: Event { package_id: Test, transaction_module: Identifier("boars"), sender: A, type_: StructTag { address: iota, module: Identifier("display"), name: Identifier("VersionUpdated"), type_params: [Struct(StructTag { address: Test, module: Identifier("boars"), name: Identifier("Boar"), type_params: [] })] }, contents: [187, 140, 105, 181, 122, 15, 118, 201, 161, 89, 13, 177, 14, 92, 210, 68, 126, 138, 208, 165, 206, 111, 45, 120, 187, 220, 213, 233, 218, 60, 254, 64, 3, 0, 15, 7, 118, 101, 99, 116, 111, 114, 115, 5, 123, 118, 101, 99, 125, 3, 105, 100, 100, 5, 123, 105, 100, 100, 125, 5, 110, 97, 109, 101, 101, 7, 123, 110, 97, 109, 101, 101, 125, 4, 110, 117, 109, 115, 6, 123, 110, 117, 109, 115, 125, 5, 98, 111, 111, 108, 115, 7, 123, 98, 111, 111, 108, 115, 125, 5, 98, 117, 121, 101, 114, 7, 123, 98, 117, 121, 101, 114, 125, 4, 110, 97, 109, 101, 6, 123, 110, 97, 109, 101, 125, 7, 99, 114, 101, 97, 116, 111, 114, 9, 123, 99, 114, 101, 97, 116, 111, 114, 125, 5, 112, 114, 105, 99, 101, 7, 123, 112, 114, 105, 99, 101, 125, 11, 112, 114, 111, 106, 101, 99, 116, 95, 117, 114, 108, 58, 85, 110, 105, 113, 117, 101, 32, 66, 111, 97, 114, 32, 102, 114, 111, 109, 32, 116, 104, 101, 32, 66, 111, 97, 114, 115, 32, 99, 111, 108, 108, 101, 99, 116, 105, 111, 110, 32, 119, 105, 116, 104, 32, 123, 110, 97, 109, 101, 125, 32, 97, 110, 100, 32, 123, 105, 100, 125, 8, 98, 97, 115, 101, 95, 117, 114, 108, 32, 104, 116, 116, 112, 115, 58, 47, 47, 103, 101, 116, 45, 97, 45, 98, 111, 97, 114, 46, 99, 111, 109, 47, 123, 105, 109, 103, 95, 117, 114, 108, 125, 11, 110, 111, 95, 116, 101, 109, 112, 108, 97, 116, 101, 23, 104, 116, 116, 112, 115, 58, 47, 47, 103, 101, 116, 45, 97, 45, 98, 111, 97, 114, 46, 99, 111, 109, 47, 3, 97, 103, 101, 21, 123, 109, 101, 116, 97, 100, 97, 116, 97, 46, 110, 101, 115, 116, 101, 100, 46, 97, 103, 101, 125, 8, 102, 117, 108, 108, 95, 117, 114, 108, 10, 123, 102, 117, 108, 108, 95, 117, 114, 108, 125, 13, 101, 115, 99, 97, 112, 101, 95, 115, 121, 110, 116, 97, 120, 8, 92, 123, 110, 97, 109, 101, 92, 125] } mutated: object(0,0), object(1,1) gas summary: computation_cost: 1000000, computation_cost_burned: 1000000, storage_cost: 5221200, storage_rebate: 3017200, non_refundable_storage_fee: 0 @@ -136,7 +136,7 @@ Checkpoint created: 4 task 15, line 185: //# view-checkpoint -CheckpointSummary { epoch: 0, seq: 4, content_digest: D6dE1yK8HSeHb7QdiXFjf9qMKPdAYHdNWBMWkb26nbJt, +CheckpointSummary { epoch: 0, seq: 4, content_digest: FCxhKY3MRduFDFzwVqwS3uNmfLcWT9VMVyx5DXgyLyxu, epoch_rolling_gas_cost_summary: GasCostSummary { computation_cost: 5000000, computation_cost_burned: 5000000, storage_cost: 36153200, storage_rebate: 9560800, non_refundable_storage_fee: 0 }} task 16, lines 187-200: @@ -195,7 +195,7 @@ Response: { }, { "key": "project_url", - "value": "Unique Boar from the Boars collection with First Boar and 0x889962831d2d9d2273e1890c6592ea21197e4e577259171eed3fe0d49e2906d7", + "value": "Unique Boar from the Boars collection with First Boar and 0x5bd1a7b5d989a691e439fb8528f532d507f97ba71a174e53339319083f15e8e8", "error": null }, { diff --git a/crates/iota-graphql-e2e-tests/tests/objects/display.move b/crates/iota-graphql-e2e-tests/tests/objects/display.move index 31963f74a61..9fe3e406192 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/display.move +++ b/crates/iota-graphql-e2e-tests/tests/objects/display.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish --sender A module Test::boars { diff --git a/crates/iota-graphql-e2e-tests/tests/objects/enum_data.exp b/crates/iota-graphql-e2e-tests/tests/objects/enum_data.exp index cb3b64ecc62..fac1fda94be 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/enum_data.exp +++ b/crates/iota-graphql-e2e-tests/tests/objects/enum_data.exp @@ -36,7 +36,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0x37f6b83e93f5faa2818235a6c07c2559447623e45725f49eaf51d21311961a88::m::Foo" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" }, "data": { "Struct": [ @@ -44,38 +44,111 @@ Response: { "name": "id", "value": { "UID": [ - 83, - 9, - 241, - 204, - 212, - 6, + 111, + 97, + 228, + 175, + 193, + 233, + 240, + 35, + 193, + 22, + 148, + 82, + 69, + 255, + 159, 203, - 57, - 197, - 134, + 28, + 96, + 171, + 92, + 232, + 178, + 14, + 43, + 32, 234, - 65, - 158, - 61, + 227, + 216, + 62, + 111, + 79, + 224 + ] + } + }, + { + "name": "balance", + "value": { + "Struct": [ + { + "name": "value", + "value": { + "Number": "299999995982800" + } + } + ] + } + } + ] + }, + "json": { + "id": "0x6f61e4afc1e9f023c116945245ff9fcb1c60ab5ce8b20e2b20eae3d83e6f4fe0", + "balance": { + "value": "299999995982800" + } + } + } + } + } + }, + { + "outputState": { + "asMoveObject": { + "contents": { + "type": { + "repr": "0xf210a282fd5df957e40bdf8c7536a035233e6ebd306132390cb10d67b4ae3d77::m::Foo" + }, + "data": { + "Struct": [ + { + "name": "id", + "value": { + "UID": [ + 235, + 17, + 204, + 36, + 83, + 225, + 192, + 186, + 116, + 44, + 86, + 207, + 239, 177, - 15, + 245, 53, - 250, - 79, - 130, - 156, - 148, - 26, - 3, - 100, - 56, - 147, - 127, - 45, - 103, - 230, - 95 + 176, + 145, + 179, + 216, + 218, + 243, + 161, + 175, + 208, + 107, + 158, + 49, + 141, + 10, + 37, + 42 ] } }, @@ -83,38 +156,38 @@ Response: { "name": "f0", "value": { "ID": [ - 83, - 9, - 241, + 235, + 17, 204, - 212, - 6, - 203, - 57, - 197, - 134, - 234, - 65, - 158, - 61, + 36, + 83, + 225, + 192, + 186, + 116, + 44, + 86, + 207, + 239, 177, - 15, + 245, 53, - 250, - 79, - 130, - 156, - 148, - 26, - 3, - 100, - 56, - 147, - 127, - 45, - 103, - 230, - 95 + 176, + 145, + 179, + 216, + 218, + 243, + 161, + 175, + 208, + 107, + 158, + 49, + 141, + 10, + 37, + 42 ] } }, @@ -154,38 +227,38 @@ Response: { "Vector": [ { "Address": [ - 83, - 9, - 241, + 235, + 17, 204, - 212, - 6, - 203, - 57, - 197, - 134, - 234, - 65, - 158, - 61, + 36, + 83, + 225, + 192, + 186, + 116, + 44, + 86, + 207, + 239, 177, - 15, + 245, 53, - 250, - 79, - 130, - 156, - 148, - 26, - 3, - 100, - 56, - 147, - 127, - 45, - 103, - 230, - 95 + 176, + 145, + 179, + 216, + 218, + 243, + 161, + 175, + 208, + 107, + 158, + 49, + 141, + 10, + 37, + 42 ] } ] @@ -252,15 +325,15 @@ Response: { ] }, "json": { - "id": "0x5309f1ccd406cb39c586ea419e3db10f35fa4f829c941a036438937f2d67e65f", - "f0": "0x5309f1ccd406cb39c586ea419e3db10f35fa4f829c941a036438937f2d67e65f", + "id": "0xeb11cc2453e1c0ba742c56cfefb1f535b091b3d8daf3a1afd06b9e318d0a252a", + "f0": "0xeb11cc2453e1c0ba742c56cfefb1f535b091b3d8daf3a1afd06b9e318d0a252a", "f1": true, "f2": 42, "f3": "43", "f4": "hello", "f5": "world", "f6": [ - "0x5309f1ccd406cb39c586ea419e3db10f35fa4f829c941a036438937f2d67e65f" + "0xeb11cc2453e1c0ba742c56cfefb1f535b091b3d8daf3a1afd06b9e318d0a252a" ], "f7": 44, "f8": { @@ -283,79 +356,6 @@ Response: { } } } - }, - { - "outputState": { - "asMoveObject": { - "contents": { - "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" - }, - "data": { - "Struct": [ - { - "name": "id", - "value": { - "UID": [ - 100, - 18, - 0, - 4, - 156, - 171, - 248, - 243, - 216, - 245, - 164, - 7, - 201, - 255, - 217, - 154, - 191, - 55, - 37, - 174, - 178, - 66, - 35, - 152, - 135, - 214, - 39, - 170, - 118, - 197, - 203, - 231 - ] - } - }, - { - "name": "balance", - "value": { - "Struct": [ - { - "name": "value", - "value": { - "Number": "299999995982800" - } - } - ] - } - } - ] - }, - "json": { - "id": "0x641200049cabf8f3d8f5a407c9ffd99abf3725aeb242239887d627aa76c5cbe7", - "balance": { - "value": "299999995982800" - } - } - } - } - } } ] } diff --git a/crates/iota-graphql-e2e-tests/tests/objects/enum_data.move b/crates/iota-graphql-e2e-tests/tests/objects/enum_data.move index 8348235b29a..655884df8e2 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/enum_data.move +++ b/crates/iota-graphql-e2e-tests/tests/objects/enum_data.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --addresses P0=0x0 P1=0x0 --accounts A --simulator --protocol-version 1 +//# init --addresses P0=0x0 P1=0x0 --accounts A --simulator --protocol-version 3 //# publish --upgradeable --sender A module P0::m { diff --git a/crates/iota-graphql-e2e-tests/tests/objects/filter_by_type.exp b/crates/iota-graphql-e2e-tests/tests/objects/filter_by_type.exp index ee3a4125263..e36b399fd15 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/filter_by_type.exp +++ b/crates/iota-graphql-e2e-tests/tests/objects/filter_by_type.exp @@ -21,7 +21,7 @@ gas summary: computation_cost: 1000000, computation_cost_burned: 1000000, storag task 4, lines 16-18: //# run 0x3::iota_system::request_add_stake --args object(0x5) object(3,0) @validator_0 --sender C -events: Event { package_id: iota_system, transaction_module: Identifier("iota_system"), sender: C, type_: StructTag { address: iota_system, module: Identifier("validator"), name: Identifier("StakingRequestEvent"), type_params: [] }, contents: [145, 201, 73, 32, 135, 23, 121, 37, 7, 101, 40, 103, 216, 99, 155, 51, 118, 111, 115, 220, 44, 253, 144, 143, 80, 103, 133, 240, 58, 235, 137, 202, 175, 163, 158, 79, 0, 218, 226, 120, 249, 119, 199, 198, 147, 10, 94, 44, 118, 232, 93, 23, 165, 38, 215, 36, 187, 206, 15, 184, 31, 176, 125, 76, 140, 202, 78, 28, 224, 186, 89, 4, 206, 166, 29, 249, 36, 45, 162, 247, 210, 158, 62, 243, 40, 251, 126, 192, 124, 8, 107, 59, 244, 124, 166, 26, 1, 0, 0, 0, 0, 0, 0, 0, 0, 228, 11, 84, 2, 0, 0, 0] } +events: Event { package_id: iota_system, transaction_module: Identifier("iota_system"), sender: C, type_: StructTag { address: iota_system, module: Identifier("validator"), name: Identifier("StakingRequestEvent"), type_params: [] }, contents: [217, 25, 175, 239, 57, 244, 176, 32, 209, 232, 109, 171, 147, 246, 3, 15, 205, 197, 172, 21, 36, 143, 47, 100, 240, 226, 221, 254, 42, 3, 157, 155, 175, 163, 158, 79, 0, 218, 226, 120, 249, 119, 199, 198, 147, 10, 94, 44, 118, 232, 93, 23, 165, 38, 215, 36, 187, 206, 15, 184, 31, 176, 125, 76, 140, 202, 78, 28, 224, 186, 89, 4, 206, 166, 29, 249, 36, 45, 162, 247, 210, 158, 62, 243, 40, 251, 126, 192, 124, 8, 107, 59, 244, 124, 166, 26, 1, 0, 0, 0, 0, 0, 0, 0, 0, 228, 11, 84, 2, 0, 0, 0] } created: object(4,0) mutated: object(_), 0x0000000000000000000000000000000000000000000000000000000000000005, object(0,0) deleted: object(3,0) @@ -134,7 +134,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field" } } } @@ -145,7 +145,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" } } } @@ -156,7 +156,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field<0x0000000000000000000000000000000000000000000000000000000000000002::object::ID,address>" } } } @@ -167,7 +167,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field" } } } @@ -178,7 +178,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field" } } } @@ -189,7 +189,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::display::Display<0x000000000000000000000000000000000000000000000000000000000000107a::nft::Nft>" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::CoinMetadata<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" } } } @@ -200,7 +200,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field<0x0000000000000000000000000000000000000000000000000000000000000002::object::ID,address>" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::display::Display<0x000000000000000000000000000000000000000000000000000000000000107a::nft::Nft>" } } } @@ -211,7 +211,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field" } } } @@ -222,7 +222,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::CoinMetadata<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" } } } @@ -233,7 +233,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field" } } } @@ -244,7 +244,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" } } } @@ -277,7 +277,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::CoinMetadata<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" } } } @@ -288,7 +288,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::CoinMetadata<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" } } } diff --git a/crates/iota-graphql-e2e-tests/tests/objects/filter_by_type.move b/crates/iota-graphql-e2e-tests/tests/objects/filter_by_type.move index 1487434869d..4f43cac4d37 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/filter_by_type.move +++ b/crates/iota-graphql-e2e-tests/tests/objects/filter_by_type.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --simulator --accounts C +//# init --protocol-version 3 --simulator --accounts C // TODO: Short term hack to get around indexer epoch issue //# create-checkpoint diff --git a/crates/iota-graphql-e2e-tests/tests/objects/pagination.exp b/crates/iota-graphql-e2e-tests/tests/objects/pagination.exp index 38db19b6da2..9c76ad922f0 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/pagination.exp +++ b/crates/iota-graphql-e2e-tests/tests/objects/pagination.exp @@ -48,19 +48,19 @@ Response: { "objects": { "edges": [ { - "cursor": "IAXGOxelPyat2ktykI7yExbqPi0zGAJwpEL4QOtk4i5JAQAAAAAAAAA=" + "cursor": "IBcnOgT0HBt/zwECt9TY8eAARlbizljFmQw20vNrznaAAQAAAAAAAAA=" }, { - "cursor": "IIhXgyfJmpDjfYgKcvjwnz8XljbqK8kWTshmtGAa1JcrAQAAAAAAAAA=" + "cursor": "IDnfdsnYm3WfTX/I86oanPj4gqpOnvKppLqzh539e51DAQAAAAAAAAA=" }, { - "cursor": "IIuOKDIL3ox4W0vwl9Iys+UJ+hOhYIxg9QXA1kvE02x1AQAAAAAAAAA=" + "cursor": "IEisGDkgq6z6Ww8fsM8DlovaXtUEQKitDSwqq47z+IUhAQAAAAAAAAA=" }, { - "cursor": "IKaEPeEks1/XBPKzfdNHH89Pmqm8lXcmxDqL2W0EAEsyAQAAAAAAAAA=" + "cursor": "IHe6f9OtHxHPsu8Ocxk0+8Le05zLLIWJCX7hPLtjxf1bAQAAAAAAAAA=" }, { - "cursor": "ILL8PT6CGnkMV6rhl3RwEvqsdGHGUNh5XsuxI1STMnrZAQAAAAAAAAA=" + "cursor": "IIOsrADQ8BewuFucSNoSsxl38aOLmCmqq1M1qt1obgy8AQAAAAAAAAA=" } ] } @@ -76,10 +76,10 @@ Response: { "objects": { "edges": [ { - "cursor": "IAXGOxelPyat2ktykI7yExbqPi0zGAJwpEL4QOtk4i5JAQAAAAAAAAA=" + "cursor": "IBcnOgT0HBt/zwECt9TY8eAARlbizljFmQw20vNrznaAAQAAAAAAAAA=" }, { - "cursor": "IIhXgyfJmpDjfYgKcvjwnz8XljbqK8kWTshmtGAa1JcrAQAAAAAAAAA=" + "cursor": "IDnfdsnYm3WfTX/I86oanPj4gqpOnvKppLqzh539e51DAQAAAAAAAAA=" } ] } @@ -95,52 +95,52 @@ Response: { "objects": { "edges": [ { - "cursor": "IAXGOxelPyat2ktykI7yExbqPi0zGAJwpEL4QOtk4i5JAQAAAAAAAAA=", + "cursor": "IBcnOgT0HBt/zwECt9TY8eAARlbizljFmQw20vNrznaAAQAAAAAAAAA=", "node": { - "address": "0x05c63b17a53f26adda4b72908ef21316ea3e2d33180270a442f840eb64e22e49" + "address": "0x17273a04f41c1b7fcf0102b7d4d8f1e0004656e2ce58c5990c36d2f36bce7680" } }, { - "cursor": "IIhXgyfJmpDjfYgKcvjwnz8XljbqK8kWTshmtGAa1JcrAQAAAAAAAAA=", + "cursor": "IDnfdsnYm3WfTX/I86oanPj4gqpOnvKppLqzh539e51DAQAAAAAAAAA=", "node": { - "address": "0x88578327c99a90e37d880a72f8f09f3f179636ea2bc9164ec866b4601ad4972b" + "address": "0x39df76c9d89b759f4d7fc8f3aa1a9cf8f882aa4e9ef2a9a4bab3879dfd7b9d43" } }, { - "cursor": "IIuOKDIL3ox4W0vwl9Iys+UJ+hOhYIxg9QXA1kvE02x1AQAAAAAAAAA=", + "cursor": "IEisGDkgq6z6Ww8fsM8DlovaXtUEQKitDSwqq47z+IUhAQAAAAAAAAA=", "node": { - "address": "0x8b8e28320bde8c785b4bf097d232b3e509fa13a1608c60f505c0d64bc4d36c75" + "address": "0x48ac183920abacfa5b0f1fb0cf03968bda5ed50440a8ad0d2c2aab8ef3f88521" } }, { - "cursor": "IKaEPeEks1/XBPKzfdNHH89Pmqm8lXcmxDqL2W0EAEsyAQAAAAAAAAA=", + "cursor": "IHe6f9OtHxHPsu8Ocxk0+8Le05zLLIWJCX7hPLtjxf1bAQAAAAAAAAA=", "node": { - "address": "0xa6843de124b35fd704f2b37dd3471fcf4f9aa9bc957726c43a8bd96d04004b32" + "address": "0x77ba7fd3ad1f11cfb2ef0e731934fbc2ded39ccb2c8589097ee13cbb63c5fd5b" } }, { - "cursor": "ILL8PT6CGnkMV6rhl3RwEvqsdGHGUNh5XsuxI1STMnrZAQAAAAAAAAA=", + "cursor": "IIOsrADQ8BewuFucSNoSsxl38aOLmCmqq1M1qt1obgy8AQAAAAAAAAA=", "node": { - "address": "0xb2fc3d3e821a790c57aae197747012faac7461c650d8795ecbb1235493327ad9" + "address": "0x83acac00d0f017b0b85b9c48da12b31977f1a38b9829aaab5335aadd686e0cbc" } } ] } }, "obj_3_0": { - "address": "0x88578327c99a90e37d880a72f8f09f3f179636ea2bc9164ec866b4601ad4972b" + "address": "0x17273a04f41c1b7fcf0102b7d4d8f1e0004656e2ce58c5990c36d2f36bce7680" }, "obj_5_0": { - "address": "0xa6843de124b35fd704f2b37dd3471fcf4f9aa9bc957726c43a8bd96d04004b32" + "address": "0x39df76c9d89b759f4d7fc8f3aa1a9cf8f882aa4e9ef2a9a4bab3879dfd7b9d43" }, "obj_6_0": { - "address": "0xb2fc3d3e821a790c57aae197747012faac7461c650d8795ecbb1235493327ad9" + "address": "0x77ba7fd3ad1f11cfb2ef0e731934fbc2ded39ccb2c8589097ee13cbb63c5fd5b" }, "obj_4_0": { - "address": "0x8b8e28320bde8c785b4bf097d232b3e509fa13a1608c60f505c0d64bc4d36c75" + "address": "0x83acac00d0f017b0b85b9c48da12b31977f1a38b9829aaab5335aadd686e0cbc" }, "obj_2_0": { - "address": "0x05c63b17a53f26adda4b72908ef21316ea3e2d33180270a442f840eb64e22e49" + "address": "0x48ac183920abacfa5b0f1fb0cf03968bda5ed50440a8ad0d2c2aab8ef3f88521" } } } @@ -153,7 +153,10 @@ Response: { "objects": { "edges": [ { - "cursor": "ILL8PT6CGnkMV6rhl3RwEvqsdGHGUNh5XsuxI1STMnrZAQAAAAAAAAA=" + "cursor": "IEisGDkgq6z6Ww8fsM8DlovaXtUEQKitDSwqq47z+IUhAQAAAAAAAAA=" + }, + { + "cursor": "IHe6f9OtHxHPsu8Ocxk0+8Le05zLLIWJCX7hPLtjxf1bAQAAAAAAAAA=" } ] } @@ -167,11 +170,7 @@ Response: { "data": { "address": { "objects": { - "edges": [ - { - "cursor": "IKaEPeEks1/XBPKzfdNHH89Pmqm8lXcmxDqL2W0EAEsyAQAAAAAAAAA=" - } - ] + "edges": [] } } } @@ -183,11 +182,7 @@ Response: { "data": { "address": { "objects": { - "edges": [ - { - "cursor": "IAXGOxelPyat2ktykI7yExbqPi0zGAJwpEL4QOtk4i5JAQAAAAAAAAA=" - } - ] + "edges": [] } } } @@ -201,15 +196,15 @@ Response: { "objects": { "edges": [ { - "cursor": "IKaEPeEks1/XBPKzfdNHH89Pmqm8lXcmxDqL2W0EAEsyAQAAAAAAAAA=", + "cursor": "IHe6f9OtHxHPsu8Ocxk0+8Le05zLLIWJCX7hPLtjxf1bAQAAAAAAAAA=", "node": { - "address": "0xa6843de124b35fd704f2b37dd3471fcf4f9aa9bc957726c43a8bd96d04004b32" + "address": "0x77ba7fd3ad1f11cfb2ef0e731934fbc2ded39ccb2c8589097ee13cbb63c5fd5b" } }, { - "cursor": "ILL8PT6CGnkMV6rhl3RwEvqsdGHGUNh5XsuxI1STMnrZAQAAAAAAAAA=", + "cursor": "IIOsrADQ8BewuFucSNoSsxl38aOLmCmqq1M1qt1obgy8AQAAAAAAAAA=", "node": { - "address": "0xb2fc3d3e821a790c57aae197747012faac7461c650d8795ecbb1235493327ad9" + "address": "0x83acac00d0f017b0b85b9c48da12b31977f1a38b9829aaab5335aadd686e0cbc" } } ] diff --git a/crates/iota-graphql-e2e-tests/tests/objects/pagination.move b/crates/iota-graphql-e2e-tests/tests/objects/pagination.move index a29772c7180..1761941b779 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/pagination.move +++ b/crates/iota-graphql-e2e-tests/tests/objects/pagination.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 A=0x42 --simulator +//# init --protocol-version 3 --addresses Test=0x0 A=0x42 --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/objects/public_transfer.exp b/crates/iota-graphql-e2e-tests/tests/objects/public_transfer.exp index 5c4f5326f97..a49b4b7a09a 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/public_transfer.exp +++ b/crates/iota-graphql-e2e-tests/tests/objects/public_transfer.exp @@ -37,7 +37,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0xd01ba1620a63906586ce0abe061e1f132f9581e42c616660afa3119a028c48a5::m::Bar" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" } } } @@ -48,7 +48,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" + "repr": "0x4bc5cdfa11066824f0581c920eb6d415e06d05af77e9ff92a3ead639290b53fb::m::Bar" } } } @@ -59,7 +59,7 @@ Response: { "asMoveObject": { "contents": { "type": { - "repr": "0xd01ba1620a63906586ce0abe061e1f132f9581e42c616660afa3119a028c48a5::m::Foo" + "repr": "0x4bc5cdfa11066824f0581c920eb6d415e06d05af77e9ff92a3ead639290b53fb::m::Foo" } } } diff --git a/crates/iota-graphql-e2e-tests/tests/objects/public_transfer.move b/crates/iota-graphql-e2e-tests/tests/objects/public_transfer.move index 4859be85c7a..878e31eaddc 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/public_transfer.move +++ b/crates/iota-graphql-e2e-tests/tests/objects/public_transfer.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses P0=0x0 --accounts A --simulator //# publish module P0::m { diff --git a/crates/iota-graphql-e2e-tests/tests/objects/received.exp b/crates/iota-graphql-e2e-tests/tests/objects/received.exp index 2132f4d9a92..e37bae0b579 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/received.exp +++ b/crates/iota-graphql-e2e-tests/tests/objects/received.exp @@ -30,7 +30,7 @@ Response: { "receivedTransactionBlocks": { "nodes": [ { - "digest": "87R2Us8BZN1RJYEvfdLp3LrxiiVWuwJA5UeMoCeYdJ4b" + "digest": "5Y8XcPc5cBi1hAXtoWbC8Zp5NkJB2pwsiwy81YtSbTxR" } ] } diff --git a/crates/iota-graphql-e2e-tests/tests/objects/received.move b/crates/iota-graphql-e2e-tests/tests/objects/received.move index c79a2dbf3d2..349b70ab0f1 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/received.move +++ b/crates/iota-graphql-e2e-tests/tests/objects/received.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 --simulator +//# init --protocol-version 3 --addresses P0=0x0 --simulator //# run-graphql { diff --git a/crates/iota-graphql-e2e-tests/tests/objects/staked_iota.exp b/crates/iota-graphql-e2e-tests/tests/objects/staked_iota.exp index af253fa6124..2ecf6ea028d 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/staked_iota.exp +++ b/crates/iota-graphql-e2e-tests/tests/objects/staked_iota.exp @@ -10,7 +10,7 @@ Response: { "objects": { "edges": [ { - "cursor": "IAeULt3Vt3s3r+CMTYVO1viD348GDaO7+cH3HhE5nbGzAAAAAAAAAAA=", + "cursor": "IBzTT9dE90ffUpMMyHxrf3Gr46ZpcTQoNSYG8PyAd4rRAAAAAAAAAAA=", "node": { "asMoveObject": { "asStakedIota": { @@ -39,7 +39,7 @@ gas summary: computation_cost: 1000000, computation_cost_burned: 1000000, storag task 3, line 38: //# run 0x3::iota_system::request_add_stake --args object(0x5) object(2,0) @validator_0 --sender C -events: Event { package_id: iota_system, transaction_module: Identifier("iota_system"), sender: C, type_: StructTag { address: iota_system, module: Identifier("validator"), name: Identifier("StakingRequestEvent"), type_params: [] }, contents: [145, 201, 73, 32, 135, 23, 121, 37, 7, 101, 40, 103, 216, 99, 155, 51, 118, 111, 115, 220, 44, 253, 144, 143, 80, 103, 133, 240, 58, 235, 137, 202, 175, 163, 158, 79, 0, 218, 226, 120, 249, 119, 199, 198, 147, 10, 94, 44, 118, 232, 93, 23, 165, 38, 215, 36, 187, 206, 15, 184, 31, 176, 125, 76, 140, 202, 78, 28, 224, 186, 89, 4, 206, 166, 29, 249, 36, 45, 162, 247, 210, 158, 62, 243, 40, 251, 126, 192, 124, 8, 107, 59, 244, 124, 166, 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 228, 11, 84, 2, 0, 0, 0] } +events: Event { package_id: iota_system, transaction_module: Identifier("iota_system"), sender: C, type_: StructTag { address: iota_system, module: Identifier("validator"), name: Identifier("StakingRequestEvent"), type_params: [] }, contents: [217, 25, 175, 239, 57, 244, 176, 32, 209, 232, 109, 171, 147, 246, 3, 15, 205, 197, 172, 21, 36, 143, 47, 100, 240, 226, 221, 254, 42, 3, 157, 155, 175, 163, 158, 79, 0, 218, 226, 120, 249, 119, 199, 198, 147, 10, 94, 44, 118, 232, 93, 23, 165, 38, 215, 36, 187, 206, 15, 184, 31, 176, 125, 76, 140, 202, 78, 28, 224, 186, 89, 4, 206, 166, 29, 249, 36, 45, 162, 247, 210, 158, 62, 243, 40, 251, 126, 192, 124, 8, 107, 59, 244, 124, 166, 26, 0, 0, 0, 0, 0, 0, 0, 0, 0, 228, 11, 84, 2, 0, 0, 0] } created: object(3,0) mutated: object(_), 0x0000000000000000000000000000000000000000000000000000000000000005, object(0,0) deleted: object(2,0) @@ -60,34 +60,34 @@ Response: { "objects": { "edges": [ { - "cursor": "IAeULt3Vt3s3r+CMTYVO1viD348GDaO7+cH3HhE5nbGzAgAAAAAAAAA=", + "cursor": "IBRoODsinaSa4FR6ECYNlPAg85qVJE3d+HBEkymgWhy5AgAAAAAAAAA=", "node": { "asMoveObject": { "asStakedIota": { - "principal": "1500000000000000", - "poolId": "0x91c949208717792507652867d8639b33766f73dc2cfd908f506785f03aeb89ca" + "principal": "15340000000000", + "poolId": "0xd919afef39f4b020d1e86dab93f6030fcdc5ac15248f2f64f0e2ddfe2a039d9b" } } } }, { - "cursor": "IA+owYRIwFujnaEiElWV25tmsgRd3aiT7h9//oDyhQb2AgAAAAAAAAA=", + "cursor": "IBzTT9dE90ffUpMMyHxrf3Gr46ZpcTQoNSYG8PyAd4rRAgAAAAAAAAA=", "node": { "asMoveObject": { "asStakedIota": { - "principal": "15340000000000", - "poolId": "0x91c949208717792507652867d8639b33766f73dc2cfd908f506785f03aeb89ca" + "principal": "1500000000000000", + "poolId": "0xd919afef39f4b020d1e86dab93f6030fcdc5ac15248f2f64f0e2ddfe2a039d9b" } } } }, { - "cursor": "IPEp6dbnEicrkwynNujLy5/Voas64Cc5Ta7ruT5oQNbTAgAAAAAAAAA=", + "cursor": "ICx+i9jLtLxeXhS/A0ahYRLGYYlMzFleNwu1meRx2FBtAgAAAAAAAAA=", "node": { "asMoveObject": { "asStakedIota": { "principal": "10000000000", - "poolId": "0x91c949208717792507652867d8639b33766f73dc2cfd908f506785f03aeb89ca" + "poolId": "0xd919afef39f4b020d1e86dab93f6030fcdc5ac15248f2f64f0e2ddfe2a039d9b" } } } @@ -98,7 +98,7 @@ Response: { "stakedIotas": { "edges": [ { - "cursor": "IPEp6dbnEicrkwynNujLy5/Voas64Cc5Ta7ruT5oQNbTAgAAAAAAAAA=", + "cursor": "ICx+i9jLtLxeXhS/A0ahYRLGYYlMzFleNwu1meRx2FBtAgAAAAAAAAA=", "node": { "principal": "10000000000" } diff --git a/crates/iota-graphql-e2e-tests/tests/objects/staked_iota.move b/crates/iota-graphql-e2e-tests/tests/objects/staked_iota.move index 78ffdc2c968..28433b412c2 100644 --- a/crates/iota-graphql-e2e-tests/tests/objects/staked_iota.move +++ b/crates/iota-graphql-e2e-tests/tests/objects/staked_iota.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --simulator --accounts C +//# init --protocol-version 3 --simulator --accounts C //# run-graphql { # Initial query yields only the validator's stake diff --git a/crates/iota-graphql-e2e-tests/tests/owner/downcasts.exp b/crates/iota-graphql-e2e-tests/tests/owner/downcasts.exp index 4c0135a646e..a9841222726 100644 --- a/crates/iota-graphql-e2e-tests/tests/owner/downcasts.exp +++ b/crates/iota-graphql-e2e-tests/tests/owner/downcasts.exp @@ -24,7 +24,7 @@ Response: { }, "coin": { "asObject": { - "digest": "MCn6VCYaKu81nt3Azx9A31GtLmmmtfzuWnShjGXxBPC" + "digest": "53tg3FNtsBq9jFpgBiemKEYtj9qFZ5CZxbpJnsc5uAcY" } } } diff --git a/crates/iota-graphql-e2e-tests/tests/owner/downcasts.move b/crates/iota-graphql-e2e-tests/tests/owner/downcasts.move index 1213b5f0a0d..5e724b2490c 100644 --- a/crates/iota-graphql-e2e-tests/tests/owner/downcasts.move +++ b/crates/iota-graphql-e2e-tests/tests/owner/downcasts.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses P0=0x0 --accounts A --simulator // Split off a gas coin, so we have an object to query //# programmable --sender A --inputs 1000 @A diff --git a/crates/iota-graphql-e2e-tests/tests/owner/root_version.exp b/crates/iota-graphql-e2e-tests/tests/owner/root_version.exp index 32e752065fb..78af306fc98 100644 --- a/crates/iota-graphql-e2e-tests/tests/owner/root_version.exp +++ b/crates/iota-graphql-e2e-tests/tests/owner/root_version.exp @@ -124,10 +124,10 @@ Response: { "version": 11, "contents": { "json": { - "id": "0x92649202a7ed897a62d638e17c21515c6a948ac7be20cc41060455601883cd00", + "id": "0xb5aecdb998631cc910744d437cb25748475a2a9e6230ca5953e8942b6994ab49", "count": "1", "wrapped": { - "id": "0xe49f0faf71c339f2bbcf1348bdc3b8ba4ece961691aed1a8d9309d400ef140e5", + "id": "0x4550d7846ec386e99c914c4e6b0ea63f9e5f61bf8fa6f3e6d44593666d107989", "count": "1" } } @@ -141,10 +141,10 @@ Response: { "version": 10, "contents": { "json": { - "id": "0x92649202a7ed897a62d638e17c21515c6a948ac7be20cc41060455601883cd00", + "id": "0xb5aecdb998631cc910744d437cb25748475a2a9e6230ca5953e8942b6994ab49", "count": "1", "wrapped": { - "id": "0xe49f0faf71c339f2bbcf1348bdc3b8ba4ece961691aed1a8d9309d400ef140e5", + "id": "0x4550d7846ec386e99c914c4e6b0ea63f9e5f61bf8fa6f3e6d44593666d107989", "count": "1" } } @@ -158,10 +158,10 @@ Response: { "version": 8, "contents": { "json": { - "id": "0x92649202a7ed897a62d638e17c21515c6a948ac7be20cc41060455601883cd00", + "id": "0xb5aecdb998631cc910744d437cb25748475a2a9e6230ca5953e8942b6994ab49", "count": "1", "wrapped": { - "id": "0xe49f0faf71c339f2bbcf1348bdc3b8ba4ece961691aed1a8d9309d400ef140e5", + "id": "0x4550d7846ec386e99c914c4e6b0ea63f9e5f61bf8fa6f3e6d44593666d107989", "count": "0" } } @@ -175,10 +175,10 @@ Response: { "version": 7, "contents": { "json": { - "id": "0x92649202a7ed897a62d638e17c21515c6a948ac7be20cc41060455601883cd00", + "id": "0xb5aecdb998631cc910744d437cb25748475a2a9e6230ca5953e8942b6994ab49", "count": "0", "wrapped": { - "id": "0xe49f0faf71c339f2bbcf1348bdc3b8ba4ece961691aed1a8d9309d400ef140e5", + "id": "0x4550d7846ec386e99c914c4e6b0ea63f9e5f61bf8fa6f3e6d44593666d107989", "count": "0" } } @@ -199,7 +199,7 @@ Response: { "version": 7, "contents": { "json": { - "id": "0x12b9cd665e2fba6d9cda7d1b651188c07f355ebb6258fab2fb07161606b4ab2a", + "id": "0x3e0b2966e7f0c0a706456cb99bb8421b32863f5a4524b02d338ff2b797804bc3", "count": "0" } } @@ -212,7 +212,7 @@ Response: { "version": 10, "contents": { "json": { - "id": "0x12b9cd665e2fba6d9cda7d1b651188c07f355ebb6258fab2fb07161606b4ab2a", + "id": "0x3e0b2966e7f0c0a706456cb99bb8421b32863f5a4524b02d338ff2b797804bc3", "count": "1" } } @@ -225,7 +225,7 @@ Response: { "version": 10, "contents": { "json": { - "id": "0x12b9cd665e2fba6d9cda7d1b651188c07f355ebb6258fab2fb07161606b4ab2a", + "id": "0x3e0b2966e7f0c0a706456cb99bb8421b32863f5a4524b02d338ff2b797804bc3", "count": "1" } } @@ -238,7 +238,7 @@ Response: { "version": 7, "contents": { "json": { - "id": "0x12b9cd665e2fba6d9cda7d1b651188c07f355ebb6258fab2fb07161606b4ab2a", + "id": "0x3e0b2966e7f0c0a706456cb99bb8421b32863f5a4524b02d338ff2b797804bc3", "count": "0" } } @@ -251,7 +251,7 @@ Response: { "version": 7, "contents": { "json": { - "id": "0x12b9cd665e2fba6d9cda7d1b651188c07f355ebb6258fab2fb07161606b4ab2a", + "id": "0x3e0b2966e7f0c0a706456cb99bb8421b32863f5a4524b02d338ff2b797804bc3", "count": "0" } } @@ -271,7 +271,7 @@ Response: { "version": 7, "contents": { "json": { - "id": "0xd466bb5f3baf46598ec3550987b3993d522e10758441b170fd2a4eab5f618320", + "id": "0x04f88b43d1ec09fb0141d0b9e699753c5cf49a45e62f9c1caa4ecf2090948bfb", "count": "0" } } @@ -284,7 +284,7 @@ Response: { "version": 11, "contents": { "json": { - "id": "0xd466bb5f3baf46598ec3550987b3993d522e10758441b170fd2a4eab5f618320", + "id": "0x04f88b43d1ec09fb0141d0b9e699753c5cf49a45e62f9c1caa4ecf2090948bfb", "count": "1" } } @@ -297,7 +297,7 @@ Response: { "version": 7, "contents": { "json": { - "id": "0xd466bb5f3baf46598ec3550987b3993d522e10758441b170fd2a4eab5f618320", + "id": "0x04f88b43d1ec09fb0141d0b9e699753c5cf49a45e62f9c1caa4ecf2090948bfb", "count": "0" } } diff --git a/crates/iota-graphql-e2e-tests/tests/owner/root_version.move b/crates/iota-graphql-e2e-tests/tests/owner/root_version.move index 843513267e6..75ac65d88f8 100644 --- a/crates/iota-graphql-e2e-tests/tests/owner/root_version.move +++ b/crates/iota-graphql-e2e-tests/tests/owner/root_version.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses P0=0x0 --accounts A --simulator //# publish module P0::M { diff --git a/crates/iota-graphql-e2e-tests/tests/packages/datatypes.exp b/crates/iota-graphql-e2e-tests/tests/packages/datatypes.exp index 0b4e2611e4b..7304306a6d1 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/datatypes.exp +++ b/crates/iota-graphql-e2e-tests/tests/packages/datatypes.exp @@ -155,7 +155,7 @@ task 5, lines 73-97: Response: { "data": { "object": { - "address": "0x416f86a3ae5c3eae593a2f345346265988d2d4a5a981c9635e506ccdb5417d5b", + "address": "0x54eaef53a812cd11c23a201ef3f2920fb6a6461815e8258402193f35d0e7d8b9", "asMovePackage": { "module": { "datatypes": { @@ -190,7 +190,7 @@ task 6, lines 99-144: Response: { "data": { "object": { - "address": "0x416f86a3ae5c3eae593a2f345346265988d2d4a5a981c9635e506ccdb5417d5b", + "address": "0x54eaef53a812cd11c23a201ef3f2920fb6a6461815e8258402193f35d0e7d8b9", "asMovePackage": { "module": { "datatypes": { diff --git a/crates/iota-graphql-e2e-tests/tests/packages/datatypes.move b/crates/iota-graphql-e2e-tests/tests/packages/datatypes.move index 9c9cf1bf15c..384d1768633 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/datatypes.move +++ b/crates/iota-graphql-e2e-tests/tests/packages/datatypes.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --addresses P0=0x0 P1=0x0 --accounts A --simulator --protocol-version 1 +//# init --addresses P0=0x0 P1=0x0 --accounts A --simulator --protocol-version 3 //# run-graphql diff --git a/crates/iota-graphql-e2e-tests/tests/packages/enums.exp b/crates/iota-graphql-e2e-tests/tests/packages/enums.exp index 49ab62fee88..4c9b4d89409 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/enums.exp +++ b/crates/iota-graphql-e2e-tests/tests/packages/enums.exp @@ -25,13 +25,19 @@ Response: { "nodes": [ { "outputState": { - "address": "0x70d8121ec58cb9e0c1e7af1e2d2b255737dc0b9a63e845d1357d78d371f3b858", + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4", "asMovePackage": null } }, { "outputState": { - "address": "0xbf719126a219999498cda62d638ec3e5bafded8aeff2183e30876ba3f4b88870", + "address": "0x87c53ab5ea6379f82a592148afc619d80f05d498f04827f8ee7b3b09a19de8c3", + "asMovePackage": null + } + }, + { + "outputState": { + "address": "0xcc4c98e77e7e93b1e93e8def3fbc0edd188d3ce404c3f321316104825983ce8c", "asMovePackage": { "module": { "enum": { @@ -87,12 +93,6 @@ Response: { } } } - }, - { - "outputState": { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431", - "asMovePackage": null - } } ] } @@ -125,13 +125,25 @@ Response: { "nodes": [ { "outputState": { - "address": "0x5f826526feff5ea347f55fc824285e273961ae527be45af5b4267fade97c6fa8", + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4", + "asMovePackage": null + } + }, + { + "outputState": { + "address": "0x87c53ab5ea6379f82a592148afc619d80f05d498f04827f8ee7b3b09a19de8c3", + "asMovePackage": null + } + }, + { + "outputState": { + "address": "0x87d6d716a945bd647ebfad919a11364eece57ca6c7f78ce90a652324f00a238c", "asMovePackage": { "module": { "s": { "module": { "package": { - "address": "0xbf719126a219999498cda62d638ec3e5bafded8aeff2183e30876ba3f4b88870" + "address": "0xcc4c98e77e7e93b1e93e8def3fbc0edd188d3ce404c3f321316104825983ce8c" } }, "name": "S", @@ -186,7 +198,7 @@ Response: { "t": { "module": { "package": { - "address": "0x5f826526feff5ea347f55fc824285e273961ae527be45af5b4267fade97c6fa8" + "address": "0x87d6d716a945bd647ebfad919a11364eece57ca6c7f78ce90a652324f00a238c" } }, "name": "T", @@ -216,12 +228,12 @@ Response: { { "name": "s", "type": { - "repr": "0xbf719126a219999498cda62d638ec3e5bafded8aeff2183e30876ba3f4b88870::m::S", + "repr": "0xcc4c98e77e7e93b1e93e8def3fbc0edd188d3ce404c3f321316104825983ce8c::m::S", "signature": { "ref": null, "body": { "datatype": { - "package": "0xbf719126a219999498cda62d638ec3e5bafded8aeff2183e30876ba3f4b88870", + "package": "0xcc4c98e77e7e93b1e93e8def3fbc0edd188d3ce404c3f321316104825983ce8c", "module": "m", "type": "S", "typeParameters": [] @@ -255,7 +267,7 @@ Response: { { "name": "t", "type": { - "repr": "0xbf719126a219999498cda62d638ec3e5bafded8aeff2183e30876ba3f4b88870::m::T<0xbf719126a219999498cda62d638ec3e5bafded8aeff2183e30876ba3f4b88870::m::S>" + "repr": "0xcc4c98e77e7e93b1e93e8def3fbc0edd188d3ce404c3f321316104825983ce8c::m::T<0xcc4c98e77e7e93b1e93e8def3fbc0edd188d3ce404c3f321316104825983ce8c::m::S>" } } ] @@ -265,18 +277,6 @@ Response: { } } } - }, - { - "outputState": { - "address": "0x70d8121ec58cb9e0c1e7af1e2d2b255737dc0b9a63e845d1357d78d371f3b858", - "asMovePackage": null - } - }, - { - "outputState": { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431", - "asMovePackage": null - } } ] } @@ -297,6 +297,16 @@ Response: { "effects": { "objectChanges": { "nodes": [ + { + "outputState": { + "asMovePackage": null + } + }, + { + "outputState": { + "asMovePackage": null + } + }, { "outputState": { "asMovePackage": { @@ -316,16 +326,6 @@ Response: { } } } - }, - { - "outputState": { - "asMovePackage": null - } - }, - { - "outputState": { - "asMovePackage": null - } } ] } diff --git a/crates/iota-graphql-e2e-tests/tests/packages/enums.move b/crates/iota-graphql-e2e-tests/tests/packages/enums.move index 3c858b24aab..21b2ab9b3fc 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/enums.move +++ b/crates/iota-graphql-e2e-tests/tests/packages/enums.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --addresses P0=0x0 P1=0x0 --accounts A --simulator --protocol-version 1 +//# init --addresses P0=0x0 P1=0x0 --accounts A --simulator --protocol-version 3 //# publish --upgradeable --sender A diff --git a/crates/iota-graphql-e2e-tests/tests/packages/friends.exp b/crates/iota-graphql-e2e-tests/tests/packages/friends.exp index ad3f49f048c..51aaa0c74ee 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/friends.exp +++ b/crates/iota-graphql-e2e-tests/tests/packages/friends.exp @@ -23,6 +23,16 @@ Response: { "effects": { "objectChanges": { "nodes": [ + { + "outputState": { + "asMovePackage": null + } + }, + { + "outputState": { + "asMovePackage": null + } + }, { "outputState": { "asMovePackage": { @@ -96,16 +106,6 @@ Response: { } } } - }, - { - "outputState": { - "asMovePackage": null - } - }, - { - "outputState": { - "asMovePackage": null - } } ] } @@ -128,9 +128,7 @@ Response: { "nodes": [ { "outputState": { - "asMovePackage": { - "module": null - } + "asMovePackage": null } }, { @@ -140,7 +138,9 @@ Response: { }, { "outputState": { - "asMovePackage": null + "asMovePackage": { + "module": null + } } } ] @@ -166,7 +166,7 @@ Response: { "effects", "objectChanges", "nodes", - 0, + 2, "outputState", "asMovePackage", "module", @@ -189,6 +189,16 @@ Response: { "effects": { "objectChanges": { "nodes": [ + { + "outputState": { + "asMovePackage": null + } + }, + { + "outputState": { + "asMovePackage": null + } + }, { "outputState": { "asMovePackage": { @@ -264,16 +274,6 @@ Response: { } } } - }, - { - "outputState": { - "asMovePackage": null - } - }, - { - "outputState": { - "asMovePackage": null - } } ] } @@ -309,11 +309,6 @@ Response: { "asMovePackage": null } }, - { - "outputState": { - "asMovePackage": null - } - }, { "outputState": { "asMovePackage": { @@ -349,6 +344,11 @@ Response: { } } } + }, + { + "outputState": { + "asMovePackage": null + } } ] } diff --git a/crates/iota-graphql-e2e-tests/tests/packages/friends.move b/crates/iota-graphql-e2e-tests/tests/packages/friends.move index 457541c820e..d20b2d1cb71 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/friends.move +++ b/crates/iota-graphql-e2e-tests/tests/packages/friends.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 P1=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses P0=0x0 P1=0x0 --accounts A --simulator //# publish --upgradeable --sender A diff --git a/crates/iota-graphql-e2e-tests/tests/packages/functions.exp b/crates/iota-graphql-e2e-tests/tests/packages/functions.exp index bb7442d617f..885e72d8399 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/functions.exp +++ b/crates/iota-graphql-e2e-tests/tests/packages/functions.exp @@ -95,19 +95,25 @@ Response: { "nodes": [ { "outputState": { - "address": "0x0d5aebe6f12ff71a237457c35211f4c584b2223a4f2702affa8d58975bc149c9", + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4", "asMovePackage": null } }, { "outputState": { - "address": "0x807e4a90bae0ebd4b5726bf685fedcd12877ab1a0feb7b67ecdc648e270356d4", + "address": "0x3a83f4839537a59d741dc28360d84b70e8d92abbecf377de4e2b82d259fbc632", + "asMovePackage": null + } + }, + { + "outputState": { + "address": "0xdbeae47ffd82a9b37300ad21bcb435145647c3026172a3067d4bd6194ad38bf2", "asMovePackage": { "module": { "function": { "module": { "package": { - "address": "0x807e4a90bae0ebd4b5726bf685fedcd12877ab1a0feb7b67ecdc648e270356d4" + "address": "0xdbeae47ffd82a9b37300ad21bcb435145647c3026172a3067d4bd6194ad38bf2" } }, "name": "f", @@ -137,12 +143,6 @@ Response: { } } } - }, - { - "outputState": { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431", - "asMovePackage": null - } } ] } @@ -175,25 +175,25 @@ Response: { "nodes": [ { "outputState": { - "address": "0x0d5aebe6f12ff71a237457c35211f4c584b2223a4f2702affa8d58975bc149c9", + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4", "asMovePackage": null } }, { "outputState": { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431", + "address": "0x3a83f4839537a59d741dc28360d84b70e8d92abbecf377de4e2b82d259fbc632", "asMovePackage": null } }, { "outputState": { - "address": "0xe57e6628d5170a1fc4209c1f4cf1cfb8bcf42f6874ab8f90fda217600741f46a", + "address": "0xe8ab1e0a297142a98aa0f732ed13f760e845c821b72b532d77c8094466b7ee3c", "asMovePackage": { "module": { "f": { "module": { "package": { - "address": "0xe57e6628d5170a1fc4209c1f4cf1cfb8bcf42f6874ab8f90fda217600741f46a" + "address": "0xe8ab1e0a297142a98aa0f732ed13f760e845c821b72b532d77c8094466b7ee3c" } }, "name": "f", @@ -223,7 +223,7 @@ Response: { "g": { "module": { "package": { - "address": "0xe57e6628d5170a1fc4209c1f4cf1cfb8bcf42f6874ab8f90fda217600741f46a" + "address": "0xe8ab1e0a297142a98aa0f732ed13f760e845c821b72b532d77c8094466b7ee3c" } }, "name": "g", diff --git a/crates/iota-graphql-e2e-tests/tests/packages/functions.move b/crates/iota-graphql-e2e-tests/tests/packages/functions.move index 775e9dcef5d..7de67e6eca5 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/functions.move +++ b/crates/iota-graphql-e2e-tests/tests/packages/functions.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 P1=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses P0=0x0 P1=0x0 --accounts A --simulator //# run-graphql diff --git a/crates/iota-graphql-e2e-tests/tests/packages/modules.exp b/crates/iota-graphql-e2e-tests/tests/packages/modules.exp index 9351971dcea..a11451f5896 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/modules.exp +++ b/crates/iota-graphql-e2e-tests/tests/packages/modules.exp @@ -22,23 +22,23 @@ Response: { "nodes": [ { "outputState": { - "address": "0x3a391c29453c2e07af1eb0befd50b903f0cc1609df8826e1a79d3e1eceab6481", + "address": "0x3079bbb8be43087ea062afccaf9c251714e7917667a46202a1de549f7fbf06ec", "asMovePackage": { "module": { "name": "m", "package": { - "address": "0x3a391c29453c2e07af1eb0befd50b903f0cc1609df8826e1a79d3e1eceab6481" + "address": "0x3079bbb8be43087ea062afccaf9c251714e7917667a46202a1de549f7fbf06ec" }, "fileFormatVersion": 6, - "bytes": "oRzrCwYAAAAIAQAGAgYKAxARBCEEBSUfB0QkCGhADKgBMAAGAQMBBQEADAEAAQIBAgAABAABAQIAAgIBAAEHBQEBAAIEAAYCAwYLAAEJAAEDAQYLAAEIAQABCQABBgsAAQkAAQgBBENvaW4ESU9UQQNiYXIEY29pbgNmb28EaW90YQFtBXZhbHVlOjkcKUU8LgevHrC+/VC5A/DMFgnfiCbhp50+Hs6rZIEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAgABAAADBQsBOAALABYCAQEAAAMIBioAAAAAAAAACgA4AQYrAAAAAAAAAAsAOAEYAgA=", - "disassembly": "// Move bytecode v6\nmodule 3a391c29453c2e07af1eb0befd50b903f0cc1609df8826e1a79d3e1eceab6481.m {\nuse 0000000000000000000000000000000000000000000000000000000000000002::coin;\nuse 0000000000000000000000000000000000000000000000000000000000000002::iota;\n\n\n\n\n\n\npublic foo(Arg0: u64, Arg1: &Coin): u64 {\nB0:\n\t0: MoveLoc[1](Arg1: &Coin)\n\t1: Call coin::value(&Coin): u64\n\t2: MoveLoc[0](Arg0: u64)\n\t3: Add\n\t4: Ret\n\n}\npublic bar(Arg0: &Coin): u64 {\nB0:\n\t0: LdU64(42)\n\t1: CopyLoc[0](Arg0: &Coin)\n\t2: Call foo(u64, &Coin): u64\n\t3: LdU64(43)\n\t4: MoveLoc[0](Arg0: &Coin)\n\t5: Call foo(u64, &Coin): u64\n\t6: Mul\n\t7: Ret\n\n}\n}" + "bytes": "oRzrCwYAAAAIAQAGAgYKAxARBCEEBSUfB0QkCGhADKgBMAAGAQMBBQEADAEAAQIBAgAABAABAQIAAgIBAAEHBQEBAAIEAAYCAwYLAAEJAAEDAQYLAAEIAQABCQABBgsAAQkAAQgBBENvaW4ESU9UQQNiYXIEY29pbgNmb28EaW90YQFtBXZhbHVlMHm7uL5DCH6gYq/Mr5wlFxTnkXZnpGICod5Un3+/BuwAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAgABAAADBQsBOAALABYCAQEAAAMIBioAAAAAAAAACgA4AQYrAAAAAAAAAAsAOAEYAgA=", + "disassembly": "// Move bytecode v6\nmodule 3079bbb8be43087ea062afccaf9c251714e7917667a46202a1de549f7fbf06ec.m {\nuse 0000000000000000000000000000000000000000000000000000000000000002::coin;\nuse 0000000000000000000000000000000000000000000000000000000000000002::iota;\n\n\n\n\n\n\npublic foo(Arg0: u64, Arg1: &Coin): u64 {\nB0:\n\t0: MoveLoc[1](Arg1: &Coin)\n\t1: Call coin::value(&Coin): u64\n\t2: MoveLoc[0](Arg0: u64)\n\t3: Add\n\t4: Ret\n\n}\npublic bar(Arg0: &Coin): u64 {\nB0:\n\t0: LdU64(42)\n\t1: CopyLoc[0](Arg0: &Coin)\n\t2: Call foo(u64, &Coin): u64\n\t3: LdU64(43)\n\t4: MoveLoc[0](Arg0: &Coin)\n\t5: Call foo(u64, &Coin): u64\n\t6: Mul\n\t7: Ret\n\n}\n}" } } } }, { "outputState": { - "address": "0xf22e2f7e20b84f57fdcf1f44c79c321533b922fa9488c23556356ae3f906b0bb", + "address": "0x5ba4893be3938566e676ec9ded7f006ac2f8a5cf9713ff00880f4d8a2bd24434", "asMovePackage": null } } @@ -63,7 +63,7 @@ Response: { "nodes": [ { "outputState": { - "address": "0x3a391c29453c2e07af1eb0befd50b903f0cc1609df8826e1a79d3e1eceab6481", + "address": "0x3079bbb8be43087ea062afccaf9c251714e7917667a46202a1de549f7fbf06ec", "asMovePackage": { "all": { "edges": [ @@ -136,7 +136,7 @@ Response: { }, { "outputState": { - "address": "0xf22e2f7e20b84f57fdcf1f44c79c321533b922fa9488c23556356ae3f906b0bb", + "address": "0x5ba4893be3938566e676ec9ded7f006ac2f8a5cf9713ff00880f4d8a2bd24434", "asMovePackage": null } } @@ -161,7 +161,7 @@ Response: { "nodes": [ { "outputState": { - "address": "0x3a391c29453c2e07af1eb0befd50b903f0cc1609df8826e1a79d3e1eceab6481", + "address": "0x3079bbb8be43087ea062afccaf9c251714e7917667a46202a1de549f7fbf06ec", "asMovePackage": { "prefix": { "edges": [ @@ -276,7 +276,7 @@ Response: { }, { "outputState": { - "address": "0xf22e2f7e20b84f57fdcf1f44c79c321533b922fa9488c23556356ae3f906b0bb", + "address": "0x5ba4893be3938566e676ec9ded7f006ac2f8a5cf9713ff00880f4d8a2bd24434", "asMovePackage": null } } diff --git a/crates/iota-graphql-e2e-tests/tests/packages/modules.move b/crates/iota-graphql-e2e-tests/tests/packages/modules.move index 5d060390370..2f94544273a 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/modules.move +++ b/crates/iota-graphql-e2e-tests/tests/packages/modules.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses pkg=0x0 --simulator +//# init --protocol-version 3 --addresses pkg=0x0 --simulator //# publish diff --git a/crates/iota-graphql-e2e-tests/tests/packages/structs.exp b/crates/iota-graphql-e2e-tests/tests/packages/structs.exp index 32e3769f08f..219a30ea5f7 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/structs.exp +++ b/crates/iota-graphql-e2e-tests/tests/packages/structs.exp @@ -154,19 +154,19 @@ Response: { "nodes": [ { "outputState": { - "address": "0x84c38f8d9019f1e508e97d3496da69da808df7482086053358996b7679e9e8b0", + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4", "asMovePackage": null } }, { "outputState": { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431", + "address": "0x3d5f2563cf49bfa78c83aaa5d4c327b682046e151dd0bb6230a6bf0337e09394", "asMovePackage": null } }, { "outputState": { - "address": "0xfc8d5ed0a9e91f3edc482b1494ad1d010721fa25b3542147604d9dc4194c5b56", + "address": "0x7782861497fc6e7cfe33fa186e469d51323147f7c25ad0583c49d49f52da3208", "asMovePackage": { "module": { "struct": { @@ -224,13 +224,25 @@ Response: { "nodes": [ { "outputState": { - "address": "0x0fcbd15c6fc4129828bd9551a7c79b672a801b095eb2009787bd81372c91b7ed", + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4", + "asMovePackage": null + } + }, + { + "outputState": { + "address": "0x3d5f2563cf49bfa78c83aaa5d4c327b682046e151dd0bb6230a6bf0337e09394", + "asMovePackage": null + } + }, + { + "outputState": { + "address": "0x60963c987e0b206d400a3b2a3a0e5ec3a182f1d4b5b52bbac9c2c2b92bfab908", "asMovePackage": { "module": { "s": { "module": { "package": { - "address": "0xfc8d5ed0a9e91f3edc482b1494ad1d010721fa25b3542147604d9dc4194c5b56" + "address": "0x7782861497fc6e7cfe33fa186e469d51323147f7c25ad0583c49d49f52da3208" } }, "name": "S", @@ -255,7 +267,7 @@ Response: { "t": { "module": { "package": { - "address": "0x0fcbd15c6fc4129828bd9551a7c79b672a801b095eb2009787bd81372c91b7ed" + "address": "0x60963c987e0b206d400a3b2a3a0e5ec3a182f1d4b5b52bbac9c2c2b92bfab908" } }, "name": "T", @@ -282,12 +294,12 @@ Response: { { "name": "s", "type": { - "repr": "0xfc8d5ed0a9e91f3edc482b1494ad1d010721fa25b3542147604d9dc4194c5b56::m::S", + "repr": "0x7782861497fc6e7cfe33fa186e469d51323147f7c25ad0583c49d49f52da3208::m::S", "signature": { "ref": null, "body": { "datatype": { - "package": "0xfc8d5ed0a9e91f3edc482b1494ad1d010721fa25b3542147604d9dc4194c5b56", + "package": "0x7782861497fc6e7cfe33fa186e469d51323147f7c25ad0583c49d49f52da3208", "module": "m", "type": "S", "typeParameters": [] @@ -316,7 +328,7 @@ Response: { { "name": "t", "type": { - "repr": "0xfc8d5ed0a9e91f3edc482b1494ad1d010721fa25b3542147604d9dc4194c5b56::m::T<0xfc8d5ed0a9e91f3edc482b1494ad1d010721fa25b3542147604d9dc4194c5b56::m::S>" + "repr": "0x7782861497fc6e7cfe33fa186e469d51323147f7c25ad0583c49d49f52da3208::m::T<0x7782861497fc6e7cfe33fa186e469d51323147f7c25ad0583c49d49f52da3208::m::S>" } } ] @@ -324,18 +336,6 @@ Response: { } } } - }, - { - "outputState": { - "address": "0x84c38f8d9019f1e508e97d3496da69da808df7482086053358996b7679e9e8b0", - "asMovePackage": null - } - }, - { - "outputState": { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431", - "asMovePackage": null - } } ] } @@ -356,6 +356,16 @@ Response: { "effects": { "objectChanges": { "nodes": [ + { + "outputState": { + "asMovePackage": null + } + }, + { + "outputState": { + "asMovePackage": null + } + }, { "outputState": { "asMovePackage": { @@ -375,16 +385,6 @@ Response: { } } } - }, - { - "outputState": { - "asMovePackage": null - } - }, - { - "outputState": { - "asMovePackage": null - } } ] } diff --git a/crates/iota-graphql-e2e-tests/tests/packages/structs.move b/crates/iota-graphql-e2e-tests/tests/packages/structs.move index abbe608e644..490897b2084 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/structs.move +++ b/crates/iota-graphql-e2e-tests/tests/packages/structs.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 P1=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses P0=0x0 P1=0x0 --accounts A --simulator //# run-graphql diff --git a/crates/iota-graphql-e2e-tests/tests/packages/types.exp b/crates/iota-graphql-e2e-tests/tests/packages/types.exp index f8cbbd79690..44f9dea3c65 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/types.exp +++ b/crates/iota-graphql-e2e-tests/tests/packages/types.exp @@ -275,7 +275,7 @@ Response: { "data": null, "errors": [ { - "message": "Internal error occurred while processing request: Error calculating layout for 0xe61abd7728c5a61dbd0ee1088e5686724388708bfd5bdc55e1e546a1e4cf92fd::m::S1: Type layout nesting exceeded limit of 128", + "message": "Internal error occurred while processing request: Error calculating layout for 0xb0f5f76d189b2ca371e1ea1ea11cecbaf049dd9b7782567aff1d024e02ac6f1a::m::S1: Type layout nesting exceeded limit of 128", "locations": [ { "line": 4, diff --git a/crates/iota-graphql-e2e-tests/tests/packages/types.move b/crates/iota-graphql-e2e-tests/tests/packages/types.move index 654aeba5954..2f547fbb9c3 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/types.move +++ b/crates/iota-graphql-e2e-tests/tests/packages/types.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 --simulator +//# init --protocol-version 3 --addresses P0=0x0 --simulator //# run-graphql # Happy path -- valid type, get everything diff --git a/crates/iota-graphql-e2e-tests/tests/packages/versioning.exp b/crates/iota-graphql-e2e-tests/tests/packages/versioning.exp index 9c380f32b9f..a4b64f00feb 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/versioning.exp +++ b/crates/iota-graphql-e2e-tests/tests/packages/versioning.exp @@ -31,14 +31,14 @@ Response: { "packageVersions": { "nodes": [ { - "address": "0xb9a97ea1dd36f9c94f0c363c663258dfe85c540c6f276d77e93421ad41bf8bcb", + "address": "0x3fb2634b6fd1f0c24c7c89776985e2713b0cab61b7f0c3bad3fa537c98b2ad8d", "version": 1 } ] } }, "firstPackage": { - "address": "0xb9a97ea1dd36f9c94f0c363c663258dfe85c540c6f276d77e93421ad41bf8bcb", + "address": "0x3fb2634b6fd1f0c24c7c89776985e2713b0cab61b7f0c3bad3fa537c98b2ad8d", "version": 1, "module": { "functions": { @@ -52,7 +52,7 @@ Response: { "packageVersions": { "nodes": [ { - "address": "0xb9a97ea1dd36f9c94f0c363c663258dfe85c540c6f276d77e93421ad41bf8bcb", + "address": "0x3fb2634b6fd1f0c24c7c89776985e2713b0cab61b7f0c3bad3fa537c98b2ad8d", "version": 1 } ] @@ -81,7 +81,7 @@ Response: { "version": 1 }, { - "address": "0xb9a97ea1dd36f9c94f0c363c663258dfe85c540c6f276d77e93421ad41bf8bcb", + "address": "0x3fb2634b6fd1f0c24c7c89776985e2713b0cab61b7f0c3bad3fa537c98b2ad8d", "version": 1 } ] @@ -120,18 +120,18 @@ Response: { "packageVersions": { "nodes": [ { - "address": "0xb9a97ea1dd36f9c94f0c363c663258dfe85c540c6f276d77e93421ad41bf8bcb", + "address": "0x3fb2634b6fd1f0c24c7c89776985e2713b0cab61b7f0c3bad3fa537c98b2ad8d", "version": 1 }, { - "address": "0x0f5c2c0da4e480bb873754b7c89ab178e2c328995b6cf89b836f08a164745acb", + "address": "0x9a446cce0055448dbf60df25fda9f0524967e073754b8273223f649199895388", "version": 2 } ] } }, "firstPackage": { - "address": "0xb9a97ea1dd36f9c94f0c363c663258dfe85c540c6f276d77e93421ad41bf8bcb", + "address": "0x3fb2634b6fd1f0c24c7c89776985e2713b0cab61b7f0c3bad3fa537c98b2ad8d", "version": 1, "module": { "functions": { @@ -145,11 +145,11 @@ Response: { "packageVersions": { "nodes": [ { - "address": "0xb9a97ea1dd36f9c94f0c363c663258dfe85c540c6f276d77e93421ad41bf8bcb", + "address": "0x3fb2634b6fd1f0c24c7c89776985e2713b0cab61b7f0c3bad3fa537c98b2ad8d", "version": 1 }, { - "address": "0x0f5c2c0da4e480bb873754b7c89ab178e2c328995b6cf89b836f08a164745acb", + "address": "0x9a446cce0055448dbf60df25fda9f0524967e073754b8273223f649199895388", "version": 2 } ] @@ -178,11 +178,11 @@ Response: { "version": 1 }, { - "address": "0xb9a97ea1dd36f9c94f0c363c663258dfe85c540c6f276d77e93421ad41bf8bcb", + "address": "0x3fb2634b6fd1f0c24c7c89776985e2713b0cab61b7f0c3bad3fa537c98b2ad8d", "version": 1 }, { - "address": "0x0f5c2c0da4e480bb873754b7c89ab178e2c328995b6cf89b836f08a164745acb", + "address": "0x9a446cce0055448dbf60df25fda9f0524967e073754b8273223f649199895388", "version": 2 } ] @@ -224,22 +224,22 @@ Response: { "packageVersions": { "nodes": [ { - "address": "0xb9a97ea1dd36f9c94f0c363c663258dfe85c540c6f276d77e93421ad41bf8bcb", + "address": "0x3fb2634b6fd1f0c24c7c89776985e2713b0cab61b7f0c3bad3fa537c98b2ad8d", "version": 1 }, { - "address": "0x0f5c2c0da4e480bb873754b7c89ab178e2c328995b6cf89b836f08a164745acb", + "address": "0x9a446cce0055448dbf60df25fda9f0524967e073754b8273223f649199895388", "version": 2 }, { - "address": "0xb4d563e260d9c18a8d6c929bb330e0dc1abfd55023e6b3e1d47524919b24e732", + "address": "0xe3d0bbdf8ce2be8a32ea519d8b62362a6e52288dfa2cb47fa7954b047b469c22", "version": 3 } ] } }, "firstPackage": { - "address": "0xb9a97ea1dd36f9c94f0c363c663258dfe85c540c6f276d77e93421ad41bf8bcb", + "address": "0x3fb2634b6fd1f0c24c7c89776985e2713b0cab61b7f0c3bad3fa537c98b2ad8d", "version": 1, "module": { "functions": { @@ -253,15 +253,15 @@ Response: { "packageVersions": { "nodes": [ { - "address": "0xb9a97ea1dd36f9c94f0c363c663258dfe85c540c6f276d77e93421ad41bf8bcb", + "address": "0x3fb2634b6fd1f0c24c7c89776985e2713b0cab61b7f0c3bad3fa537c98b2ad8d", "version": 1 }, { - "address": "0x0f5c2c0da4e480bb873754b7c89ab178e2c328995b6cf89b836f08a164745acb", + "address": "0x9a446cce0055448dbf60df25fda9f0524967e073754b8273223f649199895388", "version": 2 }, { - "address": "0xb4d563e260d9c18a8d6c929bb330e0dc1abfd55023e6b3e1d47524919b24e732", + "address": "0xe3d0bbdf8ce2be8a32ea519d8b62362a6e52288dfa2cb47fa7954b047b469c22", "version": 3 } ] @@ -290,15 +290,15 @@ Response: { "version": 1 }, { - "address": "0xb9a97ea1dd36f9c94f0c363c663258dfe85c540c6f276d77e93421ad41bf8bcb", + "address": "0x3fb2634b6fd1f0c24c7c89776985e2713b0cab61b7f0c3bad3fa537c98b2ad8d", "version": 1 }, { - "address": "0x0f5c2c0da4e480bb873754b7c89ab178e2c328995b6cf89b836f08a164745acb", + "address": "0x9a446cce0055448dbf60df25fda9f0524967e073754b8273223f649199895388", "version": 2 }, { - "address": "0xb4d563e260d9c18a8d6c929bb330e0dc1abfd55023e6b3e1d47524919b24e732", + "address": "0xe3d0bbdf8ce2be8a32ea519d8b62362a6e52288dfa2cb47fa7954b047b469c22", "version": 3 } ] @@ -715,7 +715,7 @@ Response: { "after": { "nodes": [ { - "address": "0x0f5c2c0da4e480bb873754b7c89ab178e2c328995b6cf89b836f08a164745acb", + "address": "0x9a446cce0055448dbf60df25fda9f0524967e073754b8273223f649199895388", "version": 2, "previousTransactionBlock": { "effects": { @@ -726,7 +726,7 @@ Response: { } }, { - "address": "0xb4d563e260d9c18a8d6c929bb330e0dc1abfd55023e6b3e1d47524919b24e732", + "address": "0xe3d0bbdf8ce2be8a32ea519d8b62362a6e52288dfa2cb47fa7954b047b469c22", "version": 3, "previousTransactionBlock": { "effects": { @@ -741,7 +741,7 @@ Response: { "between": { "nodes": [ { - "address": "0x0f5c2c0da4e480bb873754b7c89ab178e2c328995b6cf89b836f08a164745acb", + "address": "0x9a446cce0055448dbf60df25fda9f0524967e073754b8273223f649199895388", "version": 2, "previousTransactionBlock": { "effects": { @@ -763,15 +763,15 @@ Response: { "packageVersions": { "nodes": [ { - "address": "0xb9a97ea1dd36f9c94f0c363c663258dfe85c540c6f276d77e93421ad41bf8bcb", + "address": "0x3fb2634b6fd1f0c24c7c89776985e2713b0cab61b7f0c3bad3fa537c98b2ad8d", "version": 1 }, { - "address": "0x0f5c2c0da4e480bb873754b7c89ab178e2c328995b6cf89b836f08a164745acb", + "address": "0x9a446cce0055448dbf60df25fda9f0524967e073754b8273223f649199895388", "version": 2 }, { - "address": "0xb4d563e260d9c18a8d6c929bb330e0dc1abfd55023e6b3e1d47524919b24e732", + "address": "0xe3d0bbdf8ce2be8a32ea519d8b62362a6e52288dfa2cb47fa7954b047b469c22", "version": 3 } ] @@ -779,11 +779,11 @@ Response: { "after": { "nodes": [ { - "address": "0x0f5c2c0da4e480bb873754b7c89ab178e2c328995b6cf89b836f08a164745acb", + "address": "0x9a446cce0055448dbf60df25fda9f0524967e073754b8273223f649199895388", "version": 2 }, { - "address": "0xb4d563e260d9c18a8d6c929bb330e0dc1abfd55023e6b3e1d47524919b24e732", + "address": "0xe3d0bbdf8ce2be8a32ea519d8b62362a6e52288dfa2cb47fa7954b047b469c22", "version": 3 } ] @@ -791,11 +791,11 @@ Response: { "before": { "nodes": [ { - "address": "0xb9a97ea1dd36f9c94f0c363c663258dfe85c540c6f276d77e93421ad41bf8bcb", + "address": "0x3fb2634b6fd1f0c24c7c89776985e2713b0cab61b7f0c3bad3fa537c98b2ad8d", "version": 1 }, { - "address": "0x0f5c2c0da4e480bb873754b7c89ab178e2c328995b6cf89b836f08a164745acb", + "address": "0x9a446cce0055448dbf60df25fda9f0524967e073754b8273223f649199895388", "version": 2 } ] @@ -803,7 +803,7 @@ Response: { "between": { "nodes": [ { - "address": "0x0f5c2c0da4e480bb873754b7c89ab178e2c328995b6cf89b836f08a164745acb", + "address": "0x9a446cce0055448dbf60df25fda9f0524967e073754b8273223f649199895388", "version": 2 } ] diff --git a/crates/iota-graphql-e2e-tests/tests/packages/versioning.move b/crates/iota-graphql-e2e-tests/tests/packages/versioning.move index 21c88bce38e..02cd7c5e802 100644 --- a/crates/iota-graphql-e2e-tests/tests/packages/versioning.move +++ b/crates/iota-graphql-e2e-tests/tests/packages/versioning.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 P1=0x0 P2=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses P0=0x0 P1=0x0 P2=0x0 --accounts A --simulator //# publish --upgradeable --sender A module P0::m { diff --git a/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/balance_changes.exp b/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/balance_changes.exp index 397cfbbfa29..1f7fe20f748 100644 --- a/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/balance_changes.exp +++ b/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/balance_changes.exp @@ -48,7 +48,7 @@ Response: { "edges": [ { "node": { - "amount": "3000" + "amount": "5000" }, "cursor": "eyJpIjowLCJjIjoyfQ" }, @@ -66,19 +66,19 @@ Response: { }, { "node": { - "amount": "5000" + "amount": "3000" }, "cursor": "eyJpIjozLCJjIjoyfQ" }, { "node": { - "amount": "2000" + "amount": "4000" }, "cursor": "eyJpIjo0LCJjIjoyfQ" }, { "node": { - "amount": "4000" + "amount": "2000" }, "cursor": "eyJpIjo1LCJjIjoyfQ" } @@ -111,13 +111,13 @@ Response: { "edges": [ { "node": { - "amount": "5000" + "amount": "3000" }, "cursor": "eyJpIjozLCJjIjoxfQ" }, { "node": { - "amount": "2000" + "amount": "4000" }, "cursor": "eyJpIjo0LCJjIjoxfQ" } @@ -150,7 +150,7 @@ Response: { "edges": [ { "node": { - "amount": "3000" + "amount": "5000" }, "cursor": "eyJpIjowLCJjIjoxfQ" }, diff --git a/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/balance_changes.move b/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/balance_changes.move index 1b4157401ca..a769cc47a93 100644 --- a/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/balance_changes.move +++ b/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/balance_changes.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --simulator --accounts C O P Q R S +//# init --protocol-version 3 --simulator --accounts C O P Q R S //# programmable --sender C --inputs @C 1000 2000 3000 4000 5000 //> SplitCoins(Gas, [Input(1), Input(2), Input(3), Input(4), Input(5)]); diff --git a/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/dependencies.exp b/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/dependencies.exp index 2419d46d870..e9d9840d7af 100644 --- a/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/dependencies.exp +++ b/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/dependencies.exp @@ -65,7 +65,7 @@ Response: { "transactionBlocks": { "nodes": [ { - "digest": "Nj6ihexZgq92WRmwPhTFoHW5LqE8YKDYgBNj6uRve5S", + "digest": "6b1Kn6ooPLtqbcHyoHYneKUFwZ8Y2vgfwxnzYVscb3jx", "effects": { "dependencies": { "pageInfo": { @@ -78,26 +78,15 @@ Response: { { "cursor": "eyJpIjowLCJjIjoxfQ", "node": { - "digest": "5vV7Am92rPPM9VLDV2oqA6SzNTrMLt8V3yiCkWGYeYtb", + "digest": "CmL9WCz3trGP76o155Yi4nyAB3wVpxyLPnDbPu6jkeXQ", "kind": { "__typename": "ProgrammableTransactionBlock", "transactions": { "nodes": [ + {}, { - "module": "M1", - "functionName": "sum" - }, - { - "module": "M1", - "functionName": "sum" - }, - { - "module": "M1", - "functionName": "sum" - }, - { - "module": "M1", - "functionName": "create" + "module": "package", + "functionName": "make_immutable" } ] } @@ -107,15 +96,26 @@ Response: { { "cursor": "eyJpIjoxLCJjIjoxfQ", "node": { - "digest": "7DzPqWyTbh35vX6NZriBXftsRotmqRueMd7uib8VBrFt", + "digest": "E79gYn9CMr8F2TFJ5KjFsL6xLKTexVXz9Y1VQSecm9ap", "kind": { "__typename": "ProgrammableTransactionBlock", "transactions": { "nodes": [ - {}, { - "module": "package", - "functionName": "make_immutable" + "module": "M1", + "functionName": "sum" + }, + { + "module": "M1", + "functionName": "sum" + }, + { + "module": "M1", + "functionName": "sum" + }, + { + "module": "M1", + "functionName": "create" } ] } @@ -138,7 +138,7 @@ Response: { "transactionBlocks": { "nodes": [ { - "digest": "Nj6ihexZgq92WRmwPhTFoHW5LqE8YKDYgBNj6uRve5S", + "digest": "6b1Kn6ooPLtqbcHyoHYneKUFwZ8Y2vgfwxnzYVscb3jx", "effects": { "dependencies": { "pageInfo": { @@ -151,15 +151,26 @@ Response: { { "cursor": "eyJpIjoxLCJjIjoxfQ", "node": { - "digest": "7DzPqWyTbh35vX6NZriBXftsRotmqRueMd7uib8VBrFt", + "digest": "E79gYn9CMr8F2TFJ5KjFsL6xLKTexVXz9Y1VQSecm9ap", "kind": { "__typename": "ProgrammableTransactionBlock", "transactions": { "nodes": [ - {}, { - "module": "package", - "functionName": "make_immutable" + "module": "M1", + "functionName": "sum" + }, + { + "module": "M1", + "functionName": "sum" + }, + { + "module": "M1", + "functionName": "sum" + }, + { + "module": "M1", + "functionName": "create" } ] } diff --git a/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/dependencies.move b/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/dependencies.move index 2c880894ee6..12304eb3052 100644 --- a/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/dependencies.move +++ b/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/dependencies.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/events.move b/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/events.move index 3561024bf53..0ebbb8ee5ec 100644 --- a/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/events.move +++ b/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/events.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/object_changes.move b/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/object_changes.move index fb1adec95d0..6953a8c0e4a 100644 --- a/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/object_changes.move +++ b/crates/iota-graphql-e2e-tests/tests/transaction_block_effects/object_changes.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 A=0x42 --simulator --custom-validator-account --reference-gas-price 234 --default-gas-price 1000 +//# init --protocol-version 3 --addresses Test=0x0 A=0x42 --simulator --custom-validator-account --reference-gas-price 234 --default-gas-price 1000 //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/at_checkpoint.exp b/crates/iota-graphql-e2e-tests/tests/transactions/at_checkpoint.exp index 3bafc470eb3..a1a493ba525 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/at_checkpoint.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/at_checkpoint.exp @@ -30,7 +30,7 @@ Response: { "c0": { "nodes": [ { - "digest": "Bg37EsAT4gAL589upKSw92NUKGpxHk1ejmfkUoXdJzMo", + "digest": "HMpUemPGq81ZKbabWFBBberZ97ZQzZyd2ufocFpgaKxF", "kind": { "__typename": "GenesisTransaction" } @@ -46,7 +46,7 @@ Response: { } }, { - "digest": "973SQ6oCuqnAVPYxHBfki8ZbCj1fNLjoakbS3VjvTaPa", + "digest": "77LwCSZqEqoXZdYW5eAP1Zowenc5uFiXNUroLQqre8qW", "kind": { "__typename": "ProgrammableTransactionBlock" } @@ -87,7 +87,7 @@ Response: { "transactionBlocks": { "nodes": [ { - "digest": "Bg37EsAT4gAL589upKSw92NUKGpxHk1ejmfkUoXdJzMo", + "digest": "HMpUemPGq81ZKbabWFBBberZ97ZQzZyd2ufocFpgaKxF", "kind": { "__typename": "GenesisTransaction" } @@ -105,7 +105,7 @@ Response: { } }, { - "digest": "973SQ6oCuqnAVPYxHBfki8ZbCj1fNLjoakbS3VjvTaPa", + "digest": "77LwCSZqEqoXZdYW5eAP1Zowenc5uFiXNUroLQqre8qW", "kind": { "__typename": "ProgrammableTransactionBlock" } @@ -154,7 +154,7 @@ Response: { "transactionBlocks": { "nodes": [ { - "digest": "Bg37EsAT4gAL589upKSw92NUKGpxHk1ejmfkUoXdJzMo", + "digest": "HMpUemPGq81ZKbabWFBBberZ97ZQzZyd2ufocFpgaKxF", "kind": { "__typename": "GenesisTransaction" } @@ -172,7 +172,7 @@ Response: { } }, { - "digest": "973SQ6oCuqnAVPYxHBfki8ZbCj1fNLjoakbS3VjvTaPa", + "digest": "77LwCSZqEqoXZdYW5eAP1Zowenc5uFiXNUroLQqre8qW", "kind": { "__typename": "ProgrammableTransactionBlock" } diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/at_checkpoint.move b/crates/iota-graphql-e2e-tests/tests/transactions/at_checkpoint.move index ff2dd79fa5f..7bce26d1d5b 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/at_checkpoint.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/at_checkpoint.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --accounts A --simulator +//# init --protocol-version 3 --accounts A --simulator // Limiting transactions by the checkpoint they are in diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/errors.exp b/crates/iota-graphql-e2e-tests/tests/transactions/errors.exp index 5515b71fcc6..3a13572bcc0 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/errors.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/errors.exp @@ -25,7 +25,7 @@ Response: { { "effects": { "status": "FAILURE", - "errors": "Error in 1st command, from '0xb7d000e6ce024756ce5f7f7a50dc9ab34f9727d72b0ff528917b08087360bfed::m::boom' (instruction 1), abort code: 42" + "errors": "Error in 1st command, from '0x29bf31ca5680f5b89f41d9b5fac1c4bed644aa6a4d2ce5816ad0a97457204726::m::boom' (instruction 1), abort code: 42" } } ] @@ -54,7 +54,7 @@ Response: { { "effects": { "status": "FAILURE", - "errors": "Error in 3rd command, from '0xb7d000e6ce024756ce5f7f7a50dc9ab34f9727d72b0ff528917b08087360bfed::m::boom' (instruction 1), abort code: 42" + "errors": "Error in 3rd command, from '0x29bf31ca5680f5b89f41d9b5fac1c4bed644aa6a4d2ce5816ad0a97457204726::m::boom' (instruction 1), abort code: 42" } } ] diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/errors.move b/crates/iota-graphql-e2e-tests/tests/transactions/errors.move index af45de174f4..3ff8e57c1f1 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/errors.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/errors.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 --simulator +//# init --protocol-version 3 --addresses P0=0x0 --simulator //# publish diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/filters/kind.exp b/crates/iota-graphql-e2e-tests/tests/transactions/filters/kind.exp index ddb946f16fd..69c0ae28be8 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/filters/kind.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/filters/kind.exp @@ -60,7 +60,7 @@ Response: { }, "nodes": [ { - "digest": "DyfoVXSwG2QBCqLzozzoUBpcBzMtyqbeqxtrx16kKJ75", + "digest": "9KeLUgtFqgMkY9xSS3urrrzwNQcf2HKP4P7wjxauQANh", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -68,7 +68,7 @@ Response: { } }, { - "digest": "7aGyUPRRVEVDWCLuAhY56ZW8BrkWqkRhVo1jG2zbFdJw", + "digest": "4qdp62pQtjfoWHvNLQUXEsKyVEg85kEr2WRyQDoa5UKh", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -76,7 +76,7 @@ Response: { } }, { - "digest": "64FRv4LtnfGQ5fSQ3VsFmVdCyhU3Fm5rpsb3aYmEAwP4", + "digest": "C414Xw6dcKm2Wx1J7AeWmut3a5XQ33zuQWf4KdVwqyUF", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -84,7 +84,7 @@ Response: { } }, { - "digest": "EQHDejPTtkNQUbqmNrcyruptdePpxhBXoWxkrZJ1eS7r", + "digest": "8vA9JdWS1h1LEeHt494QPnhfxZdrS4nWYHLVmkJh49D6", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -92,7 +92,7 @@ Response: { } }, { - "digest": "GikQy5k3EytSDpKEdQLLzXyZmpzCAejQPyiBJJLJ8az4", + "digest": "FTLi7hzMz8uwbUvUJexJmbA4V4sKUc2XmM7sDcM9YYwd", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -117,7 +117,7 @@ Response: { }, "nodes": [ { - "digest": "DyfoVXSwG2QBCqLzozzoUBpcBzMtyqbeqxtrx16kKJ75", + "digest": "9KeLUgtFqgMkY9xSS3urrrzwNQcf2HKP4P7wjxauQANh", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -142,7 +142,7 @@ Response: { }, "nodes": [ { - "digest": "7aGyUPRRVEVDWCLuAhY56ZW8BrkWqkRhVo1jG2zbFdJw", + "digest": "4qdp62pQtjfoWHvNLQUXEsKyVEg85kEr2WRyQDoa5UKh", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -167,7 +167,7 @@ Response: { }, "nodes": [ { - "digest": "64FRv4LtnfGQ5fSQ3VsFmVdCyhU3Fm5rpsb3aYmEAwP4", + "digest": "C414Xw6dcKm2Wx1J7AeWmut3a5XQ33zuQWf4KdVwqyUF", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -192,7 +192,7 @@ Response: { }, "nodes": [ { - "digest": "EQHDejPTtkNQUbqmNrcyruptdePpxhBXoWxkrZJ1eS7r", + "digest": "8vA9JdWS1h1LEeHt494QPnhfxZdrS4nWYHLVmkJh49D6", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -217,7 +217,7 @@ Response: { }, "nodes": [ { - "digest": "GikQy5k3EytSDpKEdQLLzXyZmpzCAejQPyiBJJLJ8az4", + "digest": "FTLi7hzMz8uwbUvUJexJmbA4V4sKUc2XmM7sDcM9YYwd", "effects": { "checkpoint": { "sequenceNumber": 2 diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/filters/kind.move b/crates/iota-graphql-e2e-tests/tests/transactions/filters/kind.move index f93b8bc6345..99daec6dc4c 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/filters/kind.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/filters/kind.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B C D E --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B C D E --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/filters/transaction_ids.move b/crates/iota-graphql-e2e-tests/tests/transactions/filters/transaction_ids.move index 230fd6afb29..996d3a3ea7d 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/filters/transaction_ids.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/filters/transaction_ids.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/programmable.exp b/crates/iota-graphql-e2e-tests/tests/transactions/programmable.exp index ece5d01d0ae..466778a4d77 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/programmable.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/programmable.exp @@ -20,12 +20,12 @@ Response: { "transactionBlocks": { "nodes": [ { - "digest": "tvDgE7mmbtwGC6Kg1tyUptAZyjVxSxCpLPYmSSv7tLe", + "digest": "DStVeow9xBNhy4nZWSiMP4WZGsNaNFZQhWUzPHGnynyo", "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" }, "signatures": [ - "ABKf28j4W1zD01A94qW7ugvONKCQo3SWIeU025zhQFzKq5EI7FvY8fKAnxBlFfRM0SBj9V0+F5xflied0rGECQJ/UUY663bYjcm3XmNyULIgxJz1t5Z9vxfB+fp8WUoJKA==" + "ACWH/ZLI0uFUs4vyM4EJzhNIis0YFhMKDXsnOs0nUaywj7aPjj0udDW6A6IbAaJDeDsr9qrR6ZtcLNT5amXATgJ/UUY663bYjcm3XmNyULIgxJz1t5Z9vxfB+fp8WUoJKA==" ], "gasInput": { "gasSponsor": { @@ -34,7 +34,7 @@ Response: { "gasPayment": { "nodes": [ { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431" + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4" } ] }, @@ -96,7 +96,7 @@ Response: { "dependencies": { "nodes": [ { - "digest": "Bg37EsAT4gAL589upKSw92NUKGpxHk1ejmfkUoXdJzMo" + "digest": "HMpUemPGq81ZKbabWFBBberZ97ZQzZyd2ufocFpgaKxF" } ] }, @@ -116,37 +116,37 @@ Response: { "objectChanges": { "nodes": [ { - "address": "0x1b0961cd4993fc5cdc89e30b433a4de9cde01b12e15c8e2fb88754d653992822", - "idCreated": true, + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4", + "idCreated": false, "idDeleted": false, "outputState": { - "address": "0x1b0961cd4993fc5cdc89e30b433a4de9cde01b12e15c8e2fb88754d653992822", - "digest": "6C9ZpyBYqwBB1xzcMkoSmGybPA7FsStqMHgW6zEo1Snm" + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4", + "digest": "9et93fzXWULZboG1jbmPUY4Pjy5fdzYkXbZ2Tk3fff3Q" } }, { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431", - "idCreated": false, + "address": "0xa76aca10126d5d229060b1363c338e695a191c318bbbf54007151d6918134d65", + "idCreated": true, "idDeleted": false, "outputState": { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431", - "digest": "CpcjnZm8q5CZHgyAGAAQM6HaTcKtPiEUv7wBk44gpCh9" + "address": "0xa76aca10126d5d229060b1363c338e695a191c318bbbf54007151d6918134d65", + "digest": "2dSPbPpCYNgxAwhN35rNfZveTSjKRJiS7ndDWm3AsuRd" } }, { - "address": "0xfb3eadec34edec1ac979b6f1752b399ddc68480ffc15dcbe7163b0acc89c548e", + "address": "0xfa4b58cc517cbb110c3ff16e66b45e273d0d9c63ef70c5e7981a74b505136c1c", "idCreated": true, "idDeleted": false, "outputState": { - "address": "0xfb3eadec34edec1ac979b6f1752b399ddc68480ffc15dcbe7163b0acc89c548e", - "digest": "Eq9cwQBnFuUphwJmmkJnzMKdSDMG7TejrjjRbm8Pxrxk" + "address": "0xfa4b58cc517cbb110c3ff16e66b45e273d0d9c63ef70c5e7981a74b505136c1c", + "digest": "6SGX3tGkm1Mwo6XTCqAEyubcGgkfqvmqXYMq8XTxT5vu" } } ] }, "gasEffects": { "gasObject": { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431" + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4" }, "gasSummary": { "computationCost": "1000000", @@ -163,7 +163,7 @@ Response: { "sequenceNumber": 1 }, "transactionBlock": { - "digest": "tvDgE7mmbtwGC6Kg1tyUptAZyjVxSxCpLPYmSSv7tLe" + "digest": "DStVeow9xBNhy4nZWSiMP4WZGsNaNFZQhWUzPHGnynyo" } }, "expiration": null @@ -190,12 +190,12 @@ Response: { "transactionBlocks": { "nodes": [ { - "digest": "JCZvDpBa2RK5ypkUY5bwY2nTpTCpHUCc8DEz62B8nzL", + "digest": "AgriF36Cqis1HgoJTfpiCufw2R3cadyZaXqJiqWDhLzL", "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" }, "signatures": [ - "AATfZcSufT+0WT8TPvA21GyB7SWGr4L84+OZDLnfNILU446/Y4FUXZRfA8ZAlK83T23HuTDJM51bvufBlH3LJg1/UUY663bYjcm3XmNyULIgxJz1t5Z9vxfB+fp8WUoJKA==" + "AAYhpMoy694z8sKoB9lcxyPQqdxC9/T2Qe5F93FMA5HmjXHjn3drH+xngIIMqxWsqulu9VXnV300w+C7L7+IjA5/UUY663bYjcm3XmNyULIgxJz1t5Z9vxfB+fp8WUoJKA==" ], "gasInput": { "gasSponsor": { @@ -204,7 +204,7 @@ Response: { "gasPayment": { "nodes": [ { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431" + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4" } ] }, @@ -219,21 +219,21 @@ Response: { "cursor": "eyJpIjowLCJjIjoyfQ", "node": { "__typename": "OwnedOrImmutable", - "address": "0xfb3eadec34edec1ac979b6f1752b399ddc68480ffc15dcbe7163b0acc89c548e", + "address": "0xfa4b58cc517cbb110c3ff16e66b45e273d0d9c63ef70c5e7981a74b505136c1c", "version": 2, - "digest": "Eq9cwQBnFuUphwJmmkJnzMKdSDMG7TejrjjRbm8Pxrxk", + "digest": "6SGX3tGkm1Mwo6XTCqAEyubcGgkfqvmqXYMq8XTxT5vu", "object": { - "address": "0xfb3eadec34edec1ac979b6f1752b399ddc68480ffc15dcbe7163b0acc89c548e", + "address": "0xfa4b58cc517cbb110c3ff16e66b45e273d0d9c63ef70c5e7981a74b505136c1c", "version": 2, - "digest": "Eq9cwQBnFuUphwJmmkJnzMKdSDMG7TejrjjRbm8Pxrxk", + "digest": "6SGX3tGkm1Mwo6XTCqAEyubcGgkfqvmqXYMq8XTxT5vu", "asMoveObject": { "contents": { "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::package::UpgradeCap" }, "json": { - "id": "0xfb3eadec34edec1ac979b6f1752b399ddc68480ffc15dcbe7163b0acc89c548e", - "package": "0x1b0961cd4993fc5cdc89e30b433a4de9cde01b12e15c8e2fb88754d653992822", + "id": "0xfa4b58cc517cbb110c3ff16e66b45e273d0d9c63ef70c5e7981a74b505136c1c", + "package": "0xa76aca10126d5d229060b1363c338e695a191c318bbbf54007151d6918134d65", "version": "1", "policy": 0 } @@ -315,7 +315,7 @@ Response: { "0x0000000000000000000000000000000000000000000000000000000000000001", "0x0000000000000000000000000000000000000000000000000000000000000002" ], - "currentPackage": "0x1b0961cd4993fc5cdc89e30b433a4de9cde01b12e15c8e2fb88754d653992822", + "currentPackage": "0xa76aca10126d5d229060b1363c338e695a191c318bbbf54007151d6918134d65", "upgradeTicket": { "__typename": "Result", "cmd": 0, @@ -367,10 +367,10 @@ Response: { "dependencies": { "nodes": [ { - "digest": "tvDgE7mmbtwGC6Kg1tyUptAZyjVxSxCpLPYmSSv7tLe" + "digest": "DStVeow9xBNhy4nZWSiMP4WZGsNaNFZQhWUzPHGnynyo" }, { - "digest": "Bg37EsAT4gAL589upKSw92NUKGpxHk1ejmfkUoXdJzMo" + "digest": "HMpUemPGq81ZKbabWFBBberZ97ZQzZyd2ufocFpgaKxF" } ] }, @@ -390,37 +390,37 @@ Response: { "objectChanges": { "nodes": [ { - "address": "0xc4d534b3b62e9e99d3b8a20abc757dccfc3b7a32f6fff8e47cf8cf1cd906cf64", - "idCreated": true, + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4", + "idCreated": false, "idDeleted": false, "outputState": { - "address": "0xc4d534b3b62e9e99d3b8a20abc757dccfc3b7a32f6fff8e47cf8cf1cd906cf64", - "digest": "2NMQpbTkNrUxaTApQHYhN8z4jJj9bs8DFC4g7Pb5psoP" + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4", + "digest": "8c2bwMXrgv7wXFLN16zvdAdzsUffBb5UmP1cyfzswLu6" } }, { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431", - "idCreated": false, + "address": "0x92307478b83faee47dae9846622016f8c5d86bec5d72333b5d75c739b986c025", + "idCreated": true, "idDeleted": false, "outputState": { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431", - "digest": "HSn5KHX97NhZi1ovsPZQbRgYdfYnTwmE6aamc6o2bxYq" + "address": "0x92307478b83faee47dae9846622016f8c5d86bec5d72333b5d75c739b986c025", + "digest": "RnXt4c1zEop2mo3eGaCVVYqJmNBvXLSsa5cozxAnnWY" } }, { - "address": "0xfb3eadec34edec1ac979b6f1752b399ddc68480ffc15dcbe7163b0acc89c548e", + "address": "0xfa4b58cc517cbb110c3ff16e66b45e273d0d9c63ef70c5e7981a74b505136c1c", "idCreated": false, "idDeleted": false, "outputState": { - "address": "0xfb3eadec34edec1ac979b6f1752b399ddc68480ffc15dcbe7163b0acc89c548e", - "digest": "HH96DACodGczUapLY4eCrByvdAzg5iadrjtadPtj6EXw" + "address": "0xfa4b58cc517cbb110c3ff16e66b45e273d0d9c63ef70c5e7981a74b505136c1c", + "digest": "D6bV2KX6nqdty574rDYjKoigVfDB7UUPGqd4mCLV4WXD" } } ] }, "gasEffects": { "gasObject": { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431" + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4" }, "gasSummary": { "computationCost": "1000000", @@ -437,7 +437,7 @@ Response: { "sequenceNumber": 2 }, "transactionBlock": { - "digest": "JCZvDpBa2RK5ypkUY5bwY2nTpTCpHUCc8DEz62B8nzL" + "digest": "AgriF36Cqis1HgoJTfpiCufw2R3cadyZaXqJiqWDhLzL" } }, "expiration": null @@ -482,12 +482,12 @@ Response: { "transactionBlocks": { "nodes": [ { - "digest": "FiPjt6NaBmwTXNSQJDPvTEfwoWJQ1tfpe8P87ajeKfn4", + "digest": "72hb62Xx8UCvvfvsDpQntCUnr9EGFPHUVohx8YpN5iGS", "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" }, "signatures": [ - "AImFtPn31VRGRTd3pkm8Ho2hfbGtkyJHUyPyDFPCw5wSYbc4DK0zM2CfP4znyyquxSTZrkbhdc68v6DBVrGc0wN/UUY663bYjcm3XmNyULIgxJz1t5Z9vxfB+fp8WUoJKA==" + "AME48u1BDowW3Jv0BjdpKnIe1a7m2cGL5B4Ig7E25EQ0FoH+8jF4k6qzPHTn9+Et1gbTLBJ9MAE2JWPmbhHaEAV/UUY663bYjcm3XmNyULIgxJz1t5Z9vxfB+fp8WUoJKA==" ], "gasInput": { "gasSponsor": { @@ -496,7 +496,7 @@ Response: { "gasPayment": { "nodes": [ { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431" + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4" } ] }, @@ -591,7 +591,7 @@ Response: { "cursor": "eyJpIjozLCJjIjozfQ", "node": { "__typename": "MoveCallTransaction", - "package": "0xc4d534b3b62e9e99d3b8a20abc757dccfc3b7a32f6fff8e47cf8cf1cd906cf64", + "package": "0x92307478b83faee47dae9846622016f8c5d86bec5d72333b5d75c739b986c025", "module": "m", "functionName": "new", "typeArguments": [], @@ -615,7 +615,7 @@ Response: { ], "return": [ { - "repr": "0x1b0961cd4993fc5cdc89e30b433a4de9cde01b12e15c8e2fb88754d653992822::m::Foo" + "repr": "0xa76aca10126d5d229060b1363c338e695a191c318bbbf54007151d6918134d65::m::Foo" } ] } @@ -642,7 +642,7 @@ Response: { "cursor": "eyJpIjo1LCJjIjozfQ", "node": { "__typename": "MoveCallTransaction", - "package": "0xc4d534b3b62e9e99d3b8a20abc757dccfc3b7a32f6fff8e47cf8cf1cd906cf64", + "package": "0x92307478b83faee47dae9846622016f8c5d86bec5d72333b5d75c739b986c025", "module": "m", "functionName": "new", "typeArguments": [], @@ -666,7 +666,7 @@ Response: { ], "return": [ { - "repr": "0x1b0961cd4993fc5cdc89e30b433a4de9cde01b12e15c8e2fb88754d653992822::m::Foo" + "repr": "0xa76aca10126d5d229060b1363c338e695a191c318bbbf54007151d6918134d65::m::Foo" } ] } @@ -676,7 +676,7 @@ Response: { "cursor": "eyJpIjo2LCJjIjozfQ", "node": { "__typename": "MoveCallTransaction", - "package": "0xc4d534b3b62e9e99d3b8a20abc757dccfc3b7a32f6fff8e47cf8cf1cd906cf64", + "package": "0x92307478b83faee47dae9846622016f8c5d86bec5d72333b5d75c739b986c025", "module": "m", "functionName": "burn", "typeArguments": [], @@ -692,7 +692,7 @@ Response: { "typeParameters": [], "parameters": [ { - "repr": "0x1b0961cd4993fc5cdc89e30b433a4de9cde01b12e15c8e2fb88754d653992822::m::Foo" + "repr": "0xa76aca10126d5d229060b1363c338e695a191c318bbbf54007151d6918134d65::m::Foo" } ], "return": [] @@ -744,10 +744,10 @@ Response: { "dependencies": { "nodes": [ { - "digest": "JCZvDpBa2RK5ypkUY5bwY2nTpTCpHUCc8DEz62B8nzL" + "digest": "8yToACVC5ZQqVbjeTvjwSMugbuw1gBbngHmVcJiy6Wfh" }, { - "digest": "GhnCeKNzadasPxvEHpaNqHYrBn72Ge9vYoFU3h4tAeBv" + "digest": "AgriF36Cqis1HgoJTfpiCufw2R3cadyZaXqJiqWDhLzL" } ] }, @@ -767,19 +767,19 @@ Response: { "objectChanges": { "nodes": [ { - "address": "0xb4e14ddd214a4ddb9acca83017dc8fd05f8ad433e4d7e6676087e6c0368e68a0", + "address": "0x04df2cc664f044d04926fdd64a874053746e122278680308150f60b567b20940", "idCreated": true, "idDeleted": false, "outputState": { - "address": "0xb4e14ddd214a4ddb9acca83017dc8fd05f8ad433e4d7e6676087e6c0368e68a0", - "digest": "8qv3yd7bS9w4MqkxCgjzGe6QHzWnqBYC7yu84FLJi1G1", + "address": "0x04df2cc664f044d04926fdd64a874053746e122278680308150f60b567b20940", + "digest": "2xZhu1r8zVrrjx6dTGmhyd3hUspYCmLURzNT9vGGM5nC", "asMoveObject": { "contents": { "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" }, "json": { - "id": "0xb4e14ddd214a4ddb9acca83017dc8fd05f8ad433e4d7e6676087e6c0368e68a0", + "id": "0x04df2cc664f044d04926fdd64a874053746e122278680308150f60b567b20940", "balance": { "value": "2000" } @@ -789,45 +789,45 @@ Response: { } }, { - "address": "0xb83a5a53f7d1dd5b87d9d829da34ef11375bf8acae96f8ca7fe5dfbf49c87d14", - "idCreated": true, + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4", + "idCreated": false, "idDeleted": false, "outputState": { - "address": "0xb83a5a53f7d1dd5b87d9d829da34ef11375bf8acae96f8ca7fe5dfbf49c87d14", - "digest": "7DJMVrZRXv1iuuVTNmv8LEuuMBSoTLw9h6AHxFGHbWE7", + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4", + "digest": "BvV2DJy7JAHKjTFgTFri7GrHN6GnEEjZovCBMpdGuMXC", "asMoveObject": { "contents": { "type": { - "repr": "0x1b0961cd4993fc5cdc89e30b433a4de9cde01b12e15c8e2fb88754d653992822::m::Foo" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" }, "json": { - "id": "0xb83a5a53f7d1dd5b87d9d829da34ef11375bf8acae96f8ca7fe5dfbf49c87d14", - "xs": [ - "42", - "43" - ] + "id": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4", + "balance": { + "value": "299999982120400" + } } } } } }, { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431", - "idCreated": false, + "address": "0x223be7efd160093e62a1b704b39bf7c0b99ef9a213eb12766693c18829eb8358", + "idCreated": true, "idDeleted": false, "outputState": { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431", - "digest": "739cLjqpEeSvaT8Ta9qXAVV2bAFRXcmf9F6Zt3cCnXij", + "address": "0x223be7efd160093e62a1b704b39bf7c0b99ef9a213eb12766693c18829eb8358", + "digest": "n1Vsn9aEWScvXM5LdPHt5i4x9dz27TND53bvgVvZP8g", "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" + "repr": "0xa76aca10126d5d229060b1363c338e695a191c318bbbf54007151d6918134d65::m::Foo" }, "json": { - "id": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431", - "balance": { - "value": "299999982120400" - } + "id": "0x223be7efd160093e62a1b704b39bf7c0b99ef9a213eb12766693c18829eb8358", + "xs": [ + "42", + "43" + ] } } } @@ -837,7 +837,7 @@ Response: { }, "gasEffects": { "gasObject": { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431" + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4" }, "gasSummary": { "computationCost": "1000000", @@ -854,7 +854,7 @@ Response: { "sequenceNumber": 3 }, "transactionBlock": { - "digest": "FiPjt6NaBmwTXNSQJDPvTEfwoWJQ1tfpe8P87ajeKfn4" + "digest": "72hb62Xx8UCvvfvsDpQntCUnr9EGFPHUVohx8YpN5iGS" } }, "expiration": null @@ -881,12 +881,12 @@ Response: { "transactionBlocks": { "nodes": [ { - "digest": "65qUXtK1a5gysqwhVKmxj5VQd3YndguWE1q9V7dZyVez", + "digest": "Ffp1DhEFmSMobvN7x83LN18dsj4qH8Tv6f74vqntrA32", "sender": { "address": "0x8cca4e1ce0ba5904cea61df9242da2f7d29e3ef328fb7ec07c086b3bf47ca61a" }, "signatures": [ - "ANpvkSQpMbmKm+dKBWFWsUa1GUaKYomXv9GXi3cfVe4ek3G9xb0AzX27HMyl7Odsbw+wyYuW30RoRH0tmBjXwQl/UUY663bYjcm3XmNyULIgxJz1t5Z9vxfB+fp8WUoJKA==" + "ALH9L5nJjP7Cz9X1aqT2ZsV1Ry2wZ7+0bhQFcAm3OB8dLGtEu75zArMTz/28KfjFC5tzJ86m/vO2T+nXaaly3Qh/UUY663bYjcm3XmNyULIgxJz1t5Z9vxfB+fp8WUoJKA==" ], "gasInput": { "gasSponsor": { @@ -895,7 +895,7 @@ Response: { "gasPayment": { "nodes": [ { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431" + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4" } ] }, @@ -942,7 +942,7 @@ Response: { "dependencies": { "nodes": [ { - "digest": "FiPjt6NaBmwTXNSQJDPvTEfwoWJQ1tfpe8P87ajeKfn4" + "digest": "72hb62Xx8UCvvfvsDpQntCUnr9EGFPHUVohx8YpN5iGS" } ] }, @@ -962,19 +962,19 @@ Response: { "objectChanges": { "nodes": [ { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431", + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4", "idCreated": false, "idDeleted": false, "outputState": { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431", - "digest": "EXucdLBcKqADNerCwsbSi5PjvyFRjYn4wLB5Binbc4jZ" + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4", + "digest": "5ysmth2HaqLv1xLbCMQUnhd8qVZtEvyhRZ4M8mPMF9qc" } } ] }, "gasEffects": { "gasObject": { - "address": "0xc69f1954c6d13e3630075f462966e06d9f05fec29846d3d781026187e5411431" + "address": "0x18a5cb688685554c4b370c3f551466dba6fb25390920fa5836812a9fb66500c4" }, "gasSummary": { "computationCost": "1000000", @@ -991,7 +991,7 @@ Response: { "sequenceNumber": 4 }, "transactionBlock": { - "digest": "65qUXtK1a5gysqwhVKmxj5VQd3YndguWE1q9V7dZyVez" + "digest": "Ffp1DhEFmSMobvN7x83LN18dsj4qH8Tv6f74vqntrA32" } }, "expiration": null diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/programmable.move b/crates/iota-graphql-e2e-tests/tests/transactions/programmable.move index 5ebb235513f..e760cce72e5 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/programmable.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/programmable.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 P1=0x0 --accounts A --simulator +//# init --protocol-version 3 --addresses P0=0x0 P1=0x0 --accounts A --simulator //# publish --upgradeable --sender A module P0::m { diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/random.exp b/crates/iota-graphql-e2e-tests/tests/transactions/random.exp index a8828476838..602d0f9713f 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/random.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/random.exp @@ -19,7 +19,7 @@ Response: { "json": { "id": "0x0000000000000000000000000000000000000000000000000000000000000008", "inner": { - "id": "0x3f92eceed239ba443daf612dcdf621f85feba71f3698d53c9f2a6e4b4246b1c6", + "id": "0x087e07e48f02e1757a9a06af5c2be59dfc1a5717fe305d79fd77c5b3350426bc", "version": "1" } } diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/random.move b/crates/iota-graphql-e2e-tests/tests/transactions/random.move index fe4523fba2e..df163923132 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/random.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/random.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --simulator +//# init --protocol-version 3 --simulator //# create-checkpoint diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/alternating.exp b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/alternating.exp index 157691e81c5..2cdfabfbb75 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/alternating.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/alternating.exp @@ -96,7 +96,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "EnWpFFgRUBm3LbuTfjJ6oF3RXVcnzpth6L9nn4Q18RNk", + "digest": "HaHDjFPNFmzEzGV15jqjHArnNQYN3wFCehawzgSjTPDC", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -124,7 +124,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo0LCJpIjpmYWxzZX0", "node": { - "digest": "613degbde4DuvwwAZcQngGJzZkZ6GZJ9g8CwygpA1s1a", + "digest": "5p4zvwg7nsiUAUdYdzoB9swbX49YkwxfoCVLYvyUQvw5", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -135,7 +135,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo2LCJpIjpmYWxzZX0", "node": { - "digest": "AYmLYSfPbk4sVdBLcV76UmZgPTETP7XsVCaNKHTJ5qzs", + "digest": "57kQ1GWYJ16691C3z6roPwP9zoH4ZZXXcopwdp5E3jn2", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -163,7 +163,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo0LCJpIjpmYWxzZX0", "node": { - "digest": "613degbde4DuvwwAZcQngGJzZkZ6GZJ9g8CwygpA1s1a", + "digest": "5p4zvwg7nsiUAUdYdzoB9swbX49YkwxfoCVLYvyUQvw5", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -191,7 +191,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo4LCJpIjpmYWxzZX0", "node": { - "digest": "4kA8BHJRwCdE8BjFTpTfSGt9nvoii37twZRM91uxMuvi", + "digest": "HmHRDzEysmLNnMmwwFfehYukrz8hBKdV35W27g2dGcA6", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -219,7 +219,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo2LCJpIjpmYWxzZX0", "node": { - "digest": "AYmLYSfPbk4sVdBLcV76UmZgPTETP7XsVCaNKHTJ5qzs", + "digest": "57kQ1GWYJ16691C3z6roPwP9zoH4ZZXXcopwdp5E3jn2", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -230,7 +230,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo4LCJpIjpmYWxzZX0", "node": { - "digest": "4kA8BHJRwCdE8BjFTpTfSGt9nvoii37twZRM91uxMuvi", + "digest": "HmHRDzEysmLNnMmwwFfehYukrz8hBKdV35W27g2dGcA6", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -258,7 +258,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoxMCwiaSI6ZmFsc2V9", "node": { - "digest": "Dqi2D27TQ531EiXzEonQGrANt8kJh9w5JZ6SMtmRpm9b", + "digest": "Ftg9H88N7ioz51UuwfzvksAwaKnD6izS6kZKVLs5kPoh", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -286,7 +286,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoxMCwiaSI6ZmFsc2V9", "node": { - "digest": "Dqi2D27TQ531EiXzEonQGrANt8kJh9w5JZ6SMtmRpm9b", + "digest": "Ftg9H88N7ioz51UuwfzvksAwaKnD6izS6kZKVLs5kPoh", "effects": { "checkpoint": { "sequenceNumber": 3 diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/alternating.move b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/alternating.move index e5fafe88635..003c0ed37d1 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/alternating.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/alternating.move @@ -4,7 +4,7 @@ // Testing behavior of alternating between a scan-limited and normal query -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/both_cursors.exp b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/both_cursors.exp index 00684e0720f..92dae435b70 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/both_cursors.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/both_cursors.exp @@ -96,7 +96,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0Ijo0LCJpIjpmYWxzZX0", "node": { - "digest": "CQYnSfH6y3QuQjbzBYRDRYPr8sNMYp34tLf2NES2SdGm", + "digest": "5FvPDk1jtQos4igBEUNxZJyLUC475rXDuvTviKvYiy6i", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -124,7 +124,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0Ijo0LCJpIjpmYWxzZX0", "node": { - "digest": "CQYnSfH6y3QuQjbzBYRDRYPr8sNMYp34tLf2NES2SdGm", + "digest": "5FvPDk1jtQos4igBEUNxZJyLUC475rXDuvTviKvYiy6i", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -135,7 +135,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0Ijo2LCJpIjpmYWxzZX0", "node": { - "digest": "7jDvB5qNfUzvhzSer2KBHYwYvuXwfRemZ6TjpUZvcEjA", + "digest": "DwrJyK3daKxKED9MdN6EK1qeEUsrM2q88dgfPHJ4SqyJ", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -163,7 +163,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0Ijo1LCJpIjpmYWxzZX0", "node": { - "digest": "6hvDK848GxrWavhtQx6613ALsYctTEayGjqJw6z1qDvJ", + "digest": "5XRZD7sWxeBri7eWzrG9y4gWq75zX66dnzU4g82BcSea", "effects": { "checkpoint": { "sequenceNumber": 2 diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/both_cursors.move b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/both_cursors.move index 1361c415061..46bc2fae2df 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/both_cursors.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/both_cursors.move @@ -5,7 +5,7 @@ // Tests paginating forwards where first and scanLimit are equal. The 1st, 3rd, 5th, and 7th through // 10th transactions will match the filtering criteria. -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/first.exp b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/first.exp index 4f313117c03..3ceb7dba444 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/first.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/first.exp @@ -96,7 +96,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "ABweY1pDPKFhZX77kk48ofYU9KpAvXHBZUYyPK8JZf8p", + "digest": "EbDGGG7J7UrDdsn6F1C8puthPhTAyCbP8zJamEmXrvH4", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -107,7 +107,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo0LCJpIjpmYWxzZX0", "node": { - "digest": "CQYnSfH6y3QuQjbzBYRDRYPr8sNMYp34tLf2NES2SdGm", + "digest": "5FvPDk1jtQos4igBEUNxZJyLUC475rXDuvTviKvYiy6i", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -118,7 +118,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo2LCJpIjpmYWxzZX0", "node": { - "digest": "7jDvB5qNfUzvhzSer2KBHYwYvuXwfRemZ6TjpUZvcEjA", + "digest": "DwrJyK3daKxKED9MdN6EK1qeEUsrM2q88dgfPHJ4SqyJ", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -129,7 +129,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoxMCwiaSI6ZmFsc2V9", "node": { - "digest": "66KKZQ7bxz9uM5Jhc651Txh57S4s9xJMDJs6dmBkeDUz", + "digest": "CAuCiWwxT74gQYmGNLM3irCjw7AQ8dpNL9kgtRu9MrZm", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -140,7 +140,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoxMSwiaSI6ZmFsc2V9", "node": { - "digest": "78g29WY1ZjXGJmRjGWQNKnowtmDvftTDsQd9HbpUBn7g", + "digest": "3Xzrd8CC4UpV1fiPCc3VyBJC93HAAwL872CgtTNM3hyj", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -168,7 +168,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "ABweY1pDPKFhZX77kk48ofYU9KpAvXHBZUYyPK8JZf8p", + "digest": "EbDGGG7J7UrDdsn6F1C8puthPhTAyCbP8zJamEmXrvH4", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -196,7 +196,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0Ijo0LCJpIjpmYWxzZX0", "node": { - "digest": "CQYnSfH6y3QuQjbzBYRDRYPr8sNMYp34tLf2NES2SdGm", + "digest": "5FvPDk1jtQos4igBEUNxZJyLUC475rXDuvTviKvYiy6i", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -224,7 +224,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0Ijo2LCJpIjpmYWxzZX0", "node": { - "digest": "7jDvB5qNfUzvhzSer2KBHYwYvuXwfRemZ6TjpUZvcEjA", + "digest": "DwrJyK3daKxKED9MdN6EK1qeEUsrM2q88dgfPHJ4SqyJ", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -268,7 +268,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0IjoxMCwiaSI6ZmFsc2V9", "node": { - "digest": "66KKZQ7bxz9uM5Jhc651Txh57S4s9xJMDJs6dmBkeDUz", + "digest": "CAuCiWwxT74gQYmGNLM3irCjw7AQ8dpNL9kgtRu9MrZm", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -279,7 +279,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0IjoxMSwiaSI6ZmFsc2V9", "node": { - "digest": "78g29WY1ZjXGJmRjGWQNKnowtmDvftTDsQd9HbpUBn7g", + "digest": "3Xzrd8CC4UpV1fiPCc3VyBJC93HAAwL872CgtTNM3hyj", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -333,7 +333,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0IjoxMCwiaSI6ZmFsc2V9", "node": { - "digest": "66KKZQ7bxz9uM5Jhc651Txh57S4s9xJMDJs6dmBkeDUz", + "digest": "CAuCiWwxT74gQYmGNLM3irCjw7AQ8dpNL9kgtRu9MrZm", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -344,7 +344,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0IjoxMSwiaSI6ZmFsc2V9", "node": { - "digest": "78g29WY1ZjXGJmRjGWQNKnowtmDvftTDsQd9HbpUBn7g", + "digest": "3Xzrd8CC4UpV1fiPCc3VyBJC93HAAwL872CgtTNM3hyj", "effects": { "checkpoint": { "sequenceNumber": 3 diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/first.move b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/first.move index 2a2277647cc..2c19d64df48 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/first.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/first.move @@ -5,7 +5,7 @@ // Tests paginating forwards where first and scanLimit are equal. The 1st, 3rd, 5th, and 7th through // 10th transactions will match the filtering criteria. -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/last.exp b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/last.exp index f70d192b281..7b02f4f3725 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/last.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/last.exp @@ -96,7 +96,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "ABweY1pDPKFhZX77kk48ofYU9KpAvXHBZUYyPK8JZf8p", + "digest": "EbDGGG7J7UrDdsn6F1C8puthPhTAyCbP8zJamEmXrvH4", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -107,7 +107,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjozLCJpIjpmYWxzZX0", "node": { - "digest": "8v4rn79YoSv3Drbkqbz8ttsX7EJDPEoDNbvLXpZnNXvE", + "digest": "ovzzHxewTojtxtEmjJNWFJZuD8MuP6NDxUafmf4z8tf", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -118,7 +118,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo3LCJpIjpmYWxzZX0", "node": { - "digest": "6ZXmX1p1yNyfqE4fhJKqQAjv1CFfM6qJoF5MwAFc578k", + "digest": "HwhPdVvSZy2tgdA7siWYSm1JcaMJ6Cjp6YhriCQhigsg", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -129,7 +129,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo5LCJpIjpmYWxzZX0", "node": { - "digest": "EWYMoiDshwh4KMAXKQPcHUxfCHMx5zvsJ249L36moUWD", + "digest": "Az5dLVuYkHQ2T7b7ETHyLjye33TKnCRMmbxazpuBd9o1", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -140,7 +140,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoxMSwiaSI6ZmFsc2V9", "node": { - "digest": "BcK8vqpPuVeCkSM5LAEanttduQNAyhiK6psakqN8dyYY", + "digest": "BqVcj3fjjaNCaCsHF9MAfMok2MhDHnQkMKTVVtL5kRvb", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -168,7 +168,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoxMSwiaSI6ZmFsc2V9", "node": { - "digest": "BcK8vqpPuVeCkSM5LAEanttduQNAyhiK6psakqN8dyYY", + "digest": "BqVcj3fjjaNCaCsHF9MAfMok2MhDHnQkMKTVVtL5kRvb", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -196,7 +196,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0Ijo5LCJpIjpmYWxzZX0", "node": { - "digest": "EWYMoiDshwh4KMAXKQPcHUxfCHMx5zvsJ249L36moUWD", + "digest": "Az5dLVuYkHQ2T7b7ETHyLjye33TKnCRMmbxazpuBd9o1", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -224,7 +224,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0Ijo3LCJpIjpmYWxzZX0", "node": { - "digest": "6ZXmX1p1yNyfqE4fhJKqQAjv1CFfM6qJoF5MwAFc578k", + "digest": "HwhPdVvSZy2tgdA7siWYSm1JcaMJ6Cjp6YhriCQhigsg", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -268,7 +268,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "ABweY1pDPKFhZX77kk48ofYU9KpAvXHBZUYyPK8JZf8p", + "digest": "EbDGGG7J7UrDdsn6F1C8puthPhTAyCbP8zJamEmXrvH4", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -279,7 +279,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0IjozLCJpIjpmYWxzZX0", "node": { - "digest": "8v4rn79YoSv3Drbkqbz8ttsX7EJDPEoDNbvLXpZnNXvE", + "digest": "ovzzHxewTojtxtEmjJNWFJZuD8MuP6NDxUafmf4z8tf", "effects": { "checkpoint": { "sequenceNumber": 2 diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/last.move b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/last.move index 089a36ffc93..ad3c30a3549 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/last.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/equal/last.move @@ -4,7 +4,7 @@ // Mirrors scan_limit/equal/first.move, paginating backwards where first and scanLimit are equal. -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/first.exp b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/first.exp index ef865977112..3f98bcccf25 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/first.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/first.exp @@ -164,7 +164,7 @@ Response: { { "cursor": "eyJjIjo1LCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "EnWpFFgRUBm3LbuTfjJ6oF3RXVcnzpth6L9nn4Q18RNk", + "digest": "HaHDjFPNFmzEzGV15jqjHArnNQYN3wFCehawzgSjTPDC", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -175,7 +175,7 @@ Response: { { "cursor": "eyJjIjo1LCJ0IjozLCJpIjpmYWxzZX0", "node": { - "digest": "3fmkhmMgShCMM9HWgLR13ovdiBTy2Zoe6MgVfy8Wi8nt", + "digest": "FDiPztMqvdBBRZTteyNf6wKbryrRMrqSXtS3LpWxtKqZ", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -186,7 +186,7 @@ Response: { { "cursor": "eyJjIjo1LCJ0IjoxNywiaSI6ZmFsc2V9", "node": { - "digest": "8NagxHteFzUuAvwvzjFAorD1dNfqfKnqwwviDaAt8j1z", + "digest": "DLGGXHtEkErS86wgoWD3AiLzWJqRWwpw4VCgE2XZJQUN", "effects": { "checkpoint": { "sequenceNumber": 5 @@ -197,7 +197,7 @@ Response: { { "cursor": "eyJjIjo1LCJ0IjoxOCwiaSI6ZmFsc2V9", "node": { - "digest": "Ep8Q9KBacxFkYEvaxsm7CJAUqQ85WUi5z6oZudBEyFSP", + "digest": "8vmaeHkAP6N612bY4bWhWiuoTjdJejuzCdrwQ2Sougu", "effects": { "checkpoint": { "sequenceNumber": 5 @@ -208,7 +208,7 @@ Response: { { "cursor": "eyJjIjo1LCJ0IjoyMSwiaSI6ZmFsc2V9", "node": { - "digest": "636Q9XJw6aZQ4RGEnej49Z99Nn21GgPBQDmCGGhNQ2Rm", + "digest": "CRPK78UpxQyyLXcizjY81vBG4bZNG6yQ9jbda9iaSnVy", "effects": { "checkpoint": { "sequenceNumber": 5 @@ -236,7 +236,7 @@ Response: { { "cursor": "eyJjIjo1LCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "EnWpFFgRUBm3LbuTfjJ6oF3RXVcnzpth6L9nn4Q18RNk", + "digest": "HaHDjFPNFmzEzGV15jqjHArnNQYN3wFCehawzgSjTPDC", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -264,7 +264,7 @@ Response: { { "cursor": "eyJjIjo3LCJ0IjozLCJpIjpmYWxzZX0", "node": { - "digest": "3fmkhmMgShCMM9HWgLR13ovdiBTy2Zoe6MgVfy8Wi8nt", + "digest": "FDiPztMqvdBBRZTteyNf6wKbryrRMrqSXtS3LpWxtKqZ", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -308,7 +308,7 @@ Response: { { "cursor": "eyJjIjo3LCJ0IjoxNywiaSI6ZmFsc2V9", "node": { - "digest": "8NagxHteFzUuAvwvzjFAorD1dNfqfKnqwwviDaAt8j1z", + "digest": "DLGGXHtEkErS86wgoWD3AiLzWJqRWwpw4VCgE2XZJQUN", "effects": { "checkpoint": { "sequenceNumber": 5 @@ -336,7 +336,7 @@ Response: { { "cursor": "eyJjIjo3LCJ0IjoxOCwiaSI6ZmFsc2V9", "node": { - "digest": "Ep8Q9KBacxFkYEvaxsm7CJAUqQ85WUi5z6oZudBEyFSP", + "digest": "8vmaeHkAP6N612bY4bWhWiuoTjdJejuzCdrwQ2Sougu", "effects": { "checkpoint": { "sequenceNumber": 5 @@ -364,7 +364,7 @@ Response: { { "cursor": "eyJjIjo3LCJ0IjoyMSwiaSI6ZmFsc2V9", "node": { - "digest": "636Q9XJw6aZQ4RGEnej49Z99Nn21GgPBQDmCGGhNQ2Rm", + "digest": "CRPK78UpxQyyLXcizjY81vBG4bZNG6yQ9jbda9iaSnVy", "effects": { "checkpoint": { "sequenceNumber": 5 diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/first.move b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/first.move index 9269c040184..1bd0e83c3e8 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/first.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/first.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/last.exp b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/last.exp index 237670042df..265137784bf 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/last.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/last.exp @@ -164,7 +164,7 @@ Response: { { "cursor": "eyJjIjo1LCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "EnWpFFgRUBm3LbuTfjJ6oF3RXVcnzpth6L9nn4Q18RNk", + "digest": "HaHDjFPNFmzEzGV15jqjHArnNQYN3wFCehawzgSjTPDC", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -175,7 +175,7 @@ Response: { { "cursor": "eyJjIjo1LCJ0IjozLCJpIjpmYWxzZX0", "node": { - "digest": "3fmkhmMgShCMM9HWgLR13ovdiBTy2Zoe6MgVfy8Wi8nt", + "digest": "FDiPztMqvdBBRZTteyNf6wKbryrRMrqSXtS3LpWxtKqZ", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -186,7 +186,7 @@ Response: { { "cursor": "eyJjIjo1LCJ0IjoxNywiaSI6ZmFsc2V9", "node": { - "digest": "8NagxHteFzUuAvwvzjFAorD1dNfqfKnqwwviDaAt8j1z", + "digest": "DLGGXHtEkErS86wgoWD3AiLzWJqRWwpw4VCgE2XZJQUN", "effects": { "checkpoint": { "sequenceNumber": 5 @@ -197,7 +197,7 @@ Response: { { "cursor": "eyJjIjo1LCJ0IjoyMSwiaSI6ZmFsc2V9", "node": { - "digest": "9xCVVc6YyR55DpQifNUbt7ixYuficCkWkSreszCdHL1g", + "digest": "21ejBpRm4SSaoYgb2rvzno3vuFEfc7QjzPDmzZdJ5aFR", "effects": { "checkpoint": { "sequenceNumber": 5 @@ -225,7 +225,7 @@ Response: { { "cursor": "eyJjIjo1LCJ0IjoyMSwiaSI6ZmFsc2V9", "node": { - "digest": "9xCVVc6YyR55DpQifNUbt7ixYuficCkWkSreszCdHL1g", + "digest": "21ejBpRm4SSaoYgb2rvzno3vuFEfc7QjzPDmzZdJ5aFR", "effects": { "checkpoint": { "sequenceNumber": 5 @@ -253,7 +253,7 @@ Response: { { "cursor": "eyJjIjo3LCJ0IjoxNywiaSI6ZmFsc2V9", "node": { - "digest": "8NagxHteFzUuAvwvzjFAorD1dNfqfKnqwwviDaAt8j1z", + "digest": "DLGGXHtEkErS86wgoWD3AiLzWJqRWwpw4VCgE2XZJQUN", "effects": { "checkpoint": { "sequenceNumber": 5 @@ -313,7 +313,7 @@ Response: { { "cursor": "eyJjIjo3LCJ0IjozLCJpIjpmYWxzZX0", "node": { - "digest": "3fmkhmMgShCMM9HWgLR13ovdiBTy2Zoe6MgVfy8Wi8nt", + "digest": "FDiPztMqvdBBRZTteyNf6wKbryrRMrqSXtS3LpWxtKqZ", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -341,7 +341,7 @@ Response: { { "cursor": "eyJjIjo3LCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "EnWpFFgRUBm3LbuTfjJ6oF3RXVcnzpth6L9nn4Q18RNk", + "digest": "HaHDjFPNFmzEzGV15jqjHArnNQYN3wFCehawzgSjTPDC", "effects": { "checkpoint": { "sequenceNumber": 2 diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/last.move b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/last.move index 712859bdbc2..2cfcca7a8f8 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/last.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/ge_page/last.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/invalid_limits.move b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/invalid_limits.move index 71f05e500f7..0023fbfd17e 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/invalid_limits.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/invalid_limits.move @@ -4,7 +4,7 @@ // For any instance where limit is 0 or scan limit is 0, we should return an empty result -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/first.exp b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/first.exp index 377000f6c75..dada5d2480d 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/first.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/first.exp @@ -96,7 +96,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "EnWpFFgRUBm3LbuTfjJ6oF3RXVcnzpth6L9nn4Q18RNk", + "digest": "HaHDjFPNFmzEzGV15jqjHArnNQYN3wFCehawzgSjTPDC", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -107,7 +107,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjozLCJpIjpmYWxzZX0", "node": { - "digest": "GA1cT6FjP8qBTMegHadnXpjq7kTkD4Fw68jDJPci9C9r", + "digest": "DJzyBdq9ukZfiNYfRm9vWi7aXqW97F1YfwdrtTycEs1m", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -118,7 +118,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo0LCJpIjpmYWxzZX0", "node": { - "digest": "613degbde4DuvwwAZcQngGJzZkZ6GZJ9g8CwygpA1s1a", + "digest": "5p4zvwg7nsiUAUdYdzoB9swbX49YkwxfoCVLYvyUQvw5", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -129,7 +129,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo1LCJpIjpmYWxzZX0", "node": { - "digest": "GELgyKASGaZrZrgUw3C3TwaYjXwRme8StabM1zRrHDc7", + "digest": "AohmvfnFhqjmhMr4AvnEsVXHpuTne7xnFdNfZ98cEPP2", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -140,7 +140,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo2LCJpIjpmYWxzZX0", "node": { - "digest": "AYmLYSfPbk4sVdBLcV76UmZgPTETP7XsVCaNKHTJ5qzs", + "digest": "57kQ1GWYJ16691C3z6roPwP9zoH4ZZXXcopwdp5E3jn2", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -151,7 +151,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo3LCJpIjpmYWxzZX0", "node": { - "digest": "CkoEZRVMsPm3VJbX7gdwTrmfVMyHyNWDMk7tPKR6DdF2", + "digest": "89KDnhtw8jSB5ULTUJFrycuJ85D3owK5RGWtWRJmysRn", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -162,7 +162,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo4LCJpIjpmYWxzZX0", "node": { - "digest": "4kA8BHJRwCdE8BjFTpTfSGt9nvoii37twZRM91uxMuvi", + "digest": "HmHRDzEysmLNnMmwwFfehYukrz8hBKdV35W27g2dGcA6", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -173,7 +173,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo5LCJpIjpmYWxzZX0", "node": { - "digest": "ALqmnvW1wprABKAdtX8Zz49UYtTYshzXUC1fvsWx2ik9", + "digest": "BaKixTTzKMyJkmAndNNcfxFGz4gYmMJgu7GRk11zwYoi", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -184,7 +184,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoxMCwiaSI6ZmFsc2V9", "node": { - "digest": "Dqi2D27TQ531EiXzEonQGrANt8kJh9w5JZ6SMtmRpm9b", + "digest": "Ftg9H88N7ioz51UuwfzvksAwaKnD6izS6kZKVLs5kPoh", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -195,7 +195,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoxMSwiaSI6ZmFsc2V9", "node": { - "digest": "3Hs1AweW2BsmMaeicbKyhJ2DetgX5Q13Ev1S97wfBfHK", + "digest": "ANXXybMZSWDWK9pYe8Ztnxbh9GaiyX4dXA3wVdWZaxT1", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -223,7 +223,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "EnWpFFgRUBm3LbuTfjJ6oF3RXVcnzpth6L9nn4Q18RNk", + "digest": "HaHDjFPNFmzEzGV15jqjHArnNQYN3wFCehawzgSjTPDC", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -251,7 +251,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0Ijo0LCJpIjpmYWxzZX0", "node": { - "digest": "613degbde4DuvwwAZcQngGJzZkZ6GZJ9g8CwygpA1s1a", + "digest": "5p4zvwg7nsiUAUdYdzoB9swbX49YkwxfoCVLYvyUQvw5", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -279,7 +279,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0Ijo2LCJpIjpmYWxzZX0", "node": { - "digest": "AYmLYSfPbk4sVdBLcV76UmZgPTETP7XsVCaNKHTJ5qzs", + "digest": "57kQ1GWYJ16691C3z6roPwP9zoH4ZZXXcopwdp5E3jn2", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -290,7 +290,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0Ijo4LCJpIjpmYWxzZX0", "node": { - "digest": "4kA8BHJRwCdE8BjFTpTfSGt9nvoii37twZRM91uxMuvi", + "digest": "HmHRDzEysmLNnMmwwFfehYukrz8hBKdV35W27g2dGcA6", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -318,7 +318,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0IjoxMCwiaSI6ZmFsc2V9", "node": { - "digest": "Dqi2D27TQ531EiXzEonQGrANt8kJh9w5JZ6SMtmRpm9b", + "digest": "Ftg9H88N7ioz51UuwfzvksAwaKnD6izS6kZKVLs5kPoh", "effects": { "checkpoint": { "sequenceNumber": 3 diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/first.move b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/first.move index 8f14ce91b19..58d741d581c 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/first.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/first.move @@ -6,7 +6,7 @@ // However, because we have a scanLimit of 2, we'll be limited to filtering only two candidate // transactions per page, of which one will match the filtering criteria. -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/last.exp b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/last.exp index 3330dbf0293..e2b3ff0f032 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/last.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/last.exp @@ -96,7 +96,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "EnWpFFgRUBm3LbuTfjJ6oF3RXVcnzpth6L9nn4Q18RNk", + "digest": "HaHDjFPNFmzEzGV15jqjHArnNQYN3wFCehawzgSjTPDC", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -107,7 +107,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjozLCJpIjpmYWxzZX0", "node": { - "digest": "GA1cT6FjP8qBTMegHadnXpjq7kTkD4Fw68jDJPci9C9r", + "digest": "DJzyBdq9ukZfiNYfRm9vWi7aXqW97F1YfwdrtTycEs1m", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -118,7 +118,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo0LCJpIjpmYWxzZX0", "node": { - "digest": "613degbde4DuvwwAZcQngGJzZkZ6GZJ9g8CwygpA1s1a", + "digest": "5p4zvwg7nsiUAUdYdzoB9swbX49YkwxfoCVLYvyUQvw5", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -129,7 +129,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo1LCJpIjpmYWxzZX0", "node": { - "digest": "GELgyKASGaZrZrgUw3C3TwaYjXwRme8StabM1zRrHDc7", + "digest": "AohmvfnFhqjmhMr4AvnEsVXHpuTne7xnFdNfZ98cEPP2", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -140,7 +140,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo2LCJpIjpmYWxzZX0", "node": { - "digest": "AYmLYSfPbk4sVdBLcV76UmZgPTETP7XsVCaNKHTJ5qzs", + "digest": "57kQ1GWYJ16691C3z6roPwP9zoH4ZZXXcopwdp5E3jn2", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -151,7 +151,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo3LCJpIjpmYWxzZX0", "node": { - "digest": "CkoEZRVMsPm3VJbX7gdwTrmfVMyHyNWDMk7tPKR6DdF2", + "digest": "89KDnhtw8jSB5ULTUJFrycuJ85D3owK5RGWtWRJmysRn", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -162,7 +162,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo4LCJpIjpmYWxzZX0", "node": { - "digest": "4kA8BHJRwCdE8BjFTpTfSGt9nvoii37twZRM91uxMuvi", + "digest": "HmHRDzEysmLNnMmwwFfehYukrz8hBKdV35W27g2dGcA6", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -173,7 +173,7 @@ Response: { { "cursor": "eyJjIjozLCJ0Ijo5LCJpIjpmYWxzZX0", "node": { - "digest": "ALqmnvW1wprABKAdtX8Zz49UYtTYshzXUC1fvsWx2ik9", + "digest": "BaKixTTzKMyJkmAndNNcfxFGz4gYmMJgu7GRk11zwYoi", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -184,7 +184,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoxMCwiaSI6ZmFsc2V9", "node": { - "digest": "Dqi2D27TQ531EiXzEonQGrANt8kJh9w5JZ6SMtmRpm9b", + "digest": "Ftg9H88N7ioz51UuwfzvksAwaKnD6izS6kZKVLs5kPoh", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -195,7 +195,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoxMSwiaSI6ZmFsc2V9", "node": { - "digest": "3Hs1AweW2BsmMaeicbKyhJ2DetgX5Q13Ev1S97wfBfHK", + "digest": "ANXXybMZSWDWK9pYe8Ztnxbh9GaiyX4dXA3wVdWZaxT1", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -223,7 +223,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoxMCwiaSI6ZmFsc2V9", "node": { - "digest": "Dqi2D27TQ531EiXzEonQGrANt8kJh9w5JZ6SMtmRpm9b", + "digest": "Ftg9H88N7ioz51UuwfzvksAwaKnD6izS6kZKVLs5kPoh", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -251,7 +251,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0Ijo4LCJpIjpmYWxzZX0", "node": { - "digest": "4kA8BHJRwCdE8BjFTpTfSGt9nvoii37twZRM91uxMuvi", + "digest": "HmHRDzEysmLNnMmwwFfehYukrz8hBKdV35W27g2dGcA6", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -279,7 +279,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0Ijo2LCJpIjpmYWxzZX0", "node": { - "digest": "AYmLYSfPbk4sVdBLcV76UmZgPTETP7XsVCaNKHTJ5qzs", + "digest": "57kQ1GWYJ16691C3z6roPwP9zoH4ZZXXcopwdp5E3jn2", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -307,7 +307,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0Ijo0LCJpIjpmYWxzZX0", "node": { - "digest": "613degbde4DuvwwAZcQngGJzZkZ6GZJ9g8CwygpA1s1a", + "digest": "5p4zvwg7nsiUAUdYdzoB9swbX49YkwxfoCVLYvyUQvw5", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -335,7 +335,7 @@ Response: { { "cursor": "eyJjIjo0LCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "EnWpFFgRUBm3LbuTfjJ6oF3RXVcnzpth6L9nn4Q18RNk", + "digest": "HaHDjFPNFmzEzGV15jqjHArnNQYN3wFCehawzgSjTPDC", "effects": { "checkpoint": { "sequenceNumber": 2 diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/last.move b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/last.move index cd4d79a008f..ed80d9c5878 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/last.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/le_page/last.move @@ -6,7 +6,7 @@ // However, because we have a scanLimit of 2, we'll be limited to filtering only two candidate // transactions per page, of which one will match the filtering criteria. -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/require.exp b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/require.exp index f3a557539a2..9256ba2af53 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/require.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/require.exp @@ -94,7 +94,7 @@ Response: { }, "nodes": [ { - "digest": "ABweY1pDPKFhZX77kk48ofYU9KpAvXHBZUYyPK8JZf8p", + "digest": "EbDGGG7J7UrDdsn6F1C8puthPhTAyCbP8zJamEmXrvH4", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -102,7 +102,7 @@ Response: { } }, { - "digest": "8v4rn79YoSv3Drbkqbz8ttsX7EJDPEoDNbvLXpZnNXvE", + "digest": "ovzzHxewTojtxtEmjJNWFJZuD8MuP6NDxUafmf4z8tf", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -110,7 +110,7 @@ Response: { } }, { - "digest": "DeaCABc78S9mn6vSdNcNpabxYuazUfc1gsb87KMKbcyk", + "digest": "3EEE6qLkWUHTKao12yX2Z5Wa24xpArFuXXdxUm8bACqf", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -118,7 +118,7 @@ Response: { } }, { - "digest": "6YBm7aNbGN3XrFBmateAnVZ1tCdi2sattswFEHJztyrB", + "digest": "F7z3hxsjWK8Rwh1ATL55HLJE6K3vS2V8n2JipmXGiu8d", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -126,7 +126,7 @@ Response: { } }, { - "digest": "Gb2DhrKwqpXxHGt7q8XN5arTdGHf1fBKu8h8a86KYWR7", + "digest": "J7DwztQx1iKJdpmaSjt2PEJCM8VZXJMxVFpTcLPyCE8q", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -134,7 +134,7 @@ Response: { } }, { - "digest": "DigC9u79Qh7rJB9jkthYucTwZmrrnB9j9Ggf7r8wZZYZ", + "digest": "7ef7k6w9ML8t4ifRD9eMNoDGbFxRqsGttupE3opzkKjM", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -142,7 +142,7 @@ Response: { } }, { - "digest": "CMWP39HwdMdEhgHh54EonhUkmG3B5xTBdsxy493DfAxa", + "digest": "7XbbNMaQUR1R3ZujV2LKJA73HpBG8s9vAy2JTXRMgcxe", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -150,7 +150,7 @@ Response: { } }, { - "digest": "k7uqYEVDRKmcSV51NiyNASiTM6SnrFGSV4DmvWvPt2R", + "digest": "FUioT54RTwJ3UbXfUF9TTZFDU7PsuNzZsBH7Z9egjNg4", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -158,7 +158,7 @@ Response: { } }, { - "digest": "3FjYZKGUcbVYqHKSvzPuWJBTBtZHQMkVC6e23dPBLkxy", + "digest": "GPUnTY7eUNa6Uopm6oqtwuRAjNV4JyBPwGT61U6dK8A6", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -166,7 +166,7 @@ Response: { } }, { - "digest": "9c34rk4iWpmURFjPQNXCHCTFZXXwYwDXgBKn22q674jE", + "digest": "2vpWJcdbASSzWB5febpoC9Qi6kX9fF2rHJCsnzzgs6sN", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -191,7 +191,7 @@ Response: { }, "nodes": [ { - "digest": "ABweY1pDPKFhZX77kk48ofYU9KpAvXHBZUYyPK8JZf8p", + "digest": "EbDGGG7J7UrDdsn6F1C8puthPhTAyCbP8zJamEmXrvH4", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -199,7 +199,7 @@ Response: { } }, { - "digest": "8v4rn79YoSv3Drbkqbz8ttsX7EJDPEoDNbvLXpZnNXvE", + "digest": "ovzzHxewTojtxtEmjJNWFJZuD8MuP6NDxUafmf4z8tf", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -207,7 +207,7 @@ Response: { } }, { - "digest": "DeaCABc78S9mn6vSdNcNpabxYuazUfc1gsb87KMKbcyk", + "digest": "3EEE6qLkWUHTKao12yX2Z5Wa24xpArFuXXdxUm8bACqf", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -215,7 +215,7 @@ Response: { } }, { - "digest": "6YBm7aNbGN3XrFBmateAnVZ1tCdi2sattswFEHJztyrB", + "digest": "F7z3hxsjWK8Rwh1ATL55HLJE6K3vS2V8n2JipmXGiu8d", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -223,7 +223,7 @@ Response: { } }, { - "digest": "Gb2DhrKwqpXxHGt7q8XN5arTdGHf1fBKu8h8a86KYWR7", + "digest": "J7DwztQx1iKJdpmaSjt2PEJCM8VZXJMxVFpTcLPyCE8q", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -231,7 +231,7 @@ Response: { } }, { - "digest": "DigC9u79Qh7rJB9jkthYucTwZmrrnB9j9Ggf7r8wZZYZ", + "digest": "7ef7k6w9ML8t4ifRD9eMNoDGbFxRqsGttupE3opzkKjM", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -239,7 +239,7 @@ Response: { } }, { - "digest": "CMWP39HwdMdEhgHh54EonhUkmG3B5xTBdsxy493DfAxa", + "digest": "7XbbNMaQUR1R3ZujV2LKJA73HpBG8s9vAy2JTXRMgcxe", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -247,7 +247,7 @@ Response: { } }, { - "digest": "k7uqYEVDRKmcSV51NiyNASiTM6SnrFGSV4DmvWvPt2R", + "digest": "FUioT54RTwJ3UbXfUF9TTZFDU7PsuNzZsBH7Z9egjNg4", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -255,7 +255,7 @@ Response: { } }, { - "digest": "3FjYZKGUcbVYqHKSvzPuWJBTBtZHQMkVC6e23dPBLkxy", + "digest": "GPUnTY7eUNa6Uopm6oqtwuRAjNV4JyBPwGT61U6dK8A6", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -263,7 +263,7 @@ Response: { } }, { - "digest": "9c34rk4iWpmURFjPQNXCHCTFZXXwYwDXgBKn22q674jE", + "digest": "2vpWJcdbASSzWB5febpoC9Qi6kX9fF2rHJCsnzzgs6sN", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -311,7 +311,7 @@ Response: { }, "nodes": [ { - "digest": "ABweY1pDPKFhZX77kk48ofYU9KpAvXHBZUYyPK8JZf8p", + "digest": "EbDGGG7J7UrDdsn6F1C8puthPhTAyCbP8zJamEmXrvH4", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -319,7 +319,7 @@ Response: { } }, { - "digest": "8v4rn79YoSv3Drbkqbz8ttsX7EJDPEoDNbvLXpZnNXvE", + "digest": "ovzzHxewTojtxtEmjJNWFJZuD8MuP6NDxUafmf4z8tf", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -327,7 +327,7 @@ Response: { } }, { - "digest": "DeaCABc78S9mn6vSdNcNpabxYuazUfc1gsb87KMKbcyk", + "digest": "3EEE6qLkWUHTKao12yX2Z5Wa24xpArFuXXdxUm8bACqf", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -335,7 +335,7 @@ Response: { } }, { - "digest": "6YBm7aNbGN3XrFBmateAnVZ1tCdi2sattswFEHJztyrB", + "digest": "F7z3hxsjWK8Rwh1ATL55HLJE6K3vS2V8n2JipmXGiu8d", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -343,7 +343,7 @@ Response: { } }, { - "digest": "Gb2DhrKwqpXxHGt7q8XN5arTdGHf1fBKu8h8a86KYWR7", + "digest": "J7DwztQx1iKJdpmaSjt2PEJCM8VZXJMxVFpTcLPyCE8q", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -351,7 +351,7 @@ Response: { } }, { - "digest": "DigC9u79Qh7rJB9jkthYucTwZmrrnB9j9Ggf7r8wZZYZ", + "digest": "7ef7k6w9ML8t4ifRD9eMNoDGbFxRqsGttupE3opzkKjM", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -359,7 +359,7 @@ Response: { } }, { - "digest": "CMWP39HwdMdEhgHh54EonhUkmG3B5xTBdsxy493DfAxa", + "digest": "7XbbNMaQUR1R3ZujV2LKJA73HpBG8s9vAy2JTXRMgcxe", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -367,7 +367,7 @@ Response: { } }, { - "digest": "k7uqYEVDRKmcSV51NiyNASiTM6SnrFGSV4DmvWvPt2R", + "digest": "FUioT54RTwJ3UbXfUF9TTZFDU7PsuNzZsBH7Z9egjNg4", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -375,7 +375,7 @@ Response: { } }, { - "digest": "3FjYZKGUcbVYqHKSvzPuWJBTBtZHQMkVC6e23dPBLkxy", + "digest": "GPUnTY7eUNa6Uopm6oqtwuRAjNV4JyBPwGT61U6dK8A6", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -383,7 +383,7 @@ Response: { } }, { - "digest": "9c34rk4iWpmURFjPQNXCHCTFZXXwYwDXgBKn22q674jE", + "digest": "2vpWJcdbASSzWB5febpoC9Qi6kX9fF2rHJCsnzzgs6sN", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -431,7 +431,7 @@ Response: { }, "nodes": [ { - "digest": "ABweY1pDPKFhZX77kk48ofYU9KpAvXHBZUYyPK8JZf8p", + "digest": "EbDGGG7J7UrDdsn6F1C8puthPhTAyCbP8zJamEmXrvH4", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -439,7 +439,7 @@ Response: { } }, { - "digest": "8v4rn79YoSv3Drbkqbz8ttsX7EJDPEoDNbvLXpZnNXvE", + "digest": "ovzzHxewTojtxtEmjJNWFJZuD8MuP6NDxUafmf4z8tf", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -447,7 +447,7 @@ Response: { } }, { - "digest": "DeaCABc78S9mn6vSdNcNpabxYuazUfc1gsb87KMKbcyk", + "digest": "3EEE6qLkWUHTKao12yX2Z5Wa24xpArFuXXdxUm8bACqf", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -455,7 +455,7 @@ Response: { } }, { - "digest": "6YBm7aNbGN3XrFBmateAnVZ1tCdi2sattswFEHJztyrB", + "digest": "F7z3hxsjWK8Rwh1ATL55HLJE6K3vS2V8n2JipmXGiu8d", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -463,7 +463,7 @@ Response: { } }, { - "digest": "Gb2DhrKwqpXxHGt7q8XN5arTdGHf1fBKu8h8a86KYWR7", + "digest": "J7DwztQx1iKJdpmaSjt2PEJCM8VZXJMxVFpTcLPyCE8q", "effects": { "checkpoint": { "sequenceNumber": 2 @@ -471,7 +471,7 @@ Response: { } }, { - "digest": "DigC9u79Qh7rJB9jkthYucTwZmrrnB9j9Ggf7r8wZZYZ", + "digest": "7ef7k6w9ML8t4ifRD9eMNoDGbFxRqsGttupE3opzkKjM", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -479,7 +479,7 @@ Response: { } }, { - "digest": "CMWP39HwdMdEhgHh54EonhUkmG3B5xTBdsxy493DfAxa", + "digest": "7XbbNMaQUR1R3ZujV2LKJA73HpBG8s9vAy2JTXRMgcxe", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -487,7 +487,7 @@ Response: { } }, { - "digest": "k7uqYEVDRKmcSV51NiyNASiTM6SnrFGSV4DmvWvPt2R", + "digest": "FUioT54RTwJ3UbXfUF9TTZFDU7PsuNzZsBH7Z9egjNg4", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -495,7 +495,7 @@ Response: { } }, { - "digest": "3FjYZKGUcbVYqHKSvzPuWJBTBtZHQMkVC6e23dPBLkxy", + "digest": "GPUnTY7eUNa6Uopm6oqtwuRAjNV4JyBPwGT61U6dK8A6", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -503,7 +503,7 @@ Response: { } }, { - "digest": "9c34rk4iWpmURFjPQNXCHCTFZXXwYwDXgBKn22q674jE", + "digest": "2vpWJcdbASSzWB5febpoC9Qi6kX9fF2rHJCsnzzgs6sN", "effects": { "checkpoint": { "sequenceNumber": 3 @@ -592,7 +592,7 @@ Response: { { "cursor": "eyJjIjozLCJ0IjoyLCJpIjpmYWxzZX0", "node": { - "digest": "ABweY1pDPKFhZX77kk48ofYU9KpAvXHBZUYyPK8JZf8p", + "digest": "EbDGGG7J7UrDdsn6F1C8puthPhTAyCbP8zJamEmXrvH4", "effects": { "checkpoint": { "sequenceNumber": 2 diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/require.move b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/require.move index ecd32122627..a2eadb876c5 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/require.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/scan_limit/require.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses Test=0x0 --accounts A B --simulator +//# init --protocol-version 3 --addresses Test=0x0 --accounts A B --simulator //# publish module Test::M1 { diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/shared.exp b/crates/iota-graphql-e2e-tests/tests/transactions/shared.exp index 13ed627e781..6c35b3ee6c9 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/shared.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/shared.exp @@ -42,7 +42,7 @@ Response: { "transactions": { "nodes": [ { - "package": "0xf40350dd9cb77160579b0e2511443ac8079495baf4a63202dc7a62b1a383a181", + "package": "0x0f785d8315ad19c0ad6f7415b518d775e963dc5457df45e92c3ba2fcca766537", "module": "m", "functionName": "get" } @@ -55,17 +55,17 @@ Response: { "nodes": [ { "__typename": "SharedObjectRead", - "address": "0xa2f6f4538016aceca519479b3992959c2bb84f2a95db0cff1415422fa499ddbe", + "address": "0x6f07b78ba9141b1a7a6b729880fc9b63b12e557313134c722f2b04169c853a1a", "version": 2, - "digest": "53nzg2NiA6eZd4BxKfy88MmasZSGRat3313Azf266AfJ", + "digest": "bLhuLwagCRxcGmFWMCg9GDDziJhA9u5zqKfpQ81Tn2g", "object": { "asMoveObject": { "contents": { "type": { - "repr": "0xf40350dd9cb77160579b0e2511443ac8079495baf4a63202dc7a62b1a383a181::m::Foo" + "repr": "0x0f785d8315ad19c0ad6f7415b518d775e963dc5457df45e92c3ba2fcca766537::m::Foo" }, "json": { - "id": "0xa2f6f4538016aceca519479b3992959c2bb84f2a95db0cff1415422fa499ddbe", + "id": "0x6f07b78ba9141b1a7a6b729880fc9b63b12e557313134c722f2b04169c853a1a", "x": "0" } } @@ -82,7 +82,7 @@ Response: { "transactions": { "nodes": [ { - "package": "0xf40350dd9cb77160579b0e2511443ac8079495baf4a63202dc7a62b1a383a181", + "package": "0x0f785d8315ad19c0ad6f7415b518d775e963dc5457df45e92c3ba2fcca766537", "module": "m", "functionName": "inc" } @@ -102,12 +102,12 @@ Response: { "transactions": { "nodes": [ { - "package": "0xf40350dd9cb77160579b0e2511443ac8079495baf4a63202dc7a62b1a383a181", + "package": "0x0f785d8315ad19c0ad6f7415b518d775e963dc5457df45e92c3ba2fcca766537", "module": "m", "functionName": "get" }, { - "package": "0xf40350dd9cb77160579b0e2511443ac8079495baf4a63202dc7a62b1a383a181", + "package": "0x0f785d8315ad19c0ad6f7415b518d775e963dc5457df45e92c3ba2fcca766537", "module": "m", "functionName": "inc" } diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/shared.move b/crates/iota-graphql-e2e-tests/tests/transactions/shared.move index ccccc98a607..cfde0386747 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/shared.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/shared.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --addresses P0=0x0 --simulator +//# init --protocol-version 3 --addresses P0=0x0 --simulator //# publish module P0::m { diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/system.exp b/crates/iota-graphql-e2e-tests/tests/transactions/system.exp index bdad7857c2f..e7fec8f39e5 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/system.exp +++ b/crates/iota-graphql-e2e-tests/tests/transactions/system.exp @@ -7,7 +7,7 @@ Response: { "transactionBlocks": { "nodes": [ { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83", + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ", "sender": null, "signatures": [ "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA==" @@ -407,7 +407,7 @@ Response: { "json": { "id": "0x0000000000000000000000000000000000000000000000000000000000000008", "inner": { - "id": "0x3f92eceed239ba443daf612dcdf621f85feba71f3698d53c9f2a6e4b4246b1c6", + "id": "0x087e07e48f02e1757a9a06af5c2be59dfc1a5717fe305d79fd77c5b3350426bc", "version": "1" } } @@ -489,7 +489,7 @@ Response: { "json": { "id": "0x0000000000000000000000000000000000000000000000000000000000000403", "lists": { - "id": "0x799c22e253a384f3545724b35277ac6b27595603f051828e5f4cece335c1cf1c", + "id": "0xd221ff85aa726e3a39d3d1d3969341c10000dd4cf7a550e0319d6424bc755f95", "size": "0" } } @@ -586,18 +586,18 @@ Response: { { "cursor": "eyJpIjo5LCJjIjowfQ", "node": { - "address": "0x1b19651d3c2723e7f6c1eca23c13b72cd7f274524223391e75994dad311828df", + "address": "0x2c7f8520eacbe6e27a4a61d113ba4baf08de78c9a1e50fee95763451d410f0c6", "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000003::staking_pool::StakedIota" }, "json": { - "id": "0x1b19651d3c2723e7f6c1eca23c13b72cd7f274524223391e75994dad311828df", - "name": "0", - "value": { - "iota_amount": "0", - "pool_token_amount": "0" + "id": "0x2c7f8520eacbe6e27a4a61d113ba4baf08de78c9a1e50fee95763451d410f0c6", + "pool_id": "0x05c2a0cbbad1a47da3d5ec68021869986e9dd70b57d9e64f020bb138dbe842c8", + "stake_activation_epoch": "0", + "principal": { + "value": "1500000000000000" } } } @@ -608,14 +608,92 @@ Response: { { "cursor": "eyJpIjoxMCwiYyI6MH0", "node": { - "address": "0x27982f48a697a9f61e3f738993c2d00e16b89938eedb74af1849ba6c6d82303b", + "address": "0x36b1836eb54ee1b5e86133199c8f9d75b7bc99f7b5d88f574a2a210f4288d0b4", + "asMoveObject": { + "contents": { + "type": { + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::CoinMetadata<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" + }, + "json": { + "id": "0x36b1836eb54ee1b5e86133199c8f9d75b7bc99f7b5d88f574a2a210f4288d0b4", + "decimals": 9, + "name": "IOTA", + "symbol": "IOTA", + "description": "The main (gas)token of the IOTA Network.", + "icon_url": { + "url": "https://iota.org/logo.png" + } + } + } + }, + "asMovePackage": null + } + }, + { + "cursor": "eyJpIjoxMSwiYyI6MH0", + "node": { + "address": "0x398a1aa5ea188a9410153911506f47d0800414543bded1724e5db9b6ef7bcd0a", + "asMoveObject": { + "contents": { + "type": { + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::display::Display<0x000000000000000000000000000000000000000000000000000000000000107a::nft::Nft>" + }, + "json": { + "id": "0x398a1aa5ea188a9410153911506f47d0800414543bded1724e5db9b6ef7bcd0a", + "fields": { + "contents": [ + { + "key": "name", + "value": "{immutable_metadata.name}" + }, + { + "key": "image_url", + "value": "{immutable_metadata.uri}" + }, + { + "key": "description", + "value": "{immutable_metadata.description}" + }, + { + "key": "creator", + "value": "{immutable_metadata.issuer_name}" + }, + { + "key": "version", + "value": "{immutable_metadata.version}" + }, + { + "key": "media_type", + "value": "{immutable_metadata.media_type}" + }, + { + "key": "collection_name", + "value": "{immutable_metadata.collection_name}" + }, + { + "key": "immutable_issuer", + "value": "{immutable_issuer}" + } + ] + }, + "version": 1 + } + } + }, + "asMovePackage": null + } + }, + { + "cursor": "eyJpIjoxMiwiYyI6MH0", + "node": { + "address": "0x59960c037f5355cceb1ed7d285234b35012fbe18dbb7921ce3a518e15d807dea", "asMoveObject": { "contents": { "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" }, "json": { - "id": "0x27982f48a697a9f61e3f738993c2d00e16b89938eedb74af1849ba6c6d82303b", + "id": "0x59960c037f5355cceb1ed7d285234b35012fbe18dbb7921ce3a518e15d807dea", "balance": { "value": "30000000000000000" } @@ -626,7 +704,27 @@ Response: { } }, { - "cursor": "eyJpIjoxMSwiYyI6MH0", + "cursor": "eyJpIjoxMywiYyI6MH0", + "node": { + "address": "0x5ba4893be3938566e676ec9ded7f006ac2f8a5cf9713ff00880f4d8a2bd24434", + "asMoveObject": { + "contents": { + "type": { + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" + }, + "json": { + "id": "0x5ba4893be3938566e676ec9ded7f006ac2f8a5cf9713ff00880f4d8a2bd24434", + "balance": { + "value": "300000000000000" + } + } + } + }, + "asMovePackage": null + } + }, + { + "cursor": "eyJpIjoxNCwiYyI6MH0", "node": { "address": "0x6af2a2b7ca60bf76174adfd3e9c4957f8e937759603182f9b46c7f6c5f19c6d2", "asMoveObject": { @@ -639,11 +737,11 @@ Response: { "name": "1", "value": { "epoch": "0", - "protocol_version": "1", + "protocol_version": "3", "system_state_version": "1", "iota_treasury_cap": { "inner": { - "id": "0x509e22a8bb835eeb98bed81a1a50c196383345279d4238f47714f2301c9bfbe6", + "id": "0xc15f6e44edf5e6fac2777ae0d03718609673544fc01c4b6ddcfe9ed29b3bfc16", "total_supply": { "value": "31800000000000000" } @@ -890,15 +988,15 @@ Response: { "next_epoch_p2p_address": null, "next_epoch_primary_address": null, "extra_fields": { - "id": "0x6c8759462593f9ef531bbec3f6402e9a9d53e369ecb7982b59635843471eb271", + "id": "0xeb22f95292675630311381ac30b8035d3d34e5b13706b14aef83f56fb5219104", "size": "0" } }, "voting_power": "10000", - "operation_cap_id": "0xa0f947f895ffe5715ab529ed16659e9c099bee48524f56c8d5a2eab8e842bd99", + "operation_cap_id": "0x9359d1eae7919936b04321b147376bb2c4d0e5a580544ccb85e88f57fb3bfe06", "gas_price": "1000", "staking_pool": { - "id": "0x7aea65806e468642f39bb0a0f7a40a4b16afc9588d1980e8d9a9d20936541b7e", + "id": "0x05c2a0cbbad1a47da3d5ec68021869986e9dd70b57d9e64f020bb138dbe842c8", "activation_epoch": "0", "deactivation_epoch": null, "iota_balance": "1500000000000000", @@ -907,14 +1005,14 @@ Response: { }, "pool_token_balance": "1500000000000000", "exchange_rates": { - "id": "0x044bc64e9c5eaa0143019c38bea813b1c288dc66272d74cf8a1821bf0eeac85f", + "id": "0x7a91f6ef9ac734a5fd196148b31dea99a6382bb60a51d254b49a1ae58971d3d0", "size": "1" }, "pending_stake": "0", "pending_total_iota_withdraw": "0", "pending_pool_token_withdraw": "0", "extra_fields": { - "id": "0xcc8c44fff5e1f3fdc63bd2597685dba922cce4809e2f960c8128dcbcb76c3712", + "id": "0x43ef1b2999e8c2138782f7c7256e7a7328a087d1ca54d436bad1e612ff3895d7", "size": "0" } }, @@ -923,35 +1021,35 @@ Response: { "next_epoch_gas_price": "1000", "next_epoch_commission_rate": "200", "extra_fields": { - "id": "0x9a75ea182d4788588f2e81034a8a57034da6284a66159cd3289cb352e3a845aa", + "id": "0x46905279408a6ebdc8d6f376ed34b4a7653e701214874c764d0b240bc3233e56", "size": "0" } } ], "pending_active_validators": { "contents": { - "id": "0x31b89b7b349cf41a62274ebd1ef2c2789fa86b68270d9feefb7a62741d1f8e86", + "id": "0xa292cc32e3ef38fc736bf882953228f189e57e6102dff74ac640ab20d0a97b68", "size": "0" } }, "pending_removals": [], "staking_pool_mappings": { - "id": "0x6025058d2b6287ffad433b714c9a573fe15ba477a4f47862b5088ca5938cded3", + "id": "0x7306811a2d1ccba1979366cfac72b6e0bbb25830621beb3a763d82c8432c2548", "size": "1" }, "inactive_validators": { - "id": "0xaaf430f2910eb82232f946c1bda701ecb8f2c1f6928a9099ae80f5afc85fb69c", + "id": "0x4577cd777ee8f12d682dd4d86b4316d88920d01641a35fdd912c281c3918ec19", "size": "0" }, "validator_candidates": { - "id": "0xd2789ba6e8f5c3241a6d12bedc3055e57aeac6e92bf182ef7737c6533d7a22f9", + "id": "0xe08408f1bfb39d546b658cde0ab9ca8363a13a64e8aa943c37c29e2cd0ac2af7", "size": "0" }, "at_risk_validators": { "contents": [] }, "extra_fields": { - "id": "0x62d408b9b0b3c458c8424f68a6622c534d3023ce93368b5a412bd4dc665a8ebf", + "id": "0x504a9178ee1993dfc19e1cdf815cc71e78a11e3dd4cba609c407b0d8873b2be0", "size": "0" } }, @@ -972,7 +1070,7 @@ Response: { "validator_very_low_stake_threshold": "1000000000000000", "validator_low_stake_grace_period": "7", "extra_fields": { - "id": "0x61b4baf4a2f43504215e4131c43852e24775aa0afeef0628a113228ef41c7813", + "id": "0x0c77a159e1ae881ab41774c7c14c684b9385dd9cde74150bc3c80160583ccb2f", "size": "0" } }, @@ -994,7 +1092,7 @@ Response: { "safe_mode_non_refundable_storage_fee": "0", "epoch_start_timestamp_ms": "0", "extra_fields": { - "id": "0x5dc4c87d84a69a6b0f51a38b24cd4c9833b79c2a2e249dc1f24794961bfc76c1", + "id": "0x117ad5f40111d3a0c25a0ac46c76070270e52648085320c0e81d1124a8f03d3a", "size": "0" } } @@ -1005,35 +1103,17 @@ Response: { } }, { - "cursor": "eyJpIjoxMiwiYyI6MH0", - "node": { - "address": "0xa0f947f895ffe5715ab529ed16659e9c099bee48524f56c8d5a2eab8e842bd99", - "asMoveObject": { - "contents": { - "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000003::validator_cap::UnverifiedValidatorOperationCap" - }, - "json": { - "id": "0xa0f947f895ffe5715ab529ed16659e9c099bee48524f56c8d5a2eab8e842bd99", - "authorizer_validator_address": "0x28f02a953f3553f51a9365593c7d4bd0643d2085f004b18c6ca9de51682b2c80" - } - } - }, - "asMovePackage": null - } - }, - { - "cursor": "eyJpIjoxMywiYyI6MH0", + "cursor": "eyJpIjoxNSwiYyI6MH0", "node": { - "address": "0xbb5caa62668e66be9dd0a5a74534814bf789fd6ab6335e08eff62a58ee1f287e", + "address": "0x7640a5451f1ceb353262d82aa933eef348c6800c4d37d1a113d1d31124240580", "asMoveObject": { "contents": { "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field<0x0000000000000000000000000000000000000000000000000000000000000002::object::ID,address>" }, "json": { - "id": "0xbb5caa62668e66be9dd0a5a74534814bf789fd6ab6335e08eff62a58ee1f287e", - "name": "0x7aea65806e468642f39bb0a0f7a40a4b16afc9588d1980e8d9a9d20936541b7e", + "id": "0x7640a5451f1ceb353262d82aa933eef348c6800c4d37d1a113d1d31124240580", + "name": "0x05c2a0cbbad1a47da3d5ec68021869986e9dd70b57d9e64f020bb138dbe842c8", "value": "0x28f02a953f3553f51a9365593c7d4bd0643d2085f004b18c6ca9de51682b2c80" } } @@ -1042,77 +1122,17 @@ Response: { } }, { - "cursor": "eyJpIjoxNCwiYyI6MH0", - "node": { - "address": "0xc28024efc650d31011065697e676b6738732dad9f709e54a13b116672c298f25", - "asMoveObject": { - "contents": { - "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::CoinMetadata<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" - }, - "json": { - "id": "0xc28024efc650d31011065697e676b6738732dad9f709e54a13b116672c298f25", - "decimals": 9, - "name": "IOTA", - "symbol": "IOTA", - "description": "The main (gas)token of the IOTA Network.", - "icon_url": { - "url": "https://iota.org/logo.png" - } - } - } - }, - "asMovePackage": null - } - }, - { - "cursor": "eyJpIjoxNSwiYyI6MH0", + "cursor": "eyJpIjoxNiwiYyI6MH0", "node": { - "address": "0xc928f6f78781333f87efac974403df5a8565e3accb3c16a891a60a18dd2f2f42", + "address": "0x9359d1eae7919936b04321b147376bb2c4d0e5a580544ccb85e88f57fb3bfe06", "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::display::Display<0x000000000000000000000000000000000000000000000000000000000000107a::nft::Nft>" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000003::validator_cap::UnverifiedValidatorOperationCap" }, "json": { - "id": "0xc928f6f78781333f87efac974403df5a8565e3accb3c16a891a60a18dd2f2f42", - "fields": { - "contents": [ - { - "key": "name", - "value": "{immutable_metadata.name}" - }, - { - "key": "image_url", - "value": "{immutable_metadata.uri}" - }, - { - "key": "description", - "value": "{immutable_metadata.description}" - }, - { - "key": "creator", - "value": "{immutable_metadata.issuer_name}" - }, - { - "key": "version", - "value": "{immutable_metadata.version}" - }, - { - "key": "media_type", - "value": "{immutable_metadata.media_type}" - }, - { - "key": "collection_name", - "value": "{immutable_metadata.collection_name}" - }, - { - "key": "immutable_issuer", - "value": "{immutable_issuer}" - } - ] - }, - "version": 1 + "id": "0x9359d1eae7919936b04321b147376bb2c4d0e5a580544ccb85e88f57fb3bfe06", + "authorizer_validator_address": "0x28f02a953f3553f51a9365593c7d4bd0643d2085f004b18c6ca9de51682b2c80" } } }, @@ -1120,20 +1140,20 @@ Response: { } }, { - "cursor": "eyJpIjoxNiwiYyI6MH0", + "cursor": "eyJpIjoxNywiYyI6MH0", "node": { - "address": "0xdd6b3a50df67a765e8c59bd37030224138f36bba296535846f490d8dede3c502", + "address": "0xaf46abe6b9b499da57800fe44fb6be5625b46905f87d865eff3906e33f1de65a", "asMoveObject": { "contents": { "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000003::staking_pool::StakedIota" + "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field" }, "json": { - "id": "0xdd6b3a50df67a765e8c59bd37030224138f36bba296535846f490d8dede3c502", - "pool_id": "0x7aea65806e468642f39bb0a0f7a40a4b16afc9588d1980e8d9a9d20936541b7e", - "stake_activation_epoch": "0", - "principal": { - "value": "1500000000000000" + "id": "0xaf46abe6b9b499da57800fe44fb6be5625b46905f87d865eff3906e33f1de65a", + "name": "0", + "value": { + "iota_amount": "0", + "pool_token_amount": "0" } } } @@ -1142,16 +1162,16 @@ Response: { } }, { - "cursor": "eyJpIjoxNywiYyI6MH0", + "cursor": "eyJpIjoxOCwiYyI6MH0", "node": { - "address": "0xe9ee601e69a2f4274ca5f23c16190823a7af496b5b9f2eac0d2c2645634b2737", + "address": "0xfcd138dc779f2208287655fb49de71e66cda1e8dba0624407c3a9fad28cdb944", "asMoveObject": { "contents": { "type": { "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::dynamic_field::Field" }, "json": { - "id": "0xe9ee601e69a2f4274ca5f23c16190823a7af496b5b9f2eac0d2c2645634b2737", + "id": "0xfcd138dc779f2208287655fb49de71e66cda1e8dba0624407c3a9fad28cdb944", "name": "1", "value": { "version": "1", @@ -1164,26 +1184,6 @@ Response: { }, "asMovePackage": null } - }, - { - "cursor": "eyJpIjoxOCwiYyI6MH0", - "node": { - "address": "0xf22e2f7e20b84f57fdcf1f44c79c321533b922fa9488c23556356ae3f906b0bb", - "asMoveObject": { - "contents": { - "type": { - "repr": "0x0000000000000000000000000000000000000000000000000000000000000002::coin::Coin<0x0000000000000000000000000000000000000000000000000000000000000002::iota::IOTA>" - }, - "json": { - "id": "0xf22e2f7e20b84f57fdcf1f44c79c321533b922fa9488c23556356ae3f906b0bb", - "balance": { - "value": "300000000000000" - } - } - } - }, - "asMovePackage": null - } } ] } @@ -1225,7 +1225,7 @@ Response: { "idDeleted": false, "outputState": { "address": "0x0000000000000000000000000000000000000000000000000000000000000001", - "digest": "BWPzkyHoTWnZGNLBbza1xvEYP9hgm7dogXoi1DUjtpM7" + "digest": "7UFySuwUYdH2zABgJdY1psQjEczEfPvY9FrrNQagGnCp" } }, { @@ -1234,7 +1234,7 @@ Response: { "idDeleted": false, "outputState": { "address": "0x0000000000000000000000000000000000000000000000000000000000000002", - "digest": "63DagAxEDkLwscMq1tGaPAhLR9dCYKwMTRFZugpfDiMq" + "digest": "7Jhmz7CEjB6ndNqws1uZPG29ts1nH5tDTXwJvks2HJCL" } }, { @@ -1243,7 +1243,7 @@ Response: { "idDeleted": false, "outputState": { "address": "0x0000000000000000000000000000000000000000000000000000000000000003", - "digest": "3X6wUAwmH2pcVR5yHWcUMoJKs5gwhxfCRh42jq7xVZM3" + "digest": "5DgTiPyNzZGqxutLnJc58N2iLNdH2Asow723sSBWi1gA" } }, { @@ -1252,7 +1252,7 @@ Response: { "idDeleted": false, "outputState": { "address": "0x0000000000000000000000000000000000000000000000000000000000000005", - "digest": "6XeCa8r3ai3FXCgp2kcz85fDjW4rJ5FZLeQJf2Nzwm75" + "digest": "6F2JQrphEGNPFDmMoZUW3VtNsQ38P2xEqbYfwtgAzdUo" } }, { @@ -1261,7 +1261,7 @@ Response: { "idDeleted": false, "outputState": { "address": "0x0000000000000000000000000000000000000000000000000000000000000006", - "digest": "gCx8m8h2RGFjajzMdfBdk4MyevRi34XxhWobMZVRyZV" + "digest": "BrE9YtjinR8WtmugqQZyUa2myDu38zH3DcRTjLoyedFk" } }, { @@ -1270,7 +1270,7 @@ Response: { "idDeleted": false, "outputState": { "address": "0x0000000000000000000000000000000000000000000000000000000000000008", - "digest": "3oxqmiP9tw1RZ5idND2hZ6sQ7bCM8qU3MT7PYS8wWEuB" + "digest": "9UkS39k2xgNCu1W47nmUmaBkkjDJ9xzf7g7fT9ERRKdL" } }, { @@ -1279,7 +1279,7 @@ Response: { "idDeleted": false, "outputState": { "address": "0x000000000000000000000000000000000000000000000000000000000000000b", - "digest": "8m58ysCbvvBpAN54w4fww8Y6nXm4GAQfovuswD1vGFFP" + "digest": "AuzCeVavBdmNpRTcEqAEDnAJApbtGkUtVhpwZgmNy3Rq" } }, { @@ -1288,7 +1288,7 @@ Response: { "idDeleted": false, "outputState": { "address": "0x0000000000000000000000000000000000000000000000000000000000000403", - "digest": "Eg1CzeQBR3AGwxnDq8TJo8A9gHSZjFpXNMk8zeMc5jXe" + "digest": "9um7GQp72yV7a4J8Wvrhh8bpLJU7TDz64QhptKB8h9E2" } }, { @@ -1297,97 +1297,97 @@ Response: { "idDeleted": false, "outputState": { "address": "0x000000000000000000000000000000000000000000000000000000000000107a", - "digest": "Ds3kGXwgK3MWzFVW6wHPEDoPzLtU3mtRnvxF1PC7WTWp" + "digest": "tCFpGw6qvC3oDFBCXLvC8qbm4mxjAf2UjxbxytNoBXt" } }, { - "address": "0x1b19651d3c2723e7f6c1eca23c13b72cd7f274524223391e75994dad311828df", + "address": "0x2c7f8520eacbe6e27a4a61d113ba4baf08de78c9a1e50fee95763451d410f0c6", "idCreated": true, "idDeleted": false, "outputState": { - "address": "0x1b19651d3c2723e7f6c1eca23c13b72cd7f274524223391e75994dad311828df", - "digest": "EPdNi4i16VS8Ji3xn4DAPVAJ5xi4pZHpCXGuKbqeNkzz" + "address": "0x2c7f8520eacbe6e27a4a61d113ba4baf08de78c9a1e50fee95763451d410f0c6", + "digest": "FfPSmLHq4BSAvatR1zGxx7bLNdLbBKtLFcj7pLboJZyH" } }, { - "address": "0x27982f48a697a9f61e3f738993c2d00e16b89938eedb74af1849ba6c6d82303b", + "address": "0x36b1836eb54ee1b5e86133199c8f9d75b7bc99f7b5d88f574a2a210f4288d0b4", "idCreated": true, "idDeleted": false, "outputState": { - "address": "0x27982f48a697a9f61e3f738993c2d00e16b89938eedb74af1849ba6c6d82303b", - "digest": "GdkVYasCkoM1F9zBqytqzNtU5oKBsJnt7bvNeM7BQESg" + "address": "0x36b1836eb54ee1b5e86133199c8f9d75b7bc99f7b5d88f574a2a210f4288d0b4", + "digest": "APLnCXgk93zNmvjsDLvB7YVPQ5Hhw6vGhaTqnvVApVdj" } }, { - "address": "0x6af2a2b7ca60bf76174adfd3e9c4957f8e937759603182f9b46c7f6c5f19c6d2", + "address": "0x398a1aa5ea188a9410153911506f47d0800414543bded1724e5db9b6ef7bcd0a", "idCreated": true, "idDeleted": false, "outputState": { - "address": "0x6af2a2b7ca60bf76174adfd3e9c4957f8e937759603182f9b46c7f6c5f19c6d2", - "digest": "96gSoSvev5VaU8BfPEYSBpXkTfSLEhMjzcLtk3PAxBzn" + "address": "0x398a1aa5ea188a9410153911506f47d0800414543bded1724e5db9b6ef7bcd0a", + "digest": "5ZF9Rejpbm6bwMSHbYubwfzTmuqC5z5pkdk8Je5Ky7sT" } }, { - "address": "0xa0f947f895ffe5715ab529ed16659e9c099bee48524f56c8d5a2eab8e842bd99", + "address": "0x59960c037f5355cceb1ed7d285234b35012fbe18dbb7921ce3a518e15d807dea", "idCreated": true, "idDeleted": false, "outputState": { - "address": "0xa0f947f895ffe5715ab529ed16659e9c099bee48524f56c8d5a2eab8e842bd99", - "digest": "BNLVZMqwpvMRqGDAvLXND3zWvHJ3tcbfn5LgC1FvmNw7" + "address": "0x59960c037f5355cceb1ed7d285234b35012fbe18dbb7921ce3a518e15d807dea", + "digest": "7eSPEvysMcragu4j3jz232Yzq2k4KeZZukSD9PBzJmre" } }, { - "address": "0xbb5caa62668e66be9dd0a5a74534814bf789fd6ab6335e08eff62a58ee1f287e", + "address": "0x5ba4893be3938566e676ec9ded7f006ac2f8a5cf9713ff00880f4d8a2bd24434", "idCreated": true, "idDeleted": false, "outputState": { - "address": "0xbb5caa62668e66be9dd0a5a74534814bf789fd6ab6335e08eff62a58ee1f287e", - "digest": "713ee9qPuf6j6nXZVs6jQNaURCHXprZfwwbFnmuVRBDu" + "address": "0x5ba4893be3938566e676ec9ded7f006ac2f8a5cf9713ff00880f4d8a2bd24434", + "digest": "74Ahxwfe4tu6bANF1dhKmrUnFzzCY3AkHsRGpYK2jyJr" } }, { - "address": "0xc28024efc650d31011065697e676b6738732dad9f709e54a13b116672c298f25", + "address": "0x6af2a2b7ca60bf76174adfd3e9c4957f8e937759603182f9b46c7f6c5f19c6d2", "idCreated": true, "idDeleted": false, "outputState": { - "address": "0xc28024efc650d31011065697e676b6738732dad9f709e54a13b116672c298f25", - "digest": "BEPbAyocJgB6QtEAKQQUgyfYFf8RkWPWUpvRpuwobNHC" + "address": "0x6af2a2b7ca60bf76174adfd3e9c4957f8e937759603182f9b46c7f6c5f19c6d2", + "digest": "A37ZgmnXCRdm5v3yAkG56bP7hbkYjUj81VTgCCgXQGCm" } }, { - "address": "0xc928f6f78781333f87efac974403df5a8565e3accb3c16a891a60a18dd2f2f42", + "address": "0x7640a5451f1ceb353262d82aa933eef348c6800c4d37d1a113d1d31124240580", "idCreated": true, "idDeleted": false, "outputState": { - "address": "0xc928f6f78781333f87efac974403df5a8565e3accb3c16a891a60a18dd2f2f42", - "digest": "FzFjGUJ3FBJRZ4eVE7y9F3LRqxWyDY4JT1pmg7g1BU4X" + "address": "0x7640a5451f1ceb353262d82aa933eef348c6800c4d37d1a113d1d31124240580", + "digest": "2Uet64wgf5yqMNELAkYTsVqHb7yxuh9caD6yCVm7B2dc" } }, { - "address": "0xdd6b3a50df67a765e8c59bd37030224138f36bba296535846f490d8dede3c502", + "address": "0x9359d1eae7919936b04321b147376bb2c4d0e5a580544ccb85e88f57fb3bfe06", "idCreated": true, "idDeleted": false, "outputState": { - "address": "0xdd6b3a50df67a765e8c59bd37030224138f36bba296535846f490d8dede3c502", - "digest": "EJ4XFN8AQCbwg3rqF1uksR2gKLDyadnCN1dYvJ3Um713" + "address": "0x9359d1eae7919936b04321b147376bb2c4d0e5a580544ccb85e88f57fb3bfe06", + "digest": "CWiEcb4A3vxdW6gYURRnWzXi1fSr8kAFDkmotkAaSecv" } }, { - "address": "0xe9ee601e69a2f4274ca5f23c16190823a7af496b5b9f2eac0d2c2645634b2737", + "address": "0xaf46abe6b9b499da57800fe44fb6be5625b46905f87d865eff3906e33f1de65a", "idCreated": true, "idDeleted": false, "outputState": { - "address": "0xe9ee601e69a2f4274ca5f23c16190823a7af496b5b9f2eac0d2c2645634b2737", - "digest": "9oBq9oxHbMER3WeW5DudYyXecorCy5yjcaLJVBzrAJmw" + "address": "0xaf46abe6b9b499da57800fe44fb6be5625b46905f87d865eff3906e33f1de65a", + "digest": "56hwycrGTguejGkuRww33X6YkqRcm8zX8RzWHnXExTU6" } }, { - "address": "0xf22e2f7e20b84f57fdcf1f44c79c321533b922fa9488c23556356ae3f906b0bb", + "address": "0xfcd138dc779f2208287655fb49de71e66cda1e8dba0624407c3a9fad28cdb944", "idCreated": true, "idDeleted": false, "outputState": { - "address": "0xf22e2f7e20b84f57fdcf1f44c79c321533b922fa9488c23556356ae3f906b0bb", - "digest": "D6VFrTAqMGZjMktBnEhGkZ9MXyea8WisE4vdn8fBAjgV" + "address": "0xfcd138dc779f2208287655fb49de71e66cda1e8dba0624407c3a9fad28cdb944", + "digest": "5r2XEc2DfUdN8bZnp33bQRq3ppB8W4WfNnrDG9pL8q6u" } } ] @@ -1409,7 +1409,7 @@ Response: { "sequenceNumber": 0 }, "transactionBlock": { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83" + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ" } }, "expiration": null @@ -1461,7 +1461,7 @@ Response: { "dependencies": { "nodes": [ { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83" + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ" } ] }, @@ -1519,7 +1519,7 @@ Response: { "transactionBlocks": { "nodes": [ { - "digest": "7PLwaqn6pg198VkYmVKERZ2Qad3vDEWhfB7VRv5qDSc7", + "digest": "9ADUBQmAfCGqabFJTzw3VsybLPe28j7QTZ84xqbDRiVd", "sender": null, "signatures": [ "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA==" @@ -1543,7 +1543,7 @@ Response: { "node": { "__typename": "ChangeEpochTransaction", "epoch": null, - "protocolVersion": 1, + "protocolVersion": 3, "storageCharge": "0", "computationCharge": "0", "storageRebate": "0", @@ -1561,7 +1561,7 @@ Response: { "dependencies": { "nodes": [ { - "digest": "EjxArvXVR7U8yu5Yi2oDmELwV74gwJFinZKhnPuTKj83" + "digest": "HA4up8t27PnLk1un1DvTbctYWMRp2AMzkMbJLDeGVopQ" } ] }, @@ -1576,34 +1576,34 @@ Response: { "idDeleted": false, "outputState": { "address": "0x0000000000000000000000000000000000000000000000000000000000000005", - "digest": "G1xvDacDJsdTTLoLq8tjhLbTDCR6MYL9yVStX9e8GKi" + "digest": "6XciTcJGe1ojLheRewNPp2BBdt5CCUSzbMyhowmrG9tN" } }, { - "address": "0x0d8feaf1c714777ad5401781c55794e0b8f6dda166924176279914bed08951e0", + "address": "0x589e651bedffb1545420130fb727c2b7440d2616e73fba44b8faeb5faea30cb3", "idCreated": true, "idDeleted": false, "outputState": { - "address": "0x0d8feaf1c714777ad5401781c55794e0b8f6dda166924176279914bed08951e0", - "digest": "92GwpFNwQDbmoJRDkH6crad72SMMubKs3HdpMhncAwbp" + "address": "0x589e651bedffb1545420130fb727c2b7440d2616e73fba44b8faeb5faea30cb3", + "digest": "4i5J4zT287eApgZ1kErZsFvgUMH1Yx9RaNR1veh3hGb3" } }, { - "address": "0x4509fa198370254af0418248c9718bb206e810e1bce9fc8ffd0c500aef972f4d", - "idCreated": true, + "address": "0x6af2a2b7ca60bf76174adfd3e9c4957f8e937759603182f9b46c7f6c5f19c6d2", + "idCreated": false, "idDeleted": false, "outputState": { - "address": "0x4509fa198370254af0418248c9718bb206e810e1bce9fc8ffd0c500aef972f4d", - "digest": "5wumBetyyshU5rCK2f6GUDCohAWpmVrus9D4QS5bambg" + "address": "0x6af2a2b7ca60bf76174adfd3e9c4957f8e937759603182f9b46c7f6c5f19c6d2", + "digest": "GQ5Ntk3gZUZxVFBmEqPTP4NhQcuzq8ZqcoJfSztUTFtU" } }, { - "address": "0x6af2a2b7ca60bf76174adfd3e9c4957f8e937759603182f9b46c7f6c5f19c6d2", - "idCreated": false, + "address": "0x77ebd13ca27a089fcdfecdafe7a0e66e6e5e6778aa19d2f6f7bc94ec170ac35e", + "idCreated": true, "idDeleted": false, "outputState": { - "address": "0x6af2a2b7ca60bf76174adfd3e9c4957f8e937759603182f9b46c7f6c5f19c6d2", - "digest": "CNxCh8MK8JrDGgHFwQyJ3oBV6njo5bRSWcEy8HPAakha" + "address": "0x77ebd13ca27a089fcdfecdafe7a0e66e6e5e6778aa19d2f6f7bc94ec170ac35e", + "digest": "CRU1zaa52si82SonUoFsWYxk2LhkBbuidgJV4x7Grcbt" } } ] @@ -1625,7 +1625,7 @@ Response: { "sequenceNumber": 2 }, "transactionBlock": { - "digest": "7PLwaqn6pg198VkYmVKERZ2Qad3vDEWhfB7VRv5qDSc7" + "digest": "9ADUBQmAfCGqabFJTzw3VsybLPe28j7QTZ84xqbDRiVd" } }, "expiration": null diff --git a/crates/iota-graphql-e2e-tests/tests/transactions/system.move b/crates/iota-graphql-e2e-tests/tests/transactions/system.move index a3b4ee10b3e..4dd61a5b0c4 100644 --- a/crates/iota-graphql-e2e-tests/tests/transactions/system.move +++ b/crates/iota-graphql-e2e-tests/tests/transactions/system.move @@ -2,7 +2,7 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -//# init --protocol-version 1 --simulator +//# init --protocol-version 3 --simulator // Tests for representations of all the various system transactions diff --git a/crates/iota-graphql-e2e-tests/tests/validator/validator.move b/crates/iota-graphql-e2e-tests/tests/validator/validator.move index e455e09db87..19d18460378 100644 --- a/crates/iota-graphql-e2e-tests/tests/validator/validator.move +++ b/crates/iota-graphql-e2e-tests/tests/validator/validator.move @@ -4,7 +4,7 @@ // Test heavy transactions. -//# init --protocol-version 1 --simulator --accounts A --addresses P0=0x0 +//# init --protocol-version 3 --simulator --accounts A --addresses P0=0x0 //# advance-epoch diff --git a/crates/iota-graphql-rpc/README.md b/crates/iota-graphql-rpc/README.md index c7dadc63929..9f361220618 100644 --- a/crates/iota-graphql-rpc/README.md +++ b/crates/iota-graphql-rpc/README.md @@ -49,7 +49,7 @@ If you're using VS Code, make sure to update `settings.json` with the appropriat ### Using docker compose (recommended) -See [pg-services-local](../../docker/pg-services-local/README.md), which automatically sets up the GraphQL server along with an Indexer instance, the postgres database and a local network. +See [pg-services-local](../../dev-tools/pg-services-local/README.md), which automatically sets up the GraphQL server along with an Indexer instance, the postgres database and a local network. ### Using manual setup @@ -115,7 +115,7 @@ Find more example queries in the [examples](examples) directory. ### Launching the server with Indexer For local development, it might be useful to spin up an actual Indexer as well (not only the postgres instance) which writes data to the database, so you can query it with the GraphQL server. -You can run it with a local network using the `iota start` subcommand or [pg-services-local](../../docker/pg-services-local/README.md) or as a [standalone service](../iota-indexer/README.md#standalone-indexer-setup). +You can run it with a local network using the `iota start` subcommand or [pg-services-local](../../dev-tools/pg-services-local/README.md) or as a [standalone service](../iota-indexer/README.md#standalone-indexer-setup). To run it with the `iota start` subcommand, switch to the root directory of the repository and run the following command to start the Indexer with the Sync worker: diff --git a/crates/iota-graphql-rpc/schema.graphql b/crates/iota-graphql-rpc/schema.graphql index 8821213b8ef..ec5f1a351d7 100644 --- a/crates/iota-graphql-rpc/schema.graphql +++ b/crates/iota-graphql-rpc/schema.graphql @@ -951,7 +951,7 @@ of dynamic fields: an object stored in this kind of field will be considered wrapped and will not be accessible directly via its ID by external tools (explorers, wallets, etc) accessing storage. -2) Dynamic Object Fields values must be Iota objects (have the `key` and +2) Dynamic Object Fields values must be IOTA objects (have the `key` and `store` abilities, and id: UID as the first field), but will still be directly accessible off-chain via their object ID after being attached. """ @@ -1061,7 +1061,7 @@ type EndOfEpochTransactionKindEdge { } """ -Operation of the Iota network is temporally partitioned into non-overlapping +Operation of the IOTA network is temporally partitioned into non-overlapping epochs, and the network aims to keep epochs roughly the same duration as each other. During a particular epoch the following data is fixed: @@ -1664,7 +1664,7 @@ type MergeCoinsTransaction { } """ -Abilities are keywords in Iota Move that define how types behave at the +Abilities are keywords in IOTA Move that define how types behave at the compiler level. """ enum MoveAbility { @@ -2610,7 +2610,7 @@ enum MoveVisibility { } """ -Mutations are used to write to the Iota network. +Mutations are used to write to the IOTA network. """ type Mutation { """ @@ -2638,7 +2638,7 @@ type Mutation { } """ -An object in Iota is a package (set of Move bytecode modules) or object +An object in IOTA is a package (set of Move bytecode modules) or object (typed data structure with fields) with additional metadata detailing its id, version, transaction digest, owner field indicating how this object can be accessed. @@ -3279,7 +3279,7 @@ type Query { field when its parent was at `rootVersion`. If `rootVersion` is omitted, dynamic fields will be from a consistent - snapshot of the Iota state at the latest checkpoint known to the + snapshot of the IOTA state at the latest checkpoint known to the GraphQL RPC. Similarly, `Owner.asObject` will return the object's version at the latest checkpoint. """ diff --git a/crates/iota-graphql-rpc/src/commands.rs b/crates/iota-graphql-rpc/src/commands.rs index 0a3c4317242..f8ddacb7730 100644 --- a/crates/iota-graphql-rpc/src/commands.rs +++ b/crates/iota-graphql-rpc/src/commands.rs @@ -7,9 +7,9 @@ use std::path::PathBuf; use clap::*; #[derive(Parser)] -#[clap( +#[command( name = "iota-graphql-rpc", - about = "Iota GraphQL RPC", + about = "IOTA GraphQL RPC", rename_all = "kebab-case", author, version @@ -23,38 +23,38 @@ pub enum Command { }, GenerateSchema { /// Path to output GraphQL schema to, in SDL format. - #[clap(short, long)] + #[arg(short, long)] file: Option, }, StartServer { /// The title to display at the top of the page - #[clap(short, long)] + #[arg(short, long)] ide_title: Option, /// DB URL for data fetching - #[clap(short, long)] + #[arg(short, long)] db_url: Option, /// Pool size for DB connections - #[clap(long)] + #[arg(long)] db_pool_size: Option, /// Port to bind the server to - #[clap(short, long)] + #[arg(short, long)] port: Option, /// Host to bind the server to - #[clap(long)] + #[arg(long)] host: Option, /// Port to bind the prom server to - #[clap(long)] + #[arg(long)] prom_port: Option, /// Host to bind the prom server to - #[clap(long)] + #[arg(long)] prom_host: Option, /// Path to TOML file containing configuration for service. - #[clap(short, long)] + #[arg(short, long)] config: Option, /// RPC url to the Node for tx execution - #[clap(long)] + #[arg(long)] node_rpc_url: Option, }, } diff --git a/crates/iota-graphql-rpc/src/config.rs b/crates/iota-graphql-rpc/src/config.rs index 550482e4d9d..70e30f59370 100644 --- a/crates/iota-graphql-rpc/src/config.rs +++ b/crates/iota-graphql-rpc/src/config.rs @@ -439,7 +439,7 @@ impl Default for Versions { impl Default for Ide { fn default() -> Self { Self { - ide_title: "Iota GraphQL IDE".to_string(), + ide_title: "IOTA GraphQL IDE".to_string(), } } } diff --git a/crates/iota-graphql-rpc/src/mutation.rs b/crates/iota-graphql-rpc/src/mutation.rs index b236b6824a4..0cc7cfaadda 100644 --- a/crates/iota-graphql-rpc/src/mutation.rs +++ b/crates/iota-graphql-rpc/src/mutation.rs @@ -26,7 +26,7 @@ use crate::{ }; pub struct Mutation; -/// Mutations are used to write to the Iota network. +/// Mutations are used to write to the IOTA network. #[Object] impl Mutation { /// Execute a transaction, committing its effects on chain. @@ -56,11 +56,11 @@ impl Mutation { ) -> Result { let iota_sdk_client: &Option = ctx .data() - .map_err(|_| Error::Internal("Unable to fetch Iota SDK client".to_string())) + .map_err(|_| Error::Internal("Unable to fetch IOTA SDK client".to_string())) .extend()?; let iota_sdk_client = iota_sdk_client .as_ref() - .ok_or_else(|| Error::Internal("Iota SDK client not initialized".to_string())) + .ok_or_else(|| Error::Internal("IOTA SDK client not initialized".to_string())) .extend()?; let tx_data = bcs::from_bytes( &Base64::decode(&tx_bytes) diff --git a/crates/iota-graphql-rpc/src/server/exchange_rates_task.rs b/crates/iota-graphql-rpc/src/server/exchange_rates_task.rs index d39d8f3597c..e2d9b91f297 100644 --- a/crates/iota-graphql-rpc/src/server/exchange_rates_task.rs +++ b/crates/iota-graphql-rpc/src/server/exchange_rates_task.rs @@ -43,7 +43,7 @@ impl TriggerExchangeRatesTask { info!("Detected epoch boundary, triggering call to exchange rates"); let latest_iota_system_state = self.db.inner.spawn_blocking(move |this| this.get_latest_iota_system_state() - ).await.map_err(|_| error!("Failed to fetch latest Iota system state")); + ).await.map_err(|_| error!("Failed to fetch latest IOTA system state")); if let Ok(latest_iota_system_state) = latest_iota_system_state { let db = self.db.clone(); diff --git a/crates/iota-graphql-rpc/src/test_infra/cluster.rs b/crates/iota-graphql-rpc/src/test_infra/cluster.rs index 36aff4343d1..f6a0aeede83 100644 --- a/crates/iota-graphql-rpc/src/test_infra/cluster.rs +++ b/crates/iota-graphql-rpc/src/test_infra/cluster.rs @@ -105,9 +105,9 @@ pub async fn start_cluster( } } -/// Takes in a simulated instantiation of a Iota blockchain and builds a cluster -/// around it. This cluster is typically used in e2e tests to emulate and test -/// behaviors. +/// Takes in a simulated instantiation of an IOTA blockchain and builds a +/// cluster around it. This cluster is typically used in e2e tests to emulate +/// and test behaviors. pub async fn serve_executor( graphql_connection_config: ConnectionConfig, internal_data_source_rpc_port: u16, diff --git a/crates/iota-graphql-rpc/src/types/cursor.rs b/crates/iota-graphql-rpc/src/types/cursor.rs index 3d421edeef6..93af6cfbc98 100644 --- a/crates/iota-graphql-rpc/src/types/cursor.rs +++ b/crates/iota-graphql-rpc/src/types/cursor.rs @@ -250,6 +250,7 @@ impl Page> { /// two booleans indicating whether there is a previous or next page in /// the range, the `checkpoint_viewed_at` to set for consistency, and an /// iterator of cursors within that Page. + #[allow(clippy::type_complexity)] pub(crate) fn paginate_consistent_indices( &self, total: usize, diff --git a/crates/iota-graphql-rpc/src/types/dynamic_field.rs b/crates/iota-graphql-rpc/src/types/dynamic_field.rs index 612e3ab84ac..c63634adfa2 100644 --- a/crates/iota-graphql-rpc/src/types/dynamic_field.rs +++ b/crates/iota-graphql-rpc/src/types/dynamic_field.rs @@ -58,7 +58,7 @@ pub(crate) struct DynamicFieldName { /// an object stored in this kind of field will be considered wrapped and /// will not be accessible directly via its ID by external tools (explorers, /// wallets, etc) accessing storage. -/// 2) Dynamic Object Fields values must be Iota objects (have the `key` and +/// 2) Dynamic Object Fields values must be IOTA objects (have the `key` and /// `store` abilities, and id: UID as the first field), but will still be /// directly accessible off-chain via their object ID after being attached. #[Object] diff --git a/crates/iota-graphql-rpc/src/types/epoch.rs b/crates/iota-graphql-rpc/src/types/epoch.rs index bbb9607c294..696ebdfe0fd 100644 --- a/crates/iota-graphql-rpc/src/types/epoch.rs +++ b/crates/iota-graphql-rpc/src/types/epoch.rs @@ -43,7 +43,7 @@ struct EpochKey { pub checkpoint_viewed_at: u64, } -/// Operation of the Iota network is temporally partitioned into non-overlapping +/// Operation of the IOTA network is temporally partitioned into non-overlapping /// epochs, and the network aims to keep epochs roughly the same duration as /// each other. During a particular epoch the following data is fixed: /// diff --git a/crates/iota-graphql-rpc/src/types/event/mod.rs b/crates/iota-graphql-rpc/src/types/event/mod.rs index 633e9771184..cfe77f3cb45 100644 --- a/crates/iota-graphql-rpc/src/types/event/mod.rs +++ b/crates/iota-graphql-rpc/src/types/event/mod.rs @@ -42,7 +42,7 @@ mod lookups; pub(crate) use cursor::Cursor; pub(crate) use filter::EventFilter; -/// A Iota node emits one of the following events: +/// An IOTA node emits one of the following events: /// Move event /// Publish event /// Transfer object event diff --git a/crates/iota-graphql-rpc/src/types/move_type.rs b/crates/iota-graphql-rpc/src/types/move_type.rs index 44227f90de5..880d84985ca 100644 --- a/crates/iota-graphql-rpc/src/types/move_type.rs +++ b/crates/iota-graphql-rpc/src/types/move_type.rs @@ -316,7 +316,7 @@ impl TryFrom for MoveFieldLayout { } /// Error from seeing a `signer` value or type, which shouldn't be possible in -/// Iota Move. +/// IOTA Move. pub(crate) fn unexpected_signer_error() -> Error { Error::Internal("Unexpected value of type: signer.".to_string()) } diff --git a/crates/iota-graphql-rpc/src/types/move_value.rs b/crates/iota-graphql-rpc/src/types/move_value.rs index d0e31fe47e4..32a3080db8a 100644 --- a/crates/iota-graphql-rpc/src/types/move_value.rs +++ b/crates/iota-graphql-rpc/src/types/move_value.rs @@ -230,7 +230,7 @@ impl TryFrom for MoveData { }) } - // Iota does not support `signer` as a type. + // IOTA does not support `signer` as a type. V::Signer(_) => return Err(unexpected_signer_error()), }) } @@ -318,7 +318,7 @@ fn try_to_json_value(value: A::MoveValue) -> Result { .collect(), ) } - // Iota does not support `signer` as a type. + // IOTA does not support `signer` as a type. V::Signer(_) => return Err(unexpected_signer_error()), }) } diff --git a/crates/iota-graphql-rpc/src/types/object.rs b/crates/iota-graphql-rpc/src/types/object.rs index ae2d53767f3..d6211443f8a 100644 --- a/crates/iota-graphql-rpc/src/types/object.rs +++ b/crates/iota-graphql-rpc/src/types/object.rs @@ -334,7 +334,7 @@ struct LatestAtKey { checkpoint_viewed_at: u64, } -/// An object in Iota is a package (set of Move bytecode modules) or object +/// An object in IOTA is a package (set of Move bytecode modules) or object /// (typed data structure with fields) with additional metadata detailing its /// id, version, transaction digest, owner field indicating how this object can /// be accessed. diff --git a/crates/iota-graphql-rpc/src/types/open_move_type.rs b/crates/iota-graphql-rpc/src/types/open_move_type.rs index b88944de9f0..bf3c4e79126 100644 --- a/crates/iota-graphql-rpc/src/types/open_move_type.rs +++ b/crates/iota-graphql-rpc/src/types/open_move_type.rs @@ -13,7 +13,7 @@ pub(crate) struct OpenMoveType { signature: OpenMoveTypeSignature, } -/// Abilities are keywords in Iota Move that define how types behave at the +/// Abilities are keywords in IOTA Move that define how types behave at the /// compiler level. #[derive(Enum, Copy, Clone, Eq, PartialEq)] pub(crate) enum MoveAbility { diff --git a/crates/iota-graphql-rpc/src/types/query.rs b/crates/iota-graphql-rpc/src/types/query.rs index 91de38b2f8e..6786b84501c 100644 --- a/crates/iota-graphql-rpc/src/types/query.rs +++ b/crates/iota-graphql-rpc/src/types/query.rs @@ -110,11 +110,11 @@ impl Query { let iota_sdk_client: &Option = ctx .data() - .map_err(|_| Error::Internal("Unable to fetch Iota SDK client".to_string())) + .map_err(|_| Error::Internal("Unable to fetch IOTA SDK client".to_string())) .extend()?; let iota_sdk_client = iota_sdk_client .as_ref() - .ok_or_else(|| Error::Internal("Iota SDK client not initialized".to_string())) + .ok_or_else(|| Error::Internal("IOTA SDK client not initialized".to_string())) .extend()?; let (sender_address, tx_kind, gas_price, gas_sponsor, gas_budget, gas_objects) = @@ -200,7 +200,7 @@ impl Query { /// field when its parent was at `rootVersion`. /// /// If `rootVersion` is omitted, dynamic fields will be from a consistent - /// snapshot of the Iota state at the latest checkpoint known to the + /// snapshot of the IOTA state at the latest checkpoint known to the /// GraphQL RPC. Similarly, `Owner.asObject` will return the object's /// version at the latest checkpoint. async fn owner( diff --git a/crates/iota-graphql-rpc/src/types/validator.rs b/crates/iota-graphql-rpc/src/types/validator.rs index 7d70342693f..4f9b3cfb87a 100644 --- a/crates/iota-graphql-rpc/src/types/validator.rs +++ b/crates/iota-graphql-rpc/src/types/validator.rs @@ -74,7 +74,7 @@ impl Loader for Db { .inner .spawn_blocking(move |this| this.get_latest_iota_system_state()) .await - .map_err(|_| Error::Internal("Failed to fetch latest Iota system state".to_string()))?; + .map_err(|_| Error::Internal("Failed to fetch latest IOTA system state".to_string()))?; let governance_api = GovernanceReadApi::new(self.inner.clone()); let (candidate_rates, pending_rates) = tokio::try_join!( diff --git a/crates/iota-graphql-rpc/tests/snapshots/snapshot_tests__schema_sdl_export.snap b/crates/iota-graphql-rpc/tests/snapshots/snapshot_tests__schema_sdl_export.snap index b1c9a00a5ac..f9ff85e3efb 100644 --- a/crates/iota-graphql-rpc/tests/snapshots/snapshot_tests__schema_sdl_export.snap +++ b/crates/iota-graphql-rpc/tests/snapshots/snapshot_tests__schema_sdl_export.snap @@ -955,7 +955,7 @@ of dynamic fields: an object stored in this kind of field will be considered wrapped and will not be accessible directly via its ID by external tools (explorers, wallets, etc) accessing storage. -2) Dynamic Object Fields values must be Iota objects (have the `key` and +2) Dynamic Object Fields values must be IOTA objects (have the `key` and `store` abilities, and id: UID as the first field), but will still be directly accessible off-chain via their object ID after being attached. """ @@ -1065,7 +1065,7 @@ type EndOfEpochTransactionKindEdge { } """ -Operation of the Iota network is temporally partitioned into non-overlapping +Operation of the IOTA network is temporally partitioned into non-overlapping epochs, and the network aims to keep epochs roughly the same duration as each other. During a particular epoch the following data is fixed: @@ -1668,7 +1668,7 @@ type MergeCoinsTransaction { } """ -Abilities are keywords in Iota Move that define how types behave at the +Abilities are keywords in IOTA Move that define how types behave at the compiler level. """ enum MoveAbility { @@ -2614,7 +2614,7 @@ enum MoveVisibility { } """ -Mutations are used to write to the Iota network. +Mutations are used to write to the IOTA network. """ type Mutation { """ @@ -2642,7 +2642,7 @@ type Mutation { } """ -An object in Iota is a package (set of Move bytecode modules) or object +An object in IOTA is a package (set of Move bytecode modules) or object (typed data structure with fields) with additional metadata detailing its id, version, transaction digest, owner field indicating how this object can be accessed. @@ -3283,7 +3283,7 @@ type Query { field when its parent was at `rootVersion`. If `rootVersion` is omitted, dynamic fields will be from a consistent - snapshot of the Iota state at the latest checkpoint known to the + snapshot of the IOTA state at the latest checkpoint known to the GraphQL RPC. Similarly, `Owner.asObject` will return the object's version at the latest checkpoint. """ diff --git a/crates/iota-indexer-builder/Cargo.toml b/crates/iota-indexer-builder/Cargo.toml index 1b12d82425f..67e187d9be1 100644 --- a/crates/iota-indexer-builder/Cargo.toml +++ b/crates/iota-indexer-builder/Cargo.toml @@ -12,6 +12,7 @@ anyhow.workspace = true async-trait.workspace = true prometheus.workspace = true tokio = { workspace = true, features = ["full"] } +tokio-util.workspace = true tracing.workspace = true # internal dependencies diff --git a/crates/iota-indexer-builder/src/iota_datasource.rs b/crates/iota-indexer-builder/src/iota_datasource.rs index 27839b1ddd6..85e71eaf7b1 100644 --- a/crates/iota-indexer-builder/src/iota_datasource.rs +++ b/crates/iota-indexer-builder/src/iota_datasource.rs @@ -15,10 +15,8 @@ use iota_types::{ full_checkpoint_content::{CheckpointData as IotaCheckpointData, CheckpointTransaction}, messages_checkpoint::CheckpointSequenceNumber, }; -use tokio::{ - sync::{oneshot, oneshot::Sender}, - task::JoinHandle, -}; +use tokio::task::JoinHandle; +use tokio_util::sync::CancellationToken; use tracing::info; use crate::indexer_builder::{DataSender, Datasource}; @@ -53,11 +51,12 @@ impl Datasource for IotaCheckpointDatasource { target_checkpoint: u64, data_sender: DataSender, ) -> Result>, Error> { - let (exit_sender, exit_receiver) = oneshot::channel(); + let token = CancellationToken::new(); + let token_child = token.child_token(); let progress_store = PerTaskInMemProgressStore { current_checkpoint: starting_checkpoint, exit_checkpoint: target_checkpoint, - exit_sender: Some(exit_sender), + token: Some(token), }; let mut executor = IndexerExecutor::new(progress_store, 1, self.metrics.clone()); let worker = IndexerWorker::new(data_sender); @@ -76,7 +75,7 @@ impl Datasource for IotaCheckpointDatasource { Some(remote_store_url), vec![], // optional remote store access options ReaderOptions::default(), - exit_receiver, + token_child, ) .await?; Ok(()) @@ -87,7 +86,7 @@ impl Datasource for IotaCheckpointDatasource { struct PerTaskInMemProgressStore { pub current_checkpoint: u64, pub exit_checkpoint: u64, - pub exit_sender: Option>, + pub token: Option, } #[async_trait] @@ -105,8 +104,8 @@ impl ProgressStore for PerTaskInMemProgressStore { checkpoint_number: CheckpointSequenceNumber, ) -> anyhow::Result<()> { if checkpoint_number >= self.exit_checkpoint { - if let Some(sender) = self.exit_sender.take() { - let _ = sender.send(()); + if let Some(token) = self.token.take() { + token.cancel(); } } self.current_checkpoint = checkpoint_number; diff --git a/crates/iota-indexer/README.md b/crates/iota-indexer/README.md index b7126faad67..7802da7c4c8 100644 --- a/crates/iota-indexer/README.md +++ b/crates/iota-indexer/README.md @@ -19,7 +19,7 @@ For more in depth information check the [Database Schema](./schema.md). ### Using docker compose (recommended) -See [pg-services-local](../../docker/pg-services-local/README.md), which automatically sets up the Indexer Sync worker and the Indexer RPC worker along with a postgres database and local network. +See [pg-services-local](../../dev-tools/pg-services-local/README.md), which automatically sets up the Indexer Sync worker and the Indexer RPC worker along with a postgres database and local network. ### Using manual setup @@ -27,7 +27,7 @@ To run an Indexer, a running postgres instance is required. #### Database setup -You can either spin up the postgres instance as a single service via [docker-compose](../../docker/pg-services-local/README.md) or manually set up it up. +You can either spin up the postgres instance as a single service via [docker-compose](../../dev-tools/pg-services-local/README.md) or manually set up it up. If you choose for manual setup, follow the steps below: 1. Install a local [Postgres server](https://www.postgresql.org/download) and start it. @@ -63,13 +63,13 @@ To run the indexer as a standalone service with an existing fullnode, follow the ```sh # Change the RPC_CLIENT_URL to http://0.0.0.0:9000 to run indexer against local validator & fullnode -cargo run --bin iota-indexer -- --db-url "postgres://postgres:postgrespw@localhost/iota_indexer" --rpc-client-url "https://fullnode.devnet.iota.io:443" --fullnode-sync-worker --reset-db +cargo run --bin iota-indexer -- --db-url "postgres://postgres:postgrespw@localhost/iota_indexer" --rpc-client-url "https://api.devnet.iota.cafe:443" --fullnode-sync-worker --reset-db ``` - to run indexer as a reader which exposes a JSON RPC service with following [APIs](https://docs.iota.org/iota-api-ref). ``` -cargo run --bin iota-indexer -- --db-url "postgres://postgres:postgrespw@localhost/iota_indexer" --rpc-client-url "https://fullnode.devnet.iota.io:443" --rpc-server-worker +cargo run --bin iota-indexer -- --db-url "postgres://postgres:postgrespw@localhost/iota_indexer" --rpc-client-url "https://api.devnet.iota.cafe:443" --rpc-server-worker ``` More available flags can be found in this [file](https://github.com/iotaledger/iota/blob/develop/crates/iota-indexer/src/lib.rs). @@ -171,5 +171,5 @@ Note that you need an existing database for this to work. Using the DATABASE_URL ```sh # Change the RPC_CLIENT_URL to http://0.0.0.0:9000 to run indexer against local validator & fullnode -cargo run --bin iota-indexer --features mysql-feature --no-default-features -- --db-url "" --rpc-client-url "https://fullnode.devnet.iota.io:443" --fullnode-sync-worker --reset-db +cargo run --bin iota-indexer --features mysql-feature --no-default-features -- --db-url "" --rpc-client-url "https://api.devnet.iota.cafe:443" --fullnode-sync-worker --reset-db ``` diff --git a/crates/iota-indexer/examples/index_genesis_transaction.rs b/crates/iota-indexer/examples/index_genesis_transaction.rs index cc2de930778..dc0e51b7aaf 100644 --- a/crates/iota-indexer/examples/index_genesis_transaction.rs +++ b/crates/iota-indexer/examples/index_genesis_transaction.rs @@ -18,22 +18,22 @@ use iota_swarm_config::genesis_config::ValidatorGenesisConfigBuilder; use rand::rngs::OsRng; #[derive(Parser, Debug)] -#[clap(about = "Example that indexes the genesis transaction into the database")] +#[command(about = "Example that indexes the genesis transaction into the database")] struct Args { /// Remotely stored migration snapshots. - #[clap( + #[arg( long, name = "iota|", help = "Remote migration snapshots.", default_values_t = vec![SnapshotUrl::Iota], + num_args(0..) )] - #[arg(num_args(0..))] migration_snapshots: Vec, } const DEFAULT_DB_URL: &str = "postgres://postgres:postgrespw@localhost:5432/iota_indexer"; -// Build genesis with `Iota` stardust snapshot +// Build genesis with `IOTA` stardust snapshot fn genesis_builder(migration_sources: Vec) -> GenesisBuilder { // Create the builder let mut builder = GenesisBuilder::new(); diff --git a/crates/iota-indexer/src/indexer.rs b/crates/iota-indexer/src/indexer.rs index 4325feae326..b9c16356971 100644 --- a/crates/iota-indexer/src/indexer.rs +++ b/crates/iota-indexer/src/indexer.rs @@ -13,7 +13,6 @@ use iota_data_ingestion_core::{ use iota_metrics::spawn_monitored_task; use iota_types::messages_checkpoint::CheckpointSequenceNumber; use prometheus::Registry; -use tokio::sync::oneshot; use tokio_util::sync::CancellationToken; use tracing::info; @@ -71,7 +70,7 @@ impl Indexer { cancel: CancellationToken, ) -> Result<(), IndexerError> { info!( - "Iota Indexer Writer (version {:?}) started...", + "IOTA Indexer Writer (version {:?}) started...", env!("CARGO_PKG_VERSION") ); @@ -132,14 +131,6 @@ impl Indexer { store.persist_protocol_configs_and_feature_flags(chain_id)?; } - let cancel_clone = cancel.clone(); - let (exit_sender, exit_receiver) = oneshot::channel(); - // Spawn a task that links the cancellation token to the exit sender - spawn_monitored_task!(async move { - cancel_clone.cancelled().await; - let _ = exit_sender.send(()); - }); - let mut executor = IndexerExecutor::new( ShimIndexerProgressStore::new(vec![ ("primary".to_string(), primary_watermark), @@ -170,7 +161,7 @@ impl Indexer { config.remote_store_url.clone(), vec![], extra_reader_options, - exit_receiver, + cancel.child_token(), ) .await?; Ok(()) @@ -182,7 +173,7 @@ impl Indexer { db_url: String, ) -> Result<(), IndexerError> { info!( - "Iota Indexer Reader (version {:?}) started...", + "IOTA Indexer Reader (version {:?}) started...", env!("CARGO_PKG_VERSION") ); let indexer_reader = IndexerReader::::new(db_url)?; @@ -202,7 +193,7 @@ impl Indexer { metrics: IndexerMetrics, ) -> Result<(), IndexerError> { info!( - "Iota Indexer Analytical Worker (version {:?}) started...", + "IOTA Indexer Analytical Worker (version {:?}) started...", env!("CARGO_PKG_VERSION") ); let mut processor_orchestrator = ProcessorOrchestrator::new(store, metrics); diff --git a/crates/iota-indexer/src/indexer_reader.rs b/crates/iota-indexer/src/indexer_reader.rs index 1fa26c1969c..2ecf20f90bf 100644 --- a/crates/iota-indexer/src/indexer_reader.rs +++ b/crates/iota-indexer/src/indexer_reader.rs @@ -10,7 +10,8 @@ use std::{ use anyhow::{Result, anyhow}; use cached::{Cached, SizedCache}; use diesel::{ - ExpressionMethods, OptionalExtension, QueryDsl, RunQueryDsl, TextExpressionMethods, + ExpressionMethods, JoinOnDsl, NullableExpressionMethods, OptionalExtension, QueryDsl, + RunQueryDsl, SelectableHelper, TextExpressionMethods, dsl::sql, r2d2::{ConnectionManager, R2D2Connection}, sql_types::Bool, @@ -63,7 +64,7 @@ use crate::{ }, schema::{ address_metrics, checkpoints, display, epochs, events, move_call_metrics, objects, - objects_snapshot, packages, transactions, + objects_snapshot, packages, pruner_cp_watermark, transactions, tx_digests, }, store::{diesel_macro::*, package_resolver::IndexerStorePackageResolver}, types::{IndexerResult, OwnerType}, @@ -461,7 +462,16 @@ impl IndexerReader { ) -> Result { let mut stored_txn: StoredTransaction = run_query!(&self.pool, |conn| { transactions::table - .filter(transactions::transaction_digest.eq(digest.into_inner().to_vec())) + .filter( + transactions::tx_sequence_number + .nullable() + .eq(tx_digests::table + .select(tx_digests::tx_sequence_number) + // we filter the tx_digests table because it is indexed by digest, + // transactions table is not + .filter(tx_digests::tx_digest.eq(digest.into_inner().to_vec())) + .single_value()), + ) .first::(conn) })?; @@ -497,7 +507,14 @@ impl IndexerReader { .collect::>(); let transactions = run_query!(&self.pool, |conn| { transactions::table - .filter(transactions::transaction_digest.eq_any(digests)) + .inner_join( + tx_digests::table + .on(transactions::tx_sequence_number.eq(tx_digests::tx_sequence_number)), + ) + // we filter the tx_digests table because it is indexed by digest, + // transactions table is not + .filter(tx_digests::tx_digest.eq_any(digests)) + .select(StoredTransaction::as_select()) .load::(conn) })?; if cfg!(feature = "postgres-feature") { @@ -722,8 +739,21 @@ impl IndexerReader { limit: usize, is_descending: bool, ) -> IndexerResult> { + let pool = self.get_pool(); + let tx_range: (i64, i64) = run_query_async!(&pool, move |conn| { + pruner_cp_watermark::dsl::pruner_cp_watermark + .select(( + pruner_cp_watermark::min_tx_sequence_number, + pruner_cp_watermark::max_tx_sequence_number, + )) + // we filter the pruner_cp_watermark table because it is indexed by + // checkpoint_sequence_number, transactions is not + .filter(pruner_cp_watermark::checkpoint_sequence_number.eq(checkpoint_seq as i64)) + .first::<(i64, i64)>(conn) + })?; + let mut query = transactions::dsl::transactions - .filter(transactions::dsl::checkpoint_sequence_number.eq(checkpoint_seq as i64)) + .filter(transactions::tx_sequence_number.between(tx_range.0, tx_range.1)) .into_boxed(); // Translate transaction digest cursor to tx sequence number @@ -778,9 +808,11 @@ impl IndexerReader { let cursor_tx_seq = if let Some(cursor) = cursor { let pool = self.get_pool(); let tx_seq = run_query_async!(&pool, move |conn| { - transactions::dsl::transactions - .select(transactions::tx_sequence_number) - .filter(transactions::dsl::transaction_digest.eq(cursor.into_inner().to_vec())) + tx_digests::table + .select(tx_digests::tx_sequence_number) + // we filter the tx_digests table because it is indexed by digest, + // transactions (and other tables) are not + .filter(tx_digests::tx_digest.eq(cursor.into_inner().to_vec())) .first::(conn) })?; Some(tx_seq) @@ -1019,7 +1051,16 @@ impl IndexerReader { let pool = self.get_pool(); let (timestamp_ms, serialized_events) = run_query_async!(&pool, move |conn| { transactions::table - .filter(transactions::transaction_digest.eq(digest.into_inner().to_vec())) + .filter( + transactions::tx_sequence_number + .nullable() + .eq(tx_digests::table + .select(tx_digests::tx_sequence_number) + // we filter the tx_digests table because it is indexed by digest, + // transactions table is not + .filter(tx_digests::tx_digest.eq(digest.into_inner().to_vec())) + .single_value()), + ) .select((transactions::timestamp_ms, transactions::events)) .first::<(i64, StoredTransactionEvents)>(conn) })?; @@ -1037,43 +1078,68 @@ impl IndexerReader { Ok(iota_tx_events.map_or(vec![], |ste| ste.data)) } - fn query_events_by_tx_digest_query( + async fn query_events_by_tx_digest( &self, tx_digest: TransactionDigest, cursor: Option, limit: usize, descending_order: bool, - ) -> IndexerResult { - let cursor = if let Some(cursor) = cursor { + ) -> IndexerResult> { + let mut query = events::table.into_boxed(); + + if let Some(cursor) = cursor { if cursor.tx_digest != tx_digest { return Err(IndexerError::InvalidArgument( "Cursor tx_digest does not match the tx_digest in the query.".into(), )); } if descending_order { - format!("e.{EVENT_SEQUENCE_NUMBER_STR} < {}", cursor.event_seq) + query = query.filter(events::event_sequence_number.lt(cursor.event_seq as i64)); } else { - format!("e.{EVENT_SEQUENCE_NUMBER_STR} > {}", cursor.event_seq) + query = query.filter(events::event_sequence_number.gt(cursor.event_seq as i64)); } } else if descending_order { - format!("e.{EVENT_SEQUENCE_NUMBER_STR} <= {}", i64::MAX) + query = query.filter(events::event_sequence_number.le(i64::MAX)); } else { - format!("e.{EVENT_SEQUENCE_NUMBER_STR} >= {}", 0) + query = query.filter(events::event_sequence_number.ge(0)); }; - let order_clause = if descending_order { "DESC" } else { "ASC" }; - Ok(format!( - "SELECT * \ - FROM EVENTS e \ - JOIN TRANSACTIONS t \ - ON t.tx_sequence_number = e.tx_sequence_number \ - AND t.transaction_digest = '\\x{}'::bytea \ - WHERE {cursor} \ - ORDER BY e.{EVENT_SEQUENCE_NUMBER_STR} {order_clause} \ - LIMIT {limit} - ", - Hex::encode(tx_digest.into_inner()), - )) + if descending_order { + query = query.order(events::event_sequence_number.desc()); + } else { + query = query.order(events::event_sequence_number.asc()); + } + + query = query.filter( + events::tx_sequence_number.nullable().eq(tx_digests::table + .select(tx_digests::tx_sequence_number) + // we filter the tx_digests table because it is indexed by digest, + // events table is not + .filter(tx_digests::tx_digest.eq(tx_digest.into_inner().to_vec())) + .single_value()), + ); + + let pool = self.get_pool(); + let stored_events = run_query_async!(&pool, move |conn| { + query.limit(limit as i64).load::(conn) + })?; + + let mut iota_event_futures = vec![]; + for stored_event in stored_events { + iota_event_futures.push(tokio::task::spawn( + stored_event.try_into_iota_event(self.package_resolver.clone()), + )); + } + + let iota_events = futures::future::join_all(iota_event_futures) + .await + .into_iter() + .collect::, _>>() + .tap_err(|e| tracing::error!("Failed to join iota event futures: {}", e))? + .into_iter() + .collect::, _>>() + .tap_err(|e| tracing::error!("Failed to collect iota event futures: {}", e))?; + Ok(iota_events) } pub async fn query_events_in_blocking_task( @@ -1093,21 +1159,22 @@ impl IndexerReader { transactions::dsl::transactions .select(transactions::tx_sequence_number) .filter( - transactions::dsl::transaction_digest.eq(tx_digest.into_inner().to_vec()), + transactions::tx_sequence_number + .nullable() + .eq(tx_digests::table + .select(tx_digests::tx_sequence_number) + // we filter the tx_digests table because it is indexed by digest, + // transactions table is not + .filter(tx_digests::tx_digest.eq(tx_digest.into_inner().to_vec())) + .single_value()), ) .first::(conn) })?; - (tx_seq, event_seq) + (tx_seq, event_seq as i64) } else if descending_order { - let max_tx_seq: i64 = run_query_async!(&pool, move |conn| { - events::dsl::events - .select(events::tx_sequence_number) - .order(events::dsl::tx_sequence_number.desc()) - .first::(conn) - }) - .map_or(-1, |max_tx_seq| max_tx_seq + 1); - - (max_tx_seq, 0) + let max_tx_seq = i64::MAX; + let max_event_seq = i64::MAX; + (max_tx_seq, max_event_seq) } else { (-1, 0) }; @@ -1147,7 +1214,9 @@ impl IndexerReader { limit, ) } else if let EventFilter::Transaction(tx_digest) = filter { - self.query_events_by_tx_digest_query(tx_digest, cursor, limit, descending_order)? + return self + .query_events_by_tx_digest(tx_digest, cursor, limit, descending_order) + .await; } else { let main_where_clause = match filter { EventFilter::Package(package_id) => { diff --git a/crates/iota-indexer/src/lib.rs b/crates/iota-indexer/src/lib.rs index 4593d1b75f2..0b6c58f39f5 100644 --- a/crates/iota-indexer/src/lib.rs +++ b/crates/iota-indexer/src/lib.rs @@ -47,45 +47,45 @@ pub mod test_utils; pub mod types; #[derive(Parser, Clone, Debug)] -#[clap( - name = "Iota indexer", - about = "An off-fullnode service serving data from Iota protocol", +#[command( + name = "IOTA indexer", + about = "An off-fullnode service serving data from IOTA protocol", rename_all = "kebab-case" )] pub struct IndexerConfig { - #[clap(long)] + #[arg(long)] pub db_url: Option>, - #[clap(long)] + #[arg(long)] pub db_user_name: Option, - #[clap(long)] + #[arg(long)] pub db_password: Option>, - #[clap(long)] + #[arg(long)] pub db_host: Option, - #[clap(long)] + #[arg(long)] pub db_port: Option, - #[clap(long)] + #[arg(long)] pub db_name: Option, - #[clap(long, default_value = "http://0.0.0.0:9000", global = true)] + #[arg(long, default_value = "http://0.0.0.0:9000", global = true)] pub rpc_client_url: String, - #[clap(long, default_value = Some("https://checkpoints.mainnet.iota.io"), global = true)] + #[arg(long, default_value = Some("https://checkpoints.mainnet.iota.cafe"), global = true)] pub remote_store_url: Option, - #[clap(long, default_value = "0.0.0.0", global = true)] + #[arg(long, default_value = "0.0.0.0", global = true)] pub client_metric_host: String, - #[clap(long, default_value = "9184", global = true)] + #[arg(long, default_value = "9184", global = true)] pub client_metric_port: u16, - #[clap(long, default_value = "0.0.0.0", global = true)] + #[arg(long, default_value = "0.0.0.0", global = true)] pub rpc_server_url: String, - #[clap(long, default_value = "9000", global = true)] + #[arg(long, default_value = "9000", global = true)] pub rpc_server_port: u16, - #[clap(long)] + #[arg(long)] pub reset_db: bool, - #[clap(long)] + #[arg(long)] pub fullnode_sync_worker: bool, - #[clap(long)] + #[arg(long)] pub rpc_server_worker: bool, - #[clap(long)] + #[arg(long)] pub data_ingestion_path: Option, - #[clap(long)] + #[arg(long)] pub analytical_worker: bool, } @@ -149,7 +149,7 @@ impl Default for IndexerConfig { db_port: None, db_name: None, rpc_client_url: "http://127.0.0.1:9000".to_string(), - remote_store_url: Some("https://checkpoints.mainnet.iota.io".to_string()), + remote_store_url: Some("https://checkpoints.mainnet.iota.cafe".to_string()), client_metric_host: "0.0.0.0".to_string(), client_metric_port: 9184, rpc_server_url: "0.0.0.0".to_string(), diff --git a/crates/iota-indexer/src/main.rs b/crates/iota-indexer/src/main.rs index 791b16a5469..11c04da67b3 100644 --- a/crates/iota-indexer/src/main.rs +++ b/crates/iota-indexer/src/main.rs @@ -13,7 +13,7 @@ async fn main() -> Result<(), IndexerError> { .with_env() .init(); warn!( - "WARNING: Iota indexer is still experimental and we expect occasional breaking changes that require backfills." + "WARNING: IOTA indexer is still experimental and we expect occasional breaking changes that require backfills." ); let mut indexer_config = IndexerConfig::parse(); diff --git a/crates/iota-indexer/src/metrics.rs b/crates/iota-indexer/src/metrics.rs index 0e01cb3c317..4f220c1bad3 100644 --- a/crates/iota-indexer/src/metrics.rs +++ b/crates/iota-indexer/src/metrics.rs @@ -337,7 +337,7 @@ impl IndexerMetrics { ).unwrap(), fullnode_checkpoint_data_download_latency: register_histogram_with_registry!( "fullnode_checkpoint_data_download_latency", - "Time spent in downloading checkpoint and transation for a new checkpoint from the Full Node", + "Time spent in downloading checkpoint and transaction for a new checkpoint from the Full Node", DATA_INGESTION_LATENCY_SEC_BUCKETS.to_vec(), registry, ) @@ -458,7 +458,7 @@ impl IndexerMetrics { ) .unwrap(), checkpoint_db_commit_latency_transactions_chunks_transformation: register_histogram_with_registry!( - "checkpoint_db_commit_latency_transactions_transaformation", + "checkpoint_db_commit_latency_transactions_transformation", "Time spent in transactions chunks transformation prior to commit", DATA_INGESTION_LATENCY_SEC_BUCKETS.to_vec(), registry, diff --git a/crates/iota-indexer/src/models/large_objects.rs b/crates/iota-indexer/src/models/large_objects.rs index e5f179e1ba6..a386eb620a4 100644 --- a/crates/iota-indexer/src/models/large_objects.rs +++ b/crates/iota-indexer/src/models/large_objects.rs @@ -78,7 +78,8 @@ pub fn put_large_object_in_chunks( let offset = (chunk_num * chunk_size) as i64; tracing::trace!("Storing large-object chunk at offset {}", offset); // TODO: (to treat in a different issue): - // remove dangling chunks (either by using a transaction or by handlng manually) + // remove dangling chunks (either by using a transaction or by handling + // manually) // // additionally we could apply a backoff retry strategy transactional_blocking_with_retry!( diff --git a/crates/iota-indexer/tests/common/mod.rs b/crates/iota-indexer/tests/common/mod.rs index 23ccf354899..8b1089ac60a 100644 --- a/crates/iota-indexer/tests/common/mod.rs +++ b/crates/iota-indexer/tests/common/mod.rs @@ -18,11 +18,13 @@ use iota_indexer::{ test_utils::{ReaderWriterConfig, start_test_indexer}, }; use iota_json_rpc_api::ReadApiClient; -use iota_json_rpc_types::IotaTransactionBlockResponseOptions; +use iota_json_rpc_types::{IotaTransactionBlockResponseOptions, TransactionBlockBytes}; use iota_metrics::init_metrics; use iota_types::{ base_types::{ObjectID, SequenceNumber}, + crypto::AccountKeyPair, digests::TransactionDigest, + utils::to_sender_signed_transaction, }; use jsonrpsee::{ http_client::{HttpClient, HttpClientBuilder}, @@ -238,6 +240,18 @@ pub async fn indexer_wait_for_transaction( .expect("Timeout waiting for indexer to catchup to given transaction"); } +pub async fn execute_tx_and_wait_for_indexer( + indexer_client: &HttpClient, + cluster: &TestCluster, + store: &PgIndexerStore, + tx_bytes: TransactionBlockBytes, + keypair: &AccountKeyPair, +) { + let txn = to_sender_signed_transaction(tx_bytes.to_data().unwrap(), keypair); + let res = cluster.wallet.execute_transaction_must_succeed(txn).await; + indexer_wait_for_transaction(res.digest, store, indexer_client).await; +} + /// Start an Indexer instance in `Read` mode fn start_indexer_reader( fullnode_rpc_url: impl Into, diff --git a/crates/iota-indexer/tests/rpc-tests/coin_api.rs b/crates/iota-indexer/tests/rpc-tests/coin_api.rs index 208d0b166fd..4e4eec7ecb1 100644 --- a/crates/iota-indexer/tests/rpc-tests/coin_api.rs +++ b/crates/iota-indexer/tests/rpc-tests/coin_api.rs @@ -3,6 +3,8 @@ use std::{path::PathBuf, str::FromStr}; +use diesel::PgConnection; +use iota_indexer::store::PgIndexerStore; use iota_json::{call_args, type_args}; use iota_json_rpc_api::{CoinReadApiClient, TransactionBuilderClient, WriteApiClient}; use iota_json_rpc_types::{ @@ -25,15 +27,15 @@ use jsonrpsee::http_client::HttpClient; use test_cluster::TestCluster; use tokio::sync::OnceCell; -use crate::common::{ApiTestSetup, indexer_wait_for_object}; +use crate::common::{ApiTestSetup, execute_tx_and_wait_for_indexer, indexer_wait_for_object}; -static COMMON_TESTING_ADDR_AND_CUSTOM_COIN_NAME: OnceCell<(IotaAddress, String)> = +static COMMON_TESTING_ADDR_AND_CUSTOM_COIN_NAME: OnceCell<(IotaAddress, AccountKeyPair, String)> = OnceCell::const_new(); async fn get_or_init_addr_and_custom_coins( cluster: &TestCluster, indexer_client: &HttpClient, -) -> &'static (IotaAddress, String) { +) -> &'static (IotaAddress, AccountKeyPair, String) { COMMON_TESTING_ADDR_AND_CUSTOM_COIN_NAME .get_or_init(|| async { let (address, keypair): (_, AccountKeyPair) = get_key_pair(); @@ -49,7 +51,7 @@ async fn get_or_init_addr_and_custom_coins( } let (coin_name, coin_object_ref) = - create_and_mint_trusted_coin(cluster, address, keypair, 100_000) + create_and_mint_trusted_coin(cluster, address, &keypair, 100_000) .await .unwrap(); @@ -60,7 +62,7 @@ async fn get_or_init_addr_and_custom_coins( ) .await; - (address, coin_name) + (address, keypair, coin_name) }) .await } @@ -74,7 +76,7 @@ fn get_coins_basic_scenario() { .. } = ApiTestSetup::get_or_init(); runtime.block_on(async move { - let (owner, _) = get_or_init_addr_and_custom_coins(cluster, client).await; + let (owner, _, _) = get_or_init_addr_and_custom_coins(cluster, client).await; let (result_fullnode, result_indexer) = get_coins_fullnode_indexer(cluster, client, *owner, None, None, None).await; @@ -93,7 +95,7 @@ fn get_coins_with_cursor() { .. } = ApiTestSetup::get_or_init(); runtime.block_on(async move { - let (owner, _) = get_or_init_addr_and_custom_coins(cluster, client).await; + let (owner, _, _) = get_or_init_addr_and_custom_coins(cluster, client).await; let all_coins = cluster .rpc_client() .get_coins(*owner, None, None, None) @@ -118,7 +120,7 @@ fn get_coins_with_limit() { .. } = ApiTestSetup::get_or_init(); runtime.block_on(async move { - let (owner, _) = get_or_init_addr_and_custom_coins(cluster, client).await; + let (owner, _, _) = get_or_init_addr_and_custom_coins(cluster, client).await; let (result_fullnode, result_indexer) = get_coins_fullnode_indexer(cluster, client, *owner, None, None, Some(2)).await; @@ -137,7 +139,7 @@ fn get_coins_custom_coin() { .. } = ApiTestSetup::get_or_init(); runtime.block_on(async move { - let (owner, coin_name) = get_or_init_addr_and_custom_coins(cluster, client).await; + let (owner, _, coin_name) = get_or_init_addr_and_custom_coins(cluster, client).await; let (result_fullnode, result_indexer) = get_coins_fullnode_indexer( cluster, @@ -163,7 +165,7 @@ fn get_all_coins_basic_scenario() { .. } = ApiTestSetup::get_or_init(); runtime.block_on(async move { - let (owner, _) = get_or_init_addr_and_custom_coins(cluster, client).await; + let (owner, _, _) = get_or_init_addr_and_custom_coins(cluster, client).await; let (result_fullnode, result_indexer) = get_all_coins_fullnode_indexer(cluster, client, *owner, None, None).await; @@ -183,7 +185,7 @@ fn get_all_coins_with_cursor() { .. } = ApiTestSetup::get_or_init(); runtime.block_on(async move { - let (owner, _) = get_or_init_addr_and_custom_coins(cluster, client).await; + let (owner, _, _) = get_or_init_addr_and_custom_coins(cluster, client).await; let all_coins = cluster .rpc_client() @@ -218,7 +220,7 @@ fn get_all_coins_with_limit() { .. } = ApiTestSetup::get_or_init(); runtime.block_on(async move { - let (owner, _) = get_or_init_addr_and_custom_coins(cluster, client).await; + let (owner, _, _) = get_or_init_addr_and_custom_coins(cluster, client).await; let (result_fullnode, result_indexer) = get_all_coins_fullnode_indexer(cluster, client, *owner, None, Some(2)).await; @@ -237,7 +239,7 @@ fn get_balance_iota_coin() { .. } = ApiTestSetup::get_or_init(); runtime.block_on(async move { - let (owner, _) = get_or_init_addr_and_custom_coins(cluster, client).await; + let (owner, _, _) = get_or_init_addr_and_custom_coins(cluster, client).await; let (result_fullnode, result_indexer) = get_balance_fullnode_indexer(cluster, client, *owner, None).await; @@ -255,7 +257,7 @@ fn get_balance_custom_coin() { .. } = ApiTestSetup::get_or_init(); runtime.block_on(async move { - let (owner, coin_name) = get_or_init_addr_and_custom_coins(cluster, client).await; + let (owner, _, coin_name) = get_or_init_addr_and_custom_coins(cluster, client).await; let (result_fullnode, result_indexer) = get_balance_fullnode_indexer(cluster, client, *owner, Some(coin_name.to_string())) @@ -274,7 +276,35 @@ fn get_all_balances() { .. } = ApiTestSetup::get_or_init(); runtime.block_on(async move { - let (owner, _) = get_or_init_addr_and_custom_coins(cluster, client).await; + let (owner, _, _) = get_or_init_addr_and_custom_coins(cluster, client).await; + + let (mut result_fullnode, mut result_indexer) = + get_all_balances_fullnode_indexer(cluster, client, *owner).await; + + result_fullnode.sort_by_key(|balance: &Balance| balance.coin_type.clone()); + result_indexer.sort_by_key(|balance: &Balance| balance.coin_type.clone()); + + assert_eq!(result_fullnode, result_indexer); + }); +} + +#[test] +fn get_all_balances_with_zero_iotas() { + let ApiTestSetup { + runtime, + client, + cluster, + store, + } = ApiTestSetup::get_or_init(); + runtime.block_on(async move { + let (owner, keypair, _) = get_or_init_addr_and_custom_coins(cluster, client).await; + let coins_dump_address = IotaAddress::random_for_testing_only(); + + // first call is to make node and potentially the indexer cache the result + // and increase chance of producing wrong result on the second call + get_all_balances_fullnode_indexer(cluster, client, *owner).await; + + transfer_all_coins(cluster, client, store, *owner, keypair, coins_dump_address).await; let (mut result_fullnode, mut result_indexer) = get_all_balances_fullnode_indexer(cluster, client, *owner).await; @@ -295,7 +325,7 @@ fn get_coin_metadata() { .. } = ApiTestSetup::get_or_init(); runtime.block_on(async move { - let (_, coin_name) = get_or_init_addr_and_custom_coins(cluster, client).await; + let (_, _, coin_name) = get_or_init_addr_and_custom_coins(cluster, client).await; let (result_fullnode, result_indexer) = get_coin_metadata_fullnode_indexer(cluster, client, coin_name.to_string()).await; @@ -314,7 +344,7 @@ fn get_total_supply() { .. } = ApiTestSetup::get_or_init(); runtime.block_on(async move { - let (_, coin_name) = get_or_init_addr_and_custom_coins(cluster, client).await; + let (_, _, coin_name) = get_or_init_addr_and_custom_coins(cluster, client).await; let (result_fullnode, result_indexer) = get_total_supply_fullnode_indexer(cluster, client, coin_name.to_string()).await; @@ -415,7 +445,7 @@ async fn get_total_supply_fullnode_indexer( async fn create_and_mint_trusted_coin( cluster: &TestCluster, address: IotaAddress, - account_keypair: AccountKeyPair, + account_keypair: &AccountKeyPair, amount: u64, ) -> Result<(String, IotaObjectRef), anyhow::Error> { let http_client = cluster.rpc_client(); @@ -446,7 +476,7 @@ async fn create_and_mint_trusted_coin( .unwrap(); let signed_transaction = - to_sender_signed_transaction(transaction_bytes.to_data().unwrap(), &account_keypair); + to_sender_signed_transaction(transaction_bytes.to_data().unwrap(), account_keypair); let (tx_bytes, signatures) = signed_transaction.to_tx_bytes_and_signatures(); let tx_response: IotaTransactionBlockResponse = http_client @@ -513,7 +543,7 @@ async fn create_and_mint_trusted_coin( .unwrap(); let signed_transaction = - to_sender_signed_transaction(transaction_bytes.to_data().unwrap(), &account_keypair); + to_sender_signed_transaction(transaction_bytes.to_data().unwrap(), account_keypair); let (tx_bytes, signatures) = signed_transaction.to_tx_bytes_and_signatures(); let tx_response = http_client @@ -537,3 +567,29 @@ async fn create_and_mint_trusted_coin( Ok((coin_name, created_coin_obj_ref)) } + +async fn transfer_all_coins( + cluster: &TestCluster, + indexer_client: &HttpClient, + store: &PgIndexerStore, + from_address: IotaAddress, + keypair: &AccountKeyPair, + to_address: IotaAddress, +) { + let coins: Vec<_> = cluster + .rpc_client() + .get_coins(from_address, None, None, None) + .await + .unwrap() + .data + .iter() + .map(|coin| coin.coin_object_id) + .collect(); + + let tx_bytes: TransactionBlockBytes = indexer_client + .pay_all_iota(from_address, coins, to_address, 10_000_000.into()) + .await + .unwrap(); + + execute_tx_and_wait_for_indexer(indexer_client, cluster, store, tx_bytes, keypair).await; +} diff --git a/crates/iota-indexer/tests/rpc-tests/indexer_api.rs b/crates/iota-indexer/tests/rpc-tests/indexer_api.rs index 49f4e35ff80..72c42d4e799 100644 --- a/crates/iota-indexer/tests/rpc-tests/indexer_api.rs +++ b/crates/iota-indexer/tests/rpc-tests/indexer_api.rs @@ -30,8 +30,9 @@ use move_core_types::{ }; use crate::common::{ - ApiTestSetup, indexer_wait_for_checkpoint, indexer_wait_for_object, - indexer_wait_for_transaction, rpc_call_error_msg_matches, + ApiTestSetup, indexer_wait_for_checkpoint, indexer_wait_for_latest_checkpoint, + indexer_wait_for_object, indexer_wait_for_transaction, rpc_call_error_msg_matches, + start_test_cluster_with_read_write_indexer, }; #[test] @@ -185,36 +186,64 @@ fn query_events_supported_events() { }); } -#[test] -fn query_validator_epoch_info_event() { - let ApiTestSetup { - runtime, - store, - client, - cluster, - } = ApiTestSetup::get_or_init(); - - runtime.block_on(async move { - indexer_wait_for_checkpoint(store, 1).await; - - cluster.force_new_epoch().await; - - let result = client.query_events(EventFilter::MoveEventType("0x0000000000000000000000000000000000000000000000000000000000000003::validator_set::ValidatorEpochInfoEventV1".parse().unwrap()), None, None, None).await; - assert!(result.is_ok()); - assert!(!result.unwrap().data.is_empty()); +#[tokio::test] +async fn query_validator_epoch_info_event() { + let (cluster, store, client) = + &start_test_cluster_with_read_write_indexer(Some("query_validator_epoch_info_event"), None) + .await; + indexer_wait_for_checkpoint(store, 1).await; - let result = client.query_events(EventFilter::MoveEventType("0x3::validator_set::ValidatorEpochInfoEventV1".parse().unwrap()), None, None, None).await; - assert!(result.is_ok()); - assert!(!result.unwrap().data.is_empty()); + cluster.force_new_epoch().await; + indexer_wait_for_latest_checkpoint(store, cluster).await; - let result = client.query_events(EventFilter::MoveEventType("0x0003::validator_set::ValidatorEpochInfoEventV1".parse().unwrap()), None, None, None).await; - assert!(result.is_ok()); - assert!(!result.unwrap().data.is_empty()); + let result = client.query_events(EventFilter::MoveEventType("0x0000000000000000000000000000000000000000000000000000000000000003::validator_set::ValidatorEpochInfoEventV1".parse().unwrap()), None, None, None).await; + assert!(result.is_ok()); + assert!(!result.unwrap().data.is_empty()); - let result = client.query_events(EventFilter::MoveEventType("0x1::validator_set::ValidatorEpochInfoEventV1".parse().unwrap()), None, None, None).await; - assert!(result.is_ok()); - assert!(result.unwrap().data.is_empty()); - }); + let result = client + .query_events( + EventFilter::MoveEventType( + "0x3::validator_set::ValidatorEpochInfoEventV1" + .parse() + .unwrap(), + ), + None, + None, + None, + ) + .await; + assert!(result.is_ok()); + assert!(!result.unwrap().data.is_empty()); + + let result = client + .query_events( + EventFilter::MoveEventType( + "0x0003::validator_set::ValidatorEpochInfoEventV1" + .parse() + .unwrap(), + ), + None, + None, + None, + ) + .await; + assert!(result.is_ok()); + assert!(!result.unwrap().data.is_empty()); + + let result = client + .query_events( + EventFilter::MoveEventType( + "0x1::validator_set::ValidatorEpochInfoEventV1" + .parse() + .unwrap(), + ), + None, + None, + None, + ) + .await; + assert!(result.is_ok()); + assert!(result.unwrap().data.is_empty()); } #[test] diff --git a/crates/iota-indexer/tests/rpc-tests/transaction_builder.rs b/crates/iota-indexer/tests/rpc-tests/transaction_builder.rs index ae4612d465b..cfb35f33a5a 100644 --- a/crates/iota-indexer/tests/rpc-tests/transaction_builder.rs +++ b/crates/iota-indexer/tests/rpc-tests/transaction_builder.rs @@ -28,14 +28,13 @@ use iota_types::{ label::label_struct_tag_to_string, stardust_upgrade_label::stardust_upgrade_label_type, timelock::TimeLock, }, - utils::to_sender_signed_transaction, }; use jsonrpsee::http_client::HttpClient; use test_cluster::TestCluster; use crate::common::{ - ApiTestSetup, indexer_wait_for_checkpoint, indexer_wait_for_latest_checkpoint, - indexer_wait_for_object, indexer_wait_for_transaction, + ApiTestSetup, execute_tx_and_wait_for_indexer, indexer_wait_for_checkpoint, + indexer_wait_for_latest_checkpoint, indexer_wait_for_object, start_test_cluster_with_read_write_indexer, }; const FUNDED_BALANCE_PER_COIN: u64 = 10_000_000_000; @@ -772,18 +771,6 @@ fn request_withdraw_timelocked_stake_from_active() { .unwrap(); } -async fn execute_tx_and_wait_for_indexer( - indexer_client: &HttpClient, - cluster: &TestCluster, - store: &PgIndexerStore, - tx_bytes: TransactionBlockBytes, - keypair: &AccountKeyPair, -) { - let txn = to_sender_signed_transaction(tx_bytes.to_data().unwrap(), keypair); - let res = cluster.wallet.execute_transaction_must_succeed(txn).await; - indexer_wait_for_transaction(res.digest, store, indexer_client).await; -} - async fn get_address_balances(indexer_client: &HttpClient, address: IotaAddress) -> Vec { indexer_client .get_coins(address, None, None, None) diff --git a/crates/iota-indexer/tests/rpc-tests/write_api.rs b/crates/iota-indexer/tests/rpc-tests/write_api.rs index 200e80f66d9..91997444a53 100644 --- a/crates/iota-indexer/tests/rpc-tests/write_api.rs +++ b/crates/iota-indexer/tests/rpc-tests/write_api.rs @@ -2,15 +2,14 @@ // SPDX-License-Identifier: Apache-2.0 use fastcrypto::encoding::Base64; -use iota_json_rpc_api::{ - IndexerApiClient, ReadApiClient, TransactionBuilderClient, WriteApiClient, -}; +use iota_json_rpc_api::{ReadApiClient, TransactionBuilderClient, WriteApiClient}; use iota_json_rpc_types::{ IotaExecutionStatus, IotaObjectDataOptions, IotaTransactionBlockEffectsAPI, IotaTransactionBlockResponseOptions, }; use iota_types::{ base_types::{IotaAddress, ObjectID}, + crypto::{AccountKeyPair, get_key_pair}, object::Owner, programmable_transaction_builder::ProgrammableTransactionBuilder, quorum_driver_types::ExecuteTransactionRequestType, @@ -125,17 +124,28 @@ fn dev_inspect_transaction_block() { runtime.block_on(async { indexer_wait_for_checkpoint(store, 1).await; - let sender = cluster.get_address_0(); - let receiver = cluster.get_address_1(); + let (sender, _): (_, AccountKeyPair) = get_key_pair(); + let (receiver, _): (_, AccountKeyPair) = get_key_pair(); - let objects = cluster - .rpc_client() - .get_owned_objects(sender, None, None, None) - .await - .unwrap() - .data; + let gas = cluster + .fund_address_and_return_gas( + cluster.get_reference_gas_price().await, + Some(10_000_000_000), + sender, + ) + .await; - let (obj_id, seq_num, digest) = objects.first().unwrap().object().unwrap().object_ref(); + indexer_wait_for_object(client, gas.0, gas.1).await; + + let (obj_id, seq_num, digest) = cluster + .fund_address_and_return_gas( + cluster.get_reference_gas_price().await, + Some(10_000_000_000), + sender, + ) + .await; + + indexer_wait_for_object(client, obj_id, seq_num).await; let mut builder = ProgrammableTransactionBuilder::new(); builder @@ -173,15 +183,23 @@ fn dev_inspect_transaction_block() { .await .unwrap(); + // Ensure that the actual object sequence number remains unchanged after the + // checkpoint advances indexer_wait_for_checkpoint(store, latest_checkpoint_seq_number.into_inner() + 1).await; - let actual_object_info = client + let actual_object_data = client .get_object(obj_id, Some(IotaObjectDataOptions::new().with_owner())) .await + .unwrap() + .data .unwrap(); assert_eq!( - actual_object_info.data.unwrap().owner.unwrap(), + actual_object_data.version, seq_num, + "The object sequence number should not mutate" + ); + assert_eq!( + actual_object_data.owner.unwrap(), Owner::AddressOwner(sender), "The initial owner of the object should not change" ); diff --git a/crates/iota-json-rpc-api/Cargo.toml b/crates/iota-json-rpc-api/Cargo.toml index 7bec4ba6e6c..2d1478439ac 100644 --- a/crates/iota-json-rpc-api/Cargo.toml +++ b/crates/iota-json-rpc-api/Cargo.toml @@ -24,7 +24,7 @@ iota-open-rpc.workspace = true iota-open-rpc-macros.workspace = true iota-types.workspace = true # NOTE: It's important to keep the above dependency list short. -# This and the iota-sdk crate are widely used to develop on Iota and it's valuable +# This and the iota-sdk crate are widely used to develop on IOTA and it's valuable # to not have to pull in the entire iota repo for it. [dev-dependencies] diff --git a/crates/iota-json-rpc-api/src/coin.rs b/crates/iota-json-rpc-api/src/coin.rs index f3201e4ec64..23d63eee1de 100644 --- a/crates/iota-json-rpc-api/src/coin.rs +++ b/crates/iota-json-rpc-api/src/coin.rs @@ -20,7 +20,7 @@ pub trait CoinReadApi { #[method(name = "getCoins")] async fn get_coins( &self, - /// the owner's Iota address + /// the owner's IOTA address owner: IotaAddress, /// optional type name for the coin (e.g., 0x168da5bf1f48dafc111b0a488fa454aca95e0b5e::usdc::USDC), default to 0x2::iota::IOTA if not specified. coin_type: Option, @@ -35,7 +35,7 @@ pub trait CoinReadApi { #[method(name = "getAllCoins")] async fn get_all_coins( &self, - /// the owner's Iota address + /// the owner's IOTA address owner: IotaAddress, /// optional paging cursor cursor: Option, @@ -48,7 +48,7 @@ pub trait CoinReadApi { #[method(name = "getBalance")] async fn get_balance( &self, - /// the owner's Iota address + /// the owner's IOTA address owner: IotaAddress, /// optional type names for the coin (e.g., 0x168da5bf1f48dafc111b0a488fa454aca95e0b5e::usdc::USDC), default to 0x2::iota::IOTA if not specified. coin_type: Option, @@ -59,7 +59,7 @@ pub trait CoinReadApi { #[method(name = "getAllBalances")] async fn get_all_balances( &self, - /// the owner's Iota address + /// the owner's IOTA address owner: IotaAddress, ) -> RpcResult>; diff --git a/crates/iota-json-rpc-api/src/extended.rs b/crates/iota-json-rpc-api/src/extended.rs index b048fc7c425..4e7d796b8b4 100644 --- a/crates/iota-json-rpc-api/src/extended.rs +++ b/crates/iota-json-rpc-api/src/extended.rs @@ -14,7 +14,7 @@ use jsonrpsee::{core::RpcResult, proc_macros::rpc}; #[open_rpc(namespace = "iotax", tag = "Extended API")] #[rpc(server, client, namespace = "iotax")] pub trait ExtendedApi { - /// Return a list of epoch info + /// Return a list of epoch info. Exclusively served by the indexer. #[rustfmt::skip] #[method(name = "getEpochs")] async fn get_epochs( @@ -27,7 +27,8 @@ pub trait ExtendedApi { descending_order: Option, ) -> RpcResult; - /// Return a list of epoch metrics, which is a subset of epoch info + /// Return a list of epoch metrics, which is a subset of epoch info. + /// Exclusively served by the indexer. #[method(name = "getEpochMetrics")] async fn get_epoch_metrics( &self, @@ -39,31 +40,35 @@ pub trait ExtendedApi { descending_order: Option, ) -> RpcResult; - /// Return current epoch info + /// Return current epoch info. Exclusively served by the indexer. #[method(name = "getCurrentEpoch")] async fn get_current_epoch(&self) -> RpcResult; - /// Return Network metrics + /// Return Network metrics. Exclusively served by the indexer. #[method(name = "getNetworkMetrics")] async fn get_network_metrics(&self) -> RpcResult; - /// Return move call metrics + /// Return move call metrics. Exclusively served by the indexer. #[method(name = "getMoveCallMetrics")] async fn get_move_call_metrics(&self) -> RpcResult; - /// Address related metrics + /// Address related metrics. Exclusively served by the indexer. #[method(name = "getLatestAddressMetrics")] async fn get_latest_address_metrics(&self) -> RpcResult; + /// Address related metrics. Exclusively served by the indexer. #[method(name = "getCheckpointAddressMetrics")] async fn get_checkpoint_address_metrics(&self, checkpoint: u64) -> RpcResult; + /// Address related metrics. Exclusively served by the indexer. #[method(name = "getAllEpochAddressMetrics")] async fn get_all_epoch_address_metrics( &self, descending_order: Option, ) -> RpcResult>; + /// Return the total number of transactions. Exclusively served by the + /// indexer. #[method(name = "getTotalTransactions")] async fn get_total_transactions(&self) -> RpcResult>; } diff --git a/crates/iota-json-rpc-api/src/indexer.rs b/crates/iota-json-rpc-api/src/indexer.rs index b77ab45d2d6..e25363c265d 100644 --- a/crates/iota-json-rpc-api/src/indexer.rs +++ b/crates/iota-json-rpc-api/src/indexer.rs @@ -33,7 +33,7 @@ pub trait IndexerApi { #[method(name = "getOwnedObjects")] async fn get_owned_objects( &self, - /// the owner's Iota address + /// the owner's IOTA address address: IotaAddress, /// the objects query criteria. query: Option, @@ -73,7 +73,7 @@ pub trait IndexerApi { descending_order: Option, ) -> RpcResult; - /// Subscribe to a stream of Iota event + /// Subscribe to a stream of IOTA event #[rustfmt::skip] #[subscription(name = "subscribeEvent", item = IotaEvent)] fn subscribe_event( @@ -82,7 +82,7 @@ pub trait IndexerApi { filter: EventFilter, ) -> SubscriptionResult; - /// Subscribe to a stream of Iota transaction effects + /// Subscribe to a stream of IOTA transaction effects #[subscription(name = "subscribeTransaction", item = IotaTransactionBlockEffects)] fn subscribe_transaction(&self, filter: TransactionFilter) -> SubscriptionResult; diff --git a/crates/iota-json-rpc-api/src/transaction_builder.rs b/crates/iota-json-rpc-api/src/transaction_builder.rs index eb2019ae1d9..c8fd2d80b8a 100644 --- a/crates/iota-json-rpc-api/src/transaction_builder.rs +++ b/crates/iota-json-rpc-api/src/transaction_builder.rs @@ -26,7 +26,7 @@ pub trait TransactionBuilder { #[method(name = "transferObject")] async fn transfer_object( &self, - /// the transaction signer's Iota address + /// the transaction signer's IOTA address signer: IotaAddress, /// the ID of the object to be transferred object_id: ObjectID, @@ -34,22 +34,22 @@ pub trait TransactionBuilder { gas: Option, /// the gas budget, the transaction will fail if the gas cost exceed the budget gas_budget: BigInt, - /// the recipient's Iota address + /// the recipient's IOTA address recipient: IotaAddress, ) -> RpcResult; - /// Create an unsigned transaction to send IOTA coin object to a Iota address. The IOTA object is also used as the gas object. + /// Create an unsigned transaction to send IOTA coin object to an IOTA address. The IOTA object is also used as the gas object. #[rustfmt::skip] #[method(name = "transferIota")] async fn transfer_iota( &self, - /// the transaction signer's Iota address + /// the transaction signer's IOTA address signer: IotaAddress, - /// the Iota coin object to be used in this transaction + /// the IOTA coin object to be used in this transaction iota_object_id: ObjectID, /// the gas budget, the transaction will fail if the gas cost exceed the budget gas_budget: BigInt, - /// the recipient's Iota address + /// the recipient's IOTA address recipient: IotaAddress, /// the amount to be split out and transferred amount: Option>, @@ -63,9 +63,9 @@ pub trait TransactionBuilder { #[method(name = "pay")] async fn pay( &self, - /// the transaction signer's Iota address + /// the transaction signer's IOTA address signer: IotaAddress, - /// the Iota coins to be used in this transaction + /// the IOTA coins to be used in this transaction input_coins: Vec, /// the recipients' addresses, the length of this vector must be the same as amounts. recipients: Vec, @@ -90,9 +90,9 @@ pub trait TransactionBuilder { #[method(name = "payIota")] async fn pay_iota( &self, - /// the transaction signer's Iota address + /// the transaction signer's IOTA address signer: IotaAddress, - /// the Iota coins to be used in this transaction, including the coin for gas payment. + /// the IOTA coins to be used in this transaction, including the coin for gas payment. input_coins: Vec, /// the recipients' addresses, the length of this vector must be the same as amounts. recipients: Vec, @@ -113,9 +113,9 @@ pub trait TransactionBuilder { #[method(name = "payAllIota")] async fn pay_all_iota( &self, - /// the transaction signer's Iota address + /// the transaction signer's IOTA address signer: IotaAddress, - /// the Iota coins to be used in this transaction, including the coin for gas payment. + /// the IOTA coins to be used in this transaction, including the coin for gas payment. input_coins: Vec, /// the recipient address, recipient: IotaAddress, @@ -128,7 +128,7 @@ pub trait TransactionBuilder { #[method(name = "moveCall")] async fn move_call( &self, - /// the transaction signer's Iota address + /// the transaction signer's IOTA address signer: IotaAddress, /// the Move package ID, e.g. `0x2` package_object_id: ObjectID, @@ -153,7 +153,7 @@ pub trait TransactionBuilder { #[method(name = "publish")] async fn publish( &self, - /// the transaction signer's Iota address + /// the transaction signer's IOTA address sender: IotaAddress, /// the compiled bytes of a Move package compiled_modules: Vec, @@ -171,7 +171,7 @@ pub trait TransactionBuilder { #[method(name = "splitCoin")] async fn split_coin( &self, - /// the transaction signer's Iota address + /// the transaction signer's IOTA address signer: IotaAddress, /// the coin object to be spilt coin_object_id: ObjectID, @@ -188,7 +188,7 @@ pub trait TransactionBuilder { #[method(name = "splitCoinEqual")] async fn split_coin_equal( &self, - /// the transaction signer's Iota address + /// the transaction signer's IOTA address signer: IotaAddress, /// the coin object to be spilt coin_object_id: ObjectID, @@ -205,7 +205,7 @@ pub trait TransactionBuilder { #[method(name = "mergeCoins")] async fn merge_coin( &self, - /// the transaction signer's Iota address + /// the transaction signer's IOTA address signer: IotaAddress, /// the coin object to merge into, this coin will remain after the transaction primary_coin: ObjectID, @@ -222,7 +222,7 @@ pub trait TransactionBuilder { #[method(name = "batchTransaction")] async fn batch_transaction( &self, - /// the transaction signer's Iota address + /// the transaction signer's IOTA address signer: IotaAddress, /// list of transaction request parameters single_transaction_params: Vec, @@ -239,13 +239,13 @@ pub trait TransactionBuilder { #[method(name = "requestAddStake")] async fn request_add_stake( &self, - /// the transaction signer's Iota address + /// the transaction signer's IOTA address signer: IotaAddress, /// Coin object to stake coins: Vec, /// stake amount amount: Option>, - /// the validator's Iota address + /// the validator's IOTA address validator: IotaAddress, /// gas object to be used in this transaction, node will pick one from the signer's possession if not provided gas: Option, @@ -258,7 +258,7 @@ pub trait TransactionBuilder { #[method(name = "requestWithdrawStake")] async fn request_withdraw_stake( &self, - /// the transaction signer's Iota address + /// the transaction signer's IOTA address signer: IotaAddress, /// StakedIota object ID staked_iota: ObjectID, @@ -273,11 +273,11 @@ pub trait TransactionBuilder { #[method(name = "requestAddTimelockedStake")] async fn request_add_timelocked_stake( &self, - /// the transaction signer's Iota address + /// the transaction signer's IOTA address signer: IotaAddress, /// TimeLock> object to stake locked_balance: ObjectID, - /// the validator's Iota address + /// the validator's IOTA address validator: IotaAddress, /// gas object to be used in this transaction gas: ObjectID, @@ -290,7 +290,7 @@ pub trait TransactionBuilder { #[method(name = "requestWithdrawTimelockedStake")] async fn request_withdraw_timelocked_stake( &self, - /// the transaction signer's Iota address + /// the transaction signer's IOTA address signer: IotaAddress, /// TimelockedStakedIota object ID timelocked_staked_iota: ObjectID, diff --git a/crates/iota-json-rpc-tests/tests/coin_api.rs b/crates/iota-json-rpc-tests/tests/coin_api.rs index 1cdf7328abd..276ee081f48 100644 --- a/crates/iota-json-rpc-tests/tests/coin_api.rs +++ b/crates/iota-json-rpc-tests/tests/coin_api.rs @@ -28,7 +28,7 @@ use iota_types::{ quorum_driver_types::ExecuteTransactionRequestType, }; use jsonrpsee::http_client::HttpClient; -use test_cluster::TestClusterBuilder; +use test_cluster::{TestCluster, TestClusterBuilder}; async fn create_and_mint_coins( http_client: &HttpClient, @@ -88,7 +88,10 @@ async fn create_and_mint_coins( }) .unwrap(); - let coin_name = format!("{package_id}::trusted_coin::TRUSTED_COIN"); + let coin_name = format!( + "{}::trusted_coin::TRUSTED_COIN", + package_id.to_hex_literal() + ); let result: Supply = http_client .get_total_supply(coin_name.clone()) .await @@ -718,3 +721,86 @@ async fn get_all_balances() { "New coin should have correct balance" ); } + +#[sim_test] +async fn get_all_balances_after_zeroing_coins_count() { + let cluster = TestClusterBuilder::new().build().await; + let http_client = cluster.rpc_client(); + let address = cluster.get_address_0(); + let other_address = cluster.get_address_1(); + + // Get all balances + let balances: Vec = http_client.get_all_balances(address).await.unwrap(); + assert!(!balances.is_empty(), "Should have some balances"); + + // Check if IOTA balance exists + let iota_balance = balances + .iter() + .find(|b| b.coin_type == "0x2::iota::IOTA") + .expect("IOTA balance should exist"); + + assert!( + iota_balance.coin_object_count > 0, + "There should be more than 0 IOTA coins" + ); + + transfer_all_coins(&cluster, http_client, address, other_address) + .await + .unwrap(); + + let updated_balances: Vec = http_client.get_all_balances(address).await.unwrap(); + assert!( + updated_balances.is_empty(), + "Should have no balances after sending away all IOTA coins" + ); +} + +async fn transfer_all_coins( + cluster: &TestCluster, + http_client: &HttpClient, + from_address: IotaAddress, + to_address: IotaAddress, +) -> Result { + let coins = http_client + .get_coins(from_address, None, None, None) + .await + .unwrap() + .data + .iter() + .map(|coin| coin.coin_object_id) + .collect(); + + let transaction_bytes: TransactionBlockBytes = http_client + .pay_all_iota(from_address, coins, to_address, 10_000_000.into()) + .await?; + + execute_tx(cluster, http_client, transaction_bytes).await +} + +async fn execute_tx( + cluster: &TestCluster, + http_client: &HttpClient, + transaction_bytes: TransactionBlockBytes, +) -> Result { + let tx = cluster + .wallet + .sign_transaction(&transaction_bytes.to_data()?); + let (tx_bytes, signatures) = tx.to_tx_bytes_and_signatures(); + + let tx_response: IotaTransactionBlockResponse = http_client + .execute_transaction_block( + tx_bytes, + signatures, + Some( + IotaTransactionBlockResponseOptions::new() + .with_effects() + .with_object_changes() + .with_balance_changes(), + ), + Some(ExecuteTransactionRequestType::WaitForLocalExecution), + ) + .await?; + assert_eq!(tx_response.status_ok(), Some(true)); + + Ok(tx_response) +} diff --git a/crates/iota-json-rpc-types/src/iota_event.rs b/crates/iota-json-rpc-types/src/iota_event.rs index 80ed4023f0d..828ddb8450f 100644 --- a/crates/iota-json-rpc-types/src/iota_event.rs +++ b/crates/iota-json-rpc-types/src/iota_event.rs @@ -44,7 +44,7 @@ pub struct IotaEvent { #[serde_as(as = "DisplayFromStr")] /// Move module where this event was emitted. pub transaction_module: Identifier, - /// Sender's Iota address. + /// Sender's IOTA address. pub sender: IotaAddress, #[schemars(with = "String")] #[serde_as(as = "IotaStructTag")] diff --git a/crates/iota-json-rpc-types/src/iota_move.rs b/crates/iota-json-rpc-types/src/iota_move.rs index 239afb24a09..feb3ce0264d 100644 --- a/crates/iota-json-rpc-types/src/iota_move.rs +++ b/crates/iota-json-rpc-types/src/iota_move.rs @@ -376,7 +376,7 @@ impl From for IotaMoveValue { IotaMoveValue::Vector(values.into_iter().map(|value| value.into()).collect()) } MoveValue::Struct(value) => { - // Best effort Iota core type conversion + // Best effort IOTA core type conversion let MoveStruct { type_, fields } = &value; if let Some(value) = try_convert_type(type_, fields) { return value; diff --git a/crates/iota-json-rpc-types/src/iota_transaction.rs b/crates/iota-json-rpc-types/src/iota_transaction.rs index 23e4cdfc8c0..1c044f64ab2 100644 --- a/crates/iota-json-rpc-types/src/iota_transaction.rs +++ b/crates/iota-json-rpc-types/src/iota_transaction.rs @@ -1249,7 +1249,7 @@ impl DevInspectResults { #[derive(Eq, PartialEq, Clone, Debug, Serialize, Deserialize, JsonSchema)] pub enum IotaTransactionBlockBuilderMode { - /// Regular Iota Transactions that are committed on chain + /// Regular IOTA Transactions that are committed on chain Commit, /// Simulated transaction that allows calling any Move function with /// arbitrary values. diff --git a/crates/iota-json-rpc/src/lib.rs b/crates/iota-json-rpc/src/lib.rs index 4332692fac1..75280868d70 100644 --- a/crates/iota-json-rpc/src/lib.rs +++ b/crates/iota-json-rpc/src/lib.rs @@ -70,8 +70,8 @@ pub struct JsonRpcServerBuilder { pub fn iota_rpc_doc(version: &str) -> Project { Project::new( version, - "Iota JSON-RPC", - "Iota JSON-RPC API for interaction with Iota Full node. Make RPC calls using https://fullnode.NETWORK.iota.io:443, where NETWORK is the network you want to use (testnet, devnet, mainnet). By default, local networks use port 9000.", + "IOTA JSON-RPC", + "IOTA JSON-RPC API for interaction with IOTA full node or indexer. Make RPC calls using https://api.NETWORK.iota.cafe:443 (or https://indexer.NETWORK.iota.cafe:443 for the indexer), where NETWORK is the network you want to use (testnet, devnet, mainnet). By default, local networks use port 9000 (or 9124 for the indexer).", "IOTA Foundation", "https://iota.org", "info@iota.org", @@ -257,7 +257,7 @@ impl JsonRpcServerBuilder { let handle = ServerHandle { handle: ServerHandleInner::Axum(handle), }; - info!(local_addr =? addr, "Iota JSON-RPC server listening on {addr}"); + info!(local_addr =? addr, "IOTA JSON-RPC server listening on {addr}"); Ok(handle) } } diff --git a/crates/iota-keys/src/keypair_file.rs b/crates/iota-keys/src/keypair_file.rs index 1a2c9dffaf4..21bf12f6f7f 100644 --- a/crates/iota-keys/src/keypair_file.rs +++ b/crates/iota-keys/src/keypair_file.rs @@ -88,7 +88,7 @@ pub fn read_key(path: &PathBuf, require_secp256k1: bool) -> Result validate_alias(x)?, - None => random_name( - &self - .alias_names() - .into_iter() - .map(|x| x.to_string()) - .collect::>(), - ), - }; + + let new_alias_name = self.create_alias(new_alias.map(str::to_string))?; + for a in self.aliases_mut() { if a.alias == old_alias { let pk = &a.public_key_base64; diff --git a/crates/iota-keys/src/random_names.rs b/crates/iota-keys/src/random_names.rs index 8c943bccb53..5b18e2dde57 100644 --- a/crates/iota-keys/src/random_names.rs +++ b/crates/iota-keys/src/random_names.rs @@ -2,15 +2,14 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 +//! This library provides two functions to generate a random combination of an +//! adjective and a precious stone name as a well formatted string, or a list of +//! these strings. + use std::collections::HashSet; use rand::{Rng, rngs::ThreadRng, thread_rng}; -/// This library provides two functions to generate -/// a random combination of an adjective -/// and a precious stone name as a well formatted -/// string, or a list of these strings. - /// A list of adjectives const LEFT_NAMES: [&str; 108] = [ "admiring", diff --git a/crates/iota-keys/tests/tests.rs b/crates/iota-keys/tests/tests.rs index 08da6e43a76..04c65bc8c9e 100644 --- a/crates/iota-keys/tests/tests.rs +++ b/crates/iota-keys/tests/tests.rs @@ -163,6 +163,21 @@ fn update_alias_test() { let update = keystore.update_alias("o", None).unwrap(); let aliases = keystore.alias_names(); assert_eq!(vec![&update], aliases); + + // check that updating alias does not allow duplicates + keystore + .generate_and_add_new_key( + SignatureScheme::ED25519, + Some("my_alias_test".to_string()), + None, + None, + ) + .unwrap(); + assert!( + keystore + .update_alias("my_alias_test", Some(&update)) + .is_err() + ); } #[test] @@ -211,7 +226,7 @@ fn mnemonic_test() { } /// This test confirms rust's implementation of mnemonic is the same with the -/// Iota Wallet +/// IOTA Wallet #[test] fn iota_wallet_address_mnemonic_test() -> Result<(), anyhow::Error> { let phrase = "result crisp session latin must fruit genuine question prevent start coconut brave speak student dismiss"; diff --git a/crates/iota-light-client/README.md b/crates/iota-light-client/README.md index 917a59a5213..381e459ac4b 100644 --- a/crates/iota-light-client/README.md +++ b/crates/iota-light-client/README.md @@ -1,4 +1,4 @@ -This crate contains a Command Line Interface light client for Iota. +This crate contains a Command Line Interface light client for IOTA. # What is a light client? @@ -6,7 +6,7 @@ A light client allows checking the authenticity and validity of on-chain state, Running a _full node_ requires downloading the full sequence of all transaction and re-executing them. Then the full state of the blockchain is available locally to serve reads. This is however an expensive process in terms of network bandwidth needed to download the full sequence of transactions, as well as CPU to re-execute it, and storage to store the full state of the blockchain. -Alternatively, a _light client_ only needs to download minimal information to authenticate blockchain state. Specifically in Iota, the light client needs to _sync_ all end-of-epoch checkpoints that contain information about the committee in the next epoch. Sync involves downloading the checkpoints and checking their validity by checking their certificate. +Alternatively, a _light client_ only needs to download minimal information to authenticate blockchain state. Specifically in IOTA, the light client needs to _sync_ all end-of-epoch checkpoints that contain information about the committee in the next epoch. Sync involves downloading the checkpoints and checking their validity by checking their certificate. Once all end-of-epoch checkpoints are downloaded and checked, any event or current object can be checked for its validity. To do that the light client downloads the checkpoint in which the transaction was executed, and the effects structure that summarizes its effects on the system, including events emitted and objects created. The chain of validity from the checkpoint to the effects and its contents is checked via the certificate on the checkpoint and the hashes of all structures. @@ -20,15 +20,15 @@ The light client requires a config file and a directory to cache checkpoints, an ## Setup -The config file for the light client takes a URL for a full node, a directory (that must exist) and within the directory to name of the genesis blob for the Iota network. +The config file for the light client takes a URL for a full node, a directory (that must exist) and within the directory to name of the genesis blob for the IOTA network. ``` -full_node_url: "http://ord-mnt-rpcbig-06.mainnet.iota.io:9000" +full_node_url: "https://api.testnet.iota.cafe:443" checkpoint_summary_dir: "checkpoints_dir" genesis_filename: "genesis.blob" ``` -The genesis blob for the Iota mainnet can be found here: https://github.com/iotaledger/iota-genesis/blob/main/mainnet/genesis.blob +The genesis blob for the IOTA mainnet can be found here: https://github.com/iotaledger/iota-genesis/blob/main/mainnet/genesis.blob ## Sync diff --git a/crates/iota-light-client/src/bin/light_client.rs b/crates/iota-light-client/src/bin/light_client.rs index 1db8daab15f..e99ef02494c 100644 --- a/crates/iota-light-client/src/bin/light_client.rs +++ b/crates/iota-light-client/src/bin/light_client.rs @@ -13,7 +13,7 @@ use iota_light_client::utils::{ use iota_package_resolver::Resolver; use iota_types::{base_types::ObjectID, digests::TransactionDigest, object::Data}; -/// A light client for the Iota blockchain +/// A light client for the IOTA blockchain #[derive(Parser, Debug)] #[command(author, version, about, long_about = None)] struct Args { diff --git a/crates/iota-light-client/src/proof.rs b/crates/iota-light-client/src/proof.rs index 36d2ee30a6f..96058205b7f 100644 --- a/crates/iota-light-client/src/proof.rs +++ b/crates/iota-light-client/src/proof.rs @@ -13,7 +13,7 @@ use iota_types::{ transaction::Transaction, }; -/// Define aspect of Iota state that needs to be certified in a proof +/// Define aspect of IOTA state that needs to be certified in a proof #[derive(Default)] pub struct ProofTarget { /// Objects that need to be certified. diff --git a/crates/iota-metric-checker/README.md b/crates/iota-metric-checker/README.md index 599b271d082..cc7d0e90947 100644 --- a/crates/iota-metric-checker/README.md +++ b/crates/iota-metric-checker/README.md @@ -5,7 +5,7 @@ The `iota-metric-checker` crate is used for querying prometheus metrics and vali ### Example Usage ``` -RUST_LOG=debug cargo run --package iota-metric-checker --bin iota-metric-checker -- --api-key xxxxxxxx --api-user xxxx_metrics --config checks.yaml --url https://xxxx.iota.io/prometheus +RUST_LOG=debug cargo run --package iota-metric-checker --bin iota-metric-checker -- --api-key xxxxxxxx --api-user xxxx_metrics --config checks.yaml --url https://xxxx.iota.cafe/prometheus ``` ### Example Config diff --git a/crates/iota-metrics/src/metered_channel.rs b/crates/iota-metrics/src/metered_channel.rs index 14c86ec2313..ae7cc11c25d 100644 --- a/crates/iota-metrics/src/metered_channel.rs +++ b/crates/iota-metrics/src/metered_channel.rs @@ -291,7 +291,7 @@ impl Sender { } //////////////////////////////// -/// Stream API Wrappers! +// Stream API Wrappers! //////////////////////////////// /// A wrapper around [`crate::metered_channel::Receiver`] that implements @@ -356,7 +356,7 @@ impl From> for ReceiverStream { // cases. //////////////////////////////////////////////////////////////// -/// Constructor +// Constructor //////////////////////////////////////////////////////////////// /// Similar to `mpsc::channel`, `channel` creates a pair of `Sender` and diff --git a/crates/iota-move-build/Cargo.toml b/crates/iota-move-build/Cargo.toml index 5ddcd6b91f3..5d4a7e73622 100644 --- a/crates/iota-move-build/Cargo.toml +++ b/crates/iota-move-build/Cargo.toml @@ -5,7 +5,7 @@ authors = ["IOTA Foundation "] edition = "2021" license = "Apache-2.0" publish = false -description = "Logic for building Iota Move Packages" +description = "Logic for building IOTA Move Packages" [dependencies] # external dependencies diff --git a/crates/iota-move-build/src/lib.rs b/crates/iota-move-build/src/lib.rs index 309885fcbf1..d3215b3b1ef 100644 --- a/crates/iota-move-build/src/lib.rs +++ b/crates/iota-move-build/src/lib.rs @@ -54,7 +54,7 @@ use serde_reflection::Registry; #[path = "unit_tests/build_tests.rs"] mod build_tests; -/// Wrapper around the core Move `CompiledPackage` with some Iota-specific +/// Wrapper around the core Move `CompiledPackage` with some IOTA-specific /// traits and info #[derive(Debug, Clone)] pub struct CompiledPackage { @@ -65,7 +65,7 @@ pub struct CompiledPackage { pub dependency_ids: PackageDependencies, } -/// Wrapper around the core Move `BuildConfig` with some Iota-specific info +/// Wrapper around the core Move `BuildConfig` with some IOTA-specific info #[derive(Clone)] pub struct BuildConfig { pub config: MoveBuildConfig, @@ -156,7 +156,7 @@ impl BuildConfig { } /// Given a `path` and a `build_config`, build the package in that path, - /// including its dependencies. If we are building the Iota framework, + /// including its dependencies. If we are building the IOTA framework, /// we skip the check that the addresses should be 0 pub fn build(self, path: &Path) -> IotaResult { let print_diags_to_stderr = self.print_diags_to_stderr; @@ -337,7 +337,7 @@ impl CompiledPackage { } else { // Collect all module IDs from the current package to be published (module names // are not sufficient as we may have modules with the same names in - // user code and in Iota framework which would result in the latter + // user code and in IOTA framework which would result in the latter // being pulled into a set of modules to be published). let self_modules: HashSet<_> = self .package @@ -396,13 +396,13 @@ impl CompiledPackage { .filter(|m| *m.self_id().address() == BRIDGE_ADDRESS) } - /// Get bytecode modules from the Iota System that are used by this package + /// Get bytecode modules from the IOTA System that are used by this package pub fn get_iota_system_modules(&self) -> impl Iterator { self.get_modules_and_deps() .filter(|m| *m.self_id().address() == IOTA_SYSTEM_ADDRESS) } - /// Get bytecode modules from the Iota Framework that are used by this + /// Get bytecode modules from the IOTA Framework that are used by this /// package pub fn get_iota_framework_modules(&self) -> impl Iterator { self.get_modules_and_deps() diff --git a/crates/iota-move-build/src/unit_tests/build_tests.rs b/crates/iota-move-build/src/unit_tests/build_tests.rs index 992a1095bee..e32e4bf0fef 100644 --- a/crates/iota-move-build/src/unit_tests/build_tests.rs +++ b/crates/iota-move-build/src/unit_tests/build_tests.rs @@ -8,7 +8,7 @@ use crate::BuildConfig; #[test] fn generate_struct_layouts() { - // build the Iota framework and generate struct layouts to make sure nothing + // build the IOTA framework and generate struct layouts to make sure nothing // crashes let path = Path::new(env!("CARGO_MANIFEST_DIR")) .parent() diff --git a/crates/iota-move-lsp/src/bin/move-analyzer.rs b/crates/iota-move-lsp/src/bin/move-analyzer.rs index 1faf3513773..4865c4fe484 100644 --- a/crates/iota-move-lsp/src/bin/move-analyzer.rs +++ b/crates/iota-move-lsp/src/bin/move-analyzer.rs @@ -9,7 +9,7 @@ use move_analyzer::analyzer; bin_version::bin_version!(); #[derive(Parser)] -#[clap( +#[command( name = env!("CARGO_BIN_NAME"), rename_all = "kebab-case", author, diff --git a/crates/iota-move/src/build.rs b/crates/iota-move/src/build.rs index 0eb2829a2ef..2bd347d8cd6 100644 --- a/crates/iota-move/src/build.rs +++ b/crates/iota-move/src/build.rs @@ -21,22 +21,27 @@ const STRUCT_LAYOUTS_FILENAME: &str = "struct_layouts.yaml"; pub struct Build { /// Include the contents of packages in dependencies that haven't been /// published (only relevant when dumping bytecode as base64) - #[clap(long, global = true)] + #[arg(long, global = true)] pub with_unpublished_dependencies: bool, /// Whether we are printing in base64. - #[clap(long, global = true)] + #[arg(long, global = true)] pub dump_bytecode_as_base64: bool, - /// If true, generate struct layout schemas for - /// all struct types passed into `entry` functions declared by modules in - /// this package These layout schemas can be consumed by clients (e.g., - /// the TypeScript SDK) to enable serialization/deserialization of - /// transaction arguments and events. - #[clap(long, global = true)] + /// Don't specialize the package to the active chain when dumping bytecode + /// as Base64. This allows building to proceed without a network connection + /// or active environment, but it will not be able to automatically + /// determine the addresses of its dependencies. + #[arg(long, global = true, requires = "dump_bytecode_as_base64")] + pub ignore_chain: bool, + /// If true, generate struct layout schemas for all struct types passed into + /// `entry` functions declared by modules in this package These layout + /// schemas can be consumed by clients (e.g., the TypeScript SDK) to enable + /// serialization/deserialization of transaction arguments and events. + #[arg(long, global = true)] pub generate_struct_layouts: bool, /// The chain ID, if resolved. Required when the dump_bytecode_as_base64 is /// true, for automated address management, where package addresses are /// resolved for the respective chain in the Move.lock file. - #[clap(skip)] + #[arg(skip)] pub chain_id: Option, } diff --git a/crates/iota-move/src/coverage.rs b/crates/iota-move/src/coverage.rs index 836d28d36df..84e3bf47c2e 100644 --- a/crates/iota-move/src/coverage.rs +++ b/crates/iota-move/src/coverage.rs @@ -11,7 +11,7 @@ use move_package::BuildConfig; #[derive(Parser)] #[group(id = "iota-move-coverage")] pub struct Coverage { - #[clap(flatten)] + #[command(flatten)] pub coverage: coverage::Coverage, } diff --git a/crates/iota-move/src/disassemble.rs b/crates/iota-move/src/disassemble.rs index 42671f872b0..4e29995b07f 100644 --- a/crates/iota-move/src/disassemble.rs +++ b/crates/iota-move/src/disassemble.rs @@ -20,14 +20,14 @@ use move_package::BuildConfig; #[group(id = "iota-move-disassemmble")] pub struct Disassemble { /// Path to a .mv file to disassemble - #[clap(name = "module_path")] + #[arg(name = "module_path")] module_path: PathBuf, /// Whether to display the disassembly in raw Debug format - #[clap(long = "Xdebug")] + #[arg(long = "Xdebug")] debug: bool, - #[clap(short = 'i', long = "interactive")] + #[arg(short = 'i', long = "interactive")] interactive: bool, } diff --git a/crates/iota-move/src/iota-natives.bpl b/crates/iota-move/src/iota-natives.bpl index f9825430840..845c461720a 100644 --- a/crates/iota-move/src/iota-natives.bpl +++ b/crates/iota-move/src/iota-natives.bpl @@ -53,7 +53,7 @@ procedure {:inline 1} $2_tx_context_derive_id(tx_hash: Vec (int), ids_created: i {%- set T = instance.name -%} // ---------------------------------------------------------------------------------- -// Native Iota event implementation for object type `{{instance.suffix}}` +// Native IOTA event implementation for object type `{{instance.suffix}}` procedure {:inline 1} $2_event_emit{{S}}(event: {{T}}); @@ -69,7 +69,7 @@ procedure {:inline 1} $2_event_emit{{S}}(event: {{T}}); {%- set T = instance.name -%} // ---------------------------------------------------------------------------------- -// Native Iota types implementation for object type `{{instance.suffix}}` +// Native IOTA types implementation for object type `{{instance.suffix}}` procedure {:inline 1} $2_types_is_one_time_witness{{S}}(_: {{T}}) returns (res: bool); @@ -112,7 +112,7 @@ procedure {:inline 1} $2_dynamic_field_has_child_object_with_ty{{S}}(parent: int {%- set T = instance.name -%} // ---------------------------------------------------------------------------------- -// Native Iota prover implementation for object type `{{instance.suffix}}` +// Native IOTA prover implementation for object type `{{instance.suffix}}` function $2_prover_vec_remove{{S}}(v: Vec ({{T}}), elem_idx: int): Vec ({{T}}) { RemoveAtVec(v, elem_idx) diff --git a/crates/iota-move/src/lib.rs b/crates/iota-move/src/lib.rs index 6e3f4832bd7..04c94cb7800 100644 --- a/crates/iota-move/src/lib.rs +++ b/crates/iota-move/src/lib.rs @@ -38,9 +38,9 @@ pub enum Command { } #[derive(Parser)] pub struct Calib { - #[clap(name = "runs", short = 'r', long = "runs", default_value = "1")] + #[arg(name = "runs", short = 'r', long = "runs", default_value = "1")] runs: usize, - #[clap(name = "summarize", short = 's', long = "summarize")] + #[arg(name = "summarize", short = 's', long = "summarize")] summarize: bool, } diff --git a/crates/iota-move/src/main.rs b/crates/iota-move/src/main.rs index d18514582a3..b4129d7a6f9 100644 --- a/crates/iota-move/src/main.rs +++ b/crates/iota-move/src/main.rs @@ -15,29 +15,29 @@ use tracing::debug; bin_version::bin_version!(); #[derive(Parser)] -#[clap( +#[command( name = env!("CARGO_BIN_NAME"), - about = "Iota-Move CLI", + about = "IOTA Move CLI", rename_all = "kebab-case", author, version = VERSION, )] struct Args { /// Path to a package which the command should be run with respect to. - #[clap(long = "path", short = 'p', global = true)] + #[arg(long = "path", short = 'p', global = true)] pub package_path: Option, /// If true, run the Move bytecode verifier on the bytecode from a /// successful build - #[clap(long = "path", short = 'p', global = true)] + #[arg(long = "path", short = 'p', global = true)] pub run_bytecode_verifier: bool, /// If true, print build diagnostics to stderr--no printing if false - #[clap(long = "path", short = 'p', global = true)] + #[arg(long = "path", short = 'p', global = true)] pub print_diags_to_stderr: bool, /// Package build options - #[clap(flatten)] + #[command(flatten)] pub build_config: MoveBuildConfig, /// Subcommands. - #[clap(subcommand)] + #[command(subcommand)] pub cmd: iota_move::Command, } @@ -64,7 +64,7 @@ async fn main() { .with_log_file(&format!("{bin_name}.log")) .with_env() .init(); - debug!("Iota-Move CLI version: {VERSION}"); + debug!("IOTA Move CLI version: {VERSION}"); exit_main!(execute_move_command( args.package_path.as_deref(), diff --git a/crates/iota-move/src/manage_package.rs b/crates/iota-move/src/manage_package.rs index 4f2f5d64e2b..e377c9c5bd0 100644 --- a/crates/iota-move/src/manage_package.rs +++ b/crates/iota-move/src/manage_package.rs @@ -23,23 +23,23 @@ const NO_LOCK_FILE: &str = "Expected a `Move.lock` file to exist in the package #[derive(Parser)] #[group(id = "iota-move-manage-package")] pub struct ManagePackage { - #[clap(long)] + #[arg(long)] /// The environment to associate this package information with (consider /// using `iota client active-env`). pub environment: String, - #[clap(long = "network-id")] + #[arg(long = "network-id")] /// The network chain identifer. Use '35834a8a' for mainnet. pub chain_id: String, - #[clap(long = "original-id", value_parser = ObjectID::from_hex_literal)] + #[arg(long = "original-id", value_parser = ObjectID::from_hex_literal)] /// The original address (Object ID) where this package is published. pub original_id: ObjectID, - #[clap(long = "latest-id", value_parser = ObjectID::from_hex_literal)] + #[arg(long = "latest-id", value_parser = ObjectID::from_hex_literal)] /// The most recent address (Object ID) where this package is published. It /// is the same as 'original-id' if the package is immutable and /// published once. It is different from 'original-id' if the package has /// been upgraded to a different address. pub latest_id: ObjectID, - #[clap(long = "version-number")] + #[arg(long = "version-number")] /// The version number of the published package. It is '1' if the package is /// immutable and published once. It is some number greater than '1' if /// the package has been upgraded once or more. diff --git a/crates/iota-move/src/migrate.rs b/crates/iota-move/src/migrate.rs index 315b08053b3..5c72c5def7b 100644 --- a/crates/iota-move/src/migrate.rs +++ b/crates/iota-move/src/migrate.rs @@ -11,7 +11,7 @@ use move_package::BuildConfig as MoveBuildConfig; #[derive(Parser)] #[group(id = "iota-move-migrate")] pub struct Migrate { - #[clap(flatten)] + #[command(flatten)] pub migrate: migrate::Migrate, } diff --git a/crates/iota-move/src/new.rs b/crates/iota-move/src/new.rs index cc4ae3a721d..c7c4f7d107e 100644 --- a/crates/iota-move/src/new.rs +++ b/crates/iota-move/src/new.rs @@ -17,7 +17,7 @@ const IOTA_PKG_PATH: &str = "{ git = \"https://github.com/iotaledger/iota.git\", #[derive(Parser)] #[group(id = "iota-move-new")] pub struct New { - #[clap(flatten)] + #[command(flatten)] pub new: new::New, } diff --git a/crates/iota-move/src/unit_test.rs b/crates/iota-move/src/unit_test.rs index f85d7689d7d..2bfb09a79e4 100644 --- a/crates/iota-move/src/unit_test.rs +++ b/crates/iota-move/src/unit_test.rs @@ -34,7 +34,7 @@ const MAX_UNIT_TEST_INSTRUCTIONS: u64 = 1_000_000; #[derive(Parser)] #[group(id = "iota-move-test")] pub struct Test { - #[clap(flatten)] + #[command(flatten)] pub test: test::Test, } @@ -47,7 +47,7 @@ impl Test { let compute_coverage = self.test.compute_coverage; if !cfg!(debug_assertions) && compute_coverage { return Err(anyhow::anyhow!( - "The --coverage flag is currently supported only in debug builds. Please build the Iota CLI from source in debug mode." + "The --coverage flag is currently supported only in debug builds. Please build the IOTA CLI from source in debug mode." )); } // find manifest file directory from a given path or (if missing) from current diff --git a/crates/iota-network/build.rs b/crates/iota-network/build.rs index ee7d0a1a6f5..35dcae113ee 100644 --- a/crates/iota-network/build.rs +++ b/crates/iota-network/build.rs @@ -12,6 +12,9 @@ use tonic_build::manual::{Builder, Method, Service}; type Result = ::std::result::Result>; fn main() -> Result<()> { + println!("cargo::rustc-check-cfg=cfg(msim)"); + println!("cargo::rustc-check-cfg=cfg(fail_points)"); + let out_dir = if env::var("DUMP_GENERATED_GRPC").is_ok() { PathBuf::from("") } else { diff --git a/crates/iota-network/src/randomness/mod.rs b/crates/iota-network/src/randomness/mod.rs index 3bed5180149..3ada502b18e 100644 --- a/crates/iota-network/src/randomness/mod.rs +++ b/crates/iota-network/src/randomness/mod.rs @@ -952,7 +952,7 @@ impl RandomnessEventLoop { full_sig: Arc>, ) { // For simtests, we may test not sending partial signatures. - #[expect(unused_mut)] + #[cfg_attr(not(any(msim, fail_points)), expect(unused_mut))] let mut fail_point_skip_sending = false; fail_point_if!("rb-send-partial-signatures", || { fail_point_skip_sending = true; diff --git a/crates/iota-node/src/lib.rs b/crates/iota-node/src/lib.rs index 58443bb59e6..9469e669aa1 100644 --- a/crates/iota-node/src/lib.rs +++ b/crates/iota-node/src/lib.rs @@ -1575,7 +1575,7 @@ impl IotaNode { .state .get_object_cache_reader() .get_iota_system_state_object_unsafe() - .expect("Read Iota System State object cannot fail"); + .expect("Read IOTA System State object cannot fail"); #[cfg(msim)] if !self @@ -1896,7 +1896,6 @@ impl IotaNode { .store(new_value, Ordering::Relaxed); } - #[expect(unused_variables)] async fn fetch_jwks( authority: AuthorityName, provider: &OIDCProvider, @@ -2083,7 +2082,7 @@ pub async fn build_http_server( .unwrap() }); - info!(local_addr =? addr, "Iota JSON-RPC server listening on {addr}"); + info!(local_addr =? addr, "IOTA JSON-RPC server listening on {addr}"); Ok(Some(handle)) } diff --git a/crates/iota-node/src/main.rs b/crates/iota-node/src/main.rs index 09b001d78bb..eb63a09ba27 100644 --- a/crates/iota-node/src/main.rs +++ b/crates/iota-node/src/main.rs @@ -8,7 +8,7 @@ use clap::{ArgGroup, Parser}; use iota_common::sync::async_once_cell::AsyncOnceCell; use iota_config::{Config, NodeConfig, node::RunWithRange}; use iota_core::runtime::IotaRuntimes; -use iota_node::IotaNode; +use iota_node::{IotaNode, metrics}; use iota_types::{ committee::EpochId, messages_checkpoint::CheckpointSequenceNumber, multiaddr::Multiaddr, supported_protocol_versions::SupportedProtocolVersions, @@ -20,21 +20,23 @@ use tracing::{error, info}; bin_version::bin_version!(); #[derive(Parser)] -#[clap(rename_all = "kebab-case")] -#[clap(name = env!("CARGO_BIN_NAME"))] -#[clap(version = VERSION)] -#[clap(group(ArgGroup::new("exclusive").required(false)))] +#[command( + rename_all = "kebab-case", + version = VERSION, + group(ArgGroup::new("exclusive").required(false)), + name = env!("CARGO_BIN_NAME")) +] struct Args { - #[clap(long)] + #[arg(long)] pub config_path: PathBuf, - #[clap(long, help = "Specify address to listen on")] + #[arg(long, help = "Specify address to listen on")] listen_address: Option, - #[clap(long, group = "exclusive")] + #[arg(long, group = "exclusive")] run_with_range_epoch: Option, - #[clap(long, group = "exclusive")] + #[arg(long, group = "exclusive")] run_with_range_checkpoint: Option, } @@ -81,7 +83,7 @@ fn main() { drop(metrics_rt); - info!("Iota Node version: {VERSION}"); + info!("IOTA Node version: {VERSION}"); info!( "Supported protocol versions: {:?}", config.supported_protocol_versions @@ -92,6 +94,11 @@ fn main() { config.metrics_address ); + { + let _enter = runtimes.metrics.enter(); + metrics::start_metrics_push_task(&config, registry_service.clone()); + } + if let Some(listen_address) = args.listen_address { config.network_address = listen_address; } @@ -144,7 +151,7 @@ fn main() { None => "unknown".to_string(), }; - info!("Iota chain identifier: {chain_identifier}"); + info!("IOTA chain identifier: {chain_identifier}"); prometheus_registry .register(iota_metrics::uptime_metric( if is_validator { diff --git a/crates/iota-node/src/metrics.rs b/crates/iota-node/src/metrics.rs index 2d83d7bb4e6..a6670b91a58 100644 --- a/crates/iota-node/src/metrics.rs +++ b/crates/iota-node/src/metrics.rs @@ -2,14 +2,164 @@ // Modifications Copyright (c) 2024 IOTA Stiftung // SPDX-License-Identifier: Apache-2.0 -use std::time::Duration; +use std::time::{Duration, SystemTime, UNIX_EPOCH}; +use axum::http::header; +use iota_metrics::RegistryService; use iota_network::tonic::Code; use iota_network_stack::metrics::MetricsCallbackProvider; use prometheus::{ - HistogramVec, IntCounterVec, IntGaugeVec, Registry, register_histogram_vec_with_registry, - register_int_counter_vec_with_registry, register_int_gauge_vec_with_registry, + Encoder, HistogramVec, IntCounterVec, IntGaugeVec, PROTOBUF_FORMAT, Registry, + register_histogram_vec_with_registry, register_int_counter_vec_with_registry, + register_int_gauge_vec_with_registry, }; +use tracing::error; + +const METRICS_PUSH_TIMEOUT: Duration = Duration::from_secs(45); + +pub struct MetricsPushClient { + certificate: std::sync::Arc, + client: reqwest::Client, +} + +impl MetricsPushClient { + pub fn new(network_key: iota_types::crypto::NetworkKeyPair) -> Self { + use fastcrypto::traits::KeyPair; + let certificate = std::sync::Arc::new(iota_tls::SelfSignedCertificate::new( + network_key.private(), + iota_tls::IOTA_VALIDATOR_SERVER_NAME, + )); + let identity = certificate.reqwest_identity(); + let client = reqwest::Client::builder() + .identity(identity) + .build() + .unwrap(); + + Self { + certificate, + client, + } + } + + pub fn certificate(&self) -> &iota_tls::SelfSignedCertificate { + &self.certificate + } + + pub fn client(&self) -> &reqwest::Client { + &self.client + } +} + +/// Starts a task to periodically push metrics to a configured endpoint if a +/// metrics push endpoint is configured. +pub fn start_metrics_push_task(config: &iota_config::NodeConfig, registry: RegistryService) { + use fastcrypto::traits::KeyPair; + use iota_config::node::MetricsConfig; + + const DEFAULT_METRICS_PUSH_INTERVAL: Duration = Duration::from_secs(60); + + let (interval, url) = match &config.metrics { + Some(MetricsConfig { + push_interval_seconds, + push_url: Some(url), + }) => { + let interval = push_interval_seconds + .map(Duration::from_secs) + .unwrap_or(DEFAULT_METRICS_PUSH_INTERVAL); + let url = reqwest::Url::parse(url).expect("unable to parse metrics push url"); + (interval, url) + } + _ => return, + }; + + // make a copy so we can make a new client later when we hit errors posting + // metrics + let config_copy = config.clone(); + let mut client = MetricsPushClient::new(config_copy.network_key_pair().copy()); + + async fn push_metrics( + client: &MetricsPushClient, + url: &reqwest::Url, + registry: &RegistryService, + ) -> Result<(), anyhow::Error> { + // now represents a collection timestamp for all of the metrics we send to the + // proxy + let now = SystemTime::now() + .duration_since(UNIX_EPOCH) + .unwrap() + .as_millis() as i64; + + let mut metric_families = registry.gather_all(); + for mf in metric_families.iter_mut() { + for m in mf.mut_metric() { + m.set_timestamp_ms(now); + } + } + + let mut buf: Vec = vec![]; + let encoder = prometheus::ProtobufEncoder::new(); + encoder.encode(&metric_families, &mut buf)?; + + let mut s = snap::raw::Encoder::new(); + let compressed = s.compress_vec(&buf).map_err(|err| { + error!("unable to snappy encode; {err}"); + err + })?; + + let response = client + .client() + .post(url.to_owned()) + .header(reqwest::header::CONTENT_ENCODING, "snappy") + .header(header::CONTENT_TYPE, PROTOBUF_FORMAT) + .body(compressed) + .timeout(METRICS_PUSH_TIMEOUT) + .send() + .await?; + + if !response.status().is_success() { + let status = response.status(); + let body = match response.text().await { + Ok(body) => body, + Err(error) => format!("couldn't decode response body; {error}"), + }; + return Err(anyhow::anyhow!( + "metrics push failed: [{}]:{}", + status, + body + )); + } + + tracing::debug!("successfully pushed metrics to {url}"); + + Ok(()) + } + + tokio::spawn(async move { + tracing::info!(push_url =% url, interval =? interval, "Started Metrics Push Service"); + + let mut interval = tokio::time::interval(interval); + interval.set_missed_tick_behavior(tokio::time::MissedTickBehavior::Skip); + + let mut errors = 0; + loop { + interval.tick().await; + + if let Err(error) = push_metrics(&client, &url, ®istry).await { + errors += 1; + if errors >= 10 { + // If we hit 10 failures in a row, start logging errors. + tracing::error!("unable to push metrics: {error}; new client will be created"); + } else { + tracing::warn!("unable to push metrics: {error}; new client will be created"); + } + // aggressively recreate our client connection if we hit an error + client = MetricsPushClient::new(config_copy.network_key_pair().copy()); + } else { + errors = 0; + } + } + }); +} pub struct IotaNodeMetrics { pub jwk_requests: IntCounterVec, diff --git a/crates/iota-open-rpc-macros/src/lib.rs b/crates/iota-open-rpc-macros/src/lib.rs index cddbf557130..78e15b0eadc 100644 --- a/crates/iota-open-rpc-macros/src/lib.rs +++ b/crates/iota-open-rpc-macros/src/lib.rs @@ -261,7 +261,7 @@ fn parse_rpc_method(trait_data: &mut syn::ItemTrait) -> Result TokenStream2 { let attrs = attributes .attrs diff --git a/crates/iota-open-rpc/spec/openrpc.json b/crates/iota-open-rpc/spec/openrpc.json index 95f08307724..a4fd912ea8f 100644 --- a/crates/iota-open-rpc/spec/openrpc.json +++ b/crates/iota-open-rpc/spec/openrpc.json @@ -1,8 +1,8 @@ { "openrpc": "1.2.6", "info": { - "title": "Iota JSON-RPC", - "description": "Iota JSON-RPC API for interaction with Iota Full node. Make RPC calls using https://fullnode.NETWORK.iota.io:443, where NETWORK is the network you want to use (testnet, devnet, mainnet). By default, local networks use port 9000.", + "title": "IOTA JSON-RPC", + "description": "IOTA JSON-RPC API for interaction with IOTA full node or indexer. Make RPC calls using https://api.NETWORK.iota.cafe:443 (or https://indexer.NETWORK.iota.cafe:443 for the indexer), where NETWORK is the network you want to use (testnet, devnet, mainnet). By default, local networks use port 9000 (or 9124 for the indexer).", "contact": { "name": "IOTA Foundation", "url": "https://iota.org", @@ -12,7 +12,7 @@ "name": "Apache-2.0", "url": "https://raw.githubusercontent.com/iotaledger/iota/main/LICENSE" }, - "version": "0.8.0-alpha" + "version": "0.9.0-alpha" }, "methods": [ { @@ -1300,7 +1300,7 @@ "name": "Result", "value": { "minSupportedProtocolVersion": "1", - "maxSupportedProtocolVersion": "2", + "maxSupportedProtocolVersion": "3", "protocolVersion": "1", "featureFlags": { "accept_zklogin_in_multisig": false, @@ -1313,6 +1313,7 @@ "hardened_otw_check": true, "no_extraneous_module_bytes": true, "passkey_auth": true, + "relocate_event_module": false, "rethrow_serialization_type_layout_errors": true, "zklogin_auth": false }, @@ -3163,7 +3164,7 @@ "params": [ { "name": "owner", - "description": "the owner's Iota address", + "description": "the owner's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -3213,7 +3214,7 @@ "params": [ { "name": "owner", - "description": "the owner's Iota address", + "description": "the owner's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -3303,6 +3304,7 @@ "name": "Extended API" } ], + "description": "Address related metrics. Exclusively served by the indexer.", "params": [ { "name": "descending_order", @@ -3333,7 +3335,7 @@ "params": [ { "name": "owner", - "description": "the owner's Iota address", + "description": "the owner's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -3385,6 +3387,7 @@ "name": "Extended API" } ], + "description": "Address related metrics. Exclusively served by the indexer.", "params": [ { "name": "checkpoint", @@ -3462,7 +3465,7 @@ "params": [ { "name": "owner", - "description": "the owner's Iota address", + "description": "the owner's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -3623,7 +3626,7 @@ "name": "Extended API" } ], - "description": "Return current epoch info", + "description": "Return current epoch info. Exclusively served by the indexer.", "params": [], "result": { "name": "EpochInfo", @@ -3819,7 +3822,7 @@ "name": "Extended API" } ], - "description": "Return a list of epoch metrics, which is a subset of epoch info", + "description": "Return a list of epoch metrics, which is a subset of epoch info. Exclusively served by the indexer.", "params": [ { "name": "cursor", @@ -3860,7 +3863,7 @@ "name": "Extended API" } ], - "description": "Return a list of epoch info", + "description": "Return a list of epoch info. Exclusively served by the indexer.", "params": [ { "name": "cursor", @@ -3901,7 +3904,7 @@ "name": "Extended API" } ], - "description": "Address related metrics", + "description": "Address related metrics. Exclusively served by the indexer.", "params": [], "result": { "name": "AddressMetrics", @@ -3945,7 +3948,7 @@ "name": "Extended API" } ], - "description": "Return move call metrics", + "description": "Return move call metrics. Exclusively served by the indexer.", "params": [], "result": { "name": "MoveCallMetrics", @@ -3962,7 +3965,7 @@ "name": "Extended API" } ], - "description": "Return Network metrics", + "description": "Return Network metrics. Exclusively served by the indexer.", "params": [], "result": { "name": "NetworkMetrics", @@ -3983,7 +3986,7 @@ "params": [ { "name": "address", - "description": "the owner's Iota address", + "description": "the owner's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -4398,6 +4401,7 @@ "name": "Extended API" } ], + "description": "Return the total number of transactions. Exclusively served by the indexer.", "params": [], "result": { "name": "BigInt", @@ -4701,7 +4705,7 @@ "name": "PubSub" } ], - "description": "Subscribe to a stream of Iota event", + "description": "Subscribe to a stream of IOTA event", "params": [ { "name": "filter", @@ -4733,7 +4737,7 @@ "name": "PubSub" } ], - "description": "Subscribe to a stream of Iota transaction effects", + "description": "Subscribe to a stream of IOTA transaction effects", "params": [ { "name": "filter", @@ -4762,7 +4766,7 @@ "params": [ { "name": "signer", - "description": "the transaction signer's Iota address", + "description": "the transaction signer's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -4821,7 +4825,7 @@ "params": [ { "name": "signer", - "description": "the transaction signer's Iota address", + "description": "the transaction signer's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -4878,7 +4882,7 @@ "params": [ { "name": "signer", - "description": "the transaction signer's Iota address", + "description": "the transaction signer's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -4972,7 +4976,7 @@ "params": [ { "name": "signer", - "description": "the transaction signer's Iota address", + "description": "the transaction signer's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -4980,7 +4984,7 @@ }, { "name": "input_coins", - "description": "the Iota coins to be used in this transaction", + "description": "the IOTA coins to be used in this transaction", "required": true, "schema": { "type": "array", @@ -5046,7 +5050,7 @@ "params": [ { "name": "signer", - "description": "the transaction signer's Iota address", + "description": "the transaction signer's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -5054,7 +5058,7 @@ }, { "name": "input_coins", - "description": "the Iota coins to be used in this transaction, including the coin for gas payment.", + "description": "the IOTA coins to be used in this transaction, including the coin for gas payment.", "required": true, "schema": { "type": "array", @@ -5099,7 +5103,7 @@ "params": [ { "name": "signer", - "description": "the transaction signer's Iota address", + "description": "the transaction signer's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -5107,7 +5111,7 @@ }, { "name": "input_coins", - "description": "the Iota coins to be used in this transaction, including the coin for gas payment.", + "description": "the IOTA coins to be used in this transaction, including the coin for gas payment.", "required": true, "schema": { "type": "array", @@ -5166,7 +5170,7 @@ "params": [ { "name": "sender", - "description": "the transaction signer's Iota address", + "description": "the transaction signer's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -5229,7 +5233,7 @@ "params": [ { "name": "signer", - "description": "the transaction signer's Iota address", + "description": "the transaction signer's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -5255,7 +5259,7 @@ }, { "name": "validator", - "description": "the validator's Iota address", + "description": "the validator's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -5296,7 +5300,7 @@ "params": [ { "name": "signer", - "description": "the transaction signer's Iota address", + "description": "the transaction signer's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -5312,7 +5316,7 @@ }, { "name": "validator", - "description": "the validator's Iota address", + "description": "the validator's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -5354,7 +5358,7 @@ "params": [ { "name": "signer", - "description": "the transaction signer's Iota address", + "description": "the transaction signer's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -5403,7 +5407,7 @@ "params": [ { "name": "signer", - "description": "the transaction signer's Iota address", + "description": "the transaction signer's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -5453,7 +5457,7 @@ "params": [ { "name": "signer", - "description": "the transaction signer's Iota address", + "description": "the transaction signer's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -5513,7 +5517,7 @@ "params": [ { "name": "signer", - "description": "the transaction signer's Iota address", + "description": "the transaction signer's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -5566,11 +5570,11 @@ "name": "Transaction Builder API" } ], - "description": "Create an unsigned transaction to send IOTA coin object to a Iota address. The IOTA object is also used as the gas object.", + "description": "Create an unsigned transaction to send IOTA coin object to an IOTA address. The IOTA object is also used as the gas object.", "params": [ { "name": "signer", - "description": "the transaction signer's Iota address", + "description": "the transaction signer's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -5578,7 +5582,7 @@ }, { "name": "iota_object_id", - "description": "the Iota coin object to be used in this transaction", + "description": "the IOTA coin object to be used in this transaction", "required": true, "schema": { "$ref": "#/components/schemas/ObjectID" @@ -5594,7 +5598,7 @@ }, { "name": "recipient", - "description": "the recipient's Iota address", + "description": "the recipient's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -5627,7 +5631,7 @@ "params": [ { "name": "signer", - "description": "the transaction signer's Iota address", + "description": "the transaction signer's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -5658,7 +5662,7 @@ }, { "name": "recipient", - "description": "the recipient's Iota address", + "description": "the recipient's IOTA address", "required": true, "schema": { "$ref": "#/components/schemas/IotaAddress" @@ -5727,7 +5731,7 @@ } }, "AuthorityPublicKeyBytes": { - "description": "Defines the compressed version of the public key that we pass around in Iota", + "description": "Defines the compressed version of the public key that we pass around in IOTA.", "allOf": [ { "$ref": "#/components/schemas/Base64" @@ -6772,7 +6776,7 @@ "description": "Parsed json value of the event" }, "sender": { - "description": "Sender's Iota address.", + "description": "Sender's IOTA address.", "allOf": [ { "$ref": "#/components/schemas/IotaAddress" @@ -7050,7 +7054,7 @@ ] }, "EventID": { - "description": "Unique ID of a Iota Event, the ID is a combination of tx seq number and event seq number, the ID is local to this particular fullnode and will be different from other fullnode.", + "description": "Unique ID of an IOTA Event, the ID is a combination of tx seq number and event seq number, the ID is local to this particular fullnode and will be different from other fullnode.", "type": "object", "required": [ "eventSeq", @@ -8792,7 +8796,7 @@ "IotaTransactionBlockBuilderMode": { "oneOf": [ { - "description": "Regular Iota Transactions that are committed on chain", + "description": "Regular IOTA Transactions that are committed on chain", "type": "string", "enum": [ "Commit" diff --git a/crates/iota-open-rpc/src/generate_json_rpc_spec.rs b/crates/iota-open-rpc/src/generate_json_rpc_spec.rs index 6f15290df3c..c7c7e3b5305 100644 --- a/crates/iota-open-rpc/src/generate_json_rpc_spec.rs +++ b/crates/iota-open-rpc/src/generate_json_rpc_spec.rs @@ -28,12 +28,12 @@ enum Action { } #[derive(Debug, Parser)] -#[clap( - name = "Iota format generator", - about = "Trace serde (de)serialization to generate format descriptions for Iota types" +#[command( + name = "IOTA format generator", + about = "Trace serde (de)serialization to generate format descriptions for IOTA types" )] struct Options { - #[clap(value_enum, default_value = "Record", ignore_case = true)] + #[arg(value_enum, default_value = "Record", ignore_case = true)] action: Action, } diff --git a/crates/iota-open-rpc/src/lib.rs b/crates/iota-open-rpc/src/lib.rs index 829f2c0d19c..e3ced320aa9 100644 --- a/crates/iota-open-rpc/src/lib.rs +++ b/crates/iota-open-rpc/src/lib.rs @@ -17,7 +17,7 @@ use versions::Versioning; /// OPEN-RPC documentation following the OpenRPC specification /// The implementation is partial, only required fields and subset of optional -/// fields in the specification are implemented catered to Iota's need. +/// fields in the specification are implemented catered to IOTA's need. #[derive(Serialize, Deserialize, Clone)] pub struct Project { openrpc: String, diff --git a/crates/iota-package-dump/src/client.rs b/crates/iota-package-dump/src/client.rs index ee96573c704..fdd1afc50e1 100644 --- a/crates/iota-package-dump/src/client.rs +++ b/crates/iota-package-dump/src/client.rs @@ -13,7 +13,8 @@ pub(crate) struct Client { } impl Client { - /// Create a new GraphQL client, talking to a Iota GraphQL service at `url`. + /// Create a new GraphQL client, talking to an IOTA GraphQL service at + /// `url`. pub(crate) fn new(url: impl IntoUrl) -> Result { Ok(Self { inner: reqwest::Client::builder() diff --git a/crates/iota-package-resolver/src/lib.rs b/crates/iota-package-resolver/src/lib.rs index 05ed900b686..e0f7da786c9 100644 --- a/crates/iota-package-resolver/src/lib.rs +++ b/crates/iota-package-resolver/src/lib.rs @@ -2767,9 +2767,9 @@ mod tests { ); } - /// *** Test Helpers - /// ************************************************************************ - /// ** + // *** Test Helpers + // ************************************************************************ + // ** type TypeOriginTable = Vec; diff --git a/crates/iota-protocol-config-macros/src/lib.rs b/crates/iota-protocol-config-macros/src/lib.rs index a479c8df658..04dcbae99dc 100644 --- a/crates/iota-protocol-config-macros/src/lib.rs +++ b/crates/iota-protocol-config-macros/src/lib.rs @@ -63,7 +63,7 @@ pub fn accessors_macro(input: TokenStream) -> TokenStream { .path .segments .last() - .map_or(false, |segment| segment.ident == "Option") => + .is_some_and(|segment| segment.ident == "Option") => { // Extract inner type T from Option let inner_type = if let syn::PathArguments::AngleBracketed( @@ -247,7 +247,7 @@ pub fn feature_flag_getters_macro(input: TokenStream) -> TokenStream { .path .segments .last() - .map_or(false, |segment| segment.ident == "bool") => + .is_some_and(|segment| segment.ident == "bool") => { Some(( quote! { diff --git a/crates/iota-protocol-config/src/lib.rs b/crates/iota-protocol-config/src/lib.rs index 77c62e282f3..b87e01cc899 100644 --- a/crates/iota-protocol-config/src/lib.rs +++ b/crates/iota-protocol-config/src/lib.rs @@ -16,13 +16,17 @@ use tracing::{info, warn}; /// The minimum and maximum protocol versions supported by this build. const MIN_PROTOCOL_VERSION: u64 = 1; -pub const MAX_PROTOCOL_VERSION: u64 = 2; +pub const MAX_PROTOCOL_VERSION: u64 = 3; // Record history of protocol version allocations here: // // Version 1: Original version. // Version 2: Don't redistribute slashed staking rewards, fix computation of // SystemEpochInfoEventV1. +// Version 3: Set the `relocate_event_module` to be true so that the module that +// is associated as the "sending module" for an event is relocated by linkage. +// Add `Clock` based unlock to `Timelock` objects. + #[derive(Copy, Clone, Debug, Hash, Serialize, Deserialize, PartialEq, Eq, PartialOrd, Ord)] pub struct ProtocolVersion(u64); @@ -39,10 +43,10 @@ impl ProtocolVersion { #[cfg(not(msim))] const MAX_ALLOWED: Self = Self::MAX; - // We create 3 additional "fake" versions in simulator builds so that we can + // We create 2 additional "fake" versions in simulator builds so that we can // test upgrades. #[cfg(msim)] - pub const MAX_ALLOWED: Self = Self(MAX_PROTOCOL_VERSION + 3); + pub const MAX_ALLOWED: Self = Self(MAX_PROTOCOL_VERSION + 2); pub fn new(v: u64) -> Self { Self(v) @@ -182,6 +186,10 @@ struct FeatureFlags { // This flag is used to provide the correct MoveVM configuration for clients. #[serde(skip_serializing_if = "is_true")] rethrow_serialization_type_layout_errors: bool, + + // Makes the event's sending module version-aware. + #[serde(skip_serializing_if = "is_false")] + relocate_event_module: bool, } fn is_true(b: &bool) -> bool { @@ -291,6 +299,8 @@ pub struct ProtocolConfig { feature_flags: FeatureFlags, // ==== Transaction input limits ==== + + // /// Maximum serialized size of a transaction (in bytes). max_tx_size_bytes: Option, @@ -299,12 +309,12 @@ pub struct ProtocolConfig { max_input_objects: Option, /// Max size of objects a transaction can write to disk after completion. - /// Enforce by the Iota adapter. This is the sum of the serialized size + /// Enforce by the IOTA adapter. This is the sum of the serialized size /// of all objects written to disk. The max size of individual objects /// on the other hand is `max_move_object_size`. max_size_written_objects: Option, /// Max size of objects a system transaction can write to disk after - /// completion. Enforce by the Iota adapter. Similar to + /// completion. Enforce by the IOTA adapter. Similar to /// `max_size_written_objects` but for system transactions. max_size_written_objects_system_tx: Option, @@ -340,6 +350,8 @@ pub struct ProtocolConfig { max_programmable_tx_commands: Option, // ==== Move VM, Move bytecode verifier, and execution limits === + + // /// Maximum Move bytecode version the VM understands. All older versions are /// accepted. move_binary_format_version: Option, @@ -366,12 +378,12 @@ pub struct ProtocolConfig { binary_variant_instantiation_handles: Option, /// Maximum size of the `contents` part of an object, in bytes. Enforced by - /// the Iota adapter when effects are produced. + /// the IOTA adapter when effects are produced. max_move_object_size: Option, // TODO: Option 500 KB exceeds the max // computation gas cost - /// Maximum size of a Move package object, in bytes. Enforced by the Iota + /// Maximum size of a Move package object, in bytes. Enforced by the IOTA /// adapter at the end of a publish transaction. max_move_package_size: Option, @@ -507,6 +519,8 @@ pub struct ProtocolConfig { // === Object runtime internal operation limits ==== // These affect dynamic fields + + // /// Maximum number of cached objects in the object runtime ObjectStore. /// Enforced by object runtime during execution object_runtime_max_num_cached_objects: Option, @@ -524,7 +538,9 @@ pub struct ProtocolConfig { object_runtime_max_num_store_entries_system_tx: Option, // === Execution gas costs ==== - /// Base cost for any Iota transaction + + // + /// Base cost for any IOTA transaction base_tx_cost_fixed: Option, /// Additional cost for a transaction that publishes a package @@ -560,23 +576,25 @@ pub struct ProtocolConfig { // entire object, just consulting an ID -> tx digest map obj_access_cost_verify_per_byte: Option, - /// === Gas version. gas model === + // === Gas version. gas model === + // /// Gas model version, what code we are using to charge gas gas_model_version: Option, - /// === Storage gas costs === + // === Storage gas costs === - /// Per-byte cost of storing an object in the Iota global object store. Some + // + /// Per-byte cost of storing an object in the IOTA global object store. Some /// of this cost may be refundable if the object is later freed obj_data_cost_refundable: Option, - // Per-byte cost of storing an object in the Iota transaction log (e.g., in + // Per-byte cost of storing an object in the IOTA transaction log (e.g., in // CertifiedTransactionEffects) This depends on the size of various fields including the // effects TODO: Option, - /// === Tokenomics === + // === Tokenomics === // TODO: Option, - /// === Core Protocol === + // === Core Protocol === + // /// Max number of transactions per checkpoint. /// Note that this is a protocol constant and not a config as validators /// must have this set to the same value, otherwise they *will* fork. @@ -864,7 +883,7 @@ pub struct ProtocolConfig { debug_print_base_cost: Option, debug_print_stack_trace_base_cost: Option, - /// === Execution Version === + // === Execution Version === execution_version: Option, // Dictates the threshold (percentage of stake) that is used to calculate the "bad" nodes to be @@ -878,8 +897,7 @@ pub struct ProtocolConfig { // will cause the new epoch to start with JWKs from the previous epoch still valid. max_age_of_jwk_in_epochs: Option, - /// === random beacon === - + // === random beacon === /// Maximum allowed precision loss when reducing voting weights for the /// random beacon protocol. random_beacon_reduction_allowed_delta: Option, @@ -1053,6 +1071,10 @@ impl ProtocolConfig { pub fn rethrow_serialization_type_layout_errors(&self) -> bool { self.feature_flags.rethrow_serialization_type_layout_errors } + + pub fn relocate_event_module(&self) -> bool { + self.feature_flags.relocate_event_module + } } #[cfg(not(msim))] @@ -1639,6 +1661,10 @@ impl ProtocolConfig { 1 => unreachable!(), // version 2 is a new framework version but with no config changes 2 => {} + // version 3 + 3 => { + cfg.feature_flags.relocate_event_module = true; + } // Use this template when making changes: // // // modify an existing constant. diff --git a/crates/iota-protocol-config/src/snapshots/iota_protocol_config__test__Mainnet_version_3.snap b/crates/iota-protocol-config/src/snapshots/iota_protocol_config__test__Mainnet_version_3.snap new file mode 100644 index 00000000000..c5550f5ac7e --- /dev/null +++ b/crates/iota-protocol-config/src/snapshots/iota_protocol_config__test__Mainnet_version_3.snap @@ -0,0 +1,271 @@ +--- +source: crates/iota-protocol-config/src/lib.rs +expression: "ProtocolConfig::get_for_version(cur, *chain_id)" +snapshot_kind: text +--- +version: 3 +feature_flags: + consensus_transaction_ordering: ByGasPrice + per_object_congestion_control_mode: TotalTxCount + zklogin_max_epoch_upper_bound_delta: 30 + relocate_event_module: true +max_tx_size_bytes: 131072 +max_input_objects: 2048 +max_size_written_objects: 5000000 +max_size_written_objects_system_tx: 50000000 +max_serialized_tx_effects_size_bytes: 524288 +max_serialized_tx_effects_size_bytes_system_tx: 8388608 +max_gas_payment_objects: 256 +max_modules_in_publish: 64 +max_package_dependencies: 32 +max_arguments: 512 +max_type_arguments: 16 +max_type_argument_depth: 16 +max_pure_argument_size: 16384 +max_programmable_tx_commands: 1024 +move_binary_format_version: 7 +min_move_binary_format_version: 6 +binary_module_handles: 100 +binary_struct_handles: 300 +binary_function_handles: 1500 +binary_function_instantiations: 750 +binary_signatures: 1000 +binary_constant_pool: 4000 +binary_identifiers: 10000 +binary_address_identifiers: 100 +binary_struct_defs: 200 +binary_struct_def_instantiations: 100 +binary_function_defs: 1000 +binary_field_handles: 500 +binary_field_instantiations: 250 +binary_friend_decls: 100 +max_move_object_size: 256000 +max_move_package_size: 102400 +max_publish_or_upgrade_per_ptb: 5 +max_tx_gas: 50000000000 +max_gas_price: 100000 +max_gas_computation_bucket: 5000000 +gas_rounding_step: 1000 +max_loop_depth: 5 +max_generic_instantiation_length: 32 +max_function_parameters: 128 +max_basic_blocks: 1024 +max_value_stack_size: 1024 +max_type_nodes: 256 +max_push_size: 10000 +max_struct_definitions: 200 +max_function_definitions: 1000 +max_fields_in_struct: 32 +max_dependency_depth: 100 +max_num_event_emit: 1024 +max_num_new_move_object_ids: 2048 +max_num_new_move_object_ids_system_tx: 32768 +max_num_deleted_move_object_ids: 2048 +max_num_deleted_move_object_ids_system_tx: 32768 +max_num_transferred_move_object_ids: 2048 +max_num_transferred_move_object_ids_system_tx: 32768 +max_event_emit_size: 256000 +max_event_emit_size_total: 65536000 +max_move_vector_len: 262144 +max_move_identifier_len: 128 +max_move_value_depth: 128 +max_back_edges_per_function: 10000 +max_back_edges_per_module: 10000 +max_verifier_meter_ticks_per_function: 16000000 +max_meter_ticks_per_module: 16000000 +max_meter_ticks_per_package: 16000000 +object_runtime_max_num_cached_objects: 1000 +object_runtime_max_num_cached_objects_system_tx: 16000 +object_runtime_max_num_store_entries: 1000 +object_runtime_max_num_store_entries_system_tx: 16000 +base_tx_cost_fixed: 1000 +package_publish_cost_fixed: 1000 +base_tx_cost_per_byte: 0 +package_publish_cost_per_byte: 80 +obj_access_cost_read_per_byte: 15 +obj_access_cost_mutate_per_byte: 40 +obj_access_cost_delete_per_byte: 40 +obj_access_cost_verify_per_byte: 200 +gas_model_version: 1 +obj_data_cost_refundable: 100 +obj_metadata_cost_non_refundable: 50 +storage_rebate_rate: 10000 +reward_slashing_rate: 10000 +storage_gas_price: 76 +validator_target_reward: 767000000000000 +max_transactions_per_checkpoint: 10000 +max_checkpoint_size_bytes: 31457280 +buffer_stake_for_protocol_upgrade_bps: 5000 +address_from_bytes_cost_base: 52 +address_to_u256_cost_base: 52 +address_from_u256_cost_base: 52 +config_read_setting_impl_cost_base: 100 +config_read_setting_impl_cost_per_byte: 40 +dynamic_field_hash_type_and_key_cost_base: 100 +dynamic_field_hash_type_and_key_type_cost_per_byte: 2 +dynamic_field_hash_type_and_key_value_cost_per_byte: 2 +dynamic_field_hash_type_and_key_type_tag_cost_per_byte: 2 +dynamic_field_add_child_object_cost_base: 100 +dynamic_field_add_child_object_type_cost_per_byte: 10 +dynamic_field_add_child_object_value_cost_per_byte: 10 +dynamic_field_add_child_object_struct_tag_cost_per_byte: 10 +dynamic_field_borrow_child_object_cost_base: 100 +dynamic_field_borrow_child_object_child_ref_cost_per_byte: 10 +dynamic_field_borrow_child_object_type_cost_per_byte: 10 +dynamic_field_remove_child_object_cost_base: 100 +dynamic_field_remove_child_object_child_cost_per_byte: 2 +dynamic_field_remove_child_object_type_cost_per_byte: 2 +dynamic_field_has_child_object_cost_base: 100 +dynamic_field_has_child_object_with_ty_cost_base: 100 +dynamic_field_has_child_object_with_ty_type_cost_per_byte: 2 +dynamic_field_has_child_object_with_ty_type_tag_cost_per_byte: 2 +event_emit_cost_base: 52 +event_emit_value_size_derivation_cost_per_byte: 2 +event_emit_tag_size_derivation_cost_per_byte: 5 +event_emit_output_cost_per_byte: 10 +object_borrow_uid_cost_base: 52 +object_delete_impl_cost_base: 52 +object_record_new_uid_cost_base: 52 +transfer_transfer_internal_cost_base: 52 +transfer_freeze_object_cost_base: 52 +transfer_share_object_cost_base: 52 +transfer_receive_object_cost_base: 52 +tx_context_derive_id_cost_base: 52 +types_is_one_time_witness_cost_base: 52 +types_is_one_time_witness_type_tag_cost_per_byte: 2 +types_is_one_time_witness_type_cost_per_byte: 2 +validator_validate_metadata_cost_base: 52 +validator_validate_metadata_data_cost_per_byte: 2 +crypto_invalid_arguments_cost: 100 +bls12381_bls12381_min_sig_verify_cost_base: 52 +bls12381_bls12381_min_sig_verify_msg_cost_per_byte: 2 +bls12381_bls12381_min_sig_verify_msg_cost_per_block: 2 +bls12381_bls12381_min_pk_verify_cost_base: 52 +bls12381_bls12381_min_pk_verify_msg_cost_per_byte: 2 +bls12381_bls12381_min_pk_verify_msg_cost_per_block: 2 +ecdsa_k1_ecrecover_keccak256_cost_base: 52 +ecdsa_k1_ecrecover_keccak256_msg_cost_per_byte: 2 +ecdsa_k1_ecrecover_keccak256_msg_cost_per_block: 2 +ecdsa_k1_ecrecover_sha256_cost_base: 52 +ecdsa_k1_ecrecover_sha256_msg_cost_per_byte: 2 +ecdsa_k1_ecrecover_sha256_msg_cost_per_block: 2 +ecdsa_k1_decompress_pubkey_cost_base: 52 +ecdsa_k1_secp256k1_verify_keccak256_cost_base: 52 +ecdsa_k1_secp256k1_verify_keccak256_msg_cost_per_byte: 2 +ecdsa_k1_secp256k1_verify_keccak256_msg_cost_per_block: 2 +ecdsa_k1_secp256k1_verify_sha256_cost_base: 52 +ecdsa_k1_secp256k1_verify_sha256_msg_cost_per_byte: 2 +ecdsa_k1_secp256k1_verify_sha256_msg_cost_per_block: 2 +ecdsa_r1_ecrecover_keccak256_cost_base: 52 +ecdsa_r1_ecrecover_keccak256_msg_cost_per_byte: 2 +ecdsa_r1_ecrecover_keccak256_msg_cost_per_block: 2 +ecdsa_r1_ecrecover_sha256_cost_base: 52 +ecdsa_r1_ecrecover_sha256_msg_cost_per_byte: 2 +ecdsa_r1_ecrecover_sha256_msg_cost_per_block: 2 +ecdsa_r1_secp256r1_verify_keccak256_cost_base: 52 +ecdsa_r1_secp256r1_verify_keccak256_msg_cost_per_byte: 2 +ecdsa_r1_secp256r1_verify_keccak256_msg_cost_per_block: 2 +ecdsa_r1_secp256r1_verify_sha256_cost_base: 52 +ecdsa_r1_secp256r1_verify_sha256_msg_cost_per_byte: 2 +ecdsa_r1_secp256r1_verify_sha256_msg_cost_per_block: 2 +ecvrf_ecvrf_verify_cost_base: 52 +ecvrf_ecvrf_verify_alpha_string_cost_per_byte: 2 +ecvrf_ecvrf_verify_alpha_string_cost_per_block: 2 +ed25519_ed25519_verify_cost_base: 52 +ed25519_ed25519_verify_msg_cost_per_byte: 2 +ed25519_ed25519_verify_msg_cost_per_block: 2 +groth16_prepare_verifying_key_bls12381_cost_base: 52 +groth16_prepare_verifying_key_bn254_cost_base: 52 +groth16_verify_groth16_proof_internal_bls12381_cost_base: 52 +groth16_verify_groth16_proof_internal_bls12381_cost_per_public_input: 2 +groth16_verify_groth16_proof_internal_bn254_cost_base: 52 +groth16_verify_groth16_proof_internal_bn254_cost_per_public_input: 2 +groth16_verify_groth16_proof_internal_public_input_cost_per_byte: 2 +hash_blake2b256_cost_base: 52 +hash_blake2b256_data_cost_per_byte: 2 +hash_blake2b256_data_cost_per_block: 2 +hash_keccak256_cost_base: 52 +hash_keccak256_data_cost_per_byte: 2 +hash_keccak256_data_cost_per_block: 2 +group_ops_bls12381_decode_scalar_cost: 52 +group_ops_bls12381_decode_g1_cost: 52 +group_ops_bls12381_decode_g2_cost: 52 +group_ops_bls12381_decode_gt_cost: 52 +group_ops_bls12381_scalar_add_cost: 52 +group_ops_bls12381_g1_add_cost: 52 +group_ops_bls12381_g2_add_cost: 52 +group_ops_bls12381_gt_add_cost: 52 +group_ops_bls12381_scalar_sub_cost: 52 +group_ops_bls12381_g1_sub_cost: 52 +group_ops_bls12381_g2_sub_cost: 52 +group_ops_bls12381_gt_sub_cost: 52 +group_ops_bls12381_scalar_mul_cost: 52 +group_ops_bls12381_g1_mul_cost: 52 +group_ops_bls12381_g2_mul_cost: 52 +group_ops_bls12381_gt_mul_cost: 52 +group_ops_bls12381_scalar_div_cost: 52 +group_ops_bls12381_g1_div_cost: 52 +group_ops_bls12381_g2_div_cost: 52 +group_ops_bls12381_gt_div_cost: 52 +group_ops_bls12381_g1_hash_to_base_cost: 52 +group_ops_bls12381_g2_hash_to_base_cost: 52 +group_ops_bls12381_g1_hash_to_cost_per_byte: 2 +group_ops_bls12381_g2_hash_to_cost_per_byte: 2 +group_ops_bls12381_g1_msm_base_cost: 52 +group_ops_bls12381_g2_msm_base_cost: 52 +group_ops_bls12381_g1_msm_base_cost_per_input: 52 +group_ops_bls12381_g2_msm_base_cost_per_input: 52 +group_ops_bls12381_msm_max_len: 32 +group_ops_bls12381_pairing_cost: 52 +hmac_hmac_sha3_256_cost_base: 52 +hmac_hmac_sha3_256_input_cost_per_byte: 2 +hmac_hmac_sha3_256_input_cost_per_block: 2 +check_zklogin_id_cost_base: 200 +check_zklogin_issuer_cost_base: 200 +bcs_per_byte_serialized_cost: 2 +bcs_legacy_min_output_size_cost: 1 +bcs_failure_cost: 52 +hash_sha2_256_base_cost: 52 +hash_sha2_256_per_byte_cost: 2 +hash_sha2_256_legacy_min_input_len_cost: 1 +hash_sha3_256_base_cost: 52 +hash_sha3_256_per_byte_cost: 2 +hash_sha3_256_legacy_min_input_len_cost: 1 +type_name_get_base_cost: 52 +type_name_get_per_byte_cost: 2 +string_check_utf8_base_cost: 52 +string_check_utf8_per_byte_cost: 2 +string_is_char_boundary_base_cost: 52 +string_sub_string_base_cost: 52 +string_sub_string_per_byte_cost: 2 +string_index_of_base_cost: 52 +string_index_of_per_byte_pattern_cost: 2 +string_index_of_per_byte_searched_cost: 2 +vector_empty_base_cost: 52 +vector_length_base_cost: 52 +vector_push_back_base_cost: 52 +vector_push_back_legacy_per_abstract_memory_unit_cost: 2 +vector_borrow_base_cost: 52 +vector_pop_back_base_cost: 52 +vector_destroy_empty_base_cost: 52 +vector_swap_base_cost: 52 +debug_print_base_cost: 52 +debug_print_stack_trace_base_cost: 52 +execution_version: 1 +consensus_bad_nodes_stake_threshold: 20 +max_jwk_votes_per_validator_per_epoch: 240 +max_age_of_jwk_in_epochs: 1 +random_beacon_reduction_allowed_delta: 800 +random_beacon_reduction_lower_bound: 1000 +random_beacon_dkg_timeout_round: 3000 +random_beacon_min_round_interval_ms: 500 +random_beacon_dkg_version: 1 +consensus_max_transaction_size_bytes: 262144 +consensus_max_transactions_in_block_bytes: 524288 +consensus_max_num_transactions_in_block: 512 +max_deferral_rounds_for_congestion_control: 10 +min_checkpoint_interval_ms: 200 +checkpoint_summary_version_specific_data: 1 +max_soft_bundle_size: 5 +bridge_should_try_to_finalize_committee: false +max_accumulated_txn_cost_per_object_in_mysticeti_commit: 10 diff --git a/crates/iota-protocol-config/src/snapshots/iota_protocol_config__test__Testnet_version_3.snap b/crates/iota-protocol-config/src/snapshots/iota_protocol_config__test__Testnet_version_3.snap new file mode 100644 index 00000000000..148c0e0f3e0 --- /dev/null +++ b/crates/iota-protocol-config/src/snapshots/iota_protocol_config__test__Testnet_version_3.snap @@ -0,0 +1,271 @@ +--- +source: crates/iota-protocol-config/src/lib.rs +expression: "ProtocolConfig::get_for_version(cur, *chain_id)" +snapshot_kind: text +--- +version: 3 +feature_flags: + consensus_transaction_ordering: ByGasPrice + per_object_congestion_control_mode: TotalTxCount + zklogin_max_epoch_upper_bound_delta: 30 + relocate_event_module: true +max_tx_size_bytes: 131072 +max_input_objects: 2048 +max_size_written_objects: 5000000 +max_size_written_objects_system_tx: 50000000 +max_serialized_tx_effects_size_bytes: 524288 +max_serialized_tx_effects_size_bytes_system_tx: 8388608 +max_gas_payment_objects: 256 +max_modules_in_publish: 64 +max_package_dependencies: 32 +max_arguments: 512 +max_type_arguments: 16 +max_type_argument_depth: 16 +max_pure_argument_size: 16384 +max_programmable_tx_commands: 1024 +move_binary_format_version: 7 +min_move_binary_format_version: 6 +binary_module_handles: 100 +binary_struct_handles: 300 +binary_function_handles: 1500 +binary_function_instantiations: 750 +binary_signatures: 1000 +binary_constant_pool: 4000 +binary_identifiers: 10000 +binary_address_identifiers: 100 +binary_struct_defs: 200 +binary_struct_def_instantiations: 100 +binary_function_defs: 1000 +binary_field_handles: 500 +binary_field_instantiations: 250 +binary_friend_decls: 100 +max_move_object_size: 256000 +max_move_package_size: 102400 +max_publish_or_upgrade_per_ptb: 5 +max_tx_gas: 50000000000 +max_gas_price: 100000 +max_gas_computation_bucket: 5000000 +gas_rounding_step: 1000 +max_loop_depth: 5 +max_generic_instantiation_length: 32 +max_function_parameters: 128 +max_basic_blocks: 1024 +max_value_stack_size: 1024 +max_type_nodes: 256 +max_push_size: 10000 +max_struct_definitions: 200 +max_function_definitions: 1000 +max_fields_in_struct: 32 +max_dependency_depth: 100 +max_num_event_emit: 1024 +max_num_new_move_object_ids: 2048 +max_num_new_move_object_ids_system_tx: 32768 +max_num_deleted_move_object_ids: 2048 +max_num_deleted_move_object_ids_system_tx: 32768 +max_num_transferred_move_object_ids: 2048 +max_num_transferred_move_object_ids_system_tx: 32768 +max_event_emit_size: 256000 +max_event_emit_size_total: 65536000 +max_move_vector_len: 262144 +max_move_identifier_len: 128 +max_move_value_depth: 128 +max_back_edges_per_function: 10000 +max_back_edges_per_module: 10000 +max_verifier_meter_ticks_per_function: 16000000 +max_meter_ticks_per_module: 16000000 +max_meter_ticks_per_package: 16000000 +object_runtime_max_num_cached_objects: 1000 +object_runtime_max_num_cached_objects_system_tx: 16000 +object_runtime_max_num_store_entries: 1000 +object_runtime_max_num_store_entries_system_tx: 16000 +base_tx_cost_fixed: 1000 +package_publish_cost_fixed: 1000 +base_tx_cost_per_byte: 0 +package_publish_cost_per_byte: 80 +obj_access_cost_read_per_byte: 15 +obj_access_cost_mutate_per_byte: 40 +obj_access_cost_delete_per_byte: 40 +obj_access_cost_verify_per_byte: 200 +gas_model_version: 1 +obj_data_cost_refundable: 100 +obj_metadata_cost_non_refundable: 50 +storage_rebate_rate: 10000 +reward_slashing_rate: 10000 +storage_gas_price: 76 +validator_target_reward: 767000000000000 +max_transactions_per_checkpoint: 10000 +max_checkpoint_size_bytes: 31457280 +buffer_stake_for_protocol_upgrade_bps: 5000 +address_from_bytes_cost_base: 52 +address_to_u256_cost_base: 52 +address_from_u256_cost_base: 52 +config_read_setting_impl_cost_base: 100 +config_read_setting_impl_cost_per_byte: 40 +dynamic_field_hash_type_and_key_cost_base: 100 +dynamic_field_hash_type_and_key_type_cost_per_byte: 2 +dynamic_field_hash_type_and_key_value_cost_per_byte: 2 +dynamic_field_hash_type_and_key_type_tag_cost_per_byte: 2 +dynamic_field_add_child_object_cost_base: 100 +dynamic_field_add_child_object_type_cost_per_byte: 10 +dynamic_field_add_child_object_value_cost_per_byte: 10 +dynamic_field_add_child_object_struct_tag_cost_per_byte: 10 +dynamic_field_borrow_child_object_cost_base: 100 +dynamic_field_borrow_child_object_child_ref_cost_per_byte: 10 +dynamic_field_borrow_child_object_type_cost_per_byte: 10 +dynamic_field_remove_child_object_cost_base: 100 +dynamic_field_remove_child_object_child_cost_per_byte: 2 +dynamic_field_remove_child_object_type_cost_per_byte: 2 +dynamic_field_has_child_object_cost_base: 100 +dynamic_field_has_child_object_with_ty_cost_base: 100 +dynamic_field_has_child_object_with_ty_type_cost_per_byte: 2 +dynamic_field_has_child_object_with_ty_type_tag_cost_per_byte: 2 +event_emit_cost_base: 52 +event_emit_value_size_derivation_cost_per_byte: 2 +event_emit_tag_size_derivation_cost_per_byte: 5 +event_emit_output_cost_per_byte: 10 +object_borrow_uid_cost_base: 52 +object_delete_impl_cost_base: 52 +object_record_new_uid_cost_base: 52 +transfer_transfer_internal_cost_base: 52 +transfer_freeze_object_cost_base: 52 +transfer_share_object_cost_base: 52 +transfer_receive_object_cost_base: 52 +tx_context_derive_id_cost_base: 52 +types_is_one_time_witness_cost_base: 52 +types_is_one_time_witness_type_tag_cost_per_byte: 2 +types_is_one_time_witness_type_cost_per_byte: 2 +validator_validate_metadata_cost_base: 52 +validator_validate_metadata_data_cost_per_byte: 2 +crypto_invalid_arguments_cost: 100 +bls12381_bls12381_min_sig_verify_cost_base: 52 +bls12381_bls12381_min_sig_verify_msg_cost_per_byte: 2 +bls12381_bls12381_min_sig_verify_msg_cost_per_block: 2 +bls12381_bls12381_min_pk_verify_cost_base: 52 +bls12381_bls12381_min_pk_verify_msg_cost_per_byte: 2 +bls12381_bls12381_min_pk_verify_msg_cost_per_block: 2 +ecdsa_k1_ecrecover_keccak256_cost_base: 52 +ecdsa_k1_ecrecover_keccak256_msg_cost_per_byte: 2 +ecdsa_k1_ecrecover_keccak256_msg_cost_per_block: 2 +ecdsa_k1_ecrecover_sha256_cost_base: 52 +ecdsa_k1_ecrecover_sha256_msg_cost_per_byte: 2 +ecdsa_k1_ecrecover_sha256_msg_cost_per_block: 2 +ecdsa_k1_decompress_pubkey_cost_base: 52 +ecdsa_k1_secp256k1_verify_keccak256_cost_base: 52 +ecdsa_k1_secp256k1_verify_keccak256_msg_cost_per_byte: 2 +ecdsa_k1_secp256k1_verify_keccak256_msg_cost_per_block: 2 +ecdsa_k1_secp256k1_verify_sha256_cost_base: 52 +ecdsa_k1_secp256k1_verify_sha256_msg_cost_per_byte: 2 +ecdsa_k1_secp256k1_verify_sha256_msg_cost_per_block: 2 +ecdsa_r1_ecrecover_keccak256_cost_base: 52 +ecdsa_r1_ecrecover_keccak256_msg_cost_per_byte: 2 +ecdsa_r1_ecrecover_keccak256_msg_cost_per_block: 2 +ecdsa_r1_ecrecover_sha256_cost_base: 52 +ecdsa_r1_ecrecover_sha256_msg_cost_per_byte: 2 +ecdsa_r1_ecrecover_sha256_msg_cost_per_block: 2 +ecdsa_r1_secp256r1_verify_keccak256_cost_base: 52 +ecdsa_r1_secp256r1_verify_keccak256_msg_cost_per_byte: 2 +ecdsa_r1_secp256r1_verify_keccak256_msg_cost_per_block: 2 +ecdsa_r1_secp256r1_verify_sha256_cost_base: 52 +ecdsa_r1_secp256r1_verify_sha256_msg_cost_per_byte: 2 +ecdsa_r1_secp256r1_verify_sha256_msg_cost_per_block: 2 +ecvrf_ecvrf_verify_cost_base: 52 +ecvrf_ecvrf_verify_alpha_string_cost_per_byte: 2 +ecvrf_ecvrf_verify_alpha_string_cost_per_block: 2 +ed25519_ed25519_verify_cost_base: 52 +ed25519_ed25519_verify_msg_cost_per_byte: 2 +ed25519_ed25519_verify_msg_cost_per_block: 2 +groth16_prepare_verifying_key_bls12381_cost_base: 52 +groth16_prepare_verifying_key_bn254_cost_base: 52 +groth16_verify_groth16_proof_internal_bls12381_cost_base: 52 +groth16_verify_groth16_proof_internal_bls12381_cost_per_public_input: 2 +groth16_verify_groth16_proof_internal_bn254_cost_base: 52 +groth16_verify_groth16_proof_internal_bn254_cost_per_public_input: 2 +groth16_verify_groth16_proof_internal_public_input_cost_per_byte: 2 +hash_blake2b256_cost_base: 52 +hash_blake2b256_data_cost_per_byte: 2 +hash_blake2b256_data_cost_per_block: 2 +hash_keccak256_cost_base: 52 +hash_keccak256_data_cost_per_byte: 2 +hash_keccak256_data_cost_per_block: 2 +group_ops_bls12381_decode_scalar_cost: 52 +group_ops_bls12381_decode_g1_cost: 52 +group_ops_bls12381_decode_g2_cost: 52 +group_ops_bls12381_decode_gt_cost: 52 +group_ops_bls12381_scalar_add_cost: 52 +group_ops_bls12381_g1_add_cost: 52 +group_ops_bls12381_g2_add_cost: 52 +group_ops_bls12381_gt_add_cost: 52 +group_ops_bls12381_scalar_sub_cost: 52 +group_ops_bls12381_g1_sub_cost: 52 +group_ops_bls12381_g2_sub_cost: 52 +group_ops_bls12381_gt_sub_cost: 52 +group_ops_bls12381_scalar_mul_cost: 52 +group_ops_bls12381_g1_mul_cost: 52 +group_ops_bls12381_g2_mul_cost: 52 +group_ops_bls12381_gt_mul_cost: 52 +group_ops_bls12381_scalar_div_cost: 52 +group_ops_bls12381_g1_div_cost: 52 +group_ops_bls12381_g2_div_cost: 52 +group_ops_bls12381_gt_div_cost: 52 +group_ops_bls12381_g1_hash_to_base_cost: 52 +group_ops_bls12381_g2_hash_to_base_cost: 52 +group_ops_bls12381_g1_hash_to_cost_per_byte: 2 +group_ops_bls12381_g2_hash_to_cost_per_byte: 2 +group_ops_bls12381_g1_msm_base_cost: 52 +group_ops_bls12381_g2_msm_base_cost: 52 +group_ops_bls12381_g1_msm_base_cost_per_input: 52 +group_ops_bls12381_g2_msm_base_cost_per_input: 52 +group_ops_bls12381_msm_max_len: 32 +group_ops_bls12381_pairing_cost: 52 +hmac_hmac_sha3_256_cost_base: 52 +hmac_hmac_sha3_256_input_cost_per_byte: 2 +hmac_hmac_sha3_256_input_cost_per_block: 2 +check_zklogin_id_cost_base: 200 +check_zklogin_issuer_cost_base: 200 +bcs_per_byte_serialized_cost: 2 +bcs_legacy_min_output_size_cost: 1 +bcs_failure_cost: 52 +hash_sha2_256_base_cost: 52 +hash_sha2_256_per_byte_cost: 2 +hash_sha2_256_legacy_min_input_len_cost: 1 +hash_sha3_256_base_cost: 52 +hash_sha3_256_per_byte_cost: 2 +hash_sha3_256_legacy_min_input_len_cost: 1 +type_name_get_base_cost: 52 +type_name_get_per_byte_cost: 2 +string_check_utf8_base_cost: 52 +string_check_utf8_per_byte_cost: 2 +string_is_char_boundary_base_cost: 52 +string_sub_string_base_cost: 52 +string_sub_string_per_byte_cost: 2 +string_index_of_base_cost: 52 +string_index_of_per_byte_pattern_cost: 2 +string_index_of_per_byte_searched_cost: 2 +vector_empty_base_cost: 52 +vector_length_base_cost: 52 +vector_push_back_base_cost: 52 +vector_push_back_legacy_per_abstract_memory_unit_cost: 2 +vector_borrow_base_cost: 52 +vector_pop_back_base_cost: 52 +vector_destroy_empty_base_cost: 52 +vector_swap_base_cost: 52 +debug_print_base_cost: 52 +debug_print_stack_trace_base_cost: 52 +execution_version: 1 +consensus_bad_nodes_stake_threshold: 20 +max_jwk_votes_per_validator_per_epoch: 240 +max_age_of_jwk_in_epochs: 1 +random_beacon_reduction_allowed_delta: 800 +random_beacon_reduction_lower_bound: 1000 +random_beacon_dkg_timeout_round: 3000 +random_beacon_min_round_interval_ms: 500 +random_beacon_dkg_version: 1 +consensus_max_transaction_size_bytes: 262144 +consensus_max_transactions_in_block_bytes: 524288 +consensus_max_num_transactions_in_block: 512 +max_deferral_rounds_for_congestion_control: 10 +min_checkpoint_interval_ms: 200 +checkpoint_summary_version_specific_data: 1 +max_soft_bundle_size: 5 +bridge_should_try_to_finalize_committee: true +max_accumulated_txn_cost_per_object_in_mysticeti_commit: 10 diff --git a/crates/iota-protocol-config/src/snapshots/iota_protocol_config__test__version_3.snap b/crates/iota-protocol-config/src/snapshots/iota_protocol_config__test__version_3.snap new file mode 100644 index 00000000000..3e7f5807cab --- /dev/null +++ b/crates/iota-protocol-config/src/snapshots/iota_protocol_config__test__version_3.snap @@ -0,0 +1,279 @@ +--- +source: crates/iota-protocol-config/src/lib.rs +expression: "ProtocolConfig::get_for_version(cur, *chain_id)" +snapshot_kind: text +--- +version: 3 +feature_flags: + consensus_transaction_ordering: ByGasPrice + enable_poseidon: true + enable_group_ops_native_function_msm: true + per_object_congestion_control_mode: TotalTxCount + zklogin_max_epoch_upper_bound_delta: 30 + enable_vdf: true + passkey_auth: true + relocate_event_module: true +max_tx_size_bytes: 131072 +max_input_objects: 2048 +max_size_written_objects: 5000000 +max_size_written_objects_system_tx: 50000000 +max_serialized_tx_effects_size_bytes: 524288 +max_serialized_tx_effects_size_bytes_system_tx: 8388608 +max_gas_payment_objects: 256 +max_modules_in_publish: 64 +max_package_dependencies: 32 +max_arguments: 512 +max_type_arguments: 16 +max_type_argument_depth: 16 +max_pure_argument_size: 16384 +max_programmable_tx_commands: 1024 +move_binary_format_version: 7 +min_move_binary_format_version: 6 +binary_module_handles: 100 +binary_struct_handles: 300 +binary_function_handles: 1500 +binary_function_instantiations: 750 +binary_signatures: 1000 +binary_constant_pool: 4000 +binary_identifiers: 10000 +binary_address_identifiers: 100 +binary_struct_defs: 200 +binary_struct_def_instantiations: 100 +binary_function_defs: 1000 +binary_field_handles: 500 +binary_field_instantiations: 250 +binary_friend_decls: 100 +max_move_object_size: 256000 +max_move_package_size: 102400 +max_publish_or_upgrade_per_ptb: 5 +max_tx_gas: 50000000000 +max_gas_price: 100000 +max_gas_computation_bucket: 5000000 +gas_rounding_step: 1000 +max_loop_depth: 5 +max_generic_instantiation_length: 32 +max_function_parameters: 128 +max_basic_blocks: 1024 +max_value_stack_size: 1024 +max_type_nodes: 256 +max_push_size: 10000 +max_struct_definitions: 200 +max_function_definitions: 1000 +max_fields_in_struct: 32 +max_dependency_depth: 100 +max_num_event_emit: 1024 +max_num_new_move_object_ids: 2048 +max_num_new_move_object_ids_system_tx: 32768 +max_num_deleted_move_object_ids: 2048 +max_num_deleted_move_object_ids_system_tx: 32768 +max_num_transferred_move_object_ids: 2048 +max_num_transferred_move_object_ids_system_tx: 32768 +max_event_emit_size: 256000 +max_event_emit_size_total: 65536000 +max_move_vector_len: 262144 +max_move_identifier_len: 128 +max_move_value_depth: 128 +max_back_edges_per_function: 10000 +max_back_edges_per_module: 10000 +max_verifier_meter_ticks_per_function: 16000000 +max_meter_ticks_per_module: 16000000 +max_meter_ticks_per_package: 16000000 +object_runtime_max_num_cached_objects: 1000 +object_runtime_max_num_cached_objects_system_tx: 16000 +object_runtime_max_num_store_entries: 1000 +object_runtime_max_num_store_entries_system_tx: 16000 +base_tx_cost_fixed: 1000 +package_publish_cost_fixed: 1000 +base_tx_cost_per_byte: 0 +package_publish_cost_per_byte: 80 +obj_access_cost_read_per_byte: 15 +obj_access_cost_mutate_per_byte: 40 +obj_access_cost_delete_per_byte: 40 +obj_access_cost_verify_per_byte: 200 +gas_model_version: 1 +obj_data_cost_refundable: 100 +obj_metadata_cost_non_refundable: 50 +storage_rebate_rate: 10000 +reward_slashing_rate: 10000 +storage_gas_price: 76 +validator_target_reward: 767000000000000 +max_transactions_per_checkpoint: 10000 +max_checkpoint_size_bytes: 31457280 +buffer_stake_for_protocol_upgrade_bps: 5000 +address_from_bytes_cost_base: 52 +address_to_u256_cost_base: 52 +address_from_u256_cost_base: 52 +config_read_setting_impl_cost_base: 100 +config_read_setting_impl_cost_per_byte: 40 +dynamic_field_hash_type_and_key_cost_base: 100 +dynamic_field_hash_type_and_key_type_cost_per_byte: 2 +dynamic_field_hash_type_and_key_value_cost_per_byte: 2 +dynamic_field_hash_type_and_key_type_tag_cost_per_byte: 2 +dynamic_field_add_child_object_cost_base: 100 +dynamic_field_add_child_object_type_cost_per_byte: 10 +dynamic_field_add_child_object_value_cost_per_byte: 10 +dynamic_field_add_child_object_struct_tag_cost_per_byte: 10 +dynamic_field_borrow_child_object_cost_base: 100 +dynamic_field_borrow_child_object_child_ref_cost_per_byte: 10 +dynamic_field_borrow_child_object_type_cost_per_byte: 10 +dynamic_field_remove_child_object_cost_base: 100 +dynamic_field_remove_child_object_child_cost_per_byte: 2 +dynamic_field_remove_child_object_type_cost_per_byte: 2 +dynamic_field_has_child_object_cost_base: 100 +dynamic_field_has_child_object_with_ty_cost_base: 100 +dynamic_field_has_child_object_with_ty_type_cost_per_byte: 2 +dynamic_field_has_child_object_with_ty_type_tag_cost_per_byte: 2 +event_emit_cost_base: 52 +event_emit_value_size_derivation_cost_per_byte: 2 +event_emit_tag_size_derivation_cost_per_byte: 5 +event_emit_output_cost_per_byte: 10 +object_borrow_uid_cost_base: 52 +object_delete_impl_cost_base: 52 +object_record_new_uid_cost_base: 52 +transfer_transfer_internal_cost_base: 52 +transfer_freeze_object_cost_base: 52 +transfer_share_object_cost_base: 52 +transfer_receive_object_cost_base: 52 +tx_context_derive_id_cost_base: 52 +types_is_one_time_witness_cost_base: 52 +types_is_one_time_witness_type_tag_cost_per_byte: 2 +types_is_one_time_witness_type_cost_per_byte: 2 +validator_validate_metadata_cost_base: 52 +validator_validate_metadata_data_cost_per_byte: 2 +crypto_invalid_arguments_cost: 100 +bls12381_bls12381_min_sig_verify_cost_base: 52 +bls12381_bls12381_min_sig_verify_msg_cost_per_byte: 2 +bls12381_bls12381_min_sig_verify_msg_cost_per_block: 2 +bls12381_bls12381_min_pk_verify_cost_base: 52 +bls12381_bls12381_min_pk_verify_msg_cost_per_byte: 2 +bls12381_bls12381_min_pk_verify_msg_cost_per_block: 2 +ecdsa_k1_ecrecover_keccak256_cost_base: 52 +ecdsa_k1_ecrecover_keccak256_msg_cost_per_byte: 2 +ecdsa_k1_ecrecover_keccak256_msg_cost_per_block: 2 +ecdsa_k1_ecrecover_sha256_cost_base: 52 +ecdsa_k1_ecrecover_sha256_msg_cost_per_byte: 2 +ecdsa_k1_ecrecover_sha256_msg_cost_per_block: 2 +ecdsa_k1_decompress_pubkey_cost_base: 52 +ecdsa_k1_secp256k1_verify_keccak256_cost_base: 52 +ecdsa_k1_secp256k1_verify_keccak256_msg_cost_per_byte: 2 +ecdsa_k1_secp256k1_verify_keccak256_msg_cost_per_block: 2 +ecdsa_k1_secp256k1_verify_sha256_cost_base: 52 +ecdsa_k1_secp256k1_verify_sha256_msg_cost_per_byte: 2 +ecdsa_k1_secp256k1_verify_sha256_msg_cost_per_block: 2 +ecdsa_r1_ecrecover_keccak256_cost_base: 52 +ecdsa_r1_ecrecover_keccak256_msg_cost_per_byte: 2 +ecdsa_r1_ecrecover_keccak256_msg_cost_per_block: 2 +ecdsa_r1_ecrecover_sha256_cost_base: 52 +ecdsa_r1_ecrecover_sha256_msg_cost_per_byte: 2 +ecdsa_r1_ecrecover_sha256_msg_cost_per_block: 2 +ecdsa_r1_secp256r1_verify_keccak256_cost_base: 52 +ecdsa_r1_secp256r1_verify_keccak256_msg_cost_per_byte: 2 +ecdsa_r1_secp256r1_verify_keccak256_msg_cost_per_block: 2 +ecdsa_r1_secp256r1_verify_sha256_cost_base: 52 +ecdsa_r1_secp256r1_verify_sha256_msg_cost_per_byte: 2 +ecdsa_r1_secp256r1_verify_sha256_msg_cost_per_block: 2 +ecvrf_ecvrf_verify_cost_base: 52 +ecvrf_ecvrf_verify_alpha_string_cost_per_byte: 2 +ecvrf_ecvrf_verify_alpha_string_cost_per_block: 2 +ed25519_ed25519_verify_cost_base: 52 +ed25519_ed25519_verify_msg_cost_per_byte: 2 +ed25519_ed25519_verify_msg_cost_per_block: 2 +groth16_prepare_verifying_key_bls12381_cost_base: 52 +groth16_prepare_verifying_key_bn254_cost_base: 52 +groth16_verify_groth16_proof_internal_bls12381_cost_base: 52 +groth16_verify_groth16_proof_internal_bls12381_cost_per_public_input: 2 +groth16_verify_groth16_proof_internal_bn254_cost_base: 52 +groth16_verify_groth16_proof_internal_bn254_cost_per_public_input: 2 +groth16_verify_groth16_proof_internal_public_input_cost_per_byte: 2 +hash_blake2b256_cost_base: 52 +hash_blake2b256_data_cost_per_byte: 2 +hash_blake2b256_data_cost_per_block: 2 +hash_keccak256_cost_base: 52 +hash_keccak256_data_cost_per_byte: 2 +hash_keccak256_data_cost_per_block: 2 +poseidon_bn254_cost_base: 260 +poseidon_bn254_cost_per_block: 10 +group_ops_bls12381_decode_scalar_cost: 52 +group_ops_bls12381_decode_g1_cost: 52 +group_ops_bls12381_decode_g2_cost: 52 +group_ops_bls12381_decode_gt_cost: 52 +group_ops_bls12381_scalar_add_cost: 52 +group_ops_bls12381_g1_add_cost: 52 +group_ops_bls12381_g2_add_cost: 52 +group_ops_bls12381_gt_add_cost: 52 +group_ops_bls12381_scalar_sub_cost: 52 +group_ops_bls12381_g1_sub_cost: 52 +group_ops_bls12381_g2_sub_cost: 52 +group_ops_bls12381_gt_sub_cost: 52 +group_ops_bls12381_scalar_mul_cost: 52 +group_ops_bls12381_g1_mul_cost: 52 +group_ops_bls12381_g2_mul_cost: 52 +group_ops_bls12381_gt_mul_cost: 52 +group_ops_bls12381_scalar_div_cost: 52 +group_ops_bls12381_g1_div_cost: 52 +group_ops_bls12381_g2_div_cost: 52 +group_ops_bls12381_gt_div_cost: 52 +group_ops_bls12381_g1_hash_to_base_cost: 52 +group_ops_bls12381_g2_hash_to_base_cost: 52 +group_ops_bls12381_g1_hash_to_cost_per_byte: 2 +group_ops_bls12381_g2_hash_to_cost_per_byte: 2 +group_ops_bls12381_g1_msm_base_cost: 52 +group_ops_bls12381_g2_msm_base_cost: 52 +group_ops_bls12381_g1_msm_base_cost_per_input: 52 +group_ops_bls12381_g2_msm_base_cost_per_input: 52 +group_ops_bls12381_msm_max_len: 32 +group_ops_bls12381_pairing_cost: 52 +hmac_hmac_sha3_256_cost_base: 52 +hmac_hmac_sha3_256_input_cost_per_byte: 2 +hmac_hmac_sha3_256_input_cost_per_block: 2 +check_zklogin_id_cost_base: 200 +check_zklogin_issuer_cost_base: 200 +vdf_verify_vdf_cost: 1500 +vdf_hash_to_input_cost: 100 +bcs_per_byte_serialized_cost: 2 +bcs_legacy_min_output_size_cost: 1 +bcs_failure_cost: 52 +hash_sha2_256_base_cost: 52 +hash_sha2_256_per_byte_cost: 2 +hash_sha2_256_legacy_min_input_len_cost: 1 +hash_sha3_256_base_cost: 52 +hash_sha3_256_per_byte_cost: 2 +hash_sha3_256_legacy_min_input_len_cost: 1 +type_name_get_base_cost: 52 +type_name_get_per_byte_cost: 2 +string_check_utf8_base_cost: 52 +string_check_utf8_per_byte_cost: 2 +string_is_char_boundary_base_cost: 52 +string_sub_string_base_cost: 52 +string_sub_string_per_byte_cost: 2 +string_index_of_base_cost: 52 +string_index_of_per_byte_pattern_cost: 2 +string_index_of_per_byte_searched_cost: 2 +vector_empty_base_cost: 52 +vector_length_base_cost: 52 +vector_push_back_base_cost: 52 +vector_push_back_legacy_per_abstract_memory_unit_cost: 2 +vector_borrow_base_cost: 52 +vector_pop_back_base_cost: 52 +vector_destroy_empty_base_cost: 52 +vector_swap_base_cost: 52 +debug_print_base_cost: 52 +debug_print_stack_trace_base_cost: 52 +execution_version: 1 +consensus_bad_nodes_stake_threshold: 20 +max_jwk_votes_per_validator_per_epoch: 240 +max_age_of_jwk_in_epochs: 1 +random_beacon_reduction_allowed_delta: 800 +random_beacon_reduction_lower_bound: 1000 +random_beacon_dkg_timeout_round: 3000 +random_beacon_min_round_interval_ms: 500 +random_beacon_dkg_version: 1 +consensus_max_transaction_size_bytes: 262144 +consensus_max_transactions_in_block_bytes: 524288 +consensus_max_num_transactions_in_block: 512 +max_deferral_rounds_for_congestion_control: 10 +min_checkpoint_interval_ms: 200 +checkpoint_summary_version_specific_data: 1 +max_soft_bundle_size: 5 +bridge_should_try_to_finalize_committee: true +max_accumulated_txn_cost_per_object_in_mysticeti_commit: 10 diff --git a/crates/iota-proxy/Cargo.toml b/crates/iota-proxy/Cargo.toml new file mode 100644 index 00000000000..4664ad08796 --- /dev/null +++ b/crates/iota-proxy/Cargo.toml @@ -0,0 +1,61 @@ +[package] +name = "iota-proxy" +version.workspace = true +authors = ["IOTA Foundation "] +edition = "2021" +license = "Apache-2.0" +publish = false + +[dependencies] +# external dependencies +anyhow.workspace = true +axum.workspace = true +axum-extra.workspace = true +axum-server.workspace = true +bytes.workspace = true +clap.workspace = true +const-str.workspace = true +fastcrypto.workspace = true +futures.workspace = true +git-version.workspace = true +hex.workspace = true +hyper.workspace = true +itertools.workspace = true +multiaddr = "0.17.0" +once_cell.workspace = true +prometheus.workspace = true +prost.workspace = true +protobuf.workspace = true +rand.workspace = true +reqwest.workspace = true +rustls.workspace = true +rustls-pemfile = "2.1.2" +serde.workspace = true +serde_json.workspace = true +serde_with.workspace = true +serde_yaml.workspace = true +snap.workspace = true +tokio = { workspace = true, features = ["full"] } +tower.workspace = true +tower-http.workspace = true +tracing.workspace = true +url.workspace = true + +# internal dependencies +iota-metrics.workspace = true +iota-tls.workspace = true +iota-types.workspace = true +telemetry-subscribers.workspace = true + +[dev-dependencies] +# external dependencies +axum-server.workspace = true +mime = "0.3" +serde_json.workspace = true +tower.workspace = true + +# internal dependencies +iota-types = { workspace = true, features = ["test-utils"] } + +[build-dependencies] +prost-build = "0.13.1" diff --git a/crates/iota-proxy/build.rs b/crates/iota-proxy/build.rs new file mode 100644 index 00000000000..86a87cdc909 --- /dev/null +++ b/crates/iota-proxy/build.rs @@ -0,0 +1,18 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +use std::io::Result; +fn main() -> Result<()> { + println!("cargo:rerun-if-changed=build.rs"); + println!("cargo:rerun-if-env-changed=BUILD_REMOTE_WRITE"); + + // add this env var to build. you'll need protoc installed locally and a copy of + // the proto files + if option_env!("BUILD_REMOTE_WRITE").is_some() { + prost_build::compile_protos(&["protobufs/remote.proto", "protobufs/types.proto"], &[ + "protobufs/", + ])?; + } + Ok(()) +} diff --git a/crates/iota-proxy/src/admin.rs b/crates/iota-proxy/src/admin.rs new file mode 100644 index 00000000000..3b8e08217f1 --- /dev/null +++ b/crates/iota-proxy/src/admin.rs @@ -0,0 +1,278 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +use std::{fs, io::BufReader, net::SocketAddr, sync::Arc, time::Duration}; + +use anyhow::{Error, Result}; +use axum::{Extension, Router, extract::DefaultBodyLimit, middleware, routing::post}; +use fastcrypto::{ + ed25519::{Ed25519KeyPair, Ed25519PublicKey}, + traits::{KeyPair, ToFromBytes}, +}; +use iota_tls::{ + AllowAll, ClientCertVerifier, IOTA_VALIDATOR_SERVER_NAME, SelfSignedCertificate, TlsAcceptor, + rustls::ServerConfig, +}; +use tokio::signal; +use tower::ServiceBuilder; +use tower_http::{ + LatencyUnit, + timeout::TimeoutLayer, + trace::{DefaultOnFailure, DefaultOnResponse, TraceLayer}, +}; +use tracing::{Level, info}; + +use crate::{ + config::{DynamicPeerValidationConfig, RemoteWriteConfig, StaticPeerValidationConfig}, + handlers::publish_metrics, + histogram_relay::HistogramRelay, + middleware::{expect_content_length, expect_iota_proxy_header, expect_valid_public_key}, + peers::{IotaNodeProvider, IotaPeer}, + var, +}; + +/// Configure our graceful shutdown scenarios +pub async fn shutdown_signal(h: axum_server::Handle) { + let ctrl_c = async { + signal::ctrl_c() + .await + .expect("failed to install Ctrl+C handler"); + }; + + #[cfg(unix)] + let terminate = async { + signal::unix::signal(signal::unix::SignalKind::terminate()) + .expect("failed to install signal handler") + .recv() + .await; + }; + + #[cfg(not(unix))] + let terminate = std::future::pending::<()>(); + + tokio::select! { + _ = ctrl_c => {}, + _ = terminate => {}, + } + + let grace = 30; + info!( + "signal received, starting graceful shutdown, grace period {} seconds, if needed", + &grace + ); + h.graceful_shutdown(Some(Duration::from_secs(grace))) +} + +/// Reqwest client holds the global client for remote_push api calls +/// it also holds the username and password. The client has an underlying +/// connection pool. See reqwest documentation for details +#[derive(Clone)] +pub struct ReqwestClient { + pub client: reqwest::Client, + pub settings: RemoteWriteConfig, +} + +pub fn make_reqwest_client(settings: RemoteWriteConfig, user_agent: &str) -> ReqwestClient { + ReqwestClient { + client: reqwest::Client::builder() + .user_agent(user_agent) + .pool_max_idle_per_host(settings.pool_max_idle_per_host) + .timeout(Duration::from_secs(var!("MIMIR_CLIENT_TIMEOUT", 30))) + .build() + .expect("cannot create reqwest client"), + settings, + } +} + +// Labels are adhoc labels we will inject per our config +#[derive(Clone)] +pub struct Labels { + pub network: String, +} + +/// App will configure our routes. This fn is also used to instrument our tests +pub fn app( + labels: Labels, + client: ReqwestClient, + relay: HistogramRelay, + allower: Option, +) -> Router { + // build our application with a route and our sender mpsc + let mut router = Router::new() + .route("/publish/metrics", post(publish_metrics)) + .route_layer(DefaultBodyLimit::max(var!( + "MAX_BODY_SIZE", + 1024 * 1024 * 5 + ))) + .route_layer(middleware::from_fn(expect_iota_proxy_header)) + .route_layer(middleware::from_fn(expect_content_length)); + if let Some(allower) = allower { + router = router + .route_layer(middleware::from_fn(expect_valid_public_key)) + .layer(Extension(Arc::new(allower))); + } + router + // Enforce on all routes. + // If the request does not complete within the specified timeout it will be aborted + // and a 408 Request Timeout response will be sent. + .layer(TimeoutLayer::new(Duration::from_secs(var!( + "NODE_CLIENT_TIMEOUT", + 20 + )))) + .layer(Extension(relay)) + .layer(Extension(labels)) + .layer(Extension(client)) + .layer( + ServiceBuilder::new().layer( + TraceLayer::new_for_http() + .on_response( + DefaultOnResponse::new() + .level(Level::INFO) + .latency_unit(LatencyUnit::Seconds), + ) + .on_failure( + DefaultOnFailure::new() + .level(Level::ERROR) + .latency_unit(LatencyUnit::Seconds), + ), + ), + ) +} + +/// Server creates our http/https server +pub async fn server( + listener: std::net::TcpListener, + app: Router, + acceptor: Option, +) -> std::io::Result<()> { + // setup our graceful shutdown + let handle = axum_server::Handle::new(); + // Spawn a task to gracefully shutdown server. + tokio::spawn(shutdown_signal(handle.clone())); + + if let Some(verify_peers) = acceptor { + axum_server::Server::from_tcp(listener) + .acceptor(verify_peers) + .handle(handle) + .serve(app.into_make_service_with_connect_info::()) + .await + } else { + axum_server::Server::from_tcp(listener) + .handle(handle) + .serve(app.into_make_service_with_connect_info::()) + .await + } +} + +/// CertKeyPair wraps a self signed certificate and the corresponding public key +pub struct CertKeyPair(pub SelfSignedCertificate, pub Ed25519PublicKey); + +/// Generate server certs for use with peer verification +pub fn generate_self_cert(hostname: String) -> CertKeyPair { + let mut rng = rand::thread_rng(); + let keypair = Ed25519KeyPair::generate(&mut rng); + CertKeyPair( + SelfSignedCertificate::new(keypair.copy().private(), &hostname), + keypair.public().to_owned(), + ) +} + +/// Load a certificate for use by the listening service +fn load_certs(filename: &str) -> Vec> { + let certfile = fs::File::open(filename) + .unwrap_or_else(|e| panic!("cannot open certificate file: {}; {}", filename, e)); + let mut reader = BufReader::new(certfile); + rustls_pemfile::certs(&mut reader) + .collect::, _>>() + .unwrap() +} + +/// Load a private key +fn load_private_key(filename: &str) -> rustls::pki_types::PrivateKeyDer<'static> { + let keyfile = fs::File::open(filename) + .unwrap_or_else(|e| panic!("cannot open private key file {}; {}", filename, e)); + let mut reader = BufReader::new(keyfile); + + loop { + match rustls_pemfile::read_one(&mut reader).expect("cannot parse private key .pem file") { + Some(rustls_pemfile::Item::Pkcs1Key(key)) => return key.into(), + Some(rustls_pemfile::Item::Pkcs8Key(key)) => return key.into(), + Some(rustls_pemfile::Item::Sec1Key(key)) => return key.into(), + None => break, + _ => {} + } + } + + panic!( + "no keys found in {:?} (encrypted keys not supported)", + filename + ); +} + +/// load the static keys we'll use to allow external non-validator nodes to push +/// metrics +fn load_static_peers( + static_peers: Option, +) -> Result, Error> { + let Some(static_peers) = static_peers else { + return Ok(vec![]); + }; + let static_keys = static_peers + .pub_keys + .into_iter() + .map(|spk| { + let peer_id = hex::decode(spk.peer_id).unwrap(); + let public_key = Ed25519PublicKey::from_bytes(peer_id.as_ref()).unwrap(); + let s = IotaPeer { + name: spk.name.clone(), + public_key, + }; + info!( + "loaded static peer: {} public key: {}", + &s.name, &s.public_key, + ); + s + }) + .collect(); + Ok(static_keys) +} + +/// Default allow mode for server, we don't verify clients, everything is +/// accepted +pub fn create_server_cert_default_allow( + hostname: String, +) -> Result { + let CertKeyPair(server_certificate, _) = generate_self_cert(hostname); + + ClientCertVerifier::new(AllowAll, IOTA_VALIDATOR_SERVER_NAME.to_string()).rustls_server_config( + vec![server_certificate.rustls_certificate()], + server_certificate.rustls_private_key(), + ) +} + +/// Verify clients against iota blockchain, clients that are not found in +/// iota_getValidators will be rejected +pub fn create_server_cert_enforce_peer( + dynamic_peers: DynamicPeerValidationConfig, + static_peers: Option, +) -> Result<(ServerConfig, Option), iota_tls::rustls::Error> { + let (Some(certificate_path), Some(private_key_path)) = + (dynamic_peers.certificate_file, dynamic_peers.private_key) + else { + return Err(iota_tls::rustls::Error::General( + "missing certs to initialize server".into(), + )); + }; + let static_peers = load_static_peers(static_peers).map_err(|e| { + iota_tls::rustls::Error::General(format!("unable to load static pub keys: {}", e)) + })?; + let allower = IotaNodeProvider::new(dynamic_peers.url, dynamic_peers.interval, static_peers); + allower.poll_peer_list(); + let c = ClientCertVerifier::new(allower.clone(), IOTA_VALIDATOR_SERVER_NAME.to_string()) + .rustls_server_config( + load_certs(&certificate_path), + load_private_key(&private_key_path), + )?; + Ok((c, Some(allower))) +} diff --git a/crates/iota-proxy/src/config.rs b/crates/iota-proxy/src/config.rs new file mode 100644 index 00000000000..e16304c48be --- /dev/null +++ b/crates/iota-proxy/src/config.rs @@ -0,0 +1,119 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +use core::time::Duration; +use std::net::SocketAddr; + +use anyhow::{Context, Result}; +use serde::{Deserialize, Serialize, de::DeserializeOwned}; +use serde_with::{DurationSeconds, serde_as}; +use tracing::debug; + +#[serde_as] +#[derive(Clone, Debug, Deserialize, Serialize)] +#[serde(rename_all = "kebab-case")] +pub struct ProxyConfig { + pub network: String, + pub listen_address: SocketAddr, + pub remote_write: RemoteWriteConfig, + pub dynamic_peers: DynamicPeerValidationConfig, + pub static_peers: Option, + pub metrics_address: String, + pub histogram_address: String, +} + +#[serde_as] +#[derive(Clone, Debug, Deserialize, Serialize, Default)] +#[serde(rename_all = "kebab-case")] +pub struct RemoteWriteConfig { + // TODO upgrade to https + /// the remote_write url to post data to + pub url: String, + /// username is used for posting data to the remote_write api + pub username: String, + pub password: String, + + /// Sets the maximum idle connection per host allowed in the pool. + /// + #[serde(default = "pool_max_idle_per_host_default")] + pub pool_max_idle_per_host: usize, +} + +/// DynamicPeerValidationConfig controls what iota-node & iota-bridge binaries +/// that are functioning as a validator that we'll speak with. Peer in this case +/// is peers within the consensus committee, for each epoch. This membership is +/// determined dynamically for each epoch via json-rpc calls to a full node. +#[serde_as] +#[derive(Clone, Debug, Deserialize, Serialize)] +#[serde(rename_all = "kebab-case")] +pub struct DynamicPeerValidationConfig { + /// url is the json-rpc url we use to obtain valid peers on the blockchain + pub url: String, + #[serde_as(as = "DurationSeconds")] + pub interval: Duration, + /// if certificate_file and private_key are not provided, we'll create a + /// self-signed cert using this hostname + #[serde(default = "hostname_default")] + pub hostname: Option, + + /// incoming client connections to this proxy will be presented with this + /// pub key please use an absolute path + pub certificate_file: Option, + /// private key for tls + /// please use an absolute path + pub private_key: Option, +} + +/// StaticPeerValidationConfig, unlike the DynamicPeerValidationConfig, is not +/// determined dynamically from rpc calls. It instead searches a local +/// directory for pub keys that we will add to an allow list. +#[serde_as] +#[derive(Clone, Debug, Deserialize, Serialize)] +#[serde(rename_all = "kebab-case")] +pub struct StaticPeerValidationConfig { + pub pub_keys: Vec, +} + +/// StaticPubKey holds a human friendly name, ip and the key file for the pub +/// key if you don't have a valid public routable ip, use an ip from +/// 169.254.0.0/16. +#[serde_as] +#[derive(Clone, Debug, Deserialize, Serialize)] +#[serde(rename_all = "kebab-case")] +pub struct StaticPubKey { + /// friendly name we will see in metrics + pub name: String, + /// the peer_id from a node config file (Ed25519 PublicKey) + pub peer_id: String, +} + +/// the default idle worker per host (reqwest to remote write url call) +fn pool_max_idle_per_host_default() -> usize { + 8 +} + +/// the default hostname we will use if not provided +fn hostname_default() -> Option { + Some("localhost".to_string()) +} + +/// load our config file from a path +pub fn load, T: DeserializeOwned + Serialize>(path: P) -> Result { + let path = path.as_ref(); + debug!("Reading config from {:?}", path); + Ok(serde_yaml::from_reader( + std::fs::File::open(path).context(format!("cannot open {:?}", path))?, + )?) +} + +#[cfg(test)] +mod tests { + use super::*; + #[test] + fn config_load() { + const TEMPLATE: &str = include_str!("./data/config.yaml"); + + let _template: ProxyConfig = serde_yaml::from_str(TEMPLATE).unwrap(); + } +} diff --git a/crates/iota-proxy/src/consumer.rs b/crates/iota-proxy/src/consumer.rs new file mode 100644 index 00000000000..bc336f58021 --- /dev/null +++ b/crates/iota-proxy/src/consumer.rs @@ -0,0 +1,377 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +use std::io::Read; + +use anyhow::Result; +use axum::{body::Bytes, http::StatusCode}; +use bytes::buf::Reader; +use fastcrypto::ed25519::Ed25519PublicKey; +use multiaddr::Multiaddr; +use once_cell::sync::Lazy; +use prometheus::{ + Counter, CounterVec, HistogramVec, + proto::{self, MetricFamily}, + register_counter, register_counter_vec, register_histogram_vec, +}; +use prost::Message; +use protobuf::CodedInputStream; +use tracing::{debug, error}; + +use crate::{admin::ReqwestClient, prom_to_mimir::Mimir, remote_write::WriteRequest}; + +static CONSUMER_OPS_SUBMITTED: Lazy = Lazy::new(|| { + register_counter!( + "consumer_operations_submitted", + "Operations counter for the number of metric family types we submit, excluding histograms, and not the discrete timeseries counts.", + ) + .unwrap() +}); +static CONSUMER_OPS: Lazy = Lazy::new(|| { + register_counter_vec!( + "consumer_operations", + "Operations counters and status from operations performed in the consumer.", + &["operation", "status"] + ) + .unwrap() +}); +static CONSUMER_ENCODE_COMPRESS_DURATION: Lazy = Lazy::new(|| { + register_histogram_vec!( + "protobuf_compression_seconds", + "The time it takes to compress a remote_write payload in seconds.", + &["operation"], + vec![ + 1e-08, 2e-08, 4e-08, 8e-08, 1.6e-07, 3.2e-07, 6.4e-07, 1.28e-06, 2.56e-06, 5.12e-06, + 1.024e-05, 2.048e-05, 4.096e-05, 8.192e-05 + ], + ) + .unwrap() +}); +static CONSUMER_OPERATION_DURATION: Lazy = Lazy::new(|| { + register_histogram_vec!( + "consumer_operations_duration_seconds", + "The time it takes to perform various consumer operations in seconds.", + &["operation"], + vec![ + 0.0008, 0.0016, 0.0032, 0.0064, 0.0128, 0.0256, 0.0512, 0.1024, 0.2048, 0.4096, 0.8192, + 1.0, 1.25, 1.5, 1.75, 2.0, 2.25, 2.5, 2.75, 3.0, 3.25, 3.5, 3.75, 4.0, 4.25, 4.5, 4.75, + 5.0, 5.25, 5.5, 5.75, 6.0, 6.25, 6.5, 6.75, 7.0, 7.25, 7.5, 7.75, 8.0, 8.25, 8.5, 8.75, + 9.0, 9.25, 9.5, 9.75, 10.0, 10.25, 10.5, 10.75, 11.0, 11.25, 11.5, 11.75, 12.0, 12.25, + 12.5, 12.75, 13.0, 13.25, 13.5, 13.75, 14.0, 14.25, 14.5, 14.75, 15.0, 15.25, 15.5, + 15.75, 16.0, 16.25, 16.5, 16.75, 17.0, 17.25, 17.5, 17.75, 18.0, 18.25, 18.5, 18.75, + 19.0, 19.25, 19.5, 19.75, 20.0, 20.25, 20.5, 20.75, 21.0, 21.25, 21.5, 21.75, 22.0, + 22.25, 22.5, 22.75, 23.0, 23.25, 23.5, 23.75, 24.0, 24.25, 24.5, 24.75, 25.0, 26.0, + 27.0, 28.0, 29.0, 30.0 + ], + ) + .unwrap() +}); + +/// NodeMetric holds metadata and a metric payload from the calling node +#[derive(Debug)] +pub struct NodeMetric { + pub peer_addr: Multiaddr, // the sockaddr source address from the incoming request + pub public_key: Ed25519PublicKey, // the public key from the iota blockchain + pub data: Vec, // decoded protobuf of prometheus data +} + +/// The ProtobufDecoder will decode message delimited protobuf messages from +/// prom_model.proto types They are delimited by size, eg a format is such: +/// []byte{size, data, size, data, size, data}, etc etc +pub struct ProtobufDecoder { + buf: Reader, +} + +impl ProtobufDecoder { + pub fn new(buf: Reader) -> Self { + Self { buf } + } + /// parse a delimited buffer of protobufs. this is used to consume data sent + /// from a iota-node + pub fn parse(&mut self) -> Result> { + let timer = CONSUMER_OPERATION_DURATION + .with_label_values(&["decode_len_delim_protobuf"]) + .start_timer(); + let mut result: Vec = vec![]; + while !self.buf.get_ref().is_empty() { + let len = { + let mut is = CodedInputStream::from_buffered_reader(&mut self.buf); + is.read_raw_varint32() + }?; + let mut buf = vec![0; len as usize]; + self.buf.read_exact(&mut buf)?; + result.push(T::parse_from_bytes(&buf)?); + } + timer.observe_duration(); + Ok(result) + } +} + +// populate labels in place for our given metric family data +pub fn populate_labels( + name: String, // host field for grafana agent (from chain data) + network: String, // network name from ansible (via config) + data: Vec, +) -> Vec { + let timer = CONSUMER_OPERATION_DURATION + .with_label_values(&["populate_labels"]) + .start_timer(); + debug!("received metrics from {name}"); + // proto::LabelPair doesn't have pub fields so we can't use + // struct literals to construct + let mut network_label = proto::LabelPair::default(); + network_label.set_name("network".into()); + network_label.set_value(network); + + let mut host_label = proto::LabelPair::default(); + host_label.set_name("host".into()); + host_label.set_value(name); + + let labels = vec![network_label, host_label]; + + let mut data = data; + // add our extra labels to our incoming metric data + for mf in data.iter_mut() { + for m in mf.mut_metric() { + m.mut_label().extend(labels.clone()); + } + } + timer.observe_duration(); + data +} + +fn encode_compress(request: &WriteRequest) -> Result, (StatusCode, &'static str)> { + let observe = || { + let timer = CONSUMER_ENCODE_COMPRESS_DURATION + .with_label_values(&["encode_compress"]) + .start_timer(); + || { + timer.observe_duration(); + } + }(); + let mut buf = Vec::with_capacity(request.encoded_len()); + if request.encode(&mut buf).is_err() { + observe(); + CONSUMER_OPS + .with_label_values(&["encode_compress", "failed"]) + .inc(); + error!("unable to encode prompb to mimirpb"); + return Err(( + StatusCode::INTERNAL_SERVER_ERROR, + "unable to encode prompb to remote_write pb", + )); + }; + + let mut s = snap::raw::Encoder::new(); + let compressed = match s.compress_vec(&buf) { + Ok(compressed) => compressed, + Err(error) => { + observe(); + CONSUMER_OPS + .with_label_values(&["encode_compress", "failed"]) + .inc(); + error!("unable to compress to snappy block format; {error}"); + return Err(( + StatusCode::INTERNAL_SERVER_ERROR, + "unable to compress to snappy block format", + )); + } + }; + observe(); + CONSUMER_OPS + .with_label_values(&["encode_compress", "success"]) + .inc(); + Ok(compressed) +} + +async fn check_response( + request: WriteRequest, + response: reqwest::Response, +) -> Result<(), (StatusCode, &'static str)> { + match response.status() { + reqwest::StatusCode::OK => { + CONSUMER_OPS + .with_label_values(&["check_response", "OK"]) + .inc(); + debug!("({}) SUCCESS: {:?}", reqwest::StatusCode::OK, request); + Ok(()) + } + reqwest::StatusCode::BAD_REQUEST => { + let body = response + .text() + .await + .unwrap_or_else(|_| "response body cannot be decoded".into()); + + // see mimir docs on this error condition. it's not actionable from the proxy + // so we drop it. + if body.contains("err-mimir-sample-out-of-order") { + CONSUMER_OPS + .with_label_values(&["check_response", "BAD_REQUEST"]) + .inc(); + error!("({}) ERROR: {:?}", reqwest::StatusCode::BAD_REQUEST, body); + return Err(( + StatusCode::INTERNAL_SERVER_ERROR, + "IGNORING METRICS due to err-mimir-sample-out-of-order", + )); + } + CONSUMER_OPS + .with_label_values(&["check_response", "INTERNAL_SERVER_ERROR"]) + .inc(); + error!("({}) ERROR: {:?}", reqwest::StatusCode::BAD_REQUEST, body); + Err(( + StatusCode::INTERNAL_SERVER_ERROR, + "unknown bad request error encountered in remote_push", + )) + } + code => { + let body = response + .text() + .await + .unwrap_or_else(|_| "response body cannot be decoded".into()); + CONSUMER_OPS + .with_label_values(&["check_response", "INTERNAL_SERVER_ERROR"]) + .inc(); + error!("({}) ERROR: {:?}", code, body); + Err(( + StatusCode::INTERNAL_SERVER_ERROR, + "unknown error encountered in remote_push", + )) + } + } +} + +async fn convert( + mfs: Vec, +) -> Result, (StatusCode, &'static str)> { + let result = tokio::task::spawn_blocking(|| { + let timer = CONSUMER_OPERATION_DURATION + .with_label_values(&["convert_to_remote_write_task"]) + .start_timer(); + let result = Mimir::from(mfs); + timer.observe_duration(); + result.into_iter() + }) + .await; + + let result = match result { + Ok(v) => v, + Err(err) => { + error!("unable to convert to remote_write; {err}"); + return Err(( + StatusCode::INTERNAL_SERVER_ERROR, + "DROPPING METRICS; unable to convert to remote_write", + )); + } + }; + Ok(result) +} + +/// convert_to_remote_write is an expensive method due to the time it takes to +/// submit to mimir. other operations here are optimized for async, within +/// reason. The post process uses a single connection to mimir and thus incurs +/// the seriliaztion delay for each metric family sent. Possible +/// future optimizations would be to use multiple tcp connections to mimir, +/// within reason. Nevertheless we await on each post of each metric family so +/// it shouldn't block any other async work in a significant way. +pub async fn convert_to_remote_write( + rc: ReqwestClient, + node_metric: NodeMetric, +) -> (StatusCode, &'static str) { + let timer = CONSUMER_OPERATION_DURATION + .with_label_values(&["convert_to_remote_write"]) + .start_timer(); + + let remote_write_protos = match convert(node_metric.data).await { + Ok(v) => v, + Err(err) => { + timer.stop_and_discard(); + return err; + } + }; + + // a counter so we don't iterate the node data 2x + let mut mf_cnt = 0; + for request in remote_write_protos { + mf_cnt += 1; + let compressed = match encode_compress(&request) { + Ok(compressed) => compressed, + Err(error) => return error, + }; + + let response = match rc + .client + .post(rc.settings.url.to_owned()) + .header(reqwest::header::CONTENT_ENCODING, "snappy") + .header(reqwest::header::CONTENT_TYPE, "application/x-protobuf") + .header("X-Prometheus-Remote-Write-Version", "0.1.0") + .basic_auth( + rc.settings.username.to_owned(), + Some(rc.settings.password.to_owned()), + ) + .body(compressed) + .send() + .await + { + Ok(response) => response, + Err(error) => { + CONSUMER_OPS + .with_label_values(&["check_response", "INTERNAL_SERVER_ERROR"]) + .inc(); + error!("DROPPING METRICS due to post error: {error}"); + timer.stop_and_discard(); + return ( + StatusCode::INTERNAL_SERVER_ERROR, + "DROPPING METRICS due to post error", + ); + } + }; + + match check_response(request, response).await { + Ok(_) => (), + Err(err) => { + timer.stop_and_discard(); + return err; + } + } + } + CONSUMER_OPS_SUBMITTED.inc_by(mf_cnt as f64); + timer.observe_duration(); + (StatusCode::CREATED, "created") +} + +#[cfg(test)] +mod tests { + use prometheus::proto; + use protobuf; + + use crate::{ + consumer::populate_labels, + prom_to_mimir::tests::{ + create_histogram, create_labels, create_metric_family, create_metric_histogram, + }, + }; + + #[test] + fn test_populate_labels() { + let mf = create_metric_family( + "test_histogram", + "i'm a help message", + Some(proto::MetricType::HISTOGRAM), + protobuf::RepeatedField::from(vec![create_metric_histogram( + protobuf::RepeatedField::from_vec(create_labels(vec![])), + create_histogram(), + )]), + ); + + let labeled_mf = populate_labels("validator-0".into(), "unittest-network".into(), vec![mf]); + let metric = &labeled_mf[0].get_metric()[0]; + assert_eq!( + metric.get_label(), + &create_labels(vec![ + ("network", "unittest-network"), + ("host", "validator-0"), + ]) + ); + } +} diff --git a/crates/iota-proxy/src/data/config.yaml b/crates/iota-proxy/src/data/config.yaml new file mode 100644 index 00000000000..a5265bd44fb --- /dev/null +++ b/crates/iota-proxy/src/data/config.yaml @@ -0,0 +1,18 @@ +network: joenet +listen-address: 127.0.0.1:8080 +remote-write: + url: http://unittest.abcd.io/api/v1/push + username: foo + password: fooman +dynamic-peers: + url: http://127.0.0.1:9000 + interval: 30 + certificate-file: /opt/joeman/fullchain.pem + private-key: /opt/joeman/privkey.pem +static-peers: + pub-keys: + - name: joeman + p2p-address: 169.254.0.1 + peer-id: 4e2f113e61784fdcd611650f36595db8f79e9420319f42a5b571dc2f2b295af2 +metrics-address: localhost:9184 +histogram-address: localhost:9185 diff --git a/crates/iota-proxy/src/handlers.rs b/crates/iota-proxy/src/handlers.rs new file mode 100644 index 00000000000..8c1ef98d23e --- /dev/null +++ b/crates/iota-proxy/src/handlers.rs @@ -0,0 +1,72 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +use std::net::SocketAddr; + +use axum::{ + extract::{ConnectInfo, Extension}, + http::StatusCode, +}; +use multiaddr::Multiaddr; +use once_cell::sync::Lazy; +use prometheus::{CounterVec, HistogramVec, register_counter_vec, register_histogram_vec}; + +use crate::{ + admin::{Labels, ReqwestClient}, + consumer::{NodeMetric, convert_to_remote_write, populate_labels}, + histogram_relay::HistogramRelay, + middleware::LenDelimProtobuf, + peers::IotaPeer, +}; + +static HANDLER_HITS: Lazy = Lazy::new(|| { + register_counter_vec!("http_handler_hits", "Number of HTTP requests made.", &[ + "handler", "remote" + ]) + .unwrap() +}); + +static HTTP_HANDLER_DURATION: Lazy = Lazy::new(|| { + register_histogram_vec!( + "http_handler_duration_seconds", + "The HTTP request latencies in seconds.", + &["handler", "remote"], + vec![ + 1.0, 1.25, 1.5, 1.75, 2.0, 2.25, 2.5, 2.75, 3.0, 3.25, 3.5, 3.75, 4.0, 4.25, 4.5, 4.75, + 5.0, 6.0, 7.0, 8.0, 9.0, 10.0, 11.0, 12.0, 13.0, 14.0, 15.0 + ], + ) + .unwrap() +}); + +/// Publish handler which receives metrics from nodes. Nodes will call us at +/// this endpoint and we relay them to the upstream tsdb +/// +/// Clients will receive a response after successfully relaying the metrics +/// upstream +pub async fn publish_metrics( + Extension(labels): Extension, + Extension(client): Extension, + ConnectInfo(addr): ConnectInfo, + Extension(IotaPeer { name, public_key }): Extension, + Extension(relay): Extension, + LenDelimProtobuf(data): LenDelimProtobuf, +) -> (StatusCode, &'static str) { + HANDLER_HITS + .with_label_values(&["publish_metrics", &name]) + .inc(); + let timer = HTTP_HANDLER_DURATION + .with_label_values(&["publish_metrics", &name]) + .start_timer(); + let data = populate_labels(name, labels.network, data); + relay.submit(data.clone()); + let response = convert_to_remote_write(client.clone(), NodeMetric { + data, + peer_addr: Multiaddr::from(addr.ip()), + public_key, + }) + .await; + timer.observe_duration(); + response +} diff --git a/crates/iota-proxy/src/histogram_relay.rs b/crates/iota-proxy/src/histogram_relay.rs new file mode 100644 index 00000000000..27bb9559d6a --- /dev/null +++ b/crates/iota-proxy/src/histogram_relay.rs @@ -0,0 +1,266 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +use std::{ + collections::VecDeque, + net::TcpListener, + sync::{Arc, Mutex}, + time::{SystemTime, UNIX_EPOCH}, +}; + +use anyhow::{Result, bail}; +use axum::{Router, extract::Extension, http::StatusCode, routing::get}; +use once_cell::sync::Lazy; +use prometheus::{ + CounterVec, HistogramVec, + proto::{Metric, MetricFamily}, + register_counter_vec, register_histogram_vec, +}; +use tower::ServiceBuilder; +use tower_http::{ + LatencyUnit, + trace::{DefaultOnResponse, TraceLayer}, +}; +use tracing::{Level, info}; + +use crate::var; + +const METRICS_ROUTE: &str = "/metrics"; + +static RELAY_PRESSURE: Lazy = Lazy::new(|| { + register_counter_vec!( + "relay_pressure", + "HistogramRelay's number of metric families submitted, exported, overflowed to/from the queue.", + &["histogram_relay"] + ) + .unwrap() +}); +static RELAY_DURATION: Lazy = Lazy::new(|| { + register_histogram_vec!( + "relay_duration_seconds", + "HistogramRelay's submit/export fn latencies in seconds.", + &["histogram_relay"], + vec![ + 0.0008, 0.0016, 0.0032, 0.0064, 0.0128, 0.0256, 0.0512, 0.1024, 0.2048, 0.4096, 0.8192, + 1.0, 1.25, 1.5, 1.75, 2.0, 4.0, 8.0, 10.0, 12.5, 15.0 + ], + ) + .unwrap() +}); + +// Creates a new http server that has as a sole purpose to expose +// and endpoint that prometheus agent can use to poll for the metrics. +// A RegistryService is returned that can be used to get access in prometheus +// Registries. +pub fn start_prometheus_server(listener: TcpListener) -> HistogramRelay { + let relay = HistogramRelay::new(); + let app = Router::new() + .route(METRICS_ROUTE, get(metrics)) + .layer(Extension(relay.clone())) + .layer( + ServiceBuilder::new().layer( + TraceLayer::new_for_http().on_response( + DefaultOnResponse::new() + .level(Level::INFO) + .latency_unit(LatencyUnit::Seconds), + ), + ), + ); + + tokio::spawn(async move { + listener.set_nonblocking(true).unwrap(); + let listener = tokio::net::TcpListener::from_std(listener).unwrap(); + axum::serve(listener, app).await.unwrap(); + }); + relay +} + +async fn metrics(Extension(relay): Extension) -> (StatusCode, String) { + let Ok(expformat) = relay.export() else { + return ( + StatusCode::INTERNAL_SERVER_ERROR, + "unable to pop metrics from HistogramRelay".into(), + ); + }; + (StatusCode::OK, expformat) +} + +struct Wrapper(i64, Vec); + +#[derive(Clone)] +pub struct HistogramRelay(Arc>>); + +impl Default for HistogramRelay { + fn default() -> Self { + HistogramRelay(Arc::new(Mutex::new(VecDeque::new()))) + } +} +impl HistogramRelay { + pub fn new() -> Self { + Self::default() + } + /// submit will take metric family submissions and store them for scraping + /// in doing so, it will also wrap each entry in a timestamp which will be + /// use for pruning old entries on each submission call. this may not be + /// ideal long term. + pub fn submit(&self, data: Vec) { + RELAY_PRESSURE.with_label_values(&["submit"]).inc(); + let timer = RELAY_DURATION.with_label_values(&["submit"]).start_timer(); + // represents a collection timestamp + let timestamp_secs = SystemTime::now() + .duration_since(UNIX_EPOCH) + .unwrap() + .as_secs() as i64; + let mut queue = self + .0 + .lock() + .expect("couldn't get mut lock on HistogramRelay"); + queue.retain(|v| { + // 5 mins is the max time in the queue allowed + if (timestamp_secs - v.0) < var!("MAX_QUEUE_TIME_SECS", 300) { + return true; + } + RELAY_PRESSURE.with_label_values(&["overflow"]).inc(); + false + }); // drain anything 5 mins or older + + // filter out our histograms from normal metrics + let data: Vec = extract_histograms(data).collect(); + RELAY_PRESSURE + .with_label_values(&["submitted"]) + .inc_by(data.len() as f64); + queue.push_back(Wrapper(timestamp_secs, data)); + timer.observe_duration(); + } + pub fn export(&self) -> Result { + RELAY_PRESSURE.with_label_values(&["export"]).inc(); + let timer = RELAY_DURATION.with_label_values(&["export"]).start_timer(); + // totally drain all metrics whenever we get a scrape request from the metrics + // handler + let mut queue = self + .0 + .lock() + .expect("couldn't get mut lock on HistogramRelay"); + + let data: Vec = queue.drain(..).collect(); + let mut histograms = vec![]; + for mf in data { + histograms.extend(mf.1); + } + info!( + "histogram queue drained {} items; remaining count {}", + histograms.len(), + queue.len() + ); + + let encoder = prometheus::TextEncoder::new(); + let string = match encoder.encode_to_string(&histograms) { + Ok(s) => s, + Err(error) => bail!("{error}"), + }; + RELAY_PRESSURE + .with_label_values(&["exported"]) + .inc_by(histograms.len() as f64); + timer.observe_duration(); + Ok(string) + } +} + +fn extract_histograms(data: Vec) -> impl Iterator { + data.into_iter().filter_map(|mf| { + let metrics = mf.get_metric().iter().filter_map(|m| { + if !m.has_histogram() { + return None; + } + let mut v = Metric::default(); + v.set_label(protobuf::RepeatedField::from_slice(m.get_label())); + v.set_histogram(m.get_histogram().to_owned()); + v.set_timestamp_ms(m.get_timestamp_ms()); + Some(v) + }); + + let only_histograms = protobuf::RepeatedField::from_iter(metrics); + if only_histograms.len() == 0 { + return None; + } + + let mut v = MetricFamily::default(); + v.set_name(mf.get_name().to_owned()); + v.set_help(mf.get_help().to_owned()); + v.set_field_type(mf.get_field_type()); + v.set_metric(only_histograms); + Some(v) + }) +} + +#[cfg(test)] +mod tests { + use prometheus::proto; + use protobuf; + + use crate::{ + histogram_relay::extract_histograms, + prom_to_mimir::tests::{ + create_counter, create_histogram, create_labels, create_metric_counter, + create_metric_family, create_metric_histogram, + }, + }; + + #[test] + fn filter_histograms() { + struct Test { + data: Vec, + expected: Vec, + } + + let tests = vec![ + Test { + data: vec![create_metric_family( + "test_counter", + "i'm a help message", + Some(proto::MetricType::GAUGE), + protobuf::RepeatedField::from(vec![create_metric_counter( + protobuf::RepeatedField::from_vec(create_labels(vec![ + ("host", "local-test-validator"), + ("network", "unittest-network"), + ])), + create_counter(2046.0), + )]), + )], + expected: vec![], + }, + Test { + data: vec![create_metric_family( + "test_histogram", + "i'm a help message", + Some(proto::MetricType::HISTOGRAM), + protobuf::RepeatedField::from(vec![create_metric_histogram( + protobuf::RepeatedField::from_vec(create_labels(vec![ + ("host", "local-test-validator"), + ("network", "unittest-network"), + ])), + create_histogram(), + )]), + )], + expected: vec![create_metric_family( + "test_histogram", + "i'm a help message", + Some(proto::MetricType::HISTOGRAM), + protobuf::RepeatedField::from(vec![create_metric_histogram( + protobuf::RepeatedField::from_vec(create_labels(vec![ + ("host", "local-test-validator"), + ("network", "unittest-network"), + ])), + create_histogram(), + )]), + )], + }, + ]; + + for test in tests { + let extracted: Vec = extract_histograms(test.data).collect(); + assert_eq!(extracted, test.expected); + } + } +} diff --git a/crates/iota-proxy/src/lib.rs b/crates/iota-proxy/src/lib.rs new file mode 100644 index 00000000000..268c0b59982 --- /dev/null +++ b/crates/iota-proxy/src/lib.rs @@ -0,0 +1,323 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +pub mod admin; +pub mod config; +pub mod consumer; +pub mod handlers; +pub mod histogram_relay; +pub mod metrics; +pub mod middleware; +pub mod peers; +pub mod prom_to_mimir; +pub mod remote_write; + +/// var extracts environment variables at runtime with a default fallback value +/// if a default is not provided, the value is simply an empty string if not +/// found This function will return the provided default if env::var cannot find +/// the key or if the key is somehow malformed. +#[macro_export] +macro_rules! var { + ($key:expr) => { + match std::env::var($key) { + Ok(val) => val, + Err(_) => "".into(), + } + }; + ($key:expr, $default:expr) => { + match std::env::var($key) { + Ok(val) => val.parse::<_>().unwrap(), + Err(_) => $default, + } + }; +} + +#[cfg(test)] +mod tests { + use std::{net::TcpListener, time::Duration}; + + use axum::{Router, http::StatusCode, routing::post}; + use iota_tls::{ClientCertVerifier, TlsAcceptor}; + use prometheus::{Encoder, PROTOBUF_FORMAT}; + use protobuf::RepeatedField; + + use super::*; + use crate::{ + admin::{CertKeyPair, Labels}, + config::RemoteWriteConfig, + histogram_relay::HistogramRelay, + peers::IotaNodeProvider, + prom_to_mimir::tests::*, + }; + + async fn run_dummy_remote_write(listener: TcpListener) { + /// i accept everything, send me the trash + async fn handler() -> StatusCode { + StatusCode::OK + } + + // build our application with a route + let app = Router::new().route("/v1/push", post(handler)); + + // run it + listener.set_nonblocking(true).unwrap(); + let listener = tokio::net::TcpListener::from_std(listener).unwrap(); + axum::serve(listener, app).await.unwrap(); + } + + async fn run_dummy_remote_write_very_slow(listener: TcpListener) { + /// i accept everything, send me the trash, but i will sleep and never + /// return before a timeout this is for testing slow clients and + /// this is the easiest way to do so without adding a special + /// route in the server to do so + async fn handler() -> StatusCode { + // Simulate a route that hangs while waiting for a client to send data + // but the server itself doesn't delay its processing + tokio::time::sleep(Duration::from_secs(60)).await; // A very long sleep + StatusCode::OK + } + + // build our application with a route + let app = Router::new().route("/v1/push", post(handler)); + + // run it + listener.set_nonblocking(true).unwrap(); + let listener = tokio::net::TcpListener::from_std(listener).unwrap(); + axum::serve(listener, app).await.unwrap(); + } + + /// test_axum_acceptor is a basic e2e test that creates a mock remote_write + /// post endpoint and has a simple iota-node client that posts data to + /// the proxy using the protobuf format. The server processes this data + /// and sends it to the mock remote_write which accepts everything. Future + /// work is to make this more robust and expand the scope of coverage, + /// probabaly moving this test elsewhere and renaming it. + #[tokio::test] + async fn test_axum_acceptor() { + // generate self-signed certificates + let CertKeyPair(client_priv_cert, client_pub_key) = + admin::generate_self_cert("iota".into()); + let CertKeyPair(server_priv_cert, _) = admin::generate_self_cert("localhost".into()); + + // create a fake rpc server + let dummy_remote_write_listener = std::net::TcpListener::bind("localhost:0").unwrap(); + let dummy_remote_write_address = dummy_remote_write_listener.local_addr().unwrap(); + let dummy_remote_write_url = format!( + "http://localhost:{}/v1/push", + dummy_remote_write_address.port() + ); + + let _dummy_remote_write = + tokio::spawn(async move { run_dummy_remote_write(dummy_remote_write_listener).await }); + + // init the tls config and allower + let mut allower = IotaNodeProvider::new("".into(), Duration::from_secs(30), vec![]); + let tls_config = ClientCertVerifier::new( + allower.clone(), + iota_tls::IOTA_VALIDATOR_SERVER_NAME.to_string(), + ) + .rustls_server_config( + vec![server_priv_cert.rustls_certificate()], + server_priv_cert.rustls_private_key(), + ) + .unwrap(); + + let client = admin::make_reqwest_client( + RemoteWriteConfig { + url: dummy_remote_write_url.to_owned(), + username: "bar".into(), + password: "foo".into(), + ..Default::default() + }, + "dummy user agent", + ); + + let app = admin::app( + Labels { + network: "unittest-network".into(), + }, + client, + HistogramRelay::new(), + Some(allower.clone()), + ); + + let listener = std::net::TcpListener::bind("localhost:0").unwrap(); + let server_address = listener.local_addr().unwrap(); + let server_url = format!( + "https://localhost:{}/publish/metrics", + server_address.port() + ); + + let acceptor = TlsAcceptor::new(tls_config); + let _server = tokio::spawn(async move { + admin::server(listener, app, Some(acceptor)).await.unwrap(); + }); + + // build a client + let client = reqwest::Client::builder() + .add_root_certificate(server_priv_cert.reqwest_certificate()) + .identity(client_priv_cert.reqwest_identity()) + .https_only(true) + .build() + .unwrap(); + + // Client request is rejected because it isn't in the allowlist + client.get(&server_url).send().await.unwrap_err(); + + // Insert the client's public key into the allowlist and verify the request is + // successful + allower + .get_mut() + .write() + .unwrap() + .insert(client_pub_key.to_owned(), peers::IotaPeer { + name: "some-node".into(), + public_key: client_pub_key.to_owned(), + }); + + let mf = create_metric_family( + "foo_metric", + "some help this is", + None, + RepeatedField::from_vec(vec![create_metric_counter( + RepeatedField::from_vec(create_labels(vec![("some", "label")])), + create_counter(2046.0), + )]), + ); + + let mut buf = vec![]; + let encoder = prometheus::ProtobufEncoder::new(); + encoder.encode(&[mf], &mut buf).unwrap(); + + let res = client + .post(&server_url) + .header(reqwest::header::CONTENT_TYPE, PROTOBUF_FORMAT) + .body(buf) + .send() + .await + .expect("expected a successful post with a self-signed certificate"); + let status = res.status(); + let body = res.text().await.unwrap(); + assert_eq!("created", body); + assert_eq!(status, reqwest::StatusCode::CREATED); + } + + /// this is a long test to ensure we are timing out clients that are slow + #[tokio::test] + async fn test_client_timeout() { + // generate self-signed certificates + let CertKeyPair(client_priv_cert, client_pub_key) = + admin::generate_self_cert("iota".into()); + let CertKeyPair(server_priv_cert, _) = admin::generate_self_cert("localhost".into()); + + // create a fake rpc server + let dummy_remote_write_listener = std::net::TcpListener::bind("localhost:0").unwrap(); + let dummy_remote_write_address = dummy_remote_write_listener.local_addr().unwrap(); + let dummy_remote_write_url = format!( + "http://localhost:{}/v1/push", + dummy_remote_write_address.port() + ); + + let _dummy_remote_write = tokio::spawn(async move { + run_dummy_remote_write_very_slow(dummy_remote_write_listener).await + }); + + // init the tls config and allower + let mut allower = IotaNodeProvider::new("".into(), Duration::from_secs(30), vec![]); + let tls_config = ClientCertVerifier::new( + allower.clone(), + iota_tls::IOTA_VALIDATOR_SERVER_NAME.to_string(), + ) + .rustls_server_config( + vec![server_priv_cert.rustls_certificate()], + server_priv_cert.rustls_private_key(), + ) + .unwrap(); + + let client = admin::make_reqwest_client( + RemoteWriteConfig { + url: dummy_remote_write_url.to_owned(), + username: "bar".into(), + password: "foo".into(), + ..Default::default() + }, + "dummy user agent", + ); + + // this will affect other tests if they are run in parallel, but we only have + // two tests, so it shouldn't be an issue (yet) even still, the other + // tests complete very fast so those tests would also need to slow down by + // orders and orders to be bothered by this env var + std::env::set_var("NODE_CLIENT_TIMEOUT", "5"); + + let app = admin::app( + Labels { + network: "unittest-network".into(), + }, + client, + HistogramRelay::new(), + Some(allower.clone()), + ); + + let listener = std::net::TcpListener::bind("localhost:0").unwrap(); + let server_address = listener.local_addr().unwrap(); + let server_url = format!( + "https://localhost:{}/publish/metrics", + server_address.port() + ); + + let acceptor = TlsAcceptor::new(tls_config); + let _server = tokio::spawn(async move { + admin::server(listener, app, Some(acceptor)).await.unwrap(); + }); + + // build a client + let client = reqwest::Client::builder() + .add_root_certificate(server_priv_cert.reqwest_certificate()) + .identity(client_priv_cert.reqwest_identity()) + .https_only(true) + .build() + .unwrap(); + + // Client request is rejected because it isn't in the allowlist + client.get(&server_url).send().await.unwrap_err(); + + // Insert the client's public key into the allowlist and verify the request is + // successful + allower + .get_mut() + .write() + .unwrap() + .insert(client_pub_key.to_owned(), peers::IotaPeer { + name: "some-node".into(), + public_key: client_pub_key.to_owned(), + }); + + let mf = create_metric_family( + "foo_metric", + "some help this is", + None, + RepeatedField::from_vec(vec![create_metric_counter( + RepeatedField::from_vec(create_labels(vec![("some", "label")])), + create_counter(2046.0), + )]), + ); + + let mut buf = vec![]; + let encoder = prometheus::ProtobufEncoder::new(); + encoder.encode(&[mf], &mut buf).unwrap(); + + let res = client + .post(&server_url) + .header(reqwest::header::CONTENT_TYPE, PROTOBUF_FORMAT) + .body(buf) + .send() + .await + .expect("expected a successful post with a self-signed certificate"); + let status = res.status(); + assert_eq!(status, StatusCode::REQUEST_TIMEOUT); + // Clean up the environment variable + std::env::remove_var("NODE_CLIENT_TIMEOUT"); + } +} diff --git a/crates/iota-proxy/src/main.rs b/crates/iota-proxy/src/main.rs new file mode 100644 index 00000000000..68beb3901ef --- /dev/null +++ b/crates/iota-proxy/src/main.rs @@ -0,0 +1,115 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +use std::env; + +use anyhow::Result; +use clap::Parser; +use iota_proxy::{ + admin::{ + Labels, app, create_server_cert_default_allow, create_server_cert_enforce_peer, + make_reqwest_client, server, + }, + config::{ProxyConfig, load}, + histogram_relay, metrics, +}; +use iota_tls::TlsAcceptor; +use telemetry_subscribers::TelemetryConfig; +use tracing::info; + +// WARNING!!! +// +// Do not move or use similar logic to generate git revision information outside +// of a binary entry point (e.g. main.rs). Placing the below logic into a +// library can result in unessesary builds. +const GIT_REVISION: &str = { + if let Some(revision) = option_env!("GIT_REVISION") { + revision + } else { + git_version::git_version!( + args = ["--always", "--abbrev=12", "--dirty", "--exclude", "*"], + fallback = "DIRTY" + ) + } +}; + +// VERSION mimics what other iota binaries use for the same const +pub const VERSION: &str = const_str::concat!(env!("CARGO_PKG_VERSION"), "-", GIT_REVISION); + +/// user agent we use when posting to mimir +static APP_USER_AGENT: &str = const_str::concat!( + env!("CARGO_PKG_NAME"), + "/", + env!("CARGO_PKG_VERSION"), + "/", + VERSION +); + +#[derive(Parser, Debug)] +#[command(name = env!("CARGO_BIN_NAME"), rename_all = "kebab-case", version = VERSION)] +struct Args { + #[arg( + long, + short, + default_value = "./iota-proxy.yaml", + help = "Specify the config file path to use" + )] + config: String, +} + +#[tokio::main] +async fn main() -> Result<()> { + let (_guard, _handle) = TelemetryConfig::new().init(); + + let args = Args::parse(); + + let config: ProxyConfig = load(args.config)?; + + info!( + "listen on {:?} send to {:?}", + config.listen_address, config.remote_write.url + ); + + let listener = std::net::TcpListener::bind(config.listen_address).unwrap(); + + let (tls_config, allower) = + // we'll only use the dynamic peers in some cases - it makes little sense to run with the static's + // since this first mode allows all. + if config.dynamic_peers.certificate_file.is_none() || config.dynamic_peers.private_key.is_none() { + ( + create_server_cert_default_allow(config.dynamic_peers.hostname.unwrap()) + .expect("unable to create self-signed server cert"), + None, + ) + } else { + create_server_cert_enforce_peer(config.dynamic_peers, config.static_peers) + .expect("unable to create tls server config") + }; + let histogram_listener = std::net::TcpListener::bind(config.histogram_address).unwrap(); + let metrics_listener = std::net::TcpListener::bind(config.metrics_address).unwrap(); + let acceptor = TlsAcceptor::new(tls_config); + let client = make_reqwest_client(config.remote_write, APP_USER_AGENT); + let histogram_relay = histogram_relay::start_prometheus_server(histogram_listener); + let registry_service = metrics::start_prometheus_server(metrics_listener); + let prometheus_registry = registry_service.default_registry(); + prometheus_registry + .register(iota_metrics::uptime_metric( + "iota-proxy", + VERSION, + "unavailable", + )) + .unwrap(); + let app = app( + Labels { + network: config.network, + }, + client, + histogram_relay, + allower, + ); + + server(listener, app, Some(acceptor)).await.unwrap(); + + Ok(()) +} diff --git a/crates/iota-proxy/src/metrics.rs b/crates/iota-proxy/src/metrics.rs new file mode 100644 index 00000000000..ab3f8a73e43 --- /dev/null +++ b/crates/iota-proxy/src/metrics.rs @@ -0,0 +1,128 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +use std::{ + net::TcpListener, + sync::{Arc, RwLock}, +}; + +use axum::{Router, extract::Extension, http::StatusCode, routing::get}; +use iota_metrics::RegistryService; +use prometheus::{Registry, TextEncoder}; +use tower::ServiceBuilder; +use tower_http::{ + LatencyUnit, + trace::{DefaultOnResponse, TraceLayer}, +}; +use tracing::Level; + +const METRICS_ROUTE: &str = "/metrics"; +const POD_HEALTH_ROUTE: &str = "/pod_health"; + +type HealthCheckMetrics = Arc>; + +/// Do not access struct members without using HealthCheckMetrics to arc+mutex +#[derive(Debug)] +struct HealthCheck { + // eg; consumer_operations_submitted{...} + consumer_operations_submitted: f64, +} + +/// HealthCheck contains fields we believe are interesting that say whether this +/// pod should be considered health. do not use w/o using an arc+mutex +impl HealthCheck { + fn new() -> Self { + Self { + consumer_operations_submitted: 0.0, + } + } +} + +// Creates a new http server that has as a sole purpose to expose +// and endpoint that prometheus agent can use to poll for the metrics. +// A RegistryService is returned that can be used to get access in prometheus +// Registries. +pub fn start_prometheus_server(listener: TcpListener) -> RegistryService { + let registry = Registry::new(); + + let registry_service = RegistryService::new(registry); + + let pod_health_data = Arc::new(RwLock::new(HealthCheck::new())); + + let app = Router::new() + .route(METRICS_ROUTE, get(metrics)) + .route(POD_HEALTH_ROUTE, get(pod_health)) + .layer(Extension(registry_service.clone())) + .layer(Extension(pod_health_data.clone())) + .layer( + ServiceBuilder::new().layer( + TraceLayer::new_for_http().on_response( + DefaultOnResponse::new() + .level(Level::INFO) + .latency_unit(LatencyUnit::Seconds), + ), + ), + ); + + tokio::spawn(async move { + listener.set_nonblocking(true).unwrap(); + let listener = tokio::net::TcpListener::from_std(listener).unwrap(); + axum::serve(listener, app).await.unwrap(); + }); + + registry_service +} + +// DO NOT remove this handler, it is not compatible with the +// iota_metrics::metric equivalent +async fn metrics( + Extension(registry_service): Extension, + Extension(pod_health): Extension, +) -> (StatusCode, String) { + let mut metric_families = registry_service.gather_all(); + metric_families.extend(prometheus::gather()); + + if let Some(consumer_operations_submitted) = metric_families + .iter() + .filter_map(|v| { + if v.get_name() == "consumer_operations_submitted" { + // Expecting one metric, so return the first one, as it is the only one + v.get_metric().first().map(|m| m.get_counter().get_value()) + } else { + None + } + }) + .next() + { + pod_health + .write() + .expect("unable to write to pod health metrics") + .consumer_operations_submitted = consumer_operations_submitted; + }; + match TextEncoder.encode_to_string(&metric_families) { + Ok(metrics) => (StatusCode::OK, metrics), + Err(error) => ( + StatusCode::INTERNAL_SERVER_ERROR, + format!("unable to encode metrics: {error}"), + ), + } +} + +/// pod_health is called by k8s to know if this service is correctly processing +/// data +async fn pod_health(Extension(pod_health): Extension) -> (StatusCode, String) { + let consumer_operations_submitted = pod_health + .read() + .expect("unable to read pod health metrics") + .consumer_operations_submitted; + + if consumer_operations_submitted > 0.0 { + (StatusCode::OK, consumer_operations_submitted.to_string()) + } else { + ( + StatusCode::SERVICE_UNAVAILABLE, + consumer_operations_submitted.to_string(), + ) + } +} diff --git a/crates/iota-proxy/src/middleware.rs b/crates/iota-proxy/src/middleware.rs new file mode 100644 index 00000000000..708532c32f9 --- /dev/null +++ b/crates/iota-proxy/src/middleware.rs @@ -0,0 +1,165 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +use std::sync::Arc; + +use axum::{ + async_trait, + body::{Body, Bytes}, + extract::{Extension, FromRequest}, + http::{Request, StatusCode}, + middleware::Next, + response::Response, +}; +use axum_extra::{ + headers::{ContentLength, ContentType}, + typed_header::TypedHeader, +}; +use bytes::Buf; +use hyper::header::CONTENT_ENCODING; +use iota_tls::TlsConnectionInfo; +use once_cell::sync::Lazy; +use prometheus::{CounterVec, proto::MetricFamily, register_counter_vec}; +use tracing::error; + +use crate::{consumer::ProtobufDecoder, peers::IotaNodeProvider}; + +static MIDDLEWARE_OPS: Lazy = Lazy::new(|| { + register_counter_vec!( + "middleware_operations", + "Operations counters and status for axum middleware.", + &["operation", "status"] + ) + .unwrap() +}); + +static MIDDLEWARE_HEADERS: Lazy = Lazy::new(|| { + register_counter_vec!( + "middleware_headers", + "Operations counters and status for axum middleware.", + &["header", "value"] + ) + .unwrap() +}); + +/// we expect iota-node to send us an http header content-length encoding. +pub async fn expect_content_length( + TypedHeader(content_length): TypedHeader, + request: Request, + next: Next, +) -> Result { + MIDDLEWARE_HEADERS.with_label_values(&["content-length", &format!("{}", content_length.0)]); + Ok(next.run(request).await) +} + +/// we expect iota-node to send us an http header content-type encoding. +pub async fn expect_iota_proxy_header( + TypedHeader(content_type): TypedHeader, + request: Request, + next: Next, +) -> Result { + match format!("{content_type}").as_str() { + prometheus::PROTOBUF_FORMAT => Ok(next.run(request).await), + ct => { + error!("invalid content-type; {ct}"); + MIDDLEWARE_OPS + .with_label_values(&["expect_iota_proxy_header", "invalid-content-type"]) + .inc(); + Err((StatusCode::BAD_REQUEST, "invalid content-type header")) + } + } +} + +/// we expect that calling iota-nodes are known on the blockchain and we enforce +/// their pub key tls creds here +pub async fn expect_valid_public_key( + Extension(allower): Extension>, + Extension(tls_connect_info): Extension, + mut request: Request, + next: Next, +) -> Result { + let Some(public_key) = tls_connect_info.public_key() else { + error!("unable to obtain public key from connecting client"); + MIDDLEWARE_OPS + .with_label_values(&["expect_valid_public_key", "missing-public-key"]) + .inc(); + return Err((StatusCode::FORBIDDEN, "unknown clients are not allowed")); + }; + let Some(peer) = allower.get(public_key) else { + error!("node with unknown pub key tried to connect {}", public_key); + MIDDLEWARE_OPS + .with_label_values(&[ + "expect_valid_public_key", + "unknown-validator-connection-attempt", + ]) + .inc(); + return Err((StatusCode::FORBIDDEN, "unknown clients are not allowed")); + }; + request.extensions_mut().insert(peer); + Ok(next.run(request).await) +} + +// extractor that shows how to consume the request body upfront +#[derive(Debug)] +pub struct LenDelimProtobuf(pub Vec); + +#[async_trait] +impl FromRequest for LenDelimProtobuf +where + S: Send + Sync, +{ + type Rejection = (StatusCode, String); + + async fn from_request( + req: Request, + state: &S, + ) -> Result { + let should_be_snappy = req + .headers() + .get(CONTENT_ENCODING) + .map(|v| v.as_bytes() == b"snappy") + .unwrap_or(false); + + let body = Bytes::from_request(req, state).await.map_err(|e| { + let msg = format!("error extracting bytes; {e}"); + error!(msg); + MIDDLEWARE_OPS + .with_label_values(&["LenDelimProtobuf_from_request", "unable-to-extract-bytes"]) + .inc(); + (e.status(), msg) + })?; + + let intermediate = if should_be_snappy { + let mut s = snap::raw::Decoder::new(); + let decompressed = s.decompress_vec(&body).map_err(|e| { + let msg = format!("unable to decode snappy encoded protobufs; {e}"); + error!(msg); + MIDDLEWARE_OPS + .with_label_values(&[ + "LenDelimProtobuf_decompress_vec", + "unable-to-decode-snappy", + ]) + .inc(); + (StatusCode::BAD_REQUEST, msg) + })?; + Bytes::from(decompressed).reader() + } else { + body.reader() + }; + + let mut decoder = ProtobufDecoder::new(intermediate); + let decoded = decoder.parse::().map_err(|e| { + let msg = format!("unable to decode len deliminated protobufs; {e}"); + error!(msg); + MIDDLEWARE_OPS + .with_label_values(&[ + "LenDelimProtobuf_from_request", + "unable-to-decode-protobufs", + ]) + .inc(); + (StatusCode::BAD_REQUEST, msg) + })?; + Ok(Self(decoded)) + } +} diff --git a/crates/iota-proxy/src/peers.rs b/crates/iota-proxy/src/peers.rs new file mode 100644 index 00000000000..f9c47f86928 --- /dev/null +++ b/crates/iota-proxy/src/peers.rs @@ -0,0 +1,716 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +use std::{ + collections::{BTreeMap, HashMap}, + sync::{Arc, RwLock}, + time::Duration, +}; + +use anyhow::{Context, Result, bail}; +use fastcrypto::{ + ed25519::Ed25519PublicKey, + encoding::{Base64, Encoding}, + traits::ToFromBytes, +}; +use futures::stream::{self, StreamExt}; +use iota_tls::Allower; +use iota_types::{ + base_types::IotaAddress, bridge::BridgeSummary, + iota_system_state::iota_system_state_summary::IotaSystemStateSummary, +}; +use once_cell::sync::Lazy; +use prometheus::{CounterVec, HistogramVec, register_counter_vec, register_histogram_vec}; +use serde::Deserialize; +use tracing::{debug, error, info, warn}; +use url::Url; + +static JSON_RPC_STATE: Lazy = Lazy::new(|| { + register_counter_vec!( + "json_rpc_state", + "Number of successful/failed requests made.", + &["rpc_method", "status"] + ) + .unwrap() +}); +static JSON_RPC_DURATION: Lazy = Lazy::new(|| { + register_histogram_vec!( + "json_rpc_duration_seconds", + "The json-rpc latencies in seconds.", + &["rpc_method"], + vec![ + 0.0008, 0.0016, 0.0032, 0.0064, 0.0128, 0.0256, 0.0512, 0.1024, 0.2048, 0.4096, 0.8192, + 1.0, 1.25, 1.5, 1.75, 2.0, 4.0, 8.0 + ], + ) + .unwrap() +}); + +/// IotaPeers is a mapping of public key to IotaPeer data +pub type IotaPeers = Arc>>; + +type MetricsPubKeys = Arc>>; + +#[derive(Hash, PartialEq, Eq, Debug, Clone)] +pub struct IotaPeer { + pub name: String, + pub public_key: Ed25519PublicKey, +} + +/// IotaNodeProvider queries the iota blockchain and keeps a record of known +/// validators based on the response from iota_getValidators. The node name, +/// public key and other info is extracted from the chain and stored in this +/// data structure. We pass this struct to the tls verifier and it depends on +/// the state contained within. Handlers also use this data in an Extractor +/// extension to check incoming clients on the http api against known keys. +#[derive(Debug, Clone)] +pub struct IotaNodeProvider { + nodes: IotaPeers, + bridge_nodes: IotaPeers, + static_nodes: IotaPeers, + rpc_url: String, + rpc_poll_interval: Duration, +} + +impl Allower for IotaNodeProvider { + fn allowed(&self, key: &Ed25519PublicKey) -> bool { + self.static_nodes.read().unwrap().contains_key(key) + || self.nodes.read().unwrap().contains_key(key) + || self.bridge_nodes.read().unwrap().contains_key(key) + } +} + +impl IotaNodeProvider { + pub fn new(rpc_url: String, rpc_poll_interval: Duration, static_peers: Vec) -> Self { + // build our hashmap with the static pub keys. we only do this one time at + // binary startup. + let static_nodes: HashMap = static_peers + .into_iter() + .map(|v| (v.public_key.clone(), v)) + .collect(); + let static_nodes = Arc::new(RwLock::new(static_nodes)); + let iota_nodes = Arc::new(RwLock::new(HashMap::new())); + let bridge_nodes = Arc::new(RwLock::new(HashMap::new())); + Self { + nodes: iota_nodes, + bridge_nodes, + static_nodes, + rpc_url, + rpc_poll_interval, + } + } + + /// get is used to retrieve peer info in our handlers + pub fn get(&self, key: &Ed25519PublicKey) -> Option { + debug!("look for {:?}", key); + // check static nodes first + if let Some(v) = self.static_nodes.read().unwrap().get(key) { + return Some(IotaPeer { + name: v.name.to_owned(), + public_key: v.public_key.to_owned(), + }); + } + // check iota validators + if let Some(v) = self.nodes.read().unwrap().get(key) { + return Some(IotaPeer { + name: v.name.to_owned(), + public_key: v.public_key.to_owned(), + }); + } + // check bridge validators + if let Some(v) = self.bridge_nodes.read().unwrap().get(key) { + return Some(IotaPeer { + name: v.name.to_owned(), + public_key: v.public_key.to_owned(), + }); + } + None + } + + /// Get a mutable reference to the allowed iota validator map + pub fn get_mut(&mut self) -> &mut IotaPeers { + &mut self.nodes + } + + /// get_validators will retrieve known validators + async fn get_validators(url: String) -> Result { + let rpc_method = "iotax_getLatestIotaSystemState"; + let observe = || { + let timer = JSON_RPC_DURATION + .with_label_values(&[rpc_method]) + .start_timer(); + || { + timer.observe_duration(); + } + }(); + let client = reqwest::Client::builder().build().unwrap(); + let request = serde_json::json!({ + "jsonrpc": "2.0", + "method":rpc_method, + "id":1, + }); + let response = client + .post(url) + .header(reqwest::header::CONTENT_TYPE, "application/json") + .body(request.to_string()) + .send() + .await + .with_context(|| { + JSON_RPC_STATE + .with_label_values(&[rpc_method, "failed_get"]) + .inc(); + observe(); + "unable to perform json rpc" + })?; + + let raw = response.bytes().await.with_context(|| { + JSON_RPC_STATE + .with_label_values(&[rpc_method, "failed_body_extract"]) + .inc(); + observe(); + "unable to extract body bytes from json rpc" + })?; + + #[derive(Debug, Deserialize)] + struct ResponseBody { + result: IotaSystemStateSummary, + } + + let body: ResponseBody = match serde_json::from_slice(&raw) { + Ok(b) => b, + Err(error) => { + JSON_RPC_STATE + .with_label_values(&[rpc_method, "failed_json_decode"]) + .inc(); + observe(); + bail!( + "unable to decode json: {error} response from json rpc: {:?}", + raw + ) + } + }; + JSON_RPC_STATE + .with_label_values(&[rpc_method, "success"]) + .inc(); + observe(); + Ok(body.result) + } + + /// get_bridge_validators will retrieve known bridge validators + async fn get_bridge_validators(url: String) -> Result { + let rpc_method = "iotax_getLatestBridge"; + let _timer = JSON_RPC_DURATION + .with_label_values(&[rpc_method]) + .start_timer(); + let client = reqwest::Client::builder().build().unwrap(); + let request = serde_json::json!({ + "jsonrpc": "2.0", + "method":rpc_method, + "id":1, + }); + let response = client + .post(url) + .header(reqwest::header::CONTENT_TYPE, "application/json") + .body(request.to_string()) + .send() + .await + .with_context(|| { + JSON_RPC_STATE + .with_label_values(&[rpc_method, "failed_get"]) + .inc(); + "unable to perform json rpc" + })?; + + let raw = response.bytes().await.with_context(|| { + JSON_RPC_STATE + .with_label_values(&[rpc_method, "failed_body_extract"]) + .inc(); + "unable to extract body bytes from json rpc" + })?; + + #[derive(Debug, Deserialize)] + struct ResponseBody { + result: BridgeSummary, + } + let summary: BridgeSummary = match serde_json::from_slice::(&raw) { + Ok(b) => b.result, + Err(error) => { + JSON_RPC_STATE + .with_label_values(&[rpc_method, "failed_json_decode"]) + .inc(); + bail!( + "unable to decode json: {error} response from json rpc: {:?}", + raw + ) + } + }; + JSON_RPC_STATE + .with_label_values(&[rpc_method, "success"]) + .inc(); + Ok(summary) + } + + async fn update_iota_validator_set(&self) { + match Self::get_validators(self.rpc_url.to_owned()).await { + Ok(summary) => { + let validators = extract(summary); + let mut allow = self.nodes.write().unwrap(); + allow.clear(); + allow.extend(validators); + info!( + "{} iota validators managed to make it on the allow list", + allow.len() + ); + } + Err(error) => { + JSON_RPC_STATE + .with_label_values(&["update_peer_count", "failed"]) + .inc(); + error!("unable to refresh peer list: {error}"); + } + }; + } + + async fn update_bridge_validator_set(&self, metrics_keys: MetricsPubKeys) { + let iota_system = match Self::get_validators(self.rpc_url.to_owned()).await { + Ok(summary) => summary, + Err(error) => { + JSON_RPC_STATE + .with_label_values(&["update_bridge_peer_count", "failed"]) + .inc(); + error!("unable to get iota system state: {error}"); + return; + } + }; + match Self::get_bridge_validators(self.rpc_url.to_owned()).await { + Ok(summary) => { + let names = iota_system + .active_validators + .into_iter() + .map(|v| (v.iota_address, v.name)) + .collect(); + let validators = extract_bridge(summary, Arc::new(names), metrics_keys).await; + let mut allow = self.bridge_nodes.write().unwrap(); + allow.clear(); + allow.extend(validators); + info!( + "{} bridge validators managed to make it on the allow list", + allow.len() + ); + } + Err(error) => { + JSON_RPC_STATE + .with_label_values(&["update_bridge_peer_count", "failed"]) + .inc(); + error!("unable to refresh iota bridge peer list: {error}"); + } + }; + } + + /// poll_peer_list will act as a refresh interval for our cache + pub fn poll_peer_list(&self) { + info!("Started polling for peers using rpc: {}", self.rpc_url); + + let rpc_poll_interval = self.rpc_poll_interval; + let cloned_self = self.clone(); + let bridge_metrics_keys: MetricsPubKeys = Arc::new(RwLock::new(HashMap::new())); + tokio::spawn(async move { + let mut interval = tokio::time::interval(rpc_poll_interval); + interval.set_missed_tick_behavior(tokio::time::MissedTickBehavior::Skip); + + loop { + interval.tick().await; + + cloned_self.update_iota_validator_set().await; + cloned_self + .update_bridge_validator_set(bridge_metrics_keys.clone()) + .await; + } + }); + } +} + +/// extract will get the network pubkey bytes from a IotaValidatorSummary type. +/// This type comes from a full node rpc result. See get_validators for +/// details. The key here, if extracted successfully, will ultimately be stored +/// in the allow list and let us communicate with those actual peers via tls. +fn extract(summary: IotaSystemStateSummary) -> impl Iterator { + summary.active_validators.into_iter().filter_map(|vm| { + match Ed25519PublicKey::from_bytes(&vm.network_pubkey_bytes) { + Ok(public_key) => { + debug!( + "adding public key {:?} for iota validator {:?}", + public_key, vm.name + ); + Some((public_key.clone(), IotaPeer { + name: vm.name, + public_key, + })) // scoped to filter_map + } + Err(error) => { + error!( + "unable to decode public key for name: {:?} iota_address: {:?} error: {error}", + vm.name, vm.iota_address + ); + None // scoped to filter_map + } + } + }) +} + +async fn extract_bridge( + summary: BridgeSummary, + names: Arc>, + metrics_keys: MetricsPubKeys, +) -> Vec<(Ed25519PublicKey, IotaPeer)> { + { + // Clean up the cache: retain only the metrics keys of the up-to-date bridge + // validator set + let mut metrics_keys_write = metrics_keys.write().unwrap(); + metrics_keys_write.retain(|url, _| { + summary.committee.members.iter().any(|(_, cm)| { + String::from_utf8(cm.http_rest_url.clone()).ok().as_ref() == Some(url) + }) + }); + } + + let client = reqwest::Client::builder() + .timeout(Duration::from_secs(10)) + .build() + .unwrap(); + let committee_members = summary.committee.members.clone(); + let results: Vec<_> = stream::iter(committee_members) + .filter_map(|(_, cm)| { + let client = client.clone(); + let metrics_keys = metrics_keys.clone(); + let names = names.clone(); + async move { + debug!( + address =% cm.iota_address, + "Extracting metrics public key for bridge node", + ); + + // Convert the Vec to a String and handle errors properly + let url_str = match String::from_utf8(cm.http_rest_url) { + Ok(url) => url, + Err(_) => { + warn!( + address =% cm.iota_address, + "Invalid UTF-8 sequence in http_rest_url for bridge node ", + ); + return None; + } + }; + // Parse the URL + let bridge_url = match Url::parse(&url_str) { + Ok(url) => url, + Err(_) => { + warn!(url_str, "Unable to parse http_rest_url"); + return None; + } + }; + + // Append "metrics_pub_key" to the path + let bridge_url = match append_path_segment(bridge_url, "metrics_pub_key") { + Some(url) => url, + None => { + warn!(url_str, "Unable to append path segment to URL"); + return None; + } + }; + + // Use the host portion of the http_rest_url as the "name" + let bridge_host = match bridge_url.host_str() { + Some(host) => host, + None => { + warn!(url_str, "Hostname is missing from http_rest_url"); + return None; + } + }; + let bridge_name = names.get(&cm.iota_address).cloned().unwrap_or_else(|| { + warn!( + address =% cm.iota_address, + "Bridge node not found in iota committee, using base URL as the name", + ); + String::from(bridge_host) + }); + let bridge_name = format!("bridge-{}", bridge_name); + + let bridge_request_url = bridge_url.as_str(); + + let metrics_pub_key = match client.get(bridge_request_url).send().await { + Ok(response) => { + let raw = response.bytes().await.ok()?; + let metrics_pub_key: String = match serde_json::from_slice(&raw) { + Ok(key) => key, + Err(error) => { + warn!(?error, url_str, "Failed to deserialize response"); + return fallback_to_cached_key( + &metrics_keys, + &url_str, + &bridge_name, + ); + } + }; + let metrics_bytes = match Base64::decode(&metrics_pub_key) { + Ok(pubkey_bytes) => pubkey_bytes, + Err(error) => { + warn!( + ?error, + bridge_name, "unable to decode public key for bridge node", + ); + return None; + } + }; + match Ed25519PublicKey::from_bytes(&metrics_bytes) { + Ok(pubkey) => { + // Successfully fetched the key, update the cache + let mut metrics_keys_write = metrics_keys.write().unwrap(); + metrics_keys_write.insert(url_str.clone(), pubkey.clone()); + debug!( + url_str, + public_key = ?pubkey, + "Successfully added bridge peer to metrics_keys" + ); + pubkey + } + Err(error) => { + warn!( + ?error, + bridge_request_url, + "unable to decode public key for bridge node", + ); + return None; + } + } + } + Err(_) => { + return fallback_to_cached_key(&metrics_keys, &url_str, &bridge_name); + } + }; + Some((metrics_pub_key.clone(), IotaPeer { + public_key: metrics_pub_key, + name: bridge_name, + })) + } + }) + .collect() + .await; + + results +} + +fn fallback_to_cached_key( + metrics_keys: &MetricsPubKeys, + url_str: &str, + bridge_name: &str, +) -> Option<(Ed25519PublicKey, IotaPeer)> { + let metrics_keys_read = metrics_keys.read().unwrap(); + if let Some(cached_key) = metrics_keys_read.get(url_str) { + debug!( + url_str, + "Using cached metrics public key after request failure" + ); + Some((cached_key.clone(), IotaPeer { + public_key: cached_key.clone(), + name: bridge_name.to_string(), + })) + } else { + warn!( + url_str, + "Failed to fetch public key and no cached key available" + ); + None + } +} + +fn append_path_segment(mut url: Url, segment: &str) -> Option { + url.path_segments_mut().ok()?.pop_if_empty().push(segment); + Some(url) +} + +#[cfg(test)] +mod tests { + use iota_types::{ + base_types::IotaAddress, + bridge::{BridgeCommitteeSummary, BridgeSummary, MoveTypeCommitteeMember}, + iota_system_state::iota_system_state_summary::{ + IotaSystemStateSummary, IotaValidatorSummary, + }, + }; + use multiaddr::Multiaddr; + use serde::Serialize; + + use super::*; + use crate::admin::{CertKeyPair, generate_self_cert}; + + /// creates a test that binds our proxy use case to the structure in + /// iota_getLatestIotaSystemState most of the fields are garbage, but we + /// will send the results of the serde process to a private decode + /// function that should always work if the structure is valid for our use + #[test] + fn depend_on_iota_iota_system_state_summary() { + let CertKeyPair(_, client_pub_key) = generate_self_cert("iota".into()); + let p2p_address: Multiaddr = "/ip4/127.0.0.1/tcp/10000" + .parse() + .expect("expected a multiaddr value"); + // all fields here just satisfy the field types, with exception to + // active_validators, we use some of those. + let depends_on = IotaSystemStateSummary { + active_validators: vec![IotaValidatorSummary { + network_pubkey_bytes: Vec::from(client_pub_key.as_bytes()), + p2p_address: format!("{p2p_address}"), + primary_address: "empty".into(), + ..Default::default() + }], + ..Default::default() + }; + + #[derive(Debug, Serialize, Deserialize)] + struct ResponseBody { + result: IotaSystemStateSummary, + } + + let r = serde_json::to_string(&ResponseBody { result: depends_on }) + .expect("expected to serialize ResponseBody{IotaSystemStateSummary}"); + + let deserialized = serde_json::from_str::(&r) + .expect("expected to deserialize ResponseBody{IotaSystemStateSummary}"); + + let peers = extract(deserialized.result); + assert_eq!(peers.count(), 1, "peers should have been a length of 1"); + } + + #[tokio::test] + async fn test_extract_bridge_invalid_bridge_url() { + let summary = BridgeSummary { + committee: BridgeCommitteeSummary { + members: vec![(vec![], MoveTypeCommitteeMember { + iota_address: IotaAddress::ZERO, + http_rest_url: "invalid_bridge_url".as_bytes().to_vec(), + ..Default::default() + })], + ..Default::default() + }, + ..Default::default() + }; + + let metrics_keys = Arc::new(RwLock::new(HashMap::new())); + { + let mut cache = metrics_keys.write().unwrap(); + cache.insert( + "invalid_bridge_url".to_string(), + Ed25519PublicKey::from_bytes(&[1u8; 32]).unwrap(), + ); + } + let result = extract_bridge(summary, Arc::new(BTreeMap::new()), metrics_keys.clone()).await; + + assert_eq!( + result.len(), + 0, + "Should not fall back on cache if invalid bridge url is set" + ); + } + + #[tokio::test] + async fn test_extract_bridge_interrupted_response() { + let summary = BridgeSummary { + committee: BridgeCommitteeSummary { + members: vec![(vec![], MoveTypeCommitteeMember { + iota_address: IotaAddress::ZERO, + http_rest_url: "https://unresponsive_bridge_url".as_bytes().to_vec(), + ..Default::default() + })], + ..Default::default() + }, + ..Default::default() + }; + + let metrics_keys = Arc::new(RwLock::new(HashMap::new())); + { + let mut cache = metrics_keys.write().unwrap(); + cache.insert( + "https://unresponsive_bridge_url".to_string(), + Ed25519PublicKey::from_bytes(&[1u8; 32]).unwrap(), + ); + } + let result = extract_bridge(summary, Arc::new(BTreeMap::new()), metrics_keys.clone()).await; + + assert_eq!( + result.len(), + 1, + "Should fall back on cache if invalid response occurs" + ); + let allowed_peer = &result[0].1; + assert_eq!( + allowed_peer.public_key.as_bytes(), + &[1u8; 32], + "Should fall back to the cached public key" + ); + + let cache = metrics_keys.read().unwrap(); + assert!( + cache.contains_key("https://unresponsive_bridge_url"), + "Cache should still contain the original key" + ); + } + + #[test] + fn test_append_path_segment() { + let test_cases = vec![ + ( + "https://example.com", + "metrics_pub_key", + "https://example.com/metrics_pub_key", + ), + ( + "https://example.com/api", + "metrics_pub_key", + "https://example.com/api/metrics_pub_key", + ), + ( + "https://example.com/", + "metrics_pub_key", + "https://example.com/metrics_pub_key", + ), + ( + "https://example.com/api/", + "metrics_pub_key", + "https://example.com/api/metrics_pub_key", + ), + ( + "https://example.com:8080", + "metrics_pub_key", + "https://example.com:8080/metrics_pub_key", + ), + ( + "https://example.com?param=value", + "metrics_pub_key", + "https://example.com/metrics_pub_key?param=value", + ), + ( + "https://example.com:8080/api/v1?param=value", + "metrics_pub_key", + "https://example.com:8080/api/v1/metrics_pub_key?param=value", + ), + ]; + + for (input_url, segment, expected_output) in test_cases { + let url = Url::parse(input_url).unwrap(); + let result = append_path_segment(url, segment); + assert!( + result.is_some(), + "Failed to append segment for URL: {}", + input_url + ); + let result_url = result.unwrap(); + assert_eq!( + result_url.as_str(), + expected_output, + "Unexpected result for input URL: {}", + input_url + ); + } + } +} diff --git a/crates/iota-proxy/src/prom_to_mimir.rs b/crates/iota-proxy/src/prom_to_mimir.rs new file mode 100644 index 00000000000..fb625856b1f --- /dev/null +++ b/crates/iota-proxy/src/prom_to_mimir.rs @@ -0,0 +1,404 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +use itertools::Itertools; +use prometheus::proto::{Counter, Gauge, Histogram, Metric, MetricFamily, MetricType}; +use protobuf::RepeatedField; +use tracing::{debug, error}; + +use crate::{remote_write, var}; + +#[derive(Debug)] +pub struct Mimir { + state: S, +} + +impl From<&Metric> for Mimir> { + fn from(m: &Metric) -> Self { + // we consume metric labels from an owned version so we can sort them + let mut m = m.to_owned(); + let mut sorted = m.take_label(); + sorted.sort_by(|a, b| { + (a.get_name(), a.get_value()) + .partial_cmp(&(b.get_name(), b.get_value())) + .unwrap() + }); + let mut r = RepeatedField::::default(); + for label in sorted { + let lp = remote_write::Label { + name: label.get_name().into(), + value: label.get_value().into(), + }; + r.push(lp); + } + Self { state: r } + } +} + +impl IntoIterator for Mimir> { + type Item = remote_write::Label; + type IntoIter = std::vec::IntoIter; + + fn into_iter(self) -> Self::IntoIter { + self.state.into_iter() + } +} + +impl From<&Counter> for Mimir { + fn from(c: &Counter) -> Self { + Self { + state: remote_write::Sample { + value: c.get_value(), + ..Default::default() + }, + } + } +} +impl From<&Gauge> for Mimir { + fn from(c: &Gauge) -> Self { + Self { + state: remote_write::Sample { + value: c.get_value(), + ..Default::default() + }, + } + } +} +impl Mimir { + fn sample(self) -> remote_write::Sample { + self.state + } +} + +/// TODO implement histogram +impl From<&Histogram> for Mimir { + fn from(_h: &Histogram) -> Self { + Self { + state: remote_write::Histogram::default(), + } + } +} +/// TODO implement histogram +impl Mimir { + #[allow(dead_code)] + fn histogram(self) -> remote_write::Histogram { + self.state + } +} +impl From> for Mimir> { + fn from(metric_families: Vec) -> Self { + // we may have more but we'll have at least this many timeseries + let mut timeseries: Vec = + Vec::with_capacity(metric_families.len()); + + for mf in metric_families { + // TODO add From impl + let mt = match mf.get_field_type() { + MetricType::COUNTER => remote_write::metric_metadata::MetricType::Counter, + MetricType::GAUGE => remote_write::metric_metadata::MetricType::Gauge, + MetricType::HISTOGRAM => remote_write::metric_metadata::MetricType::Histogram, + MetricType::SUMMARY => remote_write::metric_metadata::MetricType::Summary, + MetricType::UNTYPED => remote_write::metric_metadata::MetricType::Unknown, + }; + + // filter out the types we don't support + match mt { + remote_write::metric_metadata::MetricType::Counter + | remote_write::metric_metadata::MetricType::Gauge => (), + other => { + debug!("{:?} is not yet implemented, skipping metric", other); + continue; + } + } + + // TODO stop using state directly + timeseries.extend(Mimir::from(mf.clone()).state); + } + + Self { + state: timeseries + .into_iter() + // the upstream remote_write should have a max sample size per request set to this + // number + .chunks(var!("MIMIR_MAX_SAMPLE_SIZE", 500)) + .into_iter() + .map(|ts| remote_write::WriteRequest { + timeseries: ts.collect(), + ..Default::default() + }) + .collect_vec(), + } + } +} + +impl IntoIterator for Mimir> { + type Item = remote_write::WriteRequest; + type IntoIter = std::vec::IntoIter; + + fn into_iter(self) -> Self::IntoIter { + self.state.into_iter() + } +} + +impl Mimir> { + pub fn repeated(self) -> RepeatedField { + self.state + } +} + +impl From for Mimir> { + fn from(mf: MetricFamily) -> Self { + let mut timeseries = vec![]; + for metric in mf.get_metric() { + let mut ts = remote_write::TimeSeries::default(); + ts.labels.extend(vec![ + // mimir requires that we use __name__ as a key that points to a value + // of the metric name + remote_write::Label { + name: "__name__".into(), + value: mf.get_name().into(), + }, + ]); + ts.labels + .extend(Mimir::>::from(metric)); + + // assumption here is that since a MetricFamily will have one MetricType, we'll + // only need to look for one of these types. Setting two different + // types on Metric at the same time in a way that is conflicting + // with the MetricFamily type will result in undefined mimir + // behavior, probably an error. + if metric.has_counter() { + let mut s = Mimir::::from(metric.get_counter()).sample(); + s.timestamp = metric.get_timestamp_ms(); + ts.samples.push(s); + } else if metric.has_gauge() { + let mut s = Mimir::::from(metric.get_gauge()).sample(); + s.timestamp = metric.get_timestamp_ms(); + ts.samples.push(s); + } else if metric.has_histogram() { + // TODO implement + // ts.mut_histograms() + // .push(Mimir::::from(metric. + // get_histogram()).histogram()); + } else if metric.has_summary() { + // TODO implement + error!("summary is not implemented for a metric type"); + } + timeseries.push(ts); + } + Self { state: timeseries } + } +} + +impl Mimir { + pub fn timeseries(self) -> remote_write::TimeSeries { + self.state + } +} + +#[cfg(test)] +pub mod tests { + use prometheus::proto; + use protobuf::RepeatedField; + + use crate::{prom_to_mimir::Mimir, remote_write}; + + // protobuf stuff + pub fn create_metric_family( + name: &str, + help: &str, + field_type: Option, + metric: RepeatedField, + ) -> proto::MetricFamily { + // no public fields, cannot use literals + let mut mf = proto::MetricFamily::default(); + mf.set_name(name.into()); + mf.set_help(help.into()); + // TODO remove the metric type serialization if we still don't use it + // after implementing histogram and summary + if let Some(ft) = field_type { + mf.set_field_type(ft); + } + mf.set_metric(metric); + mf + } + #[allow(dead_code)] + fn create_metric_gauge( + labels: RepeatedField, + gauge: proto::Gauge, + ) -> proto::Metric { + let mut m = proto::Metric::default(); + m.set_label(labels); + m.set_gauge(gauge); + m.set_timestamp_ms(12345); + m + } + + pub fn create_metric_counter( + labels: RepeatedField, + counter: proto::Counter, + ) -> proto::Metric { + let mut m = proto::Metric::default(); + m.set_label(labels); + m.set_counter(counter); + m.set_timestamp_ms(12345); + m + } + + pub fn create_metric_histogram( + labels: RepeatedField, + histogram: proto::Histogram, + ) -> proto::Metric { + let mut m = proto::Metric::default(); + m.set_label(labels); + m.set_histogram(histogram); + m.set_timestamp_ms(12345); + m + } + + pub fn create_histogram() -> proto::Histogram { + let mut h = proto::Histogram::default(); + h.set_sample_count(1); + h.set_sample_sum(1.0); + let mut b = proto::Bucket::default(); + b.set_cumulative_count(1); + b.set_upper_bound(1.0); + h.mut_bucket().push(b); + h + } + + pub fn create_labels(labels: Vec<(&str, &str)>) -> Vec { + labels + .into_iter() + .map(|(key, value)| { + let mut lp = proto::LabelPair::default(); + lp.set_name(key.into()); + lp.set_value(value.into()); + lp + }) + .collect() + } + #[allow(dead_code)] + fn create_gauge(value: f64) -> proto::Gauge { + let mut g = proto::Gauge::default(); + g.set_value(value); + g + } + + pub fn create_counter(value: f64) -> proto::Counter { + let mut c = proto::Counter::default(); + c.set_value(value); + c + } + + // end protobuf stuff + + // mimir stuff + fn create_timeseries_with_samples( + labels: Vec, + samples: Vec, + ) -> remote_write::TimeSeries { + remote_write::TimeSeries { + labels, + samples, + ..Default::default() + } + } + // end mimir stuff + + #[test] + fn metricfamily_to_timeseries() { + let tests: Vec<(proto::MetricFamily, Vec)> = vec![ + ( + create_metric_family( + "test_gauge", + "i'm a help message", + Some(proto::MetricType::GAUGE), + RepeatedField::from(vec![create_metric_gauge( + RepeatedField::from_vec(create_labels(vec![ + ("host", "local-test-validator"), + ("network", "unittest-network"), + ])), + create_gauge(2046.0), + )]), + ), + vec![create_timeseries_with_samples( + vec![ + remote_write::Label { + name: "__name__".into(), + value: "test_gauge".into(), + }, + remote_write::Label { + name: "host".into(), + value: "local-test-validator".into(), + }, + remote_write::Label { + name: "network".into(), + value: "unittest-network".into(), + }, + ], + vec![remote_write::Sample { + value: 2046.0, + timestamp: 12345, + }], + )], + ), + ( + create_metric_family( + "test_counter", + "i'm a help message", + Some(proto::MetricType::GAUGE), + RepeatedField::from(vec![create_metric_counter( + RepeatedField::from_vec(create_labels(vec![ + ("host", "local-test-validator"), + ("network", "unittest-network"), + ])), + create_counter(2046.0), + )]), + ), + vec![create_timeseries_with_samples( + vec![ + remote_write::Label { + name: "__name__".into(), + value: "test_counter".into(), + }, + remote_write::Label { + name: "host".into(), + value: "local-test-validator".into(), + }, + remote_write::Label { + name: "network".into(), + value: "unittest-network".into(), + }, + ], + vec![remote_write::Sample { + value: 2046.0, + timestamp: 12345, + }], + )], + ), + ]; + for (mf, expected_ts) in tests { + // TODO stop using state directly + for (actual, expected) in Mimir::from(mf).state.into_iter().zip(expected_ts) { + assert_eq!(actual.labels, expected.labels); + for (actual_sample, expected_sample) in + actual.samples.into_iter().zip(expected.samples) + { + assert_eq!( + actual_sample.value, expected_sample.value, + "sample values do not match" + ); + + // timestamps are injected on the iota-node and we copy it to our sample + // make sure that works + assert_eq!( + actual_sample.timestamp, expected_sample.timestamp, + "timestamp should be non-zero" + ); + } + } + } + } +} diff --git a/crates/iota-proxy/src/remote_write.rs b/crates/iota-proxy/src/remote_write.rs new file mode 100644 index 00000000000..5c32f90dc9d --- /dev/null +++ b/crates/iota-proxy/src/remote_write.rs @@ -0,0 +1,506 @@ +// Copyright (c) Mysten Labs, Inc. +// Modifications Copyright (c) 2024 IOTA Stiftung +// SPDX-License-Identifier: Apache-2.0 + +#[allow(clippy::derive_partial_eq_without_eq)] +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct MetricMetadata { + /// Represents the metric type, these match the set from Prometheus. + /// Refer to model/textparse/interface.go for details. + #[prost(enumeration = "metric_metadata::MetricType", tag = "1")] + pub r#type: i32, + #[prost(string, tag = "2")] + pub metric_family_name: ::prost::alloc::string::String, + #[prost(string, tag = "4")] + pub help: ::prost::alloc::string::String, + #[prost(string, tag = "5")] + pub unit: ::prost::alloc::string::String, +} +/// Nested message and enum types in `MetricMetadata`. +pub mod metric_metadata { + #[derive(Clone, Copy, Debug, PartialEq, Eq, Hash, PartialOrd, Ord, ::prost::Enumeration)] + #[repr(i32)] + pub enum MetricType { + Unknown = 0, + Counter = 1, + Gauge = 2, + Histogram = 3, + Gaugehistogram = 4, + Summary = 5, + Info = 6, + Stateset = 7, + } + impl MetricType { + /// String value of the enum field names used in the ProtoBuf + /// definition. + /// + /// The values are not transformed in any way and thus are considered + /// stable (if the ProtoBuf definition does not change) and safe + /// for programmatic use. + pub fn as_str_name(&self) -> &'static str { + match self { + MetricType::Unknown => "UNKNOWN", + MetricType::Counter => "COUNTER", + MetricType::Gauge => "GAUGE", + MetricType::Histogram => "HISTOGRAM", + MetricType::Gaugehistogram => "GAUGEHISTOGRAM", + MetricType::Summary => "SUMMARY", + MetricType::Info => "INFO", + MetricType::Stateset => "STATESET", + } + } + /// Creates an enum from field names used in the ProtoBuf definition. + pub fn from_str_name(value: &str) -> ::core::option::Option { + match value { + "UNKNOWN" => Some(Self::Unknown), + "COUNTER" => Some(Self::Counter), + "GAUGE" => Some(Self::Gauge), + "HISTOGRAM" => Some(Self::Histogram), + "GAUGEHISTOGRAM" => Some(Self::Gaugehistogram), + "SUMMARY" => Some(Self::Summary), + "INFO" => Some(Self::Info), + "STATESET" => Some(Self::Stateset), + _ => None, + } + } + } +} +#[allow(clippy::derive_partial_eq_without_eq)] +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct Sample { + #[prost(double, tag = "1")] + pub value: f64, + /// timestamp is in ms format, see model/timestamp/timestamp.go for + /// conversion from time.Time to Prometheus timestamp. + #[prost(int64, tag = "2")] + pub timestamp: i64, +} +#[allow(clippy::derive_partial_eq_without_eq)] +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct Exemplar { + /// Optional, can be empty. + #[prost(message, repeated, tag = "1")] + pub labels: ::prost::alloc::vec::Vec