diff --git a/.env.example b/.env.example new file mode 100644 index 0000000..2620949 --- /dev/null +++ b/.env.example @@ -0,0 +1,14 @@ +## Add URI and BIP39 mnemonic, and etherscan API key for every network that you plan to use +## These are used in `foundry.toml` (see `nodeUrl` and `accounts` calls in network definitions) + +#ETH_NODE_URI_MAINNET="" +#MNEMONIC_MAINNET="" +#MAINNET_ETHERSCAN_API_KEY="" + +#ETH_NODE_URI_POLYGON="" +#MNEMONIC_POLYGON="" +#POLYGON_ETHERSCAN_API_KEY="" + +#ETH_NODE_URI_FORK=http://localhost:8545 +#MNEMONIC_FORK="" + 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..a75ce1b --- /dev/null +++ b/.github/workflows/ci-deep.yml @@ -0,0 +1,177 @@ +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: Install dependencies + run: yarn install + + - 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: 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 + + - 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 + 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 }} + + - name: "Add Unit Test Summary" + run: | + echo "## Unit test result" >> $GITHUB_STEP_SUMMARY + echo "✅ Passed" >> $GITHUB_STEP_SUMMARY + + 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 + 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: ${{ inputs.invariantRuns || '300' }} + FOUNDRY_INVARIANT_DEPTH: ${{ 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 }} + + - name: "Add Invariant Test Summary" + run: | + echo "## Invariant test result" >> $GITHUB_STEP_SUMMARY + echo "✅ Passed" >> $GITHUB_STEP_SUMMARY + + 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 + 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: ${{ 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 }} + + - name: "Add Fuzz Test Summary" + run: | + echo "## Fuzz test result" >> $GITHUB_STEP_SUMMARY + echo "✅ Passed" >> $GITHUB_STEP_SUMMARY \ No newline at end of file diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml new file mode 100644 index 0000000..20fef55 --- /dev/null +++ b/.github/workflows/ci.yml @@ -0,0 +1,232 @@ +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: Install dependencies + run: yarn install + + - 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: 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 + + - 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 + 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 }} + + - name: "Add Unit Test Summary" + run: | + echo "## Unit test result" >> $GITHUB_STEP_SUMMARY + echo "✅ Passed" >> $GITHUB_STEP_SUMMARY + + 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 + 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 }} + + - name: "Add Invariant Test Summary" + run: | + echo "## Invariant test result" >> $GITHUB_STEP_SUMMARY + echo "✅ Passed" >> $GITHUB_STEP_SUMMARY + + 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 + key: "build-${{ github.sha }}" + + - name: Install Foundry + uses: foundry-rs/foundry-toolchain@v1 + with: + version: nightly + + - name: Run Foundry tests + run: npm run 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 }} + + - name: "Add Fuzz Test Summary" + run: | + echo "## Fuzz test result" >> $GITHUB_STEP_SUMMARY + echo "✅ Passed" >> $GITHUB_STEP_SUMMARY + + coverage: + needs: ["build", "lint"] + runs-on: "ubuntu-latest" + steps: + - uses: actions/checkout@v3 + with: + submodules: "recursive" + + - 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" + + - 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 build --build-info --force + + - name: "Run Slither analysis" + uses: "crytic/slither-action@v0.3.0" + id: slither + with: + fail-on: "none" + sarif: "results.sarif" + ignore-compile: true + + - 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/.gitignore b/.gitignore new file mode 100644 index 0000000..cc31e42 --- /dev/null +++ b/.gitignore @@ -0,0 +1,38 @@ +## Defaults +__pycache__ +.idea +.DS_Store +.deps +.docs +.env +node_modules +venv + +# Build output +slither-audit.txt +slither + +# Test output +coverage +coverage.json +lcov.info + +# Running output +gas-report.txt +gasReporterOutput.json +addresses.json +blockchain_db +yarn-error.log +broadcast + +# deployments +deployments/localhost +deployments/mainnetForkRemote + +# bin +bin + + +# foundry +/out +/cache diff --git a/.gitmodules b/.gitmodules new file mode 100644 index 0000000..dcf0d19 --- /dev/null +++ b/.gitmodules @@ -0,0 +1,12 @@ +[submodule "lib/openzeppelin-contracts"] + path = lib/openzeppelin-contracts + url = https://github.com/OpenZeppelin/openzeppelin-contracts +[submodule "lib/openzeppelin-contracts-upgradeable"] + path = lib/openzeppelin-contracts-upgradeable + url = https://github.com/OpenZeppelin/openzeppelin-contracts-upgradeable +[submodule "lib/forge-std"] + path = lib/forge-std + url = https://github.com/foundry-rs/forge-std +[submodule "lib/utils"] + path = lib/utils + url = https://github.com/AngleProtocol/utils diff --git a/.prettierignore b/.prettierignore new file mode 100644 index 0000000..e4006e6 --- /dev/null +++ b/.prettierignore @@ -0,0 +1,3 @@ +lib +cache +out \ 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..99c3260 --- /dev/null +++ b/.solhint.json @@ -0,0 +1,28 @@ +{ + "extends": "solhint:recommended", + "plugins": ["prettier"], + "rules": { + "max-line-length": ["error", 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/settings.json b/.vscode/settings.json new file mode 100644 index 0000000..e6b354a --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,7 @@ +{ + "solidity.compileUsingRemoteVersion": "0.8.20", + "[solidity]": { + "editor.defaultFormatter": "JuanBlanco.solidity", + "editor.formatOnSave": true + }, +} 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 bb6a57e..07aa577 100644 --- a/README.md +++ b/README.md @@ -1 +1,166 @@ -# angle-dao \ No newline at end of file +# Angle Angle DAO + +[![CI](https://github.com/AngleProtocol/boilerplate/actions/workflows/ci.yml/badge.svg)](https://github.com/AngleProtocol/boilerplate/actions) +[![Coverage](https://codecov.io/gh/AngleProtocol/boilerplate/branch/main/graph/badge.svg)](https://codecov.io/gh/AngleProtocol/boilerplate) + +This repository proposes a template that is based on foundry frameworks. It also provides templates for EVM compatible smart contracts (in `./contracts/example`), tests and deployment scripts. + +## Starting + +### Install packages + +You can install all dependencies by running + +```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 +``` + +### Foundry on Docker 🐳 + +**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 +``` + +To run the container: + +```bash +docker run -it --rm -v $(pwd):/app -w /app foundry sh +``` + +Then you are inside the container and can run Foundry’s commands. + +### Tests + +You can run tests as follows: + +```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 +``` + +You can also list tests: + +```bash +forge test --list +forge test --list --json --match-test "testXXX*" +``` + +### Deploying + +There is an example script in the `scripts/foundry` folder. Then you can run: + +```bash +yarn foundry:deploy --rpc-url +``` + +Example: + +```bash +yarn foundry:deploy scripts/foundry/DeployMockAgEUR.s.sol --rpc-url goerli +``` + +### Coverage + +We recommend the use of this [vscode extension](ryanluker.vscode-coverage-gutters). + +```bash +yarn hardhat:coverage +yarn foundry:coverage +``` + +### 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. + +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/dao/ANGLE.sol b/contracts/dao/ANGLE.sol new file mode 100644 index 0000000..6fc747c --- /dev/null +++ b/contracts/dao/ANGLE.sol @@ -0,0 +1,60 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts/token/ERC20/extensions/ERC20Votes.sol"; + +/// @title ANGLE +/// @author Forked but improved from https://github.com/compound-finance/compound-protocol/tree/master/contracts/Governance +/// by Angle Core Team +/// @notice Governance token of Angle's protocol +contract ANGLE is ERC20Votes { + /// @notice An event that is emitted when the minter address is changed + event MinterChanged(address minter, address newMinter); + + /// @notice Minimum time between mints + uint32 public constant MINIMUM_BETWEEN_MINTS = 1 days * 30; + + /// @notice Cap on the percentage of `totalSupply()` that can be minted at each mint + uint8 public constant MAX_MINT = 1; + + /// @notice Address which may mint new tokens + address public minter; + + /// @notice The timestamp after which minting may occur + uint256 public mintingAllowedAfter; + + /// @notice Constructs a new ANGLE token + /// @param account Initial account to grant all the tokens to + /// @param minter_ Account with minting ability + constructor(address account, address minter_) ERC20Permit("ANGLE") ERC20("ANGLE", "ANGLE") { + require(account != address(0) && minter_ != address(0), "0"); + _mint(account, 1_000_000_000e18); // 1 billion ANGLE + minter = minter_; + emit MinterChanged(address(0), minter); + mintingAllowedAfter = block.timestamp; + } + + /// @notice Changes the minter address + /// @param minter_ Address of the new minter + function setMinter(address minter_) external { + require(msg.sender == minter, "67"); + require(minter_ != address(0), "0"); + emit MinterChanged(minter, minter_); + minter = minter_; + } + + /// @notice Mints new tokens + /// @param dst Address of the destination account + /// @param amount Number of tokens to be minted + function mint(address dst, uint256 amount) external { + require(msg.sender == minter, "68"); + require(block.timestamp >= mintingAllowedAfter, "69"); + require(amount <= (totalSupply() * MAX_MINT) / 100, "70"); + // Record the mint + mintingAllowedAfter = block.timestamp + MINIMUM_BETWEEN_MINTS; + + // Mint the amount + _mint(dst, amount); + } +} diff --git a/contracts/dao/veANGLE.vy b/contracts/dao/veANGLE.vy new file mode 100644 index 0000000..674f991 --- /dev/null +++ b/contracts/dao/veANGLE.vy @@ -0,0 +1,710 @@ +# @version 0.2.16 +""" +@title Voting Escrow +@author Angle Protocol +@license MIT +@notice Votes have a weight depending on time, so that users are + committed to the future of (whatever they are voting for) +@dev Vote weight decays linearly over time. Lock time cannot be + more than `MAXTIME` (4 years). +""" + +# Original idea and credit: +# Curve Finance's veCRV +# https://github.com/curvefi/curve-dao-contracts/blob/master/contracts/VotingEscrow.vy +# veANGLE is a fork with only one view functions added to it to make veANGLE compatible +# with Compound governance system. The references to the controller have also been removed + +# Voting escrow to have time-weighted votes +# Votes have a weight depending on time, so that users are committed +# to the future of (whatever they are voting for). +# The weight in this implementation is linear, and lock cannot be more than maxtime: +# w ^ +# 1 + / +# | / +# | / +# | / +# |/ +# 0 +--------+------> time +# maxtime (4 years?) + +struct Point: + bias: int128 + slope: int128 # - dweight / dt + ts: uint256 + blk: uint256 # block +# We cannot really do block numbers per se b/c slope is per time, not per block +# and per block could be fairly bad b/c Ethereum changes blocktimes. +# What we can do is to extrapolate ***At functions + +struct LockedBalance: + amount: int128 + end: uint256 + + +interface ERC20: + def decimals() -> uint256: view + def name() -> String[64]: view + def symbol() -> String[32]: view + def transfer(to: address, amount: uint256) -> bool: nonpayable + def transferFrom(spender: address, to: address, amount: uint256) -> bool: nonpayable + + +# Interface for checking whether address belongs to a whitelisted +# type of a smart wallet. +# When new types are added - the whole contract is changed +# The check() method is modifying to be able to use caching +# for individual wallet addresses +interface SmartWalletChecker: + def check(addr: address) -> bool: nonpayable + +DEPOSIT_FOR_TYPE: constant(int128) = 0 +CREATE_LOCK_TYPE: constant(int128) = 1 +INCREASE_LOCK_AMOUNT: constant(int128) = 2 +INCREASE_UNLOCK_TIME: constant(int128) = 3 + + +event CommitOwnership: + admin: address + +event ApplyOwnership: + admin: address + +event Deposit: + provider: indexed(address) + value: uint256 + locktime: indexed(uint256) + type: int128 + ts: uint256 + +event Withdraw: + provider: indexed(address) + value: uint256 + ts: uint256 + +event Supply: + prevSupply: uint256 + supply: uint256 + + +WEEK: constant(uint256) = 7 * 86400 # all future times are rounded by week +MAXTIME: constant(uint256) = 4 * 365 * 86400 # 4 years +MULTIPLIER: constant(uint256) = 10 ** 18 + +token: public(address) +supply: public(uint256) + +locked: public(HashMap[address, LockedBalance]) + +epoch: public(uint256) +point_history: public(Point[100000000000000000000000000000]) # epoch -> unsigned point +user_point_history: public(HashMap[address, Point[1000000000]]) # user -> Point[user_epoch] +user_point_epoch: public(HashMap[address, uint256]) +slope_changes: public(HashMap[uint256, int128]) # time -> signed slope change + +name: public(String[64]) +symbol: public(String[32]) +decimals: public(uint256) + +# Checker for whitelisted (smart contract) wallets which are allowed to deposit +# The goal is to prevent tokenizing the escrow +future_smart_wallet_checker: public(address) +smart_wallet_checker: public(address) + +admin: public(address) # Can and will be a smart contract +future_admin: public(address) + +initialized: public(bool) + + +@external +def __init__(): + """ + @notice Contract constructor + @dev The contract has an initializer to prevent the take over of the implementation + """ + assert self.initialized == False #dev: contract is already initialized + self.initialized = True + +@external +def initialize(_admin: address, token_addr: address, _smart_wallet_checker: address, _name: String[64], _symbol: String[32]): + """ + @notice Contract initializer + @param _admin Future veANGLE admin + @param token_addr `ERC20ANGLE` token address + @param _smart_wallet_checker Future smart wallet checker contract + @param _name Token name + @param _symbol Token symbol + """ + assert self.initialized == False #dev: contract is already initialized + self.initialized = True + assert _admin!= ZERO_ADDRESS #dev: admin cannot be the 0 address + self.admin = _admin + self.token = token_addr + self.smart_wallet_checker = _smart_wallet_checker + self.point_history[0].blk = block.number + self.point_history[0].ts = block.timestamp + + _decimals: uint256 = ERC20(token_addr).decimals() + assert _decimals <= 255 + self.decimals = _decimals + + self.name = _name + self.symbol = _symbol + + +@external +def commit_transfer_ownership(addr: address): + """ + @notice Transfer ownership of VotingEscrow contract to `addr` + @param addr Address to have ownership transferred to + """ + assert msg.sender == self.admin # dev: admin only + assert addr != ZERO_ADDRESS # dev: future admin cannot be the 0 address + self.future_admin = addr + log CommitOwnership(addr) + +@external +def accept_transfer_ownership(): + """ + @notice Accept a pending ownership transfer + """ + _admin: address = self.future_admin + assert msg.sender == _admin # dev: future admin only + + self.admin = _admin + log ApplyOwnership(_admin) + +@external +def apply_transfer_ownership(): + """ + @notice Apply ownership transfer + """ + assert msg.sender == self.admin # dev: admin only + _admin: address = self.future_admin + assert _admin != ZERO_ADDRESS # dev: admin not set + self.admin = _admin + log ApplyOwnership(_admin) + + +@external +def commit_smart_wallet_checker(addr: address): + """ + @notice Set an external contract to check for approved smart contract wallets + @param addr Address of Smart contract checker + """ + assert msg.sender == self.admin + self.future_smart_wallet_checker = addr + + +@external +def apply_smart_wallet_checker(): + """ + @notice Apply setting external contract to check approved smart contract wallets + """ + assert msg.sender == self.admin + self.smart_wallet_checker = self.future_smart_wallet_checker + + +@internal +def assert_not_contract(addr: address): + """ + @notice Check if the call is from a whitelisted smart contract, revert if not + @param addr Address to be checked + """ + if addr != tx.origin: + checker: address = self.smart_wallet_checker + if checker != ZERO_ADDRESS: + if SmartWalletChecker(checker).check(addr): + return + raise "Smart contract depositors not allowed" + + +@external +@view +def get_last_user_slope(addr: address) -> int128: + """ + @notice Get the most recently recorded rate of voting power decrease for `addr` + @param addr Address of the user wallet + @return Value of the slope + """ + uepoch: uint256 = self.user_point_epoch[addr] + return self.user_point_history[addr][uepoch].slope + + +@external +@view +def user_point_history__ts(_addr: address, _idx: uint256) -> uint256: + """ + @notice Get the timestamp for checkpoint `_idx` for `_addr` + @param _addr User wallet address + @param _idx User epoch number + @return Epoch time of the checkpoint + """ + return self.user_point_history[_addr][_idx].ts + + +@external +@view +def locked__end(_addr: address) -> uint256: + """ + @notice Get timestamp when `_addr`'s lock finishes + @param _addr User wallet + @return Epoch time of the lock end + """ + return self.locked[_addr].end + + +@internal +def _checkpoint(addr: address, old_locked: LockedBalance, new_locked: LockedBalance): + """ + @notice Record global and per-user data to checkpoint + @param addr User's wallet address. No user checkpoint if 0x0 + @param old_locked Pevious locked amount / end lock time for the user + @param new_locked New locked amount / end lock time for the user + """ + u_old: Point = empty(Point) + u_new: Point = empty(Point) + old_dslope: int128 = 0 + new_dslope: int128 = 0 + _epoch: uint256 = self.epoch + + if addr != ZERO_ADDRESS: + # Calculate slopes and biases + # Kept at zero when they have to + if old_locked.end > block.timestamp and old_locked.amount > 0: + u_old.slope = old_locked.amount / MAXTIME + u_old.bias = u_old.slope * convert(old_locked.end - block.timestamp, int128) + if new_locked.end > block.timestamp and new_locked.amount > 0: + u_new.slope = new_locked.amount / MAXTIME + u_new.bias = u_new.slope * convert(new_locked.end - block.timestamp, int128) + + # Read values of scheduled changes in the slope + # old_locked.end can be in the past and in the future + # new_locked.end can ONLY by in the FUTURE unless everything expired: than zeros + old_dslope = self.slope_changes[old_locked.end] + if new_locked.end != 0: + if new_locked.end == old_locked.end: + new_dslope = old_dslope + else: + new_dslope = self.slope_changes[new_locked.end] + + last_point: Point = Point({bias: 0, slope: 0, ts: block.timestamp, blk: block.number}) + if _epoch > 0: + last_point = self.point_history[_epoch] + last_checkpoint: uint256 = last_point.ts + # initial_last_point is used for extrapolation to calculate block number + # (approximately, for *At methods) and save them + # as we cannot figure that out exactly from inside the contract + initial_last_point: Point = last_point + block_slope: uint256 = 0 # dblock/dt + if block.timestamp > last_point.ts: + block_slope = MULTIPLIER * (block.number - last_point.blk) / (block.timestamp - last_point.ts) + # If last point is already recorded in this block, slope=0 + # But that's ok b/c we know the block in such case + + # Go over weeks to fill history and calculate what the current point is + t_i: uint256 = (last_checkpoint / WEEK) * WEEK + for i in range(255): + # Hopefully it won't happen that this won't get used in 5 years! + # If it does, users will be able to withdraw but vote weight will be broken + t_i += WEEK + d_slope: int128 = 0 + if t_i > block.timestamp: + t_i = block.timestamp + else: + d_slope = self.slope_changes[t_i] + last_point.bias -= last_point.slope * convert(t_i - last_checkpoint, int128) + last_point.slope += d_slope + if last_point.bias < 0: # This can happen + last_point.bias = 0 + if last_point.slope < 0: # This cannot happen - just in case + last_point.slope = 0 + last_checkpoint = t_i + last_point.ts = t_i + last_point.blk = initial_last_point.blk + block_slope * (t_i - initial_last_point.ts) / MULTIPLIER + _epoch += 1 + if t_i == block.timestamp: + last_point.blk = block.number + break + else: + self.point_history[_epoch] = last_point + + self.epoch = _epoch + # Now point_history is filled until t=now + + if addr != ZERO_ADDRESS: + # If last point was in this block, the slope change has been applied already + # But in such case we have 0 slope(s) + last_point.slope += (u_new.slope - u_old.slope) + last_point.bias += (u_new.bias - u_old.bias) + if last_point.slope < 0: + last_point.slope = 0 + if last_point.bias < 0: + last_point.bias = 0 + + # Record the changed point into history + self.point_history[_epoch] = last_point + + if addr != ZERO_ADDRESS: + # Schedule the slope changes (slope is going down) + # We subtract new_user_slope from [new_locked.end] + # and add old_user_slope to [old_locked.end] + if old_locked.end > block.timestamp: + # old_dslope was - u_old.slope, so we cancel that + old_dslope += u_old.slope + if new_locked.end == old_locked.end: + old_dslope -= u_new.slope # It was a new deposit, not extension + self.slope_changes[old_locked.end] = old_dslope + + if new_locked.end > block.timestamp: + if new_locked.end > old_locked.end: + new_dslope -= u_new.slope # old slope disappeared at this point + self.slope_changes[new_locked.end] = new_dslope + # else: we recorded it already in old_dslope + + # Now handle user history + user_epoch: uint256 = self.user_point_epoch[addr] + 1 + + self.user_point_epoch[addr] = user_epoch + u_new.ts = block.timestamp + u_new.blk = block.number + self.user_point_history[addr][user_epoch] = u_new + + +@internal +def _deposit_for(_addr: address, _value: uint256, unlock_time: uint256, locked_balance: LockedBalance, type: int128, sender: address): + """ + @notice Deposit and lock tokens for a user + @param _addr User's wallet address + @param _value Amount to deposit + @param unlock_time New time when to unlock the tokens, or 0 if unchanged + @param locked_balance Previous locked amount / timestamp + """ + _locked: LockedBalance = locked_balance + supply_before: uint256 = self.supply + + self.supply = supply_before + _value + old_locked: LockedBalance = _locked + # Adding to existing lock, or if a lock is expired - creating a new one + _locked.amount += convert(_value, int128) + if unlock_time != 0: + _locked.end = unlock_time + self.locked[_addr] = _locked + + # Possibilities: + # Both old_locked.end could be current or expired (>/< block.timestamp) + # value == 0 (extend lock) or value > 0 (add to lock or extend lock) + # _locked.end > block.timestamp (always) + self._checkpoint(_addr, old_locked, _locked) + + if _value != 0: + assert ERC20(self.token).transferFrom(sender, self, _value) + + log Deposit(_addr, _value, _locked.end, type, block.timestamp) + log Supply(supply_before, supply_before + _value) + + +@external +def checkpoint(): + """ + @notice Record global data to checkpoint + """ + self._checkpoint(ZERO_ADDRESS, empty(LockedBalance), empty(LockedBalance)) + + +@external +@nonreentrant('lock') +def deposit_for(_addr: address, _value: uint256): + """ + @notice Deposit `_value` tokens for `_addr` and add to the lock + @dev Anyone (even a smart contract) can deposit for someone else, but + cannot extend their locktime and deposit for a brand new user + @param _addr User's wallet address + @param _value Amount to add to user's lock + """ + _locked: LockedBalance = self.locked[_addr] + + assert _value > 0 # dev: need non-zero value + assert _locked.amount > 0, "No existing lock found" + assert _locked.end > block.timestamp, "Cannot add to expired lock. Withdraw" + + self._deposit_for(_addr, _value, 0, self.locked[_addr], DEPOSIT_FOR_TYPE, msg.sender) + + +@external +@nonreentrant('lock') +def create_lock(_value: uint256, _unlock_time: uint256): + """ + @notice Deposit `_value` tokens for `msg.sender` and lock until `_unlock_time` + @param _value Amount to deposit + @param _unlock_time Epoch time when tokens unlock, rounded down to whole weeks + """ + self.assert_not_contract(msg.sender) + unlock_time: uint256 = (_unlock_time / WEEK) * WEEK # Locktime is rounded down to weeks + _locked: LockedBalance = self.locked[msg.sender] + + assert _value > 0 # dev: need non-zero value + assert _locked.amount == 0, "Withdraw old tokens first" + assert unlock_time > block.timestamp, "Can only lock until time in the future" + assert unlock_time <= block.timestamp + MAXTIME, "Voting lock can be 4 years max" + + self._deposit_for(msg.sender, _value, unlock_time, _locked, CREATE_LOCK_TYPE, msg.sender) + + +@external +@nonreentrant('lock') +def increase_amount(_value: uint256): + """ + @notice Deposit `_value` additional tokens for `msg.sender` + without modifying the unlock time + @param _value Amount of tokens to deposit and add to the lock + """ + self.assert_not_contract(msg.sender) + _locked: LockedBalance = self.locked[msg.sender] + + assert _value > 0 # dev: need non-zero value + assert _locked.amount > 0, "No existing lock found" + assert _locked.end > block.timestamp, "Cannot add to expired lock. Withdraw" + + self._deposit_for(msg.sender, _value, 0, _locked, INCREASE_LOCK_AMOUNT, msg.sender) + + +@external +@nonreentrant('lock') +def increase_unlock_time(_unlock_time: uint256): + """ + @notice Extend the unlock time for `msg.sender` to `_unlock_time` + @param _unlock_time New epoch time for unlocking + """ + self.assert_not_contract(msg.sender) + _locked: LockedBalance = self.locked[msg.sender] + unlock_time: uint256 = (_unlock_time / WEEK) * WEEK # Locktime is rounded down to weeks + + assert _locked.end > block.timestamp, "Lock expired" + assert _locked.amount > 0, "Nothing is locked" + assert unlock_time > _locked.end, "Can only increase lock duration" + assert unlock_time <= block.timestamp + MAXTIME, "Voting lock can be 4 years max" + + self._deposit_for(msg.sender, 0, unlock_time, _locked, INCREASE_UNLOCK_TIME, msg.sender) + + +@external +@nonreentrant('lock') +def withdraw(): + """ + @notice Withdraw all tokens for `msg.sender` + @dev Only possible if the lock has expired + """ + _locked: LockedBalance = self.locked[msg.sender] + assert block.timestamp >= _locked.end, "The lock didn't expire" + value: uint256 = convert(_locked.amount, uint256) + + old_locked: LockedBalance = _locked + _locked.end = 0 + _locked.amount = 0 + self.locked[msg.sender] = _locked + supply_before: uint256 = self.supply + self.supply = supply_before - value + + # old_locked can have either expired <= timestamp or zero end + # _locked has only 0 end + # Both can have >= 0 amount + self._checkpoint(msg.sender, old_locked, _locked) + + assert ERC20(self.token).transfer(msg.sender, value) + + log Withdraw(msg.sender, value, block.timestamp) + log Supply(supply_before, supply_before - value) + + +# The following ERC20/minime-compatible methods are not real balanceOf and supply! +# They measure the weights for the purpose of voting, so they don't represent +# real coins. + +@internal +@view +def find_block_epoch(_block: uint256, max_epoch: uint256) -> uint256: + """ + @notice Binary search to estimate timestamp for block number + @param _block Block to find + @param max_epoch Don't go beyond this epoch + @return Approximate timestamp for block + """ + # Binary search + _min: uint256 = 0 + _max: uint256 = max_epoch + for i in range(128): # Will be always enough for 128-bit numbers + if _min >= _max: + break + _mid: uint256 = (_min + _max + 1) / 2 + if self.point_history[_mid].blk <= _block: + _min = _mid + else: + _max = _mid - 1 + return _min + + +@external +@view +def balanceOf(addr: address, _t: uint256 = block.timestamp) -> uint256: + """ + @notice Get the current voting power for `msg.sender` + @dev Adheres to the ERC20 `balanceOf` interface for Aragon compatibility + @param addr User wallet address + @param _t Epoch time to return voting power at + @return User voting power + """ + _epoch: uint256 = self.user_point_epoch[addr] + if _epoch == 0: + return 0 + else: + last_point: Point = self.user_point_history[addr][_epoch] + last_point.bias -= last_point.slope * convert(_t - last_point.ts, int128) + if last_point.bias < 0: + last_point.bias = 0 + return convert(last_point.bias, uint256) + + +@internal +@view +def _balanceOfAt(addr: address, _block: uint256) -> uint256: + """ + @notice measure voting power of `addr` at block height `_block` + @param addr User's wallet address + @param _block Block to calculate the voting power at + @return Voting power + """ + # Copying and pasting totalSupply code because Vyper cannot pass by + # reference yet + assert _block <= block.number + + # Binary search + _min: uint256 = 0 + _max: uint256 = self.user_point_epoch[addr] + for i in range(128): # Will be always enough for 128-bit numbers + if _min >= _max: + break + _mid: uint256 = (_min + _max + 1) / 2 + if self.user_point_history[addr][_mid].blk <= _block: + _min = _mid + else: + _max = _mid - 1 + + upoint: Point = self.user_point_history[addr][_min] + + max_epoch: uint256 = self.epoch + _epoch: uint256 = self.find_block_epoch(_block, max_epoch) + point_0: Point = self.point_history[_epoch] + d_block: uint256 = 0 + d_t: uint256 = 0 + if _epoch < max_epoch: + point_1: Point = self.point_history[_epoch + 1] + d_block = point_1.blk - point_0.blk + d_t = point_1.ts - point_0.ts + else: + d_block = block.number - point_0.blk + d_t = block.timestamp - point_0.ts + block_time: uint256 = point_0.ts + if d_block != 0: + block_time += d_t * (_block - point_0.blk) / d_block + + upoint.bias -= upoint.slope * convert(block_time - upoint.ts, int128) + if upoint.bias >= 0: + return convert(upoint.bias, uint256) + else: + return 0 + + +@external +@view +def balanceOfAt(addr: address, _block: uint256) -> uint256: + """ + @notice Measure voting power of `addr` at block height `_block` + @dev Adheres to MiniMe `balanceOfAt` interface: https://github.com/Giveth/minime + @param addr User's wallet address + @param _block Block to calculate the voting power at + @return Voting power + """ + return self._balanceOfAt(addr,_block) + + +@external +@view +def getPastVotes(addr: address, _block: uint256) -> uint256: + """ + @notice Measure voting power of `addr` at block height `_block` + @dev Adheres to ERC20Votes `getPastVotes` interface: @openzeppelin-contracts-upgradeable/blob/master/contracts/token/ERC20/extensions/ERC20VotesCompUpgradeable.sol + @param addr User's wallet address + @param _block Block to calculate the voting power at + @return Voting power + """ + return self._balanceOfAt(addr,_block) + + +@internal +@view +def supply_at(point: Point, t: uint256) -> uint256: + """ + @notice Calculate total voting power at some point in the past + @param point The point (bias/slope) to start search from + @param t Time to calculate the total voting power at + @return Total voting power at that time + """ + last_point: Point = point + t_i: uint256 = (last_point.ts / WEEK) * WEEK + for i in range(255): + t_i += WEEK + d_slope: int128 = 0 + if t_i > t: + t_i = t + else: + d_slope = self.slope_changes[t_i] + last_point.bias -= last_point.slope * convert(t_i - last_point.ts, int128) + if t_i == t: + break + last_point.slope += d_slope + last_point.ts = t_i + + if last_point.bias < 0: + last_point.bias = 0 + return convert(last_point.bias, uint256) + + +@external +@view +def totalSupply(t: uint256 = block.timestamp) -> uint256: + """ + @notice Calculate total voting power + @dev Adheres to the ERC20 `totalSupply` interface for Aragon compatibility + @return Total voting power + """ + _epoch: uint256 = self.epoch + last_point: Point = self.point_history[_epoch] + return self.supply_at(last_point, t) + + +@external +@view +def totalSupplyAt(_block: uint256) -> uint256: + """ + @notice Calculate total voting power at some point in the past + @param _block Block to calculate the total voting power at + @return Total voting power at `_block` + """ + assert _block <= block.number + _epoch: uint256 = self.epoch + target_epoch: uint256 = self.find_block_epoch(_block, _epoch) + + point: Point = self.point_history[target_epoch] + dt: uint256 = 0 + if target_epoch < _epoch: + point_next: Point = self.point_history[target_epoch + 1] + if point.blk != point_next.blk: + dt = (_block - point.blk) * (point_next.ts - point.ts) / (point_next.blk - point.blk) + else: + if point.blk != block.number: + dt = (_block - point.blk) * (block.timestamp - point.ts) / (block.number - point.blk) + # Now dt contains info on how far are we beyond point + + return self.supply_at(point, point.ts + dt) \ No newline at end of file diff --git a/contracts/external/AccessControl.sol b/contracts/external/AccessControl.sol new file mode 100644 index 0000000..bfa8c2a --- /dev/null +++ b/contracts/external/AccessControl.sol @@ -0,0 +1,233 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts/utils/Context.sol"; +import "@openzeppelin/contracts/utils/Strings.sol"; + +import "../interfaces/IAccessControl.sol"; + +/** + * @dev This contract is fully forked from OpenZeppelin `AccessControl`. + * The only difference is the removal of the ERC165 implementation as it's not + * needed in Angle. + * + * Contract module that allows children to implement role-based access + * control mechanisms. This is a lightweight version that doesn't allow enumerating role + * members except through off-chain means by accessing the contract event logs. Some + * applications may benefit from on-chain enumerability, for those cases see + * {AccessControlEnumerable}. + * + * Roles are referred to by their `bytes32` identifier. These should be exposed + * in the external API and be unique. The best way to achieve this is by + * using `public constant` hash digests: + * + * ``` + * bytes32 public constant MY_ROLE = keccak256("MY_ROLE"); + * ``` + * + * Roles can be used to represent a set of permissions. To restrict access to a + * function call, use {hasRole}: + * + * ``` + * function foo() public { + * require(hasRole(MY_ROLE, msg.sender)); + * ... + * } + * ``` + * + * Roles can be granted and revoked dynamically via the {grantRole} and + * {revokeRole} functions. Each role has an associated admin role, and only + * accounts that have a role's admin role can call {grantRole} and {revokeRole}. + * + * By default, the admin role for all roles is `DEFAULT_ADMIN_ROLE`, which means + * that only accounts with this role will be able to grant or revoke other + * roles. More complex role relationships can be created by using + * {_setRoleAdmin}. + * + * WARNING: The `DEFAULT_ADMIN_ROLE` is also its own admin: it has permission to + * grant and revoke this role. Extra precautions should be taken to secure + * accounts that have been granted it. + */ +abstract contract AccessControl is Context, IAccessControl { + struct RoleData { + mapping(address => bool) members; + bytes32 adminRole; + } + + mapping(bytes32 => RoleData) private _roles; + + bytes32 public constant DEFAULT_ADMIN_ROLE = 0x00; + + /** + * @dev Emitted when `newAdminRole` is set as ``role``'s admin role, replacing `previousAdminRole` + * + * `DEFAULT_ADMIN_ROLE` is the starting admin for all roles, despite + * {RoleAdminChanged} not being emitted signaling this. + * + * _Available since v3.1._ + */ + event RoleAdminChanged(bytes32 indexed role, bytes32 indexed previousAdminRole, bytes32 indexed newAdminRole); + + /** + * @dev Emitted when `account` is granted `role`. + * + * `sender` is the account that originated the contract call, an admin role + * bearer except when using {_setupRole}. + */ + event RoleGranted(bytes32 indexed role, address indexed account, address indexed sender); + + /** + * @dev Emitted when `account` is revoked `role`. + * + * `sender` is the account that originated the contract call: + * - if using `revokeRole`, it is the admin role bearer + * - if using `renounceRole`, it is the role bearer (i.e. `account`) + */ + event RoleRevoked(bytes32 indexed role, address indexed account, address indexed sender); + + /** + * @dev Modifier that checks that an account has a specific role. Reverts + * with a standardized message including the required role. + * + * The format of the revert reason is given by the following regular expression: + * + * /^AccessControl: account (0x[0-9a-f]{20}) is missing role (0x[0-9a-f]{32})$/ + * + * _Available since v4.1._ + */ + modifier onlyRole(bytes32 role) { + _checkRole(role, _msgSender()); + _; + } + + /** + * @dev Returns `true` if `account` has been granted `role`. + */ + function hasRole(bytes32 role, address account) public view override returns (bool) { + return _roles[role].members[account]; + } + + /** + * @dev Revert with a standard message if `account` is missing `role`. + * + * The format of the revert reason is given by the following regular expression: + * + * /^AccessControl: account (0x[0-9a-f]{20}) is missing role (0x[0-9a-f]{32})$/ + */ + function _checkRole(bytes32 role, address account) internal view { + if (!hasRole(role, account)) { + revert( + string( + abi.encodePacked( + "AccessControl: account ", + Strings.toHexString(uint160(account), 20), + " is missing role ", + Strings.toHexString(uint256(role), 32) + ) + ) + ); + } + } + + /** + * @dev Returns the admin role that controls `role`. See {grantRole} and + * {revokeRole}. + * + * To change a role's admin, use {_setRoleAdmin}. + */ + function getRoleAdmin(bytes32 role) public view override returns (bytes32) { + return _roles[role].adminRole; + } + + /** + * @dev Grants `role` to `account`. + * + * If `account` had not been already granted `role`, emits a {RoleGranted} + * event. + * + * Requirements: + * + * - the caller must have ``role``'s admin role. + */ + function grantRole(bytes32 role, address account) external override onlyRole(getRoleAdmin(role)) { + _grantRole(role, account); + } + + /** + * @dev Revokes `role` from `account`. + * + * If `account` had been granted `role`, emits a {RoleRevoked} event. + * + * Requirements: + * + * - the caller must have ``role``'s admin role. + */ + function revokeRole(bytes32 role, address account) external override onlyRole(getRoleAdmin(role)) { + _revokeRole(role, account); + } + + /** + * @dev Revokes `role` from the calling account. + * + * Roles are often managed via {grantRole} and {revokeRole}: this function's + * purpose is to provide a mechanism for accounts to lose their privileges + * if they are compromised (such as when a trusted device is misplaced). + * + * If the calling account had been granted `role`, emits a {RoleRevoked} + * event. + * + * Requirements: + * + * - the caller must be `account`. + */ + function renounceRole(bytes32 role, address account) external override { + require(account == _msgSender(), "71"); + + _revokeRole(role, account); + } + + /** + * @dev Grants `role` to `account`. + * + * If `account` had not been already granted `role`, emits a {RoleGranted} + * event. Note that unlike {grantRole}, this function doesn't perform any + * checks on the calling account. + * + * [WARNING] + * ==== + * This function should only be called from the constructor when setting + * up the initial roles for the system. + * + * Using this function in any other way is effectively circumventing the admin + * system imposed by {AccessControl}. + * ==== + */ + function _setupRole(bytes32 role, address account) internal { + _grantRole(role, account); + } + + /** + * @dev Sets `adminRole` as ``role``'s admin role. + * + * Emits a {RoleAdminChanged} event. + */ + function _setRoleAdmin(bytes32 role, bytes32 adminRole) internal { + emit RoleAdminChanged(role, getRoleAdmin(role), adminRole); + _roles[role].adminRole = adminRole; + } + + function _grantRole(bytes32 role, address account) internal { + if (!hasRole(role, account)) { + _roles[role].members[account] = true; + emit RoleGranted(role, account, _msgSender()); + } + } + + function _revokeRole(bytes32 role, address account) internal { + if (hasRole(role, account)) { + _roles[role].members[account] = false; + emit RoleRevoked(role, account, _msgSender()); + } + } +} diff --git a/contracts/external/AccessControlUpgradeable.sol b/contracts/external/AccessControlUpgradeable.sol new file mode 100644 index 0000000..e2ac108 --- /dev/null +++ b/contracts/external/AccessControlUpgradeable.sol @@ -0,0 +1,241 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol"; +import "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol"; + +import "../interfaces/IAccessControl.sol"; + +/** + * @dev This contract is fully forked from OpenZeppelin `AccessControlUpgradeable`. + * The only difference is the removal of the ERC165 implementation as it's not + * needed in Angle. + * + * Contract module that allows children to implement role-based access + * control mechanisms. This is a lightweight version that doesn't allow enumerating role + * members except through off-chain means by accessing the contract event logs. Some + * applications may benefit from on-chain enumerability, for those cases see + * {AccessControlEnumerable}. + * + * Roles are referred to by their `bytes32` identifier. These should be exposed + * in the external API and be unique. The best way to achieve this is by + * using `public constant` hash digests: + * + * ``` + * bytes32 public constant MY_ROLE = keccak256("MY_ROLE"); + * ``` + * + * Roles can be used to represent a set of permissions. To restrict access to a + * function call, use {hasRole}: + * + * ``` + * function foo() public { + * require(hasRole(MY_ROLE, msg.sender)); + * ... + * } + * ``` + * + * Roles can be granted and revoked dynamically via the {grantRole} and + * {revokeRole} functions. Each role has an associated admin role, and only + * accounts that have a role's admin role can call {grantRole} and {revokeRole}. + * + * By default, the admin role for all roles is `DEFAULT_ADMIN_ROLE`, which means + * that only accounts with this role will be able to grant or revoke other + * roles. More complex role relationships can be created by using + * {_setRoleAdmin}. + * + * WARNING: The `DEFAULT_ADMIN_ROLE` is also its own admin: it has permission to + * grant and revoke this role. Extra precautions should be taken to secure + * accounts that have been granted it. + */ +abstract contract AccessControlUpgradeable is Initializable, IAccessControl { + function __AccessControl_init() internal initializer { + __AccessControl_init_unchained(); + } + + function __AccessControl_init_unchained() internal initializer {} + + struct RoleData { + mapping(address => bool) members; + bytes32 adminRole; + } + + mapping(bytes32 => RoleData) private _roles; + + bytes32 public constant DEFAULT_ADMIN_ROLE = 0x00; + + /** + * @dev Emitted when `newAdminRole` is set as ``role``'s admin role, replacing `previousAdminRole` + * + * `DEFAULT_ADMIN_ROLE` is the starting admin for all roles, despite + * {RoleAdminChanged} not being emitted signaling this. + * + * _Available since v3.1._ + */ + event RoleAdminChanged(bytes32 indexed role, bytes32 indexed previousAdminRole, bytes32 indexed newAdminRole); + + /** + * @dev Emitted when `account` is granted `role`. + * + * `sender` is the account that originated the contract call, an admin role + * bearer except when using {_setupRole}. + */ + event RoleGranted(bytes32 indexed role, address indexed account, address indexed sender); + + /** + * @dev Emitted when `account` is revoked `role`. + * + * `sender` is the account that originated the contract call: + * - if using `revokeRole`, it is the admin role bearer + * - if using `renounceRole`, it is the role bearer (i.e. `account`) + */ + event RoleRevoked(bytes32 indexed role, address indexed account, address indexed sender); + + /** + * @dev Modifier that checks that an account has a specific role. Reverts + * with a standardized message including the required role. + * + * The format of the revert reason is given by the following regular expression: + * + * /^AccessControl: account (0x[0-9a-f]{20}) is missing role (0x[0-9a-f]{32})$/ + * + * _Available since v4.1._ + */ + modifier onlyRole(bytes32 role) { + _checkRole(role, msg.sender); + _; + } + + /** + * @dev Returns `true` if `account` has been granted `role`. + */ + function hasRole(bytes32 role, address account) public view override returns (bool) { + return _roles[role].members[account]; + } + + /** + * @dev Revert with a standard message if `account` is missing `role`. + * + * The format of the revert reason is given by the following regular expression: + * + * /^AccessControl: account (0x[0-9a-f]{20}) is missing role (0x[0-9a-f]{32})$/ + */ + function _checkRole(bytes32 role, address account) internal view { + if (!hasRole(role, account)) { + revert( + string( + abi.encodePacked( + "AccessControl: account ", + StringsUpgradeable.toHexString(uint160(account), 20), + " is missing role ", + StringsUpgradeable.toHexString(uint256(role), 32) + ) + ) + ); + } + } + + /** + * @dev Returns the admin role that controls `role`. See {grantRole} and + * {revokeRole}. + * + * To change a role's admin, use {_setRoleAdmin}. + */ + function getRoleAdmin(bytes32 role) public view override returns (bytes32) { + return _roles[role].adminRole; + } + + /** + * @dev Grants `role` to `account`. + * + * If `account` had not been already granted `role`, emits a {RoleGranted} + * event. + * + * Requirements: + * + * - the caller must have ``role``'s admin role. + */ + function grantRole(bytes32 role, address account) external override onlyRole(getRoleAdmin(role)) { + _grantRole(role, account); + } + + /** + * @dev Revokes `role` from `account`. + * + * If `account` had been granted `role`, emits a {RoleRevoked} event. + * + * Requirements: + * + * - the caller must have ``role``'s admin role. + */ + function revokeRole(bytes32 role, address account) external override onlyRole(getRoleAdmin(role)) { + _revokeRole(role, account); + } + + /** + * @dev Revokes `role` from the calling account. + * + * Roles are often managed via {grantRole} and {revokeRole}: this function's + * purpose is to provide a mechanism for accounts to lose their privileges + * if they are compromised (such as when a trusted device is misplaced). + * + * If the calling account had been granted `role`, emits a {RoleRevoked} + * event. + * + * Requirements: + * + * - the caller must be `account`. + */ + function renounceRole(bytes32 role, address account) external override { + require(account == msg.sender, "71"); + + _revokeRole(role, account); + } + + /** + * @dev Grants `role` to `account`. + * + * If `account` had not been already granted `role`, emits a {RoleGranted} + * event. Note that unlike {grantRole}, this function doesn't perform any + * checks on the calling account. + * + * [WARNING] + * ==== + * This function should only be called from the constructor when setting + * up the initial roles for the system. + * + * Using this function in any other way is effectively circumventing the admin + * system imposed by {AccessControl}. + * ==== + */ + function _setupRole(bytes32 role, address account) internal { + _grantRole(role, account); + } + + /** + * @dev Sets `adminRole` as ``role``'s admin role. + * + * Emits a {RoleAdminChanged} event. + */ + function _setRoleAdmin(bytes32 role, bytes32 adminRole) internal { + emit RoleAdminChanged(role, getRoleAdmin(role), adminRole); + _roles[role].adminRole = adminRole; + } + + function _grantRole(bytes32 role, address account) internal { + if (!hasRole(role, account)) { + _roles[role].members[account] = true; + emit RoleGranted(role, account, msg.sender); + } + } + + function _revokeRole(bytes32 role, address account) internal { + if (hasRole(role, account)) { + _roles[role].members[account] = false; + emit RoleRevoked(role, account, msg.sender); + } + } + + uint256[49] private __gap; +} diff --git a/contracts/external/FullMath.sol b/contracts/external/FullMath.sol new file mode 100644 index 0000000..e8cec2c --- /dev/null +++ b/contracts/external/FullMath.sol @@ -0,0 +1,112 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity >=0.4.0; + +/// @title Contains 512-bit math functions +/// @notice Facilitates multiplication and division that can have overflow of an intermediate value without any loss of precision +/// @dev Handles "phantom overflow" i.e., allows multiplication and division where an intermediate value overflows 256 bits +/// @dev This contract was forked from Uniswap V3's contract `FullMath.sol` available here +/// https://github.com/Uniswap/uniswap-v3-core/blob/main/contracts/libraries/FullMath.sol +abstract contract FullMath { + /// @notice Calculates floor(a×b÷denominator) with full precision. Throws if result overflows a uint256 or denominator == 0 + /// @param a The multiplicand + /// @param b The multiplier + /// @param denominator The divisor + /// @return result The 256-bit result + /// @dev Credit to Remco Bloemen under MIT license https://xn--2-umb.com/21/muldiv + /// @dev Contrary to UniswapV3 implementation, this contract doesn't handle `a*b` overflows + /// and will revert if so + function _mulDiv( + uint256 a, + uint256 b, + uint256 denominator + ) internal pure returns (uint256 result) { + // 512-bit multiply [prod1 prod0] = a * b + // Compute the product mod 2**256 and mod 2**256 - 1 + // then use the Chinese Remainder Theorem to reconstruct + // the 512 bit result. The result is stored in two 256 + // variables such that product = prod1 * 2**256 + prod0 + uint256 prod0; // Least significant 256 bits of the product + uint256 prod1; // Most significant 256 bits of the product + assembly { + let mm := mulmod(a, b, not(0)) + prod0 := mul(a, b) + prod1 := sub(sub(mm, prod0), lt(mm, prod0)) + } + + // Handle non-overflow cases, 256 by 256 division + if (prod1 == 0) { + require(denominator > 0); + assembly { + result := div(prod0, denominator) + } + return result; + } + + // Make sure the result is less than 2**256. + // Also prevents denominator == 0 + require(denominator > prod1); + + /////////////////////////////////////////////// + // 512 by 256 division. + /////////////////////////////////////////////// + + // Make division exact by subtracting the remainder from [prod1 prod0] + // Compute remainder using mulmod + uint256 remainder; + assembly { + remainder := mulmod(a, b, denominator) + } + // Subtract 256 bit number from 512 bit number + assembly { + prod1 := sub(prod1, gt(remainder, prod0)) + prod0 := sub(prod0, remainder) + } + + // Factor powers of two out of denominator + // Compute largest power of two divisor of denominator. + // Always >= 1. + uint256 twos = denominator & (~denominator + 1); + // Divide denominator by power of two + assembly { + denominator := div(denominator, twos) + } + + // Divide [prod1 prod0] by the factors of two + assembly { + prod0 := div(prod0, twos) + } + // Shift in bits from prod1 into prod0. For this we need + // to flip `twos` such that it is 2**256 / twos. + // If twos is zero, then it becomes one + assembly { + twos := add(div(sub(0, twos), twos), 1) + } + prod0 |= prod1 * twos; + + // Invert denominator mod 2**256 + // Now that denominator is an odd number, it has an inverse + // modulo 2**256 such that denominator * inv = 1 mod 2**256. + // Compute the inverse by starting with a seed that is correct + // correct for four bits. That is, denominator * inv = 1 mod 2**4 + uint256 inv = (3 * denominator) ^ 2; + // Now use Newton-Raphson iteration to improve the precision. + // Thanks to Hensel's lifting lemma, this also works in modular + // arithmetic, doubling the correct bits in each step. + inv *= 2 - denominator * inv; // inverse mod 2**8 + inv *= 2 - denominator * inv; // inverse mod 2**16 + inv *= 2 - denominator * inv; // inverse mod 2**32 + inv *= 2 - denominator * inv; // inverse mod 2**64 + inv *= 2 - denominator * inv; // inverse mod 2**128 + inv *= 2 - denominator * inv; // inverse mod 2**256 + + // Because the division is now exact we can divide by multiplying + // with the modular inverse of denominator. This will give us the + // correct result modulo 2**256. Since the precoditions guarantee + // that the outcome is less than 2**256, this is the final result. + // We don't need to compute the high bits of the result and prod1 + // is no longer required. + result = prod0 * inv; + return result; + } +} diff --git a/contracts/external/ProxyAdmin.sol b/contracts/external/ProxyAdmin.sol new file mode 100644 index 0000000..cfaed41 --- /dev/null +++ b/contracts/external/ProxyAdmin.sol @@ -0,0 +1,81 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "./TransparentUpgradeableProxy.sol"; +import "@openzeppelin/contracts/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/SmartWalletChecker.sol b/contracts/external/SmartWalletChecker.sol new file mode 100644 index 0000000..5ad1c41 --- /dev/null +++ b/contracts/external/SmartWalletChecker.sol @@ -0,0 +1,97 @@ +// SPDX-License-Identifier: MIT + +pragma solidity 0.8.7; + +/// @notice Interface of the `SmartWalletChecker` contracts of the protocol +interface SmartWalletChecker { + function check(address) external view returns (bool); +} + +/// @title SmartWalletWhitelist +/// @author Curve Finance and adapted by Angle Core Team (https://etherscan.io/address/0xca719728ef172d0961768581fdf35cb116e0b7a4#code) +/// @notice Provides functions to check whether a wallet has been verified or not to own veANGLE +contract SmartWalletWhitelist { + /// @notice Mapping between addresses and whether they are whitelisted or not + mapping(address => bool) public wallets; + /// @notice Admin address of the contract + address public admin; + /// @notice Future admin address of the contract + //solhint-disable-next-line + address public future_admin; + /// @notice Contract which works as this contract and that can whitelist addresses + address public checker; + /// @notice Future address to become checker + //solhint-disable-next-line + address public future_checker; + + event ApproveWallet(address); + event RevokeWallet(address); + + /// @notice Constructor of the contract + /// @param _admin Admin address of the contract + constructor(address _admin) { + require(_admin != address(0), "0"); + admin = _admin; + } + + /// @notice Commits to change the admin + /// @param _admin New admin of the contract + function commitAdmin(address _admin) external { + require(msg.sender == admin, "!admin"); + future_admin = _admin; + } + + /// @notice Changes the admin to the admin that has been committed + function applyAdmin() external { + require(msg.sender == admin, "!admin"); + require(future_admin != address(0), "admin not set"); + admin = future_admin; + } + + /// @notice Commits to change the checker address + /// @param _checker New checker address + /// @dev This address can be the zero address in which case there will be no checker + function commitSetChecker(address _checker) external { + require(msg.sender == admin, "!admin"); + future_checker = _checker; + } + + /// @notice Applies the checker previously committed + function applySetChecker() external { + require(msg.sender == admin, "!admin"); + checker = future_checker; + } + + /// @notice Approves a wallet + /// @param _wallet Wallet to approve + function approveWallet(address _wallet) public { + require(msg.sender == admin, "!admin"); + wallets[_wallet] = true; + + emit ApproveWallet(_wallet); + } + + /// @notice Revokes a wallet + /// @param _wallet Wallet to revoke + function revokeWallet(address _wallet) external { + require(msg.sender == admin, "!admin"); + wallets[_wallet] = false; + + emit RevokeWallet(_wallet); + } + + /// @notice Checks whether a wallet is whitelisted + /// @param _wallet Wallet address to check + /// @dev This function can also rely on another SmartWalletChecker (a `checker` to see whether the wallet is whitelisted or not) + function check(address _wallet) external view returns (bool) { + bool _check = wallets[_wallet]; + if (_check) { + return _check; + } else { + if (checker != address(0)) { + return SmartWalletChecker(checker).check(_wallet); + } + } + return false; + } +} diff --git a/contracts/external/TransparentUpgradeableProxy.sol b/contracts/external/TransparentUpgradeableProxy.sol new file mode 100644 index 0000000..2bb6caa --- /dev/null +++ b/contracts/external/TransparentUpgradeableProxy.sol @@ -0,0 +1,125 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts/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/interfaces/IAccessControl.sol b/contracts/interfaces/IAccessControl.sol new file mode 100644 index 0000000..caf7936 --- /dev/null +++ b/contracts/interfaces/IAccessControl.sol @@ -0,0 +1,18 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +/// @title IAccessControl +/// @author Forked from OpenZeppelin +/// @notice Interface for `AccessControl` contracts +interface IAccessControl { + function hasRole(bytes32 role, address account) external view returns (bool); + + function getRoleAdmin(bytes32 role) external view returns (bytes32); + + function grantRole(bytes32 role, address account) external; + + function revokeRole(bytes32 role, address account) external; + + function renounceRole(bytes32 role, address account) external; +} diff --git a/contracts/interfaces/IAgToken.sol b/contracts/interfaces/IAgToken.sol new file mode 100644 index 0000000..f0d3a18 --- /dev/null +++ b/contracts/interfaces/IAgToken.sol @@ -0,0 +1,27 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol"; + +/// @title IAgToken +/// @author Angle Core Team +/// @notice Interface for the stablecoins `AgToken` contracts +/// @dev The only functions that are left in the interface are the functions which are used +/// at another point in the protocol by a different contract +interface IAgToken is IERC20Upgradeable { + // ======================= `StableMaster` functions ============================ + function mint(address account, uint256 amount) external; + + function burnFrom( + uint256 amount, + address burner, + address sender + ) external; + + function burnSelf(uint256 amount, address burner) external; + + // ========================= External function ================================= + + function stableMaster() external view returns (address); +} diff --git a/contracts/interfaces/IAngleMiddlemanGauge.sol b/contracts/interfaces/IAngleMiddlemanGauge.sol new file mode 100644 index 0000000..5907a6c --- /dev/null +++ b/contracts/interfaces/IAngleMiddlemanGauge.sol @@ -0,0 +1,10 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +/// @title IAngleMiddlemanGauge +/// @author Angle Core Team +/// @notice Interface for the `AngleMiddleman` contract +interface IAngleMiddlemanGauge { + function notifyReward(address gauge, uint256 amount) external; +} diff --git a/contracts/interfaces/IBondingCurve.sol b/contracts/interfaces/IBondingCurve.sol new file mode 100644 index 0000000..2662d2f --- /dev/null +++ b/contracts/interfaces/IBondingCurve.sol @@ -0,0 +1,23 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "./IAgToken.sol"; +import "./IOracle.sol"; + +/// @title IBondingCurve +/// @author Angle Core Team +/// @notice Interface for the `BondingCurve` contract +interface IBondingCurve { + // ============================ User Functions ================================= + + function buySoldToken( + IAgToken _agToken, + uint256 targetSoldTokenQuantity, + uint256 maxAmountToPayInAgToken + ) external; + + // ========================== Governance Functions ============================= + + function changeOracle(IAgToken _agToken, IOracle _oracle) external; +} diff --git a/contracts/interfaces/ICollateralSettler.sol b/contracts/interfaces/ICollateralSettler.sol new file mode 100644 index 0000000..3293462 --- /dev/null +++ b/contracts/interfaces/ICollateralSettler.sol @@ -0,0 +1,14 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +/// @title ICollateralSettler +/// @author Angle Core Team +/// @notice Interface for the collateral settlement contracts +interface ICollateralSettler { + function triggerSettlement( + uint256 _oracleValue, + uint256 _sanRate, + uint256 _stocksUsers + ) external; +} diff --git a/contracts/interfaces/ICore.sol b/contracts/interfaces/ICore.sol new file mode 100644 index 0000000..0d9fb94 --- /dev/null +++ b/contracts/interfaces/ICore.sol @@ -0,0 +1,26 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "./IStableMaster.sol"; + +/// @title ICore +/// @author Angle Core Team +/// @dev Interface for the functions of the `Core` contract +interface ICore { + function revokeStableMaster(address stableMaster) external; + + function addGovernor(address _governor) external; + + function removeGovernor(address _governor) external; + + function setGuardian(address _guardian) external; + + function revokeGuardian() external; + + function governorList() external view returns (address[] memory); + + function stablecoinList() external view returns (address[] memory); + + function guardian() external view returns (address); +} diff --git a/contracts/interfaces/ICoreBorrow.sol b/contracts/interfaces/ICoreBorrow.sol new file mode 100644 index 0000000..3cc5b43 --- /dev/null +++ b/contracts/interfaces/ICoreBorrow.sol @@ -0,0 +1,28 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +/// @title ICoreBorrow +/// @author Angle Core Team +/// @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/IERC721.sol b/contracts/interfaces/IERC721.sol new file mode 100644 index 0000000..53d07cf --- /dev/null +++ b/contracts/interfaces/IERC721.sol @@ -0,0 +1,46 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; + +interface IERC721 is IERC165 { + function balanceOf(address owner) external view returns (uint256 balance); + + function ownerOf(uint256 tokenId) external view returns (address owner); + + function safeTransferFrom( + address from, + address to, + uint256 tokenId + ) external; + + function transferFrom( + address from, + address to, + uint256 tokenId + ) external; + + function approve(address to, uint256 tokenId) external; + + function getApproved(uint256 tokenId) external view returns (address operator); + + function setApprovalForAll(address operator, bool _approved) external; + + function isApprovedForAll(address owner, address operator) external view returns (bool); + + function safeTransferFrom( + address from, + address to, + uint256 tokenId, + bytes calldata data + ) external; +} + +interface IERC721Metadata is IERC721 { + function name() external view returns (string memory); + + function symbol() external view returns (string memory); + + function tokenURI(uint256 tokenId) external view returns (string memory); +} diff --git a/contracts/interfaces/IFeeDistributor.sol b/contracts/interfaces/IFeeDistributor.sol new file mode 100644 index 0000000..8d89070 --- /dev/null +++ b/contracts/interfaces/IFeeDistributor.sol @@ -0,0 +1,21 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +/// @title IFeeDistributor +/// @author Interface of the `FeeDistributor` contract +/// @dev This interface is used by the `SurplusConverter` contract to send funds to the `FeeDistributor` +interface IFeeDistributor { + function burn(address token) external; +} + +/// @title IFeeDistributorFront +/// @author Interface for public use of the `FeeDistributor` contract +/// @dev This interface is used for user related function +interface IFeeDistributorFront { + function token() external returns (address); + + function claim(address _addr) external returns (uint256); + + function claim(address[20] memory _addr) external returns (bool); +} \ No newline at end of file diff --git a/contracts/interfaces/IFeeManager.sol b/contracts/interfaces/IFeeManager.sol new file mode 100644 index 0000000..afe30cc --- /dev/null +++ b/contracts/interfaces/IFeeManager.sol @@ -0,0 +1,42 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "./IAccessControl.sol"; + +/// @title IFeeManagerFunctions +/// @author Angle Core Team +/// @dev Interface for the `FeeManager` contract +interface IFeeManagerFunctions is IAccessControl { + // ================================= Keepers =================================== + + function updateUsersSLP() external; + + function updateHA() external; + + // ================================= Governance ================================ + + function deployCollateral( + address[] memory governorList, + address guardian, + address _perpetualManager + ) external; + + function setFees( + uint256[] memory xArray, + uint64[] memory yArray, + uint8 typeChange + ) external; + + function setHAFees(uint64 _haFeeDeposit, uint64 _haFeeWithdraw) external; +} + +/// @title IFeeManager +/// @author Angle Core Team +/// @notice Previous interface with additionnal getters for public variables and mappings +/// @dev We need these getters as they are used in other contracts of the protocol +interface IFeeManager is IFeeManagerFunctions { + function stableMaster() external view returns (address); + + function perpetualManager() external view returns (address); +} diff --git a/contracts/interfaces/IGaugeController.sol b/contracts/interfaces/IGaugeController.sol new file mode 100644 index 0000000..082898d --- /dev/null +++ b/contracts/interfaces/IGaugeController.sol @@ -0,0 +1,14 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +interface IGaugeController { + //solhint-disable-next-line + function gauge_types(address addr) external view returns (int128); + + //solhint-disable-next-line + function gauge_relative_weight_write(address addr, uint256 timestamp) external returns (uint256); + + //solhint-disable-next-line + function gauge_relative_weight(address addr, uint256 timestamp) external view returns (uint256); +} diff --git a/contracts/interfaces/IGenericLender.sol b/contracts/interfaces/IGenericLender.sol new file mode 100644 index 0000000..e875c00 --- /dev/null +++ b/contracts/interfaces/IGenericLender.sol @@ -0,0 +1,35 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "./IAccessControl.sol"; + +/// @title IGenericLender +/// @author Yearn with slight modifications from Angle Core Team +/// @dev Interface for the `GenericLender` contract, the base interface for contracts interacting +/// with lending and yield farming platforms +interface IGenericLender is IAccessControl { + function lenderName() external view returns (string memory); + + function nav() external view returns (uint256); + + function strategy() external view returns (address); + + function apr() external view returns (uint256); + + function weightedApr() external view returns (uint256); + + function withdraw(uint256 amount) external returns (uint256); + + function emergencyWithdraw(uint256 amount) external; + + function deposit() external; + + function withdrawAll() external returns (bool); + + function hasAssets() external view returns (bool); + + function aprAfterDeposit(uint256 amount) external view returns (uint256); + + function sweep(address _token, address to) external; +} diff --git a/contracts/interfaces/IKeeperRegistry.sol b/contracts/interfaces/IKeeperRegistry.sol new file mode 100644 index 0000000..452f321 --- /dev/null +++ b/contracts/interfaces/IKeeperRegistry.sol @@ -0,0 +1,14 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol"; + +/// @title IKeeperRegistry +/// @author Angle Core Team +interface IKeeperRegistry { + /// @notice Checks whether an address is whitelisted during oracle updates + /// @param caller Address for which the whitelist should be checked + /// @return Whether if the address is trusted + function isTrusted(address caller) external view returns (bool); +} diff --git a/contracts/interfaces/ILiquidityGauge.sol b/contracts/interfaces/ILiquidityGauge.sol new file mode 100644 index 0000000..377d9c2 --- /dev/null +++ b/contracts/interfaces/ILiquidityGauge.sol @@ -0,0 +1,24 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +interface ILiquidityGauge { + // solhint-disable-next-line + function staking_token() external returns (address stakingToken); + + // solhint-disable-next-line + function deposit_reward_token(address _rewardToken, uint256 _amount) external; + + function deposit( + uint256 _value, + address _addr, + // solhint-disable-next-line + bool _claim_rewards + ) external; + + // solhint-disable-next-line + function claim_rewards(address _addr) external; + + // solhint-disable-next-line + function claim_rewards(address _addr, address _receiver) external; +} diff --git a/contracts/interfaces/IOracle.sol b/contracts/interfaces/IOracle.sol new file mode 100644 index 0000000..a95bf60 --- /dev/null +++ b/contracts/interfaces/IOracle.sol @@ -0,0 +1,23 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +/// @title IOracle +/// @author Angle Core Team +/// @notice Interface for Angle's oracle contracts reading oracle rates from both UniswapV3 and Chainlink +/// from just UniswapV3 or from just Chainlink +interface IOracle { + function read() external view returns (uint256); + + function readAll() external view returns (uint256 lowerRate, uint256 upperRate); + + function readLower() external view returns (uint256); + + function readUpper() external view returns (uint256); + + function readQuote(uint256 baseAmount) external view returns (uint256); + + function readQuoteLower(uint256 baseAmount) external view returns (uint256); + + function inBase() external view returns (uint256); +} diff --git a/contracts/interfaces/IPerpetualManager.sol b/contracts/interfaces/IPerpetualManager.sol new file mode 100644 index 0000000..6f3f493 --- /dev/null +++ b/contracts/interfaces/IPerpetualManager.sol @@ -0,0 +1,112 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "./IERC721.sol"; +import "./IFeeManager.sol"; +import "./IOracle.sol"; +import "./IAccessControl.sol"; + +/// @title Interface of the contract managing perpetuals +/// @author Angle Core Team +/// @dev Front interface, meaning only user-facing functions +interface IPerpetualManagerFront is IERC721Metadata { + function openPerpetual( + address owner, + uint256 amountBrought, + uint256 amountCommitted, + uint256 maxOracleRate, + uint256 minNetMargin + ) external returns (uint256 perpetualID); + + function closePerpetual( + uint256 perpetualID, + address to, + uint256 minCashOutAmount + ) external; + + function addToPerpetual(uint256 perpetualID, uint256 amount) external; + + function removeFromPerpetual( + uint256 perpetualID, + uint256 amount, + address to + ) external; + + function liquidatePerpetuals(uint256[] memory perpetualIDs) external; + + function forceClosePerpetuals(uint256[] memory perpetualIDs) external; + + // ========================= External View Functions ============================= + + function getCashOutAmount(uint256 perpetualID, uint256 rate) external view returns (uint256, uint256); + + function isApprovedOrOwner(address spender, uint256 perpetualID) external view returns (bool); +} + +/// @title Interface of the contract managing perpetuals +/// @author Angle Core Team +/// @dev This interface does not contain user facing functions, it just has functions that are +/// interacted with in other parts of the protocol +interface IPerpetualManagerFunctions is IAccessControl { + // ================================= Governance ================================ + + function deployCollateral( + address[] memory governorList, + address guardian, + IFeeManager feeManager, + IOracle oracle_ + ) external; + + function setFeeManager(IFeeManager feeManager_) external; + + function setHAFees( + uint64[] memory _xHAFees, + uint64[] memory _yHAFees, + uint8 deposit + ) external; + + function setTargetAndLimitHAHedge(uint64 _targetHAHedge, uint64 _limitHAHedge) external; + + function setKeeperFeesLiquidationRatio(uint64 _keeperFeesLiquidationRatio) external; + + function setKeeperFeesCap(uint256 _keeperFeesLiquidationCap, uint256 _keeperFeesClosingCap) external; + + function setKeeperFeesClosing(uint64[] memory _xKeeperFeesClosing, uint64[] memory _yKeeperFeesClosing) external; + + function setLockTime(uint64 _lockTime) external; + + function setBoundsPerpetual(uint64 _maxLeverage, uint64 _maintenanceMargin) external; + + function pause() external; + + function unpause() external; + + // ==================================== Keepers ================================ + + function setFeeKeeper(uint64 feeDeposit, uint64 feesWithdraw) external; + + // =============================== StableMaster ================================ + + function setOracle(IOracle _oracle) external; +} + +/// @title IPerpetualManager +/// @author Angle Core Team +/// @notice Previous interface with additionnal getters for public variables +interface IPerpetualManager is IPerpetualManagerFunctions { + function poolManager() external view returns (address); + + function oracle() external view returns (address); + + function targetHAHedge() external view returns (uint64); + + function totalHedgeAmount() external view returns (uint256); +} + +/// @title Interface of the contract managing perpetuals with claim function +/// @author Angle Core Team +/// @dev Front interface with rewards function, meaning only user-facing functions +interface IPerpetualManagerFrontWithClaim is IPerpetualManagerFront, IPerpetualManager { + function getReward(uint256 perpetualID) external; +} diff --git a/contracts/interfaces/IPoolManager.sol b/contracts/interfaces/IPoolManager.sol new file mode 100644 index 0000000..2d7757e --- /dev/null +++ b/contracts/interfaces/IPoolManager.sol @@ -0,0 +1,84 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "./IFeeManager.sol"; +import "./IPerpetualManager.sol"; +import "./IOracle.sol"; + +// Struct for the parameters associated to a strategy interacting with a collateral `PoolManager` +// contract +struct StrategyParams { + // Timestamp of last report made by this strategy + // It is also used to check if a strategy has been initialized + uint256 lastReport; + // Total amount the strategy is expected to have + uint256 totalStrategyDebt; + // The share of the total assets in the `PoolManager` contract that the `strategy` can access to. + uint256 debtRatio; +} + +/// @title IPoolManagerFunctions +/// @author Angle Core Team +/// @notice Interface for the collateral poolManager contracts handling each one type of collateral for +/// a given stablecoin +/// @dev Only the functions used in other contracts of the protocol are left here +interface IPoolManagerFunctions { + // ============================ Constructor ==================================== + + function deployCollateral( + address[] memory governorList, + address guardian, + IPerpetualManager _perpetualManager, + IFeeManager feeManager, + IOracle oracle + ) external; + + // ============================ Yield Farming ================================== + + function creditAvailable() external view returns (uint256); + + function debtOutstanding() external view returns (uint256); + + function report( + uint256 _gain, + uint256 _loss, + uint256 _debtPayment + ) external; + + // ============================ Governance ===================================== + + function addGovernor(address _governor) external; + + function removeGovernor(address _governor) external; + + function setGuardian(address _guardian, address guardian) external; + + function revokeGuardian(address guardian) external; + + function setFeeManager(IFeeManager _feeManager) external; + + // ============================= Getters ======================================= + + function getBalance() external view returns (uint256); + + function getTotalAsset() external view returns (uint256); +} + +/// @title IPoolManager +/// @author Angle Core Team +/// @notice Previous interface with additionnal getters for public variables and mappings +/// @dev Used in other contracts of the protocol +interface IPoolManager is IPoolManagerFunctions { + function stableMaster() external view returns (address); + + function perpetualManager() external view returns (address); + + function token() external view returns (address); + + function feeManager() external view returns (address); + + function totalDebt() external view returns (uint256); + + function strategies(address _strategy) external view returns (StrategyParams memory); +} diff --git a/contracts/interfaces/IRewardsDistributor.sol b/contracts/interfaces/IRewardsDistributor.sol new file mode 100644 index 0000000..4eb3c62 --- /dev/null +++ b/contracts/interfaces/IRewardsDistributor.sol @@ -0,0 +1,52 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; + +import "./IStakingRewards.sol"; + +/// @title IRewardsDistributor +/// @author Angle Core Team, inspired from Fei protocol +/// (https://github.com/fei-protocol/fei-protocol-core/blob/master/contracts/staking/IRewardsDistributor.sol) +/// @notice Rewards Distributor interface +interface IRewardsDistributor { + // ========================= Public Parameter Getter =========================== + + function rewardToken() external view returns (IERC20); + + // ======================== External User Available Function =================== + + function drip(IStakingRewards stakingContract) external returns (uint256); + + // ========================= Governor Functions ================================ + + function governorWithdrawRewardToken(uint256 amount, address governance) external; + + function governorRecover( + address tokenAddress, + address to, + uint256 amount, + IStakingRewards stakingContract + ) external; + + function setUpdateFrequency(uint256 _frequency, IStakingRewards stakingContract) external; + + function setIncentiveAmount(uint256 _incentiveAmount, IStakingRewards stakingContract) external; + + function setAmountToDistribute(uint256 _amountToDistribute, IStakingRewards stakingContract) external; + + function setDuration(uint256 _duration, IStakingRewards stakingContract) external; + + function setStakingContract( + address _stakingContract, + uint256 _duration, + uint256 _incentiveAmount, + uint256 _dripFrequency, + uint256 _amountToDistribute + ) external; + + function setNewRewardsDistributor(address newRewardsDistributor) external; + + function removeStakingContract(IStakingRewards stakingContract) external; +} diff --git a/contracts/interfaces/ISanToken.sol b/contracts/interfaces/ISanToken.sol new file mode 100644 index 0000000..2b85817 --- /dev/null +++ b/contracts/interfaces/ISanToken.sol @@ -0,0 +1,27 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol"; + +/// @title ISanToken +/// @author Angle Core Team +/// @notice Interface for Angle's `SanToken` contract that handles sanTokens, tokens that are given to SLPs +/// contributing to a collateral for a given stablecoin +interface ISanToken is IERC20Upgradeable { + // ================================== StableMaster ============================= + + function mint(address account, uint256 amount) external; + + function burnFrom( + uint256 amount, + address burner, + address sender + ) external; + + function burnSelf(uint256 amount, address burner) external; + + function stableMaster() external view returns (address); + + function poolManager() external view returns (address); +} diff --git a/contracts/interfaces/IStableMaster.sol b/contracts/interfaces/IStableMaster.sol new file mode 100644 index 0000000..10616dc --- /dev/null +++ b/contracts/interfaces/IStableMaster.sol @@ -0,0 +1,168 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; + +// Normally just importing `IPoolManager` should be sufficient, but for clarity here +// we prefer to import all concerned interfaces +import "./IPoolManager.sol"; +import "./IOracle.sol"; +import "./IPerpetualManager.sol"; +import "./ISanToken.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 IStableMasterFunctions +/// @author Angle Core Team +/// @notice Interface for the `StableMaster` contract +interface IStableMasterFunctions { + function deploy( + address[] memory _governorList, + address _guardian, + address _agToken + ) external; + + // ============================== Lending ====================================== + + function accumulateInterest(uint256 gain) external; + + function signalLoss(uint256 loss) external; + + // ============================== HAs ========================================== + + function getStocksUsers() external view returns (uint256 maxCAmountInStable); + + function convertToSLP(uint256 amount, address user) external; + + // ============================== Keepers ====================================== + + function getCollateralRatio() external returns (uint256); + + function setFeeKeeper( + uint64 feeMint, + uint64 feeBurn, + uint64 _slippage, + uint64 _slippageFee + ) external; + + // ============================== AgToken ====================================== + + function updateStocksUsers(uint256 amount, address poolManager) external; + + // ============================= Governance ==================================== + + function setCore(address newCore) external; + + function addGovernor(address _governor) external; + + function removeGovernor(address _governor) external; + + function setGuardian(address newGuardian, address oldGuardian) external; + + function revokeGuardian(address oldGuardian) external; + + function setCapOnStableAndMaxInterests( + uint256 _capOnStableMinted, + uint256 _maxInterestsDistributed, + IPoolManager poolManager + ) external; + + function setIncentivesForSLPs( + uint64 _feesForSLPs, + uint64 _interestsForSLPs, + IPoolManager poolManager + ) external; + + function setUserFees( + IPoolManager poolManager, + uint64[] memory _xFee, + uint64[] memory _yFee, + uint8 _mint + ) external; + + function setTargetHAHedge(uint64 _targetHAHedge) external; + + function pause(bytes32 agent, IPoolManager poolManager) external; + + function unpause(bytes32 agent, IPoolManager poolManager) external; +} + +/// @title IStableMaster +/// @author Angle Core Team +/// @notice Previous interface with additionnal getters for public variables and mappings +interface IStableMaster is IStableMasterFunctions { + function agToken() external view returns (address); + + function collateralMap(IPoolManager poolManager) + external + view + returns ( + IERC20 token, + ISanToken sanToken, + IPerpetualManager perpetualManager, + IOracle oracle, + uint256 stocksUsers, + uint256 sanRate, + uint256 collatBase, + SLPData memory slpData, + MintBurnData memory feeData + ); +} diff --git a/contracts/interfaces/IStableMasterFront.sol b/contracts/interfaces/IStableMasterFront.sol new file mode 100644 index 0000000..2728f06 --- /dev/null +++ b/contracts/interfaces/IStableMasterFront.sol @@ -0,0 +1,40 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "../interfaces/IPoolManager.sol"; + +/// @title IStableMasterFront +/// @author Angle Core Team +/// @dev Front interface, meaning only user-facing functions +interface IStableMasterFront { + function mint( + uint256 amount, + address user, + IPoolManager poolManager, + uint256 minStableAmount + ) external; + + function burn( + uint256 amount, + address burner, + address dest, + IPoolManager poolManager, + uint256 minCollatAmount + ) external; + + function deposit( + uint256 amount, + address user, + IPoolManager poolManager + ) external; + + function withdraw( + uint256 amount, + address burner, + address dest, + IPoolManager poolManager + ) external; + + function agToken() external view returns (address); +} diff --git a/contracts/interfaces/IStakingRewards.sol b/contracts/interfaces/IStakingRewards.sol new file mode 100644 index 0000000..135097f --- /dev/null +++ b/contracts/interfaces/IStakingRewards.sol @@ -0,0 +1,27 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; + +/// @title IStakingRewardsFunctions +/// @author Angle Core Team +/// @notice Interface for the staking rewards contract that interact with the `RewardsDistributor` contract +interface IStakingRewardsFunctions { + function notifyRewardAmount(uint256 reward) external; + + function recoverERC20( + address tokenAddress, + address to, + uint256 tokenAmount + ) external; + + function setNewRewardsDistribution(address newRewardsDistribution) external; +} + +/// @title IStakingRewards +/// @author Angle Core Team +/// @notice Previous interface with additionnal getters for public variables +interface IStakingRewards is IStakingRewardsFunctions { + function rewardToken() external view returns (IERC20); +} diff --git a/contracts/interfaces/IStrategy.sol b/contracts/interfaces/IStrategy.sol new file mode 100644 index 0000000..782eaf3 --- /dev/null +++ b/contracts/interfaces/IStrategy.sol @@ -0,0 +1,32 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "./IAccessControl.sol"; + +/// @title IStrategy +/// @author Inspired by Yearn with slight changes from Angle Core Team +/// @notice Interface for yield farming strategies +interface IStrategy is IAccessControl { + function estimatedAPR() external view returns (uint256); + + function poolManager() external view returns (address); + + function want() external view returns (address); + + function isActive() external view returns (bool); + + function estimatedTotalAssets() external view returns (uint256); + + function harvestTrigger(uint256 callCost) external view returns (bool); + + function harvest() external; + + function withdraw(uint256 _amountNeeded) external returns (uint256 amountFreed, uint256 _loss); + + function setEmergencyExit() external; + + function addGuardian(address _guardian) external; + + function revokeGuardian(address _guardian) external; +} diff --git a/contracts/interfaces/ITimelock.sol b/contracts/interfaces/ITimelock.sol new file mode 100644 index 0000000..8beb43f --- /dev/null +++ b/contracts/interfaces/ITimelock.sol @@ -0,0 +1,44 @@ +// SPDX-License-Identifier: GPL-3.0 + +// Forked from https://github.com/compound-finance/compound-protocol/tree/master/contracts/Governance + +pragma solidity ^0.8.7; + +interface ITimelock { + function delay() external view returns (uint256); + + // solhint-disable-next-line func-name-mixedcase + function GRACE_PERIOD() external view returns (uint256); + + function acceptAdmin() external; + + function queuedTransactions(bytes32 hash) external view returns (bool); + + function queueTransaction( + address target, + uint256 value, + string calldata signature, + bytes calldata data, + uint256 eta + ) external returns (bytes32); + + function cancelTransaction( + address target, + uint256 value, + string calldata signature, + bytes calldata data, + uint256 eta + ) external; + + function executeTransaction( + address target, + uint256 value, + string calldata signature, + bytes calldata data, + uint256 eta + ) external payable returns (bytes memory); +} + +interface ANGLEInterface { + function getPriorVotes(address account, uint256 blockNumber) external view returns (uint96); +} diff --git a/contracts/interfaces/ITreasury.sol b/contracts/interfaces/ITreasury.sol new file mode 100644 index 0000000..3c72db6 --- /dev/null +++ b/contracts/interfaces/ITreasury.sol @@ -0,0 +1,16 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +/// @title ITreasury +/// @author Angle Core Team +/// @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 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); +} diff --git a/contracts/interfaces/IVaultManager.sol b/contracts/interfaces/IVaultManager.sol new file mode 100644 index 0000000..d16dd77 --- /dev/null +++ b/contracts/interfaces/IVaultManager.sol @@ -0,0 +1,127 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; +import "./ITreasury.sol"; + +// ========================= Key Structs and Enums ============================= + +/// @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 Data stored to track someone's loan (or equivalently called position) +struct Vault { + // Amount of collateral deposited in the vault + uint256 collateralAmount; + // Normalized value of the debt (that is to say of the stablecoins borrowed) + uint256 normalizedDebt; +} + +/// @notice Actions possible when composing calls to the different entry functions proposed +enum ActionBorrowType { + createVault, + closeVault, + addCollateral, + removeCollateral, + repayDebt, + borrow, + getDebtIn, + permit +} + +// ========================= Interfaces ============================= + +/// @title IVaultManagerFunctions +/// @author Angle Core Team +/// @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 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 + /// @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 + /// @return paymentData Struct containing the final transfers executed + /// @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 + function angle( + ActionBorrowType[] memory actions, + bytes[] memory datas, + address from, + address to + ) external payable returns (PaymentData memory paymentData); + + /// @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 + /// @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 final transfers executed + /// @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 + function angle( + ActionBorrowType[] memory actions, + bytes[] memory datas, + address from, + address to, + address who, + bytes memory repayData + ) external payable returns (PaymentData memory paymentData); + + /// @notice Checks whether a given address is approved for a vault or owns this vault + /// @param spender Address for which vault ownership should be checked + /// @param vaultID ID of the vault to check + /// @return Whether the `spender` address owns or is approved for `vaultID` + function isApprovedOrOwner(address spender, uint256 vaultID) external view returns (bool); + + /// @notice Allows an address to give or revoke approval for all its vaults to another address + /// @param owner Address signing the permit and giving (or revoking) its approval for all the controlled vaults + /// @param spender Address to give approval to + /// @param approved Whether to give or revoke the approval + /// @param deadline Deadline parameter for the signature to be valid + /// @dev The `v`, `r`, and `s` parameters are used as signature data + function permit( + address owner, + address spender, + bool approved, + uint256 deadline, + uint8 v, + bytes32 r, + bytes32 s + ) external; +} + +/// @title IVaultManagerStorage +/// @author Angle Core Team +/// @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 Reference to the `treasury` contract handling this `VaultManager` + function treasury() external view returns (ITreasury); + + /// @notice Reference to the collateral handled by this `VaultManager` + function collateral() external view returns (IERC20); + + /// @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); +} diff --git a/contracts/interfaces/IVeANGLE.sol b/contracts/interfaces/IVeANGLE.sol new file mode 100644 index 0000000..53e254f --- /dev/null +++ b/contracts/interfaces/IVeANGLE.sol @@ -0,0 +1,13 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol"; + +/// @title IVeANGLE +/// @author Angle Core Team +/// @notice Interface for the `VeANGLE` contract +interface IVeANGLE { + // solhint-disable-next-line func-name-mixedcase + function deposit_for(address addr, uint256 amount) external; +} diff --git a/contracts/interfaces/IWETH.sol b/contracts/interfaces/IWETH.sol new file mode 100644 index 0000000..1ca3256 --- /dev/null +++ b/contracts/interfaces/IWETH.sol @@ -0,0 +1,12 @@ +// SPDX-License-Identifier: MIT + +pragma solidity >=0.5.0; +import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; + +interface IWETH is IERC20 { + function deposit() external payable; + + function decimals() external view returns (uint256); + + function withdraw(uint256) external; +} diff --git a/contracts/interfaces/external/IWETH9.sol b/contracts/interfaces/external/IWETH9.sol new file mode 100644 index 0000000..443a0a4 --- /dev/null +++ b/contracts/interfaces/external/IWETH9.sol @@ -0,0 +1,13 @@ +// SPDX-License-Identifier: GPL-2.0-or-later +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; + +/// @title Interface for WETH9 +interface IWETH9 is IERC20 { + /// @notice Deposit ether to get wrapped ether + function deposit() external payable; + + /// @notice Withdraw wrapped ether to get ether + function withdraw(uint256) external; +} diff --git a/contracts/interfaces/external/aave/DataTypes.sol b/contracts/interfaces/external/aave/DataTypes.sol new file mode 100644 index 0000000..9d094b1 --- /dev/null +++ b/contracts/interfaces/external/aave/DataTypes.sol @@ -0,0 +1,44 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +library DataTypes { + // refer to the whitepaper, section 1.1 basic concepts for a formal description of these properties. + struct ReserveData { + //stores the reserve configuration + ReserveConfigurationMap configuration; + //the liquidity index. Expressed in ray + uint128 liquidityIndex; + //variable borrow index. Expressed in ray + uint128 variableBorrowIndex; + //the current supply rate. Expressed in ray + uint128 currentLiquidityRate; + //the current variable borrow rate. Expressed in ray + uint128 currentVariableBorrowRate; + //the current stable borrow rate. Expressed in ray + uint128 currentStableBorrowRate; + uint40 lastUpdateTimestamp; + //tokens addresses + address aTokenAddress; + address stableDebtTokenAddress; + address variableDebtTokenAddress; + //address of the interest rate strategy + address interestRateStrategyAddress; + //the id of the reserve. Represents the position in the list of the active reserves + uint8 id; + } + + struct ReserveConfigurationMap { + //bit 0-15: LTV + //bit 16-31: Liq. threshold + //bit 32-47: Liq. bonus + //bit 48-55: Decimals + //bit 56: Reserve is active + //bit 57: reserve is frozen + //bit 58: borrowing is enabled + //bit 59: stable rate borrowing enabled + //bit 60-63: reserved + //bit 64-79: reserve factor + uint256 data; + } +} diff --git a/contracts/interfaces/external/aave/IAave.sol b/contracts/interfaces/external/aave/IAave.sol new file mode 100644 index 0000000..3476753 --- /dev/null +++ b/contracts/interfaces/external/aave/IAave.sol @@ -0,0 +1,124 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import { DataTypes } from "./DataTypes.sol"; +import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol"; + +interface IAaveIncentivesController { + function getRewardsBalance(address[] calldata assets, address user) external view returns (uint256); + + function claimRewards( + address[] calldata assets, + uint256 amount, + address to + ) external returns (uint256); + + function getDistributionEnd() external view returns (uint256); + + function getAssetData(address asset) + external + view + returns ( + uint256, + uint256, + uint256 + ); +} + +interface IAToken is IERC20 { + function getIncentivesController() external view returns (IAaveIncentivesController); +} + +interface ILendingPool { + function deposit( + address asset, + uint256 amount, + address onBehalfOf, + uint16 referralCode + ) external; + + function withdraw( + address asset, + uint256 amount, + address to + ) external returns (uint256); + + function getReserveData(address asset) external view returns (DataTypes.ReserveData memory); +} + +interface ILendingPoolAddressesProvider { + function getLendingPool() external view returns (address); +} + +interface IProtocolDataProvider { + function ADDRESSES_PROVIDER() external view returns (ILendingPoolAddressesProvider); + + function getReserveConfigurationData(address asset) + external + view + returns ( + uint256 decimals, + uint256 ltv, + uint256 liquidationThreshold, + uint256 liquidationBonus, + uint256 reserveFactor, + bool usageAsCollateralEnabled, + bool borrowingEnabled, + bool stableBorrowRateEnabled, + bool isActive, + bool isFrozen + ); + + function getReserveData(address asset) + external + view + returns ( + uint256 availableLiquidity, + uint256 totalStableDebt, + uint256 totalVariableDebt, + uint256 liquidityRate, + uint256 variableBorrowRate, + uint256 stableBorrowRate, + uint256 averageStableBorrowRate, + uint256 liquidityIndex, + uint256 variableBorrowIndex, + uint40 lastUpdateTimestamp + ); +} + +interface IReserveInterestRateStrategy { + function calculateInterestRates( + address reserve, + uint256 utilizationRate, + uint256 totalStableDebt, + uint256 totalVariableDebt, + uint256 averageStableBorrowRate, + uint256 reserveFactor + ) + external + view + returns ( + uint256 liquidityRate, + uint256 stableBorrowRate, + uint256 variableBorrowRate + ); +} + +interface IStakedAave { + function stake(address to, uint256 amount) external; + + function redeem(address to, uint256 amount) external; + + function cooldown() external; + + function claimRewards(address to, uint256 amount) external; + + function getTotalRewardsBalance(address) external view returns (uint256); + + function COOLDOWN_SECONDS() external view returns (uint256); + + function stakersCooldowns(address) external view returns (uint256); + + function UNSTAKE_WINDOW() external view returns (uint256); +} diff --git a/contracts/interfaces/external/compound/CErc20I.sol b/contracts/interfaces/external/compound/CErc20I.sol new file mode 100755 index 0000000..03f24ba --- /dev/null +++ b/contracts/interfaces/external/compound/CErc20I.sol @@ -0,0 +1,15 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity >=0.5.0; + +import "./CTokenI.sol"; + +interface CErc20I is CTokenI { + function mint(uint256 mintAmount) external returns (uint256); + + function redeemUnderlying(uint256 redeemAmount) external returns (uint256); + + function underlying() external view returns (address); + + function borrow(uint256 borrowAmount) external returns (uint256); +} diff --git a/contracts/interfaces/external/compound/CEtherI.sol b/contracts/interfaces/external/compound/CEtherI.sol new file mode 100644 index 0000000..f85cdcb --- /dev/null +++ b/contracts/interfaces/external/compound/CEtherI.sol @@ -0,0 +1,13 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity >=0.5.0; + +import "./CTokenI.sol"; + +interface CEtherI is CTokenI { + function mint() external payable; + + function redeemUnderlying(uint256 redeemAmount) external returns (uint256); + + function borrow(uint256 borrowAmount) external returns (uint256); +} diff --git a/contracts/interfaces/external/compound/CTokenI.sol b/contracts/interfaces/external/compound/CTokenI.sol new file mode 100644 index 0000000..54d1941 --- /dev/null +++ b/contracts/interfaces/external/compound/CTokenI.sol @@ -0,0 +1,26 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; +import "./InterestRateModel.sol"; + +interface CTokenI { + function balanceOf(address owner) external view returns (uint256); + + function balanceOfUnderlying(address owner) external returns (uint256); + + function supplyRatePerBlock() external view returns (uint256); + + function exchangeRateCurrent() external returns (uint256); + + function exchangeRateStored() external view returns (uint256); + + function interestRateModel() external view returns (InterestRateModel); + + function totalReserves() external view returns (uint256); + + function reserveFactorMantissa() external view returns (uint256); + + function totalBorrows() external view returns (uint256); + + function totalSupply() external view returns (uint256); +} diff --git a/contracts/interfaces/external/compound/IComptroller.sol b/contracts/interfaces/external/compound/IComptroller.sol new file mode 100644 index 0000000..1a7e2bb --- /dev/null +++ b/contracts/interfaces/external/compound/IComptroller.sol @@ -0,0 +1,16 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "./CTokenI.sol"; + +interface IComptroller { + function compSupplySpeeds(address cToken) external view returns (uint256); + + function claimComp( + address[] memory holders, + CTokenI[] memory cTokens, + bool borrowers, + bool suppliers + ) external; +} diff --git a/contracts/interfaces/external/compound/InterestRateModel.sol b/contracts/interfaces/external/compound/InterestRateModel.sol new file mode 100755 index 0000000..42acca1 --- /dev/null +++ b/contracts/interfaces/external/compound/InterestRateModel.sol @@ -0,0 +1,27 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +interface InterestRateModel { + /** + * @notice Calculates the current supply interest rate per block + * @param cash The total amount of cash the market has + * @param borrows The total amount of borrows the market has outstanding + * @param reserves The total amount of reserves the market has + * @param reserveFactorMantissa The current reserve factor the market has + * @return The supply rate per block (as a percentage, and scaled by 1e18) + */ + function getSupplyRate( + uint256 cash, + uint256 borrows, + uint256 reserves, + uint256 reserveFactorMantissa + ) external view returns (uint256); + + // Rinkeby function + function getBorrowRate( + uint256 cash, + uint256 borrows, + uint256 _reserves + ) external view returns (uint256, uint256); +} diff --git a/contracts/interfaces/external/curve/Curve.sol b/contracts/interfaces/external/curve/Curve.sol new file mode 100644 index 0000000..3b267d5 --- /dev/null +++ b/contracts/interfaces/external/curve/Curve.sol @@ -0,0 +1,60 @@ +// SPDX-License-Identifier: MIT + +pragma solidity 0.8.7; + +interface ICurveFi { + function get_virtual_price() external view returns (uint256); + + function add_liquidity( + // sBTC pool + uint256[3] calldata amounts, + uint256 min_mint_amount + ) external; + + function add_liquidity( + // bUSD pool + uint256[4] calldata amounts, + uint256 min_mint_amount + ) external; + + function add_liquidity( + // stETH pool + uint256[2] calldata amounts, + uint256 min_mint_amount + ) external payable; + + function remove_liquidity_imbalance(uint256[4] calldata amounts, uint256 max_burn_amount) external; + + function remove_liquidity(uint256 _amount, uint256[4] calldata amounts) external; + + function remove_liquidity_one_coin( + uint256 _token_amount, + int128 i, + uint256 min_amount + ) external; + + function exchange( + int128 from, + int128 to, + uint256 _from_amount, + uint256 _min_to_amount + ) external payable; + + function balances(int128) external view returns (uint256); + + function get_dy( + int128 from, + int128 to, + uint256 _from_amount + ) external view returns (uint256); + + function calc_token_amount(uint256[2] calldata amounts, bool is_deposit) external view returns (uint256); +} + +interface Zap { + function remove_liquidity_one_coin( + uint256, + int128, + uint256 + ) external; +} diff --git a/contracts/interfaces/external/lido/ISteth.sol b/contracts/interfaces/external/lido/ISteth.sol new file mode 100644 index 0000000..c5e9bde --- /dev/null +++ b/contracts/interfaces/external/lido/ISteth.sol @@ -0,0 +1,13 @@ +// SPDX-License-Identifier: MIT + +pragma solidity 0.8.7; + +import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; + +interface IStETH is IERC20 { + event Submitted(address sender, uint256 amount, address referral); + + function submit(address) external payable returns (uint256); + + function getSharesByPooledEth(uint256 _ethAmount) external view returns (uint256); +} diff --git a/contracts/interfaces/external/lido/IWStETH.sol b/contracts/interfaces/external/lido/IWStETH.sol new file mode 100644 index 0000000..b6f4500 --- /dev/null +++ b/contracts/interfaces/external/lido/IWStETH.sol @@ -0,0 +1,16 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity 0.8.7; + +import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; + +/// @title IWStETH +/// @author Angle Core Team +/// @notice Interface for the `WStETH` contract +/// @dev This interface only contains functions of the `WStETH` which are called by other contracts +/// of this module +interface IWStETH is IERC20 { + function stETH() external returns (address); + + function wrap(uint256 _stETHAmount) external returns (uint256); +} diff --git a/contracts/interfaces/external/uniswap/IUniswapRouter.sol b/contracts/interfaces/external/uniswap/IUniswapRouter.sol new file mode 100644 index 0000000..c692147 --- /dev/null +++ b/contracts/interfaces/external/uniswap/IUniswapRouter.sol @@ -0,0 +1,38 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +struct ExactInputParams { + bytes path; + address recipient; + uint256 deadline; + uint256 amountIn; + uint256 amountOutMinimum; +} + +/// @title Router token swapping functionality +/// @notice Functions for swapping tokens via Uniswap V3 +interface IUniswapV3Router { + /// @notice Swaps `amountIn` of one token for as much as possible of another along the specified path + /// @param params The parameters necessary for the multi-hop swap, encoded as `ExactInputParams` in calldata + /// @return amountOut The amount of the received token + function exactInput(ExactInputParams calldata params) external payable returns (uint256 amountOut); +} + +/// @title Router for price estimation functionality +/// @notice Functions for getting the price of one token with respect to another using Uniswap V2 +/// @dev This interface is only used for non critical elements of the protocol +interface IUniswapV2Router { + /// @notice Given an input asset amount, returns the maximum output amount of the + /// other asset (accounting for fees) given reserves. + /// @param path Addresses of the pools used to get prices + function getAmountsOut(uint256 amountIn, address[] calldata path) external view returns (uint256[] memory amounts); + + function swapExactTokensForTokens( + uint256 swapAmount, + uint256 minExpected, + address[] calldata path, + address receiver, + uint256 swapDeadline + ) external; +} diff --git a/contracts/staking/AngleDistributor.sol b/contracts/staking/AngleDistributor.sol new file mode 100644 index 0000000..597befc --- /dev/null +++ b/contracts/staking/AngleDistributor.sol @@ -0,0 +1,406 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "./AngleDistributorEvents.sol"; + +/// @title AngleDistributor +/// @author Forked from contracts developed by Curve and Frax and adapted by Angle Core Team +/// - ERC20CRV.vy (https://github.com/curvefi/curve-dao-contracts/blob/master/contracts/ERC20CRV.vy) +/// - FraxGaugeFXSRewardsDistributor.sol (https://github.com/FraxFinance/frax-solidity/blob/master/src/hardhat/contracts/Curve/FraxGaugeFXSRewardsDistributor.sol) +/// @notice All the events used in `AngleDistributor` contract +contract AngleDistributor is AngleDistributorEvents, ReentrancyGuardUpgradeable, AccessControlUpgradeable { + using SafeERC20 for IERC20; + + /// @notice Role for governors only + bytes32 public constant GOVERNOR_ROLE = keccak256("GOVERNOR_ROLE"); + /// @notice Role for the guardian + bytes32 public constant GUARDIAN_ROLE = keccak256("GUARDIAN_ROLE"); + + /// @notice Length of a week in seconds + uint256 public constant WEEK = 3600 * 24 * 7; + + /// @notice Time at which the emission rate is updated + uint256 public constant RATE_REDUCTION_TIME = WEEK; + + /// @notice Reduction of the emission rate + uint256 public constant RATE_REDUCTION_COEFFICIENT = 1007827884862117171; // 1.5 ^ (1/52) * 10**18 + + /// @notice Base used for computation + uint256 public constant BASE = 10**18; + + /// @notice Maps the address of a gauge to the last time this gauge received rewards + mapping(address => uint256) public lastTimeGaugePaid; + + /// @notice Maps the address of a gauge to whether it was killed or not + /// A gauge killed in this contract cannot receive any rewards + mapping(address => bool) public killedGauges; + + /// @notice Maps the address of a type >= 2 gauge to a delegate address responsible + /// for giving rewards to the actual gauge + mapping(address => address) public delegateGauges; + + /// @notice Maps the address of a gauge delegate to whether this delegate supports the `notifyReward` interface + /// and is therefore built for automation + mapping(address => bool) public isInterfaceKnown; + + /// @notice Address of the ANGLE token given as a reward + IERC20 public rewardToken; + + /// @notice Address of the `GaugeController` contract + IGaugeController public controller; + + /// @notice Address responsible for pulling rewards of type >= 2 gauges and distributing it to the + /// associated contracts if there is not already an address delegated for this specific contract + address public delegateGauge; + + /// @notice ANGLE current emission rate, it is first defined in the initializer and then updated every week + uint256 public rate; + + /// @notice Timestamp at which the current emission epoch started + uint256 public startEpochTime; + + /// @notice Amount of ANGLE tokens distributed through staking at the start of the epoch + /// This is an informational variable used to track how much has been distributed through liquidity mining + uint256 public startEpochSupply; + + /// @notice Index of the current emission epoch + /// Here also, this variable is not useful per se inside the smart contracts of the protocol, it is + /// just an informational variable + uint256 public miningEpoch; + + /// @notice Whether ANGLE distribution through this contract is on or no + bool public distributionsOn; + + /// @notice Constructor of the contract + /// @param _rewardToken Address of the ANGLE token + /// @param _controller Address of the GaugeController + /// @param _initialRate Initial ANGLE emission rate + /// @param _startEpochSupply Amount of ANGLE tokens already distributed via liquidity mining + /// @param governor Governor address of the contract + /// @param guardian Address of the guardian of this contract + /// @param _delegateGauge Address that will be used to pull rewards for type 2 gauges + /// @dev After this contract is created, the correct amount of ANGLE tokens should be transferred to the contract + /// @dev The `_delegateGauge` can be the zero address + function initialize( + address _rewardToken, + address _controller, + uint256 _initialRate, + uint256 _startEpochSupply, + address governor, + address guardian, + address _delegateGauge + ) external initializer { + require( + _controller != address(0) && _rewardToken != address(0) && guardian != address(0) && governor != address(0), + "0" + ); + rewardToken = IERC20(_rewardToken); + controller = IGaugeController(_controller); + startEpochSupply = _startEpochSupply; + miningEpoch = 0; + // Some ANGLE tokens should be sent to the contract directly after initialization + rate = _initialRate; + delegateGauge = _delegateGauge; + distributionsOn = false; + startEpochTime = block.timestamp; + _setRoleAdmin(GOVERNOR_ROLE, GOVERNOR_ROLE); + _setRoleAdmin(GUARDIAN_ROLE, GOVERNOR_ROLE); + _setupRole(GUARDIAN_ROLE, guardian); + _setupRole(GOVERNOR_ROLE, governor); + _setupRole(GUARDIAN_ROLE, governor); + } + + /// @custom:oz-upgrades-unsafe-allow constructor + constructor() initializer {} + + // ======================== Internal Functions ================================= + + /// @notice Internal function to distribute rewards to a gauge + /// @param gaugeAddr Address of the gauge to distribute rewards to + /// @return weeksElapsed Weeks elapsed since the last call + /// @return rewardTally Amount of rewards distributed to the gauge + /// @dev The reason for having an internal function is that it's called by the `distributeReward` and the + /// `distributeRewardToMultipleGauges` + /// @dev Although they would need to be performed all the time this function is called, this function does not + /// contain checks on whether distribution is on, and on whether rate should be reduced. These are done in each external + /// function calling this function for gas efficiency + function _distributeReward(address gaugeAddr) internal returns (uint256 weeksElapsed, uint256 rewardTally) { + // Checking if the gauge has been added or if it still possible to distribute rewards to this gauge + int128 gaugeType = IGaugeController(controller).gauge_types(gaugeAddr); + require(gaugeType >= 0 && !killedGauges[gaugeAddr], "110"); + + // Calculate the elapsed time in weeks. + uint256 lastTimePaid = lastTimeGaugePaid[gaugeAddr]; + + // Edge case for first reward for this gauge + if (lastTimePaid == 0) { + weeksElapsed = 1; + if (gaugeType == 0) { + // We give a full approval for the gauges with type zero which correspond to the staking + // contracts of the protocol + rewardToken.safeApprove(gaugeAddr, type(uint256).max); + } + } else { + // Truncation desired + weeksElapsed = (block.timestamp - lastTimePaid) / WEEK; + // Return early here for 0 weeks instead of throwing, as it could have bad effects in other contracts + if (weeksElapsed == 0) { + return (0, 0); + } + } + rewardTally = 0; + // We use this variable to keep track of the emission rate across different weeks + uint256 weeklyRate = rate; + for (uint256 i = 0; i < weeksElapsed; i++) { + uint256 relWeightAtWeek; + if (i == 0) { + // Mutative, for the current week: makes sure the weight is checkpointed. Also returns the weight. + relWeightAtWeek = controller.gauge_relative_weight_write(gaugeAddr, block.timestamp); + } else { + // View + relWeightAtWeek = controller.gauge_relative_weight(gaugeAddr, (block.timestamp - WEEK * i)); + } + rewardTally += (weeklyRate * relWeightAtWeek * WEEK) / BASE; + + // To get the rate of the week prior from the current rate we just have to multiply by the weekly division + // factor + // There may be some precisions error: inferred previous values of the rate may be different to what we would + // have had if the rate had been computed correctly in these weeks: we expect from empirical observations + // this `weeklyRate` to be inferior to what the `rate` would have been + weeklyRate = (weeklyRate * RATE_REDUCTION_COEFFICIENT) / BASE; + } + + // Update the last time paid, rounded to the closest week + // in order not to have an ever moving time on when to call this function + lastTimeGaugePaid[gaugeAddr] = (block.timestamp / WEEK) * WEEK; + + // If the `gaugeType >= 2`, this means that the gauge is a gauge on another chain (and corresponds to tokens + // that need to be bridged) or is associated to an external contract of the Angle Protocol + if (gaugeType >= 2) { + // If it is defined, we use the specific delegate attached to the gauge + address delegate = delegateGauges[gaugeAddr]; + if (delegate == address(0)) { + // If not, we check if a delegate common to all gauges with type >= 2 can be used + delegate = delegateGauge; + } + if (delegate != address(0)) { + // In the case where the gauge has a delegate (specific or not), then rewards are transferred to this gauge + rewardToken.safeTransfer(delegate, rewardTally); + // If this delegate supports a specific interface, then rewards sent are notified through this + // interface + if (isInterfaceKnown[delegate]) { + IAngleMiddlemanGauge(delegate).notifyReward(gaugeAddr, rewardTally); + } + } else { + rewardToken.safeTransfer(gaugeAddr, rewardTally); + } + } else if (gaugeType == 1) { + // This is for the case of Perpetual contracts which need to be able to receive their reward tokens + rewardToken.safeTransfer(gaugeAddr, rewardTally); + IStakingRewards(gaugeAddr).notifyRewardAmount(rewardTally); + } else { + // Mainnet: Pay out the rewards directly to the gauge + ILiquidityGauge(gaugeAddr).deposit_reward_token(address(rewardToken), rewardTally); + } + emit RewardDistributed(gaugeAddr, rewardTally); + } + + /// @notice Updates mining rate and supply at the start of the epoch + /// @dev Any modifying mining call must also call this + /// @dev It is possible that more than one week past between two calls of this function, and for this reason + /// this function has been slightly modified from Curve implementation by Angle Team + function _updateMiningParameters() internal { + // When entering this function, we always have: `(block.timestamp - startEpochTime) / RATE_REDUCTION_TIME >= 1` + uint256 epochDelta = (block.timestamp - startEpochTime) / RATE_REDUCTION_TIME; + + // Storing intermediate values for the rate and for the `startEpochSupply` + uint256 _rate = rate; + uint256 _startEpochSupply = startEpochSupply; + + startEpochTime += RATE_REDUCTION_TIME * epochDelta; + miningEpoch += epochDelta; + + for (uint256 i = 0; i < epochDelta; i++) { + // Updating the intermediate values of the `startEpochSupply` + _startEpochSupply += _rate * RATE_REDUCTION_TIME; + _rate = (_rate * BASE) / RATE_REDUCTION_COEFFICIENT; + } + rate = _rate; + startEpochSupply = _startEpochSupply; + emit UpdateMiningParameters(block.timestamp, _rate, _startEpochSupply); + } + + /// @notice Toggles the fact that a gauge delegate can be used for automation or not and therefore supports + /// the `notifyReward` interface + /// @param _delegateGauge Address of the gauge to change + function _toggleInterfaceKnown(address _delegateGauge) internal { + bool isInterfaceKnownMem = isInterfaceKnown[_delegateGauge]; + isInterfaceKnown[_delegateGauge] = !isInterfaceKnownMem; + emit InterfaceKnownToggled(_delegateGauge, !isInterfaceKnownMem); + } + + // ================= Permissionless External Functions ========================= + + /// @notice Distributes rewards to a staking contract (also called gauge) + /// @param gaugeAddr Address of the gauge to send tokens too + /// @return weeksElapsed Number of weeks elapsed since the last time rewards were distributed + /// @return rewardTally Amount of tokens sent to the gauge + /// @dev Anyone can call this function to distribute rewards to the different staking contracts + function distributeReward(address gaugeAddr) external nonReentrant returns (uint256, uint256) { + // Checking if distribution is on + require(distributionsOn == true, "109"); + // Updating rate distribution parameters if need be + if (block.timestamp >= startEpochTime + RATE_REDUCTION_TIME) { + _updateMiningParameters(); + } + return _distributeReward(gaugeAddr); + } + + /// @notice Distributes rewards to multiple staking contracts + /// @param gauges Addresses of the gauge to send tokens too + /// @dev Anyone can call this function to distribute rewards to the different staking contracts + /// @dev Compared with the `distributeReward` function, this function sends rewards to multiple + /// contracts at the same time + function distributeRewardToMultipleGauges(address[] memory gauges) external nonReentrant { + // Checking if distribution is on + require(distributionsOn == true, "109"); + // Updating rate distribution parameters if need be + if (block.timestamp >= startEpochTime + RATE_REDUCTION_TIME) { + _updateMiningParameters(); + } + for (uint256 i = 0; i < gauges.length; i++) { + _distributeReward(gauges[i]); + } + } + + /// @notice Updates mining rate and supply at the start of the epoch + /// @dev Callable by any address, but only once per epoch + function updateMiningParameters() external { + require(block.timestamp >= startEpochTime + RATE_REDUCTION_TIME, "108"); + _updateMiningParameters(); + } + + // ========================= Governor Functions ================================ + + /// @notice Withdraws ERC20 tokens that could accrue on this contract + /// @param tokenAddress Address of the ERC20 token to withdraw + /// @param to Address to transfer to + /// @param amount Amount to transfer + /// @dev Added to support recovering LP Rewards and other mistaken tokens + /// from other systems to be distributed to holders + /// @dev This function could also be used to recover ANGLE tokens in case the rate got smaller + function recoverERC20( + address tokenAddress, + address to, + uint256 amount + ) external onlyRole(GOVERNOR_ROLE) { + // If the token is the ANGLE token, we need to make sure that governance is not going to withdraw + // too many tokens and that it'll be able to sustain the weekly distribution forever + // This check assumes that `distributeReward` has been called for gauges and that there are no gauges + // which have not received their past week's rewards + if (tokenAddress == address(rewardToken)) { + uint256 currentBalance = rewardToken.balanceOf(address(this)); + // The amount distributed till the end is `rate * WEEK / (1 - RATE_REDUCTION_FACTOR)` where + // `RATE_REDUCTION_FACTOR = BASE / RATE_REDUCTION_COEFFICIENT` which translates to: + require( + currentBalance >= + ((rate * RATE_REDUCTION_COEFFICIENT) * WEEK) / (RATE_REDUCTION_COEFFICIENT - BASE) + amount, + "4" + ); + } + IERC20(tokenAddress).safeTransfer(to, amount); + emit Recovered(tokenAddress, to, amount); + } + + /// @notice Sets a new gauge controller + /// @param _controller Address of the new gauge controller + function setGaugeController(address _controller) external onlyRole(GOVERNOR_ROLE) { + require(_controller != address(0), "0"); + controller = IGaugeController(_controller); + emit GaugeControllerUpdated(_controller); + } + + /// @notice Sets a new delegate gauge for pulling rewards of a type >= 2 gauges or of all type >= 2 gauges + /// @param gaugeAddr Gauge to change the delegate of + /// @param _delegateGauge Address of the new gauge delegate related to `gaugeAddr` + /// @param toggleInterface Whether we should toggle the fact that the `_delegateGauge` is built for automation or not + /// @dev This function can be used to remove delegating or introduce the pulling of rewards to a given address + /// @dev If `gaugeAddr` is the zero address, this function updates the delegate gauge common to all gauges with type >= 2 + /// @dev The `toggleInterface` parameter has been added for convenience to save one transaction when adding a gauge delegate + /// which supports the `notifyReward` interface + function setDelegateGauge( + address gaugeAddr, + address _delegateGauge, + bool toggleInterface + ) external onlyRole(GOVERNOR_ROLE) { + if (gaugeAddr != address(0)) { + delegateGauges[gaugeAddr] = _delegateGauge; + } else { + delegateGauge = _delegateGauge; + } + emit DelegateGaugeUpdated(gaugeAddr, _delegateGauge); + + if (toggleInterface) { + _toggleInterfaceKnown(_delegateGauge); + } + } + + /// @notice Changes the ANGLE emission rate + /// @param _newRate New ANGLE emission rate + /// @dev It is important to be super wary when calling this function and to make sure that `distributeReward` + /// has been called for all gauges in the past weeks. If not, gauges may get an incorrect distribution of ANGLE rewards + /// for these past weeks based on the new rate and not on the old rate + /// @dev Governance should thus make sure to call this function rarely and when it does to do it after the weekly `distributeReward` + /// calls for all existing gauges + /// @dev As this function assumes that `distributeReward` has been called during the week, it also assumes that the `startEpochSupply` + /// parameter has been put up to date + function setRate(uint256 _newRate) external onlyRole(GOVERNOR_ROLE) { + // Checking if the new rate is compatible with the amount of ANGLE tokens this contract has in balance + // This check assumes, like this function, that `distributeReward` has correctly been called before + require( + rewardToken.balanceOf(address(this)) >= + ((_newRate * RATE_REDUCTION_COEFFICIENT) * WEEK) / (RATE_REDUCTION_COEFFICIENT - BASE), + "4" + ); + rate = _newRate; + emit RateUpdated(_newRate); + } + + /// @notice Toggles the status of a gauge to either killed or unkilled + /// @param gaugeAddr Gauge to toggle the status of + /// @dev It is impossible to kill a gauge in the `GaugeController` contract, for this reason killing of gauges + /// takes place in the `AngleDistributor` contract + /// @dev This means that people could vote for a gauge in the gauge controller contract but that rewards are not going + /// to be distributed to it in the end: people would need to remove their weights on the gauge killed to end the diminution + /// in rewards + /// @dev In the case of a gauge being killed, this function resets the timestamps at which this gauge has been approved and + /// disapproves the gauge to spend the token + /// @dev It should be cautiously called by governance as it could result in less ANGLE overall rewards than initially planned + /// if people do not remove their voting weights to the killed gauge + function toggleGauge(address gaugeAddr) external onlyRole(GOVERNOR_ROLE) { + bool gaugeKilledMem = killedGauges[gaugeAddr]; + if (!gaugeKilledMem) { + delete lastTimeGaugePaid[gaugeAddr]; + rewardToken.safeApprove(gaugeAddr, 0); + } + killedGauges[gaugeAddr] = !gaugeKilledMem; + emit GaugeToggled(gaugeAddr, !gaugeKilledMem); + } + + // ========================= Guardian Function ================================= + + /// @notice Halts or activates distribution of rewards + function toggleDistributions() external onlyRole(GUARDIAN_ROLE) { + bool distributionsOnMem = distributionsOn; + distributionsOn = !distributionsOnMem; + emit DistributionsToggled(!distributionsOnMem); + } + + /// @notice Notifies that the interface of a gauge delegate is known or has changed + /// @param _delegateGauge Address of the gauge to change + /// @dev Gauge delegates that are built for automation should be toggled + function toggleInterfaceKnown(address _delegateGauge) external onlyRole(GUARDIAN_ROLE) { + _toggleInterfaceKnown(_delegateGauge); + } +} diff --git a/contracts/staking/AngleDistributorEvents.sol b/contracts/staking/AngleDistributorEvents.sol new file mode 100644 index 0000000..8e48844 --- /dev/null +++ b/contracts/staking/AngleDistributorEvents.sol @@ -0,0 +1,29 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; +import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; +import "@openzeppelin/contracts-upgradeable/security/ReentrancyGuardUpgradeable.sol"; + +import "../interfaces/IGaugeController.sol"; +import "../interfaces/ILiquidityGauge.sol"; +import "../interfaces/IAngleMiddlemanGauge.sol"; +import "../interfaces/IStakingRewards.sol"; + +import "../external/AccessControlUpgradeable.sol"; + +/// @title AngleDistributorEvents +/// @author Angle Core Team +/// @notice All the events used in `AngleDistributor` contract +contract AngleDistributorEvents { + event DelegateGaugeUpdated(address indexed _gaugeAddr, address indexed _delegateGauge); + event DistributionsToggled(bool _distributionsOn); + event GaugeControllerUpdated(address indexed _controller); + event GaugeToggled(address indexed gaugeAddr, bool newStatus); + event InterfaceKnownToggled(address indexed _delegateGauge, bool _isInterfaceKnown); + event RateUpdated(uint256 _newRate); + event Recovered(address indexed tokenAddress, address indexed to, uint256 amount); + event RewardDistributed(address indexed gaugeAddr, uint256 rewardTally); + event UpdateMiningParameters(uint256 time, uint256 rate, uint256 supply); +} diff --git a/contracts/staking/GaugeController.vy b/contracts/staking/GaugeController.vy new file mode 100644 index 0000000..da35d28 --- /dev/null +++ b/contracts/staking/GaugeController.vy @@ -0,0 +1,605 @@ +# @version 0.2.16 + +""" +@title Gauge Controller +@author Angle Protocol +@license MIT +@notice Controls liquidity gauges and the issuance of coins through the gauges +""" + +# Full fork from: +# Curve Finance's gauge controller +# https://github.com/curvefi/curve-dao-contracts/blob/master/contracts/GaugeController.vy + +# 7 * 86400 seconds - all future times are rounded by week +WEEK: constant(uint256) = 604800 + +# Cannot change weight votes more often than once in 10 days +WEIGHT_VOTE_DELAY: constant(uint256) = 10 * 86400 + + +struct Point: + bias: uint256 + slope: uint256 + +struct VotedSlope: + slope: uint256 + power: uint256 + end: uint256 + + +interface VotingEscrow: + def get_last_user_slope(addr: address) -> int128: view + def locked__end(addr: address) -> uint256: view + + +event CommitOwnership: + admin: address + +event ApplyOwnership: + admin: address + +event AddType: + name: String[64] + type_id: int128 + +event NewTypeWeight: + type_id: int128 + time: uint256 + weight: uint256 + total_weight: uint256 + +event NewGaugeWeight: + gauge_address: address + time: uint256 + weight: uint256 + total_weight: uint256 + +event VoteForGauge: + time: uint256 + user: address + gauge_addr: address + weight: uint256 + +event NewGauge: + addr: address + gauge_type: int128 + weight: uint256 + +event KilledGauge: + addr: address + +MULTIPLIER: constant(uint256) = 10 ** 18 + +admin: public(address) # Can and will be a smart contract +future_admin: public(address) # Can and will be a smart contract + +token: public(address) # ANGLE token +voting_escrow: public(address) # Voting escrow + +# Gauge parameters +# All numbers are "fixed point" on the basis of 1e18 +n_gauge_types: public(int128) +n_gauges: public(int128) +gauge_type_names: public(HashMap[int128, String[64]]) + +# Needed for enumeration +gauges: public(address[1000000000]) + +# we increment values by 1 prior to storing them here so we can rely on a value +# of zero as meaning the gauge has not been set +gauge_types_: HashMap[address, int128] + +vote_user_slopes: public(HashMap[address, HashMap[address, VotedSlope]]) # user -> gauge_addr -> VotedSlope +vote_user_power: public(HashMap[address, uint256]) # Total vote power used by user +last_user_vote: public(HashMap[address, HashMap[address, uint256]]) # Last user vote's timestamp for each gauge address + +# Past and scheduled points for gauge weight, sum of weights per type, total weight +# Point is for bias+slope +# changes_* are for changes in slope +# time_* are for the last change timestamp +# timestamps are rounded to whole weeks + +points_weight: public(HashMap[address, HashMap[uint256, Point]]) # gauge_addr -> time -> Point +changes_weight: HashMap[address, HashMap[uint256, uint256]] # gauge_addr -> time -> slope +time_weight: public(HashMap[address, uint256]) # gauge_addr -> last scheduled time (next week) + +points_sum: public(HashMap[int128, HashMap[uint256, Point]]) # type_id -> time -> Point +changes_sum: HashMap[int128, HashMap[uint256, uint256]] # type_id -> time -> slope +time_sum: public(uint256[1000000000]) # type_id -> last scheduled time (next week) + +points_total: public(HashMap[uint256, uint256]) # time -> total weight +time_total: public(uint256) # last scheduled time + +points_type_weight: public(HashMap[int128, HashMap[uint256, uint256]]) # type_id -> time -> type weight +time_type_weight: public(uint256[1000000000]) # type_id -> last scheduled time (next week) + + +@external +def __init__(_token: address, _voting_escrow: address, _admin: address): + """ + @notice Contract constructor + @param _token `ERC20ANGLE` contract address + @param _voting_escrow `VotingEscrow` contract address + """ + assert _token != ZERO_ADDRESS + assert _voting_escrow != ZERO_ADDRESS + assert _admin != ZERO_ADDRESS + + self.admin = _admin + self.token = _token + self.voting_escrow = _voting_escrow + self.time_total = block.timestamp / WEEK * WEEK + + +@external +def commit_transfer_ownership(addr: address): + """ + @notice Transfer ownership of GaugeController to `addr` + @param addr Address to have ownership transferred to + """ + assert msg.sender == self.admin # dev: admin only + assert addr != ZERO_ADDRESS # dev: future admin cannot be the 0 address + self.future_admin = addr + log CommitOwnership(addr) + + +@external +def accept_transfer_ownership(): + """ + @notice Accept a pending ownership transfer + """ + _admin: address = self.future_admin + assert msg.sender == _admin # dev: future admin only + + self.admin = _admin + log ApplyOwnership(_admin) + + +@external +@view +def gauge_types(_addr: address) -> int128: + """ + @notice Get gauge type for address + @param _addr Gauge address + @return Gauge type id + """ + gauge_type: int128 = self.gauge_types_[_addr] + assert gauge_type != 0 + + return gauge_type - 1 + + +@internal +def _get_type_weight(gauge_type: int128) -> uint256: + """ + @notice Fill historic type weights week-over-week for missed checkins + and return the type weight for the future week + @param gauge_type Gauge type id + @return Type weight + """ + t: uint256 = self.time_type_weight[gauge_type] + if t > 0: + w: uint256 = self.points_type_weight[gauge_type][t] + for i in range(500): + if t > block.timestamp: + break + t += WEEK + self.points_type_weight[gauge_type][t] = w + if t > block.timestamp: + self.time_type_weight[gauge_type] = t + return w + else: + return 0 + + +@internal +def _get_sum(gauge_type: int128) -> uint256: + """ + @notice Fill sum of gauge weights for the same type week-over-week for + missed checkins and return the sum for the future week + @param gauge_type Gauge type id + @return Sum of weights + """ + t: uint256 = self.time_sum[gauge_type] + if t > 0: + pt: Point = self.points_sum[gauge_type][t] + for i in range(500): + if t > block.timestamp: + break + t += WEEK + d_bias: uint256 = pt.slope * WEEK + if pt.bias > d_bias: + pt.bias -= d_bias + d_slope: uint256 = self.changes_sum[gauge_type][t] + pt.slope -= d_slope + else: + pt.bias = 0 + pt.slope = 0 + self.points_sum[gauge_type][t] = pt + if t > block.timestamp: + self.time_sum[gauge_type] = t + return pt.bias + else: + return 0 + + +@internal +def _get_total() -> uint256: + """ + @notice Fill historic total weights week-over-week for missed checkins + and return the total for the future week + @return Total weight + """ + t: uint256 = self.time_total + _n_gauge_types: int128 = self.n_gauge_types + if t > block.timestamp: + # If we have already checkpointed - still need to change the value + t -= WEEK + pt: uint256 = self.points_total[t] + + for gauge_type in range(100): + if gauge_type == _n_gauge_types: + break + self._get_sum(gauge_type) + self._get_type_weight(gauge_type) + + for i in range(500): + if t > block.timestamp: + break + t += WEEK + pt = 0 + # Scales as n_types * n_unchecked_weeks (hopefully 1 at most) + for gauge_type in range(100): + if gauge_type == _n_gauge_types: + break + type_sum: uint256 = self.points_sum[gauge_type][t].bias + type_weight: uint256 = self.points_type_weight[gauge_type][t] + pt += type_sum * type_weight + self.points_total[t] = pt + + if t > block.timestamp: + self.time_total = t + return pt + + +@internal +def _get_weight(gauge_addr: address) -> uint256: + """ + @notice Fill historic gauge weights week-over-week for missed checkins + and return the total for the future week + @param gauge_addr Address of the gauge + @return Gauge weight + """ + t: uint256 = self.time_weight[gauge_addr] + if t > 0: + pt: Point = self.points_weight[gauge_addr][t] + for i in range(500): + if t > block.timestamp: + break + t += WEEK + d_bias: uint256 = pt.slope * WEEK + if pt.bias > d_bias: + pt.bias -= d_bias + d_slope: uint256 = self.changes_weight[gauge_addr][t] + pt.slope -= d_slope + else: + pt.bias = 0 + pt.slope = 0 + self.points_weight[gauge_addr][t] = pt + if t > block.timestamp: + self.time_weight[gauge_addr] = t + return pt.bias + else: + return 0 + + +@external +def add_gauge(addr: address, gauge_type: int128, weight: uint256 = 0): + """ + @notice Add gauge `addr` of type `gauge_type` with weight `weight` + @param addr Gauge address + @param gauge_type Gauge type + @param weight Gauge weight + """ + assert msg.sender == self.admin + assert (gauge_type >= 0) and (gauge_type < self.n_gauge_types) + assert self.gauge_types_[addr] == 0 # dev: cannot add the same gauge twice + + n: int128 = self.n_gauges + self.n_gauges = n + 1 + self.gauges[n] = addr + + self.gauge_types_[addr] = gauge_type + 1 + next_time: uint256 = (block.timestamp + WEEK) / WEEK * WEEK + + if weight > 0: + _type_weight: uint256 = self._get_type_weight(gauge_type) + _old_sum: uint256 = self._get_sum(gauge_type) + _old_total: uint256 = self._get_total() + + self.points_sum[gauge_type][next_time].bias = weight + _old_sum + self.time_sum[gauge_type] = next_time + self.points_total[next_time] = _old_total + _type_weight * weight + self.time_total = next_time + + self.points_weight[addr][next_time].bias = weight + + if self.time_sum[gauge_type] == 0: + self.time_sum[gauge_type] = next_time + self.time_weight[addr] = next_time + + log NewGauge(addr, gauge_type, weight) + + +@external +def checkpoint(): + """ + @notice Checkpoint to fill data common for all gauges + """ + self._get_total() + + +@external +def checkpoint_gauge(addr: address): + """ + @notice Checkpoint to fill data for both a specific gauge and common for all gauges + @param addr Gauge address + """ + self._get_weight(addr) + self._get_total() + + +@internal +@view +def _gauge_relative_weight(addr: address, time: uint256) -> uint256: + """ + @notice Get Gauge relative weight (not more than 1.0) normalized to 1e18 + (e.g. 1.0 == 1e18). Inflation which will be received by it is + inflation_rate * relative_weight / 1e18 + @param addr Gauge address + @param time Relative weight at the specified timestamp in the past or present + @return Value of relative weight normalized to 1e18 + """ + t: uint256 = time / WEEK * WEEK + _total_weight: uint256 = self.points_total[t] + + if _total_weight > 0: + gauge_type: int128 = self.gauge_types_[addr] - 1 + _type_weight: uint256 = self.points_type_weight[gauge_type][t] + _gauge_weight: uint256 = self.points_weight[addr][t].bias + return MULTIPLIER * _type_weight * _gauge_weight / _total_weight + + else: + return 0 + + +@external +@view +def gauge_relative_weight(addr: address, time: uint256 = block.timestamp) -> uint256: + """ + @notice Get Gauge relative weight (not more than 1.0) normalized to 1e18 + (e.g. 1.0 == 1e18). Inflation which will be received by it is + inflation_rate * relative_weight / 1e18 + @param addr Gauge address + @param time Relative weight at the specified timestamp in the past or present + @return Value of relative weight normalized to 1e18 + """ + return self._gauge_relative_weight(addr, time) + + +@external +def gauge_relative_weight_write(addr: address, time: uint256 = block.timestamp) -> uint256: + """ + @notice Get gauge weight normalized to 1e18 and also fill all the unfilled + values for type and gauge records + @dev Any address can call, however nothing is recorded if the values are filled already + @param addr Gauge address + @param time Relative weight at the specified timestamp in the past or present + @return Value of relative weight normalized to 1e18 + """ + self._get_weight(addr) + self._get_total() # Also calculates get_sum + return self._gauge_relative_weight(addr, time) + + + + +@internal +def _change_type_weight(type_id: int128, weight: uint256): + """ + @notice Change type weight + @param type_id Type id + @param weight New type weight + """ + old_weight: uint256 = self._get_type_weight(type_id) + old_sum: uint256 = self._get_sum(type_id) + _total_weight: uint256 = self._get_total() + next_time: uint256 = (block.timestamp + WEEK) / WEEK * WEEK + + _total_weight = _total_weight + old_sum * weight - old_sum * old_weight + self.points_total[next_time] = _total_weight + self.points_type_weight[type_id][next_time] = weight + self.time_total = next_time + self.time_type_weight[type_id] = next_time + + log NewTypeWeight(type_id, next_time, weight, _total_weight) + + +@external +def add_type(_name: String[64], weight: uint256 = 0): + """ + @notice Add gauge type with name `_name` and weight `weight` + @param _name Name of gauge type + @param weight Weight of gauge type + """ + assert msg.sender == self.admin + type_id: int128 = self.n_gauge_types + self.gauge_type_names[type_id] = _name + self.n_gauge_types = type_id + 1 + if weight != 0: + self._change_type_weight(type_id, weight) + log AddType(_name, type_id) + + +@external +def change_type_weight(type_id: int128, weight: uint256): + """ + @notice Change gauge type `type_id` weight to `weight` + @param type_id Gauge type id + @param weight New Gauge weight + """ + assert msg.sender == self.admin + self._change_type_weight(type_id, weight) + + +@internal +def _change_gauge_weight(addr: address, weight: uint256): + # Change gauge weight + # Only needed when testing in reality + gauge_type: int128 = self.gauge_types_[addr] - 1 + old_gauge_weight: uint256 = self._get_weight(addr) + type_weight: uint256 = self._get_type_weight(gauge_type) + old_sum: uint256 = self._get_sum(gauge_type) + _total_weight: uint256 = self._get_total() + next_time: uint256 = (block.timestamp + WEEK) / WEEK * WEEK + + self.points_weight[addr][next_time].bias = weight + self.time_weight[addr] = next_time + + new_sum: uint256 = old_sum + weight - old_gauge_weight + self.points_sum[gauge_type][next_time].bias = new_sum + self.time_sum[gauge_type] = next_time + + _total_weight = _total_weight + new_sum * type_weight - old_sum * type_weight + self.points_total[next_time] = _total_weight + self.time_total = next_time + + log NewGaugeWeight(addr, block.timestamp, weight, _total_weight) + + +@external +def change_gauge_weight(addr: address, weight: uint256): + """ + @notice Change weight of gauge `addr` to `weight` + @param addr `GaugeController` contract address + @param weight New Gauge weight + """ + assert msg.sender == self.admin + self._change_gauge_weight(addr, weight) + + +@external +def vote_for_gauge_weights(_gauge_addr: address, _user_weight: uint256): + """ + @notice Allocate voting power for changing pool weights + @param _gauge_addr Gauge which `msg.sender` votes for + @param _user_weight Weight for a gauge in bps (units of 0.01%). Minimal is 0.01%. Ignored if 0 + """ + escrow: address = self.voting_escrow + slope: uint256 = convert(VotingEscrow(escrow).get_last_user_slope(msg.sender), uint256) + lock_end: uint256 = VotingEscrow(escrow).locked__end(msg.sender) + _n_gauges: int128 = self.n_gauges + next_time: uint256 = (block.timestamp + WEEK) / WEEK * WEEK + assert lock_end > next_time, "Your token lock expires too soon" + assert (_user_weight >= 0) and (_user_weight <= 10000), "You used all your voting power" + assert block.timestamp >= self.last_user_vote[msg.sender][_gauge_addr] + WEIGHT_VOTE_DELAY, "Cannot vote so often" + + gauge_type: int128 = self.gauge_types_[_gauge_addr] - 1 + assert gauge_type >= 0, "Gauge not added" + # Prepare slopes and biases in memory + old_slope: VotedSlope = self.vote_user_slopes[msg.sender][_gauge_addr] + old_dt: uint256 = 0 + if old_slope.end > next_time: + old_dt = old_slope.end - next_time + old_bias: uint256 = old_slope.slope * old_dt + new_slope: VotedSlope = VotedSlope({ + slope: slope * _user_weight / 10000, + end: lock_end, + power: _user_weight + }) + new_dt: uint256 = lock_end - next_time # dev: raises when expired + new_bias: uint256 = new_slope.slope * new_dt + + # Check and update powers (weights) used + power_used: uint256 = self.vote_user_power[msg.sender] + power_used = power_used + new_slope.power - old_slope.power + self.vote_user_power[msg.sender] = power_used + assert (power_used >= 0) and (power_used <= 10000), 'Used too much power' + + ## Remove old and schedule new slope changes + # Remove slope changes for old slopes + # Schedule recording of initial slope for next_time + old_weight_bias: uint256 = self._get_weight(_gauge_addr) + old_weight_slope: uint256 = self.points_weight[_gauge_addr][next_time].slope + old_sum_bias: uint256 = self._get_sum(gauge_type) + old_sum_slope: uint256 = self.points_sum[gauge_type][next_time].slope + + self.points_weight[_gauge_addr][next_time].bias = max(old_weight_bias + new_bias, old_bias) - old_bias + self.points_sum[gauge_type][next_time].bias = max(old_sum_bias + new_bias, old_bias) - old_bias + if old_slope.end > next_time: + self.points_weight[_gauge_addr][next_time].slope = max(old_weight_slope + new_slope.slope, old_slope.slope) - old_slope.slope + self.points_sum[gauge_type][next_time].slope = max(old_sum_slope + new_slope.slope, old_slope.slope) - old_slope.slope + else: + self.points_weight[_gauge_addr][next_time].slope += new_slope.slope + self.points_sum[gauge_type][next_time].slope += new_slope.slope + if old_slope.end > block.timestamp: + # Cancel old slope changes if they still didn't happen + self.changes_weight[_gauge_addr][old_slope.end] -= old_slope.slope + self.changes_sum[gauge_type][old_slope.end] -= old_slope.slope + # Add slope changes for new slopes + self.changes_weight[_gauge_addr][new_slope.end] += new_slope.slope + self.changes_sum[gauge_type][new_slope.end] += new_slope.slope + + self._get_total() + + self.vote_user_slopes[msg.sender][_gauge_addr] = new_slope + + # Record last action time + self.last_user_vote[msg.sender][_gauge_addr] = block.timestamp + + log VoteForGauge(block.timestamp, msg.sender, _gauge_addr, _user_weight) + + +@external +@view +def get_gauge_weight(addr: address) -> uint256: + """ + @notice Get current gauge weight + @param addr Gauge address + @return Gauge weight + """ + return self.points_weight[addr][self.time_weight[addr]].bias + + +@external +@view +def get_type_weight(type_id: int128) -> uint256: + """ + @notice Get current type weight + @param type_id Type id + @return Type weight + """ + return self.points_type_weight[type_id][self.time_type_weight[type_id]] + + +@external +@view +def get_total_weight() -> uint256: + """ + @notice Get current total (type-weighted) weight + @return Total weight + """ + return self.points_total[self.time_total] + + +@external +@view +def get_weights_sum_per_type(type_id: int128) -> uint256: + """ + @notice Get sum of gauge weights per type + @param type_id Type id + @return Sum of gauge weights + """ + return self.points_sum[type_id][self.time_sum[type_id]].bias + \ No newline at end of file diff --git a/contracts/staking/LiquidityGaugeV4.vy b/contracts/staking/LiquidityGaugeV4.vy new file mode 100644 index 0000000..a20241a --- /dev/null +++ b/contracts/staking/LiquidityGaugeV4.vy @@ -0,0 +1,633 @@ +# @version 0.2.16 +""" +@title Liquidity Gauge v4 +@author Angle Protocol +@license MIT +""" + +# Original idea and credit: +# Curve Finance's veCRV +# https://github.com/curvefi/curve-dao-contracts/blob/master/contracts/gauges/LiquidityGaugeV4.vy +# Mostly forked from Curve, except that now there is no direct link between the gauge controller +# and the gauges. In this implementation, ANGLE rewards are like any other token rewards. + +from vyper.interfaces import ERC20 + +implements: ERC20 + +interface VotingEscrow: + def user_point_epoch(addr: address) -> uint256: view + def user_point_history__ts(addr: address, epoch: uint256) -> uint256: view + +interface VotingEscrowBoost: + def adjusted_balance_of(_account: address) -> uint256: view + +interface ERC20Extended: + def symbol() -> String[26]: view + def decimals() -> uint256: view + + +event Deposit: + provider: indexed(address) + value: uint256 + +event Withdraw: + provider: indexed(address) + value: uint256 + +event UpdateLiquidityLimit: + user: address + original_balance: uint256 + original_supply: uint256 + working_balance: uint256 + working_supply: uint256 + +event CommitOwnership: + admin: address + +event ApplyOwnership: + admin: address + +event Transfer: + _from: indexed(address) + _to: indexed(address) + _value: uint256 + +event Approval: + _owner: indexed(address) + _spender: indexed(address) + _value: uint256 + +event RewardDataUpdate: + _token: indexed(address) + _amount: uint256 + +struct Reward: + token: address + distributor: address + period_finish: uint256 + rate: uint256 + last_update: uint256 + integral: uint256 + + +MAX_REWARDS: constant(uint256) = 8 +TOKENLESS_PRODUCTION: constant(uint256) = 40 +WEEK: constant(uint256) = 604800 + +ANGLE: public(address) +voting_escrow: public(address) +veBoost_proxy: public(address) + +staking_token: public(address) +decimal_staking_token: public(uint256) + +balanceOf: public(HashMap[address, uint256]) +totalSupply: public(uint256) +allowance: public(HashMap[address, HashMap[address, uint256]]) + +name: public(String[64]) +symbol: public(String[32]) + +working_balances: public(HashMap[address, uint256]) +working_supply: public(uint256) + +integrate_checkpoint_of: public(HashMap[address, uint256]) + +# For tracking external rewards +reward_count: public(uint256) +reward_tokens: public(address[MAX_REWARDS]) + +reward_data: public(HashMap[address, Reward]) + +# claimant -> default reward receiver +rewards_receiver: public(HashMap[address, address]) + +# reward token -> claiming address -> integral +reward_integral_for: public(HashMap[address, HashMap[address, uint256]]) + +# user -> [uint128 claimable amount][uint128 claimed amount] +claim_data: HashMap[address, HashMap[address, uint256]] + +admin: public(address) +future_admin: public(address) + +initialized: public(bool) + + +@external +def __init__(): + """ + @notice Contract constructor + @dev The contract has an initializer to prevent the take over of the implementation + """ + assert self.initialized == False #dev: contract is already initialized + self.initialized = True + +@external +def initialize(_staking_token: address, _admin: address, _ANGLE: address, _voting_escrow: address, _veBoost_proxy: address, _distributor: address): + """ + @notice Contract initializer + @param _staking_token Liquidity Pool contract address + @param _admin Admin who can kill the gauge + @param _ANGLE Address of the ANGLE token + @param _voting_escrow Address of the veANGLE contract + @param _veBoost_proxy Address of the proxy contract used to query veANGLE balances and taking into account potential delegations + @param _distributor Address of the contract responsible for distributing ANGLE tokens to this gauge + """ + assert self.initialized == False #dev: contract is already initialized + self.initialized = True + + assert _admin != ZERO_ADDRESS + assert _ANGLE != ZERO_ADDRESS + assert _voting_escrow != ZERO_ADDRESS + assert _veBoost_proxy != ZERO_ADDRESS + assert _distributor != ZERO_ADDRESS + + self.admin = _admin + self.staking_token = _staking_token + self.decimal_staking_token = ERC20Extended(_staking_token).decimals() + + symbol: String[26] = ERC20Extended(_staking_token).symbol() + self.name = concat("Angle ", symbol, " Gauge") + self.symbol = concat(symbol, "-gauge") + self.ANGLE = _ANGLE + self.voting_escrow = _voting_escrow + self.veBoost_proxy = _veBoost_proxy + + # add in all liquidityGauge the ANGLE reward - the distribution could be null though + self.reward_data[_ANGLE].distributor = _distributor + self.reward_tokens[0] = _ANGLE + self.reward_count = 1 + + +@view +@external +def decimals() -> uint256: + """ + @notice Get the number of decimals for this token + @dev Implemented as a view method to reduce gas costs + @return uint256 decimal places + """ + return self.decimal_staking_token + + +@internal +def _update_liquidity_limit(addr: address, l: uint256, L: uint256): + """ + @notice Calculate limits which depend on the amount of ANGLE token per-user. + Effectively it calculates working balances to apply amplification + of ANGLE production by ANGLE + @param addr User address + @param l User's amount of liquidity (LP tokens) + @param L Total amount of liquidity (LP tokens) + """ + # To be called after totalSupply is updated + voting_balance: uint256 = VotingEscrowBoost(self.veBoost_proxy).adjusted_balance_of(addr) + voting_total: uint256 = ERC20(self.voting_escrow).totalSupply() + + lim: uint256 = l * TOKENLESS_PRODUCTION / 100 + if voting_total > 0: + lim += L * voting_balance / voting_total * (100 - TOKENLESS_PRODUCTION) / 100 + + lim = min(l, lim) + old_bal: uint256 = self.working_balances[addr] + self.working_balances[addr] = lim + _working_supply: uint256 = self.working_supply + lim - old_bal + self.working_supply = _working_supply + + log UpdateLiquidityLimit(addr, l, L, lim, _working_supply) + + +@internal +def _checkpoint_reward(_user: address, token: address, _total_supply: uint256, _user_balance: uint256, _claim: bool, receiver: address): + """ + @notice Claim pending rewards and checkpoint rewards for a user + """ + total_supply: uint256 = _total_supply + user_balance: uint256 = _user_balance + if token == self.ANGLE : + total_supply = self.working_supply + user_balance = self.working_balances[_user] + + integral: uint256 = self.reward_data[token].integral + last_update: uint256 = min(block.timestamp, self.reward_data[token].period_finish) + duration: uint256 = last_update - self.reward_data[token].last_update + if duration != 0: + self.reward_data[token].last_update = last_update + if total_supply != 0: + integral += duration * self.reward_data[token].rate * 10**18 / total_supply + self.reward_data[token].integral = integral + + if _user != ZERO_ADDRESS: + integral_for: uint256 = self.reward_integral_for[token][_user] + new_claimable: uint256 = 0 + + if integral_for < integral: + self.reward_integral_for[token][_user] = integral + new_claimable = user_balance * (integral - integral_for) / 10**18 + + claim_data: uint256 = self.claim_data[_user][token] + total_claimable: uint256 = shift(claim_data, -128) + new_claimable + if total_claimable > 0: + total_claimed: uint256 = claim_data % 2**128 + if _claim: + response: Bytes[32] = raw_call( + token, + concat( + method_id("transfer(address,uint256)"), + convert(receiver, bytes32), + convert(total_claimable, bytes32), + ), + max_outsize=32, + ) + if len(response) != 0: + assert convert(response, bool) + self.claim_data[_user][token] = total_claimed + total_claimable + elif new_claimable > 0: + self.claim_data[_user][token] = total_claimed + shift(total_claimable, 128) + + if token == self.ANGLE : + self.integrate_checkpoint_of[_user] = block.timestamp + +@internal +def _checkpoint_rewards(_user: address, _total_supply: uint256, _claim: bool, _receiver: address, _only_checkpoint:bool = False): + """ + @notice Claim pending rewards and checkpoint rewards for a user + """ + + receiver: address = _receiver + user_balance: uint256 = 0 + if _user != ZERO_ADDRESS: + user_balance = self.balanceOf[_user] + if _claim and _receiver == ZERO_ADDRESS: + # if receiver is not explicitly declared, check if a default receiver is set + receiver = self.rewards_receiver[_user] + if receiver == ZERO_ADDRESS: + # if no default receiver is set, direct claims to the user + receiver = _user + + if _only_checkpoint: + self._checkpoint_reward(_user, self.ANGLE, _total_supply, user_balance, False, receiver) + else: + reward_count: uint256 = self.reward_count + for i in range(MAX_REWARDS): + if i == reward_count: + break + token: address = self.reward_tokens[i] + self._checkpoint_reward(_user, token, _total_supply, user_balance, _claim, receiver) + +@external +def user_checkpoint(addr: address) -> bool: + """ + @notice Record a checkpoint for `addr` + @param addr User address + @return bool success + """ + assert msg.sender == addr # dev: unauthorized + total_supply: uint256 = self.totalSupply + self._checkpoint_rewards(addr, total_supply, False, ZERO_ADDRESS, True) + self._update_liquidity_limit(addr, self.balanceOf[addr], total_supply) + return True + +@view +@external +def claimed_reward(_addr: address, _token: address) -> uint256: + """ + @notice Get the number of already-claimed reward tokens for a user + @param _addr Account to get reward amount for + @param _token Token to get reward amount for + @return uint256 Total amount of `_token` already claimed by `_addr` + """ + return self.claim_data[_addr][_token] % 2**128 + + +@view +@external +def claimable_reward(_user: address, _reward_token: address) -> uint256: + """ + @notice Get the number of claimable reward tokens for a user + @param _user Account to get reward amount for + @param _reward_token Token to get reward amount for + @return uint256 Claimable reward token amount + """ + integral: uint256 = self.reward_data[_reward_token].integral + total_supply: uint256 = self.totalSupply + user_balance: uint256 = self.balanceOf[_user] + if _reward_token == self.ANGLE : + total_supply = self.working_supply + user_balance = self.working_balances[_user] + + if total_supply != 0: + last_update: uint256 = min(block.timestamp, self.reward_data[_reward_token].period_finish) + duration: uint256 = last_update - self.reward_data[_reward_token].last_update + integral += (duration * self.reward_data[_reward_token].rate * 10**18 / total_supply) + + integral_for: uint256 = self.reward_integral_for[_reward_token][_user] + new_claimable: uint256 = user_balance * (integral - integral_for) / 10**18 + + return shift(self.claim_data[_user][_reward_token], -128) + new_claimable + + +@external +def set_rewards_receiver(_receiver: address): + """ + @notice Set the default reward receiver for the caller. + @dev When set to ZERO_ADDRESS, rewards are sent to the caller + @param _receiver Receiver address for any rewards claimed via `claim_rewards` + """ + self.rewards_receiver[msg.sender] = _receiver + + +@external +@nonreentrant('lock') +def claim_rewards(_addr: address = msg.sender, _receiver: address = ZERO_ADDRESS): + """ + @notice Claim available reward tokens for `_addr` + @param _addr Address to claim for + @param _receiver Address to transfer rewards to - if set to + ZERO_ADDRESS, uses the default reward receiver + for the caller + """ + if _receiver != ZERO_ADDRESS: + assert _addr == msg.sender # dev: cannot redirect when claiming for another user + self._checkpoint_rewards(_addr, self.totalSupply, True, _receiver) + + +@external +def kick(addr: address): + """ + @notice Kick `addr` for abusing their boost + @dev Only if either they had another voting event, or their voting escrow lock expired + @param addr Address to kick + """ + t_last: uint256 = self.integrate_checkpoint_of[addr] + t_ve: uint256 = VotingEscrow(self.voting_escrow).user_point_history__ts( + addr, VotingEscrow(self.voting_escrow).user_point_epoch(addr) + ) + _balance: uint256 = self.balanceOf[addr] + + assert ERC20(self.voting_escrow).balanceOf(addr) == 0 or t_ve > t_last # dev: kick not allowed + assert self.working_balances[addr] > _balance * TOKENLESS_PRODUCTION / 100 # dev: kick not needed + + total_supply: uint256 = self.totalSupply + self._checkpoint_rewards(addr, total_supply, False, ZERO_ADDRESS, True) + + self._update_liquidity_limit(addr, self.balanceOf[addr], total_supply) + + +@external +@nonreentrant('lock') +def deposit(_value: uint256, _addr: address = msg.sender, _claim_rewards: bool = False): + """ + @notice Deposit `_value` LP tokens + @dev Depositting also claims pending reward tokens + @param _value Number of tokens to deposit + @param _addr Address to deposit for + """ + total_supply: uint256 = self.totalSupply + + if _value != 0: + is_rewards: bool = self.reward_count != 0 + if is_rewards: + self._checkpoint_rewards(_addr, total_supply, _claim_rewards, ZERO_ADDRESS) + + total_supply += _value + new_balance: uint256 = self.balanceOf[_addr] + _value + self.balanceOf[_addr] = new_balance + self.totalSupply = total_supply + + self._update_liquidity_limit(_addr, new_balance, total_supply) + + ERC20(self.staking_token).transferFrom(msg.sender, self, _value) + else: + self._checkpoint_rewards(_addr, total_supply, False, ZERO_ADDRESS, True) + + log Deposit(_addr, _value) + log Transfer(ZERO_ADDRESS, _addr, _value) + + +@external +@nonreentrant('lock') +def withdraw(_value: uint256, _claim_rewards: bool = False): + """ + @notice Withdraw `_value` LP tokens + @dev Withdrawing also claims pending reward tokens + @param _value Number of tokens to withdraw + """ + total_supply: uint256 = self.totalSupply + + if _value != 0: + is_rewards: bool = self.reward_count != 0 + if is_rewards: + self._checkpoint_rewards(msg.sender, total_supply, _claim_rewards, ZERO_ADDRESS) + + total_supply -= _value + new_balance: uint256 = self.balanceOf[msg.sender] - _value + self.balanceOf[msg.sender] = new_balance + self.totalSupply = total_supply + + self._update_liquidity_limit(msg.sender, new_balance, total_supply) + + ERC20(self.staking_token).transfer(msg.sender, _value) + else: + self._checkpoint_rewards(msg.sender, total_supply, False, ZERO_ADDRESS, True) + + log Withdraw(msg.sender, _value) + log Transfer(msg.sender, ZERO_ADDRESS, _value) + + +@internal +def _transfer(_from: address, _to: address, _value: uint256): + total_supply: uint256 = self.totalSupply + + if _value != 0: + is_rewards: bool = self.reward_count != 0 + if is_rewards: + self._checkpoint_rewards(_from, total_supply, False, ZERO_ADDRESS) + new_balance: uint256 = self.balanceOf[_from] - _value + self.balanceOf[_from] = new_balance + self._update_liquidity_limit(_from, new_balance, total_supply) + + if is_rewards: + self._checkpoint_rewards(_to, total_supply, False, ZERO_ADDRESS) + new_balance = self.balanceOf[_to] + _value + self.balanceOf[_to] = new_balance + self._update_liquidity_limit(_to, new_balance, total_supply) + else: + self._checkpoint_rewards(_from, total_supply, False, ZERO_ADDRESS, True) + self._checkpoint_rewards(_to, total_supply, False, ZERO_ADDRESS, True) + + log Transfer(_from, _to, _value) + + +@external +@nonreentrant('lock') +def transfer(_to : address, _value : uint256) -> bool: + """ + @notice Transfer token for a specified address + @dev Transferring claims pending reward tokens for the sender and receiver + @param _to The address to transfer to. + @param _value The amount to be transferred. + """ + self._transfer(msg.sender, _to, _value) + + return True + + +@external +@nonreentrant('lock') +def transferFrom(_from : address, _to : address, _value : uint256) -> bool: + """ + @notice Transfer tokens from one address to another. + @dev Transferring claims pending reward tokens for the sender and receiver + @param _from address The address which you want to send tokens from + @param _to address The address which you want to transfer to + @param _value uint256 the amount of tokens to be transferred + """ + _allowance: uint256 = self.allowance[_from][msg.sender] + if _allowance != MAX_UINT256: + self.allowance[_from][msg.sender] = _allowance - _value + + self._transfer(_from, _to, _value) + + return True + + +@external +def approve(_spender : address, _value : uint256) -> bool: + """ + @notice Approve the passed address to transfer the specified amount of + tokens on behalf of msg.sender + @dev Beware that changing an allowance via this method brings the risk + that someone may use both the old and new allowance by unfortunate + transaction ordering. This may be mitigated with the use of + {incraseAllowance} and {decreaseAllowance}. + https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 + @param _spender The address which will transfer the funds + @param _value The amount of tokens that may be transferred + @return bool success + """ + self.allowance[msg.sender][_spender] = _value + log Approval(msg.sender, _spender, _value) + + return True + + +@external +def increaseAllowance(_spender: address, _added_value: uint256) -> bool: + """ + @notice Increase the allowance granted to `_spender` by the caller + @dev This is alternative to {approve} that can be used as a mitigation for + the potential race condition + @param _spender The address which will transfer the funds + @param _added_value The amount of to increase the allowance + @return bool success + """ + allowance: uint256 = self.allowance[msg.sender][_spender] + _added_value + self.allowance[msg.sender][_spender] = allowance + + log Approval(msg.sender, _spender, allowance) + + return True + + +@external +def decreaseAllowance(_spender: address, _subtracted_value: uint256) -> bool: + """ + @notice Decrease the allowance granted to `_spender` by the caller + @dev This is alternative to {approve} that can be used as a mitigation for + the potential race condition + @param _spender The address which will transfer the funds + @param _subtracted_value The amount of to decrease the allowance + @return bool success + """ + allowance: uint256 = self.allowance[msg.sender][_spender] - _subtracted_value + self.allowance[msg.sender][_spender] = allowance + + log Approval(msg.sender, _spender, allowance) + + return True + +@external +def add_reward(_reward_token: address, _distributor: address): + """ + @notice Set the active reward contract + """ + assert msg.sender == self.admin # dev: only owner + + reward_count: uint256 = self.reward_count + assert reward_count < MAX_REWARDS + assert self.reward_data[_reward_token].distributor == ZERO_ADDRESS + + self.reward_data[_reward_token].distributor = _distributor + self.reward_tokens[reward_count] = _reward_token + self.reward_count = reward_count + 1 + +@external +def set_reward_distributor(_reward_token: address, _distributor: address): + current_distributor: address = self.reward_data[_reward_token].distributor + + assert msg.sender == current_distributor or msg.sender == self.admin + assert current_distributor != ZERO_ADDRESS + assert _distributor != ZERO_ADDRESS + + self.reward_data[_reward_token].distributor = _distributor + + +@external +@nonreentrant("lock") +def deposit_reward_token(_reward_token: address, _amount: uint256): + assert msg.sender == self.reward_data[_reward_token].distributor + + self._checkpoint_rewards(ZERO_ADDRESS, self.totalSupply, False, ZERO_ADDRESS) + + response: Bytes[32] = raw_call( + _reward_token, + concat( + method_id("transferFrom(address,address,uint256)"), + convert(msg.sender, bytes32), + convert(self, bytes32), + convert(_amount, bytes32), + ), + max_outsize=32, + ) + if len(response) != 0: + assert convert(response, bool) + + period_finish: uint256 = self.reward_data[_reward_token].period_finish + if block.timestamp >= period_finish: + self.reward_data[_reward_token].rate = _amount / WEEK + else: + remaining: uint256 = period_finish - block.timestamp + leftover: uint256 = remaining * self.reward_data[_reward_token].rate + self.reward_data[_reward_token].rate = (_amount + leftover) / WEEK + + self.reward_data[_reward_token].last_update = block.timestamp + self.reward_data[_reward_token].period_finish = block.timestamp + WEEK + + log RewardDataUpdate(_reward_token,_amount) + +@external +def commit_transfer_ownership(addr: address): + """ + @notice Transfer ownership of Gauge to `addr` + @param addr Address to have ownership transferred to + """ + assert msg.sender == self.admin # dev: admin only + assert addr != ZERO_ADDRESS # dev: future admin cannot be the 0 address + + self.future_admin = addr + log CommitOwnership(addr) + + +@external +def accept_transfer_ownership(): + """ + @notice Accept a pending ownership transfer + """ + _admin: address = self.future_admin + assert msg.sender == _admin # dev: future admin only + + self.admin = _admin + log ApplyOwnership(_admin) diff --git a/contracts/staking/LiquidityGaugeV4UpgradedToken.vy b/contracts/staking/LiquidityGaugeV4UpgradedToken.vy new file mode 100644 index 0000000..8f0899a --- /dev/null +++ b/contracts/staking/LiquidityGaugeV4UpgradedToken.vy @@ -0,0 +1,670 @@ +# @version 0.2.16 +""" +@title Liquidity Gauge v4 +@author Angle Protocol +@license MIT +""" + +# Original idea and credit: +# Curve Finance's veCRV +# https://github.com/curvefi/curve-dao-contracts/blob/master/contracts/gauges/LiquidityGaugeV4.vy +# Mostly forked from Curve, except that now there is no direct link between the gauge controller +# and the gauges. In this implementation, there is a parameter to scale the value of the staking token +# which can be used if the staking_token and corresponding balances have to be upgraded + +from vyper.interfaces import ERC20 + +implements: ERC20 + +interface VotingEscrow: + def user_point_epoch(addr: address) -> uint256: view + def user_point_history__ts(addr: address, epoch: uint256) -> uint256: view + +interface VotingEscrowBoost: + def adjusted_balance_of(_account: address) -> uint256: view + +interface ERC20Extended: + def symbol() -> String[26]: view + def decimals() -> uint256: view + + +event Deposit: + provider: indexed(address) + value: uint256 + +event Withdraw: + provider: indexed(address) + value: uint256 + +event UpdateLiquidityLimit: + user: address + original_balance: uint256 + original_supply: uint256 + working_balance: uint256 + working_supply: uint256 + +event CommitOwnership: + admin: address + +event ApplyOwnership: + admin: address + +event Transfer: + _from: indexed(address) + _to: indexed(address) + _value: uint256 + +event Approval: + _owner: indexed(address) + _spender: indexed(address) + _value: uint256 + +event RewardDataUpdate: + _token: indexed(address) + _amount: uint256 + +struct Reward: + token: address + distributor: address + period_finish: uint256 + rate: uint256 + last_update: uint256 + integral: uint256 + + +MAX_REWARDS: constant(uint256) = 8 +TOKENLESS_PRODUCTION: constant(uint256) = 40 +WEEK: constant(uint256) = 604800 + +ANGLE: public(address) +voting_escrow: public(address) +veBoost_proxy: public(address) + +staking_token: public(address) +decimal_staking_token: public(uint256) + +balanceOf: public(HashMap[address, uint256]) +totalSupply: public(uint256) +allowance: public(HashMap[address, HashMap[address, uint256]]) + +name: public(String[64]) +symbol: public(String[32]) + +working_balances: public(HashMap[address, uint256]) +working_supply: public(uint256) + +integrate_checkpoint_of: public(HashMap[address, uint256]) + +# For tracking external rewards +reward_count: public(uint256) +reward_tokens: public(address[MAX_REWARDS]) + +reward_data: public(HashMap[address, Reward]) + +# claimant -> default reward receiver +rewards_receiver: public(HashMap[address, address]) + +# reward token -> claiming address -> integral +reward_integral_for: public(HashMap[address, HashMap[address, uint256]]) + +# user -> [uint128 claimable amount][uint128 claimed amount] +claim_data: HashMap[address, HashMap[address, uint256]] + +admin: public(address) +future_admin: public(address) + +initialized: public(bool) + +scaling_factor:public(uint256) + +@external +def __init__(): + """ + @notice Contract constructor + @dev The contract has an initializer to prevent the take over of the implementation + """ + assert self.initialized == False #dev: contract is already initialized + self.initialized = True + +@external +def initialize(_staking_token: address, _admin: address, _ANGLE: address, _voting_escrow: address, _veBoost_proxy: address, _distributor: address): + """ + @notice Contract initializer + @param _staking_token Liquidity Pool contract address + @param _admin Admin who can kill the gauge + @param _ANGLE Address of the ANGLE token + @param _voting_escrow Address of the veANGLE contract + @param _veBoost_proxy Address of the proxy contract used to query veANGLE balances and taking into account potential delegations + @param _distributor Address of the contract responsible for distributing ANGLE tokens to this gauge + """ + assert self.initialized == False #dev: contract is already initialized + self.initialized = True + + assert _admin != ZERO_ADDRESS + assert _ANGLE != ZERO_ADDRESS + assert _voting_escrow != ZERO_ADDRESS + assert _veBoost_proxy != ZERO_ADDRESS + assert _distributor != ZERO_ADDRESS + + self.admin = _admin + self.staking_token = _staking_token + self.decimal_staking_token = ERC20Extended(_staking_token).decimals() + + symbol: String[26] = ERC20Extended(_staking_token).symbol() + self.name = concat("Angle ", symbol, " Gauge") + self.symbol = concat(symbol, "-gauge") + self.ANGLE = _ANGLE + self.voting_escrow = _voting_escrow + self.veBoost_proxy = _veBoost_proxy + + # add in all liquidityGauge the ANGLE reward - the distribution could be null though + self.reward_data[_ANGLE].distributor = _distributor + self.reward_tokens[0] = _ANGLE + self.reward_count = 1 + + +@view +@external +def decimals() -> uint256: + """ + @notice Get the number of decimals for this token + @dev Implemented as a view method to reduce gas costs + @return uint256 decimal places + """ + return self.decimal_staking_token + + +@internal +def _update_liquidity_limit(addr: address, l: uint256, L: uint256): + """ + @notice Calculate limits which depend on the amount of ANGLE token per-user. + Effectively it calculates working balances to apply amplification + of ANGLE production by ANGLE + @param addr User address + @param l User's amount of liquidity (LP tokens) + @param L Total amount of liquidity (LP tokens) + """ + # To be called after totalSupply is updated + voting_balance: uint256 = VotingEscrowBoost(self.veBoost_proxy).adjusted_balance_of(addr) + voting_total: uint256 = ERC20(self.voting_escrow).totalSupply() + + lim: uint256 = l * TOKENLESS_PRODUCTION / 100 + if voting_total > 0: + lim += L * voting_balance / voting_total * (100 - TOKENLESS_PRODUCTION) / 100 + + lim = min(l, lim) + old_bal: uint256 = self.working_balances[addr] + self.working_balances[addr] = lim + _working_supply: uint256 = self.working_supply + lim - old_bal + self.working_supply = _working_supply + + log UpdateLiquidityLimit(addr, l, L, lim, _working_supply) + + +@internal +def _checkpoint_reward(_user: address, token: address, _total_supply: uint256, _user_balance: uint256, _claim: bool, receiver: address): + """ + @notice Claim pending rewards and checkpoint rewards for a user + """ + total_supply: uint256 = _total_supply + user_balance: uint256 = _user_balance + if token == self.ANGLE : + total_supply = self.working_supply + user_balance = self.working_balances[_user] + + integral: uint256 = self.reward_data[token].integral + last_update: uint256 = min(block.timestamp, self.reward_data[token].period_finish) + duration: uint256 = last_update - self.reward_data[token].last_update + if duration != 0: + self.reward_data[token].last_update = last_update + if total_supply != 0: + integral += duration * self.reward_data[token].rate * 10**18 / total_supply + self.reward_data[token].integral = integral + + if _user != ZERO_ADDRESS: + integral_for: uint256 = self.reward_integral_for[token][_user] + new_claimable: uint256 = 0 + + if integral_for < integral: + self.reward_integral_for[token][_user] = integral + new_claimable = user_balance * (integral - integral_for) / 10**18 + + claim_data: uint256 = self.claim_data[_user][token] + total_claimable: uint256 = shift(claim_data, -128) + new_claimable + if total_claimable > 0: + total_claimed: uint256 = claim_data % 2**128 + if _claim: + response: Bytes[32] = raw_call( + token, + concat( + method_id("transfer(address,uint256)"), + convert(receiver, bytes32), + convert(total_claimable, bytes32), + ), + max_outsize=32, + ) + if len(response) != 0: + assert convert(response, bool) + self.claim_data[_user][token] = total_claimed + total_claimable + elif new_claimable > 0: + self.claim_data[_user][token] = total_claimed + shift(total_claimable, 128) + + if token == self.ANGLE : + self.integrate_checkpoint_of[_user] = block.timestamp + +@internal +def _checkpoint_rewards(_user: address, _total_supply: uint256, _claim: bool, _receiver: address, _only_checkpoint:bool = False): + """ + @notice Claim pending rewards and checkpoint rewards for a user + """ + + receiver: address = _receiver + user_balance: uint256 = 0 + if _user != ZERO_ADDRESS: + user_balance = self.balanceOf[_user] + if _claim and _receiver == ZERO_ADDRESS: + # if receiver is not explicitly declared, check if a default receiver is set + receiver = self.rewards_receiver[_user] + if receiver == ZERO_ADDRESS: + # if no default receiver is set, direct claims to the user + receiver = _user + + if _only_checkpoint: + self._checkpoint_reward(_user, self.ANGLE, _total_supply, user_balance, False, receiver) + else: + reward_count: uint256 = self.reward_count + for i in range(MAX_REWARDS): + if i == reward_count: + break + token: address = self.reward_tokens[i] + self._checkpoint_reward(_user, token, _total_supply, user_balance, _claim, receiver) + +@external +def user_checkpoint(addr: address) -> bool: + """ + @notice Record a checkpoint for `addr` + @param addr User address + @return bool success + """ + assert msg.sender == addr # dev: unauthorized + total_supply: uint256 = self.totalSupply + self._checkpoint_rewards(addr, total_supply, False, ZERO_ADDRESS, True) + self._update_liquidity_limit(addr, self.balanceOf[addr], total_supply) + return True + +@view +@external +def claimed_reward(_addr: address, _token: address) -> uint256: + """ + @notice Get the number of already-claimed reward tokens for a user + @param _addr Account to get reward amount for + @param _token Token to get reward amount for + @return uint256 Total amount of `_token` already claimed by `_addr` + """ + return self.claim_data[_addr][_token] % 2**128 + + +@view +@external +def claimable_reward(_user: address, _reward_token: address) -> uint256: + """ + @notice Get the number of claimable reward tokens for a user + @param _user Account to get reward amount for + @param _reward_token Token to get reward amount for + @return uint256 Claimable reward token amount + """ + integral: uint256 = self.reward_data[_reward_token].integral + total_supply: uint256 = self.totalSupply + user_balance: uint256 = self.balanceOf[_user] + if _reward_token == self.ANGLE : + total_supply = self.working_supply + user_balance = self.working_balances[_user] + + if total_supply != 0: + last_update: uint256 = min(block.timestamp, self.reward_data[_reward_token].period_finish) + duration: uint256 = last_update - self.reward_data[_reward_token].last_update + integral += (duration * self.reward_data[_reward_token].rate * 10**18 / total_supply) + + integral_for: uint256 = self.reward_integral_for[_reward_token][_user] + new_claimable: uint256 = user_balance * (integral - integral_for) / 10**18 + + return shift(self.claim_data[_user][_reward_token], -128) + new_claimable + + +@external +def set_rewards_receiver(_receiver: address): + """ + @notice Set the default reward receiver for the caller. + @dev When set to ZERO_ADDRESS, rewards are sent to the caller + @param _receiver Receiver address for any rewards claimed via `claim_rewards` + """ + self.rewards_receiver[msg.sender] = _receiver + + +@external +@nonreentrant('lock') +def claim_rewards(_addr: address = msg.sender, _receiver: address = ZERO_ADDRESS): + """ + @notice Claim available reward tokens for `_addr` + @param _addr Address to claim for + @param _receiver Address to transfer rewards to - if set to + ZERO_ADDRESS, uses the default reward receiver + for the caller + """ + if _receiver != ZERO_ADDRESS: + assert _addr == msg.sender # dev: cannot redirect when claiming for another user + self._checkpoint_rewards(_addr, self.totalSupply, True, _receiver) + + +@external +def kick(addr: address): + """ + @notice Kick `addr` for abusing their boost + @dev Only if either they had another voting event, or their voting escrow lock expired + @param addr Address to kick + """ + t_last: uint256 = self.integrate_checkpoint_of[addr] + t_ve: uint256 = VotingEscrow(self.voting_escrow).user_point_history__ts( + addr, VotingEscrow(self.voting_escrow).user_point_epoch(addr) + ) + _balance: uint256 = self.balanceOf[addr] + + assert ERC20(self.voting_escrow).balanceOf(addr) == 0 or t_ve > t_last # dev: kick not allowed + assert self.working_balances[addr] > _balance * TOKENLESS_PRODUCTION / 100 # dev: kick not needed + + total_supply: uint256 = self.totalSupply + self._checkpoint_rewards(addr, total_supply, False, ZERO_ADDRESS, True) + + self._update_liquidity_limit(addr, self.balanceOf[addr], total_supply) + + +@external +@nonreentrant('lock') +def deposit(_value: uint256, _addr: address = msg.sender, _claim_rewards: bool = False): + """ + @notice Deposit `_value` LP tokens + @dev Depositting also claims pending reward tokens + @param _value Number of tokens to deposit + @param _addr Address to deposit for + """ + total_supply: uint256 = self.totalSupply + scaled_value: uint256 = _value + + _scaling_factor: uint256 = self.scaling_factor + if _scaling_factor != 0: + scaled_value = _value * _scaling_factor / 10**18 + + if scaled_value != 0: + is_rewards: bool = self.reward_count != 0 + if is_rewards: + self._checkpoint_rewards(_addr, total_supply, _claim_rewards, ZERO_ADDRESS) + + total_supply += scaled_value + new_balance: uint256 = self.balanceOf[_addr] + scaled_value + self.balanceOf[_addr] = new_balance + self.totalSupply = total_supply + + self._update_liquidity_limit(_addr, new_balance, total_supply) + + ERC20(self.staking_token).transferFrom(msg.sender, self, _value) + else: + self._checkpoint_rewards(_addr, total_supply, False, ZERO_ADDRESS, True) + + log Deposit(_addr, _value) + log Transfer(ZERO_ADDRESS, _addr, scaled_value) + + +@external +@nonreentrant('lock') +def withdraw(_value: uint256, _claim_rewards: bool = False): + """ + @notice Withdraw `_value` LP tokens + @dev Withdrawing also claims pending reward tokens + @param _value Number of tokens to withdraw + """ + total_supply: uint256 = self.totalSupply + _scaling_factor: uint256 = self.scaling_factor + scaled_value: uint256 = _value + + if _scaling_factor != 0: + scaled_value = _value * 10**18 / _scaling_factor + + if _value != 0: + + is_rewards: bool = self.reward_count != 0 + if is_rewards: + self._checkpoint_rewards(msg.sender, total_supply, _claim_rewards, ZERO_ADDRESS) + + total_supply -= _value + new_balance: uint256 = self.balanceOf[msg.sender] - _value + self.balanceOf[msg.sender] = new_balance + self.totalSupply = total_supply + + self._update_liquidity_limit(msg.sender, new_balance, total_supply) + ERC20(self.staking_token).transfer(msg.sender, scaled_value) + else: + self._checkpoint_rewards(msg.sender, total_supply, False, ZERO_ADDRESS, True) + + log Withdraw(msg.sender, scaled_value) + log Transfer(msg.sender, ZERO_ADDRESS, _value) + + +@internal +def _transfer(_from: address, _to: address, _value: uint256): + total_supply: uint256 = self.totalSupply + + if _value != 0: + is_rewards: bool = self.reward_count != 0 + if is_rewards: + self._checkpoint_rewards(_from, total_supply, False, ZERO_ADDRESS) + new_balance: uint256 = self.balanceOf[_from] - _value + self.balanceOf[_from] = new_balance + self._update_liquidity_limit(_from, new_balance, total_supply) + + if is_rewards: + self._checkpoint_rewards(_to, total_supply, False, ZERO_ADDRESS) + new_balance = self.balanceOf[_to] + _value + self.balanceOf[_to] = new_balance + self._update_liquidity_limit(_to, new_balance, total_supply) + else: + self._checkpoint_rewards(_from, total_supply, False, ZERO_ADDRESS, True) + self._checkpoint_rewards(_to, total_supply, False, ZERO_ADDRESS, True) + + log Transfer(_from, _to, _value) + + +@external +@nonreentrant('lock') +def transfer(_to : address, _value : uint256) -> bool: + """ + @notice Transfer token for a specified address + @dev Transferring claims pending reward tokens for the sender and receiver + @param _to The address to transfer to. + @param _value The amount to be transferred. + """ + self._transfer(msg.sender, _to, _value) + + return True + + +@external +@nonreentrant('lock') +def transferFrom(_from : address, _to : address, _value : uint256) -> bool: + """ + @notice Transfer tokens from one address to another. + @dev Transferring claims pending reward tokens for the sender and receiver + @param _from address The address which you want to send tokens from + @param _to address The address which you want to transfer to + @param _value uint256 the amount of tokens to be transferred + """ + _allowance: uint256 = self.allowance[_from][msg.sender] + if _allowance != MAX_UINT256: + self.allowance[_from][msg.sender] = _allowance - _value + + self._transfer(_from, _to, _value) + + return True + + +@external +def approve(_spender : address, _value : uint256) -> bool: + """ + @notice Approve the passed address to transfer the specified amount of + tokens on behalf of msg.sender + @dev Beware that changing an allowance via this method brings the risk + that someone may use both the old and new allowance by unfortunate + transaction ordering. This may be mitigated with the use of + {incraseAllowance} and {decreaseAllowance}. + https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 + @param _spender The address which will transfer the funds + @param _value The amount of tokens that may be transferred + @return bool success + """ + self.allowance[msg.sender][_spender] = _value + log Approval(msg.sender, _spender, _value) + + return True + + +@external +def increaseAllowance(_spender: address, _added_value: uint256) -> bool: + """ + @notice Increase the allowance granted to `_spender` by the caller + @dev This is alternative to {approve} that can be used as a mitigation for + the potential race condition + @param _spender The address which will transfer the funds + @param _added_value The amount of to increase the allowance + @return bool success + """ + allowance: uint256 = self.allowance[msg.sender][_spender] + _added_value + self.allowance[msg.sender][_spender] = allowance + + log Approval(msg.sender, _spender, allowance) + + return True + + +@external +def decreaseAllowance(_spender: address, _subtracted_value: uint256) -> bool: + """ + @notice Decrease the allowance granted to `_spender` by the caller + @dev This is alternative to {approve} that can be used as a mitigation for + the potential race condition + @param _spender The address which will transfer the funds + @param _subtracted_value The amount of to decrease the allowance + @return bool success + """ + allowance: uint256 = self.allowance[msg.sender][_spender] - _subtracted_value + self.allowance[msg.sender][_spender] = allowance + + log Approval(msg.sender, _spender, allowance) + + return True + +@external +def add_reward(_reward_token: address, _distributor: address): + """ + @notice Set the active reward contract + """ + assert msg.sender == self.admin # dev: only owner + + reward_count: uint256 = self.reward_count + assert reward_count < MAX_REWARDS + assert self.reward_data[_reward_token].distributor == ZERO_ADDRESS + + self.reward_data[_reward_token].distributor = _distributor + self.reward_tokens[reward_count] = _reward_token + self.reward_count = reward_count + 1 + +@external +def set_reward_distributor(_reward_token: address, _distributor: address): + current_distributor: address = self.reward_data[_reward_token].distributor + + assert msg.sender == current_distributor or msg.sender == self.admin + assert current_distributor != ZERO_ADDRESS + assert _distributor != ZERO_ADDRESS + + self.reward_data[_reward_token].distributor = _distributor + + +@external +@nonreentrant("lock") +def deposit_reward_token(_reward_token: address, _amount: uint256): + assert msg.sender == self.reward_data[_reward_token].distributor + + self._checkpoint_rewards(ZERO_ADDRESS, self.totalSupply, False, ZERO_ADDRESS) + + response: Bytes[32] = raw_call( + _reward_token, + concat( + method_id("transferFrom(address,address,uint256)"), + convert(msg.sender, bytes32), + convert(self, bytes32), + convert(_amount, bytes32), + ), + max_outsize=32, + ) + if len(response) != 0: + assert convert(response, bool) + + period_finish: uint256 = self.reward_data[_reward_token].period_finish + if block.timestamp >= period_finish: + self.reward_data[_reward_token].rate = _amount / WEEK + else: + remaining: uint256 = period_finish - block.timestamp + leftover: uint256 = remaining * self.reward_data[_reward_token].rate + self.reward_data[_reward_token].rate = (_amount + leftover) / WEEK + + self.reward_data[_reward_token].last_update = block.timestamp + self.reward_data[_reward_token].period_finish = block.timestamp + WEEK + + log RewardDataUpdate(_reward_token,_amount) + +@external +def commit_transfer_ownership(addr: address): + """ + @notice Transfer ownership of Gauge to `addr` + @param addr Address to have ownership transferred to + """ + assert msg.sender == self.admin # dev: admin only + assert addr != ZERO_ADDRESS # dev: future admin cannot be the 0 address + + self.future_admin = addr + log CommitOwnership(addr) + + +@external +def accept_transfer_ownership(): + """ + @notice Accept a pending ownership transfer + """ + _admin: address = self.future_admin + assert msg.sender == _admin # dev: future admin only + + self.admin = _admin + log ApplyOwnership(_admin) + +@external +def recover_erc20(token: address, addr: address, amount: uint256): + """ + @notice Recovers tokens sent or accruing to this contract + @param token Token to be recovered + @param addr Address to send the token to + @param amount Amount of tokens to send + """ + assert msg.sender == self.admin # dev: only owner + if token == self: + self._transfer(self, addr, amount) + else: + ERC20(token).transfer(addr, amount) + + +@external +def set_staking_token_and_scaling_factor(token: address, _value: uint256): + """ + @notice Sets the staking token + """ + assert msg.sender == self.admin # dev: only owner + self.staking_token = token + self.decimal_staking_token = ERC20Extended(token).decimals() + self.scaling_factor = _value diff --git a/contracts/staking/RewardsDistributor.sol b/contracts/staking/RewardsDistributor.sol new file mode 100644 index 0000000..df6b97d --- /dev/null +++ b/contracts/staking/RewardsDistributor.sol @@ -0,0 +1,328 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "./RewardsDistributorEvents.sol"; + +/// @notice Distribution parameters for a given contract +struct StakingParameters { + // Amount of rewards distributed since the beginning + uint256 distributedRewards; + // Last time rewards were distributed to the staking contract + uint256 lastDistributionTime; + // Frequency with which rewards should be given to the underlying contract + uint256 updateFrequency; + // Number of tokens distributed for the person calling the update function + uint256 incentiveAmount; + // Time at which reward distribution started for this reward contract + uint256 timeStarted; + // Amount of time during which rewards will be distributed + uint256 duration; + // Amount of tokens to distribute to the concerned contract + uint256 amountToDistribute; +} + +/// @title RewardsDistributor +/// @author Angle Core Team (forked form FEI Protocol) +/// @notice Controls and handles the distribution of governance tokens to the different staking contracts of the protocol +/// @dev Inspired from FEI contract: +/// https://github.com/fei-protocol/fei-protocol-core/blob/master/contracts/staking/FeiRewardsDistributor.sol +contract RewardsDistributor is RewardsDistributorEvents, IRewardsDistributor, AccessControl { + using SafeERC20 for IERC20; + + /// @notice Role for governors only + bytes32 public constant GOVERNOR_ROLE = keccak256("GOVERNOR_ROLE"); + /// @notice Role for guardians and governors + bytes32 public constant GUARDIAN_ROLE = keccak256("GUARDIAN_ROLE"); + + // ============================ Reference to a contract ======================== + + /// @notice Token used as a reward + IERC20 public immutable override rewardToken; + + // ============================== Parameters =================================== + + /// @notice Maps a `StakingContract` to its distribution parameters + mapping(IStakingRewards => StakingParameters) public stakingContractsMap; + + /// @notice List of all the staking contracts handled by the rewards distributor + /// Used to be able to change the rewards distributor and propagate a new reference to the underlying + /// staking contract + IStakingRewards[] public stakingContractsList; + + // ============================ Constructor ==================================== + + /// @notice Initializes the distributor contract with a first set of parameters + /// @param governorList List of the governor addresses of the protocol + /// @param guardian The guardian address, optional + /// @param rewardTokenAddress The ERC20 token to distribute + constructor( + address[] memory governorList, + address guardian, + address rewardTokenAddress + ) { + require(rewardTokenAddress != address(0) && guardian != address(0), "0"); + require(governorList.length > 0, "47"); + rewardToken = IERC20(rewardTokenAddress); + // Since this contract is independent from the rest of the protocol + // When updating the governor list, governors should make sure to still update the roles + // in this contract + _setRoleAdmin(GOVERNOR_ROLE, GOVERNOR_ROLE); + _setRoleAdmin(GUARDIAN_ROLE, GOVERNOR_ROLE); + for (uint256 i = 0; i < governorList.length; i++) { + require(governorList[i] != address(0), "0"); + _setupRole(GOVERNOR_ROLE, governorList[i]); + _setupRole(GUARDIAN_ROLE, governorList[i]); + } + _setupRole(GUARDIAN_ROLE, guardian); + } + + // ============================ External Functions ============================= + + /// @notice Sends reward tokens to the staking contract + /// @param stakingContract Reference to the staking contract + /// @dev The way to pause this function is to set `updateFrequency` to infinity, + /// or to completely delete the contract + /// @dev A keeper calling this function could be frontran by a miner seeing the potential profit + /// from calling this function + /// @dev This function automatically computes the amount of reward tokens to send to the staking + /// contract based on the time elapsed since the last drip, on the amount to distribute and on + /// the duration of the distribution + function drip(IStakingRewards stakingContract) external override returns (uint256) { + StakingParameters storage stakingParams = stakingContractsMap[stakingContract]; + require(stakingParams.duration > 0, "80"); + require(_isDripAvailable(stakingParams), "81"); + + uint256 dripAmount = _computeDripAmount(stakingParams); + stakingParams.lastDistributionTime = block.timestamp; + require(dripAmount != 0, "82"); + stakingParams.distributedRewards += dripAmount; + emit Dripped(msg.sender, dripAmount, address(stakingContract)); + + rewardToken.safeTransfer(address(stakingContract), dripAmount); + IStakingRewards(stakingContract).notifyRewardAmount(dripAmount); + _incentivize(stakingParams); + + return dripAmount; + } + + // =========================== Governor Functions ============================== + + /// @notice Sends tokens back to governance treasury or another address + /// @param amount Amount of tokens to send back to treasury + /// @param to Address to send the tokens to + /// @dev Only callable by governance and not by the guardian + function governorWithdrawRewardToken(uint256 amount, address to) external override onlyRole(GOVERNOR_ROLE) { + emit RewardTokenWithdrawn(amount); + rewardToken.safeTransfer(to, amount); + } + + /// @notice Function to withdraw ERC20 tokens that could accrue on a staking contract + /// @param tokenAddress Address of the ERC20 to recover + /// @param to Address to transfer to + /// @param amount Amount to transfer + /// @param stakingContract Reference to the staking contract + /// @dev A use case would be to claim tokens if the staked tokens accumulate rewards or if tokens were + /// mistakenly sent to staking contracts + function governorRecover( + address tokenAddress, + address to, + uint256 amount, + IStakingRewards stakingContract + ) external override onlyRole(GOVERNOR_ROLE) { + stakingContract.recoverERC20(tokenAddress, to, amount); + } + + /// @notice Sets a new rewards distributor contract and automatically makes this contract useless + /// @param newRewardsDistributor Address of the new rewards distributor contract + /// @dev This contract is not upgradeable, setting a new contract could allow for upgrades, which should be + /// propagated across all staking contracts + /// @dev This function transfers all the reward tokens to the new address + /// @dev The new rewards distributor contract should be initialized correctly with all the staking contracts + /// from the staking contract list + function setNewRewardsDistributor(address newRewardsDistributor) external override onlyRole(GOVERNOR_ROLE) { + // Checking the compatibility of the reward tokens. It is checked at the initialization of each staking contract + // in the `setStakingContract` function that reward tokens are compatible with the `rewardsDistributor`. If + // the `newRewardsDistributor` has a compatible rewards token, then all staking contracts will automatically be + // compatible with it + require(address(IRewardsDistributor(newRewardsDistributor).rewardToken()) == address(rewardToken), "83"); + require(newRewardsDistributor != address(this), "84"); + for (uint256 i = 0; i < stakingContractsList.length; i++) { + stakingContractsList[i].setNewRewardsDistribution(newRewardsDistributor); + } + rewardToken.safeTransfer(newRewardsDistributor, rewardToken.balanceOf(address(this))); + // The functions `setStakingContract` should then be called for each staking contract in the `newRewardsDistributor` + emit NewRewardsDistributor(newRewardsDistributor); + } + + /// @notice Deletes a staking contract from the staking contract map and removes it from the + /// `stakingContractsList` + /// @param stakingContract Contract to remove + /// @dev Allows to clean some space and to avoid keeping in memory contracts which became useless + /// @dev It is also a way governance has to completely stop rewards distribution from a contract + function removeStakingContract(IStakingRewards stakingContract) external override onlyRole(GOVERNOR_ROLE) { + uint256 indexMet; + uint256 stakingContractsListLength = stakingContractsList.length; + require(stakingContractsListLength >= 1, "80"); + for (uint256 i = 0; i < stakingContractsListLength - 1; i++) { + if (stakingContractsList[i] == stakingContract) { + indexMet = 1; + stakingContractsList[i] = stakingContractsList[stakingContractsListLength - 1]; + break; + } + } + require(indexMet == 1 || stakingContractsList[stakingContractsListLength - 1] == stakingContract, "80"); + + stakingContractsList.pop(); + + delete stakingContractsMap[stakingContract]; + emit DeletedStakingContract(address(stakingContract)); + } + + // =================== Guardian Functions (for parameters) ===================== + + /// @notice Notifies and initializes a new staking contract + /// @param _stakingContract Address of the staking contract + /// @param _duration Time frame during which tokens will be distributed + /// @param _incentiveAmount Incentive amount given to keepers calling the update function + /// @param _updateFrequency Frequency when it is possible to call the update function and give tokens to the staking contract + /// @param _amountToDistribute Amount of gov tokens to give to the staking contract across all drips + /// @dev Called by governance to activate a contract + /// @dev After setting a new staking contract, everything is as if the contract had already been set for `_updateFrequency` + /// meaning that it is possible to `drip` the staking contract immediately after that + function setStakingContract( + address _stakingContract, + uint256 _duration, + uint256 _incentiveAmount, + uint256 _updateFrequency, + uint256 _amountToDistribute + ) external override onlyRole(GOVERNOR_ROLE) { + require(_duration > 0, "85"); + require(_duration >= _updateFrequency && block.timestamp >= _updateFrequency, "86"); + + IStakingRewards stakingContract = IStakingRewards(_stakingContract); + + require(stakingContract.rewardToken() == rewardToken, "83"); + + StakingParameters storage stakingParams = stakingContractsMap[stakingContract]; + + stakingParams.updateFrequency = _updateFrequency; + stakingParams.incentiveAmount = _incentiveAmount; + stakingParams.lastDistributionTime = block.timestamp - _updateFrequency; + // In order to allow a drip whenever a `stakingContract` is set, we consider that staking has already started + // `_updateFrequency` ago + stakingParams.timeStarted = block.timestamp - _updateFrequency; + stakingParams.duration = _duration; + stakingParams.amountToDistribute = _amountToDistribute; + stakingContractsList.push(stakingContract); + + emit NewStakingContract(_stakingContract); + } + + /// @notice Sets the update frequency + /// @param _updateFrequency New update frequency + /// @param stakingContract Reference to the staking contract + function setUpdateFrequency(uint256 _updateFrequency, IStakingRewards stakingContract) + external + override + onlyRole(GUARDIAN_ROLE) + { + StakingParameters storage stakingParams = stakingContractsMap[stakingContract]; + require(stakingParams.duration > 0, "80"); + require(stakingParams.duration >= _updateFrequency, "87"); + stakingParams.updateFrequency = _updateFrequency; + emit FrequencyUpdated(_updateFrequency, address(stakingContract)); + } + + /// @notice Sets the incentive amount for calling drip + /// @param _incentiveAmount New incentive amount + /// @param stakingContract Reference to the staking contract + function setIncentiveAmount(uint256 _incentiveAmount, IStakingRewards stakingContract) + external + override + onlyRole(GUARDIAN_ROLE) + { + StakingParameters storage stakingParams = stakingContractsMap[stakingContract]; + require(stakingParams.duration > 0, "80"); + stakingParams.incentiveAmount = _incentiveAmount; + emit IncentiveUpdated(_incentiveAmount, address(stakingContract)); + } + + /// @notice Sets the new amount to distribute to a staking contract + /// @param _amountToDistribute New amount to distribute + /// @param stakingContract Reference to the staking contract + function setAmountToDistribute(uint256 _amountToDistribute, IStakingRewards stakingContract) + external + override + onlyRole(GUARDIAN_ROLE) + { + StakingParameters storage stakingParams = stakingContractsMap[stakingContract]; + require(stakingParams.duration > 0, "80"); + require(stakingParams.distributedRewards < _amountToDistribute, "88"); + stakingParams.amountToDistribute = _amountToDistribute; + emit AmountToDistributeUpdated(_amountToDistribute, address(stakingContract)); + } + + /// @notice Sets the new duration with which tokens will be distributed to the staking contract + /// @param _duration New duration + /// @param stakingContract Reference to the staking contract + function setDuration(uint256 _duration, IStakingRewards stakingContract) external override onlyRole(GUARDIAN_ROLE) { + StakingParameters storage stakingParams = stakingContractsMap[stakingContract]; + require(stakingParams.duration > 0, "80"); + require(_duration >= stakingParams.updateFrequency, "87"); + uint256 timeElapsed = _timeSinceStart(stakingParams); + require(timeElapsed < stakingParams.duration && timeElapsed < _duration, "66"); + stakingParams.duration = _duration; + emit DurationUpdated(_duration, address(stakingContract)); + } + + // =========================== Internal Functions ============================== + + /// @notice Gives the next time when `drip` could be called + /// @param stakingParams Parameters of the concerned staking contract + /// @return Block timestamp when `drip` will next be available + function _nextDripAvailable(StakingParameters memory stakingParams) internal pure returns (uint256) { + return stakingParams.lastDistributionTime + stakingParams.updateFrequency; + } + + /// @notice Tells if `drip` can currently be called + /// @param stakingParams Parameters of the concerned staking contract + /// @return If the `updateFrequency` has passed since the last drip + function _isDripAvailable(StakingParameters memory stakingParams) internal view returns (bool) { + return block.timestamp >= _nextDripAvailable(stakingParams); + } + + /// @notice Computes the amount of tokens to give at the current drip + /// @param stakingParams Parameters of the concerned staking contract + /// @dev Constant drip amount across time + function _computeDripAmount(StakingParameters memory stakingParams) internal view returns (uint256) { + if (stakingParams.distributedRewards >= stakingParams.amountToDistribute) { + return 0; + } + uint256 dripAmount = (stakingParams.amountToDistribute * + (block.timestamp - stakingParams.lastDistributionTime)) / stakingParams.duration; + uint256 timeLeft = stakingParams.duration - _timeSinceStart(stakingParams); + uint256 rewardsLeftToDistribute = stakingParams.amountToDistribute - stakingParams.distributedRewards; + if (timeLeft < stakingParams.updateFrequency || rewardsLeftToDistribute < dripAmount || timeLeft == 0) { + return rewardsLeftToDistribute; + } else { + return dripAmount; + } + } + + /// @notice Computes the time since distribution has started for the staking contract + /// @param stakingParams Parameters of the concerned staking contract + /// @return The time since distribution has started for the staking contract + function _timeSinceStart(StakingParameters memory stakingParams) internal view returns (uint256) { + uint256 _duration = stakingParams.duration; + // `block.timestamp` is always greater than `timeStarted` + uint256 timePassed = block.timestamp - stakingParams.timeStarted; + return timePassed > _duration ? _duration : timePassed; + } + + /// @notice Incentivizes the person calling the drip function + /// @param stakingParams Parameters of the concerned staking contract + function _incentivize(StakingParameters memory stakingParams) internal { + rewardToken.safeTransfer(msg.sender, stakingParams.incentiveAmount); + } +} diff --git a/contracts/staking/RewardsDistributorEvents.sol b/contracts/staking/RewardsDistributorEvents.sol new file mode 100644 index 0000000..f2c37db --- /dev/null +++ b/contracts/staking/RewardsDistributorEvents.sol @@ -0,0 +1,34 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; +import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; + +import "../external/AccessControl.sol"; + +import "../interfaces/IRewardsDistributor.sol"; +import "../interfaces/IStakingRewards.sol"; + +/// @title RewardsDistributorEvents +/// @author Angle Core Team +/// @notice All the events used in `RewardsDistributor` contract +contract RewardsDistributorEvents { + event Dripped(address indexed _caller, uint256 _amount, address _stakingContract); + + event RewardTokenWithdrawn(uint256 _amount); + + event FrequencyUpdated(uint256 _frequency, address indexed _stakingContract); + + event IncentiveUpdated(uint256 _incentiveAmount, address indexed _stakingContract); + + event AmountToDistributeUpdated(uint256 _amountToDistribute, address indexed _stakingContract); + + event DurationUpdated(uint256 _duration, address indexed _stakingContract); + + event NewStakingContract(address indexed _stakingContract); + + event DeletedStakingContract(address indexed stakingContract); + + event NewRewardsDistributor(address indexed newRewardsDistributor); +} diff --git a/contracts/staking/StakingRewards.sol b/contracts/staking/StakingRewards.sol new file mode 100644 index 0000000..ee14dee --- /dev/null +++ b/contracts/staking/StakingRewards.sol @@ -0,0 +1,284 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "./StakingRewardsEvents.sol"; + +/// @title StakingRewards +/// @author Forked form SetProtocol +/// https://github.com/SetProtocol/index-coop-contracts/blob/master/contracts/staking/StakingRewards.sol +/// @notice The `StakingRewards` contracts allows to stake an ERC20 token to receive as reward another ERC20 +/// @dev This contracts is managed by the reward distributor and implements the staking interface +contract StakingRewards is StakingRewardsEvents, IStakingRewards, ReentrancyGuard { + using SafeERC20 for IERC20; + + /// @notice Checks to see if it is the `rewardsDistribution` calling this contract + /// @dev There is no Access Control here, because it can be handled cheaply through these modifiers + modifier onlyRewardsDistribution() { + require(msg.sender == rewardsDistribution, "1"); + _; + } + + // ============================ References to contracts ======================== + + /// @notice ERC20 token given as reward + IERC20 public immutable override rewardToken; + + /// @notice ERC20 token used for staking + IERC20 public immutable stakingToken; + + /// @notice Base of the staked token, it is going to be used in the case of sanTokens + /// which are not in base 10**18 + uint256 public immutable stakingBase; + + /// @notice Rewards Distribution contract for this staking contract + address public rewardsDistribution; + + // ============================ Staking parameters ============================= + + /// @notice Time at which distribution ends + uint256 public periodFinish; + + /// @notice Reward per second given to the staking contract, split among the staked tokens + uint256 public rewardRate; + + /// @notice Duration of the reward distribution + uint256 public rewardsDuration; + + /// @notice Last time `rewardPerTokenStored` was updated + uint256 public lastUpdateTime; + + /// @notice Helps to compute the amount earned by someone + /// Cumulates rewards accumulated for one token since the beginning. + /// Stored as a uint so it is actually a float times the base of the reward token + uint256 public rewardPerTokenStored; + + /// @notice Stores for each account the `rewardPerToken`: we do the difference + /// between the current and the old value to compute what has been earned by an account + mapping(address => uint256) public userRewardPerTokenPaid; + + /// @notice Stores for each account the accumulated rewards + mapping(address => uint256) public rewards; + + uint256 private _totalSupply; + + mapping(address => uint256) private _balances; + + // ============================ Constructor ==================================== + + /// @notice Initializes the staking contract with a first set of parameters + /// @param _rewardsDistribution Address owning the rewards token + /// @param _rewardToken ERC20 token given as reward + /// @param _stakingToken ERC20 token used for staking + /// @param _rewardsDuration Duration of the staking contract + constructor( + address _rewardsDistribution, + address _rewardToken, + address _stakingToken, + uint256 _rewardsDuration + ) { + require(_stakingToken != address(0) && _rewardToken != address(0) && _rewardsDistribution != address(0), "0"); + + // We are not checking the compatibility of the reward token between the distributor and this contract here + // because it is checked by the `RewardsDistributor` when activating the staking contract + // Parameters + rewardToken = IERC20(_rewardToken); + stakingToken = IERC20(_stakingToken); + rewardsDuration = _rewardsDuration; + rewardsDistribution = _rewardsDistribution; + + stakingBase = 10**IERC20Metadata(_stakingToken).decimals(); + } + + // ============================ Modifiers ====================================== + + /// @notice Checks to see if the calling address is the zero address + /// @param account Address to check + modifier zeroCheck(address account) { + require(account != address(0), "0"); + _; + } + + /// @notice Called frequently to update the staking parameters associated to an address + /// @param account Address of the account to update + modifier updateReward(address account) { + rewardPerTokenStored = rewardPerToken(); + lastUpdateTime = lastTimeRewardApplicable(); + if (account != address(0)) { + rewards[account] = earned(account); + userRewardPerTokenPaid[account] = rewardPerTokenStored; + } + _; + } + + // ============================ View functions ================================= + + /// @notice Accesses the total supply + /// @dev Used instead of having a public variable to respect the ERC20 standard + function totalSupply() external view returns (uint256) { + return _totalSupply; + } + + /// @notice Accesses the number of token staked by an account + /// @param account Account to query the balance of + /// @dev Used instead of having a public variable to respect the ERC20 standard + function balanceOf(address account) external view returns (uint256) { + return _balances[account]; + } + + /// @notice Queries the last timestamp at which a reward was distributed + /// @dev Returns the current timestamp if a reward is being distributed and the end of the staking + /// period if staking is done + function lastTimeRewardApplicable() public view returns (uint256) { + return Math.min(block.timestamp, periodFinish); + } + + /// @notice Used to actualize the `rewardPerTokenStored` + /// @dev It adds to the reward per token: the time elapsed since the `rewardPerTokenStored` was + /// last updated multiplied by the `rewardRate` divided by the number of tokens + function rewardPerToken() public view returns (uint256) { + if (_totalSupply == 0) { + return rewardPerTokenStored; + } + return + rewardPerTokenStored + + (((lastTimeRewardApplicable() - lastUpdateTime) * rewardRate * stakingBase) / _totalSupply); + } + + /// @notice Returns how much a given account earned rewards + /// @param account Address for which the request is made + /// @return How much a given account earned rewards + /// @dev It adds to the rewards the amount of reward earned since last time that is the difference + /// in reward per token from now and last time multiplied by the number of tokens staked by the person + function earned(address account) public view returns (uint256) { + return + (_balances[account] * (rewardPerToken() - userRewardPerTokenPaid[account])) / + stakingBase + + rewards[account]; + } + + // ======================== Mutative functions forked ========================== + + /// @notice Lets someone stake a given amount of `stakingTokens` + /// @param amount Amount of ERC20 staking token that the `msg.sender` wants to stake + function stake(uint256 amount) external nonReentrant updateReward(msg.sender) { + _stake(amount, msg.sender); + } + + /// @notice Lets a user withdraw a given amount of collateral from the staking contract + /// @param amount Amount of the ERC20 staking token that the `msg.sender` wants to withdraw + function withdraw(uint256 amount) public nonReentrant updateReward(msg.sender) { + require(amount > 0, "89"); + _totalSupply = _totalSupply - amount; + _balances[msg.sender] = _balances[msg.sender] - amount; + stakingToken.safeTransfer(msg.sender, amount); + emit Withdrawn(msg.sender, amount); + } + + /// @notice Triggers a payment of the reward earned to the msg.sender + function getReward() public nonReentrant updateReward(msg.sender) { + uint256 reward = rewards[msg.sender]; + if (reward > 0) { + rewards[msg.sender] = 0; + rewardToken.safeTransfer(msg.sender, reward); + emit RewardPaid(msg.sender, reward); + } + } + + /// @notice Exits someone + /// @dev This function lets the caller withdraw its staking and claim rewards + // Attention here, there may be reentrancy attacks because of the following call + // to an external contract done before other things are modified, yet since the `rewardToken` + // is mostly going to be a trusted contract controlled by governance (namely the ANGLE token), + // this is not an issue. If the `rewardToken` changes to an untrusted contract, this need to be updated. + function exit() external { + withdraw(_balances[msg.sender]); + getReward(); + } + + // ====================== Functions added by Angle Core Team =================== + + /// @notice Allows to stake on behalf of another address + /// @param amount Amount to stake + /// @param onBehalf Address to stake onBehalf of + function stakeOnBehalf(uint256 amount, address onBehalf) + external + nonReentrant + zeroCheck(onBehalf) + updateReward(onBehalf) + { + _stake(amount, onBehalf); + } + + /// @notice Internal function to stake called by `stake` and `stakeOnBehalf` + /// @param amount Amount to stake + /// @param onBehalf Address to stake on behalf of + /// @dev Before calling this function, it has already been verified whether this address was a zero address or not + function _stake(uint256 amount, address onBehalf) internal { + require(amount > 0, "90"); + stakingToken.safeTransferFrom(msg.sender, address(this), amount); + _totalSupply = _totalSupply + amount; + _balances[onBehalf] = _balances[onBehalf] + amount; + emit Staked(onBehalf, amount); + } + + // ====================== Restricted Functions ================================= + + /// @notice Adds rewards to be distributed + /// @param reward Amount of reward tokens to distribute + /// @dev This reward will be distributed during `rewardsDuration` set previously + function notifyRewardAmount(uint256 reward) + external + override + onlyRewardsDistribution + nonReentrant + updateReward(address(0)) + { + if (block.timestamp >= periodFinish) { + // If no reward is currently being distributed, the new rate is just `reward / duration` + rewardRate = reward / rewardsDuration; + } else { + // Otherwise, cancel the future reward and add the amount left to distribute to reward + uint256 remaining = periodFinish - block.timestamp; + uint256 leftover = remaining * rewardRate; + rewardRate = (reward + leftover) / rewardsDuration; + } + + // Ensures the provided reward amount is not more than the balance in the contract. + // This keeps the reward rate in the right range, preventing overflows due to + // very high values of `rewardRate` in the earned and `rewardsPerToken` functions; + // Reward + leftover must be less than 2^256 / 10^18 to avoid overflow. + uint256 balance = rewardToken.balanceOf(address(this)); + require(rewardRate <= balance / rewardsDuration, "91"); + + lastUpdateTime = block.timestamp; + periodFinish = block.timestamp + rewardsDuration; // Change the duration + emit RewardAdded(reward); + } + + /// @notice Withdraws ERC20 tokens that could accrue on this contract + /// @param tokenAddress Address of the ERC20 token to withdraw + /// @param to Address to transfer to + /// @param amount Amount to transfer + /// @dev A use case would be to claim tokens if the staked tokens accumulate rewards + function recoverERC20( + address tokenAddress, + address to, + uint256 amount + ) external override onlyRewardsDistribution { + require(tokenAddress != address(stakingToken) && tokenAddress != address(rewardToken), "20"); + + IERC20(tokenAddress).safeTransfer(to, amount); + emit Recovered(tokenAddress, to, amount); + } + + /// @notice Changes the rewards distributor associated to this contract + /// @param _rewardsDistribution Address of the new rewards distributor contract + /// @dev This function was also added by Angle Core Team + /// @dev A compatibility check of the reward token is already performed in the current `RewardsDistributor` implementation + /// which has right to call this function + function setNewRewardsDistribution(address _rewardsDistribution) external override onlyRewardsDistribution { + rewardsDistribution = _rewardsDistribution; + emit RewardsDistributionUpdated(_rewardsDistribution); + } +} diff --git a/contracts/staking/StakingRewardsEvents.sol b/contracts/staking/StakingRewardsEvents.sol new file mode 100644 index 0000000..0841c86 --- /dev/null +++ b/contracts/staking/StakingRewardsEvents.sol @@ -0,0 +1,31 @@ +// SPDX-License-Identifier: GPL-3.0 + +pragma solidity ^0.8.7; + +import "@openzeppelin/contracts/token/ERC20/IERC20.sol"; +import "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol"; +import "@openzeppelin/contracts/utils/math/Math.sol"; +import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; +import "@openzeppelin/contracts/security/ReentrancyGuard.sol"; + +import "../external/AccessControl.sol"; + +import "../interfaces/IRewardsDistributor.sol"; +import "../interfaces/IStakingRewards.sol"; + +/// @title StakingRewardsEvents +/// @author Angle Core Team +/// @notice All the events used in `StakingRewards` contract +contract StakingRewardsEvents { + event RewardAdded(uint256 reward); + + event Staked(address indexed user, uint256 amount); + + event Withdrawn(address indexed user, uint256 amount); + + event RewardPaid(address indexed user, uint256 reward); + + event Recovered(address indexed tokenAddress, address indexed to, uint256 amount); + + event RewardsDistributionUpdated(address indexed _rewardsDistribution); +} diff --git a/contracts/staking/veBoost.vy b/contracts/staking/veBoost.vy new file mode 100644 index 0000000..4b4a1a0 --- /dev/null +++ b/contracts/staking/veBoost.vy @@ -0,0 +1,1000 @@ +# @version 0.2.16 +""" +@title Voting Escrow Delegation +@author Angle Protocol +@license MIT +@dev Provides test functions only available in test mode (`brownie test`) +""" + +# Full fork from: +# Curve Finance's veBoost + +interface ERC721Receiver: + def onERC721Received( + _operator: address, _from: address, _token_id: uint256, _data: Bytes[4096] + ) -> bytes32: + nonpayable + +interface VotingEscrow: + def balanceOf(_account: address) -> int256: view + def locked__end(_addr: address) -> uint256: view + + +event Approval: + _owner: indexed(address) + _approved: indexed(address) + _token_id: indexed(uint256) + +event ApprovalForAll: + _owner: indexed(address) + _operator: indexed(address) + _approved: bool + +event Transfer: + _from: indexed(address) + _to: indexed(address) + _token_id: indexed(uint256) + +event BurnBoost: + _delegator: indexed(address) + _receiver: indexed(address) + _token_id: indexed(uint256) + +event DelegateBoost: + _delegator: indexed(address) + _receiver: indexed(address) + _token_id: indexed(uint256) + _amount: uint256 + _cancel_time: uint256 + _expire_time: uint256 + +event ExtendBoost: + _delegator: indexed(address) + _receiver: indexed(address) + _token_id: indexed(uint256) + _amount: uint256 + _expire_time: uint256 + _cancel_time: uint256 + +event TransferBoost: + _from: indexed(address) + _to: indexed(address) + _token_id: indexed(uint256) + _amount: uint256 + _expire_time: uint256 + +event GreyListUpdated: + _receiver: indexed(address) + _delegator: indexed(address) + _status: bool + + +struct Boost: + # [bias uint128][slope int128] + delegated: uint256 + received: uint256 + # [total active delegations 128][next expiry 128] + expiry_data: uint256 + +struct Token: + # [bias uint128][slope int128] + data: uint256 + # [delegator pos 128][cancel time 128] + dinfo: uint256 + # [global 128][local 128] + position: uint256 + expire_time: uint256 + +struct Point: + bias: int256 + slope: int256 + + +IDENTITY_PRECOMPILE: constant(address) = 0x0000000000000000000000000000000000000004 +MAX_PCT: constant(uint256) = 10_000 +WEEK: constant(uint256) = 86400 * 7 + +voting_escrow: public(address) + +balanceOf: public(HashMap[address, uint256]) +getApproved: public(HashMap[uint256, address]) +isApprovedForAll: public(HashMap[address, HashMap[address, bool]]) +ownerOf: public(HashMap[uint256, address]) + +name: public(String[32]) +symbol: public(String[32]) +base_uri: public(String[128]) + +totalSupply: public(uint256) +# use totalSupply to determine the length +tokenByIndex: public(HashMap[uint256, uint256]) +# use balanceOf to determine the length +tokenOfOwnerByIndex: public(HashMap[address, uint256[MAX_UINT256]]) + +boost: HashMap[address, Boost] +boost_tokens: HashMap[uint256, Token] + +token_of_delegator_by_index: public(HashMap[address, uint256[MAX_UINT256]]) +total_minted: public(HashMap[address, uint256]) +# address => timestamp => # of delegations expiring +account_expiries: public(HashMap[address, HashMap[uint256, uint256]]) + +admin: public(address) # Can and will be a smart contract +future_admin: public(address) + +# The grey list - per-user black and white lists +# users can make this a blacklist or a whitelist - defaults to blacklist +# gray_list[_receiver][_delegator] +# by default is blacklist, with no delegators blacklisted +# if [_receiver][ZERO_ADDRESS] is False = Blacklist, True = Whitelist +# if this is a blacklist, receivers disallow any delegations from _delegator if it is True +# if this is a whitelist, receivers only allow delegations from _delegator if it is True +# Delegation will go through if: not (grey_list[_receiver][ZERO_ADDRESS] ^ grey_list[_receiver][_delegator]) +grey_list: public(HashMap[address, HashMap[address, bool]]) + + +@external +def __init__(_admin: address, _voting_escrow: address, _name: String[32], _symbol: String[32], _base_uri: String[128]): + # These zero checks were added by Angle Core Team + assert _voting_escrow != ZERO_ADDRESS + assert _admin != ZERO_ADDRESS + + self.voting_escrow = _voting_escrow + + self.name = _name + self.symbol = _symbol + self.base_uri = _base_uri + + self.admin = _admin + + +@internal +def _approve(_owner: address, _approved: address, _token_id: uint256): + self.getApproved[_token_id] = _approved + log Approval(_owner, _approved, _token_id) + + +@view +@internal +def _is_approved_or_owner(_spender: address, _token_id: uint256) -> bool: + owner: address = self.ownerOf[_token_id] + return ( + _spender == owner + or _spender == self.getApproved[_token_id] + or self.isApprovedForAll[owner][_spender] + ) + + +@internal +def _update_enumeration_data(_from: address, _to: address, _token_id: uint256): + delegator: address = convert(shift(_token_id, -96), address) + position_data: uint256 = self.boost_tokens[_token_id].position + local_pos: uint256 = position_data % 2 ** 128 + global_pos: uint256 = shift(position_data, -128) + # position in the delegator array of minted tokens + delegator_pos: uint256 = shift(self.boost_tokens[_token_id].dinfo, -128) + + if _from == ZERO_ADDRESS: + # minting - This is called before updates to balance and totalSupply + local_pos = self.balanceOf[_to] + global_pos = self.totalSupply + position_data = shift(global_pos, 128) + local_pos + # this is a new token so we get the index of a new spot + delegator_pos = self.total_minted[delegator] + + self.tokenByIndex[global_pos] = _token_id + self.tokenOfOwnerByIndex[_to][local_pos] = _token_id + self.boost_tokens[_token_id].position = position_data + + # we only mint tokens in the create_boost fn, and this is called + # before we update the cancel_time so we can just set the value + # of dinfo to the shifted position + self.boost_tokens[_token_id].dinfo = shift(delegator_pos, 128) + self.token_of_delegator_by_index[delegator][delegator_pos] = _token_id + self.total_minted[delegator] = delegator_pos + 1 + + elif _to == ZERO_ADDRESS: + # burning - This is called after updates to balance and totalSupply + # we operate on both the global array and local array + last_global_index: uint256 = self.totalSupply + last_local_index: uint256 = self.balanceOf[_from] + last_delegator_pos: uint256 = self.total_minted[delegator] - 1 + + if global_pos != last_global_index: + # swap - set the token we're burnings position to the token in the last index + last_global_token: uint256 = self.tokenByIndex[last_global_index] + last_global_token_pos: uint256 = self.boost_tokens[last_global_token].position + # update the global position of the last global token + self.boost_tokens[last_global_token].position = shift(global_pos, 128) + (last_global_token_pos % 2 ** 128) + self.tokenByIndex[global_pos] = last_global_token + self.tokenByIndex[last_global_index] = 0 + + if local_pos != last_local_index: + # swap - set the token we're burnings position to the token in the last index + last_local_token: uint256 = self.tokenOfOwnerByIndex[_from][last_local_index] + last_local_token_pos: uint256 = self.boost_tokens[last_local_token].position + # update the local position of the last local token + self.boost_tokens[last_local_token].position = shift(last_local_token_pos / 2 ** 128, 128) + local_pos + self.tokenOfOwnerByIndex[_from][local_pos] = last_local_token + self.tokenOfOwnerByIndex[_from][last_local_index] = 0 + self.boost_tokens[_token_id].position = 0 + + if delegator_pos != last_delegator_pos: + last_delegator_token: uint256 = self.token_of_delegator_by_index[delegator][last_delegator_pos] + last_delegator_token_dinfo: uint256 = self.boost_tokens[last_delegator_token].dinfo + # update the last tokens position data and maintain the correct cancel time + self.boost_tokens[last_delegator_token].dinfo = shift(delegator_pos, 128) + (last_delegator_token_dinfo % 2 ** 128) + self.token_of_delegator_by_index[delegator][delegator_pos] = last_delegator_token + self.token_of_delegator_by_index[delegator][last_delegator_pos] = 0 + self.boost_tokens[_token_id].dinfo = 0 # we are burning the token so we can just set to 0 + self.total_minted[delegator] = last_delegator_pos + + else: + # transfering - called between balance updates + from_last_index: uint256 = self.balanceOf[_from] + + if local_pos != from_last_index: + # swap - set the token we're burnings position to the token in the last index + last_local_token: uint256 = self.tokenOfOwnerByIndex[_from][from_last_index] + last_local_token_pos: uint256 = self.boost_tokens[last_local_token].position + # update the local position of the last local token + self.boost_tokens[last_local_token].position = shift(last_local_token_pos / 2 ** 128, 128) + local_pos + self.tokenOfOwnerByIndex[_from][local_pos] = last_local_token + self.tokenOfOwnerByIndex[_from][from_last_index] = 0 + + # to is simple we just add to the end of the list + local_pos = self.balanceOf[_to] + self.tokenOfOwnerByIndex[_to][local_pos] = _token_id + self.boost_tokens[_token_id].position = shift(global_pos, 128) + local_pos + + +@internal +def _burn(_token_id: uint256): + owner: address = self.ownerOf[_token_id] + + self._approve(owner, ZERO_ADDRESS, _token_id) + + self.balanceOf[owner] -= 1 + self.ownerOf[_token_id] = ZERO_ADDRESS + self.totalSupply -= 1 + + self._update_enumeration_data(owner, ZERO_ADDRESS, _token_id) + + log Transfer(owner, ZERO_ADDRESS, _token_id) + + +@internal +def _mint(_to: address, _token_id: uint256): + assert _to != ZERO_ADDRESS # dev: minting to ZERO_ADDRESS disallowed + assert self.ownerOf[_token_id] == ZERO_ADDRESS # dev: token exists + + self._update_enumeration_data(ZERO_ADDRESS, _to, _token_id) + + self.balanceOf[_to] += 1 + self.ownerOf[_token_id] = _to + self.totalSupply += 1 + + log Transfer(ZERO_ADDRESS, _to, _token_id) + + +@internal +def _mint_boost(_token_id: uint256, _delegator: address, _receiver: address, _bias: int256, _slope: int256, _cancel_time: uint256, _expire_time: uint256): + is_whitelist: uint256 = convert(self.grey_list[_receiver][ZERO_ADDRESS], uint256) + delegator_status: uint256 = convert(self.grey_list[_receiver][_delegator], uint256) + assert not convert(bitwise_xor(is_whitelist, delegator_status), bool) # dev: mint boost not allowed + + data: uint256 = shift(convert(_bias, uint256), 128) + convert(abs(_slope), uint256) + self.boost[_delegator].delegated += data + self.boost[_receiver].received += data + + token: Token = self.boost_tokens[_token_id] + token.data = data + token.dinfo = token.dinfo + _cancel_time + token.expire_time = _expire_time + self.boost_tokens[_token_id] = token + + +@internal +def _burn_boost(_token_id: uint256, _delegator: address, _receiver: address, _bias: int256, _slope: int256): + token: Token = self.boost_tokens[_token_id] + expire_time: uint256 = token.expire_time + + if expire_time == 0: + return + + self.boost[_delegator].delegated -= token.data + self.boost[_receiver].received -= token.data + + token.data = 0 + # maintain the same position in the delegator array, but remove the cancel time + token.dinfo = shift(token.dinfo / 2 ** 128, 128) + token.expire_time = 0 + self.boost_tokens[_token_id] = token + + # update the next expiry data + expiry_data: uint256 = self.boost[_delegator].expiry_data + next_expiry: uint256 = expiry_data % 2 ** 128 + active_delegations: uint256 = shift(expiry_data, -128) - 1 + + expiries: uint256 = self.account_expiries[_delegator][expire_time] + + if active_delegations != 0 and expire_time == next_expiry and expiries == 0: + # Will be passed if + # active_delegations == 0, no more active boost tokens + # or + # expire_time != next_expiry, the cancelled boost token isn't the next expiring boost token + # or + # expiries != 0, the cancelled boost token isn't the only one expiring at expire_time + for i in range(1, 513): # ~10 years + # we essentially allow for a boost token be expired for up to 6 years + # 10 yrs - 4 yrs (max vecRV lock time) = ~ 6 yrs + if i == 512: + raise "Failed to find next expiry" + week_ts: uint256 = expire_time + WEEK * (i + 1) + if self.account_expiries[_delegator][week_ts] > 0: + next_expiry = week_ts + break + elif active_delegations == 0: + next_expiry = 0 + + self.boost[_delegator].expiry_data = shift(active_delegations, 128) + next_expiry + self.account_expiries[_delegator][expire_time] = expiries - 1 + + +@internal +def _transfer_boost(_from: address, _to: address, _bias: int256, _slope: int256): + data: uint256 = shift(convert(_bias, uint256), 128) + convert(abs(_slope), uint256) + self.boost[_from].received -= data + self.boost[_to].received += data + + +@pure +@internal +def _deconstruct_bias_slope(_data: uint256) -> Point: + return Point({bias: convert(shift(_data, -128), int256), slope: -convert(_data % 2 ** 128, int256)}) + + +@pure +@internal +def _calc_bias_slope(_x: int256, _y: int256, _expire_time: int256) -> Point: + # SLOPE: (y2 - y1) / (x2 - x1) + # BIAS: y = mx + b -> y - mx = b + slope: int256 = -_y / (_expire_time - _x) + return Point({bias: _y - slope * _x, slope: slope}) + + +@internal +def _transfer(_from: address, _to: address, _token_id: uint256): + assert self.ownerOf[_token_id] == _from # dev: _from is not owner + assert _to != ZERO_ADDRESS # dev: transfers to ZERO_ADDRESS are disallowed + + delegator: address = convert(shift(_token_id, -96), address) + is_whitelist: uint256 = convert(self.grey_list[_to][ZERO_ADDRESS], uint256) + delegator_status: uint256 = convert(self.grey_list[_to][delegator], uint256) + assert not convert(bitwise_xor(is_whitelist, delegator_status), bool) # dev: transfer boost not allowed + + # clear previous token approval + self._approve(_from, ZERO_ADDRESS, _token_id) + + self.balanceOf[_from] -= 1 + self._update_enumeration_data(_from, _to, _token_id) + self.balanceOf[_to] += 1 + self.ownerOf[_token_id] = _to + + tpoint: Point = self._deconstruct_bias_slope(self.boost_tokens[_token_id].data) + tvalue: int256 = tpoint.slope * convert(block.timestamp, int256) + tpoint.bias + + # if the boost value is negative, reset the slope and bias + if tvalue > 0: + self._transfer_boost(_from, _to, tpoint.bias, tpoint.slope) + # y = mx + b -> y - b = mx -> (y - b)/m = x -> -b / m = x (x-intercept) + expiry: uint256 = convert(-tpoint.bias / tpoint.slope, uint256) + log TransferBoost(_from, _to, _token_id, convert(tvalue, uint256), expiry) + else: + self._burn_boost(_token_id, delegator, _from, tpoint.bias, tpoint.slope) + log BurnBoost(delegator, _from, _token_id) + + log Transfer(_from, _to, _token_id) + + +@internal +def _cancel_boost(_token_id: uint256, _caller: address): + receiver: address = self.ownerOf[_token_id] + assert receiver != ZERO_ADDRESS # dev: token does not exist + delegator: address = convert(shift(_token_id, -96), address) + + token: Token = self.boost_tokens[_token_id] + tpoint: Point = self._deconstruct_bias_slope(token.data) + tvalue: int256 = tpoint.slope * convert(block.timestamp, int256) + tpoint.bias + + # if not (the owner or operator or the boost value is negative) + if not (_caller == receiver or self.isApprovedForAll[receiver][_caller] or tvalue <= 0): + if _caller == delegator or self.isApprovedForAll[delegator][_caller]: + # if delegator or operator, wait till after cancel time + assert (token.dinfo % 2 ** 128) <= block.timestamp # dev: must wait for cancel time + else: + # All others are disallowed + raise "Not allowed!" + self._burn_boost(_token_id, delegator, receiver, tpoint.bias, tpoint.slope) + + log BurnBoost(delegator, receiver, _token_id) + + +@internal +def _set_delegation_status(_receiver: address, _delegator: address, _status: bool): + self.grey_list[_receiver][_delegator] = _status + log GreyListUpdated(_receiver, _delegator, _status) + + +@pure +@internal +def _uint_to_string(_value: uint256) -> String[78]: + # NOTE: Odd that this works with a raw_call inside, despite being marked + # a pure function + if _value == 0: + return "0" + + buffer: Bytes[78] = b"" + digits: uint256 = 78 + + for i in range(78): + # go forward to find the # of digits, and set it + # only if we have found the last index + if digits == 78 and _value / 10 ** i == 0: + digits = i + + value: uint256 = ((_value / 10 ** (77 - i)) % 10) + 48 + char: Bytes[1] = slice(convert(value, bytes32), 31, 1) + buffer = raw_call( + IDENTITY_PRECOMPILE, + concat(buffer, char), + max_outsize=78, + is_static_call=True + ) + + return convert(slice(buffer, 78 - digits, digits), String[78]) + + +@external +def approve(_approved: address, _token_id: uint256): + """ + @notice Change or reaffirm the approved address for an NFT. + @dev The zero address indicates there is no approved address. + Throws unless `msg.sender` is the current NFT owner, or an authorized + operator of the current owner. + @param _approved The new approved NFT controller. + @param _token_id The NFT to approve. + """ + owner: address = self.ownerOf[_token_id] + assert ( + msg.sender == owner or self.isApprovedForAll[owner][msg.sender] + ) # dev: must be owner or operator + self._approve(owner, _approved, _token_id) + + +@external +def safeTransferFrom(_from: address, _to: address, _token_id: uint256, _data: Bytes[4096] = b""): + """ + @notice Transfers the ownership of an NFT from one address to another address + @dev Throws unless `msg.sender` is the current owner, an authorized + operator, or the approved address for this NFT. Throws if `_from` is + not the current owner. Throws if `_to` is the zero address. Throws if + `_tokenId` is not a valid NFT. When transfer is complete, this function + checks if `_to` is a smart contract (code size > 0). If so, it calls + `onERC721Received` on `_to` and throws if the return value is not + `bytes4(keccak256("onERC721Received(address,address,uint256,bytes)"))`. + @param _from The current owner of the NFT + @param _to The new owner + @param _token_id The NFT to transfer + @param _data Additional data with no specified format, sent in call to `_to`, max length 4096 + """ + assert self._is_approved_or_owner(msg.sender, _token_id) # dev: neither owner nor approved + self._transfer(_from, _to, _token_id) + + if _to.is_contract: + response: bytes32 = ERC721Receiver(_to).onERC721Received( + msg.sender, _from, _token_id, _data + ) + assert slice(response, 0, 4) == method_id( + "onERC721Received(address,address,uint256,bytes)" + ) # dev: invalid response + + +@external +def setApprovalForAll(_operator: address, _approved: bool): + """ + @notice Enable or disable approval for a third party ("operator") to manage + all of `msg.sender`'s assets. + @dev Emits the ApprovalForAll event. Multiple operators per account are allowed. + @param _operator Address to add to the set of authorized operators. + @param _approved True if the operator is approved, false to revoke approval. + """ + self.isApprovedForAll[msg.sender][_operator] = _approved + log ApprovalForAll(msg.sender, _operator, _approved) + + +@external +def transferFrom(_from: address, _to: address, _token_id: uint256): + """ + @notice Transfer ownership of an NFT -- THE CALLER IS RESPONSIBLE + TO CONFIRM THAT `_to` IS CAPABLE OF RECEIVING NFTS OR ELSE + THEY MAY BE PERMANENTLY LOST + @dev Throws unless `msg.sender` is the current owner, an authorized + operator, or the approved address for this NFT. Throws if `_from` is + not the current owner. Throws if `_to` is the ZERO_ADDRESS. + @param _from The current owner of the NFT + @param _to The new owner + @param _token_id The NFT to transfer + """ + assert self._is_approved_or_owner(msg.sender, _token_id) # dev: neither owner nor approved + self._transfer(_from, _to, _token_id) + + +@view +@external +def tokenURI(_token_id: uint256) -> String[256]: + return concat(self.base_uri, self._uint_to_string(_token_id)) + + +@external +def burn(_token_id: uint256): + """ + @notice Destroy a token + @dev Only callable by the token owner, their operator, or an approved account. + Burning a token with a currently active boost, burns the boost. + @param _token_id The token to burn + """ + assert self._is_approved_or_owner(msg.sender, _token_id) # dev: neither owner nor approved + + tdata: uint256 = self.boost_tokens[_token_id].data + if tdata != 0: + tpoint: Point = self._deconstruct_bias_slope(tdata) + + delegator: address = convert(shift(_token_id, -96), address) + owner: address = self.ownerOf[_token_id] + + self._burn_boost(_token_id, delegator, owner, tpoint.bias, tpoint.slope) + + log BurnBoost(delegator, owner, _token_id) + + self._burn(_token_id) + + +@external +def create_boost( + _delegator: address, + _receiver: address, + _percentage: int256, + _cancel_time: uint256, + _expire_time: uint256, + _id: uint256, +): + """ + @notice Create a boost and delegate it to another account. + @dev Delegated boost can become negative, and requires active management, else + the adjusted veCRV balance of the delegator's account will decrease until reaching 0 + @param _delegator The account to delegate boost from + @param _receiver The account to receive the delegated boost + @param _percentage Since veCRV is a constantly decreasing asset, we use percentage to determine + the amount of delegator's boost to delegate + @param _cancel_time A point in time before _expire_time in which the delegator or their operator + can cancel the delegated boost + @param _expire_time The point in time, atleast a day in the future, at which the value of the boost + will reach 0. After which the negative value is deducted from the delegator's account (and the + receiver's received boost only) until it is cancelled. This value is rounded down to the nearest + WEEK. + @param _id The token id, within the range of [0, 2 ** 96). Useful for contracts given operator status + to have specific ranges. + """ + assert msg.sender == _delegator or self.isApprovedForAll[_delegator][msg.sender] # dev: only delegator or operator + + expire_time: uint256 = (_expire_time / WEEK) * WEEK + + expiry_data: uint256 = self.boost[_delegator].expiry_data + next_expiry: uint256 = expiry_data % 2 ** 128 + + if next_expiry == 0: + next_expiry = MAX_UINT256 + + assert block.timestamp < next_expiry # dev: negative boost token is in circulation + assert _percentage > 0 # dev: percentage must be greater than 0 bps + assert _percentage <= MAX_PCT # dev: percentage must be less than 10_000 bps + assert _cancel_time <= expire_time # dev: cancel time is after expiry + + assert expire_time >= block.timestamp + WEEK # dev: boost duration must be atleast WEEK + assert expire_time <= VotingEscrow(self.voting_escrow).locked__end(_delegator) # dev: boost expiration is past voting escrow lock expiry + assert _id < 2 ** 96 # dev: id out of bounds + + # [delegator address 160][cancel_time uint40][id uint56] + token_id: uint256 = shift(convert(_delegator, uint256), 96) + _id + # check if the token exists here before we expend more gas by minting it + self._mint(_receiver, token_id) + + # delegated slope and bias + point: Point = self._deconstruct_bias_slope(self.boost[_delegator].delegated) + + time: int256 = convert(block.timestamp, int256) + + # delegated boost will be positive, if any of circulating boosts are negative + # we have already reverted + delegated_boost: int256 = point.slope * time + point.bias + y: int256 = _percentage * (VotingEscrow(self.voting_escrow).balanceOf(_delegator) - delegated_boost) / MAX_PCT + assert y > 0 # dev: no boost + + point = self._calc_bias_slope(time, y, convert(expire_time, int256)) + assert point.slope < 0 # dev: invalid slope + + self._mint_boost(token_id, _delegator, _receiver, point.bias, point.slope, _cancel_time, expire_time) + + # increase the number of expiries for the user + if expire_time < next_expiry: + next_expiry = expire_time + + active_delegations: uint256 = shift(expiry_data, -128) + self.account_expiries[_delegator][expire_time] += 1 + self.boost[_delegator].expiry_data = shift(active_delegations + 1, 128) + next_expiry + + log DelegateBoost(_delegator, _receiver, token_id, convert(y, uint256), _cancel_time, _expire_time) + + +@external +def extend_boost(_token_id: uint256, _percentage: int256, _expire_time: uint256, _cancel_time: uint256): + """ + @notice Extend the boost of an existing boost or expired boost + @dev The extension can not decrease the value of the boost. If there are + any outstanding negative value boosts which cause the delegable boost + of an account to be negative this call will revert + @param _token_id The token to extend the boost of + @param _percentage The percentage of delegable boost to delegate + AFTER burning the token's current boost + @param _expire_time The new time at which the boost value will become + 0, and eventually negative. Must be greater than the previous expiry time, + and atleast a WEEK from now, and less than the veCRV lock expiry of the + delegator's account. This value is rounded down to the nearest WEEK. + """ + delegator: address = convert(shift(_token_id, -96), address) + receiver: address = self.ownerOf[_token_id] + + assert msg.sender == delegator or self.isApprovedForAll[delegator][msg.sender] # dev: only delegator or operator + assert receiver != ZERO_ADDRESS # dev: boost token does not exist + assert _percentage > 0 # dev: percentage must be greater than 0 bps + assert _percentage <= MAX_PCT # dev: percentage must be less than 10_000 bps + + # timestamp when delegating account's voting escrow ends - also our second point (lock_expiry, 0) + token: Token = self.boost_tokens[_token_id] + + expire_time: uint256 = (_expire_time / WEEK) * WEEK + + assert _cancel_time <= expire_time # dev: cancel time is after expiry + assert expire_time >= block.timestamp + WEEK # dev: boost duration must be atleast one day + assert expire_time <= VotingEscrow(self.voting_escrow).locked__end(delegator) # dev: boost expiration is past voting escrow lock expiry + + point: Point = self._deconstruct_bias_slope(token.data) + + time: int256 = convert(block.timestamp, int256) + tvalue: int256 = point.slope * time + point.bias + + # Can extend a token by increasing it's amount but not it's expiry time + assert expire_time >= token.expire_time # dev: new expiration must be greater than old token expiry + + # if we are extending an unexpired boost, the cancel time must the same or greater + # else we can adjust the cancel time to our preference + if _cancel_time < (token.dinfo % 2 ** 128): + assert block.timestamp >= token.expire_time # dev: cancel time reduction disallowed + + # storage variables have been updated: next_expiry + active_delegations + self._burn_boost(_token_id, delegator, receiver, point.bias, point.slope) + + expiry_data: uint256 = self.boost[delegator].expiry_data + next_expiry: uint256 = expiry_data % 2 ** 128 + + if next_expiry == 0: + next_expiry = MAX_UINT256 + + assert block.timestamp < next_expiry # dev: negative outstanding boosts + + # delegated slope and bias + point = self._deconstruct_bias_slope(self.boost[delegator].delegated) + + # verify delegated boost isn't negative, else it'll inflate out vecrv balance + delegated_boost: int256 = point.slope * time + point.bias + y: int256 = _percentage * (VotingEscrow(self.voting_escrow).balanceOf(delegator) - delegated_boost) / MAX_PCT + # a delegator can snipe the exact moment a token expires and create a boost + # with 10_000 or some percentage of their boost, which is perfectly fine. + # this check is here so the user can't extend a boost unless they actually + # have any to give + assert y > 0 # dev: no boost + assert y >= tvalue # dev: cannot reduce value of boost + + point = self._calc_bias_slope(time, y, convert(expire_time, int256)) + assert point.slope < 0 # dev: invalid slope + + self._mint_boost(_token_id, delegator, receiver, point.bias, point.slope, _cancel_time, expire_time) + + # increase the number of expiries for the user + if expire_time < next_expiry: + next_expiry = expire_time + + active_delegations: uint256 = shift(expiry_data, -128) + self.account_expiries[delegator][expire_time] += 1 + self.boost[delegator].expiry_data = shift(active_delegations + 1, 128) + next_expiry + + log ExtendBoost(delegator, receiver, _token_id, convert(y, uint256), expire_time, _cancel_time) + + +@external +def cancel_boost(_token_id: uint256): + """ + @notice Cancel an outstanding boost + @dev This does not burn the token, only the boost it represents. The owner + of the token or their operator can cancel a boost at any time. The + delegator or their operator can only cancel a token after the cancel + time. Anyone can cancel the boost if the value of it is negative. + @param _token_id The token to cancel + """ + self._cancel_boost(_token_id, msg.sender) + + +@external +def batch_cancel_boosts(_token_ids: uint256[256]): + """ + @notice Cancel many outstanding boosts + @dev This does not burn the token, only the boost it represents. The owner + of the token or their operator can cancel a boost at any time. The + delegator or their operator can only cancel a token after the cancel + time. Anyone can cancel the boost if the value of it is negative. + @param _token_ids A list of 256 token ids to nullify. The list must + be padded with 0 values if less than 256 token ids are provided. + """ + + for _token_id in _token_ids: + if _token_id == 0: + break + self._cancel_boost(_token_id, msg.sender) + + +@external +def set_delegation_status(_receiver: address, _delegator: address, _status: bool): + """ + @notice Set or reaffirm the blacklist/whitelist status of a delegator for a receiver. + @dev Setting delegator as the ZERO_ADDRESS enables users to deactivate delegations globally + and enable the white list. The ability of a delegator to delegate to a receiver + is determined by ~(grey_list[_receiver][ZERO_ADDRESS] ^ grey_list[_receiver][_delegator]). + @param _receiver The account which we will be updating it's list + @param _delegator The account to disallow/allow delegations from + @param _status Boolean of the status to set the _delegator account to + """ + assert msg.sender == _receiver or self.isApprovedForAll[_receiver][msg.sender] + self._set_delegation_status(_receiver, _delegator, _status) + + +@external +def batch_set_delegation_status(_receiver: address, _delegators: address[256], _status: uint256[256]): + """ + @notice Set or reaffirm the blacklist/whitelist status of multiple delegators for a receiver. + @dev Setting delegator as the ZERO_ADDRESS enables users to deactivate delegations globally + and enable the white list. The ability of a delegator to delegate to a receiver + is determined by ~(grey_list[_receiver][ZERO_ADDRESS] ^ grey_list[_receiver][_delegator]). + @param _receiver The account which we will be updating it's list + @param _delegators List of 256 accounts to disallow/allow delegations from + @param _status List of 256 0s and 1s (booleans) of the status to set the _delegator_i account to. + if the value is not 0 or 1, execution will break, effectively stopping at the index. + + """ + assert msg.sender == _receiver or self.isApprovedForAll[_receiver][msg.sender] # dev: only receiver or operator + + for i in range(256): + if _status[i] > 1: + break + self._set_delegation_status(_receiver, _delegators[i], convert(_status[i], bool)) + + +@view +@external +def adjusted_balance_of(_account: address) -> uint256: + """ + @notice Adjusted veCRV balance after accounting for delegations and boosts + @dev If boosts/delegations have a negative value, they're effective value is 0 + @param _account The account to query the adjusted balance of + """ + next_expiry: uint256 = self.boost[_account].expiry_data % 2 ** 128 + if next_expiry != 0 and next_expiry < block.timestamp: + # if the account has a negative boost in circulation + # we over penalize by setting their adjusted balance to 0 + # this is because we don't want to iterate to find the real + # value + return 0 + + adjusted_balance: int256 = VotingEscrow(self.voting_escrow).balanceOf(_account) + + boost: Boost = self.boost[_account] + time: int256 = convert(block.timestamp, int256) + + if boost.delegated != 0: + dpoint: Point = self._deconstruct_bias_slope(boost.delegated) + + # we take the absolute value, since delegated boost can be negative + # if any outstanding negative boosts are in circulation + # this can inflate the vecrv balance of a user + # taking the absolute value has the effect that it costs + # a user to negatively impact another's vecrv balance + adjusted_balance -= abs(dpoint.slope * time + dpoint.bias) + + if boost.received != 0: + rpoint: Point = self._deconstruct_bias_slope(boost.received) + + # similar to delegated boost, our received boost can be negative + # if any outstanding negative boosts are in our possession + # However, unlike delegated boost, we do not negatively impact + # our adjusted balance due to negative boosts. Instead we take + # whichever is greater between 0 and the value of our received + # boosts. + adjusted_balance += max(rpoint.slope * time + rpoint.bias, empty(int256)) + + # since we took the absolute value of our delegated boost, it now instead of + # becoming negative is positive, and will continue to increase ... + # meaning if we keep a negative outstanding delegated balance for long + # enought it will not only decrease our vecrv_balance but also our received + # boost, however we return the maximum between our adjusted balance and 0 + # when delegating boost, received boost isn't used for determining how + # much we can delegate. + return convert(max(adjusted_balance, empty(int256)), uint256) + + +@view +@external +def delegated_boost(_account: address) -> uint256: + """ + @notice Query the total effective delegated boost value of an account. + @dev This value can be greater than the veCRV balance of + an account if the account has outstanding negative + value boosts. + @param _account The account to query + """ + dpoint: Point = self._deconstruct_bias_slope(self.boost[_account].delegated) + time: int256 = convert(block.timestamp, int256) + return convert(abs(dpoint.slope * time + dpoint.bias), uint256) + + +@view +@external +def received_boost(_account: address) -> uint256: + """ + @notice Query the total effective received boost value of an account + @dev This value can be 0, even with delegations which have a large value, + if the account has any outstanding negative value boosts. + @param _account The account to query + """ + rpoint: Point = self._deconstruct_bias_slope(self.boost[_account].received) + time: int256 = convert(block.timestamp, int256) + return convert(max(rpoint.slope * time + rpoint.bias, empty(int256)), uint256) + + +@view +@external +def token_boost(_token_id: uint256) -> int256: + """ + @notice Query the effective value of a boost + @dev The effective value of a boost is negative after it's expiration + date. + @param _token_id The token id to query + """ + tpoint: Point = self._deconstruct_bias_slope(self.boost_tokens[_token_id].data) + time: int256 = convert(block.timestamp, int256) + return tpoint.slope * time + tpoint.bias + + +@view +@external +def token_expiry(_token_id: uint256) -> uint256: + """ + @notice Query the timestamp of a boost token's expiry + @dev The effective value of a boost is negative after it's expiration + date. + @param _token_id The token id to query + """ + return self.boost_tokens[_token_id].expire_time + + +@view +@external +def token_cancel_time(_token_id: uint256) -> uint256: + """ + @notice Query the timestamp of a boost token's cancel time. This is + the point at which the delegator can nullify the boost. A receiver + can cancel a token at any point. Anyone can nullify a token's boost + after it's expiration. + @param _token_id The token id to query + """ + return self.boost_tokens[_token_id].dinfo % 2 ** 128 + + +@view +@external +def calc_boost_bias_slope( + _delegator: address, + _percentage: int256, + _expire_time: int256, + _extend_token_id: uint256 = 0 +) -> (int256, int256): + """ + @notice Calculate the bias and slope for a boost. + @param _delegator The account to delegate boost from + @param _percentage The percentage of the _delegator's delegable + veCRV to delegate. + @param _expire_time The time at which the boost value of the token + will reach 0, and subsequently become negative + @param _extend_token_id OPTIONAL token id, which if set will first nullify + the boost of the token, before calculating the bias and slope. Useful + for calculating the new bias and slope when extending a token, or + determining the bias and slope of a subsequent token after cancelling + an existing one. Will have no effect if _delegator is not the delegator + of the token. + """ + time: int256 = convert(block.timestamp, int256) + assert _percentage > 0 # dev: percentage must be greater than 0 + assert _percentage <= MAX_PCT # dev: percentage must be less than or equal to 100% + assert _expire_time > time + WEEK # dev: Invalid min expiry time + + lock_expiry: int256 = convert(VotingEscrow(self.voting_escrow).locked__end(_delegator), int256) + assert _expire_time <= lock_expiry + + ddata: uint256 = self.boost[_delegator].delegated + + if _extend_token_id != 0 and convert(shift(_extend_token_id, -96), address) == _delegator: + # decrease the delegated bias and slope by the token's bias and slope + # only if it is the delegator's and it is within the bounds of existence + ddata -= self.boost_tokens[_extend_token_id].data + + dpoint: Point = self._deconstruct_bias_slope(ddata) + + delegated_boost: int256 = dpoint.slope * time + dpoint.bias + assert delegated_boost >= 0 # dev: outstanding negative boosts + + y: int256 = _percentage * (VotingEscrow(self.voting_escrow).balanceOf(_delegator) - delegated_boost) / MAX_PCT + assert y > 0 # dev: no boost + + slope: int256 = -y / (_expire_time - time) + assert slope < 0 # dev: invalid slope + + bias: int256 = y - slope * time + + return bias, slope + + +@pure +@external +def get_token_id(_delegator: address, _id: uint256) -> uint256: + """ + @notice Simple method to get the token id's mintable by a delegator + @param _delegator The address of the delegator + @param _id The id value, must be less than 2 ** 96 + """ + assert _id < 2 ** 96 # dev: invalid _id + return shift(convert(_delegator, uint256), 96) + _id + + +@external +def commit_transfer_ownership(_addr: address): + """ + @notice Transfer ownership of contract to `addr` + @param _addr Address to have ownership transferred to + """ + assert msg.sender == self.admin # dev: admin only + assert _addr != ZERO_ADDRESS # dev: future admin cannot be the 0 address + self.future_admin = _addr + +@external +def accept_transfer_ownership(): + """ + @notice Accept admin role, only callable by future admin + """ + future_admin: address = self.future_admin + assert msg.sender == future_admin + self.admin = future_admin + +@external +def set_base_uri(_base_uri: String[128]): + assert msg.sender == self.admin + self.base_uri = _base_uri + diff --git a/contracts/staking/veBoostProxy.vy b/contracts/staking/veBoostProxy.vy new file mode 100644 index 0000000..ab28a74 --- /dev/null +++ b/contracts/staking/veBoostProxy.vy @@ -0,0 +1,117 @@ +# @version 0.2.16 +""" +@title Voting Escrow Delegation Proxy +@author Angle Protocol +@license MIT +""" + +# Full fork from: +# Curve Finance's veBoostProxy + +from vyper.interfaces import ERC20 + +interface VeDelegation: + def adjusted_balance_of(_account: address) -> uint256: view + + +event CommitAdmin: + admin: address + +event ApplyAdmin: + admin: address + +event DelegationSet: + delegation: address + + +voting_escrow: public(address) + + +delegation: public(address) + +admin: public(address) +future_admin: public(address) + + +@external +def __init__(_voting_escrow: address, _delegation: address, _admin: address): + + assert _voting_escrow != ZERO_ADDRESS + assert _admin != ZERO_ADDRESS + + self.voting_escrow = _voting_escrow + + self.delegation = _delegation + + self.admin = _admin + + log DelegationSet(_delegation) + + +@view +@external +def adjusted_balance_of(_account: address) -> uint256: + """ + @notice Get the adjusted veCRV balance from the active boost delegation contract + @param _account The account to query the adjusted veCRV balance of + @return veCRV balance + """ + _delegation: address = self.delegation + if _delegation == ZERO_ADDRESS: + return ERC20(self.voting_escrow).balanceOf(_account) + return VeDelegation(_delegation).adjusted_balance_of(_account) + + +@external +def kill_delegation(): + """ + @notice Set delegation contract to 0x00, disabling boost delegation + @dev Callable by the emergency admin in case of an issue with the delegation logic + """ + assert msg.sender == self.admin + + self.delegation = ZERO_ADDRESS + log DelegationSet(ZERO_ADDRESS) + + +@external +def set_delegation(_delegation: address): + """ + @notice Set the delegation contract + @dev Only callable by the ownership admin + @param _delegation `VotingEscrowDelegation` deployment address + """ + assert msg.sender == self.admin + + # call `adjusted_balance_of` to make sure it works + VeDelegation(_delegation).adjusted_balance_of(msg.sender) + + self.delegation = _delegation + log DelegationSet(_delegation) + + +@external +def commit_admin(_admin: address): + """ + @notice Set admin to `_admin` + @param _admin Ownership admin + """ + assert msg.sender == self.admin, "Access denied" + + self.future_admin = _admin + + log CommitAdmin(_admin) + + +@external +def accept_transfer_ownership(): + """ + @notice Accept a pending ownership transfer + """ + _admin: address = self.future_admin + assert msg.sender == _admin # dev: future admin only + + self.admin = _admin + + log ApplyAdmin(_admin) + \ No newline at end of file diff --git a/contracts/utils/VyperDeployer.sol b/contracts/utils/VyperDeployer.sol new file mode 100644 index 0000000..0650a23 --- /dev/null +++ b/contracts/utils/VyperDeployer.sol @@ -0,0 +1,115 @@ +// SPDX-License-Identifier: UNLICENSED + +pragma solidity ^0.8.7; + +///@notice This cheat codes interface is named _CheatCodes so you can use the CheatCodes interface in other testing files without errors +interface _CheatCodes { + function ffi(string[] calldata) external returns (bytes memory); +} + +contract VyperDeployer { + address constant HEVM_ADDRESS = address(bytes20(uint160(uint256(keccak256("hevm cheat code"))))); + + /// @notice Initializes cheat codes in order to use ffi to compile Vyper contracts + _CheatCodes cheatCodes = _CheatCodes(HEVM_ADDRESS); + + ///@notice Compiles a Vyper contract and returns the address that the contract was deployeod to + ///@notice If deployment fails, an error will be thrown + ///@param fileName - The file name of the Vyper contract. For example, the file name for "SimpleStore.vy" is "SimpleStore" + ///@return deployedAddress - The address that the contract was deployed to + + function deployContract(string memory fileName) public returns (address) { + ///@notice create a list of strings with the commands necessary to compile Vyper contracts + string[] memory cmds = new string[](2); + cmds[0] = "vyper"; + cmds[1] = string.concat("vyper_contracts/", fileName, ".vy"); + + ///@notice compile the Vyper contract and return the bytecode + bytes memory bytecode = cheatCodes.ffi(cmds); + + ///@notice deploy the bytecode with the create instruction + address deployedAddress; + assembly { + deployedAddress := create(0, add(bytecode, 0x20), mload(bytecode)) + } + + ///@notice check that the deployment was successful + require(deployedAddress != address(0), "VyperDeployer could not deploy contract"); + + ///@notice return the address that the contract was deployed to + return deployedAddress; + } + + ///@notice Compiles a Vyper contract with constructor arguments and returns the address that the contract was deployeod to + ///@notice If deployment fails, an error will be thrown + ///@param fileName - The file name of the Vyper contract. For example, the file name for "SimpleStore.vy" is "SimpleStore" + ///@return deployedAddress - The address that the contract was deployed to + function deployContract(string memory fileName, bytes calldata args) public returns (address) { + ///@notice create a list of strings with the commands necessary to compile Vyper contracts + string[] memory cmds = new string[](2); + cmds[0] = "vyper"; + cmds[1] = string.concat("vyper_contracts/", fileName, ".vy"); + + ///@notice compile the Vyper contract and return the bytecode + bytes memory _bytecode = cheatCodes.ffi(cmds); + + //add args to the deployment bytecode + bytes memory bytecode = abi.encodePacked(_bytecode, args); + + ///@notice deploy the bytecode with the create instruction + address deployedAddress; + assembly { + deployedAddress := create(0, add(bytecode, 0x20), mload(bytecode)) + } + + ///@notice check that the deployment was successful + require(deployedAddress != address(0), "VyperDeployer could not deploy contract"); + + ///@notice return the address that the contract was deployed to + return deployedAddress; + } + + /// @dev Consider listening to the Blueprint if you haven't already + /// @param fileName - The file name of the Blueprint Contract + function deployBlueprint(string memory fileName) public returns (address) { + ///@notice create a list of strings with the commands necessary to compile Vyper contracts + string[] memory cmds = new string[](2); + cmds[0] = "vyper"; + cmds[1] = string.concat("vyper_contracts/", fileName, ".vy"); + + ///@notice compile the Vyper contract and return the bytecode + bytes memory bytecode = cheatCodes.ffi(cmds); + + require(bytecode.length > 0, "Initcodes length must be greater than 0"); + + /// @notice prepend needed items for Blueprint ERC + /// See https://eips.ethereum.org/EIPS/eip-5202 for more details + bytes memory eip_5202_bytecode = bytes.concat( + hex"fe", // EIP_5202_EXECUTION_HALT_BYTE + hex"71", // EIP_5202_BLUEPRINT_IDENTIFIER_BYTE + hex"00", // EIP_5202_VERSION_BYTE + bytecode + ); + + bytes2 len = bytes2(uint16(eip_5202_bytecode.length)); + + /// @notice prepend the deploy preamble + bytes memory deployBytecode = bytes.concat( + hex"61", // DEPLOY_PREAMBLE_INITIAL_BYTE + len, // DEPLOY_PREAMBLE_BYTE_LENGTH + hex"3d81600a3d39f3", // DEPLOY_PREABLE_POST_LENGTH_BYTES + eip_5202_bytecode + ); + + ///@notice check that the deployment was successful + address deployedAddress; + assembly { + deployedAddress := create(0, add(deployBytecode, 0x20), mload(deployBytecode)) + } + + require(deployedAddress != address(0), "VyperDeployer could not deploy contract"); + + ///@notice return the address that the contract was deployed to + return deployedAddress; + } +} diff --git a/foundry.toml b/foundry.toml new file mode 100644 index 0000000..e314232 --- /dev/null +++ b/foundry.toml @@ -0,0 +1,73 @@ +[profile.default] +src = "contracts" +out = "out" +test = "test" +libs = ["lib"] +script = "scripts" +cache_path = "cache" +gas_reports = ["*"] +via_ir = true +sizes = true +optimizer = true +optimizer_runs = 1000 +solc_version = "0.8.7" +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..ea784a1 --- /dev/null +++ b/helpers/fork.sh @@ -0,0 +1,37 @@ +#! /bin/bash + +source lib/utils/helpers/common.sh + +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" + echo "- 11: Linea" + + read option + + uri=$(chain_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/forge-std b/lib/forge-std new file mode 160000 index 0000000..b6a506d --- /dev/null +++ b/lib/forge-std @@ -0,0 +1 @@ +Subproject commit b6a506db2262cad5ff982a87789ee6d1558ec861 diff --git a/lib/openzeppelin-contracts b/lib/openzeppelin-contracts new file mode 160000 index 0000000..8d7a871 --- /dev/null +++ b/lib/openzeppelin-contracts @@ -0,0 +1 @@ +Subproject commit 8d7a871609117b2d95074f5f5e92e4c0506584b7 diff --git a/lib/openzeppelin-contracts-upgradeable b/lib/openzeppelin-contracts-upgradeable new file mode 160000 index 0000000..ebf264c --- /dev/null +++ b/lib/openzeppelin-contracts-upgradeable @@ -0,0 +1 @@ +Subproject commit ebf264cc4b812e6557ed7d539e947211acd5670c diff --git a/lib/utils b/lib/utils new file mode 160000 index 0000000..03a279e --- /dev/null +++ b/lib/utils @@ -0,0 +1 @@ +Subproject commit 03a279efc8569b5b6f1a2925effeb5e966a02f03 diff --git a/package.json b/package.json new file mode 100644 index 0000000..4b559c4 --- /dev/null +++ b/package.json @@ -0,0 +1,42 @@ +{ + "name": "angle-boilerplate", + "version": "1.0.0", + "description": "", + "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 mainnet scripts/DeployMockAgEUR.s.sol", + "deploy:fork": "source .env && forge script --skip test --slow --fork-url fork --broadcast scripts/DeployMockAgEUR.s.sol -vvvv", + "gas": "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": "source .env && forge script --skip test --fork-url fork --broadcast -vvvv", + "test:unit": "forge test -vvv --gas-report --match-path \"test/unit/**/*.sol\"", + "test:invariant": "forge test -vvv --gas-report --match-path \"test/invariant/**/*.sol\"", + "test:fuzz": "forge test -vvv --gas-report --match-path \"test/fuzz/**/*.sol\"", + "test": "FOUNDRY_PROFILE=dev forge test -vvv", + "slither": "slither .", + "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\"" + }, + "keywords": [], + "author": "Angle Core Team", + "license": "GPL-3.0", + "bugs": { + "url": "https://github.com/AngleProtocol/boilerplate/issues" + }, + "devDependencies": { + "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..20ce6b6 --- /dev/null +++ b/remappings.txt @@ -0,0 +1,5 @@ +ds-test/=lib/forge-std/lib/ds-test/src/ +forge-std/=lib/forge-std/src/ +@openzeppelin/contracts/=lib/openzeppelin-contracts/contracts +@openzeppelin/contracts-upgradeable/=lib/openzeppelin-contracts-upgradeable/contracts +utils/=lib/utils/ \ No newline at end of file diff --git a/scripts/BasicScript.s.sol b/scripts/BasicScript.s.sol new file mode 100644 index 0000000..8c1e77d --- /dev/null +++ b/scripts/BasicScript.s.sol @@ -0,0 +1,20 @@ +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.7; + +import "forge-std/Script.sol"; +import { console } from "forge-std/console.sol"; + +contract MyScript is Script { + function test() external { + vm.startBroadcast(); + + address _sender = address(uint160(uint256(keccak256(abi.encodePacked("sender"))))); + address _receiver = address(uint160(uint256(keccak256(abi.encodePacked("receiver"))))); + + // deal(address(token), _sender, 1 ether); + // vm.prank(_sender); + // token.transfer(_receiver, 1 ether); + + vm.stopBroadcast(); + } +} diff --git a/scripts/Simulate.s.sol b/scripts/Simulate.s.sol new file mode 100644 index 0000000..d922378 --- /dev/null +++ b/scripts/Simulate.s.sol @@ -0,0 +1,22 @@ +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.7; + +import "forge-std/Script.sol"; +import { console } from "forge-std/console.sol"; + +contract Simulate is Script { + error WrongCall(); + + function run() external { + // TODO replace with your inputs + address sender = address(0x0274a704a6D9129F90A62dDC6f6024b33EcDad36); + address contractAddress = address(0x3Ef3D8bA38EBe18DB133cEc108f4D14CE00Dd9Ae); + // remove the 0x + bytes + memory data = hex"71ee95c0000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000000c00000000000000000000000000000000000000000000000000000000000000100000000000000000000000000000000000000000000000000000000000000014000000000000000000000000000000000000000000000000000000000000000010000000000000000000000000274a704a6d9129f90a62ddc6f6024b33ecdad3600000000000000000000000000000000000000000000000000000000000000010000000000000000000000001f9840a85d5af5bf1d1762f925bdaddc4201f984000000000000000000000000000000000000000000000000000000000000000100000000000000000000000000000000000000000000000427abf85e5d121ac000000000000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000000000000000000000020000000000000000000000000000000000000000000000000000000000000000811e1782af6373843046497b3be2c5b25f13037b02c218c5a1a5be24666434d169ca949b7c3d8c763a566075f024a19b4565aba390e00f197fff97adb2f9ef8b0eee43dfe473f564bbd71b106e37928d4f052afe2abf8a42e5f07aded57af2766943bc7bedecf531200d4ca454185d135e6933aa3cff5c53a55f45033135d3a01aae71a2ff3d1e34342bdb86d6348a9da5c8384085f743bb2451aa846b5f667690ccb7b7b1fe363d3e286addaaff93de4a258308fae35fb008580bc25873284f63b37592378acc9f27d211017550c57ae97d0e9cc944d3f90bf54147a69fedeed81940d2088ad7b84767ae8569a9202e23aa0f8204a30b365916d5cefb60c1dfe"; + + vm.prank(sender); + (bool success, ) = contractAddress.call(data); + if (!success) revert WrongCall(); + } +} diff --git a/scripts/Utils.s.sol b/scripts/Utils.s.sol new file mode 100644 index 0000000..e377053 --- /dev/null +++ b/scripts/Utils.s.sol @@ -0,0 +1,8 @@ +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.7; + +import "utils/src/CommonUtils.sol"; +import { console } from "forge-std/console.sol"; +import "forge-std/Script.sol"; + +contract Utils is Script, CommonUtils {} diff --git a/slither.config.json b/slither.config.json new file mode 100644 index 0000000..81e2b31 --- /dev/null +++ b/slither.config.json @@ -0,0 +1,10 @@ +{ + "detectors_to_exclude": "naming-convention,solc-version", + "filter_paths": "(lib|test|external|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/test/fuzz/.gitkeep b/test/fuzz/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/test/invariant/.gitkeep b/test/invariant/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/test/unit/.gitkeep b/test/unit/.gitkeep new file mode 100644 index 0000000..e69de29 diff --git a/utils/forwardUtils.js b/utils/forwardUtils.js new file mode 100644 index 0000000..237c7e0 --- /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); +}); \ No newline at end of file diff --git a/yarn.lock b/yarn.lock new file mode 100644 index 0000000..54f44d1 --- /dev/null +++ b/yarn.lock @@ -0,0 +1,505 @@ +# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. +# yarn lockfile v1 + + +"@babel/code-frame@^7.0.0": + version "7.22.13" + resolved "https://registry.yarnpkg.com/@babel/code-frame/-/code-frame-7.22.13.tgz#e3c1c099402598483b7a8c46a721d1038803755e" + integrity sha512-XktuhWlJ5g+3TJXc5upd9Ks1HutSArik6jf2eAjYFyIOf4ej3RN+184cZbzDvbPnuTJIUhPKKJE3cIsYTiAT3w== + dependencies: + "@babel/highlight" "^7.22.13" + chalk "^2.4.2" + +"@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.22.13": + version "7.22.20" + resolved "https://registry.yarnpkg.com/@babel/highlight/-/highlight-7.22.20.tgz#4ca92b71d80554b01427815e06f2df965b9c1f54" + integrity sha512-dkdMCN3py0+ksCgYmGG8jKeGA/8Tk+gJwSYYlFGxG5lmhfKNoAy004YpLxpS1W2J8m/EK2Ew+yOs9pVRwO89mg== + dependencies: + "@babel/helper-validator-identifier" "^7.22.20" + chalk "^2.4.2" + js-tokens "^4.0.0" + +"@solidity-parser/parser@^0.16.0": + version "0.16.1" + resolved "https://registry.yarnpkg.com/@solidity-parser/parser/-/parser-0.16.1.tgz#f7c8a686974e1536da0105466c4db6727311253c" + integrity sha512-PdhRFNhbTtu3x8Axm0uYpqOy/lODYQK+MlYSgqIsq2L8SFYEHJPHNUiOTAJbDGzNjjr1/n9AcIayxafR/fWmYw== + dependencies: + antlr4ts "^0.5.0-alpha.4" + +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== + +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== + +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== + +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" + +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.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== + +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== + +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" + +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== + +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== + +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@^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" + +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== + +ignore@^5.2.4: + version "5.2.4" + resolved "https://registry.yarnpkg.com/ignore/-/ignore-5.2.4.tgz#a291c0c6178ff1b960befe47fcdec301674a6324" + integrity sha512-MAb38BcSbH0eHNBxn7ql2NH/kX33OkB3lZ1BNdh7ENeRChHTYsTvWrMubiIAMNS2llXEEgZ1MUOBtXChP3kaFQ== + +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: + 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== + +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" + +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== + +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.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.21: + version "4.17.21" + resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz#679591c564c3bffaae8454cf0b3df370c3d6911c" + integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg== + +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" + +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" + +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" + +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-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== + +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.1.3" + resolved "https://registry.yarnpkg.com/prettier-plugin-solidity/-/prettier-plugin-solidity-1.1.3.tgz#9a35124f578404caf617634a8cab80862d726cba" + integrity sha512-fQ9yucPi2sBbA2U2Xjh6m4isUTJ7S7QLc/XDDsktqqxYfTwdYKJ0EnnywXHwCGAaYbQNK+HIYPL1OemxuMsgeg== + dependencies: + "@solidity-parser/parser" "^0.16.0" + semver "^7.3.8" + solidity-comments-extractor "^0.0.7" + +prettier@^2.0.0, 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== + +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== + +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== + +semver@^7.3.8, semver@^7.5.2: + version "7.5.4" + resolved "https://registry.yarnpkg.com/semver/-/semver-7.5.4.tgz#483986ec4ed38e1c6c48c34894a9182dbff68a6e" + integrity sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA== + 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.7: + version "0.0.7" + resolved "https://registry.yarnpkg.com/solidity-comments-extractor/-/solidity-comments-extractor-0.0.7.tgz#99d8f1361438f84019795d928b931f4e5c39ca19" + integrity sha512-wciNMLg/Irp8OKGrh3S2tfvZiZ0NEyILfcRCXCD4mp7SgK/i9gzLfhY2hY7VMCQJ3kH9UB9BzNdibIVMchzyYw== + +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" + +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" + +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" + +table@^6.8.1: + version "6.8.1" + resolved "https://registry.yarnpkg.com/table/-/table-6.8.1.tgz#ea2b71359fe03b017a5fbc296204471158080bdf" + integrity sha512-Y4X9zqrCftUhMeH2EptSSERdVKt/nEdijTOacGD/97EKjhQ/Qs8RTlEGABSJNNN8lac9kheH+af7yAkEWlgneA== + 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== + +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" + +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== + +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==