From c1369104b6e790dec4f6e2340e73d3e1c7562e21 Mon Sep 17 00:00:00 2001
From: 0xtekgrinder <72015889+0xtekgrinder@users.noreply.github.com>
Date: Sat, 27 Apr 2024 00:52:42 -0400
Subject: [PATCH] merge pull request #1 from AngleProtocol/feat/migration
* feat: add AgTokens, CoreBorrow and FlashLoan
* chore: update solhint config
* chore: upgrade solhint
* chore: remove deployments and e2e tests
* tests: migrate agTokens from hardhat to foundry
* feat: treasury contract
* tests: FlashAngle foundry
* tests: migrate CoreBorrow test to foundry
* refactor: foundry tests are being run tby the ci
* refactor: use package.json from foundry boilerplate
* chore: install foundry dependencies in CI
* chore: add submodules again
* chore: add .gitkeep for invariants and fuzz tests
* tests: add AgTokenSideChainMultiBridge foundry test
* chore: remove hardhat from the repository
* forge install: LayerZero
* chore: add layerZero dependency
* style: prettier and lint files
* chore: comment out invariant and fuzz tests
* feat: AgToken sidechain
---
.env.example | 26 +
.github/actions/setup-repo/action.yml | 35 +
.github/assets/logo.svg | 14 +
.github/workflows/ci-deep.yml | 173 ++
.github/workflows/ci.yml | 242 +++
.github/workflows/test.yml | 34 -
.gitignore | 61 +-
.gitmodules | 15 +
.npmrc | 1 +
.prettierignore | 8 +
.prettierrc | 12 +
.solhint.json | 28 +
.solhintignore | 12 +
.vscode/tasks.json | 20 +
CODE_OF_CONDUCT.md | 128 ++
CONTRIBUTING.md | 9 +
LICENSE | 674 +++++++
README.md | 171 +-
contracts/agToken/AgEUR.sol | 112 ++
contracts/agToken/AgToken.sol | 151 ++
.../agToken/AgTokenSideChainMultiBridge.sol | 273 +++
.../agToken/TokenSideChainMultiBridge.sol | 333 ++++
.../agToken/layerZero/LayerZeroBridge.sol | 130 ++
.../layerZero/LayerZeroBridgeToken.sol | 151 ++
.../agToken/layerZero/utils/IOFTCore.sol | 110 ++
.../layerZero/utils/NonblockingLzApp.sol | 281 +++
contracts/agToken/layerZero/utils/OFTCore.sol | 185 ++
contracts/agToken/nameable/AgEURNameable.sol | 37 +
.../agToken/nameable/AgTokenNameable.sol | 38 +
.../AgTokenSideChainMultiBridgeNameable.sol | 31 +
.../TokenPolygonUpgradeableNameable.sol | 31 +
.../polygon/TokenPolygonUpgradeable.sol | 424 +++++
.../polygon/utils/ERC20UpgradeableCustom.sol | 377 ++++
contracts/coreBorrow/CoreBorrow.sol | 188 ++
contracts/external/ProxyAdmin.sol | 81 +
.../external/TransparentUpgradeableProxy.sol | 121 ++
contracts/flashloan/FlashAngle.sol | 179 ++
contracts/interfaces/IAgToken.sol | 69 +
.../IAgTokenSideChainMultiBridge.sol | 28 +
contracts/interfaces/ICoreBorrow.sol | 28 +
contracts/interfaces/IFlashAngle.sol | 37 +
contracts/interfaces/IOracle.sol | 34 +
contracts/interfaces/ITreasury.sol | 45 +
contracts/interfaces/IVaultManager.sol | 266 +++
.../interfaces/coreModule/IOracleCore.sol | 11 +
.../coreModule/IPerpetualManager.sol | 31 +
.../interfaces/coreModule/IStableMaster.sol | 96 +
.../create2/ImmutableCreate2Factory.sol | 17 +
contracts/mock/MockCoreBorrow.sol | 53 +
contracts/mock/MockFlashLoanModule.sol | 38 +
contracts/mock/MockFlashLoanReceiver.sol | 30 +
contracts/mock/MockLayerZero.sol | 76 +
contracts/mock/MockToken.sol | 54 +
contracts/mock/MockTokenPermit.sol | 87 +
contracts/mock/MockTreasury.sol | 85 +
contracts/treasury/Treasury.sol | 413 ++++
contracts/utils/Constants.sol | 15 +
foundry.toml | 79 +-
helpers/fork.sh | 73 +
lib/LayerZero | 1 +
lib/chainlink | 1 +
lib/openzeppelin-contracts | 1 +
lib/openzeppelin-contracts-upgradeable | 1 +
lib/utils | 1 +
package.json | 53 +
remappings.txt | 9 +
script/Counter.s.sol | 12 -
scripts/BasicScript.s.sol | 24 +
scripts/utils/Constants.s.sol | 11 +
scripts/utils/FindInitCode.s.sol | 35 +
scripts/utils/VanityAddress.s.sol | 38 +
scripts/vanity.json | 4 +
slither.config.json | 10 +
slither.sh | 42 +
src/Counter.sol | 14 -
test/Counter.t.sol | 24 -
test/fuzz/.gitkeep | 0
test/invariants/.gitkeep | 0
test/units/BaseTest.t.sol | 46 +
test/units/agToken/AgToken.t.sol | 232 +++
.../agToken/AgTokenSideChainMultiBridge.t.sol | 671 +++++++
test/units/coreBorrow/CoreBorrow.t.sol | 303 +++
test/units/flashAngle/FlashAngle.t.sol | 268 +++
test/units/utils/Solenv.sol | 809 ++++++++
utils/forwardUtils.js | 22 +
yarn.lock | 1691 +++++++++++++++++
86 files changed, 10747 insertions(+), 137 deletions(-)
create mode 100644 .env.example
create mode 100644 .github/actions/setup-repo/action.yml
create mode 100644 .github/assets/logo.svg
create mode 100644 .github/workflows/ci-deep.yml
create mode 100644 .github/workflows/ci.yml
delete mode 100644 .github/workflows/test.yml
create mode 100644 .npmrc
create mode 100644 .prettierignore
create mode 100644 .prettierrc
create mode 100644 .solhint.json
create mode 100644 .solhintignore
create mode 100644 .vscode/tasks.json
create mode 100644 CODE_OF_CONDUCT.md
create mode 100644 CONTRIBUTING.md
create mode 100644 LICENSE
create mode 100644 contracts/agToken/AgEUR.sol
create mode 100644 contracts/agToken/AgToken.sol
create mode 100644 contracts/agToken/AgTokenSideChainMultiBridge.sol
create mode 100644 contracts/agToken/TokenSideChainMultiBridge.sol
create mode 100644 contracts/agToken/layerZero/LayerZeroBridge.sol
create mode 100644 contracts/agToken/layerZero/LayerZeroBridgeToken.sol
create mode 100644 contracts/agToken/layerZero/utils/IOFTCore.sol
create mode 100644 contracts/agToken/layerZero/utils/NonblockingLzApp.sol
create mode 100644 contracts/agToken/layerZero/utils/OFTCore.sol
create mode 100644 contracts/agToken/nameable/AgEURNameable.sol
create mode 100644 contracts/agToken/nameable/AgTokenNameable.sol
create mode 100644 contracts/agToken/nameable/AgTokenSideChainMultiBridgeNameable.sol
create mode 100644 contracts/agToken/nameable/TokenPolygonUpgradeableNameable.sol
create mode 100644 contracts/agToken/polygon/TokenPolygonUpgradeable.sol
create mode 100644 contracts/agToken/polygon/utils/ERC20UpgradeableCustom.sol
create mode 100644 contracts/coreBorrow/CoreBorrow.sol
create mode 100644 contracts/external/ProxyAdmin.sol
create mode 100644 contracts/external/TransparentUpgradeableProxy.sol
create mode 100644 contracts/flashloan/FlashAngle.sol
create mode 100644 contracts/interfaces/IAgToken.sol
create mode 100644 contracts/interfaces/IAgTokenSideChainMultiBridge.sol
create mode 100644 contracts/interfaces/ICoreBorrow.sol
create mode 100644 contracts/interfaces/IFlashAngle.sol
create mode 100644 contracts/interfaces/IOracle.sol
create mode 100644 contracts/interfaces/ITreasury.sol
create mode 100644 contracts/interfaces/IVaultManager.sol
create mode 100644 contracts/interfaces/coreModule/IOracleCore.sol
create mode 100644 contracts/interfaces/coreModule/IPerpetualManager.sol
create mode 100644 contracts/interfaces/coreModule/IStableMaster.sol
create mode 100644 contracts/interfaces/external/create2/ImmutableCreate2Factory.sol
create mode 100644 contracts/mock/MockCoreBorrow.sol
create mode 100644 contracts/mock/MockFlashLoanModule.sol
create mode 100644 contracts/mock/MockFlashLoanReceiver.sol
create mode 100644 contracts/mock/MockLayerZero.sol
create mode 100644 contracts/mock/MockToken.sol
create mode 100644 contracts/mock/MockTokenPermit.sol
create mode 100644 contracts/mock/MockTreasury.sol
create mode 100644 contracts/treasury/Treasury.sol
create mode 100644 contracts/utils/Constants.sol
create mode 100644 helpers/fork.sh
create mode 160000 lib/LayerZero
create mode 160000 lib/chainlink
create mode 160000 lib/openzeppelin-contracts
create mode 160000 lib/openzeppelin-contracts-upgradeable
create mode 160000 lib/utils
create mode 100644 package.json
create mode 100644 remappings.txt
delete mode 100644 script/Counter.s.sol
create mode 100644 scripts/BasicScript.s.sol
create mode 100644 scripts/utils/Constants.s.sol
create mode 100644 scripts/utils/FindInitCode.s.sol
create mode 100644 scripts/utils/VanityAddress.s.sol
create mode 100644 scripts/vanity.json
create mode 100644 slither.config.json
create mode 100755 slither.sh
delete mode 100644 src/Counter.sol
delete mode 100644 test/Counter.t.sol
create mode 100644 test/fuzz/.gitkeep
create mode 100644 test/invariants/.gitkeep
create mode 100644 test/units/BaseTest.t.sol
create mode 100644 test/units/agToken/AgToken.t.sol
create mode 100644 test/units/agToken/AgTokenSideChainMultiBridge.t.sol
create mode 100644 test/units/coreBorrow/CoreBorrow.t.sol
create mode 100644 test/units/flashAngle/FlashAngle.t.sol
create mode 100644 test/units/utils/Solenv.sol
create mode 100755 utils/forwardUtils.js
create mode 100644 yarn.lock
diff --git a/.env.example b/.env.example
new file mode 100644
index 0000000..5898837
--- /dev/null
+++ b/.env.example
@@ -0,0 +1,26 @@
+## Add URI and BIP39 mnemonic, and etherscan API key for every network that you plan to use
+## These are used in `hardhat.config.ts` (see `nodeUrl` and `accounts` calls in network definitions)
+#ETH_NODE_URI_FORK=""
+#MNEMONIC_FORK=""
+#ETHERSCAN_API_KEY=""
+
+## To do mainnet fork in Polygon
+#ETH_NODE_URI_FORKPOLYGON=""
+
+## To do a local Ethereum mainnet fork
+#ETH_NODE_URI_MAINNETFORKREMOTE=""
+
+#ETH_NODE_URI_GOERLI=""
+#MNEMONIC_GOERLI=""
+#GOERLI_ETHERSCAN_API_KEY=""
+
+#ETH_NODE_URI_MAINNET=""
+#MNEMONIC_MAINNET=""
+#MAINNET_ETHERSCAN_API_KEY=""
+
+#ETH_NODE_URI_POLYGON=""
+#MNEMONIC_POLYGON=""
+#POLYGON_ETHERSCAN_API_KEY=""
+
+#ETH_NODE_URI_FANTOM=""
+
diff --git a/.github/actions/setup-repo/action.yml b/.github/actions/setup-repo/action.yml
new file mode 100644
index 0000000..1932a67
--- /dev/null
+++ b/.github/actions/setup-repo/action.yml
@@ -0,0 +1,35 @@
+name: Setup repo
+description: Runs all steps to setup the repo (install node_modules, build, etc...)
+inputs:
+ registry-token:
+ description: 'PAT to access registries'
+runs:
+ using: 'composite'
+ steps:
+ - name: Get yarn cache directory path
+ id: yarn-cache-dir-path
+ shell: bash
+ run: |
+ echo "::set-output name=dir::$(yarn cache dir)"
+ echo "::set-output name=version::$(yarn -v)"
+
+ - uses: actions/setup-node@v3
+ with:
+ node-version: '20'
+
+ - uses: actions/cache@v2
+ id: yarn-cache
+ with:
+ path: |
+ **/node_modules
+ ${{ steps.yarn-cache-dir-path.outputs.dir }}
+
+ key: ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }}
+ restore-keys: |
+ ${{ runner.os }}-yarn-
+
+ - name: Install dependencies
+ shell: bash
+ run: echo "//npm.pkg.github.com/:_authToken=$GH_REGISTRY_ACCESS_TOKEN" >> .npmrc && yarn install --frozen-lockfile --verbose && rm -f .npmrc
+ env:
+ GH_REGISTRY_ACCESS_TOKEN: ${{ inputs.registry-token }}
diff --git a/.github/assets/logo.svg b/.github/assets/logo.svg
new file mode 100644
index 0000000..ccd539a
--- /dev/null
+++ b/.github/assets/logo.svg
@@ -0,0 +1,14 @@
+
diff --git a/.github/workflows/ci-deep.yml b/.github/workflows/ci-deep.yml
new file mode 100644
index 0000000..c1a1435
--- /dev/null
+++ b/.github/workflows/ci-deep.yml
@@ -0,0 +1,173 @@
+name: "CI Deep"
+
+env:
+ FOUNDRY_PROFILE: "ci"
+
+on:
+ schedule:
+ - cron: "0 3 * * 0" # at 3:00am UTC every Sunday
+ workflow_dispatch:
+ inputs:
+ fuzzRuns:
+ default: "10000"
+ description: "Unit: number of fuzz runs."
+ required: false
+ invariantRuns:
+ default: "300"
+ description: "Unit: number of invariant runs."
+ required: false
+ invariantDepth:
+ default: "50"
+ description: "Unit: invariant depth."
+ required: false
+
+jobs:
+ lint:
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v3
+
+ - uses: actions/setup-node@v3
+ with:
+ node-version: 18
+ cache: "yarn"
+
+ - name: Setup repo
+ uses: ./.github/actions/setup-repo
+ with:
+ registry-token: ${{ secrets.GH_REGISTRY_ACCESS_TOKEN }}
+
+ - name: Run solhint
+ run: yarn lint:check
+
+ - name: "Add lint summary"
+ run: |
+ echo "## Lint result" >> $GITHUB_STEP_SUMMARY
+ echo "✅ Passed" >> $GITHUB_STEP_SUMMARY
+
+ build:
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v3
+ with:
+ submodules: "recursive"
+
+ - name: Install Foundry
+ uses: foundry-rs/foundry-toolchain@v1
+ with:
+ version: nightly
+
+ - name: Setup repo
+ uses: ./.github/actions/setup-repo
+ with:
+ registry-token: ${{ secrets.GH_REGISTRY_ACCESS_TOKEN }}
+
+ - name: Compile foundry
+ run: yarn compile --sizes
+
+ - name: "Cache the build so that it can be re-used by the other jobs"
+ uses: "actions/cache/save@v3"
+ with:
+ key: "build-${{ github.sha }}"
+ path: |
+ cache-forge
+ out
+ node_modules
+
+ - name: "Add build summary"
+ run: |
+ echo "## Build result" >> $GITHUB_STEP_SUMMARY
+ echo "✅ Passed" >> $GITHUB_STEP_SUMMARY
+
+ test-unit:
+ needs: ["build", "lint"]
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v3
+ with:
+ submodules: "recursive"
+
+ - uses: actions/cache/restore@v3
+ with:
+ fail-on-cache-miss: true
+ path: |
+ cache-forge
+ out
+ node_modules
+ key: "build-${{ github.sha }}"
+
+ - name: Install Foundry
+ uses: foundry-rs/foundry-toolchain@v1
+ with:
+ version: nightly
+
+ - name: Run Foundry tests
+ run: yarn test:unit
+ env:
+ ETH_NODE_URI_POLYGON: ${{ secrets.ETH_NODE_URI_POLYGON }}
+ ETH_NODE_URI_ARBITRUM: ${{ secrets.ETH_NODE_URI_ARBITRUM }}
+ ETH_NODE_URI_OPTIMISM: ${{ secrets.ETH_NODE_URI_OPTIMISM }}
+ ETH_NODE_URI_MAINNET: ${{ secrets.ETH_NODE_URI_MAINNET }}
+
+ # test-invariant:
+ # needs: ["build", "lint"]
+ # runs-on: ubuntu-latest
+ # steps:
+ # - uses: actions/checkout@v3
+ # with:
+ # submodules: "recursive"
+#
+ # - uses: actions/cache/restore@v3
+ # with:
+ # fail-on-cache-miss: true
+ # path: |
+ # cache-forge
+ # out
+ # node_modules
+ # key: "build-${{ github.sha }}"
+#
+ # - name: Install Foundry
+ # uses: foundry-rs/foundry-toolchain@v1
+ # with:
+ # version: nightly
+#
+ # - name: Run Foundry tests
+ # run: yarn test:invariant
+ # env:
+ # FOUNDRY_INVARIANT_RUNS: ${{ github.event.inputs.invariantRuns || '300' }}
+ # FOUNDRY_INVARIANT_DEPTH: ${{ github.event.inputs.invariantDepth || '50' }}
+ # ETH_NODE_URI_POLYGON: ${{ secrets.ETH_NODE_URI_POLYGON }}
+ # ETH_NODE_URI_ARBITRUM: ${{ secrets.ETH_NODE_URI_ARBITRUM }}
+ # ETH_NODE_URI_OPTIMISM: ${{ secrets.ETH_NODE_URI_OPTIMISM }}
+ # ETH_NODE_URI_MAINNET: ${{ secrets.ETH_NODE_URI_MAINNET }}
+#
+ # test-fuzz:
+ # needs: ["build", "lint"]
+ # runs-on: ubuntu-latest
+ # steps:
+ # - uses: actions/checkout@v3
+ # with:
+ # submodules: "recursive"
+#
+ # - uses: actions/cache/restore@v3
+ # with:
+ # fail-on-cache-miss: true
+ # path: |
+ # cache-forge
+ # out
+ # node_modules
+ # key: "build-${{ github.sha }}"
+#
+ # - name: Install Foundry
+ # uses: foundry-rs/foundry-toolchain@v1
+ # with:
+ # version: nightly
+#
+ # - name: Run Foundry tests
+ # run: yarn test:fuzz
+ # env:
+ # FOUNDRY_FUZZ_RUNS: ${{ github.event.inputs.fuzzRuns || '10000' }}
+ # ETH_NODE_URI_POLYGON: ${{ secrets.ETH_NODE_URI_POLYGON }}
+ # ETH_NODE_URI_ARBITRUM: ${{ secrets.ETH_NODE_URI_ARBITRUM }}
+ # ETH_NODE_URI_OPTIMISM: ${{ secrets.ETH_NODE_URI_OPTIMISM }}
+ # ETH_NODE_URI_MAINNET: ${{ secrets.ETH_NODE_URI_MAINNET }}
\ No newline at end of file
diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml
new file mode 100644
index 0000000..cb4b2ee
--- /dev/null
+++ b/.github/workflows/ci.yml
@@ -0,0 +1,242 @@
+name: "CI"
+
+env:
+ FOUNDRY_PROFILE: "ci"
+
+on:
+ workflow_dispatch:
+ pull_request:
+ push:
+ branches:
+ - "main"
+
+jobs:
+ lint:
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v3
+
+ - uses: actions/setup-node@v3
+ with:
+ node-version: 18
+ cache: "yarn"
+
+ - name: Setup repo
+ uses: ./.github/actions/setup-repo
+ with:
+ registry-token: ${{ secrets.GH_REGISTRY_ACCESS_TOKEN }}
+
+ - name: Run solhint
+ run: yarn lint:check
+
+ - name: "Add lint summary"
+ run: |
+ echo "## Lint result" >> $GITHUB_STEP_SUMMARY
+ echo "✅ Passed" >> $GITHUB_STEP_SUMMARY
+
+ build:
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v3
+ with:
+ submodules: "recursive"
+
+ - name: Install Foundry
+ uses: foundry-rs/foundry-toolchain@v1
+ with:
+ version: nightly
+
+ - name: Setup repo
+ uses: ./.github/actions/setup-repo
+ with:
+ registry-token: ${{ secrets.GH_REGISTRY_ACCESS_TOKEN }}
+
+ - name: Compile foundry
+ run: yarn compile --sizes
+
+ - name: "Cache the build so that it can be re-used by the other jobs"
+ uses: "actions/cache/save@v3"
+ with:
+ key: "build-${{ github.sha }}"
+ path: |
+ cache-forge
+ out
+ node_modules
+
+ - name: "Add build summary"
+ run: |
+ echo "## Build result" >> $GITHUB_STEP_SUMMARY
+ echo "✅ Passed" >> $GITHUB_STEP_SUMMARY
+
+ test-unit:
+ needs: ["build", "lint"]
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v3
+ with:
+ submodules: "recursive"
+
+ - uses: actions/cache/restore@v3
+ with:
+ fail-on-cache-miss: true
+ path: |
+ cache-forge
+ out
+ node_modules
+ key: "build-${{ github.sha }}"
+
+ - name: Install Foundry
+ uses: foundry-rs/foundry-toolchain@v1
+ with:
+ version: nightly
+
+ - name: Run Foundry tests
+ run: yarn test:unit
+ env:
+ ETH_NODE_URI_POLYGON: ${{ secrets.ETH_NODE_URI_POLYGON }}
+ ETH_NODE_URI_ARBITRUM: ${{ secrets.ETH_NODE_URI_ARBITRUM }}
+ ETH_NODE_URI_OPTIMISM: ${{ secrets.ETH_NODE_URI_OPTIMISM }}
+ ETH_NODE_URI_MAINNET: ${{ secrets.ETH_NODE_URI_MAINNET }}
+
+ # test-invariant:
+ # needs: ["build", "lint"]
+ # runs-on: ubuntu-latest
+ # steps:
+ # - uses: actions/checkout@v3
+ # with:
+ # submodules: "recursive"
+#
+ # - uses: actions/cache/restore@v3
+ # with:
+ # fail-on-cache-miss: true
+ # path: |
+ # cache-forge
+ # out
+ # node_modules
+ # key: "build-${{ github.sha }}"
+#
+ # - name: Install Foundry
+ # uses: foundry-rs/foundry-toolchain@v1
+ # with:
+ # version: nightly
+#
+ # - name: Run Foundry tests
+ # run: yarn test:invariant
+ # env:
+ # FOUNDRY_INVARIANT_RUNS: "8"
+ # FOUNDRY_INVARIANT_DEPTH: "256"
+ # ETH_NODE_URI_POLYGON: ${{ secrets.ETH_NODE_URI_POLYGON }}
+ # ETH_NODE_URI_ARBITRUM: ${{ secrets.ETH_NODE_URI_ARBITRUM }}
+ # ETH_NODE_URI_OPTIMISM: ${{ secrets.ETH_NODE_URI_OPTIMISM }}
+ # ETH_NODE_URI_MAINNET: ${{ secrets.ETH_NODE_URI_MAINNET }}
+#
+ # test-fuzz:
+ # needs: ["build", "lint"]
+ # runs-on: ubuntu-latest
+ # steps:
+ # - uses: actions/checkout@v3
+ # with:
+ # submodules: "recursive"
+#
+ # - uses: actions/cache/restore@v3
+ # with:
+ # fail-on-cache-miss: true
+ # path: |
+ # cache-forge
+ # out
+ # node_modules
+ # key: "build-${{ github.sha }}"
+#
+ # - name: Install Foundry
+ # uses: foundry-rs/foundry-toolchain@v1
+ # with:
+ # version: nightly
+#
+ # - name: Run Foundry tests
+ # run: yarn test:fuzz
+ # env:
+ # FOUNDRY_FUZZ_RUNS: "5000"
+ # ETH_NODE_URI_POLYGON: ${{ secrets.ETH_NODE_URI_POLYGON }}
+ # ETH_NODE_URI_ARBITRUM: ${{ secrets.ETH_NODE_URI_ARBITRUM }}
+ # ETH_NODE_URI_OPTIMISM: ${{ secrets.ETH_NODE_URI_OPTIMISM }}
+ # ETH_NODE_URI_MAINNET: ${{ secrets.ETH_NODE_URI_MAINNET }}
+
+ coverage:
+ needs: ["build", "lint"]
+ runs-on: "ubuntu-latest"
+ steps:
+ - uses: actions/checkout@v3
+ with:
+ submodules: "recursive"
+
+ - uses: actions/cache/restore@v3
+ with:
+ fail-on-cache-miss: true
+ path: |
+ cache-forge
+ out
+ node_modules
+ key: "build-${{ github.sha }}"
+
+ - name: "Install Foundry"
+ uses: "foundry-rs/foundry-toolchain@v1"
+
+ - name: "Install lcov"
+ run: "sudo apt-get install lcov"
+
+ - name: "Generate the coverage report using the unit and the integration tests"
+ run: "yarn ci:coverage"
+ env:
+ ETH_NODE_URI_POLYGON: ${{ secrets.ETH_NODE_URI_POLYGON }}
+ ETH_NODE_URI_ARBITRUM: ${{ secrets.ETH_NODE_URI_ARBITRUM }}
+ ETH_NODE_URI_OPTIMISM: ${{ secrets.ETH_NODE_URI_OPTIMISM }}
+ ETH_NODE_URI_MAINNET: ${{ secrets.ETH_NODE_URI_MAINNET }}
+
+ - name: "Upload coverage report to Codecov"
+ uses: "codecov/codecov-action@v3"
+ with:
+ files: "./lcov.info"
+ token: ${{ secrets.CODECOV_TOKEN }}
+
+ - name: "Add coverage summary"
+ run: |
+ echo "## Coverage result" >> $GITHUB_STEP_SUMMARY
+ echo "✅ Uploaded to Codecov" >> $GITHUB_STEP_SUMMARY
+
+ slither-analyze:
+ needs: ["build", "lint"]
+ runs-on: "ubuntu-latest"
+ permissions:
+ actions: "read"
+ contents: "read"
+ security-events: "write"
+ steps:
+ - uses: actions/checkout@v3
+ with:
+ submodules: "recursive"
+
+ - name: Install Foundry
+ uses: foundry-rs/foundry-toolchain@v1
+ with:
+ version: nightly
+
+ - name: Compile foundry
+ run: forge clean && forge build --build-info --force
+
+ - name: "Run Slither analysis"
+ uses: "crytic/slither-action@v0.3.2"
+ id: "slither"
+ with:
+ ignore-compile: true
+ fail-on: "none"
+ sarif: "results.sarif"
+
+ - name: "Upload SARIF file to GitHub code scanning"
+ uses: "github/codeql-action/upload-sarif@v2"
+ with:
+ sarif_file: ${{ steps.slither.outputs.sarif }}
+
+ - name: "Add Slither summary"
+ run: |
+ echo "## Slither result" >> $GITHUB_STEP_SUMMARY
+ echo "✅ Uploaded to GitHub code scanning" >> $GITHUB_STEP_SUMMARY
\ No newline at end of file
diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml
deleted file mode 100644
index 9282e82..0000000
--- a/.github/workflows/test.yml
+++ /dev/null
@@ -1,34 +0,0 @@
-name: test
-
-on: workflow_dispatch
-
-env:
- FOUNDRY_PROFILE: ci
-
-jobs:
- check:
- strategy:
- fail-fast: true
-
- name: Foundry project
- runs-on: ubuntu-latest
- steps:
- - uses: actions/checkout@v4
- with:
- submodules: recursive
-
- - name: Install Foundry
- uses: foundry-rs/foundry-toolchain@v1
- with:
- version: nightly
-
- - name: Run Forge build
- run: |
- forge --version
- forge build --sizes
- id: build
-
- - name: Run Forge tests
- run: |
- forge test -vvv
- id: test
diff --git a/.gitignore b/.gitignore
index 85198aa..ebe338c 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,14 +1,53 @@
-# Compiler files
-cache/
-out/
+## Defaults
+__pycache__
+.idea
+.DS_Store
+.deps
+.docs
+.env
+node_modules
+tenderly.yaml
+settings.json
+venv
-# Ignores development broadcast logs
-!/broadcast
-/broadcast/*/31337/
-/broadcast/**/dry-run/
+# Build output
+/cache
+/cache-hh
+build
+export
+**/artifacts
+.openzeppelin
+lcov.info
+docgen/docs
+docgen/SUMMARY.md
+solidity-flattenedContracts
+./crytic-export
+typechain
+slither-audit.txt
+slither
-# Docs
-docs/
+# Test output
+coverage
+coverage.json
-# Dotenv file
-.env
+# Running output
+gas-report.txt
+gasReporterOutput.json
+addresses.json
+blockchain_db
+ganache*
+yarn-error.log
+
+# deployments
+deployments/localhost
+deployments/mainnetForkRemote
+
+# bin
+bin
+
+# temporary delete
+typechain/cacheIndex.ts
+
+# foundry
+/out
+/cache-forge
diff --git a/.gitmodules b/.gitmodules
index 888d42d..2a230a9 100644
--- a/.gitmodules
+++ b/.gitmodules
@@ -1,3 +1,18 @@
[submodule "lib/forge-std"]
path = lib/forge-std
url = https://github.com/foundry-rs/forge-std
+[submodule "lib/openzeppelin-contracts"]
+ path = lib/openzeppelin-contracts
+ url = https://github.com/OpenZeppelin/openzeppelin-contracts
+[submodule "lib/utils"]
+ path = lib/utils
+ url = https://github.com/AngleProtocol/utils
+[submodule "lib/chainlink"]
+ path = lib/chainlink
+ url = https://github.com/smartcontractkit/chainlink
+[submodule "lib/openzeppelin-contracts-upgradeable"]
+ path = lib/openzeppelin-contracts-upgradeable
+ url = https://github.com/OpenZeppelin/openzeppelin-contracts-upgradeable
+[submodule "lib/LayerZero"]
+ path = lib/LayerZero
+ url = https://github.com/LayerZero-Labs/LayerZero
diff --git a/.npmrc b/.npmrc
new file mode 100644
index 0000000..af66bba
--- /dev/null
+++ b/.npmrc
@@ -0,0 +1 @@
+@angleprotocol:registry=https://npm.pkg.github.com
diff --git a/.prettierignore b/.prettierignore
new file mode 100644
index 0000000..de01738
--- /dev/null
+++ b/.prettierignore
@@ -0,0 +1,8 @@
+lib
+artifacts
+node_modules
+cache-forge
+cache-hh
+export
+out
+typechain
\ No newline at end of file
diff --git a/.prettierrc b/.prettierrc
new file mode 100644
index 0000000..295b1da
--- /dev/null
+++ b/.prettierrc
@@ -0,0 +1,12 @@
+{
+ "overrides": [
+ {
+ "files": "*.sol",
+ "options": {
+ "printWidth": 120,
+ "singleQuote": false,
+ "bracketSpacing": true
+ }
+ }
+ ]
+}
diff --git a/.solhint.json b/.solhint.json
new file mode 100644
index 0000000..939f263
--- /dev/null
+++ b/.solhint.json
@@ -0,0 +1,28 @@
+{
+ "extends": "solhint:recommended",
+ "plugins": ["prettier"],
+ "rules": {
+ "max-line-length": ["warn", 120],
+ "avoid-call-value": "warn",
+ "avoid-low-level-calls": "off",
+ "avoid-tx-origin": "warn",
+ "const-name-snakecase": "warn",
+ "contract-name-camelcase": "warn",
+ "imports-on-top": "warn",
+ "prettier/prettier": "error",
+ "ordering": "off",
+ "max-states-count": "off",
+ "mark-callable-contracts": "off",
+ "no-empty-blocks": "off",
+ "no-global-import": "off",
+ "not-rely-on-time": "off",
+ "compiler-version": "off",
+ "private-vars-leading-underscore": "warn",
+ "reentrancy": "warn",
+ "no-inline-assembly": "off",
+ "no-complex-fallback": "off",
+ "reason-string": "off",
+ "func-visibility": ["warn", { "ignoreConstructors": true }],
+ "explicit-types": ["error","explicit"]
+ }
+}
diff --git a/.solhintignore b/.solhintignore
new file mode 100644
index 0000000..6a51c68
--- /dev/null
+++ b/.solhintignore
@@ -0,0 +1,12 @@
+# Doesn't need to lint dev files
+lib
+scripts
+test
+
+# Doesn't need to lint build files
+node_modules
+cache-forge
+out
+
+# Doesn't need to lint utils
+external
\ No newline at end of file
diff --git a/.vscode/tasks.json b/.vscode/tasks.json
new file mode 100644
index 0000000..216fc85
--- /dev/null
+++ b/.vscode/tasks.json
@@ -0,0 +1,20 @@
+{
+ "version": "2.0.0",
+ "tasks": [
+ {
+ "label": "Generate Header",
+ "type": "shell",
+ "command": "headers ${input:header}",
+ "presentation": {
+ "reveal": "never"
+ }
+ }
+ ],
+ "inputs": [
+ {
+ "id": "header",
+ "description": "Header",
+ "type": "promptString"
+ }
+ ]
+}
diff --git a/CODE_OF_CONDUCT.md b/CODE_OF_CONDUCT.md
new file mode 100644
index 0000000..ef8ff11
--- /dev/null
+++ b/CODE_OF_CONDUCT.md
@@ -0,0 +1,128 @@
+# Contributor Covenant Code of Conduct
+
+## Our Pledge
+
+We as members, contributors, and leaders pledge to make participation in our
+community a harassment-free experience for everyone, regardless of age, body
+size, visible or invisible disability, ethnicity, sex characteristics, gender
+identity and expression, level of experience, education, socio-economic status,
+nationality, personal appearance, race, religion, or sexual identity
+and orientation.
+
+We pledge to act and interact in ways that contribute to an open, welcoming,
+diverse, inclusive, and healthy community.
+
+## Our Standards
+
+Examples of behavior that contributes to a positive environment for our
+community include:
+
+- Demonstrating empathy and kindness toward other people
+- Being respectful of differing opinions, viewpoints, and experiences
+- Giving and gracefully accepting constructive feedback
+- Accepting responsibility and apologizing to those affected by our mistakes,
+ and learning from the experience
+- Focusing on what is best not just for us as individuals, but for the
+ overall community
+
+Examples of unacceptable behavior include:
+
+- The use of sexualized language or imagery, and sexual attention or
+ advances of any kind
+- Trolling, insulting or derogatory comments, and personal or political attacks
+- Public or private harassment
+- Publishing others' private information, such as a physical or email
+ address, without their explicit permission
+- Other conduct which could reasonably be considered inappropriate in a
+ professional setting
+
+## Enforcement Responsibilities
+
+Community leaders are responsible for clarifying and enforcing our standards of
+acceptable behavior and will take appropriate and fair corrective action in
+response to any behavior that they deem inappropriate, threatening, offensive,
+or harmful.
+
+Community leaders have the right and responsibility to remove, edit, or reject
+comments, commits, code, wiki edits, issues, and other contributions that are
+not aligned to this Code of Conduct, and will communicate reasons for moderation
+decisions when appropriate.
+
+## Scope
+
+This Code of Conduct applies within all community spaces, and also applies when
+an individual is officially representing the community in public spaces.
+Examples of representing our community include using an official e-mail address,
+posting via an official social media account, or acting as an appointed
+representative at an online or offline event.
+
+## Enforcement
+
+Instances of abusive, harassing, or otherwise unacceptable behavior may be
+reported to the community leaders responsible for enforcement at
+contact@angle.money.
+All complaints will be reviewed and investigated promptly and fairly.
+
+All community leaders are obligated to respect the privacy and security of the
+reporter of any incident.
+
+## Enforcement Guidelines
+
+Community leaders will follow these Community Impact Guidelines in determining
+the consequences for any action they deem in violation of this Code of Conduct:
+
+### 1. Correction
+
+**Community Impact**: Use of inappropriate language or other behavior deemed
+unprofessional or unwelcome in the community.
+
+**Consequence**: A private, written warning from community leaders, providing
+clarity around the nature of the violation and an explanation of why the
+behavior was inappropriate. A public apology may be requested.
+
+### 2. Warning
+
+**Community Impact**: A violation through a single incident or series
+of actions.
+
+**Consequence**: A warning with consequences for continued behavior. No
+interaction with the people involved, including unsolicited interaction with
+those enforcing the Code of Conduct, for a specified period of time. This
+includes avoiding interactions in community spaces as well as external channels
+like social media. Violating these terms may lead to a temporary or
+permanent ban.
+
+### 3. Temporary Ban
+
+**Community Impact**: A serious violation of community standards, including
+sustained inappropriate behavior.
+
+**Consequence**: A temporary ban from any sort of interaction or public
+communication with the community for a specified period of time. No public or
+private interaction with the people involved, including unsolicited interaction
+with those enforcing the Code of Conduct, is allowed during this period.
+Violating these terms may lead to a permanent ban.
+
+### 4. Permanent Ban
+
+**Community Impact**: Demonstrating a pattern of violation of community
+standards, including sustained inappropriate behavior, harassment of an
+individual, or aggression toward or disparagement of classes of individuals.
+
+**Consequence**: A permanent ban from any sort of public interaction within
+the community.
+
+## Attribution
+
+This Code of Conduct is adapted from the [Contributor Covenant][homepage],
+version 2.0, available at
+https://www.contributor-covenant.org/version/2/0/code_of_conduct.html.
+
+Community Impact Guidelines were inspired by [Mozilla's code of conduct
+enforcement ladder](https://github.com/mozilla/diversity).
+
+[homepage]: https://www.contributor-covenant.org
+
+For answers to common questions about this code of conduct, see the FAQ at
+https://www.contributor-covenant.org/faq. Translations are available at
+https://www.contributor-covenant.org/translations.
diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
new file mode 100644
index 0000000..de7db21
--- /dev/null
+++ b/CONTRIBUTING.md
@@ -0,0 +1,9 @@
+# Submit a change or a new feature
+
+First of all thank you for your interest in this repository!
+
+This is only the beginning of the Angle protocol and codebase, and anyone is welcome to improve it.
+
+To submit some code, please work in a fork, reach out to explain what you've done and open a Pull Request from your fork.
+
+Feel free to reach out in the [#developers channel](https://discord.gg/HcRB8QMeKU) of our Discord Server if you need a hand!
diff --git a/LICENSE b/LICENSE
new file mode 100644
index 0000000..f288702
--- /dev/null
+++ b/LICENSE
@@ -0,0 +1,674 @@
+ GNU GENERAL PUBLIC LICENSE
+ Version 3, 29 June 2007
+
+ Copyright (C) 2007 Free Software Foundation, Inc.
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+ Preamble
+
+ The GNU General Public License is a free, copyleft license for
+software and other kinds of works.
+
+ The licenses for most software and other practical works are designed
+to take away your freedom to share and change the works. By contrast,
+the GNU General Public License is intended to guarantee your freedom to
+share and change all versions of a program--to make sure it remains free
+software for all its users. We, the Free Software Foundation, use the
+GNU General Public License for most of our software; it applies also to
+any other work released this way by its authors. You can apply it to
+your programs, too.
+
+ When we speak of free software, we are referring to freedom, not
+price. Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+them if you wish), that you receive source code or can get it if you
+want it, that you can change the software or use pieces of it in new
+free programs, and that you know you can do these things.
+
+ To protect your rights, we need to prevent others from denying you
+these rights or asking you to surrender the rights. Therefore, you have
+certain responsibilities if you distribute copies of the software, or if
+you modify it: responsibilities to respect the freedom of others.
+
+ For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must pass on to the recipients the same
+freedoms that you received. You must make sure that they, too, receive
+or can get the source code. And you must show them these terms so they
+know their rights.
+
+ Developers that use the GNU GPL protect your rights with two steps:
+(1) assert copyright on the software, and (2) offer you this License
+giving you legal permission to copy, distribute and/or modify it.
+
+ For the developers' and authors' protection, the GPL clearly explains
+that there is no warranty for this free software. For both users' and
+authors' sake, the GPL requires that modified versions be marked as
+changed, so that their problems will not be attributed erroneously to
+authors of previous versions.
+
+ Some devices are designed to deny users access to install or run
+modified versions of the software inside them, although the manufacturer
+can do so. This is fundamentally incompatible with the aim of
+protecting users' freedom to change the software. The systematic
+pattern of such abuse occurs in the area of products for individuals to
+use, which is precisely where it is most unacceptable. Therefore, we
+have designed this version of the GPL to prohibit the practice for those
+products. If such problems arise substantially in other domains, we
+stand ready to extend this provision to those domains in future versions
+of the GPL, as needed to protect the freedom of users.
+
+ Finally, every program is threatened constantly by software patents.
+States should not allow patents to restrict development and use of
+software on general-purpose computers, but in those that do, we wish to
+avoid the special danger that patents applied to a free program could
+make it effectively proprietary. To prevent this, the GPL assures that
+patents cannot be used to render the program non-free.
+
+ The precise terms and conditions for copying, distribution and
+modification follow.
+
+ TERMS AND CONDITIONS
+
+ 0. Definitions.
+
+ "This License" refers to version 3 of the GNU General Public License.
+
+ "Copyright" also means copyright-like laws that apply to other kinds of
+works, such as semiconductor masks.
+
+ "The Program" refers to any copyrightable work licensed under this
+License. Each licensee is addressed as "you". "Licensees" and
+"recipients" may be individuals or organizations.
+
+ To "modify" a work means to copy from or adapt all or part of the work
+in a fashion requiring copyright permission, other than the making of an
+exact copy. The resulting work is called a "modified version" of the
+earlier work or a work "based on" the earlier work.
+
+ A "covered work" means either the unmodified Program or a work based
+on the Program.
+
+ To "propagate" a work means to do anything with it that, without
+permission, would make you directly or secondarily liable for
+infringement under applicable copyright law, except executing it on a
+computer or modifying a private copy. Propagation includes copying,
+distribution (with or without modification), making available to the
+public, and in some countries other activities as well.
+
+ To "convey" a work means any kind of propagation that enables other
+parties to make or receive copies. Mere interaction with a user through
+a computer network, with no transfer of a copy, is not conveying.
+
+ An interactive user interface displays "Appropriate Legal Notices"
+to the extent that it includes a convenient and prominently visible
+feature that (1) displays an appropriate copyright notice, and (2)
+tells the user that there is no warranty for the work (except to the
+extent that warranties are provided), that licensees may convey the
+work under this License, and how to view a copy of this License. If
+the interface presents a list of user commands or options, such as a
+menu, a prominent item in the list meets this criterion.
+
+ 1. Source Code.
+
+ The "source code" for a work means the preferred form of the work
+for making modifications to it. "Object code" means any non-source
+form of a work.
+
+ A "Standard Interface" means an interface that either is an official
+standard defined by a recognized standards body, or, in the case of
+interfaces specified for a particular programming language, one that
+is widely used among developers working in that language.
+
+ The "System Libraries" of an executable work include anything, other
+than the work as a whole, that (a) is included in the normal form of
+packaging a Major Component, but which is not part of that Major
+Component, and (b) serves only to enable use of the work with that
+Major Component, or to implement a Standard Interface for which an
+implementation is available to the public in source code form. A
+"Major Component", in this context, means a major essential component
+(kernel, window system, and so on) of the specific operating system
+(if any) on which the executable work runs, or a compiler used to
+produce the work, or an object code interpreter used to run it.
+
+ The "Corresponding Source" for a work in object code form means all
+the source code needed to generate, install, and (for an executable
+work) run the object code and to modify the work, including scripts to
+control those activities. However, it does not include the work's
+System Libraries, or general-purpose tools or generally available free
+programs which are used unmodified in performing those activities but
+which are not part of the work. For example, Corresponding Source
+includes interface definition files associated with source files for
+the work, and the source code for shared libraries and dynamically
+linked subprograms that the work is specifically designed to require,
+such as by intimate data communication or control flow between those
+subprograms and other parts of the work.
+
+ The Corresponding Source need not include anything that users
+can regenerate automatically from other parts of the Corresponding
+Source.
+
+ The Corresponding Source for a work in source code form is that
+same work.
+
+ 2. Basic Permissions.
+
+ All rights granted under this License are granted for the term of
+copyright on the Program, and are irrevocable provided the stated
+conditions are met. This License explicitly affirms your unlimited
+permission to run the unmodified Program. The output from running a
+covered work is covered by this License only if the output, given its
+content, constitutes a covered work. This License acknowledges your
+rights of fair use or other equivalent, as provided by copyright law.
+
+ You may make, run and propagate covered works that you do not
+convey, without conditions so long as your license otherwise remains
+in force. You may convey covered works to others for the sole purpose
+of having them make modifications exclusively for you, or provide you
+with facilities for running those works, provided that you comply with
+the terms of this License in conveying all material for which you do
+not control copyright. Those thus making or running the covered works
+for you must do so exclusively on your behalf, under your direction
+and control, on terms that prohibit them from making any copies of
+your copyrighted material outside their relationship with you.
+
+ Conveying under any other circumstances is permitted solely under
+the conditions stated below. Sublicensing is not allowed; section 10
+makes it unnecessary.
+
+ 3. Protecting Users' Legal Rights From Anti-Circumvention Law.
+
+ No covered work shall be deemed part of an effective technological
+measure under any applicable law fulfilling obligations under article
+11 of the WIPO copyright treaty adopted on 20 December 1996, or
+similar laws prohibiting or restricting circumvention of such
+measures.
+
+ When you convey a covered work, you waive any legal power to forbid
+circumvention of technological measures to the extent such circumvention
+is effected by exercising rights under this License with respect to
+the covered work, and you disclaim any intention to limit operation or
+modification of the work as a means of enforcing, against the work's
+users, your or third parties' legal rights to forbid circumvention of
+technological measures.
+
+ 4. Conveying Verbatim Copies.
+
+ You may convey verbatim copies of the Program's source code as you
+receive it, in any medium, provided that you conspicuously and
+appropriately publish on each copy an appropriate copyright notice;
+keep intact all notices stating that this License and any
+non-permissive terms added in accord with section 7 apply to the code;
+keep intact all notices of the absence of any warranty; and give all
+recipients a copy of this License along with the Program.
+
+ You may charge any price or no price for each copy that you convey,
+and you may offer support or warranty protection for a fee.
+
+ 5. Conveying Modified Source Versions.
+
+ You may convey a work based on the Program, or the modifications to
+produce it from the Program, in the form of source code under the
+terms of section 4, provided that you also meet all of these conditions:
+
+ a) The work must carry prominent notices stating that you modified
+ it, and giving a relevant date.
+
+ b) The work must carry prominent notices stating that it is
+ released under this License and any conditions added under section
+ 7. This requirement modifies the requirement in section 4 to
+ "keep intact all notices".
+
+ c) You must license the entire work, as a whole, under this
+ License to anyone who comes into possession of a copy. This
+ License will therefore apply, along with any applicable section 7
+ additional terms, to the whole of the work, and all its parts,
+ regardless of how they are packaged. This License gives no
+ permission to license the work in any other way, but it does not
+ invalidate such permission if you have separately received it.
+
+ d) If the work has interactive user interfaces, each must display
+ Appropriate Legal Notices; however, if the Program has interactive
+ interfaces that do not display Appropriate Legal Notices, your
+ work need not make them do so.
+
+ A compilation of a covered work with other separate and independent
+works, which are not by their nature extensions of the covered work,
+and which are not combined with it such as to form a larger program,
+in or on a volume of a storage or distribution medium, is called an
+"aggregate" if the compilation and its resulting copyright are not
+used to limit the access or legal rights of the compilation's users
+beyond what the individual works permit. Inclusion of a covered work
+in an aggregate does not cause this License to apply to the other
+parts of the aggregate.
+
+ 6. Conveying Non-Source Forms.
+
+ You may convey a covered work in object code form under the terms
+of sections 4 and 5, provided that you also convey the
+machine-readable Corresponding Source under the terms of this License,
+in one of these ways:
+
+ a) Convey the object code in, or embodied in, a physical product
+ (including a physical distribution medium), accompanied by the
+ Corresponding Source fixed on a durable physical medium
+ customarily used for software interchange.
+
+ b) Convey the object code in, or embodied in, a physical product
+ (including a physical distribution medium), accompanied by a
+ written offer, valid for at least three years and valid for as
+ long as you offer spare parts or customer support for that product
+ model, to give anyone who possesses the object code either (1) a
+ copy of the Corresponding Source for all the software in the
+ product that is covered by this License, on a durable physical
+ medium customarily used for software interchange, for a price no
+ more than your reasonable cost of physically performing this
+ conveying of source, or (2) access to copy the
+ Corresponding Source from a network server at no charge.
+
+ c) Convey individual copies of the object code with a copy of the
+ written offer to provide the Corresponding Source. This
+ alternative is allowed only occasionally and noncommercially, and
+ only if you received the object code with such an offer, in accord
+ with subsection 6b.
+
+ d) Convey the object code by offering access from a designated
+ place (gratis or for a charge), and offer equivalent access to the
+ Corresponding Source in the same way through the same place at no
+ further charge. You need not require recipients to copy the
+ Corresponding Source along with the object code. If the place to
+ copy the object code is a network server, the Corresponding Source
+ may be on a different server (operated by you or a third party)
+ that supports equivalent copying facilities, provided you maintain
+ clear directions next to the object code saying where to find the
+ Corresponding Source. Regardless of what server hosts the
+ Corresponding Source, you remain obligated to ensure that it is
+ available for as long as needed to satisfy these requirements.
+
+ e) Convey the object code using peer-to-peer transmission, provided
+ you inform other peers where the object code and Corresponding
+ Source of the work are being offered to the general public at no
+ charge under subsection 6d.
+
+ A separable portion of the object code, whose source code is excluded
+from the Corresponding Source as a System Library, need not be
+included in conveying the object code work.
+
+ A "User Product" is either (1) a "consumer product", which means any
+tangible personal property which is normally used for personal, family,
+or household purposes, or (2) anything designed or sold for incorporation
+into a dwelling. In determining whether a product is a consumer product,
+doubtful cases shall be resolved in favor of coverage. For a particular
+product received by a particular user, "normally used" refers to a
+typical or common use of that class of product, regardless of the status
+of the particular user or of the way in which the particular user
+actually uses, or expects or is expected to use, the product. A product
+is a consumer product regardless of whether the product has substantial
+commercial, industrial or non-consumer uses, unless such uses represent
+the only significant mode of use of the product.
+
+ "Installation Information" for a User Product means any methods,
+procedures, authorization keys, or other information required to install
+and execute modified versions of a covered work in that User Product from
+a modified version of its Corresponding Source. The information must
+suffice to ensure that the continued functioning of the modified object
+code is in no case prevented or interfered with solely because
+modification has been made.
+
+ If you convey an object code work under this section in, or with, or
+specifically for use in, a User Product, and the conveying occurs as
+part of a transaction in which the right of possession and use of the
+User Product is transferred to the recipient in perpetuity or for a
+fixed term (regardless of how the transaction is characterized), the
+Corresponding Source conveyed under this section must be accompanied
+by the Installation Information. But this requirement does not apply
+if neither you nor any third party retains the ability to install
+modified object code on the User Product (for example, the work has
+been installed in ROM).
+
+ The requirement to provide Installation Information does not include a
+requirement to continue to provide support service, warranty, or updates
+for a work that has been modified or installed by the recipient, or for
+the User Product in which it has been modified or installed. Access to a
+network may be denied when the modification itself materially and
+adversely affects the operation of the network or violates the rules and
+protocols for communication across the network.
+
+ Corresponding Source conveyed, and Installation Information provided,
+in accord with this section must be in a format that is publicly
+documented (and with an implementation available to the public in
+source code form), and must require no special password or key for
+unpacking, reading or copying.
+
+ 7. Additional Terms.
+
+ "Additional permissions" are terms that supplement the terms of this
+License by making exceptions from one or more of its conditions.
+Additional permissions that are applicable to the entire Program shall
+be treated as though they were included in this License, to the extent
+that they are valid under applicable law. If additional permissions
+apply only to part of the Program, that part may be used separately
+under those permissions, but the entire Program remains governed by
+this License without regard to the additional permissions.
+
+ When you convey a copy of a covered work, you may at your option
+remove any additional permissions from that copy, or from any part of
+it. (Additional permissions may be written to require their own
+removal in certain cases when you modify the work.) You may place
+additional permissions on material, added by you to a covered work,
+for which you have or can give appropriate copyright permission.
+
+ Notwithstanding any other provision of this License, for material you
+add to a covered work, you may (if authorized by the copyright holders of
+that material) supplement the terms of this License with terms:
+
+ a) Disclaiming warranty or limiting liability differently from the
+ terms of sections 15 and 16 of this License; or
+
+ b) Requiring preservation of specified reasonable legal notices or
+ author attributions in that material or in the Appropriate Legal
+ Notices displayed by works containing it; or
+
+ c) Prohibiting misrepresentation of the origin of that material, or
+ requiring that modified versions of such material be marked in
+ reasonable ways as different from the original version; or
+
+ d) Limiting the use for publicity purposes of names of licensors or
+ authors of the material; or
+
+ e) Declining to grant rights under trademark law for use of some
+ trade names, trademarks, or service marks; or
+
+ f) Requiring indemnification of licensors and authors of that
+ material by anyone who conveys the material (or modified versions of
+ it) with contractual assumptions of liability to the recipient, for
+ any liability that these contractual assumptions directly impose on
+ those licensors and authors.
+
+ All other non-permissive additional terms are considered "further
+restrictions" within the meaning of section 10. If the Program as you
+received it, or any part of it, contains a notice stating that it is
+governed by this License along with a term that is a further
+restriction, you may remove that term. If a license document contains
+a further restriction but permits relicensing or conveying under this
+License, you may add to a covered work material governed by the terms
+of that license document, provided that the further restriction does
+not survive such relicensing or conveying.
+
+ If you add terms to a covered work in accord with this section, you
+must place, in the relevant source files, a statement of the
+additional terms that apply to those files, or a notice indicating
+where to find the applicable terms.
+
+ Additional terms, permissive or non-permissive, may be stated in the
+form of a separately written license, or stated as exceptions;
+the above requirements apply either way.
+
+ 8. Termination.
+
+ You may not propagate or modify a covered work except as expressly
+provided under this License. Any attempt otherwise to propagate or
+modify it is void, and will automatically terminate your rights under
+this License (including any patent licenses granted under the third
+paragraph of section 11).
+
+ However, if you cease all violation of this License, then your
+license from a particular copyright holder is reinstated (a)
+provisionally, unless and until the copyright holder explicitly and
+finally terminates your license, and (b) permanently, if the copyright
+holder fails to notify you of the violation by some reasonable means
+prior to 60 days after the cessation.
+
+ Moreover, your license from a particular copyright holder is
+reinstated permanently if the copyright holder notifies you of the
+violation by some reasonable means, this is the first time you have
+received notice of violation of this License (for any work) from that
+copyright holder, and you cure the violation prior to 30 days after
+your receipt of the notice.
+
+ Termination of your rights under this section does not terminate the
+licenses of parties who have received copies or rights from you under
+this License. If your rights have been terminated and not permanently
+reinstated, you do not qualify to receive new licenses for the same
+material under section 10.
+
+ 9. Acceptance Not Required for Having Copies.
+
+ You are not required to accept this License in order to receive or
+run a copy of the Program. Ancillary propagation of a covered work
+occurring solely as a consequence of using peer-to-peer transmission
+to receive a copy likewise does not require acceptance. However,
+nothing other than this License grants you permission to propagate or
+modify any covered work. These actions infringe copyright if you do
+not accept this License. Therefore, by modifying or propagating a
+covered work, you indicate your acceptance of this License to do so.
+
+ 10. Automatic Licensing of Downstream Recipients.
+
+ Each time you convey a covered work, the recipient automatically
+receives a license from the original licensors, to run, modify and
+propagate that work, subject to this License. You are not responsible
+for enforcing compliance by third parties with this License.
+
+ An "entity transaction" is a transaction transferring control of an
+organization, or substantially all assets of one, or subdividing an
+organization, or merging organizations. If propagation of a covered
+work results from an entity transaction, each party to that
+transaction who receives a copy of the work also receives whatever
+licenses to the work the party's predecessor in interest had or could
+give under the previous paragraph, plus a right to possession of the
+Corresponding Source of the work from the predecessor in interest, if
+the predecessor has it or can get it with reasonable efforts.
+
+ You may not impose any further restrictions on the exercise of the
+rights granted or affirmed under this License. For example, you may
+not impose a license fee, royalty, or other charge for exercise of
+rights granted under this License, and you may not initiate litigation
+(including a cross-claim or counterclaim in a lawsuit) alleging that
+any patent claim is infringed by making, using, selling, offering for
+sale, or importing the Program or any portion of it.
+
+ 11. Patents.
+
+ A "contributor" is a copyright holder who authorizes use under this
+License of the Program or a work on which the Program is based. The
+work thus licensed is called the contributor's "contributor version".
+
+ A contributor's "essential patent claims" are all patent claims
+owned or controlled by the contributor, whether already acquired or
+hereafter acquired, that would be infringed by some manner, permitted
+by this License, of making, using, or selling its contributor version,
+but do not include claims that would be infringed only as a
+consequence of further modification of the contributor version. For
+purposes of this definition, "control" includes the right to grant
+patent sublicenses in a manner consistent with the requirements of
+this License.
+
+ Each contributor grants you a non-exclusive, worldwide, royalty-free
+patent license under the contributor's essential patent claims, to
+make, use, sell, offer for sale, import and otherwise run, modify and
+propagate the contents of its contributor version.
+
+ In the following three paragraphs, a "patent license" is any express
+agreement or commitment, however denominated, not to enforce a patent
+(such as an express permission to practice a patent or covenant not to
+sue for patent infringement). To "grant" such a patent license to a
+party means to make such an agreement or commitment not to enforce a
+patent against the party.
+
+ If you convey a covered work, knowingly relying on a patent license,
+and the Corresponding Source of the work is not available for anyone
+to copy, free of charge and under the terms of this License, through a
+publicly available network server or other readily accessible means,
+then you must either (1) cause the Corresponding Source to be so
+available, or (2) arrange to deprive yourself of the benefit of the
+patent license for this particular work, or (3) arrange, in a manner
+consistent with the requirements of this License, to extend the patent
+license to downstream recipients. "Knowingly relying" means you have
+actual knowledge that, but for the patent license, your conveying the
+covered work in a country, or your recipient's use of the covered work
+in a country, would infringe one or more identifiable patents in that
+country that you have reason to believe are valid.
+
+ If, pursuant to or in connection with a single transaction or
+arrangement, you convey, or propagate by procuring conveyance of, a
+covered work, and grant a patent license to some of the parties
+receiving the covered work authorizing them to use, propagate, modify
+or convey a specific copy of the covered work, then the patent license
+you grant is automatically extended to all recipients of the covered
+work and works based on it.
+
+ A patent license is "discriminatory" if it does not include within
+the scope of its coverage, prohibits the exercise of, or is
+conditioned on the non-exercise of one or more of the rights that are
+specifically granted under this License. You may not convey a covered
+work if you are a party to an arrangement with a third party that is
+in the business of distributing software, under which you make payment
+to the third party based on the extent of your activity of conveying
+the work, and under which the third party grants, to any of the
+parties who would receive the covered work from you, a discriminatory
+patent license (a) in connection with copies of the covered work
+conveyed by you (or copies made from those copies), or (b) primarily
+for and in connection with specific products or compilations that
+contain the covered work, unless you entered into that arrangement,
+or that patent license was granted, prior to 28 March 2007.
+
+ Nothing in this License shall be construed as excluding or limiting
+any implied license or other defenses to infringement that may
+otherwise be available to you under applicable patent law.
+
+ 12. No Surrender of Others' Freedom.
+
+ If conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License. If you cannot convey a
+covered work so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you may
+not convey it at all. For example, if you agree to terms that obligate you
+to collect a royalty for further conveying from those to whom you convey
+the Program, the only way you could satisfy both those terms and this
+License would be to refrain entirely from conveying the Program.
+
+ 13. Use with the GNU Affero General Public License.
+
+ Notwithstanding any other provision of this License, you have
+permission to link or combine any covered work with a work licensed
+under version 3 of the GNU Affero General Public License into a single
+combined work, and to convey the resulting work. The terms of this
+License will continue to apply to the part which is the covered work,
+but the special requirements of the GNU Affero General Public License,
+section 13, concerning interaction through a network will apply to the
+combination as such.
+
+ 14. Revised Versions of this License.
+
+ The Free Software Foundation may publish revised and/or new versions of
+the GNU General Public License from time to time. Such new versions will
+be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+ Each version is given a distinguishing version number. If the
+Program specifies that a certain numbered version of the GNU General
+Public License "or any later version" applies to it, you have the
+option of following the terms and conditions either of that numbered
+version or of any later version published by the Free Software
+Foundation. If the Program does not specify a version number of the
+GNU General Public License, you may choose any version ever published
+by the Free Software Foundation.
+
+ If the Program specifies that a proxy can decide which future
+versions of the GNU General Public License can be used, that proxy's
+public statement of acceptance of a version permanently authorizes you
+to choose that version for the Program.
+
+ Later license versions may give you additional or different
+permissions. However, no additional obligations are imposed on any
+author or copyright holder as a result of your choosing to follow a
+later version.
+
+ 15. Disclaimer of Warranty.
+
+ THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
+APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
+HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
+OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
+THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
+IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
+ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
+
+ 16. Limitation of Liability.
+
+ IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
+THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
+GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
+USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
+DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
+PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
+EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
+SUCH DAMAGES.
+
+ 17. Interpretation of Sections 15 and 16.
+
+ If the disclaimer of warranty and limitation of liability provided
+above cannot be given local legal effect according to their terms,
+reviewing courts shall apply local law that most closely approximates
+an absolute waiver of all civil liability in connection with the
+Program, unless a warranty or assumption of liability accompanies a
+copy of the Program in return for a fee.
+
+ END OF TERMS AND CONDITIONS
+
+ How to Apply These Terms to Your New Programs
+
+ If you develop a new program, and you want it to be of the greatest
+possible use to the public, the best way to achieve this is to make it
+free software which everyone can redistribute and change under these terms.
+
+ To do so, attach the following notices to the program. It is safest
+to attach them to the start of each source file to most effectively
+state the exclusion of warranty; and each file should have at least
+the "copyright" line and a pointer to where the full notice is found.
+
+
+ Copyright (C)
+
+ This program is free software: you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation, either version 3 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program. If not, see .
+
+Also add information on how to contact you by electronic and paper mail.
+
+ If the program does terminal interaction, make it output a short
+notice like this when it starts in an interactive mode:
+
+ Copyright (C)
+ This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
+ This is free software, and you are welcome to redistribute it
+ under certain conditions; type `show c' for details.
+
+The hypothetical commands `show w' and `show c' should show the appropriate
+parts of the General Public License. Of course, your program's commands
+might be different; for a GUI interface, you would use an "about box".
+
+ You should also get your employer (if you work as a programmer) or school,
+if any, to sign a "copyright disclaimer" for the program, if necessary.
+For more information on this, and how to apply and follow the GNU GPL, see
+.
+
+ The GNU General Public License does not permit incorporating your program
+into proprietary programs. If your program is a subroutine library, you
+may consider it more useful to permit linking proprietary applications with
+the library. If this is what you want to do, use the GNU Lesser General
+Public License instead of this License. But first, please read
+.
diff --git a/README.md b/README.md
index 9265b45..014c19c 100644
--- a/README.md
+++ b/README.md
@@ -1,66 +1,165 @@
-## Foundry
+# Angle Tokens
-**Foundry is a blazing fast, portable and modular toolkit for Ethereum application development written in Rust.**
+[![CI](https://github.com/AngleProtocol/boilerplate/actions/workflows/ci.yml/badge.svg)](https://github.com/AngleProtocol/boilerplate/actions)
-Foundry consists of:
+This repository contains all the contracts of the Angle Tokens with associated contracts (CoreBorrow, FlashAngle ...)
-- **Forge**: Ethereum testing framework (like Truffle, Hardhat and DappTools).
-- **Cast**: Swiss army knife for interacting with EVM smart contracts, sending transactions and getting chain data.
-- **Anvil**: Local Ethereum node, akin to Ganache, Hardhat Network.
-- **Chisel**: Fast, utilitarian, and verbose solidity REPL.
+## Starting
-## Documentation
+### Install packages
-https://book.getfoundry.sh/
+You can install all dependencies by running
-## Usage
+```bash
+yarn
+forge i
+```
+
+### Create `.env` file
+
+In order to interact with non local networks, you must create an `.env` that has:
+
+- `PRIVATE_KEY`
+- `MNEMONIC`
+- network key (eg. `ALCHEMY_NETWORK_KEY`)
+- `ETHERSCAN_API_KEY`
+
+For additional keys, you can check the `.env.example` file.
+
+Warning: always keep your confidential information safe.
+
+## Headers
+
+To automatically create headers, follow:
+
+## Hardhat Command line completion
+
+Follow these instructions to have hardhat command line arguments completion:
+
+## Foundry Installation
+
+```bash
+curl -L https://foundry.paradigm.xyz | bash
+
+source /root/.zshrc
+# or, if you're under bash: source /root/.bashrc
+
+foundryup
+```
+
+To install the standard library:
+
+```bash
+forge install foundry-rs/forge-std
+```
+
+To update libraries:
+
+```bash
+forge update
+```
-### Build
+### Foundry on Docker 🐳
-```shell
-$ forge build
+**If you don’t want to install Rust and Foundry on your computer, you can use Docker**
+Image is available here [ghcr.io/foundry-rs/foundry](http://ghcr.io/foundry-rs/foundry).
+
+```bash
+docker pull ghcr.io/foundry-rs/foundry
+docker tag ghcr.io/foundry-rs/foundry:latest foundry:latest
```
-### Test
+To run the container:
-```shell
-$ forge test
+```bash
+docker run -it --rm -v $(pwd):/app -w /app foundry sh
```
-### Format
+Then you are inside the container and can run Foundry’s commands.
+
+### Tests
+
+You can run tests as follows:
-```shell
-$ forge fmt
+```bash
+forge test -vvvv --watch
+forge test -vvvv --match-path contracts/forge-tests/KeeperMulticall.t.sol
+forge test -vvvv --match-test "testAbc*"
+forge test -vvvv --fork-url https://eth-mainnet.alchemyapi.io/v2/Lc7oIGYeL_QvInzI0Wiu_pOZZDEKBrdf
```
-### Gas Snapshots
+You can also list tests:
-```shell
-$ forge snapshot
+```bash
+forge test --list
+forge test --list --json --match-test "testXXX*"
```
-### Anvil
+### Deploying
-```shell
-$ anvil
+There is an example script in the `scripts/foundry` folder. Then you can run:
+
+```bash
+yarn foundry:deploy --rpc-url
```
-### Deploy
+Example:
-```shell
-$ forge script script/Counter.s.sol:CounterScript --rpc-url --private-key
+```bash
+yarn foundry:deploy scripts/foundry/DeployMockAgEUR.s.sol --rpc-url goerli
```
-### Cast
+### Coverage
+
+We recommend the use of this [vscode extension](ryanluker.vscode-coverage-gutters).
-```shell
-$ cast
+```bash
+yarn hardhat:coverage
+yarn foundry:coverage
```
-### Help
+### Simulate
+
+You can simulate your transaction live or in fork mode. For both option you need to
+complete the `scripts/foundry/Simulate.s.sol` with your values: address sending the tx,
+address caled and the data to give to this address call.
-```shell
-$ forge --help
-$ anvil --help
-$ cast --help
+For live simulation
+
+```bash
+yarn foundry:simulate
```
+
+For fork simulation
+
+```bash
+yarn foundry:fork
+yarn foundry:simulate:fork
+```
+
+For fork simulation at a given block
+
+```bash
+yarn foundry:fork:block ${XXXX}
+yarn foundry:simulate:fork
+```
+
+### Gas report
+
+```bash
+yarn foundry:gas
+```
+
+## Slither
+
+```bash
+pip3 install slither-analyzer
+pip3 install solc-select
+solc-select install 0.8.11
+solc-select use 0.8.11
+slither .
+```
+
+## Media
+
+Don't hesitate to reach out on [Twitter](https://twitter.com/AngleProtocol) 🐦
diff --git a/contracts/agToken/AgEUR.sol b/contracts/agToken/AgEUR.sol
new file mode 100644
index 0000000..3338bd7
--- /dev/null
+++ b/contracts/agToken/AgEUR.sol
@@ -0,0 +1,112 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "../interfaces/IAgToken.sol";
+import "../interfaces/coreModule/IStableMaster.sol";
+import "../interfaces/ITreasury.sol";
+import "oz-upgradeable/token/ERC20/extensions/draft-ERC20PermitUpgradeable.sol";
+
+/// @title AgEUR
+/// @author Angle Labs, Inc.
+/// @notice Base contract for agEUR, Angle's Euro stablecoin
+/// @dev This contract is an upgraded version of the agEUR contract that was first deployed on Ethereum mainnet
+contract AgEUR is IAgToken, ERC20PermitUpgradeable {
+ // ================================= REFERENCES ================================
+
+ /// @notice Reference to the `StableMaster` contract associated to agEUR
+ address public stableMaster;
+
+ /// @custom:oz-upgrades-unsafe-allow constructor
+ constructor() initializer {}
+
+ // ============================== ADDED PARAMETERS =============================
+
+ /// @inheritdoc IAgToken
+ mapping(address => bool) public isMinter;
+ /// @notice Reference to the treasury contract which can grant minting rights
+ address public treasury;
+ /// @notice Boolean used to check whether the contract had been reinitialized after an upgrade
+ bool public treasuryInitialized;
+
+ // =================================== EVENTS ==================================
+
+ event TreasuryUpdated(address indexed _treasury);
+ event MinterToggled(address indexed minter);
+
+ // =================================== ERRORS ==================================
+
+ error BurnAmountExceedsAllowance();
+ error InvalidSender();
+ error InvalidTreasury();
+ error NotGovernor();
+ error NotMinter();
+ error NotTreasury();
+ error TreasuryAlreadyInitialized();
+
+ // ================================= MODIFIERS =================================
+
+ /// @notice Checks to see if it is the `Treasury` calling this contract
+ modifier onlyTreasury() {
+ if (msg.sender != treasury) revert NotTreasury();
+ _;
+ }
+
+ /// @notice Checks whether the sender has the minting right
+ modifier onlyMinter() {
+ if (!isMinter[msg.sender]) revert NotMinter();
+ _;
+ }
+
+ // ============================= EXTERNAL FUNCTION =============================
+
+ /// @notice Allows anyone to burn stablecoins
+ /// @param amount Amount of stablecoins to burn
+ /// @dev This function can typically be called if there is a settlement mechanism to burn stablecoins
+ function burnStablecoin(uint256 amount) external {
+ _burn(msg.sender, amount);
+ }
+
+ // ========================= MINTER ROLE ONLY FUNCTIONS ========================
+
+ /// @inheritdoc IAgToken
+ function burnSelf(uint256 amount, address burner) external onlyMinter {
+ _burn(burner, amount);
+ }
+
+ /// @inheritdoc IAgToken
+ function burnFrom(uint256 amount, address burner, address sender) external onlyMinter {
+ if (burner != sender) {
+ uint256 currentAllowance = allowance(burner, sender);
+ if (currentAllowance < amount) revert BurnAmountExceedsAllowance();
+ _approve(burner, sender, currentAllowance - amount);
+ }
+ _burn(burner, amount);
+ }
+
+ /// @inheritdoc IAgToken
+ function mint(address account, uint256 amount) external onlyMinter {
+ _mint(account, amount);
+ }
+
+ // ========================== TREASURY ONLY FUNCTIONS ==========================
+
+ /// @inheritdoc IAgToken
+ function addMinter(address minter) external onlyTreasury {
+ isMinter[minter] = true;
+ emit MinterToggled(minter);
+ }
+
+ /// @inheritdoc IAgToken
+ function removeMinter(address minter) external {
+ if (msg.sender != minter && msg.sender != address(treasury)) revert InvalidSender();
+ isMinter[minter] = false;
+ emit MinterToggled(minter);
+ }
+
+ /// @inheritdoc IAgToken
+ function setTreasury(address _treasury) external onlyTreasury {
+ treasury = _treasury;
+ emit TreasuryUpdated(_treasury);
+ }
+}
diff --git a/contracts/agToken/AgToken.sol b/contracts/agToken/AgToken.sol
new file mode 100644
index 0000000..75db571
--- /dev/null
+++ b/contracts/agToken/AgToken.sol
@@ -0,0 +1,151 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+/*
+ * █
+ ***** ▓▓▓
+ * ▓▓▓▓▓▓▓
+ * ///. ▓▓▓▓▓▓▓▓▓▓▓▓▓
+ ***** //////// ▓▓▓▓▓▓▓
+ * ///////////// ▓▓▓
+ ▓▓ ////////////////// █ ▓▓
+ ▓▓ ▓▓ /////////////////////// ▓▓ ▓▓
+ ▓▓ ▓▓ //////////////////////////// ▓▓ ▓▓
+ ▓▓ ▓▓ /////////▓▓▓///////▓▓▓///////// ▓▓ ▓▓
+ ▓▓ ,////////////////////////////////////// ▓▓ ▓▓
+ ▓▓ ////////////////////////////////////////// ▓▓
+ ▓▓ //////////////////////▓▓▓▓/////////////////////
+ ,////////////////////////////////////////////////////
+ .//////////////////////////////////////////////////////////
+ .//////////////////////////██.,//////////////////////////█
+ .//////////////////////████..,./////////////////////██
+ ...////////////////███████.....,.////////////////███
+ ,.,////////////████████ ........,///////////████
+ .,.,//////█████████ ,.......///////████
+ ,..//████████ ........./████
+ ..,██████ .....,███
+ .██ ,.,█
+
+
+
+ ▓▓ ▓▓▓▓▓▓▓▓▓▓ ▓▓▓▓▓▓▓▓▓▓ ▓▓ ▓▓▓▓▓▓▓▓▓▓
+ ▓▓▓▓▓▓ ▓▓▓ ▓▓▓ ▓▓▓ ▓▓ ▓▓ ▓▓▓▓
+ ▓▓▓ ▓▓▓ ▓▓▓ ▓▓▓ ▓▓▓ ▓▓▓ ▓▓ ▓▓▓▓▓
+ ▓▓▓ ▓▓ ▓▓▓ ▓▓▓ ▓▓▓▓▓▓▓▓▓▓ ▓▓▓▓▓▓▓▓▓▓ ▓▓▓▓▓▓▓▓▓▓
+*/
+
+import "../interfaces/IAgToken.sol";
+import "../interfaces/ITreasury.sol";
+import "oz-upgradeable/token/ERC20/extensions/draft-ERC20PermitUpgradeable.sol";
+
+/// @title AgToken
+/// @author Angle Labs, Inc.
+/// @notice Base contract for Angle agTokens on Ethereum and on other chains
+/// @dev By default, agTokens are ERC-20 tokens with 18 decimals
+contract AgToken is IAgToken, ERC20PermitUpgradeable {
+ // =========================== PARAMETERS / VARIABLES ==========================
+
+ /// @inheritdoc IAgToken
+ mapping(address => bool) public isMinter;
+ /// @notice Reference to the treasury contract which can grant minting rights
+ address public treasury;
+
+ // =================================== EVENTS ==================================
+
+ event TreasuryUpdated(address indexed _treasury);
+ event MinterToggled(address indexed minter);
+
+ // =================================== ERRORS ==================================
+
+ error BurnAmountExceedsAllowance();
+ error InvalidSender();
+ error InvalidTreasury();
+ error NotMinter();
+ error NotTreasury();
+
+ // ================================ CONSTRUCTOR ================================
+
+ /// @custom:oz-upgrades-unsafe-allow constructor
+ constructor() initializer {}
+
+ /// @notice Initializes the `AgToken` contract
+ function initialize(string memory name_, string memory symbol_, address _treasury) external {
+ _initialize(name_, symbol_, _treasury);
+ }
+
+ /// @notice Initializes the contract
+ function _initialize(string memory name_, string memory symbol_, address _treasury) internal virtual initializer {
+ if (address(ITreasury(_treasury).stablecoin()) != address(this)) revert InvalidTreasury();
+ __ERC20Permit_init(name_);
+ __ERC20_init(name_, symbol_);
+ treasury = _treasury;
+ emit TreasuryUpdated(_treasury);
+ }
+
+ // ================================= MODIFIERS =================================
+
+ /// @notice Checks to see if it is the `Treasury` calling this contract
+ modifier onlyTreasury() {
+ if (msg.sender != treasury) revert NotTreasury();
+ _;
+ }
+
+ /// @notice Checks whether the sender has the minting right
+ modifier onlyMinter() {
+ if (!isMinter[msg.sender]) revert NotMinter();
+ _;
+ }
+
+ // ============================= EXTERNAL FUNCTION =============================
+
+ /// @notice Allows anyone to burn stablecoins
+ /// @param amount Amount of stablecoins to burn
+ /// @dev This function can typically be called if there is a settlement mechanism to burn stablecoins
+ function burnStablecoin(uint256 amount) external {
+ _burn(msg.sender, amount);
+ }
+
+ // ========================= MINTER ROLE ONLY FUNCTIONS ========================
+
+ /// @inheritdoc IAgToken
+ function burnSelf(uint256 amount, address burner) external onlyMinter {
+ _burn(burner, amount);
+ }
+
+ /// @inheritdoc IAgToken
+ function burnFrom(uint256 amount, address burner, address sender) external onlyMinter {
+ if (burner != sender) {
+ uint256 currentAllowance = allowance(burner, sender);
+ if (currentAllowance < amount) revert BurnAmountExceedsAllowance();
+ _approve(burner, sender, currentAllowance - amount);
+ }
+ _burn(burner, amount);
+ }
+
+ /// @inheritdoc IAgToken
+ function mint(address account, uint256 amount) external onlyMinter {
+ _mint(account, amount);
+ }
+
+ // ========================== GOVERNANCE ONLY FUNCTIONS ==========================
+
+ /// @inheritdoc IAgToken
+ function addMinter(address minter) external onlyTreasury {
+ isMinter[minter] = true;
+ emit MinterToggled(minter);
+ }
+
+ /// @inheritdoc IAgToken
+ function removeMinter(address minter) external {
+ if (msg.sender != address(treasury) && msg.sender != minter) revert InvalidSender();
+ isMinter[minter] = false;
+ emit MinterToggled(minter);
+ }
+
+ /// @inheritdoc IAgToken
+ function setTreasury(address _treasury) external virtual onlyTreasury {
+ treasury = _treasury;
+ emit TreasuryUpdated(_treasury);
+ }
+}
diff --git a/contracts/agToken/AgTokenSideChainMultiBridge.sol b/contracts/agToken/AgTokenSideChainMultiBridge.sol
new file mode 100644
index 0000000..956d1b9
--- /dev/null
+++ b/contracts/agToken/AgTokenSideChainMultiBridge.sol
@@ -0,0 +1,273 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "./AgToken.sol";
+import "oz/token/ERC20/IERC20.sol";
+import "oz/token/ERC20/utils/SafeERC20.sol";
+
+/// @title AgTokenSideChainMultiBridge
+/// @author Angle Labs, Inc.
+/// @notice Contract for Angle agTokens on other chains than Ethereum mainnet
+/// @dev This contract supports bridge tokens having a minting right on the stablecoin
+/// (also referred to as the canonicalor the native token)
+contract AgTokenSideChainMultiBridge is AgToken {
+ using SafeERC20 for IERC20;
+
+ /// @notice Base used for fee computation
+ uint256 public constant BASE_PARAMS = 1e9;
+
+ // =============================== BRIDGING DATA ===============================
+
+ /// @notice Struct with some data about a specific bridge token
+ struct BridgeDetails {
+ // Limit on the balance of bridge token held by the contract: it is designed
+ // to reduce the exposure of the system to hacks
+ uint256 limit;
+ // Limit on the hourly volume of token minted through this bridge
+ // Technically the limit over a rolling hour is hourlyLimit x2 as hourly limit
+ // is enforced only between x:00 and x+1:00
+ uint256 hourlyLimit;
+ // Fee taken for swapping in and out the token
+ uint64 fee;
+ // Whether the associated token is allowed or not
+ bool allowed;
+ // Whether swapping in and out from the associated token is paused or not
+ bool paused;
+ }
+
+ /// @notice Maps a bridge token to data
+ mapping(address => BridgeDetails) public bridges;
+ /// @notice List of all bridge tokens
+ address[] public bridgeTokensList;
+ /// @notice Maps a bridge token to the associated hourly volume
+ mapping(address => mapping(uint256 => uint256)) public usage;
+ /// @notice Maps an address to whether it is exempt of fees for when it comes to swapping in and out
+ mapping(address => uint256) public isFeeExempt;
+ /// @notice Limit to the amount of tokens that can be sent from that chain to another chain
+ uint256 public chainTotalHourlyLimit;
+ /// @notice Usage per hour on that chain. Maps an hourly timestamp to the total volume swapped out on the chain
+ mapping(uint256 => uint256) public chainTotalUsage;
+
+ uint256[44] private __gapMultiBridge;
+
+ // =================================== EVENTS ==================================
+
+ event BridgeTokenAdded(address indexed bridgeToken, uint256 limit, uint256 hourlyLimit, uint64 fee, bool paused);
+ event BridgeTokenToggled(address indexed bridgeToken, bool toggleStatus);
+ event BridgeTokenRemoved(address indexed bridgeToken);
+ event BridgeTokenFeeUpdated(address indexed bridgeToken, uint64 fee);
+ event BridgeTokenLimitUpdated(address indexed bridgeToken, uint256 limit);
+ event BridgeTokenHourlyLimitUpdated(address indexed bridgeToken, uint256 hourlyLimit);
+ event HourlyLimitUpdated(uint256 hourlyLimit);
+ event Recovered(address indexed token, address indexed to, uint256 amount);
+ event FeeToggled(address indexed theAddress, uint256 toggleStatus);
+
+ // =================================== ERRORS ==================================
+
+ error AssetStillControlledInReserves();
+ error HourlyLimitExceeded();
+ error InvalidToken();
+ error NotGovernor();
+ error NotGovernorOrGuardian();
+ error TooBigAmount();
+ error TooHighParameterValue();
+ error ZeroAddress();
+
+ // ================================= MODIFIERS =================================
+
+ /// @notice Checks whether the `msg.sender` has the governor role or not
+ modifier onlyGovernor() {
+ if (!ITreasury(treasury).isGovernor(msg.sender)) revert NotGovernor();
+ _;
+ }
+
+ /// @notice Checks whether the `msg.sender` has the governor role or the guardian role
+ modifier onlyGovernorOrGuardian() {
+ if (!ITreasury(treasury).isGovernorOrGuardian(msg.sender)) revert NotGovernorOrGuardian();
+ _;
+ }
+
+ // ===================== EXTERNAL PERMISSIONLESS FUNCTIONS =====================
+
+ /// @notice Returns the list of all supported bridge tokens
+ /// @dev Helpful for UIs
+ function allBridgeTokens() external view returns (address[] memory) {
+ return bridgeTokensList;
+ }
+
+ /// @notice Returns the current volume for a bridge, for the current hour
+ /// @param bridgeToken Bridge used to mint
+ /// @dev Helpful for UIs
+ function currentUsage(address bridgeToken) external view returns (uint256) {
+ return usage[bridgeToken][block.timestamp / 3600];
+ }
+
+ /// @notice Returns the current total volume on the chain for the current hour
+ /// @dev Helpful for UIs
+ function currentTotalUsage() external view returns (uint256) {
+ return chainTotalUsage[block.timestamp / 3600];
+ }
+
+ /// @notice Mints the canonical token from a supported bridge token
+ /// @param bridgeToken Bridge token to use to mint
+ /// @param amount Amount of bridge tokens to send
+ /// @param to Address to which the stablecoin should be sent
+ /// @return Amount of the canonical stablecoin actually minted
+ /// @dev Some fees may be taken by the protocol depending on the token used and on the address calling
+ function swapIn(address bridgeToken, uint256 amount, address to) external returns (uint256) {
+ BridgeDetails memory bridgeDetails = bridges[bridgeToken];
+ if (!bridgeDetails.allowed || bridgeDetails.paused) revert InvalidToken();
+ uint256 balance = IERC20(bridgeToken).balanceOf(address(this));
+ if (balance + amount > bridgeDetails.limit) {
+ // In case someone maliciously sends tokens to this contract
+ // Or the limit changes
+ if (bridgeDetails.limit > balance) amount = bridgeDetails.limit - balance;
+ else {
+ amount = 0;
+ }
+ }
+
+ // Checking requirement on the hourly volume
+ uint256 hour = block.timestamp / 3600;
+ uint256 hourlyUsage = usage[bridgeToken][hour];
+ if (hourlyUsage + amount > bridgeDetails.hourlyLimit) {
+ // Edge case when the hourly limit changes
+ amount = bridgeDetails.hourlyLimit > hourlyUsage ? bridgeDetails.hourlyLimit - hourlyUsage : 0;
+ }
+ usage[bridgeToken][hour] = hourlyUsage + amount;
+
+ IERC20(bridgeToken).safeTransferFrom(msg.sender, address(this), amount);
+ uint256 canonicalOut = amount;
+ // Computing fees
+ if (isFeeExempt[msg.sender] == 0) {
+ canonicalOut -= (canonicalOut * bridgeDetails.fee) / BASE_PARAMS;
+ }
+ _mint(to, canonicalOut);
+ return canonicalOut;
+ }
+
+ /// @notice Burns the canonical token in exchange for a bridge token
+ /// @param bridgeToken Bridge token required
+ /// @param amount Amount of canonical tokens to burn
+ /// @param to Address to which the bridge token should be sent
+ /// @return Amount of bridge tokens actually sent back
+ /// @dev Some fees may be taken by the protocol depending on the token used and on the address calling
+ function swapOut(address bridgeToken, uint256 amount, address to) external returns (uint256) {
+ BridgeDetails memory bridgeDetails = bridges[bridgeToken];
+ if (!bridgeDetails.allowed || bridgeDetails.paused) revert InvalidToken();
+
+ uint256 hour = block.timestamp / 3600;
+ uint256 hourlyUsage = chainTotalUsage[hour] + amount;
+ // If the amount being swapped out exceeds the limit, we revert
+ // We don't want to change the amount being swapped out.
+ // The user can decide to send another tx with the correct amount to reach the limit
+ if (hourlyUsage > chainTotalHourlyLimit) revert HourlyLimitExceeded();
+ chainTotalUsage[hour] = hourlyUsage;
+
+ _burn(msg.sender, amount);
+ uint256 bridgeOut = amount;
+ if (isFeeExempt[msg.sender] == 0) {
+ bridgeOut -= (bridgeOut * bridgeDetails.fee) / BASE_PARAMS;
+ }
+ IERC20(bridgeToken).safeTransfer(to, bridgeOut);
+ return bridgeOut;
+ }
+
+ // ============================ GOVERNANCE FUNCTIONS ===========================
+
+ /// @notice Adds support for a bridge token
+ /// @param bridgeToken Bridge token to add: it should be a version of the stablecoin from another bridge
+ /// @param limit Limit on the balance of bridge token this contract could hold
+ /// @param hourlyLimit Limit on the hourly volume for this bridge
+ /// @param paused Whether swapping for this token should be paused or not
+ /// @param fee Fee taken upon swapping for or against this token
+ function addBridgeToken(
+ address bridgeToken,
+ uint256 limit,
+ uint256 hourlyLimit,
+ uint64 fee,
+ bool paused
+ ) external onlyGovernor {
+ if (bridges[bridgeToken].allowed || bridgeToken == address(0)) revert InvalidToken();
+ if (fee > BASE_PARAMS) revert TooHighParameterValue();
+ BridgeDetails memory _bridge;
+ _bridge.limit = limit;
+ _bridge.hourlyLimit = hourlyLimit;
+ _bridge.paused = paused;
+ _bridge.fee = fee;
+ _bridge.allowed = true;
+ bridges[bridgeToken] = _bridge;
+ bridgeTokensList.push(bridgeToken);
+ emit BridgeTokenAdded(bridgeToken, limit, hourlyLimit, fee, paused);
+ }
+
+ /// @notice Removes support for a token
+ /// @param bridgeToken Address of the bridge token to remove support for
+ function removeBridgeToken(address bridgeToken) external onlyGovernor {
+ if (IERC20(bridgeToken).balanceOf(address(this)) != 0) revert AssetStillControlledInReserves();
+ delete bridges[bridgeToken];
+ // Deletion from `bridgeTokensList` loop
+ uint256 bridgeTokensListLength = bridgeTokensList.length;
+ for (uint256 i; i < bridgeTokensListLength - 1; ++i) {
+ if (bridgeTokensList[i] == bridgeToken) {
+ // Replace the `bridgeToken` to remove with the last of the list
+ bridgeTokensList[i] = bridgeTokensList[bridgeTokensListLength - 1];
+ break;
+ }
+ }
+ // Remove last element in array
+ bridgeTokensList.pop();
+ emit BridgeTokenRemoved(bridgeToken);
+ }
+
+ /// @notice Recovers any ERC20 token
+ /// @dev Can be used to withdraw bridge tokens for them to be de-bridged on mainnet
+ function recoverERC20(address tokenAddress, address to, uint256 amountToRecover) external onlyGovernor {
+ IERC20(tokenAddress).safeTransfer(to, amountToRecover);
+ emit Recovered(tokenAddress, to, amountToRecover);
+ }
+
+ /// @notice Updates the `limit` amount for `bridgeToken`
+ function setLimit(address bridgeToken, uint256 limit) external onlyGovernorOrGuardian {
+ if (!bridges[bridgeToken].allowed) revert InvalidToken();
+ bridges[bridgeToken].limit = limit;
+ emit BridgeTokenLimitUpdated(bridgeToken, limit);
+ }
+
+ /// @notice Updates the `hourlyLimit` amount for `bridgeToken`
+ function setHourlyLimit(address bridgeToken, uint256 hourlyLimit) external onlyGovernorOrGuardian {
+ if (!bridges[bridgeToken].allowed) revert InvalidToken();
+ bridges[bridgeToken].hourlyLimit = hourlyLimit;
+ emit BridgeTokenHourlyLimitUpdated(bridgeToken, hourlyLimit);
+ }
+
+ /// @notice Updates the `chainTotalHourlyLimit` amount
+ function setChainTotalHourlyLimit(uint256 hourlyLimit) external onlyGovernorOrGuardian {
+ chainTotalHourlyLimit = hourlyLimit;
+ emit HourlyLimitUpdated(hourlyLimit);
+ }
+
+ /// @notice Updates the `fee` value for `bridgeToken`
+ function setSwapFee(address bridgeToken, uint64 fee) external onlyGovernorOrGuardian {
+ if (!bridges[bridgeToken].allowed) revert InvalidToken();
+ if (fee > BASE_PARAMS) revert TooHighParameterValue();
+ bridges[bridgeToken].fee = fee;
+ emit BridgeTokenFeeUpdated(bridgeToken, fee);
+ }
+
+ /// @notice Pauses or unpauses swapping in and out for a token
+ function toggleBridge(address bridgeToken) external onlyGovernorOrGuardian {
+ if (!bridges[bridgeToken].allowed) revert InvalidToken();
+ bool pausedStatus = bridges[bridgeToken].paused;
+ bridges[bridgeToken].paused = !pausedStatus;
+ emit BridgeTokenToggled(bridgeToken, !pausedStatus);
+ }
+
+ /// @notice Toggles fees for the address `theAddress`
+ function toggleFeesForAddress(address theAddress) external onlyGovernorOrGuardian {
+ uint256 feeExemptStatus = 1 - isFeeExempt[theAddress];
+ isFeeExempt[theAddress] = feeExemptStatus;
+ emit FeeToggled(theAddress, feeExemptStatus);
+ }
+}
diff --git a/contracts/agToken/TokenSideChainMultiBridge.sol b/contracts/agToken/TokenSideChainMultiBridge.sol
new file mode 100644
index 0000000..a3270ca
--- /dev/null
+++ b/contracts/agToken/TokenSideChainMultiBridge.sol
@@ -0,0 +1,333 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "oz/token/ERC20/IERC20.sol";
+import "oz/token/ERC20/utils/SafeERC20.sol";
+import "oz-upgradeable/token/ERC20/extensions/draft-ERC20PermitUpgradeable.sol";
+
+import "../interfaces/ICoreBorrow.sol";
+
+/// @title TokenSideChainMultiBridge
+/// @author Angle Labs, Inc.
+/// @notice Contract for an ERC20 token that exists natively on a chain on other chains than this native chain
+/// @dev This contract supports bridge tokens having a minting right on the ERC20 token (also referred to as the canonical
+/// of the chain)
+/// @dev This implementation assumes that the token bridged has 18 decimals
+contract TokenSideChainMultiBridge is ERC20PermitUpgradeable {
+ using SafeERC20 for IERC20;
+
+ /// @notice Struct with some data about a specific bridge token
+ struct BridgeDetails {
+ // Limit on the balance of bridge token held by the contract: it is designed
+ // to reduce the exposure of the system to hacks
+ uint256 limit;
+ // Limit on the hourly volume of token minted through this bridge
+ // Technically the limit over a rolling hour is hourlyLimit x2 as hourly limit
+ // is enforced only between x:00 and x+1:00
+ uint256 hourlyLimit;
+ // Fee taken for swapping in and out the token
+ uint64 fee;
+ // Whether the associated token is allowed or not
+ bool allowed;
+ // Whether swapping in and out from the associated token is paused or not
+ bool paused;
+ }
+
+ /// @notice Base used for fee computation
+ uint256 public constant BASE_PARAMS = 10 ** 9;
+
+ // ================================= PARAMETER =================================
+
+ /// @notice Reference to the core contract which handles access control
+ ICoreBorrow public core;
+
+ // =============================== BRIDGING DATA ===============================
+
+ /// @notice Maps a bridge token to data
+ mapping(address => BridgeDetails) public bridges;
+ /// @notice List of all bridge tokens
+ address[] public bridgeTokensList;
+ /// @notice Maps a bridge token to the associated hourly volume
+ mapping(address => mapping(uint256 => uint256)) public usage;
+ /// @notice Maps an address to whether it is exempt of fees for when it comes to swapping in and out
+ mapping(address => uint256) public isFeeExempt;
+ /// @notice Limit to the amount of tokens that can be sent from that chain to another chain
+ uint256 public chainTotalHourlyLimit;
+ /// @notice Usage per hour on that chain. Maps an hourly timestamp to the total volume swapped out on the chain
+ mapping(uint256 => uint256) public chainTotalUsage;
+
+ // =================================== ERRORS ==================================
+
+ error AssetStillControlledInReserves();
+ error HourlyLimitExceeded();
+ error InvalidToken();
+ error NotGovernor();
+ error NotGovernorOrGuardian();
+ error TooHighParameterValue();
+ error ZeroAddress();
+
+ // =================================== EVENTS ==================================
+
+ event BridgeTokenAdded(address indexed bridgeToken, uint256 limit, uint256 hourlyLimit, uint64 fee, bool paused);
+ event BridgeTokenToggled(address indexed bridgeToken, bool toggleStatus);
+ event BridgeTokenRemoved(address indexed bridgeToken);
+ event BridgeTokenFeeUpdated(address indexed bridgeToken, uint64 fee);
+ event BridgeTokenLimitUpdated(address indexed bridgeToken, uint256 limit);
+ event BridgeTokenHourlyLimitUpdated(address indexed bridgeToken, uint256 hourlyLimit);
+ event HourlyLimitUpdated(uint256 hourlyLimit);
+ event Recovered(address indexed token, address indexed to, uint256 amount);
+ event FeeToggled(address indexed theAddress, uint256 toggleStatus);
+
+ // ================================= MODIFIERS =================================
+
+ /// @notice Checks whether the `msg.sender` has the governor role or not
+ modifier onlyGovernor() {
+ if (!core.isGovernor(msg.sender)) revert NotGovernor();
+ _;
+ }
+
+ /// @notice Checks whether the `msg.sender` has the governor role or the guardian role
+ modifier onlyGovernorOrGuardian() {
+ if (!core.isGovernorOrGuardian(msg.sender)) revert NotGovernorOrGuardian();
+ _;
+ }
+
+ // ================================ CONSTRUCTOR ================================
+
+ /// @notice Initializes the contract
+ /// @param name_ Name of the token
+ /// @param symbol_ Symbol of the token
+ /// @param _core Reference to the `CoreBorrow` contract
+ /// @dev This implementation allows to add directly at deployment a bridge token
+ function initialize(
+ string memory name_,
+ string memory symbol_,
+ ICoreBorrow _core,
+ address bridgeToken,
+ uint256 limit,
+ uint256 hourlyLimit,
+ uint64 fee,
+ bool paused,
+ uint256 _chainTotalHourlyLimit
+ ) external initializer {
+ __ERC20Permit_init(name_);
+ __ERC20_init(name_, symbol_);
+ if (address(_core) == address(0)) revert ZeroAddress();
+ core = _core;
+ _addBridgeToken(bridgeToken, limit, hourlyLimit, fee, paused);
+ _setChainTotalHourlyLimit(_chainTotalHourlyLimit);
+ }
+
+ /// @custom:oz-upgrades-unsafe-allow constructor
+ constructor() initializer {}
+
+ // ===================== EXTERNAL PERMISSIONLESS FUNCTIONS =====================
+
+ /// @notice Returns the list of all supported bridge tokens
+ /// @dev Helpful for UIs
+ function allBridgeTokens() external view returns (address[] memory) {
+ return bridgeTokensList;
+ }
+
+ /// @notice Returns the current volume for a bridge, for the current hour
+ /// @param bridgeToken Bridge used to mint
+ /// @dev Helpful for UIs
+ function currentUsage(address bridgeToken) external view returns (uint256) {
+ return usage[bridgeToken][block.timestamp / 3600];
+ }
+
+ /// @notice Returns the current total volume on the chain for the current hour
+ /// @dev Helpful for UIs
+ function currentTotalUsage() external view returns (uint256) {
+ return chainTotalUsage[block.timestamp / 3600];
+ }
+
+ /// @notice Mints the canonical token from a supported bridge token
+ /// @param bridgeToken Bridge token to use to mint
+ /// @param amount Amount of bridge tokens to send
+ /// @param to Address to which the token should be sent
+ /// @return Amount of canonical token actually minted
+ /// @dev Some fees may be taken by the protocol depending on the token used and on the address calling
+ function swapIn(address bridgeToken, uint256 amount, address to) external returns (uint256) {
+ BridgeDetails memory bridgeDetails = bridges[bridgeToken];
+ if (!bridgeDetails.allowed || bridgeDetails.paused) revert InvalidToken();
+ uint256 balance = IERC20(bridgeToken).balanceOf(address(this));
+ if (balance + amount > bridgeDetails.limit) {
+ // In case someone maliciously sends tokens to this contract
+ // Or the limit changes
+ if (bridgeDetails.limit > balance) amount = bridgeDetails.limit - balance;
+ else {
+ amount = 0;
+ }
+ }
+
+ // Checking requirement on the hourly volume
+ uint256 hour = block.timestamp / 3600;
+ uint256 hourlyUsage = usage[bridgeToken][hour];
+ if (hourlyUsage + amount > bridgeDetails.hourlyLimit) {
+ // Edge case when the hourly limit changes
+ amount = bridgeDetails.hourlyLimit > hourlyUsage ? bridgeDetails.hourlyLimit - hourlyUsage : 0;
+ }
+ usage[bridgeToken][hour] = hourlyUsage + amount;
+
+ IERC20(bridgeToken).safeTransferFrom(msg.sender, address(this), amount);
+ uint256 canonicalOut = amount;
+ // Computing fees
+ if (isFeeExempt[msg.sender] == 0) {
+ canonicalOut -= (canonicalOut * bridgeDetails.fee) / BASE_PARAMS;
+ }
+ _mint(to, canonicalOut);
+ return canonicalOut;
+ }
+
+ /// @notice Burns the canonical token in exchange for a bridge token
+ /// @param bridgeToken Bridge token required
+ /// @param amount Amount of canonical tokens to burn
+ /// @param to Address to which the bridge token should be sent
+ /// @return Amount of bridge tokens actually sent back
+ /// @dev Some fees may be taken by the protocol depending on the token used and on the address calling
+ function swapOut(address bridgeToken, uint256 amount, address to) external returns (uint256) {
+ BridgeDetails memory bridgeDetails = bridges[bridgeToken];
+ if (!bridgeDetails.allowed || bridgeDetails.paused) revert InvalidToken();
+
+ uint256 hour = block.timestamp / 3600;
+ uint256 hourlyUsage = chainTotalUsage[hour] + amount;
+ // If the amount being swapped out exceeds the limit, we revert
+ // We don't want to change the amount being swapped out.
+ // The user can decide to send another tx with the correct amount to reach the limit
+ if (hourlyUsage > chainTotalHourlyLimit) revert HourlyLimitExceeded();
+ chainTotalUsage[hour] = hourlyUsage;
+
+ _burn(msg.sender, amount);
+ uint256 bridgeOut = amount;
+ if (isFeeExempt[msg.sender] == 0) {
+ bridgeOut -= (bridgeOut * bridgeDetails.fee) / BASE_PARAMS;
+ }
+ IERC20(bridgeToken).safeTransfer(to, bridgeOut);
+ return bridgeOut;
+ }
+
+ // ============================ GOVERNANCE FUNCTIONS ===========================
+
+ /// @notice Sets a new `core` contract
+ /// @dev One sanity check that can be performed here is to verify whether at least the governor
+ /// calling the contract is still a governor in the new core
+ function setCore(ICoreBorrow _core) external onlyGovernor {
+ if (!_core.isGovernor(msg.sender)) revert NotGovernor();
+ core = _core;
+ }
+
+ /// @notice Adds support for a bridge token
+ /// @param bridgeToken Bridge token to add: it should be a version of the canonical token from another bridge
+ /// @param limit Limit on the balance of bridge token this contract could hold
+ /// @param hourlyLimit Limit on the hourly volume for this bridge
+ /// @param paused Whether swapping for this token should be paused or not
+ /// @param fee Fee taken upon swapping for or against this token
+ function addBridgeToken(
+ address bridgeToken,
+ uint256 limit,
+ uint256 hourlyLimit,
+ uint64 fee,
+ bool paused
+ ) external onlyGovernor {
+ _addBridgeToken(bridgeToken, limit, hourlyLimit, fee, paused);
+ }
+
+ /// @notice Removes support for a token
+ /// @param bridgeToken Address of the bridge token to remove support for
+ function removeBridgeToken(address bridgeToken) external onlyGovernor {
+ if (IERC20(bridgeToken).balanceOf(address(this)) != 0) revert AssetStillControlledInReserves();
+ delete bridges[bridgeToken];
+ // Deletion from `bridgeTokensList` loop
+ uint256 bridgeTokensListLength = bridgeTokensList.length;
+ for (uint256 i = 0; i < bridgeTokensListLength - 1; i++) {
+ if (bridgeTokensList[i] == bridgeToken) {
+ // Replace the `bridgeToken` to remove with the last of the list
+ bridgeTokensList[i] = bridgeTokensList[bridgeTokensListLength - 1];
+ break;
+ }
+ }
+ // Remove last element in array
+ bridgeTokensList.pop();
+ emit BridgeTokenRemoved(bridgeToken);
+ }
+
+ /// @notice Recovers any ERC20 token
+ /// @dev Can be used to withdraw bridge tokens for them to be de-bridged on mainnet
+ function recoverERC20(address tokenAddress, address to, uint256 amountToRecover) external onlyGovernor {
+ IERC20(tokenAddress).safeTransfer(to, amountToRecover);
+ emit Recovered(tokenAddress, to, amountToRecover);
+ }
+
+ /// @notice Updates the `limit` amount for `bridgeToken`
+ function setLimit(address bridgeToken, uint256 limit) external onlyGovernorOrGuardian {
+ if (!bridges[bridgeToken].allowed) revert InvalidToken();
+ bridges[bridgeToken].limit = limit;
+ emit BridgeTokenLimitUpdated(bridgeToken, limit);
+ }
+
+ /// @notice Updates the `hourlyLimit` amount for `bridgeToken`
+ function setHourlyLimit(address bridgeToken, uint256 hourlyLimit) external onlyGovernorOrGuardian {
+ if (!bridges[bridgeToken].allowed) revert InvalidToken();
+ bridges[bridgeToken].hourlyLimit = hourlyLimit;
+ emit BridgeTokenHourlyLimitUpdated(bridgeToken, hourlyLimit);
+ }
+
+ /// @notice Updates the `chainTotalHourlyLimit` amount
+ function setChainTotalHourlyLimit(uint256 hourlyLimit) external onlyGovernorOrGuardian {
+ _setChainTotalHourlyLimit(hourlyLimit);
+ }
+
+ /// @notice Updates the `fee` value for `bridgeToken`
+ function setSwapFee(address bridgeToken, uint64 fee) external onlyGovernorOrGuardian {
+ if (!bridges[bridgeToken].allowed) revert InvalidToken();
+ if (fee > BASE_PARAMS) revert TooHighParameterValue();
+ bridges[bridgeToken].fee = fee;
+ emit BridgeTokenFeeUpdated(bridgeToken, fee);
+ }
+
+ /// @notice Pauses or unpauses swapping in and out for a token
+ function toggleBridge(address bridgeToken) external onlyGovernorOrGuardian {
+ if (!bridges[bridgeToken].allowed) revert InvalidToken();
+ bool pausedStatus = bridges[bridgeToken].paused;
+ bridges[bridgeToken].paused = !pausedStatus;
+ emit BridgeTokenToggled(bridgeToken, !pausedStatus);
+ }
+
+ /// @notice Toggles fees for the address `theAddress`
+ function toggleFeesForAddress(address theAddress) external onlyGovernorOrGuardian {
+ uint256 feeExemptStatus = 1 - isFeeExempt[theAddress];
+ isFeeExempt[theAddress] = feeExemptStatus;
+ emit FeeToggled(theAddress, feeExemptStatus);
+ }
+
+ // ============================= INTERNAL FUNCTIONS ============================
+
+ /// @notice Internal version of the `addBridgeToken` function
+ function _addBridgeToken(
+ address bridgeToken,
+ uint256 limit,
+ uint256 hourlyLimit,
+ uint64 fee,
+ bool paused
+ ) internal {
+ if (bridges[bridgeToken].allowed || bridgeToken == address(0)) revert InvalidToken();
+ if (fee > BASE_PARAMS) revert TooHighParameterValue();
+ BridgeDetails memory _bridge;
+ _bridge.limit = limit;
+ _bridge.hourlyLimit = hourlyLimit;
+ _bridge.paused = paused;
+ _bridge.fee = fee;
+ _bridge.allowed = true;
+ bridges[bridgeToken] = _bridge;
+ bridgeTokensList.push(bridgeToken);
+ emit BridgeTokenAdded(bridgeToken, limit, hourlyLimit, fee, paused);
+ }
+
+ /// @notice Internal version of the `setChainTotalHourlyLimit`
+ function _setChainTotalHourlyLimit(uint256 hourlyLimit) internal {
+ chainTotalHourlyLimit = hourlyLimit;
+ emit HourlyLimitUpdated(hourlyLimit);
+ }
+}
diff --git a/contracts/agToken/layerZero/LayerZeroBridge.sol b/contracts/agToken/layerZero/LayerZeroBridge.sol
new file mode 100644
index 0000000..0aee30c
--- /dev/null
+++ b/contracts/agToken/layerZero/LayerZeroBridge.sol
@@ -0,0 +1,130 @@
+// SPDX-License-Identifier: MIT
+
+pragma solidity ^0.8.12;
+
+import "./utils/OFTCore.sol";
+import "oz-upgradeable/security/PausableUpgradeable.sol";
+import "oz/token/ERC20/extensions/draft-IERC20Permit.sol";
+
+/// @title LayerZeroBridge
+/// @author Angle Labs, Inc., forked from https://github.com/LayerZero-Labs/solidity-examples/blob/main/contracts/token/oft/OFT.sol
+/// @notice Contract to be deployed on Ethereum for bridging an AgToken using a bridge intermediate token and LayerZero
+contract LayerZeroBridge is OFTCore, PausableUpgradeable {
+ /// @notice Name of the contract for indexing purposes
+ string public name;
+
+ /// @notice Address of the bridgeable token
+ /// @dev Immutable
+ IERC20 public canonicalToken;
+
+ /// @notice Maps an address to the amount of token bridged but not received
+ mapping(address => uint256) public balanceOf;
+
+ // ================================ CONSTRUCTOR ================================
+ /// @notice Initializes the contract
+ /// @param _name Name of the token corresponding to this contract
+ /// @param _lzEndpoint Layer zero endpoint to pass messages
+ /// @param _treasury Address of the treasury contract used for access control
+ function initialize(string memory _name, address _lzEndpoint, address _treasury) external initializer {
+ __LzAppUpgradeable_init(_lzEndpoint, _treasury);
+ name = _name;
+ canonicalToken = IERC20(address(ITreasury(_treasury).stablecoin()));
+ }
+
+ /// @custom:oz-upgrades-unsafe-allow constructor
+ constructor() initializer {}
+
+ // ===================== EXTERNAL PERMISSIONLESS FUNCTIONS =====================
+
+ /// @inheritdoc OFTCore
+ function sendWithPermit(
+ uint16 _dstChainId,
+ bytes memory _toAddress,
+ uint256 _amount,
+ address payable _refundAddress,
+ address _zroPaymentAddress,
+ bytes memory _adapterParams,
+ uint256 deadline,
+ uint8 v,
+ bytes32 r,
+ bytes32 s
+ ) public payable override {
+ IERC20Permit(address(canonicalToken)).permit(msg.sender, address(this), _amount, deadline, v, r, s);
+ send(_dstChainId, _toAddress, _amount, _refundAddress, _zroPaymentAddress, _adapterParams);
+ }
+
+ /// @inheritdoc OFTCore
+ function withdraw(uint256 amount, address recipient) external override returns (uint256) {
+ return _withdraw(amount, msg.sender, recipient);
+ }
+
+ /// @notice Withdraws amount of `token` from the contract and sends it to the recipient
+ /// @param amount Amount to withdraw
+ /// @param recipient Address to withdraw for
+ /// @return The amount of canonical token sent
+ function withdrawFor(uint256 amount, address recipient) external returns (uint256) {
+ return _withdraw(amount, recipient, recipient);
+ }
+
+ // ============================= INTERNAL FUNCTIONS ============================
+
+ /// @notice Withdraws `amount` from the balance of the `from` address and sends these tokens to the `to` address
+ /// @dev It's important to make sure that `from` is either the `msg.sender` or that `from` and `to` are the same
+ /// addresses
+ function _withdraw(uint256 amount, address from, address to) internal whenNotPaused returns (uint256) {
+ balanceOf[from] -= amount; // Will overflow if the amount is too big
+ canonicalToken.transfer(to, amount);
+ return amount;
+ }
+
+ /// @inheritdoc OFTCore
+ function _debitFrom(uint16, bytes memory, uint256 _amount) internal override whenNotPaused returns (uint256) {
+ // No need to use safeTransferFrom as we know this implementation reverts on failure
+ canonicalToken.transferFrom(msg.sender, address(this), _amount);
+ return _amount;
+ }
+
+ /// @inheritdoc OFTCore
+ function _debitCreditFrom(uint16, bytes memory, uint256 _amount) internal override whenNotPaused returns (uint256) {
+ balanceOf[msg.sender] -= _amount;
+ return _amount;
+ }
+
+ /// @inheritdoc OFTCore
+ function _creditTo(uint16, address _toAddress, uint256 _amount) internal override whenNotPaused returns (uint256) {
+ // Should never revert as all the LayerZero bridge tokens come from
+ // this contract
+ uint256 balance = canonicalToken.balanceOf(address(this));
+ if (balance < _amount) {
+ balanceOf[_toAddress] = _amount - balance;
+ if (balance != 0) canonicalToken.transfer(_toAddress, balance);
+ } else {
+ canonicalToken.transfer(_toAddress, _amount);
+ }
+ return _amount;
+ }
+
+ // =============================== VIEW FUNCTIONS ==============================
+
+ /// @inheritdoc ERC165Upgradeable
+ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
+ return interfaceId == type(IOFTCore).interfaceId || super.supportsInterface(interfaceId);
+ }
+
+ // ============================ GOVERNANCE FUNCTIONS ===========================
+
+ /// @notice Pauses bridging through the contract
+ /// @param pause Future pause status
+ function pauseSendTokens(bool pause) external onlyGovernorOrGuardian {
+ pause ? _pause() : _unpause();
+ }
+
+ /// @notice Decreases the balance of an address
+ /// @param amount Amount to withdraw from balance
+ /// @param recipient Address to withdraw from
+ function sweep(uint256 amount, address recipient) external onlyGovernorOrGuardian {
+ balanceOf[recipient] -= amount; // Will overflow if the amount is too big
+ }
+
+ uint256[47] private __gap;
+}
diff --git a/contracts/agToken/layerZero/LayerZeroBridgeToken.sol b/contracts/agToken/layerZero/LayerZeroBridgeToken.sol
new file mode 100644
index 0000000..5e11903
--- /dev/null
+++ b/contracts/agToken/layerZero/LayerZeroBridgeToken.sol
@@ -0,0 +1,151 @@
+// SPDX-License-Identifier: MIT
+
+pragma solidity ^0.8.12;
+
+import "./utils/OFTCore.sol";
+import "../../interfaces/IAgTokenSideChainMultiBridge.sol";
+import "oz-upgradeable/security/PausableUpgradeable.sol";
+import "oz-upgradeable/token/ERC20/ERC20Upgradeable.sol";
+
+/// @title LayerZeroBridgeToken
+/// @author Angle Labs, Inc., forked from https://github.com/LayerZero-Labs/solidity-examples/blob/main/contracts/token/oft/OFT.sol
+/// @notice Contract to be deployed on a L2/sidechain for bridging an AgToken using a bridge intermediate token and LayerZero
+contract LayerZeroBridgeToken is OFTCore, ERC20Upgradeable, PausableUpgradeable {
+ /// @notice Address of the bridgeable token
+ /// @dev Immutable
+ IAgTokenSideChainMultiBridge public canonicalToken;
+
+ // =================================== ERROR ===================================
+
+ error InvalidAllowance();
+
+ // ================================ CONSTRUCTOR ================================
+
+ /// @notice Initializes the contract
+ /// @param _name Name of the token corresponding to this contract
+ /// @param _symbol Symbol of the token corresponding to this contract
+ /// @param _lzEndpoint Layer zero endpoint to pass messages
+ /// @param _treasury Address of the treasury contract used for access control
+ /// @param initialSupply Initial supply to mint to the canonical token address
+ /// @dev The initial supply corresponds to the initial amount that could be bridged using this OFT
+ function initialize(
+ string memory _name,
+ string memory _symbol,
+ address _lzEndpoint,
+ address _treasury,
+ uint256 initialSupply
+ ) external initializer {
+ __ERC20_init_unchained(_name, _symbol);
+ __LzAppUpgradeable_init(_lzEndpoint, _treasury);
+
+ canonicalToken = IAgTokenSideChainMultiBridge(address(ITreasury(_treasury).stablecoin()));
+ _approve(address(this), address(canonicalToken), type(uint256).max);
+ _mint(address(canonicalToken), initialSupply);
+ }
+
+ /// @custom:oz-upgrades-unsafe-allow constructor
+ constructor() initializer {}
+
+ // ===================== EXTERNAL PERMISSIONLESS FUNCTIONS =====================
+
+ /// @inheritdoc OFTCore
+ function sendWithPermit(
+ uint16 _dstChainId,
+ bytes memory _toAddress,
+ uint256 _amount,
+ address payable _refundAddress,
+ address _zroPaymentAddress,
+ bytes memory _adapterParams,
+ uint256 deadline,
+ uint8 v,
+ bytes32 r,
+ bytes32 s
+ ) public payable override {
+ canonicalToken.permit(msg.sender, address(this), _amount, deadline, v, r, s);
+ send(_dstChainId, _toAddress, _amount, _refundAddress, _zroPaymentAddress, _adapterParams);
+ }
+
+ /// @inheritdoc OFTCore
+ function withdraw(uint256 amount, address recipient) external override returns (uint256 amountMinted) {
+ // Does not check allowances as transfers from `msg.sender`
+ _transfer(msg.sender, address(this), amount);
+ amountMinted = canonicalToken.swapIn(address(this), amount, recipient);
+ uint256 leftover = balanceOf(address(this));
+ if (leftover != 0) {
+ _transfer(address(this), msg.sender, leftover);
+ }
+ }
+
+ // ============================= INTERNAL FUNCTIONS ============================
+
+ /// @inheritdoc OFTCore
+ function _debitFrom(
+ uint16,
+ bytes memory,
+ uint256 _amount
+ ) internal override whenNotPaused returns (uint256 amountSwapped) {
+ // No need to use safeTransferFrom as we know this implementation reverts on failure
+ canonicalToken.transferFrom(msg.sender, address(this), _amount);
+
+ // Swap canonical for this bridge token. There may be some fees
+ amountSwapped = canonicalToken.swapOut(address(this), _amount, address(this));
+ _burn(address(this), amountSwapped);
+ }
+
+ /// @inheritdoc OFTCore
+ function _debitCreditFrom(uint16, bytes memory, uint256 _amount) internal override whenNotPaused returns (uint256) {
+ _burn(msg.sender, _amount);
+ return _amount;
+ }
+
+ /// @inheritdoc OFTCore
+ function _creditTo(
+ uint16,
+ address _toAddress,
+ uint256 _amount
+ ) internal override whenNotPaused returns (uint256 amountMinted) {
+ _mint(address(this), _amount);
+ amountMinted = canonicalToken.swapIn(address(this), _amount, _toAddress);
+ uint256 leftover = balanceOf(address(this));
+ if (leftover != 0) {
+ _transfer(address(this), _toAddress, leftover);
+ }
+ }
+
+ // =============================== VIEW FUNCTIONS ==============================
+
+ /// @inheritdoc ERC165Upgradeable
+ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
+ return
+ interfaceId == type(IOFT).interfaceId ||
+ interfaceId == type(IERC20).interfaceId ||
+ super.supportsInterface(interfaceId);
+ }
+
+ // ============================ GOVERNANCE FUNCTIONS ===========================
+
+ /// @notice Mints the intermediate contract to the `canonicalToken`
+ /// @dev Used to increase the bridging capacity
+ function mint(uint256 amount) external onlyGovernorOrGuardian {
+ _mint(address(canonicalToken), amount);
+ }
+
+ /// @notice Burns the intermediate contract from the `canonicalToken`
+ /// @dev Used to decrease the bridging capacity
+ function burn(uint256 amount) external onlyGovernorOrGuardian {
+ _burn(address(canonicalToken), amount);
+ }
+
+ /// @notice Increases allowance of the `canonicalToken`
+ function setupAllowance() public onlyGovernorOrGuardian {
+ _approve(address(this), address(canonicalToken), type(uint256).max);
+ }
+
+ /// @notice Pauses bridging through the contract
+ /// @param pause Future pause status
+ function pauseSendTokens(bool pause) external onlyGovernorOrGuardian {
+ pause ? _pause() : _unpause();
+ }
+
+ uint256[49] private __gap;
+}
diff --git a/contracts/agToken/layerZero/utils/IOFTCore.sol b/contracts/agToken/layerZero/utils/IOFTCore.sol
new file mode 100644
index 0000000..61b9306
--- /dev/null
+++ b/contracts/agToken/layerZero/utils/IOFTCore.sol
@@ -0,0 +1,110 @@
+// SPDX-License-Identifier: MIT
+
+pragma solidity ^0.8.12;
+
+import "oz/token/ERC20/IERC20.sol";
+import "oz/utils/introspection/IERC165.sol";
+
+/**
+ * @dev Interface of the IOFT core standard
+ * @dev Forked from https://github.com/LayerZero-Labs/solidity-examples/blob/main/contracts/token/oft/IOFTCore.sol
+ */
+interface IOFTCore is IERC165 {
+ /// @notice Estimates send token `_tokenId` to (`_dstChainId`, `_toAddress`)
+ /// @param _dstChainId L0 defined chain id to send tokens too
+ /// @param _toAddress dynamic bytes array which contains the address to whom you are sending
+ /// tokens to on the dstChain
+ /// @param _amount amount of the tokens to transfer
+ /// @param _useZro indicates to use zro to pay L0 fees
+ /// @param _adapterParams flexible bytes array to indicate messaging adapter services in L0
+ function estimateSendFee(
+ uint16 _dstChainId,
+ bytes calldata _toAddress,
+ uint256 _amount,
+ bool _useZro,
+ bytes calldata _adapterParams
+ ) external view returns (uint256 nativeFee, uint256 zroFee);
+
+ /// @notice Sends `_amount` amount of token to (`_dstChainId`, `_toAddress`)
+ /// @param _dstChainId the destination chain identifier
+ /// @param _toAddress can be any size depending on the `dstChainId`.
+ /// @param _amount the quantity of tokens in wei
+ /// @param _refundAddress the address LayerZero refunds if too much message fee is sent
+ /// @param _zroPaymentAddress set to address(0x0) if not paying in ZRO (LayerZero Token)
+ /// @param _adapterParams is a flexible bytes array to indicate messaging adapter services
+ function send(
+ uint16 _dstChainId,
+ bytes calldata _toAddress,
+ uint256 _amount,
+ address payable _refundAddress,
+ address _zroPaymentAddress,
+ bytes calldata _adapterParams
+ ) external payable;
+
+ /// @notice Sends `_amount` amount of credit to (`_dstChainId`, `_toAddress`)
+ /// @param _dstChainId the destination chain identifier
+ /// @param _toAddress can be any size depending on the `dstChainId`.
+ /// @param _amount the quantity of credit to send in wei
+ /// @param _refundAddress the address LayerZero refunds if too much message fee is sent
+ /// @param _zroPaymentAddress set to address(0x0) if not paying in ZRO (LayerZero Token)
+ /// @param _adapterParams is a flexible bytes array to indicate messaging adapter services
+ function sendCredit(
+ uint16 _dstChainId,
+ bytes calldata _toAddress,
+ uint256 _amount,
+ address payable _refundAddress,
+ address _zroPaymentAddress,
+ bytes calldata _adapterParams
+ ) external payable;
+
+ /// @notice Sends `_amount` amount of token to (`_dstChainId`, `_toAddress`)
+ /// @param _dstChainId The destination chain identifier
+ /// @param _toAddress Can be any size depending on the `dstChainId`.
+ /// @param _amount Quantity of tokens in wei
+ /// @param _refundAddress Address LayerZero refunds if too much message fee is sent
+ /// @param _zroPaymentAddress Set to address(0x0) if not paying in ZRO (LayerZero Token)
+ /// @param _adapterParams Flexible bytes array to indicate messaging adapter services
+ /// @param deadline Deadline parameter for the signature to be valid
+ /// @dev The `v`, `r`, and `s` parameters are used as signature data
+ function sendWithPermit(
+ uint16 _dstChainId,
+ bytes memory _toAddress,
+ uint256 _amount,
+ address payable _refundAddress,
+ address _zroPaymentAddress,
+ bytes memory _adapterParams,
+ uint256 deadline,
+ uint8 v,
+ bytes32 r,
+ bytes32 s
+ ) external payable;
+
+ /// @notice Withdraws amount of canonical token from the `msg.sender` balance and sends it to the recipient
+ /// @param amount Amount to withdraw
+ /// @param recipient Address to send the canonical token to
+ /// @return The amount of canonical token sent
+ function withdraw(uint256 amount, address recipient) external returns (uint256);
+
+ /// @dev Emitted when `_amount` tokens are moved from the `_sender` to (`_dstChainId`, `_toAddress`)
+ /// `_nonce` is the outbound nonce
+ event SendToChain(
+ address indexed _sender,
+ uint16 indexed _dstChainId,
+ bytes indexed _toAddress,
+ uint256 _amount,
+ uint64 _nonce
+ );
+
+ /// @dev Emitted when `_amount` tokens are received from `_srcChainId` into the `_toAddress` on the local chain.
+ /// `_nonce` is the inbound nonce.
+ event ReceiveFromChain(
+ uint16 indexed _srcChainId,
+ bytes indexed _srcAddress,
+ address indexed _toAddress,
+ uint256 _amount,
+ uint64 _nonce
+ );
+}
+
+/// @dev Interface of the OFT standard
+interface IOFT is IOFTCore, IERC20 {}
diff --git a/contracts/agToken/layerZero/utils/NonblockingLzApp.sol b/contracts/agToken/layerZero/utils/NonblockingLzApp.sol
new file mode 100644
index 0000000..9390e0f
--- /dev/null
+++ b/contracts/agToken/layerZero/utils/NonblockingLzApp.sol
@@ -0,0 +1,281 @@
+// SPDX-License-Identifier: MIT
+
+pragma solidity ^0.8.12;
+
+import "oz-upgradeable/proxy/utils/Initializable.sol";
+import "layer-zero/interfaces/ILayerZeroReceiver.sol";
+import "layer-zero/interfaces/ILayerZeroUserApplicationConfig.sol";
+import "layer-zero/interfaces/ILayerZeroEndpoint.sol";
+import "../../../interfaces/ITreasury.sol";
+
+/// @title NonblockingLzApp
+/// @author Angle Labs, Inc., forked from https://github.com/LayerZero-Labs/solidity-examples/
+/// @notice Base contract for bridging using LayerZero
+abstract contract NonblockingLzApp is Initializable, ILayerZeroReceiver, ILayerZeroUserApplicationConfig {
+ /// @notice Layer Zero endpoint
+ ILayerZeroEndpoint public lzEndpoint;
+
+ /// @notice Maps chainIds to failed messages to retry them
+ mapping(uint16 => mapping(bytes => mapping(uint64 => bytes32))) public failedMessages;
+
+ /// @notice Maps chainIds to their OFT address
+ mapping(uint16 => bytes) public trustedRemoteLookup;
+
+ /// @notice Reference to the treasury contract to fetch access control
+ address public treasury;
+
+ /// @notice Maps pairs of (`to` chain, `packetType`) to the minimum amount of gas needed on the destination chain
+ mapping(uint16 => mapping(uint16 => uint256)) public minDstGasLookup;
+
+ /// @notice For future LayerZero compatibility
+ address public precrime;
+
+ // ================================== Events ===================================
+
+ event SetTrustedRemote(uint16 _srcChainId, bytes _srcAddress);
+ event MessageFailed(uint16 _srcChainId, bytes _srcAddress, uint64 _nonce, bytes _payload);
+
+ // =============================== Errors ================================
+
+ error NotGovernor();
+ error NotGovernorOrGuardian();
+ error InsufficientGas();
+ error InvalidEndpoint();
+ error InvalidSource();
+ error InvalidCaller();
+ error InvalidParams();
+ error InvalidPayload();
+ error ZeroAddress();
+
+ // ============================= Constructor ===================================
+
+ //solhint-disable-next-line
+ function __LzAppUpgradeable_init(address _endpoint, address _treasury) internal {
+ if (_endpoint == address(0) || _treasury == address(0)) revert ZeroAddress();
+ lzEndpoint = ILayerZeroEndpoint(_endpoint);
+ treasury = _treasury;
+ }
+
+ // =============================== Modifiers ===================================
+
+ /// @notice Checks whether the `msg.sender` has the governor role or the guardian role
+ modifier onlyGovernorOrGuardian() {
+ if (!ITreasury(treasury).isGovernorOrGuardian(msg.sender)) revert NotGovernorOrGuardian();
+ _;
+ }
+
+ // ==================== External Permissionless Functions ======================
+
+ /// @notice Receives a message from the LZ endpoint and process it
+ /// @param _srcChainId ChainId of the source chain - LayerZero standard
+ /// @param _srcAddress Sender of the source chain
+ /// @param _nonce Nounce of the message
+ /// @param _payload Data: recipient address and amount
+ function lzReceive(
+ uint16 _srcChainId,
+ bytes memory _srcAddress,
+ uint64 _nonce,
+ bytes memory _payload
+ ) public virtual override {
+ // lzReceive must be called by the endpoint for security
+ if (msg.sender != address(lzEndpoint)) revert InvalidEndpoint();
+
+ bytes memory trustedRemote = trustedRemoteLookup[_srcChainId];
+ // if will still block the message pathway from (srcChainId, srcAddress).
+ // should not receive message from untrusted remote.
+ if (_srcAddress.length != trustedRemote.length || keccak256(_srcAddress) != keccak256(trustedRemote))
+ revert InvalidSource();
+
+ _blockingLzReceive(_srcChainId, _srcAddress, _nonce, _payload);
+ }
+
+ /// @notice Retries a message that previously failed and was stored
+ /// @param _srcChainId ChainId of the source chain - LayerZero standard
+ /// @param _srcAddress Sender of the source chain
+ /// @param _nonce Nounce of the message
+ /// @param _payload Data: recipient address and amount
+ function retryMessage(
+ uint16 _srcChainId,
+ bytes memory _srcAddress,
+ uint64 _nonce,
+ bytes memory _payload
+ ) public payable virtual {
+ // assert there is message to retry
+ bytes32 payloadHash = failedMessages[_srcChainId][_srcAddress][_nonce];
+ if (payloadHash == bytes32(0) || keccak256(_payload) != payloadHash) revert InvalidPayload();
+ // clear the stored message
+ failedMessages[_srcChainId][_srcAddress][_nonce] = bytes32(0);
+ // execute the message. revert if it fails again
+ _nonblockingLzReceive(_srcChainId, _srcAddress, _nonce, _payload);
+ }
+
+ // ============================= Internal Functions ===================================
+
+ /// @notice Handles message receptions in a non blocking way
+ /// @param _srcChainId ChainId of the source chain - LayerZero standard
+ /// @param _srcAddress Sender of the source chain
+ /// @param _nonce Nounce of the message
+ /// @param _payload Data: recipient address and amount
+ /// @dev public for the needs of try / catch but effectively internal
+ function nonblockingLzReceive(
+ uint16 _srcChainId,
+ bytes memory _srcAddress,
+ uint64 _nonce,
+ bytes memory _payload
+ ) public virtual {
+ // only internal transaction
+ if (msg.sender != address(this)) revert InvalidCaller();
+ _nonblockingLzReceive(_srcChainId, _srcAddress, _nonce, _payload);
+ }
+
+ /// @notice Handles message receptions in a non blocking way
+ /// @param _srcChainId ChainId of the source chain - LayerZero standard
+ /// @param _srcAddress Sender of the source chain
+ /// @param _nonce Nounce of the message
+ /// @param _payload Data: recipient address and amount
+ function _nonblockingLzReceive(
+ uint16 _srcChainId,
+ bytes memory _srcAddress,
+ uint64 _nonce,
+ bytes memory _payload
+ ) internal virtual;
+
+ /// @notice Handles message receptions in a blocking way
+ /// @param _srcChainId ChainId of the source chain - LayerZero standard
+ /// @param _srcAddress Sender of the source chain
+ /// @param _nonce Nounce of the message
+ /// @param _payload Data: recipient address and amount
+ function _blockingLzReceive(
+ uint16 _srcChainId,
+ bytes memory _srcAddress,
+ uint64 _nonce,
+ bytes memory _payload
+ ) internal {
+ // try-catch all errors/exceptions
+ try this.nonblockingLzReceive(_srcChainId, _srcAddress, _nonce, _payload) {
+ // do nothing
+ } catch {
+ // error / exception
+ failedMessages[_srcChainId][_srcAddress][_nonce] = keccak256(_payload);
+ emit MessageFailed(_srcChainId, _srcAddress, _nonce, _payload);
+ }
+ }
+
+ /// @notice Sends a message to the LZ endpoint and process it
+ /// @param _dstChainId L0 defined chain id to send tokens too
+ /// @param _payload Data: recipient address and amount
+ /// @param _refundAddress Address LayerZero refunds if too much message fee is sent
+ /// @param _zroPaymentAddress Set to address(0x0) if not paying in ZRO (LayerZero Token)
+ /// @param _adapterParams Flexible bytes array to indicate messaging adapter services in L0
+ function _lzSend(
+ uint16 _dstChainId,
+ bytes memory _payload,
+ address payable _refundAddress,
+ address _zroPaymentAddress,
+ bytes memory _adapterParams
+ ) internal virtual {
+ bytes memory trustedRemote = trustedRemoteLookup[_dstChainId];
+ if (trustedRemote.length == 0) revert InvalidSource();
+ //solhint-disable-next-line
+ lzEndpoint.send{ value: msg.value }(
+ _dstChainId,
+ trustedRemote,
+ _payload,
+ _refundAddress,
+ _zroPaymentAddress,
+ _adapterParams
+ );
+ }
+
+ /// @notice Checks the gas limit of a given transaction
+ function _checkGasLimit(
+ uint16 _dstChainId,
+ uint16 _type,
+ bytes memory _adapterParams,
+ uint256 _extraGas
+ ) internal view virtual {
+ uint256 minGasLimit = minDstGasLookup[_dstChainId][_type] + _extraGas;
+ if (minGasLimit == 0 || minGasLimit > _getGasLimit(_adapterParams)) revert InsufficientGas();
+ }
+
+ /// @notice Gets the gas limit from the `_adapterParams` parameter
+ function _getGasLimit(bytes memory _adapterParams) internal pure virtual returns (uint256 gasLimit) {
+ if (_adapterParams.length < 34) revert InvalidParams();
+ // solhint-disable-next-line
+ assembly {
+ gasLimit := mload(add(_adapterParams, 34))
+ }
+ }
+
+ // ======================= Governance Functions ================================
+
+ /// @notice Sets the corresponding address on an other chain.
+ /// @param _srcChainId ChainId of the source chain - LayerZero standard
+ /// @param _srcAddress Address on the source chain
+ /// @dev Used for both receiving and sending message
+ /// @dev There can only be one trusted source per chain
+ /// @dev Allows owner to set it multiple times.
+ function setTrustedRemote(uint16 _srcChainId, bytes calldata _srcAddress) external onlyGovernorOrGuardian {
+ trustedRemoteLookup[_srcChainId] = _srcAddress;
+ emit SetTrustedRemote(_srcChainId, _srcAddress);
+ }
+
+ /// @notice Fetches the default LZ config
+ function getConfig(
+ uint16 _version,
+ uint16 _chainId,
+ address,
+ uint256 _configType
+ ) external view returns (bytes memory) {
+ return lzEndpoint.getConfig(_version, _chainId, address(this), _configType);
+ }
+
+ /// @notice Overrides the default LZ config
+ function setConfig(
+ uint16 _version,
+ uint16 _chainId,
+ uint256 _configType,
+ bytes calldata _config
+ ) external override onlyGovernorOrGuardian {
+ lzEndpoint.setConfig(_version, _chainId, _configType, _config);
+ }
+
+ /// @notice Overrides the default LZ config
+ function setSendVersion(uint16 _version) external override onlyGovernorOrGuardian {
+ lzEndpoint.setSendVersion(_version);
+ }
+
+ /// @notice Overrides the default LZ config
+ function setReceiveVersion(uint16 _version) external override onlyGovernorOrGuardian {
+ lzEndpoint.setReceiveVersion(_version);
+ }
+
+ /// @notice Unpauses the receive functionalities
+ function forceResumeReceive(
+ uint16 _srcChainId,
+ bytes calldata _srcAddress
+ ) external override onlyGovernorOrGuardian {
+ lzEndpoint.forceResumeReceive(_srcChainId, _srcAddress);
+ }
+
+ /// @notice Sets the minimum gas parameter for a packet type on a given chain
+ function setMinDstGas(uint16 _dstChainId, uint16 _packetType, uint256 _minGas) external onlyGovernorOrGuardian {
+ if (_minGas == 0) revert InvalidParams();
+ minDstGasLookup[_dstChainId][_packetType] = _minGas;
+ }
+
+ /// @notice Sets the precrime variable
+ function setPrecrime(address _precrime) external onlyGovernorOrGuardian {
+ precrime = _precrime;
+ }
+
+ // ======================= View Functions ================================
+
+ /// @notice Checks if the `_srcAddress` corresponds to the trusted source
+ function isTrustedRemote(uint16 _srcChainId, bytes calldata _srcAddress) external view returns (bool) {
+ bytes memory trustedSource = trustedRemoteLookup[_srcChainId];
+ return keccak256(trustedSource) == keccak256(_srcAddress);
+ }
+
+ uint256[44] private __gap;
+}
diff --git a/contracts/agToken/layerZero/utils/OFTCore.sol b/contracts/agToken/layerZero/utils/OFTCore.sol
new file mode 100644
index 0000000..ec7641d
--- /dev/null
+++ b/contracts/agToken/layerZero/utils/OFTCore.sol
@@ -0,0 +1,185 @@
+// SPDX-License-Identifier: MIT
+
+pragma solidity ^0.8.12;
+
+import "./NonblockingLzApp.sol";
+import "./IOFTCore.sol";
+import "oz-upgradeable/utils/introspection/ERC165Upgradeable.sol";
+
+/// @title OFTCore
+/// @author Forked from https://github.com/LayerZero-Labs/solidity-examples/blob/main/contracts/token/oft/OFTCore.sol
+/// but with slight modifications from the Angle Labs, Inc. which added return values to the `_creditTo`
+/// and `_debitFrom` functions
+/// @notice Base contract for bridging using LayerZero
+abstract contract OFTCore is NonblockingLzApp, ERC165Upgradeable, IOFTCore {
+ /// @notice Amount of additional gas specified
+ uint256 public constant EXTRA_GAS = 200000;
+ /// @notice Packet type for token transfer
+ uint16 public constant PT_SEND = 0;
+
+ /// @notice Whether to use custom parameters in transactions
+ uint8 public useCustomAdapterParams;
+
+ // ==================== External Permissionless Functions ======================
+
+ /// @inheritdoc IOFTCore
+ function sendWithPermit(
+ uint16 _dstChainId,
+ bytes memory _toAddress,
+ uint256 _amount,
+ address payable _refundAddress,
+ address _zroPaymentAddress,
+ bytes memory _adapterParams,
+ uint256 deadline,
+ uint8 v,
+ bytes32 r,
+ bytes32 s
+ ) public payable virtual;
+
+ /// @inheritdoc IOFTCore
+ function send(
+ uint16 _dstChainId,
+ bytes memory _toAddress,
+ uint256 _amount,
+ address payable _refundAddress,
+ address _zroPaymentAddress,
+ bytes memory _adapterParams
+ ) public payable virtual {
+ _checkAdapterParams(_dstChainId, PT_SEND, _adapterParams, EXTRA_GAS);
+ _amount = _debitFrom(_dstChainId, _toAddress, _amount);
+
+ bytes memory payload = abi.encode(_toAddress, _amount);
+ _lzSend(_dstChainId, payload, _refundAddress, _zroPaymentAddress, _adapterParams);
+
+ uint64 nonce = lzEndpoint.getOutboundNonce(_dstChainId, address(this));
+ emit SendToChain(msg.sender, _dstChainId, _toAddress, _amount, nonce);
+ }
+
+ /// @inheritdoc IOFTCore
+ function sendCredit(
+ uint16 _dstChainId,
+ bytes memory _toAddress,
+ uint256 _amount,
+ address payable _refundAddress,
+ address _zroPaymentAddress,
+ bytes memory _adapterParams
+ ) public payable virtual {
+ _checkAdapterParams(_dstChainId, PT_SEND, _adapterParams, EXTRA_GAS);
+ _amount = _debitCreditFrom(_dstChainId, _toAddress, _amount);
+
+ _send(_dstChainId, _toAddress, _amount, _refundAddress, _zroPaymentAddress, _adapterParams);
+ }
+
+ /// @inheritdoc IOFTCore
+ function withdraw(uint256 amount, address recipient) external virtual returns (uint256);
+
+ /// @notice Sets whether custom adapter parameters can be used or not
+ function setUseCustomAdapterParams(uint8 _useCustomAdapterParams) public virtual onlyGovernorOrGuardian {
+ useCustomAdapterParams = _useCustomAdapterParams;
+ }
+
+ // =========================== Internal Functions ==============================
+
+ /// @notice Internal function to send `_amount` amount of token to (`_dstChainId`, `_toAddress`)
+ /// @param _dstChainId the destination chain identifier
+ /// @param _toAddress can be any size depending on the `dstChainId`.
+ /// @param _amount the quantity of tokens in wei
+ /// @param _refundAddress the address LayerZero refunds if too much message fee is sent
+ /// @param _zroPaymentAddress set to address(0x0) if not paying in ZRO (LayerZero Token)
+ /// @param _adapterParams is a flexible bytes array to indicate messaging adapter services
+ /// @dev Accounting and checks should be performed beforehand
+ function _send(
+ uint16 _dstChainId,
+ bytes memory _toAddress,
+ uint256 _amount,
+ address payable _refundAddress,
+ address _zroPaymentAddress,
+ bytes memory _adapterParams
+ ) internal {
+ bytes memory payload = abi.encode(_toAddress, _amount);
+ _lzSend(_dstChainId, payload, _refundAddress, _zroPaymentAddress, _adapterParams);
+
+ uint64 nonce = lzEndpoint.getOutboundNonce(_dstChainId, address(this));
+ emit SendToChain(msg.sender, _dstChainId, _toAddress, _amount, nonce);
+ }
+
+ /// @inheritdoc NonblockingLzApp
+ function _nonblockingLzReceive(
+ uint16 _srcChainId,
+ bytes memory _srcAddress,
+ uint64 _nonce,
+ bytes memory _payload
+ ) internal virtual override {
+ // decode and load the toAddress
+ (bytes memory toAddressBytes, uint256 amount) = abi.decode(_payload, (bytes, uint256));
+ address toAddress;
+ //solhint-disable-next-line
+ assembly {
+ toAddress := mload(add(toAddressBytes, 20))
+ }
+ amount = _creditTo(_srcChainId, toAddress, amount);
+
+ emit ReceiveFromChain(_srcChainId, _srcAddress, toAddress, amount, _nonce);
+ }
+
+ /// @notice Checks the adapter parameters given during the smart contract call
+ function _checkAdapterParams(
+ uint16 _dstChainId,
+ uint16 _pkType,
+ bytes memory _adapterParams,
+ uint256 _extraGas
+ ) internal virtual {
+ if (useCustomAdapterParams > 0) _checkGasLimit(_dstChainId, _pkType, _adapterParams, _extraGas);
+ else if (_adapterParams.length != 0) revert InvalidParams();
+ }
+
+ /// @notice Makes accountability when bridging from this contract using canonical token
+ /// @param _dstChainId ChainId of the destination chain - LayerZero standard
+ /// @param _toAddress Recipient on the destination chain
+ /// @param _amount Amount to bridge
+ function _debitFrom(
+ uint16 _dstChainId,
+ bytes memory _toAddress,
+ uint256 _amount
+ ) internal virtual returns (uint256);
+
+ /// @notice Makes accountability when bridging from this contract's credit
+ /// @param _dstChainId ChainId of the destination chain - LayerZero standard
+ /// @param _toAddress Recipient on the destination chain
+ /// @param _amount Amount to bridge
+ function _debitCreditFrom(
+ uint16 _dstChainId,
+ bytes memory _toAddress,
+ uint256 _amount
+ ) internal virtual returns (uint256);
+
+ /// @notice Makes accountability when bridging to this contract
+ /// @param _srcChainId ChainId of the source chain - LayerZero standard
+ /// @param _toAddress Recipient on this chain
+ /// @param _amount Amount to bridge
+ function _creditTo(uint16 _srcChainId, address _toAddress, uint256 _amount) internal virtual returns (uint256);
+
+ // ========================== View Functions ===================================
+
+ /// @inheritdoc ERC165Upgradeable
+ function supportsInterface(
+ bytes4 interfaceId
+ ) public view virtual override(ERC165Upgradeable, IERC165) returns (bool) {
+ return interfaceId == type(IOFTCore).interfaceId || super.supportsInterface(interfaceId);
+ }
+
+ /// @inheritdoc IOFTCore
+ function estimateSendFee(
+ uint16 _dstChainId,
+ bytes memory _toAddress,
+ uint256 _amount,
+ bool _useZro,
+ bytes memory _adapterParams
+ ) public view virtual override returns (uint256 nativeFee, uint256 zroFee) {
+ // mock the payload for send()
+ bytes memory payload = abi.encode(_toAddress, _amount);
+ return lzEndpoint.estimateFees(_dstChainId, address(this), payload, _useZro, _adapterParams);
+ }
+
+ uint256[49] private __gap;
+}
diff --git a/contracts/agToken/nameable/AgEURNameable.sol b/contracts/agToken/nameable/AgEURNameable.sol
new file mode 100644
index 0000000..b8cf714
--- /dev/null
+++ b/contracts/agToken/nameable/AgEURNameable.sol
@@ -0,0 +1,37 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "../AgEUR.sol";
+
+/// @title AgEURNameable
+/// @author Angle Labs, Inc.
+contract AgEURNameable is AgEUR {
+ /// @notice Checks whether the `msg.sender` has the governor role or not
+ modifier onlyGovernor() {
+ if (!ITreasury(treasury).isGovernor(msg.sender)) revert NotGovernor();
+ _;
+ }
+
+ string internal __name;
+
+ string internal __symbol;
+
+ uint256[48] private __gapNameable;
+
+ /// @inheritdoc ERC20Upgradeable
+ function name() public view override returns (string memory) {
+ return __name;
+ }
+
+ /// @inheritdoc ERC20Upgradeable
+ function symbol() public view override returns (string memory) {
+ return __symbol;
+ }
+
+ /// @notice Updates the name and symbol of the token
+ function setNameAndSymbol(string memory newName, string memory newSymbol) external onlyGovernor {
+ __name = newName;
+ __symbol = newSymbol;
+ }
+}
diff --git a/contracts/agToken/nameable/AgTokenNameable.sol b/contracts/agToken/nameable/AgTokenNameable.sol
new file mode 100644
index 0000000..783c37d
--- /dev/null
+++ b/contracts/agToken/nameable/AgTokenNameable.sol
@@ -0,0 +1,38 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "../AgToken.sol";
+
+/// @title AgTokenNameable
+/// @author Angle Labs, Inc.
+contract AgTokenNameable is AgToken {
+ error NotGovernor();
+
+ /// @notice Checks whether the `msg.sender` has the governor role or not
+ modifier onlyGovernor() {
+ if (!ITreasury(treasury).isGovernor(msg.sender)) revert NotGovernor();
+ _;
+ }
+ string internal __name;
+
+ string internal __symbol;
+
+ uint256[48] private __gapNameable;
+
+ /// @inheritdoc ERC20Upgradeable
+ function name() public view override returns (string memory) {
+ return __name;
+ }
+
+ /// @inheritdoc ERC20Upgradeable
+ function symbol() public view override returns (string memory) {
+ return __symbol;
+ }
+
+ /// @notice Updates the name and symbol of the token
+ function setNameAndSymbol(string memory newName, string memory newSymbol) external onlyGovernor {
+ __name = newName;
+ __symbol = newSymbol;
+ }
+}
diff --git a/contracts/agToken/nameable/AgTokenSideChainMultiBridgeNameable.sol b/contracts/agToken/nameable/AgTokenSideChainMultiBridgeNameable.sol
new file mode 100644
index 0000000..ff00a6c
--- /dev/null
+++ b/contracts/agToken/nameable/AgTokenSideChainMultiBridgeNameable.sol
@@ -0,0 +1,31 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "../AgTokenSideChainMultiBridge.sol";
+
+/// @title AgTokenSideChainMultiBridgeNameable
+/// @author Angle Labs, Inc.
+contract AgTokenSideChainMultiBridgeNameable is AgTokenSideChainMultiBridge {
+ string internal __name;
+
+ string internal __symbol;
+
+ uint256[48] private __gapNameable;
+
+ /// @inheritdoc ERC20Upgradeable
+ function name() public view override returns (string memory) {
+ return __name;
+ }
+
+ /// @inheritdoc ERC20Upgradeable
+ function symbol() public view override returns (string memory) {
+ return __symbol;
+ }
+
+ /// @notice Updates the name and symbol of the token
+ function setNameAndSymbol(string memory newName, string memory newSymbol) external onlyGovernor {
+ __name = newName;
+ __symbol = newSymbol;
+ }
+}
diff --git a/contracts/agToken/nameable/TokenPolygonUpgradeableNameable.sol b/contracts/agToken/nameable/TokenPolygonUpgradeableNameable.sol
new file mode 100644
index 0000000..948c67b
--- /dev/null
+++ b/contracts/agToken/nameable/TokenPolygonUpgradeableNameable.sol
@@ -0,0 +1,31 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "../polygon/TokenPolygonUpgradeable.sol";
+
+/// @title TokenPolygonUpgradeableNameable
+/// @author Angle Labs, Inc.
+contract TokenPolygonUpgradeableNameable is TokenPolygonUpgradeable {
+ string internal __name;
+
+ string internal __symbol;
+
+ uint256[48] private __gapNameable;
+
+ /// @inheritdoc ERC20UpgradeableCustom
+ function name() public view override returns (string memory) {
+ return __name;
+ }
+
+ /// @inheritdoc ERC20UpgradeableCustom
+ function symbol() public view override returns (string memory) {
+ return __symbol;
+ }
+
+ /// @notice Updates the name and symbol of the token
+ function setNameAndSymbol(string memory newName, string memory newSymbol) external onlyGovernor {
+ __name = newName;
+ __symbol = newSymbol;
+ }
+}
diff --git a/contracts/agToken/polygon/TokenPolygonUpgradeable.sol b/contracts/agToken/polygon/TokenPolygonUpgradeable.sol
new file mode 100644
index 0000000..14b627e
--- /dev/null
+++ b/contracts/agToken/polygon/TokenPolygonUpgradeable.sol
@@ -0,0 +1,424 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.12;
+
+import "./utils/ERC20UpgradeableCustom.sol";
+import "oz-upgradeable/access/AccessControlUpgradeable.sol";
+import "oz-upgradeable/proxy/utils/Initializable.sol";
+import "oz-upgradeable/utils/cryptography/draft-EIP712Upgradeable.sol";
+import "oz/token/ERC20/IERC20.sol";
+import "oz/token/ERC20/utils/SafeERC20.sol";
+
+import "../../interfaces/IAgToken.sol";
+import "../../interfaces/ITreasury.sol";
+
+interface IChildToken {
+ function deposit(address user, bytes calldata depositData) external;
+
+ function withdraw(uint256 amount) external;
+}
+
+contract TokenPolygonUpgradeable is
+ Initializable,
+ ERC20UpgradeableCustom,
+ AccessControlUpgradeable,
+ EIP712Upgradeable,
+ IChildToken
+{
+ bytes32 public constant DEPOSITOR_ROLE = keccak256("DEPOSITOR_ROLE");
+
+ /// @dev Emitted when the child chain manager changes
+ event ChildChainManagerAdded(address newAddress);
+ event ChildChainManagerRevoked(address oldAddress);
+
+ constructor() initializer {}
+
+ function initialize(
+ string memory _name,
+ string memory _symbol,
+ address childChainManager,
+ address guardian
+ ) public initializer {
+ __ERC20_init(_name, _symbol);
+ __AccessControl_init();
+ _setupRole(DEFAULT_ADMIN_ROLE, guardian);
+ _setupRole(DEPOSITOR_ROLE, childChainManager);
+ __EIP712_init(_name, "1");
+ }
+
+ /**
+ * @notice Called when the bridge has tokens to mint
+ * @param user Address to mint the token to
+ * @param depositData Encoded amount to mint
+ */
+ function deposit(address user, bytes calldata depositData) external override {
+ require(hasRole(DEPOSITOR_ROLE, msg.sender));
+ uint256 amount = abi.decode(depositData, (uint256));
+ _mint(user, amount);
+ }
+
+ /**
+ * @notice Called when user wants to withdraw tokens back to root chain
+ * @dev Should burn user's tokens. This transaction will be verified when exiting on root chain
+ * @param amount Amount of tokens to withdraw
+ */
+ function withdraw(uint256 amount) external override {
+ _burn(_msgSender(), amount);
+ }
+
+ // =============================================================================
+ // ======================= New data added for the upgrade ======================
+ // =============================================================================
+
+ mapping(address => bool) public isMinter;
+ /// @notice Reference to the treasury contract which can grant minting rights
+ address public treasury;
+ /// @notice Boolean to check whether the contract has been reinitialized after its upgrade
+ bool public treasuryInitialized;
+
+ using SafeERC20 for IERC20;
+
+ /// @notice Base used for fee computation
+ uint256 public constant BASE_PARAMS = 10 ** 9;
+
+ // =============================== Bridging Data ===============================
+
+ /// @notice Struct with some data about a specific bridge token
+ struct BridgeDetails {
+ // Limit on the balance of bridge token held by the contract: it is designed
+ // to reduce the exposure of the system to hacks
+ uint256 limit;
+ // Limit on the hourly volume of token minted through this bridge
+ // Technically the limit over a rolling hour is hourlyLimit x2 as hourly limit
+ // is enforced only between x:00 and x+1:00
+ uint256 hourlyLimit;
+ // Fee taken for swapping in and out the token
+ uint64 fee;
+ // Whether the associated token is allowed or not
+ bool allowed;
+ // Whether swapping in and out from the associated token is paused or not
+ bool paused;
+ }
+
+ /// @notice Maps a bridge token to data
+ mapping(address => BridgeDetails) public bridges;
+ /// @notice List of all bridge tokens
+ address[] public bridgeTokensList;
+ /// @notice Maps a bridge token to the associated hourly volume
+ mapping(address => mapping(uint256 => uint256)) public usage;
+ /// @notice Maps an address to whether it is exempt of fees for when it comes to swapping in and out
+ mapping(address => uint256) public isFeeExempt;
+ /// @notice Limit to the amount of tokens that can be sent from that chain to another chain
+ uint256 public chainTotalHourlyLimit;
+ /// @notice Usage per hour on that chain. Maps an hourly timestamp to the total volume swapped out on the chain
+ mapping(uint256 => uint256) public chainTotalUsage;
+
+ uint256[42] private __gap;
+
+ // ================================== Events ===================================
+
+ event BridgeTokenAdded(address indexed bridgeToken, uint256 limit, uint256 hourlyLimit, uint64 fee, bool paused);
+ event BridgeTokenToggled(address indexed bridgeToken, bool toggleStatus);
+ event BridgeTokenRemoved(address indexed bridgeToken);
+ event BridgeTokenFeeUpdated(address indexed bridgeToken, uint64 fee);
+ event BridgeTokenLimitUpdated(address indexed bridgeToken, uint256 limit);
+ event BridgeTokenHourlyLimitUpdated(address indexed bridgeToken, uint256 hourlyLimit);
+ event HourlyLimitUpdated(uint256 hourlyLimit);
+ event FeeToggled(address indexed theAddress, uint256 toggleStatus);
+ event KeeperToggled(address indexed keeper, bool toggleStatus);
+ event MinterToggled(address indexed minter);
+ event Recovered(address indexed token, address indexed to, uint256 amount);
+ event TreasuryUpdated(address indexed _treasury);
+
+ // ================================== Errors ===================================
+
+ error AssetStillControlledInReserves();
+ error BurnAmountExceedsAllowance();
+ error HourlyLimitExceeded();
+ error InvalidSender();
+ error InvalidToken();
+ error InvalidTreasury();
+ error NotGovernor();
+ error NotGovernorOrGuardian();
+ error NotMinter();
+ error NotTreasury();
+ error TooBigAmount();
+ error TooHighParameterValue();
+ error TreasuryAlreadyInitialized();
+ error ZeroAddress();
+
+ /// @notice Checks to see if it is the `Treasury` calling this contract
+ /// @dev There is no Access Control here, because it can be handled cheaply through this modifier
+ modifier onlyTreasury() {
+ if (msg.sender != treasury) revert NotTreasury();
+ _;
+ }
+
+ /// @notice Checks whether the sender has the minting right
+ modifier onlyMinter() {
+ if (!isMinter[msg.sender]) revert NotMinter();
+ _;
+ }
+
+ /// @notice Checks whether the `msg.sender` has the governor role or not
+ modifier onlyGovernor() {
+ if (!ITreasury(treasury).isGovernor(msg.sender)) revert NotGovernor();
+ _;
+ }
+
+ /// @notice Checks whether the `msg.sender` has the governor role or the guardian role
+ modifier onlyGovernorOrGuardian() {
+ if (!ITreasury(treasury).isGovernorOrGuardian(msg.sender)) revert NotGovernorOrGuardian();
+ _;
+ }
+
+ /// @notice Sets up the treasury contract on Polygon after the upgrade
+ /// @param _treasury Address of the treasury contract
+ function setUpTreasury(address _treasury) external {
+ // Only governor on Polygon
+ if (msg.sender != 0xdA2D2f638D6fcbE306236583845e5822554c02EA) revert NotGovernor();
+ if (address(ITreasury(_treasury).stablecoin()) != address(this)) revert InvalidTreasury();
+ if (treasuryInitialized) revert TreasuryAlreadyInitialized();
+ treasury = _treasury;
+ treasuryInitialized = true;
+ emit TreasuryUpdated(_treasury);
+ }
+
+ // =========================== External Function ===============================
+
+ /// @notice Allows anyone to burn agToken without redeeming collateral back
+ /// @param amount Amount of stablecoins to burn
+ /// @dev This function can typically be called if there is a settlement mechanism to burn stablecoins
+ function burnStablecoin(uint256 amount) external {
+ _burnCustom(msg.sender, amount);
+ }
+
+ // ======================= Minter Role Only Functions ==========================
+
+ function burnSelf(uint256 amount, address burner) external onlyMinter {
+ _burnCustom(burner, amount);
+ }
+
+ function burnFrom(uint256 amount, address burner, address sender) external onlyMinter {
+ _burnFromNoRedeem(amount, burner, sender);
+ }
+
+ function mint(address account, uint256 amount) external onlyMinter {
+ _mint(account, amount);
+ }
+
+ // ======================= Treasury Only Functions =============================
+
+ function addMinter(address minter) external onlyTreasury {
+ isMinter[minter] = true;
+ emit MinterToggled(minter);
+ }
+
+ function removeMinter(address minter) external {
+ if (msg.sender != address(treasury) && msg.sender != minter) revert InvalidSender();
+ isMinter[minter] = false;
+ emit MinterToggled(minter);
+ }
+
+ function setTreasury(address _treasury) external onlyTreasury {
+ treasury = _treasury;
+ emit TreasuryUpdated(_treasury);
+ }
+
+ // ============================ Internal Function ==============================
+
+ /// @notice Internal version of the function `burnFromNoRedeem`
+ /// @param amount Amount to burn
+ /// @dev It is at the level of this function that allowance checks are performed
+ function _burnFromNoRedeem(uint256 amount, address burner, address sender) internal {
+ if (burner != sender) {
+ uint256 currentAllowance = allowance(burner, sender);
+ if (currentAllowance < amount) revert BurnAmountExceedsAllowance();
+ _approve(burner, sender, currentAllowance - amount);
+ }
+ _burnCustom(burner, amount);
+ }
+
+ // ==================== External Permissionless Functions ======================
+
+ /// @notice Returns the list of all supported bridge tokens
+ /// @dev Helpful for UIs
+ function allBridgeTokens() external view returns (address[] memory) {
+ return bridgeTokensList;
+ }
+
+ /// @notice Returns the current volume for a bridge, for the current hour
+ /// @dev Helpful for UIs
+ function currentUsage(address bridge) external view returns (uint256) {
+ return usage[bridge][block.timestamp / 3600];
+ }
+
+ /// @notice Returns the current total volume on the chain for the current hour
+ /// @dev Helpful for UIs
+ function currentTotalUsage() external view returns (uint256) {
+ return chainTotalUsage[block.timestamp / 3600];
+ }
+
+ /// @notice Mints the canonical token from a supported bridge token
+ /// @param bridgeToken Bridge token to use to mint
+ /// @param amount Amount of bridge tokens to send
+ /// @param to Address to which the stablecoin should be sent
+ /// @dev Some fees may be taken by the protocol depending on the token used and on the address calling
+ function swapIn(address bridgeToken, uint256 amount, address to) external returns (uint256) {
+ BridgeDetails memory bridgeDetails = bridges[bridgeToken];
+ if (!bridgeDetails.allowed || bridgeDetails.paused) revert InvalidToken();
+ uint256 balance = IERC20(bridgeToken).balanceOf(address(this));
+ if (balance + amount > bridgeDetails.limit) {
+ // In case someone maliciously sends tokens to this contract
+ // Or the limit changes
+ if (bridgeDetails.limit > balance) amount = bridgeDetails.limit - balance;
+ else {
+ amount = 0;
+ }
+ }
+
+ // Checking requirement on the hourly volume
+ uint256 hour = block.timestamp / 3600;
+ uint256 hourlyUsage = usage[bridgeToken][hour] + amount;
+ if (hourlyUsage > bridgeDetails.hourlyLimit) {
+ // Edge case when the hourly limit changes
+ if (bridgeDetails.hourlyLimit > usage[bridgeToken][hour])
+ amount = bridgeDetails.hourlyLimit - usage[bridgeToken][hour];
+ else {
+ amount = 0;
+ }
+ }
+ usage[bridgeToken][hour] += amount;
+
+ IERC20(bridgeToken).safeTransferFrom(msg.sender, address(this), amount);
+ uint256 canonicalOut = amount;
+ // Computing fees
+ if (isFeeExempt[msg.sender] == 0) {
+ canonicalOut -= (canonicalOut * bridgeDetails.fee) / BASE_PARAMS;
+ }
+ _mint(to, canonicalOut);
+ return canonicalOut;
+ }
+
+ /// @notice Burns the canonical token in exchange for a bridge token
+ /// @param bridgeToken Bridge token required
+ /// @param amount Amount of canonical tokens to burn
+ /// @param to Address to which the bridge token should be sent
+ /// @dev Some fees may be taken by the protocol depending on the token used and on the address calling
+ function swapOut(address bridgeToken, uint256 amount, address to) external returns (uint256) {
+ BridgeDetails memory bridgeDetails = bridges[bridgeToken];
+ if (!bridgeDetails.allowed || bridgeDetails.paused) revert InvalidToken();
+
+ uint256 hour = block.timestamp / 3600;
+ uint256 hourlyUsage = chainTotalUsage[hour] + amount;
+ // If the amount being swapped out exceeds the limit, we revert
+ // We don't want to change the amount being swapped out.
+ // The user can decide to send another tx with the correct amount to reach the limit
+ if (hourlyUsage > chainTotalHourlyLimit) revert HourlyLimitExceeded();
+ chainTotalUsage[hour] = hourlyUsage;
+
+ _burnCustom(msg.sender, amount);
+ uint256 bridgeOut = amount;
+ if (isFeeExempt[msg.sender] == 0) {
+ bridgeOut -= (bridgeOut * bridgeDetails.fee) / BASE_PARAMS;
+ }
+ IERC20(bridgeToken).safeTransfer(to, bridgeOut);
+ return bridgeOut;
+ }
+
+ // ======================= Governance Functions ================================
+
+ /// @notice Adds support for a bridge token
+ /// @param bridgeToken Bridge token to add: it should be a version of the stablecoin from another bridge
+ /// @param limit Limit on the balance of bridge token this contract could hold
+ /// @param hourlyLimit Limit on the hourly volume for this bridge
+ /// @param paused Whether swapping for this token should be paused or not
+ /// @param fee Fee taken upon swapping for or against this token
+ function addBridgeToken(
+ address bridgeToken,
+ uint256 limit,
+ uint256 hourlyLimit,
+ uint64 fee,
+ bool paused
+ ) external onlyGovernor {
+ if (bridges[bridgeToken].allowed || bridgeToken == address(0)) revert InvalidToken();
+ if (fee > BASE_PARAMS) revert TooHighParameterValue();
+ BridgeDetails memory _bridge;
+ _bridge.limit = limit;
+ _bridge.hourlyLimit = hourlyLimit;
+ _bridge.paused = paused;
+ _bridge.fee = fee;
+ _bridge.allowed = true;
+ bridges[bridgeToken] = _bridge;
+ bridgeTokensList.push(bridgeToken);
+ emit BridgeTokenAdded(bridgeToken, limit, hourlyLimit, fee, paused);
+ }
+
+ /// @notice Removes support for a token
+ /// @param bridgeToken Address of the bridge token to remove support for
+ function removeBridgeToken(address bridgeToken) external onlyGovernor {
+ if (IERC20(bridgeToken).balanceOf(address(this)) != 0) revert AssetStillControlledInReserves();
+ delete bridges[bridgeToken];
+ // Deletion from `bridgeTokensList` loop
+ uint256 bridgeTokensListLength = bridgeTokensList.length;
+ for (uint256 i; i < bridgeTokensListLength - 1; ++i) {
+ if (bridgeTokensList[i] == bridgeToken) {
+ // Replace the `bridgeToken` to remove with the last of the list
+ bridgeTokensList[i] = bridgeTokensList[bridgeTokensListLength - 1];
+ break;
+ }
+ }
+ // Remove last element in array
+ bridgeTokensList.pop();
+ emit BridgeTokenRemoved(bridgeToken);
+ }
+
+ /// @notice Recovers any ERC20 token
+ /// @dev Can be used to withdraw bridge tokens for them to be de-bridged on mainnet
+ function recoverERC20(address tokenAddress, address to, uint256 amountToRecover) external onlyGovernor {
+ IERC20(tokenAddress).safeTransfer(to, amountToRecover);
+ emit Recovered(tokenAddress, to, amountToRecover);
+ }
+
+ /// @notice Updates the `limit` amount for `bridgeToken`
+ function setLimit(address bridgeToken, uint256 limit) external onlyGovernorOrGuardian {
+ if (!bridges[bridgeToken].allowed) revert InvalidToken();
+ bridges[bridgeToken].limit = limit;
+ emit BridgeTokenLimitUpdated(bridgeToken, limit);
+ }
+
+ /// @notice Updates the `hourlyLimit` amount for `bridgeToken`
+ function setHourlyLimit(address bridgeToken, uint256 hourlyLimit) external onlyGovernorOrGuardian {
+ if (!bridges[bridgeToken].allowed) revert InvalidToken();
+ bridges[bridgeToken].hourlyLimit = hourlyLimit;
+ emit BridgeTokenHourlyLimitUpdated(bridgeToken, hourlyLimit);
+ }
+
+ /// @notice Updates the `chainTotalHourlyLimit` amount
+ function setChainTotalHourlyLimit(uint256 hourlyLimit) external onlyGovernorOrGuardian {
+ chainTotalHourlyLimit = hourlyLimit;
+ emit HourlyLimitUpdated(hourlyLimit);
+ }
+
+ /// @notice Updates the `fee` value for `bridgeToken`
+ function setSwapFee(address bridgeToken, uint64 fee) external onlyGovernorOrGuardian {
+ if (!bridges[bridgeToken].allowed) revert InvalidToken();
+ if (fee > BASE_PARAMS) revert TooHighParameterValue();
+ bridges[bridgeToken].fee = fee;
+ emit BridgeTokenFeeUpdated(bridgeToken, fee);
+ }
+
+ /// @notice Pauses or unpauses swapping in and out for a token
+ function toggleBridge(address bridgeToken) external onlyGovernorOrGuardian {
+ if (!bridges[bridgeToken].allowed) revert InvalidToken();
+ bool pausedStatus = bridges[bridgeToken].paused;
+ bridges[bridgeToken].paused = !pausedStatus;
+ emit BridgeTokenToggled(bridgeToken, !pausedStatus);
+ }
+
+ /// @notice Toggles fees for the address `theAddress`
+ function toggleFeesForAddress(address theAddress) external onlyGovernorOrGuardian {
+ uint256 feeExemptStatus = 1 - isFeeExempt[theAddress];
+ isFeeExempt[theAddress] = feeExemptStatus;
+ emit FeeToggled(theAddress, feeExemptStatus);
+ }
+}
diff --git a/contracts/agToken/polygon/utils/ERC20UpgradeableCustom.sol b/contracts/agToken/polygon/utils/ERC20UpgradeableCustom.sol
new file mode 100644
index 0000000..43515d6
--- /dev/null
+++ b/contracts/agToken/polygon/utils/ERC20UpgradeableCustom.sol
@@ -0,0 +1,377 @@
+// SPDX-License-Identifier: MIT
+// OpenZeppelin Contracts v4.4.1 (token/ERC20/ERC20.sol)
+
+// solhint-disable custom-errors
+
+pragma solidity ^0.8.0;
+
+import "oz-upgradeable/token/ERC20/IERC20Upgradeable.sol";
+import "oz-upgradeable/token/ERC20/extensions/IERC20MetadataUpgradeable.sol";
+import "oz-upgradeable/utils/ContextUpgradeable.sol";
+import "oz-upgradeable/proxy/utils/Initializable.sol";
+
+/**
+ * @dev Implementation of the {IERC20} interface modified by Angle Labs, Inc.
+ *
+ * This implementation has a custom burn function to avoid having a {Transfer} event to the zero address
+ * in some specific burn cases to avoid having Polygon PoS bridge catching this event
+ *
+ * TIP: For a detailed writeup see our guide
+ * https://forum.zeppelin.solutions/t/how-to-implement-erc20-supply-mechanisms/226[How
+ * to implement supply mechanisms].
+ *
+ * We have followed general OpenZeppelin Contracts guidelines: functions revert
+ * instead returning `false` on failure. This behavior is nonetheless
+ * conventional and does not conflict with the expectations of ERC20
+ * applications.
+ *
+ * Additionally, an {Approval} event is emitted on calls to {transferFrom}.
+ * This allows applications to reconstruct the allowance for all accounts just
+ * by listening to said events. Other implementations of the EIP may not emit
+ * these events, as it isn't required by the specification.
+ *
+ * Finally, the non-standard {decreaseAllowance} and {increaseAllowance}
+ * functions have been added to mitigate the well-known issues around setting
+ * allowances. See {IERC20-approve}.
+ */
+contract ERC20UpgradeableCustom is Initializable, ContextUpgradeable, IERC20Upgradeable, IERC20MetadataUpgradeable {
+ mapping(address => uint256) private _balances;
+
+ mapping(address => mapping(address => uint256)) private _allowances;
+
+ uint256 private _totalSupply;
+
+ string private _name;
+ string private _symbol;
+
+ /**
+ * @dev Sets the values for {name} and {symbol}.
+ *
+ * The default value of {decimals} is 18. To select a different value for
+ * {decimals} you should overload it.
+ *
+ * All two of these values are immutable: they can only be set once during
+ * construction.
+ */
+ //solhint-disable-next-line
+ function __ERC20_init(string memory name_, string memory symbol_) internal onlyInitializing {
+ __Context_init_unchained();
+ __ERC20_init_unchained(name_, symbol_);
+ }
+
+ //solhint-disable-next-line
+ function __ERC20_init_unchained(string memory name_, string memory symbol_) internal onlyInitializing {
+ _name = name_;
+ _symbol = symbol_;
+ }
+
+ /**
+ * @dev Returns the name of the token.
+ */
+ function name() public view virtual override returns (string memory) {
+ return _name;
+ }
+
+ /**
+ * @dev Returns the symbol of the token, usually a shorter version of the
+ * name.
+ */
+ function symbol() public view virtual override returns (string memory) {
+ return _symbol;
+ }
+
+ /**
+ * @dev Returns the number of decimals used to get its user representation.
+ * For example, if `decimals` equals `2`, a balance of `505` tokens should
+ * be displayed to a user as `5.05` (`505 / 10 ** 2`).
+ *
+ * Tokens usually opt for a value of 18, imitating the relationship between
+ * Ether and Wei. This is the value {ERC20} uses, unless this function is
+ * overridden;
+ *
+ * NOTE: This information is only used for _display_ purposes: it in
+ * no way affects any of the arithmetic of the contract, including
+ * {IERC20-balanceOf} and {IERC20-transfer}.
+ */
+ function decimals() public view virtual override returns (uint8) {
+ return 18;
+ }
+
+ /**
+ * @dev See {IERC20-totalSupply}.
+ */
+ function totalSupply() public view virtual override returns (uint256) {
+ return _totalSupply;
+ }
+
+ /**
+ * @dev See {IERC20-balanceOf}.
+ */
+ function balanceOf(address account) public view virtual override returns (uint256) {
+ return _balances[account];
+ }
+
+ /**
+ * @dev See {IERC20-transfer}.
+ *
+ * Requirements:
+ *
+ * - `recipient` cannot be the zero address.
+ * - the caller must have a balance of at least `amount`.
+ */
+ function transfer(address recipient, uint256 amount) public virtual override returns (bool) {
+ _transfer(_msgSender(), recipient, amount);
+ return true;
+ }
+
+ /**
+ * @dev See {IERC20-allowance}.
+ */
+ function allowance(address owner, address spender) public view virtual override returns (uint256) {
+ return _allowances[owner][spender];
+ }
+
+ /**
+ * @dev See {IERC20-approve}.
+ *
+ * Requirements:
+ *
+ * - `spender` cannot be the zero address.
+ */
+ function approve(address spender, uint256 amount) public virtual override returns (bool) {
+ _approve(_msgSender(), spender, amount);
+ return true;
+ }
+
+ /**
+ * @dev See {IERC20-transferFrom}.
+ *
+ * Emits an {Approval} event indicating the updated allowance. This is not
+ * required by the EIP. See the note at the beginning of {ERC20}.
+ *
+ * Requirements:
+ *
+ * - `sender` and `recipient` cannot be the zero address.
+ * - `sender` must have a balance of at least `amount`.
+ * - the caller must have allowance for `sender`'s tokens of at least
+ * `amount`.
+ */
+ function transferFrom(address sender, address recipient, uint256 amount) public virtual override returns (bool) {
+ _transfer(sender, recipient, amount);
+
+ uint256 currentAllowance = _allowances[sender][_msgSender()];
+ require(currentAllowance >= amount, "ERC20: transfer amount exceeds allowance");
+ unchecked {
+ _approve(sender, _msgSender(), currentAllowance - amount);
+ }
+
+ return true;
+ }
+
+ /**
+ * @dev Atomically increases the allowance granted to `spender` by the caller.
+ *
+ * This is an alternative to {approve} that can be used as a mitigation for
+ * problems described in {IERC20-approve}.
+ *
+ * Emits an {Approval} event indicating the updated allowance.
+ *
+ * Requirements:
+ *
+ * - `spender` cannot be the zero address.
+ */
+ function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {
+ _approve(_msgSender(), spender, _allowances[_msgSender()][spender] + addedValue);
+ return true;
+ }
+
+ /**
+ * @dev Atomically decreases the allowance granted to `spender` by the caller.
+ *
+ * This is an alternative to {approve} that can be used as a mitigation for
+ * problems described in {IERC20-approve}.
+ *
+ * Emits an {Approval} event indicating the updated allowance.
+ *
+ * Requirements:
+ *
+ * - `spender` cannot be the zero address.
+ * - `spender` must have allowance for the caller of at least
+ * `subtractedValue`.
+ */
+ function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) {
+ uint256 currentAllowance = _allowances[_msgSender()][spender];
+ require(currentAllowance >= subtractedValue, "ERC20: decreased allowance below zero");
+ unchecked {
+ _approve(_msgSender(), spender, currentAllowance - subtractedValue);
+ }
+
+ return true;
+ }
+
+ /**
+ * @dev Moves `amount` of tokens from `sender` to `recipient`.
+ *
+ * This internal function is equivalent to {transfer}, and can be used to
+ * e.g. implement automatic token fees, slashing mechanisms, etc.
+ *
+ * Emits a {Transfer} event.
+ *
+ * Requirements:
+ *
+ * - `sender` cannot be the zero address.
+ * - `recipient` cannot be the zero address.
+ * - `sender` must have a balance of at least `amount`.
+ */
+ function _transfer(address sender, address recipient, uint256 amount) internal virtual {
+ require(sender != address(0), "ERC20: transfer from the zero address");
+ require(recipient != address(0), "ERC20: transfer to the zero address");
+
+ _beforeTokenTransfer(sender, recipient, amount);
+
+ uint256 senderBalance = _balances[sender];
+ require(senderBalance >= amount, "ERC20: transfer amount exceeds balance");
+ unchecked {
+ _balances[sender] = senderBalance - amount;
+ }
+ _balances[recipient] += amount;
+
+ emit Transfer(sender, recipient, amount);
+
+ _afterTokenTransfer(sender, recipient, amount);
+ }
+
+ /** @dev Creates `amount` tokens and assigns them to `account`, increasing
+ * the total supply.
+ *
+ * Emits a {Transfer} event with `from` set to the zero address.
+ *
+ * Requirements:
+ *
+ * - `account` cannot be the zero address.
+ */
+ function _mint(address account, uint256 amount) internal virtual {
+ require(account != address(0), "ERC20: mint to the zero address");
+
+ _beforeTokenTransfer(address(0), account, amount);
+
+ _totalSupply += amount;
+ _balances[account] += amount;
+ emit Transfer(address(0), account, amount);
+
+ _afterTokenTransfer(address(0), account, amount);
+ }
+
+ /**
+ * @dev Destroys `amount` tokens from `account`, reducing the
+ * total supply.
+ *
+ * Emits a {Transfer} event with `to` set to the zero address.
+ *
+ * Requirements:
+ *
+ * - `account` cannot be the zero address.
+ * - `account` must have at least `amount` tokens.
+ */
+ function _burn(address account, uint256 amount) internal virtual {
+ require(account != address(0), "ERC20: burn from the zero address");
+
+ _beforeTokenTransfer(account, address(0), amount);
+
+ uint256 accountBalance = _balances[account];
+ require(accountBalance >= amount, "ERC20: burn amount exceeds balance");
+ unchecked {
+ _balances[account] = accountBalance - amount;
+ }
+ _totalSupply -= amount;
+
+ emit Transfer(account, address(0), amount);
+
+ _afterTokenTransfer(account, address(0), amount);
+ }
+
+ /**
+ * @dev Destroys `amount` tokens from `account`, reducing the
+ * total supply.
+ *
+ * Contrary to the other `burn` function, the {Transfer} event is not to the zero address
+ * but rather to this address: the reason is that not all burn events should be caught up
+ * by the PoS bridge
+ *
+ * Requirements:
+ *
+ * - `account` cannot be the zero address.
+ * - `account` must have at least `amount` tokens.
+ */
+ function _burnCustom(address account, uint256 amount) internal virtual {
+ require(account != address(0), "ERC20: burn from the zero address");
+
+ _beforeTokenTransfer(account, address(0), amount);
+
+ uint256 accountBalance = _balances[account];
+ require(accountBalance >= amount, "ERC20: burn amount exceeds balance");
+ unchecked {
+ _balances[account] = accountBalance - amount;
+ }
+ _totalSupply -= amount;
+
+ emit Transfer(account, address(this), amount);
+
+ _afterTokenTransfer(account, address(0), amount);
+ }
+
+ /**
+ * @dev Sets `amount` as the allowance of `spender` over the `owner` s tokens.
+ *
+ * This internal function is equivalent to `approve`, and can be used to
+ * e.g. set automatic allowances for certain subsystems, etc.
+ *
+ * Emits an {Approval} event.
+ *
+ * Requirements:
+ *
+ * - `owner` cannot be the zero address.
+ * - `spender` cannot be the zero address.
+ */
+ function _approve(address owner, address spender, uint256 amount) internal virtual {
+ require(owner != address(0), "ERC20: approve from the zero address");
+ require(spender != address(0), "ERC20: approve to the zero address");
+
+ _allowances[owner][spender] = amount;
+ emit Approval(owner, spender, amount);
+ }
+
+ /**
+ * @dev Hook that is called before any transfer of tokens. This includes
+ * minting and burning.
+ *
+ * Calling conditions:
+ *
+ * - when `from` and `to` are both non-zero, `amount` of `from`'s tokens
+ * will be transferred to `to`.
+ * - when `from` is zero, `amount` tokens will be minted for `to`.
+ * - when `to` is zero, `amount` of `from`'s tokens will be burned.
+ * - `from` and `to` are never both zero.
+ *
+ * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
+ */
+ function _beforeTokenTransfer(address from, address to, uint256 amount) internal virtual {}
+
+ /**
+ * @dev Hook that is called after any transfer of tokens. This includes
+ * minting and burning.
+ *
+ * Calling conditions:
+ *
+ * - when `from` and `to` are both non-zero, `amount` of `from`'s tokens
+ * has been transferred to `to`.
+ * - when `from` is zero, `amount` tokens have been minted for `to`.
+ * - when `to` is zero, `amount` of `from`'s tokens have been burned.
+ * - `from` and `to` are never both zero.
+ *
+ * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
+ */
+ function _afterTokenTransfer(address from, address to, uint256 amount) internal virtual {}
+
+ uint256[45] private __gap;
+}
diff --git a/contracts/coreBorrow/CoreBorrow.sol b/contracts/coreBorrow/CoreBorrow.sol
new file mode 100644
index 0000000..5d77c05
--- /dev/null
+++ b/contracts/coreBorrow/CoreBorrow.sol
@@ -0,0 +1,188 @@
+// SPDX-License-Identifier: GPL-3.0
+
+/*
+ * █
+ ***** ▓▓▓
+ * ▓▓▓▓▓▓▓
+ * ///. ▓▓▓▓▓▓▓▓▓▓▓▓▓
+ ***** //////// ▓▓▓▓▓▓▓
+ * ///////////// ▓▓▓
+ ▓▓ ////////////////// █ ▓▓
+ ▓▓ ▓▓ /////////////////////// ▓▓ ▓▓
+ ▓▓ ▓▓ //////////////////////////// ▓▓ ▓▓
+ ▓▓ ▓▓ /////////▓▓▓///////▓▓▓///////// ▓▓ ▓▓
+ ▓▓ ,////////////////////////////////////// ▓▓ ▓▓
+ ▓▓ ////////////////////////////////////////// ▓▓
+ ▓▓ //////////////////////▓▓▓▓/////////////////////
+ ,////////////////////////////////////////////////////
+ .//////////////////////////////////////////////////////////
+ .//////////////////////////██.,//////////////////////////█
+ .//////////////////////████..,./////////////////////██
+ ...////////////////███████.....,.////////////////███
+ ,.,////////////████████ ........,///////////████
+ .,.,//////█████████ ,.......///////████
+ ,..//████████ ........./████
+ ..,██████ .....,███
+ .██ ,.,█
+
+
+
+ ▓▓ ▓▓▓▓▓▓▓▓▓▓ ▓▓▓▓▓▓▓▓▓▓ ▓▓ ▓▓▓▓▓▓▓▓▓▓
+ ▓▓▓▓▓▓ ▓▓▓ ▓▓▓ ▓▓▓ ▓▓ ▓▓ ▓▓▓▓
+ ▓▓▓ ▓▓▓ ▓▓▓ ▓▓▓ ▓▓▓ ▓▓▓ ▓▓ ▓▓▓▓▓
+ ▓▓▓ ▓▓ ▓▓▓ ▓▓▓ ▓▓▓▓▓▓▓▓▓▓ ▓▓▓▓▓▓▓▓▓▓ ▓▓▓▓▓▓▓▓▓▓
+*/
+
+pragma solidity ^0.8.12;
+
+import "oz-upgradeable/access/AccessControlEnumerableUpgradeable.sol";
+import "oz-upgradeable/proxy/utils/Initializable.sol";
+
+import "../interfaces/ICoreBorrow.sol";
+import "../interfaces/IFlashAngle.sol";
+import "../interfaces/ITreasury.sol";
+
+/// @title CoreBorrow
+/// @author Angle Labs, Inc.
+/// @notice Core contract of the borrowing module. This contract handles the access control across all contracts
+/// (it is read by all treasury contracts), and manages the `flashLoanModule`. It has no minting rights over the
+/// stablecoin contracts
+contract CoreBorrow is ICoreBorrow, Initializable, AccessControlEnumerableUpgradeable {
+ /// @notice Role for guardians
+ bytes32 public constant GUARDIAN_ROLE = keccak256("GUARDIAN_ROLE");
+ /// @notice Role for governors
+ bytes32 public constant GOVERNOR_ROLE = keccak256("GOVERNOR_ROLE");
+ /// @notice Role for treasury contract
+ bytes32 public constant FLASHLOANER_TREASURY_ROLE = keccak256("FLASHLOANER_TREASURY_ROLE");
+
+ // ============================= Reference =====================================
+
+ /// @notice Reference to the `flashLoanModule` with minting rights over the different stablecoins of the protocol
+ address public flashLoanModule;
+
+ // =============================== Events ======================================
+
+ event FlashLoanModuleUpdated(address indexed _flashloanModule);
+ event CoreUpdated(address indexed _core);
+
+ // =============================== Errors ======================================
+
+ error InvalidCore();
+ error IncompatibleGovernorAndGuardian();
+ error NotEnoughGovernorsLeft();
+ error ZeroAddress();
+
+ /// @notice Initializes the `CoreBorrow` contract and the access control of the borrowing module
+ /// @param governor Address of the governor of the Angle Protocol
+ /// @param guardian Guardian address of the protocol
+ function initialize(address governor, address guardian) public initializer {
+ if (governor == address(0) || guardian == address(0)) revert ZeroAddress();
+ if (governor == guardian) revert IncompatibleGovernorAndGuardian();
+ _setupRole(GOVERNOR_ROLE, governor);
+ _setupRole(GUARDIAN_ROLE, guardian);
+ _setupRole(GUARDIAN_ROLE, governor);
+ _setRoleAdmin(GUARDIAN_ROLE, GOVERNOR_ROLE);
+ _setRoleAdmin(GOVERNOR_ROLE, GOVERNOR_ROLE);
+ _setRoleAdmin(FLASHLOANER_TREASURY_ROLE, GOVERNOR_ROLE);
+ }
+
+ /// @custom:oz-upgrades-unsafe-allow constructor
+ constructor() initializer {}
+
+ // =========================== View Functions ==================================
+
+ /// @inheritdoc ICoreBorrow
+ function isFlashLoanerTreasury(address treasury) external view returns (bool) {
+ return hasRole(FLASHLOANER_TREASURY_ROLE, treasury);
+ }
+
+ /// @inheritdoc ICoreBorrow
+ function isGovernor(address admin) external view returns (bool) {
+ return hasRole(GOVERNOR_ROLE, admin);
+ }
+
+ /// @inheritdoc ICoreBorrow
+ function isGovernorOrGuardian(address admin) external view returns (bool) {
+ return hasRole(GUARDIAN_ROLE, admin);
+ }
+
+ // =========================== Governor Functions ==============================
+
+ /// @notice Grants the `FLASHLOANER_TREASURY_ROLE` to a `treasury` contract
+ /// @param treasury Contract to grant the role to
+ /// @dev This function can be used to allow flash loans on a stablecoin of the protocol
+ function addFlashLoanerTreasuryRole(address treasury) external {
+ grantRole(FLASHLOANER_TREASURY_ROLE, treasury);
+ address _flashLoanModule = flashLoanModule;
+ if (_flashLoanModule != address(0)) {
+ // This call will revert if `treasury` is the zero address or if it is not linked
+ // to this `CoreBorrow` contract
+ ITreasury(treasury).setFlashLoanModule(_flashLoanModule);
+ IFlashAngle(_flashLoanModule).addStablecoinSupport(treasury);
+ }
+ }
+
+ /// @notice Adds a governor in the protocol
+ /// @param governor Address to grant the role to
+ /// @dev It is necessary to call this function to grant a governor role to make sure
+ /// all governors also have the guardian role
+ function addGovernor(address governor) external {
+ grantRole(GOVERNOR_ROLE, governor);
+ grantRole(GUARDIAN_ROLE, governor);
+ }
+
+ /// @notice Revokes the flash loan ability for a stablecoin
+ /// @param treasury Treasury address associated with the stablecoin for which flash loans
+ /// should no longer be available
+ function removeFlashLoanerTreasuryRole(address treasury) external {
+ revokeRole(FLASHLOANER_TREASURY_ROLE, treasury);
+ ITreasury(treasury).setFlashLoanModule(address(0));
+ address _flashLoanModule = flashLoanModule;
+ if (_flashLoanModule != address(0)) {
+ IFlashAngle(flashLoanModule).removeStablecoinSupport(treasury);
+ }
+ }
+
+ /// @notice Revokes a governor from the protocol
+ /// @param governor Address to remove the role to
+ /// @dev It is necessary to call this function to remove a governor role to make sure
+ /// the address also loses its guardian role
+ function removeGovernor(address governor) external {
+ if (getRoleMemberCount(GOVERNOR_ROLE) <= 1) revert NotEnoughGovernorsLeft();
+ revokeRole(GUARDIAN_ROLE, governor);
+ revokeRole(GOVERNOR_ROLE, governor);
+ }
+
+ /// @notice Changes the `flashLoanModule` of the protocol
+ /// @param _flashLoanModule Address of the new flash loan module
+ function setFlashLoanModule(address _flashLoanModule) external onlyRole(GOVERNOR_ROLE) {
+ if (_flashLoanModule != address(0)) {
+ if (address(IFlashAngle(_flashLoanModule).core()) != address(this)) revert InvalidCore();
+ }
+ uint256 count = getRoleMemberCount(FLASHLOANER_TREASURY_ROLE);
+ for (uint256 i; i < count; ++i) {
+ ITreasury(getRoleMember(FLASHLOANER_TREASURY_ROLE, i)).setFlashLoanModule(_flashLoanModule);
+ }
+ flashLoanModule = _flashLoanModule;
+ emit FlashLoanModuleUpdated(_flashLoanModule);
+ }
+
+ /// @notice Changes the core contract of the protocol
+ /// @param _core New core contract
+ /// @dev This function verifies that all governors of the current core contract are also governors
+ /// of the new core contract. It also notifies the `flashLoanModule` of the change.
+ /// @dev Governance wishing to change the core contract should also make sure to call `setCore`
+ /// in the different treasury contracts
+ function setCore(ICoreBorrow _core) external onlyRole(GOVERNOR_ROLE) {
+ uint256 count = getRoleMemberCount(GOVERNOR_ROLE);
+ bool success;
+ for (uint256 i; i < count; ++i) {
+ success = _core.isGovernor(getRoleMember(GOVERNOR_ROLE, i));
+ if (!success) break;
+ }
+ if (!success) revert InvalidCore();
+ address _flashLoanModule = flashLoanModule;
+ if (_flashLoanModule != address(0)) IFlashAngle(_flashLoanModule).setCore(address(_core));
+ emit CoreUpdated(address(_core));
+ }
+}
diff --git a/contracts/external/ProxyAdmin.sol b/contracts/external/ProxyAdmin.sol
new file mode 100644
index 0000000..29abc23
--- /dev/null
+++ b/contracts/external/ProxyAdmin.sol
@@ -0,0 +1,81 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "./TransparentUpgradeableProxy.sol";
+import "oz/access/Ownable.sol";
+
+/**
+ * @dev This is an auxiliary contract meant to be assigned as the admin of a {TransparentUpgradeableProxy}. For an
+ * explanation of why you would want to use this see the documentation for {TransparentUpgradeableProxy}.
+ * This contract was fully forked from OpenZeppelin `ProxyAdmin`
+ */
+contract ProxyAdmin is Ownable {
+ /**
+ * @dev Returns the current implementation of `proxy`.
+ *
+ * Requirements:
+ *
+ * - This contract must be the admin of `proxy`.
+ */
+ function getProxyImplementation(TransparentUpgradeableProxy proxy) public view virtual returns (address) {
+ // We need to manually run the static call since the getter cannot be flagged as view
+ // bytes4(keccak256("implementation()")) == 0x5c60da1b
+ (bool success, bytes memory returndata) = address(proxy).staticcall(hex"5c60da1b");
+ require(success);
+ return abi.decode(returndata, (address));
+ }
+
+ /**
+ * @dev Returns the current admin of `proxy`.
+ *
+ * Requirements:
+ *
+ * - This contract must be the admin of `proxy`.
+ */
+ function getProxyAdmin(TransparentUpgradeableProxy proxy) public view virtual returns (address) {
+ // We need to manually run the static call since the getter cannot be flagged as view
+ // bytes4(keccak256("admin()")) == 0xf851a440
+ (bool success, bytes memory returndata) = address(proxy).staticcall(hex"f851a440");
+ require(success);
+ return abi.decode(returndata, (address));
+ }
+
+ /**
+ * @dev Changes the admin of `proxy` to `newAdmin`.
+ *
+ * Requirements:
+ *
+ * - This contract must be the current admin of `proxy`.
+ */
+ function changeProxyAdmin(TransparentUpgradeableProxy proxy, address newAdmin) public virtual onlyOwner {
+ proxy.changeAdmin(newAdmin);
+ }
+
+ /**
+ * @dev Upgrades `proxy` to `implementation`. See {TransparentUpgradeableProxy-upgradeTo}.
+ *
+ * Requirements:
+ *
+ * - This contract must be the admin of `proxy`.
+ */
+ function upgrade(TransparentUpgradeableProxy proxy, address implementation) public virtual onlyOwner {
+ proxy.upgradeTo(implementation);
+ }
+
+ /**
+ * @dev Upgrades `proxy` to `implementation` and calls a function on the new implementation. See
+ * {TransparentUpgradeableProxy-upgradeToAndCall}.
+ *
+ * Requirements:
+ *
+ * - This contract must be the admin of `proxy`.
+ */
+ function upgradeAndCall(
+ TransparentUpgradeableProxy proxy,
+ address implementation,
+ bytes memory data
+ ) public payable virtual onlyOwner {
+ proxy.upgradeToAndCall{ value: msg.value }(implementation, data);
+ }
+}
diff --git a/contracts/external/TransparentUpgradeableProxy.sol b/contracts/external/TransparentUpgradeableProxy.sol
new file mode 100644
index 0000000..b3490d9
--- /dev/null
+++ b/contracts/external/TransparentUpgradeableProxy.sol
@@ -0,0 +1,121 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "oz/proxy/ERC1967/ERC1967Proxy.sol";
+
+/**
+ * @dev This contract implements a proxy that is upgradeable by an admin. It is fully forked from OpenZeppelin
+ * `TransparentUpgradeableProxy`
+ *
+ * To avoid https://medium.com/nomic-labs-blog/malicious-backdoors-in-ethereum-proxies-62629adf3357[proxy selector
+ * clashing], which can potentially be used in an attack, this contract uses the
+ * https://blog.openzeppelin.com/the-transparent-proxy-pattern/[transparent proxy pattern]. This pattern implies two
+ * things that go hand in hand:
+ *
+ * 1. If any account other than the admin calls the proxy, the call will be forwarded to the implementation, even if
+ * that call matches one of the admin functions exposed by the proxy itself.
+ * 2. If the admin calls the proxy, it can access the admin functions, but its calls will never be forwarded to the
+ * implementation. If the admin tries to call a function on the implementation it will fail with an error that says
+ * "admin cannot fallback to proxy target".
+ *
+ * These properties mean that the admin account can only be used for admin actions like upgrading the proxy or changing
+ * the admin, so it's best if it's a dedicated account that is not used for anything else. This will avoid headaches due
+ * to sudden errors when trying to call a function from the proxy implementation.
+ *
+ * Our recommendation is for the dedicated account to be an instance of the {ProxyAdmin} contract. If set up this way,
+ * you should think of the `ProxyAdmin` instance as the real administrative interface of your proxy.
+ */
+contract TransparentUpgradeableProxy is ERC1967Proxy {
+ /**
+ * @dev Initializes an upgradeable proxy managed by `_admin`, backed by the implementation at `_logic`, and
+ * optionally initialized with `_data` as explained in {ERC1967Proxy-constructor}.
+ */
+ constructor(address _logic, address admin_, bytes memory _data) payable ERC1967Proxy(_logic, _data) {
+ assert(_ADMIN_SLOT == bytes32(uint256(keccak256("eip1967.proxy.admin")) - 1));
+ _changeAdmin(admin_);
+ }
+
+ /**
+ * @dev Modifier used internally that will delegate the call to the implementation unless the sender is the admin.
+ */
+ modifier ifAdmin() {
+ if (msg.sender == _getAdmin()) {
+ _;
+ } else {
+ _fallback();
+ }
+ }
+
+ /**
+ * @dev Returns the current admin.
+ *
+ * NOTE: Only the admin can call this function. See {ProxyAdmin-getProxyAdmin}.
+ *
+ * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using the
+ * https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
+ * `0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103`
+ */
+ function admin() external ifAdmin returns (address admin_) {
+ admin_ = _getAdmin();
+ }
+
+ /**
+ * @dev Returns the current implementation.
+ *
+ * NOTE: Only the admin can call this function. See {ProxyAdmin-getProxyImplementation}.
+ *
+ * TIP: To get this value clients can read directly from the storage slot shown below (specified by EIP1967) using the
+ * https://eth.wiki/json-rpc/API#eth_getstorageat[`eth_getStorageAt`] RPC call.
+ * `0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc`
+ */
+ function implementation() external ifAdmin returns (address implementation_) {
+ implementation_ = _implementation();
+ }
+
+ /**
+ * @dev Changes the admin of the proxy.
+ *
+ * Emits an {AdminChanged} event.
+ *
+ * NOTE: Only the admin can call this function. See {ProxyAdmin-changeProxyAdmin}.
+ */
+ function changeAdmin(address newAdmin) external virtual ifAdmin {
+ _changeAdmin(newAdmin);
+ }
+
+ /**
+ * @dev Upgrade the implementation of the proxy.
+ *
+ * NOTE: Only the admin can call this function. See {ProxyAdmin-upgrade}.
+ */
+ function upgradeTo(address newImplementation) external ifAdmin {
+ _upgradeToAndCall(newImplementation, bytes(""), false);
+ }
+
+ /**
+ * @dev Upgrade the implementation of the proxy, and then call a function from the new implementation as specified
+ * by `data`, which should be an encoded function call. This is useful to initialize new storage variables in the
+ * proxied contract.
+ *
+ * NOTE: Only the admin can call this function. See {ProxyAdmin-upgradeAndCall}.
+ */
+ function upgradeToAndCall(address newImplementation, bytes calldata data) external payable ifAdmin {
+ _upgradeToAndCall(newImplementation, data, true);
+ }
+
+ /**
+ * @dev Returns the current admin.
+ */
+ function _admin() internal view virtual returns (address) {
+ return _getAdmin();
+ }
+
+ /**
+ * @dev Makes sure the admin cannot access the fallback function. See {Proxy-_beforeFallback}.
+ */
+ function _beforeFallback() internal virtual override {
+ require(msg.sender != _getAdmin(), "TransparentUpgradeableProxy: admin cannot fallback to proxy target");
+ super._beforeFallback();
+ }
+}
diff --git a/contracts/flashloan/FlashAngle.sol b/contracts/flashloan/FlashAngle.sol
new file mode 100644
index 0000000..7b966db
--- /dev/null
+++ b/contracts/flashloan/FlashAngle.sol
@@ -0,0 +1,179 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "oz-upgradeable/security/ReentrancyGuardUpgradeable.sol";
+import "oz-upgradeable/proxy/utils/Initializable.sol";
+import "oz/interfaces/IERC3156FlashBorrower.sol";
+import "oz/interfaces/IERC3156FlashLender.sol";
+import "oz/token/ERC20/IERC20.sol";
+import "oz/token/ERC20/utils/SafeERC20.sol";
+
+import "../interfaces/IAgToken.sol";
+import "../interfaces/ICoreBorrow.sol";
+import "../interfaces/IFlashAngle.sol";
+import "../interfaces/ITreasury.sol";
+
+/// @title FlashAngle
+/// @author Angle Labs, Inc.
+/// @notice Contract to take flash loans on top of several AgToken contracts
+contract FlashAngle is IERC3156FlashLender, IFlashAngle, Initializable, ReentrancyGuardUpgradeable {
+ using SafeERC20 for IERC20;
+ /// @notice Base used for parameter computation
+ uint256 public constant BASE_PARAMS = 10 ** 9;
+ /// @notice Success message received when calling a `FlashBorrower` contract
+ bytes32 public constant CALLBACK_SUCCESS = keccak256("ERC3156FlashBorrower.onFlashLoan");
+
+ /// @notice Struct encoding for a given stablecoin the parameters
+ struct StablecoinData {
+ // Maximum amount borrowable for this stablecoin
+ uint256 maxBorrowable;
+ // Flash loan fee taken by the protocol for a flash loan on this stablecoin
+ uint64 flashLoanFee;
+ // Treasury address responsible of the stablecoin
+ address treasury;
+ }
+
+ // ======================= Parameters and References ===========================
+
+ /// @notice Maps a stablecoin to the data and parameters for flash loans
+ mapping(IAgToken => StablecoinData) public stablecoinMap;
+ /// @inheritdoc IFlashAngle
+ ICoreBorrow public core;
+
+ // =============================== Event =======================================
+
+ event FlashLoan(address indexed stablecoin, uint256 amount, IERC3156FlashBorrower indexed receiver);
+ event FlashLoanParametersUpdated(IAgToken indexed stablecoin, uint64 _flashLoanFee, uint256 _maxBorrowable);
+
+ // =============================== Errors ======================================
+
+ error InvalidReturnMessage();
+ error NotCore();
+ error NotGovernorOrGuardian();
+ error NotTreasury();
+ error TooBigAmount();
+ error TooHighParameterValue();
+ error UnsupportedStablecoin();
+ error ZeroAddress();
+
+ /// @notice Initializes the contract
+ /// @param _core Core address handling this module
+ function initialize(ICoreBorrow _core) public initializer {
+ if (address(_core) == address(0)) revert ZeroAddress();
+ core = _core;
+ }
+
+ /// @custom:oz-upgrades-unsafe-allow constructor
+ constructor() initializer {}
+
+ // =================================== Modifiers ===============================
+
+ /// @notice Checks whether the sender is the core contract
+ modifier onlyCore() {
+ if (msg.sender != address(core)) revert NotCore();
+ _;
+ }
+
+ /// @notice Checks whether a given stablecoin has been initialized in this contract
+ /// @param stablecoin Stablecoin to check
+ /// @dev To check whether a stablecoin has been initialized, we just need to check whether its associated
+ /// `treasury` address is not null in the `stablecoinMap`. This is what's checked in the `CoreBorrow` contract
+ /// when adding support for a stablecoin
+ modifier onlyExistingStablecoin(IAgToken stablecoin) {
+ if (stablecoinMap[stablecoin].treasury == address(0)) revert UnsupportedStablecoin();
+ _;
+ }
+
+ // ================================ ERC3156 Spec ===============================
+
+ /// @inheritdoc IERC3156FlashLender
+ function flashFee(address token, uint256 amount) external view returns (uint256) {
+ return _flashFee(token, amount);
+ }
+
+ /// @inheritdoc IERC3156FlashLender
+ function maxFlashLoan(address token) external view returns (uint256) {
+ // It will be 0 anyway if the token was not added
+ return stablecoinMap[IAgToken(token)].maxBorrowable;
+ }
+
+ /// @inheritdoc IERC3156FlashLender
+ function flashLoan(
+ IERC3156FlashBorrower receiver,
+ address token,
+ uint256 amount,
+ bytes calldata data
+ ) external nonReentrant returns (bool) {
+ uint256 fee = _flashFee(token, amount);
+ if (amount > stablecoinMap[IAgToken(token)].maxBorrowable) revert TooBigAmount();
+ IAgToken(token).mint(address(receiver), amount);
+ if (receiver.onFlashLoan(msg.sender, token, amount, fee, data) != CALLBACK_SUCCESS)
+ revert InvalidReturnMessage();
+ // Token must be an agToken here so normally no need to use `safeTransferFrom`, but out of safety
+ // and in case governance whitelists an agToken which does not have a correct implementation, we prefer
+ // to use `safeTransferFrom` here
+ IERC20(token).safeTransferFrom(address(receiver), address(this), amount + fee);
+ IAgToken(token).burnSelf(amount, address(this));
+ emit FlashLoan(token, amount, receiver);
+ return true;
+ }
+
+ /// @notice Internal function to compute the fee induced for taking a flash loan of `amount` of `token`
+ /// @param token The loan currency
+ /// @param amount The amount of tokens lent
+ /// @dev This function will revert if the `token` requested is not whitelisted here
+ function _flashFee(
+ address token,
+ uint256 amount
+ ) internal view onlyExistingStablecoin(IAgToken(token)) returns (uint256) {
+ return (amount * stablecoinMap[IAgToken(token)].flashLoanFee) / BASE_PARAMS;
+ }
+
+ // ============================ Treasury Only Function =========================
+
+ /// @inheritdoc IFlashAngle
+ function accrueInterestToTreasury(IAgToken stablecoin) external returns (uint256 balance) {
+ address treasury = stablecoinMap[stablecoin].treasury;
+ if (msg.sender != treasury) revert NotTreasury();
+ balance = stablecoin.balanceOf(address(this));
+ IERC20(address(stablecoin)).safeTransfer(treasury, balance);
+ }
+
+ // =========================== Governance Only Function ========================
+
+ /// @notice Sets the parameters for a given stablecoin
+ /// @param stablecoin Stablecoin to change the parameters for
+ /// @param _flashLoanFee New flash loan fee for this stablecoin
+ /// @param _maxBorrowable Maximum amount that can be borrowed in a single flash loan
+ /// @dev Setting a `maxBorrowable` parameter equal to 0 is a way to pause the functionality
+ /// @dev Parameters can only be modified for whitelisted stablecoins
+ function setFlashLoanParameters(
+ IAgToken stablecoin,
+ uint64 _flashLoanFee,
+ uint256 _maxBorrowable
+ ) external onlyExistingStablecoin(stablecoin) {
+ if (!core.isGovernorOrGuardian(msg.sender)) revert NotGovernorOrGuardian();
+ if (_flashLoanFee > BASE_PARAMS) revert TooHighParameterValue();
+ stablecoinMap[stablecoin].flashLoanFee = _flashLoanFee;
+ stablecoinMap[stablecoin].maxBorrowable = _maxBorrowable;
+ emit FlashLoanParametersUpdated(stablecoin, _flashLoanFee, _maxBorrowable);
+ }
+
+ // =========================== CoreBorrow Only Functions =======================
+
+ /// @inheritdoc IFlashAngle
+ function addStablecoinSupport(address _treasury) external onlyCore {
+ stablecoinMap[IAgToken(ITreasury(_treasury).stablecoin())].treasury = _treasury;
+ }
+
+ /// @inheritdoc IFlashAngle
+ function removeStablecoinSupport(address _treasury) external onlyCore {
+ delete stablecoinMap[IAgToken(ITreasury(_treasury).stablecoin())];
+ }
+
+ /// @inheritdoc IFlashAngle
+ function setCore(address _core) external onlyCore {
+ core = ICoreBorrow(_core);
+ }
+}
diff --git a/contracts/interfaces/IAgToken.sol b/contracts/interfaces/IAgToken.sol
new file mode 100644
index 0000000..dae0d05
--- /dev/null
+++ b/contracts/interfaces/IAgToken.sol
@@ -0,0 +1,69 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "oz-upgradeable/token/ERC20/IERC20Upgradeable.sol";
+
+/// @title IAgToken
+/// @author Angle Labs, Inc.
+/// @notice Interface for the stablecoins `AgToken` contracts
+/// @dev This interface only contains functions of the `AgToken` contract which are called by other contracts
+/// of this module or of the first module of the Angle Protocol
+interface IAgToken is IERC20Upgradeable {
+ // ======================= Minter Role Only Functions ===========================
+
+ /// @notice Lets the `StableMaster` contract or another whitelisted contract mint agTokens
+ /// @param account Address to mint to
+ /// @param amount Amount to mint
+ /// @dev The contracts allowed to issue agTokens are the `StableMaster` contract, `VaultManager` contracts
+ /// associated to this stablecoin as well as the flash loan module (if activated) and potentially contracts
+ /// whitelisted by governance
+ function mint(address account, uint256 amount) external;
+
+ /// @notice Burns `amount` tokens from a `burner` address after being asked to by `sender`
+ /// @param amount Amount of tokens to burn
+ /// @param burner Address to burn from
+ /// @param sender Address which requested the burn from `burner`
+ /// @dev This method is to be called by a contract with the minter right after being requested
+ /// to do so by a `sender` address willing to burn tokens from another `burner` address
+ /// @dev The method checks the allowance between the `sender` and the `burner`
+ function burnFrom(uint256 amount, address burner, address sender) external;
+
+ /// @notice Burns `amount` tokens from a `burner` address
+ /// @param amount Amount of tokens to burn
+ /// @param burner Address to burn from
+ /// @dev This method is to be called by a contract with a minter right on the AgToken after being
+ /// requested to do so by an address willing to burn tokens from its address
+ function burnSelf(uint256 amount, address burner) external;
+
+ /// @notice Allows anyone to burn stablecoins
+ /// @param amount Amount of stablecoins to burn
+ /// @dev This function can typically be called if there is a settlement mechanism to burn stablecoins
+ function burnStablecoin(uint256 amount) external;
+
+ // ========================= Treasury Only Functions ===========================
+
+ /// @notice Adds a minter in the contract
+ /// @param minter Minter address to add
+ /// @dev Zero address checks are performed directly in the `Treasury` contract
+ function addMinter(address minter) external;
+
+ /// @notice Removes a minter from the contract
+ /// @param minter Minter address to remove
+ /// @dev This function can also be called by a minter wishing to revoke itself
+ function removeMinter(address minter) external;
+
+ /// @notice Sets a new treasury contract
+ /// @param _treasury New treasury address
+ function setTreasury(address _treasury) external;
+
+ // ========================= External functions ================================
+
+ /// @notice Checks whether an address has the right to mint agTokens
+ /// @param minter Address for which the minting right should be checked
+ /// @return Whether the address has the right to mint agTokens or not
+ function isMinter(address minter) external view returns (bool);
+
+ /// @notice Get the associated treasury
+ function treasury() external view returns (address);
+}
diff --git a/contracts/interfaces/IAgTokenSideChainMultiBridge.sol b/contracts/interfaces/IAgTokenSideChainMultiBridge.sol
new file mode 100644
index 0000000..2bc9115
--- /dev/null
+++ b/contracts/interfaces/IAgTokenSideChainMultiBridge.sol
@@ -0,0 +1,28 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "oz-upgradeable/token/ERC20/extensions/draft-IERC20PermitUpgradeable.sol";
+import "oz-upgradeable/token/ERC20/IERC20Upgradeable.sol";
+
+/// @title IAgTokenSideChainMultiBridge
+/// @author Angle Labs, Inc.
+/// @notice Interface for the canonical `AgToken` contracts
+/// @dev This interface only contains functions useful for bridge tokens to interact with the canonical token
+interface IAgTokenSideChainMultiBridge is IERC20PermitUpgradeable, IERC20Upgradeable {
+ /// @notice Mints the canonical token from a supported bridge token
+ /// @param bridgeToken Bridge token to use to mint
+ /// @param amount Amount of bridge tokens to send
+ /// @param to Address to which the stablecoin should be sent
+ /// @return Amount of the canonical stablecoin actually minted
+ /// @dev Some fees may be taken by the protocol depending on the token used and on the address calling
+ function swapIn(address bridgeToken, uint256 amount, address to) external returns (uint256);
+
+ /// @notice Burns the canonical token in exchange for a bridge token
+ /// @param bridgeToken Bridge token required
+ /// @param amount Amount of canonical tokens to burn
+ /// @param to Address to which the bridge token should be sent
+ /// @return Amount of bridge tokens actually sent back
+ /// @dev Some fees may be taken by the protocol depending on the token used and on the address calling
+ function swapOut(address bridgeToken, uint256 amount, address to) external returns (uint256);
+}
diff --git a/contracts/interfaces/ICoreBorrow.sol b/contracts/interfaces/ICoreBorrow.sol
new file mode 100644
index 0000000..fabca63
--- /dev/null
+++ b/contracts/interfaces/ICoreBorrow.sol
@@ -0,0 +1,28 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+/// @title ICoreBorrow
+/// @author Angle Labs, Inc.
+/// @notice Interface for the `CoreBorrow` contract
+/// @dev This interface only contains functions of the `CoreBorrow` contract which are called by other contracts
+/// of this module
+interface ICoreBorrow {
+ /// @notice Checks if an address corresponds to a treasury of a stablecoin with a flash loan
+ /// module initialized on it
+ /// @param treasury Address to check
+ /// @return Whether the address has the `FLASHLOANER_TREASURY_ROLE` or not
+ function isFlashLoanerTreasury(address treasury) external view returns (bool);
+
+ /// @notice Checks whether an address is governor of the Angle Protocol or not
+ /// @param admin Address to check
+ /// @return Whether the address has the `GOVERNOR_ROLE` or not
+ function isGovernor(address admin) external view returns (bool);
+
+ /// @notice Checks whether an address is governor or a guardian of the Angle Protocol or not
+ /// @param admin Address to check
+ /// @return Whether the address has the `GUARDIAN_ROLE` or not
+ /// @dev Governance should make sure when adding a governor to also give this governor the guardian
+ /// role by calling the `addGovernor` function
+ function isGovernorOrGuardian(address admin) external view returns (bool);
+}
diff --git a/contracts/interfaces/IFlashAngle.sol b/contracts/interfaces/IFlashAngle.sol
new file mode 100644
index 0000000..e97715d
--- /dev/null
+++ b/contracts/interfaces/IFlashAngle.sol
@@ -0,0 +1,37 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "./IAgToken.sol";
+import "./ICoreBorrow.sol";
+
+/// @title IFlashAngle
+/// @author Angle Labs, Inc.
+/// @notice Interface for the `FlashAngle` contract
+/// @dev This interface only contains functions of the contract which are called by other contracts
+/// of this module
+interface IFlashAngle {
+ /// @notice Reference to the `CoreBorrow` contract managing the FlashLoan module
+ function core() external view returns (ICoreBorrow);
+
+ /// @notice Sends the fees taken from flash loans to the treasury contract associated to the stablecoin
+ /// @param stablecoin Stablecoin from which profits should be sent
+ /// @return balance Amount of profits sent
+ /// @dev This function can only be called by the treasury contract
+ function accrueInterestToTreasury(IAgToken stablecoin) external returns (uint256 balance);
+
+ /// @notice Adds support for a stablecoin
+ /// @param _treasury Treasury associated to the stablecoin to add support for
+ /// @dev This function can only be called by the `CoreBorrow` contract
+ function addStablecoinSupport(address _treasury) external;
+
+ /// @notice Removes support for a stablecoin
+ /// @param _treasury Treasury associated to the stablecoin to remove support for
+ /// @dev This function can only be called by the `CoreBorrow` contract
+ function removeStablecoinSupport(address _treasury) external;
+
+ /// @notice Sets a new core contract
+ /// @param _core Core contract address to set
+ /// @dev This function can only be called by the `CoreBorrow` contract
+ function setCore(address _core) external;
+}
diff --git a/contracts/interfaces/IOracle.sol b/contracts/interfaces/IOracle.sol
new file mode 100644
index 0000000..4f23c53
--- /dev/null
+++ b/contracts/interfaces/IOracle.sol
@@ -0,0 +1,34 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "./ITreasury.sol";
+import "chainlink/src/v0.8/interfaces/AggregatorV3Interface.sol";
+
+/// @title IOracle
+/// @author Angle Labs, Inc.
+/// @notice Interface for the `Oracle` contract
+/// @dev This interface only contains functions of the contract which are called by other contracts
+/// of this module
+interface IOracle {
+ /// @notice Reads the rate from the Chainlink circuit and other data provided
+ /// @return quoteAmount The current rate between the in-currency and out-currency in the base
+ /// of the out currency
+ /// @dev For instance if the out currency is EUR (and hence agEUR), then the base of the returned
+ /// value is 10**18
+ function read() external view returns (uint256);
+
+ /// @notice Changes the treasury contract
+ /// @param _treasury Address of the new treasury contract
+ /// @dev This function can be called by an approved `VaultManager` contract which can call
+ /// this function after being requested to do so by a `treasury` contract
+ /// @dev In some situations (like reactor contracts), the `VaultManager` may not directly be linked
+ /// to the `oracle` contract and as such we may need governors to be able to call this function as well
+ function setTreasury(address _treasury) external;
+
+ /// @notice Reference to the `treasury` contract handling this `VaultManager`
+ function treasury() external view returns (ITreasury treasury);
+
+ /// @notice Array with the list of Chainlink feeds in the order in which they are read
+ function circuitChainlink() external view returns (AggregatorV3Interface[] memory);
+}
diff --git a/contracts/interfaces/ITreasury.sol b/contracts/interfaces/ITreasury.sol
new file mode 100644
index 0000000..5cdfca6
--- /dev/null
+++ b/contracts/interfaces/ITreasury.sol
@@ -0,0 +1,45 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "./IAgToken.sol";
+import "./ICoreBorrow.sol";
+import "./IFlashAngle.sol";
+
+/// @title ITreasury
+/// @author Angle Labs, Inc.
+/// @notice Interface for the `Treasury` contract
+/// @dev This interface only contains functions of the `Treasury` which are called by other contracts
+/// of this module
+interface ITreasury {
+ /// @notice Stablecoin handled by this `treasury` contract
+ function stablecoin() external view returns (IAgToken);
+
+ /// @notice Checks whether a given address has the governor role
+ /// @param admin Address to check
+ /// @return Whether the address has the governor role
+ /// @dev Access control is only kept in the `CoreBorrow` contract
+ function isGovernor(address admin) external view returns (bool);
+
+ /// @notice Checks whether a given address has the guardian or the governor role
+ /// @param admin Address to check
+ /// @return Whether the address has the guardian or the governor role
+ /// @dev Access control is only kept in the `CoreBorrow` contract which means that this function
+ /// queries the `CoreBorrow` contract
+ function isGovernorOrGuardian(address admin) external view returns (bool);
+
+ /// @notice Checks whether a given address has well been initialized in this contract
+ /// as a `VaultManager`
+ /// @param _vaultManager Address to check
+ /// @return Whether the address has been initialized or not
+ function isVaultManager(address _vaultManager) external view returns (bool);
+
+ /// @notice Sets a new flash loan module for this stablecoin
+ /// @param _flashLoanModule Reference to the new flash loan module
+ /// @dev This function removes the minting right to the old flash loan module and grants
+ /// it to the new module
+ function setFlashLoanModule(address _flashLoanModule) external;
+
+ /// @notice Gets the vault manager list
+ function vaultManagerList(uint256 i) external returns (address);
+}
diff --git a/contracts/interfaces/IVaultManager.sol b/contracts/interfaces/IVaultManager.sol
new file mode 100644
index 0000000..fa23e55
--- /dev/null
+++ b/contracts/interfaces/IVaultManager.sol
@@ -0,0 +1,266 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "oz/interfaces/IERC721Metadata.sol";
+import "oz/token/ERC20/IERC20.sol";
+import "./ITreasury.sol";
+import "./IOracle.sol";
+
+// ========================= Key Structs and Enums =============================
+
+/// @notice Parameters associated to a given `VaultManager` contract: these all correspond
+/// to parameters which signification is detailed in the `VaultManagerStorage` file
+struct VaultParameters {
+ uint256 debtCeiling;
+ uint64 collateralFactor;
+ uint64 targetHealthFactor;
+ uint64 interestRate;
+ uint64 liquidationSurcharge;
+ uint64 maxLiquidationDiscount;
+ bool whitelistingActivated;
+ uint256 baseBoost;
+}
+
+/// @notice Data stored to track someone's loan (or equivalently called position)
+struct Vault {
+ // Amount of collateral deposited in the vault, in collateral decimals. For example, if the collateral
+ // is USDC with 6 decimals, then `collateralAmount` will be in base 10**6
+ uint256 collateralAmount;
+ // Normalized value of the debt (that is to say of the stablecoins borrowed). It is expressed
+ // in the base of Angle stablecoins (i.e. `BASE_TOKENS = 10**18`)
+ uint256 normalizedDebt;
+}
+
+/// @notice For a given `vaultID`, this encodes a liquidation opportunity that is to say details about the maximum
+/// amount that could be repaid by liquidating the position
+/// @dev All the values are null in the case of a vault which cannot be liquidated under these conditions
+struct LiquidationOpportunity {
+ // Maximum stablecoin amount that can be repaid upon liquidating the vault
+ uint256 maxStablecoinAmountToRepay;
+ // Collateral amount given to the person in the case where the maximum amount to repay is given
+ uint256 maxCollateralAmountGiven;
+ // Threshold value of stablecoin amount to repay: it is ok for a liquidator to repay below threshold,
+ // but if this threshold is non null and the liquidator wants to repay more than threshold, it should repay
+ // the max stablecoin amount given in this vault
+ uint256 thresholdRepayAmount;
+ // Discount proposed to the liquidator on the collateral
+ uint256 discount;
+ // Amount of debt in the vault
+ uint256 currentDebt;
+}
+
+/// @notice Data stored during a liquidation process to keep in memory what's due to a liquidator and some
+/// essential data for vaults being liquidated
+struct LiquidatorData {
+ // Current amount of stablecoins the liquidator should give to the contract
+ uint256 stablecoinAmountToReceive;
+ // Current amount of collateral the contract should give to the liquidator
+ uint256 collateralAmountToGive;
+ // Bad debt accrued across the liquidation process
+ uint256 badDebtFromLiquidation;
+ // Oracle value (in stablecoin base) at the time of the liquidation
+ uint256 oracleValue;
+ // Value of the `interestAccumulator` at the time of the call
+ uint256 newInterestAccumulator;
+}
+
+/// @notice Data to track during a series of action the amount to give or receive in stablecoins and collateral
+/// to the caller or associated addresses
+struct PaymentData {
+ // Stablecoin amount the contract should give
+ uint256 stablecoinAmountToGive;
+ // Stablecoin amount owed to the contract
+ uint256 stablecoinAmountToReceive;
+ // Collateral amount the contract should give
+ uint256 collateralAmountToGive;
+ // Collateral amount owed to the contract
+ uint256 collateralAmountToReceive;
+}
+
+/// @notice Actions possible when composing calls to the different entry functions proposed
+enum ActionType {
+ createVault,
+ closeVault,
+ addCollateral,
+ removeCollateral,
+ repayDebt,
+ borrow,
+ getDebtIn,
+ permit
+}
+
+// ========================= Interfaces =============================
+
+/// @title IVaultManagerFunctions
+/// @author Angle Labs, Inc.
+/// @notice Interface for the `VaultManager` contract
+/// @dev This interface only contains functions of the contract which are called by other contracts
+/// of this module (without getters)
+interface IVaultManagerFunctions {
+ /// @notice Accrues interest accumulated across all vaults to the surplus and sends the surplus to the treasury
+ /// @return surplusValue Value of the surplus communicated to the `Treasury`
+ /// @return badDebtValue Value of the bad debt communicated to the `Treasury`
+ /// @dev `surplus` and `badDebt` should be reset to 0 once their current value have been given to the
+ /// `treasury` contract
+ function accrueInterestToTreasury() external returns (uint256 surplusValue, uint256 badDebtValue);
+
+ /// @notice Removes debt from a vault after being requested to do so by another `VaultManager` contract
+ /// @param vaultID ID of the vault to remove debt from
+ /// @param amountStablecoins Amount of stablecoins to remove from the debt: this amount is to be converted to an
+ /// internal debt amount
+ /// @param senderBorrowFee Borrowing fees from the contract which requested this: this is to make sure that people
+ /// are not arbitraging difference in minting fees
+ /// @param senderRepayFee Repay fees from the contract which requested this: this is to make sure that people
+ /// are not arbitraging differences in repay fees
+ /// @dev This function can only be called from a vaultManager registered in the same Treasury
+ function getDebtOut(
+ uint256 vaultID,
+ uint256 amountStablecoins,
+ uint256 senderBorrowFee,
+ uint256 senderRepayFee
+ ) external;
+
+ /// @notice Gets the current debt of a vault
+ /// @param vaultID ID of the vault to check
+ /// @return Debt of the vault
+ function getVaultDebt(uint256 vaultID) external view returns (uint256);
+
+ /// @notice Gets the total debt across all vaults
+ /// @return Total debt across all vaults, taking into account the interest accumulated
+ /// over time
+ function getTotalDebt() external view returns (uint256);
+
+ /// @notice Sets the treasury contract
+ /// @param _treasury New treasury contract
+ /// @dev All required checks when setting up a treasury contract are performed in the contract
+ /// calling this function
+ function setTreasury(address _treasury) external;
+
+ /// @notice Creates a vault
+ /// @param toVault Address for which the va
+ /// @return vaultID ID of the vault created
+ /// @dev This function just creates the vault without doing any collateral or
+ function createVault(address toVault) external returns (uint256);
+
+ /// @notice Allows composability between calls to the different entry points of this module. Any user calling
+ /// this function can perform any of the allowed actions in the order of their choice
+ /// @param actions Set of actions to perform
+ /// @param datas Data to be decoded for each action: it can include like the `vaultID` or the `stablecoinAmount`
+ /// to borrow
+ /// @param from Address from which stablecoins will be taken if one action includes burning stablecoins.
+ /// This address should either be the `msg.sender` or be approved by the latter
+ /// @param to Address to which stablecoins and/or collateral will be sent in case of
+ /// @param who Address of the contract to handle in case of repayment of stablecoins from received collateral
+ /// @param repayData Data to pass to the repayment contract in case of
+ /// @return paymentData Struct containing the accounting changes from the protocol's perspective (like how much
+ /// of collateral or how much has been received). Note that the values in the struct are not aggregated and you
+ /// could have in the output a positive amount of stablecoins to receive as well as a positive amount of
+ /// stablecoins to give
+ /// @dev This function is optimized to reduce gas cost due to payment from or to the user and that expensive calls
+ /// or computations (like `oracleValue`) are done only once
+ /// @dev When specifying `vaultID` in `data`, it is important to know that if you specify `vaultID = 0`,
+ /// it will simply use the latest `vaultID`. This is the default behavior, and unless you're engaging into some
+ /// complex protocol actions, it is encouraged to use `vaultID = 0` only when the first action of the
+ /// batch is `createVault`
+ function angle(
+ ActionType[] memory actions,
+ bytes[] memory datas,
+ address from,
+ address to,
+ address who,
+ bytes memory repayData
+ ) external returns (PaymentData memory paymentData);
+
+ /// @notice This function is a wrapper built on top of the function above. It enables users to interact with
+ /// the contract without having to provide `who` and `repayData` parameters
+ function angle(
+ ActionType[] memory actions,
+ bytes[] memory datas,
+ address from,
+ address to
+ ) external returns (PaymentData memory paymentData);
+
+ /// @notice Initializes the `VaultManager` contract
+ /// @param _treasury Treasury address handling the contract
+ /// @param _collateral Collateral supported by this contract
+ /// @param _oracle Oracle contract used
+ /// @param _symbol Symbol used to define the `VaultManager` name and symbol
+ /// @dev The parameters and the oracle are the only elements which could be modified once the
+ /// contract has been initialized
+ /// @dev For the contract to be fully initialized, governance needs to set the parameters for the liquidation
+ /// boost
+ function initialize(
+ ITreasury _treasury,
+ IERC20 _collateral,
+ IOracle _oracle,
+ VaultParameters calldata params,
+ string memory _symbol
+ ) external;
+
+ /// @notice Minimum amount of debt a vault can have, expressed in `BASE_TOKENS` that is to say the base
+ /// of the agTokens
+ function dust() external view returns (uint256);
+
+ /// @notice Pauses external permissionless functions of the contract
+ function togglePause() external;
+}
+
+/// @title IVaultManagerStorage
+/// @author Angle Labs, Inc.
+/// @notice Interface for the `VaultManager` contract
+/// @dev This interface contains getters of the contract's public variables used by other contracts
+/// of this module
+interface IVaultManagerStorage {
+ /// @notice Encodes the maximum ratio stablecoin/collateral a vault can have before being liquidated. It's what
+ /// determines the minimum collateral ratio of a position
+ function collateralFactor() external view returns (uint64);
+
+ /// @notice Stablecoin handled by this contract. Another `VaultManager` contract could have
+ /// the same rights as this `VaultManager` on the stablecoin contract
+ function stablecoin() external view returns (IAgToken);
+
+ /// @notice Reference to the `treasury` contract handling this `VaultManager`
+ function treasury() external view returns (ITreasury);
+
+ /// @notice Oracle contract to get access to the price of the collateral with respect to the stablecoin
+ function oracle() external view returns (IOracle);
+
+ /// @notice The `interestAccumulator` variable keeps track of the interest that should accrue to the protocol.
+ /// The stored value is not necessarily the true value: this one is recomputed every time an action takes place
+ /// within the protocol. It is in base `BASE_INTEREST`
+ function interestAccumulator() external view returns (uint256);
+
+ /// @notice Reference to the collateral handled by this `VaultManager`
+ function collateral() external view returns (IERC20);
+
+ /// @notice Total normalized amount of stablecoins borrowed, not taking into account the potential
+ /// bad debt accumulated. This value is expressed in the base of Angle stablecoins (`BASE_TOKENS = 10**18`)
+ function totalNormalizedDebt() external view returns (uint256);
+
+ /// @notice Maximum amount of stablecoins that can be issued with this contract. It is expressed in `BASE_TOKENS`
+ function debtCeiling() external view returns (uint256);
+
+ /// @notice Maps a `vaultID` to its data (namely collateral amount and normalized debt)
+ function vaultData(uint256 vaultID) external view returns (uint256 collateralAmount, uint256 normalizedDebt);
+
+ /// @notice ID of the last vault created. The `vaultIDCount` variables serves as a counter to generate a unique
+ /// `vaultID` for each vault: it is like `tokenID` in basic ERC721 contracts
+ function vaultIDCount() external view returns (uint256);
+}
+
+/// @title IVaultManager
+/// @author Angle Labs, Inc.
+/// @notice Interface for the `VaultManager` contract
+interface IVaultManager is IVaultManagerFunctions, IVaultManagerStorage, IERC721Metadata {
+ function isApprovedOrOwner(address spender, uint256 vaultID) external view returns (bool);
+}
+
+/// @title IVaultManagerListing
+/// @author Angle Labs, Inc.
+/// @notice Interface for the `VaultManagerListing` contract
+interface IVaultManagerListing is IVaultManager {
+ /// @notice Get the collateral owned by `user` in the contract
+ /// @dev This function effectively sums the collateral amounts of all the vaults owned by `user`
+ function getUserCollateral(address user) external view returns (uint256);
+}
diff --git a/contracts/interfaces/coreModule/IOracleCore.sol b/contracts/interfaces/coreModule/IOracleCore.sol
new file mode 100644
index 0000000..1785ba2
--- /dev/null
+++ b/contracts/interfaces/coreModule/IOracleCore.sol
@@ -0,0 +1,11 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+/// @title IOracleCore
+/// @author Angle Labs, Inc.
+interface IOracleCore {
+ function readUpper() external view returns (uint256);
+
+ function readQuoteLower(uint256 baseAmount) external view returns (uint256);
+}
diff --git a/contracts/interfaces/coreModule/IPerpetualManager.sol b/contracts/interfaces/coreModule/IPerpetualManager.sol
new file mode 100644
index 0000000..d5be036
--- /dev/null
+++ b/contracts/interfaces/coreModule/IPerpetualManager.sol
@@ -0,0 +1,31 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+/// @title IPerpetualManager
+/// @author Angle Labs, Inc.
+interface IPerpetualManager {
+ function totalHedgeAmount() external view returns (uint256);
+
+ function maintenanceMargin() external view returns (uint64);
+
+ function maxLeverage() external view returns (uint64);
+
+ function targetHAHedge() external view returns (uint64);
+
+ function limitHAHedge() external view returns (uint64);
+
+ function lockTime() external view returns (uint64);
+
+ function haBonusMalusDeposit() external view returns (uint64);
+
+ function haBonusMalusWithdraw() external view returns (uint64);
+
+ function xHAFeesDeposit(uint256) external view returns (uint64);
+
+ function yHAFeesDeposit(uint256) external view returns (uint64);
+
+ function xHAFeesWithdraw(uint256) external view returns (uint64);
+
+ function yHAFeesWithdraw(uint256) external view returns (uint64);
+}
diff --git a/contracts/interfaces/coreModule/IStableMaster.sol b/contracts/interfaces/coreModule/IStableMaster.sol
new file mode 100644
index 0000000..0e39741
--- /dev/null
+++ b/contracts/interfaces/coreModule/IStableMaster.sol
@@ -0,0 +1,96 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "./IPerpetualManager.sol";
+import "./IOracleCore.sol";
+
+// Struct to handle all the parameters to manage the fees
+// related to a given collateral pool (associated to the stablecoin)
+struct MintBurnData {
+ // Values of the thresholds to compute the minting fees
+ // depending on HA hedge (scaled by `BASE_PARAMS`)
+ uint64[] xFeeMint;
+ // Values of the fees at thresholds (scaled by `BASE_PARAMS`)
+ uint64[] yFeeMint;
+ // Values of the thresholds to compute the burning fees
+ // depending on HA hedge (scaled by `BASE_PARAMS`)
+ uint64[] xFeeBurn;
+ // Values of the fees at thresholds (scaled by `BASE_PARAMS`)
+ uint64[] yFeeBurn;
+ // Max proportion of collateral from users that can be covered by HAs
+ // It is exactly the same as the parameter of the same name in `PerpetualManager`, whenever one is updated
+ // the other changes accordingly
+ uint64 targetHAHedge;
+ // Minting fees correction set by the `FeeManager` contract: they are going to be multiplied
+ // to the value of the fees computed using the hedge curve
+ // Scaled by `BASE_PARAMS`
+ uint64 bonusMalusMint;
+ // Burning fees correction set by the `FeeManager` contract: they are going to be multiplied
+ // to the value of the fees computed using the hedge curve
+ // Scaled by `BASE_PARAMS`
+ uint64 bonusMalusBurn;
+ // Parameter used to limit the number of stablecoins that can be issued using the concerned collateral
+ uint256 capOnStableMinted;
+}
+
+// Struct to handle all the variables and parameters to handle SLPs in the protocol
+// including the fraction of interests they receive or the fees to be distributed to
+// them
+struct SLPData {
+ // Last timestamp at which the `sanRate` has been updated for SLPs
+ uint256 lastBlockUpdated;
+ // Fees accumulated from previous blocks and to be distributed to SLPs
+ uint256 lockedInterests;
+ // Max interests used to update the `sanRate` in a single block
+ // Should be in collateral token base
+ uint256 maxInterestsDistributed;
+ // Amount of fees left aside for SLPs and that will be distributed
+ // when the protocol is collateralized back again
+ uint256 feesAside;
+ // Part of the fees normally going to SLPs that is left aside
+ // before the protocol is collateralized back again (depends on collateral ratio)
+ // Updated by keepers and scaled by `BASE_PARAMS`
+ uint64 slippageFee;
+ // Portion of the fees from users minting and burning
+ // that goes to SLPs (the rest goes to surplus)
+ uint64 feesForSLPs;
+ // Slippage factor that's applied to SLPs exiting (depends on collateral ratio)
+ // If `slippage = BASE_PARAMS`, SLPs can get nothing, if `slippage = 0` they get their full claim
+ // Updated by keepers and scaled by `BASE_PARAMS`
+ uint64 slippage;
+ // Portion of the interests from lending
+ // that goes to SLPs (the rest goes to surplus)
+ uint64 interestsForSLPs;
+}
+
+/// @title IStableMaster
+/// @author Angle Labs, Inc.
+interface IStableMaster {
+ function agToken() external view returns (address);
+
+ function updateStocksUsers(uint256 amount, address poolManager) external;
+
+ function collateralMap(
+ address poolManager
+ )
+ external
+ view
+ returns (
+ address token,
+ address sanToken,
+ IPerpetualManager perpetualManager,
+ IOracleCore oracle,
+ uint256 stocksUsers,
+ uint256 sanRate,
+ uint256 collatBase,
+ SLPData memory slpData,
+ MintBurnData memory feeData
+ );
+
+ function paused(bytes32) external view returns (bool);
+
+ function deposit(uint256 amount, address user, address poolManager) external;
+
+ function withdraw(uint256 amount, address burner, address dest, address poolManager) external;
+}
diff --git a/contracts/interfaces/external/create2/ImmutableCreate2Factory.sol b/contracts/interfaces/external/create2/ImmutableCreate2Factory.sol
new file mode 100644
index 0000000..a4cffdf
--- /dev/null
+++ b/contracts/interfaces/external/create2/ImmutableCreate2Factory.sol
@@ -0,0 +1,17 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+interface ImmutableCreate2Factory {
+ function safeCreate2(bytes32 salt, bytes memory initCode) external payable returns (address deploymentAddress);
+
+ function findCreate2Address(
+ bytes32 salt,
+ bytes calldata initCode
+ ) external view returns (address deploymentAddress);
+
+ function findCreate2AddressViaHash(
+ bytes32 salt,
+ bytes32 initCodeHash
+ ) external view returns (address deploymentAddress);
+}
diff --git a/contracts/mock/MockCoreBorrow.sol b/contracts/mock/MockCoreBorrow.sol
new file mode 100644
index 0000000..b67ce73
--- /dev/null
+++ b/contracts/mock/MockCoreBorrow.sol
@@ -0,0 +1,53 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "../interfaces/ICoreBorrow.sol";
+import "../interfaces/IFlashAngle.sol";
+import "../interfaces/ITreasury.sol";
+
+contract MockCoreBorrow is ICoreBorrow {
+ mapping(address => bool) public flashLoaners;
+ mapping(address => bool) public governors;
+ mapping(address => bool) public guardians;
+
+ function isFlashLoanerTreasury(address treasury) external view override returns (bool) {
+ return flashLoaners[treasury];
+ }
+
+ function isGovernor(address admin) external view override returns (bool) {
+ return governors[admin];
+ }
+
+ function isGovernorOrGuardian(address admin) external view override returns (bool) {
+ return governors[admin] || guardians[admin];
+ }
+
+ function toggleGovernor(address admin) external {
+ governors[admin] = !governors[admin];
+ }
+
+ function toggleGuardian(address admin) external {
+ guardians[admin] = !guardians[admin];
+ }
+
+ function toggleFlashLoaners(address admin) external {
+ flashLoaners[admin] = !flashLoaners[admin];
+ }
+
+ function addStablecoinSupport(IFlashAngle flashAngle, address _treasury) external {
+ flashAngle.addStablecoinSupport(_treasury);
+ }
+
+ function removeStablecoinSupport(IFlashAngle flashAngle, address _treasury) external {
+ flashAngle.removeStablecoinSupport(_treasury);
+ }
+
+ function setCore(IFlashAngle flashAngle, address _core) external {
+ flashAngle.setCore(_core);
+ }
+
+ function setFlashLoanModule(ITreasury _treasury, address _flashLoanModule) external {
+ _treasury.setFlashLoanModule(_flashLoanModule);
+ }
+}
diff --git a/contracts/mock/MockFlashLoanModule.sol b/contracts/mock/MockFlashLoanModule.sol
new file mode 100644
index 0000000..13bc01d
--- /dev/null
+++ b/contracts/mock/MockFlashLoanModule.sol
@@ -0,0 +1,38 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "../interfaces/IFlashAngle.sol";
+import "../interfaces/ICoreBorrow.sol";
+
+contract MockFlashLoanModule is IFlashAngle {
+ ICoreBorrow public override core;
+ mapping(address => bool) public stablecoinsSupported;
+ mapping(IAgToken => uint256) public interestAccrued;
+ uint256 public surplusValue;
+
+ constructor(ICoreBorrow _core) {
+ core = _core;
+ }
+
+ function accrueInterestToTreasury(IAgToken stablecoin) external override returns (uint256 balance) {
+ balance = surplusValue;
+ interestAccrued[stablecoin] += balance;
+ }
+
+ function addStablecoinSupport(address _treasury) external override {
+ stablecoinsSupported[_treasury] = true;
+ }
+
+ function removeStablecoinSupport(address _treasury) external override {
+ stablecoinsSupported[_treasury] = false;
+ }
+
+ function setCore(address _core) external override {
+ core = ICoreBorrow(_core);
+ }
+
+ function setSurplusValue(uint256 _surplusValue) external {
+ surplusValue = _surplusValue;
+ }
+}
diff --git a/contracts/mock/MockFlashLoanReceiver.sol b/contracts/mock/MockFlashLoanReceiver.sol
new file mode 100644
index 0000000..c29e49d
--- /dev/null
+++ b/contracts/mock/MockFlashLoanReceiver.sol
@@ -0,0 +1,30 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.7;
+
+import "oz/token/ERC20/ERC20.sol";
+
+import "oz/interfaces/IERC3156FlashBorrower.sol";
+
+import "oz/interfaces/IERC3156FlashLender.sol";
+
+contract MockFlashLoanReceiver is IERC3156FlashBorrower {
+ bytes32 public constant CALLBACK_SUCCESS = keccak256("ERC3156FlashBorrower.onFlashLoan");
+
+ constructor() {}
+
+ function onFlashLoan(
+ address,
+ address token,
+ uint256 amount,
+ uint256 fee,
+ bytes calldata data
+ ) external override returns (bytes32) {
+ IERC20(token).approve(msg.sender, amount + fee);
+ if (amount >= 10 ** 21) return keccak256("error");
+ if (amount == 2 * 10 ** 18) {
+ IERC3156FlashLender(msg.sender).flashLoan(IERC3156FlashBorrower(address(this)), token, amount, data);
+ return keccak256("reentrant");
+ } else return CALLBACK_SUCCESS;
+ }
+}
diff --git a/contracts/mock/MockLayerZero.sol b/contracts/mock/MockLayerZero.sol
new file mode 100644
index 0000000..645da7d
--- /dev/null
+++ b/contracts/mock/MockLayerZero.sol
@@ -0,0 +1,76 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "layer-zero/interfaces/ILayerZeroReceiver.sol";
+
+contract MockLayerZero {
+ mapping(uint16 => uint256) public counters;
+ uint256 public config;
+ mapping(uint16 => uint64) public outboundNonce;
+ uint256 public resumeReceived;
+ uint256 public sendVersion;
+ uint256 public receiveVersion;
+
+ /// @notice Initiate with a fixe change rate
+ constructor() {}
+
+ function send(
+ uint16 _dstChainId,
+ bytes calldata,
+ bytes calldata,
+ address,
+ address,
+ bytes calldata
+ ) external payable {
+ counters[_dstChainId] += 1;
+ }
+
+ function getOutboundNonce(uint16 _dstChainId, address) external view returns (uint64) {
+ return outboundNonce[_dstChainId];
+ }
+
+ function setOutBoundNonce(uint16 _from, uint64 value) external {
+ outboundNonce[_from] = value;
+ }
+
+ function lzReceive(
+ address lzApp,
+ uint16 _srcChainId,
+ bytes memory _srcAddress,
+ uint64 _nonce,
+ bytes memory _payload
+ ) public {
+ ILayerZeroReceiver(lzApp).lzReceive(_srcChainId, _srcAddress, _nonce, _payload);
+ }
+
+ function estimateFees(
+ uint16,
+ address,
+ bytes calldata,
+ bool,
+ bytes calldata
+ ) external pure returns (uint256 nativeFee, uint256 zroFee) {
+ return (123, 456);
+ }
+
+ function setConfig(uint16, uint16, uint256 _configType, bytes calldata) external {
+ config = _configType;
+ }
+
+ function getConfig(uint16, uint16, address, uint256) external view returns (bytes memory) {
+ return abi.encodePacked(config);
+ }
+
+ function setSendVersion(uint16 _version) external {
+ sendVersion = _version;
+ }
+
+ function setReceiveVersion(uint16 _version) external {
+ receiveVersion = _version;
+ }
+
+ function forceResumeReceive(uint16, bytes calldata) external {
+ resumeReceived = 1 - resumeReceived;
+ }
+}
diff --git a/contracts/mock/MockToken.sol b/contracts/mock/MockToken.sol
new file mode 100644
index 0000000..a7ecb57
--- /dev/null
+++ b/contracts/mock/MockToken.sol
@@ -0,0 +1,54 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.7;
+
+import "oz/token/ERC20/ERC20.sol";
+
+contract MockToken is ERC20 {
+ event Minting(address indexed _to, address indexed _minter, uint256 _amount);
+
+ event Burning(address indexed _from, address indexed _burner, uint256 _amount);
+
+ uint8 internal _decimal;
+ mapping(address => bool) public minters;
+ address public treasury;
+
+ constructor(string memory name_, string memory symbol_, uint8 decimal_) ERC20(name_, symbol_) {
+ _decimal = decimal_;
+ }
+
+ function decimals() public view override returns (uint8) {
+ return _decimal;
+ }
+
+ function mint(address account, uint256 amount) external {
+ _mint(account, amount);
+ emit Minting(account, msg.sender, amount);
+ }
+
+ function burn(address account, uint256 amount) public {
+ _burn(account, amount);
+ emit Burning(account, msg.sender, amount);
+ }
+
+ function setAllowance(address from, address to) public {
+ _approve(from, to, type(uint256).max);
+ }
+
+ function burnSelf(uint256 amount, address account) public {
+ _burn(account, amount);
+ emit Burning(account, msg.sender, amount);
+ }
+
+ function addMinter(address minter) public {
+ minters[minter] = true;
+ }
+
+ function removeMinter(address minter) public {
+ minters[minter] = false;
+ }
+
+ function setTreasury(address _treasury) public {
+ treasury = _treasury;
+ }
+}
diff --git a/contracts/mock/MockTokenPermit.sol b/contracts/mock/MockTokenPermit.sol
new file mode 100644
index 0000000..f1ee3da
--- /dev/null
+++ b/contracts/mock/MockTokenPermit.sol
@@ -0,0 +1,87 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.7;
+
+import "oz/token/ERC20/extensions/draft-ERC20Permit.sol";
+import "oz/token/ERC20/IERC20.sol";
+import "oz/token/ERC20/utils/SafeERC20.sol";
+
+contract MockTokenPermit is ERC20Permit {
+ using SafeERC20 for IERC20;
+ event Minting(address indexed _to, address indexed _minter, uint256 _amount);
+
+ event Burning(address indexed _from, address indexed _burner, uint256 _amount);
+
+ uint8 internal _decimal;
+ mapping(address => bool) public minters;
+ address public treasury;
+ uint256 public fees;
+
+ bool public reverts;
+
+ constructor(string memory name_, string memory symbol_, uint8 decimal_) ERC20Permit(name_) ERC20(name_, symbol_) {
+ _decimal = decimal_;
+ }
+
+ function decimals() public view override returns (uint8) {
+ return _decimal;
+ }
+
+ function mint(address account, uint256 amount) external {
+ _mint(account, amount);
+ emit Minting(account, msg.sender, amount);
+ }
+
+ function burn(address account, uint256 amount) public {
+ _burn(account, amount);
+ emit Burning(account, msg.sender, amount);
+ }
+
+ function setAllowance(address from, address to) public {
+ _approve(from, to, type(uint256).max);
+ }
+
+ function burnSelf(uint256 amount, address account) public {
+ _burn(account, amount);
+ emit Burning(account, msg.sender, amount);
+ }
+
+ function addMinter(address minter) public {
+ minters[minter] = true;
+ }
+
+ function removeMinter(address minter) public {
+ minters[minter] = false;
+ }
+
+ function setTreasury(address _treasury) public {
+ treasury = _treasury;
+ }
+
+ function setFees(uint256 _fees) public {
+ fees = _fees;
+ }
+
+ function recoverERC20(IERC20 token, address to, uint256 amount) external {
+ token.safeTransfer(to, amount);
+ }
+
+ function swapIn(address bridgeToken, uint256 amount, address to) external returns (uint256) {
+ require(!reverts);
+
+ IERC20(bridgeToken).safeTransferFrom(msg.sender, address(this), amount);
+ uint256 canonicalOut = amount;
+ canonicalOut -= (canonicalOut * fees) / 10 ** 9;
+ _mint(to, canonicalOut);
+ return canonicalOut;
+ }
+
+ function swapOut(address bridgeToken, uint256 amount, address to) external returns (uint256) {
+ require(!reverts);
+ _burn(msg.sender, amount);
+ uint256 bridgeOut = amount;
+ bridgeOut -= (bridgeOut * fees) / 10 ** 9;
+ IERC20(bridgeToken).safeTransfer(to, bridgeOut);
+ return bridgeOut;
+ }
+}
diff --git a/contracts/mock/MockTreasury.sol b/contracts/mock/MockTreasury.sol
new file mode 100644
index 0000000..44068cd
--- /dev/null
+++ b/contracts/mock/MockTreasury.sol
@@ -0,0 +1,85 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "../interfaces/ITreasury.sol";
+import "../interfaces/IFlashAngle.sol";
+import "../interfaces/IVaultManager.sol";
+
+contract MockTreasury is ITreasury {
+ IAgToken public override stablecoin;
+ address public governor;
+ address public guardian;
+ address public vaultManager1;
+ address public vaultManager2;
+ address public flashLoanModule;
+ address[] public vaultManagerList;
+
+ constructor(
+ IAgToken _stablecoin,
+ address _governor,
+ address _guardian,
+ address _vaultManager1,
+ address _vaultManager2,
+ address _flashLoanModule
+ ) {
+ stablecoin = _stablecoin;
+ governor = _governor;
+ guardian = _guardian;
+ vaultManager1 = _vaultManager1;
+ vaultManager2 = _vaultManager2;
+ flashLoanModule = _flashLoanModule;
+ }
+
+ function isGovernor(address admin) external view override returns (bool) {
+ return (admin == governor);
+ }
+
+ function isGovernorOrGuardian(address admin) external view override returns (bool) {
+ return (admin == governor || admin == guardian);
+ }
+
+ function isVaultManager(address _vaultManager) external view override returns (bool) {
+ return (_vaultManager == vaultManager1 || _vaultManager == vaultManager2);
+ }
+
+ function setStablecoin(IAgToken _stablecoin) external {
+ stablecoin = _stablecoin;
+ }
+
+ function setFlashLoanModule(address _flashLoanModule) external override {
+ flashLoanModule = _flashLoanModule;
+ }
+
+ function setGovernor(address _governor) external {
+ governor = _governor;
+ }
+
+ function setVaultManager(address _vaultManager) external {
+ vaultManager1 = _vaultManager;
+ }
+
+ function setVaultManager2(address _vaultManager) external {
+ vaultManager2 = _vaultManager;
+ }
+
+ function setTreasury(address _agTokenOrVaultManager, address _treasury) external {
+ IAgToken(_agTokenOrVaultManager).setTreasury(_treasury);
+ }
+
+ function addMinter(IAgToken _agToken, address _minter) external {
+ _agToken.addMinter(_minter);
+ }
+
+ function removeMinter(IAgToken _agToken, address _minter) external {
+ _agToken.removeMinter(_minter);
+ }
+
+ function accrueInterestToTreasury(IFlashAngle flashAngle) external returns (uint256 balance) {
+ balance = flashAngle.accrueInterestToTreasury(stablecoin);
+ }
+
+ function accrueInterestToTreasuryVaultManager(IVaultManager _vaultManager) external returns (uint256, uint256) {
+ return _vaultManager.accrueInterestToTreasury();
+ }
+}
diff --git a/contracts/treasury/Treasury.sol b/contracts/treasury/Treasury.sol
new file mode 100644
index 0000000..63c172d
--- /dev/null
+++ b/contracts/treasury/Treasury.sol
@@ -0,0 +1,413 @@
+// SPDX-License-Identifier: GPL-3.0
+
+/*
+ * █
+ ***** ▓▓▓
+ * ▓▓▓▓▓▓▓
+ * ///. ▓▓▓▓▓▓▓▓▓▓▓▓▓
+ ***** //////// ▓▓▓▓▓▓▓
+ * ///////////// ▓▓▓
+ ▓▓ ////////////////// █ ▓▓
+ ▓▓ ▓▓ /////////////////////// ▓▓ ▓▓
+ ▓▓ ▓▓ //////////////////////////// ▓▓ ▓▓
+ ▓▓ ▓▓ /////////▓▓▓///////▓▓▓///////// ▓▓ ▓▓
+ ▓▓ ,////////////////////////////////////// ▓▓ ▓▓
+ ▓▓ ////////////////////////////////////////// ▓▓
+ ▓▓ //////////////////////▓▓▓▓/////////////////////
+ ,////////////////////////////////////////////////////
+ .//////////////////////////////////////////////////////////
+ .//////////////////////////██.,//////////////////////////█
+ .//////////////////////████..,./////////////////////██
+ ...////////////////███████.....,.////////////////███
+ ,.,////////////████████ ........,///////////████
+ .,.,//////█████████ ,.......///////████
+ ,..//████████ ........./████
+ ..,██████ .....,███
+ .██ ,.,█
+
+
+
+ ▓▓ ▓▓▓▓▓▓▓▓▓▓ ▓▓▓▓▓▓▓▓▓▓ ▓▓ ▓▓▓▓▓▓▓▓▓▓
+ ▓▓▓▓▓▓ ▓▓▓ ▓▓▓ ▓▓▓ ▓▓ ▓▓ ▓▓▓▓
+ ▓▓▓ ▓▓▓ ▓▓▓ ▓▓▓ ▓▓▓ ▓▓▓ ▓▓ ▓▓▓▓▓
+ ▓▓▓ ▓▓ ▓▓▓ ▓▓▓ ▓▓▓▓▓▓▓▓▓▓ ▓▓▓▓▓▓▓▓▓▓ ▓▓▓▓▓▓▓▓▓▓
+*/
+
+pragma solidity ^0.8.12;
+
+import "oz-upgradeable/proxy/utils/Initializable.sol";
+import "oz/token/ERC20/utils/SafeERC20.sol";
+import "oz/token/ERC20/IERC20.sol";
+
+import "../interfaces/IAgToken.sol";
+import "../interfaces/ICoreBorrow.sol";
+import "../interfaces/IFlashAngle.sol";
+import "../interfaces/ITreasury.sol";
+import "../interfaces/IVaultManager.sol";
+
+/// @title Treasury
+/// @author Angle Labs, Inc.
+/// @notice Treasury of Angle Borrowing Module doing the accounting across all VaultManagers for
+/// a given stablecoin
+contract Treasury is ITreasury, Initializable {
+ using SafeERC20 for IERC20;
+
+ /// @notice Base used for parameter computation
+ uint256 public constant BASE_9 = 1e9;
+
+ // ================================= REFERENCES ================================
+
+ /// @notice Reference to the `CoreBorrow` contract of the module which handles all AccessControl logic
+ ICoreBorrow public core;
+ /// @notice Flash Loan Module with a minter right on the stablecoin
+ IFlashAngle public flashLoanModule;
+ /// @inheritdoc ITreasury
+ IAgToken public stablecoin;
+ /// @notice Address responsible for handling the surplus made by the treasury
+ address public surplusManager;
+ /// @notice List of the accepted `VaultManager` of the protocol
+ address[] public vaultManagerList;
+ /// @notice Maps an address to 1 if it was initialized as a `VaultManager` contract
+ mapping(address => uint256) public vaultManagerMap;
+
+ // ================================= VARIABLES =================================
+
+ /// @notice Amount of bad debt (unbacked stablecoin) accumulated across all `VaultManager` contracts
+ /// linked to this stablecoin
+ uint256 public badDebt;
+ /// @notice Surplus amount accumulated by the contract waiting to be distributed to governance. Technically
+ /// only a share of this `surplusBuffer` will go to governance. Once a share of the surplus buffer has been
+ /// given to governance, then this surplus is reset
+ uint256 public surplusBuffer;
+
+ // ================================= PARAMETER =================================
+
+ /// @notice Share of the `surplusBuffer` distributed to governance (in `BASE_9`)
+ uint64 public surplusForGovernance;
+
+ // =================================== EVENTS ==================================
+
+ event BadDebtUpdated(uint256 badDebtValue);
+ event CoreUpdated(address indexed _core);
+ event NewTreasurySet(address indexed _treasury);
+ event Recovered(address indexed token, address indexed to, uint256 amount);
+ event SurplusBufferUpdated(uint256 surplusBufferValue);
+ event SurplusForGovernanceUpdated(uint64 _surplusForGovernance);
+ event SurplusManagerUpdated(address indexed _surplusManager);
+ event VaultManagerToggled(address indexed vaultManager);
+
+ // =================================== ERRORS ==================================
+
+ error AlreadyVaultManager();
+ error InvalidAddress();
+ error InvalidTreasury();
+ error NotCore();
+ error NotGovernor();
+ error NotVaultManager();
+ error RightsNotRemoved();
+ error TooBigAmount();
+ error TooHighParameterValue();
+ error ZeroAddress();
+
+ // ================================== MODIFIER =================================
+
+ /// @notice Checks whether the `msg.sender` has the governor role or not
+ modifier onlyGovernor() {
+ if (!core.isGovernor(msg.sender)) revert NotGovernor();
+ _;
+ }
+
+ /// @notice Initializes the treasury contract
+ /// @param _core Address of the `CoreBorrow` contract of the module
+ /// @param _stablecoin Address of the stablecoin
+ function initialize(ICoreBorrow _core, IAgToken _stablecoin) public virtual initializer {
+ if (address(_stablecoin) == address(0) || address(_core) == address(0)) revert ZeroAddress();
+ core = _core;
+ stablecoin = _stablecoin;
+ }
+
+ /// @custom:oz-upgrades-unsafe-allow constructor
+ constructor() initializer {}
+
+ // =============================== VIEW FUNCTIONS ==============================
+
+ /// @inheritdoc ITreasury
+ function isGovernor(address admin) external view returns (bool) {
+ return core.isGovernor(admin);
+ }
+
+ /// @inheritdoc ITreasury
+ function isGovernorOrGuardian(address admin) external view returns (bool) {
+ return core.isGovernorOrGuardian(admin);
+ }
+
+ /// @inheritdoc ITreasury
+ function isVaultManager(address _vaultManager) external view returns (bool) {
+ return vaultManagerMap[_vaultManager] == 1;
+ }
+
+ // ===================== EXTERNAL PERMISSIONLESS FUNCTIONS =====================
+
+ /// @notice Fetches the surplus accrued across all the `VaultManager` contracts controlled by this
+ /// `Treasury` contract as well as from the fees of the `FlashLoan` module
+ /// @return Surplus buffer value at the end of the call
+ /// @return Bad debt value at the end of the call
+ /// @dev This function pools surplus and bad debt across all contracts and then updates the `surplusBuffer`
+ /// (or the `badDebt` if more losses were made than profits)
+ function fetchSurplusFromAll() external returns (uint256, uint256) {
+ return _fetchSurplusFromAll();
+ }
+
+ /// @notice Fetches the surplus accrued in the flash loan module and updates the `surplusBuffer`
+ /// @return Surplus buffer value at the end of the call
+ /// @return Bad debt value at the end of the call
+ /// @dev This function fails if the `flashLoanModule` has not been initialized yet
+ function fetchSurplusFromFlashLoan() external returns (uint256, uint256) {
+ uint256 surplusBufferValue = surplusBuffer + flashLoanModule.accrueInterestToTreasury(stablecoin);
+ return _updateSurplusAndBadDebt(surplusBufferValue, badDebt);
+ }
+
+ /// @notice Pushes the surplus buffer to the `surplusManager` contract
+ /// @return governanceAllocation Amount transferred to governance
+ /// @dev It makes sure to fetch the surplus from all the contracts handled by this treasury to avoid
+ /// the situation where rewards are still distributed to governance even though a `VaultManager` has made
+ /// a big loss
+ /// @dev Typically this function is to be called once every week by a keeper to distribute rewards to veANGLE
+ /// holders
+ /// @dev `stablecoin` must be an AgToken and hence `transfer` reverts if the call is not successful
+ function pushSurplus() external returns (uint256 governanceAllocation) {
+ address _surplusManager = surplusManager;
+ if (_surplusManager == address(0)) {
+ revert ZeroAddress();
+ }
+ (uint256 surplusBufferValue, ) = _fetchSurplusFromAll();
+ surplusBuffer = 0;
+ emit SurplusBufferUpdated(0);
+ governanceAllocation = (surplusForGovernance * surplusBufferValue) / BASE_9;
+ stablecoin.transfer(_surplusManager, governanceAllocation);
+ }
+
+ /// @notice Updates the bad debt of the protocol in case where the protocol has accumulated some revenue
+ /// from an external source
+ /// @param amount Amount to reduce the bad debt of
+ /// @return badDebtValue Value of the bad debt at the end of the call
+ /// @dev If the protocol has made a loss and managed to make some profits to recover for this loss (through
+ /// a program like Olympus Pro), then this function needs to be called
+ /// @dev `badDebt` is simply reduced here by burning stablecoins
+ /// @dev It is impossible to burn more than the `badDebt` otherwise this function could be used to manipulate
+ /// the `surplusBuffer` and hence the amount going to governance
+ function updateBadDebt(uint256 amount) external returns (uint256 badDebtValue) {
+ stablecoin.burnSelf(amount, address(this));
+ badDebtValue = badDebt - amount;
+ badDebt = badDebtValue;
+ emit BadDebtUpdated(badDebtValue);
+ }
+
+ // ========================= INTERNAL UTILITY FUNCTIONS ========================
+
+ /// @notice Internal version of the `fetchSurplusFromAll` function
+ function _fetchSurplusFromAll() internal returns (uint256 surplusBufferValue, uint256 badDebtValue) {
+ (surplusBufferValue, badDebtValue) = _fetchSurplusFromList(vaultManagerList);
+ // It will fail anyway if the `flashLoanModule` is the zero address
+ if (address(flashLoanModule) != address(0))
+ surplusBufferValue += flashLoanModule.accrueInterestToTreasury(stablecoin);
+ (surplusBufferValue, badDebtValue) = _updateSurplusAndBadDebt(surplusBufferValue, badDebtValue);
+ }
+
+ /// @notice Fetches the surplus from a list of `VaultManager` addresses without modifying the
+ /// `surplusBuffer` and `badDebtValue`
+ /// @return surplusBufferValue Value the `surplusBuffer` should have after the call if it was updated
+ /// @return badDebtValue Value the `badDebt` should have after the call if it was updated
+ /// @dev This internal function is never to be called alone, and should always be called in conjunction
+ /// with the `_updateSurplusAndBadDebt` function
+ function _fetchSurplusFromList(
+ address[] memory vaultManagers
+ ) internal returns (uint256 surplusBufferValue, uint256 badDebtValue) {
+ badDebtValue = badDebt;
+ surplusBufferValue = surplusBuffer;
+ uint256 newSurplus;
+ uint256 newBadDebt;
+ uint256 vaultManagersLength = vaultManagers.length;
+ for (uint256 i; i < vaultManagersLength; ++i) {
+ (newSurplus, newBadDebt) = IVaultManager(vaultManagers[i]).accrueInterestToTreasury();
+ surplusBufferValue += newSurplus;
+ badDebtValue += newBadDebt;
+ }
+ }
+
+ /// @notice Updates the `surplusBuffer` and the `badDebt` from updated values after calling the flash loan module
+ /// and/or a list of `VaultManager` contracts
+ /// @param surplusBufferValue Value of the surplus buffer after the calls to the different modules
+ /// @param badDebtValue Value of the bad debt after the calls to the different modules
+ /// @return Value of the `surplusBuffer` corrected from the `badDebt`
+ /// @return Value of the `badDebt` corrected from the `surplusBuffer` and from the surplus the treasury
+ /// had accumulated previously
+ /// @dev When calling this function, it is possible that there is a positive `surplusBufferValue`
+ /// and `badDebtValue`, this function tries to reconcile both values and makes sure that we either have
+ /// surplus or bad debt but not both at the same time
+ function _updateSurplusAndBadDebt(
+ uint256 surplusBufferValue,
+ uint256 badDebtValue
+ ) internal returns (uint256, uint256) {
+ if (badDebtValue != 0) {
+ // If we have bad debt we need to burn stablecoins that accrued to the protocol
+ // We still need to make sure that we're not burning too much or as much as we can if the debt is big
+ uint256 balance = stablecoin.balanceOf(address(this));
+ // We are going to burn `min(balance, badDebtValue)`
+ uint256 toBurn = balance <= badDebtValue ? balance : badDebtValue;
+ stablecoin.burnSelf(toBurn, address(this));
+ // If we burned more than `surplusBuffer`, we set surplus to 0. It means we had to tap into Treasury reserve
+ surplusBufferValue = toBurn >= surplusBufferValue ? 0 : surplusBufferValue - toBurn;
+ badDebtValue -= toBurn;
+ // Note here that the stablecoin balance is necessarily greater than the surplus buffer, and so if
+ // `surplusBuffer >= toBurn`, then `badDebtValue = toBurn`
+ }
+ surplusBuffer = surplusBufferValue;
+ badDebt = badDebtValue;
+ emit SurplusBufferUpdated(surplusBufferValue);
+ emit BadDebtUpdated(badDebtValue);
+ return (surplusBufferValue, badDebtValue);
+ }
+
+ /// @notice Adds a new `VaultManager`
+ /// @param vaultManager `VaultManager` contract to add
+ /// @dev This contract should have already been initialized with a correct treasury address
+ /// @dev It's this function that gives the minter right to the `VaultManager`
+ function _addVaultManager(address vaultManager) internal virtual {
+ if (vaultManagerMap[vaultManager] == 1) revert AlreadyVaultManager();
+ if (address(IVaultManager(vaultManager).treasury()) != address(this)) revert InvalidTreasury();
+ vaultManagerMap[vaultManager] = 1;
+ vaultManagerList.push(vaultManager);
+ emit VaultManagerToggled(vaultManager);
+ stablecoin.addMinter(vaultManager);
+ }
+
+ // ============================= GOVERNOR FUNCTIONS ============================
+
+ /// @notice Adds a new minter for the stablecoin
+ /// @param minter Minter address to add
+ function addMinter(address minter) external virtual onlyGovernor {
+ if (minter == address(0)) revert ZeroAddress();
+ stablecoin.addMinter(minter);
+ }
+
+ /// @notice External wrapper for `_addVaultManager`
+ function addVaultManager(address vaultManager) external virtual onlyGovernor {
+ _addVaultManager(vaultManager);
+ }
+
+ /// @notice Removes a minter from the stablecoin contract
+ /// @param minter Minter address to remove
+ function removeMinter(address minter) external virtual onlyGovernor {
+ // To remove the minter role to a `VaultManager` you have to go through the `removeVaultManager` function
+ if (vaultManagerMap[minter] == 1) revert InvalidAddress();
+ stablecoin.removeMinter(minter);
+ }
+
+ /// @notice Removes a `VaultManager`
+ /// @param vaultManager `VaultManager` contract to remove
+ /// @dev A removed `VaultManager` loses its minter right on the stablecoin
+ function removeVaultManager(address vaultManager) external onlyGovernor {
+ if (vaultManagerMap[vaultManager] != 1) revert NotVaultManager();
+ delete vaultManagerMap[vaultManager];
+ // deletion from `vaultManagerList` loop
+ uint256 vaultManagerListLength = vaultManagerList.length;
+ for (uint256 i; i < vaultManagerListLength - 1; ++i) {
+ if (vaultManagerList[i] == vaultManager) {
+ // replace the `VaultManager` to remove with the last of the list
+ vaultManagerList[i] = vaultManagerList[vaultManagerListLength - 1];
+ break;
+ }
+ }
+ // remove last element in array
+ vaultManagerList.pop();
+ emit VaultManagerToggled(vaultManager);
+ stablecoin.removeMinter(vaultManager);
+ }
+
+ /// @notice Allows to recover any ERC20 token, including the stablecoin handled by this contract, and to send it
+ /// to a contract
+ /// @param tokenAddress Address of the token to recover
+ /// @param to Address of the contract to send collateral to
+ /// @param amountToRecover Amount of collateral to transfer
+ /// @dev It is impossible to recover the stablecoin of the protocol if there is some bad debt for it
+ /// @dev In this case, the function makes sure to fetch the surplus/bad debt from all the `VaultManager` contracts
+ /// and from the flash loan module
+ /// @dev If the token to recover is the stablecoin, tokens recovered are fetched
+ /// from the surplus and not from the `surplusBuffer`
+ function recoverERC20(address tokenAddress, address to, uint256 amountToRecover) external onlyGovernor {
+ // Cannot recover stablecoin if badDebt or tap into the surplus buffer
+ if (tokenAddress == address(stablecoin)) {
+ _fetchSurplusFromAll();
+ // If balance is non zero then this means, after the call to `fetchSurplusFromAll` that
+ // bad debt is necessarily null
+ uint256 balance = stablecoin.balanceOf(address(this));
+ if (amountToRecover + surplusBuffer > balance) revert TooBigAmount();
+ stablecoin.transfer(to, amountToRecover);
+ } else {
+ IERC20(tokenAddress).safeTransfer(to, amountToRecover);
+ }
+ emit Recovered(tokenAddress, to, amountToRecover);
+ }
+
+ /// @notice Changes the treasury contract and communicates this change to all `VaultManager` contract
+ /// @param _treasury New treasury address for this stablecoin
+ /// @dev This function is basically a way to remove rights to this contract and grant them to a new one
+ /// @dev It could be used to set a new core contract
+ function setTreasury(address _treasury) external virtual onlyGovernor {
+ if (ITreasury(_treasury).stablecoin() != stablecoin) revert InvalidTreasury();
+ // Flash loan role should be removed before calling this function
+ if (core.isFlashLoanerTreasury(address(this))) revert RightsNotRemoved();
+ emit NewTreasurySet(_treasury);
+ uint256 vaultManagerListLength = vaultManagerList.length;
+ for (uint256 i; i < vaultManagerListLength; ++i) {
+ IVaultManager(vaultManagerList[i]).setTreasury(_treasury);
+ }
+ // A `TreasuryUpdated` event is triggered in the stablecoin
+ stablecoin.setTreasury(_treasury);
+ }
+
+ /// @notice Sets the `surplusForGovernance` parameter
+ /// @param _surplusForGovernance New value of the parameter
+ /// @dev To pause surplus distribution, governance needs to set a zero value for `surplusForGovernance`
+ /// which means
+ function setSurplusForGovernance(uint64 _surplusForGovernance) external onlyGovernor {
+ if (_surplusForGovernance > BASE_9) revert TooHighParameterValue();
+ surplusForGovernance = _surplusForGovernance;
+ emit SurplusForGovernanceUpdated(_surplusForGovernance);
+ }
+
+ /// @notice Sets the `surplusManager` contract responsible for handling the surplus of the
+ /// protocol
+ /// @param _surplusManager New address responsible for handling the surplus
+ function setSurplusManager(address _surplusManager) external onlyGovernor {
+ if (_surplusManager == address(0)) revert ZeroAddress();
+ surplusManager = _surplusManager;
+ emit SurplusManagerUpdated(_surplusManager);
+ }
+
+ /// @notice Sets a new `core` contract
+ /// @dev This function should typically be called on all treasury contracts after the `setCore`
+ /// function has been called on the `CoreBorrow` contract
+ /// @dev One sanity check that can be performed here is to verify whether at least the governor
+ /// calling the contract is still a governor in the new core
+ function setCore(ICoreBorrow _core) external onlyGovernor {
+ if (!_core.isGovernor(msg.sender)) revert NotGovernor();
+ core = ICoreBorrow(_core);
+ emit CoreUpdated(address(_core));
+ }
+
+ /// @inheritdoc ITreasury
+ function setFlashLoanModule(address _flashLoanModule) external {
+ if (msg.sender != address(core)) revert NotCore();
+ address oldFlashLoanModule = address(flashLoanModule);
+ flashLoanModule = IFlashAngle(_flashLoanModule);
+ if (oldFlashLoanModule != address(0)) {
+ stablecoin.removeMinter(oldFlashLoanModule);
+ }
+ // We may want to cancel the module
+ if (_flashLoanModule != address(0)) {
+ stablecoin.addMinter(_flashLoanModule);
+ }
+ }
+}
diff --git a/contracts/utils/Constants.sol b/contracts/utils/Constants.sol
new file mode 100644
index 0000000..93344b3
--- /dev/null
+++ b/contracts/utils/Constants.sol
@@ -0,0 +1,15 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+error InsufficientAssets();
+
+/// @title Constants
+/// @author Angle Labs, Inc.
+/// @notice Constants and errors for Angle Protocol contracts
+contract Constants {
+ uint256 internal constant _BASE_9 = 1e9;
+ uint256 internal constant _BASE_18 = 1e18;
+ uint256 internal constant _BASE_27 = 1e27;
+ uint256 internal constant _BASE_36 = 1e36;
+}
diff --git a/foundry.toml b/foundry.toml
index 25b918f..07fba83 100644
--- a/foundry.toml
+++ b/foundry.toml
@@ -1,6 +1,73 @@
-[profile.default]
-src = "src"
-out = "out"
-libs = ["lib"]
-
-# See more config options https://github.com/foundry-rs/foundry/blob/master/crates/config/README.md#all-options
+[profile.default]
+src = "contracts"
+out = "out"
+test = "test"
+libs = ["lib"]
+script = "scripts"
+cache_path = "cache-forge"
+gas_reports = ["*"]
+via_ir = true
+sizes = true
+optimizer = true
+optimizer_runs = 1000
+solc_version = "0.8.22"
+ffi = true
+
+[fuzz]
+runs = 10000
+
+[invariant]
+runs = 1000
+depth = 30
+
+[rpc_endpoints]
+arbitrum = "${ETH_NODE_URI_ARBITRUM}"
+gnosis = "${ETH_NODE_URI_GNOSIS}"
+mainnet = "${ETH_NODE_URI_MAINNET}"
+optimism = "${ETH_NODE_URI_OPTIMISM}"
+polygon = "${ETH_NODE_URI_POLYGON}"
+fork = "${ETH_NODE_URI_FORK}"
+avalanche = "${ETH_NODE_URI_AVALANCHE}"
+celo = "${ETH_NODE_URI_CELO}"
+polygonzkevm = "${ETH_NODE_URI_POLYGONZKEVM}"
+bsc = "${ETH_NODE_URI_BSC}"
+base = "${ETH_NODE_URI_BASE}"
+
+[etherscan]
+arbitrum = { key = "${ARBITRUM_ETHERSCAN_API_KEY}" }
+gnosis = { key = "${GNOSIS_ETHERSCAN_API_KEY}" , url = "https://api.gnosisscan.io/api"}
+mainnet = { key = "${MAINNET_ETHERSCAN_API_KEY}" }
+optimism = { key = "${OPTIMISM_ETHERSCAN_API_KEY}" }
+polygon = { key = "${POLYGON_ETHERSCAN_API_KEY}" }
+avalanche = { key = "${AVALANCHE_ETHERSCAN_API_KEY}" }
+celo = { key = "${CELO_ETHERSCAN_API_KEY}", url = "https://api.celoscan.io/api" }
+base = { key = "${BASE_ETHERSCAN_API_KEY}", url = "https://api.basescan.org/api" }
+polygonzkevm = { key = "${POLYGONZKEVM_ETHERSCAN_API_KEY}", url = "https://api-zkevm.polygonscan.com/api" }
+bsc = { key = "${BSC_ETHERSCAN_API_KEY}"}
+
+[profile.dev]
+optimizer = true
+via_ir = false
+src = "test"
+gas_reports = ["*"]
+
+[profile.dev.fuzz]
+runs = 2000
+
+[profile.dev.invariant]
+runs = 10
+depth = 1
+fail_on_revert = false
+
+[profile.ci]
+src = "test"
+via_ir = false
+gas_reports = ["*"]
+
+[profile.ci.fuzz]
+runs = 100
+
+[profile.ci.invariant]
+runs = 10
+depth = 30
+fail_on_revert = false
diff --git a/helpers/fork.sh b/helpers/fork.sh
new file mode 100644
index 0000000..b686d16
--- /dev/null
+++ b/helpers/fork.sh
@@ -0,0 +1,73 @@
+#! /bin/bash
+
+function option_to_uri {
+ option=$1
+
+ case $option in
+ "1")
+ echo $ETH_NODE_URI_MAINNET
+ ;;
+ "2")
+ echo $ETH_NODE_URI_ARBITRUM
+ ;;
+ "3")
+ echo $ETH_NODE_URI_POLYGON
+ ;;
+ "4")
+ echo $ETH_NODE_URI_GNOSIS
+ ;;
+ "5")
+ echo $ETH_NODE_URI_AVALANCHE
+ ;;
+ "6")
+ echo $ETH_NODE_URI_BASE
+ ;;
+ "7")
+ echo $ETH_NODE_URI_BSC
+ ;;
+ "8")
+ echo $ETH_NODE_URI_CELO
+ ;;
+ "9")
+ echo $ETH_NODE_URI_POLYGON_ZKEVM
+ ;;
+ "10")
+ echo $ETH_NODE_URI_OPTIMISM
+ ;;
+ *)
+ ;;
+ esac
+}
+
+function main {
+ if [ ! -f .env ]; then
+ echo ".env not found!"
+ exit 1
+ fi
+ source .env
+
+ echo "Which chain would you like to fork ?"
+ echo "- 1: Ethereum Mainnet"
+ echo "- 2: Arbitrum"
+ echo "- 3: Polygon"
+ echo "- 4: Gnosis"
+ echo "- 5: Avalanche"
+ echo "- 6: Base"
+ echo "- 7: Binance Smart Chain"
+ echo "- 8: Celo"
+ echo "- 9: Polygon ZkEvm"
+ echo "- 10: Optimism"
+
+ read option
+
+ uri=$(option_to_uri $option)
+ if [ -z "$uri" ]; then
+ echo "Unknown network"
+ exit 1
+ fi
+
+ echo "Forking $uri"
+ anvil --fork-url $uri
+}
+
+main
diff --git a/lib/LayerZero b/lib/LayerZero
new file mode 160000
index 0000000..48c21c3
--- /dev/null
+++ b/lib/LayerZero
@@ -0,0 +1 @@
+Subproject commit 48c21c3921931798184367fc02d3a8132b041942
diff --git a/lib/chainlink b/lib/chainlink
new file mode 160000
index 0000000..7680667
--- /dev/null
+++ b/lib/chainlink
@@ -0,0 +1 @@
+Subproject commit 7680667f309b13ba1d336bd1680026ff6f6d575f
diff --git a/lib/openzeppelin-contracts b/lib/openzeppelin-contracts
new file mode 160000
index 0000000..ecd2ca2
--- /dev/null
+++ b/lib/openzeppelin-contracts
@@ -0,0 +1 @@
+Subproject commit ecd2ca2cd7cac116f7a37d0e474bbb3d7d5e1c4d
diff --git a/lib/openzeppelin-contracts-upgradeable b/lib/openzeppelin-contracts-upgradeable
new file mode 160000
index 0000000..0a2cb9a
--- /dev/null
+++ b/lib/openzeppelin-contracts-upgradeable
@@ -0,0 +1 @@
+Subproject commit 0a2cb9a445c365870ed7a8ab461b12acf3e27d63
diff --git a/lib/utils b/lib/utils
new file mode 160000
index 0000000..41388c6
--- /dev/null
+++ b/lib/utils
@@ -0,0 +1 @@
+Subproject commit 41388c6ea9c45b05b92155356d6e700802fdb566
diff --git a/package.json b/package.json
new file mode 100644
index 0000000..b7d42e4
--- /dev/null
+++ b/package.json
@@ -0,0 +1,53 @@
+{
+ "name": "ag-tokens",
+ "version": "1.0.0",
+ "description": "",
+ "main": "index.js",
+ "scripts": {
+ "ci:coverage": "forge coverage --report lcov && yarn lcov:clean",
+ "coverage": "FOUNDRY_PROFILE=dev forge coverage --report lcov && yarn lcov:clean && yarn lcov:generate-html",
+ "compile": "forge build",
+ "compile:dev": "FOUNDRY_PROFILE=dev forge build",
+ "deploy": "forge script --skip test --broadcast --verify --slow -vvvv --rpc-url",
+ "deploy:fork": "forge script --skip test --slow --fork-url fork --broadcast -vvvv",
+ "generate": "FOUNDRY_PROFILE=dev forge script scripts/utils/GenerateSelectors.s.sol",
+ "deploy:check": "FOUNDRY_PROFILE=dev forge script --fork-url fork scripts/test/CheckTransmuter.s.sol",
+ "gas": "FOUNDRY_PROFILE=dev yarn test --gas-report",
+ "fork": "bash helpers/fork.sh",
+ "run": "docker run -it --rm -v $(pwd):/app -w /app ghcr.io/foundry-rs/foundry sh",
+ "script:fork": "FOUNDRY_PROFILE=dev forge script --skip test --fork-url fork --broadcast -vvvv",
+ "test:unit": "forge test -vvvv --gas-report --match-path \"test/units/**/*.sol\"",
+ "test:invariant": "forge test -vvv --gas-report --match-path \"test/invariants/**/*.sol\"",
+ "test:fuzz": "forge test -vvv --gas-report --match-path \"test/fuzz/**/*.sol\"",
+ "slither": "chmod +x ./slither.sh && ./slither.sh",
+ "test": "forge test -vvvv",
+ "lcov:clean": "lcov --remove lcov.info -o lcov.info 'test/**' 'scripts/**' 'contracts/transmuter/configs/**' 'contracts/utils/**'",
+ "lcov:generate-html": "genhtml lcov.info --output=coverage",
+ "size": "forge build --skip test --sizes",
+ "size:dev": "FOUNDRY_PROFILE=dev forge build --skip test --sizes",
+ "prettier": "prettier --write '**/*.sol'",
+ "lint": "yarn lint:check --fix",
+ "lint:check": "solhint --max-warnings 20 \"**/*.sol\"",
+ "vanity": "forge script --skip test --slow -vvvv --rpc-url mainnet ./scripts/utils/VanityAddress.s.sol",
+ "verify": "forge verify-contract --num-of-optimizations 1000 --watch --constructor-args 0000000000000000000000000000000000ffe8b47b3e2130213b802212439497000000000000000000000000fda462548ce04282f4b6d6619823a7c64fdc018500000000000000000000000000000000000000000000000000000000000000600000000000000000000000000000000000000000000000000000000000000000 --compiler-version v0.8.19+commit.7dd6d404 0x0022228a2cc5E7eF0274A7Baa600d44da5aB5776 lib/openzeppelin-contracts/contracts/proxy/transparent/TransparentUpgradeableProxy.sol:TransparentUpgradeableProxy --chain"
+ },
+ "repository": {
+ "type": "git",
+ "url": "git+https://github.com/AngleProtocol/AgTokens.git"
+ },
+ "keywords": [],
+ "author": "Angle Core Team",
+ "license": "GPL-3.0",
+ "bugs": {
+ "url": "https://github.com/AngleProtocol/AgTokens/issues"
+ },
+ "devDependencies": {
+ "@angleprotocol/sdk": "0.38.8",
+ "prettier": "^2.0.0",
+ "prettier-plugin-solidity": "^1.1.3",
+ "solhint": "^3.5.1",
+ "solhint-plugin-prettier": "^0.0.5"
+ },
+ "dependencies": {
+ }
+}
diff --git a/remappings.txt b/remappings.txt
new file mode 100644
index 0000000..c0843c1
--- /dev/null
+++ b/remappings.txt
@@ -0,0 +1,9 @@
+ds-test/=lib/forge-std/lib/ds-test/src/
+forge-std/=lib/forge-std/src/
+oz/=lib/openzeppelin-contracts/contracts/
+oz-upgradeable/=lib/openzeppelin-contracts-upgradeable/contracts/
+chainlink/=lib/chainlink/contracts/
+utils/=lib/utils/
+contracts/=contracts/
+chainlink/=lib/chainlink/contracts/
+layer-zero/=lib/LayerZero/contracts/
\ No newline at end of file
diff --git a/script/Counter.s.sol b/script/Counter.s.sol
deleted file mode 100644
index df9ee8b..0000000
--- a/script/Counter.s.sol
+++ /dev/null
@@ -1,12 +0,0 @@
-// SPDX-License-Identifier: UNLICENSED
-pragma solidity ^0.8.13;
-
-import {Script, console} from "forge-std/Script.sol";
-
-contract CounterScript is Script {
- function setUp() public {}
-
- function run() public {
- vm.broadcast();
- }
-}
diff --git a/scripts/BasicScript.s.sol b/scripts/BasicScript.s.sol
new file mode 100644
index 0000000..539bf0c
--- /dev/null
+++ b/scripts/BasicScript.s.sol
@@ -0,0 +1,24 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.12;
+
+import "forge-std/Script.sol";
+import "contracts/mock/MockToken.sol";
+import { console } from "forge-std/console.sol";
+
+contract MyScript is Script {
+ function run() external {
+ vm.startBroadcast();
+
+ MockToken token = new MockToken("Name", "SYM", 18);
+ address _sender = address(uint160(uint256(keccak256(abi.encodePacked("sender")))));
+ address _receiver = address(uint160(uint256(keccak256(abi.encodePacked("receiver")))));
+
+ console.log(address(token));
+
+ // deal(address(token), _sender, 1 ether);
+ // vm.prank(_sender);
+ // token.transfer(_receiver, 1 ether);
+
+ vm.stopBroadcast();
+ }
+}
diff --git a/scripts/utils/Constants.s.sol b/scripts/utils/Constants.s.sol
new file mode 100644
index 0000000..f57822c
--- /dev/null
+++ b/scripts/utils/Constants.s.sol
@@ -0,0 +1,11 @@
+// SPDX-License-Identifier: GPL-3.0
+pragma solidity ^0.8.12;
+
+import "contracts/utils/Constants.sol";
+import "utils/src/Constants.sol";
+
+/*//////////////////////////////////////////////////////////////////////////////////////////////////////////////////
+ MAINNET CONSTANTS
+//////////////////////////////////////////////////////////////////////////////////////////////////////////////////*/
+
+address constant DEPLOYER = 0xfdA462548Ce04282f4B6D6619823a7C64Fdc0185;
diff --git a/scripts/utils/FindInitCode.s.sol b/scripts/utils/FindInitCode.s.sol
new file mode 100644
index 0000000..b0e93b4
--- /dev/null
+++ b/scripts/utils/FindInitCode.s.sol
@@ -0,0 +1,35 @@
+// SPDX-License-Identifier: GPL-3.0
+pragma solidity ^0.8.12;
+
+import { console } from "forge-std/console.sol";
+import "forge-std/Script.sol";
+import { StdAssertions } from "forge-std/Test.sol";
+import { stdJson } from "forge-std/StdJson.sol";
+import "stringutils/strings.sol";
+import "./Constants.s.sol";
+
+import { TransparentUpgradeableProxy } from "oz/proxy/transparent/TransparentUpgradeableProxy.sol";
+
+import { ImmutableCreate2Factory } from "contracts/interfaces/external/create2/ImmutableCreate2Factory.sol";
+
+/// @dev Script to run to find the init code of a contract to get a vanity address from it
+contract FindInitCode is Script, StdAssertions {
+ using stdJson for string;
+ using strings for *;
+
+ function run() external {
+ // To maintain chain consistency, we do as if we deployed with the deployer as a proxyAdmin before transferring
+ // to another address
+ // We use a contract that is widely deployed across many chains as an implementation to make it resilient
+ // to possible implementation changes
+ bytes memory emptyData;
+ bytes memory initCode = abi.encodePacked(
+ type(TransparentUpgradeableProxy).creationCode,
+ abi.encode(IMMUTABLE_CREATE2_FACTORY_ADDRESS, DEPLOYER, emptyData)
+ );
+ console.log("Proxy bytecode");
+ console.logBytes(initCode);
+ console.logBytes(abi.encode(IMMUTABLE_CREATE2_FACTORY_ADDRESS, DEPLOYER, emptyData));
+ console.log("");
+ }
+}
diff --git a/scripts/utils/VanityAddress.s.sol b/scripts/utils/VanityAddress.s.sol
new file mode 100644
index 0000000..6608a0f
--- /dev/null
+++ b/scripts/utils/VanityAddress.s.sol
@@ -0,0 +1,38 @@
+// SPDX-License-Identifier: GPL-3.0
+
+pragma solidity ^0.8.12;
+
+import "stringutils/strings.sol";
+import "forge-std/Script.sol";
+import { StdAssertions } from "forge-std/Test.sol";
+import { stdJson } from "forge-std/StdJson.sol";
+import { console } from "forge-std/console.sol";
+import { VanityAddress } from "utils/src/VanityAddress.sol";
+import "./Constants.s.sol";
+
+contract VanityAddressScript is Script, VanityAddress {
+ using stdJson for string;
+
+ string constant JSON_VANITY_PATH = "./scripts/vanity.json";
+
+ using stdJson for string;
+
+ function run() external {
+ // Deploy diamond
+ bytes
+ memory initCode = hex"60406080815262000f5f80380380620000188162000364565b9283398101906060818303126200035f576200003481620003a0565b9160209262000045848401620003a0565b8584015190936001600160401b0391908282116200035f57019280601f850112156200035f57835193620000836200007d86620003b5565b62000364565b94808652878601928882840101116200035f578288620000a49301620003d1565b823b1562000305577f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc80546001600160a01b03199081166001600160a01b0386811691821790935590959194600093909290917fbc7cd75a20ee27fd9adebab32041f755214dbc6bffa90cc0225b39da2e5c2d3b8580a2805115801590620002fd575b620001f5575b50505050507fb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103937f7e644d79422f17c01e4894b5f4f588d331ebfa28653d42ae832dc59e38c9798f86865493815196818616885216958684820152a18315620001a357501617905551610b0a9081620004558239f35b60849086519062461bcd60e51b82526004820152602660248201527f455243313936373a206e65772061646d696e20697320746865207a65726f206160448201526564647265737360d01b6064820152fd5b8951946060860190811186821017620002e9578a52602785527f416464726573733a206c6f772d6c6576656c2064656c65676174652063616c6c89860152660819985a5b195960ca1b8a860152823b156200029657928092819262000280969551915af43d156200028c573d620002706200007d82620003b5565b9081528092893d92013e620003f6565b5038808080806200012d565b60609150620003f6565b895162461bcd60e51b8152600481018a9052602660248201527f416464726573733a2064656c65676174652063616c6c20746f206e6f6e2d636f6044820152651b9d1c9858dd60d21b6064820152608490fd5b634e487b7160e01b85526041600452602485fd5b508362000127565b865162461bcd60e51b815260048101879052602d60248201527f455243313936373a206e657720696d706c656d656e746174696f6e206973206e60448201526c1bdd08184818dbdb9d1c9858dd609a1b6064820152608490fd5b600080fd5b6040519190601f01601f191682016001600160401b038111838210176200038a57604052565b634e487b7160e01b600052604160045260246000fd5b51906001600160a01b03821682036200035f57565b6001600160401b0381116200038a57601f01601f191660200190565b60005b838110620003e55750506000910152565b8181015183820152602001620003d4565b9091901562000403575090565b815115620004145750805190602001fd5b6044604051809262461bcd60e51b825260206004830152620004468151809281602486015260208686019101620003d1565b601f01601f19168101030190fdfe6080604052600436101561002c575b361561001f575b61001d6104dd565b005b6100276104dd565b610015565b6000803560e01c9081633659cfe614610093575080634f1ef2861461008a5780635c60da1b146100815780638f283970146100785763f851a4400361000e57610073610455565b61000e565b506100736102f0565b5061007361023b565b50610073610157565b3461012c5760207ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc36011261012c576100ca61012f565b73ffffffffffffffffffffffffffffffffffffffff7fb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103541633146000146101235761012090610117610639565b908382526106f3565b80f35b506101206104dd565b80fd5b6004359073ffffffffffffffffffffffffffffffffffffffff8216820361015257565b600080fd5b5060407ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101525761018a61012f565b60243567ffffffffffffffff9182821161015257366023830112156101525781600401359283116101525736602484840101116101525773ffffffffffffffffffffffffffffffffffffffff7fb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d61035416331460001461023057600060208480602461021e61021961001d996106aa565b610666565b96828852018387013784010152610833565b50505061001d6104dd565b50346101525760007ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc360112610152576020600073ffffffffffffffffffffffffffffffffffffffff90817fb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103541633146000146102e25750807f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc5416905b60405191168152f35b906102eb6104dd565b6102d9565b50346101525760207ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101525761032861012f565b73ffffffffffffffffffffffffffffffffffffffff907fb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d610391808354163314600014610230577f7e644d79422f17c01e4894b5f4f588d331ebfa28653d42ae832dc59e38c9798f604084549281519481851686521693846020820152a181156103d1577fffffffffffffffffffffffff000000000000000000000000000000000000000016179055005b60846040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602660248201527f455243313936373a206e65772061646d696e20697320746865207a65726f206160448201527f64647265737300000000000000000000000000000000000000000000000000006064820152fd5b50346101525760007ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc360112610152576020600073ffffffffffffffffffffffffffffffffffffffff7fb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d61038181541633146000146104d85754604051911681529050f35b506102eb5b5073ffffffffffffffffffffffffffffffffffffffff807fb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d61035416331461055f577f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc54166000808092368280378136915af43d82803e1561055b573d90f35b3d90fd5b60a46040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152604260248201527f5472616e73706172656e745570677261646561626c6550726f78793a2061646d60448201527f696e2063616e6e6f742066616c6c6261636b20746f2070726f7879207461726760648201527f65740000000000000000000000000000000000000000000000000000000000006084820152fd5b507f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b604051906020820182811067ffffffffffffffff82111761065957604052565b610661610609565b604052565b907fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0601f604051930116820182811067ffffffffffffffff82111761065957604052565b7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0601f60209267ffffffffffffffff81116106e6575b01160190565b6106ee610609565b6106e0565b803b156107af5773ffffffffffffffffffffffffffffffffffffffff81167f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc817fffffffffffffffffffffffff00000000000000000000000000000000000000008254161790557fbc7cd75a20ee27fd9adebab32041f755214dbc6bffa90cc0225b39da2e5c2d3b600080a28151158015906107a7575b610792575050565b6107a49161079e6108d9565b91610957565b50565b50600061078a565b60846040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602d60248201527f455243313936373a206e657720696d706c656d656e746174696f6e206973206e60448201527f6f74206120636f6e7472616374000000000000000000000000000000000000006064820152fd5b803b156107af5773ffffffffffffffffffffffffffffffffffffffff81167f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc817fffffffffffffffffffffffff00000000000000000000000000000000000000008254161790557fbc7cd75a20ee27fd9adebab32041f755214dbc6bffa90cc0225b39da2e5c2d3b600080a28151158015906108d157610792575050565b50600161078a565b604051906060820182811067ffffffffffffffff82111761094a575b604052602782527f206661696c6564000000000000000000000000000000000000000000000000006040837f416464726573733a206c6f772d6c6576656c2064656c65676174652063616c6c60208201520152565b610952610609565b6108f5565b9190823b156109a0576000816109959460208394519201905af43d15610998573d90610985610219836106aa565b9182523d6000602084013e610a24565b90565b606090610a24565b60846040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602660248201527f416464726573733a2064656c65676174652063616c6c20746f206e6f6e2d636f60448201527f6e747261637400000000000000000000000000000000000000000000000000006064820152fd5b90919015610a30575090565b815115610a405750805190602001fd5b604051907f08c379a000000000000000000000000000000000000000000000000000000000825281602080600483015282519283602484015260005b848110610abd575050507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0601f836000604480968601015201168101030190fd5b818101830151868201604401528593508201610a7c56fea26469706673582212206eca7257d81e920296a8c93c0ac0d93d9bb0927ce3e4cefa34ff129bc0cdbceb64736f6c634300081100330000000000000000000000000000000000ffe8b47b3e2130213b802212439497000000000000000000000000fda462548ce04282f4b6d6619823a7c64fdc018500000000000000000000000000000000000000000000000000000000000000600000000000000000000000000000000000000000000000000000000000000000";
+ string memory json = vm.readFile(JSON_VANITY_PATH);
+ uint256 initInt = json.readUint(string.concat("$.", "init"));
+ uint256 iterations = 3000000;
+
+ (address computed, uint256 found) = minePrefix(initCode, DEPLOYER, 0x000020, initInt, iterations);
+ console.log("Computed: ", computed);
+ console.log("Found: ", found);
+
+ // write result to json vanity path
+ json = "";
+ vm.serializeUint(json, "init", found);
+ string memory finalJson = vm.serializeAddress(json, "salt", computed);
+ vm.writeFile(JSON_VANITY_PATH, finalJson);
+ }
+}
diff --git a/scripts/vanity.json b/scripts/vanity.json
new file mode 100644
index 0000000..da1671c
--- /dev/null
+++ b/scripts/vanity.json
@@ -0,0 +1,4 @@
+{
+ "init": 8223543,
+ "salt": "0xfda462548ce04282f4b6d6619823a7c64fdc01850000000000000000007d7b37"
+}
diff --git a/slither.config.json b/slither.config.json
new file mode 100644
index 0000000..7180139
--- /dev/null
+++ b/slither.config.json
@@ -0,0 +1,10 @@
+{
+ "detectors_to_exclude": "naming-convention,solc-version",
+ "filter_paths": "(lib|test|external|node_modules|scripts)",
+ "solc_remaps": [
+ "ds-test/=lib/ds-test/src/",
+ "forge-std/=lib/forge-std/src/",
+ "oz/=lib/openzeppelin-contracts/contracts/",
+ "oz-upgradeable/=lib/openzeppelin-contracts-upgradeable/contracts/"
+ ]
+}
diff --git a/slither.sh b/slither.sh
new file mode 100755
index 0000000..19869b6
--- /dev/null
+++ b/slither.sh
@@ -0,0 +1,42 @@
+# Use git grep to find all files containing '0.8.19'
+files=$(git grep -l '0.8.19')
+
+# Loop over files and replace '0.8.19' with '0.8.18'
+for file in $files
+do
+ sed -i '' 's/0.8.19/0.8.18/g' "$file"
+done
+
+# # # Ignore test files
+# find "test/" -type f -name "*.sol" | while read file; do
+# # Replace '.sol' extension with '.txt'
+# mv "$file" "${file%.sol}.txt"
+# done
+
+# mv "test/mock/MockLib.txt" "test/mock/MockLib.sol"
+
+# Function to handle error
+reset() {
+ # find "test/" -type f -name "*.txt" | while read file; do
+ # # Replace '.sol' extension with '.txt'
+ # mv "$file" "${file%.txt}.sol"
+ # done
+
+ files=$(git grep -l '0.8.18')
+ for file in $files
+ do
+ sed -i '' 's/0.8.18/0.8.19/g' "$file"
+ done
+}
+
+trap 'reset' ERR
+
+# pip3 install slither-analyzer
+# pip3 install solc-select
+# solc-select install 0.8.18
+# solc-select use 0.8.18
+FOUNDRY_PROFILE=dev forge build --skip test
+
+slither test/mock/Slither.sol --show-ignored-findings --foundry-ignore-compile
+
+reset
\ No newline at end of file
diff --git a/src/Counter.sol b/src/Counter.sol
deleted file mode 100644
index aded799..0000000
--- a/src/Counter.sol
+++ /dev/null
@@ -1,14 +0,0 @@
-// SPDX-License-Identifier: UNLICENSED
-pragma solidity ^0.8.13;
-
-contract Counter {
- uint256 public number;
-
- function setNumber(uint256 newNumber) public {
- number = newNumber;
- }
-
- function increment() public {
- number++;
- }
-}
diff --git a/test/Counter.t.sol b/test/Counter.t.sol
deleted file mode 100644
index 54b724f..0000000
--- a/test/Counter.t.sol
+++ /dev/null
@@ -1,24 +0,0 @@
-// SPDX-License-Identifier: UNLICENSED
-pragma solidity ^0.8.13;
-
-import {Test, console} from "forge-std/Test.sol";
-import {Counter} from "../src/Counter.sol";
-
-contract CounterTest is Test {
- Counter public counter;
-
- function setUp() public {
- counter = new Counter();
- counter.setNumber(0);
- }
-
- function test_Increment() public {
- counter.increment();
- assertEq(counter.number(), 1);
- }
-
- function testFuzz_SetNumber(uint256 x) public {
- counter.setNumber(x);
- assertEq(counter.number(), x);
- }
-}
diff --git a/test/fuzz/.gitkeep b/test/fuzz/.gitkeep
new file mode 100644
index 0000000..e69de29
diff --git a/test/invariants/.gitkeep b/test/invariants/.gitkeep
new file mode 100644
index 0000000..e69de29
diff --git a/test/units/BaseTest.t.sol b/test/units/BaseTest.t.sol
new file mode 100644
index 0000000..00e51ec
--- /dev/null
+++ b/test/units/BaseTest.t.sol
@@ -0,0 +1,46 @@
+// SPDX-License-Identifier: UNLICENSED
+pragma solidity ^0.8.12;
+
+import { Test, stdMath, StdStorage, stdStorage } from "forge-std/Test.sol";
+import "../../contracts/external/ProxyAdmin.sol";
+import "../../contracts/external/TransparentUpgradeableProxy.sol";
+import "../../contracts/mock/MockCoreBorrow.sol";
+import { Math } from "oz/utils/math/Math.sol";
+import { console } from "forge-std/console.sol";
+import { CommonUtils } from "utils/src/CommonUtils.sol";
+
+contract BaseTest is Test, CommonUtils {
+ ProxyAdmin public proxyAdmin;
+ MockCoreBorrow public coreBorrow;
+
+ address internal constant _GOVERNOR = 0xdC4e6DFe07EFCa50a197DF15D9200883eF4Eb1c8;
+ address internal constant _GUARDIAN = 0x0C2553e4B9dFA9f83b1A6D3EAB96c4bAaB42d430;
+ address internal constant _KEEPER = address(uint160(uint256(keccak256(abi.encodePacked("_keeper")))));
+ address internal constant _ANGLE = 0x31429d1856aD1377A8A0079410B297e1a9e214c2;
+ address internal constant _GOVERNOR_POLYGON = 0xdA2D2f638D6fcbE306236583845e5822554c02EA;
+
+ address internal constant _alice = address(uint160(uint256(keccak256(abi.encodePacked("_alice")))));
+ address internal constant _bob = address(uint160(uint256(keccak256(abi.encodePacked("_bob")))));
+ address internal constant _charlie = address(uint160(uint256(keccak256(abi.encodePacked("_charlie")))));
+ address internal constant _dylan = address(uint160(uint256(keccak256(abi.encodePacked("_dylan")))));
+
+ uint256 internal _ethereum;
+ uint256 internal _polygon;
+
+ uint256 public constant BASE_PARAMS = 10 ** 9;
+ uint256 public constant BASE_STAKER = 10 ** 36;
+ uint256 public constant BASE_18 = 10 ** 18;
+
+ function setUp() public virtual {
+ proxyAdmin = new ProxyAdmin();
+ coreBorrow = new MockCoreBorrow();
+ coreBorrow.toggleGuardian(_GUARDIAN);
+ coreBorrow.toggleGovernor(_GOVERNOR);
+ vm.label(_GOVERNOR, "Governor");
+ vm.label(_GUARDIAN, "Guardian");
+ vm.label(_alice, "Alice");
+ vm.label(_bob, "Bob");
+ vm.label(_charlie, "Charlie");
+ vm.label(_dylan, "Dylan");
+ }
+}
diff --git a/test/units/agToken/AgToken.t.sol b/test/units/agToken/AgToken.t.sol
new file mode 100644
index 0000000..3e293df
--- /dev/null
+++ b/test/units/agToken/AgToken.t.sol
@@ -0,0 +1,232 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import { stdStorage, StdStorage } from "forge-std/Test.sol";
+import "../BaseTest.t.sol";
+import { IAgToken, AgToken } from "contracts/agToken/AgToken.sol";
+import { MockTreasury } from "contracts/mock/MockTreasury.sol";
+
+contract AgTokenTest is BaseTest {
+ using stdStorage for StdStorage;
+
+ address internal _hacker = address(uint160(uint256(keccak256(abi.encodePacked("hacker")))));
+
+ AgToken internal _agToken;
+ MockTreasury internal _treasury;
+ AgToken internal _agTokenImplem;
+
+ string constant _NAME = "Angle stablecoin gold";
+ string constant _SYMBOL = "agGold";
+
+ function setUp() public override {
+ super.setUp();
+
+ _agTokenImplem = new AgToken();
+ _agToken = AgToken(_deployUpgradeable(address(proxyAdmin), address(_agTokenImplem), ""));
+
+ _treasury = new MockTreasury(
+ IAgToken(address(_agToken)),
+ _GOVERNOR,
+ _GUARDIAN,
+ address(0),
+ address(0),
+ address(0)
+ );
+
+ _agToken.initialize(_NAME, _SYMBOL, address(_treasury));
+
+ vm.startPrank(_GOVERNOR);
+ _treasury.setStablecoin(_agToken);
+ _treasury.addMinter(_agToken, _alice);
+ vm.stopPrank();
+ }
+
+ // ================================= INITIALIZE ================================
+
+ function test_initialize_Constructor() public {
+ assertEq(_agToken.name(), _NAME);
+ assertEq(_agToken.symbol(), _SYMBOL);
+ assertEq(_agToken.decimals(), 18);
+ assertEq(_agToken.treasury(), address(_treasury));
+ }
+
+ function test_initialize_AlreadyInitalizeFail() public {
+ string memory name2 = "Angle stablecoin XXX";
+ string memory symbol2 = "agXXX";
+ vm.expectRevert();
+ _agToken.initialize(name2, symbol2, _alice);
+ }
+
+ function test_initialize_WrongTreasuryAddress() public {
+ string memory name2 = "Angle stablecoin XXX";
+ string memory symbol2 = "agXXX";
+ MockTreasury treasury = new MockTreasury(
+ IAgToken(address(0)),
+ address(0),
+ address(0),
+ address(0),
+ address(0),
+ address(0)
+ );
+ bytes memory emptyData;
+ _agToken = AgToken(_deployUpgradeable(address(proxyAdmin), address(_agTokenImplem), emptyData));
+
+ vm.expectRevert(AgToken.InvalidTreasury.selector);
+ _agToken.initialize(name2, symbol2, address(treasury));
+ }
+
+ // ================================= MINT ================================
+
+ function test_mint_WrongSender() public {
+ vm.expectRevert(AgToken.NotMinter.selector);
+ vm.prank(_bob);
+ _agToken.mint(_hacker, 1e18);
+ }
+
+ function test_mint_ZeroAddress() public {
+ vm.expectRevert("ERC20: mint to the zero address");
+ vm.prank(_alice);
+ _agToken.mint(address(0), 1e18);
+ }
+
+ function test_mint_Normal() public {
+ uint256 amount = 1e18;
+ vm.prank(_alice);
+ _agToken.mint(_alice, amount);
+ assertEq(_agToken.balanceOf(_alice), amount);
+ assertEq(_agToken.totalSupply(), amount);
+ }
+
+ // ================================= BurnStablecoin ================================
+
+ function test_burnStablecoin_BurnStablecoin() public {
+ uint256 amount = 1e18;
+ vm.startPrank(_alice);
+ _agToken.mint(_alice, amount);
+ _agToken.burnStablecoin(amount);
+ vm.stopPrank();
+ assertEq(_agToken.balanceOf(_alice), 0);
+ assertEq(_agToken.totalSupply(), 0);
+ }
+
+ function test_burnStablecoin_BurnGreaterThanBalance() public {
+ uint256 amount = 1e18;
+ vm.startPrank(_alice);
+ _agToken.mint(_alice, amount);
+ vm.expectRevert("ERC20: burn amount exceeds balance");
+ _agToken.burnStablecoin(amount + 1);
+ vm.stopPrank();
+ }
+
+ // ================================= BurnSelf ================================
+
+ function test_burnSelf_NotMinter() public {
+ vm.expectRevert(AgToken.NotMinter.selector);
+ _agToken.burnSelf(1e18, _alice);
+ }
+
+ function test_burnSelf_Normal() public {
+ uint256 amount = 1e18;
+ vm.startPrank(_alice);
+ _agToken.mint(_alice, amount);
+ _agToken.burnSelf(amount, _alice);
+ vm.stopPrank();
+ assertEq(_agToken.balanceOf(_alice), 0);
+ assertEq(_agToken.totalSupply(), 0);
+ }
+
+ // ================================= BurnFrom ================================
+
+ function test_burnFrom_NotMinter() public {
+ vm.prank(_bob);
+ vm.expectRevert(AgToken.NotMinter.selector);
+ _agToken.burnFrom(1e18, _bob, _alice);
+ }
+
+ function test_burnFrom_NoApproval() public {
+ vm.prank(_alice);
+ vm.expectRevert(AgToken.BurnAmountExceedsAllowance.selector);
+ _agToken.burnFrom(1e18, _alice, _bob);
+ }
+
+ function test_burnFrom_WithApproval() public {
+ uint256 amount = 1e18;
+ vm.startPrank(_alice);
+ _agToken.mint(_alice, amount);
+ _agToken.approve(_bob, amount * 2);
+ vm.stopPrank();
+ assertEq(_agToken.allowance(_alice, _bob), amount * 2);
+
+ vm.prank(_alice);
+ _agToken.burnFrom(amount, _alice, _bob);
+
+ assertEq(_agToken.balanceOf(_alice), 0);
+ assertEq(_agToken.totalSupply(), 0);
+ assertEq(_agToken.allowance(_alice, _bob), amount);
+ }
+
+ function test_burnFrom_WithoutApprovalBurnerIsSender() public {
+ uint256 amount = 1e18;
+ vm.startPrank(_alice);
+ _agToken.mint(_alice, amount);
+ _agToken.burnFrom(amount, _alice, _alice);
+ vm.stopPrank();
+
+ assertEq(_agToken.balanceOf(_alice), 0);
+ assertEq(_agToken.totalSupply(), 0);
+ }
+
+ // ================================= AddMinter ================================
+
+ function test_addMinter_NotTreasury() public {
+ vm.expectRevert(AgToken.NotTreasury.selector);
+ _agToken.addMinter(_alice);
+ }
+
+ function test_addMinter_MinterToggled() public {
+ vm.prank(_GOVERNOR);
+ _treasury.addMinter(_agToken, _alice);
+ assert(_agToken.isMinter(_alice));
+ }
+
+ // ================================= RemoveMinter ================================
+
+ function test_removeMinter_NotTreasury() public {
+ vm.expectRevert(AgToken.InvalidSender.selector);
+ _agToken.removeMinter(_alice);
+ }
+
+ function test_removeMinter_Normal() public {
+ vm.prank(_GOVERNOR);
+ _treasury.addMinter(_agToken, _bob);
+ assertTrue(_agToken.isMinter(_bob));
+
+ vm.prank(_GOVERNOR);
+ _treasury.removeMinter(_agToken, _bob);
+ assertFalse(_agToken.isMinter(_bob));
+ }
+
+ // ================================= SetTreasury ================================
+
+ function test_setTreasury_NotTreasury() public {
+ vm.expectRevert(AgToken.NotTreasury.selector);
+ vm.prank(_alice);
+ _agToken.setTreasury(_alice);
+ }
+
+ function test_setTreasury_Normal() public {
+ vm.prank(_GOVERNOR);
+ _treasury.setTreasury(address(_agToken), _alice);
+
+ assertEq(_agToken.treasury(), _alice);
+ }
+
+ function test_setTreasury_NormalReset() public {
+ vm.prank(_GOVERNOR);
+ _treasury.setTreasury(address(_agToken), _alice);
+ vm.prank(_alice);
+ _agToken.setTreasury(address(_treasury));
+
+ assertEq(_agToken.treasury(), address(_treasury));
+ }
+}
diff --git a/test/units/agToken/AgTokenSideChainMultiBridge.t.sol b/test/units/agToken/AgTokenSideChainMultiBridge.t.sol
new file mode 100644
index 0000000..63d78c0
--- /dev/null
+++ b/test/units/agToken/AgTokenSideChainMultiBridge.t.sol
@@ -0,0 +1,671 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import "../BaseTest.t.sol";
+import { MockTreasury } from "contracts/mock/MockTreasury.sol";
+import { AgTokenSideChainMultiBridge } from "contracts/agToken/AgTokenSideChainMultiBridge.sol";
+import { MockToken } from "contracts/mock/MockToken.sol";
+
+contract AgTokenSideChainMultiBridgeTest is BaseTest {
+ AgTokenSideChainMultiBridge internal _agToken;
+ AgTokenSideChainMultiBridge internal _agTokenImplem;
+ MockTreasury internal _treasury;
+ MockToken internal _bridgeToken;
+
+ function setUp() public override {
+ super.setUp();
+
+ _agTokenImplem = new AgTokenSideChainMultiBridge();
+ _agToken = AgTokenSideChainMultiBridge(_deployUpgradeable(address(proxyAdmin), address(_agTokenImplem), ""));
+
+ _treasury = new MockTreasury(
+ IAgToken(address(_agToken)),
+ _GOVERNOR,
+ _GUARDIAN,
+ address(0),
+ address(0),
+ address(0)
+ );
+
+ _agToken.initialize("agEUR", "agEUR", address(_treasury));
+
+ _treasury.addMinter(_agToken, _alice);
+ _treasury.addMinter(_agToken, _GOVERNOR);
+ vm.prank(_alice);
+ _agToken.mint(_alice, 1e18);
+
+ _bridgeToken = new MockToken("any-agEUR", "any-agEUR", 18);
+ vm.prank(_GOVERNOR);
+ _agToken.addBridgeToken(address(_bridgeToken), 10e18, 1e18, 5e8, false);
+ }
+
+ // ================================= INITIALIZE ================================
+
+ function test_initialize_Constructor() public view {
+ (uint256 limit, uint256 hourlyLimit, uint64 fee, bool allowed, bool paused) = _agToken.bridges(
+ address(_bridgeToken)
+ );
+ assertEq(limit, 10e18);
+ assertEq(hourlyLimit, 1e18);
+ assertEq(fee, 5e8);
+ assertTrue(allowed);
+ assertFalse(paused);
+ assertEq(_agToken.bridgeTokensList(0), address(_bridgeToken));
+ assertEq(_agToken.allBridgeTokens()[0], address(_bridgeToken));
+ }
+
+ function test_initialize_NotGovernor() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.NotGovernor.selector);
+ vm.prank(_bob);
+ _agToken.addBridgeToken(address(_bridgeToken), 10e18, 1e18, 5e8, false);
+ }
+
+ function test_initialize_TooHighParameterValue() public {
+ MockToken bridgeToken2 = new MockToken("any-agEUR", "any-agEUR", 18);
+ vm.expectRevert(AgTokenSideChainMultiBridge.TooHighParameterValue.selector);
+ vm.prank(_GOVERNOR);
+ _agToken.addBridgeToken(address(bridgeToken2), 1e18, 1e17, 2e9, false);
+ }
+
+ function test_initialize_ZeroAddress() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.InvalidToken.selector);
+ vm.prank(_GOVERNOR);
+ _agToken.addBridgeToken(address(0), 1e18, 1e17, 5e8, false);
+ }
+
+ function test_initialize_AlreadyAdded() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.InvalidToken.selector);
+ vm.prank(_GOVERNOR);
+ _agToken.addBridgeToken(address(_bridgeToken), 1e18, 1e17, 5e8, false);
+ }
+
+ function test_initialize_SecondTokenAdded() public {
+ MockToken bridgeToken2 = new MockToken("synapse-agEUR", "synapse-agEUR", 18);
+
+ vm.prank(_GOVERNOR);
+ _agToken.addBridgeToken(address(bridgeToken2), 1e18, 1e17, 5e8, false);
+ (uint256 limit, uint256 hourlyLimit, uint64 fee, bool allowed, bool paused) = _agToken.bridges(
+ address(bridgeToken2)
+ );
+ assertEq(limit, 1e18);
+ assertEq(hourlyLimit, 1e17);
+ assertEq(fee, 5e8);
+ assertTrue(allowed);
+ assertFalse(paused);
+ assertEq(_agToken.bridgeTokensList(1), address(bridgeToken2));
+ assertEq(_agToken.allBridgeTokens()[1], address(bridgeToken2));
+ }
+
+ // ================================= AddBridgeToken ================================
+
+ function test_addBridgeToken_Normal() public view {
+ (uint256 limit, uint256 hourlyLimit, uint64 fee, bool allowed, bool paused) = _agToken.bridges(
+ address(_bridgeToken)
+ );
+
+ assertEq(limit, 10e18);
+ assertEq(hourlyLimit, 1e18);
+ assertEq(fee, 5e8);
+ assertTrue(allowed);
+ assertFalse(paused);
+ assertEq(_agToken.bridgeTokensList(0), address(_bridgeToken));
+ assertEq(_agToken.allBridgeTokens()[0], address(_bridgeToken));
+ }
+
+ function test_addBridgeToken_NotGovernor() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.NotGovernor.selector);
+ vm.prank(_alice);
+ _agToken.addBridgeToken(address(_bridgeToken), 10e18, 1e18, 5e8, false);
+ }
+
+ function test_addBridgeToken_TooHighParameterValue() public {
+ MockToken bridgeToken2 = new MockToken("any-agEUR", "any-agEUR", 18);
+ vm.expectRevert(AgTokenSideChainMultiBridge.TooHighParameterValue.selector);
+ vm.prank(_GOVERNOR);
+ _agToken.addBridgeToken(address(bridgeToken2), 1e18, 1e17, 2e9, false);
+ }
+
+ function test_addBridgeToken_ZeroAddress() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.InvalidToken.selector);
+ vm.prank(_GOVERNOR);
+ _agToken.addBridgeToken(address(0), 1e18, 1e17, 5e8, false);
+ }
+
+ function test_addBridgeToken_AlreadyAdded() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.InvalidToken.selector);
+ vm.prank(_GOVERNOR);
+ _agToken.addBridgeToken(address(_bridgeToken), 1e18, 1e17, 5e8, false);
+ }
+
+ function test_addBridgeToken_SecondTokenAdded() public {
+ MockToken bridgeToken2 = new MockToken("synapse-agEUR", "synapse-agEUR", 18);
+
+ vm.prank(_GOVERNOR);
+ _agToken.addBridgeToken(address(bridgeToken2), 1e18, 1e17, 5e8, false);
+ (uint256 limit, uint256 hourlyLimit, uint64 fee, bool allowed, bool paused) = _agToken.bridges(
+ address(bridgeToken2)
+ );
+ assertEq(limit, 1e18);
+ assertEq(hourlyLimit, 1e17);
+ assertEq(fee, 5e8);
+ assertTrue(allowed);
+ assertFalse(paused);
+ assertEq(_agToken.bridgeTokensList(1), address(bridgeToken2));
+ assertEq(_agToken.allBridgeTokens()[1], address(bridgeToken2));
+ }
+
+ // ================================= RemoveBridgeToken ================================
+
+ function test_removeBridgeToken_NotGovernor() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.NotGovernor.selector);
+ vm.prank(_alice);
+ _agToken.removeBridgeToken(address(_bridgeToken));
+ }
+
+ function test_removeBridgeToken_NonNullBalance() public {
+ _bridgeToken.mint(address(_agToken), 1e18);
+ vm.expectRevert(AgTokenSideChainMultiBridge.AssetStillControlledInReserves.selector);
+ vm.prank(_GOVERNOR);
+ _agToken.removeBridgeToken(address(_bridgeToken));
+ }
+
+ function test_removeBridgeToken_NormalOneToken() public {
+ vm.prank(_GOVERNOR);
+ _agToken.removeBridgeToken(address(_bridgeToken));
+
+ (uint256 limit, uint256 hourlyLimit, uint64 fee, bool allowed, bool paused) = _agToken.bridges(
+ address(_bridgeToken)
+ );
+ assertEq(limit, 0);
+ assertEq(hourlyLimit, 0);
+ assertEq(fee, 0);
+ assertFalse(allowed);
+ assertFalse(paused);
+ }
+
+ function test_removeBridgeToken_TwoTokensAndFirstIsRemoved() public {
+ MockToken bridgeToken2 = new MockToken("synapse-agEUR", "synapse-agEUR", 18);
+ vm.startPrank(_GOVERNOR);
+ _agToken.addBridgeToken(address(bridgeToken2), 100e18, 10e18, 3e7, true);
+ _agToken.removeBridgeToken(address(bridgeToken2));
+ vm.stopPrank();
+
+ (uint256 limit, uint256 hourlyLimit, uint64 fee, bool allowed, bool paused) = _agToken.bridges(
+ address(bridgeToken2)
+ );
+ assertEq(limit, 0);
+ assertEq(hourlyLimit, 0);
+ assertEq(fee, 0);
+ assertFalse(allowed);
+ assertFalse(paused);
+ assertEq(_agToken.bridgeTokensList(0), address(_bridgeToken));
+ assertEq(_agToken.allBridgeTokens()[0], address(_bridgeToken));
+ }
+
+ // ================================= RecoverERC20 ================================
+
+ function test_recoverERC20_NotGovernor() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.NotGovernor.selector);
+ vm.prank(_alice);
+ _agToken.recoverERC20(address(_bridgeToken), _bob, 1e18);
+ }
+
+ function test_recoverERC20_InvalidBalance() public {
+ vm.expectRevert();
+ vm.prank(_GOVERNOR);
+ _agToken.recoverERC20(address(_bridgeToken), _bob, 1e19);
+ }
+
+ function test_recoverERC20_Normal() public {
+ _bridgeToken.mint(address(_agToken), 1e18);
+ assertEq(_bridgeToken.balanceOf(address(_agToken)), 1e18);
+ vm.prank(_GOVERNOR);
+ _agToken.recoverERC20(address(_bridgeToken), _bob, 1e18);
+ assertEq(_bridgeToken.balanceOf(address(_agToken)), 0);
+ }
+
+ // ================================= SetLimit ================================
+
+ function test_setLimit_NotGovernorOrGuardian() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.NotGovernorOrGuardian.selector);
+ vm.prank(_alice);
+ _agToken.setLimit(address(_bridgeToken), 1e18);
+ }
+
+ function test_setLimit_InvalidToken() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.InvalidToken.selector);
+ vm.prank(_GOVERNOR);
+ _agToken.setLimit(_alice, 1e18);
+ }
+
+ function test_setLimit_Normal() public {
+ vm.prank(_GOVERNOR);
+ _agToken.setLimit(address(_bridgeToken), 1000e18);
+ (uint256 limit, , , , ) = _agToken.bridges(address(_bridgeToken));
+ assertEq(limit, 1000e18);
+ }
+
+ // ================================= SetHourlyLimit ================================
+
+ function test_setHourlyLimit_NotGovernorOrGuardian() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.NotGovernorOrGuardian.selector);
+ vm.prank(_alice);
+ _agToken.setHourlyLimit(address(_bridgeToken), 1e18);
+ }
+
+ function test_setHourlyLimit_InvalidToken() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.InvalidToken.selector);
+ vm.prank(_GOVERNOR);
+ _agToken.setHourlyLimit(_alice, 1e18);
+ }
+
+ function test_setHourlyLimit_Normal() public {
+ vm.prank(_GOVERNOR);
+ _agToken.setHourlyLimit(address(_bridgeToken), 1000e18);
+ (, uint256 hourlyLimit, , , ) = _agToken.bridges(address(_bridgeToken));
+ assertEq(hourlyLimit, 1000e18);
+ }
+
+ // ================================= SetChainTotalHourlyLimit ================================
+
+ function test_setChainTotalHourlyLimit_NotGovernorOrGuardian() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.NotGovernorOrGuardian.selector);
+ vm.prank(_alice);
+ _agToken.setChainTotalHourlyLimit(1e18);
+ }
+
+ function test_setChainTotalHourlyLimit_Normal() public {
+ vm.prank(_GOVERNOR);
+ _agToken.setChainTotalHourlyLimit(1000e18);
+ assertEq(_agToken.chainTotalHourlyLimit(), 1000e18);
+ }
+
+ // ================================= SetSwapFee ================================
+
+ function test_setSwapFee_NotGovernorOrGuardian() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.NotGovernorOrGuardian.selector);
+ vm.prank(_alice);
+ _agToken.setSwapFee(address(_bridgeToken), 5e8);
+ }
+
+ function test_setSwapFee_InvalidToken() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.InvalidToken.selector);
+ vm.prank(_GOVERNOR);
+ _agToken.setSwapFee(_alice, 5e8);
+ }
+
+ function test_setSwapFee_TooHighParameterValue() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.TooHighParameterValue.selector);
+ vm.prank(_GOVERNOR);
+ _agToken.setSwapFee(address(_bridgeToken), 2e9);
+ }
+
+ function test_setSwapFee_Normal() public {
+ vm.prank(_GOVERNOR);
+ _agToken.setSwapFee(address(_bridgeToken), 3e8);
+ (, , uint64 fee, , ) = _agToken.bridges(address(_bridgeToken));
+ assertEq(fee, 3e8);
+ }
+
+ // ================================= ToggleBridge ================================
+
+ function test_toggleBridge_NotGovernorOrGuardian() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.NotGovernorOrGuardian.selector);
+ vm.prank(_alice);
+ _agToken.toggleBridge(address(_bridgeToken));
+ }
+
+ function test_toggleBridge_NonExistingBridge() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.InvalidToken.selector);
+ vm.prank(_GOVERNOR);
+ _agToken.toggleBridge(_alice);
+ }
+
+ function test_toggleBridge_Paused() public {
+ vm.prank(_GOVERNOR);
+ _agToken.toggleBridge(address(_bridgeToken));
+ (, , , , bool paused) = _agToken.bridges(address(_bridgeToken));
+ assertTrue(paused);
+ }
+
+ function test_toggleBridge_Unpaused() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.toggleBridge(address(_bridgeToken));
+ _agToken.toggleBridge(address(_bridgeToken));
+ vm.stopPrank();
+ (, , , , bool paused) = _agToken.bridges(address(_bridgeToken));
+ assertFalse(paused);
+ }
+
+ // ================================= ToggleFeesForAddress ================================
+
+ function test_toggleFeesForAddress_NotGovernorOrGuardian() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.NotGovernorOrGuardian.selector);
+ vm.prank(_alice);
+ _agToken.toggleFeesForAddress(_alice);
+ }
+
+ function test_toggleFeesForAddress_AddressExempted() public {
+ vm.prank(_GOVERNOR);
+ _agToken.toggleFeesForAddress(_alice);
+ assertEq(_agToken.isFeeExempt(_alice), 1);
+ }
+
+ function test_toggleFeesForAddress_AddressNotExempted() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.toggleFeesForAddress(_alice);
+ _agToken.toggleFeesForAddress(_alice);
+ vm.stopPrank();
+ assertEq(_agToken.isFeeExempt(_alice), 0);
+ }
+
+ // ================================= SwapIn ================================
+
+ function test_swapIn_IncorrectBridgeToken() public {
+ vm.expectRevert(AgTokenSideChainMultiBridge.InvalidToken.selector);
+ _agToken.swapIn(_bob, 1e18, _alice);
+ }
+
+ function test_swapIn_PausedBridge() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.toggleBridge(address(_bridgeToken));
+ vm.stopPrank();
+ vm.expectRevert(AgTokenSideChainMultiBridge.InvalidToken.selector);
+ _agToken.swapIn(address(_bridgeToken), 1e18, _alice);
+ }
+
+ function test_swapIn_InsufficentBalanceOrApproval() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.setLimit(address(_bridgeToken), 100e18);
+ _agToken.setHourlyLimit(address(_bridgeToken), 100e18);
+ _bridgeToken.mint(_GOVERNOR, 10e18);
+ vm.expectRevert();
+ _agToken.swapIn(address(_bridgeToken), 50e18, _alice);
+ _bridgeToken.approve(address(_agToken), 100e18);
+ vm.expectRevert();
+ _agToken.swapIn(address(_bridgeToken), 50e18, _alice);
+ vm.stopPrank();
+ }
+
+ function test_swapIn_ZeroLimitSwaps() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.setLimit(address(_bridgeToken), 0);
+ _agToken.swapIn(address(_bridgeToken), 1e18, _alice);
+ vm.stopPrank();
+ assertEq(_agToken.balanceOf(_bob), 0);
+ }
+
+ function test_swapIn_AmountGreaterThanLimit() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.setLimit(address(_bridgeToken), 10e18);
+ _agToken.setHourlyLimit(address(_bridgeToken), 10e18);
+ _agToken.setSwapFee(address(_bridgeToken), 0);
+ _bridgeToken.mint(_GOVERNOR, 100e18);
+ _bridgeToken.approve(address(_agToken), 100e18);
+ _agToken.swapIn(address(_bridgeToken), 100e18, _bob);
+ vm.stopPrank();
+ assertEq(_agToken.balanceOf(_bob), 10e18);
+ assertEq(_bridgeToken.balanceOf(_GOVERNOR), 90e18);
+ assertEq(_agToken.currentUsage(address(_bridgeToken)), 10e18);
+ }
+
+ function test_swapIn_AmountGreaterThanHourlyLimit() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.setLimit(address(_bridgeToken), 10e18);
+ _agToken.setHourlyLimit(address(_bridgeToken), 1e18);
+ _agToken.setSwapFee(address(_bridgeToken), 0);
+ _bridgeToken.mint(_GOVERNOR, 2e18);
+ _bridgeToken.approve(address(_agToken), 2e18);
+ assertEq(_agToken.balanceOf(_bob), 0);
+ _agToken.swapIn(address(_bridgeToken), 2e18, _bob);
+ vm.stopPrank();
+ assertEq(_agToken.balanceOf(_bob), 1e18);
+ assertEq(_bridgeToken.balanceOf(_GOVERNOR), 1e18);
+ assertEq(_agToken.currentUsage(address(_bridgeToken)), 1e18);
+ }
+
+ function test_swapIn_TotalAmountGreaterThanHourlyLimit() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.setLimit(address(_bridgeToken), 10e18);
+ _agToken.setHourlyLimit(address(_bridgeToken), 2e18);
+ _bridgeToken.mint(_GOVERNOR, 3e18);
+ _bridgeToken.approve(address(_agToken), 3e18);
+ _agToken.swapIn(address(_bridgeToken), 1e18, _alice);
+ assertEq(_agToken.currentUsage(address(_bridgeToken)), 1e18);
+ _agToken.swapIn(address(_bridgeToken), 2e18, _alice);
+ vm.stopPrank();
+ assertEq(_agToken.balanceOf(_alice), 2e18);
+ assertEq(_bridgeToken.balanceOf(_GOVERNOR), 1e18);
+ assertEq(_agToken.currentUsage(address(_bridgeToken)), 2e18);
+ }
+
+ function test_swapIn_HourlyLimitOverTwoHours() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.setSwapFee(address(_bridgeToken), 0);
+ _agToken.setLimit(address(_bridgeToken), 10e18);
+ _agToken.setHourlyLimit(address(_bridgeToken), 2e18);
+ _bridgeToken.mint(_GOVERNOR, 3e18);
+ _bridgeToken.approve(address(_agToken), 3e18);
+ _agToken.swapIn(address(_bridgeToken), 1e18, _bob);
+ assertEq(_agToken.currentUsage(address(_bridgeToken)), 1e18);
+ assertEq(_agToken.balanceOf(_bob), 1e18);
+ assertEq(_bridgeToken.balanceOf(_GOVERNOR), 2e18);
+ assertEq(_bridgeToken.balanceOf(address(_agToken)), 1e18);
+
+ uint256 hour = block.timestamp / 3600;
+ assertEq(_agToken.usage(address(_bridgeToken), hour), 1e18);
+ vm.warp(block.timestamp + 3600);
+ hour = block.timestamp / 3600;
+ assertEq(_agToken.usage(address(_bridgeToken), hour - 1), 1e18);
+ assertEq(_agToken.usage(address(_bridgeToken), hour), 0);
+ assertEq(_agToken.currentUsage(address(_bridgeToken)), 0);
+ _agToken.swapIn(address(_bridgeToken), 2e18, _bob);
+ assertEq(_agToken.usage(address(_bridgeToken), hour), 2e18);
+ assertEq(_bridgeToken.balanceOf(_GOVERNOR), 0);
+ assertEq(_bridgeToken.balanceOf(address(_agToken)), 3e18);
+ assertEq(_agToken.balanceOf(_bob), 3e18);
+ }
+
+ function test_swapIn_WithSomeTransactionFees() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.setSwapFee(address(_bridgeToken), 5e8);
+ _agToken.setLimit(address(_bridgeToken), 100e18);
+ _agToken.setHourlyLimit(address(_bridgeToken), 10e18);
+ _bridgeToken.mint(_GOVERNOR, 10e18);
+ _bridgeToken.approve(address(_agToken), 10e18);
+ _agToken.swapIn(address(_bridgeToken), 10e18, _bob);
+ vm.stopPrank();
+ assertEq(_bridgeToken.balanceOf(address(_agToken)), 10e18);
+ assertEq(_bridgeToken.balanceOf(_GOVERNOR), 0e18);
+ assertEq(_agToken.balanceOf(_bob), 5e18);
+ }
+
+ function test_swapIn_WithSomeTransactionFesAndExempt() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.toggleFeesForAddress(_GOVERNOR);
+ _agToken.setSwapFee(address(_bridgeToken), 5e8);
+ _agToken.setLimit(address(_bridgeToken), 100e18);
+ _agToken.setHourlyLimit(address(_bridgeToken), 100e18);
+ _bridgeToken.mint(_GOVERNOR, 10e18);
+ _bridgeToken.approve(address(_agToken), 10e18);
+ _agToken.swapIn(address(_bridgeToken), 10e18, _bob);
+ vm.stopPrank();
+ assertEq(_bridgeToken.balanceOf(address(_agToken)), 10e18);
+ assertEq(_bridgeToken.balanceOf(_GOVERNOR), 0e18);
+ assertEq(_agToken.balanceOf(_bob), 10e18);
+ }
+
+ function test_swapIn_WithoutTransactionsFeesAndExempt() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.setSwapFee(address(_bridgeToken), 0);
+ _agToken.setLimit(address(_bridgeToken), 100e18);
+ _agToken.setHourlyLimit(address(_bridgeToken), 100e18);
+ _bridgeToken.mint(_GOVERNOR, 10e18);
+ _bridgeToken.approve(address(_agToken), 10e18);
+ _agToken.swapIn(address(_bridgeToken), 10e18, _bob);
+ vm.stopPrank();
+ assertEq(_bridgeToken.balanceOf(address(_agToken)), 10e18);
+ assertEq(_bridgeToken.balanceOf(_GOVERNOR), 0e18);
+ assertEq(_agToken.balanceOf(_bob), 10e18);
+ }
+
+ function test_swapIn_WithWeirdTransactionFees() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.setSwapFee(address(_bridgeToken), 4e5);
+ _agToken.setLimit(address(_bridgeToken), 100e18);
+ _agToken.setHourlyLimit(address(_bridgeToken), 100e18);
+ _bridgeToken.mint(_GOVERNOR, 100e18);
+ _bridgeToken.approve(address(_agToken), 100e18);
+ _agToken.swapIn(address(_bridgeToken), 100e18, _bob);
+ vm.stopPrank();
+ assertEq(_bridgeToken.balanceOf(address(_agToken)), 100e18);
+ assertEq(_bridgeToken.balanceOf(_GOVERNOR), 0e18);
+ assertEq(_agToken.balanceOf(_bob), 9996e16);
+ }
+
+ // ================================= SwapOut ================================
+
+ function test_swapOut_IncorrectBridgeToken() public {
+ vm.prank(_GOVERNOR);
+ _agToken.setChainTotalHourlyLimit(type(uint256).max);
+
+ vm.expectRevert(AgTokenSideChainMultiBridge.InvalidToken.selector);
+ _agToken.swapOut(_bob, 1e18, _alice);
+ }
+
+ function test_swapOut_BridgeTokenPaused() public {
+ vm.prank(_GOVERNOR);
+ _agToken.toggleBridge(address(_bridgeToken));
+
+ vm.prank(_GOVERNOR);
+ vm.expectRevert(AgTokenSideChainMultiBridge.InvalidToken.selector);
+ _agToken.swapOut(address(_bridgeToken), 1e18, _alice);
+ }
+
+ function test_swapOut_InvalidBridgeTokenBalance() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.setChainTotalHourlyLimit(type(uint256).max);
+ _agToken.setSwapFee(address(_bridgeToken), 5e8);
+ _agToken.mint(_GOVERNOR, 100e18);
+ vm.expectRevert();
+ _agToken.swapOut(address(_bridgeToken), 1e18, _alice);
+ vm.stopPrank();
+ }
+
+ function test_swapOut_HourlyLimitExceeded() public {
+ uint256 limit = 10e18;
+ vm.startPrank(_GOVERNOR);
+ _agToken.setChainTotalHourlyLimit(limit);
+
+ _agToken.mint(_GOVERNOR, 100e18);
+ _bridgeToken.mint(address(_agToken), 100e18);
+ _agToken.swapOut(address(_bridgeToken), 9e18, _bob);
+ vm.expectRevert(AgTokenSideChainMultiBridge.HourlyLimitExceeded.selector);
+ _agToken.swapOut(address(_bridgeToken), 2e18, _bob);
+ vm.stopPrank();
+ }
+
+ function test_swapOut_HourlyLimitExceededAtDifferentHours() public {
+ uint256 limit = 10e18;
+ vm.startPrank(_GOVERNOR);
+ _agToken.setChainTotalHourlyLimit(limit);
+
+ _agToken.mint(_GOVERNOR, 100e18);
+ _bridgeToken.mint(address(_agToken), 100e18);
+
+ _agToken.swapOut(address(_bridgeToken), 9e18, _bob);
+ vm.warp(block.timestamp + 3600);
+ _agToken.swapOut(address(_bridgeToken), 2e18, _bob);
+ vm.expectRevert(AgTokenSideChainMultiBridge.HourlyLimitExceeded.selector);
+ _agToken.swapOut(address(_bridgeToken), 81e17, _bob);
+ vm.stopPrank();
+ }
+
+ function test_swapOut_WithValidBridgeTokenBalance() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.setChainTotalHourlyLimit(type(uint256).max);
+ _agToken.setSwapFee(address(_bridgeToken), 5e8);
+ _agToken.mint(_GOVERNOR, 100e18);
+ _bridgeToken.mint(address(_agToken), 100e18);
+ _agToken.swapOut(address(_bridgeToken), 100e18, _bob);
+ vm.stopPrank();
+ assertEq(_agToken.balanceOf(_GOVERNOR), 0);
+ assertEq(_bridgeToken.balanceOf(_bob), 50e18);
+ assertEq(_bridgeToken.balanceOf(_GOVERNOR), 0);
+ assertEq(_bridgeToken.balanceOf(address(_agToken)), 50e18);
+ }
+
+ function test_swapOut_WithValidBridgeTokenBalanceButFeeExemption() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.setChainTotalHourlyLimit(type(uint256).max);
+ _agToken.toggleFeesForAddress(_GOVERNOR);
+ _agToken.setSwapFee(address(_bridgeToken), 5e8);
+ _agToken.mint(_GOVERNOR, 100e18);
+ _bridgeToken.mint(address(_agToken), 100e18);
+ _agToken.swapOut(address(_bridgeToken), 100e18, _bob);
+ vm.stopPrank();
+ assertEq(_agToken.balanceOf(_GOVERNOR), 0);
+ assertEq(_bridgeToken.balanceOf(_bob), 100e18);
+ assertEq(_bridgeToken.balanceOf(_GOVERNOR), 0);
+ assertEq(_bridgeToken.balanceOf(address(_agToken)), 0);
+ }
+
+ function test_swapOut_WithWeirdTransactionFees() public {
+ vm.startPrank(_GOVERNOR);
+ _agToken.setChainTotalHourlyLimit(type(uint256).max);
+ _agToken.setSwapFee(address(_bridgeToken), 4e5);
+ _agToken.mint(_GOVERNOR, 100e18);
+ _bridgeToken.mint(address(_agToken), 100e18);
+ _agToken.swapOut(address(_bridgeToken), 100e18, _bob);
+ vm.stopPrank();
+ assertEq(_agToken.balanceOf(_GOVERNOR), 0);
+ assertEq(_bridgeToken.balanceOf(_bob), 9996e16);
+ assertEq(_bridgeToken.balanceOf(_GOVERNOR), 0);
+ assertEq(_bridgeToken.balanceOf(address(_agToken)), 4e16);
+ }
+
+ function test_swapOut_HourlyLimitAtDifferentHours() public {
+ uint256 limit = 10e18;
+ vm.startPrank(_GOVERNOR);
+ _agToken.setChainTotalHourlyLimit(limit);
+
+ _agToken.mint(_GOVERNOR, 100e18);
+ _bridgeToken.mint(address(_agToken), 100e18);
+
+ _agToken.swapOut(address(_bridgeToken), 9e18, _bob);
+ assertEq(_agToken.chainTotalUsage(0), 9e18);
+ vm.warp(block.timestamp + 3600);
+ _agToken.swapOut(address(_bridgeToken), 2e18, _bob);
+ assertEq(_agToken.chainTotalUsage(1), 2e18);
+
+ _agToken.swapOut(address(_bridgeToken), 8e18, _bob);
+ assertEq(_agToken.chainTotalUsage(1), 10e18);
+ vm.stopPrank();
+ }
+
+ function test_swapOut_HourlyLimitUpdatedByGovernance() public {
+ uint256 limit = 10e18;
+ vm.startPrank(_GOVERNOR);
+ _agToken.setChainTotalHourlyLimit(limit);
+
+ _agToken.mint(_GOVERNOR, 100e18);
+ _bridgeToken.mint(address(_agToken), 100e18);
+
+ _agToken.swapOut(address(_bridgeToken), 9e18, _bob);
+ vm.expectRevert(AgTokenSideChainMultiBridge.HourlyLimitExceeded.selector);
+ _agToken.swapOut(address(_bridgeToken), 2e18, _bob);
+
+ uint256 hour = block.timestamp / 3600;
+ assertEq(_agToken.chainTotalUsage(hour), 9e18);
+
+ _agToken.setChainTotalHourlyLimit(11e18);
+ _agToken.swapOut(address(_bridgeToken), 2e18, _bob);
+ assertEq(_agToken.chainTotalUsage(hour), 11e18);
+
+ vm.expectRevert(AgTokenSideChainMultiBridge.HourlyLimitExceeded.selector);
+ _agToken.swapOut(address(_bridgeToken), 1e17, _bob);
+
+ assertEq(_agToken.chainTotalUsage(hour), 11e18);
+ vm.stopPrank();
+ }
+}
diff --git a/test/units/coreBorrow/CoreBorrow.t.sol b/test/units/coreBorrow/CoreBorrow.t.sol
new file mode 100644
index 0000000..e5d3e44
--- /dev/null
+++ b/test/units/coreBorrow/CoreBorrow.t.sol
@@ -0,0 +1,303 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import "../BaseTest.t.sol";
+import { CoreBorrow, ICoreBorrow } from "contracts/coreBorrow/CoreBorrow.sol";
+import { MockTreasury } from "contracts/mock/MockTreasury.sol";
+import { IAgToken } from "contracts/interfaces/IAgToken.sol";
+import { MockFlashLoanModule } from "contracts/mock/MockFlashLoanModule.sol";
+
+contract CoreBorrowTest is BaseTest {
+ MockTreasury internal _treasury;
+ CoreBorrow internal _coreBorrowImplem;
+ CoreBorrow internal _coreBorrow;
+ MockFlashLoanModule internal _flashAngle;
+
+ bytes32 constant GUARDIAN_ROLE = keccak256("GUARDIAN_ROLE");
+ bytes32 constant GOVERNOR_ROLE = keccak256("GOVERNOR_ROLE");
+ bytes32 constant FLASHLOANER_TREASURY_ROLE = keccak256("FLASHLOANER_TREASURY_ROLE");
+
+ function setUp() public override {
+ super.setUp();
+
+ _coreBorrowImplem = new CoreBorrow();
+ _coreBorrow = CoreBorrow(_deployUpgradeable(address(proxyAdmin), address(_coreBorrowImplem), ""));
+
+ _treasury = new MockTreasury(IAgToken(address(0)), address(0), address(0), address(0), address(0), address(0));
+ _flashAngle = new MockFlashLoanModule(_coreBorrow);
+
+ _coreBorrow.initialize(_GOVERNOR, _GUARDIAN);
+ }
+
+ // ================================= INITIALIZE ================================
+
+ function test_initialize_Constructor() public {
+ assertEq(_coreBorrow.isGovernor(_GOVERNOR), true);
+ assertEq(_coreBorrow.isGovernor(_GUARDIAN), false);
+ assertEq(_coreBorrow.isGovernorOrGuardian(_GUARDIAN), true);
+ assertEq(_coreBorrow.isGovernorOrGuardian(_GOVERNOR), true);
+ assertEq(_coreBorrow.isFlashLoanerTreasury(_GOVERNOR), false);
+ assertEq(_coreBorrow.isFlashLoanerTreasury(_GUARDIAN), false);
+ assertEq(_coreBorrow.getRoleAdmin(GUARDIAN_ROLE), GOVERNOR_ROLE);
+ assertEq(_coreBorrow.getRoleAdmin(GOVERNOR_ROLE), GOVERNOR_ROLE);
+ assertEq(_coreBorrow.getRoleAdmin(FLASHLOANER_TREASURY_ROLE), GOVERNOR_ROLE);
+ assertEq(_coreBorrow.hasRole(GOVERNOR_ROLE, _GOVERNOR), true);
+ assertEq(_coreBorrow.hasRole(GUARDIAN_ROLE, _GUARDIAN), true);
+ assertEq(_coreBorrow.hasRole(GUARDIAN_ROLE, _GOVERNOR), true);
+ assertEq(_coreBorrow.hasRole(FLASHLOANER_TREASURY_ROLE, _GOVERNOR), false);
+ assertEq(_coreBorrow.flashLoanModule(), address(0));
+ }
+
+ function test_initialize_AlreadyInitialized() public {
+ vm.expectRevert("Initializable: contract is already initialized");
+ _coreBorrow.initialize(_GOVERNOR, _GUARDIAN);
+ }
+
+ function test_initialize_InvalidGovernorGuardian() public {
+ _coreBorrow = CoreBorrow(_deployUpgradeable(address(proxyAdmin), address(_coreBorrowImplem), ""));
+ vm.expectRevert(CoreBorrow.IncompatibleGovernorAndGuardian.selector);
+ _coreBorrow.initialize(_GOVERNOR, _GOVERNOR);
+
+ vm.expectRevert(CoreBorrow.ZeroAddress.selector);
+ _coreBorrow.initialize(address(0), _GUARDIAN);
+
+ vm.expectRevert(CoreBorrow.ZeroAddress.selector);
+ _coreBorrow.initialize(_GOVERNOR, address(0));
+ }
+
+ // ================================= AddGovernor ================================
+
+ function test_addGovernor_NotGovernor() public {
+ vm.expectRevert();
+ vm.prank(_alice);
+ _coreBorrow.addGovernor(_alice);
+ }
+
+ function test_addGovernor_Normal() public {
+ vm.prank(_GOVERNOR);
+ _coreBorrow.addGovernor(_alice);
+ assertEq(_coreBorrow.isGovernor(_alice), true);
+ assertEq(_coreBorrow.isGovernorOrGuardian(_alice), true);
+ assertEq(_coreBorrow.hasRole(GOVERNOR_ROLE, _alice), true);
+ assertEq(_coreBorrow.hasRole(GUARDIAN_ROLE, _alice), true);
+ }
+
+ function test_addGovernor_Multi() public {
+ vm.prank(_GOVERNOR);
+ _coreBorrow.addGovernor(_alice);
+ vm.prank(_alice);
+ _coreBorrow.addGovernor(_bob);
+ assertEq(_coreBorrow.isGovernor(_bob), true);
+ assertEq(_coreBorrow.isGovernorOrGuardian(_bob), true);
+ assertEq(_coreBorrow.hasRole(GOVERNOR_ROLE, _bob), true);
+ assertEq(_coreBorrow.hasRole(GUARDIAN_ROLE, _bob), true);
+ }
+
+ // ================================= RemoveGovernor ================================
+
+ function test_removeGovernor_NotGovernor() public {
+ vm.expectRevert();
+ vm.prank(_alice);
+ _coreBorrow.removeGovernor(_alice);
+ }
+
+ function test_removeGovernor_NotEnoughGovernorsLeft() public {
+ vm.expectRevert(CoreBorrow.NotEnoughGovernorsLeft.selector);
+ _coreBorrow.removeGovernor(_GOVERNOR);
+ }
+
+ function test_removeGovernor_NormalAfterAsk() public {
+ vm.prank(_GOVERNOR);
+ _coreBorrow.addGovernor(_alice);
+ vm.prank(_alice);
+ _coreBorrow.removeGovernor(_alice);
+
+ assertEq(_coreBorrow.isGovernor(_alice), false);
+ assertEq(_coreBorrow.isGovernorOrGuardian(_alice), false);
+ assertEq(_coreBorrow.hasRole(GOVERNOR_ROLE, _alice), false);
+ assertEq(_coreBorrow.hasRole(GUARDIAN_ROLE, _alice), false);
+ }
+
+ function test_removeGovernor_Normal() public {
+ vm.prank(_GOVERNOR);
+ _coreBorrow.addGovernor(_alice);
+ vm.prank(_GOVERNOR);
+ _coreBorrow.removeGovernor(_alice);
+
+ assertEq(_coreBorrow.isGovernor(_alice), false);
+ assertEq(_coreBorrow.isGovernorOrGuardian(_alice), false);
+ assertEq(_coreBorrow.hasRole(GOVERNOR_ROLE, _alice), false);
+ assertEq(_coreBorrow.hasRole(GUARDIAN_ROLE, _alice), false);
+ }
+
+ // ================================= SetFlashLoanModule ================================
+
+ function test_setFlashLoanModule_NotGovernor() public {
+ vm.expectRevert();
+ vm.prank(_alice);
+ _coreBorrow.setFlashLoanModule(_GOVERNOR);
+ }
+
+ function test_setFlashLoanModule_ZeroAddress() public {
+ vm.prank(_GOVERNOR);
+ _coreBorrow.setFlashLoanModule(address(0));
+ assertEq(_coreBorrow.flashLoanModule(), address(0));
+ }
+
+ function test_setFlashLoanModule_NoTreasury() public {
+ vm.prank(_GOVERNOR);
+ _coreBorrow.setFlashLoanModule(address(_flashAngle));
+ assertEq(_coreBorrow.flashLoanModule(), address(_flashAngle));
+ }
+
+ function test_setFlashLoanModule_WrongCore() public {
+ _flashAngle = new MockFlashLoanModule(CoreBorrow(_GOVERNOR));
+ vm.expectRevert(CoreBorrow.InvalidCore.selector);
+ vm.prank(_GOVERNOR);
+ _coreBorrow.setFlashLoanModule(address(_flashAngle));
+ }
+
+ function test_setFlashLoanModule_Normal() public {
+ vm.startPrank(_GOVERNOR);
+ _coreBorrow.addFlashLoanerTreasuryRole(address(_treasury));
+ _coreBorrow.setFlashLoanModule(address(_flashAngle));
+ vm.stopPrank();
+ assertEq(_coreBorrow.flashLoanModule(), address(_flashAngle));
+ assertEq(_treasury.flashLoanModule(), address(_flashAngle));
+ }
+
+ // ================================= AddFlashLoanerTreasuryRole ================================
+
+ function test_addFlashLoanerTreasuryRole_NotGovernor() public {
+ vm.expectRevert();
+ vm.prank(_alice);
+ _coreBorrow.addFlashLoanerTreasuryRole(address(_treasury));
+ }
+
+ function test_addFlashLoanerTreasuryRole_ZeroFlashLoandModule() public {
+ vm.prank(_GOVERNOR);
+ _coreBorrow.addFlashLoanerTreasuryRole(address(_treasury));
+ assertEq(_coreBorrow.isFlashLoanerTreasury(address(_treasury)), true);
+ assertEq(_coreBorrow.hasRole(FLASHLOANER_TREASURY_ROLE, address(_treasury)), true);
+ }
+
+ function test_addFlashLoanerTreasuryRole_Normal() public {
+ vm.startPrank(_GOVERNOR);
+ _coreBorrow.setFlashLoanModule(address(_flashAngle));
+ _coreBorrow.addFlashLoanerTreasuryRole(address(_treasury));
+ vm.stopPrank();
+ assertEq(_coreBorrow.isFlashLoanerTreasury(address(_treasury)), true);
+ assertEq(_coreBorrow.hasRole(FLASHLOANER_TREASURY_ROLE, address(_treasury)), true);
+ assertEq(_coreBorrow.flashLoanModule(), address(_flashAngle));
+ assertEq(_treasury.flashLoanModule(), address(_flashAngle));
+ assertEq(_flashAngle.stablecoinsSupported(address(_treasury)), true);
+ }
+
+ // ================================= RemoveFlashLoanerTreasuryRole ================================
+
+ function test_removeFlashLoanerTreasuryRole_NotGovernor() public {
+ vm.expectRevert();
+ vm.prank(_alice);
+ _coreBorrow.removeFlashLoanerTreasuryRole(address(_treasury));
+ }
+
+ function test_removeFlashLoanerTreasuryRole_ZeroFlashLoanModule() public {
+ vm.startPrank(_GOVERNOR);
+ _coreBorrow.addFlashLoanerTreasuryRole(address(_treasury));
+ _coreBorrow.removeFlashLoanerTreasuryRole(address(_treasury));
+ vm.stopPrank();
+ assertEq(_coreBorrow.isFlashLoanerTreasury(address(_treasury)), false);
+ assertEq(_coreBorrow.hasRole(FLASHLOANER_TREASURY_ROLE, address(_treasury)), false);
+ }
+
+ function test_removeFlashLoanerTreasuryRole_Normal() public {
+ vm.startPrank(_GOVERNOR);
+ _coreBorrow.setFlashLoanModule(address(_flashAngle));
+ _coreBorrow.addFlashLoanerTreasuryRole(address(_treasury));
+ _coreBorrow.removeFlashLoanerTreasuryRole(address(_treasury));
+ vm.stopPrank();
+ assertEq(_coreBorrow.isFlashLoanerTreasury(address(_treasury)), false);
+ assertEq(_coreBorrow.hasRole(FLASHLOANER_TREASURY_ROLE, address(_treasury)), false);
+ assertEq(_coreBorrow.flashLoanModule(), address(_flashAngle));
+ assertEq(_treasury.flashLoanModule(), address(0));
+ assertEq(_flashAngle.stablecoinsSupported(address(_treasury)), false);
+ }
+
+ // ================================= SetCore ================================
+
+ function test_setCore_NotGovernor() public {
+ vm.expectRevert();
+ vm.prank(_alice);
+ _coreBorrow.setCore(ICoreBorrow(address(_treasury)));
+ }
+
+ function test_setCore_GoodGovernorNoFlashLoanModule() public {
+ CoreBorrow coreBorrowRevert = CoreBorrow(
+ _deployUpgradeable(address(proxyAdmin), address(_coreBorrowImplem), "")
+ );
+ coreBorrowRevert.initialize(_GOVERNOR, _GUARDIAN);
+
+ vm.expectEmit(true, true, true, true);
+ emit CoreBorrow.CoreUpdated(address(coreBorrowRevert));
+ vm.prank(_GOVERNOR);
+ _coreBorrow.setCore(ICoreBorrow(address(coreBorrowRevert)));
+ }
+
+ function test_setCore_GoodGovernorAndFlashLoanModule() public {
+ CoreBorrow coreBorrowRevert = CoreBorrow(
+ _deployUpgradeable(address(proxyAdmin), address(_coreBorrowImplem), "")
+ );
+ coreBorrowRevert.initialize(_GOVERNOR, _GUARDIAN);
+
+ vm.startPrank(_GOVERNOR);
+ _coreBorrow.setFlashLoanModule(address(_flashAngle));
+ vm.expectEmit(true, true, true, true);
+ emit CoreBorrow.CoreUpdated(address(coreBorrowRevert));
+ _coreBorrow.setCore(ICoreBorrow(address(coreBorrowRevert)));
+ vm.stopPrank();
+ assertEq(address(_flashAngle.core()), address(coreBorrowRevert));
+ }
+
+ function test_setCore_WrongGovernor() public {
+ CoreBorrow coreBorrowRevert = CoreBorrow(
+ _deployUpgradeable(address(proxyAdmin), address(_coreBorrowImplem), "")
+ );
+ coreBorrowRevert.initialize(_GOVERNOR_POLYGON, _alice);
+
+ vm.expectRevert(CoreBorrow.InvalidCore.selector);
+ vm.prank(_GOVERNOR);
+ _coreBorrow.setCore(ICoreBorrow(address(coreBorrowRevert)));
+ }
+
+ // ================================= GrantGuardianRole ================================
+
+ function test_grantGuardianRole_NotGuardian() public {
+ vm.expectRevert();
+ vm.prank(_alice);
+ _coreBorrow.grantRole(GUARDIAN_ROLE, _alice);
+ }
+
+ function test_grantGuardianRole_Normal() public {
+ vm.prank(_GOVERNOR);
+ _coreBorrow.grantRole(GUARDIAN_ROLE, _alice);
+ assertEq(_coreBorrow.isGovernorOrGuardian(_alice), true);
+ assertEq(_coreBorrow.hasRole(GUARDIAN_ROLE, _alice), true);
+ }
+
+ // ================================= RevokeGuardianRole ================================
+
+ function test_revokeGuardianRole_NotGuardian() public {
+ vm.expectRevert();
+ vm.prank(_alice);
+ _coreBorrow.revokeRole(GUARDIAN_ROLE, _alice);
+ }
+
+ function test_revokeGuardianRole_Normal() public {
+ vm.startPrank(_GOVERNOR);
+ _coreBorrow.grantRole(GUARDIAN_ROLE, _alice);
+ _coreBorrow.revokeRole(GUARDIAN_ROLE, _alice);
+ vm.stopPrank();
+ assertEq(_coreBorrow.isGovernorOrGuardian(_alice), false);
+ assertEq(_coreBorrow.hasRole(GUARDIAN_ROLE, _alice), false);
+ }
+}
diff --git a/test/units/flashAngle/FlashAngle.t.sol b/test/units/flashAngle/FlashAngle.t.sol
new file mode 100644
index 0000000..1f53d19
--- /dev/null
+++ b/test/units/flashAngle/FlashAngle.t.sol
@@ -0,0 +1,268 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.17;
+
+import { stdStorage, StdStorage } from "forge-std/Test.sol";
+import "../BaseTest.t.sol";
+import { MockToken } from "contracts/mock/MockToken.sol";
+import { MockFlashLoanReceiver } from "contracts/mock/MockFlashLoanReceiver.sol";
+import { MockCoreBorrow, ICoreBorrow } from "contracts/mock/MockCoreBorrow.sol";
+import { FlashAngle, IFlashAngle, IERC3156FlashBorrower } from "contracts/flashloan/FlashAngle.sol";
+import { MockTreasury } from "contracts/mock/MockTreasury.sol";
+import { IAgToken } from "contracts/interfaces/IAgToken.sol";
+
+contract FlashAngleTest is BaseTest {
+ using stdStorage for StdStorage;
+
+ MockToken internal _token;
+ MockTreasury internal _treasury;
+ MockFlashLoanReceiver internal _flashLoanReceiver;
+ MockCoreBorrow internal _coreBorrow;
+ FlashAngle internal _flashAngleImplem;
+ FlashAngle internal _flashAngle;
+
+ function setUp() public override {
+ super.setUp();
+
+ _token = new MockToken("agEUR", "agEUR", 18);
+
+ _coreBorrow = new MockCoreBorrow();
+ _flashLoanReceiver = new MockFlashLoanReceiver();
+
+ _treasury = new MockTreasury(
+ IAgToken(address(_token)),
+ address(0),
+ address(0),
+ address(0),
+ address(0),
+ address(0)
+ );
+
+ _flashAngleImplem = new FlashAngle();
+ _flashAngle = FlashAngle(_deployUpgradeable(address(proxyAdmin), address(_flashAngleImplem), ""));
+ _flashAngle.initialize(ICoreBorrow(address(_coreBorrow)));
+
+ _coreBorrow.addStablecoinSupport(IFlashAngle(address(_flashAngle)), address(_treasury));
+ _coreBorrow.toggleGovernor(_GOVERNOR);
+ _coreBorrow.toggleGuardian(_GUARDIAN);
+ }
+
+ // ================================= INITIALIZE ================================
+
+ function test_initialize_Constructor() public {
+ assertEq(address(_flashAngle.core()), address(_coreBorrow));
+ assertEq(address(_treasury.stablecoin()), address(_token));
+ (uint256 maxBorrowable, uint64 flashLoanFee, address treasury) = _flashAngle.stablecoinMap(
+ IAgToken(address(_token))
+ );
+ assertEq(treasury, address(_treasury));
+ assertEq(flashLoanFee, 0);
+ assertEq(maxBorrowable, 0);
+ }
+
+ function test_initialize_AlreadyInitiliazed() public {
+ vm.expectRevert("Initializable: contract is already initialized");
+ _flashAngle.initialize(ICoreBorrow(_alice));
+ }
+
+ function test_initialize_ZeroAddress() public {
+ vm.expectRevert("Initializable: contract is already initialized");
+ _flashAngle.initialize(ICoreBorrow(address(0)));
+ }
+
+ // ================================= AddStableCoinSupport ================================
+
+ function test_addStablecoinSupport_NotCore() public {
+ vm.expectRevert(FlashAngle.NotCore.selector);
+ _flashAngle.addStablecoinSupport(_GUARDIAN);
+ }
+
+ function test_addStablecoinSupport_Normal() public {
+ _treasury = new MockTreasury(
+ IAgToken(address(_token)),
+ address(0),
+ address(0),
+ address(0),
+ address(0),
+ address(0)
+ );
+ _coreBorrow.addStablecoinSupport(IFlashAngle(address(_flashAngle)), address(_treasury));
+ (uint256 maxBorrowable, uint64 flashLoanFee, address treasury) = _flashAngle.stablecoinMap(
+ IAgToken(address(_token))
+ );
+ assertEq(treasury, address(_treasury));
+ assertEq(flashLoanFee, 0);
+ assertEq(maxBorrowable, 0);
+ }
+
+ // ================================= RemoveStableCoinSupport ================================
+
+ function test_removeStablecoinSupport_NotCore() public {
+ vm.expectRevert(FlashAngle.NotCore.selector);
+ _flashAngle.removeStablecoinSupport(address(_treasury));
+ }
+
+ function test_removeStablecoinSupport_Normal() public {
+ _coreBorrow.removeStablecoinSupport(_flashAngle, address(_treasury));
+ (uint256 maxBorrowable, uint64 flashLoanFee, address treasury) = _flashAngle.stablecoinMap(
+ IAgToken(address(_token))
+ );
+ assertEq(treasury, address(0));
+ assertEq(flashLoanFee, 0);
+ assertEq(maxBorrowable, 0);
+ }
+
+ // ================================= SetCore ================================
+
+ function test_setCore_NotCore() public {
+ vm.expectRevert(FlashAngle.NotCore.selector);
+ _flashAngle.setCore(_alice);
+ }
+
+ function test_setCore_Normal() public {
+ _coreBorrow.setCore(_flashAngle, _alice);
+ assertEq(address(_flashAngle.core()), _alice);
+ }
+
+ // ================================= SetFlashLoanParameters ================================
+
+ function test_setFlashLoanParameters_UnsupportedStablecoin() public {
+ vm.expectRevert(FlashAngle.UnsupportedStablecoin.selector);
+ _flashAngle.setFlashLoanParameters(IAgToken(address(0)), 1e18, 0);
+ }
+
+ function test_setFlashLoanParameters_NotGovernorOrGuardian() public {
+ vm.expectRevert(FlashAngle.NotGovernorOrGuardian.selector);
+ _flashAngle.setFlashLoanParameters(IAgToken(address(_token)), 1e18, 0);
+ }
+
+ function test_setFlashLoanParameters_TooHighFee() public {
+ vm.expectRevert(FlashAngle.TooHighParameterValue.selector);
+ vm.prank(_GOVERNOR);
+ _flashAngle.setFlashLoanParameters(IAgToken(address(_token)), 1e18, 0);
+ }
+
+ function test_setFlashLoanParameters_Normal() public {
+ vm.prank(_GOVERNOR);
+ _flashAngle.setFlashLoanParameters(IAgToken(address(_token)), 5e8, 100e18);
+ (uint256 maxBorrowable, uint64 flashLoanFee, address treasury) = _flashAngle.stablecoinMap(
+ IAgToken(address(_token))
+ );
+ assertEq(treasury, address(_treasury));
+ assertEq(flashLoanFee, 5e8);
+ assertEq(maxBorrowable, 100e18);
+ }
+
+ // ================================= FlashFee ================================
+
+ function test_flashFee_UnsupportedStablecoin() public {
+ vm.expectRevert(FlashAngle.UnsupportedStablecoin.selector);
+ _flashAngle.flashFee(address(0), 1e18);
+ }
+
+ function test_flashFee_NormalNoFlashFee() public {
+ assertEq(_flashAngle.flashFee(address(_token), 1e18), 0);
+ }
+
+ function test_flashFee_NormalWithFlashFee() public {
+ vm.prank(_GOVERNOR);
+ _flashAngle.setFlashLoanParameters(IAgToken(address(_token)), 5e8, 100e18);
+ assertEq(_flashAngle.flashFee(address(_token), 1e18), 5e17);
+ }
+
+ // ================================= MaxFlashLoan ================================
+
+ function test_maxFlashLoan_NonExistingToken() public {
+ assertEq(_flashAngle.maxFlashLoan(_GUARDIAN), 0);
+ }
+
+ function test_maxFlashLoan_UninitalizedParameters() public {
+ assertEq(_flashAngle.maxFlashLoan(address(_token)), 0);
+ }
+
+ function test_maxFlashLoan_Normal() public {
+ vm.prank(_GOVERNOR);
+ _flashAngle.setFlashLoanParameters(IAgToken(address(_token)), 5e8, 100e18);
+ assertEq(_flashAngle.maxFlashLoan(address(_token)), 100e18);
+ }
+
+ // ================================= AccruteInterestToTreasury ================================
+
+ function test_accrueInterestToTreasury_NotTreasury() public {
+ vm.expectRevert(FlashAngle.NotTreasury.selector);
+ _flashAngle.accrueInterestToTreasury(IAgToken(address(_token)));
+ }
+
+ function test_accureInterestToTreasury_InvalidStablecoin() public {
+ vm.expectRevert(FlashAngle.NotTreasury.selector);
+ _flashAngle.accrueInterestToTreasury(IAgToken(address(_token)));
+ }
+
+ function test_accrueInterestToTreasury_NormalZeroBalance() public {
+ _treasury.accrueInterestToTreasury(_flashAngle);
+ assertEq(_token.balanceOf(address(_flashAngle)), 0);
+ }
+
+ function test_accrueInterestToTreasury_Normal() public {
+ _token.mint(address(_flashAngle), 1e18);
+ _treasury.accrueInterestToTreasury(_flashAngle);
+ assertEq(_token.balanceOf(address(_flashAngle)), 0);
+ }
+
+ // ================================= FlashLoan ================================
+
+ function test_flashLoan_UnsupportedStablecoin() public {
+ vm.expectRevert(FlashAngle.UnsupportedStablecoin.selector);
+ _flashAngle.flashLoan(_flashLoanReceiver, _GUARDIAN, 0, "");
+ }
+
+ function test_flashloan_TooBigAmount() public {
+ vm.expectRevert(FlashAngle.TooBigAmount.selector);
+ _flashAngle.flashLoan(_flashLoanReceiver, address(_token), 1e18, "");
+ }
+
+ function test_flashLoan_InvalidReturnMessage() public {
+ vm.prank(_GOVERNOR);
+ _flashAngle.setFlashLoanParameters(IAgToken(address(_token)), 5e8, 10000e18);
+
+ vm.expectRevert(FlashAngle.InvalidReturnMessage.selector);
+ _flashAngle.flashLoan(_flashLoanReceiver, address(_token), 1001e18, "");
+ }
+
+ function test_flashloan_TooSmallBalance() public {
+ vm.prank(_GOVERNOR);
+ _flashAngle.setFlashLoanParameters(IAgToken(address(_token)), 5e8, 10000e18);
+
+ vm.expectRevert("ERC20: transfer amount exceeds balance");
+ _flashAngle.flashLoan(_flashLoanReceiver, address(_token), 100e18, "");
+ }
+
+ function test_flashLoan_Normal() public {
+ vm.prank(_GOVERNOR);
+ _flashAngle.setFlashLoanParameters(IAgToken(address(_token)), 5e8, 10000e18);
+ _token.mint(address(_flashLoanReceiver), 50e18);
+
+ assertEq(_token.balanceOf(address(_flashLoanReceiver)), 50e18);
+ assertEq(_token.balanceOf(address(_flashAngle)), 0);
+ assertEq(_flashAngle.flashFee(address(_token), 100e18), 50e18);
+
+ vm.expectEmit(true, true, true, true);
+ emit FlashAngle.FlashLoan(address(_token), 100e18, IERC3156FlashBorrower(address(_flashLoanReceiver)));
+ _flashAngle.flashLoan(_flashLoanReceiver, address(_token), 100e18, "FlashLoanReceiver");
+
+ assertEq(_token.balanceOf(address(_flashLoanReceiver)), 0);
+ assertEq(_token.balanceOf(address(_flashAngle)), 50e18);
+ }
+
+ function test_flashloan_Reentrant() public {
+ vm.prank(_GOVERNOR);
+ _flashAngle.setFlashLoanParameters(IAgToken(address(_token)), 5e8, 10000e18);
+ _token.mint(address(_flashLoanReceiver), 50e18);
+
+ assertEq(_token.balanceOf(address(_flashLoanReceiver)), 50e18);
+ assertEq(_token.balanceOf(address(_flashAngle)), 0);
+ assertEq(_flashAngle.flashFee(address(_token), 100e18), 50e18);
+
+ vm.expectRevert("ReentrancyGuard: reentrant call");
+ _flashAngle.flashLoan(_flashLoanReceiver, address(_token), 2e18, "");
+ }
+}
diff --git a/test/units/utils/Solenv.sol b/test/units/utils/Solenv.sol
new file mode 100644
index 0000000..3716b32
--- /dev/null
+++ b/test/units/utils/Solenv.sol
@@ -0,0 +1,809 @@
+// SPDX-License-Identifier: MIT
+pragma solidity ^0.8.12;
+
+import { Vm } from "forge-std/Vm.sol";
+
+/*
+Copied from "memester-xyz/solenv"
+https://github.com/memester-xyz/solenv
+*/
+
+library Solenv {
+ using strings for *;
+
+ Vm constant vm = Vm(address(bytes20(uint160(uint256(keccak256("hevm cheat code"))))));
+
+ function config(string memory filename) internal {
+ string[] memory inputs = new string[](3);
+ inputs[0] = "sh";
+ inputs[1] = "-c";
+ inputs[2] = string(
+ bytes.concat('cast abi-encode "response(bytes)" $(xxd -p -c 1000000 ', bytes(filename), ")")
+ );
+
+ bytes memory res = vm.ffi(inputs);
+
+ strings.slice memory data = abi.decode(res, (string)).toSlice();
+
+ strings.slice memory lineDelim = "\n".toSlice();
+ strings.slice memory keyDelim = "=".toSlice();
+ strings.slice memory commentDelim = "#".toSlice();
+
+ uint256 length = data.count(lineDelim) + 1;
+ for (uint256 i; i < length; ++i) {
+ strings.slice memory line = data.split(lineDelim);
+ if (!line.startsWith(commentDelim)) {
+ string memory key = line.split(keyDelim).toString();
+ // Ignore empty lines
+ if (bytes(key).length != 0) {
+ vm.setEnv(key, line.toString());
+ }
+ }
+ }
+ }
+
+ function config() internal {
+ config(".env");
+ }
+}
+
+/*
+ * @title String & slice utility library for Solidity contracts.
+ * @author Nick Johnson
+ *
+ * @dev Functionality in this library is largely implemented using an
+ * abstraction called a 'slice'. A slice represents a part of a string -
+ * anything from the entire string to a single character, or even no
+ * characters at all (a 0-length slice). Since a slice only has to specify
+ * an offset and a length, copying and manipulating slices is a lot less
+ * expensive than copying and manipulating the strings they reference.
+ *
+ * To further reduce gas costs, most functions on slice that need to return
+ * a slice modify the original one instead of allocating a new one; for
+ * instance, `s.split(".")` will return the text up to the first '.',
+ * modifying s to only contain the remainder of the string after the '.'.
+ * In situations where you do not want to modify the original slice, you
+ * can make a copy first with `.copy()`, for example:
+ * `s.copy().split(".")`. Try and avoid using this idiom in loops; since
+ * Solidity has no memory management, it will result in allocating many
+ * short-lived slices that are later discarded.
+ *
+ * Functions that return two slices come in two versions: a non-allocating
+ * version that takes the second slice as an argument, modifying it in
+ * place, and an allocating version that allocates and returns the second
+ * slice; see `nextRune` for example.
+ *
+ * Functions that have to copy string data will return strings rather than
+ * slices; these can be cast back to slices for further processing if
+ * required.
+ *
+ * For convenience, some functions are provided with non-modifying
+ * variants that create a new slice and return both; for instance,
+ * `s.splitNew('.')` leaves s unmodified, and returns two values
+ * corresponding to the left and right parts of the string.
+ */
+
+pragma solidity ^0.8.0;
+
+library strings {
+ struct slice {
+ uint256 _len;
+ uint256 _ptr;
+ }
+
+ function memcpy(uint256 dest, uint256 src, uint256 len) private pure {
+ // Copy word-length chunks while possible
+ for (; len >= 32; len -= 32) {
+ assembly {
+ mstore(dest, mload(src))
+ }
+ dest += 32;
+ src += 32;
+ }
+
+ // Copy remaining bytes
+ uint256 mask = type(uint256).max;
+ if (len > 0) {
+ mask = 256 ** (32 - len) - 1;
+ }
+ assembly {
+ let srcpart := and(mload(src), not(mask))
+ let destpart := and(mload(dest), mask)
+ mstore(dest, or(destpart, srcpart))
+ }
+ }
+
+ /*
+ * @dev Returns a slice containing the entire string.
+ * @param self The string to make a slice from.
+ * @return A newly allocated slice containing the entire string.
+ */
+ function toSlice(string memory self) internal pure returns (slice memory) {
+ uint256 ptr;
+ assembly {
+ ptr := add(self, 0x20)
+ }
+ return slice(bytes(self).length, ptr);
+ }
+
+ /*
+ * @dev Returns the length of a null-terminated bytes32 string.
+ * @param self The value to find the length of.
+ * @return The length of the string, from 0 to 32.
+ */
+ function len(bytes32 self) internal pure returns (uint256) {
+ uint256 ret;
+ if (self == 0) return 0;
+ if (uint256(self) & type(uint128).max == 0) {
+ ret += 16;
+ self = bytes32(uint256(self) / 0x100000000000000000000000000000000);
+ }
+ if (uint256(self) & type(uint64).max == 0) {
+ ret += 8;
+ self = bytes32(uint256(self) / 0x10000000000000000);
+ }
+ if (uint256(self) & type(uint32).max == 0) {
+ ret += 4;
+ self = bytes32(uint256(self) / 0x100000000);
+ }
+ if (uint256(self) & type(uint16).max == 0) {
+ ret += 2;
+ self = bytes32(uint256(self) / 0x10000);
+ }
+ if (uint256(self) & type(uint8).max == 0) {
+ ret += 1;
+ }
+ return 32 - ret;
+ }
+
+ /*
+ * @dev Returns a slice containing the entire bytes32, interpreted as a
+ * null-terminated utf-8 string.
+ * @param self The bytes32 value to convert to a slice.
+ * @return A new slice containing the value of the input argument up to the
+ * first null.
+ */
+ function toSliceB32(bytes32 self) internal pure returns (slice memory ret) {
+ // Allocate space for `self` in memory, copy it there, and point ret at it
+ assembly {
+ let ptr := mload(0x40)
+ mstore(0x40, add(ptr, 0x20))
+ mstore(ptr, self)
+ mstore(add(ret, 0x20), ptr)
+ }
+ ret._len = len(self);
+ }
+
+ /*
+ * @dev Returns a new slice containing the same data as the current slice.
+ * @param self The slice to copy.
+ * @return A new slice containing the same data as `self`.
+ */
+ function copy(slice memory self) internal pure returns (slice memory) {
+ return slice(self._len, self._ptr);
+ }
+
+ /*
+ * @dev Copies a slice to a new string.
+ * @param self The slice to copy.
+ * @return A newly allocated string containing the slice's text.
+ */
+ function toString(slice memory self) internal pure returns (string memory) {
+ string memory ret = new string(self._len);
+ uint256 retptr;
+ assembly {
+ retptr := add(ret, 32)
+ }
+
+ memcpy(retptr, self._ptr, self._len);
+ return ret;
+ }
+
+ /*
+ * @dev Returns the length in runes of the slice. Note that this operation
+ * takes time proportional to the length of the slice; avoid using it
+ * in loops, and call `slice.empty()` if you only need to know whether
+ * the slice is empty or not.
+ * @param self The slice to operate on.
+ * @return The length of the slice in runes.
+ */
+ function len(slice memory self) internal pure returns (uint256 l) {
+ // Starting at ptr-31 means the LSB will be the byte we care about
+ uint256 ptr = self._ptr - 31;
+ uint256 end = ptr + self._len;
+ for (l = 0; ptr < end; l++) {
+ uint8 b;
+ assembly {
+ b := and(mload(ptr), 0xFF)
+ }
+ if (b < 0x80) {
+ ptr += 1;
+ } else if (b < 0xE0) {
+ ptr += 2;
+ } else if (b < 0xF0) {
+ ptr += 3;
+ } else if (b < 0xF8) {
+ ptr += 4;
+ } else if (b < 0xFC) {
+ ptr += 5;
+ } else {
+ ptr += 6;
+ }
+ }
+ }
+
+ /*
+ * @dev Returns true if the slice is empty (has a length of 0).
+ * @param self The slice to operate on.
+ * @return True if the slice is empty, False otherwise.
+ */
+ function empty(slice memory self) internal pure returns (bool) {
+ return self._len == 0;
+ }
+
+ /*
+ * @dev Returns a positive number if `other` comes lexicographically after
+ * `self`, a negative number if it comes before, or zero if the
+ * contents of the two slices are equal. Comparison is done per-rune,
+ * on unicode codepoints.
+ * @param self The first slice to compare.
+ * @param other The second slice to compare.
+ * @return The result of the comparison.
+ */
+ function compare(slice memory self, slice memory other) internal pure returns (int256) {
+ uint256 shortest = self._len;
+ if (other._len < self._len) shortest = other._len;
+
+ uint256 selfptr = self._ptr;
+ uint256 otherptr = other._ptr;
+ for (uint256 idx = 0; idx < shortest; idx += 32) {
+ uint256 a;
+ uint256 b;
+ assembly {
+ a := mload(selfptr)
+ b := mload(otherptr)
+ }
+ if (a != b) {
+ // Mask out irrelevant bytes and check again
+ uint256 mask = type(uint256).max; // 0xffff...
+ if (shortest < 32) {
+ mask = ~(2 ** (8 * (32 - shortest + idx)) - 1);
+ }
+ unchecked {
+ uint256 diff = (a & mask) - (b & mask);
+ if (diff != 0) return int256(diff);
+ }
+ }
+ selfptr += 32;
+ otherptr += 32;
+ }
+ return int256(self._len) - int256(other._len);
+ }
+
+ /*
+ * @dev Returns true if the two slices contain the same text.
+ * @param self The first slice to compare.
+ * @param self The second slice to compare.
+ * @return True if the slices are equal, false otherwise.
+ */
+ function equals(slice memory self, slice memory other) internal pure returns (bool) {
+ return compare(self, other) == 0;
+ }
+
+ /*
+ * @dev Extracts the first rune in the slice into `rune`, advancing the
+ * slice to point to the next rune and returning `self`.
+ * @param self The slice to operate on.
+ * @param rune The slice that will contain the first rune.
+ * @return `rune`.
+ */
+ function nextRune(slice memory self, slice memory rune) internal pure returns (slice memory) {
+ rune._ptr = self._ptr;
+
+ if (self._len == 0) {
+ rune._len = 0;
+ return rune;
+ }
+
+ uint256 l;
+ uint256 b;
+ // Load the first byte of the rune into the LSBs of b
+ assembly {
+ b := and(mload(sub(mload(add(self, 32)), 31)), 0xFF)
+ }
+ if (b < 0x80) {
+ l = 1;
+ } else if (b < 0xE0) {
+ l = 2;
+ } else if (b < 0xF0) {
+ l = 3;
+ } else {
+ l = 4;
+ }
+
+ // Check for truncated codepoints
+ if (l > self._len) {
+ rune._len = self._len;
+ self._ptr += self._len;
+ self._len = 0;
+ return rune;
+ }
+
+ self._ptr += l;
+ self._len -= l;
+ rune._len = l;
+ return rune;
+ }
+
+ /*
+ * @dev Returns the first rune in the slice, advancing the slice to point
+ * to the next rune.
+ * @param self The slice to operate on.
+ * @return A slice containing only the first rune from `self`.
+ */
+ function nextRune(slice memory self) internal pure returns (slice memory ret) {
+ nextRune(self, ret);
+ }
+
+ /*
+ * @dev Returns the number of the first codepoint in the slice.
+ * @param self The slice to operate on.
+ * @return The number of the first codepoint in the slice.
+ */
+ function ord(slice memory self) internal pure returns (uint256 ret) {
+ if (self._len == 0) {
+ return 0;
+ }
+
+ uint256 word;
+ uint256 length;
+ uint256 divisor = 2 ** 248;
+
+ // Load the rune into the MSBs of b
+ assembly {
+ word := mload(mload(add(self, 32)))
+ }
+ uint256 b = word / divisor;
+ if (b < 0x80) {
+ ret = b;
+ length = 1;
+ } else if (b < 0xE0) {
+ ret = b & 0x1F;
+ length = 2;
+ } else if (b < 0xF0) {
+ ret = b & 0x0F;
+ length = 3;
+ } else {
+ ret = b & 0x07;
+ length = 4;
+ }
+
+ // Check for truncated codepoints
+ if (length > self._len) {
+ return 0;
+ }
+
+ for (uint256 i = 1; i < length; ++i) {
+ divisor = divisor / 256;
+ b = (word / divisor) & 0xFF;
+ if (b & 0xC0 != 0x80) {
+ // Invalid UTF-8 sequence
+ return 0;
+ }
+ ret = (ret * 64) | (b & 0x3F);
+ }
+
+ return ret;
+ }
+
+ /*
+ * @dev Returns the keccak-256 hash of the slice.
+ * @param self The slice to hash.
+ * @return The hash of the slice.
+ */
+ function keccak(slice memory self) internal pure returns (bytes32 ret) {
+ assembly {
+ ret := keccak256(mload(add(self, 32)), mload(self))
+ }
+ }
+
+ /*
+ * @dev Returns true if `self` starts with `needle`.
+ * @param self The slice to operate on.
+ * @param needle The slice to search for.
+ * @return True if the slice starts with the provided text, false otherwise.
+ */
+ function startsWith(slice memory self, slice memory needle) internal pure returns (bool) {
+ if (self._len < needle._len) {
+ return false;
+ }
+
+ if (self._ptr == needle._ptr) {
+ return true;
+ }
+
+ bool equal;
+ assembly {
+ let length := mload(needle)
+ let selfptr := mload(add(self, 0x20))
+ let needleptr := mload(add(needle, 0x20))
+ equal := eq(keccak256(selfptr, length), keccak256(needleptr, length))
+ }
+ return equal;
+ }
+
+ /*
+ * @dev If `self` starts with `needle`, `needle` is removed from the
+ * beginning of `self`. Otherwise, `self` is unmodified.
+ * @param self The slice to operate on.
+ * @param needle The slice to search for.
+ * @return `self`
+ */
+ function beyond(slice memory self, slice memory needle) internal pure returns (slice memory) {
+ if (self._len < needle._len) {
+ return self;
+ }
+
+ bool equal = true;
+ if (self._ptr != needle._ptr) {
+ assembly {
+ let length := mload(needle)
+ let selfptr := mload(add(self, 0x20))
+ let needleptr := mload(add(needle, 0x20))
+ equal := eq(keccak256(selfptr, length), keccak256(needleptr, length))
+ }
+ }
+
+ if (equal) {
+ self._len -= needle._len;
+ self._ptr += needle._len;
+ }
+
+ return self;
+ }
+
+ /*
+ * @dev Returns true if the slice ends with `needle`.
+ * @param self The slice to operate on.
+ * @param needle The slice to search for.
+ * @return True if the slice starts with the provided text, false otherwise.
+ */
+ function endsWith(slice memory self, slice memory needle) internal pure returns (bool) {
+ if (self._len < needle._len) {
+ return false;
+ }
+
+ uint256 selfptr = self._ptr + self._len - needle._len;
+
+ if (selfptr == needle._ptr) {
+ return true;
+ }
+
+ bool equal;
+ assembly {
+ let length := mload(needle)
+ let needleptr := mload(add(needle, 0x20))
+ equal := eq(keccak256(selfptr, length), keccak256(needleptr, length))
+ }
+
+ return equal;
+ }
+
+ /*
+ * @dev If `self` ends with `needle`, `needle` is removed from the
+ * end of `self`. Otherwise, `self` is unmodified.
+ * @param self The slice to operate on.
+ * @param needle The slice to search for.
+ * @return `self`
+ */
+ function until(slice memory self, slice memory needle) internal pure returns (slice memory) {
+ if (self._len < needle._len) {
+ return self;
+ }
+
+ uint256 selfptr = self._ptr + self._len - needle._len;
+ bool equal = true;
+ if (selfptr != needle._ptr) {
+ assembly {
+ let length := mload(needle)
+ let needleptr := mload(add(needle, 0x20))
+ equal := eq(keccak256(selfptr, length), keccak256(needleptr, length))
+ }
+ }
+
+ if (equal) {
+ self._len -= needle._len;
+ }
+
+ return self;
+ }
+
+ // Returns the memory address of the first byte of the first occurrence of
+ // `needle` in `self`, or the first byte after `self` if not found.
+ function findPtr(
+ uint256 selflen,
+ uint256 selfptr,
+ uint256 needlelen,
+ uint256 needleptr
+ ) private pure returns (uint256) {
+ uint256 ptr = selfptr;
+ uint256 idx;
+
+ if (needlelen <= selflen) {
+ if (needlelen <= 32) {
+ bytes32 mask;
+ if (needlelen > 0) {
+ mask = bytes32(~(2 ** (8 * (32 - needlelen)) - 1));
+ }
+
+ bytes32 needledata;
+ assembly {
+ needledata := and(mload(needleptr), mask)
+ }
+
+ uint256 end = selfptr + selflen - needlelen;
+ bytes32 ptrdata;
+ assembly {
+ ptrdata := and(mload(ptr), mask)
+ }
+
+ while (ptrdata != needledata) {
+ if (ptr >= end) return selfptr + selflen;
+ ptr++;
+ assembly {
+ ptrdata := and(mload(ptr), mask)
+ }
+ }
+ return ptr;
+ } else {
+ // For long needles, use hashing
+ bytes32 hash;
+ assembly {
+ hash := keccak256(needleptr, needlelen)
+ }
+
+ for (idx = 0; idx <= selflen - needlelen; idx++) {
+ bytes32 testHash;
+ assembly {
+ testHash := keccak256(ptr, needlelen)
+ }
+ if (hash == testHash) return ptr;
+ ptr += 1;
+ }
+ }
+ }
+ return selfptr + selflen;
+ }
+
+ // Returns the memory address of the first byte after the last occurrence of
+ // `needle` in `self`, or the address of `self` if not found.
+ function rfindPtr(
+ uint256 selflen,
+ uint256 selfptr,
+ uint256 needlelen,
+ uint256 needleptr
+ ) private pure returns (uint256) {
+ uint256 ptr;
+
+ if (needlelen <= selflen) {
+ if (needlelen <= 32) {
+ bytes32 mask;
+ if (needlelen > 0) {
+ mask = bytes32(~(2 ** (8 * (32 - needlelen)) - 1));
+ }
+
+ bytes32 needledata;
+ assembly {
+ needledata := and(mload(needleptr), mask)
+ }
+
+ ptr = selfptr + selflen - needlelen;
+ bytes32 ptrdata;
+ assembly {
+ ptrdata := and(mload(ptr), mask)
+ }
+
+ while (ptrdata != needledata) {
+ if (ptr <= selfptr) return selfptr;
+ ptr--;
+ assembly {
+ ptrdata := and(mload(ptr), mask)
+ }
+ }
+ return ptr + needlelen;
+ } else {
+ // For long needles, use hashing
+ bytes32 hash;
+ assembly {
+ hash := keccak256(needleptr, needlelen)
+ }
+ ptr = selfptr + (selflen - needlelen);
+ while (ptr >= selfptr) {
+ bytes32 testHash;
+ assembly {
+ testHash := keccak256(ptr, needlelen)
+ }
+ if (hash == testHash) return ptr + needlelen;
+ ptr -= 1;
+ }
+ }
+ }
+ return selfptr;
+ }
+
+ /*
+ * @dev Modifies `self` to contain everything from the first occurrence of
+ * `needle` to the end of the slice. `self` is set to the empty slice
+ * if `needle` is not found.
+ * @param self The slice to search and modify.
+ * @param needle The text to search for.
+ * @return `self`.
+ */
+ function find(slice memory self, slice memory needle) internal pure returns (slice memory) {
+ uint256 ptr = findPtr(self._len, self._ptr, needle._len, needle._ptr);
+ self._len -= ptr - self._ptr;
+ self._ptr = ptr;
+ return self;
+ }
+
+ /*
+ * @dev Modifies `self` to contain the part of the string from the start of
+ * `self` to the end of the first occurrence of `needle`. If `needle`
+ * is not found, `self` is set to the empty slice.
+ * @param self The slice to search and modify.
+ * @param needle The text to search for.
+ * @return `self`.
+ */
+ function rfind(slice memory self, slice memory needle) internal pure returns (slice memory) {
+ uint256 ptr = rfindPtr(self._len, self._ptr, needle._len, needle._ptr);
+ self._len = ptr - self._ptr;
+ return self;
+ }
+
+ /*
+ * @dev Splits the slice, setting `self` to everything after the first
+ * occurrence of `needle`, and `token` to everything before it. If
+ * `needle` does not occur in `self`, `self` is set to the empty slice,
+ * and `token` is set to the entirety of `self`.
+ * @param self The slice to split.
+ * @param needle The text to search for in `self`.
+ * @param token An output parameter to which the first token is written.
+ * @return `token`.
+ */
+ function split(slice memory self, slice memory needle, slice memory token) internal pure returns (slice memory) {
+ uint256 ptr = findPtr(self._len, self._ptr, needle._len, needle._ptr);
+ token._ptr = self._ptr;
+ token._len = ptr - self._ptr;
+ if (ptr == self._ptr + self._len) {
+ // Not found
+ self._len = 0;
+ } else {
+ self._len -= token._len + needle._len;
+ self._ptr = ptr + needle._len;
+ }
+ return token;
+ }
+
+ /*
+ * @dev Splits the slice, setting `self` to everything after the first
+ * occurrence of `needle`, and returning everything before it. If
+ * `needle` does not occur in `self`, `self` is set to the empty slice,
+ * and the entirety of `self` is returned.
+ * @param self The slice to split.
+ * @param needle The text to search for in `self`.
+ * @return The part of `self` up to the first occurrence of `delim`.
+ */
+ function split(slice memory self, slice memory needle) internal pure returns (slice memory token) {
+ split(self, needle, token);
+ }
+
+ /*
+ * @dev Splits the slice, setting `self` to everything before the last
+ * occurrence of `needle`, and `token` to everything after it. If
+ * `needle` does not occur in `self`, `self` is set to the empty slice,
+ * and `token` is set to the entirety of `self`.
+ * @param self The slice to split.
+ * @param needle The text to search for in `self`.
+ * @param token An output parameter to which the first token is written.
+ * @return `token`.
+ */
+ function rsplit(slice memory self, slice memory needle, slice memory token) internal pure returns (slice memory) {
+ uint256 ptr = rfindPtr(self._len, self._ptr, needle._len, needle._ptr);
+ token._ptr = ptr;
+ token._len = self._len - (ptr - self._ptr);
+ if (ptr == self._ptr) {
+ // Not found
+ self._len = 0;
+ } else {
+ self._len -= token._len + needle._len;
+ }
+ return token;
+ }
+
+ /*
+ * @dev Splits the slice, setting `self` to everything before the last
+ * occurrence of `needle`, and returning everything after it. If
+ * `needle` does not occur in `self`, `self` is set to the empty slice,
+ * and the entirety of `self` is returned.
+ * @param self The slice to split.
+ * @param needle The text to search for in `self`.
+ * @return The part of `self` after the last occurrence of `delim`.
+ */
+ function rsplit(slice memory self, slice memory needle) internal pure returns (slice memory token) {
+ rsplit(self, needle, token);
+ }
+
+ /*
+ * @dev Counts the number of nonoverlapping occurrences of `needle` in `self`.
+ * @param self The slice to search.
+ * @param needle The text to search for in `self`.
+ * @return The number of occurrences of `needle` found in `self`.
+ */
+ function count(slice memory self, slice memory needle) internal pure returns (uint256 cnt) {
+ uint256 ptr = findPtr(self._len, self._ptr, needle._len, needle._ptr) + needle._len;
+ while (ptr <= self._ptr + self._len) {
+ cnt++;
+ ptr = findPtr(self._len - (ptr - self._ptr), ptr, needle._len, needle._ptr) + needle._len;
+ }
+ }
+
+ /*
+ * @dev Returns True if `self` contains `needle`.
+ * @param self The slice to search.
+ * @param needle The text to search for in `self`.
+ * @return True if `needle` is found in `self`, false otherwise.
+ */
+ function contains(slice memory self, slice memory needle) internal pure returns (bool) {
+ return rfindPtr(self._len, self._ptr, needle._len, needle._ptr) != self._ptr;
+ }
+
+ /*
+ * @dev Returns a newly allocated string containing the concatenation of
+ * `self` and `other`.
+ * @param self The first slice to concatenate.
+ * @param other The second slice to concatenate.
+ * @return The concatenation of the two strings.
+ */
+ function concat(slice memory self, slice memory other) internal pure returns (string memory) {
+ string memory ret = new string(self._len + other._len);
+ uint256 retptr;
+ assembly {
+ retptr := add(ret, 32)
+ }
+ memcpy(retptr, self._ptr, self._len);
+ memcpy(retptr + self._len, other._ptr, other._len);
+ return ret;
+ }
+
+ /*
+ * @dev Joins an array of slices, using `self` as a delimiter, returning a
+ * newly allocated string.
+ * @param self The delimiter to use.
+ * @param parts A list of slices to join.
+ * @return A newly allocated string containing all the slices in `parts`,
+ * joined with `self`.
+ */
+ function join(slice memory self, slice[] memory parts) internal pure returns (string memory) {
+ if (parts.length == 0) return "";
+
+ uint256 length = self._len * (parts.length - 1);
+ for (uint256 i; i < parts.length; ++i) length += parts[i]._len;
+
+ string memory ret = new string(length);
+ uint256 retptr;
+ assembly {
+ retptr := add(ret, 32)
+ }
+
+ for (uint256 i; i < parts.length; ++i) {
+ memcpy(retptr, parts[i]._ptr, parts[i]._len);
+ retptr += parts[i]._len;
+ if (i < parts.length - 1) {
+ memcpy(retptr, self._ptr, self._len);
+ retptr += self._len;
+ }
+ }
+
+ return ret;
+ }
+}
diff --git a/utils/forwardUtils.js b/utils/forwardUtils.js
new file mode 100755
index 0000000..60e7e28
--- /dev/null
+++ b/utils/forwardUtils.js
@@ -0,0 +1,22 @@
+const { exec } = require("child_process");
+
+if (process.argv.length < 3) {
+ console.error('Please provide a chain input as an argument.');
+ process.exit(1);
+}
+
+const command = process.argv[2];
+const extraArgs = process.argv.slice(3).join(' ');
+
+
+exec(`node lib/utils/utils/${command}.js ${extraArgs}`, (error, stdout, stderr) => {
+ if (error) {
+ console.log(error);
+ return;
+ }
+ if (stderr) {
+ console.log(stderr);
+ return;
+ }
+ console.log(stdout);
+});
diff --git a/yarn.lock b/yarn.lock
new file mode 100644
index 0000000..1edf638
--- /dev/null
+++ b/yarn.lock
@@ -0,0 +1,1691 @@
+# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
+# yarn lockfile v1
+
+
+"@angleprotocol/sdk@0.38.8":
+ version "0.38.8"
+ resolved "https://npm.pkg.github.com/download/@angleprotocol/sdk/0.38.8/dde9467c39c877b5c928791f2e1f7a2c5e92cbbb#dde9467c39c877b5c928791f2e1f7a2c5e92cbbb"
+ integrity sha512-mHosGBuF03jXcCTWUqsP6BV7ukoKCEugWC0zIHdTXboSERrWj9T36xlecwjO+6PnbbCS16oc/9OBUmlH7nrqZQ==
+ dependencies:
+ "@apollo/client" "^3.7.17"
+ "@typechain/ethers-v5" "^10.0.0"
+ "@types/lodash" "^4.14.180"
+ ethers "^5.6.4"
+ graphql "^15.7.1"
+ graphql-request "^3.6.1"
+ jsbi "^4.3.0"
+ keccak256 "^1.0.6"
+ lodash "^4.17.21"
+ merkletreejs "^0.3.10"
+ tiny-invariant "^1.1.0"
+ typechain "^8.3.2"
+
+"@apollo/client@^3.7.17":
+ version "3.9.11"
+ resolved "https://registry.yarnpkg.com/@apollo/client/-/client-3.9.11.tgz#737e5c35c21d6f3b78423033ad81837a8a6992e0"
+ integrity sha512-H7e9m7cRcFO93tokwzqrsbnfKorkpV24xU30hFH5u2g6B+c1DMo/ouyF/YrBPdrTzqxQCjTUmds/FLmJ7626GA==
+ dependencies:
+ "@graphql-typed-document-node/core" "^3.1.1"
+ "@wry/caches" "^1.0.0"
+ "@wry/equality" "^0.5.6"
+ "@wry/trie" "^0.5.0"
+ graphql-tag "^2.12.6"
+ hoist-non-react-statics "^3.3.2"
+ optimism "^0.18.0"
+ prop-types "^15.7.2"
+ rehackt "0.0.6"
+ response-iterator "^0.2.6"
+ symbol-observable "^4.0.0"
+ ts-invariant "^0.10.3"
+ tslib "^2.3.0"
+ zen-observable-ts "^1.2.5"
+
+"@babel/code-frame@^7.0.0":
+ version "7.24.2"
+ resolved "https://registry.yarnpkg.com/@babel/code-frame/-/code-frame-7.24.2.tgz#718b4b19841809a58b29b68cde80bc5e1aa6d9ae"
+ integrity sha512-y5+tLQyV8pg3fsiln67BVLD1P13Eg4lh5RW9mF0zUuvLrv9uIQ4MCL+CRT+FTsBlBjcIan6PGsLcBN0m3ClUyQ==
+ dependencies:
+ "@babel/highlight" "^7.24.2"
+ picocolors "^1.0.0"
+
+"@babel/helper-validator-identifier@^7.22.20":
+ version "7.22.20"
+ resolved "https://registry.yarnpkg.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.22.20.tgz#c4ae002c61d2879e724581d96665583dbc1dc0e0"
+ integrity sha512-Y4OZ+ytlatR8AI+8KZfKuL5urKp7qey08ha31L8b3BwewJAoJamTzyvxPR/5D+KkdJCGPq/+8TukHBlY10FX9A==
+
+"@babel/highlight@^7.24.2":
+ version "7.24.2"
+ resolved "https://registry.yarnpkg.com/@babel/highlight/-/highlight-7.24.2.tgz#3f539503efc83d3c59080a10e6634306e0370d26"
+ integrity sha512-Yac1ao4flkTxTteCDZLEvdxg2fZfz1v8M4QpaGypq/WPDqg3ijHYbDfs+LG5hvzSoqaSZ9/Z9lKSP3CjZjv+pA==
+ dependencies:
+ "@babel/helper-validator-identifier" "^7.22.20"
+ chalk "^2.4.2"
+ js-tokens "^4.0.0"
+ picocolors "^1.0.0"
+
+"@ethereumjs/rlp@^4.0.1":
+ version "4.0.1"
+ resolved "https://registry.yarnpkg.com/@ethereumjs/rlp/-/rlp-4.0.1.tgz#626fabfd9081baab3d0a3074b0c7ecaf674aaa41"
+ integrity sha512-tqsQiBQDQdmPWE1xkkBq4rlSW5QZpLOUJ5RJh2/9fug+q9tnUhuZoVLk7s0scUIKTOzEtR72DFBXI4WiZcMpvw==
+
+"@ethereumjs/util@^8.1.0":
+ version "8.1.0"
+ resolved "https://registry.yarnpkg.com/@ethereumjs/util/-/util-8.1.0.tgz#299df97fb6b034e0577ce9f94c7d9d1004409ed4"
+ integrity sha512-zQ0IqbdX8FZ9aw11vP+dZkKDkS+kgIvQPHnSAXzP9pLu+Rfu3D3XEeLbicvoXJTYnhZiPmsZUxgdzXwNKxRPbA==
+ dependencies:
+ "@ethereumjs/rlp" "^4.0.1"
+ ethereum-cryptography "^2.0.0"
+ micro-ftch "^0.3.1"
+
+"@ethersproject/abi@5.7.0", "@ethersproject/abi@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/abi/-/abi-5.7.0.tgz#b3f3e045bbbeed1af3947335c247ad625a44e449"
+ integrity sha512-351ktp42TiRcYB3H1OP8yajPeAQstMW/yCFokj/AthP9bLHzQFPlOrxOcwYEDkUAICmOHljvN4K39OMTMUa9RA==
+ dependencies:
+ "@ethersproject/address" "^5.7.0"
+ "@ethersproject/bignumber" "^5.7.0"
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/constants" "^5.7.0"
+ "@ethersproject/hash" "^5.7.0"
+ "@ethersproject/keccak256" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ "@ethersproject/properties" "^5.7.0"
+ "@ethersproject/strings" "^5.7.0"
+
+"@ethersproject/abstract-provider@5.7.0", "@ethersproject/abstract-provider@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/abstract-provider/-/abstract-provider-5.7.0.tgz#b0a8550f88b6bf9d51f90e4795d48294630cb9ef"
+ integrity sha512-R41c9UkchKCpAqStMYUpdunjo3pkEvZC3FAwZn5S5MGbXoMQOHIdHItezTETxAO5bevtMApSyEhn9+CHcDsWBw==
+ dependencies:
+ "@ethersproject/bignumber" "^5.7.0"
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ "@ethersproject/networks" "^5.7.0"
+ "@ethersproject/properties" "^5.7.0"
+ "@ethersproject/transactions" "^5.7.0"
+ "@ethersproject/web" "^5.7.0"
+
+"@ethersproject/abstract-signer@5.7.0", "@ethersproject/abstract-signer@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz#13f4f32117868452191a4649723cb086d2b596b2"
+ integrity sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==
+ dependencies:
+ "@ethersproject/abstract-provider" "^5.7.0"
+ "@ethersproject/bignumber" "^5.7.0"
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ "@ethersproject/properties" "^5.7.0"
+
+"@ethersproject/address@5.7.0", "@ethersproject/address@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/address/-/address-5.7.0.tgz#19b56c4d74a3b0a46bfdbb6cfcc0a153fc697f37"
+ integrity sha512-9wYhYt7aghVGo758POM5nqcOMaE168Q6aRLJZwUmiqSrAungkG74gSSeKEIR7ukixesdRZGPgVqme6vmxs1fkA==
+ dependencies:
+ "@ethersproject/bignumber" "^5.7.0"
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/keccak256" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ "@ethersproject/rlp" "^5.7.0"
+
+"@ethersproject/base64@5.7.0", "@ethersproject/base64@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/base64/-/base64-5.7.0.tgz#ac4ee92aa36c1628173e221d0d01f53692059e1c"
+ integrity sha512-Dr8tcHt2mEbsZr/mwTPIQAf3Ai0Bks/7gTw9dSqk1mQvhW3XvRlmDJr/4n+wg1JmCl16NZue17CDh8xb/vZ0sQ==
+ dependencies:
+ "@ethersproject/bytes" "^5.7.0"
+
+"@ethersproject/basex@5.7.0", "@ethersproject/basex@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/basex/-/basex-5.7.0.tgz#97034dc7e8938a8ca943ab20f8a5e492ece4020b"
+ integrity sha512-ywlh43GwZLv2Voc2gQVTKBoVQ1mti3d8HK5aMxsfu/nRDnMmNqaSJ3r3n85HBByT8OpoY96SXM1FogC533T4zw==
+ dependencies:
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/properties" "^5.7.0"
+
+"@ethersproject/bignumber@5.7.0", "@ethersproject/bignumber@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/bignumber/-/bignumber-5.7.0.tgz#e2f03837f268ba655ffba03a57853e18a18dc9c2"
+ integrity sha512-n1CAdIHRWjSucQO3MC1zPSVgV/6dy/fjL9pMrPP9peL+QxEg9wOsVqwD4+818B6LUEtaXzVHQiuivzRoxPxUGw==
+ dependencies:
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ bn.js "^5.2.1"
+
+"@ethersproject/bytes@5.7.0", "@ethersproject/bytes@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/bytes/-/bytes-5.7.0.tgz#a00f6ea8d7e7534d6d87f47188af1148d71f155d"
+ integrity sha512-nsbxwgFXWh9NyYWo+U8atvmMsSdKJprTcICAkvbBffT75qDocbuggBU0SJiVK2MuTrp0q+xvLkTnGMPK1+uA9A==
+ dependencies:
+ "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/constants@5.7.0", "@ethersproject/constants@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/constants/-/constants-5.7.0.tgz#df80a9705a7e08984161f09014ea012d1c75295e"
+ integrity sha512-DHI+y5dBNvkpYUMiRQyxRBYBefZkJfo70VUkUAsRjcPs47muV9evftfZ0PJVCXYbAiCgght0DtcF9srFQmIgWA==
+ dependencies:
+ "@ethersproject/bignumber" "^5.7.0"
+
+"@ethersproject/contracts@5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/contracts/-/contracts-5.7.0.tgz#c305e775abd07e48aa590e1a877ed5c316f8bd1e"
+ integrity sha512-5GJbzEU3X+d33CdfPhcyS+z8MzsTrBGk/sc+G+59+tPa9yFkl6HQ9D6L0QMgNTA9q8dT0XKxxkyp883XsQvbbg==
+ dependencies:
+ "@ethersproject/abi" "^5.7.0"
+ "@ethersproject/abstract-provider" "^5.7.0"
+ "@ethersproject/abstract-signer" "^5.7.0"
+ "@ethersproject/address" "^5.7.0"
+ "@ethersproject/bignumber" "^5.7.0"
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/constants" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ "@ethersproject/properties" "^5.7.0"
+ "@ethersproject/transactions" "^5.7.0"
+
+"@ethersproject/hash@5.7.0", "@ethersproject/hash@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/hash/-/hash-5.7.0.tgz#eb7aca84a588508369562e16e514b539ba5240a7"
+ integrity sha512-qX5WrQfnah1EFnO5zJv1v46a8HW0+E5xuBBDTwMFZLuVTx0tbU2kkx15NqdjxecrLGatQN9FGQKpb1FKdHCt+g==
+ dependencies:
+ "@ethersproject/abstract-signer" "^5.7.0"
+ "@ethersproject/address" "^5.7.0"
+ "@ethersproject/base64" "^5.7.0"
+ "@ethersproject/bignumber" "^5.7.0"
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/keccak256" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ "@ethersproject/properties" "^5.7.0"
+ "@ethersproject/strings" "^5.7.0"
+
+"@ethersproject/hdnode@5.7.0", "@ethersproject/hdnode@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/hdnode/-/hdnode-5.7.0.tgz#e627ddc6b466bc77aebf1a6b9e47405ca5aef9cf"
+ integrity sha512-OmyYo9EENBPPf4ERhR7oj6uAtUAhYGqOnIS+jE5pTXvdKBS99ikzq1E7Iv0ZQZ5V36Lqx1qZLeak0Ra16qpeOg==
+ dependencies:
+ "@ethersproject/abstract-signer" "^5.7.0"
+ "@ethersproject/basex" "^5.7.0"
+ "@ethersproject/bignumber" "^5.7.0"
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ "@ethersproject/pbkdf2" "^5.7.0"
+ "@ethersproject/properties" "^5.7.0"
+ "@ethersproject/sha2" "^5.7.0"
+ "@ethersproject/signing-key" "^5.7.0"
+ "@ethersproject/strings" "^5.7.0"
+ "@ethersproject/transactions" "^5.7.0"
+ "@ethersproject/wordlists" "^5.7.0"
+
+"@ethersproject/json-wallets@5.7.0", "@ethersproject/json-wallets@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/json-wallets/-/json-wallets-5.7.0.tgz#5e3355287b548c32b368d91014919ebebddd5360"
+ integrity sha512-8oee5Xgu6+RKgJTkvEMl2wDgSPSAQ9MB/3JYjFV9jlKvcYHUXZC+cQp0njgmxdHkYWn8s6/IqIZYm0YWCjO/0g==
+ dependencies:
+ "@ethersproject/abstract-signer" "^5.7.0"
+ "@ethersproject/address" "^5.7.0"
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/hdnode" "^5.7.0"
+ "@ethersproject/keccak256" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ "@ethersproject/pbkdf2" "^5.7.0"
+ "@ethersproject/properties" "^5.7.0"
+ "@ethersproject/random" "^5.7.0"
+ "@ethersproject/strings" "^5.7.0"
+ "@ethersproject/transactions" "^5.7.0"
+ aes-js "3.0.0"
+ scrypt-js "3.0.1"
+
+"@ethersproject/keccak256@5.7.0", "@ethersproject/keccak256@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/keccak256/-/keccak256-5.7.0.tgz#3186350c6e1cd6aba7940384ec7d6d9db01f335a"
+ integrity sha512-2UcPboeL/iW+pSg6vZ6ydF8tCnv3Iu/8tUmLLzWWGzxWKFFqOBQFLo6uLUv6BDrLgCDfN28RJ/wtByx+jZ4KBg==
+ dependencies:
+ "@ethersproject/bytes" "^5.7.0"
+ js-sha3 "0.8.0"
+
+"@ethersproject/logger@5.7.0", "@ethersproject/logger@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/logger/-/logger-5.7.0.tgz#6ce9ae168e74fecf287be17062b590852c311892"
+ integrity sha512-0odtFdXu/XHtjQXJYA3u9G0G8btm0ND5Cu8M7i5vhEcE8/HmF4Lbdqanwyv4uQTr2tx6b7fQRmgLrsnpQlmnig==
+
+"@ethersproject/networks@5.7.1", "@ethersproject/networks@^5.7.0":
+ version "5.7.1"
+ resolved "https://registry.yarnpkg.com/@ethersproject/networks/-/networks-5.7.1.tgz#118e1a981d757d45ccea6bb58d9fd3d9db14ead6"
+ integrity sha512-n/MufjFYv3yFcUyfhnXotyDlNdFb7onmkSy8aQERi2PjNcnWQ66xXxa3XlS8nCcA8aJKJjIIMNJTC7tu80GwpQ==
+ dependencies:
+ "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/pbkdf2@5.7.0", "@ethersproject/pbkdf2@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/pbkdf2/-/pbkdf2-5.7.0.tgz#d2267d0a1f6e123f3771007338c47cccd83d3102"
+ integrity sha512-oR/dBRZR6GTyaofd86DehG72hY6NpAjhabkhxgr3X2FpJtJuodEl2auADWBZfhDHgVCbu3/H/Ocq2uC6dpNjjw==
+ dependencies:
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/sha2" "^5.7.0"
+
+"@ethersproject/properties@5.7.0", "@ethersproject/properties@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/properties/-/properties-5.7.0.tgz#a6e12cb0439b878aaf470f1902a176033067ed30"
+ integrity sha512-J87jy8suntrAkIZtecpxEPxY//szqr1mlBaYlQ0r4RCaiD2hjheqF9s1LVE8vVuJCXisjIP+JgtK/Do54ej4Sw==
+ dependencies:
+ "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/providers@5.7.2":
+ version "5.7.2"
+ resolved "https://registry.yarnpkg.com/@ethersproject/providers/-/providers-5.7.2.tgz#f8b1a4f275d7ce58cf0a2eec222269a08beb18cb"
+ integrity sha512-g34EWZ1WWAVgr4aptGlVBF8mhl3VWjv+8hoAnzStu8Ah22VHBsuGzP17eb6xDVRzw895G4W7vvx60lFFur/1Rg==
+ dependencies:
+ "@ethersproject/abstract-provider" "^5.7.0"
+ "@ethersproject/abstract-signer" "^5.7.0"
+ "@ethersproject/address" "^5.7.0"
+ "@ethersproject/base64" "^5.7.0"
+ "@ethersproject/basex" "^5.7.0"
+ "@ethersproject/bignumber" "^5.7.0"
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/constants" "^5.7.0"
+ "@ethersproject/hash" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ "@ethersproject/networks" "^5.7.0"
+ "@ethersproject/properties" "^5.7.0"
+ "@ethersproject/random" "^5.7.0"
+ "@ethersproject/rlp" "^5.7.0"
+ "@ethersproject/sha2" "^5.7.0"
+ "@ethersproject/strings" "^5.7.0"
+ "@ethersproject/transactions" "^5.7.0"
+ "@ethersproject/web" "^5.7.0"
+ bech32 "1.1.4"
+ ws "7.4.6"
+
+"@ethersproject/random@5.7.0", "@ethersproject/random@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/random/-/random-5.7.0.tgz#af19dcbc2484aae078bb03656ec05df66253280c"
+ integrity sha512-19WjScqRA8IIeWclFme75VMXSBvi4e6InrUNuaR4s5pTF2qNhcGdCUwdxUVGtDDqC00sDLCO93jPQoDUH4HVmQ==
+ dependencies:
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/rlp@5.7.0", "@ethersproject/rlp@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/rlp/-/rlp-5.7.0.tgz#de39e4d5918b9d74d46de93af80b7685a9c21304"
+ integrity sha512-rBxzX2vK8mVF7b0Tol44t5Tb8gomOHkj5guL+HhzQ1yBh/ydjGnpw6at+X6Iw0Kp3OzzzkcKp8N9r0W4kYSs9w==
+ dependencies:
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/sha2@5.7.0", "@ethersproject/sha2@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/sha2/-/sha2-5.7.0.tgz#9a5f7a7824ef784f7f7680984e593a800480c9fb"
+ integrity sha512-gKlH42riwb3KYp0reLsFTokByAKoJdgFCwI+CCiX/k+Jm2mbNs6oOaCjYQSlI1+XBVejwH2KrmCbMAT/GnRDQw==
+ dependencies:
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ hash.js "1.1.7"
+
+"@ethersproject/signing-key@5.7.0", "@ethersproject/signing-key@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/signing-key/-/signing-key-5.7.0.tgz#06b2df39411b00bc57c7c09b01d1e41cf1b16ab3"
+ integrity sha512-MZdy2nL3wO0u7gkB4nA/pEf8lu1TlFswPNmy8AiYkfKTdO6eXBJyUdmHO/ehm/htHw9K/qF8ujnTyUAD+Ry54Q==
+ dependencies:
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ "@ethersproject/properties" "^5.7.0"
+ bn.js "^5.2.1"
+ elliptic "6.5.4"
+ hash.js "1.1.7"
+
+"@ethersproject/solidity@5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/solidity/-/solidity-5.7.0.tgz#5e9c911d8a2acce2a5ebb48a5e2e0af20b631cb8"
+ integrity sha512-HmabMd2Dt/raavyaGukF4XxizWKhKQ24DoLtdNbBmNKUOPqwjsKQSdV9GQtj9CBEea9DlzETlVER1gYeXXBGaA==
+ dependencies:
+ "@ethersproject/bignumber" "^5.7.0"
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/keccak256" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ "@ethersproject/sha2" "^5.7.0"
+ "@ethersproject/strings" "^5.7.0"
+
+"@ethersproject/strings@5.7.0", "@ethersproject/strings@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/strings/-/strings-5.7.0.tgz#54c9d2a7c57ae8f1205c88a9d3a56471e14d5ed2"
+ integrity sha512-/9nu+lj0YswRNSH0NXYqrh8775XNyEdUQAuf3f+SmOrnVewcJ5SBNAjF7lpgehKi4abvNNXyf+HX86czCdJ8Mg==
+ dependencies:
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/constants" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/transactions@5.7.0", "@ethersproject/transactions@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/transactions/-/transactions-5.7.0.tgz#91318fc24063e057885a6af13fdb703e1f993d3b"
+ integrity sha512-kmcNicCp1lp8qanMTC3RIikGgoJ80ztTyvtsFvCYpSCfkjhD0jZ2LOrnbcuxuToLIUYYf+4XwD1rP+B/erDIhQ==
+ dependencies:
+ "@ethersproject/address" "^5.7.0"
+ "@ethersproject/bignumber" "^5.7.0"
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/constants" "^5.7.0"
+ "@ethersproject/keccak256" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ "@ethersproject/properties" "^5.7.0"
+ "@ethersproject/rlp" "^5.7.0"
+ "@ethersproject/signing-key" "^5.7.0"
+
+"@ethersproject/units@5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/units/-/units-5.7.0.tgz#637b563d7e14f42deeee39245275d477aae1d8b1"
+ integrity sha512-pD3xLMy3SJu9kG5xDGI7+xhTEmGXlEqXU4OfNapmfnxLVY4EMSSRp7j1k7eezutBPH7RBN/7QPnwR7hzNlEFeg==
+ dependencies:
+ "@ethersproject/bignumber" "^5.7.0"
+ "@ethersproject/constants" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/wallet@5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/wallet/-/wallet-5.7.0.tgz#4e5d0790d96fe21d61d38fb40324e6c7ef350b2d"
+ integrity sha512-MhmXlJXEJFBFVKrDLB4ZdDzxcBxQ3rLyCkhNqVu3CDYvR97E+8r01UgrI+TI99Le+aYm/in/0vp86guJuM7FCA==
+ dependencies:
+ "@ethersproject/abstract-provider" "^5.7.0"
+ "@ethersproject/abstract-signer" "^5.7.0"
+ "@ethersproject/address" "^5.7.0"
+ "@ethersproject/bignumber" "^5.7.0"
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/hash" "^5.7.0"
+ "@ethersproject/hdnode" "^5.7.0"
+ "@ethersproject/json-wallets" "^5.7.0"
+ "@ethersproject/keccak256" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ "@ethersproject/properties" "^5.7.0"
+ "@ethersproject/random" "^5.7.0"
+ "@ethersproject/signing-key" "^5.7.0"
+ "@ethersproject/transactions" "^5.7.0"
+ "@ethersproject/wordlists" "^5.7.0"
+
+"@ethersproject/web@5.7.1", "@ethersproject/web@^5.7.0":
+ version "5.7.1"
+ resolved "https://registry.yarnpkg.com/@ethersproject/web/-/web-5.7.1.tgz#de1f285b373149bee5928f4eb7bcb87ee5fbb4ae"
+ integrity sha512-Gueu8lSvyjBWL4cYsWsjh6MtMwM0+H4HvqFPZfB6dV8ctbP9zFAO73VG1cMWae0FLPCtz0peKPpZY8/ugJJX2w==
+ dependencies:
+ "@ethersproject/base64" "^5.7.0"
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ "@ethersproject/properties" "^5.7.0"
+ "@ethersproject/strings" "^5.7.0"
+
+"@ethersproject/wordlists@5.7.0", "@ethersproject/wordlists@^5.7.0":
+ version "5.7.0"
+ resolved "https://registry.yarnpkg.com/@ethersproject/wordlists/-/wordlists-5.7.0.tgz#8fb2c07185d68c3e09eb3bfd6e779ba2774627f5"
+ integrity sha512-S2TFNJNfHWVHNE6cNDjbVlZ6MgE17MIxMbMg2zv3wn+3XSJGosL1m9ZVv3GXCf/2ymSsQ+hRI5IzoMJTG6aoVA==
+ dependencies:
+ "@ethersproject/bytes" "^5.7.0"
+ "@ethersproject/hash" "^5.7.0"
+ "@ethersproject/logger" "^5.7.0"
+ "@ethersproject/properties" "^5.7.0"
+ "@ethersproject/strings" "^5.7.0"
+
+"@graphql-typed-document-node/core@^3.1.1":
+ version "3.2.0"
+ resolved "https://registry.yarnpkg.com/@graphql-typed-document-node/core/-/core-3.2.0.tgz#5f3d96ec6b2354ad6d8a28bf216a1d97b5426861"
+ integrity sha512-mB9oAsNCm9aM3/SOv4YtBMqZbYj10R7dkq8byBqxGY/ncFwhf2oQzMV+LCRlWoDSEBJ3COiR1yeDvMtsoOsuFQ==
+
+"@noble/curves@1.3.0", "@noble/curves@~1.3.0":
+ version "1.3.0"
+ resolved "https://registry.yarnpkg.com/@noble/curves/-/curves-1.3.0.tgz#01be46da4fd195822dab821e72f71bf4aeec635e"
+ integrity sha512-t01iSXPuN+Eqzb4eBX0S5oubSqXbK/xXa1Ne18Hj8f9pStxztHCE2gfboSp/dZRLSqfuLpRK2nDXDK+W9puocA==
+ dependencies:
+ "@noble/hashes" "1.3.3"
+
+"@noble/hashes@1.3.3", "@noble/hashes@~1.3.2":
+ version "1.3.3"
+ resolved "https://registry.yarnpkg.com/@noble/hashes/-/hashes-1.3.3.tgz#39908da56a4adc270147bb07968bf3b16cfe1699"
+ integrity sha512-V7/fPHgl+jsVPXqqeOzT8egNj2iBIVt+ECeMMG8TdcnTikP3oaBtUVqpT/gYCR68aEBJSF+XbYUxStjbFMqIIA==
+
+"@noble/hashes@^1.4.0":
+ version "1.4.0"
+ resolved "https://registry.yarnpkg.com/@noble/hashes/-/hashes-1.4.0.tgz#45814aa329f30e4fe0ba49426f49dfccdd066426"
+ integrity sha512-V1JJ1WTRUqHHrOSh597hURcMqVKVGL/ea3kv0gSnEdsEZ0/+VyPghM1lMNGc00z7CIQorSvbKpuJkxvuHbvdbg==
+
+"@scure/base@~1.1.4":
+ version "1.1.6"
+ resolved "https://registry.yarnpkg.com/@scure/base/-/base-1.1.6.tgz#8ce5d304b436e4c84f896e0550c83e4d88cb917d"
+ integrity sha512-ok9AWwhcgYuGG3Zfhyqg+zwl+Wn5uE+dwC0NV/2qQkx4dABbb/bx96vWu8NSj+BNjjSjno+JRYRjle1jV08k3g==
+
+"@scure/bip32@1.3.3":
+ version "1.3.3"
+ resolved "https://registry.yarnpkg.com/@scure/bip32/-/bip32-1.3.3.tgz#a9624991dc8767087c57999a5d79488f48eae6c8"
+ integrity sha512-LJaN3HwRbfQK0X1xFSi0Q9amqOgzQnnDngIt+ZlsBC3Bm7/nE7K0kwshZHyaru79yIVRv/e1mQAjZyuZG6jOFQ==
+ dependencies:
+ "@noble/curves" "~1.3.0"
+ "@noble/hashes" "~1.3.2"
+ "@scure/base" "~1.1.4"
+
+"@scure/bip39@1.2.2":
+ version "1.2.2"
+ resolved "https://registry.yarnpkg.com/@scure/bip39/-/bip39-1.2.2.tgz#f3426813f4ced11a47489cbcf7294aa963966527"
+ integrity sha512-HYf9TUXG80beW+hGAt3TRM8wU6pQoYur9iNypTROm42dorCGmLnFe3eWjz3gOq6G62H2WRh0FCzAR1PI+29zIA==
+ dependencies:
+ "@noble/hashes" "~1.3.2"
+ "@scure/base" "~1.1.4"
+
+"@solidity-parser/parser@^0.16.0":
+ version "0.16.2"
+ resolved "https://registry.yarnpkg.com/@solidity-parser/parser/-/parser-0.16.2.tgz#42cb1e3d88b3e8029b0c9befff00b634cd92d2fa"
+ integrity sha512-PI9NfoA3P8XK2VBkK5oIfRgKDsicwDZfkVq9ZTBCQYGOP1N2owgY2dyLGyU5/J/hQs8KRk55kdmvTLjy3Mu3vg==
+ dependencies:
+ antlr4ts "^0.5.0-alpha.4"
+
+"@solidity-parser/parser@^0.17.0":
+ version "0.17.0"
+ resolved "https://registry.yarnpkg.com/@solidity-parser/parser/-/parser-0.17.0.tgz#52a2fcc97ff609f72011014e4c5b485ec52243ef"
+ integrity sha512-Nko8R0/kUo391jsEHHxrGM07QFdnPGvlmox4rmH0kNiNAashItAilhy4Mv4pK5gQmW5f4sXAF58fwJbmlkGcVw==
+
+"@typechain/ethers-v5@^10.0.0":
+ version "10.2.1"
+ resolved "https://registry.yarnpkg.com/@typechain/ethers-v5/-/ethers-v5-10.2.1.tgz#50241e6957683281ecfa03fb5a6724d8a3ce2391"
+ integrity sha512-n3tQmCZjRE6IU4h6lqUGiQ1j866n5MTCBJreNEHHVWXa2u9GJTaeYyU1/k+1qLutkyw+sS6VAN+AbeiTqsxd/A==
+ dependencies:
+ lodash "^4.17.15"
+ ts-essentials "^7.0.1"
+
+"@types/lodash@^4.14.180":
+ version "4.17.0"
+ resolved "https://registry.yarnpkg.com/@types/lodash/-/lodash-4.17.0.tgz#d774355e41f372d5350a4d0714abb48194a489c3"
+ integrity sha512-t7dhREVv6dbNj0q17X12j7yDG4bD/DHYX7o5/DbDxobP0HnGPgpRz2Ej77aL7TZT3DSw13fqUTj8J4mMnqa7WA==
+
+"@types/prettier@^2.1.1":
+ version "2.7.3"
+ resolved "https://registry.yarnpkg.com/@types/prettier/-/prettier-2.7.3.tgz#3e51a17e291d01d17d3fc61422015a933af7a08f"
+ integrity sha512-+68kP9yzs4LMp7VNh8gdzMSPZFL44MLGqiHWvttYJe+6qnuVr4Ek9wSBQoveqY/r+LwjCcU29kNVkidwim+kYA==
+
+"@wry/caches@^1.0.0":
+ version "1.0.1"
+ resolved "https://registry.yarnpkg.com/@wry/caches/-/caches-1.0.1.tgz#8641fd3b6e09230b86ce8b93558d44cf1ece7e52"
+ integrity sha512-bXuaUNLVVkD20wcGBWRyo7j9N3TxePEWFZj2Y+r9OoUzfqmavM84+mFykRicNsBqatba5JLay1t48wxaXaWnlA==
+ dependencies:
+ tslib "^2.3.0"
+
+"@wry/context@^0.7.0":
+ version "0.7.4"
+ resolved "https://registry.yarnpkg.com/@wry/context/-/context-0.7.4.tgz#e32d750fa075955c4ab2cfb8c48095e1d42d5990"
+ integrity sha512-jmT7Sb4ZQWI5iyu3lobQxICu2nC/vbUhP0vIdd6tHC9PTfenmRmuIFqktc6GH9cgi+ZHnsLWPvfSvc4DrYmKiQ==
+ dependencies:
+ tslib "^2.3.0"
+
+"@wry/equality@^0.5.6":
+ version "0.5.7"
+ resolved "https://registry.yarnpkg.com/@wry/equality/-/equality-0.5.7.tgz#72ec1a73760943d439d56b7b1e9985aec5d497bb"
+ integrity sha512-BRFORjsTuQv5gxcXsuDXx6oGRhuVsEGwZy6LOzRRfgu+eSfxbhUQ9L9YtSEIuIjY/o7g3iWFjrc5eSY1GXP2Dw==
+ dependencies:
+ tslib "^2.3.0"
+
+"@wry/trie@^0.4.3":
+ version "0.4.3"
+ resolved "https://registry.yarnpkg.com/@wry/trie/-/trie-0.4.3.tgz#077d52c22365871bf3ffcbab8e95cb8bc5689af4"
+ integrity sha512-I6bHwH0fSf6RqQcnnXLJKhkSXG45MFral3GxPaY4uAl0LYDZM+YDVDAiU9bYwjTuysy1S0IeecWtmq1SZA3M1w==
+ dependencies:
+ tslib "^2.3.0"
+
+"@wry/trie@^0.5.0":
+ version "0.5.0"
+ resolved "https://registry.yarnpkg.com/@wry/trie/-/trie-0.5.0.tgz#11e783f3a53f6e4cd1d42d2d1323f5bc3fa99c94"
+ integrity sha512-FNoYzHawTMk/6KMQoEG5O4PuioX19UbwdQKF44yw0nLfOypfQdjtfZzo/UIJWAJ23sNIFbD1Ug9lbaDGMwbqQA==
+ dependencies:
+ tslib "^2.3.0"
+
+aes-js@3.0.0:
+ version "3.0.0"
+ resolved "https://registry.yarnpkg.com/aes-js/-/aes-js-3.0.0.tgz#e21df10ad6c2053295bcbb8dab40b09dbea87e4d"
+ integrity sha512-H7wUZRn8WpTq9jocdxQ2c8x2sKo9ZVmzfRE13GiNJXfp7NcKYEdvl3vspKjXox6RIG2VtaRe4JFvxG4rqp2Zuw==
+
+ajv@^6.12.6:
+ version "6.12.6"
+ resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.12.6.tgz#baf5a62e802b07d977034586f8c3baf5adf26df4"
+ integrity sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==
+ dependencies:
+ fast-deep-equal "^3.1.1"
+ fast-json-stable-stringify "^2.0.0"
+ json-schema-traverse "^0.4.1"
+ uri-js "^4.2.2"
+
+ajv@^8.0.1:
+ version "8.12.0"
+ resolved "https://registry.yarnpkg.com/ajv/-/ajv-8.12.0.tgz#d1a0527323e22f53562c567c00991577dfbe19d1"
+ integrity sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==
+ dependencies:
+ fast-deep-equal "^3.1.1"
+ json-schema-traverse "^1.0.0"
+ require-from-string "^2.0.2"
+ uri-js "^4.2.2"
+
+ansi-regex@^5.0.1:
+ version "5.0.1"
+ resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.1.tgz#082cb2c89c9fe8659a311a53bd6a4dc5301db304"
+ integrity sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==
+
+ansi-styles@^3.2.1:
+ version "3.2.1"
+ resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-3.2.1.tgz#41fbb20243e50b12be0f04b8dedbf07520ce841d"
+ integrity sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==
+ dependencies:
+ color-convert "^1.9.0"
+
+ansi-styles@^4.0.0, ansi-styles@^4.1.0:
+ version "4.3.0"
+ resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.3.0.tgz#edd803628ae71c04c85ae7a0906edad34b648937"
+ integrity sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==
+ dependencies:
+ color-convert "^2.0.1"
+
+antlr4@^4.11.0:
+ version "4.13.1"
+ resolved "https://registry.yarnpkg.com/antlr4/-/antlr4-4.13.1.tgz#1e0a1830a08faeb86217cb2e6c34716004e4253d"
+ integrity sha512-kiXTspaRYvnIArgE97z5YVVf/cDVQABr3abFRR6mE7yesLMkgu4ujuyV/sgxafQ8wgve0DJQUJ38Z8tkgA2izA==
+
+antlr4ts@^0.5.0-alpha.4:
+ version "0.5.0-alpha.4"
+ resolved "https://registry.yarnpkg.com/antlr4ts/-/antlr4ts-0.5.0-alpha.4.tgz#71702865a87478ed0b40c0709f422cf14d51652a"
+ integrity sha512-WPQDt1B74OfPv/IMS2ekXAKkTZIHl88uMetg6q3OTqgFxZ/dxDXI0EWLyZid/1Pe6hTftyg5N7gel5wNAGxXyQ==
+
+argparse@^2.0.1:
+ version "2.0.1"
+ resolved "https://registry.yarnpkg.com/argparse/-/argparse-2.0.1.tgz#246f50f3ca78a3240f6c997e8a9bd1eac49e4b38"
+ integrity sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==
+
+array-back@^3.0.1, array-back@^3.1.0:
+ version "3.1.0"
+ resolved "https://registry.yarnpkg.com/array-back/-/array-back-3.1.0.tgz#b8859d7a508871c9a7b2cf42f99428f65e96bfb0"
+ integrity sha512-TkuxA4UCOvxuDK6NZYXCalszEzj+TLszyASooky+i742l9TqsOdYCMJJupxRic61hwquNtppB3hgcuq9SVSH1Q==
+
+array-back@^4.0.1, array-back@^4.0.2:
+ version "4.0.2"
+ resolved "https://registry.yarnpkg.com/array-back/-/array-back-4.0.2.tgz#8004e999a6274586beeb27342168652fdb89fa1e"
+ integrity sha512-NbdMezxqf94cnNfWLL7V/im0Ub+Anbb0IoZhvzie8+4HJ4nMQuzHuy49FkGYCJK2yAloZ3meiB6AVMClbrI1vg==
+
+ast-parents@^0.0.1:
+ version "0.0.1"
+ resolved "https://registry.yarnpkg.com/ast-parents/-/ast-parents-0.0.1.tgz#508fd0f05d0c48775d9eccda2e174423261e8dd3"
+ integrity sha512-XHusKxKz3zoYk1ic8Un640joHbFMhbqneyoZfoKnEGtf2ey9Uh/IdpcQplODdO/kENaMIWsD0nJm4+wX3UNLHA==
+
+astral-regex@^2.0.0:
+ version "2.0.0"
+ resolved "https://registry.yarnpkg.com/astral-regex/-/astral-regex-2.0.0.tgz#483143c567aeed4785759c0865786dc77d7d2e31"
+ integrity sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==
+
+asynckit@^0.4.0:
+ version "0.4.0"
+ resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79"
+ integrity sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==
+
+balanced-match@^1.0.0:
+ version "1.0.2"
+ resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee"
+ integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==
+
+base64-js@^1.3.1:
+ version "1.5.1"
+ resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.5.1.tgz#1b1b440160a5bf7ad40b650f095963481903930a"
+ integrity sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==
+
+bech32@1.1.4:
+ version "1.1.4"
+ resolved "https://registry.yarnpkg.com/bech32/-/bech32-1.1.4.tgz#e38c9f37bf179b8eb16ae3a772b40c356d4832e9"
+ integrity sha512-s0IrSOzLlbvX7yp4WBfPITzpAU8sqQcpsmwXDiKwrG4r491vwCO/XpejasRNl0piBMe/DvP4Tz0mIS/X1DPJBQ==
+
+bignumber.js@^9.0.1:
+ version "9.1.2"
+ resolved "https://registry.yarnpkg.com/bignumber.js/-/bignumber.js-9.1.2.tgz#b7c4242259c008903b13707983b5f4bbd31eda0c"
+ integrity sha512-2/mKyZH9K85bzOEfhXDBFZTGd1CTs+5IHpeFQo9luiBG7hghdC851Pj2WAhb6E3R6b9tZj/XKhbg4fum+Kepug==
+
+bn.js@4.11.6:
+ version "4.11.6"
+ resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-4.11.6.tgz#53344adb14617a13f6e8dd2ce28905d1c0ba3215"
+ integrity sha512-XWwnNNFCuuSQ0m3r3C4LE3EiORltHd9M05pq6FOlVeiophzRbMo50Sbz1ehl8K3Z+jw9+vmgnXefY1hz8X+2wA==
+
+bn.js@^4.11.9:
+ version "4.12.0"
+ resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-4.12.0.tgz#775b3f278efbb9718eec7361f483fb36fbbfea88"
+ integrity sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==
+
+bn.js@^5.2.0, bn.js@^5.2.1:
+ version "5.2.1"
+ resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-5.2.1.tgz#0bc527a6a0d18d0aa8d5b0538ce4a77dccfa7b70"
+ integrity sha512-eXRvHzWyYPBuB4NBy0cmYQjGitUrtqwbvlzP3G6VFnNRbsZQIxQ10PbKKHt8gZ/HW/D/747aDl+QkDqg3KQLMQ==
+
+brace-expansion@^1.1.7:
+ version "1.1.11"
+ resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd"
+ integrity sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==
+ dependencies:
+ balanced-match "^1.0.0"
+ concat-map "0.0.1"
+
+brace-expansion@^2.0.1:
+ version "2.0.1"
+ resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-2.0.1.tgz#1edc459e0f0c548486ecf9fc99f2221364b9a0ae"
+ integrity sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==
+ dependencies:
+ balanced-match "^1.0.0"
+
+brorand@^1.1.0:
+ version "1.1.0"
+ resolved "https://registry.yarnpkg.com/brorand/-/brorand-1.1.0.tgz#12c25efe40a45e3c323eb8675a0a0ce57b22371f"
+ integrity sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w==
+
+buffer-reverse@^1.0.1:
+ version "1.0.1"
+ resolved "https://registry.yarnpkg.com/buffer-reverse/-/buffer-reverse-1.0.1.tgz#49283c8efa6f901bc01fa3304d06027971ae2f60"
+ integrity sha512-M87YIUBsZ6N924W57vDwT/aOu8hw7ZgdByz6ijksLjmHJELBASmYTTlNHRgjE+pTsT9oJXGaDSgqqwfdHotDUg==
+
+buffer@^6.0.3:
+ version "6.0.3"
+ resolved "https://registry.yarnpkg.com/buffer/-/buffer-6.0.3.tgz#2ace578459cc8fbe2a70aaa8f52ee63b6a74c6c6"
+ integrity sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==
+ dependencies:
+ base64-js "^1.3.1"
+ ieee754 "^1.2.1"
+
+callsites@^3.0.0:
+ version "3.1.0"
+ resolved "https://registry.yarnpkg.com/callsites/-/callsites-3.1.0.tgz#b3630abd8943432f54b3f0519238e33cd7df2f73"
+ integrity sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==
+
+chalk@^2.4.2:
+ version "2.4.2"
+ resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.4.2.tgz#cd42541677a54333cf541a49108c1432b44c9424"
+ integrity sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==
+ dependencies:
+ ansi-styles "^3.2.1"
+ escape-string-regexp "^1.0.5"
+ supports-color "^5.3.0"
+
+chalk@^4.1.0, chalk@^4.1.2:
+ version "4.1.2"
+ resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.2.tgz#aac4e2b7734a740867aeb16bf02aad556a1e7a01"
+ integrity sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==
+ dependencies:
+ ansi-styles "^4.1.0"
+ supports-color "^7.1.0"
+
+color-convert@^1.9.0:
+ version "1.9.3"
+ resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-1.9.3.tgz#bb71850690e1f136567de629d2d5471deda4c1e8"
+ integrity sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==
+ dependencies:
+ color-name "1.1.3"
+
+color-convert@^2.0.1:
+ version "2.0.1"
+ resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3"
+ integrity sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==
+ dependencies:
+ color-name "~1.1.4"
+
+color-name@1.1.3:
+ version "1.1.3"
+ resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.3.tgz#a7d0558bd89c42f795dd42328f740831ca53bc25"
+ integrity sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==
+
+color-name@~1.1.4:
+ version "1.1.4"
+ resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2"
+ integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==
+
+combined-stream@^1.0.8:
+ version "1.0.8"
+ resolved "https://registry.yarnpkg.com/combined-stream/-/combined-stream-1.0.8.tgz#c3d45a8b34fd730631a110a8a2520682b31d5a7f"
+ integrity sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==
+ dependencies:
+ delayed-stream "~1.0.0"
+
+command-line-args@^5.1.1:
+ version "5.2.1"
+ resolved "https://registry.yarnpkg.com/command-line-args/-/command-line-args-5.2.1.tgz#c44c32e437a57d7c51157696893c5909e9cec42e"
+ integrity sha512-H4UfQhZyakIjC74I9d34fGYDwk3XpSr17QhEd0Q3I9Xq1CETHo4Hcuo87WyWHpAF1aSLjLRf5lD9ZGX2qStUvg==
+ dependencies:
+ array-back "^3.1.0"
+ find-replace "^3.0.0"
+ lodash.camelcase "^4.3.0"
+ typical "^4.0.0"
+
+command-line-usage@^6.1.0:
+ version "6.1.3"
+ resolved "https://registry.yarnpkg.com/command-line-usage/-/command-line-usage-6.1.3.tgz#428fa5acde6a838779dfa30e44686f4b6761d957"
+ integrity sha512-sH5ZSPr+7UStsloltmDh7Ce5fb8XPlHyoPzTpyyMuYCtervL65+ubVZ6Q61cFtFl62UyJlc8/JwERRbAFPUqgw==
+ dependencies:
+ array-back "^4.0.2"
+ chalk "^2.4.2"
+ table-layout "^1.0.2"
+ typical "^5.2.0"
+
+commander@^10.0.0:
+ version "10.0.1"
+ resolved "https://registry.yarnpkg.com/commander/-/commander-10.0.1.tgz#881ee46b4f77d1c1dccc5823433aa39b022cbe06"
+ integrity sha512-y4Mg2tXshplEbSGzx7amzPwKKOCGuoSRP/CjEdwwk0FOGlUbq6lKuoyDZTNZkmxHdJtp54hdfY/JUrdL7Xfdug==
+
+concat-map@0.0.1:
+ version "0.0.1"
+ resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b"
+ integrity sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==
+
+cosmiconfig@^8.0.0:
+ version "8.3.6"
+ resolved "https://registry.yarnpkg.com/cosmiconfig/-/cosmiconfig-8.3.6.tgz#060a2b871d66dba6c8538ea1118ba1ac16f5fae3"
+ integrity sha512-kcZ6+W5QzcJ3P1Mt+83OUv/oHFqZHIx8DuxG6eZ5RGMERoLqp4BuGjhHLYGK+Kf5XVkQvqBSmAy/nGWN3qDgEA==
+ dependencies:
+ import-fresh "^3.3.0"
+ js-yaml "^4.1.0"
+ parse-json "^5.2.0"
+ path-type "^4.0.0"
+
+cross-fetch@^3.0.6:
+ version "3.1.8"
+ resolved "https://registry.yarnpkg.com/cross-fetch/-/cross-fetch-3.1.8.tgz#0327eba65fd68a7d119f8fb2bf9334a1a7956f82"
+ integrity sha512-cvA+JwZoU0Xq+h6WkMvAUqPEYy92Obet6UdKLfW60qn99ftItKjB5T+BkyWOFWe2pUyfQ+IJHmpOTznqk1M6Kg==
+ dependencies:
+ node-fetch "^2.6.12"
+
+crypto-js@^4.2.0:
+ version "4.2.0"
+ resolved "https://registry.yarnpkg.com/crypto-js/-/crypto-js-4.2.0.tgz#4d931639ecdfd12ff80e8186dba6af2c2e856631"
+ integrity sha512-KALDyEYgpY+Rlob/iriUtjV6d5Eq+Y191A5g4UqLAi8CyGP9N1+FdVbkc1SxKc2r4YAYqG8JzO2KGL+AizD70Q==
+
+debug@^4.3.1:
+ version "4.3.4"
+ resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.4.tgz#1319f6579357f2338d3337d2cdd4914bb5dcc865"
+ integrity sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==
+ dependencies:
+ ms "2.1.2"
+
+deep-extend@~0.6.0:
+ version "0.6.0"
+ resolved "https://registry.yarnpkg.com/deep-extend/-/deep-extend-0.6.0.tgz#c4fa7c95404a17a9c3e8ca7e1537312b736330ac"
+ integrity sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==
+
+delayed-stream@~1.0.0:
+ version "1.0.0"
+ resolved "https://registry.yarnpkg.com/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619"
+ integrity sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==
+
+elliptic@6.5.4:
+ version "6.5.4"
+ resolved "https://registry.yarnpkg.com/elliptic/-/elliptic-6.5.4.tgz#da37cebd31e79a1367e941b592ed1fbebd58abbb"
+ integrity sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ==
+ dependencies:
+ bn.js "^4.11.9"
+ brorand "^1.1.0"
+ hash.js "^1.0.0"
+ hmac-drbg "^1.0.1"
+ inherits "^2.0.4"
+ minimalistic-assert "^1.0.1"
+ minimalistic-crypto-utils "^1.0.1"
+
+emoji-regex@^8.0.0:
+ version "8.0.0"
+ resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-8.0.0.tgz#e818fd69ce5ccfcb404594f842963bf53164cc37"
+ integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==
+
+error-ex@^1.3.1:
+ version "1.3.2"
+ resolved "https://registry.yarnpkg.com/error-ex/-/error-ex-1.3.2.tgz#b4ac40648107fdcdcfae242f428bea8a14d4f1bf"
+ integrity sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==
+ dependencies:
+ is-arrayish "^0.2.1"
+
+escape-string-regexp@^1.0.5:
+ version "1.0.5"
+ resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4"
+ integrity sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==
+
+ethereum-bloom-filters@^1.0.6:
+ version "1.1.0"
+ resolved "https://registry.yarnpkg.com/ethereum-bloom-filters/-/ethereum-bloom-filters-1.1.0.tgz#b3fc1eb789509ee30db0bf99a2988ccacb8d0397"
+ integrity sha512-J1gDRkLpuGNvWYzWslBQR9cDV4nd4kfvVTE/Wy4Kkm4yb3EYRSlyi0eB/inTsSTTVyA0+HyzHgbr95Fn/Z1fSw==
+ dependencies:
+ "@noble/hashes" "^1.4.0"
+
+ethereum-cryptography@^2.0.0, ethereum-cryptography@^2.1.2:
+ version "2.1.3"
+ resolved "https://registry.yarnpkg.com/ethereum-cryptography/-/ethereum-cryptography-2.1.3.tgz#1352270ed3b339fe25af5ceeadcf1b9c8e30768a"
+ integrity sha512-BlwbIL7/P45W8FGW2r7LGuvoEZ+7PWsniMvQ4p5s2xCyw9tmaDlpfsN9HjAucbF+t/qpVHwZUisgfK24TCW8aA==
+ dependencies:
+ "@noble/curves" "1.3.0"
+ "@noble/hashes" "1.3.3"
+ "@scure/bip32" "1.3.3"
+ "@scure/bip39" "1.2.2"
+
+ethers@^5.6.4:
+ version "5.7.2"
+ resolved "https://registry.yarnpkg.com/ethers/-/ethers-5.7.2.tgz#3a7deeabbb8c030d4126b24f84e525466145872e"
+ integrity sha512-wswUsmWo1aOK8rR7DIKiWSw9DbLWe6x98Jrn8wcTflTVvaXhAMaB5zGAXy0GYQEQp9iO1iSHWVyARQm11zUtyg==
+ dependencies:
+ "@ethersproject/abi" "5.7.0"
+ "@ethersproject/abstract-provider" "5.7.0"
+ "@ethersproject/abstract-signer" "5.7.0"
+ "@ethersproject/address" "5.7.0"
+ "@ethersproject/base64" "5.7.0"
+ "@ethersproject/basex" "5.7.0"
+ "@ethersproject/bignumber" "5.7.0"
+ "@ethersproject/bytes" "5.7.0"
+ "@ethersproject/constants" "5.7.0"
+ "@ethersproject/contracts" "5.7.0"
+ "@ethersproject/hash" "5.7.0"
+ "@ethersproject/hdnode" "5.7.0"
+ "@ethersproject/json-wallets" "5.7.0"
+ "@ethersproject/keccak256" "5.7.0"
+ "@ethersproject/logger" "5.7.0"
+ "@ethersproject/networks" "5.7.1"
+ "@ethersproject/pbkdf2" "5.7.0"
+ "@ethersproject/properties" "5.7.0"
+ "@ethersproject/providers" "5.7.2"
+ "@ethersproject/random" "5.7.0"
+ "@ethersproject/rlp" "5.7.0"
+ "@ethersproject/sha2" "5.7.0"
+ "@ethersproject/signing-key" "5.7.0"
+ "@ethersproject/solidity" "5.7.0"
+ "@ethersproject/strings" "5.7.0"
+ "@ethersproject/transactions" "5.7.0"
+ "@ethersproject/units" "5.7.0"
+ "@ethersproject/wallet" "5.7.0"
+ "@ethersproject/web" "5.7.1"
+ "@ethersproject/wordlists" "5.7.0"
+
+ethjs-unit@0.1.6:
+ version "0.1.6"
+ resolved "https://registry.yarnpkg.com/ethjs-unit/-/ethjs-unit-0.1.6.tgz#c665921e476e87bce2a9d588a6fe0405b2c41699"
+ integrity sha512-/Sn9Y0oKl0uqQuvgFk/zQgR7aw1g36qX/jzSQ5lSwlO0GigPymk4eGQfeNTD03w1dPOqfz8V77Cy43jH56pagw==
+ dependencies:
+ bn.js "4.11.6"
+ number-to-bn "1.7.0"
+
+extract-files@^9.0.0:
+ version "9.0.0"
+ resolved "https://registry.yarnpkg.com/extract-files/-/extract-files-9.0.0.tgz#8a7744f2437f81f5ed3250ed9f1550de902fe54a"
+ integrity sha512-CvdFfHkC95B4bBBk36hcEmvdR2awOdhhVUYH6S/zrVj3477zven/fJMYg7121h4T1xHZC+tetUpubpAhxwI7hQ==
+
+fast-deep-equal@^3.1.1:
+ version "3.1.3"
+ resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525"
+ integrity sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==
+
+fast-diff@^1.1.2, fast-diff@^1.2.0:
+ version "1.3.0"
+ resolved "https://registry.yarnpkg.com/fast-diff/-/fast-diff-1.3.0.tgz#ece407fa550a64d638536cd727e129c61616e0f0"
+ integrity sha512-VxPP4NqbUjj6MaAOafWeUn2cXWLcCtljklUtZf0Ind4XQ+QPtmA0b18zZy0jIQx+ExRVCR/ZQpBmik5lXshNsw==
+
+fast-json-stable-stringify@^2.0.0:
+ version "2.1.0"
+ resolved "https://registry.yarnpkg.com/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz#874bf69c6f404c2b5d99c481341399fd55892633"
+ integrity sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==
+
+find-replace@^3.0.0:
+ version "3.0.0"
+ resolved "https://registry.yarnpkg.com/find-replace/-/find-replace-3.0.0.tgz#3e7e23d3b05167a76f770c9fbd5258b0def68c38"
+ integrity sha512-6Tb2myMioCAgv5kfvP5/PkZZ/ntTpVK39fHY7WkWBgvbeE+VHd/tZuZ4mrC+bxh4cfOZeYKVPaJIZtZXV7GNCQ==
+ dependencies:
+ array-back "^3.0.1"
+
+form-data@^3.0.0:
+ version "3.0.1"
+ resolved "https://registry.yarnpkg.com/form-data/-/form-data-3.0.1.tgz#ebd53791b78356a99af9a300d4282c4d5eb9755f"
+ integrity sha512-RHkBKtLWUVwd7SqRIvCZMEvAMoGUp0XU+seQiZejj0COz3RI3hWP4sCv3gZWWLjJTd7rGwcsF5eKZGii0r/hbg==
+ dependencies:
+ asynckit "^0.4.0"
+ combined-stream "^1.0.8"
+ mime-types "^2.1.12"
+
+fs-extra@^7.0.0:
+ version "7.0.1"
+ resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-7.0.1.tgz#4f189c44aa123b895f722804f55ea23eadc348e9"
+ integrity sha512-YJDaCJZEnBmcbw13fvdAM9AwNOJwOzrE4pqMqBq5nFiEqXUqHwlK4B+3pUw6JNvfSPtX05xFHtYy/1ni01eGCw==
+ dependencies:
+ graceful-fs "^4.1.2"
+ jsonfile "^4.0.0"
+ universalify "^0.1.0"
+
+fs.realpath@^1.0.0:
+ version "1.0.0"
+ resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f"
+ integrity sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==
+
+glob@7.1.7:
+ version "7.1.7"
+ resolved "https://registry.yarnpkg.com/glob/-/glob-7.1.7.tgz#3b193e9233f01d42d0b3f78294bbeeb418f94a90"
+ integrity sha512-OvD9ENzPLbegENnYP5UUfJIirTg4+XwMWGaQfQTY0JenxNvvIKP3U3/tAQSPIu/lHxXYSZmpXlUHeqAIdKzBLQ==
+ dependencies:
+ fs.realpath "^1.0.0"
+ inflight "^1.0.4"
+ inherits "2"
+ minimatch "^3.0.4"
+ once "^1.3.0"
+ path-is-absolute "^1.0.0"
+
+glob@^8.0.3:
+ version "8.1.0"
+ resolved "https://registry.yarnpkg.com/glob/-/glob-8.1.0.tgz#d388f656593ef708ee3e34640fdfb99a9fd1c33e"
+ integrity sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==
+ dependencies:
+ fs.realpath "^1.0.0"
+ inflight "^1.0.4"
+ inherits "2"
+ minimatch "^5.0.1"
+ once "^1.3.0"
+
+graceful-fs@^4.1.2, graceful-fs@^4.1.6:
+ version "4.2.11"
+ resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.11.tgz#4183e4e8bf08bb6e05bbb2f7d2e0c8f712ca40e3"
+ integrity sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==
+
+graphql-request@^3.6.1:
+ version "3.7.0"
+ resolved "https://registry.yarnpkg.com/graphql-request/-/graphql-request-3.7.0.tgz#c7406e537084f8b9788541e3e6704340ca13055b"
+ integrity sha512-dw5PxHCgBneN2DDNqpWu8QkbbJ07oOziy8z+bK/TAXufsOLaETuVO4GkXrbs0WjhdKhBMN3BkpN/RIvUHkmNUQ==
+ dependencies:
+ cross-fetch "^3.0.6"
+ extract-files "^9.0.0"
+ form-data "^3.0.0"
+
+graphql-tag@^2.12.6:
+ version "2.12.6"
+ resolved "https://registry.yarnpkg.com/graphql-tag/-/graphql-tag-2.12.6.tgz#d441a569c1d2537ef10ca3d1633b48725329b5f1"
+ integrity sha512-FdSNcu2QQcWnM2VNvSCCDCVS5PpPqpzgFT8+GXzqJuoDd0CBncxCY278u4mhRO7tMgo2JjgJA5aZ+nWSQ/Z+xg==
+ dependencies:
+ tslib "^2.1.0"
+
+graphql@^15.7.1:
+ version "15.8.0"
+ resolved "https://registry.yarnpkg.com/graphql/-/graphql-15.8.0.tgz#33410e96b012fa3bdb1091cc99a94769db212b38"
+ integrity sha512-5gghUc24tP9HRznNpV2+FIoq3xKkj5dTQqf4v0CpdPbFVwFkWoxOM+o+2OC9ZSvjEMTjfmG9QT+gcvggTwW1zw==
+
+has-flag@^3.0.0:
+ version "3.0.0"
+ resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-3.0.0.tgz#b5d454dc2199ae225699f3467e5a07f3b955bafd"
+ integrity sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==
+
+has-flag@^4.0.0:
+ version "4.0.0"
+ resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b"
+ integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==
+
+hash.js@1.1.7, hash.js@^1.0.0, hash.js@^1.0.3:
+ version "1.1.7"
+ resolved "https://registry.yarnpkg.com/hash.js/-/hash.js-1.1.7.tgz#0babca538e8d4ee4a0f8988d68866537a003cf42"
+ integrity sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==
+ dependencies:
+ inherits "^2.0.3"
+ minimalistic-assert "^1.0.1"
+
+hmac-drbg@^1.0.1:
+ version "1.0.1"
+ resolved "https://registry.yarnpkg.com/hmac-drbg/-/hmac-drbg-1.0.1.tgz#d2745701025a6c775a6c545793ed502fc0c649a1"
+ integrity sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg==
+ dependencies:
+ hash.js "^1.0.3"
+ minimalistic-assert "^1.0.0"
+ minimalistic-crypto-utils "^1.0.1"
+
+hoist-non-react-statics@^3.3.2:
+ version "3.3.2"
+ resolved "https://registry.yarnpkg.com/hoist-non-react-statics/-/hoist-non-react-statics-3.3.2.tgz#ece0acaf71d62c2969c2ec59feff42a4b1a85b45"
+ integrity sha512-/gGivxi8JPKWNm/W0jSmzcMPpfpPLc3dY/6GxhX2hQ9iGj3aDfklV4ET7NjKpSinLpJ5vafa9iiGIEZg10SfBw==
+ dependencies:
+ react-is "^16.7.0"
+
+ieee754@^1.2.1:
+ version "1.2.1"
+ resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.2.1.tgz#8eb7a10a63fff25d15a57b001586d177d1b0d352"
+ integrity sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==
+
+ignore@^5.2.4:
+ version "5.3.1"
+ resolved "https://registry.yarnpkg.com/ignore/-/ignore-5.3.1.tgz#5073e554cd42c5b33b394375f538b8593e34d4ef"
+ integrity sha512-5Fytz/IraMjqpwfd34ke28PTVMjZjJG2MPn5t7OE4eUCUNf8BAa7b5WUS9/Qvr6mwOQS7Mk6vdsMno5he+T8Xw==
+
+import-fresh@^3.3.0:
+ version "3.3.0"
+ resolved "https://registry.yarnpkg.com/import-fresh/-/import-fresh-3.3.0.tgz#37162c25fcb9ebaa2e6e53d5b4d88ce17d9e0c2b"
+ integrity sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==
+ dependencies:
+ parent-module "^1.0.0"
+ resolve-from "^4.0.0"
+
+inflight@^1.0.4:
+ version "1.0.6"
+ resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9"
+ integrity sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==
+ dependencies:
+ once "^1.3.0"
+ wrappy "1"
+
+inherits@2, inherits@^2.0.3, inherits@^2.0.4:
+ version "2.0.4"
+ resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c"
+ integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==
+
+is-arrayish@^0.2.1:
+ version "0.2.1"
+ resolved "https://registry.yarnpkg.com/is-arrayish/-/is-arrayish-0.2.1.tgz#77c99840527aa8ecb1a8ba697b80645a7a926a9d"
+ integrity sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==
+
+is-fullwidth-code-point@^3.0.0:
+ version "3.0.0"
+ resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz#f116f8064fe90b3f7844a38997c0b75051269f1d"
+ integrity sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==
+
+is-hex-prefixed@1.0.0:
+ version "1.0.0"
+ resolved "https://registry.yarnpkg.com/is-hex-prefixed/-/is-hex-prefixed-1.0.0.tgz#7d8d37e6ad77e5d127148913c573e082d777f554"
+ integrity sha512-WvtOiug1VFrE9v1Cydwm+FnXd3+w9GaeVUss5W4v/SLy3UW00vP+6iNF2SdnfiBoLy4bTqVdkftNGTUeOFVsbA==
+
+js-sha3@0.8.0, js-sha3@^0.8.0:
+ version "0.8.0"
+ resolved "https://registry.yarnpkg.com/js-sha3/-/js-sha3-0.8.0.tgz#b9b7a5da73afad7dedd0f8c463954cbde6818840"
+ integrity sha512-gF1cRrHhIzNfToc802P800N8PpXS+evLLXfsVpowqmAFR9uwbi89WvXg2QspOmXL8QL86J4T1EpFu+yUkwJY3Q==
+
+"js-tokens@^3.0.0 || ^4.0.0", js-tokens@^4.0.0:
+ version "4.0.0"
+ resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-4.0.0.tgz#19203fb59991df98e3a287050d4647cdeaf32499"
+ integrity sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==
+
+js-yaml@^4.1.0:
+ version "4.1.0"
+ resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-4.1.0.tgz#c1fb65f8f5017901cdd2c951864ba18458a10602"
+ integrity sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==
+ dependencies:
+ argparse "^2.0.1"
+
+jsbi@^4.3.0:
+ version "4.3.0"
+ resolved "https://registry.yarnpkg.com/jsbi/-/jsbi-4.3.0.tgz#b54ee074fb6fcbc00619559305c8f7e912b04741"
+ integrity sha512-SnZNcinB4RIcnEyZqFPdGPVgrg2AcnykiBy0sHVJQKHYeaLUvi3Exj+iaPpLnFVkDPZIV4U0yvgC9/R4uEAZ9g==
+
+json-parse-even-better-errors@^2.3.0:
+ version "2.3.1"
+ resolved "https://registry.yarnpkg.com/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz#7c47805a94319928e05777405dc12e1f7a4ee02d"
+ integrity sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==
+
+json-schema-traverse@^0.4.1:
+ version "0.4.1"
+ resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660"
+ integrity sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==
+
+json-schema-traverse@^1.0.0:
+ version "1.0.0"
+ resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz#ae7bcb3656ab77a73ba5c49bf654f38e6b6860e2"
+ integrity sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==
+
+jsonfile@^4.0.0:
+ version "4.0.0"
+ resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-4.0.0.tgz#8771aae0799b64076b76640fca058f9c10e33ecb"
+ integrity sha512-m6F1R3z8jjlf2imQHS2Qez5sjKWQzbuuhuJ/FKYFRZvPE3PuHcSMVZzfsLhGVOkfd20obL5SWEBew5ShlquNxg==
+ optionalDependencies:
+ graceful-fs "^4.1.6"
+
+keccak256@^1.0.6:
+ version "1.0.6"
+ resolved "https://registry.yarnpkg.com/keccak256/-/keccak256-1.0.6.tgz#dd32fb771558fed51ce4e45a035ae7515573da58"
+ integrity sha512-8GLiM01PkdJVGUhR1e6M/AvWnSqYS0HaERI+K/QtStGDGlSTx2B1zTqZk4Zlqu5TxHJNTxWAdP9Y+WI50OApUw==
+ dependencies:
+ bn.js "^5.2.0"
+ buffer "^6.0.3"
+ keccak "^3.0.2"
+
+keccak@^3.0.2:
+ version "3.0.4"
+ resolved "https://registry.yarnpkg.com/keccak/-/keccak-3.0.4.tgz#edc09b89e633c0549da444432ecf062ffadee86d"
+ integrity sha512-3vKuW0jV8J3XNTzvfyicFR5qvxrSAGl7KIhvgOu5cmWwM7tZRj3fMbj/pfIf4be7aznbc+prBWGjywox/g2Y6Q==
+ dependencies:
+ node-addon-api "^2.0.0"
+ node-gyp-build "^4.2.0"
+ readable-stream "^3.6.0"
+
+lines-and-columns@^1.1.6:
+ version "1.2.4"
+ resolved "https://registry.yarnpkg.com/lines-and-columns/-/lines-and-columns-1.2.4.tgz#eca284f75d2965079309dc0ad9255abb2ebc1632"
+ integrity sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==
+
+lodash.camelcase@^4.3.0:
+ version "4.3.0"
+ resolved "https://registry.yarnpkg.com/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz#b28aa6288a2b9fc651035c7711f65ab6190331a6"
+ integrity sha512-TwuEnCnxbc3rAvhf/LbG7tJUDzhqXyFnv3dtzLOPgCG/hODL7WFnsbwktkD7yUV0RrreP/l1PALq/YSg6VvjlA==
+
+lodash.truncate@^4.4.2:
+ version "4.4.2"
+ resolved "https://registry.yarnpkg.com/lodash.truncate/-/lodash.truncate-4.4.2.tgz#5a350da0b1113b837ecfffd5812cbe58d6eae193"
+ integrity sha512-jttmRe7bRse52OsWIMDLaXxWqRAmtIUccAQ3garviCqJjafXOfNMO0yMfNpdD6zbGaTU0P5Nz7e7gAT6cKmJRw==
+
+lodash@^4.17.15, lodash@^4.17.21:
+ version "4.17.21"
+ resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz#679591c564c3bffaae8454cf0b3df370c3d6911c"
+ integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==
+
+loose-envify@^1.4.0:
+ version "1.4.0"
+ resolved "https://registry.yarnpkg.com/loose-envify/-/loose-envify-1.4.0.tgz#71ee51fa7be4caec1a63839f7e682d8132d30caf"
+ integrity sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==
+ dependencies:
+ js-tokens "^3.0.0 || ^4.0.0"
+
+lru-cache@^6.0.0:
+ version "6.0.0"
+ resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-6.0.0.tgz#6d6fe6570ebd96aaf90fcad1dafa3b2566db3a94"
+ integrity sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==
+ dependencies:
+ yallist "^4.0.0"
+
+merkletreejs@^0.3.10:
+ version "0.3.11"
+ resolved "https://registry.yarnpkg.com/merkletreejs/-/merkletreejs-0.3.11.tgz#e0de05c3ca1fd368de05a12cb8efb954ef6fc04f"
+ integrity sha512-LJKTl4iVNTndhL+3Uz/tfkjD0klIWsHlUzgtuNnNrsf7bAlXR30m+xYB7lHr5Z/l6e/yAIsr26Dabx6Buo4VGQ==
+ dependencies:
+ bignumber.js "^9.0.1"
+ buffer-reverse "^1.0.1"
+ crypto-js "^4.2.0"
+ treeify "^1.1.0"
+ web3-utils "^1.3.4"
+
+micro-ftch@^0.3.1:
+ version "0.3.1"
+ resolved "https://registry.yarnpkg.com/micro-ftch/-/micro-ftch-0.3.1.tgz#6cb83388de4c1f279a034fb0cf96dfc050853c5f"
+ integrity sha512-/0LLxhzP0tfiR5hcQebtudP56gUurs2CLkGarnCiB/OqEyUFQ6U3paQi/tgLv0hBJYt2rnr9MNpxz4fiiugstg==
+
+mime-db@1.52.0:
+ version "1.52.0"
+ resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.52.0.tgz#bbabcdc02859f4987301c856e3387ce5ec43bf70"
+ integrity sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==
+
+mime-types@^2.1.12:
+ version "2.1.35"
+ resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.35.tgz#381a871b62a734450660ae3deee44813f70d959a"
+ integrity sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==
+ dependencies:
+ mime-db "1.52.0"
+
+minimalistic-assert@^1.0.0, minimalistic-assert@^1.0.1:
+ version "1.0.1"
+ resolved "https://registry.yarnpkg.com/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz#2e194de044626d4a10e7f7fbc00ce73e83e4d5c7"
+ integrity sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==
+
+minimalistic-crypto-utils@^1.0.1:
+ version "1.0.1"
+ resolved "https://registry.yarnpkg.com/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz#f6c00c1c0b082246e5c4d99dfb8c7c083b2b582a"
+ integrity sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg==
+
+minimatch@^3.0.4:
+ version "3.1.2"
+ resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.1.2.tgz#19cd194bfd3e428f049a70817c038d89ab4be35b"
+ integrity sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==
+ dependencies:
+ brace-expansion "^1.1.7"
+
+minimatch@^5.0.1:
+ version "5.1.6"
+ resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-5.1.6.tgz#1cfcb8cf5522ea69952cd2af95ae09477f122a96"
+ integrity sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==
+ dependencies:
+ brace-expansion "^2.0.1"
+
+mkdirp@^1.0.4:
+ version "1.0.4"
+ resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-1.0.4.tgz#3eb5ed62622756d79a5f0e2a221dfebad75c2f7e"
+ integrity sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==
+
+ms@2.1.2:
+ version "2.1.2"
+ resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.2.tgz#d09d1f357b443f493382a8eb3ccd183872ae6009"
+ integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==
+
+node-addon-api@^2.0.0:
+ version "2.0.2"
+ resolved "https://registry.yarnpkg.com/node-addon-api/-/node-addon-api-2.0.2.tgz#432cfa82962ce494b132e9d72a15b29f71ff5d32"
+ integrity sha512-Ntyt4AIXyaLIuMHF6IOoTakB3K+RWxwtsHNRxllEoA6vPwP9o4866g6YWDLUdnucilZhmkxiHwHr11gAENw+QA==
+
+node-fetch@^2.6.12:
+ version "2.7.0"
+ resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-2.7.0.tgz#d0f0fa6e3e2dc1d27efcd8ad99d550bda94d187d"
+ integrity sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==
+ dependencies:
+ whatwg-url "^5.0.0"
+
+node-gyp-build@^4.2.0:
+ version "4.8.0"
+ resolved "https://registry.yarnpkg.com/node-gyp-build/-/node-gyp-build-4.8.0.tgz#3fee9c1731df4581a3f9ead74664369ff00d26dd"
+ integrity sha512-u6fs2AEUljNho3EYTJNBfImO5QTo/J/1Etd+NVdCj7qWKUSN/bSLkZwhDv7I+w/MSC6qJ4cknepkAYykDdK8og==
+
+number-to-bn@1.7.0:
+ version "1.7.0"
+ resolved "https://registry.yarnpkg.com/number-to-bn/-/number-to-bn-1.7.0.tgz#bb3623592f7e5f9e0030b1977bd41a0c53fe1ea0"
+ integrity sha512-wsJ9gfSz1/s4ZsJN01lyonwuxA1tml6X1yBDnfpMglypcBRFZZkus26EdPSlqS5GJfYddVZa22p3VNb3z5m5Ig==
+ dependencies:
+ bn.js "4.11.6"
+ strip-hex-prefix "1.0.0"
+
+object-assign@^4.1.1:
+ version "4.1.1"
+ resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863"
+ integrity sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==
+
+once@^1.3.0:
+ version "1.4.0"
+ resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1"
+ integrity sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==
+ dependencies:
+ wrappy "1"
+
+optimism@^0.18.0:
+ version "0.18.0"
+ resolved "https://registry.yarnpkg.com/optimism/-/optimism-0.18.0.tgz#e7bb38b24715f3fdad8a9a7fc18e999144bbfa63"
+ integrity sha512-tGn8+REwLRNFnb9WmcY5IfpOqeX2kpaYJ1s6Ae3mn12AeydLkR3j+jSCmVQFoXqU8D41PAJ1RG1rCRNWmNZVmQ==
+ dependencies:
+ "@wry/caches" "^1.0.0"
+ "@wry/context" "^0.7.0"
+ "@wry/trie" "^0.4.3"
+ tslib "^2.3.0"
+
+parent-module@^1.0.0:
+ version "1.0.1"
+ resolved "https://registry.yarnpkg.com/parent-module/-/parent-module-1.0.1.tgz#691d2709e78c79fae3a156622452d00762caaaa2"
+ integrity sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==
+ dependencies:
+ callsites "^3.0.0"
+
+parse-json@^5.2.0:
+ version "5.2.0"
+ resolved "https://registry.yarnpkg.com/parse-json/-/parse-json-5.2.0.tgz#c76fc66dee54231c962b22bcc8a72cf2f99753cd"
+ integrity sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==
+ dependencies:
+ "@babel/code-frame" "^7.0.0"
+ error-ex "^1.3.1"
+ json-parse-even-better-errors "^2.3.0"
+ lines-and-columns "^1.1.6"
+
+path-is-absolute@^1.0.0:
+ version "1.0.1"
+ resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz#174b9268735534ffbc7ace6bf53a5a9e1b5c5f5f"
+ integrity sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==
+
+path-type@^4.0.0:
+ version "4.0.0"
+ resolved "https://registry.yarnpkg.com/path-type/-/path-type-4.0.0.tgz#84ed01c0a7ba380afe09d90a8c180dcd9d03043b"
+ integrity sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==
+
+picocolors@^1.0.0:
+ version "1.0.0"
+ resolved "https://registry.yarnpkg.com/picocolors/-/picocolors-1.0.0.tgz#cb5bdc74ff3f51892236eaf79d68bc44564ab81c"
+ integrity sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==
+
+pluralize@^8.0.0:
+ version "8.0.0"
+ resolved "https://registry.yarnpkg.com/pluralize/-/pluralize-8.0.0.tgz#1a6fa16a38d12a1901e0320fa017051c539ce3b1"
+ integrity sha512-Nc3IT5yHzflTfbjgqWcCPpo7DaKy4FnpB0l/zCAW0Tc7jxAiuqSxHasntB3D7887LSrA93kDJ9IXovxJYxyLCA==
+
+prettier-linter-helpers@^1.0.0:
+ version "1.0.0"
+ resolved "https://registry.yarnpkg.com/prettier-linter-helpers/-/prettier-linter-helpers-1.0.0.tgz#d23d41fe1375646de2d0104d3454a3008802cf7b"
+ integrity sha512-GbK2cP9nraSSUF9N2XwUwqfzlAFlMNYYl+ShE/V+H8a9uNl/oUqB1w2EL54Jh0OlyRSd8RfWYJ3coVS4TROP2w==
+ dependencies:
+ fast-diff "^1.1.2"
+
+prettier-plugin-solidity@^1.1.3:
+ version "1.3.1"
+ resolved "https://registry.yarnpkg.com/prettier-plugin-solidity/-/prettier-plugin-solidity-1.3.1.tgz#59944d3155b249f7f234dee29f433524b9a4abcf"
+ integrity sha512-MN4OP5I2gHAzHZG1wcuJl0FsLS3c4Cc5494bbg+6oQWBPuEamjwDvmGfFMZ6NFzsh3Efd9UUxeT7ImgjNH4ozA==
+ dependencies:
+ "@solidity-parser/parser" "^0.17.0"
+ semver "^7.5.4"
+ solidity-comments-extractor "^0.0.8"
+
+prettier@^2.0.0, prettier@^2.3.1, prettier@^2.8.3:
+ version "2.8.8"
+ resolved "https://registry.yarnpkg.com/prettier/-/prettier-2.8.8.tgz#e8c5d7e98a4305ffe3de2e1fc4aca1a71c28b1da"
+ integrity sha512-tdN8qQGvNjw4CHbY+XXk0JgCXn9QiF21a55rBe5LJAU+kDyC4WQn4+awm2Xfk2lQMk5fKup9XgzTZtGkjBdP9Q==
+
+prop-types@^15.7.2:
+ version "15.8.1"
+ resolved "https://registry.yarnpkg.com/prop-types/-/prop-types-15.8.1.tgz#67d87bf1a694f48435cf332c24af10214a3140b5"
+ integrity sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==
+ dependencies:
+ loose-envify "^1.4.0"
+ object-assign "^4.1.1"
+ react-is "^16.13.1"
+
+punycode@^2.1.0:
+ version "2.3.1"
+ resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.3.1.tgz#027422e2faec0b25e1549c3e1bd8309b9133b6e5"
+ integrity sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==
+
+randombytes@^2.1.0:
+ version "2.1.0"
+ resolved "https://registry.yarnpkg.com/randombytes/-/randombytes-2.1.0.tgz#df6f84372f0270dc65cdf6291349ab7a473d4f2a"
+ integrity sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==
+ dependencies:
+ safe-buffer "^5.1.0"
+
+react-is@^16.13.1, react-is@^16.7.0:
+ version "16.13.1"
+ resolved "https://registry.yarnpkg.com/react-is/-/react-is-16.13.1.tgz#789729a4dc36de2999dc156dd6c1d9c18cea56a4"
+ integrity sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==
+
+readable-stream@^3.6.0:
+ version "3.6.2"
+ resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-3.6.2.tgz#56a9b36ea965c00c5a93ef31eb111a0f11056967"
+ integrity sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==
+ dependencies:
+ inherits "^2.0.3"
+ string_decoder "^1.1.1"
+ util-deprecate "^1.0.1"
+
+reduce-flatten@^2.0.0:
+ version "2.0.0"
+ resolved "https://registry.yarnpkg.com/reduce-flatten/-/reduce-flatten-2.0.0.tgz#734fd84e65f375d7ca4465c69798c25c9d10ae27"
+ integrity sha512-EJ4UNY/U1t2P/2k6oqotuX2Cc3T6nxJwsM0N0asT7dhrtH1ltUxDn4NalSYmPE2rCkVpcf/X6R0wDwcFpzhd4w==
+
+rehackt@0.0.6:
+ version "0.0.6"
+ resolved "https://registry.yarnpkg.com/rehackt/-/rehackt-0.0.6.tgz#7a0a2247f2295e7548915417e44fbbf03bf004f4"
+ integrity sha512-l3WEzkt4ntlEc/IB3/mF6SRgNHA6zfQR7BlGOgBTOmx7IJJXojDASav+NsgXHFjHn+6RmwqsGPFgZpabWpeOdw==
+
+require-from-string@^2.0.2:
+ version "2.0.2"
+ resolved "https://registry.yarnpkg.com/require-from-string/-/require-from-string-2.0.2.tgz#89a7fdd938261267318eafe14f9c32e598c36909"
+ integrity sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==
+
+resolve-from@^4.0.0:
+ version "4.0.0"
+ resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-4.0.0.tgz#4abcd852ad32dd7baabfe9b40e00a36db5f392e6"
+ integrity sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==
+
+response-iterator@^0.2.6:
+ version "0.2.6"
+ resolved "https://registry.yarnpkg.com/response-iterator/-/response-iterator-0.2.6.tgz#249005fb14d2e4eeb478a3f735a28fd8b4c9f3da"
+ integrity sha512-pVzEEzrsg23Sh053rmDUvLSkGXluZio0qu8VT6ukrYuvtjVfCbDZH9d6PGXb8HZfzdNZt8feXv/jvUzlhRgLnw==
+
+safe-buffer@^5.1.0, safe-buffer@~5.2.0:
+ version "5.2.1"
+ resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6"
+ integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==
+
+scrypt-js@3.0.1:
+ version "3.0.1"
+ resolved "https://registry.yarnpkg.com/scrypt-js/-/scrypt-js-3.0.1.tgz#d314a57c2aef69d1ad98a138a21fe9eafa9ee312"
+ integrity sha512-cdwTTnqPu0Hyvf5in5asVdZocVDTNRmR7XEcJuIzMjJeSHybHl7vpB66AzwTaIg6CLSbtjcxc8fqcySfnTkccA==
+
+semver@^7.5.2, semver@^7.5.4:
+ version "7.6.0"
+ resolved "https://registry.yarnpkg.com/semver/-/semver-7.6.0.tgz#1a46a4db4bffcccd97b743b5005c8325f23d4e2d"
+ integrity sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==
+ dependencies:
+ lru-cache "^6.0.0"
+
+slice-ansi@^4.0.0:
+ version "4.0.0"
+ resolved "https://registry.yarnpkg.com/slice-ansi/-/slice-ansi-4.0.0.tgz#500e8dd0fd55b05815086255b3195adf2a45fe6b"
+ integrity sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ==
+ dependencies:
+ ansi-styles "^4.0.0"
+ astral-regex "^2.0.0"
+ is-fullwidth-code-point "^3.0.0"
+
+solhint-plugin-prettier@^0.0.5:
+ version "0.0.5"
+ resolved "https://registry.yarnpkg.com/solhint-plugin-prettier/-/solhint-plugin-prettier-0.0.5.tgz#e3b22800ba435cd640a9eca805a7f8bc3e3e6a6b"
+ integrity sha512-7jmWcnVshIrO2FFinIvDQmhQpfpS2rRRn3RejiYgnjIE68xO2bvrYvjqVNfrio4xH9ghOqn83tKuTzLjEbmGIA==
+ dependencies:
+ prettier-linter-helpers "^1.0.0"
+
+solhint@^3.5.1:
+ version "3.6.2"
+ resolved "https://registry.yarnpkg.com/solhint/-/solhint-3.6.2.tgz#2b2acbec8fdc37b2c68206a71ba89c7f519943fe"
+ integrity sha512-85EeLbmkcPwD+3JR7aEMKsVC9YrRSxd4qkXuMzrlf7+z2Eqdfm1wHWq1ffTuo5aDhoZxp2I9yF3QkxZOxOL7aQ==
+ dependencies:
+ "@solidity-parser/parser" "^0.16.0"
+ ajv "^6.12.6"
+ antlr4 "^4.11.0"
+ ast-parents "^0.0.1"
+ chalk "^4.1.2"
+ commander "^10.0.0"
+ cosmiconfig "^8.0.0"
+ fast-diff "^1.2.0"
+ glob "^8.0.3"
+ ignore "^5.2.4"
+ js-yaml "^4.1.0"
+ lodash "^4.17.21"
+ pluralize "^8.0.0"
+ semver "^7.5.2"
+ strip-ansi "^6.0.1"
+ table "^6.8.1"
+ text-table "^0.2.0"
+ optionalDependencies:
+ prettier "^2.8.3"
+
+solidity-comments-extractor@^0.0.8:
+ version "0.0.8"
+ resolved "https://registry.yarnpkg.com/solidity-comments-extractor/-/solidity-comments-extractor-0.0.8.tgz#f6e148ab0c49f30c1abcbecb8b8df01ed8e879f8"
+ integrity sha512-htM7Vn6LhHreR+EglVMd2s+sZhcXAirB1Zlyrv5zBuTxieCvjfnRpd7iZk75m/u6NOlEyQ94C6TWbBn2cY7w8g==
+
+string-format@^2.0.0:
+ version "2.0.0"
+ resolved "https://registry.yarnpkg.com/string-format/-/string-format-2.0.0.tgz#f2df2e7097440d3b65de31b6d40d54c96eaffb9b"
+ integrity sha512-bbEs3scLeYNXLecRRuk6uJxdXUSj6le/8rNPHChIJTn2V79aXVTR1EH2OH5zLKKoz0V02fOUKZZcw01pLUShZA==
+
+string-width@^4.2.3:
+ version "4.2.3"
+ resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010"
+ integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==
+ dependencies:
+ emoji-regex "^8.0.0"
+ is-fullwidth-code-point "^3.0.0"
+ strip-ansi "^6.0.1"
+
+string_decoder@^1.1.1:
+ version "1.3.0"
+ resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.3.0.tgz#42f114594a46cf1a8e30b0a84f56c78c3edac21e"
+ integrity sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==
+ dependencies:
+ safe-buffer "~5.2.0"
+
+strip-ansi@^6.0.1:
+ version "6.0.1"
+ resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9"
+ integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==
+ dependencies:
+ ansi-regex "^5.0.1"
+
+strip-hex-prefix@1.0.0:
+ version "1.0.0"
+ resolved "https://registry.yarnpkg.com/strip-hex-prefix/-/strip-hex-prefix-1.0.0.tgz#0c5f155fef1151373377de9dbb588da05500e36f"
+ integrity sha512-q8d4ue7JGEiVcypji1bALTos+0pWtyGlivAWyPuTkHzuTCJqrK9sWxYQZUq6Nq3cuyv3bm734IhHvHtGGURU6A==
+ dependencies:
+ is-hex-prefixed "1.0.0"
+
+supports-color@^5.3.0:
+ version "5.5.0"
+ resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-5.5.0.tgz#e2e69a44ac8772f78a1ec0b35b689df6530efc8f"
+ integrity sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==
+ dependencies:
+ has-flag "^3.0.0"
+
+supports-color@^7.1.0:
+ version "7.2.0"
+ resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-7.2.0.tgz#1b7dcdcb32b8138801b3e478ba6a51caa89648da"
+ integrity sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==
+ dependencies:
+ has-flag "^4.0.0"
+
+symbol-observable@^4.0.0:
+ version "4.0.0"
+ resolved "https://registry.yarnpkg.com/symbol-observable/-/symbol-observable-4.0.0.tgz#5b425f192279e87f2f9b937ac8540d1984b39205"
+ integrity sha512-b19dMThMV4HVFynSAM1++gBHAbk2Tc/osgLIBZMKsyqh34jb2e8Os7T6ZW/Bt3pJFdBTd2JwAnAAEQV7rSNvcQ==
+
+table-layout@^1.0.2:
+ version "1.0.2"
+ resolved "https://registry.yarnpkg.com/table-layout/-/table-layout-1.0.2.tgz#c4038a1853b0136d63365a734b6931cf4fad4a04"
+ integrity sha512-qd/R7n5rQTRFi+Zf2sk5XVVd9UQl6ZkduPFC3S7WEGJAmetDTjY3qPN50eSKzwuzEyQKy5TN2TiZdkIjos2L6A==
+ dependencies:
+ array-back "^4.0.1"
+ deep-extend "~0.6.0"
+ typical "^5.2.0"
+ wordwrapjs "^4.0.0"
+
+table@^6.8.1:
+ version "6.8.2"
+ resolved "https://registry.yarnpkg.com/table/-/table-6.8.2.tgz#c5504ccf201213fa227248bdc8c5569716ac6c58"
+ integrity sha512-w2sfv80nrAh2VCbqR5AK27wswXhqcck2AhfnNW76beQXskGZ1V12GwS//yYVa3d3fcvAip2OUnbDAjW2k3v9fA==
+ dependencies:
+ ajv "^8.0.1"
+ lodash.truncate "^4.4.2"
+ slice-ansi "^4.0.0"
+ string-width "^4.2.3"
+ strip-ansi "^6.0.1"
+
+text-table@^0.2.0:
+ version "0.2.0"
+ resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4"
+ integrity sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==
+
+tiny-invariant@^1.1.0:
+ version "1.3.3"
+ resolved "https://registry.yarnpkg.com/tiny-invariant/-/tiny-invariant-1.3.3.tgz#46680b7a873a0d5d10005995eb90a70d74d60127"
+ integrity sha512-+FbBPE1o9QAYvviau/qC5SE3caw21q3xkvWKBtja5vgqOWIHHJ3ioaq1VPfn/Szqctz2bU/oYeKd9/z5BL+PVg==
+
+tr46@~0.0.3:
+ version "0.0.3"
+ resolved "https://registry.yarnpkg.com/tr46/-/tr46-0.0.3.tgz#8184fd347dac9cdc185992f3a6622e14b9d9ab6a"
+ integrity sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==
+
+treeify@^1.1.0:
+ version "1.1.0"
+ resolved "https://registry.yarnpkg.com/treeify/-/treeify-1.1.0.tgz#4e31c6a463accd0943879f30667c4fdaff411bb8"
+ integrity sha512-1m4RA7xVAJrSGrrXGs0L3YTwyvBs2S8PbRHaLZAkFw7JR8oIFwYtysxlBZhYIa7xSyiYJKZ3iGrrk55cGA3i9A==
+
+ts-command-line-args@^2.2.0:
+ version "2.5.1"
+ resolved "https://registry.yarnpkg.com/ts-command-line-args/-/ts-command-line-args-2.5.1.tgz#e64456b580d1d4f6d948824c274cf6fa5f45f7f0"
+ integrity sha512-H69ZwTw3rFHb5WYpQya40YAX2/w7Ut75uUECbgBIsLmM+BNuYnxsltfyyLMxy6sEeKxgijLTnQtLd0nKd6+IYw==
+ dependencies:
+ chalk "^4.1.0"
+ command-line-args "^5.1.1"
+ command-line-usage "^6.1.0"
+ string-format "^2.0.0"
+
+ts-essentials@^7.0.1:
+ version "7.0.3"
+ resolved "https://registry.yarnpkg.com/ts-essentials/-/ts-essentials-7.0.3.tgz#686fd155a02133eedcc5362dc8b5056cde3e5a38"
+ integrity sha512-8+gr5+lqO3G84KdiTSMRLtuyJ+nTBVRKuCrK4lidMPdVeEp0uqC875uE5NMcaA7YYMN7XsNiFQuMvasF8HT/xQ==
+
+ts-invariant@^0.10.3:
+ version "0.10.3"
+ resolved "https://registry.yarnpkg.com/ts-invariant/-/ts-invariant-0.10.3.tgz#3e048ff96e91459ffca01304dbc7f61c1f642f6c"
+ integrity sha512-uivwYcQaxAucv1CzRp2n/QdYPo4ILf9VXgH19zEIjFx2EJufV16P0JtJVpYHy89DItG6Kwj2oIUjrcK5au+4tQ==
+ dependencies:
+ tslib "^2.1.0"
+
+tslib@^2.1.0, tslib@^2.3.0:
+ version "2.6.2"
+ resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.6.2.tgz#703ac29425e7b37cd6fd456e92404d46d1f3e4ae"
+ integrity sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==
+
+typechain@^8.3.2:
+ version "8.3.2"
+ resolved "https://registry.yarnpkg.com/typechain/-/typechain-8.3.2.tgz#1090dd8d9c57b6ef2aed3640a516bdbf01b00d73"
+ integrity sha512-x/sQYr5w9K7yv3es7jo4KTX05CLxOf7TRWwoHlrjRh8H82G64g+k7VuWPJlgMo6qrjfCulOdfBjiaDtmhFYD/Q==
+ dependencies:
+ "@types/prettier" "^2.1.1"
+ debug "^4.3.1"
+ fs-extra "^7.0.0"
+ glob "7.1.7"
+ js-sha3 "^0.8.0"
+ lodash "^4.17.15"
+ mkdirp "^1.0.4"
+ prettier "^2.3.1"
+ ts-command-line-args "^2.2.0"
+ ts-essentials "^7.0.1"
+
+typical@^4.0.0:
+ version "4.0.0"
+ resolved "https://registry.yarnpkg.com/typical/-/typical-4.0.0.tgz#cbeaff3b9d7ae1e2bbfaf5a4e6f11eccfde94fc4"
+ integrity sha512-VAH4IvQ7BDFYglMd7BPRDfLgxZZX4O4TFcRDA6EN5X7erNJJq+McIEp8np9aVtxrCJ6qx4GTYVfOWNjcqwZgRw==
+
+typical@^5.2.0:
+ version "5.2.0"
+ resolved "https://registry.yarnpkg.com/typical/-/typical-5.2.0.tgz#4daaac4f2b5315460804f0acf6cb69c52bb93066"
+ integrity sha512-dvdQgNDNJo+8B2uBQoqdb11eUCE1JQXhvjC/CZtgvZseVd5TYMXnq0+vuUemXbd/Se29cTaUuPX3YIc2xgbvIg==
+
+universalify@^0.1.0:
+ version "0.1.2"
+ resolved "https://registry.yarnpkg.com/universalify/-/universalify-0.1.2.tgz#b646f69be3942dabcecc9d6639c80dc105efaa66"
+ integrity sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==
+
+uri-js@^4.2.2:
+ version "4.4.1"
+ resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz#9b1a52595225859e55f669d928f88c6c57f2a77e"
+ integrity sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==
+ dependencies:
+ punycode "^2.1.0"
+
+utf8@3.0.0:
+ version "3.0.0"
+ resolved "https://registry.yarnpkg.com/utf8/-/utf8-3.0.0.tgz#f052eed1364d696e769ef058b183df88c87f69d1"
+ integrity sha512-E8VjFIQ/TyQgp+TZfS6l8yp/xWppSAHzidGiRrqe4bK4XP9pTRyKFgGJpO3SN7zdX4DeomTrwaseCHovfpFcqQ==
+
+util-deprecate@^1.0.1:
+ version "1.0.2"
+ resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf"
+ integrity sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==
+
+web3-utils@^1.3.4:
+ version "1.10.4"
+ resolved "https://registry.yarnpkg.com/web3-utils/-/web3-utils-1.10.4.tgz#0daee7d6841641655d8b3726baf33b08eda1cbec"
+ integrity sha512-tsu8FiKJLk2PzhDl9fXbGUWTkkVXYhtTA+SmEFkKft+9BgwLxfCRpU96sWv7ICC8zixBNd3JURVoiR3dUXgP8A==
+ dependencies:
+ "@ethereumjs/util" "^8.1.0"
+ bn.js "^5.2.1"
+ ethereum-bloom-filters "^1.0.6"
+ ethereum-cryptography "^2.1.2"
+ ethjs-unit "0.1.6"
+ number-to-bn "1.7.0"
+ randombytes "^2.1.0"
+ utf8 "3.0.0"
+
+webidl-conversions@^3.0.0:
+ version "3.0.1"
+ resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-3.0.1.tgz#24534275e2a7bc6be7bc86611cc16ae0a5654871"
+ integrity sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==
+
+whatwg-url@^5.0.0:
+ version "5.0.0"
+ resolved "https://registry.yarnpkg.com/whatwg-url/-/whatwg-url-5.0.0.tgz#966454e8765462e37644d3626f6742ce8b70965d"
+ integrity sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==
+ dependencies:
+ tr46 "~0.0.3"
+ webidl-conversions "^3.0.0"
+
+wordwrapjs@^4.0.0:
+ version "4.0.1"
+ resolved "https://registry.yarnpkg.com/wordwrapjs/-/wordwrapjs-4.0.1.tgz#d9790bccfb110a0fc7836b5ebce0937b37a8b98f"
+ integrity sha512-kKlNACbvHrkpIw6oPeYDSmdCTu2hdMHoyXLTcUKala++lx5Y+wjJ/e474Jqv5abnVmwxw08DiTuHmw69lJGksA==
+ dependencies:
+ reduce-flatten "^2.0.0"
+ typical "^5.2.0"
+
+wrappy@1:
+ version "1.0.2"
+ resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f"
+ integrity sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==
+
+ws@7.4.6:
+ version "7.4.6"
+ resolved "https://registry.yarnpkg.com/ws/-/ws-7.4.6.tgz#5654ca8ecdeee47c33a9a4bf6d28e2be2980377c"
+ integrity sha512-YmhHDO4MzaDLB+M9ym/mDA5z0naX8j7SIlT8f8z+I0VtzsRbekxEutHSme7NPS2qE8StCYQNUnfWdXta/Yu85A==
+
+yallist@^4.0.0:
+ version "4.0.0"
+ resolved "https://registry.yarnpkg.com/yallist/-/yallist-4.0.0.tgz#9bb92790d9c0effec63be73519e11a35019a3a72"
+ integrity sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==
+
+zen-observable-ts@^1.2.5:
+ version "1.2.5"
+ resolved "https://registry.yarnpkg.com/zen-observable-ts/-/zen-observable-ts-1.2.5.tgz#6c6d9ea3d3a842812c6e9519209365a122ba8b58"
+ integrity sha512-QZWQekv6iB72Naeake9hS1KxHlotfRpe+WGNbNx5/ta+R3DNjVO2bswf63gXlWDcs+EMd7XY8HfVQyP1X6T4Zg==
+ dependencies:
+ zen-observable "0.8.15"
+
+zen-observable@0.8.15:
+ version "0.8.15"
+ resolved "https://registry.yarnpkg.com/zen-observable/-/zen-observable-0.8.15.tgz#96415c512d8e3ffd920afd3889604e30b9eaac15"
+ integrity sha512-PQ2PC7R9rslx84ndNBZB/Dkv8V8fZEpk83RLgXtYd0fwUgEjseMn1Dgajh2x6S8QbZAFa9p2qVCEuYZNgve0dQ==