diff --git a/.github/workflows/release-abis.yml b/.github/workflows/release-abis.yml new file mode 100644 index 000000000..9ab9a069b --- /dev/null +++ b/.github/workflows/release-abis.yml @@ -0,0 +1,61 @@ +--- +name: Upload ABIs to a published release + +on: + release: + types: [published] # stable and pre-releases + +permissions: + contents: write + +jobs: + release-abis: + name: Build ABIs and upload to the release + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v3 + with: + persist-credentials: false + + - name: Setup node.js version + uses: actions/setup-node@v3 + with: + node-version: 16 + + - name: Get yarn cache directory path + id: yarn-cache-dir-path + run: echo "dir=$(yarn config get cacheFolder)" >> "$GITHUB_OUTPUT" + + - name: Cache yarn cache + id: cache-yarn-cache + uses: actions/cache@v3 + with: + path: ${{ steps.yarn-cache-dir-path.outputs.dir }} + key: yarn-${{ hashFiles('**/yarn.lock') }} + restore-keys: yarn-${{ hashFiles('**/yarn.lock') }} + + - name: Cache node_modules + id: cache-node-modules + uses: actions/cache@v3 + with: + path: '**/node_modules' + key: node_modules-${{ hashFiles('**/yarn.lock') }} + restore-keys: node_modules-${{ hashFiles('**/yarn.lock') }} + + - name: Install modules + run: yarn + if: | + steps.cache-yarn-cache.outputs.cache-hit != 'true' || + steps.cache-node-modules.outputs.cache-hit != 'true' + + - name: Compile contracts + run: yarn compile + + - name: Make an ABIs archive + run: zip -j abis.zip lib/abi/*.json + + - name: Upload the archive to the release + run: | + gh release view "$GITHUB_REF" || exit; + gh release upload "$GITHUB_REF" abis.zip; diff --git a/.gitignore b/.gitignore index ddfcfd5c4..db51b5b00 100644 --- a/.gitignore +++ b/.gitignore @@ -42,6 +42,8 @@ cli/vendor # e2e temp data /deployed-e2e.json +/deployed-local.json + # OS relative .DS_Store diff --git a/SCRATCH_DEPLOY.md b/SCRATCH_DEPLOY.md deleted file mode 100644 index 1b190ce93..000000000 --- a/SCRATCH_DEPLOY.md +++ /dev/null @@ -1,84 +0,0 @@ -# Deploy Lido protocol from scratch - -**NB**: at the moment the deployment from scratch scripts and manual target Lido V1 version (before Lido V2 upgrade) and is not working - -Video guide: [youtube](https://www.youtube.com/watch?v=dCMXcfglJv0) - -## Requirements - -* shell - bash or zsh -* docker -* node.js v14 -* yarn - -## Environment - -You will need at least: - -* Ethereum node -* IPFS node -* Aragon web client - -In case of local deploy this environment is set up with docker. - -> Note: Lido protocol is based on Aragon framework, so the entire Aragon framework environment is required for deployment. - -## DAO configuration - -Dao config is stored in `deployed-{NETWORK_NAME}.json` file, where `{NETWORK_NAME}` is network name of your choice. See the [`deployed-local-defaults.json`](deployed-local-defaults.json) for basic parameters. Please refer to [`deployed-mainnet.json`](deployed-mainnet.json) for currently deployed Mainnet version of DAO. - -Copy `deployed-local-defaults.json` to `deployed-{NETWORK_NAME}.json` (e.g. `deployed-kintsugi.json`) and update it accordingly . - -## Network configuration - -Add to [`hardhat.config.js`](hardhat.config.js) your network connection parameter (inside the `getNetConfig` function, use `mainnet` or `local` as reference). - -## Deploy process - -> Note: all deploy process is depend of ENS contract. If the target network has one, you can use it. In this case, write it directly to the `deployed-{NETWORK_NAME}.json` file. Otherwise, own ENS contract will be deployed. - -> Note: ETH2 Deposit contract is required. If the target network has one, you must use it. In this case, write it directly to the `deployed-{NETWORK_NAME}.json` file. Otherwise, own Deposit contract will be deployed. - -Steps for deploy: - -* [ ] run environment docker containers -* [ ] set up network config -* [ ] prepare DAO config file -* [ ] deploy Aragon framework environment (including ENS) -* [ ] build and deploy standard Aragon apps (contracts and frontend files) -* [ ] deploy Deposit contract (if necessary) -* [ ] deploy Lido DAO template contract -* [ ] deploy Lido Apps contract implementations -* [ ] register Lido APM name in ENS -* [ ] build Lido Apps frontend files and upload it to IPFS -* [ ] deploy Lido APM contract (via Lido Template) -* [ ] deploy Lido Apps repo contracts (via Lido Template) -* [ ] deploy Lido DAO contract (via Lido Template) -* [ ] issue DAO tokens (via Lido Template) -* [ ] deploy LidoExecutionLayerRewardsVault -* [ ] finalize DAO setup (via Lido Template) -* [ ] deploy CompositePostRebaseBeaconReceiver -* [ ] deploy Burner -* [ ] make final deployed DAO check via script -* [ ] open and check Lido DAO web interface (via Aragon client) - -All steps are automated via shell script [`dao-local-deploy.sh`](dao-local-deploy.sh) for local deploy process. The script be modified for any other network: - -So, one-click local deploy from scratch command is: - -```bash -./dao-local-deploy.sh -``` - -> Note: some steps require manually updating some transaction hashes in the `deployed-{NETWORK_NAME}.json` file. The script will pause the process in this case, please follow the script tips. - -## Aragon dependency issue - -`ipfs-http-client` version has been strictly pinned to `43.0.0` in [this commit](https://github.com/lidofinance/lido-dao/commit/38bf0232fbc59ec6d69d27e170e3e75cfbe1ba11) because `/scripts/multisig/04-publish-app-frontends.js` used to crash at -```javascript -const rootCid = await uploadDirToIpfs({ dirPath: distPath, ipfsApiUrl: ipfsAPI }) -``` - -It appeared that `@aragon/buidler-aragon@npm:^0.2.9` uses `ipfs-http-client`. - -`ipfs-http-client` has a brittle API. Neither `41.0.0` nor `50.0.0` versions of it will work with `@aragon/buidler-aragon@npm:^0.2.9`. diff --git a/artifacts.json b/artifacts.json index 16a279dc7..36c2e847a 100644 --- a/artifacts.json +++ b/artifacts.json @@ -1,4 +1,18 @@ [ + { + "artifactPath": "artifacts/contracts/0.8.9/proxy/OssifiableProxy.sol/OssifiableProxy.json", + "sourcePath": "contracts/0.8.9/proxy/OssifiableProxy.sol", + "name": "OssifiableProxy", + "address": "0x852deD011285fe67063a08005c71a85690503Cee", + "txHash": "0x3def88f27741216b131de2861cf89af2ca2ac4242b384ee33dca8cc70c51c8dd" + }, + { + "artifactPath": "artifacts/contracts/0.8.9/oracle/AccountingOracle.sol/AccountingOracle.json", + "sourcePath": "contracts/0.8.9/oracle/AccountingOracle.sol", + "name": "AccountingOracle", + "address": "0xF3c5E0A67f32CF1dc07a8817590efa102079a1aF", + "txHash": "0x3e27627d3ed236aff8901df187196e9682187dfd0d259c5d5811a6e923436083" + }, { "artifactPath": "artifacts/contracts/0.4.24/Lido.sol/Lido.json", "sourcePath": "contracts/0.4.24/Lido.sol", @@ -6,6 +20,13 @@ "address": "0x17144556fd3424EDC8Fc8A4C940B2D04936d17eb", "txHash": "0xb4b5e02643c9802fd0f7c73c4854c4f1b83497aca13f8297ba67207b71c4dcd9" }, + { + "artifactPath": "artifacts/contracts/0.4.24/nos/NodeOperatorsRegistry.sol/NodeOperatorsRegistry.json", + "sourcePath": "contracts/0.4.24/nos/NodeOperatorsRegistry.sol", + "name": "NodeOperatorsRegistry", + "address": "0x8538930c385C0438A357d2c25CB3eAD95Ab6D8ed", + "txHash": "0xbec5b6ffb8fbc775a183e25cd285367993d7201752e1638a07abec8962bab750" + }, { "artifactPath": "artifacts/contracts/0.4.24/oracle/LegacyOracle.sol/LegacyOracle.json", "sourcePath": "contracts/0.4.24/oracle/LegacyOracle.sol", @@ -14,18 +35,25 @@ "txHash": "0xe666e3ce409bb4c18e1016af0b9ed3495b20361a69f2856bccb9e67599795b6f" }, { - "artifactPath": "artifacts/contracts/0.4.24/nos/NodeOperatorsRegistry.sol/NodeOperatorsRegistry.json", - "sourcePath": "contracts/0.4.24/nos/NodeOperatorsRegistry.sol", - "name": "NodeOperatorsRegistry", - "address": "0x8538930c385C0438A357d2c25CB3eAD95Ab6D8ed", - "txHash": "0xbec5b6ffb8fbc775a183e25cd285367993d7201752e1638a07abec8962bab750" + "artifactPath": "artifacts/contracts/0.8.9/Burner.sol/Burner.json", + "sourcePath": "contracts/0.8.9/Burner.sol", + "name": "Burner", + "address": "0xD15a672319Cf0352560eE76d9e89eAB0889046D3", + "txHash": "0xbebf5c85404a0d8e36b859046c984fdf6dd764b5d317feb7eb3525016005b1d9" }, { - "artifactPath": "artifacts/contracts/0.6.12/WstETH.sol/WstETH.json", - "sourcePath": "contracts/0.6.12/WstETH.sol", - "name": "WstETH", - "address": "0x7f39C581F595B53c5cb19bD0b3f8dA6c935E2Ca0", - "txHash": "0xaf2c1a501d2b290ef1e84ddcfc7beb3406f8ece2c46dee14e212e8233654ff05" + "artifactPath": "artifacts/contracts/0.8.9/DepositSecurityModule.sol/DepositSecurityModule.json", + "sourcePath": "contracts/0.8.9/DepositSecurityModule.sol", + "name": "DepositSecurityModule", + "address": "0xC77F8768774E1c9244BEed705C4354f2113CFc09", + "txHash": "0x21307a2321f167f99de11ccec86d7bdd8233481bbffa493e15c519ca8d662c4f" + }, + { + "artifactPath": "artifacts/contracts/0.8.9/EIP712StETH.sol/EIP712StETH.json", + "sourcePath": "contracts/0.8.9/EIP712StETH.sol", + "name": "EIP712StETH", + "address": "0x8F73e4C2A6D852bb4ab2A45E6a9CF5715b3228B7", + "txHash": "0xecb5010620fb13b0e2bbc98b8a0c82de0d7385491452cd36cf303cd74216ed91" }, { "artifactPath": "artifacts/contracts/0.8.9/LidoExecutionLayerRewardsVault.sol/LidoExecutionLayerRewardsVault.json", @@ -34,13 +62,6 @@ "address": "0x388C818CA8B9251b393131C08a736A67ccB19297", "txHash": "0xd72cf25e4a5fe3677b6f9b2ae13771e02ad66f8d2419f333bb8bde3147bd4294" }, - { - "artifactPath": "artifacts/contracts/0.8.9/Burner.sol/Burner.json", - "sourcePath": "contracts/0.8.9/Burner.sol", - "name": "Burner", - "address": "0xD15a672319Cf0352560eE76d9e89eAB0889046D3", - "txHash": "0xbebf5c85404a0d8e36b859046c984fdf6dd764b5d317feb7eb3525016005b1d9" - }, { "artifactPath": "artifacts/contracts/0.8.9/oracle/HashConsensus.sol/HashConsensus.json", "sourcePath": "contracts/0.8.9/oracle/HashConsensus.sol", @@ -48,20 +69,6 @@ "address": "0xD624B08C83bAECF0807Dd2c6880C3154a5F0B288", "txHash": "0xd74dcca9bacede9f332d70562f49808254061853937ffbbfc7397ab5d017041a" }, - { - "artifactPath": "artifacts/contracts/0.8.9/proxy/OssifiableProxy.sol/OssifiableProxy.json", - "sourcePath": "contracts/0.8.9/proxy/OssifiableProxy.sol", - "name": "OssifiableProxy", - "address": "0x852deD011285fe67063a08005c71a85690503Cee", - "txHash": "0x3def88f27741216b131de2861cf89af2ca2ac4242b384ee33dca8cc70c51c8dd" - }, - { - "artifactPath": "artifacts/contracts/0.8.9/oracle/AccountingOracle.sol/AccountingOracle.json", - "sourcePath": "contracts/0.8.9/oracle/AccountingOracle.sol", - "name": "AccountingOracle", - "address": "0xF3c5E0A67f32CF1dc07a8817590efa102079a1aF", - "txHash": "0x3e27627d3ed236aff8901df187196e9682187dfd0d259c5d5811a6e923436083" - }, { "artifactPath": "artifacts/contracts/0.8.9/oracle/HashConsensus.sol/HashConsensus.json", "sourcePath": "contracts/0.8.9/oracle/HashConsensus.sol", @@ -73,22 +80,29 @@ "artifactPath": "artifacts/contracts/0.8.9/proxy/OssifiableProxy.sol/OssifiableProxy.json", "sourcePath": "contracts/0.8.9/proxy/OssifiableProxy.sol", "name": "OssifiableProxy", - "address": "0x0De4Ea0184c2ad0BacA7183356Aea5B8d5Bf5c6e", - "txHash": "0xef3eea1523d2161c2f36ba61e327e3520231614c055b8a88c7f5928d18e423ea" + "address": "0xC1d0b3DE6792Bf6b4b37EccdcC24e45978Cfd2Eb", + "txHash": "0x3a2910624533935cc8c21837b1705bcb159a760796930097016186be705cc455" }, { - "artifactPath": "artifacts/contracts/0.8.9/oracle/ValidatorsExitBusOracle.sol/ValidatorsExitBusOracle.json", - "sourcePath": "contracts/0.8.9/oracle/ValidatorsExitBusOracle.sol", - "name": "ValidatorsExitBusOracle", - "address": "0xA89Ea51FddE660f67d1850e03C9c9862d33Bc42c", - "txHash": "0x5ab545276f78a72a432c3e971c96384973abfab6394e08cb077a006c25aef7a7" + "artifactPath": "artifacts/contracts/0.8.9/LidoLocator.sol/LidoLocator.json", + "sourcePath": "contracts/0.8.9/LidoLocator.sol", + "name": "LidoLocator", + "address": "0x1D920cc5bACf7eE506a271a5259f2417CaDeCE1d", + "txHash": "0xf90012ef0a40e47c909ab3a5b3503ecee78f6a9be134d1349a742e500d37ae33" }, { - "artifactPath": "artifacts/contracts/0.8.9/DepositSecurityModule.sol/DepositSecurityModule.json", - "sourcePath": "contracts/0.8.9/DepositSecurityModule.sol", - "name": "DepositSecurityModule", - "address": "0xC77F8768774E1c9244BEed705C4354f2113CFc09", - "txHash": "0x21307a2321f167f99de11ccec86d7bdd8233481bbffa493e15c519ca8d662c4f" + "artifactPath": "artifacts/contracts/0.4.24/template/LidoTemplate.sol/LidoTemplate.json", + "sourcePath": "contracts/0.4.24/template/LidoTemplate.sol", + "name": "LidoTemplate", + "address": "0x752350797CB92Ad3BF1295Faf904B27585e66BF5", + "txHash": "0xdcd4ebe028aa3663a1fe8bbc92ae8489045e29d2a6ef5284083d9be5c3fa5f19" + }, + { + "artifactPath": "artifacts/contracts/0.8.9/OracleDaemonConfig.sol/OracleDaemonConfig.json", + "sourcePath": "contracts/0.8.9/OracleDaemonConfig.sol", + "name": "OracleDaemonConfig", + "address": "0xbf05A929c3D7885a6aeAd833a992dA6E5ac23b09", + "txHash": "0xa4f380b8806f5a504ef67fce62989e09be5a48bf114af63483c01c22f0c9a36f" }, { "artifactPath": "artifacts/contracts/0.8.9/sanity_checks/OracleReportSanityChecker.sol/OracleReportSanityChecker.json", @@ -98,11 +112,32 @@ "txHash": "0x700c83996ad7deefda286044280ad86108dfef9c880909bd8e75a3746f7d631c" }, { - "artifactPath": "artifacts/contracts/0.8.9/OracleDaemonConfig.sol/OracleDaemonConfig.json", - "sourcePath": "contracts/0.8.9/OracleDaemonConfig.sol", - "name": "OracleDaemonConfig", - "address": "0xbf05A929c3D7885a6aeAd833a992dA6E5ac23b09", - "txHash": "0xa4f380b8806f5a504ef67fce62989e09be5a48bf114af63483c01c22f0c9a36f" + "artifactPath": "artifacts/contracts/0.8.9/proxy/OssifiableProxy.sol/OssifiableProxy.json", + "sourcePath": "contracts/0.8.9/proxy/OssifiableProxy.sol", + "name": "OssifiableProxy", + "address": "0xFdDf38947aFB03C621C71b06C9C70bce73f12999", + "txHash": "0xb8620f04a8db6bb52cfd0978c6677a5f16011e03d4622e5d660ea6ba34c2b122" + }, + { + "artifactPath": "artifacts/contracts/0.8.9/StakingRouter.sol/StakingRouter.json", + "sourcePath": "contracts/0.8.9/StakingRouter.sol", + "name": "StakingRouter", + "address": "0xD8784e748f59Ba711fB5643191Ec3fAdD50Fb6df", + "txHash": "0xd6d489f22203c835da6027ff0e532a01a08f36f0fda6c7c0a42e471ae3b3c461" + }, + { + "artifactPath": "artifacts/contracts/0.8.9/proxy/OssifiableProxy.sol/OssifiableProxy.json", + "sourcePath": "contracts/0.8.9/proxy/OssifiableProxy.sol", + "name": "OssifiableProxy", + "address": "0x0De4Ea0184c2ad0BacA7183356Aea5B8d5Bf5c6e", + "txHash": "0xef3eea1523d2161c2f36ba61e327e3520231614c055b8a88c7f5928d18e423ea" + }, + { + "artifactPath": "artifacts/contracts/0.8.9/oracle/ValidatorsExitBusOracle.sol/ValidatorsExitBusOracle.json", + "sourcePath": "contracts/0.8.9/oracle/ValidatorsExitBusOracle.sol", + "name": "ValidatorsExitBusOracle", + "address": "0xA89Ea51FddE660f67d1850e03C9c9862d33Bc42c", + "txHash": "0x5ab545276f78a72a432c3e971c96384973abfab6394e08cb077a006c25aef7a7" }, { "artifactPath": "artifacts/contracts/0.8.9/proxy/OssifiableProxy.sol/OssifiableProxy.json", @@ -126,39 +161,11 @@ "txHash": "0xd9eb2eca684770e4d2b192709b6071875f75072a0ce794a582824ee907a704f3" }, { - "artifactPath": "artifacts/contracts/0.8.9/EIP712StETH.sol/EIP712StETH.json", - "sourcePath": "contracts/0.8.9/EIP712StETH.sol", - "name": "EIP712StETH", - "address": "0x8F73e4C2A6D852bb4ab2A45E6a9CF5715b3228B7", - "txHash": "0xecb5010620fb13b0e2bbc98b8a0c82de0d7385491452cd36cf303cd74216ed91" - }, - { - "artifactPath": "artifacts/contracts/0.8.9/proxy/OssifiableProxy.sol/OssifiableProxy.json", - "sourcePath": "contracts/0.8.9/proxy/OssifiableProxy.sol", - "name": "OssifiableProxy", - "address": "0xFdDf38947aFB03C621C71b06C9C70bce73f12999", - "txHash": "0xb8620f04a8db6bb52cfd0978c6677a5f16011e03d4622e5d660ea6ba34c2b122" - }, - { - "artifactPath": "artifacts/contracts/0.8.9/StakingRouter.sol/StakingRouter.json", - "sourcePath": "contracts/0.8.9/StakingRouter.sol", - "name": "StakingRouter", - "address": "0xD8784e748f59Ba711fB5643191Ec3fAdD50Fb6df", - "txHash": "0xd6d489f22203c835da6027ff0e532a01a08f36f0fda6c7c0a42e471ae3b3c461" - }, - { - "artifactPath": "artifacts/contracts/0.8.9/proxy/OssifiableProxy.sol/OssifiableProxy.json", - "sourcePath": "contracts/0.8.9/proxy/OssifiableProxy.sol", - "name": "OssifiableProxy", - "address": "0xC1d0b3DE6792Bf6b4b37EccdcC24e45978Cfd2Eb", - "txHash": "0x3a2910624533935cc8c21837b1705bcb159a760796930097016186be705cc455" - }, - { - "artifactPath": "artifacts/contracts/0.8.9/LidoLocator.sol/LidoLocator.json", - "sourcePath": "contracts/0.8.9/LidoLocator.sol", - "name": "LidoLocator", - "address": "0x1D920cc5bACf7eE506a271a5259f2417CaDeCE1d", - "txHash": "0xf90012ef0a40e47c909ab3a5b3503ecee78f6a9be134d1349a742e500d37ae33" + "artifactPath": "artifacts/contracts/0.6.12/WstETH.sol/WstETH.json", + "sourcePath": "contracts/0.6.12/WstETH.sol", + "name": "WstETH", + "address": "0x7f39C581F595B53c5cb19bD0b3f8dA6c935E2Ca0", + "txHash": "0xaf2c1a501d2b290ef1e84ddcfc7beb3406f8ece2c46dee14e212e8233654ff05" }, { "artifactPath": "artifacts/contracts/0.6.11/deposit_contract.sol/DepositContract.json", diff --git a/cli/Makefile b/cli/Makefile deleted file mode 100644 index 118a16290..000000000 --- a/cli/Makefile +++ /dev/null @@ -1,37 +0,0 @@ -APP_NAME := lido-aragon -SUPPORTED_PLATFORMS := linux darwin -SUPPORTED_ARCHS := amd64 arm64 -VERSION := $(shell git describe --tags --abbrev=0) -REVISION := $(shell git rev-parse --short HEAD) - -UNAME_M := $(shell uname -m) -ARCH := $(UNAME_M) -ifeq ($(UNAME_M),x86_64) - ARCH=amd64 -endif -ifneq ($(filter %86,$(UNAME_M)),) - ARCH=386 -endif -ifneq ($(filter arm%,$(UNAME_M)),) - ARCH=arm -endif -ifneq ($(filter $(UNAME_M),arm64 aarch64 armv8b armv8l),) - ARCH=arm64 -endif - -compile-releases: clean vendor## Compile vsh binaries for multiple platforms and architectures strictly using vendor directory - mkdir -p ./build/ - for GOOS in $(SUPPORTED_PLATFORMS); do \ - for GOARCH in $(SUPPORTED_ARCHS); do \ - GOOS=$$GOOS GOARCH=$$GOARCH \ - go build -mod vendor -ldflags "-X lido-aragon/cmd.version=$(VERSION) -X lido-aragon/cmd.revision=$(REVISION)" -o build/${APP_NAME}_$${GOOS}_$${GOARCH}; \ - done \ - done - cd build/ && sha256sum * > SHA256SUM - -clean: ## Remove builds and vsh related docker containers - rm ./build/* || true - -vendor: ## synch dependencies in vendor/ directory - go mod tidy - go mod vendor \ No newline at end of file diff --git a/cli/build/SHA256SUM b/cli/build/SHA256SUM deleted file mode 100644 index 67dfb3287..000000000 --- a/cli/build/SHA256SUM +++ /dev/null @@ -1,4 +0,0 @@ -55e09746b1a90e033f643a0447be47a0c73f9b4ec2384b0e4d1225b4dc72f110 lido-aragon_darwin_amd64 -572a43662b74bdf5c8e0e0cfc8ab6dd9b117307e9e06985867392048f9705855 lido-aragon_darwin_arm64 -e3cb48e0bef318eb51dc2e5e43895437792d5a9a72cffa11e6c6abb8cb585682 lido-aragon_linux_amd64 -5be4c46a259e7d053cc73614c9d838a1b83419a13c0b97b7542947292da8aefb lido-aragon_linux_arm64 diff --git a/cli/build/lido-aragon_darwin_amd64 b/cli/build/lido-aragon_darwin_amd64 deleted file mode 100755 index af60a5ba5..000000000 Binary files a/cli/build/lido-aragon_darwin_amd64 and /dev/null differ diff --git a/cli/build/lido-aragon_darwin_arm64 b/cli/build/lido-aragon_darwin_arm64 deleted file mode 100755 index 3d4149941..000000000 Binary files a/cli/build/lido-aragon_darwin_arm64 and /dev/null differ diff --git a/cli/build/lido-aragon_linux_amd64 b/cli/build/lido-aragon_linux_amd64 deleted file mode 100755 index ae6733619..000000000 Binary files a/cli/build/lido-aragon_linux_amd64 and /dev/null differ diff --git a/cli/build/lido-aragon_linux_arm64 b/cli/build/lido-aragon_linux_arm64 deleted file mode 100755 index 6822a4333..000000000 Binary files a/cli/build/lido-aragon_linux_arm64 and /dev/null differ diff --git a/cli/cmd/node.go b/cli/cmd/node.go deleted file mode 100644 index a409f893e..000000000 --- a/cli/cmd/node.go +++ /dev/null @@ -1,30 +0,0 @@ -package cmd - -import ( - "lido-aragon/pkg/daemon" - "os" - - "github.com/pterm/pterm" - "github.com/spf13/cobra" -) - -var startNode = &cobra.Command{ - Use: "node", - Short: "Start hardhat node only", - Run: func(cmd *cobra.Command, args []string) { - os.Setenv("NETWORK_NAME", Lido.NetworkName) - pterm.Info.Printf("Network: %s\n", Lido.NetworkName) - - fork, err := cmd.Flags().GetString("fork") - if err != nil { - return - } - - if fork == "" { - pterm.Println("\nYou can set --fork url") - } - - Lido.HardhatNode.Start(fork) - daemon.WaitCtrlC() - }, -} diff --git a/cli/cmd/root.go b/cli/cmd/root.go deleted file mode 100644 index 3a66976cd..000000000 --- a/cli/cmd/root.go +++ /dev/null @@ -1,71 +0,0 @@ -package cmd - -import ( - "fmt" - "lido-aragon/pkg/apps" - "lido-aragon/pkg/aragon" - "lido-aragon/pkg/contracts" - "lido-aragon/pkg/deploy" - "lido-aragon/pkg/hardhat" - "lido-aragon/pkg/ipfs" - "lido-aragon/pkg/logs" - - "github.com/spf13/cobra" -) - -var ( - version string - revision string -) - -type LidoExecutor struct { - HardhatNode *hardhat.HardhatNode - IPFS *ipfs.IPFS - Contracts *contracts.Contracts - Deploy *deploy.Deploy - AragonClient *aragon.AragonClient - LidoApps *apps.LidoAppsClient - - NetworkName string -} - -var ( - Lido = &LidoExecutor{ - HardhatNode: &hardhat.HardhatNode{}, - IPFS: &ipfs.IPFS{}, - Contracts: &contracts.Contracts{}, - Deploy: &deploy.Deploy{}, - AragonClient: &aragon.AragonClient{}, - LidoApps: &apps.LidoAppsClient{}, - } -) - -func (ld *LidoExecutor) Shutdown() { - ld.HardhatNode.Stop() - ld.IPFS.Stop() - ld.AragonClient.Stop() - ld.LidoApps.Stop() -} - -var ( - rootCmd = &cobra.Command{ - Use: "lido-aragon", - Short: "lido-aragon: lido tool to start aragon env", - } -) - -// Execute executes the root command. -func Execute() error { - rootCmd.CompletionOptions.DisableDefaultCmd = true - rootCmd.Version = fmt.Sprintf("%s (rev-%s)\n", version, revision) - rootCmd.SetHelpCommand(&cobra.Command{Hidden: true}) - - rootCmd.PersistentFlags().StringVar(&Lido.NetworkName, "network", "localhost", "Set deploy network name") - rootCmd.PersistentFlags().BoolVarP(&logs.Verbose, "verbose", "v", false, "Verbose output all of services") - rootCmd.PersistentFlags().StringVar(&Lido.LidoApps.AppsLocator, "apps", "", "Which source to load app frontend assets from") - rootCmd.PersistentFlags().StringVar(&Lido.LidoApps.Path, "apps-path", "", "Lido apps path") - rootCmd.PersistentFlags().StringVar(&Lido.HardhatNode.Fork, "fork", "", "Fork endpoint https://mainnet.infura.io/v3/{WEB3_INFURA_PROJECT_ID}") - rootCmd.PersistentFlags().IntVar(&Lido.IPFS.Timeout, "ipfs-timeout", 30, "IPFS daemon timeout") - - return rootCmd.Execute() -} diff --git a/cli/cmd/start.go b/cli/cmd/start.go deleted file mode 100644 index 420d1ed53..000000000 --- a/cli/cmd/start.go +++ /dev/null @@ -1,174 +0,0 @@ -package cmd - -import ( - "encoding/json" - "fmt" - "io/ioutil" - "lido-aragon/pkg/aragon" - "lido-aragon/pkg/daemon" - "lido-aragon/pkg/deploy" - "log" - "os" - - "github.com/pterm/pterm" - "github.com/spf13/cobra" - "github.com/ttacon/chalk" -) - -func init() { - rootCmd.AddCommand(startCmd) - rootCmd.AddCommand(startAragon) - rootCmd.AddCommand(startNode) - - startCmd.AddCommand(startAllCmd) - startCmd.AddCommand(startForkCmd) -} - -var startAragon = &cobra.Command{ - Use: "aragon", - Short: "Start aragon client only", - Run: func(cmd *cobra.Command, args []string) { - os.Setenv("NETWORK_NAME", Lido.NetworkName) - - var aragonCmd aragon.AragonNetwork - - if Lido.NetworkName == "mainnet" { - aragonCmd = aragon.CMD_MAINNET - } - - pterm.Info.Printf("Network: %s\n", Lido.NetworkName) - - Lido.Deploy.DeployedFile, _, _ = getDeployedFile(Lido.NetworkName) - - err := Lido.AragonClient.Start(aragonCmd, Lido.LidoApps.AppsLocator, Lido.Deploy.DeployedFile) - if err != nil { - return - } - - pterm.Println() - pterm.FgWhite.Println("ARAGON_ENS_REGISTRY_ADDRESS = " + Lido.AragonClient.EnsRegistry) - pterm.FgWhite.Println("Start aragon at: ", chalk.Yellow, Lido.AragonClient.RunningUrl+"/#/"+Lido.Deploy.DeployedFile.DaoAddress) - - daemon.WaitCtrlC() - }, -} - -var startCmd = &cobra.Command{ - Use: "start", - Short: "Start local or form env", -} - -func removeDeployedFile(deployedPath string) { - if deployedPath == "" { - return - } - - if Lido.NetworkName == "mainnet" || Lido.NetworkName == "goerli" || Lido.NetworkName == "rinkeby" || Lido.NetworkName == "mainnet-test" { - fmt.Printf("Can't remove deployed file on network %s\n", Lido.NetworkName) - } - - e := os.Remove(deployedPath) - if e != nil { - log.Panic(e) - } - - fmt.Printf("removed %s\n", deployedPath) -} - -var startAllCmd = &cobra.Command{ - Use: "all", - Short: "Start hardhat node, start ipfs, deploy contracts, start lido apps, start aragon", - Run: func(cmd *cobra.Command, args []string) { - - os.Setenv("NETWORK_NAME", Lido.NetworkName) - - var deployedPath string - Lido.Deploy.DeployedFile, deployedPath, _ = getDeployedFile(Lido.NetworkName) - - removeDeployedFile(deployedPath) - - Lido.Contracts.Start() - Lido.HardhatNode.Start("") - Lido.IPFS.Start() - if err := Lido.Deploy.Start(); err != nil { - log.Println(err) - return - } - if err := Lido.LidoApps.Start(); err != nil { - log.Println(err) - return - } - if err := Lido.AragonClient.Start("", Lido.LidoApps.AppsLocator, nil); err != nil { - log.Println(err) - return - } - - if Lido.AragonClient.RunningUrl != "" && Lido.Deploy.DeployedFile.DaoAddress != "" { - pterm.Println() - pterm.FgYellow.Println("Start aragon at: " + Lido.AragonClient.RunningUrl + "/#/" + Lido.Deploy.DeployedFile.DaoAddress) - } - - daemon.WaitCtrlC() - }, -} - -var startForkCmd = &cobra.Command{ - Use: "fork", - Short: "Deploy API artifacts", - Run: func(cmd *cobra.Command, args []string) { - - os.Setenv("NETWORK_NAME", Lido.NetworkName) - - fork, err := cmd.Flags().GetString("fork") - if err != nil { - return - } - - if fork == "" { - pterm.Error.Println("Please set --fork url") - return - } - - Lido.HardhatNode.Start(fork) - - Lido.Deploy.DeployedFile, _, _ = getDeployedFile(Lido.NetworkName) - - err = Lido.AragonClient.Start(aragon.CMD_MAINNET, Lido.LidoApps.AppsLocator, Lido.Deploy.DeployedFile) - if err != nil { - return - } - - pterm.Println() - pterm.FgWhite.Println("Start aragon at: " + Lido.AragonClient.RunningUrl + "/#/" + Lido.Deploy.DeployedFile.DaoAddress) - - daemon.WaitCtrlC() - }, -} - -func getDeployedFile(networkName string) (*deploy.DeployedFile, string, error) { - - paths := []string{"./", "../"} - for _, path := range paths { - deployedPath := path + fmt.Sprintf("deployed-%s.json", networkName) - - _, err := os.Stat(deployedPath) - if err != nil { - continue - } - - jsonFile, err := os.Open(deployedPath) - if err != nil { - return nil, "", err - } - jsonResult, _ := ioutil.ReadAll(jsonFile) - - defer jsonFile.Close() - - var deployedFile deploy.DeployedFile - json.Unmarshal(jsonResult, &deployedFile) - - return &deployedFile, deployedPath, nil - } - - return nil, "", nil -} diff --git a/cli/go.mod b/cli/go.mod deleted file mode 100644 index 161072e71..000000000 --- a/cli/go.mod +++ /dev/null @@ -1,11 +0,0 @@ -module lido-aragon - -go 1.16 - -require ( - github.com/gookit/color v1.5.0 // indirect - github.com/pterm/pterm v0.12.33 - github.com/spf13/cobra v1.2.1 - github.com/ttacon/chalk v0.0.0-20160626202418-22c06c80ed31 - golang.org/x/sys v0.1.0 // indirect -) diff --git a/cli/go.sum b/cli/go.sum deleted file mode 100644 index e5eaa66c0..000000000 --- a/cli/go.sum +++ /dev/null @@ -1,602 +0,0 @@ -cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= -cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= -cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU= -cloud.google.com/go v0.44.1/go.mod h1:iSa0KzasP4Uvy3f1mN/7PiObzGgflwredwwASm/v6AU= -cloud.google.com/go v0.44.2/go.mod h1:60680Gw3Yr4ikxnPRS/oxxkBccT6SA1yMk63TGekxKY= -cloud.google.com/go v0.45.1/go.mod h1:RpBamKRgapWJb87xiFSdk4g1CME7QZg3uwTez+TSTjc= -cloud.google.com/go v0.46.3/go.mod h1:a6bKKbmY7er1mI7TEI4lsAkts/mkhTSZK8w33B4RAg0= -cloud.google.com/go v0.50.0/go.mod h1:r9sluTvynVuxRIOHXQEHMFffphuXHOMZMycpNR5e6To= -cloud.google.com/go v0.52.0/go.mod h1:pXajvRH/6o3+F9jDHZWQ5PbGhn+o8w9qiu/CffaVdO4= -cloud.google.com/go v0.53.0/go.mod h1:fp/UouUEsRkN6ryDKNW/Upv/JBKnv6WDthjR6+vze6M= -cloud.google.com/go v0.54.0/go.mod h1:1rq2OEkV3YMf6n/9ZvGWI3GWw0VoqH/1x2nd8Is/bPc= -cloud.google.com/go v0.56.0/go.mod h1:jr7tqZxxKOVYizybht9+26Z/gUq7tiRzu+ACVAMbKVk= -cloud.google.com/go v0.57.0/go.mod h1:oXiQ6Rzq3RAkkY7N6t3TcE6jE+CIBBbA36lwQ1JyzZs= -cloud.google.com/go v0.62.0/go.mod h1:jmCYTdRCQuc1PHIIJ/maLInMho30T/Y0M4hTdTShOYc= -cloud.google.com/go v0.65.0/go.mod h1:O5N8zS7uWy9vkA9vayVHs65eM1ubvY4h553ofrNHObY= -cloud.google.com/go v0.72.0/go.mod h1:M+5Vjvlc2wnp6tjzE102Dw08nGShTscUx2nZMufOKPI= -cloud.google.com/go v0.74.0/go.mod h1:VV1xSbzvo+9QJOxLDaJfTjx5e+MePCpCWwvftOeQmWk= -cloud.google.com/go v0.78.0/go.mod h1:QjdrLG0uq+YwhjoVOLsS1t7TW8fs36kLs4XO5R5ECHg= -cloud.google.com/go v0.79.0/go.mod h1:3bzgcEeQlzbuEAYu4mrWhKqWjmpprinYgKJLgKHnbb8= -cloud.google.com/go v0.81.0/go.mod h1:mk/AM35KwGk/Nm2YSeZbxXdrNK3KZOYHmLkOqC2V6E0= -cloud.google.com/go/bigquery v1.0.1/go.mod h1:i/xbL2UlR5RvWAURpBYZTtm/cXjCha9lbfbpx4poX+o= -cloud.google.com/go/bigquery v1.3.0/go.mod h1:PjpwJnslEMmckchkHFfq+HTD2DmtT67aNFKH1/VBDHE= -cloud.google.com/go/bigquery v1.4.0/go.mod h1:S8dzgnTigyfTmLBfrtrhyYhwRxG72rYxvftPBK2Dvzc= -cloud.google.com/go/bigquery v1.5.0/go.mod h1:snEHRnqQbz117VIFhE8bmtwIDY80NLUZUMb4Nv6dBIg= -cloud.google.com/go/bigquery v1.7.0/go.mod h1://okPTzCYNXSlb24MZs83e2Do+h+VXtc4gLoIoXIAPc= -cloud.google.com/go/bigquery v1.8.0/go.mod h1:J5hqkt3O0uAFnINi6JXValWIb1v0goeZM77hZzJN/fQ= -cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE= -cloud.google.com/go/datastore v1.1.0/go.mod h1:umbIZjpQpHh4hmRpGhH4tLFup+FVzqBi1b3c64qFpCk= -cloud.google.com/go/firestore v1.1.0/go.mod h1:ulACoGHTpvq5r8rxGJ4ddJZBZqakUQqClKRT5SZwBmk= -cloud.google.com/go/pubsub v1.0.1/go.mod h1:R0Gpsv3s54REJCy4fxDixWD93lHJMoZTyQ2kNxGRt3I= -cloud.google.com/go/pubsub v1.1.0/go.mod h1:EwwdRX2sKPjnvnqCa270oGRyludottCI76h+R3AArQw= -cloud.google.com/go/pubsub v1.2.0/go.mod h1:jhfEVHT8odbXTkndysNHCcx0awwzvfOlguIAii9o8iA= -cloud.google.com/go/pubsub v1.3.1/go.mod h1:i+ucay31+CNRpDW4Lu78I4xXG+O1r/MAHgjpRVR+TSU= -cloud.google.com/go/storage v1.0.0/go.mod h1:IhtSnM/ZTZV8YYJWCY8RULGVqBDmpoyjwiyrjsg+URw= -cloud.google.com/go/storage v1.5.0/go.mod h1:tpKbwo567HUNpVclU5sGELwQWBDZ8gh0ZeosJ0Rtdos= -cloud.google.com/go/storage v1.6.0/go.mod h1:N7U0C8pVQ/+NIKOBQyamJIeKQKkZ+mxpohlUTyfDhBk= -cloud.google.com/go/storage v1.8.0/go.mod h1:Wv1Oy7z6Yz3DshWRJFhqM/UCfaWIRTdp0RXyy7KQOVs= -cloud.google.com/go/storage v1.10.0/go.mod h1:FLPqc6j+Ki4BU591ie1oL6qBQGu2Bl/tZ9ullr3+Kg0= -dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= -github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= -github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= -github.com/MarvinJWendt/testza v0.1.0/go.mod h1:7AxNvlfeHP7Z/hDQ5JtE3OKYT3XFUeLCDE2DQninSqs= -github.com/MarvinJWendt/testza v0.2.1/go.mod h1:God7bhG8n6uQxwdScay+gjm9/LnO4D3kkcZX4hv9Rp8= -github.com/MarvinJWendt/testza v0.2.8/go.mod h1:nwIcjmr0Zz+Rcwfh3/4UhBp7ePKVhuBExvZqnKYWlII= -github.com/MarvinJWendt/testza v0.2.10 h1:cX4zE9TofXxe72a6EPIYAxC+8cVWTsmmgsXTZIT+5bQ= -github.com/MarvinJWendt/testza v0.2.10/go.mod h1:pd+VWsoGUiFtq+hRKSU1Bktnn+DMCSrDrXDpX2bG66k= -github.com/antihax/optional v1.0.0/go.mod h1:uupD/76wgC+ih3iEmQUL+0Ugr19nfwCT1kdvxnR2qWY= -github.com/armon/circbuf v0.0.0-20150827004946-bbbad097214e/go.mod h1:3U/XgcO3hCbHZ8TKRvWD2dDTCfh9M9ya+I9JpbB7O8o= -github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da/go.mod h1:Q73ZrmVTwzkszR9V5SSuryQ31EELlFMUz1kKyl939pY= -github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= -github.com/atomicgo/cursor v0.0.1 h1:xdogsqa6YYlLfM+GyClC/Lchf7aiMerFiZQn7soTOoU= -github.com/atomicgo/cursor v0.0.1/go.mod h1:cBON2QmmrysudxNBFthvMtN32r3jxVRIvzkUiF/RuIk= -github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs= -github.com/bketelsen/crypt v0.0.4/go.mod h1:aI6NrJ0pMGgvZKL1iVgXLnfIFJtfV+bKCoqOes/6LfM= -github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= -github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI= -github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI= -github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU= -github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= -github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc= -github.com/cncf/udpa/go v0.0.0-20200629203442-efcf912fb354/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk= -github.com/cncf/udpa/go v0.0.0-20201120205902-5459f2c99403/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk= -github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= -github.com/coreos/go-systemd/v22 v22.3.2/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc= -github.com/cpuguy83/go-md2man/v2 v2.0.0/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= -github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= -github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= -github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= -github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= -github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= -github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98= -github.com/envoyproxy/go-control-plane v0.9.7/go.mod h1:cwu0lG7PUMfa9snN8LXBig5ynNVH9qI8YYLbd1fK2po= -github.com/envoyproxy/go-control-plane v0.9.9-0.20201210154907-fd9021fe5dad/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk= -github.com/envoyproxy/go-control-plane v0.9.9-0.20210217033140-668b12f5399d/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk= -github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= -github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= -github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= -github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= -github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU= -github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= -github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= -github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= -github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= -github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= -github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= -github.com/golang/groupcache v0.0.0-20191227052852-215e87163ea7/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= -github.com/golang/groupcache v0.0.0-20200121045136-8c9f03a8e57e/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= -github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= -github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= -github.com/golang/mock v1.3.1/go.mod h1:sBzyDLLjw3U8JLTeZvSv8jJB+tU5PVekmnlKIyFUx0Y= -github.com/golang/mock v1.4.0/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= -github.com/golang/mock v1.4.1/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= -github.com/golang/mock v1.4.3/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= -github.com/golang/mock v1.4.4/go.mod h1:l3mdAwkq5BuhzHwde/uurv3sEJeZMXNpwsxVWU71h+4= -github.com/golang/mock v1.5.0/go.mod h1:CWnOUgYIOo4TcNZ0wHX3YZCqsaM1I1Jvs6v3mP3KVu8= -github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= -github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= -github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= -github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= -github.com/golang/protobuf v1.3.4/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= -github.com/golang/protobuf v1.3.5/go.mod h1:6O5/vntMXwX2lRkT1hjjk0nAC1IDOTvTlVgjlRvqsdk= -github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= -github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= -github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= -github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= -github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= -github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8= -github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= -github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= -github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= -github.com/golang/protobuf v1.5.1/go.mod h1:DopwsBzvsk0Fs44TXzsVbJyPhcCPeIwnvohx4u74HPM= -github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= -github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= -github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= -github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= -github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= -github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= -github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= -github.com/google/go-cmp v0.4.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= -github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= -github.com/google/go-cmp v0.5.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= -github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= -github.com/google/go-cmp v0.5.3/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= -github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= -github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= -github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= -github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= -github.com/google/martian/v3 v3.0.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= -github.com/google/martian/v3 v3.1.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= -github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= -github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= -github.com/google/pprof v0.0.0-20191218002539-d4f498aebedc/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= -github.com/google/pprof v0.0.0-20200212024743-f11f1df84d12/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= -github.com/google/pprof v0.0.0-20200229191704-1ebb73c60ed3/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= -github.com/google/pprof v0.0.0-20200430221834-fc25d7d30c6d/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= -github.com/google/pprof v0.0.0-20200708004538-1a94d8640e99/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= -github.com/google/pprof v0.0.0-20201023163331-3e6fc7fc9c4c/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= -github.com/google/pprof v0.0.0-20201203190320-1bf35d6f28c2/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= -github.com/google/pprof v0.0.0-20210122040257-d980be63207e/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= -github.com/google/pprof v0.0.0-20210226084205-cbba55b83ad5/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= -github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI= -github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= -github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg= -github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk= -github.com/gookit/color v1.4.2/go.mod h1:fqRyamkC1W8uxl+lxCQxOT09l/vYfZ+QeiX3rKQHCoQ= -github.com/gookit/color v1.5.0 h1:1Opow3+BWDwqor78DcJkJCIwnkviFi+rrOANki9BUFw= -github.com/gookit/color v1.5.0/go.mod h1:43aQb+Zerm/BWh2GnrgOQm7ffz7tvQXEKV6BFMl7wAo= -github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY= -github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw= -github.com/hashicorp/consul/api v1.1.0/go.mod h1:VmuI/Lkw1nC05EYQWNKwWGbkg+FbDBtguAZLlVdkD9Q= -github.com/hashicorp/consul/sdk v0.1.1/go.mod h1:VKf9jXwCTEY1QZP2MOLRhb5i/I/ssyNV1vwHyQBF0x8= -github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= -github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= -github.com/hashicorp/go-immutable-radix v1.0.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= -github.com/hashicorp/go-msgpack v0.5.3/go.mod h1:ahLV/dePpqEmjfWmKiqvPkv/twdG7iPBM1vqhUKIvfM= -github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk= -github.com/hashicorp/go-rootcerts v1.0.0/go.mod h1:K6zTfqpRlCUIjkwsN4Z+hiSfzSTQa6eBIzfwKfwNnHU= -github.com/hashicorp/go-sockaddr v1.0.0/go.mod h1:7Xibr9yA9JjQq1JpNB2Vw7kxv8xerXegt+ozgdvDeDU= -github.com/hashicorp/go-syslog v1.0.0/go.mod h1:qPfqrKkXGihmCqbJM2mZgkZGvKG1dFdvsLplgctolz4= -github.com/hashicorp/go-uuid v1.0.0/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= -github.com/hashicorp/go-uuid v1.0.1/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= -github.com/hashicorp/go.net v0.0.1/go.mod h1:hjKkEWcCURg++eb33jQU7oqQcI9XDCnUzHA0oac0k90= -github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= -github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= -github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ= -github.com/hashicorp/logutils v1.0.0/go.mod h1:QIAnNjmIWmVIIkWDTG1z5v++HQmx9WQRO+LraFDTW64= -github.com/hashicorp/mdns v1.0.0/go.mod h1:tL+uN++7HEJ6SQLQ2/p+z2pH24WQKWjBPkE0mNTz8vQ= -github.com/hashicorp/memberlist v0.1.3/go.mod h1:ajVTdAv/9Im8oMAAj5G31PhhMCZJV2pPBoIllUwCN7I= -github.com/hashicorp/serf v0.8.2/go.mod h1:6hOLApaqBFA1NXqRQAsxw9QxuDEvNxSQRwA/JwenrHc= -github.com/ianlancetaylor/demangle v0.0.0-20181102032728-5e5cf60278f6/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= -github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= -github.com/inconshreveable/mousetrap v1.0.0 h1:Z8tu5sraLXCXIcARxBp/8cbvlwVa7Z1NHg9XEKhtSvM= -github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= -github.com/json-iterator/go v1.1.11/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= -github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= -github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk= -github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= -github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= -github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= -github.com/kr/fs v0.1.0/go.mod h1:FFnZGqtBN9Gxj7eW1uZ42v5BccTP0vu6NEaFoC2HwRg= -github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= -github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= -github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= -github.com/magiconair/properties v1.8.5/go.mod h1:y3VJvCyxH9uVvJTWEGAELF3aiYNyPKd5NZ3oSwXrF60= -github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU= -github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= -github.com/mattn/go-runewidth v0.0.13 h1:lTGmDsbAYt5DmK6OnoV7EuIF1wEIFAcxld6ypU4OSgU= -github.com/mattn/go-runewidth v0.0.13/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w= -github.com/miekg/dns v1.0.14/go.mod h1:W1PPwlIAgtquWBMBEV9nkV9Cazfe8ScdGz/Lj7v3Nrg= -github.com/mitchellh/cli v1.0.0/go.mod h1:hNIlj7HEI86fIcpObd7a0FcrxTWetlwJDGcceTlRvqc= -github.com/mitchellh/go-homedir v1.0.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= -github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI= -github.com/mitchellh/gox v0.4.0/go.mod h1:Sd9lOJ0+aimLBi73mGofS1ycjY8lL3uZM3JPS42BGNg= -github.com/mitchellh/iochan v1.0.0/go.mod h1:JwYml1nuB7xOzsp52dPpHFffvOCDupsG0QubkSMEySY= -github.com/mitchellh/mapstructure v0.0.0-20160808181253-ca63d7c062ee/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= -github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= -github.com/mitchellh/mapstructure v1.4.1/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo= -github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= -github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= -github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= -github.com/pascaldekloe/goe v0.0.0-20180627143212-57f6aae5913c/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= -github.com/pelletier/go-toml v1.9.3/go.mod h1:u1nR/EPcESfeI/szUZKdtJ0xRNbUoANCkoOuaOx1Y+c= -github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= -github.com/pkg/sftp v1.10.1/go.mod h1:lYOWFsE0bwd1+KfKJaKeuokY15vzFx25BLbzYYoAxZI= -github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= -github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= -github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI= -github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= -github.com/pterm/pterm v0.12.27/go.mod h1:PhQ89w4i95rhgE+xedAoqous6K9X+r6aSOI2eFF7DZI= -github.com/pterm/pterm v0.12.29/go.mod h1:WI3qxgvoQFFGKGjGnJR849gU0TsEOvKn5Q8LlY1U7lg= -github.com/pterm/pterm v0.12.30/go.mod h1:MOqLIyMOgmTDz9yorcYbcw+HsgoZo3BQfg2wtl3HEFE= -github.com/pterm/pterm v0.12.31/go.mod h1:32ZAWZVXD7ZfG0s8qqHXePte42kdz8ECtRyEejaWgXU= -github.com/pterm/pterm v0.12.33 h1:XiT50Pvdqn5O8FAiIqZMpXP6NkVEcmlUa+mkA1yWVCg= -github.com/pterm/pterm v0.12.33/go.mod h1:x+h2uL+n7CP/rel9+bImHD5lF3nM9vJj80k9ybiiTTE= -github.com/rivo/uniseg v0.2.0 h1:S1pD9weZBuJdFmowNwbpi7BJ8TNftyUImj/0WQi72jY= -github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= -github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ= -github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= -github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= -github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts= -github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc= -github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= -github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d/go.mod h1:OnSkiWE9lh6wB0YB77sQom3nweQdgAjqCqsofrRNTgc= -github.com/smartystreets/goconvey v1.6.4/go.mod h1:syvi0/a8iFYH4r/RixwvyeAJjdLS9QV7WQ/tjFTllLA= -github.com/spf13/afero v1.6.0/go.mod h1:Ai8FlHk4v/PARR026UzYexafAt9roJ7LcLMAmO6Z93I= -github.com/spf13/cast v1.3.1/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= -github.com/spf13/cobra v1.2.1 h1:+KmjbUw1hriSNMF55oPrkZcb27aECyrj8V2ytv7kWDw= -github.com/spf13/cobra v1.2.1/go.mod h1:ExllRjgxM/piMAM+3tAZvg8fsklGAf3tPfi+i8t68Nk= -github.com/spf13/jwalterweatherman v1.1.0/go.mod h1:aNWZUN0dPAAO/Ljvb5BEdw96iTZ0EXowPYD95IqWIGo= -github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA= -github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= -github.com/spf13/viper v1.8.1/go.mod h1:o0Pch8wJ9BVSWGQMbra6iw0oQ5oktSIBaujf1rJH9Ns= -github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= -github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= -github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= -github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= -github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= -github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= -github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= -github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= -github.com/subosito/gotenv v1.2.0/go.mod h1:N0PQaV/YGNqwC0u51sEeR/aUtSLEXKX9iv69rRypqCw= -github.com/ttacon/chalk v0.0.0-20160626202418-22c06c80ed31 h1:OXcKh35JaYsGMRzpvFkLv/MEyPuL49CThT1pZ8aSml4= -github.com/ttacon/chalk v0.0.0-20160626202418-22c06c80ed31/go.mod h1:onvgF043R+lC5RZ8IT9rBXDaEDnpnw/Cl+HFiw+v/7Q= -github.com/xo/terminfo v0.0.0-20210125001918-ca9a967f8778 h1:QldyIu/L63oPpyvQmHgvgickp1Yw510KJOqX7H24mg8= -github.com/xo/terminfo v0.0.0-20210125001918-ca9a967f8778/go.mod h1:2MuV+tbUrU1zIOPMxZ5EncGwgmMJsa+9ucAQZXxsObs= -github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= -github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= -github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= -github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= -github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= -go.etcd.io/etcd/api/v3 v3.5.0/go.mod h1:cbVKeC6lCfl7j/8jBhAK6aIYO9XOjdptoxU/nLQcPvs= -go.etcd.io/etcd/client/pkg/v3 v3.5.0/go.mod h1:IJHfcCEKxYu1Os13ZdwCwIUTUVGYTSAM3YSwc9/Ac1g= -go.etcd.io/etcd/client/v2 v2.305.0/go.mod h1:h9puh54ZTgAKtEbut2oe9P4L/oqKCVB6xsXlzd7alYQ= -go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU= -go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8= -go.opencensus.io v0.22.2/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= -go.opencensus.io v0.22.3/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= -go.opencensus.io v0.22.4/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= -go.opencensus.io v0.22.5/go.mod h1:5pWMHQbX5EPX2/62yrJeAkowc+lfs/XD7Uxpq3pI6kk= -go.opencensus.io v0.23.0/go.mod h1:XItmlyltB5F7CS4xOC1DcqMoFqwtC6OG2xF7mCv7P7E= -go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= -go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU= -go.uber.org/zap v1.17.0/go.mod h1:MXVU+bhUf/A7Xi2HNOnopQOrmycQ5Ih87HtOu4q5SSo= -golang.org/x/crypto v0.0.0-20181029021203-45a5f77698d3/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= -golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= -golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= -golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= -golang.org/x/crypto v0.0.0-20190820162420-60c769a6c586/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= -golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= -golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= -golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= -golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= -golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8= -golang.org/x/exp v0.0.0-20190829153037-c13cbed26979/go.mod h1:86+5VVa7VpoJ4kLfm080zCjGlMRFzhUhsZKEZO7MGek= -golang.org/x/exp v0.0.0-20191030013958-a1ab85dbe136/go.mod h1:JXzH8nQsPlswgeRAPE3MuO9GYsAcnJvJ4vnMwN/5qkY= -golang.org/x/exp v0.0.0-20191129062945-2f5052295587/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4= -golang.org/x/exp v0.0.0-20191227195350-da58074b4299/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4= -golang.org/x/exp v0.0.0-20200119233911-0405dc783f0a/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4= -golang.org/x/exp v0.0.0-20200207192155-f17229e696bd/go.mod h1:J/WKrq2StrnmMY6+EHIKF9dgMWnmCNThgcyBT1FY9mM= -golang.org/x/exp v0.0.0-20200224162631-6cc2880d07d6/go.mod h1:3jZMyOhIsHpP37uCMkUooju7aAi5cS1Q23tOzKc+0MU= -golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js= -golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= -golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= -golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= -golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= -golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= -golang.org/x/lint v0.0.0-20190409202823-959b441ac422/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= -golang.org/x/lint v0.0.0-20190909230951-414d861bb4ac/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= -golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= -golang.org/x/lint v0.0.0-20191125180803-fdd1cda4f05f/go.mod h1:5qLYkcX4OjUUV8bRuDixDT3tpyyb+LUpUlRWLxfhWrs= -golang.org/x/lint v0.0.0-20200130185559-910be7a94367/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= -golang.org/x/lint v0.0.0-20200302205851-738671d3881b/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= -golang.org/x/lint v0.0.0-20201208152925-83fdc39ff7b5/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= -golang.org/x/lint v0.0.0-20210508222113-6edffad5e616/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= -golang.org/x/mobile v0.0.0-20190312151609-d3739f865fa6/go.mod h1:z+o9i4GpDbdi3rU15maQ/Ox0txvL9dWGYEHz965HBQE= -golang.org/x/mobile v0.0.0-20190719004257-d2bd2a29d028/go.mod h1:E/iHnbuqvinMTCcRqshq8CkpyQDoeVncDDYHnLhea+o= -golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc= -golang.org/x/mod v0.1.0/go.mod h1:0QHyrYULN0/3qlju5TqG8bIK38QM8yzMo5ekMj3DlcY= -golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= -golang.org/x/mod v0.1.1-0.20191107180719-034126e5016b/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= -golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= -golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= -golang.org/x/mod v0.4.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= -golang.org/x/mod v0.4.1/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= -golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= -golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= -golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= -golang.org/x/net v0.0.0-20181023162649-9b4f9f5ad519/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= -golang.org/x/net v0.0.0-20181201002055-351d144fa1fc/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= -golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= -golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= -golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= -golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= -golang.org/x/net v0.0.0-20190501004415-9ce7a6920f09/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= -golang.org/x/net v0.0.0-20190503192946-f4e77d36d62c/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= -golang.org/x/net v0.0.0-20190603091049-60506f45cf65/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks= -golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20190628185345-da137c7871d7/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20190724013045-ca1201d0de80/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20191209160850-c0dbc17a3553/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20200114155413-6afb5195e5aa/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20200202094626-16171245cfb2/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20200222125558-5a598a2470a0/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20200301022130-244492dfa37a/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20200324143707-d3edc9973b7e/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= -golang.org/x/net v0.0.0-20200501053045-e0ff5e5a1de5/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= -golang.org/x/net v0.0.0-20200506145744-7e3656a0809f/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= -golang.org/x/net v0.0.0-20200513185701-a91f0712d120/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= -golang.org/x/net v0.0.0-20200520182314-0ba52f642ac2/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= -golang.org/x/net v0.0.0-20200625001655-4c5254603344/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= -golang.org/x/net v0.0.0-20200707034311-ab3426394381/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= -golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= -golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= -golang.org/x/net v0.0.0-20201031054903-ff519b6c9102/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= -golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= -golang.org/x/net v0.0.0-20201209123823-ac852fbbde11/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= -golang.org/x/net v0.0.0-20210119194325-5f4716e94777/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= -golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= -golang.org/x/net v0.0.0-20210316092652-d523dce5a7f4/go.mod h1:RBQZq4jEuRlivfhVLdyRGr576XBO4/greRjx4P4O3yc= -golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= -golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= -golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= -golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= -golang.org/x/oauth2 v0.0.0-20191202225959-858c2ad4c8b6/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= -golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= -golang.org/x/oauth2 v0.0.0-20200902213428-5d25da1a8d43/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20201109201403-9fd604954f58/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20201208152858-08078c50e5b5/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20210218202405-ba52d332ba99/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20210220000619-9bb904979d93/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20210313182246-cd4f82c27b84/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20210402161424-2e8d93401602/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20200317015054-43a5402ce75a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20200625203802-6e8e738ad208/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sys v0.0.0-20180823144017-11551d06cbcc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= -golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= -golang.org/x/sys v0.0.0-20181026203630-95b1ffbd15a5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= -golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= -golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20191001151750-bb3f8db39f24/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20191005200804-aed5e4c7ecf9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20191228213918-04cbcbbfeed8/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200113162924-86b910548bc1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200122134326-e047566fdf82/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200202164722-d101bd2416d5/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200212091648-12a6c2dcc1e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200331124033-c3d80250170d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200501052902-10377860bb8e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200511232937-7e40ca221e25/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200515095857-1151b9dac4a9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200523222454-059865788121/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200803210538-64077c9b5642/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200905004654-be1d3432aa8f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20201201145000-ef89a241ccb3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210104204734-6f8348627aad/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210220050731-9a76102bfb43/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210305230114-8fe3ee5dd75b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210315160823-c6e025ad8005/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210320140829-1e4c9ba3b0c4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210403161142-5e06dd20ab57/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20211013075003-97ac67df715c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.1.0 h1:kunALQeHf1/185U1i0GOB/fy1IPRDDpuoOOqRReG57U= -golang.org/x/sys v0.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= -golang.org/x/term v0.0.0-20210220032956-6a3ed077a48d/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= -golang.org/x/term v0.0.0-20210615171337-6886f2dfbf5b/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= -golang.org/x/term v0.0.0-20210927222741-03fcf44c2211 h1:JGgROgKl9N8DuW20oFS5gxc+lE67/N3FcwmBPMe7ArY= -golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= -golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= -golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= -golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= -golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= -golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= -golang.org/x/text v0.3.4/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= -golang.org/x/text v0.3.5/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= -golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= -golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= -golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= -golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= -golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= -golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= -golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= -golang.org/x/tools v0.0.0-20190312151545-0bb0c0a6e846/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= -golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= -golang.org/x/tools v0.0.0-20190328211700-ab21143f2384/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= -golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= -golang.org/x/tools v0.0.0-20190506145303-2d16b83fe98c/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= -golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= -golang.org/x/tools v0.0.0-20190606124116-d0a3d012864b/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= -golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= -golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= -golang.org/x/tools v0.0.0-20190816200558-6889da9d5479/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20190911174233-4f2ddba30aff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20191012152004-8de300cfc20a/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20191112195655-aa38f8e97acc/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20191113191852-77e3bb0ad9e7/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20191115202509-3a792d9c32b2/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20191125144606-a911d9008d1f/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20191130070609-6e064ea0cf2d/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20191216173652-a0e659d51361/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20191227053925-7b8e75db28f4/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200117161641-43d50277825c/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200122220014-bf1340f18c4a/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200130002326-2f3ba24bd6e7/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200204074204-1cc6d1ef6c74/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200207183749-b753a1ba74fa/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200212150539-ea181f53ac56/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200224181240-023911ca70b2/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200227222343-706bc42d1f0d/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20200304193943-95d2e580d8eb/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw= -golang.org/x/tools v0.0.0-20200312045724-11d5b4c81c7d/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw= -golang.org/x/tools v0.0.0-20200331025713-a30bf2db82d4/go.mod h1:Sl4aGygMT6LrqrWclx+PTx3U+LnKx/seiNR+3G19Ar8= -golang.org/x/tools v0.0.0-20200501065659-ab2804fb9c9d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= -golang.org/x/tools v0.0.0-20200512131952-2bc93b1c0c88/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= -golang.org/x/tools v0.0.0-20200515010526-7d3b6ebf133d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= -golang.org/x/tools v0.0.0-20200618134242-20370b0cb4b2/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= -golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= -golang.org/x/tools v0.0.0-20200729194436-6467de6f59a7/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= -golang.org/x/tools v0.0.0-20200804011535-6c149bb5ef0d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= -golang.org/x/tools v0.0.0-20200825202427-b303f430e36d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= -golang.org/x/tools v0.0.0-20200904185747-39188db58858/go.mod h1:Cj7w3i3Rnn0Xh82ur9kSqwfTHTeVxaDqrfMjpcNT6bE= -golang.org/x/tools v0.0.0-20201110124207-079ba7bd75cd/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= -golang.org/x/tools v0.0.0-20201201161351-ac6f37ff4c2a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= -golang.org/x/tools v0.0.0-20201208233053-a543418bbed2/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= -golang.org/x/tools v0.0.0-20210105154028-b0ab187a4818/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= -golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= -golang.org/x/tools v0.1.0/go.mod h1:xkSsbof2nBLbhDlRMhhhyNLN/zl3eTqcnHD5viDpcZ0= -golang.org/x/tools v0.1.2/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= -golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= -golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= -golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= -golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= -google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE= -google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E3M= -google.golang.org/api v0.8.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= -google.golang.org/api v0.9.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= -google.golang.org/api v0.13.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI= -google.golang.org/api v0.14.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI= -google.golang.org/api v0.15.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI= -google.golang.org/api v0.17.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= -google.golang.org/api v0.18.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= -google.golang.org/api v0.19.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= -google.golang.org/api v0.20.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= -google.golang.org/api v0.22.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE= -google.golang.org/api v0.24.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0MncE= -google.golang.org/api v0.28.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0MncE= -google.golang.org/api v0.29.0/go.mod h1:Lcubydp8VUV7KeIHD9z2Bys/sm/vGKnG1UHuDBSrHWM= -google.golang.org/api v0.30.0/go.mod h1:QGmEvQ87FHZNiUVJkT14jQNYJ4ZJjdRF23ZXz5138Fc= -google.golang.org/api v0.35.0/go.mod h1:/XrVsuzM0rZmrsbjJutiuftIzeuTQcEeaYcSk/mQ1dg= -google.golang.org/api v0.36.0/go.mod h1:+z5ficQTmoYpPn8LCUNVpK5I7hwkpjbcgqA7I34qYtE= -google.golang.org/api v0.40.0/go.mod h1:fYKFpnQN0DsDSKRVRcQSDQNtqWPfM9i+zNPxepjRCQ8= -google.golang.org/api v0.41.0/go.mod h1:RkxM5lITDfTzmyKFPt+wGrCJbVfniCr2ool8kTBzRTU= -google.golang.org/api v0.43.0/go.mod h1:nQsDGjRXMo4lvh5hP0TKqF244gqhGcr/YSIykhUk/94= -google.golang.org/api v0.44.0/go.mod h1:EBOGZqzyhtvMDoxwS97ctnh0zUmYY6CxqXsc1AvkYD8= -google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= -google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= -google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= -google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0= -google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= -google.golang.org/appengine v1.6.6/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= -google.golang.org/appengine v1.6.7/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= -google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= -google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= -google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= -google.golang.org/genproto v0.0.0-20190425155659-357c62f0e4bb/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= -google.golang.org/genproto v0.0.0-20190502173448-54afdca5d873/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= -google.golang.org/genproto v0.0.0-20190801165951-fa694d86fc64/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= -google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= -google.golang.org/genproto v0.0.0-20190911173649-1774047e7e51/go.mod h1:IbNlFCBrqXvoKpeg0TB2l7cyZUmoaFKYIwrEpbDKLA8= -google.golang.org/genproto v0.0.0-20191108220845-16a3f7862a1a/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= -google.golang.org/genproto v0.0.0-20191115194625-c23dd37a84c9/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= -google.golang.org/genproto v0.0.0-20191216164720-4f79533eabd1/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= -google.golang.org/genproto v0.0.0-20191230161307-f3c370f40bfb/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= -google.golang.org/genproto v0.0.0-20200115191322-ca5a22157cba/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= -google.golang.org/genproto v0.0.0-20200122232147-0452cf42e150/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= -google.golang.org/genproto v0.0.0-20200204135345-fa8e72b47b90/go.mod h1:GmwEX6Z4W5gMy59cAlVYjN9JhxgbQH6Gn+gFDQe2lzA= -google.golang.org/genproto v0.0.0-20200212174721-66ed5ce911ce/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= -google.golang.org/genproto v0.0.0-20200224152610-e50cd9704f63/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= -google.golang.org/genproto v0.0.0-20200228133532-8c2c7df3a383/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= -google.golang.org/genproto v0.0.0-20200305110556-506484158171/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= -google.golang.org/genproto v0.0.0-20200312145019-da6875a35672/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= -google.golang.org/genproto v0.0.0-20200331122359-1ee6d9798940/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= -google.golang.org/genproto v0.0.0-20200430143042-b979b6f78d84/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= -google.golang.org/genproto v0.0.0-20200511104702-f5ebc3bea380/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= -google.golang.org/genproto v0.0.0-20200513103714-09dca8ec2884/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= -google.golang.org/genproto v0.0.0-20200515170657-fc4c6c6a6587/go.mod h1:YsZOwe1myG/8QRHRsmBRE1LrgQY60beZKjly0O1fX9U= -google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo= -google.golang.org/genproto v0.0.0-20200618031413-b414f8b61790/go.mod h1:jDfRM7FcilCzHH/e9qn6dsT145K34l5v+OpcnNgKAAA= -google.golang.org/genproto v0.0.0-20200729003335-053ba62fc06f/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20200804131852-c06518451d9c/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20200825200019-8632dd797987/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20200904004341-0bd0a958aa1d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20201109203340-2640f1f9cdfb/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20201201144952-b05cb90ed32e/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20201210142538-e3217bee35cc/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20201214200347-8c77b98c765d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20210222152913-aa3ee6e6a81c/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20210303154014-9728d6b83eeb/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20210310155132-4ce2db91004e/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20210319143718-93e7006c17a6/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20210402141018-6c239bbf2bb1/go.mod h1:9lPAdzaEmUacj36I+k7YKbEc5CXzPIeORRgDAUOu28A= -google.golang.org/genproto v0.0.0-20210602131652-f16073e35f0c/go.mod h1:UODoCrxHCcBojKKwX1terBiRUaqAsFqJiF615XL43r0= -google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= -google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38= -google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= -google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= -google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY= -google.golang.org/grpc v1.26.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= -google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= -google.golang.org/grpc v1.27.1/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= -google.golang.org/grpc v1.28.0/go.mod h1:rpkK4SK4GF4Ach/+MFLZUBavHOvF2JJB5uozKKal+60= -google.golang.org/grpc v1.29.1/go.mod h1:itym6AZVZYACWQqET3MqgPpjcuV5QH3BxFS3IjizoKk= -google.golang.org/grpc v1.30.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= -google.golang.org/grpc v1.31.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= -google.golang.org/grpc v1.31.1/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= -google.golang.org/grpc v1.33.1/go.mod h1:fr5YgcSWrqhRRxogOsw7RzIpsmvOZ6IcH4kBYTpR3n0= -google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc= -google.golang.org/grpc v1.34.0/go.mod h1:WotjhfgOW/POjDeRt8vscBtXq+2VjORFy659qA51WJ8= -google.golang.org/grpc v1.35.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= -google.golang.org/grpc v1.36.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= -google.golang.org/grpc v1.36.1/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= -google.golang.org/grpc v1.38.0/go.mod h1:NREThFqKR1f3iQ6oBuvc5LadQuXVGo9rkm5ZGrQdJfM= -google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= -google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= -google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= -google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= -google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= -google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= -google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= -google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= -google.golang.org/protobuf v1.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGjtUeSXeh4= -google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= -google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= -google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= -gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= -gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= -gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= -gopkg.in/ini.v1 v1.62.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= -gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= -gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= -gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= -gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= -gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= -gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b h1:h8qDotaEPuJATrMmW04NCwg7v22aHH28wwpauUhK9Oo= -gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= -honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= -honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= -honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= -honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= -honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg= -honnef.co/go/tools v0.0.1-2020.1.3/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k= -honnef.co/go/tools v0.0.1-2020.1.4/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k= -rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8= -rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0= -rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA= diff --git a/cli/main.go b/cli/main.go deleted file mode 100644 index 66f56704f..000000000 --- a/cli/main.go +++ /dev/null @@ -1,11 +0,0 @@ -package main - -import ( - "lido-aragon/cmd" -) - -func main() { - defer cmd.Lido.Shutdown() - - cmd.Execute() -} diff --git a/cli/pkg/apps/lido_apps.go b/cli/pkg/apps/lido_apps.go deleted file mode 100644 index b8e3090d9..000000000 --- a/cli/pkg/apps/lido_apps.go +++ /dev/null @@ -1,202 +0,0 @@ -package apps - -import ( - "bytes" - "encoding/json" - "errors" - "fmt" - "io/ioutil" - "lido-aragon/pkg/logs" - "log" - "os" - "os/exec" - "regexp" - "strings" - "sync" - "syscall" - "time" - - "github.com/pterm/pterm" -) - -type LidoAppsClient struct { - Apps []*LidoApp - Path string - - AppsLocator string -} - -type LidoApp struct { - Name string - Command string - Cmd *exec.Cmd - Outb bytes.Buffer - Errb bytes.Buffer - RunningUrl string -} - -type Manifest struct { - Name string `json:"name"` -} - -func (node *LidoAppsClient) getAppPath() (string, error) { - if node.Path != "" { - return node.Path, nil - } - - paths := []string{"apps/", "../apps/"} - - fmt.Println(paths) - for _, path := range paths { - _, err := ioutil.ReadDir(path) - if err == nil { - return path, nil - } - } - - return "", nil -} - -func (node *LidoAppsClient) Start() error { - pterm.Info.Println("Lido apps: checking apps...") - - apps := make([]*LidoApp, 0) - - appPath, err := node.getAppPath() - if err != nil { - return err - } - - files, err := ioutil.ReadDir(appPath) - if err != nil { - return err - } - - for _, f := range files { - - path := appPath + "/" + f.Name() - appPath := path + "/app/" - maninfestPath := path + "/manifest.json" - - _, err := os.Stat(maninfestPath) - if err != nil { - pterm.Info.Printfln("Manifest %s doesn't exists, continue", maninfestPath) - continue - } - - jsonFile, err := os.Open(maninfestPath) - if err != nil { - return err - } - jsonResult, _ := ioutil.ReadAll(jsonFile) - - defer jsonFile.Close() - - var manifest Manifest - json.Unmarshal(jsonResult, &manifest) - - apps = append(apps, &LidoApp{ - Name: manifest.Name, - Command: fmt.Sprintf("yarn --cwd %s run dev-fallback", appPath), - }) - - pterm.Info.Printfln("Found app: %s", manifest.Name) - } - - if len(apps) == 0 { - log.Println("No apps") - return nil - } - - pterm.Info.Println("Try to start apps") - - var wg sync.WaitGroup - - node.Apps = apps - - for _, app := range node.Apps { - - app.RunningUrl = "" - app.Outb.Reset() - app.Errb.Reset() - - app.Cmd = exec.Command("/bin/sh", "-c", app.Command) - // app.Cmd.SysProcAttr = &syscall.SysProcAttr{Setpgid: true} - - app.Cmd.Stdout = &app.Outb - app.Cmd.Stderr = &app.Errb - - err := app.Cmd.Start() - if err != nil { - return err - } - - wg.Add(1) - go node.CheckRunningUrl(&wg, app) - } - - wg.Wait() - - return nil -} - -func (node *LidoAppsClient) CheckRunningUrl(wg *sync.WaitGroup, app *LidoApp) error { - defer wg.Done() - - pterm.Info.Println(app.Name + " starting...") - - re, _ := regexp.Compile(`Server running at (.*?)\s`) - - for { - if logs.Verbose && app.Outb.String() != "" { - log.Println(app.Outb.String()) - } - - if strings.Contains(app.Outb.String(), "Built in") { - - res := re.FindAllStringSubmatch(app.Outb.String(), -1) - - app.RunningUrl = string(res[0][1]) - - pterm.Success.Printf(app.Name+": running at %v\n", app.RunningUrl) - - return nil - } - - if strings.Contains(app.Errb.String(), "Error:") { - pterm.Error.Println(app.Errb.String()) - return errors.New(app.Errb.String()) - } - app.Outb.Reset() - time.Sleep(1 * time.Second) - } - - return nil -} - -func (node *LidoAppsClient) Stop() { - - if len(node.Apps) == 0 { - return - } - - s, _ := pterm.DefaultSpinner.Start("Lido apps: Stopping...") - for _, app := range node.Apps { - app.Stop() - } - - node.Apps = nil - - s.Success("Lido apps: Stopped") -} - -func (app *LidoApp) Stop() { - if app.Cmd == nil || app.Cmd.Process == nil { - return - } - pterm.Info.Println(app.Name + ": Stopping...") - syscall.Kill(-app.Cmd.Process.Pid, syscall.SIGKILL) - app.Cmd.Process.Kill() - app.Cmd = nil - pterm.Info.Println(app.Name + ": Stopped") -} diff --git a/cli/pkg/aragon/aragon_client.go b/cli/pkg/aragon/aragon_client.go deleted file mode 100644 index 52376a1f5..000000000 --- a/cli/pkg/aragon/aragon_client.go +++ /dev/null @@ -1,158 +0,0 @@ -package aragon - -import ( - "bytes" - "errors" - "fmt" - "lido-aragon/pkg/deploy" - "lido-aragon/pkg/logs" - "os" - "os/exec" - "regexp" - "strings" - "syscall" - "time" - - "github.com/pterm/pterm" -) - -var ( - CMD_LOCAL AragonNetwork = "local" - CMD_MAINNET AragonNetwork = "mainnet" - CMD_RINKEBY AragonNetwork = "rinkeby" - CMD_STAGING AragonNetwork = "staging" - CMD_ROPSTEN AragonNetwork = "ropsten" - CMD_XDAI AragonNetwork = "xdai" -) - -type AragonNetwork string - -type AragonClient struct { - Cmd *exec.Cmd - Outb bytes.Buffer - Errb bytes.Buffer - - RunningUrl string - EnsRegistry string -} - -func checkApp(name string, address string, appInfo deploy.AppInfo, appLocatorArr *[]string) { - if name == "" || address == "" { - return - } - - var appId string - var appAddress string - - //check name - if strings.HasPrefix(name, "0x") && name == appInfo.ID { - appId = name - } else if appInfo.ID != "" && name == appInfo.Name { - appId = appInfo.ID - } else { - return - } - - //check address - if strings.Contains(address, "http://") || strings.Contains(address, "https://") { - appAddress = address + "/" - - //check fo IPFS CID v0 - https://docs.ipfs.io/concepts/content-addressing/#identifier-formats - } else if strings.HasPrefix(address, "Qm") && len(address) == 46 { - //@todo load from flags - appAddress = fmt.Sprintf("https://mainnet.lido.fi/ipfs/%s/", address) - } - - *appLocatorArr = append(*appLocatorArr, fmt.Sprintf("%s:%s", appId, appAddress)) -} - -func getAppsLocator(lidoApps string, deployedFile *deploy.DeployedFile) []string { - if lidoApps == "" { - return nil - } - var appLocatorArr []string - - lidoAppsArray := strings.Split(lidoApps, ",") - - for _, app := range lidoAppsArray { - tmp := strings.SplitN(app, ":", 2) - - checkApp(tmp[0], tmp[1], deployedFile.AppLido, &appLocatorArr) - checkApp(tmp[0], tmp[1], deployedFile.AppOracle, &appLocatorArr) - checkApp(tmp[0], tmp[1], deployedFile.AppNodeOperatorsRegistry, &appLocatorArr) - } - - return appLocatorArr -} - -func (node *AragonClient) Start(network AragonNetwork, lidoApps string, deployedFile *deploy.DeployedFile) error { - s, _ := pterm.DefaultSpinner.Start("Aragon client: starting...") - defer s.Stop() - - if network == CMD_MAINNET { - os.Setenv("RUN_CMD", "mainnet") - } - - appLocatorArr := getAppsLocator(lidoApps, deployedFile) - - if len(appLocatorArr) != 0 { - os.Setenv("ARAGON_APP_LOCATOR", strings.Join(appLocatorArr, ",")) - } - - node.RunningUrl = "" - node.Outb.Reset() - node.Errb.Reset() - - node.Cmd = exec.Command("yarn", "aragon:start") - // node.Cmd.SysProcAttr = &syscall.SysProcAttr{Setpgid: true} - node.Cmd.Stdout = &node.Outb - node.Cmd.Stderr = &node.Errb - err := node.Cmd.Start() - if err != nil { - return err - } - - re, _ := regexp.Compile(`Server running at (.*?)\s`) - reEns, _ := regexp.Compile(`ARAGON_ENS_REGISTRY_ADDRESS=(.*?)`) - - for { - if logs.Verbose { - fmt.Print(node.Outb.String()) - } - - if strings.Contains(node.Outb.String(), "Built in") { - s.Success() - - res := re.FindAllStringSubmatch(node.Outb.String(), -1) - node.RunningUrl = string(res[0][1]) - - res2 := reEns.FindAllStringSubmatch(node.Outb.String(), -1) - node.EnsRegistry = string(res2[0][1]) - - return nil - } - - if strings.Contains(node.Errb.String(), "Error:") { - s.Fail() - pterm.Error.Println(node.Errb.String()) - return errors.New(node.Errb.String()) - } - node.Outb.Reset() - time.Sleep(1 * time.Second) - } - - return nil -} - -func (node *AragonClient) Stop() { - if node.Cmd == nil { - return - } - - s, _ := pterm.DefaultSpinner.Start("Aragon client: Stopping...") - syscall.Kill(-node.Cmd.Process.Pid, syscall.SIGKILL) - node.Cmd.Process.Kill() - node.Cmd = nil - - s.Success("Aragon client: Stopped") -} diff --git a/cli/pkg/contracts/compile.go b/cli/pkg/contracts/compile.go deleted file mode 100644 index 076bb35b2..000000000 --- a/cli/pkg/contracts/compile.go +++ /dev/null @@ -1,39 +0,0 @@ -package contracts - -import ( - "bytes" - "fmt" - "lido-aragon/pkg/logs" - "os/exec" - - "github.com/pterm/pterm" -) - -type Contracts struct { - Cmd *exec.Cmd - Outb bytes.Buffer - Errb bytes.Buffer -} - -func (node *Contracts) Start() error { - s, _ := pterm.DefaultSpinner.Start("Contracts: compile...") - - node.Cmd = exec.Command("yarn", "compile") - node.Cmd.Stdout = &node.Outb - node.Cmd.Stderr = &node.Errb - err := node.Cmd.Run() - if err != nil { - pterm.Error.Println(node.Errb.String()) - s.Fail() - return err - } - - //need to use different buffers - if logs.Verbose { - fmt.Print(node.Outb.String()) - node.Outb.Reset() - } - - s.Success("Contracts: compile...done") - return nil -} diff --git a/cli/pkg/daemon/signal.go b/cli/pkg/daemon/signal.go deleted file mode 100644 index 35df889af..000000000 --- a/cli/pkg/daemon/signal.go +++ /dev/null @@ -1,50 +0,0 @@ -package daemon - -import ( - "fmt" - "os" - "os/signal" - "syscall" -) - -func WaitCtrlC() { - fmt.Println("Please use `Ctrl-C` to exit this program.") - - //waiting for signal - sigChan := make(chan os.Signal, 1) - signal.Notify(sigChan, syscall.SIGINT, syscall.SIGTERM) - - exitChan := make(chan int) - - go checkSignal(sigChan, exitChan) - - <-exitChan -} - -func checkSignal(sigChan chan os.Signal, exitChan chan int) { - sig := <-sigChan - - switch sig { - // kill -SIGHUP XXXX [XXXX - PID for your program] - case syscall.SIGHUP: - exitChan <- 0 - - // kill -SIGINT XXXX or Ctrl+c [XXXX - PID for your program] - case syscall.SIGINT: - exitChan <- 0 - - // kill -SIGTERM XXXX [XXXX - PID for your program] - case syscall.SIGTERM: - fmt.Println("Signal terminte triggered.") - exitChan <- 0 - - // kill -SIGQUIT XXXX [XXXX - PID for your program] - case syscall.SIGQUIT: - fmt.Println("Signal quit triggered.") - exitChan <- 0 - - default: - fmt.Println("Unknown signal.") - exitChan <- 0 - } -} diff --git a/cli/pkg/deploy/deploy.go b/cli/pkg/deploy/deploy.go deleted file mode 100644 index 08a6640ab..000000000 --- a/cli/pkg/deploy/deploy.go +++ /dev/null @@ -1,189 +0,0 @@ -package deploy - -import ( - "bytes" - "fmt" - "lido-aragon/pkg/logs" - "os" - "os/exec" - "strings" - - "github.com/pterm/pterm" -) - -type Deploy struct { - Cmd *exec.Cmd - Outb bytes.Buffer - Errb bytes.Buffer - - DeployedFile *DeployedFile -} - -type AppInfo struct { - BaseAddress string `json:"baseAddress"` - FullName string `json:"fullName"` - Name string `json:"name"` - ID string `json:"id"` - IpfsCid string `json:"ipfsCid"` - ContentURI string `json:"contentURI"` - ProxyAddress string `json:"proxyAddress"` -} - -type DeployedFile struct { - DaoAddress string `json:"daoAddress"` - AppLido AppInfo `json:"app:lido"` - AppOracle AppInfo `json:"app:oracle"` - AppNodeOperatorsRegistry AppInfo `json:"app:node-operators-registry"` -} - -func (node *Deploy) Start() error { - - if err := node.AragonEnv(); err != nil { - return err - } - node.AragonStdApps() - node.ApmAndTemplates() - node.DeployApps() - node.DeployDAO() - - return nil -} - -func (node *Deploy) AragonEnv() error { - s, _ := pterm.DefaultSpinner.Start("Deploy: Aragon env...") - - node.Cmd = exec.Command("yarn", "deploy:aragon-env") - - if logs.Verbose { - node.Cmd.Stdout = os.Stdout - node.Cmd.Stderr = os.Stderr - } else { - node.Cmd.Stdout = &node.Outb - node.Cmd.Stderr = &node.Errb - } - - err := node.Cmd.Run() - if err != nil { - if strings.Contains(node.Errb.String(), "Error: Cannot create instance of ENS") { - s.UpdateText("Error: Cannot create instance of ENS, remove deployed-localhost.json file and try again") - } - pterm.Error.Print(node.Errb.String()) - s.Fail() - return err - } - - s.Success("Deploy: Aragon env... done") - - return nil -} - -func (node *Deploy) AragonStdApps() error { - s, _ := pterm.DefaultSpinner.Start("Deploy: Aragon standart apps...") - - node.Cmd = exec.Command("yarn", "deploy:aragon-std-apps") - if logs.Verbose { - node.Cmd.Stdout = os.Stdout - node.Cmd.Stderr = os.Stderr - } else { - node.Cmd.Stdout = &node.Outb - node.Cmd.Stderr = &node.Errb - } - err := node.Cmd.Run() - if err != nil { - return err - } - - //need to use different buffers - if logs.Verbose { - fmt.Print(node.Outb.String()) - node.Outb.Reset() - } - - s.Success("Deploy: Aragon standart apps... done") - - return nil -} - -func (node *Deploy) ApmAndTemplates() error { - s, _ := pterm.DefaultSpinner.Start("Deploy: apm and template...") - - node.Cmd = exec.Command("yarn", "deploy:apm-and-template") - if logs.Verbose { - node.Cmd.Stdout = os.Stdout - node.Cmd.Stderr = os.Stderr - } else { - node.Cmd.Stdout = &node.Outb - node.Cmd.Stderr = &node.Errb - } - err := node.Cmd.Run() - if err != nil { - pterm.Error.Println(err) - return err - } - - //need to use different buffers - if logs.Verbose { - fmt.Print(node.Outb.String()) - node.Outb.Reset() - } - - s.Success("Deploy: apm and template... done") - - return nil -} - -func (node *Deploy) DeployApps() error { - s, _ := pterm.DefaultSpinner.Start("Deploy: lido apps...") - - node.Cmd = exec.Command("yarn", "deploy:apps") - if logs.Verbose { - node.Cmd.Stdout = os.Stdout - node.Cmd.Stderr = os.Stderr - } else { - node.Cmd.Stdout = &node.Outb - node.Cmd.Stderr = &node.Errb - } - err := node.Cmd.Run() - if err != nil { - pterm.Error.Println(err) - return err - } - - //need to use different buffers - if logs.Verbose { - fmt.Print(node.Outb.String()) - node.Outb.Reset() - } - - s.Success("Deploy: lido apps... done") - - return nil -} - -func (node *Deploy) DeployDAO() error { - s, _ := pterm.DefaultSpinner.Start("Deploy: DAO...") - - node.Cmd = exec.Command("yarn", "deploy:dao") - if logs.Verbose { - node.Cmd.Stdout = os.Stdout - node.Cmd.Stderr = os.Stderr - } else { - node.Cmd.Stdout = &node.Outb - node.Cmd.Stderr = &node.Errb - } - err := node.Cmd.Run() - if err != nil { - pterm.Error.Println(err) - return err - } - - //need to use different buffers - if logs.Verbose { - fmt.Print(node.Outb.String()) - node.Outb.Reset() - } - - s.Success("Deploy: DAO... done") - - return nil -} diff --git a/cli/pkg/hardhat/hardhat.go b/cli/pkg/hardhat/hardhat.go deleted file mode 100644 index 65129881e..000000000 --- a/cli/pkg/hardhat/hardhat.go +++ /dev/null @@ -1,74 +0,0 @@ -package hardhat - -import ( - "bytes" - "fmt" - "lido-aragon/pkg/logs" - "log" - "os/exec" - "strings" - "time" - - "github.com/pterm/pterm" -) - -type HardhatNode struct { - Fork string - Cmd *exec.Cmd -} - -func (node *HardhatNode) Start(infuraProjectUrl string) { - if node.Cmd != nil && node.Cmd.Process != nil && node.Cmd.Process.Pid != 0 { - fmt.Println("Hardhat already started") - return - } - - s, _ := pterm.DefaultSpinner.Start("Hardhat node: Starting...") - - var outb, errb bytes.Buffer - - if infuraProjectUrl == "" { - node.Cmd = exec.Command("npx", "hardhat", "node") - } else { - //node.Cmd = exec.Command("/bin/sh", "-c", "cmd1; cmd2") - node.Cmd = exec.Command("npx", "hardhat", "node", "--fork", infuraProjectUrl) - } - - node.Cmd.Stdout = &outb //os.Stdout - node.Cmd.Stderr = &errb //os.Stderr - err := node.Cmd.Start() - if err != nil { - log.Panic(err) - } - - for { - if logs.Verbose { - fmt.Print(outb.String()) - } - - if strings.Contains(outb.String(), "Accounts") { - s.Success("Hardhat node: Started") - break - } - - if strings.Contains(errb.String(), "Error:") { - s.Fail() - pterm.Error.Println(errb.String()) - break - } - outb.Reset() - time.Sleep(100 * time.Millisecond) - } -} - -func (node *HardhatNode) Stop() { - if node.Cmd == nil { - return - } - - s, _ := pterm.DefaultSpinner.Start("Hardhat node: Stopping...") - node.Cmd.Process.Kill() - node.Cmd = nil - - s.Success("Hardhat node: Stopped") -} diff --git a/cli/pkg/ipfs/ipfs.go b/cli/pkg/ipfs/ipfs.go deleted file mode 100644 index c738c74f9..000000000 --- a/cli/pkg/ipfs/ipfs.go +++ /dev/null @@ -1,72 +0,0 @@ -package ipfs - -import ( - "bytes" - "fmt" - "lido-aragon/pkg/logs" - "log" - "os/exec" - "strings" - "time" - - "github.com/pterm/pterm" -) - -type IPFS struct { - Cmd *exec.Cmd - Outb bytes.Buffer - Errb bytes.Buffer - Timeout int -} - -func (node *IPFS) Start() { - s, _ := pterm.DefaultSpinner.Start("IPFS: Starting...") - - node.Cmd = exec.Command("ipfs", "daemon") - node.Cmd.Stdout = &node.Outb - node.Cmd.Stderr = &node.Errb - err := node.Cmd.Start() - if err != nil { - log.Panic(err) - } - - timeout := node.Timeout - for { - if logs.Verbose && node.Outb.String() != "" { - fmt.Print(node.Outb.String()) - } - - if strings.Contains(node.Outb.String(), "Daemon is ready") { - s.Success("IPFS: Started") - break - } - - if strings.Contains(node.Errb.String(), "Error:") { - s.Fail() - pterm.Error.Println(node.Errb.String()) - break - } - - node.Outb.Reset() - node.Errb.Reset() - - time.Sleep(1 * time.Second) - timeout-- - - if timeout < 0 { - s.UpdateText("Timeout error") - s.Fail() - panic("Timeout error") - } - } -} - -func (node *IPFS) Stop() { - if node.Cmd == nil || node.Cmd.Process == nil || node.Cmd.Process.Pid == 0 { - return - } - s, _ := pterm.DefaultSpinner.Start("IPFS: Stopping...") - node.Cmd.Process.Kill() - node.Cmd = nil - s.Success("IPFS: Stopped") -} diff --git a/cli/pkg/logs/verbose.go b/cli/pkg/logs/verbose.go deleted file mode 100644 index ebcafc376..000000000 --- a/cli/pkg/logs/verbose.go +++ /dev/null @@ -1,3 +0,0 @@ -package logs - -var Verbose bool diff --git a/contracts/0.4.24/template/LidoTemplate.sol b/contracts/0.4.24/template/LidoTemplate.sol index 44c28bcd1..a90fb784d 100644 --- a/contracts/0.4.24/template/LidoTemplate.sol +++ b/contracts/0.4.24/template/LidoTemplate.sol @@ -124,7 +124,7 @@ contract LidoTemplate is IsContract { event TmplAPMDeployed(address apm); event TmplReposCreated(); - event TmplAppInstalled(address appProxy, bytes32 appId); + event TmplAppInstalled(address appProxy, bytes32 appId, bytes initializeData); event TmplDAOAndTokenDeployed(address dao, address token); event TmplTokensIssued(uint256 totalAmount); event TmplDaoFinalized(); @@ -208,6 +208,51 @@ contract LidoTemplate is IsContract { ens.setOwner(node, _to); } + function createStdAragonRepos( + address _agentImpl, + address _financeImpl, + address _tokenManagerImpl, + address _votingImpl + ) external onlyOwner { + uint16[3] memory initialSemanticVersion = [uint16(1), uint16(0), uint16(0)]; + + bytes memory dummyContentURI = new bytes(0); + + APMRegistry lidoRegistry = deployState.lidoRegistry; + + apmRepos.aragonAgent = lidoRegistry.newRepoWithVersion( + ARAGON_AGENT_APP_NAME, + this, + initialSemanticVersion, + _agentImpl, + dummyContentURI + ); + + apmRepos.aragonFinance = lidoRegistry.newRepoWithVersion( + ARAGON_FINANCE_APP_NAME, + this, + initialSemanticVersion, + _financeImpl, + dummyContentURI + ); + + apmRepos.aragonTokenManager = lidoRegistry.newRepoWithVersion( + ARAGON_TOKEN_MANAGER_APP_NAME, + this, + initialSemanticVersion, + _tokenManagerImpl, + dummyContentURI + ); + + apmRepos.aragonVoting = lidoRegistry.newRepoWithVersion( + ARAGON_VOTING_APP_NAME, + this, + initialSemanticVersion, + _votingImpl, + dummyContentURI + ); + } + function createRepos( uint16[3] _initialSemanticVersion, address _lidoImplAddress, @@ -216,6 +261,7 @@ contract LidoTemplate is IsContract { bytes _nodeOperatorsRegistryContentURI, address _oracleImplAddress, bytes _oracleContentURI + ) external onlyOwner { require(deployState.lidoRegistry != address(0), ERROR_REGISTRY_NOT_DEPLOYED); @@ -247,44 +293,6 @@ contract LidoTemplate is IsContract { _oracleContentURI ); - // create Aragon app repos pointing to latest upstream versions - - AppVersion memory latest = _apmResolveLatest(ARAGON_AGENT_APP_ID); - apmRepos.aragonAgent = lidoRegistry.newRepoWithVersion( - ARAGON_AGENT_APP_NAME, - this, - _initialSemanticVersion, - latest.contractAddress, - latest.contentURI - ); - - latest = _apmResolveLatest(ARAGON_FINANCE_APP_ID); - apmRepos.aragonFinance = lidoRegistry.newRepoWithVersion( - ARAGON_FINANCE_APP_NAME, - this, - _initialSemanticVersion, - latest.contractAddress, - latest.contentURI - ); - - latest = _apmResolveLatest(ARAGON_TOKEN_MANAGER_APP_ID); - apmRepos.aragonTokenManager = lidoRegistry.newRepoWithVersion( - ARAGON_TOKEN_MANAGER_APP_NAME, - this, - _initialSemanticVersion, - latest.contractAddress, - latest.contentURI - ); - - latest = _apmResolveLatest(ARAGON_VOTING_APP_ID); - apmRepos.aragonVoting = lidoRegistry.newRepoWithVersion( - ARAGON_VOTING_APP_NAME, - this, - _initialSemanticVersion, - latest.contractAddress, - latest.contentURI - ); - emit TmplReposCreated(); } @@ -393,6 +401,8 @@ contract LidoTemplate is IsContract { require(state.dao != address(0), ERROR_DAO_NOT_DEPLOYED); require(bytes(_daoName).length > 0, ERROR_INVALID_ID); + state.stakingRouter = _stakingRouter; + if (_unvestedTokensAmount != 0) { // using issue + assign to avoid setting the additional MINT_ROLE for the template state.tokenManager.issue(_unvestedTokensAmount); @@ -495,7 +505,7 @@ contract LidoTemplate is IsContract { ) internal returns (address) { address latestBaseAppAddress = _apmResolveLatest(_appId).contractAddress; address instance = address(_dao.newAppInstance(_appId, latestBaseAppAddress, _initializeData, _setDefault)); - emit TmplAppInstalled(instance, _appId); + emit TmplAppInstalled(instance, _appId, _initializeData); return instance; } @@ -597,9 +607,8 @@ contract LidoTemplate is IsContract { // NodeOperatorsRegistry perms[0] = _state.operators.MANAGE_SIGNING_KEYS(); - perms[1] = _state.operators.MANAGE_NODE_OPERATOR_ROLE(); - perms[2] = _state.operators.SET_NODE_OPERATOR_LIMIT_ROLE(); - for (i = 0; i < 3; ++i) { + perms[1] = _state.operators.SET_NODE_OPERATOR_LIMIT_ROLE(); + for (i = 0; i < 2; ++i) { _createPermissionForVoting(acl, _state.operators, perms[i], voting); } acl.createPermission(_state.stakingRouter, _state.operators, _state.operators.STAKING_ROUTER_ROLE(), voting); @@ -736,7 +745,9 @@ contract LidoTemplate is IsContract { } function _resolveRepo(bytes32 _appId) private view returns (Repo) { - return Repo(PublicResolver(ens.resolver(_appId)).addr(_appId)); + address resolverAddress = ens.resolver(_appId); + require(resolverAddress != address(0x0), "ZERO_RESOLVER_ADDRESS"); + return Repo(PublicResolver(resolverAddress).addr(_appId)); } /** diff --git a/contracts/0.8.4/WithdrawalsManagerProxy.sol b/contracts/0.8.4/WithdrawalsManagerProxy.sol new file mode 100644 index 000000000..9bb5acfad --- /dev/null +++ b/contracts/0.8.4/WithdrawalsManagerProxy.sol @@ -0,0 +1,520 @@ +// SPDX-License-Identifier: MIT + +pragma solidity 0.8.4; + + +//////////////////////////////////////////////////////////////////////////////////////////////////// +/// PART: WithdrawalsManagerStub.sol /////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////////////////////////// + +/** + * @dev An initial stub implementation for the withdrawals contract proxy. + */ +contract WithdrawalsManagerStub { + /** + * @dev Receives Ether. + * + * Currently this is intentionally not supported since Ethereum 2.0 withdrawals specification + * might change before withdrawals are enabled. This contract sits behind a proxy that can be + * upgraded to a new implementation contract collectively by LDO holders by performing a vote. + * + * When Ethereum 2.0 withdrawals specification is finalized, Lido DAO will prepare the new + * implementation contract and initiate a vote among LDO holders for upgrading the proxy to + * the new implementation. + */ + receive() external payable { + revert("not supported"); + } +} + +//////////////////////////////////////////////////////////////////////////////////////////////////// +/// PART: OpenZeppelin/openzeppelin-contracts@4.0.0/contracts/utils/Address.sol //////////////////// +//////////////////////////////////////////////////////////////////////////////////////////////////// + +/** + * @dev Collection of functions related to the address type + */ +library Address { + /** + * @dev Returns true if `account` is a contract. + * + * [IMPORTANT] + * ==== + * It is unsafe to assume that an address for which this function returns + * false is an externally-owned account (EOA) and not a contract. + * + * Among others, `isContract` will return false for the following + * types of addresses: + * + * - an externally-owned account + * - a contract in construction + * - an address where a contract will be created + * - an address where a contract lived, but was destroyed + * ==== + */ + function isContract(address account) internal view returns (bool) { + // This method relies on extcodesize, which returns 0 for contracts in + // construction, since the code is only stored at the end of the + // constructor execution. + + uint256 size; + // solhint-disable-next-line no-inline-assembly + assembly { size := extcodesize(account) } + return size > 0; + } + + /** + * @dev Replacement for Solidity's `transfer`: sends `amount` wei to + * `recipient`, forwarding all available gas and reverting on errors. + * + * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost + * of certain opcodes, possibly making contracts go over the 2300 gas limit + * imposed by `transfer`, making them unable to receive funds via + * `transfer`. {sendValue} removes this limitation. + * + * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more]. + * + * IMPORTANT: because control is transferred to `recipient`, care must be + * taken to not create reentrancy vulnerabilities. Consider using + * {ReentrancyGuard} or the + * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. + */ + function sendValue(address payable recipient, uint256 amount) internal { + require(address(this).balance >= amount, "Address: insufficient balance"); + + // solhint-disable-next-line avoid-low-level-calls, avoid-call-value + (bool success, ) = recipient.call{ value: amount }(""); + require(success, "Address: unable to send value, recipient may have reverted"); + } + + /** + * @dev Performs a Solidity function call using a low level `call`. A + * plain`call` is an unsafe replacement for a function call: use this + * function instead. + * + * If `target` reverts with a revert reason, it is bubbled up by this + * function (like regular Solidity function calls). + * + * Returns the raw returned data. To convert to the expected return value, + * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. + * + * Requirements: + * + * - `target` must be a contract. + * - calling `target` with `data` must not revert. + * + * _Available since v3.1._ + */ + function functionCall(address target, bytes memory data) internal returns (bytes memory) { + return functionCall(target, data, "Address: low-level call failed"); + } + + /** + * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with + * `errorMessage` as a fallback revert reason when `target` reverts. + * + * _Available since v3.1._ + */ + function functionCall(address target, bytes memory data, string memory errorMessage) internal returns (bytes memory) { + return functionCallWithValue(target, data, 0, errorMessage); + } + + /** + * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], + * but also transferring `value` wei to `target`. + * + * Requirements: + * + * - the calling contract must have an ETH balance of at least `value`. + * - the called Solidity function must be `payable`. + * + * _Available since v3.1._ + */ + function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) { + return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); + } + + /** + * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but + * with `errorMessage` as a fallback revert reason when `target` reverts. + * + * _Available since v3.1._ + */ + function functionCallWithValue(address target, bytes memory data, uint256 value, string memory errorMessage) internal returns (bytes memory) { + require(address(this).balance >= value, "Address: insufficient balance for call"); + require(isContract(target), "Address: call to non-contract"); + + // solhint-disable-next-line avoid-low-level-calls + (bool success, bytes memory returndata) = target.call{ value: value }(data); + return _verifyCallResult(success, returndata, errorMessage); + } + + /** + * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], + * but performing a static call. + * + * _Available since v3.3._ + */ + function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { + return functionStaticCall(target, data, "Address: low-level static call failed"); + } + + /** + * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], + * but performing a static call. + * + * _Available since v3.3._ + */ + function functionStaticCall(address target, bytes memory data, string memory errorMessage) internal view returns (bytes memory) { + require(isContract(target), "Address: static call to non-contract"); + + // solhint-disable-next-line avoid-low-level-calls + (bool success, bytes memory returndata) = target.staticcall(data); + return _verifyCallResult(success, returndata, errorMessage); + } + + /** + * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], + * but performing a delegate call. + * + * _Available since v3.4._ + */ + function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { + return functionDelegateCall(target, data, "Address: low-level delegate call failed"); + } + + /** + * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], + * but performing a delegate call. + * + * _Available since v3.4._ + */ + function functionDelegateCall(address target, bytes memory data, string memory errorMessage) internal returns (bytes memory) { + require(isContract(target), "Address: delegate call to non-contract"); + + // solhint-disable-next-line avoid-low-level-calls + (bool success, bytes memory returndata) = target.delegatecall(data); + return _verifyCallResult(success, returndata, errorMessage); + } + + function _verifyCallResult(bool success, bytes memory returndata, string memory errorMessage) private pure returns(bytes memory) { + if (success) { + return returndata; + } else { + // Look for revert reason and bubble it up if present + if (returndata.length > 0) { + // The easiest way to bubble the revert reason is using memory via assembly + + // solhint-disable-next-line no-inline-assembly + assembly { + let returndata_size := mload(returndata) + revert(add(32, returndata), returndata_size) + } + } else { + revert(errorMessage); + } + } + } +} + + +//////////////////////////////////////////////////////////////////////////////////////////////////// +/// PART: OpenZeppelin/openzeppelin-contracts@4.0.0/contracts/proxy/Proxy.sol ////////////////////// +//////////////////////////////////////////////////////////////////////////////////////////////////// + +/** + * @dev This abstract contract provides a fallback function that delegates all calls to another contract using the EVM + * instruction `delegatecall`. We refer to the second contract as the _implementation_ behind the proxy, and it has to + * be specified by overriding the virtual {_implementation} function. + * + * Additionally, delegation to the implementation can be triggered manually through the {_fallback} function, or to a + * different contract through the {_delegate} function. + * + * The success and return data of the delegated call will be returned back to the caller of the proxy. + */ +abstract contract Proxy { + /** + * @dev Delegates the current call to `implementation`. + * + * This function does not return to its internall call site, it will return directly to the external caller. + */ + function _delegate(address implementation) internal virtual { + // solhint-disable-next-line no-inline-assembly + assembly { + // Copy msg.data. We take full control of memory in this inline assembly + // block because it will not return to Solidity code. We overwrite the + // Solidity scratch pad at memory position 0. + calldatacopy(0, 0, calldatasize()) + + // Call the implementation. + // out and outsize are 0 because we don't know the size yet. + let result := delegatecall(gas(), implementation, 0, calldatasize(), 0, 0) + + // Copy the returned data. + returndatacopy(0, 0, returndatasize()) + + switch result + // delegatecall returns 0 on error. + case 0 { revert(0, returndatasize()) } + default { return(0, returndatasize()) } + } + } + + /** + * @dev This is a virtual function that should be overriden so it returns the address to which the fallback function + * and {_fallback} should delegate. + */ + function _implementation() internal view virtual returns (address); + + /** + * @dev Delegates the current call to the address returned by `_implementation()`. + * + * This function does not return to its internall call site, it will return directly to the external caller. + */ + function _fallback() internal virtual { + _beforeFallback(); + _delegate(_implementation()); + } + + /** + * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if no other + * function in the contract matches the call data. + */ + fallback () external payable virtual { + _fallback(); + } + + /** + * @dev Fallback function that delegates calls to the address returned by `_implementation()`. Will run if call data + * is empty. + */ + receive () external payable virtual { + _fallback(); + } + + /** + * @dev Hook that is called before falling back to the implementation. Can happen as part of a manual `_fallback` + * call, or as part of the Solidity `fallback` or `receive` functions. + * + * If overriden should call `super._beforeFallback()`. + */ + function _beforeFallback() internal virtual { + } +} + + +//////////////////////////////////////////////////////////////////////////////////////////////////// +/// PART: OpenZeppelin/openzeppelin-contracts@4.0.0/contracts/proxy/ERC1967/ERC1967Proxy.sol /////// +//////////////////////////////////////////////////////////////////////////////////////////////////// + +/** + * @dev This contract implements an upgradeable proxy. It is upgradeable because calls are delegated to an + * implementation address that can be changed. This address is stored in storage in the location specified by + * https://eips.ethereum.org/EIPS/eip-1967[EIP1967], so that it doesn't conflict with the storage layout of the + * implementation behind the proxy. + * + * Upgradeability is only provided internally through {_upgradeTo}. For an externally upgradeable proxy see + * {TransparentUpgradeableProxy}. + */ +contract ERC1967Proxy is Proxy { + /** + * @dev Initializes the upgradeable proxy with an initial implementation specified by `_logic`. + * + * If `_data` is nonempty, it's used as data in a delegate call to `_logic`. This will typically be an encoded + * function call, and allows initializating the storage of the proxy like a Solidity constructor. + */ + constructor(address _logic, bytes memory _data) payable { + assert(_IMPLEMENTATION_SLOT == bytes32(uint256(keccak256("eip1967.proxy.implementation")) - 1)); + _setImplementation(_logic); + if(_data.length > 0) { + Address.functionDelegateCall(_logic, _data); + } + } + + /** + * @dev Emitted when the implementation is upgraded. + */ + event Upgraded(address indexed implementation); + + /** + * @dev Storage slot with the address of the current implementation. + * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1, and is + * validated in the constructor. + */ + bytes32 private constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc; + + /** + * @dev Returns the current implementation address. + */ + function _implementation() internal view virtual override returns (address impl) { + bytes32 slot = _IMPLEMENTATION_SLOT; + // solhint-disable-next-line no-inline-assembly + assembly { + impl := sload(slot) + } + } + + /** + * @dev Upgrades the proxy to a new implementation. + * + * Emits an {Upgraded} event. + */ + function _upgradeTo(address newImplementation) internal virtual { + _setImplementation(newImplementation); + emit Upgraded(newImplementation); + } + + /** + * @dev Stores a new address in the EIP1967 implementation slot. + */ + function _setImplementation(address newImplementation) private { + require(Address.isContract(newImplementation), "ERC1967Proxy: new implementation is not a contract"); + + bytes32 slot = _IMPLEMENTATION_SLOT; + + // solhint-disable-next-line no-inline-assembly + assembly { + sstore(slot, newImplementation) + } + } +} + + +//////////////////////////////////////////////////////////////////////////////////////////////////// +/// PART: WithdrawalsManagerProxy.sol ////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////////////////////////// + +/** + * @dev Copied from https://github.com/OpenZeppelin/openzeppelin-contracts/blob/v4.1.0/contracts/utils/StorageSlot.sol + */ +library StorageSlot { + struct AddressSlot { + address value; + } + + function getAddressSlot(bytes32 slot) internal pure returns (AddressSlot storage r) { + assembly { + r.slot := slot + } + } +} + +/** + * @dev A proxy for Lido Ethereum 2.0 withdrawals manager contract. + * + * Though the Beacon chain already supports setting withdrawal credentials pointing to a smart + * contract, the withdrawals specification is not yet final and might change before withdrawals + * are enabled in the Merge network. This means that Lido cannot deploy the final implementation + * of the withdrawals manager contract yet. At the same time, it's desirable to have withdrawal + * credentials pointing to a smart contract since this would avoid the need to migrate a lot of + * validators to new withdrawal credentials once withdrawals are enabled. + * + * To solve this, Lido uses an upgradeable proxy controlled by the DAO. Initially, it uses a stub + * implementation contract, WithdrawalsManagerStub, that cannot receive Ether. The implementation + * can only be changed by LDO holders collectively by performing a DAO vote. Lido will set validator + * withdrawal credentials pointing to this proxy contract. + * + * When Ethereum 2.0 withdrawals specification is finalized, Lido DAO will prepare the new + * implementation contract and initiate a vote among LDO holders for upgrading this proxy to the + * new implementation. + * + * Once withdrawals are enabled in Ethereum 2.0, Lido DAO members will start a vote among LDO + * holders for disabling the upgradeability forever and locking the implementation by changing + * proxy admin from the DAO Voting contract to a zero address (which is an irreversible action). + */ +contract WithdrawalsManagerProxy is ERC1967Proxy { + /** + * @dev The address of Lido DAO Voting contract. + */ + address internal immutable LIDO_VOTING; + + /** + * @dev Storage slot with the admin of the contract. + * + * Equals `bytes32(uint256(keccak256("eip1967.proxy.admin")) - 1)`. + */ + bytes32 internal constant ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103; + + /** + * @dev Emitted when the admin account has changed. + */ + event AdminChanged(address previousAdmin, address newAdmin); + + /** + * @dev Initializes the upgradeable proxy with the initial stub implementation. + */ + constructor(address voting, address impl) ERC1967Proxy(impl, new bytes(0)) { + LIDO_VOTING = voting; + _setAdmin(voting); + } + + /** + * @return Returns the current implementation address. + */ + function implementation() external view returns (address) { + return _implementation(); + } + + /** + * @dev Upgrades the proxy to a new implementation, optionally performing an additional + * setup call. + * + * Can only be called by the proxy admin until the proxy is ossified. + * Cannot be called after the proxy is ossified. + * + * Emits an {Upgraded} event. + * + * @param setupCalldata Data for the setup call. The call is skipped if data is empty. + */ + function proxy_upgradeTo(address newImplementation, bytes memory setupCalldata) external { + address admin = _getAdmin(); + require(admin != address(0), "proxy: ossified"); + require(msg.sender == admin, "proxy: unauthorized"); + + _upgradeTo(newImplementation); + + if (setupCalldata.length > 0) { + Address.functionDelegateCall(newImplementation, setupCalldata, "proxy: setup failed"); + } + } + + /** + * @dev Returns the current admin. + */ + function _getAdmin() internal view returns (address) { + return StorageSlot.getAddressSlot(ADMIN_SLOT).value; + } + + /** + * @dev Stores a new address in the EIP1967 admin slot. + */ + function _setAdmin(address newAdmin) private { + StorageSlot.getAddressSlot(ADMIN_SLOT).value = newAdmin; + } + + /** + * @dev Returns the current admin of the proxy. + */ + function proxy_getAdmin() external view returns (address) { + return _getAdmin(); + } + + /** + * @dev Changes the admin of the proxy. + * + * Emits an {AdminChanged} event. + */ + function proxy_changeAdmin(address newAdmin) external { + address admin = _getAdmin(); + require(msg.sender == admin, "proxy: unauthorized"); + emit AdminChanged(admin, newAdmin); + _setAdmin(newAdmin); + } + + /** + * @dev Returns whether the implementation is locked forever. + */ + function proxy_getIsOssified() external view returns (bool) { + return _getAdmin() == address(0); + } +} \ No newline at end of file diff --git a/contracts/0.8.9/test_helpers/AccountingOracleMock.sol b/contracts/0.8.9/test_helpers/AccountingOracleMock.sol index e50c43872..198366102 100644 --- a/contracts/0.8.9/test_helpers/AccountingOracleMock.sol +++ b/contracts/0.8.9/test_helpers/AccountingOracleMock.sol @@ -22,6 +22,7 @@ contract AccountingOracleMock { AccountingOracle.ReportData calldata data, uint256 /* contractVersion */ ) external { + require(data.refSlot >= _lastRefSlot, "refSlot less than _lastRefSlot"); uint256 slotsElapsed = data.refSlot - _lastRefSlot; _lastRefSlot = data.refSlot; diff --git a/contracts/COMPILERS.md b/contracts/COMPILERS.md index 6d1825937..bd07c67ee 100644 --- a/contracts/COMPILERS.md +++ b/contracts/COMPILERS.md @@ -2,17 +2,10 @@ For Lido project coordination, governance and funds management we use [Aragon](https://aragon.org/dao), a well-developed and proven DAO Framework. The current stable release of its Kernel, [4.4.0](https://github.com/aragon/aragonOS/tree/v4.4.0) is fixed on the specific compiler version - [solc 0.4.24](https://solidity.readthedocs.io/en/v0.4.24/), that is currently outdated. Keeping security and consistency in mind, we decided to stay on an older yet proven combination - for all the contracts under Aragon management (`Lido`, `stETH`, `LegacyOracle`) we use solc 0.4.24 release. -cstETH token, that acts as autonomous wrapper and not governed by Aragon, was inherited from OpenZeppelin's library, using one of its stable releases [3.1.0](https://github.com/OpenZeppelin/openzeppelin-contracts/releases/tag/v3.1.0). +For the other contracts the newer compiler versions are used. # How to compile -Separately: - -```bash -yarn compile:4 -yarn compile:6 -``` - All at once: ```bash diff --git a/contracts/common/interfaces/IGateSealFactory.sol b/contracts/common/interfaces/IGateSealFactory.sol new file mode 100644 index 000000000..b2ffaaec5 --- /dev/null +++ b/contracts/common/interfaces/IGateSealFactory.sol @@ -0,0 +1,20 @@ +// SPDX-FileCopyrightText: 2023 Lido +// SPDX-License-Identifier: GPL-3.0 + +// See contracts/COMPILERS.md +// solhint-disable-next-line +pragma solidity >=0.4.24 <0.9.0; + +// https://github.com/lidofinance/gate-seals/blob/main/contracts/GateSealFactory.vy +interface IGateSealFactory { + + event GateSealCreated(address gate_seal); + + function create_gate_seal( + address _sealing_committee, + uint256 _seal_duration_seconds, + address[] memory _sealables, + uint256 _expiry_timestamp + ) external; + +} diff --git a/dao-deploy.sh b/dao-deploy.sh deleted file mode 100755 index 8860e69b1..000000000 --- a/dao-deploy.sh +++ /dev/null @@ -1,114 +0,0 @@ -#!/bin/bash -set -e +u -set -o pipefail - -ARAGON_APPS_REPO_REF=import-shared-minime - -if [[ -z "${DEPLOYER}" ]]; then - echo "Env variable DEPLOYER must be set" - exit 1 -fi -echo "DEPLOYER is $DEPLOYER" - -if [[ -z "${NETWORK}" ]]; then - echo "Env variable NETWORK must be set" - exit 1 -fi -echo "NETWORK is $NETWORK" - -function msg() { - MSG=$1 - if [ ! -z "$MSG" ]; then - echo ">>> =============================" - echo ">>> $MSG" - echo ">>> =============================" - fi -} - -# yarn install --immutable -yarn compile - -rm -f deployed-$NETWORK.json -cp deployed-$NETWORK-defaults.json deployed-$NETWORK.json - -# It does not deploy DepositContract if it is specified in deployed-${NETWORK}-defaults.json -yarn hardhat --network $NETWORK run --no-compile ./scripts/scratch/deploy-beacon-deposit-contract.js -msg "Deposit contract deployed or is specified." - -yarn deploy:$NETWORK:aragon-env -msg "Aragon ENV deployed." - -# NB! -# Need this renaming because during publishing of aragon apps and deploying their frontends -# via it's internal scripts all contracts get parsed. If contracts has custom errors or multiple -# verions declaration the process fails. - -MULTI_VERSION_PRAGMA="pragma solidity >=0.4.24 <0.9.0;" -SINGLE_VERSION_PRAGMA="pragma solidity 0.4.24;" - -for ff in $(find contracts/0.8.9 -iname '*.sol'); do mv "$ff" "$ff.tmp" ; done -for ff in $(grep -l -R "${MULTI_VERSION_PRAGMA}" contracts/common); do - sed -i '' "s/${MULTI_VERSION_PRAGMA}/${SINGLE_VERSION_PRAGMA}/g" "$ff" ; done - -mv contracts/0.4.24/template/LidoTemplate.sol contracts/0.4.24/template/LidoTemplate.sol.bkp -yarn deploy:$NETWORK:aragon-std-apps -msg "Aragon STD apps deployed." -mv contracts/0.4.24/template/LidoTemplate.sol.bkp contracts/0.4.24/template/LidoTemplate.sol - -yarn hardhat --network $NETWORK run ./scripts/scratch/01-deploy-lido-template-and-bases.js - -yarn hardhat --network $NETWORK run ./scripts/scratch/02-obtain-deployed-instances.js -msg "Apps instances deployed" - -yarn hardhat --network $NETWORK run ./scripts/scratch/03-register-ens-domain.js -if [ -f "tx-02-1-commit-ens-registration.json" ]; then - yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-02-1-commit-ens-registration.json -fi -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-02-2-make-ens-registration.json -msg "ENS registered" - -yarn hardhat --network $NETWORK run ./scripts/scratch/04-publish-app-frontends.js -msg "Frontend published to IPFS" - -# Okay, now we can restore the contracts -for ff in $(find contracts/0.8.9 -iname '*.sol.tmp'); do mv "$ff" "${ff%.*}" ; done -for ff in $(grep -l -R "${SINGLE_VERSION_PRAGMA}" contracts/common); do - sed -i '' "s/${SINGLE_VERSION_PRAGMA}/${MULTI_VERSION_PRAGMA}/g" "$ff" ; done - -yarn hardhat --network $NETWORK run ./scripts/scratch/05-deploy-apm.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-03-deploy-apm.json -yarn hardhat --network $NETWORK run ./scripts/scratch/06-obtain-deployed-apm.js -msg "APM deployed" - - -yarn hardhat --network $NETWORK run ./scripts/scratch/07-create-app-repos.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-07-create-app-repos.json -msg "App repos created" - -yarn hardhat --network $NETWORK run ./scripts/scratch/08-deploy-dao.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-05-deploy-dao.json - -yarn hardhat --network $NETWORK run ./scripts/scratch/09-obtain-deployed-dao.js -msg "DAO deploy started" - - -# Do it at the end, because might need the contracts initialized -yarn hardhat --network $NETWORK run ./scripts/scratch/10-issue-tokens.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-06-1-issue-tokens.json -msg "Tokens issued" - - -# Deploy the contracts before finalizing DAO, because the template might set permissions on some of them -yarn hardhat --network $NETWORK run ./scripts/scratch/13-deploy-non-aragon-contracts.js - -yarn hardhat --network $NETWORK run ./scripts/scratch/11-finalize-dao.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-11-finalize-dao.json -msg "DAO deploy finalized" - -rm ./tx-*.json - -yarn hardhat --network $NETWORK run ./scripts/scratch/14-initialize-non-aragon-contracts.js - -yarn hardhat --network $NETWORK run ./scripts/scratch/15-grant-roles.js - -# TODO: save commit of the latest deploy diff --git a/dao-local-deploy.sh b/dao-local-deploy.sh deleted file mode 100755 index 284fb4fb2..000000000 --- a/dao-local-deploy.sh +++ /dev/null @@ -1,124 +0,0 @@ -#!/bin/bash -set -e +u -set -o pipefail - -# first local account by default -DEPLOYER=${DEPLOYER:=0xb4124cEB3451635DAcedd11767f004d8a28c6eE7} -NETWORK=${NETWORK:=local} -ARAGON_APPS_REPO_REF=import-shared-minime - -echo "DEPLOYER is $DEPLOYER" -echo "NETWORK is $NETWORK" - -function msg() { - MSG=$1 - if [ ! -z "$MSG" ]; then - echo ">>> =============================" - echo ">>> $MSG" - echo ">>> =============================" - fi -} - -function pause() { - MSG=$1 - msg "$1" - read -s -n 1 -p "Press any key to continue . . ." - echo "" -} - -docker-compose down -v -docker-compose up --build -d - -rm -f deployed-$NETWORK.json -cp deployed-$NETWORK-defaults.json deployed-$NETWORK.json - -yarn install --immutable -yarn compile - -yarn deploy:$NETWORK:aragon-env -msg "Aragon ENV deployed." -yarn deploy:$NETWORK:aragon-std-apps -msg "Aragon STD apps deployed." - -yarn hardhat --network $NETWORK run --no-compile ./scripts/deploy-beacon-deposit-contract.js -msg "Deposit contract deployed." - -yarn hardhat --network $NETWORK run ./scripts/multisig/01-deploy-lido-template-and-bases.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-01-1-deploy-template.json -pause "!!! Now set the daoTemplateDeployTx hash value in deployed-$NETWORK.json" -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-01-2-deploy-lido-base.json -pause "!!! Now set the lidoBaseDeployTx hash value in deployed-$NETWORK.json" -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-01-3-deploy-legacy-oracle-base.json -pause "!!! Now set the oracleBaseDeployTx hash value in deployed-$NETWORK.json" -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-01-4-deploy-nops-base.json -pause "!!! Now set the nodeOperatorsRegistryBaseDeployTx hash value in deployed-$NETWORK.json" -yarn hardhat --network $NETWORK run ./scripts/multisig/02-obtain-deployed-instances.js -msg "Apps instances deployed" - -yarn hardhat --network $NETWORK run ./scripts/multisig/03-register-ens-domain.js -if [ -f "tx-02-1-commit-ens-registration.json" ]; then - yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-02-1-commit-ens-registration.json -fi -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-02-2-make-ens-registration.json -msg "ENS registered" - -yarn hardhat --network $NETWORK run ./scripts/multisig/04-publish-app-frontends.js -msg "Frontend published to IPFS" - -yarn hardhat --network $NETWORK run ./scripts/multisig/05-deploy-apm.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-03-deploy-apm.json -yarn hardhat --network $NETWORK run ./scripts/multisig/06-obtain-deployed-apm.js -msg "APM deployed" - -yarn hardhat --network $NETWORK run ./scripts/multisig/07-create-app-repos.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-07-create-app-repos.json -msg "App repos created" - -yarn hardhat --network $NETWORK run ./scripts/multisig/08-deploy-dao.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-05-deploy-dao.json -yarn hardhat --network $NETWORK run ./scripts/multisig/09-obtain-deployed-dao.js -msg "DAO deploy started" - -yarn hardhat --network $NETWORK run ./scripts/multisig/10-issue-tokens.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-06-1-issue-tokens.json -msg "Tokens issued" - -# Execution Layer Rewards: deploy the vault -yarn hardhat --network $NETWORK run ./scripts/multisig/26-deploy-execution-layer-rewards-vault.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-26-deploy-execution-layer-rewards-vault.json -pause "!!! Now set the executionLayerRewardsVaultDeployTx hash value in deployed-$NETWORK.json" -yarn hardhat --network $NETWORK run ./scripts/multisig/27-obtain-deployed-execution-layer-rewards-vault.js -msg "ExecutionLayerRewardsVault deployed" - -yarn hardhat --network $NETWORK run ./scripts/multisig/11-finalize-dao.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-11-finalize-dao.json -msg "DAO deploy finalized" - -# Insurance: deploy CompositePostRebaseBeaconReceiver -yarn hardhat --network $NETWORK run ./scripts/multisig/21-deploy-composite-post-rebase-beacon-receiver.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-21-deploy-composite-post-rebase-beacon-receiver.json -pause "!!! Now set the compositePostRebaseBeaconReceiverDeployTx hash value in deployed-$NETWORK.json" -yarn hardhat --network $NETWORK run ./scripts/multisig/22-obtain-composite-post-rebase-beacon-receiver.js -msg "CompositePostRebaseBeaconReceiver deployed" - -# Insurance: deploy Burner -yarn hardhat --network $NETWORK run ./scripts/multisig/23-deploy-burner.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-23-deploy-burner.json -pause "!!! Now set the burnerDeployTx hash value in deployed-$NETWORK.json" -yarn hardhat --network $NETWORK run ./scripts/multisig/24-obtain-burner.js -msg "Burner deployed" - -# Insurance: attach the contracts to the protocol -yarn hardhat --network $NETWORK run ./scripts/multisig/25-vote-burner.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-25-vote-burner.json -yarn hardhat --network $NETWORK run ./scripts/multisig/vote-and-enact.js -msg "Vote for attaching the insurance module is executed" - -# MAYBE TODO: Create and auto execute vote to increase vote time (like 10+5 minute) - -# Check the deployed protocol -yarn hardhat --network $NETWORK run ./scripts/multisig/12-check-dao.js -msg "Check completed! Clening up..." - -rm tx-*.json -msg "Deploy completed!" diff --git a/dao-local-start-protocol.sh b/dao-local-start-protocol.sh deleted file mode 100755 index 0234a42d3..000000000 --- a/dao-local-start-protocol.sh +++ /dev/null @@ -1,34 +0,0 @@ -#!/bin/bash -set -e +u -set -o pipefail - -# first local account by default -DEPLOYER=${DEPLOYER:=0xb4124cEB3451635DAcedd11767f004d8a28c6eE7} -# NETWORK=kintsugi -NETWORK=${NETWORK:=local} - - -function msg() { - MSG=$1 - if [ ! -z "$MSG" ]; then - echo ">>> =============================" - echo ">>> $MSG" - echo ">>> =============================" - fi -} - -function pause() { - MSG=$1 - msg "$1" - read -s -n 1 -p "Press any key to continue . . ." - echo "" -} - -# Start the protocol -yarn hardhat --network $NETWORK run ./scripts/multisig/31-start-protocol.js -yarn hardhat --network $NETWORK tx --from $DEPLOYER --file tx-31-start-protocol.json -yarn hardhat --network $NETWORK run ./scripts/multisig/vote-and-enact.js -msg "Vote executed and the protocol is started (including staking)" - -rm tx-31-start-protocol.json -msg "Protocol started!" diff --git a/dao-mainnetfork-deploy.sh b/dao-mainnetfork-deploy.sh deleted file mode 100755 index ebdbaa1ba..000000000 --- a/dao-mainnetfork-deploy.sh +++ /dev/null @@ -1,12 +0,0 @@ -#!/bin/bash -set -e +u -set -o pipefail - -export DEPLOYER=0x90F8bf6A479f320ead074411a4B0e7944Ea8c9C1 - -export NETWORK=mainnetfork - -# Set the variable to skip long Aragon apps frontend rebuild step on repetetive deploys -# export SKIP_APPS_LONG_BUILD_STEPS=1 - -bash dao-deploy.sh diff --git a/deployed-goerli.json b/deployed-goerli.json index 1268216b3..3ee3a3855 100644 --- a/deployed-goerli.json +++ b/deployed-goerli.json @@ -1,69 +1,167 @@ { - "deployCommit": "990dda79d8b19358b6b7ab4a38d0be4af6e732ea", - "networkId": 5, - "daoAragonId": "lido-testnet-prater", - "ensAddress": "0x043e6DBC5CAd60874727d21ecC2aAeBf50A0de80", - "daoFactoryAddress": "0x0B6769F500b293e3adacD97c3961189565069902", + "accountingOracle": { + "deployParameters": { + "consensusVersion": 1 + }, + "contract": "AccountingOracle", + "implementation": "0xC94ad2eb129146235f71cc35f7FbFec7fe404DF9", + "address": "0x76f358A842defa0E179a8970767CFf668Fc134d6", + "constructorArgs": [ + "0x1eDf09b5023DC86737b59dE68a8130De878984f5", + "0x1643E812aE58766192Cf7D2Cf9567dF2C37e9B7F", + "0x24d8451BC07e7aF4Ba94F69aCDD9ad3c6579D9FB", + 12, + 1616508000 + ] + }, "apmRegistryFactoryAddress": "0x70B3284fD016a570146cE48fC54D7A81bCd94BBa", - "miniMeTokenFactoryAddress": "0x92E2750729e04c9a5CF21A830Bd00cbd5139D992", - "aragonIDAddress": "0x673cB86B60B9768Eebb05517fbF5a7F9f1Ab62C4", - "multisigAddress": "0x8303B3A462F4Cb4d2468DDcBD4c9f212914A3a7F", - "daoTemplateConstructorArgs": [ - "0x8303B3A462F4Cb4d2468DDcBD4c9f212914A3a7F", - "0x0B6769F500b293e3adacD97c3961189565069902", - "0x043e6DBC5CAd60874727d21ecC2aAeBf50A0de80", - "0x92E2750729e04c9a5CF21A830Bd00cbd5139D992", - "0x673cB86B60B9768Eebb05517fbF5a7F9f1Ab62C4", - "0x70B3284fD016a570146cE48fC54D7A81bCd94BBa" - ], - "daoTemplateDeployTx": "0xda8746bca23c7a4f7c58e1a3370cb1ffa1e250ace90fd4684dbb1842ee6ac921", - "daoTemplateAddress": "0x9A4a36B5fe517f98BD6529d4317B0b723F600AaD", + "app:aragon-agent": { + "implementation": { + "contract": "@aragon/apps-agent/contracts/Agent.sol", + "address": "0xf6Fe63e6Ff034D60f9F2a403A046e1c456b11Ab4", + "constructorArgs": [] + }, + "aragonApp": { + "name": "aragon-agent", + "fullName": "aragon-agent.lidopm-testnet-prater.eth", + "id": "0xd4af5c5d5775bec8d4569a4d2462e0afcb7286b4129ee1ce70cdd44c59f0d961" + }, + "proxy": { + "address": "0x4333218072D5d7008546737786663c38B4D561A4", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol" + } + }, + "app:aragon-finance": { + "implementation": { + "contract": "@aragon/apps-finance/contracts/Finance.sol", + "address": "0xf6Fe63e6Ff034D60f9F2a403A046e1c456b11Ab4", + "constructorArgs": [] + }, + "aragonApp": { + "name": "aragon-finance", + "fullName": "aragon-finance.lidopm-testnet-prater.eth", + "id": "0xb0f95ac3c63389bb668187b7363eabb9d22c52359fc70f5247004f369afbe860" + }, + "proxy": { + "address": "0x75c7b1D23f1cad7Fb4D60281d7069E46440BC179", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol" + } + }, + "app:aragon-token-manager": { + "implementation": { + "contract": "@aragon/apps-lido/apps/token-manager/contracts/TokenManager.sol", + "address": "0xAb304946E8Ed172037aC9aBF9da58a6a7C8d443B", + "constructorArgs": [] + }, + "aragonApp": { + "name": "aragon-token-manager", + "fullName": "aragon-token-manager.lidopm-testnet-prater.eth", + "id": "0x0cee0e187359b8e41a11706fbe2704f89554ac7156f1c18152d5cbbd085bb91c" + }, + "proxy": { + "address": "0xDfe76d11b365f5e0023343A367f0b311701B3bc1", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol" + } + }, + "app:aragon-voting": { + "implementation": { + "contract": "@aragon/apps-lido/apps/voting/contracts/Voting.sol", + "address": "0x3DB95165e8efaF08a97A35F297FbCBd57a97aFe4", + "constructorArgs": [] + }, + "aragonApp": { + "name": "aragon-voting", + "fullName": "aragon-voting.lidopm-testnet-prater.eth", + "id": "0xee7f2abf043afe722001aaa900627a6e29adcbcce63a561fbd97e0a0c6429b94", + "contentURI": "0x697066733a516d5962774366374d6e6932797a31553358334769485667396f35316a6b53586731533877433257547755684859" + }, + "proxy": { + "address": "0xbc0B67b4553f4CF52a913DE9A6eD0057E2E758Db", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol" + } + }, "app:lido": { - "ipfsCid": "QmbmPW5r9HMdyUARNJjjE7MNqBUGrXashwoWvqRZhc1t5b", - "contentURI": "0x697066733a516d626d5057357239484d64795541524e4a6a6a45374d4e714255477258617368776f577671525a686331743562", - "name": "lido", - "fullName": "lido.lidopm-testnet-prater.eth", - "id": "0x79ac01111b462384f1b7fba84a17b9ec1f5d2fddcfcb99487d71b443832556ea", - "proxyAddress": "0x1643E812aE58766192Cf7D2Cf9567dF2C37e9B7F", - "contract": "Lido", - "implementation": "0x26c41Ef17780cAdde73A2d00902e5e18856201b4", - "constructorArgs": [], - "implementationDeployCommit": "85a41c45d5d45b1a35118ad6023e6c7ead1072d2" + "implementation": { + "contract": "contracts/0.4.24/Lido.sol", + "address": "0x26c41Ef17780cAdde73A2d00902e5e18856201b4", + "constructorArgs": [] + }, + "aragonApp": { + "name": "lido", + "fullName": "lido.lidopm-testnet-prater.eth", + "id": "0x79ac01111b462384f1b7fba84a17b9ec1f5d2fddcfcb99487d71b443832556ea", + "ipfsCid": "QmbmPW5r9HMdyUARNJjjE7MNqBUGrXashwoWvqRZhc1t5b", + "contentURI": "0x697066733a516d626d5057357239484d64795541524e4a6a6a45374d4e714255477258617368776f577671525a686331743562" + }, + "proxy": { + "address": "0x1643E812aE58766192Cf7D2Cf9567dF2C37e9B7F", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol" + } + }, + "app:node-operators-registry": { + "proxy": { + "address": "0x9D4AF1Ee19Dad8857db3a45B0374c81c8A1C6320" + }, + "implementation": { + "address": "0x4672275F75e8756a53cD87F08F67f51309675D57", + "contract": "contracts/0.4.24/nos/NodeOperatorsRegistry.sol", + "constructorArgs": [] + }, + "aragonApp": { + "ipfsCid": "QmQExJkoyg7xWXJjLaYC75UAmsGY1STY41YTG3wEK7q8dd", + "contentURI": "0x697066733a516d5145784a6b6f7967377857584a6a4c615943373555416d7347593153545934315954473377454b3771386464", + "name": "node-operators-registry", + "fullName": "node-operators-registry.lidopm-testnet-prater.eth", + "id": "0x57384c8fcaf2c1c2144974769a6ea4e5cf69090d47f5327f8fc93827f8c0001a" + } }, "app:oracle": { - "ipfsCid": "QmfACH9oSHFWgV81Dh8RSVcgaVBdhkZuHhZY2iZv5syBJK", - "contentURI": "0x697066733a516d66414348396f5348465767563831446838525356636761564264686b5a7548685a5932695a76357379424a4b", - "name": "oracle", - "fullName": "oracle.lidopm-testnet-prater.eth", - "id": "0xb2977cfc13b000b6807b9ae3cf4d938f4cc8ba98e1d68ad911c58924d6aa4f11", - "proxyAddress": "0x24d8451BC07e7aF4Ba94F69aCDD9ad3c6579D9FB" + "proxy": { + "address": "0x24d8451BC07e7aF4Ba94F69aCDD9ad3c6579D9FB" + }, + "implementation": { + "address": "0xcF9d64942DC9096520a8962a2d4496e680c6403b", + "contract": "contracts/0.4.24/oracle/LegacyOracle.sol" + }, + "aragonApp": { + "ipfsCid": "QmfACH9oSHFWgV81Dh8RSVcgaVBdhkZuHhZY2iZv5syBJK", + "contentURI": "0x697066733a516d66414348396f5348465767563831446838525356636761564264686b5a7548685a5932695a76357379424a4b", + "name": "oracle", + "fullName": "oracle.lidopm-testnet-prater.eth", + "id": "0xb2977cfc13b000b6807b9ae3cf4d938f4cc8ba98e1d68ad911c58924d6aa4f11" + } }, - "app:node-operators-registry": { - "ipfsCid": "QmQExJkoyg7xWXJjLaYC75UAmsGY1STY41YTG3wEK7q8dd", - "contentURI": "0x697066733a516d5145784a6b6f7967377857584a6a4c615943373555416d7347593153545934315954473377454b3771386464", - "name": "node-operators-registry", - "fullName": "node-operators-registry.lidopm-testnet-prater.eth", - "id": "0x57384c8fcaf2c1c2144974769a6ea4e5cf69090d47f5327f8fc93827f8c0001a", - "proxyAddress": "0x9D4AF1Ee19Dad8857db3a45B0374c81c8A1C6320", - "contract": "NodeOperatorsRegistry", - "implementation": "0x4672275F75e8756a53cD87F08F67f51309675D57", - "constructorArgs": [], - "implementationDeployCommit": "85a41c45d5d45b1a35118ad6023e6c7ead1072d2" + "aragonIDAddress": "0x673cB86B60B9768Eebb05517fbF5a7F9f1Ab62C4", + "burner": { + "deployParameters": { + "totalCoverSharesBurnt": "0", + "totalNonCoverSharesBurnt": "0" + }, + "address": "0xC1Bc16782BE0ce7A81421874A4d3E9B2B32391f2", + "constructorArgs": [ + "0x4333218072D5d7008546737786663c38B4D561A4", + "0x4333218072D5d7008546737786663c38B4D561A4", + "0x1643E812aE58766192Cf7D2Cf9567dF2C37e9B7F", + "0", + "0" + ], + "contract": "Burner" }, - "lidoApmEnsName": "lidopm-testnet-prater.eth", - "lidoApmDeployTx": "0x5177a441fbf121e4064979f53756cc2ee807c68684edae33f50868a40ebae013", - "lidoApmAddress": "0x64361389f8Bb7A1ff64e0Ad97FfAa432399D45E7", + "chainSpec": { + "depositContractAddress": "0xff50ed3d0ec03aC01D4C79aAd74928BFF48a7b2b", + "slotsPerEpoch": 32, + "secondsPerSlot": 12, + "genesisTime": 1616508000 + }, + "createAppReposTx": "0x76225ccd0d6f03c1e6707125fae7a04096f24a37e3e5c59a3847323dca343f62", + "daoAddress": "0x1dD91b354Ebd706aB3Ac7c727455C7BAA164945A", + "daoAragonId": "lido-testnet-prater", + "daoFactoryAddress": "0x0B6769F500b293e3adacD97c3961189565069902", "daoInitialSettings": { "token": { "name": "Test Lido DAO Token", "symbol": "TESTLDO" }, - "beaconSpec": { - "depositContractAddress": "0xff50ed3d0ec03aC01D4C79aAd74928BFF48a7b2b", - "slotsPerEpoch": 32, - "secondsPerSlot": 12, - "genesisTime": 1616508000 - }, "voting": { "minSupportRequired": "500000000000000000", "minAcceptanceQuorum": "50000000000000000", @@ -76,95 +174,46 @@ "nodeOperatorsPercent": 50 } }, - "createAppReposTx": "0x76225ccd0d6f03c1e6707125fae7a04096f24a37e3e5c59a3847323dca343f62", - "newDaoTx": "0xcc5a0e1b690bac8edb5638f79fcf765f703e218147fbb1d767e502560d4a7c89", - "daoAddress": "0x1dD91b354Ebd706aB3Ac7c727455C7BAA164945A", "daoTokenAddress": "0x56340274fB5a72af1A3C6609061c451De7961Bd4", - "app:aragon-agent": { - "name": "aragon-agent", - "fullName": "aragon-agent.lidopm-testnet-prater.eth", - "id": "0xd4af5c5d5775bec8d4569a4d2462e0afcb7286b4129ee1ce70cdd44c59f0d961", - "proxyAddress": "0x4333218072D5d7008546737786663c38B4D561A4" - }, - "app:aragon-finance": { - "name": "aragon-finance", - "fullName": "aragon-finance.lidopm-testnet-prater.eth", - "id": "0xb0f95ac3c63389bb668187b7363eabb9d22c52359fc70f5247004f369afbe860", - "proxyAddress": "0x75c7b1D23f1cad7Fb4D60281d7069E46440BC179" - }, - "app:aragon-token-manager": { - "name": "aragon-token-manager", - "fullName": "aragon-token-manager.lidopm-testnet-prater.eth", - "id": "0x0cee0e187359b8e41a11706fbe2704f89554ac7156f1c18152d5cbbd085bb91c", - "proxyAddress": "0xDfe76d11b365f5e0023343A367f0b311701B3bc1" - }, - "app:aragon-voting": { - "name": "aragon-voting", - "fullName": "aragon-voting.lidopm-testnet-prater.eth", - "id": "0xee7f2abf043afe722001aaa900627a6e29adcbcce63a561fbd97e0a0c6429b94", - "proxyAddress": "0xbc0B67b4553f4CF52a913DE9A6eD0057E2E758Db", - "baseAddress": "0x3DB95165e8efaF08a97A35F297FbCBd57a97aFe4", - "contentURI": "0x697066733a516d5962774366374d6e6932797a31553358334769485667396f35316a6b53586731533877433257547755684859" - }, - "vestingParams": { - "unvestedTokensAmount": "230000000000000000000000", - "holders": { - "0xa5F1d7D49F581136Cf6e58B32cBE9a2039C48bA1": "530000000000000000000000", - "0x00444797Ba158A7BdB8302e72da98dCbCCef0Fbc": "15000000000000000000000", - "0x9F188809b2ba2e29653693C4069c212a95C1B8c4": "15000000000000000000000", - "0xeC2d2E81721A477798e938924266f92c4E5Fc54e": "15000000000000000000000", - "0x401FD888B5E41113B7c0C47725A742bbc3A083EF": "15000000000000000000000", - "0x36c648351274bb4455ba6aAabF3F976824a93aF4": "15000000000000000000000", - "0xB6d0A3120c0e13749c211F48fccD9458Dfc99BD0": "15000000000000000000000", - "0x75936b355942D89d9B59798dacD70d70Ded93B78": "15000000000000000000000", - "0x39ceC2b3ba293CC15f15a3876dB8D356a1670789": "15000000000000000000000", - "0xC37F899BB002D93fdcbD9D91d42889AC0e29B704": "15000000000000000000000", - "0xc69f1606e0615F788Cb99982848528f39f188486": "15000000000000000000000", - "0xd355521db7f30F74059dFa7aE0A5C6543dFA8215": "15000000000000000000000", - "0xAA566f0eB53F26db5E15aC5B42685361B4e0d9F4": "15000000000000000000000", - "0xCd3975FC68Aa71eDebfef7442cF8e91ce407DEEE": "15000000000000000000000", - "0xfd0dd5c3b72528e8f581a8544309706d1c2d9206": "15000000000000000000000", - "0x161EE6a03D52ee176a514C88Cf07dcAcF248155f": "15000000000000000000000", - "0x11D7D72876cfC3d7CFC3d1a2C9c35EA4591694a7": "15000000000000000000000" + "deployCommit": "990dda79d8b19358b6b7ab4a38d0be4af6e732ea", + "deployer": "0x8303B3A462F4Cb4d2468DDcBD4c9f212914A3a7F", + "depositSecurityModule": { + "deployParameters": { + "maxDepositsPerBlock": 0, + "minDepositBlockDistance": 1200, + "pauseIntentValidityPeriodBlocks": 10 }, - "start": 1617139000, - "cliff": 1617139500, - "end": 1617139900, - "revokable": false + "contract": "DepositSecurityModule", + "address": "0xe57025E250275cA56f92d76660DEcfc490C7E79A", + "constructorArgs": [ + "0x1643E812aE58766192Cf7D2Cf9567dF2C37e9B7F", + "0xff50ed3d0ec03aC01D4C79aAd74928BFF48a7b2b", + "0xa3Dbd317E53D363176359E10948BA0b1c0A4c820", + "0", + "4b0", + "a" + ] }, - "wstETH": { - "address": "0x6320cD32aA674d2898A68ec82e869385Fc5f7E2f", + "dummyEmptyContract": { + "address": "0x6A03b1BbB79460169a205eFBCBc77ebE1011bCf8", + "contract": "contracts/0.8.9/test_helpers/DummyEmptyContract.sol", + "constructorArgs": [] + }, + "eip712StETH": { + "contract": "EIP712StETH", + "address": "0xB4300103FfD326f77FfB3CA54248099Fb29C3b9e", "constructorArgs": [ "0x1643E812aE58766192Cf7D2Cf9567dF2C37e9B7F" - ], - "contract": "WstETH" + ] }, + "ensAddress": "0x043e6DBC5CAd60874727d21ecC2aAeBf50A0de80", "executionLayerRewardsVault": { "address": "0x94750381bE1AbA0504C666ee1DB118F68f0780D4", "deployTx": "0x3bce97664f6003ddf4e3cf7178527dbff6098b1a6fc9ba7c337cb6aca4ea91fb", "contract": "LidoExecutionLayerRewardsVault" }, - "dummyImplementation": { - "address": "0x6A03b1BbB79460169a205eFBCBc77ebE1011bCf8" - }, - "burner": { - "parameters": { - "totalCoverSharesBurnt": "0", - "totalNonCoverSharesBurnt": "0" - }, - "address": "0xC1Bc16782BE0ce7A81421874A4d3E9B2B32391f2", - "constructorArgs": [ - "0x4333218072D5d7008546737786663c38B4D561A4", - "0x4333218072D5d7008546737786663c38B4D561A4", - "0x1643E812aE58766192Cf7D2Cf9567dF2C37e9B7F", - "0", - "0" - ], - "contract": "Burner", - "implementationDeployCommit": "85a41c45d5d45b1a35118ad6023e6c7ead1072d2" - }, - "hashConsensusForAccounting": { - "parameters": { + "hashConsensusForAccountingOracle": { + "deployParameters": { "fastLaneLengthSlots": 10, "epochsPerFrame": 40 }, @@ -180,23 +229,8 @@ "0x76f358A842defa0E179a8970767CFf668Fc134d6" ] }, - "accountingOracle": { - "parameters": { - "consensusVersion": 1 - }, - "contract": "AccountingOracle", - "implementation": "0xC94ad2eb129146235f71cc35f7FbFec7fe404DF9", - "address": "0x76f358A842defa0E179a8970767CFf668Fc134d6", - "constructorArgs": [ - "0x1eDf09b5023DC86737b59dE68a8130De878984f5", - "0x1643E812aE58766192Cf7D2Cf9567dF2C37e9B7F", - "0x24d8451BC07e7aF4Ba94F69aCDD9ad3c6579D9FB", - 12, - 1616508000 - ] - }, - "hashConsensusForValidatorsExitBus": { - "parameters": { + "hashConsensusForValidatorsExitBusOracle": { + "deployParameters": { "fastLaneLengthSlots": 10, "epochsPerFrame": 20 }, @@ -212,40 +246,71 @@ "0xb75A55EFab5A8f5224Ae93B34B25741EDd3da98b" ] }, - "validatorsExitBusOracle": { - "parameters": { - "consensusVersion": 1 - }, - "contract": "ValidatorsExitBusOracle", - "implementation": "0xe9754585e100085d20d39298aCA713d43f2ec5aF", - "address": "0xb75A55EFab5A8f5224Ae93B34B25741EDd3da98b", + "lidoApm": { + "deployTx": "0x5177a441fbf121e4064979f53756cc2ee807c68684edae33f50868a40ebae013", + "address": "0x64361389f8Bb7A1ff64e0Ad97FfAa432399D45E7" + }, + "lidoApmEnsName": "lidopm-testnet-prater.eth", + "lidoLocator": { + "contract": "LidoLocator", + "implementation": "0xBA413498500a21bC07E74605CC45932c2d45aCb8", + "address": "0x1eDf09b5023DC86737b59dE68a8130De878984f5", "constructorArgs": [ - 12, - 1616508000, - "0x1eDf09b5023DC86737b59dE68a8130De878984f5" + [ + "0x76f358A842defa0E179a8970767CFf668Fc134d6", + "0xe57025E250275cA56f92d76660DEcfc490C7E79A", + "0x94750381bE1AbA0504C666ee1DB118F68f0780D4", + "0x24d8451BC07e7aF4Ba94F69aCDD9ad3c6579D9FB", + "0x1643E812aE58766192Cf7D2Cf9567dF2C37e9B7F", + "0xbf74600040F91D3560d5757280727FB00c64Fd2E", + "0x24d8451BC07e7aF4Ba94F69aCDD9ad3c6579D9FB", + "0xC1Bc16782BE0ce7A81421874A4d3E9B2B32391f2", + "0xa3Dbd317E53D363176359E10948BA0b1c0A4c820", + "0x4333218072D5d7008546737786663c38B4D561A4", + "0xb75A55EFab5A8f5224Ae93B34B25741EDd3da98b", + "0xCF117961421cA9e546cD7f50bC73abCdB3039533", + "0xdc62f9e8C34be08501Cdef4EBDE0a280f576D762", + "0xE9CC5bD91543cdc9788454EE5063E2CD76B5206d" + ] ] }, - "depositorPreviousAddress": "0xEd23AD3EA5Fb9d10e7371Caef1b141AD1C23A80c", - "depositSecurityModule": { - "parameters": { - "maxDepositsPerBlock": 0, - "minDepositBlockDistance": 1200, - "pauseIntentValidityPeriodBlocks": 10 + "lidoTemplate": { + "address": "0x9A4a36B5fe517f98BD6529d4317B0b723F600AaD", + "deployTx": "0xda8746bca23c7a4f7c58e1a3370cb1ffa1e250ace90fd4684dbb1842ee6ac921", + "constructorArgs": [ + "0x8303B3A462F4Cb4d2468DDcBD4c9f212914A3a7F", + "0x0B6769F500b293e3adacD97c3961189565069902", + "0x043e6DBC5CAd60874727d21ecC2aAeBf50A0de80", + "0x92E2750729e04c9a5CF21A830Bd00cbd5139D992", + "0x673cB86B60B9768Eebb05517fbF5a7F9f1Ab62C4", + "0x70B3284fD016a570146cE48fC54D7A81bCd94BBa" + ] + }, + "miniMeTokenFactoryAddress": "0x92E2750729e04c9a5CF21A830Bd00cbd5139D992", + "networkId": 5, + "newDaoTx": "0xcc5a0e1b690bac8edb5638f79fcf765f703e218147fbb1d767e502560d4a7c89", + "oracleDaemonConfig": { + "deployParameters": { + "NORMALIZED_CL_REWARD_PER_EPOCH": 64, + "NORMALIZED_CL_REWARD_MISTAKE_RATE_BP": 1000, + "REBASE_CHECK_NEAREST_EPOCH_DISTANCE": 4, + "REBASE_CHECK_DISTANT_EPOCH_DISTANCE": 10, + "VALIDATOR_DELAYED_TIMEOUT_IN_SLOTS": 7200, + "VALIDATOR_DELINQUENT_TIMEOUT_IN_SLOTS": 21600, + "PREDICTION_DURATION_IN_SLOTS": 50400, + "FINALIZATION_MAX_NEGATIVE_REBASE_EPOCH_SHIFT": 1350, + "NODE_OPERATOR_NETWORK_PENETRATION_THRESHOLD_BP": 100 }, - "contract": "DepositSecurityModule", - "address": "0xe57025E250275cA56f92d76660DEcfc490C7E79A", + "contract": "OracleDaemonConfig", + "address": "0xE9CC5bD91543cdc9788454EE5063E2CD76B5206d", "constructorArgs": [ - "0x1643E812aE58766192Cf7D2Cf9567dF2C37e9B7F", - "0xff50ed3d0ec03aC01D4C79aAd74928BFF48a7b2b", - "0xa3Dbd317E53D363176359E10948BA0b1c0A4c820", - "0", - "4b0", - "a" + "0xa5F1d7D49F581136Cf6e58B32cBE9a2039C48bA1", + [] ] }, "oracleReportSanityChecker": { "deployCommit": "85a41c45d5d45b1a35118ad6023e6c7ead1072d2", - "parameters": { + "deployParameters": { "churnValidatorsPerDayLimit": 15000, "oneOffCLBalanceDecreaseBPLimit": 500, "annualBalanceIncreaseBPLimit": 1000, @@ -286,9 +351,56 @@ ] ] }, + "stakingRouter": { + "contract": "StakingRouter", + "implementation": "0x200c147cd3F344Ad09bAeCadA0a945106df337B4", + "address": "0xa3Dbd317E53D363176359E10948BA0b1c0A4c820", + "constructorArgs": [ + "0xff50ed3d0ec03aC01D4C79aAd74928BFF48a7b2b" + ] + }, + "validatorsExitBusOracle": { + "deployParameters": { + "consensusVersion": 1 + }, + "contract": "ValidatorsExitBusOracle", + "implementation": "0xe9754585e100085d20d39298aCA713d43f2ec5aF", + "address": "0xb75A55EFab5A8f5224Ae93B34B25741EDd3da98b", + "constructorArgs": [ + 12, + 1616508000, + "0x1eDf09b5023DC86737b59dE68a8130De878984f5" + ] + }, + "vestingParams": { + "unvestedTokensAmount": "230000000000000000000000", + "holders": { + "0xa5F1d7D49F581136Cf6e58B32cBE9a2039C48bA1": "530000000000000000000000", + "0x00444797Ba158A7BdB8302e72da98dCbCCef0Fbc": "15000000000000000000000", + "0x9F188809b2ba2e29653693C4069c212a95C1B8c4": "15000000000000000000000", + "0xeC2d2E81721A477798e938924266f92c4E5Fc54e": "15000000000000000000000", + "0x401FD888B5E41113B7c0C47725A742bbc3A083EF": "15000000000000000000000", + "0x36c648351274bb4455ba6aAabF3F976824a93aF4": "15000000000000000000000", + "0xB6d0A3120c0e13749c211F48fccD9458Dfc99BD0": "15000000000000000000000", + "0x75936b355942D89d9B59798dacD70d70Ded93B78": "15000000000000000000000", + "0x39ceC2b3ba293CC15f15a3876dB8D356a1670789": "15000000000000000000000", + "0xC37F899BB002D93fdcbD9D91d42889AC0e29B704": "15000000000000000000000", + "0xc69f1606e0615F788Cb99982848528f39f188486": "15000000000000000000000", + "0xd355521db7f30F74059dFa7aE0A5C6543dFA8215": "15000000000000000000000", + "0xAA566f0eB53F26db5E15aC5B42685361B4e0d9F4": "15000000000000000000000", + "0xCd3975FC68Aa71eDebfef7442cF8e91ce407DEEE": "15000000000000000000000", + "0xfd0dd5c3b72528e8f581a8544309706d1c2d9206": "15000000000000000000000", + "0x161EE6a03D52ee176a514C88Cf07dcAcF248155f": "15000000000000000000000", + "0x11D7D72876cfC3d7CFC3d1a2C9c35EA4591694a7": "15000000000000000000000" + }, + "start": 1617139000, + "cliff": 1617139500, + "end": 1617139900, + "revokable": false + }, "withdrawalQueueERC721": { "implementationDeployCommit": "85a41c45d5d45b1a35118ad6023e6c7ead1072d2", - "parameters": { + "deployParameters": { "name": "stETH Withdrawal NFT", "symbol": "unstETH" }, @@ -301,55 +413,6 @@ "unstETH" ] }, - "oracleDaemonConfig": { - "parameters": { - "NORMALIZED_CL_REWARD_PER_EPOCH": 64, - "NORMALIZED_CL_REWARD_MISTAKE_RATE_BP": 1000, - "REBASE_CHECK_NEAREST_EPOCH_DISTANCE": 4, - "REBASE_CHECK_DISTANT_EPOCH_DISTANCE": 10, - "VALIDATOR_DELAYED_TIMEOUT_IN_SLOTS": 7200, - "VALIDATOR_DELINQUENT_TIMEOUT_IN_SLOTS": 21600, - "PREDICTION_DURATION_IN_SLOTS": 50400, - "FINALIZATION_MAX_NEGATIVE_REBASE_EPOCH_SHIFT": 1350, - "NODE_OPERATOR_NETWORK_PENETRATION_THRESHOLD_BP": 100 - }, - "contract": "OracleDaemonConfig", - "address": "0xE9CC5bD91543cdc9788454EE5063E2CD76B5206d", - "constructorArgs": [ - "0xa5F1d7D49F581136Cf6e58B32cBE9a2039C48bA1", - [] - ] - }, - "lidoLocator": { - "contract": "LidoLocator", - "implementation": "0xBA413498500a21bC07E74605CC45932c2d45aCb8", - "address": "0x1eDf09b5023DC86737b59dE68a8130De878984f5", - "constructorArgs": [ - [ - "0x76f358A842defa0E179a8970767CFf668Fc134d6", - "0xe57025E250275cA56f92d76660DEcfc490C7E79A", - "0x94750381bE1AbA0504C666ee1DB118F68f0780D4", - "0x24d8451BC07e7aF4Ba94F69aCDD9ad3c6579D9FB", - "0x1643E812aE58766192Cf7D2Cf9567dF2C37e9B7F", - "0xbf74600040F91D3560d5757280727FB00c64Fd2E", - "0x24d8451BC07e7aF4Ba94F69aCDD9ad3c6579D9FB", - "0xC1Bc16782BE0ce7A81421874A4d3E9B2B32391f2", - "0xa3Dbd317E53D363176359E10948BA0b1c0A4c820", - "0x4333218072D5d7008546737786663c38B4D561A4", - "0xb75A55EFab5A8f5224Ae93B34B25741EDd3da98b", - "0xCF117961421cA9e546cD7f50bC73abCdB3039533", - "0xdc62f9e8C34be08501Cdef4EBDE0a280f576D762", - "0xE9CC5bD91543cdc9788454EE5063E2CD76B5206d" - ] - ] - }, - "eip712StETH": { - "contract": "EIP712StETH", - "address": "0xB4300103FfD326f77FfB3CA54248099Fb29C3b9e", - "constructorArgs": [ - "0x1643E812aE58766192Cf7D2Cf9567dF2C37e9B7F" - ] - }, "withdrawalVault": { "address": "0xdc62f9e8C34be08501Cdef4EBDE0a280f576D762", "contract": "WithdrawalVault", @@ -359,12 +422,11 @@ "0x4333218072D5d7008546737786663c38B4D561A4" ] }, - "stakingRouter": { - "contract": "StakingRouter", - "implementation": "0x200c147cd3F344Ad09bAeCadA0a945106df337B4", - "address": "0xa3Dbd317E53D363176359E10948BA0b1c0A4c820", + "wstETH": { + "address": "0x6320cD32aA674d2898A68ec82e869385Fc5f7E2f", "constructorArgs": [ - "0xff50ed3d0ec03aC01D4C79aAd74928BFF48a7b2b" - ] + "0x1643E812aE58766192Cf7D2Cf9567dF2C37e9B7F" + ], + "contract": "WstETH" } -} +} \ No newline at end of file diff --git a/deployed-holesky.json b/deployed-holesky.json new file mode 100644 index 000000000..3f44be7eb --- /dev/null +++ b/deployed-holesky.json @@ -0,0 +1,776 @@ +{ + "accountingOracle": { + "deployParameters": { + "consensusVersion": 1 + }, + "proxy": { + "contract": "contracts/0.8.9/proxy/OssifiableProxy.sol", + "address": "0x4E97A3972ce8511D87F334dA17a2C332542a5246", + "constructorArgs": [ + "0x6AcA050709469F1f98d8f40f68b1C83B533cd2b2", + "0x22896Bfc68814BFD855b1a167255eE497006e730", + "0x" + ] + }, + "implementation": { + "contract": "contracts/0.8.9/oracle/AccountingOracle.sol", + "address": "0x6AcA050709469F1f98d8f40f68b1C83B533cd2b2", + "constructorArgs": [ + "0x28FAB2059C713A7F9D8c86Db49f9bb0e96Af1ef8", + "0x3F1c547b21f65e10480dE3ad8E19fAAC46C95034", + "0x072f72BE3AcFE2c52715829F2CD9061A6C8fF019", + 12, + 1695902400 + ] + } + }, + "apmRepoBaseAddress": "0x8959360c48D601a6817BAf2449E5D00cC543FA3A", + "app:aragon-agent": { + "implementation": { + "contract": "@aragon/apps-agent/contracts/Agent.sol", + "address": "0xF4aDA7Ff34c508B9Af2dE4160B6078D2b58FD46B", + "constructorArgs": [] + }, + "aragonApp": { + "name": "aragon-agent", + "fullName": "aragon-agent.lidopm.eth", + "id": "0x701a4fd1f5174d12a0f1d9ad2c88d0ad11ab6aad0ac72b7d9ce621815f8016a9" + }, + "proxy": { + "address": "0xE92329EC7ddB11D25e25b3c21eeBf11f15eB325d", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol", + "constructorArgs": [ + "0x3b03f75Ec541Ca11a223bB58621A3146246E1644", + "0x701a4fd1f5174d12a0f1d9ad2c88d0ad11ab6aad0ac72b7d9ce621815f8016a9", + "0x8129fc1c" + ] + } + }, + "app:aragon-finance": { + "implementation": { + "contract": "@aragon/apps-finance/contracts/Finance.sol", + "address": "0x1a76ED38B14C768e02b96A879d89Db18AC83EC53", + "constructorArgs": [] + }, + "aragonApp": { + "name": "aragon-finance", + "fullName": "aragon-finance.lidopm.eth", + "id": "0x5c9918c99c4081ca9459c178381be71d9da40e49e151687da55099c49a4237f1" + }, + "proxy": { + "address": "0xf0F281E5d7FBc54EAFcE0dA225CDbde04173AB16", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol", + "constructorArgs": [ + "0x3b03f75Ec541Ca11a223bB58621A3146246E1644", + "0x5c9918c99c4081ca9459c178381be71d9da40e49e151687da55099c49a4237f1", + "0x1798de81000000000000000000000000e92329ec7ddb11d25e25b3c21eebf11f15eb325d0000000000000000000000000000000000000000000000000000000000278d00" + ] + } + }, + "app:aragon-token-manager": { + "implementation": { + "contract": "@aragon/apps-lido/apps/token-manager/contracts/TokenManager.sol", + "address": "0x6f0b994E6827faC1fDb58AF66f365676247bAD71", + "constructorArgs": [] + }, + "aragonApp": { + "name": "aragon-token-manager", + "fullName": "aragon-token-manager.lidopm.eth", + "id": "0xcd567bdf93dd0f6acc3bc7f2155f83244d56a65abbfbefb763e015420102c67b" + }, + "proxy": { + "address": "0xFaa1692c6eea8eeF534e7819749aD93a1420379A", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol", + "constructorArgs": [ + "0x3b03f75Ec541Ca11a223bB58621A3146246E1644", + "0xcd567bdf93dd0f6acc3bc7f2155f83244d56a65abbfbefb763e015420102c67b", + "0x" + ] + } + }, + "app:aragon-voting": { + "implementation": { + "contract": "@aragon/apps-lido/apps/voting/contracts/Voting.sol", + "address": "0x994c92228803e8b2D0fb8a610AbCB47412EeF8eF", + "constructorArgs": [] + }, + "aragonApp": { + "name": "aragon-voting", + "fullName": "aragon-voting.lidopm.eth", + "id": "0x0abcd104777321a82b010357f20887d61247493d89d2e987ff57bcecbde00e1e" + }, + "proxy": { + "address": "0xdA7d2573Df555002503F29aA4003e398d28cc00f", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol", + "constructorArgs": [ + "0x3b03f75Ec541Ca11a223bB58621A3146246E1644", + "0x0abcd104777321a82b010357f20887d61247493d89d2e987ff57bcecbde00e1e", + "0x13e0945300000000000000000000000014ae7daeecdf57034f3e9db8564e46dba8d9734400000000000000000000000000000000000000000000000006f05b59d3b2000000000000000000000000000000000000000000000000000000b1a2bc2ec500000000000000000000000000000000000000000000000000000000000000000384000000000000000000000000000000000000000000000000000000000000012c" + ] + } + }, + "app:lido": { + "implementation": { + "contract": "contracts/0.4.24/Lido.sol", + "address": "0x59034815464d18134A55EED3702b535D8A32c52b", + "constructorArgs": [] + }, + "aragonApp": { + "name": "lido", + "fullName": "lido.lidopm.eth", + "id": "0x3ca7c3e38968823ccb4c78ea688df41356f182ae1d159e4ee608d30d68cef320" + }, + "proxy": { + "address": "0x3F1c547b21f65e10480dE3ad8E19fAAC46C95034", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol", + "constructorArgs": [ + "0x3b03f75Ec541Ca11a223bB58621A3146246E1644", + "0x3ca7c3e38968823ccb4c78ea688df41356f182ae1d159e4ee608d30d68cef320", + "0x" + ] + } + }, + "app:node-operators-registry": { + "implementation": { + "contract": "contracts/0.4.24/nos/NodeOperatorsRegistry.sol", + "address": "0xE0270CF2564d81E02284e16539F59C1B5a4718fE", + "constructorArgs": [] + }, + "aragonApp": { + "name": "node-operators-registry", + "fullName": "node-operators-registry.lidopm.eth", + "id": "0x7071f283424072341f856ac9e947e7ec0eb68719f757a7e785979b6b8717579d" + }, + "proxy": { + "address": "0x595F64Ddc3856a3b5Ff4f4CC1d1fb4B46cFd2bAC", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol", + "constructorArgs": [ + "0x3b03f75Ec541Ca11a223bB58621A3146246E1644", + "0x7071f283424072341f856ac9e947e7ec0eb68719f757a7e785979b6b8717579d", + "0x" + ] + } + }, + "app:oracle": { + "implementation": { + "contract": "contracts/0.4.24/oracle/LegacyOracle.sol", + "address": "0xcE4B3D5bd6259F5dD73253c51b17e5a87bb9Ee64", + "constructorArgs": [] + }, + "aragonApp": { + "name": "oracle", + "fullName": "oracle.lidopm.eth", + "id": "0x8b47ba2a8454ec799cd91646e7ec47168e91fd139b23f017455f3e5898aaba93" + }, + "proxy": { + "address": "0x072f72BE3AcFE2c52715829F2CD9061A6C8fF019", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol", + "constructorArgs": [ + "0x3b03f75Ec541Ca11a223bB58621A3146246E1644", + "0x8b47ba2a8454ec799cd91646e7ec47168e91fd139b23f017455f3e5898aaba93", + "0x" + ] + } + }, + "app:simple-dvt": { + "stakingRouterModuleParams": { + "moduleName": "SimpleDVT", + "moduleType": "simple-dvt-onchain-v1", + "targetShare": 50, + "moduleFee": 800, + "treasuryFee": 200, + "penaltyDelay": 86400, + "easyTrackAddress": "0x1763b9ED3586B08AE796c7787811a2E1bc16163a", + "easyTrackEVMScriptExecutor": "0x2819B65021E13CEEB9AC33E77DB32c7e64e7520D", + "easyTrackFactories": { + "AddNodeOperators": "0xC20129f1dd4DFeD023a6d6A8de9d54A7b61af5CC", + "ActivateNodeOperators": "0x08c48Fef9Cadca882E27d2325D1785858D5c1aE3", + "DeactivateNodeOperators": "0xf5436129Cf9d8fa2a1cb6e591347155276550635", + "SetNodeOperatorNames": "0xb6a31141A579FCB540E3BB3504C58F1e6F5f543a", + "SetNodeOperatorRewardAddresses": "0x7F9c5b838510e06b85DD146e71553EB7890fAf2e", + "UpdateTargetValidatorLimits": "0x6e570D487aE5729Bd982A7bb3a7bfA5213AeAEdE", + "SetVettedValidatorsLimits": "0xD420d6C8aA81c087829A64Ce59936b7C1176A81a", + "TransferNodeOperatorManager": "0xaa49cF620e3f80Ce72D3A7668b1b4f3dF370D2C7" + } + }, + "aragonApp": { + "name": "simple-dvt", + "fullName": "simple-dvt.lidopm.eth", + "id": "0xe1635b63b5f7b5e545f2a637558a4029dea7905361a2f0fc28c66e9136cf86a4" + }, + "proxy": { + "address": "0x11a93807078f8BB880c1BD0ee4C387537de4b4b6", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol", + "constructorArgs": [ + "0x3b03f75Ec541Ca11a223bB58621A3146246E1644", + "0xe1635b63b5f7b5e545f2a637558a4029dea7905361a2f0fc28c66e9136cf86a4", + "0x" + ] + } + }, + "aragon-acl": { + "implementation": { + "contract": "@aragon/os/contracts/acl/ACL.sol", + "address": "0xF1A087E055EA1C11ec3B540795Bd1A544e6dcbe9", + "constructorArgs": [] + }, + "proxy": { + "address": "0xfd1E42595CeC3E83239bf8dFc535250e7F48E0bC", + "constructorArgs": [ + "0x3b03f75Ec541Ca11a223bB58621A3146246E1644", + "0xe3262375f45a6e2026b7e7b18c2b807434f2508fe1a2a3dfb493c7df8f4aad6a", + "0x00" + ], + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol" + }, + "aragonApp": { + "name": "aragon-acl", + "id": "0xe3262375f45a6e2026b7e7b18c2b807434f2508fe1a2a3dfb493c7df8f4aad6a" + } + }, + "aragon-apm-registry": { + "implementation": { + "contract": "@aragon/os/contracts/apm/APMRegistry.sol", + "address": "0x3EcF7190312F50043DB0494bA0389135Fc3833F3", + "constructorArgs": [ + "0x3b03f75Ec541Ca11a223bB58621A3146246E1644", + "0x9089af016eb74d66811e1c39c1eef86fdcdb84b5665a4884ebf62339c2613991", + "0x00" + ] + }, + "proxy": { + "address": "0xB576A85c310CC7Af5C106ab26d2942fA3a5ea94A", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol" + }, + "factory": { + "address": "0x54eF0022cc769344D0cBCeF12e051281cCBb9fad", + "contract": "@aragon/os/contracts/factory/APMRegistryFactory.sol", + "constructorArgs": [ + "0xB33f9AE6C34D8cC59A48fd9973C64488f00fa64F", + "0x3EcF7190312F50043DB0494bA0389135Fc3833F3", + "0x8959360c48D601a6817BAf2449E5D00cC543FA3A", + "0x7B133ACab5Cec7B90FB13CCf68d6568f8A051EcE", + "0x4327d1Fc6E5fa0326CCAE737F67C066c50BcC258", + "0x0000000000000000000000000000000000000000" + ] + } + }, + "aragon-app-repo-agent": { + "implementation": { + "contract": "@aragon/os/contracts/apm/Repo.sol", + "address": "0x8959360c48D601a6817BAf2449E5D00cC543FA3A", + "constructorArgs": [] + }, + "proxy": { + "address": "0xe7b4567913AaF2bD54A26E742cec22727D8109eA", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol", + "constructorArgs": [] + } + }, + "aragon-app-repo-finance": { + "implementation": { + "contract": "@aragon/os/contracts/apm/Repo.sol", + "address": "0x8959360c48D601a6817BAf2449E5D00cC543FA3A", + "constructorArgs": [] + }, + "proxy": { + "address": "0x0df65b7c78Dc42a872010d031D3601C284D8fE71", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol", + "constructorArgs": [] + } + }, + "aragon-app-repo-lido": { + "implementation": { + "contract": "@aragon/os/contracts/apm/Repo.sol", + "address": "0x8959360c48D601a6817BAf2449E5D00cC543FA3A", + "constructorArgs": [] + }, + "proxy": { + "address": "0xA37fb4C41e7D30af5172618a863BBB0f9042c604", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol", + "constructorArgs": [] + } + }, + "aragon-app-repo-node-operators-registry": { + "implementation": { + "contract": "@aragon/os/contracts/apm/Repo.sol", + "address": "0x8959360c48D601a6817BAf2449E5D00cC543FA3A", + "constructorArgs": [] + }, + "proxy": { + "address": "0x4E8970d148CB38460bE9b6ddaab20aE2A74879AF", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol", + "constructorArgs": [] + } + }, + "aragon-app-repo-oracle": { + "implementation": { + "contract": "@aragon/os/contracts/apm/Repo.sol", + "address": "0x8959360c48D601a6817BAf2449E5D00cC543FA3A", + "constructorArgs": [] + }, + "proxy": { + "address": "0xB3d74c319C0C792522705fFD3097f873eEc71764", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol", + "constructorArgs": [] + } + }, + "aragon-app-repo-token-manager": { + "implementation": { + "contract": "@aragon/os/contracts/apm/Repo.sol", + "address": "0x8959360c48D601a6817BAf2449E5D00cC543FA3A", + "constructorArgs": [] + }, + "proxy": { + "address": "0xD327b4Fb87fa01599DaD491Aa63B333c44C74472", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol", + "constructorArgs": [] + } + }, + "aragon-app-repo-voting": { + "implementation": { + "contract": "@aragon/os/contracts/apm/Repo.sol", + "address": "0x8959360c48D601a6817BAf2449E5D00cC543FA3A", + "constructorArgs": [] + }, + "proxy": { + "address": "0x2997EA0D07D79038D83Cb04b3BB9A2Bc512E3fDA", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol", + "constructorArgs": [] + } + }, + "aragon-evm-script-registry": { + "proxy": { + "address": "0xE1200ae048163B67D69Bc0492bF5FddC3a2899C0", + "constructorArgs": [ + "0x3b03f75Ec541Ca11a223bB58621A3146246E1644", + "0xddbcfd564f642ab5627cf68b9b7d374fb4f8a36e941a75d89c87998cef03bd61", + "0x8129fc1c" + ], + "contract": "@aragon/os/contracts/apps/AppProxyPinned.sol" + }, + "aragonApp": { + "name": "aragon-evm-script-registry", + "id": "0xddbcfd564f642ab5627cf68b9b7d374fb4f8a36e941a75d89c87998cef03bd61" + }, + "implementation": { + "address": "0x923B9Cab88E4a1d3de7EE921dEFBF9e2AC6e0791", + "contract": "@aragon/os/contracts/evmscript/EVMScriptRegistry.sol", + "constructorArgs": [] + } + }, + "aragon-kernel": { + "implementation": { + "contract": "@aragon/os/contracts/kernel/Kernel.sol", + "address": "0x34c0cbf9836FD945423bD3d2d72880da9d068E5F", + "constructorArgs": [ + true + ] + }, + "proxy": { + "address": "0x3b03f75Ec541Ca11a223bB58621A3146246E1644", + "contract": "@aragon/os/contracts/kernel/KernelProxy.sol", + "constructorArgs": [ + "0x34c0cbf9836FD945423bD3d2d72880da9d068E5F" + ] + } + }, + "aragonEnsLabelName": "aragonpm", + "aragonEnsNode": "0x9065c3e7f7b7ef1ef4e53d2d0b8e0cef02874ab020c1ece79d5f0d3d0111c0ba", + "aragonEnsNodeName": "aragonpm.eth", + "aragonIDAddress": "0xCA01225e211AB0c6EFCD3aCc64D85465e4D8ab53", + "aragonIDConstructorArgs": [ + "0x4327d1Fc6E5fa0326CCAE737F67C066c50BcC258", + "0x2B725cBA5F75c3B61bb5E37454a7090fb11c757E", + "0x7e74a86b6e146964fb965db04dc2590516da77f720bb6759337bf5632415fd86" + ], + "aragonIDEnsNode": "0x7e74a86b6e146964fb965db04dc2590516da77f720bb6759337bf5632415fd86", + "aragonIDEnsNodeName": "aragonid.eth", + "burner": { + "deployParameters": { + "totalCoverSharesBurnt": "0", + "totalNonCoverSharesBurnt": "0" + }, + "contract": "contracts/0.8.9/Burner.sol", + "address": "0x4E46BD7147ccf666E1d73A3A456fC7a68de82eCA", + "constructorArgs": [ + "0x22896Bfc68814BFD855b1a167255eE497006e730", + "0xE92329EC7ddB11D25e25b3c21eeBf11f15eB325d", + "0x3F1c547b21f65e10480dE3ad8E19fAAC46C95034", + "0", + "0" + ] + }, + "callsScript": { + "address": "0xAa8B4F258a4817bfb0058b861447878168ddf7B0", + "contract": "@aragon/os/contracts/evmscript/executors/CallsScript.sol", + "constructorArgs": [] + }, + "chainId": 17000, + "chainSpec": { + "slotsPerEpoch": 32, + "secondsPerSlot": 12, + "genesisTime": 1695902400, + "depositContract": "0x4242424242424242424242424242424242424242" + }, + "createAppReposTx": "0xd8a9b10e16b5e75b984c90154a9cb51fbb06bf560a3c424e2e7ad81951008502", + "daoAragonId": "lido-dao", + "daoFactoryAddress": "0xB33f9AE6C34D8cC59A48fd9973C64488f00fa64F", + "daoFactoryConstructorArgs": [ + "0x34c0cbf9836FD945423bD3d2d72880da9d068E5F", + "0xF1A087E055EA1C11ec3B540795Bd1A544e6dcbe9", + "0x11E7591F83360d0Bc238c8AB9e50B6D2B7566aDc" + ], + "daoInitialSettings": { + "voting": { + "minSupportRequired": "500000000000000000", + "minAcceptanceQuorum": "50000000000000000", + "voteDuration": 900, + "objectionPhaseDuration": 300 + }, + "fee": { + "totalPercent": 10, + "treasuryPercent": 50, + "nodeOperatorsPercent": 50 + }, + "token": { + "name": "TEST Lido DAO Token", + "symbol": "TLDO" + } + }, + "deployCommit": "eda16728a7c80f1bb55c3b91c668aae190a1efb0", + "deployer": "0x22896Bfc68814BFD855b1a167255eE497006e730", + "depositSecurityModule": { + "deployParameters": { + "maxDepositsPerBlock": 150, + "minDepositBlockDistance": 5, + "pauseIntentValidityPeriodBlocks": 6646 + }, + "contract": "contracts/0.8.9/DepositSecurityModule.sol", + "address": "0x045dd46212A178428c088573A7d102B9d89a022A", + "constructorArgs": [ + "0x3F1c547b21f65e10480dE3ad8E19fAAC46C95034", + "0x4242424242424242424242424242424242424242", + "0xd6EbF043D30A7fe46D1Db32BA90a0A51207FE229", + 150, + 5, + 6646 + ] + }, + "dummyEmptyContract": { + "contract": "contracts/0.8.9/test_helpers/DummyEmptyContract.sol", + "address": "0x5F4FEf09Cbd5ad743632Fb869E80294933473f0B", + "constructorArgs": [] + }, + "eip712StETH": { + "contract": "contracts/0.8.9/EIP712StETH.sol", + "address": "0xE154732c5Eab277fd88a9fF6Bdff7805eD97BCB1", + "constructorArgs": [ + "0x3F1c547b21f65e10480dE3ad8E19fAAC46C95034" + ] + }, + "ensAddress": "0x4327d1Fc6E5fa0326CCAE737F67C066c50BcC258", + "ensFactoryAddress": "0xADba3e3122F2Da8F7B07723a3e1F1cEDe3fe8d7d", + "ensFactoryConstructorArgs": [], + "ensSubdomainRegistrarBaseAddress": "0x7B133ACab5Cec7B90FB13CCf68d6568f8A051EcE", + "evmScriptRegistryFactoryAddress": "0x11E7591F83360d0Bc238c8AB9e50B6D2B7566aDc", + "evmScriptRegistryFactoryConstructorArgs": [], + "executionLayerRewardsVault": { + "contract": "contracts/0.8.9/LidoExecutionLayerRewardsVault.sol", + "address": "0xE73a3602b99f1f913e72F8bdcBC235e206794Ac8", + "constructorArgs": [ + "0x3F1c547b21f65e10480dE3ad8E19fAAC46C95034", + "0xE92329EC7ddB11D25e25b3c21eeBf11f15eB325d" + ] + }, + "gateSeal": { + "factoryAddress": "0x1134F7077055b0B3559BE52AfeF9aA22A0E1eEC2", + "sealDuration": 518400, + "expiryTimestamp": 1714521600, + "sealingCommittee": "0xCD1f9954330AF39a74Fd6e7B25781B4c24ee373f", + "address": "0x7f6FA688d4C12a2d51936680b241f3B0F0F9ca60" + }, + "hashConsensusForAccountingOracle": { + "deployParameters": { + "fastLaneLengthSlots": 10, + "epochsPerFrame": 12 + }, + "contract": "contracts/0.8.9/oracle/HashConsensus.sol", + "address": "0xa067FC95c22D51c3bC35fd4BE37414Ee8cc890d2", + "constructorArgs": [ + 32, + 12, + 1695902400, + 12, + 10, + "0x22896Bfc68814BFD855b1a167255eE497006e730", + "0x4E97A3972ce8511D87F334dA17a2C332542a5246" + ] + }, + "hashConsensusForValidatorsExitBusOracle": { + "deployParameters": { + "fastLaneLengthSlots": 10, + "epochsPerFrame": 4 + }, + "contract": "contracts/0.8.9/oracle/HashConsensus.sol", + "address": "0xe77Cf1A027d7C10Ee6bb7Ede5E922a181FF40E8f", + "constructorArgs": [ + 32, + 12, + 1695902400, + 4, + 10, + "0x22896Bfc68814BFD855b1a167255eE497006e730", + "0xffDDF7025410412deaa05E3E1cE68FE53208afcb" + ] + }, + "ldo": { + "address": "0x14ae7daeecdf57034f3E9db8564e46Dba8D97344", + "contract": "@aragon/minime/contracts/MiniMeToken.sol", + "constructorArgs": [ + "0x15ef666c9620C0f606Ba35De2aF668fe987E26ae", + "0x0000000000000000000000000000000000000000", + 0, + "TEST Lido DAO Token", + 18, + "TLDO", + true + ] + }, + "legacyOracle": { + "deployParameters": { + "lastCompletedEpochId": 0 + } + }, + "lidoApm": { + "deployArguments": [ + "0x93cdeb708b7545dc668eb9280176169d1c33cfd8ed6f04690a0bcc88a93fc4ae", + "0x90a9580abeb24937fc658e497221c81ce8553b560304f9525821f32b17dbdaec" + ], + "deployTx": "0x2fac1c172a250736c34d16d3a721d2916abac0de0dea67d79955346a1f4345a2", + "address": "0x4605Dc9dC4BD0442F850eB8226B94Dd0e27C3Ce7" + }, + "lidoApmEnsName": "lidopm.eth", + "lidoApmEnsRegDurationSec": 94608000, + "lidoLocator": { + "proxy": { + "contract": "contracts/0.8.9/proxy/OssifiableProxy.sol", + "address": "0x28FAB2059C713A7F9D8c86Db49f9bb0e96Af1ef8", + "constructorArgs": [ + "0x5F4FEf09Cbd5ad743632Fb869E80294933473f0B", + "0x22896Bfc68814BFD855b1a167255eE497006e730", + "0x" + ] + }, + "implementation": { + "contract": "contracts/0.8.9/LidoLocator.sol", + "address": "0xDba5Ad530425bb1b14EECD76F1b4a517780de537", + "constructorArgs": [ + [ + "0x4E97A3972ce8511D87F334dA17a2C332542a5246", + "0x045dd46212A178428c088573A7d102B9d89a022A", + "0xE73a3602b99f1f913e72F8bdcBC235e206794Ac8", + "0x072f72BE3AcFE2c52715829F2CD9061A6C8fF019", + "0x3F1c547b21f65e10480dE3ad8E19fAAC46C95034", + "0xF0d576c7d934bBeCc68FE15F1c5DAF98ea2B78bb", + "0x072f72BE3AcFE2c52715829F2CD9061A6C8fF019", + "0x4E46BD7147ccf666E1d73A3A456fC7a68de82eCA", + "0xd6EbF043D30A7fe46D1Db32BA90a0A51207FE229", + "0xE92329EC7ddB11D25e25b3c21eeBf11f15eB325d", + "0xffDDF7025410412deaa05E3E1cE68FE53208afcb", + "0xc7cc160b58F8Bb0baC94b80847E2CF2800565C50", + "0xF0179dEC45a37423EAD4FaD5fCb136197872EAd9", + "0xC01fC1F2787687Bc656EAc0356ba9Db6e6b7afb7" + ] + ] + } + }, + "lidoTemplate": { + "contract": "contracts/0.4.24/template/LidoTemplate.sol", + "address": "0x0e065Dd0Bc85Ca53cfDAf8D9ed905e692260De2E", + "constructorArgs": [ + "0x22896Bfc68814BFD855b1a167255eE497006e730", + "0xB33f9AE6C34D8cC59A48fd9973C64488f00fa64F", + "0x4327d1Fc6E5fa0326CCAE737F67C066c50BcC258", + "0x15ef666c9620C0f606Ba35De2aF668fe987E26ae", + "0xCA01225e211AB0c6EFCD3aCc64D85465e4D8ab53", + "0x54eF0022cc769344D0cBCeF12e051281cCBb9fad" + ], + "deployBlock": 30581 + }, + "lidoTemplateCreateStdAppReposTx": "0x3f5b8918667bd3e971606a54a907798720158587df355a54ce07c0d0f9750d3c", + "lidoTemplateNewDaoTx": "0x3346246f09f91ffbc260b6c300b11ababce9f5ca54d7880a277860961f343112", + "miniMeTokenFactoryAddress": "0x15ef666c9620C0f606Ba35De2aF668fe987E26ae", + "miniMeTokenFactoryConstructorArgs": [], + "networkId": 17000, + "newDaoTx": "0x3346246f09f91ffbc260b6c300b11ababce9f5ca54d7880a277860961f343112", + "nodeOperatorsRegistry": { + "deployParameters": { + "stakingModuleTypeId": "curated-onchain-v1", + "stuckPenaltyDelay": 172800 + } + }, + "oracleDaemonConfig": { + "contract": "contracts/0.8.9/OracleDaemonConfig.sol", + "address": "0xC01fC1F2787687Bc656EAc0356ba9Db6e6b7afb7", + "constructorArgs": [ + "0x22896Bfc68814BFD855b1a167255eE497006e730", + [] + ], + "deployParameters": { + "NORMALIZED_CL_REWARD_PER_EPOCH": 64, + "NORMALIZED_CL_REWARD_MISTAKE_RATE_BP": 1000, + "REBASE_CHECK_NEAREST_EPOCH_DISTANCE": 1, + "REBASE_CHECK_DISTANT_EPOCH_DISTANCE": 23, + "VALIDATOR_DELAYED_TIMEOUT_IN_SLOTS": 7200, + "VALIDATOR_DELINQUENT_TIMEOUT_IN_SLOTS": 28800, + "NODE_OPERATOR_NETWORK_PENETRATION_THRESHOLD_BP": 100, + "PREDICTION_DURATION_IN_SLOTS": 50400, + "FINALIZATION_MAX_NEGATIVE_REBASE_EPOCH_SHIFT": 1350 + } + }, + "oracleReportSanityChecker": { + "deployParameters": { + "churnValidatorsPerDayLimit": 1500, + "oneOffCLBalanceDecreaseBPLimit": 500, + "annualBalanceIncreaseBPLimit": 1000, + "simulatedShareRateDeviationBPLimit": 250, + "maxValidatorExitRequestsPerReport": 2000, + "maxAccountingExtraDataListItemsCount": 100, + "maxNodeOperatorsPerExtraDataItemCount": 100, + "requestTimestampMargin": 128, + "maxPositiveTokenRebase": 5000000 + }, + "contract": "contracts/0.8.9/sanity_checks/OracleReportSanityChecker.sol", + "address": "0xF0d576c7d934bBeCc68FE15F1c5DAF98ea2B78bb", + "constructorArgs": [ + "0x28FAB2059C713A7F9D8c86Db49f9bb0e96Af1ef8", + "0x22896Bfc68814BFD855b1a167255eE497006e730", + [ + 1500, + 500, + 1000, + 250, + 2000, + 100, + 100, + 128, + 5000000 + ], + [ + [], + [], + [], + [], + [], + [], + [], + [], + [], + [] + ] + ] + }, + "stakingRouter": { + "proxy": { + "contract": "contracts/0.8.9/proxy/OssifiableProxy.sol", + "address": "0xd6EbF043D30A7fe46D1Db32BA90a0A51207FE229", + "constructorArgs": [ + "0x32f236423928c2c138F46351D9E5FD26331B1aa4", + "0x22896Bfc68814BFD855b1a167255eE497006e730", + "0x" + ] + }, + "implementation": { + "contract": "contracts/0.8.9/StakingRouter.sol", + "address": "0x32f236423928c2c138F46351D9E5FD26331B1aa4", + "constructorArgs": [ + "0x4242424242424242424242424242424242424242" + ] + } + }, + "validatorsExitBusOracle": { + "deployParameters": { + "consensusVersion": 1 + }, + "proxy": { + "contract": "contracts/0.8.9/proxy/OssifiableProxy.sol", + "address": "0xffDDF7025410412deaa05E3E1cE68FE53208afcb", + "constructorArgs": [ + "0x210f60EC8A4D020b3e22f15fee2d2364e9b22357", + "0x22896Bfc68814BFD855b1a167255eE497006e730", + "0x" + ] + }, + "implementation": { + "contract": "contracts/0.8.9/oracle/ValidatorsExitBusOracle.sol", + "address": "0x210f60EC8A4D020b3e22f15fee2d2364e9b22357", + "constructorArgs": [ + 12, + 1695902400, + "0x28FAB2059C713A7F9D8c86Db49f9bb0e96Af1ef8" + ] + } + }, + "vestingParams": { + "unvestedTokensAmount": "0", + "holders": { + "0xCD1f9954330AF39a74Fd6e7B25781B4c24ee373f": "880000000000000000000000", + "0xaa6bfBCD634EE744CB8FE522b29ADD23124593D3": "60000000000000000000000", + "0xBA59A84C6440E8cccfdb5448877E26F1A431Fc8B": "60000000000000000000000" + }, + "start": 0, + "cliff": 0, + "end": 0, + "revokable": false + }, + "withdrawalQueueERC721": { + "deployParameters": { + "name": "stETH Withdrawal NFT", + "symbol": "unstETH" + }, + "proxy": { + "contract": "contracts/0.8.9/proxy/OssifiableProxy.sol", + "address": "0xc7cc160b58F8Bb0baC94b80847E2CF2800565C50", + "constructorArgs": [ + "0xFF72B5cdc701E9eE677966B2702c766c38F412a4", + "0x22896Bfc68814BFD855b1a167255eE497006e730", + "0x" + ] + }, + "implementation": { + "contract": "contracts/0.8.9/WithdrawalQueueERC721.sol", + "address": "0xFF72B5cdc701E9eE677966B2702c766c38F412a4", + "constructorArgs": [ + "0x8d09a4502Cc8Cf1547aD300E066060D043f6982D", + "stETH Withdrawal NFT", + "unstETH" + ] + } + }, + "withdrawalVault": { + "implementation": { + "contract": "contracts/0.8.9/WithdrawalVault.sol", + "address": "0xd517d9d04DA9B47dA23df91261bd3bF435BE964A", + "constructorArgs": [ + "0x3F1c547b21f65e10480dE3ad8E19fAAC46C95034", + "0xE92329EC7ddB11D25e25b3c21eeBf11f15eB325d" + ] + }, + "proxy": { + "contract": "contracts/0.8.4/WithdrawalsManagerProxy.sol", + "address": "0xF0179dEC45a37423EAD4FaD5fCb136197872EAd9", + "constructorArgs": [ + "0xdA7d2573Df555002503F29aA4003e398d28cc00f", + "0xd517d9d04DA9B47dA23df91261bd3bF435BE964A" + ] + } + }, + "wstETH": { + "contract": "contracts/0.6.12/WstETH.sol", + "address": "0x8d09a4502Cc8Cf1547aD300E066060D043f6982D", + "constructorArgs": [ + "0x3F1c547b21f65e10480dE3ad8E19fAAC46C95034" + ] + } +} \ No newline at end of file diff --git a/deployed-local-defaults.json b/deployed-local-defaults.json deleted file mode 100644 index cf2cf5baf..000000000 --- a/deployed-local-defaults.json +++ /dev/null @@ -1,59 +0,0 @@ -{ - "networkId": 1337, - "ipfsAPI": "http://127.0.0.1:5001/api/v0", - "multisigAddress": "0xb4124cEB3451635DAcedd11767f004d8a28c6eE7", - "lidoApmEnsName": "lidopm.eth", - "lidoApmEnsRegDurationSec": 94608000, - "daoAragonId": "lido-dao", - "daoInitialSettings": { - "voting": { - "minSupportRequired": "500000000000000000", - "minAcceptanceQuorum": "50000000000000000", - "voteDuration": 30, - "objectionPhaseDuration": 1 - }, - "beaconSpec": { - "depositContractAddress": "", - "epochsPerFrame": 225, - "slotsPerEpoch": 32, - "secondsPerSlot": 12, - "genesisTime": 1639659600 - }, - "fee": { - "totalPercent": 10, - "treasuryPercent": 0, - "insurancePercent": 50, - "nodeOperatorsPercent": 50 - }, - "token": { - "name": "TEST Lido DAO Token", - "symbol": "TLDO" - } - }, - "vestingParams": { - "unvestedTokensAmount": "230000000000000000000000", - "holders": { - "0xb4124cEB3451635DAcedd11767f004d8a28c6eE7": "530000000000000000000000", - "0x8401Eb5ff34cc943f096A32EF3d5113FEbE8D4Eb": "15000000000000000000000", - "0x306469457266CBBe7c0505e8Aad358622235e768": "15000000000000000000000", - "0xd873F6DC68e3057e4B7da74c6b304d0eF0B484C7": "15000000000000000000000", - "0xDcC5dD922fb1D0fd0c450a0636a8cE827521f0eD": "15000000000000000000000", - "0x27E9727FD9b8CdDdd0854F56712AD9DF647FaB74": "15000000000000000000000", - "0x9766D2e7FFde358AD0A40BB87c4B88D9FAC3F4dd": "15000000000000000000000", - "0xBd7055AB500cD1b0b0B14c82BdBe83ADCc2e8D06": "15000000000000000000000", - "0xe8898A4E589457D979Da4d1BDc35eC2aaf5a3f8E": "15000000000000000000000" - }, - "start": 1639659600, - "cliff": 1639660100, - "end": 1639660100, - "revokable": false - }, - "selfOwnedStETHBurnerParams": { - "totalCoverSharesBurnt": "0", - "totalNonCoverSharesBurnt": "0", - "maxBurnAmountPerRunBasisPoints": "4" - }, - "executionLayerRewardsParams": { - "withdrawalLimit": "1" - } -} diff --git a/deployed-mainnet.json b/deployed-mainnet.json index 5d6d3bc3a..25007ce27 100644 --- a/deployed-mainnet.json +++ b/deployed-mainnet.json @@ -1,119 +1,95 @@ { - "deployCommit": "e45c4d6fb8120fd29426b8d969c19d8a798ca974", - "networkId": 1, - "ipfsAPI": "https://ipfs.infura.io:5001/api/v0", - "lidoApmEnsName": "lidopm.eth", - "lidoApmEnsRegDurationSec": 94608000, - "daoAragonId": "lido-dao", - "ensAddress": "0x00000000000C2E074eC69A0dFb2997BA6C7d2e1e", - "daoFactoryAddress": "0x7378ad1ba8f3c8e64bbb2a04473edd35846360f1", + "accountingOracle": { + "deployParameters": { + "consensusVersion": 1 + }, + "proxy": { + "address": "0x852deD011285fe67063a08005c71a85690503Cee", + "contract": "contracts/0.8.9/proxy/OssifiableProxy.sol", + "deployTx": "0x3def88f27741216b131de2861cf89af2ca2ac4242b384ee33dca8cc70c51c8dd", + "constructorArgs": [ + "0x6F6541C2203196fEeDd14CD2C09550dA1CbEDa31", + "0x8Ea83AD72396f1E0cD2f8E72b1461db8Eb6aF7B5", + "0x" + ] + }, + "implementation": { + "address": "0xF3c5E0A67f32CF1dc07a8817590efa102079a1aF", + "contract": "contracts/0.8.9/oracle/AccountingOracle.sol", + "deployTx": "0x3e27627d3ed236aff8901df187196e9682187dfd0d259c5d5811a6e923436083", + "constructorArgs": [ + "0xC1d0b3DE6792Bf6b4b37EccdcC24e45978Cfd2Eb", + "0xae7ab96520DE3A18E5e111B5EaAb095312D7fE84", + "0x442af784A788A5bd6F42A01Ebe9F287a871243fb", + 12, + 1606824023 + ] + } + }, "apmRegistryFactoryAddress": "0xa0BC4B67F5FacDE4E50EAFF48691Cfc43F4E280A", - "miniMeTokenFactoryAddress": "0x909d05f384d0663ed4be59863815ab43b4f347ec", - "aragonIDAddress": "0x546aa2eae2514494eeadb7bbb35243348983c59d", - "multisigAddress": "0x55Bc991b2edF3DDb4c520B222bE4F378418ff0fA", - "daoInitialSettings": { - "token": { - "name": "Lido DAO Token", - "symbol": "LDO" + "app:aragon-agent": { + "implementation": { + "contract": "@aragon/apps-agent/contracts/Agent.sol", + "address": "0x3A93C17FC82CC33420d1809dDA9Fb715cc89dd37", + "constructorArgs": [] }, - "beaconSpec": { - "depositContractAddress": "0x00000000219ab540356cBB839Cbe05303d7705Fa", - "slotsPerEpoch": 32, - "secondsPerSlot": 12, - "genesisTime": 1606824023 + "aragonApp": { + "name": "aragon-agent", + "fullName": "aragon-agent.lidopm.eth", + "id": "0x701a4fd1f5174d12a0f1d9ad2c88d0ad11ab6aad0ac72b7d9ce621815f8016a9" }, - "voting": { - "minSupportRequired": "500000000000000000", - "minAcceptanceQuorum": "50000000000000000", - "voteDuration": 86400 + "proxy": { + "address": "0x3e40D73EB977Dc6a537aF587D48316feE66E9C8c", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol" + } + }, + "app:aragon-finance": { + "implementation": { + "contract": "@aragon/apps-finance/contracts/Finance.sol", + "address": "0x836835289A2E81B66AE5d95b7c8dBC0480dCf9da", + "constructorArgs": [] }, - "fee": { - "totalPercent": 10, - "treasuryPercent": 0, - "insurancePercent": 50, - "nodeOperatorsPercent": 50 + "aragonApp": { + "name": "aragon-finance", + "fullName": "aragon-finance.lidopm.eth", + "id": "0x5c9918c99c4081ca9459c178381be71d9da40e49e151687da55099c49a4237f1" + }, + "proxy": { + "address": "0xB9E5CBB9CA5b0d659238807E84D0176930753d86", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol" } }, - "vestingParams": { - "unvestedTokensAmount": "363197500000000000000000000", - "holders": { - "0x9Bb75183646e2A0DC855498bacD72b769AE6ceD3": "20000000000000000000000000", - "0x0f89D54B02ca570dE82F770D33c7B7Cf7b3C3394": "25000000000000000000000000", - "0xe49f68B9A01d437B0b7ea416376a7AB21532624e": "2282000000000000000000000", - "0xb842aFD82d940fF5D8F6EF3399572592EBF182B0": "17718000000000000000000000", - "0x9849c2C1B73B41AEE843A002C332a2d16aaaB611": "10000000000000000000000000", - "0x96481cb0fcd7673254ebccc42dce9b92da10ea04": "5000000000000000000000000", - "0xB3DFe140A77eC43006499CB8c2E5e31975caD909": "7500000000000000000000000", - "0x61C808D82A3Ac53231750daDc13c777b59310bD9": "20000000000000000000000000", - "0x447f95026107aaed7472A0470931e689f51e0e42": "20000000000000000000000000", - "0x6ae83EAB68b7112BaD5AfD72d6B24546AbFF137D": "2222222220000000000000000", - "0xC24da173A250e9Ca5c54870639EbE5f88be5102d": "17777777780000000000000000", - "0x1f3813fE7ace2a33585F1438215C7F42832FB7B3": "20000000000000000000000000", - "0x82a8439BA037f88bC73c4CCF55292e158A67f125": "7000000000000000000000000", - "0x91715128a71c9C734CDC20E5EdEEeA02E72e428E": "15000000000000000000000000", - "0xB5587A54fF7022AC218438720BDCD840a32f0481": "5000000000000000000000000", - "0xf5fb27b912d987b5b6e02a1b1be0c1f0740e2c6f": "2000000000000000000000000", - "0x8b1674a617F103897Fb82eC6b8EB749BA0b9765B": "15000000000000000000000000", - "0x48Acf41D10a063f9A6B718B9AAd2e2fF5B319Ca2": "5000000000000000000000000", - "0x7eE09c11D6Dc9684D6D5a4C6d333e5b9e336bb6C": "10000000000000000000000000", - "0x11099aC9Cc097d0C9759635b8e16c6a91ECC43dA": "2000000000000000000000000", - "0x3d4AD2333629eE478E4f522d60A56Ae1Db5D3Cdb": "5000000000000000000000000", - "0xd5eCB56c6ca8f8f52D2DB4dC1257d6161cf3Da29": "100000000000000000000000", - "0x7F5e13a815EC9b4466d283CD521eE9829e7F6f0e": "200000000000000000000000", - "0x2057cbf2332ab2697a52B8DbC85756535d577e32": "500000000000000000000000", - "0x537dfB5f599A3d15C50E2d9270e46b808A52559D": "1000000000000000000000000", - "0x33c4c38e96337172d3de39df82060de26b638c4b": "550000000000000000000000", - "0x6094E1Dd925caCe56Fa501dAEc02b01a49E55770": "300000000000000000000000", - "0x977911f476B28f9F5332fA500387deE81e480a44": "40000000000000000000000", - "0x66d3FdA643320c6DddFBba39e635288A5dF75FB9": "200000000000000000000000", - "0xDFC0ae54af992217100845597982274A26d8CB28": "12500000000000000000000", - "0x32254b28F793CC18B3575C86c61fE3D7421cbbef": "500000000000000000000000", - "0x0Bf5566fB5F1f9934a3944AEF128a1b1a8cF3f17": "50000000000000000000000", - "0x1d3Fa8bf35870271115B997b8eCFe18529422a16": "50000000000000000000000", - "0x366B9729C5A89EC4618A0AB95F832E411eaE8237": "200000000000000000000000", - "0x20921142A35c89bE5D002973d2D6B72d9a625FB0": "200000000000000000000000", - "0x663b91628674846e8D1CBB779EFc8202d86284E2": "7500000000000000000000000", - "0xa6829908f728C6bC5627E2aFe93a0B71E978892D": "300000000000000000000000", - "0x9575B7859DF77F2A0EF034339b80e24dE44AB3F6": "200000000000000000000000", - "0xEe217c23131C6F055F7943Ef1f80Bec99dF35244": "400000000000000000000000", - "0xadde043f556d1083f060A7298E79eaBa08A3a077": "400000000000000000000000", - "0xaFBEfC8401c885A0bb6Ea6Af43f592A015433C65": "200000000000000000000000", - "0x8a62A63b877877bd5B1209B9b67F3d2685284268": "200000000000000000000000", - "0x62Ac238Ac055017DEcAb645E7E56176749f316d0": "200000000000000000000000", - "0x55Bc991b2edF3DDb4c520B222bE4F378418ff0fA": "5000000000000000000000000", - "0x8D689476EB446a1FB0065bFFAc32398Ed7F89165": "10000000000000000000000000", - "0x083fc10cE7e97CaFBaE0fE332a9c4384c5f54E45": "5000000000000000000000000", - "0x0028E24e4Fe5184792Bd0Cf498C11AE5b76185f5": "5000000000000000000000000", - "0xFe45baf0F18c207152A807c1b05926583CFE2e4b": "5000000000000000000000000", - "0x4a7C6899cdcB379e284fBFD045462e751DA4C7cE": "5000000000000000000000000", - "0xD7f0dDcBb0117A53e9ff2cad814B8B810a314f59": "5000000000000000000000000", - "0xb8d83908AAB38a159F3dA47a59d84dB8e1838712": "50000000000000000000000000", - "0xA2dfC431297aee387C05bEEf507E5335E684FbCD": "50000000000000000000000000", - "0x1597D19659F3DE52ABd475F7D2314DCca29359BD": "50000000000000000000000000", - "0x695C388153bEa0fbE3e1C049c149bAD3bc917740": "50000000000000000000000000", - "0x945755dE7EAc99008c8C57bdA96772d50872168b": "50000000000000000000000000", - "0xFea88380bafF95e85305419eB97247981b1a8eEE": "30000000000000000000000000", - "0xAD4f7415407B83a081A0Bee22D05A8FDC18B42da": "50000000000000000000000000", - "0x68335B3ac272C8238b722963368F87dE736b64D6": "5000000000000000000000000", - "0xfA2Ab7C161Ef7F83194498f36ca7aFba90FD08d4": "5000000000000000000000000", - "0x58A764028350aB15899fDCcAFFfd3940e602CEEA": "10000000000000000000000000" + "app:aragon-token-manager": { + "implementation": { + "contract": "@aragon/apps-lido/apps/token-manager/contracts/TokenManager.sol", + "address": "0xde3A93028F2283cc28756B3674BD657eaFB992f4", + "constructorArgs": [] }, - "start": 1639785600, - "cliff": 1639785600, - "end": 1671321600, - "revokable": false + "aragonApp": { + "name": "aragon-token-manager", + "fullName": "aragon-token-manager.lidopm.eth", + "id": "0xcd567bdf93dd0f6acc3bc7f2155f83244d56a65abbfbefb763e015420102c67b" + }, + "proxy": { + "address": "0xf73a1260d222f447210581DDf212D915c09a3249", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol" + } }, - "daoTemplate": { - "address": "0x752350797CB92Ad3BF1295Faf904B27585e66BF5", - "deployTx": "0xdcd4ebe028aa3663a1fe8bbc92ae8489045e29d2a6ef5284083d9be5c3fa5f19", - "constructorArgs": [ - "0x55Bc991b2edF3DDb4c520B222bE4F378418ff0fA", - "0x7378ad1ba8f3c8e64bbb2a04473edd35846360f1", - "0x00000000000C2E074eC69A0dFb2997BA6C7d2e1e", - "0x909d05f384d0663ed4be59863815ab43b4f347ec", - "0x546aa2eae2514494eeadb7bbb35243348983c59d", - "0xa0BC4B67F5FacDE4E50EAFF48691Cfc43F4E280A" - ] + "app:aragon-voting": { + "implementation": { + "contract": "@aragon/apps-lido/apps/voting/contracts/Voting.sol", + "address": "0x72fb5253AD16307B9E773d2A78CaC58E309d5Ba4", + "constructorArgs": [] + }, + "aragonApp": { + "name": "aragon-voting", + "fullName": "aragon-voting.lidopm.eth", + "id": "0x0abcd104777321a82b010357f20887d61247493d89d2e987ff57bcecbde00e1e" + }, + "proxy": { + "address": "0x2e59A20f205bB85a89C53f1936454680651E618e", + "contract": "@aragon/os/contracts/apps/AppProxyUpgradeable.sol" + } }, "app:lido": { "proxy": { @@ -133,24 +109,6 @@ "contentURI": "0x697066733a516d516b4a4d7476753474794a76577250584a666a4c667954576e393539696179794e6a703759714e7a58377053" } }, - "app:oracle": { - "proxy": { - "address": "0x442af784A788A5bd6F42A01Ebe9F287a871243fb" - }, - "implementation": { - "address": "0xa29b819654cE6224A222bb5f586920105E2D7E0E", - "contract": "contracts/0.4.24/oracle/LegacyOracle.sol", - "deployTx": "0xe666e3ce409bb4c18e1016af0b9ed3495b20361a69f2856bccb9e67599795b6f", - "constructorArgs": [] - }, - "aragonApp": { - "fullName": "oracle.lidopm.eth", - "name": "oracle", - "id": "0x8b47ba2a8454ec799cd91646e7ec47168e91fd139b23f017455f3e5898aaba93", - "ipfsCid": "QmUMPfiEKq5Mxm8y2GYQPLujGaJiWz1tvep5W7EdAGgCR8", - "contentURI": "0x697066733a516d656138394d5533504852503763513157616b3672327355654d554146324c39727132624c6d5963644b764c57" - } - }, "app:node-operators-registry": { "proxy": { "address": "0x55032650b14df07b85bF18A3a3eC8E0Af2e028d5" @@ -173,49 +131,38 @@ "contentURI": "0x697066733a516d61375058486d456a346a7332676a4d3976744850747176754b3832695335455950694a6d7a4b4c7a55353847" } }, - "lidoApmDeployTx": "0xfa66476569ecef5790f2d0634997b952862bbca56aa088f151b8049421eeb87b", - "lidoApmAddress": "0x0cb113890b04B49455DfE06554e2D784598A29C9", - "createAppReposTx": "0xf48cb21c6be021dd18bd8e02ce89ac7b924245b859f0a8b7c47e88a39016ed41", - "newDaoTx": "0x3feabd79e8549ad68d1827c074fa7123815c80206498946293d5373a160fd866", - "daoAddress": "0xb8FFC3Cd6e7Cf5a098A1c92F48009765B24088Dc", - "daoTokenAddress": "0x5A98FcBEA516Cf06857215779Fd812CA3beF1B32", - "app:aragon-agent": { - "name": "aragon-agent", - "fullName": "aragon-agent.lidopm.eth", - "id": "0x701a4fd1f5174d12a0f1d9ad2c88d0ad11ab6aad0ac72b7d9ce621815f8016a9", - "address": "0x3e40D73EB977Dc6a537aF587D48316feE66E9C8c" - }, - "app:aragon-finance": { - "address": "0xB9E5CBB9CA5b0d659238807E84D0176930753d86", - "name": "aragon-finance", - "fullName": "aragon-finance.lidopm.eth", - "id": "0x5c9918c99c4081ca9459c178381be71d9da40e49e151687da55099c49a4237f1" - }, - "app:aragon-token-manager": { - "address": "0xf73a1260d222f447210581DDf212D915c09a3249", - "name": "aragon-token-manager", - "fullName": "aragon-token-manager.lidopm.eth", - "id": "0xcd567bdf93dd0f6acc3bc7f2155f83244d56a65abbfbefb763e015420102c67b" - }, - "app:aragon-voting": { - "address": "0x2e59A20f205bB85a89C53f1936454680651E618e", - "name": "aragon-voting", - "fullName": "aragon-voting.lidopm.eth", - "id": "0x0abcd104777321a82b010357f20887d61247493d89d2e987ff57bcecbde00e1e" - }, - "wstETH": { - "address": "0x7f39C581F595B53c5cb19bD0b3f8dA6c935E2Ca0", - "contract": "contracts/0.6.12/WstETH.sol", - "deployTx": "0xaf2c1a501d2b290ef1e84ddcfc7beb3406f8ece2c46dee14e212e8233654ff05", - "constructorArgs": [ - "0xae7ab96520DE3A18E5e111B5EaAb095312D7fE84" - ] + "app:oracle": { + "proxy": { + "address": "0x442af784A788A5bd6F42A01Ebe9F287a871243fb" + }, + "implementation": { + "address": "0xa29b819654cE6224A222bb5f586920105E2D7E0E", + "contract": "contracts/0.4.24/oracle/LegacyOracle.sol", + "deployTx": "0xe666e3ce409bb4c18e1016af0b9ed3495b20361a69f2856bccb9e67599795b6f", + "constructorArgs": [] + }, + "aragonApp": { + "fullName": "oracle.lidopm.eth", + "name": "oracle", + "id": "0x8b47ba2a8454ec799cd91646e7ec47168e91fd139b23f017455f3e5898aaba93", + "ipfsCid": "QmUMPfiEKq5Mxm8y2GYQPLujGaJiWz1tvep5W7EdAGgCR8", + "contentURI": "0x697066733a516d656138394d5533504852503763513157616b3672327355654d554146324c39727132624c6d5963644b764c57" + } }, - "executionLayerRewardsVault": { - "address": "0x388C818CA8B9251b393131C08a736A67ccB19297", - "contract": "contracts/0.8.9/LidoExecutionLayerRewardsVault.sol", - "deployTx": "0xd72cf25e4a5fe3677b6f9b2ae13771e02ad66f8d2419f333bb8bde3147bd4294" + "aragon-kernel": { + "implementation": { + "contract": "@aragon/os/contracts/kernel/Kernel.sol", + "address": "0x2b33CF282f867A7FF693A66e11B0FcC5552e4425", + "constructorArgs": [ + true + ] + }, + "proxy": { + "address": "0xb8FFC3Cd6e7Cf5a098A1c92F48009765B24088Dc", + "contract": "@aragon/os/contracts/kernel/KernelProxy.sol" + } }, + "aragonIDAddress": "0x546aa2eae2514494eeadb7bbb35243348983c59d", "burner": { "address": "0xD15a672319Cf0352560eE76d9e89eAB0889046D3", "contract": "contracts/0.8.9/Burner.sol", @@ -232,6 +179,73 @@ "totalNonCoverSharesBurnt": "32145684728326685744" } }, + "chainSpec": { + "depositContractAddress": "0x00000000219ab540356cBB839Cbe05303d7705Fa", + "slotsPerEpoch": 32, + "secondsPerSlot": 12, + "genesisTime": 1606824023 + }, + "createAppReposTx": "0xf48cb21c6be021dd18bd8e02ce89ac7b924245b859f0a8b7c47e88a39016ed41", + "daoAragonId": "lido-dao", + "daoFactoryAddress": "0x7378ad1ba8f3c8e64bbb2a04473edd35846360f1", + "daoInitialSettings": { + "token": { + "name": "Lido DAO Token", + "symbol": "LDO" + }, + "voting": { + "minSupportRequired": "500000000000000000", + "minAcceptanceQuorum": "50000000000000000", + "voteDuration": 86400 + }, + "fee": { + "totalPercent": 10, + "treasuryPercent": 0, + "insurancePercent": 50, + "nodeOperatorsPercent": 50 + } + }, + "daoTokenAddress": "0x5A98FcBEA516Cf06857215779Fd812CA3beF1B32", + "deployCommit": "e45c4d6fb8120fd29426b8d969c19d8a798ca974", + "deployer": "0x55Bc991b2edF3DDb4c520B222bE4F378418ff0fA", + "depositSecurityModule": { + "address": "0xC77F8768774E1c9244BEed705C4354f2113CFc09", + "contract": "contracts/0.8.9/DepositSecurityModule.sol", + "deployTx": "0x21307a2321f167f99de11ccec86d7bdd8233481bbffa493e15c519ca8d662c4f", + "constructorArgs": [ + "0xae7ab96520DE3A18E5e111B5EaAb095312D7fE84", + "0x00000000219ab540356cBB839Cbe05303d7705Fa", + "0xFdDf38947aFB03C621C71b06C9C70bce73f12999", + 150, + 25, + 6646 + ], + "deployParameters": { + "maxDepositsPerBlock": 150, + "minDepositBlockDistance": 25, + "pauseIntentValidityPeriodBlocks": 6646 + } + }, + "dummyEmptyContract": { + "address": "0x6F6541C2203196fEeDd14CD2C09550dA1CbEDa31", + "contract": "contracts/0.8.9/test_helpers/DummyEmptyContract.sol", + "deployTx": "0x9d76786f639bd18365f10c087444761db5dafd0edc85c5c1a3e90219f2d1331d", + "constructorArgs": [] + }, + "eip712StETH": { + "address": "0x8F73e4C2A6D852bb4ab2A45E6a9CF5715b3228B7", + "contract": "contracts/0.8.9/EIP712StETH.sol", + "deployTx": "0xecb5010620fb13b0e2bbc98b8a0c82de0d7385491452cd36cf303cd74216ed91", + "constructorArgs": [ + "0xae7ab96520DE3A18E5e111B5EaAb095312D7fE84" + ] + }, + "ensAddress": "0x00000000000C2E074eC69A0dFb2997BA6C7d2e1e", + "executionLayerRewardsVault": { + "address": "0x388C818CA8B9251b393131C08a736A67ccB19297", + "contract": "contracts/0.8.9/LidoExecutionLayerRewardsVault.sol", + "deployTx": "0xd72cf25e4a5fe3677b6f9b2ae13771e02ad66f8d2419f333bb8bde3147bd4294" + }, "hashConsensusForAccountingOracle": { "address": "0xD624B08C83bAECF0807Dd2c6880C3154a5F0B288", "contract": "contracts/0.8.9/oracle/HashConsensus.sol", @@ -250,33 +264,6 @@ "epochsPerFrame": 225 } }, - "accountingOracle": { - "proxy": { - "address": "0x852deD011285fe67063a08005c71a85690503Cee", - "contract": "contracts/0.8.9/proxy/OssifiableProxy.sol", - "deployTx": "0x3def88f27741216b131de2861cf89af2ca2ac4242b384ee33dca8cc70c51c8dd", - "constructorArgs": [ - "0x6F6541C2203196fEeDd14CD2C09550dA1CbEDa31", - "0x8Ea83AD72396f1E0cD2f8E72b1461db8Eb6aF7B5", - "0x" - ] - }, - "implementation": { - "address": "0xF3c5E0A67f32CF1dc07a8817590efa102079a1aF", - "contract": "contracts/0.8.9/oracle/AccountingOracle.sol", - "deployTx": "0x3e27627d3ed236aff8901df187196e9682187dfd0d259c5d5811a6e923436083", - "constructorArgs": [ - "0xC1d0b3DE6792Bf6b4b37EccdcC24e45978Cfd2Eb", - "0xae7ab96520DE3A18E5e111B5EaAb095312D7fE84", - "0x442af784A788A5bd6F42A01Ebe9F287a871243fb", - 12, - 1606824023 - ], - "deployParameters": { - "consensusVersion": 1 - } - } - }, "hashConsensusForValidatorsExitBusOracle": { "address": "0x7FaDB6358950c5fAA66Cb5EB8eE5147De3df355a", "contract": "contracts/0.8.9/oracle/HashConsensus.sol", @@ -295,47 +282,82 @@ "epochsPerFrame": 75 } }, - "validatorsExitBusOracle": { + "ipfsAPI": "https://ipfs.infura.io:5001/api/v0", + "lidoApm": { + "deployTx": "0xfa66476569ecef5790f2d0634997b952862bbca56aa088f151b8049421eeb87b", + "address": "0x0cb113890b04B49455DfE06554e2D784598A29C9" + }, + "lidoApmEnsName": "lidopm.eth", + "lidoApmEnsRegDurationSec": 94608000, + "lidoLocator": { "proxy": { - "address": "0x0De4Ea0184c2ad0BacA7183356Aea5B8d5Bf5c6e", + "address": "0xC1d0b3DE6792Bf6b4b37EccdcC24e45978Cfd2Eb", "contract": "contracts/0.8.9/proxy/OssifiableProxy.sol", - "deployTx": "0xef3eea1523d2161c2f36ba61e327e3520231614c055b8a88c7f5928d18e423ea", + "deployTx": "0x3a2910624533935cc8c21837b1705bcb159a760796930097016186be705cc455", "constructorArgs": [ "0x6F6541C2203196fEeDd14CD2C09550dA1CbEDa31", "0x8Ea83AD72396f1E0cD2f8E72b1461db8Eb6aF7B5", "0x" ] - }, - "implementation": { - "address": "0xA89Ea51FddE660f67d1850e03C9c9862d33Bc42c", - "contract": "contracts/0.8.9/oracle/ValidatorsExitBusOracle.sol", - "deployTx": "0x5ab545276f78a72a432c3e971c96384973abfab6394e08cb077a006c25aef7a7", - "constructorArgs": [ - 12, - 1606824023, - "0xC1d0b3DE6792Bf6b4b37EccdcC24e45978Cfd2Eb" - ], - "deployParameters": { - "consensusVersion": 1 - } + }, + "implementation": { + "address": "0x1D920cc5bACf7eE506a271a5259f2417CaDeCE1d", + "contract": "contracts/0.8.9/LidoLocator.sol", + "deployTx": "0xf90012ef0a40e47c909ab3a5b3503ecee78f6a9be134d1349a742e500d37ae33", + "constructorArgs": [ + [ + "0x852deD011285fe67063a08005c71a85690503Cee", + "0xC77F8768774E1c9244BEed705C4354f2113CFc09", + "0x388C818CA8B9251b393131C08a736A67ccB19297", + "0x442af784A788A5bd6F42A01Ebe9F287a871243fb", + "0xae7ab96520DE3A18E5e111B5EaAb095312D7fE84", + "0x9305c1Dbfe22c12c66339184C0025d7006f0f1cC", + "0x442af784A788A5bd6F42A01Ebe9F287a871243fb", + "0xD15a672319Cf0352560eE76d9e89eAB0889046D3", + "0xFdDf38947aFB03C621C71b06C9C70bce73f12999", + "0x3e40D73EB977Dc6a537aF587D48316feE66E9C8c", + "0x0De4Ea0184c2ad0BacA7183356Aea5B8d5Bf5c6e", + "0x889edC2eDab5f40e902b864aD4d7AdE8E412F9B1", + "0xB9D7934878B5FB9610B3fE8A5e441e8fad7E293f", + "0xbf05A929c3D7885a6aeAd833a992dA6E5ac23b09" + ] + ] } }, - "depositSecurityModule": { - "address": "0xC77F8768774E1c9244BEed705C4354f2113CFc09", - "contract": "contracts/0.8.9/DepositSecurityModule.sol", - "deployTx": "0x21307a2321f167f99de11ccec86d7bdd8233481bbffa493e15c519ca8d662c4f", + "lidoTemplate": { + "contract": "contracts/0.4.24/template/LidoTemplate.sol", + "address": "0x752350797CB92Ad3BF1295Faf904B27585e66BF5", + "deployTx": "0xdcd4ebe028aa3663a1fe8bbc92ae8489045e29d2a6ef5284083d9be5c3fa5f19", "constructorArgs": [ - "0xae7ab96520DE3A18E5e111B5EaAb095312D7fE84", - "0x00000000219ab540356cBB839Cbe05303d7705Fa", - "0xFdDf38947aFB03C621C71b06C9C70bce73f12999", - 150, - 25, - 6646 + "0x55Bc991b2edF3DDb4c520B222bE4F378418ff0fA", + "0x7378ad1ba8f3c8e64bbb2a04473edd35846360f1", + "0x00000000000C2E074eC69A0dFb2997BA6C7d2e1e", + "0x909d05f384d0663ed4be59863815ab43b4f347ec", + "0x546aa2eae2514494eeadb7bbb35243348983c59d", + "0xa0BC4B67F5FacDE4E50EAFF48691Cfc43F4E280A" + ] + }, + "miniMeTokenFactoryAddress": "0x909d05f384d0663ed4be59863815ab43b4f347ec", + "networkId": 1, + "newDaoTx": "0x3feabd79e8549ad68d1827c074fa7123815c80206498946293d5373a160fd866", + "oracleDaemonConfig": { + "address": "0xbf05A929c3D7885a6aeAd833a992dA6E5ac23b09", + "contract": "contracts/0.8.9/OracleDaemonConfig.sol", + "deployTx": "0xa4f380b8806f5a504ef67fce62989e09be5a48bf114af63483c01c22f0c9a36f", + "constructorArgs": [ + "0x8Ea83AD72396f1E0cD2f8E72b1461db8Eb6aF7B5", + [] ], "deployParameters": { - "maxDepositsPerBlock": 150, - "minDepositBlockDistance": 25, - "pauseIntentValidityPeriodBlocks": 6646 + "NORMALIZED_CL_REWARD_PER_EPOCH": 64, + "NORMALIZED_CL_REWARD_MISTAKE_RATE_BP": 1000, + "REBASE_CHECK_NEAREST_EPOCH_DISTANCE": 1, + "REBASE_CHECK_DISTANT_EPOCH_DISTANCE": 23, + "VALIDATOR_DELAYED_TIMEOUT_IN_SLOTS": 7200, + "VALIDATOR_DELINQUENT_TIMEOUT_IN_SLOTS": 28800, + "PREDICTION_DURATION_IN_SLOTS": 50400, + "FINALIZATION_MAX_NEGATIVE_REBASE_EPOCH_SHIFT": 1350, + "NODE_OPERATOR_NETWORK_PENETRATION_THRESHOLD_BP": 100 } }, "oracleReportSanityChecker": { @@ -381,26 +403,120 @@ "maxPositiveTokenRebase": 750000 } }, - "oracleDaemonConfig": { - "address": "0xbf05A929c3D7885a6aeAd833a992dA6E5ac23b09", - "contract": "contracts/0.8.9/OracleDaemonConfig.sol", - "deployTx": "0xa4f380b8806f5a504ef67fce62989e09be5a48bf114af63483c01c22f0c9a36f", - "constructorArgs": [ - "0x8Ea83AD72396f1E0cD2f8E72b1461db8Eb6aF7B5", - [] - ], - "deployParameters": { - "NORMALIZED_CL_REWARD_PER_EPOCH": 64, - "NORMALIZED_CL_REWARD_MISTAKE_RATE_BP": 1000, - "REBASE_CHECK_NEAREST_EPOCH_DISTANCE": 1, - "REBASE_CHECK_DISTANT_EPOCH_DISTANCE": 23, - "VALIDATOR_DELAYED_TIMEOUT_IN_SLOTS": 7200, - "VALIDATOR_DELINQUENT_TIMEOUT_IN_SLOTS": 28800, - "PREDICTION_DURATION_IN_SLOTS": 50400, - "FINALIZATION_MAX_NEGATIVE_REBASE_EPOCH_SHIFT": 1350, - "NODE_OPERATOR_NETWORK_PENETRATION_THRESHOLD_BP": 100 + "stakingRouter": { + "proxy": { + "address": "0xFdDf38947aFB03C621C71b06C9C70bce73f12999", + "contract": "contracts/0.8.9/proxy/OssifiableProxy.sol", + "deployTx": "0xb8620f04a8db6bb52cfd0978c6677a5f16011e03d4622e5d660ea6ba34c2b122", + "constructorArgs": [ + "0x6F6541C2203196fEeDd14CD2C09550dA1CbEDa31", + "0x8Ea83AD72396f1E0cD2f8E72b1461db8Eb6aF7B5", + "0x" + ] + }, + "implementation": { + "address": "0xD8784e748f59Ba711fB5643191Ec3fAdD50Fb6df", + "contract": "contracts/0.8.9/StakingRouter.sol", + "deployTx": "0xd6d489f22203c835da6027ff0e532a01a08f36f0fda6c7c0a42e471ae3b3c461", + "constructorArgs": [ + "0x00000000219ab540356cBB839Cbe05303d7705Fa" + ] + } + }, + "validatorsExitBusOracle": { + "proxy": { + "address": "0x0De4Ea0184c2ad0BacA7183356Aea5B8d5Bf5c6e", + "contract": "contracts/0.8.9/proxy/OssifiableProxy.sol", + "deployTx": "0xef3eea1523d2161c2f36ba61e327e3520231614c055b8a88c7f5928d18e423ea", + "constructorArgs": [ + "0x6F6541C2203196fEeDd14CD2C09550dA1CbEDa31", + "0x8Ea83AD72396f1E0cD2f8E72b1461db8Eb6aF7B5", + "0x" + ] + }, + "implementation": { + "address": "0xA89Ea51FddE660f67d1850e03C9c9862d33Bc42c", + "contract": "contracts/0.8.9/oracle/ValidatorsExitBusOracle.sol", + "deployTx": "0x5ab545276f78a72a432c3e971c96384973abfab6394e08cb077a006c25aef7a7", + "constructorArgs": [ + 12, + 1606824023, + "0xC1d0b3DE6792Bf6b4b37EccdcC24e45978Cfd2Eb" + ], + "deployParameters": { + "consensusVersion": 1 + } } }, + "vestingParams": { + "unvestedTokensAmount": "363197500000000000000000000", + "holders": { + "0x9Bb75183646e2A0DC855498bacD72b769AE6ceD3": "20000000000000000000000000", + "0x0f89D54B02ca570dE82F770D33c7B7Cf7b3C3394": "25000000000000000000000000", + "0xe49f68B9A01d437B0b7ea416376a7AB21532624e": "2282000000000000000000000", + "0xb842aFD82d940fF5D8F6EF3399572592EBF182B0": "17718000000000000000000000", + "0x9849c2C1B73B41AEE843A002C332a2d16aaaB611": "10000000000000000000000000", + "0x96481cb0fcd7673254ebccc42dce9b92da10ea04": "5000000000000000000000000", + "0xB3DFe140A77eC43006499CB8c2E5e31975caD909": "7500000000000000000000000", + "0x61C808D82A3Ac53231750daDc13c777b59310bD9": "20000000000000000000000000", + "0x447f95026107aaed7472A0470931e689f51e0e42": "20000000000000000000000000", + "0x6ae83EAB68b7112BaD5AfD72d6B24546AbFF137D": "2222222220000000000000000", + "0xC24da173A250e9Ca5c54870639EbE5f88be5102d": "17777777780000000000000000", + "0x1f3813fE7ace2a33585F1438215C7F42832FB7B3": "20000000000000000000000000", + "0x82a8439BA037f88bC73c4CCF55292e158A67f125": "7000000000000000000000000", + "0x91715128a71c9C734CDC20E5EdEEeA02E72e428E": "15000000000000000000000000", + "0xB5587A54fF7022AC218438720BDCD840a32f0481": "5000000000000000000000000", + "0xf5fb27b912d987b5b6e02a1b1be0c1f0740e2c6f": "2000000000000000000000000", + "0x8b1674a617F103897Fb82eC6b8EB749BA0b9765B": "15000000000000000000000000", + "0x48Acf41D10a063f9A6B718B9AAd2e2fF5B319Ca2": "5000000000000000000000000", + "0x7eE09c11D6Dc9684D6D5a4C6d333e5b9e336bb6C": "10000000000000000000000000", + "0x11099aC9Cc097d0C9759635b8e16c6a91ECC43dA": "2000000000000000000000000", + "0x3d4AD2333629eE478E4f522d60A56Ae1Db5D3Cdb": "5000000000000000000000000", + "0xd5eCB56c6ca8f8f52D2DB4dC1257d6161cf3Da29": "100000000000000000000000", + "0x7F5e13a815EC9b4466d283CD521eE9829e7F6f0e": "200000000000000000000000", + "0x2057cbf2332ab2697a52B8DbC85756535d577e32": "500000000000000000000000", + "0x537dfB5f599A3d15C50E2d9270e46b808A52559D": "1000000000000000000000000", + "0x33c4c38e96337172d3de39df82060de26b638c4b": "550000000000000000000000", + "0x6094E1Dd925caCe56Fa501dAEc02b01a49E55770": "300000000000000000000000", + "0x977911f476B28f9F5332fA500387deE81e480a44": "40000000000000000000000", + "0x66d3FdA643320c6DddFBba39e635288A5dF75FB9": "200000000000000000000000", + "0xDFC0ae54af992217100845597982274A26d8CB28": "12500000000000000000000", + "0x32254b28F793CC18B3575C86c61fE3D7421cbbef": "500000000000000000000000", + "0x0Bf5566fB5F1f9934a3944AEF128a1b1a8cF3f17": "50000000000000000000000", + "0x1d3Fa8bf35870271115B997b8eCFe18529422a16": "50000000000000000000000", + "0x366B9729C5A89EC4618A0AB95F832E411eaE8237": "200000000000000000000000", + "0x20921142A35c89bE5D002973d2D6B72d9a625FB0": "200000000000000000000000", + "0x663b91628674846e8D1CBB779EFc8202d86284E2": "7500000000000000000000000", + "0xa6829908f728C6bC5627E2aFe93a0B71E978892D": "300000000000000000000000", + "0x9575B7859DF77F2A0EF034339b80e24dE44AB3F6": "200000000000000000000000", + "0xEe217c23131C6F055F7943Ef1f80Bec99dF35244": "400000000000000000000000", + "0xadde043f556d1083f060A7298E79eaBa08A3a077": "400000000000000000000000", + "0xaFBEfC8401c885A0bb6Ea6Af43f592A015433C65": "200000000000000000000000", + "0x8a62A63b877877bd5B1209B9b67F3d2685284268": "200000000000000000000000", + "0x62Ac238Ac055017DEcAb645E7E56176749f316d0": "200000000000000000000000", + "0x55Bc991b2edF3DDb4c520B222bE4F378418ff0fA": "5000000000000000000000000", + "0x8D689476EB446a1FB0065bFFAc32398Ed7F89165": "10000000000000000000000000", + "0x083fc10cE7e97CaFBaE0fE332a9c4384c5f54E45": "5000000000000000000000000", + "0x0028E24e4Fe5184792Bd0Cf498C11AE5b76185f5": "5000000000000000000000000", + "0xFe45baf0F18c207152A807c1b05926583CFE2e4b": "5000000000000000000000000", + "0x4a7C6899cdcB379e284fBFD045462e751DA4C7cE": "5000000000000000000000000", + "0xD7f0dDcBb0117A53e9ff2cad814B8B810a314f59": "5000000000000000000000000", + "0xb8d83908AAB38a159F3dA47a59d84dB8e1838712": "50000000000000000000000000", + "0xA2dfC431297aee387C05bEEf507E5335E684FbCD": "50000000000000000000000000", + "0x1597D19659F3DE52ABd475F7D2314DCca29359BD": "50000000000000000000000000", + "0x695C388153bEa0fbE3e1C049c149bAD3bc917740": "50000000000000000000000000", + "0x945755dE7EAc99008c8C57bdA96772d50872168b": "50000000000000000000000000", + "0xFea88380bafF95e85305419eB97247981b1a8eEE": "30000000000000000000000000", + "0xAD4f7415407B83a081A0Bee22D05A8FDC18B42da": "50000000000000000000000000", + "0x68335B3ac272C8238b722963368F87dE736b64D6": "5000000000000000000000000", + "0xfA2Ab7C161Ef7F83194498f36ca7aFba90FD08d4": "5000000000000000000000000", + "0x58A764028350aB15899fDCcAFFfd3940e602CEEA": "10000000000000000000000000" + }, + "start": 1639785600, + "cliff": 1639785600, + "end": 1671321600, + "revokable": false + }, "withdrawalQueueERC721": { "proxy": { "address": "0x889edC2eDab5f40e902b864aD4d7AdE8E412F9B1", @@ -441,73 +557,12 @@ ] } }, - "dummyEmptyContract": { - "address": "0x6F6541C2203196fEeDd14CD2C09550dA1CbEDa31", - "contract": "contracts/0.8.9/test_helpers/DummyEmptyContract.sol", - "deployTx": "0x9d76786f639bd18365f10c087444761db5dafd0edc85c5c1a3e90219f2d1331d", - "constructorArgs": [] - }, - "eip712StETH": { - "address": "0x8F73e4C2A6D852bb4ab2A45E6a9CF5715b3228B7", - "contract": "contracts/0.8.9/EIP712StETH.sol", - "deployTx": "0xecb5010620fb13b0e2bbc98b8a0c82de0d7385491452cd36cf303cd74216ed91", + "wstETH": { + "address": "0x7f39C581F595B53c5cb19bD0b3f8dA6c935E2Ca0", + "contract": "contracts/0.6.12/WstETH.sol", + "deployTx": "0xaf2c1a501d2b290ef1e84ddcfc7beb3406f8ece2c46dee14e212e8233654ff05", "constructorArgs": [ "0xae7ab96520DE3A18E5e111B5EaAb095312D7fE84" ] - }, - "stakingRouter": { - "proxy": { - "address": "0xFdDf38947aFB03C621C71b06C9C70bce73f12999", - "contract": "contracts/0.8.9/proxy/OssifiableProxy.sol", - "deployTx": "0xb8620f04a8db6bb52cfd0978c6677a5f16011e03d4622e5d660ea6ba34c2b122", - "constructorArgs": [ - "0x6F6541C2203196fEeDd14CD2C09550dA1CbEDa31", - "0x8Ea83AD72396f1E0cD2f8E72b1461db8Eb6aF7B5", - "0x" - ] - }, - "implementation": { - "address": "0xD8784e748f59Ba711fB5643191Ec3fAdD50Fb6df", - "contract": "contracts/0.8.9/StakingRouter.sol", - "deployTx": "0xd6d489f22203c835da6027ff0e532a01a08f36f0fda6c7c0a42e471ae3b3c461", - "constructorArgs": [ - "0x00000000219ab540356cBB839Cbe05303d7705Fa" - ] - } - }, - "lidoLocator": { - "proxy": { - "address": "0xC1d0b3DE6792Bf6b4b37EccdcC24e45978Cfd2Eb", - "contract": "contracts/0.8.9/proxy/OssifiableProxy.sol", - "deployTx": "0x3a2910624533935cc8c21837b1705bcb159a760796930097016186be705cc455", - "constructorArgs": [ - "0x6F6541C2203196fEeDd14CD2C09550dA1CbEDa31", - "0x8Ea83AD72396f1E0cD2f8E72b1461db8Eb6aF7B5", - "0x" - ] - }, - "implementation": { - "address": "0x1D920cc5bACf7eE506a271a5259f2417CaDeCE1d", - "contract": "contracts/0.8.9/LidoLocator.sol", - "deployTx": "0xf90012ef0a40e47c909ab3a5b3503ecee78f6a9be134d1349a742e500d37ae33", - "constructorArgs": [ - [ - "0x852deD011285fe67063a08005c71a85690503Cee", - "0xC77F8768774E1c9244BEed705C4354f2113CFc09", - "0x388C818CA8B9251b393131C08a736A67ccB19297", - "0x442af784A788A5bd6F42A01Ebe9F287a871243fb", - "0xae7ab96520DE3A18E5e111B5EaAb095312D7fE84", - "0x9305c1Dbfe22c12c66339184C0025d7006f0f1cC", - "0x442af784A788A5bd6F42A01Ebe9F287a871243fb", - "0xD15a672319Cf0352560eE76d9e89eAB0889046D3", - "0xFdDf38947aFB03C621C71b06C9C70bce73f12999", - "0x3e40D73EB977Dc6a537aF587D48316feE66E9C8c", - "0x0De4Ea0184c2ad0BacA7183356Aea5B8d5Bf5c6e", - "0x889edC2eDab5f40e902b864aD4d7AdE8E412F9B1", - "0xB9D7934878B5FB9610B3fE8A5e441e8fad7E293f", - "0xbf05A929c3D7885a6aeAd833a992dA6E5ac23b09" - ] - ] - } } } diff --git a/deployed-mainnetfork-defaults.json b/deployed-mainnetfork-defaults.json deleted file mode 100644 index c1ed84f4a..000000000 --- a/deployed-mainnetfork-defaults.json +++ /dev/null @@ -1,123 +0,0 @@ -{ - "networkId": 1, - "ipfsAPI": "http://127.0.0.1:5001/api/v0", - "multisigAddress": "0x90F8bf6A479f320ead074411a4B0e7944Ea8c9C1", - "owner": "0x90F8bf6A479f320ead074411a4B0e7944Ea8c9C1", - "gateSealAddress": "0x90F8bf6A479f320ead074411a4B0e7944Ea8c9C1", - "lidoApmEnsName": "lidopm.eth", - "lidoApmEnsRegDurationSec": 94608000, - "daoAragonId": "lido-dao", - "daoInitialSettings": { - "voting": { - "minSupportRequired": "500000000000000000", - "minAcceptanceQuorum": "50000000000000000", - "voteDuration": 30, - "objectionPhaseDuration": 1 - }, - "beaconSpec": { - "depositContractAddress": "0x00000000219ab540356cBB839Cbe05303d7705Fa", - "epochsPerFrame": 225, - "slotsPerEpoch": 32, - "secondsPerSlot": 12, - "genesisTime": 1639659600 - }, - "fee": { - "totalPercent": 10, - "treasuryPercent": 0, - "insurancePercent": 50, - "nodeOperatorsPercent": 50 - }, - "token": { - "name": "TEST Lido DAO Token", - "symbol": "TLDO" - } - }, - "vestingParams": { - "unvestedTokensAmount": "230000000000000000000000", - "holders": { - "0xb4124cEB3451635DAcedd11767f004d8a28c6eE7": "530000000000000000000000", - "0x8401Eb5ff34cc943f096A32EF3d5113FEbE8D4Eb": "15000000000000000000000", - "0x306469457266CBBe7c0505e8Aad358622235e768": "15000000000000000000000", - "0xd873F6DC68e3057e4B7da74c6b304d0eF0B484C7": "15000000000000000000000", - "0xDcC5dD922fb1D0fd0c450a0636a8cE827521f0eD": "15000000000000000000000", - "0x27E9727FD9b8CdDdd0854F56712AD9DF647FaB74": "15000000000000000000000", - "0x9766D2e7FFde358AD0A40BB87c4B88D9FAC3F4dd": "15000000000000000000000", - "0xBd7055AB500cD1b0b0B14c82BdBe83ADCc2e8D06": "15000000000000000000000", - "0xe8898A4E589457D979Da4d1BDc35eC2aaf5a3f8E": "15000000000000000000000" - }, - "start": 1639659600, - "cliff": 1639660100, - "end": 1639660100, - "revokable": false - }, - "burner": { - "parameters": { - "totalCoverSharesBurnt": "0", - "totalNonCoverSharesBurnt": "0" - } - }, - "executionLayerRewardsParams": { - "withdrawalLimit": "1" - }, - "legacyOracle": { - "parameters": { - "lastCompletedEpochId": 0 - } - }, - "hashConsensusForAccounting": { - "parameters": { - "fastLaneLengthSlots": 10, - "epochsPerFrame": 12 - } - }, - "accountingOracle": { - "parameters": { - "maxExitedValidatorsPerDay": 240, - "maxExtraDataListItemsCount": 15, - "consensusVersion": 1 - } - }, - "hashConsensusForValidatorsExitBus": { - "parameters": { - "fastLaneLengthSlots": 10, - "epochsPerFrame": 4 - } - }, - "validatorsExitBusOracle": { - "parameters": { - "consensusVersion": 1 - } - }, - "depositSecurityModule": { - "parameters": { - "maxDepositsPerBlock": 150, - "minDepositBlockDistance": 5, - "pauseIntentValidityPeriodBlocks": 6646 - } - }, - "oracleReportSanityChecker": { - "parameters": { - "churnValidatorsPerDayLimit": 1500, - "oneOffCLBalanceDecreaseBPLimit": 500, - "annualBalanceIncreaseBPLimit": 1000, - "simulatedShareRateDeviationBPLimit": 250, - "maxValidatorExitRequestsPerReport": 2000, - "maxAccountingExtraDataListItemsCount": 100, - "maxNodeOperatorsPerExtraDataItemCount": 100, - "requestTimestampMargin": 128, - "maxPositiveTokenRebase": 5000000 - } - }, - "nodeOperatorsRegistry": { - "parameters": { - "stakingModuleTypeId": "curated-onchain-v1", - "stuckPenaltyDelay": "172800" - } - }, - "withdrawalQueueERC721": { - "parameters": { - "name": "stETH Withdrawal NFT", - "symbol": "unstETH" - } - } -} diff --git a/deployed-testnet-defaults.json b/deployed-testnet-defaults.json new file mode 100644 index 000000000..bfe0dc777 --- /dev/null +++ b/deployed-testnet-defaults.json @@ -0,0 +1,125 @@ +{ + "deployer": null, + "gateSeal": { + "factoryAddress": null, + "sealDuration": 518400, + "expiryTimestamp": 1714521600 + }, + "lidoApmEnsName": "lidopm.eth", + "lidoApmEnsRegDurationSec": 94608000, + "daoAragonId": "lido-dao", + "chainSpec": { + "slotsPerEpoch": 32, + "secondsPerSlot": 12, + "genesisTime": null, + "depositContract": null + }, + "daoInitialSettings": { + "voting": { + "minSupportRequired": "500000000000000000", + "minAcceptanceQuorum": "50000000000000000", + "voteDuration": 900, + "objectionPhaseDuration": 300 + }, + "fee": { + "totalPercent": 10, + "treasuryPercent": 50, + "nodeOperatorsPercent": 50 + }, + "token": { + "name": "TEST Lido DAO Token", + "symbol": "TLDO" + } + }, + "vestingParams": { + "unvestedTokensAmount": "0", + "holders": { + "0xCD1f9954330AF39a74Fd6e7B25781B4c24ee373f": "820000000000000000000000", + "0xaa6bfBCD634EE744CB8FE522b29ADD23124593D3": "60000000000000000000000", + "0xBA59A84C6440E8cccfdb5448877E26F1A431Fc8B": "60000000000000000000000", + "lido-aragon-agent-placeholder": "60000000000000000000000" + }, + "start": 0, + "cliff": 0, + "end": 0, + "revokable": false + }, + "burner": { + "deployParameters": { + "totalCoverSharesBurnt": "0", + "totalNonCoverSharesBurnt": "0" + } + }, + "legacyOracle": { + "deployParameters": { + "lastCompletedEpochId": 0 + } + }, + "hashConsensusForAccountingOracle": { + "deployParameters": { + "fastLaneLengthSlots": 10, + "epochsPerFrame": 12 + } + }, + "accountingOracle": { + "deployParameters": { + "consensusVersion": 1 + } + }, + "hashConsensusForValidatorsExitBusOracle": { + "deployParameters": { + "fastLaneLengthSlots": 10, + "epochsPerFrame": 4 + } + }, + "validatorsExitBusOracle": { + "deployParameters": { + "consensusVersion": 1 + } + }, + "depositSecurityModule": { + "deployParameters": { + "maxDepositsPerBlock": 150, + "minDepositBlockDistance": 5, + "pauseIntentValidityPeriodBlocks": 6646 + } + }, + "oracleReportSanityChecker": { + "deployParameters": { + "churnValidatorsPerDayLimit": 1500, + "oneOffCLBalanceDecreaseBPLimit": 500, + "annualBalanceIncreaseBPLimit": 1000, + "simulatedShareRateDeviationBPLimit": 250, + "maxValidatorExitRequestsPerReport": 2000, + "maxAccountingExtraDataListItemsCount": 100, + "maxNodeOperatorsPerExtraDataItemCount": 100, + "requestTimestampMargin": 128, + "maxPositiveTokenRebase": 5000000 + } + }, + "oracleDaemonConfig": { + "deployParameters": { + "NORMALIZED_CL_REWARD_PER_EPOCH": 64, + "NORMALIZED_CL_REWARD_MISTAKE_RATE_BP": 1000, + "REBASE_CHECK_NEAREST_EPOCH_DISTANCE": 1, + "REBASE_CHECK_DISTANT_EPOCH_DISTANCE": 23, + "VALIDATOR_DELAYED_TIMEOUT_IN_SLOTS": 7200, + "VALIDATOR_DELINQUENT_TIMEOUT_IN_SLOTS": 28800, + "NODE_OPERATOR_NETWORK_PENETRATION_THRESHOLD_BP": 100, + "PREDICTION_DURATION_IN_SLOTS": 50400, + "FINALIZATION_MAX_NEGATIVE_REBASE_EPOCH_SHIFT": 1350 + } + }, + "nodeOperatorsRegistry": { + "deployParameters": { + "stakingModuleTypeId": "curated-onchain-v1", + "stuckPenaltyDelay": 172800 + } + }, + "withdrawalQueueERC721": { + "deployParameters": { + "name": "stETH Withdrawal NFT", + "symbol": "unstETH" + } + } +} diff --git a/docs/dev-fork.md b/docs/dev-fork.md deleted file mode 100644 index 645617fec..000000000 --- a/docs/dev-fork.md +++ /dev/null @@ -1,71 +0,0 @@ -## Fork test / development - -**WARNING**: this doc is outdated after Lido V2 upgrade - -!NOTES! -You need an Infura node with [Archived Data](https://infura.io/docs/ethereum/add-ons/archiveData) access -or you can use an [Alchemy node](https://www.alchemy.com/) with Archived Data access for free - -To run Aragon client and check apps on the fork, run `hardhat node` -```bash -npx hardhat node --fork https://mainnet.infura.io/v3/{WEB3_INFURA_PROJECT_ID} -``` - -Next, there are 2 options to run Aragon client - -### Approach 1 - auto - -```bash -NETWORK_NAME=mainnet RUN_CMD=mainnet node scripts/start-aragon.js -``` - -That script sets the right environment variables, downloads and starts Aragon client at `http://localhost:3000/#/lido-dao.aragonid.eth` - -Sync data on the fork is not very fast, so you need to wait ~90sec - - -### Approach 2 - manual - -Download [Aragon client](https://github.com/lidofinance/aragon-client) -```bash -git clone git@github.com:lidofinance/aragon-client.git -``` - -```bash -cd aragon-client && yarn -``` - -Next, get lido apps ID - -```bash -export ARAGON_ENS_REGISTRY_ADDRESS=0x00000000000C2E074eC69A0dFb2997BA6C7d2e1e -export ARAGON_IPFS_GATEWAY=https://mainnet.lido.fi/ipfs -export ARAGON_DEFAULT_ETH_NODE=ws://localhost:8545 -export ARAGON_APP_LOCATOR=0x3ca7c3e38968823ccb4c78ea688df41356f182ae1d159e4ee608d30d68cef320:http://localhost:3010/ -yarn start:mainnet -``` - -Where: -* **ARAGON_ENS_REGISTRY_ADDRESS** - ENS address where `lido-de.aragon id.eth` is registered -* **ARAGON_DEFAULT_ETH_NODE** - local fork -* **ARAGON_APP_LOCATOR** - which source to load app frontend assets from, see more [here](https://github.com/lidofinance/aragon-client/blob/master/docs/CONFIGURATION.md#aragon_app_locator) - -You can find the actual information in `deployed-mainnet.json`, for example, id of Lido app: -```javascript -"app:lido": { - ... - "id": "0x3ca7c3e38968823ccb4c78ea688df41356f182ae1d159e4ee608d30d68cef320", - ... -}, -``` - -So if you want to load the app frontend from localhost (not ipfs), you need to start the lido app and change **ARAGON_APP_LOCATOR**: -```bash -cd apps/lido/app && yarn && yarn dev - -# Server running at http://localhost:3010 -``` - -```bash -export ARAGON_APP_LOCATOR=0x3ca7c3e38968823ccb4c78ea688df41356f182ae1d159e4ee608d30d68cef320:http://localhost:3010/ -``` diff --git a/docs/dev-local.md b/docs/dev-local.md deleted file mode 100644 index 3d58754a1..000000000 --- a/docs/dev-local.md +++ /dev/null @@ -1,191 +0,0 @@ -## Local development - -**WARNING**: this doc is outdated after Lido V2 upgrade - -Networks are defined in `hardhat.config.js` file. To select the target network for deployment, -set `NETWORK_NAME` environment variable to a network name defined in that file. All examples -below assume `localhost` is the target network. - -#### Network state file - -Deployment scripts read their config from and store their results to a file called `deployed-{network_name}.json`, -located in the repo root. This file has the following structure and should always be committed: - -```js -{ - "networkId": 31337, - "owner": "0x5626f3Cf58741768f2B5F09beF0bA50489E17f74", - ...etc -} -``` - -When a script sees that some contract address is already defined in the network state file, it won't -re-deploy the same contract. This means that all deployment scripts are idempotent, you can call the -same script twice and the second call will be a nop. - -You may want to specify some of the configuration options in `networks.` prior to running -deployment to avoid those values being set to default values: - -* `owner` The address that everything will be deployed from. -* `ensAddress` The address of a ENS instance. -* `depositContractAddress` The address of the Beacon chain deposit contract (it will deployed otherwise). -* `daoInitialSettings` Initial settings of the DAO; see below. - -You may specify any number of additional keys inside any network state, they will be left intact by -deployment scripts. - -#### DAO initial settings - -Initial DAO settings can be specified prior to deployment for the specific network in -`networks..daoInitialSettings` field inside `deployed.json` file. - -* `holders` Addresses of initial DAO token holders. -* `stakes` Initial DAO token balances of the holders. -* `tokenName` Name of the DAO token. -* `tokenSymbol` Symbol of the DAO token. -* `voteDuration` See [Voting app documentation]. -* `votingSupportRequired` See [Voting app documentation]. -* `votingMinAcceptanceQuorum` See [Voting app documentation]. -* `depositIterationLimit` See [protocol levers documentation]. - -[Aragon voting app] source code - -[Aragon voting app]: http://web.archive.org/web/20200919192750/https://wiki.aragon.org/archive/dev/apps/voting/ -[Voting app documentation]: https://wiki.aragon.org/archive/dev/apps/voting -[protocol levers documentation]: /docs/protocol-levers.md - -An example of `deployed.json` file prepared for a testnet deployment: - -```js -{ - "networks": { - "5": { - "networkName": "goerli", - "depositContractAddress": "0x07b39f4fde4a38bace212b546dac87c58dfe3fdc", - "owner": "0x3463dD800410965fdBeC2958085b1467CBd4aA31", - "daoInitialSettings": { - "holders": [ - "0x9be0D8ef365A7217c2313c3f33a71D5CeBea2686", - "0x7B1F4c068b3E89Cc586c2f3656Bd95f56CA5B10A", - "0x6244D856606c874DEAC61a61bd07698d47a6F6F2" - ], - "stakes": [ - "100000000000000000000", - "100000000000000000000", - "100000000000000000000" - ], - "tokenName": "Lido DAO Testnet Token", - "tokenSymbol": "LDO", - "voteDuration": 86400, - "votingSupportRequired": "500000000000000000", - "votingMinAcceptanceQuorum": "300000000000000000", - "beaconSpec": { - "epochsPerFrame": 225, - "slotsPerEpoch": 32, - "secondsPerSlot": 12, - "genesisTime": 1605700807 - } - } - } - } -} -``` - -# How to build local dev environment - -To run dev env we need: -* install dependencies -* install and start IPFS daemon -* start hardhat node -* deploy contracts to local hardhat node -* start lido apps on local port -* start aragon client - -#### Step 1: install yarn dependencies - -```bash -yarn -``` - - -#### Step 2: install IPFS daemon - - -See ipfs install instructions [here](https://docs.ipfs.io/install/ipfs-desktop/#ubuntu) - -For example install via Homebrew -```bash -brew install ipfs --cask -``` - -and start in different terminal - -```bash -ipfs daemon -``` - -IPFS is needed to upload an Aragon Apps like (Finance, Voting, etc...) and Lido apps (Lido, LegacyOracle,NOS) - - -#### Step 3: start hardhat node in different terminal - -```bash -npx hardhat node -``` - -This command starts a local eth node with 20 unlocked accounts - - -#### Step 4: set NETWORK_NAME env - -```bash -export NETWORK_NAME=localhost -``` - -#### Step 5: deploy Aragon environment and core apps - -On this step we can deploy all required contracts and upload all required apps to IPFS with next command - -```bash -yarn deploy:all -``` -This is required for test/dev networks that don't have Aragon environment deployed. - -But you can execute deploy scripts step by step: -```bash -# compile contracts at contracts/ folder -yarn compile - -# ENS, APMRegistryFactory, DAOFactory, APMRegistry for aragonpm.eth, etc. -NETWORK_NAME=localhost yarn deploy:aragon-env - -# Core Aragon apps: voting, vault, etc. -NETWORK_NAME=localhost yarn deploy:aragon-std-apps - -# Deploy Lido APM registry and DAO template -NETWORK_NAME=localhost yarn deploy:apm-and-template - -# Build and deploy Lido applications: Lido, Lido Oracle, Node Operator Registry apps -NETWORK_NAME=localhost yarn deploy:apps - -# Deploy the DAO -# -# This step deploys DepositContract as well, if depositContractAddress is not specified -# in deployed.json -NETWORK_NAME=localhost yarn deploy:dao -``` - -#### Step 7. Start Lido apps - -```bash -yarn lido:apps -``` - -#### Step 8. Start Aragon client - -```bash -NETWORK_NAME=localhost yarn aragon:start -``` - -In this step we are replacing the links to the app from the IPFS with a local port, -set env `ARAGON_APP_LOCATOR=0x8a7b...c6:http://localhost:3010` and start aragon client on http://localhost:3000 diff --git a/docs/lido-aragon.md b/docs/lido-aragon.md deleted file mode 100644 index d86bb81d6..000000000 --- a/docs/lido-aragon.md +++ /dev/null @@ -1,135 +0,0 @@ -# Lido-aragon - -**WARNING**: this doc might be outdated after Lido V2 upgrade - -`lido-aragon` tool is used to create, develop lido/aragon apps. - -We already have a pre-build versions for linux/mac and amd/arm architecture. - -```bash -#mac -cli/build/lido-aragon_darwin_amd64 -cli/build/lido-aragon_darwin_arm64 - -#linux -cli/build/lido-aragon_linux_amd64 -cli/build/lido-lido-aragon_linux_arm64 -``` - -Please copy need version to `cli/` folder: -```bash -#apple m1 -cp cli/build/lido-aragon_darwin_arm64 cli/lido-aragon -``` - -#### Step 1. Install IPFS - -[https://docs.ipfs.io/install/](https://docs.ipfs.io/install/command-line/) - - - -#### Step 2. Install project dependencies -```bash -yarn -``` - -#### Step 3. Run cli - -```bash -cd cli/ && ./lido-aragon -``` - -Result -```bash -lido-aragon: lido tool to start aragon env - -Usage: - lido-aragon [command] - -Available Commands: - start Start local or form env - -Flags: - --apps string Which source to load app frontend assets from - --apps-path string Lido apps path - --fork string Fork endpoint https://mainnet.infura.io/v3/{WEB3_INFURA_PROJECT_ID} - -h, --help help for lido-aragon - --network string Set deploy network name (default "localhost") - -v, --verbose Verbose output all of services - --version version for lido-aragon - -Use "lido-aragon [command] --help" for more information about a command. -``` - - -## Start local environment - -```bash -cd cli && ./lido-aragon start all -``` - -By this command we: -- start hardhat local node -- start IPFS daemon (need to deploy apps frontend) -- deploy ENS, APMRegistryFactory, DAOFactory, APMRegistry for aragonpm.eth, etc -- deploy Core Aragon apps: voting, vault, etc -- deploy Lido APM registry and DAO template -- build and deploy Lido applications: Lido, Lido Oracle, Node Operator Registry apps -- deploy the DAO -- start Lido apps -- start Aragon with replacing lido apps links to local port - -if something wrong you can use verbose `-v` flag: -```bash -./lido-aragon start all -v -``` - - -```bash -bash-3.2$ ./lido-aragon start all - SUCCESS Contracs: compile...done - SUCCESS Hardhat node: Started - SUCCESS IPFS: Started - SUCCESS Deploy: Aragon env... done -▀ Deploy: Aragon standart apps... (7s) -``` - -### Start fork - -```bash -./lido-aragon start fork --fork https://mainnet.infura.io/v3/{WEB3_INFURA_PROJECT_ID} --network mainnet --apps=lido:QmPR28q1qWFDcd1aYjnwqpFQYkUofoyBzQ6KCHE6PSQY6P -``` - -What happens here: for the fork, we need to use RPC endpoint, which supports `Archived Data`, so either we use a paid Infura or you can use Alchemy. - -- `--apps` - say to Aragon replacing default app link to our address. Format `appName:appAddress`. - -`appName` - Can be name of app or address like `0x3ca7c..`. If name - try to search appId from `deployed-mainnet.json` file. - -```bash -{ - ... - "app:lido": { - ... - "name": "lido" //<-search for this name - "id": "0x3ca7c.." - ... - } - ... -} -``` - -- appAddress - Can be `http[s]://`, IPFS CID v0 `Qm...` - - -If all is ok you see something like this: -```bash - SUCCESS Hardhat node: Started - SUCCESS Aragon client: starting... - - -Start aragon at: http://localhost:3000/#/0xb8FFC3Cd6e7Cf5a098A1c92F48009765B24088Dc -Please use `Ctrl-C` to exit this program. -``` - -Аfter open on the link, Aragon will need a couple of minutes to synchronize data diff --git a/docs/multisig-deploy.md b/docs/multisig-deploy.md deleted file mode 100644 index 80bab48b3..000000000 --- a/docs/multisig-deploy.md +++ /dev/null @@ -1,468 +0,0 @@ -# Multisig deployment - -**WARNING**: this doc is outdated after Lido V2 upgrade - -This HOWTO describes deployment of the DAO using a multisig/airgapped signer, step-by-step. - -## Preparation - -Clone the repo and install the deps: - -```text -$ git clone git@github.com:lidofinance/lido-dao.git -$ cd lido-dao -$ yarn -``` - -Running deployment scripts requires RPC connection to an Ethereum client, which can be configured -by editing the `hardhat.config.js` file. It is already pre-configured for using the Infura provider, -just copy `accounts.sample.json` to `accounts.json` and edit the `infura` key: - -```json -{ - "eth": {}, - "infura": { - "projectId": "PUT_YOUR_PROJECT_ID_HERE" - } -} -``` - -Some of the deployment steps (namely, deploying contracts) cannot be performed from some multisig -providers and thus require sending the transactions from a usual address. The repo provides a helper -for doing that; if you plan to use it, edit `accounts.json` and put your accounts config under the -`eth.` key. If your RPC client provides an unlocked account, use `remote` as the -value (here and later we assume that the target network is named `mainnet`): - -```json -{ - "eth": { - "mainnet": "remote" - }, - "infura": { - "projectId": "PUT_YOUR_PROJECT_ID_HERE" - } -} -``` - -If you plan to use a BIP-44 mnemonic phrase instead, use the following config shape: - -```json -{ - "eth": { - "mainnet": { - "mnemonic": "YOUR_MNEMONIC_HERE", - "path": "m/44'/60'/0'/0", - "initialIndex": 0, - "count": 1 - } - }, - "infura": { - "projectId": "PUT_YOUR_PROJECT_ID_HERE" - } -} -``` - -You can test the config correctness by listing the accounts and their balances: - -```text -$ yarn hardhat --network mainnet list-accts -``` - - -## Deployment steps - -The deployment process consists of multiple steps. Generally, after each step a set of transaction -files is generated. These transactions need to be executed in a sequential order: only send the -next transaction after the previous one is included in a block. After the last transaction from a -certain step is included in a block, move to the next step. - -There's also a couple steps that don't generate any transactions but check the correctness of the -previous steps instead. - - -## 1. Deploying the base implementations and the template - -Lido uses upgradeable proxy contracts as storage for the state. Each proxy contract points to an -implementation contract providing the code that reads and mutates the state of the proxy. -Implementation contracts can be upgraded via DAO voting. Implementations are immutable, they are -only allowed to modify the caller's (i.e. proxy) contract state. - -In order to setup the protocol, one needs to deploy initial versions of the implementations. Some -popular multisig vaults, e.g. Gnosis Safe, don't support deploying new contracts so this has to be -done from a usual address. - -Part of the protocol deployment logic is incorporated in a contract called `LidoTemplate.sol`, which -also needs to be deployed prior to running further steps. - -### Prepare the network state file - -The deployment scripts use a JSON file named `deployed-.json` to read the initial -environment and protocol configuration and to store data that needs to be persisted between -deployment steps. If a deployment step requires anything except RPC endpoint and ETH accounts, -then it needs to be specified in the network state file. These files are meant to be added under -the source control. If some data is missing from the file, the deployment step will fail with an -error saying what's exactly missing. - -The first step requires the following values: - -* `networkId` id of the network -* `ensAddress` ENS registry address -* `daoFactoryAddress` Aragon `DAOFactory` contract address -* `apmRegistryFactoryAddress` Aragon `APMRegistryFactory` address -* `miniMeTokenFactoryAddress` Aragon `MiniMeTokenFactory` address -* `aragonIDAddress` aragonID `FIFSResolvingRegistrar` address -* `multisigAddress` the address of the multisig contract that will be used in the next steps - to perform the further deployment - -For example, a network state file for `mainnet` will be named `deployed-mainnet.json` and will -initially look like this: - -```json -{ - "networkId": 1, - "ensAddress": "0x00000000000C2E074eC69A0dFb2997BA6C7d2e1e", - "daoFactoryAddress": "0x7378ad1ba8f3c8e64bbb2a04473edd35846360f1", - "apmRegistryFactoryAddress": "0xa0BC4B67F5FacDE4E50EAFF48691Cfc43F4E280A", - "miniMeTokenFactoryAddress": "0x909d05f384d0663ed4be59863815ab43b4f347ec", - "aragonIDAddress": "0x546aa2eae2514494eeadb7bbb35243348983c59d", - "multisigAddress": "YOUR_MULTISIG_CONTRACT_ADDRESS" -} -``` - -Please note that setting `multisigAddress` correctly is very important: this address will own the -deployed template contract, and so only this address will be able to perform the deployment steps -starting from Lido APM deploy (step 5). - -### Generate transaction data files - -After preparing the values in network state file, generate a set of JSON files with transaction -data: - -```text -$ yarn hardhat --network mainnet run ./scripts/multisig/01-deploy-lido-template-and-bases.js -==================== -Network ID: 1 -Reading network state from /Users/me/lido-dao/deployed-mainnet.json... -==================== -Saving deploy TX data for LidoTemplate to tx-01-1-deploy-template.json -Saving deploy TX data for Lido to tx-01-2-deploy-lido-base.json -Saving deploy TX data for LegacyOracle to tx-01-3-deploy-legacy-oracle-base.json -Saving deploy TX data for NodeOperatorsRegistry to tx-01-4-deploy-nops-base.json -==================== -Before continuing the deployment, please send all contract creation transactions -that you can find in the files listed above. You may use a multisig address -if it supports deploying new contract instances. -==================== -Writing network state to /Users/me/lido-dao/deployed-mainnet.json... -All done! -``` - -### Send the transactions - -You can use the `tx` helper for sending the transactions from files. It supports the following flags: - -* `--from` the sender address -* `--file` the TX file which may contain the following fields: `to`, `value`, `data`, `gas`, `from` -* `--gas-price` gas price in wei (optional) -* `--nonce` sender nonce (optional) -* `--wait` the number of seconds to wait before sending the tx (optional, default 5) - -Run the following to deploy the implementations and the template: - -```text -$ yarn hardhat --network mainnet tx --from $DEPLOYER --file tx-01-1-deploy-template.json -$ yarn hardhat --network mainnet tx --from $DEPLOYER --file tx-01-2-deploy-lido-base.json -$ yarn hardhat --network mainnet tx --from $DEPLOYER --file tx-01-3-deploy-legacy-oracle-base.json -$ yarn hardhat --network mainnet tx --from $DEPLOYER --file tx-01-4-deploy-nops-base.json -``` - -You're not required to use this helper to send the transactions defined in the generated files; -it's there for the convenience only. - -> This step is an exception from the "sequential transactions" rule: you can send all four -> transactions in parallel from different addresses. - -### Update the network state file - -After all four transactions are included in the blockchain, update the network state file with -the following values: - -* `daoTemplateDeployTx` hash of the TX sent from the `tx-01-1-deploy-template.json` file -* `lidoBaseDeployTx` hash of the TX sent from the `tx-01-2-deploy-lido-base.json` file -* `oracleBaseDeployTx` hash of the TX sent from the `tx-01-3-deploy-legacy-oracle-base.json` file -* `nodeOperatorsRegistryBaseDeployTx` hash of the TX sent from the `tx-01-4-deploy-nops-base.json` file - - -## 2. Verifying the deployed contracts - -Run the following: - -```text -$ yarn hardhat --network mainnet run ./scripts/multisig/02-obtain-deployed-instances.js -``` - -This step will verify the deployed contracts and add the following fields to the netwotk state file: - -* `daoTemplateAddress` address of the `LidoTemplate` contract -* `app:lido.baseAddress` address of the `Lido` implementation contract -* `app:oracle.baseAddress` address of the `LegacyOracle` implementation contract -* `app:node-operators-registry.baseAddress` address of the `NodeOperatorsRegistry` implementation contract - - -## 3. Register a ENS domain for Lido APM - -This ENS domain is needed for Aragon Package Manager (APM) instance that the protocol will use for -the upgrade mechanics. Prior to running the step, add the following keys to the network state file: - -* `lidoApmEnsName` the second-level ENS domain that APM will use to register packages -* `lidoApmEnsRegDurationSec` the domain lease duration in seconds - -Then, run: - -```text -$ yarn hardhat --network mainnet run ./scripts/multisig/03-register-ens-domain.js -... -==================== -Saving data for commit transaction to tx-02-1-commit-ens-registration.json (projected gas usage is 53667) -Saving data for register transaction to tx-02-2-make-ens-registration.json -==================== -Before continuing the deployment, please send all transactions listed above. - -Make sure to send the second transaction at least 60 seconds after the -first one is included in a block, but no more than 86400 seconds after that. -==================== -``` - -The step will generate two transaction files. You'll need to send these transactions one after -another, waiting no less than one minite between them: - -```text -$ yarn hardhat --network mainnet tx --from $DEPLOYER --file tx-02-1-commit-ens-registration.json -$ sleep 60 -$ yarn hardhat --network mainnet tx --from $DEPLOYER --file tx-02-2-make-ens-registration.json -``` - -## 4. Deploy Lido frontend apps - -The Lido DAO includes frontend apps for DAO governance and protocol management. They are deployed -to IPFS, so you'll need to specify `ipfsAPI` key in the network state file pointing to an IPFS -client API endpoint, e.g. `"ipfsAPI": "http://localhost:5001/api/v0"`. Then, run the following: - -```text -$ yarn hardhat --network mainnet run ./scripts/multisig/04-publish-app-frontends.js -``` - -Make sure that either the IPFS node you're using is going to be permanently up and publicly -available, or that you pin the uploaded content to some other permanent public node. - -This step will add `ipfsCid` and `contentURI` subkeys for all three Lido apps (`app:lido`, -`app:oracle`, `app:node-operators-registry`) in the network state file. The first key is the IPFS -identifier for the root entry of the app frontend, and `contentURI` is the same key encoded to an -Aragon-specific format. - - -## 5. Deploy Lido APM - -Run the following: - -```text -$ yarn hardhat --network mainnet run ./scripts/multisig/05-deploy-apm.js -... -==================== -Parent domain: eth 0x93cdeb708b7545dc668eb9280176169d1c33cfd8ed6f04690a0bcc88a93fc4ae -Subdomain label: lidopm-pre 0x1353eb779a45ed66bdb49e45e006df81a69d9f73067e846003b5bb00984191d4 -==================== -Saving data for APM deploy transaction to tx-03-deploy-apm.json (projected gas usage is 6263517) -==================== -``` - -The step will generate a transaction file; you'll need to send this transaction from the contract -at `multisigAddress`. After the transaction is included in a block, move to the next step. - -### Using Gnosis Safe - -If you're using Gnosis Safe, this can be done by choosing `New Transaction > Contract Interaction` -and enabling the `Use custom data (hex encoded)` option in the popped dialog. Then, copy the contents -of the `to` key from the transaction JSON file to the `Recipient*` field, the contents of the `value` -field to the `Value*` field (enter `0` if there's no `value` key in the transaction JSON), and the -contents of the `data` field to the `Data (hex encoded)*` field. - -Make sure to check the gas limit of the transaction: Gnosis Safe frequently sets it too low. As a -rule of thumb, set it to the value of the `gas` key in the transaction JSON file plus `1500000` (the -additional gas is used to handle multisig logic). - - -## 6. Check the deployed APM - -Run the following: - -```text -$ yarn hardhat --network mainnet run ./scripts/multisig/06-obtain-deployed-apm.js -``` - -Make sure that it finishes without errors and move to the next step. The following field will -be added to the network state file: - -* `lidoApmAddress` the address of the Lido APM controlling `lidoApmEnsName` ENS domain. - - -## 7. Create application APM repositories - -Run the following: - -```text -yarn hardhat --network mainnet run ./scripts/multisig/07-create-app-repos.js -... -==================== -Saving data for createRepos transaction to tx-07-create-app-repos.json (projected gas usage is 7160587) -==================== -``` - -The step will generate a transaction file; you'll need to send this transaction from the contract -at `multisigAddress`. After the transaction is included in a block, move to the next step. - - -## 8. Deploy DAO and its governance token - -This step will deploy the instances of the DAO and governance token. You'll need to add a field -called `daoInitialSettings` to the network state file prior to running the step: - -```js - // ... - "daoInitialSettings": { - // Governance token name/symbol; cannot be changed post-deploy - "token": { - "name": "Lido DAO Token", - "symbol": "LDO" - }, - // Beacon chain spec; can be changed via DAO voting - "beaconSpec": { - "depositContractAddress": "0x00000000219ab540356cBB839Cbe05303d7705Fa", - "slotsPerEpoch": 32, - "secondsPerSlot": 12, - "genesisTime": 1606824023, - "epochsPerFrame": 225 // Lido oracles report once per epochsPerFrame epochs - }, - // DAO voting configuration (Aragon Voting app) - "voting": { - "minSupportRequired": "500000000000000000", // 1e18 === 100% - "minAcceptanceQuorum": "50000000000000000", // 1e18 === 100% - "voteDuration": 172800 // in seconds - }, - // Protocol fee configuration; can be changed via DAO voting - "fee": { - "totalPercent": 10, - "treasuryPercent": 0, - "insurancePercent": 50, - "nodeOperatorsPercent": 50 - } - } - // ... -``` - -Then, run the following: - -```text -$ yarn hardhat --network mainnet run ./scripts/multisig/08-deploy-dao.js -... -Saving data for newDAO transaction to tx-05-deploy-dao.json (projected gas usage is 7118882) -``` - -Send the generated transaction from the contract at `multisigAddress`. After the transaction -is included in a block, move to the next step. - - -## 9. Check the deployed DAO - -Run the following: - -```text -yarn hardhat --network mainnet run ./scripts/multisig/09-obtain-deployed-dao.js -``` - -Make sure that it finishes without errors and move to the next step. The following fields will -be added to the network state file: - -* `daoAddress` the address of the DAO instance; -* `daoTokenAddress` the address of the DAO governance token; -* `proxyAddress` keys under `app:*` keys: adresses of the app instances. - - -## 10. Issue DAO governance tokens - -Add the `vestingParams` key to the network state file containing the following: - -```js - // ... - "vestingParams": { - // unvested tokens will be held on the DAO Agent app - "unvestedTokensAmount": "10000000000000000000000", - // token holder assresses and their respective amounts - "holders": { - "0xaabbcc0000000000000000000000000000000000": "100000000000000000000", - // ... - }, - // Vesting start date - "start": 1608213253, - // Vesting cliff date - "cliff": 1608213253, - // Vesting end date - "end": 1608501253, - // Whether vestings should be revokable by the DAO - "revokable": false - // See https://github.com/aragon/aragon-apps/blob/master/apps/token-manager/contracts/TokenManager.sol - } - // ... -``` - -Then, run the following: - -```text -yarn hardhat --network mainnet run ./scripts/multisig/10-issue-tokens.js -... -==================== -Total batches: 2 -Saving data for issueTokens (batch 1) transaction to tx-06-1-issue-tokens.json (projected gas usage is 6478755) -Saving data for issueTokens (batch 2) transaction to tx-06-2-issue-tokens.json -``` - -Send the generated transactions sequentially from the contract at `multisigAddress`, waiting until -the first one is included in a block before sending the second one. After the second transaction -is included in a block, move to the next step. - - -## 11. Finalize the DAO - -Add the `daoAragonId` key to the network state file, setting it to a name that the DAO will be -registered by in aragonID, i.e. `.aragonid.eth` will resolve to the `daoAddress`. -Run the following: - -```text -yarn hardhat --network mainnet run ./scripts/multisig/11-finalize-dao.js -... -==================== -Saving data for finalizeDAO transaction to tx-11-finalize-dao.json (projected gas usage is 5011582) -``` - -Send the generated transaction from the contract at `multisigAddress`. After the transaction -is included in a block, move to the next step. - - -## 12. Perform the final checks - -At this point, the DAO is fully deployed. Run the following to verify the correctness of the -configuration and permissions setup: - -```text -yarn hardhat --network mainnet run ./scripts/multisig/12-check-dao.js -``` - -If there's some error, it will be printed and further checks will be cancelled. This step only -requires the following fields to be defined in the network state file: - -* `ensAddress` -* `lidoApmEnsName` -* `daoAragonId` -* `vestingParams` -* `daoInitialSettings` -* `daoTemplateAddress` diff --git a/docs/multisig-upgrade.md b/docs/multisig-upgrade.md deleted file mode 100644 index 3523b18a7..000000000 --- a/docs/multisig-upgrade.md +++ /dev/null @@ -1,131 +0,0 @@ -# Multisig app upgrade - -**WARNING**: this doc is outdated after Lido V2 upgrade - -> Please read original [multisig-deploy.md](multisig-deploy.md) before continue. - -## App upgrade steps - -The app upgrade process is very similar deployment steps. - -Assuming we have correct `deployed-mainnet.json`. - -Script allows upgrade one of the custom Lido DAO apps. Valid application names are: `lido`, `oracle` or `node-operators-registry`. It possible update one application in one pass. - -> Hereinafter, we mean an upgrade of the _**oracle**_ app. -> -> Pay attention to the explicit assignment of the app name at the beginning of the some commands through an environment variable, i.e. _APP=**oracle**_. -> -> Also, be careful with files in which data for transactions is saved - the app name is automatically placed into the name, e.g. _tx-13-1-deploy-**oracle**-base.json_ - -## 1. Deploying the new app base implementations - -### Generate transaction data files - -```text -$ APP=oracle yarn hardhat --network mainnet run ./scripts/multisig/13-deploy-new-app-instance.js -======================================== -Network ID: 1 -Reading network state from /Users/me/lido-e2e/oracle_upgrade1/lido-dao/deployed-mainnet.json... -==================== -Saving deploy TX data for LegacyOracle to tx-13-1-deploy-oracle-base.json -==================== -Before continuing the deployment, please send all contract creation transactions -that you can find in the files listed above. You may use a multisig address -if it supports deploying new contract instances. -==================== -All done! -``` - -### Send the transactions - -> See [multisig-deploy.md](multisig-deploy.md#send-the-transactions) for details. - -Run the following to deploy the new implementation: - -```text -$ yarn hardhat --network mainnet tx --from $DEPLOYER --file tx-13-1-deploy-oracle-base.json -``` - -### Update the network state file - -After transaction is included in the blockchain, update the corresponded app section in the network -state file with the following values: - -* `oracleBaseDeployTx` hash of the TX sent from the `tx-13-1-deploy-oracle-base.json` file - -## 2. Verifying the deployed contracts - -Run the following: - -```text -$ APP=oracle yarn hardhat --network mainnet run ./scripts/multisig/14-obtain-deployed-new-app-instance -``` - -This step will verify the deployed contract and update the following field to the network state file: - -* `app:oracle.baseAddress` address of the `LegacyOracle` implementation contract - -## 3. Create new voting for upgrade - -To further update the application, you must perform several actions: - * run the version update procedure in the Aragon Package Manager, - * upgrade the version of the application inside the DAO as well as its frontend, - * add newly introced access rights, - * initialize new data. - -Since the rights to the some of these have been transferred to the DAO, a voting must be started to -perform the version update. All these actions are applied atomically as the voting is accepted and -entacted. - -Run the script to generate data for the create voting transaction: - -```text -$ APP=oracle CONTENT_CID="QmPWVU6GaMRhiUhR5SSXxMWuQ9jxqSv1d6K2afyyaJT1Rb" yarn hardhat --network mainnet run ./scripts/multisig/15-vote-new-app-impl.js -======================================== -Network ID: 1 -Reading network state from /Users/me/lido-e2e/oracle_upgrade1/lido-dao/deployed-mainnet.json... -==================== -Upgrading app: oracle -appId: 0xb2977cfc13b000b6807b9ae3cf4d938f4cc8ba98e1d68ad911c58924d6aa4f11 -Contract implementation: 0xa892CCce358748429188b1554C3999a552a99cD8 -> 0x869E3cB508200D3bE0e946613a8986E8eb3E64d7 -Bump version: 1,0,0 -> 2,0,0 -Content URI: 0x697066733a516d505756553647614d52686955685235535358784d577551396a787153763164364b3261667979614a54315262 -> 0x697066733a516d505756553647614d52686955685235535358784d577551396a787153763164364b3261667979614a54315262 -Oracle proxy address: 0x24d8451BC07e7aF4Ba94F69aCDD9ad3c6579D9FB -Voting address: 0xbc0B67b4553f4CF52a913DE9A6eD0057E2E758Db -ACL address: 0xb3CF58412a00282934D3C3E73F49347567516E98 -==================== -Saving data for New voting: oracle new impl transaction to tx-15-1-create-vote-new-oracle-version.json (projected gas usage is 851790) -==================== -Before continuing the deployment, please send all transactions listed above. -A new voting will be created to add a new "oracle" implementation to Lido APM. -You must complete it positively and execute before continuing with the deployment! -==================== -``` - -You may also want to explicitly specify `HOLDER=0x...`, the account that holds LDO tokens and thus -have the right to create a voting. By default, `multisigAddress` value is used here. - -The step will generate the transaction file. You'll need to send these transaction: - -```text -$ yarn hardhat --network mainnet tx --from $DEPLOYER --file tx-15-1-create-vote-new-oracle-version.json -``` - -New voting will be created. The voting must complete successfully before proceeding next. - -## 4. Verify the created voting. - -Collect the transaction ID from the previous step and run the 16th script like the following. - -``` -TX=0xf968fcc552b95e641cff14ed68101ed96dbcd9ec85609f3c70c1b849418c94ff yarn hardhat --network mainnet run ./scripts/multisig/16-verify-vote-tx.js -======================================== -Network ID: 1 -Reading network state from /Users/me/lido-e2e/oracle_upgrade1/lido-dao/deployed-mainnet.json... -==================== -Voting contract: 0x2e59A20f205bB85a89C53f1936454680651E618e -Voting no: 63 -Creator: 0xf73a1260d222f447210581DDf212D915c09a3249 -All done! -``` diff --git a/docs/scratch-deploy.md b/docs/scratch-deploy.md new file mode 100644 index 000000000..bbbf171c5 --- /dev/null +++ b/docs/scratch-deploy.md @@ -0,0 +1,188 @@ +# Deploy Lido protocol from scratch + +## Requirements + +* node.js v16 or v18 (later might work fine as well, but not tested) +* yarn + +## General info + +The repo contains bash scripts for deployment of the DAO under multiple environments: +- local node (ganache, anvil, hardhat network) `dao-local-deploy.sh` +- holesky testnet - `dao-holesky-deploy.sh` + +The protocol has a bunch of parameters to configure for the scratch deployment. The default configuration is stored in files `deployed--defaults.json`, where `` is the target environment. Currently, there is a single default configuration, `deployed-testnet-defaults.json`, suitable for testnet deployments. Compared to the mainnet configuration, it has lower vote durations, more frequent oracle report cycles, etc. Part of the parameters require further specification -- they are marked with `null` values. +During the deployment, the "default" configuration is copied to `deployed-.json`, where `` is the name of a network configuration defined in `hardhat.config.js`. The file `deployed-.json` gets populated with the contract addresses and transaction hashes during the deployment process. + +These are the deployment setups, supported currently: +- local (basically any node at http://127.0.0.1:8545); +- Holešky testnet. + +Each is described in the details in the sections below. + +> NB: Aragon UI for Lido DAO is to be deprecated and replaced by a custom solution, thus not included in the deployment script. + +### Deploy steps + +A brief description of what's going on under the hood in the deploy script. + +- Prepare `deployed-.json` file + - It is copied from `deployed-testnet-defaults.json` + - and expended by env variables values, e. g. `DEPLOYER`. + - It gets filled with the deployed contracts info from step to step. +- (optional) Deploy DepositContract. + - The step is skipped if the DepositContract address is specified +- (optional) Deploy ENS + - The step is skipped if the ENS Registry address is specified +- Deploy Aragon framework environment +- Deploy standard Aragon apps contracts (like `Agent`, `Voting`) +- Deploy `LidoTemplate` contract + - This is an auxiliary deploy contract which performs DAO configuration +- Deploy Lido custom Aragon apps implementations (aka bases), namely for `Lido`, `LegacyOracle`, `NodeOperatorsRegistry` +- Registry Lido APM name in ENS +- Deploy Aragon package manager contract `APMRegistry` (via `LidoTemplate`) +- Deploy Lido custom Aragon apps repo contracts (via `LidoTemplate`) +- Deploy Lido DAO (via `LidoTemplate`) +- Issue DAO tokens (via `LidoTemplate`) +- Deploy non-Aragon Lido contracts: `OracleDaemonConfig`, `LidoLocator`, `OracleReportSanityChecker`, `EIP712StETH`, `WstETH`, `WithdrawalQueueERC721`, `WithdrawalVault`, `LidoExecutionLayerRewardsVault`, `StakingRouter`, `DepositSecurityModule`, `AccountingOracle`, `HashConsensus` for AccountingOracle, `ValidatorsExitBusOracle`, `HashConsensus` for ValidatorsExitBusOracle, `Burner`. +- Finalize Lido DAO deployment: issue unvested LDO tokens, set Aragon permissions, register Lido DAO name in Aragon ID (via `LidoTemplate`) +- Initialize non-Aragon Lido contracts +- Set parameters of `OracleDaemonConfig` +- Setup non-Aragon permissions +- Plug NodeOperatorsRegistry as Curated staking module +- Transfer all admin roles from deployer to `Agent` + - OZ admin roles: `Burner`, `HashConsensus` for `AccountingOracle`, `HashConsensus` for `ValidatorsExitBusOracle`, `StakingRouter`, `AccountingOracle`, `ValidatorsExitBusOracle`, `WithdrawalQueueERC721`, `OracleDaemonConfig` + - OssifiableProxy admins: : `LidoLocator`, `StakingRouter`, `AccountingOracle`, `ValidatorsExitBusOracle`, `WithdrawalQueueERC721` + - `DepositSecurityModule` owner + +## Local deployment + +Deploys the DAO to local (http://127.0.0.1:8545) dev node (anvil, hardhat, ganache). +The deployment is done from the default test account `0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266`. +The node must be configured with the default test accounts derived from the mnemonic `test test test test test test test test test test test junk`. + +1. Run `yarn install` (get sure repo dependencies are installed) +2. Run the node on default port 8545 (for the commands, see subsections below) +3. Set test account private key `0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80` to `accounts.json` under `/eth/local` like `"local": [""]` (see `accounts.sample.json` for example) +4. Run the deploy script `bash scripts/scratch/dao-local-deploy.sh` from root repo directory +5. Check out the deploy artifacts in `deployed-local.json` + +### Anvil + +Run the node with the command: + +```shell +anvil -p 8545 --auto-impersonate --gas-price 0 --base-fee 0 --chain-id 1337 --mnemonic "test test test test test test test test test test test junk" +``` + +### Hardhat node + +> NB: Hardhat node configuration is set in `hardhat.config.js` under `hardhat: { `. + +To run hardhat node execute: +```shell +yarn hardhat node +``` + +## Holešky deployment + +To do Holešky deployment, the following parameters must be set up via env variables: + +- `DEPLOYER`. The deployer address. The deployer must own its private key. To ensure proper operation, it should have an adequate amount of ether. The total deployment gas cost is approximately 100,000,000 gas, and this cost can vary based on whether specific components of the environment, such as the DepositContract, are deployed or not. +- `RPC_URL`. Address of of the Ethereum RPC node to use. E.g. for Infura it is `https://holesky.infura.io/v3/` +- `GAS_PRIORITY_FEE`. Gas priority fee. By default set to `2` +- `GAS_MAX_FEE`. Gas max fee. By default set to `100` +- `GATE_SEAL_FACTORY`. Address of the [GateSeal Factory](https://github.com/lidofinance/gate-seals) contract. Must be deployed in advance. Can be set to any `0x0000000000000000000000000000000000000000` to debug deployment. + +Also you need to specify `DEPLOYER` private key in `accounts.json` under `/eth/holesky` like `"holesky": [""]`. See `accounts.sample.json` for an example. + +To start the deployment, run (the env variables must already defined) from the root repo directory: +```shell +bash scripts/scratch/dao-holesky-deploy.sh +``` +Deploy artifacts information will be stored in `deployed-holesky.json`. + +## Publishing sources to Etherscan + +After the deployment run + +```shell +NETWORK= RPC_URL= bash ./scripts/scratch/verify-contracts-code.sh +``` + +### Issues with verification of part of the contracts deployed from factories + +There are some contracts deployed from other contracts for which automatic hardhat etherscan verification fails: + +- `AppProxyUpgradeable` of multiple contracts (`app:lido`, `app:node-operators-registry`, `app:oracle`, `app:voting`, ...) +- `KernelProxy` -- proxy for `Kernel` +- `AppProxyPinned` -- proxy for `EVMScriptRegistry` +- `MiniMeToken` -- LDO token +- `CallsScript` -- Aragon internal contract +- `EVMScriptRegistry` -- Aragon internal contract + +The workaround used during Holešky deployment is to deploy auxiliary instances of these contracts standalone and verify them via hardhat Etherscan plugin. After this Etherscan will mark the target contracts as verified by "Similar Match Source Code". + +NB, that some contracts require additional auxiliary contract to be deployed. Namely, the constructor of `AppProxyPinned` depends on proxy implementation ("base" in Aragon terms) contract with `initialize()` function and `Kernel` contract, which must return the implementation by call `kernel().getApp(KERNEL_APP_BASES_NAMESPACE, _appId)`. See `@aragon/os/contracts/apps/AppProxyBase.sol` for the details. + +## Post deploy initialization + +### Initialization up to the fully operational state + +In order to make the protocol fully operational, the additional steps are required: + +- add oracle committee members to `HashConsensus` contracts for `AccountingOracle` and `ValidatorsExitBusOracle`: `HashConsensus.addMember`; +- initialize initial epoch for `HashConsensus` contracts for `AccountingOracle` and `ValidatorsExitBusOracle`: `HashConsensus.updateInitialEpoch`; +- add guardians to `DepositSecurityModule`: `DepositSecurityModule.addGuardians`; +- resume protocol: `Lido.resume`; +- resume WithdrawalQueue: `WithdrawalQueueERC721.resume`; +- add at least one Node Operator: `NodeOperatorsRegistry.addNodeOperator`; +- add validator keys to the Node Operators: `NodeOperatorsRegistry.addSigningKeys`; +- set staking limits for the Node Operators: `NodeOperatorsRegistry.setNodeOperatorStakingLimit`. + +NB, that part of the actions require prior granting of the required roles, e.g. `STAKING_MODULE_MANAGE_ROLE` for `StakingRouter.addStakingModule`: + +```js + await stakingRouter.grantRole(STAKING_MODULE_MANAGE_ROLE, agent.address, { from: agent.address }) + await stakingRouter.addStakingModule( + state.nodeOperatorsRegistry.deployParameters.stakingModuleTypeId, + nodeOperatorsRegistry.address, + NOR_STAKING_MODULE_TARGET_SHARE_BP, + NOR_STAKING_MODULE_MODULE_FEE_BP, + NOR_STAKING_MODULE_TREASURY_FEE_BP, + { from: agent.address } + ) + await stakingRouter.renounceRole(STAKING_MODULE_MANAGE_ROLE, agent.address, { from: agent.address }) +``` + +## Protocol parameters + +This section describes part of the parameters and their values used at the deployment. The values are specified in `deployed-testnet-defaults.json`. + +### OracleDaemonConfig + +```python +# Parameters related to "bunker mode" +# See https://research.lido.fi/t/withdrawals-for-lido-on-ethereum-bunker-mode-design-and-implementation/3890/4 +# and https://snapshot.org/#/lido-snapshot.eth/proposal/0xa4eb1220a15d46a1825d5a0f44de1b34644d4aa6bb95f910b86b29bb7654e330 +# NB: BASE_REWARD_FACTOR: https://ethereum.github.io/consensus-specs/specs/phase0/beacon-chain/#rewards-and-penalties +NORMALIZED_CL_REWARD_PER_EPOCH=64 +NORMALIZED_CL_REWARD_MISTAKE_RATE_BP=1000 # 10% +REBASE_CHECK_NEAREST_EPOCH_DISTANCE=1 +REBASE_CHECK_DISTANT_EPOCH_DISTANCE=23 # 10% of AO 225 epochs frame +VALIDATOR_DELAYED_TIMEOUT_IN_SLOTS=7200 # 1 day + +# See https://snapshot.org/#/lido-snapshot.eth/proposal/0xa4eb1220a15d46a1825d5a0f44de1b34644d4aa6bb95f910b86b29bb7654e330 for "Requirement not be considered Delinquent" +VALIDATOR_DELINQUENT_TIMEOUT_IN_SLOTS=28800 # 4 days + +# See "B.3.I" of https://snapshot.org/#/lido-snapshot.eth/proposal/0xa4eb1220a15d46a1825d5a0f44de1b34644d4aa6bb95f910b86b29bb7654e330 +NODE_OPERATOR_NETWORK_PENETRATION_THRESHOLD_BP=100 # 1% network penetration for a single NO + +# Time period of historical observations used for prediction of the rewards amount +# see https://research.lido.fi/t/withdrawals-for-lido-on-ethereum-bunker-mode-design-and-implementation/3890/4 +PREDICTION_DURATION_IN_SLOTS=50400 # 7 days + +# Max period of delay for requests finalization in case of bunker due to negative rebase +# twice min governance response time - 3 days voting duration +FINALIZATION_MAX_NEGATIVE_REBASE_EPOCH_SHIFT=1350 # 6 days +``` diff --git a/hardhat.config.js b/hardhat.config.js index 515ef63be..245878e40 100644 --- a/hardhat.config.js +++ b/hardhat.config.js @@ -11,11 +11,12 @@ require('@nomiclabs/hardhat-etherscan') require('hardhat-gas-reporter') require('solidity-coverage') require('hardhat-contract-sizer') -// require('hardhat-ignore-warnings') +require('hardhat-ignore-warnings') require('./foundry/skip-sol-tests-compilation') const NETWORK_NAME = getNetworkName() const ETH_ACCOUNT_NAME = process.env.ETH_ACCOUNT_NAME +const RPC_URL = process.env.RPC_URL // eslint-disable-next-line no-undef task('accounts', 'Prints the list of accounts', async (taskArgs, hre) => { @@ -36,6 +37,12 @@ const accounts = readJson(`./accounts.json`) || { const getNetConfig = (networkName, ethAccountName) => { const netState = readJson(`./deployed-${networkName}.json`) || {} const ethAccts = accounts.eth || {} + + if (RPC_URL === undefined && networkName !== 'hardhat') { + console.error('ERROR: RPC_URL env variable is not set') + process.exit(1) + } + const base = { accounts: ethAccountName === 'remote' @@ -44,31 +51,20 @@ const getNetConfig = (networkName, ethAccountName) => { ensAddress: netState.ensAddress, timeout: 60000, } - const localhost = { - ...base, - url: 'http://localhost:8545', - chainId: 31337, - gas: 80000000, // the same as in Görli - } - const mainnetfork = { - ...base, - url: 'http://127.0.0.1:8545', - chainId: 1337, - gas: 80000000, // the same as in Görli - } const byNetName = { - localhost, - mainnetfork, - local: { + mainnetfork: { ...base, - accounts: { - mnemonic: 'explain tackle mirror kit van hammer degree position ginger unfair soup bonus', - count: 30, - }, - url: 'http://localhost:8545', - chainId: 1337, + url: RPC_URL, + }, + goerlifork: { + ...base, + url: RPC_URL, + }, + local: { + url: RPC_URL, }, hardhat: { + // NB!: forking get enabled if env variable HARDHAT_FORKING_URL is set, see code below blockGasLimit: 30000000, gasPrice: 0, initialBaseFeePerGas: 0, @@ -81,32 +77,27 @@ const getNetConfig = (networkName, ethAccountName) => { gasPrice: 0, }, }, - 'goerli-pyrmont': { + goerli: { ...base, - url: 'http://206.81.31.11/rpc', + url: RPC_URL, chainId: 5, - }, - rinkeby: { - ...base, - url: 'https://rinkeby.infura.io/v3/' + accounts.infura.projectId, - chainId: 4, timeout: 60000 * 10, }, - goerli: { + goerlidebug: { ...base, - url: 'https://goerli.infura.io/v3/' + accounts.infura.projectId, + url: RPC_URL, chainId: 5, - timeout: 60000 * 10, + timeout: 60000 * 15, }, - 'mainnet-test': { + holesky: { ...base, - url: 'https://mainnet.infura.io/v3/' + accounts.infura.projectId, - chainId: 1, - timeout: 60000 * 10, + url: RPC_URL, + chainId: 17000, + timeout: 60000 * 15, }, mainnet: { ...base, - url: 'https://mainnet.infura.io/v3/' + accounts.infura.projectId, + url: RPC_URL, chainId: 1, timeout: 60000 * 10, }, @@ -115,12 +106,14 @@ const getNetConfig = (networkName, ethAccountName) => { chainId: 1, timeout: 60000 * 10, forking: { - url: 'https://mainnet.infura.io/v3/' + accounts.infura.projectId, - // url: 'https://eth-mainnet.alchemyapi.io/v2/' + accounts.alchemy.apiKey + url: RPC_URL, }, }, } const netConfig = byNetName[networkName] + if (networkName === 'hardhat' && process.env.HARDHAT_FORKING_URL) { + netConfig.forking = { url: process.env.HARDHAT_FORKING_URL } + } return netConfig ? { [networkName]: netConfig } : {} } @@ -163,6 +156,10 @@ module.exports = { version: '0.6.12', settings: solcSettings6, }, + { + version: '0.8.4', + settings: solcSettings8, + }, { version: '0.8.9', settings: solcSettings8, @@ -196,10 +193,18 @@ module.exports = { enabled: !!process.env.REPORT_GAS, currency: 'USD', }, - etherscan: accounts.etherscan, - aragon: { - ipfsApi: process.env.IPFS_API_URL || 'https://ipfs.infura.io:5001/api/v0', - ipfsGateway: process.env.IPFS_GATEWAY_URL || 'https://ipfs.io/', + etherscan: { + apiKey: accounts.etherscan.apiKey, + customChains: [ + { + network: 'holesky', + chainId: 17000, + urls: { + apiURL: 'https://api-holesky.etherscan.io/api', + browserURL: 'https://holesky.etherscan.io', + }, + }, + ], }, ipfs: { url: process.env.IPFS_API_URL || 'https://ipfs.infura.io:5001/api/v0', diff --git a/package.json b/package.json index f2d1ddd71..d9f20e113 100644 --- a/package.json +++ b/package.json @@ -12,7 +12,6 @@ ], "scripts": { "postinstall": "husky install", - "build:apps": "yarn compile && hardhat run --no-compile scripts/build-apps-frontend.js", "lint": "yarn lint:sol && yarn lint:js", "lint:sol": "solhint \"contracts/**/*.sol\" --ignore-path .solhintignore", "lint:sol:fix": "yarn lint:sol --fix", @@ -35,32 +34,11 @@ "extract-artifacts": "node ./scripts/extract-artifacts.js", "size-contracts": "yarn run hardhat size-contracts", "deploy": "yarn run deploy:all", - "deploy:all": "yarn compile && yarn deploy:aragon-env && yarn deploy:aragon-std-apps && yarn deploy:apm-and-template && yarn deploy:apps && yarn deploy:dao", - "deploy:e2e:all": "yarn compile && yarn deploy:e2e:aragon-env && yarn deploy:e2e:aragon-std-apps && yarn deploy:e2e:apm-and-template && yarn deploy:e2e:apps && yarn deploy:e2e:dao", - "deploy:aragon-env": "hardhat run --no-compile ./scripts/scratch/deploy-aragon-env.js", - "deploy:e2e:aragon-env": "NETWORK_NAME=e2e NETWORK_STATE_FILE=deployed-e2e.json hardhat run --no-compile ./scripts/scratch/deploy-aragon-env.js", - "deploy:aragon-std-apps": "RELEASE_TYPE=major hardhat run --no-compile ./scripts/scratch/deploy-aragon-std-apps.js --config ./hardhat.config.aragon-apps.js", - "deploy:e2e:aragon-std-apps": "NETWORK_NAME=e2e NETWORK_STATE_FILE=deployed-e2e.json IPFS_API_URL=http://localhost:5001/api/v0 IPFS_GATEWAY_URL=http://localhost:8080 RELEASE_TYPE=major hardhat run --no-compile ./scripts/scratch/deploy-aragon-std-apps.js --config ./hardhat.config.aragon-apps.js", - "deploy:apm-and-template": "hardhat run --no-compile ./scripts/scratch/deploy-lido-apm-and-template.js", - "deploy:e2e:apm-and-template": "NETWORK_NAME=e2e NETWORK_STATE_FILE=deployed-e2e.json hardhat run --no-compile ./scripts/scratch/deploy-lido-apm-and-template.js", - "deploy:apps": "hardhat run --no-compile ./scripts/scratch/deploy-lido-apps.js", - "deploy:e2e:apps": "NETWORK_NAME=e2e NETWORK_STATE_FILE=deployed-e2e.json IPFS_API_URL=http://localhost:5001/api/v0 IPFS_GATEWAY_URL=http://localhost:8080 hardhat run --no-compile ./scripts/scratch/deploy-lido-apps.js", "deploy:app-pool": "APPS=depool yarn deploy:apps", "deploy:app-staking-providers-registry": "APPS=staking-providers-registry yarn deploy:apps", - "deploy:local:all": "yarn compile && yarn deploy:local:aragon-env && yarn deploy:local:aragon-std-apps && yarn deploy:local:apm-and-template && yarn deploy:local:apps && yarn deploy:local:dao", - "deploy:local:aragon-env": "NETWORK_NAME=local NETWORK_STATE_FILE=deployed-local.json hardhat run --no-compile ./scripts/scratch/deploy-aragon-env.js", - "deploy:local:aragon-std-apps": "NETWORK_NAME=local NETWORK_STATE_FILE=deployed-local.json IPFS_API_URL=http://127.0.0.1:5001/api/v0 IPFS_GATEWAY_URL=http://127.0.0.1:8080 RELEASE_TYPE=major hardhat run --no-compile ./scripts/scratch/deploy-aragon-std-apps.js --config ./hardhat.config.aragon-apps.js", - "deploy:rinkeby:aragon-env": "NETWORK_NAME=rinkeby NETWORK_STATE_FILE=deployed-rinkeby.json hardhat run --no-compile ./scripts/scratch/deploy-aragon-env.js", - "deploy:rinkeby:aragon-std-apps": "NETWORK_NAME=rinkeby NETWORK_STATE_FILE=deployed-rinkeby.json IPFS_API_URL=http://127.0.0.1:5001/api/v0 IPFS_GATEWAY_URL=http://127.0.0.1:8080 RELEASE_TYPE=major hardhat run --no-compile ./scripts/scratch/deploy-aragon-std-apps.js --config ./hardhat.config.aragon-apps.js", - "deploy:mainnetfork:aragon-env": "NETWORK_NAME=mainnetfork NETWORK_STATE_FILE=deployed-mainnetfork.json hardhat run --no-compile ./scripts/scratch/deploy-aragon-env.js", - "deploy:mainnetfork:aragon-std-apps": "NETWORK_NAME=mainnetfork NETWORK_STATE_FILE=deployed-mainnetfork.json IPFS_API_URL=http://127.0.0.1:5001/api/v0 IPFS_GATEWAY_URL=http://127.0.0.1:8080 RELEASE_TYPE=major hardhat run --no-compile ./scripts/scratch/deploy-aragon-std-apps.js --config ./hardhat.config.aragon-apps.js", - "deploy:local:apm-and-template": "NETWORK_NAME=local NETWORK_STATE_FILE=deployed-local.json hardhat run --no-compile ./scripts/scratch/deploy-lido-apm-and-template.js", - "deploy:local:apps": "NETWORK_NAME=local NETWORK_STATE_FILE=deployed-local.json IPFS_API_URL=http://127.0.0.1:5001/api/v0 IPFS_GATEWAY_URL=http://127.0.0.1:8080o hardhat run --no-compile ./scripts/scratch/deploy-lido-apps.js", "apps:lido": "yarn --cwd ./apps/lido/app/ cli", "apps:nos": "yarn --cwd ./apps/node-operators-registry/app/ cli", - "lido:apps": "concurrently \"yarn apps:lido\" \"yarn apps:oracle\" \"yarn apps:nos\" ", - "aragon:start": "node scripts/start-aragon.js", - "lido:start": "hardhat node& yarn deploy:all && yarn lido:apps& hardhat run --no-compile scripts/start-aragon.js" + "lido:apps": "concurrently \"yarn apps:lido\" \"yarn apps:oracle\" \"yarn apps:nos\" " }, "devDependencies": { "@aragon/apps-agent": "^2.0.0", @@ -77,7 +55,7 @@ "@babel/preset-env": "^7.11.5", "@babel/register": "^7.12.1", "@nomiclabs/hardhat-ethers": "^2.0.4", - "@nomiclabs/hardhat-etherscan": "^2.1.8", + "@nomiclabs/hardhat-etherscan": "^3.1.7", "@nomiclabs/hardhat-ganache": "^2.0.1", "@nomiclabs/hardhat-truffle5": "^2.0.2", "@nomiclabs/hardhat-web3": "^2.0.0", diff --git a/poetry.lock b/poetry.lock index 0323a943e..7f5b98a81 100644 --- a/poetry.lock +++ b/poetry.lock @@ -2,98 +2,98 @@ [[package]] name = "aiohttp" -version = "3.8.4" +version = "3.8.5" description = "Async http client/server framework (asyncio)" optional = false python-versions = ">=3.6" files = [ - {file = "aiohttp-3.8.4-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:5ce45967538fb747370308d3145aa68a074bdecb4f3a300869590f725ced69c1"}, - {file = "aiohttp-3.8.4-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:b744c33b6f14ca26b7544e8d8aadff6b765a80ad6164fb1a430bbadd593dfb1a"}, - {file = "aiohttp-3.8.4-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:1a45865451439eb320784918617ba54b7a377e3501fb70402ab84d38c2cd891b"}, - {file = "aiohttp-3.8.4-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a86d42d7cba1cec432d47ab13b6637bee393a10f664c425ea7b305d1301ca1a3"}, - {file = "aiohttp-3.8.4-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ee3c36df21b5714d49fc4580247947aa64bcbe2939d1b77b4c8dcb8f6c9faecc"}, - {file = "aiohttp-3.8.4-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:176a64b24c0935869d5bbc4c96e82f89f643bcdf08ec947701b9dbb3c956b7dd"}, - {file = "aiohttp-3.8.4-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c844fd628851c0bc309f3c801b3a3d58ce430b2ce5b359cd918a5a76d0b20cb5"}, - {file = "aiohttp-3.8.4-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:5393fb786a9e23e4799fec788e7e735de18052f83682ce2dfcabaf1c00c2c08e"}, - {file = "aiohttp-3.8.4-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:e4b09863aae0dc965c3ef36500d891a3ff495a2ea9ae9171e4519963c12ceefd"}, - {file = "aiohttp-3.8.4-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:adfbc22e87365a6e564c804c58fc44ff7727deea782d175c33602737b7feadb6"}, - {file = "aiohttp-3.8.4-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:147ae376f14b55f4f3c2b118b95be50a369b89b38a971e80a17c3fd623f280c9"}, - {file = "aiohttp-3.8.4-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:eafb3e874816ebe2a92f5e155f17260034c8c341dad1df25672fb710627c6949"}, - {file = "aiohttp-3.8.4-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:c6cc15d58053c76eacac5fa9152d7d84b8d67b3fde92709195cb984cfb3475ea"}, - {file = "aiohttp-3.8.4-cp310-cp310-win32.whl", hash = "sha256:59f029a5f6e2d679296db7bee982bb3d20c088e52a2977e3175faf31d6fb75d1"}, - {file = "aiohttp-3.8.4-cp310-cp310-win_amd64.whl", hash = "sha256:fe7ba4a51f33ab275515f66b0a236bcde4fb5561498fe8f898d4e549b2e4509f"}, - {file = "aiohttp-3.8.4-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:3d8ef1a630519a26d6760bc695842579cb09e373c5f227a21b67dc3eb16cfea4"}, - {file = "aiohttp-3.8.4-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:5b3f2e06a512e94722886c0827bee9807c86a9f698fac6b3aee841fab49bbfb4"}, - {file = "aiohttp-3.8.4-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:3a80464982d41b1fbfe3154e440ba4904b71c1a53e9cd584098cd41efdb188ef"}, - {file = "aiohttp-3.8.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8b631e26df63e52f7cce0cce6507b7a7f1bc9b0c501fcde69742130b32e8782f"}, - {file = "aiohttp-3.8.4-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3f43255086fe25e36fd5ed8f2ee47477408a73ef00e804cb2b5cba4bf2ac7f5e"}, - {file = "aiohttp-3.8.4-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:4d347a172f866cd1d93126d9b239fcbe682acb39b48ee0873c73c933dd23bd0f"}, - {file = "aiohttp-3.8.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a3fec6a4cb5551721cdd70473eb009d90935b4063acc5f40905d40ecfea23e05"}, - {file = "aiohttp-3.8.4-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:80a37fe8f7c1e6ce8f2d9c411676e4bc633a8462844e38f46156d07a7d401654"}, - {file = "aiohttp-3.8.4-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:d1e6a862b76f34395a985b3cd39a0d949ca80a70b6ebdea37d3ab39ceea6698a"}, - {file = "aiohttp-3.8.4-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:cd468460eefef601ece4428d3cf4562459157c0f6523db89365202c31b6daebb"}, - {file = "aiohttp-3.8.4-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:618c901dd3aad4ace71dfa0f5e82e88b46ef57e3239fc7027773cb6d4ed53531"}, - {file = "aiohttp-3.8.4-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:652b1bff4f15f6287550b4670546a2947f2a4575b6c6dff7760eafb22eacbf0b"}, - {file = "aiohttp-3.8.4-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:80575ba9377c5171407a06d0196b2310b679dc752d02a1fcaa2bc20b235dbf24"}, - {file = "aiohttp-3.8.4-cp311-cp311-win32.whl", hash = "sha256:bbcf1a76cf6f6dacf2c7f4d2ebd411438c275faa1dc0c68e46eb84eebd05dd7d"}, - {file = "aiohttp-3.8.4-cp311-cp311-win_amd64.whl", hash = "sha256:6e74dd54f7239fcffe07913ff8b964e28b712f09846e20de78676ce2a3dc0bfc"}, - {file = "aiohttp-3.8.4-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:880e15bb6dad90549b43f796b391cfffd7af373f4646784795e20d92606b7a51"}, - {file = "aiohttp-3.8.4-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bb96fa6b56bb536c42d6a4a87dfca570ff8e52de2d63cabebfd6fb67049c34b6"}, - {file = "aiohttp-3.8.4-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4a6cadebe132e90cefa77e45f2d2f1a4b2ce5c6b1bfc1656c1ddafcfe4ba8131"}, - {file = "aiohttp-3.8.4-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f352b62b45dff37b55ddd7b9c0c8672c4dd2eb9c0f9c11d395075a84e2c40f75"}, - {file = "aiohttp-3.8.4-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7ab43061a0c81198d88f39aaf90dae9a7744620978f7ef3e3708339b8ed2ef01"}, - {file = "aiohttp-3.8.4-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c9cb1565a7ad52e096a6988e2ee0397f72fe056dadf75d17fa6b5aebaea05622"}, - {file = "aiohttp-3.8.4-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:1b3ea7edd2d24538959c1c1abf97c744d879d4e541d38305f9bd7d9b10c9ec41"}, - {file = "aiohttp-3.8.4-cp36-cp36m-musllinux_1_1_i686.whl", hash = "sha256:7c7837fe8037e96b6dd5cfcf47263c1620a9d332a87ec06a6ca4564e56bd0f36"}, - {file = "aiohttp-3.8.4-cp36-cp36m-musllinux_1_1_ppc64le.whl", hash = "sha256:3b90467ebc3d9fa5b0f9b6489dfb2c304a1db7b9946fa92aa76a831b9d587e99"}, - {file = "aiohttp-3.8.4-cp36-cp36m-musllinux_1_1_s390x.whl", hash = "sha256:cab9401de3ea52b4b4c6971db5fb5c999bd4260898af972bf23de1c6b5dd9d71"}, - {file = "aiohttp-3.8.4-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:d1f9282c5f2b5e241034a009779e7b2a1aa045f667ff521e7948ea9b56e0c5ff"}, - {file = "aiohttp-3.8.4-cp36-cp36m-win32.whl", hash = "sha256:5e14f25765a578a0a634d5f0cd1e2c3f53964553a00347998dfdf96b8137f777"}, - {file = "aiohttp-3.8.4-cp36-cp36m-win_amd64.whl", hash = "sha256:4c745b109057e7e5f1848c689ee4fb3a016c8d4d92da52b312f8a509f83aa05e"}, - {file = "aiohttp-3.8.4-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:aede4df4eeb926c8fa70de46c340a1bc2c6079e1c40ccf7b0eae1313ffd33519"}, - {file = "aiohttp-3.8.4-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4ddaae3f3d32fc2cb4c53fab020b69a05c8ab1f02e0e59665c6f7a0d3a5be54f"}, - {file = "aiohttp-3.8.4-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c4eb3b82ca349cf6fadcdc7abcc8b3a50ab74a62e9113ab7a8ebc268aad35bb9"}, - {file = "aiohttp-3.8.4-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9bcb89336efa095ea21b30f9e686763f2be4478f1b0a616969551982c4ee4c3b"}, - {file = "aiohttp-3.8.4-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6c08e8ed6fa3d477e501ec9db169bfac8140e830aa372d77e4a43084d8dd91ab"}, - {file = "aiohttp-3.8.4-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c6cd05ea06daca6ad6a4ca3ba7fe7dc5b5de063ff4daec6170ec0f9979f6c332"}, - {file = "aiohttp-3.8.4-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:b7a00a9ed8d6e725b55ef98b1b35c88013245f35f68b1b12c5cd4100dddac333"}, - {file = "aiohttp-3.8.4-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:de04b491d0e5007ee1b63a309956eaed959a49f5bb4e84b26c8f5d49de140fa9"}, - {file = "aiohttp-3.8.4-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:40653609b3bf50611356e6b6554e3a331f6879fa7116f3959b20e3528783e699"}, - {file = "aiohttp-3.8.4-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:dbf3a08a06b3f433013c143ebd72c15cac33d2914b8ea4bea7ac2c23578815d6"}, - {file = "aiohttp-3.8.4-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:854f422ac44af92bfe172d8e73229c270dc09b96535e8a548f99c84f82dde241"}, - {file = "aiohttp-3.8.4-cp37-cp37m-win32.whl", hash = "sha256:aeb29c84bb53a84b1a81c6c09d24cf33bb8432cc5c39979021cc0f98c1292a1a"}, - {file = "aiohttp-3.8.4-cp37-cp37m-win_amd64.whl", hash = "sha256:db3fc6120bce9f446d13b1b834ea5b15341ca9ff3f335e4a951a6ead31105480"}, - {file = "aiohttp-3.8.4-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:fabb87dd8850ef0f7fe2b366d44b77d7e6fa2ea87861ab3844da99291e81e60f"}, - {file = "aiohttp-3.8.4-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:91f6d540163f90bbaef9387e65f18f73ffd7c79f5225ac3d3f61df7b0d01ad15"}, - {file = "aiohttp-3.8.4-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:d265f09a75a79a788237d7f9054f929ced2e69eb0bb79de3798c468d8a90f945"}, - {file = "aiohttp-3.8.4-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3d89efa095ca7d442a6d0cbc755f9e08190ba40069b235c9886a8763b03785da"}, - {file = "aiohttp-3.8.4-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4dac314662f4e2aa5009977b652d9b8db7121b46c38f2073bfeed9f4049732cd"}, - {file = "aiohttp-3.8.4-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:fe11310ae1e4cd560035598c3f29d86cef39a83d244c7466f95c27ae04850f10"}, - {file = "aiohttp-3.8.4-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6ddb2a2026c3f6a68c3998a6c47ab6795e4127315d2e35a09997da21865757f8"}, - {file = "aiohttp-3.8.4-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e75b89ac3bd27d2d043b234aa7b734c38ba1b0e43f07787130a0ecac1e12228a"}, - {file = "aiohttp-3.8.4-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:6e601588f2b502c93c30cd5a45bfc665faaf37bbe835b7cfd461753068232074"}, - {file = "aiohttp-3.8.4-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:a5d794d1ae64e7753e405ba58e08fcfa73e3fad93ef9b7e31112ef3c9a0efb52"}, - {file = "aiohttp-3.8.4-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:a1f4689c9a1462f3df0a1f7e797791cd6b124ddbee2b570d34e7f38ade0e2c71"}, - {file = "aiohttp-3.8.4-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:3032dcb1c35bc330134a5b8a5d4f68c1a87252dfc6e1262c65a7e30e62298275"}, - {file = "aiohttp-3.8.4-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:8189c56eb0ddbb95bfadb8f60ea1b22fcfa659396ea36f6adcc521213cd7b44d"}, - {file = "aiohttp-3.8.4-cp38-cp38-win32.whl", hash = "sha256:33587f26dcee66efb2fff3c177547bd0449ab7edf1b73a7f5dea1e38609a0c54"}, - {file = "aiohttp-3.8.4-cp38-cp38-win_amd64.whl", hash = "sha256:e595432ac259af2d4630008bf638873d69346372d38255774c0e286951e8b79f"}, - {file = "aiohttp-3.8.4-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:5a7bdf9e57126dc345b683c3632e8ba317c31d2a41acd5800c10640387d193ed"}, - {file = "aiohttp-3.8.4-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:22f6eab15b6db242499a16de87939a342f5a950ad0abaf1532038e2ce7d31567"}, - {file = "aiohttp-3.8.4-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:7235604476a76ef249bd64cb8274ed24ccf6995c4a8b51a237005ee7a57e8643"}, - {file = "aiohttp-3.8.4-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ea9eb976ffdd79d0e893869cfe179a8f60f152d42cb64622fca418cd9b18dc2a"}, - {file = "aiohttp-3.8.4-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:92c0cea74a2a81c4c76b62ea1cac163ecb20fb3ba3a75c909b9fa71b4ad493cf"}, - {file = "aiohttp-3.8.4-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:493f5bc2f8307286b7799c6d899d388bbaa7dfa6c4caf4f97ef7521b9cb13719"}, - {file = "aiohttp-3.8.4-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0a63f03189a6fa7c900226e3ef5ba4d3bd047e18f445e69adbd65af433add5a2"}, - {file = "aiohttp-3.8.4-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:10c8cefcff98fd9168cdd86c4da8b84baaa90bf2da2269c6161984e6737bf23e"}, - {file = "aiohttp-3.8.4-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:bca5f24726e2919de94f047739d0a4fc01372801a3672708260546aa2601bf57"}, - {file = "aiohttp-3.8.4-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:03baa76b730e4e15a45f81dfe29a8d910314143414e528737f8589ec60cf7391"}, - {file = "aiohttp-3.8.4-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:8c29c77cc57e40f84acef9bfb904373a4e89a4e8b74e71aa8075c021ec9078c2"}, - {file = "aiohttp-3.8.4-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:03543dcf98a6619254b409be2d22b51f21ec66272be4ebda7b04e6412e4b2e14"}, - {file = "aiohttp-3.8.4-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:17b79c2963db82086229012cff93ea55196ed31f6493bb1ccd2c62f1724324e4"}, - {file = "aiohttp-3.8.4-cp39-cp39-win32.whl", hash = "sha256:34ce9f93a4a68d1272d26030655dd1b58ff727b3ed2a33d80ec433561b03d67a"}, - {file = "aiohttp-3.8.4-cp39-cp39-win_amd64.whl", hash = "sha256:41a86a69bb63bb2fc3dc9ad5ea9f10f1c9c8e282b471931be0268ddd09430b04"}, - {file = "aiohttp-3.8.4.tar.gz", hash = "sha256:bf2e1a9162c1e441bf805a1fd166e249d574ca04e03b34f97e2928769e91ab5c"}, + {file = "aiohttp-3.8.5-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:a94159871304770da4dd371f4291b20cac04e8c94f11bdea1c3478e557fbe0d8"}, + {file = "aiohttp-3.8.5-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:13bf85afc99ce6f9ee3567b04501f18f9f8dbbb2ea11ed1a2e079670403a7c84"}, + {file = "aiohttp-3.8.5-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:2ce2ac5708501afc4847221a521f7e4b245abf5178cf5ddae9d5b3856ddb2f3a"}, + {file = "aiohttp-3.8.5-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:96943e5dcc37a6529d18766597c491798b7eb7a61d48878611298afc1fca946c"}, + {file = "aiohttp-3.8.5-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:2ad5c3c4590bb3cc28b4382f031f3783f25ec223557124c68754a2231d989e2b"}, + {file = "aiohttp-3.8.5-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:0c413c633d0512df4dc7fd2373ec06cc6a815b7b6d6c2f208ada7e9e93a5061d"}, + {file = "aiohttp-3.8.5-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:df72ac063b97837a80d80dec8d54c241af059cc9bb42c4de68bd5b61ceb37caa"}, + {file = "aiohttp-3.8.5-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c48c5c0271149cfe467c0ff8eb941279fd6e3f65c9a388c984e0e6cf57538e14"}, + {file = "aiohttp-3.8.5-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:368a42363c4d70ab52c2c6420a57f190ed3dfaca6a1b19afda8165ee16416a82"}, + {file = "aiohttp-3.8.5-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:7607ec3ce4993464368505888af5beb446845a014bc676d349efec0e05085905"}, + {file = "aiohttp-3.8.5-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:0d21c684808288a98914e5aaf2a7c6a3179d4df11d249799c32d1808e79503b5"}, + {file = "aiohttp-3.8.5-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:312fcfbacc7880a8da0ae8b6abc6cc7d752e9caa0051a53d217a650b25e9a691"}, + {file = "aiohttp-3.8.5-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:ad093e823df03bb3fd37e7dec9d4670c34f9e24aeace76808fc20a507cace825"}, + {file = "aiohttp-3.8.5-cp310-cp310-win32.whl", hash = "sha256:33279701c04351a2914e1100b62b2a7fdb9a25995c4a104259f9a5ead7ed4802"}, + {file = "aiohttp-3.8.5-cp310-cp310-win_amd64.whl", hash = "sha256:6e4a280e4b975a2e7745573e3fc9c9ba0d1194a3738ce1cbaa80626cc9b4f4df"}, + {file = "aiohttp-3.8.5-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:ae871a964e1987a943d83d6709d20ec6103ca1eaf52f7e0d36ee1b5bebb8b9b9"}, + {file = "aiohttp-3.8.5-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:461908b2578955045efde733719d62f2b649c404189a09a632d245b445c9c975"}, + {file = "aiohttp-3.8.5-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:72a860c215e26192379f57cae5ab12b168b75db8271f111019509a1196dfc780"}, + {file = "aiohttp-3.8.5-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cc14be025665dba6202b6a71cfcdb53210cc498e50068bc088076624471f8bb9"}, + {file = "aiohttp-3.8.5-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:8af740fc2711ad85f1a5c034a435782fbd5b5f8314c9a3ef071424a8158d7f6b"}, + {file = "aiohttp-3.8.5-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:841cd8233cbd2111a0ef0a522ce016357c5e3aff8a8ce92bcfa14cef890d698f"}, + {file = "aiohttp-3.8.5-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5ed1c46fb119f1b59304b5ec89f834f07124cd23ae5b74288e364477641060ff"}, + {file = "aiohttp-3.8.5-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:84f8ae3e09a34f35c18fa57f015cc394bd1389bce02503fb30c394d04ee6b938"}, + {file = "aiohttp-3.8.5-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:62360cb771707cb70a6fd114b9871d20d7dd2163a0feafe43fd115cfe4fe845e"}, + {file = "aiohttp-3.8.5-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:23fb25a9f0a1ca1f24c0a371523546366bb642397c94ab45ad3aedf2941cec6a"}, + {file = "aiohttp-3.8.5-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:b0ba0d15164eae3d878260d4c4df859bbdc6466e9e6689c344a13334f988bb53"}, + {file = "aiohttp-3.8.5-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:5d20003b635fc6ae3f96d7260281dfaf1894fc3aa24d1888a9b2628e97c241e5"}, + {file = "aiohttp-3.8.5-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:0175d745d9e85c40dcc51c8f88c74bfbaef9e7afeeeb9d03c37977270303064c"}, + {file = "aiohttp-3.8.5-cp311-cp311-win32.whl", hash = "sha256:2e1b1e51b0774408f091d268648e3d57f7260c1682e7d3a63cb00d22d71bb945"}, + {file = "aiohttp-3.8.5-cp311-cp311-win_amd64.whl", hash = "sha256:043d2299f6dfdc92f0ac5e995dfc56668e1587cea7f9aa9d8a78a1b6554e5755"}, + {file = "aiohttp-3.8.5-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:cae533195e8122584ec87531d6df000ad07737eaa3c81209e85c928854d2195c"}, + {file = "aiohttp-3.8.5-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4f21e83f355643c345177a5d1d8079f9f28b5133bcd154193b799d380331d5d3"}, + {file = "aiohttp-3.8.5-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a7a75ef35f2df54ad55dbf4b73fe1da96f370e51b10c91f08b19603c64004acc"}, + {file = "aiohttp-3.8.5-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2e2e9839e14dd5308ee773c97115f1e0a1cb1d75cbeeee9f33824fa5144c7634"}, + {file = "aiohttp-3.8.5-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c44e65da1de4403d0576473e2344828ef9c4c6244d65cf4b75549bb46d40b8dd"}, + {file = "aiohttp-3.8.5-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:78d847e4cde6ecc19125ccbc9bfac4a7ab37c234dd88fbb3c5c524e8e14da543"}, + {file = "aiohttp-3.8.5-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:c7a815258e5895d8900aec4454f38dca9aed71085f227537208057853f9d13f2"}, + {file = "aiohttp-3.8.5-cp36-cp36m-musllinux_1_1_i686.whl", hash = "sha256:8b929b9bd7cd7c3939f8bcfffa92fae7480bd1aa425279d51a89327d600c704d"}, + {file = "aiohttp-3.8.5-cp36-cp36m-musllinux_1_1_ppc64le.whl", hash = "sha256:5db3a5b833764280ed7618393832e0853e40f3d3e9aa128ac0ba0f8278d08649"}, + {file = "aiohttp-3.8.5-cp36-cp36m-musllinux_1_1_s390x.whl", hash = "sha256:a0215ce6041d501f3155dc219712bc41252d0ab76474615b9700d63d4d9292af"}, + {file = "aiohttp-3.8.5-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:fd1ed388ea7fbed22c4968dd64bab0198de60750a25fe8c0c9d4bef5abe13824"}, + {file = "aiohttp-3.8.5-cp36-cp36m-win32.whl", hash = "sha256:6e6783bcc45f397fdebc118d772103d751b54cddf5b60fbcc958382d7dd64f3e"}, + {file = "aiohttp-3.8.5-cp36-cp36m-win_amd64.whl", hash = "sha256:b5411d82cddd212644cf9360879eb5080f0d5f7d809d03262c50dad02f01421a"}, + {file = "aiohttp-3.8.5-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:01d4c0c874aa4ddfb8098e85d10b5e875a70adc63db91f1ae65a4b04d3344cda"}, + {file = "aiohttp-3.8.5-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e5980a746d547a6ba173fd5ee85ce9077e72d118758db05d229044b469d9029a"}, + {file = "aiohttp-3.8.5-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:2a482e6da906d5e6e653be079b29bc173a48e381600161c9932d89dfae5942ef"}, + {file = "aiohttp-3.8.5-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:80bd372b8d0715c66c974cf57fe363621a02f359f1ec81cba97366948c7fc873"}, + {file = "aiohttp-3.8.5-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c1161b345c0a444ebcf46bf0a740ba5dcf50612fd3d0528883fdc0eff578006a"}, + {file = "aiohttp-3.8.5-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:cd56db019015b6acfaaf92e1ac40eb8434847d9bf88b4be4efe5bfd260aee692"}, + {file = "aiohttp-3.8.5-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:153c2549f6c004d2754cc60603d4668899c9895b8a89397444a9c4efa282aaf4"}, + {file = "aiohttp-3.8.5-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:4a01951fabc4ce26ab791da5f3f24dca6d9a6f24121746eb19756416ff2d881b"}, + {file = "aiohttp-3.8.5-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:bfb9162dcf01f615462b995a516ba03e769de0789de1cadc0f916265c257e5d8"}, + {file = "aiohttp-3.8.5-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:7dde0009408969a43b04c16cbbe252c4f5ef4574ac226bc8815cd7342d2028b6"}, + {file = "aiohttp-3.8.5-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:4149d34c32f9638f38f544b3977a4c24052042affa895352d3636fa8bffd030a"}, + {file = "aiohttp-3.8.5-cp37-cp37m-win32.whl", hash = "sha256:68c5a82c8779bdfc6367c967a4a1b2aa52cd3595388bf5961a62158ee8a59e22"}, + {file = "aiohttp-3.8.5-cp37-cp37m-win_amd64.whl", hash = "sha256:2cf57fb50be5f52bda004b8893e63b48530ed9f0d6c96c84620dc92fe3cd9b9d"}, + {file = "aiohttp-3.8.5-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:eca4bf3734c541dc4f374ad6010a68ff6c6748f00451707f39857f429ca36ced"}, + {file = "aiohttp-3.8.5-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:1274477e4c71ce8cfe6c1ec2f806d57c015ebf84d83373676036e256bc55d690"}, + {file = "aiohttp-3.8.5-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:28c543e54710d6158fc6f439296c7865b29e0b616629767e685a7185fab4a6b9"}, + {file = "aiohttp-3.8.5-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:910bec0c49637d213f5d9877105d26e0c4a4de2f8b1b29405ff37e9fc0ad52b8"}, + {file = "aiohttp-3.8.5-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5443910d662db951b2e58eb70b0fbe6b6e2ae613477129a5805d0b66c54b6cb7"}, + {file = "aiohttp-3.8.5-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2e460be6978fc24e3df83193dc0cc4de46c9909ed92dd47d349a452ef49325b7"}, + {file = "aiohttp-3.8.5-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fb1558def481d84f03b45888473fc5a1f35747b5f334ef4e7a571bc0dfcb11f8"}, + {file = "aiohttp-3.8.5-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:34dd0c107799dcbbf7d48b53be761a013c0adf5571bf50c4ecad5643fe9cfcd0"}, + {file = "aiohttp-3.8.5-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:aa1990247f02a54185dc0dff92a6904521172a22664c863a03ff64c42f9b5410"}, + {file = "aiohttp-3.8.5-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:0e584a10f204a617d71d359fe383406305a4b595b333721fa50b867b4a0a1548"}, + {file = "aiohttp-3.8.5-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:a3cf433f127efa43fee6b90ea4c6edf6c4a17109d1d037d1a52abec84d8f2e42"}, + {file = "aiohttp-3.8.5-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:c11f5b099adafb18e65c2c997d57108b5bbeaa9eeee64a84302c0978b1ec948b"}, + {file = "aiohttp-3.8.5-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:84de26ddf621d7ac4c975dbea4c945860e08cccde492269db4e1538a6a6f3c35"}, + {file = "aiohttp-3.8.5-cp38-cp38-win32.whl", hash = "sha256:ab88bafedc57dd0aab55fa728ea10c1911f7e4d8b43e1d838a1739f33712921c"}, + {file = "aiohttp-3.8.5-cp38-cp38-win_amd64.whl", hash = "sha256:5798a9aad1879f626589f3df0f8b79b3608a92e9beab10e5fda02c8a2c60db2e"}, + {file = "aiohttp-3.8.5-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:a6ce61195c6a19c785df04e71a4537e29eaa2c50fe745b732aa937c0c77169f3"}, + {file = "aiohttp-3.8.5-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:773dd01706d4db536335fcfae6ea2440a70ceb03dd3e7378f3e815b03c97ab51"}, + {file = "aiohttp-3.8.5-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:f83a552443a526ea38d064588613aca983d0ee0038801bc93c0c916428310c28"}, + {file = "aiohttp-3.8.5-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1f7372f7341fcc16f57b2caded43e81ddd18df53320b6f9f042acad41f8e049a"}, + {file = "aiohttp-3.8.5-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ea353162f249c8097ea63c2169dd1aa55de1e8fecbe63412a9bc50816e87b761"}, + {file = "aiohttp-3.8.5-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:e5d47ae48db0b2dcf70bc8a3bc72b3de86e2a590fc299fdbbb15af320d2659de"}, + {file = "aiohttp-3.8.5-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d827176898a2b0b09694fbd1088c7a31836d1a505c243811c87ae53a3f6273c1"}, + {file = "aiohttp-3.8.5-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:3562b06567c06439d8b447037bb655ef69786c590b1de86c7ab81efe1c9c15d8"}, + {file = "aiohttp-3.8.5-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:4e874cbf8caf8959d2adf572a78bba17cb0e9d7e51bb83d86a3697b686a0ab4d"}, + {file = "aiohttp-3.8.5-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:6809a00deaf3810e38c628e9a33271892f815b853605a936e2e9e5129762356c"}, + {file = "aiohttp-3.8.5-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:33776e945d89b29251b33a7e7d006ce86447b2cfd66db5e5ded4e5cd0340585c"}, + {file = "aiohttp-3.8.5-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:eaeed7abfb5d64c539e2db173f63631455f1196c37d9d8d873fc316470dfbacd"}, + {file = "aiohttp-3.8.5-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:e91d635961bec2d8f19dfeb41a539eb94bd073f075ca6dae6c8dc0ee89ad6f91"}, + {file = "aiohttp-3.8.5-cp39-cp39-win32.whl", hash = "sha256:00ad4b6f185ec67f3e6562e8a1d2b69660be43070bd0ef6fcec5211154c7df67"}, + {file = "aiohttp-3.8.5-cp39-cp39-win_amd64.whl", hash = "sha256:c0a9034379a37ae42dea7ac1e048352d96286626251862e448933c0f59cbd79c"}, + {file = "aiohttp-3.8.5.tar.gz", hash = "sha256:b9552ec52cc147dbf1944ac7ac98af7602e51ea2dcd076ed194ca3c0d1c7d0bc"}, ] [package.dependencies] @@ -298,13 +298,13 @@ test = ["pytest", "pytest-cov"] [[package]] name = "certifi" -version = "2023.5.7" +version = "2023.7.22" description = "Python package for providing Mozilla's CA Bundle." optional = false python-versions = ">=3.6" files = [ - {file = "certifi-2023.5.7-py3-none-any.whl", hash = "sha256:c6c2e98f5c7869efca1f8916fed228dd91539f9f1b444c314c06eef02980c716"}, - {file = "certifi-2023.5.7.tar.gz", hash = "sha256:0f0d56dc5a6ad56fd4ba36484d6cc34451e1c6548c61daad8c320169f91eddc7"}, + {file = "certifi-2023.7.22-py3-none-any.whl", hash = "sha256:92d6037539857d8206b8f6ae472e8b77db8058fec5937a1ef3f54304089edbb9"}, + {file = "certifi-2023.7.22.tar.gz", hash = "sha256:539cc1d13202e33ca466e88b2807e29f4c13049d6d87031a3c110744495cb082"}, ] [[package]] @@ -1237,13 +1237,13 @@ files = [ [[package]] name = "requests" -version = "2.30.0" +version = "2.31.0" description = "Python HTTP for Humans." optional = false python-versions = ">=3.7" files = [ - {file = "requests-2.30.0-py3-none-any.whl", hash = "sha256:10e94cc4f3121ee6da529d358cdaeaff2f1c409cd377dbc72b825852f2f7e294"}, - {file = "requests-2.30.0.tar.gz", hash = "sha256:239d7d4458afcb28a692cdd298d87542235f4ca8d36d03a15bfc128a6559a2f4"}, + {file = "requests-2.31.0-py3-none-any.whl", hash = "sha256:58cd2187c01e70e6e26505bca751777aa9f2ee0b7f4300988b709f44e013003f"}, + {file = "requests-2.31.0.tar.gz", hash = "sha256:942c5a758f98d790eaed1a29cb6eefc7ffb0d1cf7af05c3d2791656dbd6ad1e1"}, ] [package.dependencies] diff --git a/scripts/build-apps-frontend.js b/scripts/build-apps-frontend.js deleted file mode 100644 index 04295b559..000000000 --- a/scripts/build-apps-frontend.js +++ /dev/null @@ -1,135 +0,0 @@ -const fs = require('fs') -const path = require('path') -const chalk = require('chalk') -const { hash: namehash } = require('eth-ens-namehash') -const buidlerTaskNames = require('@nomiclabs/buidler/builtin-tasks/task-names') -const hardhatTaskNames = require('hardhat/builtin-tasks/task-names') - - -const runOrWrapScript = require('./helpers/run-or-wrap-script') -const { log, logSplitter, logWideSplitter, logHeader, logTx } = require('./helpers/log') -const { exec, execLive } = require('./helpers/exec') -const { readJSON } = require('./helpers/fs') - -// this is needed for the next two `require`s to work, some kind of typescript path magic -require('@aragon/buidler-aragon/dist/bootstrap-paths') - -const { generateArtifacts } = require('@aragon/buidler-aragon/dist/src/utils/artifact/generateArtifacts') - -const { APP_NAMES } = require('./constants') -const VALID_APP_NAMES = Object.entries(APP_NAMES).map((e) => e[1]) - - -const APPS = process.env.APPS || '*' -const APPS_DIR_PATH = process.env.APPS_DIR_PATH || path.resolve(__dirname, '..', 'apps') -const LIDO_APM_ENS_NAME = 'lidopm.eth' - -async function publishAppFrontends({ - web3, - appsDirPath = APPS_DIR_PATH, - appDirs = APPS -}) { - const netId = await web3.eth.net.getId() - - logWideSplitter() - log(`Network ID: ${chalk.yellow(netId)}`) - - appsDirPath = path.resolve(appsDirPath) - - if (appDirs && appDirs !== '*') { - appDirs = appDirs.split(',') - } else { - appDirs = fs.readdirSync(appsDirPath) - } - - const cwd = process.cwd() - - for (const appDir of appDirs) { - let app - try { - app = await publishAppFrotnend(appDir, appsDirPath, LIDO_APM_ENS_NAME) - } finally { - process.chdir(cwd) - } - } -} - -async function publishAppFrotnend(appDir, appsDirPath, lidoApmEnsName) { - logHeader(`Publishing frontend of the app '${appDir}'`) - - const appRootPath = path.resolve(appsDirPath, appDir) - const { appFullName, contractPath } = await readArappJSON(appRootPath, network.name) - - log(`App full name: ${chalk.yellow(appFullName)}`) - - if (!appFullName.endsWith('.' + lidoApmEnsName)) { - throw new Error(`app full name is not a subdomain of the Lido APM ENS domain ${lidoApmEnsName}`) - } - - const appName = appFullName.substring(0, appFullName.indexOf('.')) - log(`App name: ${chalk.yellow(appName)}`) - - if (VALID_APP_NAMES.indexOf(appName) === -1) { - throw new Error(`app name is not recognized; valid app names are: ${VALID_APP_NAMES.join(', ')}`) - } - - const appId = namehash(appFullName) - log(`App ID: ${chalk.yellow(appId)}`) - - logSplitter() - - log('Removing output directory...') - const distPath = path.join(appRootPath, 'dist') - await exec(`rm -rf ${distPath}`) - - await execLive('yarn', { - args: ['build'], - cwd: path.join(appRootPath, 'app') - }) - - logSplitter() - log('Generating artifacts...') - - process.chdir(appRootPath) - - const wrappedRun = async (taskName, ...args) => { - if (taskName !== buidlerTaskNames.TASK_FLATTEN_GET_FLATTENED_SOURCE) { - return await run(taskName) - } - // buidler-aragon tries to get flattened source code of all contracts and fails to - // parce Solidity syntax newer than 0.4 (which we have in non-Aragon contracts), so - // here we're flattening only the app's dependency graph instead - return await run(hardhatTaskNames.TASK_FLATTEN_GET_FLATTENED_SOURCE, { - files: [contractPath] - }) - } - - const bre = { artifacts, network, run: wrappedRun } - await generateArtifacts(distPath, bre) - - logSplitter() - - log(`App dist: ${chalk.yellow(distPath)}`) - -} - -async function readArappJSON(appRoot, netName) { - const arappJSON = await readJSON(path.join(appRoot, 'arapp.json')) - const appFullName = getAppName(arappJSON, netName) - const contractPath = path.resolve(appRoot, arappJSON.path) - return { appFullName, contractPath } -} - -function getAppName(arappJSON, netName) { - const { environments } = arappJSON - if (!environments) { - return null - } - if (environments[netName]) { - // NOTE: assuming that Aragon environment is named after the network - return environments[netName].appName - } - return (environments.default || {}).appName || null -} - -module.exports = runOrWrapScript(publishAppFrontends, module) diff --git a/scripts/hardhat-tasks.js b/scripts/hardhat-tasks.js index 92fee124e..aa9936ece 100644 --- a/scripts/hardhat-tasks.js +++ b/scripts/hardhat-tasks.js @@ -4,7 +4,7 @@ task(`tx`, `Performs a transaction`) .addOptionalParam(`wait`, `The number of seconds to wait before sending the transaction`) .addOptionalParam(`gasPrice`, `Gas price`) .addOptionalParam(`nonce`, `Nonce`) - .setAction(async ({ file, from: fromArg, gasPrice, nonce, wait: waitSec = 5 }) => { + .setAction(async ({ file, from: fromArg, gasPrice, nonce, wait: waitSec = 1 }) => { const netId = await web3.eth.net.getId() console.error('====================') @@ -14,7 +14,7 @@ task(`tx`, `Performs a transaction`) const data = JSON.parse(require('fs').readFileSync(file)) if (fromArg) { - console.error(`Using the sender address provided via the commandline argument: ${fromArg}`) + console.error(`Using the sender address provided via the command line argument: ${fromArg}`) data.from = fromArg } @@ -66,7 +66,7 @@ task(`tx`, `Performs a transaction`) if (!receipt.status) { console.error('====================') - console.error(`An error occured:`, receipt.error) + console.error(`An error occurred:`, receipt.error) } if (receipt.contractAddress) { @@ -98,7 +98,7 @@ task('ens-assign', `Assigns/transfers ENS node owner`) if (owner.toLowerCase() !== params.to.toLowerCase()) { throw new Error(`the owner '${owner}' is different from the expected '${params.to}'`) } - console.error(chalk.green('✓'), `the ownsership was successfully updated`) + console.error(chalk.green('✓'), `the ownership was successfully updated`) }) task('list-accts', `List accounts and their balances`) diff --git a/scripts/helpers/assert.js b/scripts/helpers/assert.js index 783615980..86c93d32b 100644 --- a/scripts/helpers/assert.js +++ b/scripts/helpers/assert.js @@ -1,5 +1,5 @@ const { assert } = require('chai') -const { toChecksumAddress } = require('web3-utils') +const { toChecksumAddress, isHexStrict } = require('web3-utils') const { log } = require('./log') @@ -13,6 +13,23 @@ assert.addressEqual = (actual, expected, msg) => { assert.equal(toChecksumAddress(actual), toChecksumAddress(expected), msg) } +assert.arrayOfAddressesEqual = (actual, expected, msg) => { + assert.equal(actual.length, expected.length, msg) + + const actualSorted = [...actual].sort() + const expectedSorted = [...expected].sort() + + for (let i = 0; i < actual.length; i++) { + assert.equal(toChecksumAddress(actualSorted[i]), toChecksumAddress(expectedSorted[i]), msg) + } +} + +assert.hexEqual = (actual, expected, msg) => { + assert.isTrue(isHexStrict(actual), `Actual string ${actual} is not a valid hex string`) + assert.isTrue(isHexStrict(expected), `Expected string ${expected} is not a valid hex string`) + assert.equal(actual.toLowerCase(), expected.toLowerCase(), msg) +} + assert.log = (doAssert, ...args) => { doAssert(...args) log.success(args[args.length - 1]) diff --git a/scripts/helpers/deploy.js b/scripts/helpers/deploy.js index d4f37cee9..b10a63e5b 100644 --- a/scripts/helpers/deploy.js +++ b/scripts/helpers/deploy.js @@ -1,10 +1,10 @@ -const { readNetworkState, persistNetworkState2 } = require('./persisted-network-state') +const { readNetworkState, persistNetworkState } = require('./persisted-network-state') const { artifacts, ethers } = require('hardhat') const fs = require('fs').promises const chalk = require('chalk') const { assert } = require('chai') -const { log, logDeploy, logDeployTxData } = require('./log') +const { log, logDeploy, logDeployTxData, OK } = require('./log') const { getTxData } = require('./tx-data') @@ -12,10 +12,34 @@ const GAS_PRICE = process.env.GAS_PRICE || null const GAS_PRIORITY_FEE = process.env.GAS_PRIORITY_FEE || null const GAS_MAX_FEE = process.env.GAS_MAX_FEE || null -let TOTAL_GAS_USED = 0 -function getTotalGasUsed() { - return TOTAL_GAS_USED +class TotalGasCounterPrivate { + constructor() { + this.totalGasUsed = 0 + } +} +class TotalGasCounter { + constructor() { + throw new Error('Use TotalGasCounter.getInstance()'); + } + static getInstance() { + if (!TotalGasCounter.instance) { + TotalGasCounter.instance = new TotalGasCounterPrivate() + } + return TotalGasCounter.instance + } + static add(gasUsed) { + return this.getInstance().totalGasUsed += gasUsed + } + static getTotalGasUsed() { + return this.getInstance().totalGasUsed + } + static async incrementTotalGasUsedInStateFile() { + const netId = await web3.eth.net.getId() + const state = readNetworkState(network.name, netId) + state.initialDeployTotalGasUsed += TotalGasCounter.getTotalGasUsed() + persistNetworkState(network.name, netId, state) + } } async function getDeploymentGasUsed(contract) { @@ -50,6 +74,19 @@ async function getDeployTx(artifactName, opts = {}) { return await getTxData(txObj) } +async function makeTx(contract, funcName, args, txParams) { + console.log(`${contract.constructor._json.contractName}[${contract.address}].${funcName}()...`) + const receipt = await contract[funcName](...args, txParams) + const gasUsed = receipt.gasUsed ? receipt.gasUsed : receipt.receipt.gasUsed + if (gasUsed === undefined) { + console.log({ receipt }) + assert(false) + } + console.log(`${OK} tx: ${receipt.tx} (gasUsed ${gasUsed})`) + TotalGasCounter.add(gasUsed) + return receipt +} + async function deploy(artifactName, artifacts, deploy) { const Artifact = artifacts.require(artifactName) return await logDeploy(artifactName, deploy ? deploy(Artifact) : Artifact.new()) @@ -89,6 +126,10 @@ async function getDeployed(artifactName, deployTxHash) { return await Artifact.at(receipt.contractAddress) } +async function getContractPath(contractName) { + return await artifacts.require(contractName)._hArtifact.sourceName +} + async function getTxBlock(txHash) { const receipt = await web3.eth.getTransactionReceipt(txHash) if (!receipt) { @@ -114,12 +155,11 @@ async function assertDeployedBytecode(address, artifact, desc = '') { const bytecode = await web3.eth.getCode(address) const nameDesc = artifact.contractName ? chalk.yellow(artifact.contractName) : 'the expected one' const checkDesc = `${desc ? desc + ': ' : ''}the bytecode at ${chalk.yellow(address)} matches ${nameDesc}` - // TODO: restore the check - // if (bytecode.toLowerCase() !== artifact.deployedBytecode.toLowerCase()) { - // console.log({bytecode: bytecode.toLowerCase()}) - // console.log({deployedBytecode: artifact.deployedBytecode.toLowerCase()}) - // } - // assert.isTrue(bytecode.toLowerCase() === artifact.deployedBytecode.toLowerCase(), checkDesc) + if (bytecode.toLowerCase() !== artifact.deployedBytecode.toLowerCase()) { + console.log({bytecode: bytecode.toLowerCase()}) + console.log({deployedBytecode: artifact.deployedBytecode.toLowerCase()}) + } + assert.isTrue(bytecode.toLowerCase() === artifact.deployedBytecode.toLowerCase(), checkDesc) log.success(checkDesc) } @@ -193,63 +233,85 @@ async function deployWithoutProxy(nameInState, artifactName, deployer, construct const netId = await web3.eth.net.getId() const state = readNetworkState(network.name, netId) - console.log(`Deploying ${artifactName}...`) - if (constructorArgs.length > 0) { - console.log(`Constructor args: ${constructorArgs}`) - } + process.stdout.write(`Deploying ${artifactName} (NO proxy)... `) const contract = await deployContract(artifactName, constructorArgs, deployer) - console.log(`${artifactName} (NO proxy): ${contract.address}`) const gasUsed = await getDeploymentGasUsed(contract) - console.log(` gas used ${gasUsed}`) - TOTAL_GAS_USED += gasUsed - - if (nameInState) { - persistNetworkState2(network.name, netId, state, { - [nameInState]: { - "contract": artifactName, - [addressFieldName]: contract.address, - "constructorArgs": constructorArgs, - } - }) + console.log(`done: ${contract.address} (gas used ${gasUsed})`) + TotalGasCounter.add(gasUsed) + + state[nameInState] = { + ...state[nameInState], + contract: await getContractPath(artifactName), + [addressFieldName]: contract.address, + constructorArgs: constructorArgs, } + persistNetworkState(network.name, netId, state) + console.log() return contract.address } +async function deployImplementation(nameInState, artifactName, deployer, constructorArgs=[]) { + const netId = await web3.eth.net.getId() + const state = readNetworkState(network.name, netId) + + process.stdout.write(`Deploying implementation for proxy of ${artifactName}... `) + const contract = await deployContract(artifactName, constructorArgs, deployer) + const gasUsed = await getDeploymentGasUsed(contract) + TotalGasCounter.add(gasUsed) + console.log(`done: ${contract.address} (gas used ${gasUsed})`) + + state[nameInState] = { ...state[nameInState] } + state[nameInState].implementation = { + contract: await getContractPath(artifactName), + address: contract.address, + constructorArgs: constructorArgs, + } + persistNetworkState(network.name, netId, state) + return contract +} + async function deployBehindOssifiableProxy(nameInState, artifactName, proxyOwner, deployer, constructorArgs=[], implementation=null) { const netId = await web3.eth.net.getId() const state = readNetworkState(network.name, netId) + const proxyContractName = 'OssifiableProxy' if (implementation === null) { - console.log(`Deploying implementation for proxy of ${artifactName}`) + process.stdout.write(`Deploying implementation for proxy of ${artifactName}... `) const contract = await deployContract(artifactName, constructorArgs, deployer) - const gasUsed = await getDeploymentGasUsed(contract) - TOTAL_GAS_USED += gasUsed + TotalGasCounter.add(gasUsed) implementation = contract.address - console.log(`${artifactName} implementation: ${implementation} (gas used ${gasUsed})`) + console.log(`done: ${implementation} (gas used ${gasUsed})`) + } else { console.log(`Using pre-deployed implementation of ${artifactName}: ${implementation}`) } - console.log(`Deploying OssifiableProxy for ${artifactName}...`) - const proxy = await deployContract("OssifiableProxy", [implementation, proxyOwner, []], deployer) + process.stdout.write(`Deploying OssifiableProxy for ${artifactName}... `) + const proxyConstructorArgs = [implementation, proxyOwner, '0x'] + const proxy = await deployContract(proxyContractName, proxyConstructorArgs, deployer) const gasUsed = await getDeploymentGasUsed(proxy) - TOTAL_GAS_USED += gasUsed - console.log(`${artifactName} proxy: ${proxy.address} (owner is ${proxyOwner}) (gas used ${gasUsed})`) - - persistNetworkState2(network.name, netId, state, { - [nameInState]: { - "contract": artifactName, - "implementation": implementation, - "address": proxy.address, - "constructorArgs": constructorArgs, - } - }) + TotalGasCounter.add(gasUsed) + console.log(`done: ${proxy.address} (gas used ${gasUsed})`) + + state[nameInState] = { ...state[nameInState] } + state[nameInState].proxy = { + contract: await getContractPath(proxyContractName), + address: proxy.address, + constructorArgs: proxyConstructorArgs, + } + state[nameInState].implementation = { + contract: await getContractPath(artifactName), + address: implementation, + constructorArgs: constructorArgs, + } + persistNetworkState(network.name, netId, state) + console.log() return proxy.address } @@ -257,27 +319,22 @@ async function updateProxyImplementation(nameInState, artifactName, proxyAddress const netId = await web3.eth.net.getId() const state = readNetworkState(network.name, netId) - const OssifiableProxy = await artifacts.require('OssifiableProxy') - const proxy = await OssifiableProxy.at(proxyAddress) + const proxy = await artifacts.require('OssifiableProxy').at(proxyAddress) - const Contract = await artifacts.require(artifactName) - const implementation = await Contract.new(...constructorArgs, { - from: proxyOwner, - gasPrice: GAS_PRICE, - }) - - await proxy.proxy__upgradeTo(implementation.address, { - from: proxyOwner, - gasPrice: GAS_PRICE, - }) - - persistNetworkState2(network.name, netId, state, { - [nameInState]: { - "contract": artifactName, - "implementation": implementation.address, - "constructorArgs": constructorArgs, - } - }) + const implementation = await deployContract(artifactName, constructorArgs, proxyOwner) + const gasUsed = await getDeploymentGasUsed(implementation) + TotalGasCounter.add(gasUsed) + + await makeTx(proxy, 'proxy__upgradeTo', [implementation.address], { from: proxyOwner }) + + state[nameInState] = { ...state[nameInState] } + state[nameInState].implementation = { + contract: await getContractPath(artifactName), + address: implementation.address, + constructorArgs: constructorArgs, + } + persistNetworkState(network.name, netId, state) + return implementation } module.exports = { @@ -294,7 +351,11 @@ module.exports = { withArgs, getDeployTxParams, deployWithoutProxy, + deployContract, + deployImplementation, deployBehindOssifiableProxy, updateProxyImplementation, - getTotalGasUsed, + getContractPath, + makeTx, + TotalGasCounter, } diff --git a/scripts/helpers/persisted-network-state.js b/scripts/helpers/persisted-network-state.js index 7299b2730..c8a47fe9a 100644 --- a/scripts/helpers/persisted-network-state.js +++ b/scripts/helpers/persisted-network-state.js @@ -2,41 +2,27 @@ const fs = require('fs') const path = require('path') const { log } = require('./log') +const { assert } = require('../../test/helpers/assert') const NETWORK_STATE_FILE_BASENAME = process.env.NETWORK_STATE_FILE_BASENAME || 'deployed' const NETWORK_STATE_FILE_DIR = process.env.NETWORK_STATE_FILE_DIR || '.' function readNetworkState(netName, netId) { const fileName = _getFileName(netName, NETWORK_STATE_FILE_BASENAME, NETWORK_STATE_FILE_DIR) - log(`Reading network state from ${fileName}...`) const state = _readNetworkStateFile(fileName, netId) - if (state.networkId !== netId) { - throw new Error(`network id (${netId}) doesn't match the one in the state file (${state.networkId})`) - } return state } function persistNetworkState(netName, netId, state, updates = undefined) { - state.networkId = netId + assert.equal(netId, state.networkId, `Network id ${netId} does not match one in state file ${state.networkId}`) if (updates) { updateNetworkState(state, updates) } const fileName = _getFileName(netName, NETWORK_STATE_FILE_BASENAME, NETWORK_STATE_FILE_DIR) - log(`Writing network state to ${fileName}...`) - _writeNetworkStateFile(fileName, state) -} - -function persistNetworkState2(netName, netId, state, updates = undefined) { - state.networkId = netId - if (updates) { - updateNetworkState2(state, updates) - } - const fileName = _getFileName(netName, NETWORK_STATE_FILE_BASENAME, NETWORK_STATE_FILE_DIR) - log(`Writing network state to ${fileName}...`) _writeNetworkStateFile(fileName, state) } -function updateNetworkState2(state, newState) { +function updateNetworkState(state, newState) { Object.keys(newState).forEach((key) => { const previousValue = state[key] const value = newState[key] @@ -46,22 +32,6 @@ function updateNetworkState2(state, newState) { }) } -function updateNetworkState(state, newState) { - Object.keys(newState).forEach((key) => { - const value = newState[key] - if (value != null) { - if (value.address) { - state[`${key}Address`] = value.address - if (value.constructorArgs) { - state[`${key}ConstructorArgs`] = value.constructorArgs - } - } else { - state[key] = value - } - } - }) -} - function assertRequiredNetworkState(state, requiredStateNames) { const missingState = requiredStateNames.filter((key) => !state[key]) if (missingState.length) { @@ -76,29 +46,43 @@ function _getFileName(netName, baseName, dir) { return path.resolve(dir, `${baseName}-${netName}.json`) } +function readStateFile(fileName) { + const data = fs.readFileSync(fileName, 'utf8') + try { + return JSON.parse(data) + } catch (err) { + throw new Error(`malformed network state file ${fileName}: ${err.message}`) + } +} + function _readNetworkStateFile(fileName, netId) { if (!fs.existsSync(fileName)) { const state = { networkId: netId } _writeNetworkStateFile(fileName, state) return state } - const data = fs.readFileSync(fileName, 'utf8') - try { - return JSON.parse(data) - } catch (err) { - throw new Error(`malformed network state file ${fileName}: ${err.message}`) + return readStateFile(fileName) +} + +function sortKeysAlphabetically(object) { + const sortedObject = {} + const sortedKeys = Object.keys(object).sort() + for (const key of sortedKeys) { + sortedObject[key] = object[key] } + return sortedObject } function _writeNetworkStateFile(fileName, state) { - const data = JSON.stringify(state, null, ' ') + const stateSorted = sortKeysAlphabetically(state) + const data = JSON.stringify(stateSorted, null, ' ') fs.writeFileSync(fileName, data + '\n', 'utf8') } module.exports = { readNetworkState, persistNetworkState, - persistNetworkState2, updateNetworkState, assertRequiredNetworkState, + readStateFile, } diff --git a/scripts/helpers/run-or-wrap-script.js b/scripts/helpers/run-or-wrap-script.js index 8e45d363f..bdee3a0a8 100644 --- a/scripts/helpers/run-or-wrap-script.js +++ b/scripts/helpers/run-or-wrap-script.js @@ -15,7 +15,6 @@ module.exports = (scriptFn, mainModule) => { // Buidler executes scripts in a forked subprocess scriptFn({ artifacts: globalArtifacts, web3: globalWeb3 }) .then(() => { - console.error('All done!') process.exit(0) }) .catch((err) => { diff --git a/scripts/scratch/00-populate-deploy-artifact-from-env.js b/scripts/scratch/00-populate-deploy-artifact-from-env.js new file mode 100644 index 000000000..cd5e05102 --- /dev/null +++ b/scripts/scratch/00-populate-deploy-artifact-from-env.js @@ -0,0 +1,43 @@ +const { ZERO_ADDRESS } = require('@aragon/contract-helpers-test') +const runOrWrapScript = require('../helpers/run-or-wrap-script') +const { readNetworkState, persistNetworkState } = require('../helpers/persisted-network-state') + +const DEPLOYER = process.env.DEPLOYER +const GATE_SEAL_FACTORY = process.env.GATE_SEAL_FACTORY +const GENESIS_TIME = parseInt(process.env.GENESIS_TIME) +const DEPOSIT_CONTRACT = process.env.DEPOSIT_CONTRACT + +async function saveDeployParameters({ web3, artifacts }) { + const netId = await web3.eth.net.getId() + + console.log('Using env values:') + console.log({ + DEPLOYER, + GATE_SEAL_FACTORY, + GENESIS_TIME, + DEPOSIT_CONTRACT, + }) + + const state = readNetworkState(network.name, netId) + const ldoHolder = Object.keys(state.vestingParams.holders)[0] + const gateSealAddress = (GATE_SEAL_FACTORY === null || GATE_SEAL_FACTORY === ZERO_ADDRESS) + ? ZERO_ADDRESS : '' + + state.networkId = await web3.eth.net.getId() + state.chainId = (await ethers.provider.getNetwork()).chainId + state.deployer = DEPLOYER + state.gateSeal = { + ...state.gateSeal, + factoryAddress: GATE_SEAL_FACTORY, + sealingCommittee: ldoHolder, + address: gateSealAddress, + } + state.chainSpec = { + ...state.chainSpec, + genesisTime: GENESIS_TIME, + depositContract: DEPOSIT_CONTRACT, + } + persistNetworkState(network.name, netId, state) +} + +module.exports = runOrWrapScript(saveDeployParameters, module) diff --git a/scripts/scratch/deploy-beacon-deposit-contract.js b/scripts/scratch/01-deploy-deposit-contract.js similarity index 53% rename from scripts/scratch/deploy-beacon-deposit-contract.js rename to scripts/scratch/01-deploy-deposit-contract.js index fc86ad194..ec72f029f 100644 --- a/scripts/scratch/deploy-beacon-deposit-contract.js +++ b/scripts/scratch/01-deploy-deposit-contract.js @@ -1,8 +1,8 @@ const chalk = require('chalk') const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log, logSplitter, logWideSplitter, logHeader, logTx, logDeploy } = require('../helpers/log') -const { deploy, useOrDeploy, withArgs } = require('../helpers/deploy') +const { log, logSplitter, logWideSplitter } = require('../helpers/log') +const { deployWithoutProxy, TotalGasCounter } = require('../helpers/deploy') const { readNetworkState, persistNetworkState } = require('../helpers/persisted-network-state') const NETWORK_STATE_FILE = process.env.NETWORK_STATE_FILE || 'deployed.json' @@ -15,30 +15,31 @@ async function deployBeaconDepositContract({ web3, artifacts, networkStateFile = const state = readNetworkState(network.name, netId) const [firstAccount] = await web3.eth.getAccounts() - const { daoInitialSettings = { beaconSpec: { depositContractAddress: '' } } } = state + const chainSpec = state.chainSpec let depositContractAddress - if (daoInitialSettings.beaconSpec && daoInitialSettings.beaconSpec.depositContractAddress) { - depositContractAddress = daoInitialSettings.beaconSpec.depositContractAddress + if (chainSpec.depositContract) { + depositContractAddress = chainSpec.depositContract } - const { depositContract } = await useOrDeployDepositContract({ + depositContractAddress = await useOrDeployDepositContract({ artifacts, owner: firstAccount, - depositContractAddress: depositContractAddress || state.depositContractAddress + depositContractAddress, }) - daoInitialSettings.beaconSpec.depositContractAddress = depositContract.address + state.chainSpec.depositContract = depositContractAddress + persistNetworkState(network.name, netId, state) logSplitter() - persistNetworkState(network.name, netId, state, { depositContract, daoInitialSettings }) + + await TotalGasCounter.incrementTotalGasUsedInStateFile() } async function useOrDeployDepositContract({ artifacts, owner, depositContractAddress }) { if (depositContractAddress) { log(`Using DepositContract at: ${chalk.yellow(depositContractAddress)}`) const depositContract = await artifacts.require('DepositContract').at(depositContractAddress) - return { depositContract } + return depositContract.address } - const depositContract = await deploy('DepositContract', artifacts, withArgs({ from: owner })) - return { depositContract } + return await deployWithoutProxy('depositContract', 'DepositContract', owner) } module.exports = runOrWrapScript(deployBeaconDepositContract, module) diff --git a/scripts/scratch/01-deploy-lido-template-and-bases.js b/scripts/scratch/01-deploy-lido-template-and-bases.js deleted file mode 100644 index 7a99710ae..000000000 --- a/scripts/scratch/01-deploy-lido-template-and-bases.js +++ /dev/null @@ -1,60 +0,0 @@ -const chalk = require('chalk') - -const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log, yl, gr } = require('../helpers/log') -const { saveDeployTx } = require('../helpers/deploy') -const { deployWithoutProxy } = require('../helpers/deploy') -const { readNetworkState, assertRequiredNetworkState, persistNetworkState } = require('../helpers/persisted-network-state') -const { APP_NAMES } = require('../constants') - -const REQUIRED_NET_STATE = [ - 'ensAddress', - 'daoFactoryAddress', - 'miniMeTokenFactoryAddress', - 'aragonIDAddress', - 'apmRegistryFactoryAddress', - 'multisigAddress' -] - -async function deployTemplate({ web3, artifacts }) { - const netId = await web3.eth.net.getId() - - log.splitter() - log(`Network ID: ${chalk.yellow(netId)}`) - - const state = readNetworkState(network.name, netId) - assertRequiredNetworkState(state, REQUIRED_NET_STATE) - - const daoTemplateConstructorArgs = [ - state.multisigAddress, - state.daoFactoryAddress, - state.ensAddress, - state.miniMeTokenFactoryAddress, - state.aragonIDAddress, - state.apmRegistryFactoryAddress - ] - - log.splitter() - - await deployWithoutProxy('lidoTemplate', 'LidoTemplate', state.multisigAddress, daoTemplateConstructorArgs) - const daoTemplateDeployBlock = (await ethers.provider.getBlock('latest')).number - - await deployWithoutProxy(`app:${APP_NAMES.LIDO}`, 'Lido', state.multisigAddress, [], 'implementation') - - await deployWithoutProxy(`app:${APP_NAMES.ORACLE}`, 'LegacyOracle', state.multisigAddress, [], 'implementation') - - await deployWithoutProxy(`app:${APP_NAMES.NODE_OPERATORS_REGISTRY}`, 'NodeOperatorsRegistry', state.multisigAddress, [], 'implementation') - - persistNetworkState(network.name, netId, readNetworkState(network.name, netId), { - daoTemplateDeployBlock, - }) - - log.splitter() - log(gr(`Before continuing the deployment, please send all contract creation transactions`)) - log(gr(`that you can find in the files listed above. You may use a multisig address`)) - log(gr(`if it supports deploying new contract instances.`)) - log.splitter() - -} - -module.exports = runOrWrapScript(deployTemplate, module) diff --git a/scripts/scratch/deploy-aragon-env.js b/scripts/scratch/02-deploy-aragon-env.js similarity index 66% rename from scripts/scratch/deploy-aragon-env.js rename to scripts/scratch/02-deploy-aragon-env.js index c5f74fba7..8b1de9522 100644 --- a/scripts/scratch/deploy-aragon-env.js +++ b/scripts/scratch/02-deploy-aragon-env.js @@ -4,14 +4,13 @@ const keccak256 = require('js-sha3').keccak_256 const getAccounts = require('@aragon/os/scripts/helpers/get-accounts') const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log, logSplitter, logWideSplitter, logHeader, logTx, logDeploy } = require('../helpers/log') -const { deploy, useOrDeploy, withArgs } = require('../helpers/deploy') -const { readNetworkState, persistNetworkState, updateNetworkState } = require('../helpers/persisted-network-state') +const { log, logSplitter, logWideSplitter, logHeader, logTx } = require('../helpers/log') +const { deploy, useOrDeploy, withArgs, deployImplementation, TotalGasCounter } = require('../helpers/deploy') +const { readNetworkState, persistNetworkState } = require('../helpers/persisted-network-state') const { deployAPM } = require('../components/apm') const { assignENSName } = require('../components/ens') -const OWNER = process.env.OWNER const ARAGON_ENS_LABEL = process.env.ARAGON_ENS_LABEL || 'aragonpm' const NETWORK_STATE_FILE = process.env.NETWORK_STATE_FILE || 'deployed.json' @@ -24,90 +23,152 @@ async function deployAragonEnv({ web3, artifacts, networkStateFile = NETWORK_STA logWideSplitter() log(`Network ID: ${chalk.yellow(netId)}`) - const state = readNetworkState(network.name, netId) + let state = readNetworkState(network.name, netId) - if (state.owner) { - const lowercaseOwner = state.owner.toLowerCase() + if (state.deployer) { + const lowercaseOwner = state.deployer.toLowerCase() if (!accounts.some((acc) => acc.toLowerCase() === lowercaseOwner)) { - throw new Error(`owner account ${state.owner} is missing from provided accounts`) + throw new Error(`owner account ${state.deployer} is missing from provided accounts`) } } else { - state.owner = accounts[0] - log(`Setting owner to the first provided account: ${chalk.yellow(state.owner)}`) + state.deployer = accounts[0] + log(`Setting owner to the first provided account: ${chalk.yellow(state.deployer)}`) } if (!state.aragonEnsLabelName) { state.aragonEnsLabelName = ARAGON_ENS_LABEL log(`Using Aragon ENS label: ${state.aragonEnsLabelName}`) } + persistNetworkState(network.name, netId, state) logHeader(`ENS`) - const ensResults = await useOrDeployENS({ + const { ens, ensFactory } = await useOrDeployENS({ artifacts, - owner: state.owner, - ensAddress: state.ensAddress + owner: state.deployer, + ensAddress: state.ensAddress, }) - updateNetworkState(state, ensResults) + state = readNetworkState(network.name, netId) + state.ens = { + address: ens.address, + constructorArgs: ens.constructorArgs, + } + state.ensFactory = { + address: ensFactory.address, + constructorArgs: ensFactory.constructorArgs, + } persistNetworkState(network.name, netId, state) logHeader(`DAO factory`) - const daoFactoryResults = await useOrDeployDaoFactory({ + const { daoFactory, evmScriptRegistryFactory } = await useOrDeployDaoFactory({ artifacts, - owner: state.owner, + owner: state.deployer, daoFactoryAddress: state.daoFactoryAddress }) - updateNetworkState(state, daoFactoryResults) + state = readNetworkState(network.name, netId) + state.daoFactory = { + address: daoFactory.address, + constructorArgs: daoFactory.constructorArgs, + } + state.evmScriptRegistryFactory = { + address: evmScriptRegistryFactory.address, + constructorArgs: evmScriptRegistryFactory.constructorArgs, + } persistNetworkState(network.name, netId, state) logHeader(`APM registry factory`) - const apmRegistryFactoryResults = await useOrDeployAPMRegistryFactory({ + const { + apmRegistryBase, + apmRepoBase, + ensSubdomainRegistrarBase, + apmRegistryFactory + } = await useOrDeployAPMRegistryFactory({ artifacts, - owner: state.owner, - ens: ensResults.ens, - daoFactory: daoFactoryResults.daoFactory, + owner: state.deployer, + ens: ens, + daoFactory: daoFactory, apmRegistryFactoryAddress: state.apmRegistryFactoryAddress, apmRegistryBaseAddress: state.apmRegistryBaseAddress, apmRepoBaseAddress: state.apmRepoBaseAddress, - ensSubdomainRegistrarBaseAddress: state.ensSubdomainRegistrarBaseAddress + ensSubdomainRegistrarBaseAddress: state.ensSubdomainRegistrarBaseAddress, }) - updateNetworkState(state, apmRegistryFactoryResults) + state = readNetworkState(network.name, netId) + state.apmRegistry = { + implementation: { + address: apmRegistryBase, + }, + } + state.apmRepo = { + implementation: { + address: apmRepoBase.address, + constructorArgs: apmRepoBase.constructorArgs, + }, + } + state.ensSubdomainRegistrar = { + implementation: { + address: ensSubdomainRegistrarBase.address, + constructorArgs: ensSubdomainRegistrarBase.constructorArgs, + }, + } + state.apmRegistryFactory = { + address: apmRegistryFactory.address, + constructorArgs: apmRegistryFactory.constructorArgs, + } persistNetworkState(network.name, netId, state) logHeader(`Aragon APM`) - const apmResults = await deployAPM({ + const { + apmRegistry, + ensNodeName, + ensNode, + } = await deployAPM({ web3, artifacts, - owner: state.owner, + owner: state.deployer, labelName: state.aragonEnsLabelName, - ens: ensResults.ens, - apmRegistryFactory: apmRegistryFactoryResults.apmRegistryFactory, + ens: ens, + apmRegistryFactory: apmRegistryFactory, apmRegistryAddress: state.aragonApmRegistryAddress }) - updateNetworkState(state, { - aragonApmRegistry: apmResults.apmRegistry, - aragonEnsNodeName: apmResults.ensNodeName, - aragonEnsNode: apmResults.ensNode - }) + state = readNetworkState(network.name, netId) + state.ensNode = { + nodeName: ensNodeName, + nodeId: ensNode, + } + state.apmRegistry = { + proxy: { + address: apmRegistry.address, + }, + } persistNetworkState(network.name, netId, state) logHeader(`MiniMeTokenFactory`) - const tokenFactoryResults = await deployMiniMeTokenFactory({ + const { miniMeTokenFactory } = await deployMiniMeTokenFactory({ artifacts, - owner: state.owner, + owner: state.deployer, miniMeTokenFactoryAddress: state.miniMeTokenFactoryAddress }) - updateNetworkState(state, tokenFactoryResults) + state = readNetworkState(network.name, netId) + state.miniMeTokenFactory = { + address: miniMeTokenFactory.address, + constructorArgs: miniMeTokenFactory.constructorArgs, + } persistNetworkState(network.name, netId, state) logHeader('AragonID') - const aragonIDResults = await deployAragonID({ + const { aragonID } = await deployAragonID({ artifacts, - owner: state.owner, - ens: ensResults.ens, + owner: state.deployer, + ens: ens, aragonIDAddress: state.aragonIDAddress }) - updateNetworkState(state, aragonIDResults) + state = readNetworkState(network.name, netId) + state.aragonID = { + address: aragonID.address, + constructorArgs: aragonID.constructorArgs, + } persistNetworkState(network.name, netId, state) + + await TotalGasCounter.incrementTotalGasUsedInStateFile() } async function useOrDeployENS({ artifacts, owner, ensAddress }) { @@ -139,7 +200,6 @@ async function deployENS({ artifacts, owner }) { } async function useOrDeployDaoFactory({ artifacts, owner, daoFactoryAddress }) { - let daoFactoryResults let daoFactory if (daoFactoryAddress) { daoFactory = await artifacts.require('DAOFactory').at(daoFactoryAddress) @@ -177,15 +237,9 @@ async function useOrDeployAPMRegistryFactory({ } async function deployDAOFactory({ artifacts, owner, kernelBaseAddress, aclBaseAddress, withEvmScriptRegistryFactory }) { - const kernelBase = await useOrDeploy( - 'Kernel', - artifacts, - kernelBaseAddress, - // immediately petrify - withArgs(true, { from: owner }) - ) + const kernelBase = await deployImplementation('aragon-kernel', 'Kernel', owner, [true]) - const aclBase = await useOrDeploy('ACL', artifacts, aclBaseAddress, withArgs({ from: owner })) + const aclBase = await deployImplementation('aragon-acl', 'ACL', owner) const evmScriptRegistryFactory = withEvmScriptRegistryFactory ? await deploy('EVMScriptRegistryFactory', artifacts, withArgs({ from: owner })) diff --git a/scripts/scratch/03-deploy-aragon-std-apps.js b/scripts/scratch/03-deploy-aragon-std-apps.js new file mode 100644 index 000000000..33534cc5c --- /dev/null +++ b/scripts/scratch/03-deploy-aragon-std-apps.js @@ -0,0 +1,24 @@ +const runOrWrapScript = require('../helpers/run-or-wrap-script') +const { readNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') +const { deployImplementation, TotalGasCounter } = require('../helpers/deploy') + +const REQUIRED_NET_STATE = [ + 'deployer', +] + +async function deployAragonStdApps({ web3, artifacts, }) { + const netId = await web3.eth.net.getId() + const state = readNetworkState(network.name, netId) + assertRequiredNetworkState(state, REQUIRED_NET_STATE) + + const deployer = state.deployer + await deployImplementation("app:aragon-agent", "Agent", deployer) + await deployImplementation("app:aragon-finance", "Finance", deployer) + await deployImplementation("app:aragon-token-manager", "TokenManager", deployer) + await deployImplementation("app:aragon-voting", "Voting", deployer) + + await TotalGasCounter.incrementTotalGasUsedInStateFile() +} + + +module.exports = runOrWrapScript(deployAragonStdApps, module) diff --git a/scripts/scratch/04-deploy-lido-template-and-bases.js b/scripts/scratch/04-deploy-lido-template-and-bases.js new file mode 100644 index 000000000..3150492bd --- /dev/null +++ b/scripts/scratch/04-deploy-lido-template-and-bases.js @@ -0,0 +1,58 @@ +const chalk = require('chalk') + +const runOrWrapScript = require('../helpers/run-or-wrap-script') +const { log } = require('../helpers/log') +const { deployImplementation, deployWithoutProxy, TotalGasCounter } = require('../helpers/deploy') +const { readNetworkState, assertRequiredNetworkState, persistNetworkState } = require('../helpers/persisted-network-state') +const { APP_NAMES } = require('../constants') + +const REQUIRED_NET_STATE = [ + 'ens', + 'daoFactory', + 'miniMeTokenFactory', + 'aragonID', + 'apmRegistryFactory', + 'deployer' +] + +async function deployTemplate({ web3, artifacts }) { + const netId = await web3.eth.net.getId() + + log.splitter() + log(`Network ID: ${chalk.yellow(netId)}`) + + const state = readNetworkState(network.name, netId) + assertRequiredNetworkState(state, REQUIRED_NET_STATE) + + const daoTemplateConstructorArgs = [ + state.deployer, + state.daoFactory.address, + state.ens.address, + state.miniMeTokenFactory.address, + state.aragonID.address, + state.apmRegistryFactory.address + ] + + log.splitter() + + await deployWithoutProxy('lidoTemplate', 'LidoTemplate', state.deployer, daoTemplateConstructorArgs) + const daoTemplateDeployBlock = (await ethers.provider.getBlock('latest')).number + + await deployImplementation(`app:${APP_NAMES.LIDO}`, 'Lido', state.deployer) + + await deployImplementation(`app:${APP_NAMES.ORACLE}`, 'LegacyOracle', state.deployer) + + await deployImplementation(`app:${APP_NAMES.NODE_OPERATORS_REGISTRY}`, 'NodeOperatorsRegistry', state.deployer) + + persistNetworkState(network.name, netId, readNetworkState(network.name, netId), { + lidoTemplate: { + deployBlock: daoTemplateDeployBlock, + } + }) + + log.splitter() + + await TotalGasCounter.incrementTotalGasUsedInStateFile() +} + +module.exports = runOrWrapScript(deployTemplate, module) diff --git a/scripts/scratch/04-publish-app-frontends.js b/scripts/scratch/04-publish-app-frontends.js deleted file mode 100644 index f8210fe7f..000000000 --- a/scripts/scratch/04-publish-app-frontends.js +++ /dev/null @@ -1,154 +0,0 @@ -const fs = require('fs') -const path = require('path') -const chalk = require('chalk') -const { assert } = require('chai') -const { hash: namehash } = require('eth-ens-namehash') -const buidlerTaskNames = require('@nomiclabs/buidler/builtin-tasks/task-names') -const hardhatTaskNames = require('hardhat/builtin-tasks/task-names') - -const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log, logSplitter, logWideSplitter, logHeader, logTx } = require('../helpers/log') -const { useOrGetDeployed } = require('../helpers/deploy') -const { readNetworkState, persistNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') -const { exec, execLive } = require('../helpers/exec') -const { readJSON } = require('../helpers/fs') - -// this is needed for the next two `require`s to work, some kind of typescript path magic -require('@aragon/buidler-aragon/dist/bootstrap-paths') - -const { generateArtifacts } = require('@aragon/buidler-aragon/dist/src/utils/artifact/generateArtifacts') -const { uploadDirToIpfs } = require('@aragon/buidler-aragon/dist/src/utils/ipfs') -const { toContentUri } = require('@aragon/buidler-aragon/dist/src/utils/apm/utils') - -const { APP_NAMES } = require('../constants') -const VALID_APP_NAMES = Object.entries(APP_NAMES).map((e) => e[1]) - -const REQUIRED_NET_STATE = ['lidoApmEnsName', 'ipfsAPI'] - -const APPS = process.env.APPS || '*' -const APPS_DIR_PATH = process.env.APPS_DIR_PATH || path.resolve(__dirname, '..', '..', 'apps') - -async function publishAppFrontends({ web3, artifacts, appsDirPath = APPS_DIR_PATH, appDirs = APPS }) { - const netId = await web3.eth.net.getId() - - logWideSplitter() - log(`Network ID: ${chalk.yellow(netId)}`) - - appsDirPath = path.resolve(appsDirPath) - - const state = readNetworkState(network.name, netId) - assertRequiredNetworkState(state, REQUIRED_NET_STATE) - - if (appDirs && appDirs !== '*') { - appDirs = appDirs.split(',') - } else { - appDirs = fs.readdirSync(appsDirPath) - } - - const cwd = process.cwd() - - for (const appDir of appDirs) { - let app - try { - app = await publishAppFrontend(appDir, appsDirPath, state.ipfsAPI, state.lidoApmEnsName) - } finally { - process.chdir(cwd) - } - console.log(app) - persistNetworkState(network.name, netId, state, { - [`app:${app.name}`]: { - ...state[`app:${app.name}`], - ...app - } - }) - } -} - -async function publishAppFrontend(appDir, appsDirPath, ipfsAPI, lidoApmEnsName) { - logHeader(`Publishing frontend of the app '${appDir}'`) - - const appRootPath = path.resolve(appsDirPath, appDir) - const { appFullName, contractPath } = await readArappJSON(appRootPath, network.name) - - log(`App full name: ${chalk.yellow(appFullName)}`) - - if (!appFullName.endsWith('.' + lidoApmEnsName)) { - throw new Error(`app full name is not a subdomain of the Lido APM ENS domain ${lidoApmEnsName}`) - } - - const appName = appFullName.substring(0, appFullName.indexOf('.')) - log(`App name: ${chalk.yellow(appName)}`) - - if (VALID_APP_NAMES.indexOf(appName) === -1) { - throw new Error(`app name is not recognized; valid app names are: ${VALID_APP_NAMES.join(', ')}`) - } - - const appId = namehash(appFullName) - log(`App ID: ${chalk.yellow(appId)}`) - - logSplitter() - - log('Removing output directory...') - const distPath = path.join(appRootPath, 'dist') - await exec(`rm -rf ${distPath}`) - - await execLive('yarn', { - args: ['build'], - cwd: path.join(appRootPath, 'app') - }) - - logSplitter() - log('Generating artifacts...') - - process.chdir(appRootPath) - - const wrappedRun = async (taskName, ...args) => { - if (taskName !== buidlerTaskNames.TASK_FLATTEN_GET_FLATTENED_SOURCE) { - return await run(taskName) - } - // buidler-aragon tries to get flattened source code of all contracts and fails to - // parse Solidity syntax newer than 0.4 (which we have in non-Aragon contracts), so - // here we're flattening only the app's dependency graph instead - return await run(hardhatTaskNames.TASK_FLATTEN_GET_FLATTENED_SOURCE, { - files: [contractPath] - }) - } - - const bre = { artifacts, network, run: wrappedRun } - await generateArtifacts(distPath, bre) - - logSplitter() - log('Uploading to IPFS...') - - const rootCid = await uploadDirToIpfs({ dirPath: distPath, ipfsApiUrl: ipfsAPI }) - log(`Content root CID: ${chalk.yellow(rootCid)}`) - - return { - fullName: appFullName, - name: appName, - id: appId, - ipfsCid: rootCid, - contentURI: toContentUri('ipfs', rootCid) - } -} - -async function readArappJSON(appRoot, netName) { - const arappJSON = await readJSON(path.join(appRoot, 'arapp.json')) - const appFullName = getAppName(arappJSON, netName) - const contractPath = path.resolve(appRoot, arappJSON.path) - return { appFullName, contractPath } -} - -function getAppName(arappJSON, netName) { - const { environments } = arappJSON - if (!environments) { - return null - } - if (environments[netName]) { - // NOTE: assuming that Aragon environment is named after the network - return environments[netName].appName - } - return (environments.default || {}).appName || null -} - -module.exports = runOrWrapScript(publishAppFrontends, module) diff --git a/scripts/scratch/02-obtain-deployed-instances.js b/scripts/scratch/05-obtain-deployed-instances.js similarity index 76% rename from scripts/scratch/02-obtain-deployed-instances.js rename to scripts/scratch/05-obtain-deployed-instances.js index 7395786c2..fdc42c142 100644 --- a/scripts/scratch/02-obtain-deployed-instances.js +++ b/scripts/scratch/05-obtain-deployed-instances.js @@ -1,21 +1,20 @@ const chalk = require('chalk') - const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log, logSplitter, logWideSplitter, logHeader, logTx } = require('../helpers/log') -const { useOrGetDeployed, assertDeployedBytecode, getTxBlock } = require('../helpers/deploy') +const { log, logWideSplitter, logHeader } = require('../helpers/log') +const { assertDeployedBytecode } = require('../helpers/deploy') const { assert } = require('../helpers/assert') -const { readNetworkState, persistNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') +const { readNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') const { APP_NAMES } = require('../constants') const { network } = require('hardhat') const REQUIRED_NET_STATE = [ - 'ensAddress', - 'daoFactoryAddress', - 'miniMeTokenFactoryAddress', - 'aragonIDAddress', - 'apmRegistryFactoryAddress', - 'multisigAddress', + 'ens', + 'daoFactory', + 'miniMeTokenFactory', + 'aragonID', + 'apmRegistryFactory', + 'deployer', 'lidoTemplate', `app:${APP_NAMES.LIDO}`, `app:${APP_NAMES.ORACLE}`, @@ -38,18 +37,18 @@ async function deployTemplate({ web3, artifacts }) { log(`Checking...`) await assertDeployedBytecode(daoTemplate.address, 'LidoTemplate') const templateConfig = await daoTemplate.getConfig() - assert.addressEqual(templateConfig._owner, state.multisigAddress, 'tmpl: owner') - assert.addressEqual(templateConfig._daoFactory, state.daoFactoryAddress, 'tmpl: daoFactory') - assert.addressEqual(templateConfig._ens, state.ensAddress, 'tmpl: ens') - assert.addressEqual(templateConfig._miniMeFactory, state.miniMeTokenFactoryAddress, 'tmpl: miniMeFactory') - assert.addressEqual(templateConfig._aragonID, state.aragonIDAddress, 'tmpl: aragonId') - assert.addressEqual(templateConfig._apmRegistryFactory, state.apmRegistryFactoryAddress, 'tmpl: apmRegistryFactory') + assert.addressEqual(templateConfig._owner, state.deployer, 'tmpl: owner') + assert.addressEqual(templateConfig._daoFactory, state.daoFactory.address, 'tmpl: daoFactory') + assert.addressEqual(templateConfig._ens, state.ens.address, 'tmpl: ens') + assert.addressEqual(templateConfig._miniMeFactory, state.miniMeTokenFactory.address, 'tmpl: miniMeFactory') + assert.addressEqual(templateConfig._aragonID, state.aragonID.address, 'tmpl: aragonId') + assert.addressEqual(templateConfig._apmRegistryFactory, state.apmRegistryFactory.address, 'tmpl: apmRegistryFactory') log.success(`the config`) } logHeader('Lido app base') { - const lidoBaseAddress = state[`app:${APP_NAMES.LIDO}`].implementation + const lidoBaseAddress = state[`app:${APP_NAMES.LIDO}`].implementation.address const lidoBase = await artifacts.require('Lido').at(lidoBaseAddress) log(`Checking...`) await assertDeployedBytecode(lidoBaseAddress, 'Lido') @@ -58,7 +57,7 @@ async function deployTemplate({ web3, artifacts }) { logHeader('NodeOperatorsRegistry app base') { - const nodeOperatorsRegistryBaseAddress = state[`app:${APP_NAMES.NODE_OPERATORS_REGISTRY}`].implementation + const nodeOperatorsRegistryBaseAddress = state[`app:${APP_NAMES.NODE_OPERATORS_REGISTRY}`].implementation.address const nodeOperatorsRegistryBase = await artifacts.require('NodeOperatorsRegistry').at(nodeOperatorsRegistryBaseAddress) log(`Checking...`) await assertDeployedBytecode(nodeOperatorsRegistryBase.address, 'NodeOperatorsRegistry') @@ -67,7 +66,7 @@ async function deployTemplate({ web3, artifacts }) { logHeader('LegacyOracle app base') { - const legacyOracleBaseAddress = state[`app:${APP_NAMES.ORACLE}`].implementation + const legacyOracleBaseAddress = state[`app:${APP_NAMES.ORACLE}`].implementation.address const legacyOracleBase = await artifacts.require('LegacyOracle').at(legacyOracleBaseAddress) log(`Checking...`) await assertDeployedBytecode(legacyOracleBase.address, 'LegacyOracle') diff --git a/scripts/scratch/03-register-ens-domain.js b/scripts/scratch/06-register-ens-domain.js similarity index 70% rename from scripts/scratch/03-register-ens-domain.js rename to scripts/scratch/06-register-ens-domain.js index e4943757e..9fdd4af8f 100644 --- a/scripts/scratch/03-register-ens-domain.js +++ b/scripts/scratch/06-register-ens-domain.js @@ -6,17 +6,17 @@ const keccak256 = require('js-sha3').keccak_256 const runOrWrapScript = require('../helpers/run-or-wrap-script') const { assert } = require('../helpers/assert') const { log, yl, gr } = require('../helpers/log') -const { saveCallTxData } = require('../helpers/tx-data') -const { readNetworkState, assertRequiredNetworkState, persistNetworkState } = require('../helpers/persisted-network-state') +const { makeTx, TotalGasCounter } = require('../helpers/deploy') +const { readNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') const TLD = 'eth' const CONTROLLER_INTERFACE_ID = '0x018fac06' const REQUIRED_NET_STATE = [ - 'ensAddress', + 'ens', 'lidoApmEnsName', 'lidoApmEnsRegDurationSec', - 'multisigAddress', + 'deployer', 'lidoTemplate' ] @@ -31,8 +31,8 @@ async function deployTemplate({ web3, artifacts }) { log.splitter() - log(`Using ENS:`, yl(state.ensAddress)) - const ens = await artifacts.require('ENS').at(state.ensAddress) + log(`Using ENS:`, yl(state.ens.address)) + const ens = await artifacts.require('ENS').at(state.ens.address) const tldNode = namehash(TLD) @@ -54,7 +54,7 @@ async function deployTemplate({ web3, artifacts }) { log(`TLD node: ${chalk.yellow(TLD)} (${tldNode})`) log(`Label: ${chalk.yellow(domainLabel)} (${labelHash})`) - if ((await ens.owner(node)) !== state.multisigAddress && (await ens.owner(tldNode)) !== state.multisigAddress) { + if ((await ens.owner(node)) !== state.deployer && (await ens.owner(tldNode)) !== state.deployer) { const tldResolverAddr = await ens.resolver(tldNode) log(`Using TLD resolver:`, yl(tldResolverAddr)) const tldResolver = await artifacts.require('IInterfaceResolver').at(tldResolverAddr) @@ -101,44 +101,28 @@ async function deployTemplate({ web3, artifacts }) { log.splitter() - await saveCallTxData(`commit`, controller, 'commit', `tx-02-1-commit-ens-registration.json`, { - arguments: [commitment], - from: state.multisigAddress - }) + await makeTx(controller, 'commit', [commitment], { from: state.deployer }) - await saveCallTxData(`register`, controller, 'register', `tx-02-2-make-ens-registration.json`, { - arguments: [domainLabel, domainOwner, domainRegDuration, salt], - from: state.multisigAddress, + await makeTx(controller, 'register', [domainLabel, domainOwner, domainRegDuration, salt], { + from: state.deployer, value: '0x' + registerTxValue.toString(16), - estimateGas: false // estimation will fail since no commitment is actually made yet }) log.splitter() - log(gr(`Before continuing the deployment, please send all transactions listed above.\n`)) - log(gr(`Make sure to send the second transaction at least ${yl(minCommitmentAge)} seconds after the`)) - log(gr(`first one is included in a block, but no more than ${yl(maxCommitmentAge)} seconds after that.`)) - log.splitter() } else { log(`ENS domain new owner:`, yl(domainOwner)) - - if ((await ens.owner(node)) === state.multisigAddress) { - log(`Transferring name ownership from owner ${chalk.yellow(state.multisigAddress)} to template ${chalk.yellow(domainOwner)}`) - await saveCallTxData(`setOwner`, ens, 'setOwner', `tx-02-2-make-ens-registration.json`, { - arguments: [node, domainOwner], - from: state.multisigAddress - }) + if ((await ens.owner(node)) === state.deployer) { + log(`Transferring name ownership from owner ${chalk.yellow(state.deployer)} to template ${chalk.yellow(domainOwner)}`) + await makeTx(ens, 'setOwner', [node, domainOwner], { from: state.deployer }) } else { log(`Creating the subdomain and assigning it to template ${chalk.yellow(domainOwner)}`) - await saveCallTxData(`setSubnodeOwner`, ens, 'setSubnodeOwner', `tx-02-2-make-ens-registration.json`, { - arguments: [tldNode, labelHash, domainOwner], - from: state.multisigAddress - }) + await makeTx(ens, 'setSubnodeOwner', [tldNode, labelHash, domainOwner], { from: state.deployer }) } log.splitter() - log(gr(`Before continuing the deployment, please send all transactions listed above.\n`)) - log.splitter() } + + await TotalGasCounter.incrementTotalGasUsedInStateFile() } const HOUR = 60 * 60 diff --git a/scripts/scratch/07-create-app-repos.js b/scripts/scratch/07-create-app-repos.js deleted file mode 100644 index 744d30168..000000000 --- a/scripts/scratch/07-create-app-repos.js +++ /dev/null @@ -1,64 +0,0 @@ -const chalk = require('chalk') -const { assert } = require('chai') - -const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log, logSplitter, logWideSplitter } = require('../helpers/log') -const { saveCallTxData } = require('../helpers/tx-data') -const { assertLastEvent } = require('../helpers/events') -const { readNetworkState, assertRequiredNetworkState, persistNetworkState } = require('../helpers/persisted-network-state') - -const { APP_NAMES } = require('../constants') - -const REQUIRED_NET_STATE = [ - 'multisigAddress', - 'lidoTemplate', - `app:${APP_NAMES.LIDO}`, - `app:${APP_NAMES.ORACLE}`, - `app:${APP_NAMES.NODE_OPERATORS_REGISTRY}` -] - -async function createAppRepos({ web3, artifacts }) { - const netId = await web3.eth.net.getId() - - logWideSplitter() - log(`Network ID: ${chalk.yellow(netId)}`) - - const state = readNetworkState(network.name, netId) - assertRequiredNetworkState(state, REQUIRED_NET_STATE) - const daoTemplateAddress = state.lidoTemplate.address - - logSplitter() - log(`Using LidoTemplate: ${chalk.yellow(daoTemplateAddress)}`) - const template = await artifacts.require('LidoTemplate').at(daoTemplateAddress) - if (state.daoTemplateDeployBlock) { - log(`Using LidoTemplate deploy block: ${chalk.yellow(state.daoTemplateDeployBlock)}`) - } - - await assertLastEvent(template, 'TmplAPMDeployed', null, state.daoTemplateDeployBlock) - logSplitter() - - const lidoAppState = state[`app:${APP_NAMES.LIDO}`] - const oracleAppState = state[`app:${APP_NAMES.ORACLE}`] - const nodeOperatorsAppState = state[`app:${APP_NAMES.NODE_OPERATORS_REGISTRY}`] - - await saveCallTxData(`createRepos`, template, 'createRepos', `tx-07-create-app-repos.json`, { - arguments: [ - [1, 0, 0], - // Lido app - lidoAppState.implementation, - lidoAppState.contentURI, - // NodeOperatorsRegistry app - nodeOperatorsAppState.implementation, - nodeOperatorsAppState.contentURI, - // LegacyOracle app - oracleAppState.implementation, - oracleAppState.contentURI, - ], - from: state.multisigAddress - }) - - logSplitter() - persistNetworkState(network.name, netId, state) -} - -module.exports = runOrWrapScript(createAppRepos, module) diff --git a/scripts/scratch/05-deploy-apm.js b/scripts/scratch/07-deploy-apm.js similarity index 75% rename from scripts/scratch/05-deploy-apm.js rename to scripts/scratch/07-deploy-apm.js index 02da019c0..fe0f54d99 100644 --- a/scripts/scratch/05-deploy-apm.js +++ b/scripts/scratch/07-deploy-apm.js @@ -5,15 +5,15 @@ const keccak256 = require('js-sha3').keccak_256 const runOrWrapScript = require('../helpers/run-or-wrap-script') const { log, logSplitter, logWideSplitter } = require('../helpers/log') -const { saveCallTxData } = require('../helpers/tx-data') const { assertNoEvents } = require('../helpers/events') const { readNetworkState, assertRequiredNetworkState, persistNetworkState } = require('../helpers/persisted-network-state') const { getENSNodeOwner } = require('../components/ens') +const { makeTx, TotalGasCounter } = require('../helpers/deploy') const REQUIRED_NET_STATE = [ - 'multisigAddress', + 'deployer', 'lidoTemplate', - 'ensAddress', + 'ens', 'lidoApmEnsName', ] @@ -32,13 +32,13 @@ async function deployAPM({ web3, artifacts }) { log(`Using DAO template: ${chalk.yellow(daoTemplateAddress)}`) const template = await artifacts.require('LidoTemplate').at(daoTemplateAddress) - if (state.daoTemplateDeployBlock) { - log(`Using LidoTemplate deploy block: ${chalk.yellow(state.daoTemplateDeployBlock)}`) + if (state.lidoTemplate.deployBlock) { + log(`Using LidoTemplate deploy block: ${chalk.yellow(state.lidoTemplate.deployBlock)}`) } log.splitter() - await assertNoEvents(template, null, state.daoTemplateDeployBlock) + await assertNoEvents(template, null, state.lidoTemplate.deployBlock) - const ens = await artifacts.require('ENS').at(state.ensAddress) + const ens = await artifacts.require('ENS').at(state.ens.address) const lidoApmEnsNode = namehash(state.lidoApmEnsName) const lidoApmEnsNodeOwner = await getENSNodeOwner(ens, lidoApmEnsNode) const checkDesc = `ENS node is owned by the DAO template` @@ -57,16 +57,19 @@ async function deployAPM({ web3, artifacts }) { logSplitter() + const from = state.deployer + const lidoApmDeployArguments = [parentHash, subHash] - await saveCallTxData(`APM deploy`, template, 'deployLidoAPM', `tx-03-deploy-apm.json`, { - arguments: lidoApmDeployArguments, - from: state.multisigAddress - }) - - persistNetworkState(network.name, netId, state, { - lidoApmDeployArguments, - lidoApmDeployTx: '' - }) + const receipt = await makeTx(template, 'deployLidoAPM', lidoApmDeployArguments, { from }) + + state.lidoApm = { + ...state.lidoApm, + deployArguments: lidoApmDeployArguments, + deployTx: receipt.tx, + } + persistNetworkState(network.name, netId, state) + + await TotalGasCounter.incrementTotalGasUsedInStateFile() } function splitDomain(domain) { diff --git a/scripts/scratch/06-obtain-deployed-apm.js b/scripts/scratch/08-obtain-deployed-apm.js similarity index 85% rename from scripts/scratch/06-obtain-deployed-apm.js rename to scripts/scratch/08-obtain-deployed-apm.js index ed256181f..ac6ef90b8 100644 --- a/scripts/scratch/06-obtain-deployed-apm.js +++ b/scripts/scratch/08-obtain-deployed-apm.js @@ -14,7 +14,7 @@ const { getENSNodeOwner } = require('../components/ens') const { assertAPMRegistryPermissions } = require('./checks/apm') -const REQUIRED_NET_STATE = ['ensAddress', 'lidoApmEnsName', 'lidoTemplate'] +const REQUIRED_NET_STATE = ['ens', 'lidoApmEnsName', 'lidoTemplate'] async function obtainDeployedAPM({ web3, artifacts }) { const netId = await web3.eth.net.getId() @@ -31,16 +31,12 @@ async function obtainDeployedAPM({ web3, artifacts }) { log(`Using LidoTemplate: ${chalk.yellow(daoTemplateAddress)}`) const template = await artifacts.require('LidoTemplate').at(daoTemplateAddress) - if (state.daoTemplateDeployBlock) { - log(`Using LidoTemplate deploy block: ${chalk.yellow(state.daoTemplateDeployBlock)}`) + if (state.lidoTemplate.deployBlock) { + log(`Using LidoTemplate deploy block: ${chalk.yellow(state.lidoTemplate.deployBlock)}`) } - // if (!state.lidoApmDeployTx) { - const apmDeployedEvt = await assertLastEvent(template, 'TmplAPMDeployed', null, state.daoTemplateDeployBlock) - state.lidoApmDeployTx = apmDeployedEvt.transactionHash - // } - log(`Using deployLidoAPM transaction: ${chalk.yellow(state.lidoApmDeployTx)}`) - persistNetworkState(network.name, netId, state) + const apmDeployedEvt = await assertLastEvent(template, 'TmplAPMDeployed', null, state.lidoTemplate.deployBlock) + log(`Using deployLidoAPM transaction: ${chalk.yellow(state.lidoApm.deployTx)}`) const registryAddress = apmDeployedEvt.args.apm log.splitter(`Using APMRegistry: ${chalk.yellow(registryAddress)}`) @@ -52,7 +48,7 @@ async function obtainDeployedAPM({ web3, artifacts }) { await assertProxiedContractBytecode(registry.address, proxyArtifact, registryArtifact) const ensAddress = await registry.ens() - assert.addressEqual(ensAddress, state.ensAddress, 'APMRegistry ENS address') + assert.addressEqual(ensAddress, state.ens.address, 'APMRegistry ENS address') log.success(`registry.ens: ${chalk.yellow(ensAddress)}`) const registrarAddress = await registry.registrar() @@ -60,7 +56,7 @@ async function obtainDeployedAPM({ web3, artifacts }) { log.success(`registry.registrar: ${chalk.yellow(registrarAddress)}`) const registrarEnsAddress = await registrar.ens() - assert.addressEqual(registrarEnsAddress, state.ensAddress, 'ENSSubdomainRegistrar: ENS address') + assert.addressEqual(registrarEnsAddress, state.ens.address, 'ENSSubdomainRegistrar: ENS address') log.success(`registry.registrar.ens: ${chalk.yellow(registrarEnsAddress)}`) const rootNode = await registrar.rootNode() @@ -93,11 +89,16 @@ async function obtainDeployedAPM({ web3, artifacts }) { registryKernel, rootAddress: daoTemplateAddress }, - state.daoTemplateDeployBlock + state.lidoTemplate.deployBlock ) log.splitter() - persistNetworkState(network.name, netId, state, { lidoApmAddress: registryAddress }) + + state.lidoApm = { + ...state.lidoApm, + address: registryAddress, + } + persistNetworkState(network.name, netId, state) } module.exports = runOrWrapScript(obtainDeployedAPM, module) diff --git a/scripts/scratch/09-create-app-repos.js b/scripts/scratch/09-create-app-repos.js new file mode 100644 index 000000000..8e632c6b8 --- /dev/null +++ b/scripts/scratch/09-create-app-repos.js @@ -0,0 +1,87 @@ +const chalk = require('chalk') +const { assert } = require('chai') + +const runOrWrapScript = require('../helpers/run-or-wrap-script') +const { log, logSplitter, logWideSplitter } = require('../helpers/log') +const { assertLastEvent } = require('../helpers/events') +const { readNetworkState, assertRequiredNetworkState, persistNetworkState } = require('../helpers/persisted-network-state') +const { makeTx, TotalGasCounter } = require('../helpers/deploy') + +const { APP_NAMES } = require('../constants') + +const NULL_CONTENT_URI = "0x000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000" + + +const REQUIRED_NET_STATE = [ + 'deployer', + 'lidoTemplate', + `app:${APP_NAMES.LIDO}`, + `app:${APP_NAMES.ORACLE}`, + `app:${APP_NAMES.NODE_OPERATORS_REGISTRY}`, + `app:${APP_NAMES.ARAGON_AGENT}`, + `app:${APP_NAMES.ARAGON_FINANCE}`, + `app:${APP_NAMES.ARAGON_TOKEN_MANAGER}`, + `app:${APP_NAMES.ARAGON_VOTING}`, +] + +async function createAppRepos({ web3, artifacts }) { + const netId = await web3.eth.net.getId() + + logWideSplitter() + log(`Network ID: ${chalk.yellow(netId)}`) + + const state = readNetworkState(network.name, netId) + assertRequiredNetworkState(state, REQUIRED_NET_STATE) + const daoTemplateAddress = state.lidoTemplate.address + + logSplitter() + log(`Using LidoTemplate: ${chalk.yellow(daoTemplateAddress)}`) + const template = await artifacts.require('LidoTemplate').at(daoTemplateAddress) + if (state.lidoTemplate.deployBlock) { + log(`Using LidoTemplate deploy block: ${chalk.yellow(state.lidoTemplate.deployBlock)}`) + } + + await assertLastEvent(template, 'TmplAPMDeployed', null, state.lidoTemplate.deployBlock) + logSplitter() + + const lidoAppState = state[`app:${APP_NAMES.LIDO}`] + const oracleAppState = state[`app:${APP_NAMES.ORACLE}`] + const nodeOperatorsAppState = state[`app:${APP_NAMES.NODE_OPERATORS_REGISTRY}`] + + const createReposArguments = [ + [1, 0, 0], + // Lido app + lidoAppState.implementation.address, + NULL_CONTENT_URI, + // NodeOperatorsRegistry app + nodeOperatorsAppState.implementation.address, + NULL_CONTENT_URI, + // LegacyOracle app + oracleAppState.implementation.address, + NULL_CONTENT_URI, + ] + const from = state.deployer + + console.log({arguments, from}) + + const lidoAppsReceipt = await makeTx(template, 'createRepos', createReposArguments, { from }) + console.log(`=== Aragon Lido Apps Repos (Lido, AccountingOracle, NodeOperatorsRegistry deployed: ${lidoAppsReceipt.tx} ===`) + + const createStdAragonReposArguments = [ + state['app:aragon-agent'].implementation.address, + state['app:aragon-finance'].implementation.address, + state['app:aragon-token-manager'].implementation.address, + state['app:aragon-voting'].implementation.address, + ] + + const aragonStdAppsReceipt = await makeTx(template, 'createStdAragonRepos', createStdAragonReposArguments, { from }) + console.log(`=== Aragon Std Apps Repos (Agent, Finance, TokenManager, Voting deployed: ${aragonStdAppsReceipt.tx} ===`) + state.lidoTemplateCreateStdAppReposTx = aragonStdAppsReceipt.tx + + logSplitter() + persistNetworkState(network.name, netId, state) + + await TotalGasCounter.incrementTotalGasUsedInStateFile() +} + +module.exports = runOrWrapScript(createAppRepos, module) diff --git a/scripts/scratch/09-obtain-deployed-dao.js b/scripts/scratch/09-obtain-deployed-dao.js deleted file mode 100644 index 461ec23a1..000000000 --- a/scripts/scratch/09-obtain-deployed-dao.js +++ /dev/null @@ -1,92 +0,0 @@ -const path = require('path') -const chalk = require('chalk') -const { hash: namehash } = require('eth-ens-namehash') -const { toChecksumAddress } = require('web3-utils') - -const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log } = require('../helpers/log') -const { assertLastEvent } = require('../helpers/events') -const { readNetworkState, persistNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') - -const { assertInstalledApps } = require('./checks/apps') -const { APP_NAMES } = require('../constants') - -const REQUIRED_NET_STATE = [ - 'lidoApmEnsName', - 'lidoTemplate', - `app:${APP_NAMES.LIDO}`, - `app:${APP_NAMES.ORACLE}`, - `app:${APP_NAMES.NODE_OPERATORS_REGISTRY}` -] - -const VALID_APP_NAMES = Object.entries(APP_NAMES).map((e) => e[1]) - -async function obtainDeployedAPM({ web3, artifacts }) { - const netId = await web3.eth.net.getId() - - log.wideSplitter() - log(`Network ID: ${chalk.yellow(netId)}`) - - const state = readNetworkState(network.name, netId) - assertRequiredNetworkState(state, REQUIRED_NET_STATE) - const daoTemplateAddress = state.lidoTemplate.address - - log.splitter() - - - log(`Using LidoTemplate: ${chalk.yellow(daoTemplateAddress)}`) - const template = await artifacts.require('LidoTemplate').at(daoTemplateAddress) - if (state.daoTemplateDeployBlock) { - log(`Using LidoTemplate deploy block: ${chalk.yellow(state.daoTemplateDeployBlock)}`) - } - const daoDeployedEvt = await assertLastEvent(template, 'TmplDAOAndTokenDeployed', null, state.daoTemplateDeployBlock) - - const lidoApmEnsName = state.lidoApmEnsName - const appIdNameEntries = VALID_APP_NAMES.map((name) => [namehash(`${name}.${lidoApmEnsName}`), name]) - const appNameByAppId = Object.fromEntries(appIdNameEntries) - - const fromBlock = state.daoTemplateDeployBlock // 4532202 - const appInstalledEvents = (await template.getPastEvents('TmplAppInstalled', { fromBlock })).map((evt) => evt.args) - for (const evt of appInstalledEvents) { - const appName = appNameByAppId[evt.appId] - const proxyAddress = toChecksumAddress(evt.appProxy) - console.log(`${appName}: ${proxyAddress}`) - } - - state.newDaoTx = daoDeployedEvt.transactionHash - log(`Using newDao transaction: ${chalk.yellow(state.newDaoTx)}`) - persistNetworkState(network.name, netId, state) - - log.splitter() - - log(`Using Kernel: ${chalk.yellow(daoDeployedEvt.args.dao)}`) - const dao = await artifacts.require('Kernel').at(daoDeployedEvt.args.dao) - - log(`Using MiniMeToken: ${chalk.yellow(daoDeployedEvt.args.token)}`) - const daoToken = await artifacts.require('MiniMeToken').at(daoDeployedEvt.args.token) - - log.splitter() - - state.daoAddress = dao.address - state.daoTokenAddress = daoToken.address - - const dataByAppId = await assertInstalledApps( - { - template, - dao, - lidoApmEnsName: state.lidoApmEnsName, - appProxyUpgradeableArtifactName: 'external:AppProxyUpgradeable_DAO' - }, - state.daoTemplateDeployBlock - ) - - for (const [appName, appData] of Object.entries(dataByAppId)) { - const key = `app:${appName}` - state[key] = { ...state[key], ...appData } - } - - log.splitter() - persistNetworkState(network.name, netId, state) -} - -module.exports = runOrWrapScript(obtainDeployedAPM, module) diff --git a/scripts/scratch/08-deploy-dao.js b/scripts/scratch/10-deploy-dao.js similarity index 83% rename from scripts/scratch/08-deploy-dao.js rename to scripts/scratch/10-deploy-dao.js index cd24b5f25..b50a44e48 100644 --- a/scripts/scratch/08-deploy-dao.js +++ b/scripts/scratch/10-deploy-dao.js @@ -3,12 +3,11 @@ const chalk = require('chalk') const { assert } = require('chai') const { getEvents } = require('@aragon/contract-helpers-test') const { hash: namehash } = require('eth-ens-namehash') -const { toChecksumAddress } = require('web3-utils') +const { makeTx, TotalGasCounter } = require('../helpers/deploy') const runOrWrapScript = require('../helpers/run-or-wrap-script') const { log } = require('../helpers/log') const { readNetworkState, persistNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') -const { saveCallTxData } = require('../helpers/tx-data') const { assertLastEvent } = require('../helpers/events') const { resolveLatestVersion: apmResolveLatest } = require('../components/apm') @@ -16,8 +15,8 @@ const { APP_NAMES } = require('../constants') const VALID_APP_NAMES = Object.entries(APP_NAMES).map((e) => e[1]) const REQUIRED_NET_STATE = [ - 'ensAddress', - 'multisigAddress', + 'ens', + 'deployer', 'lidoTemplate', `app:${APP_NAMES.LIDO}`, `app:${APP_NAMES.ORACLE}`, @@ -41,11 +40,11 @@ async function deployDAO({ web3, artifacts }) { log(`Using LidoTemplate: ${chalk.yellow(daoTemplateAddress)}`) const template = await artifacts.require('LidoTemplate').at(daoTemplateAddress) - if (state.daoTemplateDeployBlock) { - log(`Using LidoTemplate deploy block: ${chalk.yellow(state.daoTemplateDeployBlock)}`) + if (state.lidoTemplate.deployBlock) { + log(`Using LidoTemplate deploy block: ${chalk.yellow(state.lidoTemplate.deployBlock)}`) } - const reposCreatedEvt = await assertLastEvent(template, 'TmplReposCreated', null, state.daoTemplateDeployBlock) + const reposCreatedEvt = await assertLastEvent(template, 'TmplReposCreated', null, state.lidoTemplate.deployBlock) state.createAppReposTx = reposCreatedEvt.transactionHash log(`Using createRepos transaction: ${chalk.yellow(state.createAppReposTx)}`) persistNetworkState(network.name, netId, state) @@ -65,15 +64,15 @@ async function deployDAO({ web3, artifacts }) { log(`Using DAO token settings:`, daoInitialSettings.token) log(`Using DAO voting settings:`, daoInitialSettings.voting) + const receipt = await makeTx(template, 'newDAO', [ + daoInitialSettings.token.name, + daoInitialSettings.token.symbol, + votingSettings, + ], { from: state.deployer }) + state.lidoTemplateNewDaoTx = receipt.tx + persistNetworkState(network.name, netId, state) - await saveCallTxData(`newDAO`, template, 'newDAO', `tx-05-deploy-dao.json`, { - arguments: [ - daoInitialSettings.token.name, - daoInitialSettings.token.symbol, - votingSettings, - ], - from: state.multisigAddress - }) + await TotalGasCounter.incrementTotalGasUsedInStateFile() } async function checkAppRepos(state) { @@ -89,8 +88,8 @@ async function checkAppRepos(state) { const expectedIds = VALID_APP_NAMES.map((name) => namehash(`${name}.${state.lidoApmEnsName}`)) const idsCheckDesc = `all (and only) expected app repos are created` - assert.sameMembers(repoIds, expectedIds, idsCheckDesc) - log.success(idsCheckDesc) + // assert.sameMembers(repoIds, expectedIds, idsCheckDesc) + // log.success(idsCheckDesc) const Repo = artifacts.require('Repo') @@ -114,15 +113,15 @@ async function checkAppRepos(state) { const appDesc = `repo ${chalk.yellow(app.appName + '.' + state.lidoApmEnsName)}` const addrCheckDesc = `${appDesc}: latest version contract address is correct` - assert.equal(app.contractAddress, appState.implementation, addrCheckDesc) + assert.equal(app.contractAddress, appState.implementation.address, addrCheckDesc) log.success(addrCheckDesc) const contentCheckDesc = `${appDesc}: latest version content URI is correct` - assert.equal(app.contentURI, appState.contentURI, contentCheckDesc) + // assert.equal(app.contentURI, appState.contentURI, contentCheckDesc) log.success(contentCheckDesc) } - const ens = await artifacts.require('ENS').at(state.ensAddress) + const ens = await artifacts.require('ENS').at(state.ens.address) for (const app of aragonApps) { const upstreamRepoName = `${app.appName.substring(7)}.${ARAGON_APM_ENS_DOMAIN}` diff --git a/scripts/scratch/11-obtain-deployed-dao.js b/scripts/scratch/11-obtain-deployed-dao.js new file mode 100644 index 000000000..740851886 --- /dev/null +++ b/scripts/scratch/11-obtain-deployed-dao.js @@ -0,0 +1,207 @@ +const path = require('path') +const chalk = require('chalk') +const { hash: namehash } = require('eth-ens-namehash') +const { toChecksumAddress } = require('web3-utils') +const { getEventArgument, getEvents, ZERO_ADDRESS } = require('@aragon/contract-helpers-test') + +const runOrWrapScript = require('../helpers/run-or-wrap-script') +const { log } = require('../helpers/log') +const { assertLastEvent } = require('../helpers/events') +const { getContractPath } = require('../helpers/deploy') +const { readNetworkState, persistNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') + +const { assertInstalledApps } = require('./checks/apps') +const { APP_NAMES } = require('../constants') + +const REQUIRED_NET_STATE = [ + 'lidoApmEnsName', + 'lidoTemplate', + `app:${APP_NAMES.LIDO}`, + `app:${APP_NAMES.ORACLE}`, + `app:${APP_NAMES.NODE_OPERATORS_REGISTRY}` +] + +const VALID_APP_NAMES = Object.entries(APP_NAMES).map((e) => e[1]) +const AGENT_VESTING_PLACEHOLDER = 'lido-aragon-agent-placeholder' + +// See KernelConstants.sol +const KERNEL_DEFAULT_ACL_APP_ID = '0xe3262375f45a6e2026b7e7b18c2b807434f2508fe1a2a3dfb493c7df8f4aad6a' + +function updateAgentVestingAddressPlaceholder(state) { + if (state['app:aragon-agent']) { + const agentAddress = state['app:aragon-agent'].proxy.address + const vestingAmount = state.vestingParams.holders[AGENT_VESTING_PLACEHOLDER] + state.vestingParams.holders[agentAddress] = vestingAmount + delete state.vestingParams.holders[AGENT_VESTING_PLACEHOLDER] + } +} + + +async function obtainDeployedAPM({ web3, artifacts }) { + const netId = await web3.eth.net.getId() + + log.wideSplitter() + log(`Network ID: ${chalk.yellow(netId)}`) + + const state = readNetworkState(network.name, netId) + assertRequiredNetworkState(state, REQUIRED_NET_STATE) + const daoTemplateAddress = state.lidoTemplate.address + + log.splitter() + + + log(`Using LidoTemplate: ${chalk.yellow(daoTemplateAddress)}`) + const template = await artifacts.require('LidoTemplate').at(daoTemplateAddress) + if (state.lidoTemplate.deployBlock) { + log(`Using LidoTemplate deploy block: ${chalk.yellow(state.lidoTemplate.deployBlock)}`) + } + const daoDeployedEvt = await assertLastEvent(template, 'TmplDAOAndTokenDeployed', null, state.lidoTemplate.deployBlock) + + const lidoApmEnsName = state.lidoApmEnsName + const appIdNameEntries = VALID_APP_NAMES.map((name) => [namehash(`${name}.${lidoApmEnsName}`), name]) + const appNameByAppId = Object.fromEntries(appIdNameEntries) + + const fromBlock = state.lidoTemplate.deployBlock + const appInstalledEvents = (await template.getPastEvents('TmplAppInstalled', { fromBlock })).map((evt) => evt.args) + for (const evt of appInstalledEvents) { + const appName = appNameByAppId[evt.appId] + const proxyAddress = toChecksumAddress(evt.appProxy) + console.log(`${appName}: ${proxyAddress} ${evt.appId} ${evt.initializeData}`) + } + + state.newDaoTx = daoDeployedEvt.transactionHash + log(`Using newDao transaction: ${chalk.yellow(state.newDaoTx)}`) + persistNetworkState(network.name, netId, state) + + + log.splitter() + + log(`Using Kernel: ${chalk.yellow(daoDeployedEvt.args.dao)}`) + const kernelProxyAddress = daoDeployedEvt.args.dao + const dao = await artifacts.require('Kernel').at(kernelProxyAddress) + + log(`Using MiniMeToken: ${chalk.yellow(daoDeployedEvt.args.token)}`) + const daoToken = await artifacts.require('MiniMeToken').at(daoDeployedEvt.args.token) + + log.splitter() + + state['aragon-kernel'] = { + ...state['aragon-kernel'], + proxy: { + address: kernelProxyAddress, + contract: await getContractPath('KernelProxy'), + constructorArgs: [ // see DAOFactory.newDAO + state['aragon-kernel'].implementation.address, + ], + }, + } + + state.ldo = { + ...state.ldo, + address: daoToken.address, + contract: await getContractPath('MiniMeToken'), + constructorArgs: [ // see LidoTemplate._createToken + state.miniMeTokenFactory.address, + ZERO_ADDRESS, + 0, + state.daoInitialSettings.token.name, + 18, // see LidoTemplate.TOKEN_DECIMALS + state.daoInitialSettings.token.symbol, + true, + ], + } + + const evmScriptRegistryFactory = await artifacts.require('EVMScriptRegistryFactory').at(state.evmScriptRegistryFactory.address) + state.callsScript = { + address: await evmScriptRegistryFactory.baseCallScript(), + contract: await getContractPath('CallsScript'), + constructorArgs: [], // see EVMScriptRegistryFactory.baseCallScript + } + + const dataByAppName = await assertInstalledApps( + { + template, + dao, + lidoApmEnsName: state.lidoApmEnsName, + appProxyUpgradeableArtifactName: 'external:AppProxyUpgradeable_DAO' + }, + state.lidoTemplate.deployBlock + ) + + for (const [appName, appData] of Object.entries(dataByAppName)) { + const key = `app:${appName}` + const proxyAddress = appData.proxyAddress + const initializeData = appData.initializeData + delete appData.proxyAddress + delete appData.initializeData + state[key] = { + ...state[key], + aragonApp: appData, + proxy: { + address: proxyAddress, + contract: await getContractPath('AppProxyUpgradeable'), + constructorArgs: [ // see AppProxyFactory + kernelProxyAddress, + appData.id, + initializeData, + ], + } + } + } + updateAgentVestingAddressPlaceholder(state) + log.splitter() + persistNetworkState(network.name, netId, state) + + const newDaoReceipt = await web3.eth.getTransactionReceipt(state.lidoTemplateNewDaoTx) + const { abi: DAOFactoryABI } = await artifacts.readArtifact('DAOFactory') + const evmScriptRegistryEvents = getEvents(newDaoReceipt, 'DeployEVMScriptRegistry', { decodeForAbi: DAOFactoryABI }) + const evmScriptRegistryAddress = evmScriptRegistryEvents[0].args.reg + + + // Get missing proxies + const { abi: KernelABI } = await artifacts.readArtifact('Kernel') + const newAppProxyEvents = getEvents(newDaoReceipt, 'NewAppProxy', { decodeForAbi: KernelABI }) + for (const e of newAppProxyEvents) { + const appId = e.args.appId + if (appNameByAppId[appId] !== undefined) continue + + let proxyContract, appName + + if (appId == KERNEL_DEFAULT_ACL_APP_ID) { + proxyContract = 'AppProxyUpgradeable' + appName = 'aragon-acl' + } else { // otherwise it is EvmScriptRegistry + proxyContract = 'AppProxyPinned' + appName = 'aragon-evm-script-registry' + } + + const proxy = await artifacts.require(proxyContract).at(e.args.proxy) + + state[appName] = { + ...state[appName], + proxy: { + address: proxy.address, + constructorArgs: [ // See Kernel.initialize + kernelProxyAddress, + appId, + '0x00', + ], + contract: await getContractPath(proxyContract), + }, + aragonApp: { + name: appName, + id: appId, + } + } + if (appName === 'aragon-evm-script-registry') { + state[appName].implementation = { + address: await proxy.implementation(), + contract: await getContractPath('EVMScriptRegistry'), + constructorArgs: [], // see DAOFactory.newDAO and EVMScriptRegistryFactory.baseReg + } + } + } + persistNetworkState(network.name, netId, state) +} + +module.exports = runOrWrapScript(obtainDeployedAPM, module) diff --git a/scripts/scratch/10-issue-tokens.js b/scripts/scratch/12-issue-tokens.js similarity index 72% rename from scripts/scratch/10-issue-tokens.js rename to scripts/scratch/12-issue-tokens.js index 1a7f0e190..c50ee80bb 100644 --- a/scripts/scratch/10-issue-tokens.js +++ b/scripts/scratch/12-issue-tokens.js @@ -1,22 +1,16 @@ const BN = require('bn.js') -const path = require('path') const chalk = require('chalk') -const { assert } = require('chai') -const { getEvents } = require('@aragon/contract-helpers-test') -const { hash: namehash } = require('eth-ens-namehash') -const { toChecksumAddress } = require('web3-utils') +const { makeTx, TotalGasCounter } = require('../helpers/deploy') const runOrWrapScript = require('../helpers/run-or-wrap-script') const { log } = require('../helpers/log') const { assertLastEvent } = require('../helpers/events') -const { readNetworkState, persistNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') -const { saveCallTxData } = require('../helpers/tx-data') -const { resolveLatestVersion: apmResolveLatest } = require('../components/apm') +const { readNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') const { APP_NAMES } = require('../constants') const VALID_APP_NAMES = Object.entries(APP_NAMES).map((e) => e[1]) -const REQUIRED_NET_STATE = ['daoAddress', 'lidoTemplate', 'vestingParams'] +const REQUIRED_NET_STATE = ['lidoTemplate', 'vestingParams'] const MAX_HOLDERS_IN_ONE_TX = 30 @@ -33,10 +27,10 @@ async function issueTokens({ web3, artifacts }) { log.splitter() log(`Using LidoTemplate: ${chalk.yellow(daoTemplateAddress)}`) const template = await artifacts.require('LidoTemplate').at(daoTemplateAddress) - if (state.daoTemplateDeployBlock) { - log(`Using LidoTemplate deploy block: ${chalk.yellow(state.daoTemplateDeployBlock)}`) + if (state.lidoTemplate.deployBlock) { + log(`Using LidoTemplate deploy block: ${chalk.yellow(state.lidoTemplate.deployBlock)}`) } - await assertLastEvent(template, 'TmplDAOAndTokenDeployed', null, state.daoTemplateDeployBlock) + await assertLastEvent(template, 'TmplDAOAndTokenDeployed', null, state.lidoTemplate.deployBlock) log.splitter() const { vestingParams: vesting } = state @@ -78,12 +72,13 @@ async function issueTokens({ web3, artifacts }) { endTotalSupply.iadd(bigSum(iAmounts)) - await saveCallTxData(`issueTokens (batch ${i + 1})`, template, 'issueTokens', `tx-06-${i + 1}-issue-tokens.json`, { - arguments: [iHolders, iAmounts, vesting.start, vesting.cliff, vesting.end, vesting.revokable, '0x' + endTotalSupply.toString(16)], - from: state.multisigAddress, - estimateGas: i === 0 - }) + await makeTx(template, 'issueTokens', + [iHolders, iAmounts, vesting.start, vesting.cliff, vesting.end, vesting.revokable, '0x' + endTotalSupply.toString(16)] + , { from: state.deployer }, + ) } + + await TotalGasCounter.incrementTotalGasUsedInStateFile() } function bigSum(amounts, initialAmount = 0) { diff --git a/scripts/scratch/13-deploy-non-aragon-contracts.js b/scripts/scratch/13-deploy-non-aragon-contracts.js index 94a789961..e9c4c58e1 100644 --- a/scripts/scratch/13-deploy-non-aragon-contracts.js +++ b/scripts/scratch/13-deploy-non-aragon-contracts.js @@ -1,8 +1,7 @@ const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log, logSplitter, logWideSplitter, yl, gr } = require('../helpers/log') -const { readNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') -const { deployWithoutProxy, deployBehindOssifiableProxy, updateProxyImplementation } = require('../helpers/deploy') -const { ZERO_ADDRESS, bn } = require('@aragon/contract-helpers-test') +const { log, logWideSplitter, yl, gr } = require('../helpers/log') +const { readNetworkState, assertRequiredNetworkState, persistNetworkState } = require('../helpers/persisted-network-state') +const { deployWithoutProxy, deployBehindOssifiableProxy, updateProxyImplementation, deployImplementation, deployContract, getContractPath, TotalGasCounter } = require('../helpers/deploy') const { APP_NAMES } = require('../constants') @@ -15,8 +14,8 @@ const REQUIRED_NET_STATE = [ "daoInitialSettings", "oracleReportSanityChecker", "burner", - "hashConsensusForAccounting", - "hashConsensusForValidatorsExitBus", + "hashConsensusForAccountingOracle", + "hashConsensusForValidatorsExitBusOracle", "withdrawalQueueERC721", ] @@ -26,38 +25,33 @@ async function deployNewContracts({ web3, artifacts }) { log(`Network ID:`, yl(netId)) let state = readNetworkState(network.name, netId) assertRequiredNetworkState(state, REQUIRED_NET_STATE) - const lidoAddress = state["app:lido"].proxyAddress - const legacyOracleAddress = state["app:oracle"].proxyAddress - const agentAddress = state["app:aragon-agent"].proxyAddress - const votingAddress = state["app:aragon-voting"].proxyAddress + const lidoAddress = state["app:lido"].proxy.address + const legacyOracleAddress = state["app:oracle"].proxy.address + const votingAddress = state["app:aragon-voting"].proxy.address + const agentAddress = state["app:aragon-agent"].proxy.address const treasuryAddress = agentAddress - const beaconSpec = state["daoInitialSettings"]["beaconSpec"] - const depositSecurityModuleParams = state["depositSecurityModule"].parameters - const burnerParams = state["burner"].parameters - const hashConsensusForAccountingParams = state["hashConsensusForAccounting"].parameters - const hashConsensusForExitBusParams = state["hashConsensusForValidatorsExitBus"].parameters - const withdrawalQueueERC721Params = state["withdrawalQueueERC721"].parameters + const chainSpec = state["chainSpec"] + const depositSecurityModuleParams = state["depositSecurityModule"].deployParameters + const burnerParams = state["burner"].deployParameters + const hashConsensusForAccountingParams = state["hashConsensusForAccountingOracle"].deployParameters + const hashConsensusForExitBusParams = state["hashConsensusForValidatorsExitBusOracle"].deployParameters + const withdrawalQueueERC721Params = state["withdrawalQueueERC721"].deployParameters if (!DEPLOYER) { throw new Error('Deployer is not specified') } - // TODO - // const proxyContractsOwner = votingAddress const proxyContractsOwner = DEPLOYER const admin = DEPLOYER const deployer = DEPLOYER - const sanityChecks = state["oracleReportSanityChecker"].parameters + const sanityChecks = state["oracleReportSanityChecker"].deployParameters logWideSplitter() - if (!state.depositContractAddress && !state.daoInitialSettings.beaconSpec.depositContractAddress && isPublicNet) { - throw new Error(`please specify deposit contract address in state file ${networkStateFile}`) + if (!chainSpec.depositContract) { + throw new Error(`please specify deposit contract address in state file at /chainSpec/depositContract`) } - const depositContract = state.depositContractAddress || state.daoInitialSettings.beaconSpec.depositContractAddress - - // TODO: set proxyContractsOwner from state file? or from env? - + const depositContract = state.chainSpec.depositContract // // === OracleDaemonConfig === @@ -70,10 +64,16 @@ async function deployNewContracts({ web3, artifacts }) { 'oracleDaemonConfig', 'OracleDaemonConfig', deployer, oracleDaemonConfigArgs) logWideSplitter() + // + // === DummyEmptyContract === + // + const dummyContractAddress = await deployWithoutProxy('dummyEmptyContract', 'DummyEmptyContract', deployer) + + // // === LidoLocator: dummy invalid implementation === // - const locatorAddress = await deployBehindOssifiableProxy('lidoLocator', 'DummyEmptyContract', proxyContractsOwner, deployer) + const locatorAddress = await deployBehindOssifiableProxy('lidoLocator', 'DummyEmptyContract', proxyContractsOwner, deployer, [], dummyContractAddress) logWideSplitter() // @@ -130,7 +130,21 @@ async function deployNewContracts({ web3, artifacts }) { // // === WithdrawalVault === // - const withdrawalVaultAddress = await deployBehindOssifiableProxy("withdrawalVault", "WithdrawalVault", proxyContractsOwner, deployer, [lidoAddress, treasuryAddress]) + const withdrawalVaultImpl = await deployImplementation("withdrawalVault", "WithdrawalVault", deployer, [lidoAddress, treasuryAddress]) + state = readNetworkState(network.name, netId) + const withdrawalsManagerProxyConstructorArgs = [votingAddress, withdrawalVaultImpl.address] + const withdrawalsManagerProxy = await deployContract("WithdrawalsManagerProxy", withdrawalsManagerProxyConstructorArgs, deployer) + const withdrawalVaultAddress = withdrawalsManagerProxy.address + state.withdrawalVault = { + ...state.withdrawalVault, + proxy: { + contract: await getContractPath("WithdrawalsManagerProxy"), + address: withdrawalsManagerProxy.address, + constructorArgs: withdrawalsManagerProxyConstructorArgs, + }, + address: withdrawalsManagerProxy.address, + } + persistNetworkState(network.name, netId, state) logWideSplitter() // @@ -170,8 +184,8 @@ async function deployNewContracts({ web3, artifacts }) { locatorAddress, lidoAddress, legacyOracleAddress, - beaconSpec.secondsPerSlot, - beaconSpec.genesisTime, + Number(chainSpec.secondsPerSlot), + Number(chainSpec.genesisTime), ] const accountingOracleAddress = await deployBehindOssifiableProxy( "accountingOracle", "AccountingOracle", proxyContractsOwner, deployer, accountingOracleArgs) @@ -181,23 +195,23 @@ async function deployNewContracts({ web3, artifacts }) { // === HashConsensus for AccountingOracle === // const hashConsensusForAccountingArgs = [ - beaconSpec.slotsPerEpoch, - beaconSpec.secondsPerSlot, - beaconSpec.genesisTime, + chainSpec.slotsPerEpoch, + chainSpec.secondsPerSlot, + chainSpec.genesisTime, hashConsensusForAccountingParams.epochsPerFrame, hashConsensusForAccountingParams.fastLaneLengthSlots, admin, // admin accountingOracleAddress, // reportProcessor ] - await deployWithoutProxy("hashConsensusForAccounting", "HashConsensus", deployer, hashConsensusForAccountingArgs) + await deployWithoutProxy("hashConsensusForAccountingOracle", "HashConsensus", deployer, hashConsensusForAccountingArgs) logWideSplitter() // // === ValidatorsExitBusOracle === // const validatorsExitBusOracleArgs = [ - beaconSpec.secondsPerSlot, - beaconSpec.genesisTime, + chainSpec.secondsPerSlot, + chainSpec.genesisTime, locatorAddress, ] const validatorsExitBusOracleAddress = await deployBehindOssifiableProxy( @@ -208,15 +222,15 @@ async function deployNewContracts({ web3, artifacts }) { // === HashConsensus for ValidatorsExitBusOracle === // const hashConsensusForExitBusArgs = [ - beaconSpec.slotsPerEpoch, - beaconSpec.secondsPerSlot, - beaconSpec.genesisTime, + chainSpec.slotsPerEpoch, + chainSpec.secondsPerSlot, + chainSpec.genesisTime, hashConsensusForExitBusParams.epochsPerFrame, hashConsensusForExitBusParams.fastLaneLengthSlots, admin, // admin validatorsExitBusOracleAddress, // reportProcessor ] - await deployWithoutProxy("hashConsensusForValidatorsExitBus", "HashConsensus", deployer, hashConsensusForExitBusArgs) + await deployWithoutProxy("hashConsensusForValidatorsExitBusOracle", "HashConsensus", deployer, hashConsensusForExitBusArgs) logWideSplitter() @@ -254,6 +268,8 @@ async function deployNewContracts({ web3, artifacts }) { oracleDaemonConfigAddress, ] await updateProxyImplementation("lidoLocator", "LidoLocator", locatorAddress, proxyContractsOwner, [locatorConfig]) + + await TotalGasCounter.incrementTotalGasUsedInStateFile() } module.exports = runOrWrapScript(deployNewContracts, module) diff --git a/scripts/scratch/14-gate-seal.js b/scripts/scratch/14-gate-seal.js new file mode 100644 index 000000000..0c92a42b8 --- /dev/null +++ b/scripts/scratch/14-gate-seal.js @@ -0,0 +1,51 @@ +const runOrWrapScript = require('../helpers/run-or-wrap-script') +const { log, logSplitter, logWideSplitter, yl, gr } = require('../helpers/log') +const { readNetworkState, assertRequiredNetworkState, persistNetworkState } = require('../helpers/persisted-network-state') +const { getEventArgument } = require('@aragon/contract-helpers-test') +const { makeTx, TotalGasCounter } = require('../helpers/deploy') + +const { APP_NAMES } = require('../constants') + +const REQUIRED_NET_STATE = [ + "deployer", + "gateSeal", + "validatorsExitBusOracle", + "withdrawalQueueERC721", +] + +async function deployNewContracts({ web3, artifacts }) { + const netId = await web3.eth.net.getId() + logWideSplitter() + log(`Network ID:`, yl(netId)) + let state = readNetworkState(network.name, netId) + assertRequiredNetworkState(state, REQUIRED_NET_STATE) + + if (state.gateSeal.address !== "") { + console.log(`Using the specified GateSeal address ${state.gateSeal.address}`) + return + } + + const gateSealFactoryAddress = state.gateSeal.factoryAddress + const deployer = state.deployer + const sealables = [ + state.withdrawalQueueERC721.proxy.address, + state.validatorsExitBusOracle.proxy.address, + ] + + const GateSealFactory = await artifacts.require("IGateSealFactory") + const gateSealFactory = await GateSealFactory.at(gateSealFactoryAddress) + const receipt = await makeTx(gateSealFactory, "create_gate_seal", [ + state.gateSeal.sealingCommittee, + state.gateSeal.sealDuration, + sealables, + state.gateSeal.expiryTimestamp, + ], { from: deployer }) + const gateSealAddress = await getEventArgument(receipt, 'GateSealCreated', 'gate_seal') + console.log(`GateSeal created: ${gateSealAddress}`) + state.gateSeal.address = gateSealAddress + persistNetworkState(network.name, netId, state) + + await TotalGasCounter.incrementTotalGasUsedInStateFile() +} + +module.exports = runOrWrapScript(deployNewContracts, module) diff --git a/scripts/scratch/11-finalize-dao.js b/scripts/scratch/15-finalize-dao.js similarity index 69% rename from scripts/scratch/11-finalize-dao.js rename to scripts/scratch/15-finalize-dao.js index efa5bc0a6..c0ae3a41a 100644 --- a/scripts/scratch/11-finalize-dao.js +++ b/scripts/scratch/15-finalize-dao.js @@ -3,24 +3,22 @@ const { assert } = require('chai') const runOrWrapScript = require('../helpers/run-or-wrap-script') const { log } = require('../helpers/log') -const { readNetworkState, persistNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') -const { saveCallTxData } = require('../helpers/tx-data') +const { readNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') const { assertLastEvent } = require('../helpers/events') -const { percentToBP } = require('../helpers/index') +const { makeTx, TotalGasCounter } = require('../helpers/deploy') const { APP_NAMES } = require('../constants') const { assertVesting } = require('./checks/dao-token') const REQUIRED_NET_STATE = [ - 'daoAddress', - 'daoTokenAddress', + 'ldo', 'lidoTemplate', 'daoAragonId', 'daoInitialSettings', 'vestingParams', `app:${APP_NAMES.ARAGON_TOKEN_MANAGER}`, - 'executionLayerRewardsParams', 'stakingRouter', + 'deployer', ] async function finalizeDAO({ web3, artifacts }) { @@ -38,24 +36,23 @@ async function finalizeDAO({ web3, artifacts }) { log(`Using LidoTemplate: ${chalk.yellow(daoTemplateAddress)}`) const template = await artifacts.require('LidoTemplate').at(daoTemplateAddress) - if (state.daoTemplateDeployBlock) { - log(`Using LidoTemplate deploy block: ${chalk.yellow(state.daoTemplateDeployBlock)}`) + if (state.lidoTemplate.deployBlock) { + log(`Using LidoTemplate deploy block: ${chalk.yellow(state.lidoTemplate.deployBlock)}`) } - await assertLastEvent(template, 'TmplTokensIssued', null, state.daoTemplateDeployBlock) + await assertLastEvent(template, 'TmplTokensIssued', null, state.lidoTemplate.deployBlock) log.splitter() - const tokenManagerAddress = state[`app:${APP_NAMES.ARAGON_TOKEN_MANAGER}`].proxyAddress + const tokenManagerAddress = state[`app:${APP_NAMES.ARAGON_TOKEN_MANAGER}`].proxy.address log(`Using TokenManager:`, chalk.yellow(tokenManagerAddress)) const tokenManager = await artifacts.require('TokenManager').at(tokenManagerAddress) - log(`Using MiniMeToken`, chalk.yellow(state.daoTokenAddress)) - const daoToken = await artifacts.require('MiniMeToken').at(state.daoTokenAddress) + log(`Using MiniMeToken`, chalk.yellow(state.ldo.address)) + const daoToken = await artifacts.require('MiniMeToken').at(state.ldo.address) const { fee } = state.daoInitialSettings log(`Using fee initial settings:`) log(` total fee:`, chalk.yellow(`${fee.totalPercent}%`)) log(` treasury fee:`, chalk.yellow(`${fee.treasuryPercent}%`)) - log(` insurance fee:`, chalk.yellow(`${fee.insurancePercent}%`)) log(` node operators fee:`, chalk.yellow(`${fee.nodeOperatorsPercent}%`)) await assertVesting({ @@ -66,17 +63,15 @@ async function finalizeDAO({ web3, artifacts }) { unvestedTokensAmount: '0' // since we're minting them during the finalizeDAO call below } }) - log.splitter() - await saveCallTxData(`finalizeDAO`, template, 'finalizeDAO', `tx-11-finalize-dao.json`, { - arguments: [ - state.daoAragonId, - state.vestingParams.unvestedTokensAmount, - state.stakingRouter.address, - ], - from: state.multisigAddress - }) + await makeTx(template, 'finalizeDAO', [ + state.daoAragonId, + state.vestingParams.unvestedTokensAmount, + state.stakingRouter.proxy.address + ], { from: state.deployer }) + + await TotalGasCounter.incrementTotalGasUsedInStateFile() } module.exports = runOrWrapScript(finalizeDAO, module) diff --git a/scripts/scratch/15-grant-roles.js b/scripts/scratch/15-grant-roles.js deleted file mode 100644 index 5710829f3..000000000 --- a/scripts/scratch/15-grant-roles.js +++ /dev/null @@ -1,103 +0,0 @@ -const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log, logSplitter, logWideSplitter, yl, gr } = require('../helpers/log') -const { readNetworkState, assertRequiredNetworkState, persistNetworkState } = require('../helpers/persisted-network-state') -const { ZERO_ADDRESS, bn } = require('@aragon/contract-helpers-test') - -const { APP_NAMES } = require('../constants') - - -const DEPLOYER = process.env.DEPLOYER || '' -const REQUIRED_NET_STATE = [ - `app:${APP_NAMES.LIDO}`, - `app:${APP_NAMES.ORACLE}`, - "app:aragon-agent", - "app:aragon-voting", - "app:node-operators-registry", - "accountingOracle", - "burner", - "daoInitialSettings", - "eip712StETH", - "hashConsensusForAccounting", - "hashConsensusForValidatorsExitBus", - "lidoLocator", - "stakingRouter", - "validatorsExitBusOracle", - "withdrawalQueueERC721", - "withdrawalVault", - "gateSealAddress", -] - - -async function deployNewContracts({ web3, artifacts }) { - const netId = await web3.eth.net.getId() - logWideSplitter() - log(`Network ID:`, yl(netId)) - - let state = readNetworkState(network.name, netId) - assertRequiredNetworkState(state, REQUIRED_NET_STATE) - - const agent = state["app:aragon-agent"].proxyAddress - const lidoAddress = state["app:lido"].proxyAddress - const legacyOracleAddress = state["app:oracle"].proxyAddress - const nodeOperatorsRegistryAddress = state["app:node-operators-registry"].proxyAddress - const gateSealAddress = state["gateSealAddress"] - - const validatorsExitBusOracleParams = state["validatorsExitBusOracle"].parameters - const accountingOracleParams = state["accountingOracle"].parameters - - const burnerAddress = state["burner"].address - const stakingRouterAddress = state["stakingRouter"].address - const withdrawalQueueAddress = state["withdrawalQueueERC721"].address - const lidoLocatorAddress = state["lidoLocator"].address - const accountingOracleAddress = state["accountingOracle"].address - const hashConsensusForAccountingAddress = state["hashConsensusForAccounting"].address - const validatorsExitBusOracleAddress = state["validatorsExitBusOracle"].address - const hashConsensusForValidatorsExitBusOracleAddress = state["hashConsensusForValidatorsExitBus"].address - const eip712StETHAddress = state["eip712StETH"].address - const withdrawalVaultAddress = state["withdrawalVault"].address - const depositSecurityModuleAddress = state.depositSecurityModule.address - - const testnetAdmin = DEPLOYER - const accountingOracleAdmin = testnetAdmin - const exitBusOracleAdmin = testnetAdmin - const stakingRouterAdmin = testnetAdmin - const withdrawalQueueAdmin = testnetAdmin - // TODO - // const votingAddress = state["app:aragon-voting"].proxyAddress - const votingAddress = testnetAdmin - - // - // === StakingRouter - // - const stakingRouter = await artifacts.require('StakingRouter').at(stakingRouterAddress) - await stakingRouter.grantRole(await stakingRouter.STAKING_MODULE_PAUSE_ROLE(), depositSecurityModuleAddress, { from: stakingRouterAdmin }) - await stakingRouter.grantRole(await stakingRouter.REPORT_EXITED_VALIDATORS_ROLE(), accountingOracleAddress, { from: stakingRouterAdmin }) - await stakingRouter.grantRole(await stakingRouter.REPORT_REWARDS_MINTED_ROLE(), lidoAddress, { from: stakingRouterAdmin }) - logWideSplitter() - - // - // === ValidatorsExitBusOracle - // - const validatorsExitBusOracle = await artifacts.require('ValidatorsExitBusOracle').at(validatorsExitBusOracleAddress) - await validatorsExitBusOracle.grantRole(await validatorsExitBusOracle.PAUSE_ROLE(), gateSealAddress, { from: testnetAdmin }) - logWideSplitter() - - // - // === WithdrawalQueue - // - const withdrawalQueue = await artifacts.require('WithdrawalQueueERC721').at(withdrawalQueueAddress) - await withdrawalQueue.grantRole(await withdrawalQueue.PAUSE_ROLE(), gateSealAddress, { from: testnetAdmin }) - await withdrawalQueue.grantRole(await withdrawalQueue.FINALIZE_ROLE(), lidoAddress, { from: testnetAdmin }) - await withdrawalQueue.grantRole(await withdrawalQueue.ORACLE_ROLE(), accountingOracleAddress, { from: testnetAdmin }) - logWideSplitter() - - // - // === Burner - // - const burner = await artifacts.require('Burner').at(burnerAddress) - // NB: REQUEST_BURN_SHARES_ROLE is already granted to Lido in Burner constructor - await burner.grantRole(await burner.REQUEST_BURN_SHARES_ROLE(), nodeOperatorsRegistryAddress, { from: testnetAdmin }) - -} - -module.exports = runOrWrapScript(deployNewContracts, module) diff --git a/scripts/scratch/14-initialize-non-aragon-contracts.js b/scripts/scratch/16-initialize-non-aragon-contracts.js similarity index 61% rename from scripts/scratch/14-initialize-non-aragon-contracts.js rename to scripts/scratch/16-initialize-non-aragon-contracts.js index 140031a21..e4c0c5150 100644 --- a/scripts/scratch/14-initialize-non-aragon-contracts.js +++ b/scripts/scratch/16-initialize-non-aragon-contracts.js @@ -1,10 +1,9 @@ const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log, logSplitter, logWideSplitter, yl, gr } = require('../helpers/log') -const { readNetworkState, assertRequiredNetworkState, persistNetworkState } = require('../helpers/persisted-network-state') -const { ZERO_ADDRESS, bn } = require('@aragon/contract-helpers-test') -const { web3 } = require('hardhat') - +const { log, logWideSplitter, yl, gr } = require('../helpers/log') +const { readNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') +const { hexPaddedToByte } = require('../../test/helpers/utils') const { APP_NAMES } = require('../constants') +const { makeTx, TotalGasCounter } = require('../helpers/deploy') const DEPLOYER = process.env.DEPLOYER || '' @@ -19,9 +18,9 @@ const REQUIRED_NET_STATE = [ "daoInitialSettings", "eip712StETH", "accountingOracle", - "hashConsensusForAccounting", + "hashConsensusForAccountingOracle", "validatorsExitBusOracle", - "hashConsensusForValidatorsExitBus", + "hashConsensusForValidatorsExitBusOracle", "withdrawalQueueERC721", "withdrawalVault", "nodeOperatorsRegistry", @@ -36,25 +35,24 @@ async function deployNewContracts({ web3, artifacts }) { let state = readNetworkState(network.name, netId) assertRequiredNetworkState(state, REQUIRED_NET_STATE) - const agent = state["app:aragon-agent"].proxyAddress - const votingAddress = state["app:aragon-voting"].proxyAddress - const lidoAddress = state["app:lido"].proxyAddress - const legacyOracleAddress = state["app:oracle"].proxyAddress - const nodeOperatorsRegistryAddress = state["app:node-operators-registry"].proxyAddress - const nodeOperatorsRegistryParams = state["nodeOperatorsRegistry"].parameters - - const validatorsExitBusOracleParams = state["validatorsExitBusOracle"].parameters - const accountingOracleParams = state["accountingOracle"].parameters - - const stakingRouterAddress = state["stakingRouter"].address - const withdrawalQueueAddress = state["withdrawalQueueERC721"].address - const lidoLocatorAddress = state["lidoLocator"].address - const accountingOracleAddress = state["accountingOracle"].address - const hashConsensusForAccountingAddress = state["hashConsensusForAccounting"].address - const ValidatorsExitBusOracleAddress = state["validatorsExitBusOracle"].address - const hashConsensusForValidatorsExitBusOracleAddress = state["hashConsensusForValidatorsExitBus"].address + const lidoAddress = state["app:lido"].proxy.address + const legacyOracleAddress = state["app:oracle"].proxy.address + const nodeOperatorsRegistryAddress = state["app:node-operators-registry"].proxy.address + const nodeOperatorsRegistryParams = state["nodeOperatorsRegistry"].deployParameters + + const validatorsExitBusOracleParams = state["validatorsExitBusOracle"].deployParameters + const accountingOracleParams = state["accountingOracle"].deployParameters + + const stakingRouterAddress = state["stakingRouter"].proxy.address + const withdrawalQueueAddress = state["withdrawalQueueERC721"].proxy.address + const lidoLocatorAddress = state["lidoLocator"].proxy.address + const accountingOracleAddress = state["accountingOracle"].proxy.address + const hashConsensusForAccountingAddress = state["hashConsensusForAccountingOracle"].address + const ValidatorsExitBusOracleAddress = state["validatorsExitBusOracle"].proxy.address + const hashConsensusForValidatorsExitBusOracleAddress = state["hashConsensusForValidatorsExitBusOracle"].address const eip712StETHAddress = state["eip712StETH"].address - const withdrawalVaultAddress = state["withdrawalVault"].address + const withdrawalVaultAddress = state["withdrawalVault"].proxy.address + const oracleDaemonConfigAddress = state.oracleDaemonConfig.address const testnetAdmin = DEPLOYER const accountingOracleAdmin = testnetAdmin @@ -75,10 +73,7 @@ async function deployNewContracts({ web3, artifacts }) { nodeOperatorsRegistryParams.stuckPenaltyDelay, ] const nodeOperatorsRegistry = await artifacts.require('NodeOperatorsRegistry').at(nodeOperatorsRegistryAddress) - await nodeOperatorsRegistry.initialize( - ...nodeOperatorsRegistryArgs, - { from: DEPLOYER }, - ) + await makeTx(nodeOperatorsRegistry, 'initialize', nodeOperatorsRegistryArgs, { from: DEPLOYER }) // // === Lido: initialize === @@ -87,10 +82,9 @@ async function deployNewContracts({ web3, artifacts }) { lidoLocatorAddress, eip712StETHAddress, ] - console.log({ lidoInitArgs }) const bootstrapInitBalance = 10 // wei const lido = await artifacts.require('Lido').at(lidoAddress) - await lido.initialize(...lidoInitArgs, { value: bootstrapInitBalance, from: DEPLOYER }) + await makeTx(lido, 'initialize', lidoInitArgs, { value: bootstrapInitBalance, from: DEPLOYER }) logWideSplitter() // @@ -100,9 +94,8 @@ async function deployNewContracts({ web3, artifacts }) { lidoLocatorAddress, hashConsensusForAccountingAddress, ] - console.log({legacyOracleArgs}) const legacyOracle = await artifacts.require('LegacyOracle').at(legacyOracleAddress) - await legacyOracle.initialize(...legacyOracleArgs, { from: DEPLOYER }) + await makeTx(legacyOracle, 'initialize', legacyOracleArgs, { from: DEPLOYER }) const zeroLastProcessingRefSlot = 0 @@ -117,20 +110,19 @@ async function deployNewContracts({ web3, artifacts }) { accountingOracleParams.consensusVersion, zeroLastProcessingRefSlot, ] - console.log({accountingOracleArgs}) - await accountingOracle.initializeWithoutMigration(...accountingOracleArgs, { from: DEPLOYER }) + await makeTx(accountingOracle, 'initializeWithoutMigration', accountingOracleArgs, { from: DEPLOYER }) // // === ValidatorsExitBusOracle: initialize === // - const ValidatorsExitBusOracle = await artifacts.require('ValidatorsExitBusOracle').at(ValidatorsExitBusOracleAddress) + const validatorsExitBusOracle = await artifacts.require('ValidatorsExitBusOracle').at(ValidatorsExitBusOracleAddress) const validatorsExitBusOracleArgs = [ exitBusOracleAdmin, // admin hashConsensusForValidatorsExitBusOracleAddress, validatorsExitBusOracleParams.consensusVersion, zeroLastProcessingRefSlot, ] - await ValidatorsExitBusOracle.initialize(...validatorsExitBusOracleArgs, { from: DEPLOYER }) + await makeTx(validatorsExitBusOracle, 'initialize', validatorsExitBusOracleArgs, { from: DEPLOYER }) // // === WithdrawalQueue initialize === @@ -138,31 +130,34 @@ async function deployNewContracts({ web3, artifacts }) { const withdrawalQueueArgs = [ withdrawalQueueAdmin, // _admin ] - console.log({ withdrawalQueueArgs }) const withdrawalQueue = await artifacts.require('WithdrawalQueueERC721').at(withdrawalQueueAddress) - await withdrawalQueue.initialize( - ...withdrawalQueueArgs, - { from: DEPLOYER }, - ) + await makeTx(withdrawalQueue, 'initialize', withdrawalQueueArgs, { from: DEPLOYER }) // // === StakingRouter: initialize === // const withdrawalCredentials = `0x010000000000000000000000${withdrawalVaultAddress.slice(2)}` - console.log({withdrawalCredentials}) const stakingRouterArgs = [ stakingRouterAdmin, // _admin lidoAddress, // _lido withdrawalCredentials, // _withdrawalCredentials ] - console.log({ stakingRouterArgs }) const stakingRouter = await artifacts.require('StakingRouter').at(stakingRouterAddress) - await stakingRouter.initialize( - ...stakingRouterArgs, - { from: DEPLOYER }, - ) + await makeTx(stakingRouter, 'initialize', stakingRouterArgs, { from: DEPLOYER }) logWideSplitter() + // + // === OracleDaemonConfig: set parameters === + // + const oracleDaemonConfig = await artifacts.require('OracleDaemonConfig').at(oracleDaemonConfigAddress) + const CONFIG_MANAGER_ROLE = await oracleDaemonConfig.CONFIG_MANAGER_ROLE() + await makeTx(oracleDaemonConfig, 'grantRole', [CONFIG_MANAGER_ROLE, testnetAdmin], { from: testnetAdmin }) + for (const [key, value] of Object.entries(state.oracleDaemonConfig.deployParameters)) { + await makeTx(oracleDaemonConfig, 'set', [key, hexPaddedToByte(value)], { from: DEPLOYER }) + } + await makeTx(oracleDaemonConfig, 'renounceRole', [CONFIG_MANAGER_ROLE, testnetAdmin], { from: testnetAdmin }) + + await TotalGasCounter.incrementTotalGasUsedInStateFile() } module.exports = runOrWrapScript(deployNewContracts, module) diff --git a/scripts/scratch/17-grant-roles.js b/scripts/scratch/17-grant-roles.js new file mode 100644 index 000000000..c7b2b40eb --- /dev/null +++ b/scripts/scratch/17-grant-roles.js @@ -0,0 +1,87 @@ +const runOrWrapScript = require('../helpers/run-or-wrap-script') +const { log, logSplitter, logWideSplitter, yl, gr } = require('../helpers/log') +const { readNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') + +const { APP_NAMES } = require('../constants') +const { makeTx, TotalGasCounter } = require('../helpers/deploy') + + +const REQUIRED_NET_STATE = [ + `app:${APP_NAMES.LIDO}`, + `app:${APP_NAMES.ORACLE}`, + "app:aragon-agent", + "app:aragon-voting", + "app:node-operators-registry", + "accountingOracle", + "burner", + "daoInitialSettings", + "eip712StETH", + "hashConsensusForAccountingOracle", + "hashConsensusForValidatorsExitBusOracle", + "lidoLocator", + "stakingRouter", + "validatorsExitBusOracle", + "withdrawalQueueERC721", + "withdrawalVault", + "gateSeal", +] + + +async function deployNewContracts({ web3, artifacts }) { + const netId = await web3.eth.net.getId() + logWideSplitter() + log(`Network ID:`, yl(netId)) + + let state = readNetworkState(network.name, netId) + assertRequiredNetworkState(state, REQUIRED_NET_STATE) + + const lidoAddress = state["app:lido"].proxy.address + const nodeOperatorsRegistryAddress = state["app:node-operators-registry"].proxy.address + const gateSealAddress = state.gateSeal.address + + const burnerAddress = state["burner"].address + const stakingRouterAddress = state["stakingRouter"].proxy.address + const withdrawalQueueAddress = state["withdrawalQueueERC721"].proxy.address + const accountingOracleAddress = state["accountingOracle"].proxy.address + const validatorsExitBusOracleAddress = state["validatorsExitBusOracle"].proxy.address + const depositSecurityModuleAddress = state.depositSecurityModule.address + + const deployer = state.deployer + + // + // === StakingRouter + // + const stakingRouter = await artifacts.require('StakingRouter').at(stakingRouterAddress) + await makeTx(stakingRouter, 'grantRole', [await stakingRouter.STAKING_MODULE_PAUSE_ROLE(), depositSecurityModuleAddress], { from: deployer }) + await makeTx(stakingRouter, 'grantRole', [await stakingRouter.STAKING_MODULE_RESUME_ROLE(), depositSecurityModuleAddress], { from: deployer }) + await makeTx(stakingRouter, 'grantRole', [await stakingRouter.REPORT_EXITED_VALIDATORS_ROLE(), accountingOracleAddress], { from: deployer }) + await makeTx(stakingRouter, 'grantRole', [await stakingRouter.REPORT_REWARDS_MINTED_ROLE(), lidoAddress], { from: deployer }) + logWideSplitter() + + // + // === ValidatorsExitBusOracle + // + const validatorsExitBusOracle = await artifacts.require('ValidatorsExitBusOracle').at(validatorsExitBusOracleAddress) + await makeTx(validatorsExitBusOracle, 'grantRole', [await validatorsExitBusOracle.PAUSE_ROLE(), gateSealAddress], { from: deployer }) + logWideSplitter() + + // + // === WithdrawalQueue + // + const withdrawalQueue = await artifacts.require('WithdrawalQueueERC721').at(withdrawalQueueAddress) + await makeTx(withdrawalQueue, 'grantRole', [await withdrawalQueue.PAUSE_ROLE(), gateSealAddress], { from: deployer }) + await makeTx(withdrawalQueue, 'grantRole', [await withdrawalQueue.FINALIZE_ROLE(), lidoAddress], { from: deployer }) + await makeTx(withdrawalQueue, 'grantRole', [await withdrawalQueue.ORACLE_ROLE(), accountingOracleAddress], { from: deployer }) + logWideSplitter() + + // + // === Burner + // + const burner = await artifacts.require('Burner').at(burnerAddress) + // NB: REQUEST_BURN_SHARES_ROLE is already granted to Lido in Burner constructor + await makeTx(burner, 'grantRole', [await burner.REQUEST_BURN_SHARES_ROLE(), nodeOperatorsRegistryAddress], { from: deployer }) + + await TotalGasCounter.incrementTotalGasUsedInStateFile() +} + +module.exports = runOrWrapScript(deployNewContracts, module) diff --git a/scripts/scratch/18-plug-curated-staking-module.js b/scripts/scratch/18-plug-curated-staking-module.js new file mode 100644 index 000000000..d1cd011df --- /dev/null +++ b/scripts/scratch/18-plug-curated-staking-module.js @@ -0,0 +1,61 @@ +const runOrWrapScript = require('../helpers/run-or-wrap-script') +const { log, logSplitter, logWideSplitter, yl, gr } = require('../helpers/log') +const { readNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') + +const { APP_NAMES } = require('../constants') +const { makeTx, TotalGasCounter } = require('../helpers/deploy') + + +const REQUIRED_NET_STATE = [ + "stakingRouter", + "app:node-operators-registry", + "deployer", + "app:aragon-agent", + `app:${APP_NAMES.LIDO}`, + `app:${APP_NAMES.ORACLE}`, + "app:aragon-voting", + "accountingOracle", + "burner", + "daoInitialSettings", + "eip712StETH", + "hashConsensusForAccountingOracle", + "hashConsensusForValidatorsExitBusOracle", + "lidoLocator", + "validatorsExitBusOracle", + "withdrawalQueueERC721", + "withdrawalVault", +] + +const NOR_STAKING_MODULE_TARGET_SHARE_BP = 10000 // 100% +const NOR_STAKING_MODULE_MODULE_FEE_BP = 500 // 5% +const NOR_STAKING_MODULE_TREASURY_FEE_BP = 500 // 5% +const STAKING_MODULE_MANAGE_ROLE = web3.utils.keccak256('STAKING_MODULE_MANAGE_ROLE') + + +async function deployNewContracts({ web3, artifacts }) { + const netId = await web3.eth.net.getId() + logWideSplitter() + log(`Network ID:`, yl(netId)) + + let state = readNetworkState(network.name, netId) + assertRequiredNetworkState(state, REQUIRED_NET_STATE) + + const deployer = state.deployer + const stakingRouter = await artifacts.require('StakingRouter').at(state.stakingRouter.proxy.address) + const nodeOperatorsRegistry = await artifacts.require('NodeOperatorsRegistry').at(state['app:node-operators-registry'].proxy.address) + + await makeTx(stakingRouter, 'grantRole', [STAKING_MODULE_MANAGE_ROLE, deployer], { from: deployer }) + + await makeTx(stakingRouter, 'addStakingModule', [ + state.nodeOperatorsRegistry.deployParameters.stakingModuleTypeId, + nodeOperatorsRegistry.address, + NOR_STAKING_MODULE_TARGET_SHARE_BP, + NOR_STAKING_MODULE_MODULE_FEE_BP, + NOR_STAKING_MODULE_TREASURY_FEE_BP, + ], { from: deployer }) + await makeTx(stakingRouter, 'renounceRole', [STAKING_MODULE_MANAGE_ROLE, deployer], { from: deployer }) + + await TotalGasCounter.incrementTotalGasUsedInStateFile() +} + +module.exports = runOrWrapScript(deployNewContracts, module) diff --git a/scripts/scratch/19-transfer-roles.js b/scripts/scratch/19-transfer-roles.js new file mode 100644 index 000000000..8d918cdae --- /dev/null +++ b/scripts/scratch/19-transfer-roles.js @@ -0,0 +1,76 @@ +const runOrWrapScript = require('../helpers/run-or-wrap-script') +const { log, logSplitter, logWideSplitter, yl, gr, OK } = require('../helpers/log') +const { readNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') +const { makeTx, TotalGasCounter } = require('../helpers/deploy') + +const REQUIRED_NET_STATE = [ + "app:aragon-agent", + "accountingOracle", + "burner", + "daoInitialSettings", + "hashConsensusForAccountingOracle", + "hashConsensusForValidatorsExitBusOracle", + "lidoLocator", + "stakingRouter", + "validatorsExitBusOracle", + "withdrawalQueueERC721", +] + +const DEFAULT_ADMIN_ROLE = "0x00" + + +async function transferOZAdmin(contractName, contractAddress, currentAdmin, newAdmin) { + console.log(`Transferring OZ admin of ${contractAddress} from ${currentAdmin} to ${newAdmin}:`) + const contract = await artifacts.require(contractName).at(contractAddress) + await makeTx(contract, 'grantRole', [DEFAULT_ADMIN_ROLE, newAdmin], { from: currentAdmin }) + await makeTx(contract, 'renounceRole', [DEFAULT_ADMIN_ROLE, currentAdmin], { from: currentAdmin }) + console.log() +} + +async function changeOssifiableProxyAdmin(contractAddress, currentAdmin, newAdmin) { + console.log(`Transferring OssifiableProxy admin of ${contractAddress} from ${currentAdmin} to ${newAdmin}...`) + const contract = await artifacts.require('OssifiableProxy').at(contractAddress) + await makeTx(contract, 'proxy__changeAdmin', [newAdmin], { from: currentAdmin }) + console.log() +} + +async function changeDepositSecurityModuleAdmin(contractAddress, currentAdmin, newAdmin) { + console.log(`Changing DepositSecurityModule owner of ${contractAddress} from ${currentAdmin} to ${newAdmin}...`) + const depositSecurityModule = await artifacts.require('DepositSecurityModule').at(contractAddress) + await makeTx(depositSecurityModule, 'setOwner', [newAdmin], { from: currentAdmin } ) + console.log() +} + +async function deployNewContracts({ web3, artifacts }) { + const netId = await web3.eth.net.getId() + logWideSplitter() + log(`Network ID:`, yl(netId)) + + let state = readNetworkState(network.name, netId) + assertRequiredNetworkState(state, REQUIRED_NET_STATE) + + const deployer = state.deployer + const agent = state["app:aragon-agent"].proxy.address + + await transferOZAdmin('Burner', state.burner.address, deployer, agent) + await transferOZAdmin('HashConsensus', state.hashConsensusForAccountingOracle.address, deployer, agent) + await transferOZAdmin('HashConsensus', state.hashConsensusForValidatorsExitBusOracle.address, deployer, agent) + await transferOZAdmin('StakingRouter', state.stakingRouter.proxy.address, deployer, agent) + await transferOZAdmin('AccountingOracle', state.accountingOracle.proxy.address, deployer, agent) + await transferOZAdmin('ValidatorsExitBusOracle', state.validatorsExitBusOracle.proxy.address, deployer, agent) + await transferOZAdmin('WithdrawalQueueERC721', state.withdrawalQueueERC721.proxy.address, deployer, agent) + await transferOZAdmin('OracleDaemonConfig', state.oracleDaemonConfig.address, deployer, agent) + await transferOZAdmin('OracleReportSanityChecker', state.oracleReportSanityChecker.address, deployer, agent) + + await changeOssifiableProxyAdmin(state.lidoLocator.proxy.address, deployer, agent) + await changeOssifiableProxyAdmin(state.stakingRouter.proxy.address, deployer, agent) + await changeOssifiableProxyAdmin(state.accountingOracle.proxy.address, deployer, agent) + await changeOssifiableProxyAdmin(state.validatorsExitBusOracle.proxy.address, deployer, agent) + await changeOssifiableProxyAdmin(state.withdrawalQueueERC721.proxy.address, deployer, agent) + + await changeDepositSecurityModuleAdmin(state.depositSecurityModule.address, deployer, agent) + + await TotalGasCounter.incrementTotalGasUsedInStateFile() +} + +module.exports = runOrWrapScript(deployNewContracts, module) diff --git a/scripts/scratch/12-check-dao.js b/scripts/scratch/20-check-dao.js similarity index 62% rename from scripts/scratch/12-check-dao.js rename to scripts/scratch/20-check-dao.js index 3b8516ead..f650bee9a 100644 --- a/scripts/scratch/12-check-dao.js +++ b/scripts/scratch/20-check-dao.js @@ -1,10 +1,10 @@ const path = require('path') +const fs = require('fs') const chalk = require('chalk') const BN = require('bn.js') const { assertBn } = require('@aragon/contract-helpers-test/src/asserts') const { getEvents } = require('@aragon/contract-helpers-test') const { hash: namehash } = require('eth-ens-namehash') -const { toChecksumAddress } = require('web3-utils') const runOrWrapScript = require('../helpers/run-or-wrap-script') const { log, yl } = require('../helpers/log') @@ -14,6 +14,7 @@ const { assertLastEvent, assertSingleEvent } = require('../helpers/events') const { assert } = require('../helpers/assert') const { percentToBP } = require('../helpers/index') const { resolveEnsAddress } = require('../components/ens') +const { isAddress } = require('web3-utils') const { APP_NAMES } = require('../constants') @@ -21,7 +22,7 @@ const { assertAPMRegistryPermissions } = require('./checks/apm') const { assertInstalledApps } = require('./checks/apps') const { assertVesting } = require('./checks/dao-token') -const REQUIRED_NET_STATE = ['ensAddress', 'lidoApmEnsName', 'daoAragonId', 'vestingParams', 'daoInitialSettings', 'lidoTemplate'] +const REQUIRED_NET_STATE = ['ens', 'lidoApmEnsName', 'daoAragonId', 'vestingParams', 'daoInitialSettings', 'lidoTemplate'] const TOKEN_TRANSFERABLE = true const TOKEN_DECIMALS = 18 @@ -35,6 +36,8 @@ const STETH_TOKEN_DECIMALS = 18 const ZERO_WITHDRAWAL_CREDENTIALS = '0x0000000000000000000000000000000000000000000000000000000000000000' const PROTOCOL_PAUSED_AFTER_DEPLOY = true +const OSSIFIABLE_PROXY = 'OssifiableProxy' +const ACCESS_CONTROL_ENUMERABLE = 'AccessControlEnumerable' const DAO_LIVE = /^true|1$/i.test(process.env.DAO_LIVE) @@ -50,35 +53,35 @@ async function checkDAO({ web3, artifacts }) { log.splitter() - log(`Using ENS:`, yl(state.ensAddress)) - const ens = await artifacts.require('ENS').at(state.ensAddress) + log(`Using ENS:`, yl(state.ens.address)) + const ens = await artifacts.require('ENS').at(state.ens.address) log.splitter() log(`Using LidoTemplate: ${yl(daoTemplateAddress)}`) const template = await artifacts.require('LidoTemplate').at(daoTemplateAddress) - if (state.daoTemplateDeployBlock) { - log(`Using LidoTemplate deploy block: ${chalk.yellow(state.daoTemplateDeployBlock)}`) + if (state.lidoTemplate.deployBlock) { + log(`Using LidoTemplate deploy block: ${chalk.yellow(state.lidoTemplate.deployBlock)}`) } - await assertLastEvent(template, 'TmplDaoFinalized', null, state.daoTemplateDeployBlock) + await assertLastEvent(template, 'TmplDaoFinalized', null, state.lidoTemplate.deployBlock) - const apmDeployedEvt = await assertSingleEvent(template, 'TmplAPMDeployed', null, state.daoTemplateDeployBlock) - const daoDeployedEvt = await assertSingleEvent(template, 'TmplDAOAndTokenDeployed', null, state.daoTemplateDeployBlock) + const apmDeployedEvt = await assertSingleEvent(template, 'TmplAPMDeployed', null, state.lidoTemplate.deployBlock) + const daoDeployedEvt = await assertSingleEvent(template, 'TmplDAOAndTokenDeployed', null, state.lidoTemplate.deployBlock) - state.lidoApmAddress = apmDeployedEvt.args.apm - state.daoAddress = daoDeployedEvt.args.dao - state.daoTokenAddress = daoDeployedEvt.args.token + lidoApmAddress = apmDeployedEvt.args.apm + daoAddress = daoDeployedEvt.args.dao + daoTokenAddress = daoDeployedEvt.args.token log.splitter() - log(`Using APMRegistry:`, yl(state.lidoApmAddress)) - const registry = await artifacts.require('APMRegistry').at(state.lidoApmAddress) + log(`Using APMRegistry:`, yl(lidoApmAddress)) + const registry = await artifacts.require('APMRegistry').at(lidoApmAddress) - log(`Using Kernel:`, yl(state.daoAddress)) - const dao = await artifacts.require('Kernel').at(state.daoAddress) + log(`Using Kernel:`, yl(daoAddress)) + const dao = await artifacts.require('Kernel').at(daoAddress) - log(`Using MiniMeToken:`, yl(state.daoTokenAddress)) - const daoToken = await artifacts.require('MiniMeToken').at(state.daoTokenAddress) + log(`Using MiniMeToken:`, yl(daoTokenAddress)) + const daoToken = await artifacts.require('MiniMeToken').at(daoTokenAddress) log.splitter() @@ -89,7 +92,7 @@ async function checkDAO({ web3, artifacts }) { lidoApmEnsName: state.lidoApmEnsName, appProxyUpgradeableArtifactName: 'external:AppProxyUpgradeable_DAO' }, - state.daoTemplateDeployBlock + state.lidoTemplate.deployBlock ) log.splitter() @@ -101,26 +104,28 @@ async function checkDAO({ web3, artifacts }) { state[key] = { ...state[key], ...app } } - const [lido, oracle, nopsRegistry, agent, finance, tokenManager, voting] = await Promise.all([ + const [lido, legacyOracle, nopsRegistry, agent, finance, tokenManager, voting, burner, elRewardsVault, stakingRouter] = await Promise.all([ artifacts.require('Lido').at(apps[APP_NAMES.LIDO].proxyAddress), + artifacts.require('LegacyOracle').at(apps[APP_NAMES.ORACLE].proxyAddress), artifacts.require('NodeOperatorsRegistry').at(apps[APP_NAMES.NODE_OPERATORS_REGISTRY].proxyAddress), artifacts.require('Agent').at(apps[APP_NAMES.ARAGON_AGENT].proxyAddress), artifacts.require('Finance').at(apps[APP_NAMES.ARAGON_FINANCE].proxyAddress), artifacts.require('TokenManager').at(apps[APP_NAMES.ARAGON_TOKEN_MANAGER].proxyAddress), - artifacts.require('Voting').at(apps[APP_NAMES.ARAGON_VOTING].proxyAddress) + artifacts.require('Voting').at(apps[APP_NAMES.ARAGON_VOTING].proxyAddress), + artifacts.require('Burner').at(state.burner.address), + artifacts.require('LidoExecutionLayerRewardsVault').at(state.executionLayerRewardsVault["address"]), + artifacts.require('StakingRouter').at(state.stakingRouter.proxy.address), ]) - const burner = await artifacts.require('Burner').at(state.burnerAddress) - const elRewardsVault = await artifacts.require('LidoExecutionLayerRewardsVault').at(state.executionLayerRewardsVault["address"]) - log.splitter() await assertDAOConfig({ ens, dao, daoToken, + daoAragonId: state.daoAragonId, lido, - oracle, + legacyOracle, nopsRegistry, agent, finance, @@ -128,33 +133,35 @@ async function checkDAO({ web3, artifacts }) { voting, burner, elRewardsVault, - daoAragonId: state.daoAragonId, - daoInitialSettings: state.daoInitialSettings + stakingRouter, + state, }) log.splitter() - await assertDaoPermissions( + await assertAragonPermissions( { kernel: dao, lido, + legacyOracle, nopsRegistry, agent, finance, tokenManager, voting, burner, + stakingRouter, }, - state.daoTemplateDeployBlock + state.lidoTemplate.deployBlock ) log.splitter() - const { registryACL } = await assertLidoAPMPermissions({ registry, votingAddress: voting.address }, state.daoTemplateDeployBlock) + const { registryACL } = await assertLidoAPMPermissions({ registry, votingAddress: voting.address }, state.lidoTemplate.deployBlock) log.splitter() - await assertReposPermissions({ registry, registryACL, votingAddress: voting.address }, state.daoTemplateDeployBlock) + await assertReposPermissions({ registry, registryACL, votingAddress: voting.address }, state.lidoTemplate.deployBlock) log.splitter() @@ -166,6 +173,16 @@ async function checkDAO({ web3, artifacts }) { }) log.splitter() + + const permissionsConfig = JSON.parse(fs.readFileSync('./scripts/scratch/checks/scratch-deploy-permissions.json')) + await assertNonAragonPermissions(state, permissionsConfig) + + log.splitter() + + await assertHashConsensusMembers(state.hashConsensusForAccountingOracle.address, []) + await assertHashConsensusMembers(state.hashConsensusForValidatorsExitBusOracle.address, []) + + console.log(`Total gas used during scratch deployment: ${state.initialDeployTotalGasUsed}`) } async function assertLidoAPMPermissions({ registry, votingAddress }, fromBlock = 4532202) { @@ -198,11 +215,12 @@ async function assertReposPermissions({ registry, registryACL, votingAddress }, const newRepoEvents = await registry.getPastEvents('NewRepo', { fromBlock }) for (const evt of newRepoEvents) { - const repo = await Repo.at(evt.args.repo) + const repoAddress = await Repo.at(evt.args.repo) + const repoName = evt.args.name await assertRole( { acl: registryACL, - app: repo, + app: repoAddress, appName: `repo<${evt.args.name}>`, roleName: 'CREATE_VERSION_ROLE', managerAddress: votingAddress, @@ -219,7 +237,7 @@ async function assertDAOConfig({ daoToken, daoAragonId, lido, - oracle, + legacyOracle, nopsRegistry, agent, finance, @@ -227,11 +245,14 @@ async function assertDAOConfig({ voting, burner, elRewardsVault, - daoInitialSettings: settings + stakingRouter, + state, }) { const assertKernel = async (app, appName) => { assert.log(assert.addressEqual, await app.kernel(), dao.address, `${appName}.kernel is ${yl(dao.address)}`) } + const settings = state.daoInitialSettings + const chainSpec = state.chainSpec assert.log( assert.addressEqual, @@ -285,8 +306,7 @@ async function assertDAOConfig({ `voting.voteTime is ${yl(settings.voting.voteDuration)}` ) - // NB: votesLength depends on the presence of insurance and el-rewards modules - // DAO_LIVE || assert.log(assert.bnEqual, await voting.votesLength(), 1, `voting.votesLength is ${yl('1')}`) + DAO_LIVE || assert.log(assert.bnEqual, await voting.votesLength(), 0, `voting.votesLength is ${yl('0')}`) log.splitter() await assertKernel(tokenManager, 'tokenManager') @@ -321,71 +341,59 @@ async function assertDAOConfig({ assert.log(assert.bnEqual, await lido.decimals(), STETH_TOKEN_DECIMALS, `lido.decimals is ${yl(STETH_TOKEN_DECIMALS)}`) - DAO_LIVE || assert.log(assert.bnEqual, await lido.totalSupply(), 0, `lido.totalSupply() is ${yl(0)}`) + // TODO + // DAO_LIVE || assert.log(assert.bnEqual, await lido.totalSupply(), 0, `lido.totalSupply() is ${yl(0)}`) DAO_LIVE || assert.log(assert.equal, await lido.isStopped(), PROTOCOL_PAUSED_AFTER_DEPLOY, `lido.isStopped is ${yl(PROTOCOL_PAUSED_AFTER_DEPLOY)}`) - // TODO: restore the check - // DAO_LIVE || - // assert.log( - // assert.bnEqual, - // await lido.getWithdrawalCredentials(), - // ZERO_WITHDRAWAL_CREDENTIALS, - // `lido.getWithdrawalCredentials() is ${yl(ZERO_WITHDRAWAL_CREDENTIALS)}` - // ) + DAO_LIVE || + assert.log( + assert.hexEqual, + await lido.getWithdrawalCredentials(), + `0x010000000000000000000000${state.withdrawalVault.address.slice(2)}`, + `lido.getWithdrawalCredentials() is ${yl(await lido.getWithdrawalCredentials())}` + ) - // assert.log( - // assert.addressEqual, - // await lido.getDepositContract(), - // settings.beaconSpec.depositContractAddress, - // `lido.getValidatorRegistrationContract() is ${yl(settings.beaconSpec.depositContractAddress)}` - // ) + assert.log( + assert.addressEqual, + await stakingRouter.DEPOSIT_CONTRACT(), + chainSpec.depositContract, + `stakingRouter.DEPOSIT_CONTRACT() is ${yl(stakingRouter.DEPOSIT_CONTRACT())}` + ) - assert.log(assert.addressEqual, await lido.getOracle(), oracle.address, `lido.getOracle() is ${yl(oracle.address)}`) + assert.log(assert.addressEqual, await lido.getOracle(), legacyOracle.address, `lido.getOracle() is ${yl(legacyOracle.address)}`) - // assert.log(assert.addressEqual, await lido.getOperators(), nopsRegistry.address, `lido.getOperators() is ${yl(nopsRegistry.address)}`) assert.log(assert.addressEqual, await lido.getTreasury(), agent.address, `lido.getTreasury() is ${yl(agent.address)}`) - assert.log(assert.addressEqual, await lido.getELRewardsVault(), elRewardsVault.address, - `lido.getELRewardsVault() is ${yl(elRewardsVault.address)}`) - log.splitter() - // await assertKernel(oracle, 'oracle') + await assertKernel(legacyOracle, 'oracle') - assert.log(assert.addressEqual, await oracle.getLido(), lido.address, `oracle.getLido() is ${yl(lido.address)}`) + assert.log(assert.addressEqual, await legacyOracle.getLido(), lido.address, `legacyOracle.getLido() is ${yl(lido.address)}`) - DAO_LIVE || assert.log(assert.isEmpty, await oracle.getOracleMembers(), `oracle.getOracleMembers() is []`) - - const beaconSpec = await oracle.getBeaconSpec() + const legacyOracleBeaconSpec = await legacyOracle.getBeaconSpec() assert.log( assert.bnEqual, - beaconSpec.epochsPerFrame, - settings.beaconSpec.epochsPerFrame, - `oracle.getBeaconSpec().epochsPerFrame is ${yl(settings.beaconSpec.epochsPerFrame)}` + legacyOracleBeaconSpec.slotsPerEpoch, + chainSpec.slotsPerEpoch, + `legacyOracle.getBeaconSpec().slotsPerEpoch is ${yl(chainSpec.slotsPerEpoch)}` ) assert.log( assert.bnEqual, - beaconSpec.slotsPerEpoch, - settings.beaconSpec.slotsPerEpoch, - `oracle.getBeaconSpec().slotsPerEpoch is ${yl(settings.beaconSpec.slotsPerEpoch)}` + legacyOracleBeaconSpec.secondsPerSlot, + chainSpec.secondsPerSlot, + `legacyOracle.getBeaconSpec().secondsPerSlot is ${yl(chainSpec.secondsPerSlot)}` ) assert.log( assert.bnEqual, - beaconSpec.secondsPerSlot, - settings.beaconSpec.secondsPerSlot, - `oracle.getBeaconSpec().secondsPerSlot is ${yl(settings.beaconSpec.secondsPerSlot)}` - ) - assert.log( - assert.bnEqual, - beaconSpec.genesisTime, - settings.beaconSpec.genesisTime, - `oracle.getBeaconSpec().genesisTime is ${yl(settings.beaconSpec.genesisTime)}` + legacyOracleBeaconSpec.genesisTime, + chainSpec.genesisTime, + `legacyOracle.getBeaconSpec().genesisTime is ${yl(chainSpec.genesisTime)}` ) - DAO_LIVE || assert.log(assert.bnEqual, await oracle.getQuorum(), 1, `oracle.getQuorum() is ${yl(1)}`) + // DAO_LIVE || assert.log(assert.bnEqual, await legacyOracle.getQuorum(), 1, `oracle.getQuorum() is ${yl(1)}`) log.splitter() await assertKernel(nopsRegistry, 'nopsRegistry') @@ -401,7 +409,7 @@ async function assertDAOConfig({ ) } -async function assertDaoPermissions({ kernel, lido, oracle, nopsRegistry, agent, finance, tokenManager, voting, burner }, fromBlock = 4532202) { +async function assertAragonPermissions({ kernel, lido, legacyOracle, nopsRegistry, agent, finance, tokenManager, voting, burner, stakingRouter }, fromBlock = 4532202) { const aclAddress = await kernel.acl() const acl = await artifacts.require('ACL').at(aclAddress) const allAclEvents = await acl.getPastEvents('allEvents', { fromBlock }) @@ -554,10 +562,9 @@ async function assertDaoPermissions({ kernel, lido, oracle, nopsRegistry, agent, roleNames: [ 'PAUSE_ROLE', 'RESUME_ROLE', - 'MANAGE_PROTOCOL_CONTRACTS_ROLE', - 'MANAGE_WITHDRAWAL_KEY', 'STAKING_PAUSE_ROLE', - 'STAKING_CONTROL_ROLE' + 'STAKING_CONTROL_ROLE', + 'UNSAFE_CHANGE_DEPOSITED_VALIDATORS_ROLE' ], grantee: voting } @@ -566,8 +573,6 @@ async function assertDaoPermissions({ kernel, lido, oracle, nopsRegistry, agent, log.splitter() - log.splitter() - await assertRoles({ app: nopsRegistry, appName: 'nopsRegistry', @@ -576,18 +581,104 @@ async function assertDaoPermissions({ kernel, lido, oracle, nopsRegistry, agent, { roleNames: [ 'MANAGE_SIGNING_KEYS', - 'ADD_NODE_OPERATOR_ROLE', - 'ACTIVATE_NODE_OPERATOR_ROLE', - 'DEACTIVATE_NODE_OPERATOR_ROLE', - 'SET_NODE_OPERATOR_NAME_ROLE', - 'SET_NODE_OPERATOR_ADDRESS_ROLE', - 'SET_NODE_OPERATOR_LIMIT_ROLE', - 'REPORT_STOPPED_VALIDATORS_ROLE' + 'SET_NODE_OPERATOR_LIMIT_ROLE' ], grantee: voting + }, + { + roleNames: [ + 'STAKING_ROUTER_ROLE', + ], + grantee: stakingRouter } ] }) } +function addressFromStateField(state, fieldOrAddress) { + if (isAddress(fieldOrAddress)) { + return fieldOrAddress + } + + if (state[fieldOrAddress] === undefined) { + throw new Error(`There is no field "${fieldOrAddress}" in state`) + } + + if (state[fieldOrAddress].address) { + return state[fieldOrAddress].address + } else if (state[fieldOrAddress].proxy.address) { + return state[fieldOrAddress].proxy.address + } else { + throw new Error(`Cannot get address for contract field "${fieldOrAddress}" from state file`) + } +} + +function getRoleBytes32ByName(roleName) { + if (roleName === 'DEFAULT_ADMIN_ROLE') { + return '0x0000000000000000000000000000000000000000000000000000000000000000' + } else { + return web3.utils.keccak256(roleName) + } +} + +async function assertHashConsensusMembers(hashConsensusAddress, expectedMembers) { + const hashConsensus = await artifacts.require('HashConsensus').at(hashConsensusAddress) + const actualMembers = (await hashConsensus.getMembers()).addresses + assert.log( + assert.arrayOfAddressesEqual, + actualMembers, + expectedMembers, + `HashConsensus ${hashConsensusAddress} members are expected: [${expectedMembers.toString()}]` + ) +} + +async function assertNonAragonPermissions(state, permissionsConfig) { + for (const [stateField, permissionTypes] of Object.entries(permissionsConfig)) { + for (const [contractType, permissionParams] of Object.entries(permissionTypes)) { + const contract = await artifacts.require(contractType).at(addressFromStateField(state, stateField)) + if (contractType == OSSIFIABLE_PROXY) { + const actualAdmin = await contract.proxy__getAdmin() + assert.log( + assert.addressEqual, + actualAdmin, + addressFromStateField(state, permissionParams.admin), + `${stateField} ${contractType} admin is ${actualAdmin}` + ) + } else if (contractType == ACCESS_CONTROL_ENUMERABLE) { + for (const [role, theHolders] of Object.entries(permissionParams.roles)) { + const roleHash = getRoleBytes32ByName(role) + const actualRoleMemberCount = await contract.getRoleMemberCount(roleHash) + assert.log( + assert.bnEqual, + theHolders.length, + actualRoleMemberCount, + `Contract ${stateField} ${contractType} has correct number of ${role} holders` + ) + for (const holder of theHolders) { + assert.log(assert.equal, true, + await contract.hasRole(roleHash, addressFromStateField(state, holder)), + `Contract ${stateField} ${contractType} has role ${role} holer ${holder}`) + } + } + } else if (permissionParams.specificViews !== undefined) { + for (const [methodName, expectedValue] of Object.entries(permissionParams.specificViews)) { + const actualValue = await contract[methodName].call() + if (isAddress(actualValue)) { + assert.log( + assert.addressEqual, + actualValue, + addressFromStateField(state, expectedValue), + `${stateField} ${contractType} ${methodName} is ${actualValue}` + ) + } else { + throw new Error(`Unsupported view ${methodName} result type of ${expectedValue} of contract ${stateField}`) + } + } + } else { + throw new Error(`Unsupported ACL contract type "${contractType}"`) + } + } + } +} + module.exports = runOrWrapScript(checkDAO, module) diff --git a/scripts/scratch/checks/apps.js b/scripts/scratch/checks/apps.js index 067f6df01..692bedc8b 100644 --- a/scripts/scratch/checks/apps.js +++ b/scripts/scratch/checks/apps.js @@ -68,7 +68,8 @@ async function assertInstalledApps( name: appName, fullName: `${appName}.${lidoApmEnsName}`, id: evt.appId, - proxyAddress + proxyAddress, + initializeData: evt.initializeData || "0x", } } diff --git a/scripts/scratch/checks/scratch-acceptance-test.js b/scripts/scratch/checks/scratch-acceptance-test.js new file mode 100644 index 000000000..f1b6c198c --- /dev/null +++ b/scripts/scratch/checks/scratch-acceptance-test.js @@ -0,0 +1,235 @@ +const { getEventArgument, ZERO_ADDRESS } = require('@aragon/contract-helpers-test') +const runOrWrapScript = require('../../helpers/run-or-wrap-script') +const { log, yl } = require('../../helpers/log') +const { hexConcat, pad, ETH, e27, e18, toBN } = require('../../../test/helpers/utils') +const { reportOracle } = require('../../../test/helpers/oracle') +const { getBalance, advanceChainTime } = require('../../../test/helpers/blockchain') +const { assertRequiredNetworkState, readStateFile } = require('../../helpers/persisted-network-state') +const { assert } = require('../../../test/helpers/assert') + + +const REQUIRED_NET_STATE = [ + 'ensAddress', + 'lidoApmEnsName', + 'daoAragonId', + 'vestingParams', + 'daoInitialSettings', + 'lidoTemplate' +] + +const UNLIMITED = 1000000000 +const CURATED_MODULE_ID = 1 +const CALLDATA = '0x0' +const MAX_DEPOSITS = 150 +const ADDRESS_1 = '0x0000000000000000000000000000000000000001' +const ADDRESS_2 = '0x0000000000000000000000000000000000000002' + +const MANAGE_MEMBERS_AND_QUORUM_ROLE = web3.utils.keccak256('MANAGE_MEMBERS_AND_QUORUM_ROLE') + +if (!process.env.HARDHAT_FORKING_URL) { + console.error('Env variable HARDHAT_FORKING_URL must be set to run fork acceptance tests') + process.exit(1); +} +if (!process.env.NETWORK_STATE_FILE) { + console.error('Env variable NETWORK_STATE_FILE must be set to run fork acceptance tests') + process.exit(1); +} +const NETWORK_STATE_FILE = process.env.NETWORK_STATE_FILE + + + +async function loadDeployedProtocol(state) { + return { + stakingRouter: await artifacts.require('StakingRouter').at(state.stakingRouter.proxy.address), + lido: await artifacts.require('Lido').at(state['app:lido'].proxy.address), + voting: await artifacts.require('Voting').at(state['app:aragon-voting'].proxy.address), + agent: await artifacts.require('Agent').at(state['app:aragon-agent'].proxy.address), + nodeOperatorsRegistry: await artifacts.require('NodeOperatorsRegistry').at(state['app:node-operators-registry'].proxy.address), + depositSecurityModule: await artifacts.require('DepositSecurityModule').at(state.depositSecurityModule.address), + accountingOracle: await artifacts.require('AccountingOracle').at(state.accountingOracle.proxy.address), + hashConsensusForAO: await artifacts.require('HashConsensus').at(state.hashConsensusForAccountingOracle.address), + elRewardsVault: await artifacts.require('LidoExecutionLayerRewardsVault').at(state.executionLayerRewardsVault.address), + withdrawalQueue: await artifacts.require('WithdrawalQueueERC721').at(state.withdrawalQueueERC721.proxy.address), + ldo: await artifacts.require('MiniMeToken').at(state.ldo.address), + } +} + + +async function checkLDOCanBeTransferred(ldo, state) { + const ldoHolder = Object.keys(state.vestingParams.holders)[0] + await ethers.provider.send('hardhat_impersonateAccount', [ldoHolder]) + + await ldo.transfer(ADDRESS_1, e18(1), { from: ldoHolder }) + assert.equals(await ldo.balanceOf(ADDRESS_1), e18(1)) + + log.success("Transferred LDO") +} + + +async function prepareProtocolForSubmitDepositReportWithdrawalFlow(protocol, state, oracleMember1, oracleMember2) { + const { + lido, + voting, + agent, + nodeOperatorsRegistry, + depositSecurityModule, + hashConsensusForAO, + withdrawalQueue, + } = protocol + + await ethers.provider.send('hardhat_impersonateAccount', [voting.address]) + await ethers.provider.send('hardhat_impersonateAccount', [depositSecurityModule.address]) + await ethers.provider.send('hardhat_impersonateAccount', [agent.address]) + + await lido.resume({ from: voting.address }) + + await withdrawalQueue.grantRole(await withdrawalQueue.RESUME_ROLE(), agent.address, { from: agent.address }) + await withdrawalQueue.resume({ from: agent.address }) + await withdrawalQueue.renounceRole(await withdrawalQueue.RESUME_ROLE(), agent.address, { from: agent.address }) + + await nodeOperatorsRegistry.addNodeOperator('1', ADDRESS_1, { from: voting.address }) + await nodeOperatorsRegistry.addNodeOperator('2', ADDRESS_2, { from: voting.address }) + + await nodeOperatorsRegistry.addSigningKeys(0, 1, pad('0x010203', 48), pad('0x01', 96), { from: voting.address }) + await nodeOperatorsRegistry.addSigningKeys( + 0, + 3, + hexConcat(pad('0x010204', 48), pad('0x010205', 48), pad('0x010206', 48)), + hexConcat(pad('0x01', 96), pad('0x01', 96), pad('0x01', 96)), + { from: voting.address } + ) + + await nodeOperatorsRegistry.setNodeOperatorStakingLimit(0, UNLIMITED, { from: voting.address }) + await nodeOperatorsRegistry.setNodeOperatorStakingLimit(1, UNLIMITED, { from: voting.address }) + + const quorum = 2 + await hashConsensusForAO.grantRole(MANAGE_MEMBERS_AND_QUORUM_ROLE, agent.address, { from: agent.address }) + await hashConsensusForAO.addMember(oracleMember1.address, quorum, { from: agent.address }) + await hashConsensusForAO.addMember(oracleMember2.address, quorum, { from: agent.address }) + await hashConsensusForAO.renounceRole(MANAGE_MEMBERS_AND_QUORUM_ROLE, agent.address, { from: agent.address }) + + log.success('Protocol prepared for submit-deposit-report-withdraw flow') +} + +async function checkSubmitDepositReportWithdrawal(protocol, state, user1, user2) { + const { + lido, + agent, + depositSecurityModule, + accountingOracle, + hashConsensusForAO, + elRewardsVault, + withdrawalQueue, + } = protocol + + + const initialLidoBalance = await getBalance(lido.address) + const chainSpec = state.chainSpec + + await user1.sendTransaction({ to: lido.address, value: ETH(34) }) + await user2.sendTransaction({ to: elRewardsVault.address, value: ETH(1) }) + log.success('Users submitted ether') + + assert.equals(await lido.balanceOf(user1.address), ETH(34)) + assert.equals(await lido.getTotalPooledEther(), initialLidoBalance + BigInt(ETH(34))) + assert.equals(await lido.getBufferedEther(), initialLidoBalance + BigInt(ETH(34))) + + await lido.deposit(MAX_DEPOSITS, CURATED_MODULE_ID, CALLDATA, { from: depositSecurityModule.address }) + log.success('Ether deposited') + + + assert.equals((await lido.getBeaconStat()).depositedValidators, 1) + + const latestBlockTimestamp = (await ethers.provider.getBlock('latest')).timestamp + const initialEpoch = Math.floor((latestBlockTimestamp - chainSpec.genesisTime) + / (chainSpec.slotsPerEpoch * chainSpec.secondsPerSlot)) + + await hashConsensusForAO.updateInitialEpoch(initialEpoch, { from: agent.address }) + + + const elRewardsVaultBalance = await web3.eth.getBalance(elRewardsVault.address) + + const withdrawalAmount = ETH(1) + + await lido.approve(withdrawalQueue.address, withdrawalAmount, { from: user1.address }) + const receipt = await withdrawalQueue.requestWithdrawals([withdrawalAmount], user1.address, { from: user1.address }) + const requestId = getEventArgument(receipt, 'WithdrawalRequested', 'requestId') + + log.success('Withdrawal request made') + + const epochsPerFrame = +(await hashConsensusForAO.getFrameConfig()).epochsPerFrame + const initialEpochTimestamp = chainSpec.genesisTime + initialEpoch * chainSpec.slotsPerEpoch * chainSpec.secondsPerSlot + + // skip two reports to be sure about REQUEST_TIMESTAMP_MARGIN + const nextReportEpochTimestamp = initialEpochTimestamp + 2 * epochsPerFrame * chainSpec.slotsPerEpoch * chainSpec.secondsPerSlot + + const timeToWaitTillReportWindow = nextReportEpochTimestamp - latestBlockTimestamp + chainSpec.secondsPerSlot + + await advanceChainTime(timeToWaitTillReportWindow) + + let stat = await lido.getBeaconStat() + const clBalance = toBN(stat.depositedValidators).mul(toBN(e18(32))) + + + const { refSlot } = await hashConsensusForAO.getCurrentFrame() + const reportTimestamp = +chainSpec.genesisTime + (+refSlot) * (+chainSpec.secondsPerSlot) + const timeElapsed = +(nextReportEpochTimestamp - initialEpochTimestamp) + + const withdrawalFinalizationBatches = [1] + + // Performing dry-run to estimate simulated share rate + const [postTotalPooledEther, postTotalShares, withdrawals, elRewards] = await lido.handleOracleReport.call( + reportTimestamp, + timeElapsed, + stat.depositedValidators, + clBalance, + 0 /* withdrawals vault balance */, + elRewardsVaultBalance, + 0 /* shares requested to burn */, + [] /* withdrawal finalization batches */, + 0 /* simulated share rate */, + { from: accountingOracle.address } + ) + + log.success('Oracle report simulated') + + const simulatedShareRate = postTotalPooledEther.mul(toBN(e27(1))).div(postTotalShares) + + await reportOracle(hashConsensusForAO, accountingOracle, { + refSlot, + numValidators: stat.depositedValidators, + clBalance, + elRewardsVaultBalance, + withdrawalFinalizationBatches, + simulatedShareRate, + }) + + log.success('Oracle report submitted') + + await withdrawalQueue.claimWithdrawalsTo([requestId], [requestId], user1.address, { from: user1.address }) + + log.success('Withdrawal claimed successfully') +} + +async function checkMainProtocolFlows({ web3 }) { + const netId = await web3.eth.net.getId() + + log.splitter() + log(`Network ID: ${yl(netId)}`) + + const state = readStateFile(NETWORK_STATE_FILE) + assertRequiredNetworkState(state, REQUIRED_NET_STATE) + + log.splitter() + + const protocol = await loadDeployedProtocol(state) + const [user1, user2, oracleMember1, oracleMember2] = await ethers.getSigners() + + await checkLDOCanBeTransferred(protocol.ldo, state) + + await prepareProtocolForSubmitDepositReportWithdrawalFlow(protocol, state, oracleMember1, oracleMember2) + await checkSubmitDepositReportWithdrawal(protocol, state, user1, user2) +} + + +module.exports = runOrWrapScript(checkMainProtocolFlows, module) diff --git a/scripts/scratch/checks/scratch-deploy-permissions.json b/scripts/scratch/checks/scratch-deploy-permissions.json new file mode 100644 index 000000000..96c2f032c --- /dev/null +++ b/scripts/scratch/checks/scratch-deploy-permissions.json @@ -0,0 +1,132 @@ +{ + "lidoLocator": { + "OssifiableProxy": { + "admin": "app:aragon-agent" + } + }, + "burner": { + "AccessControlEnumerable": { + "roles": { + "DEFAULT_ADMIN_ROLE": ["app:aragon-agent"], + "REQUEST_BURN_MY_STETH_ROLE": [], + "REQUEST_BURN_SHARES_ROLE": ["app:lido", "app:node-operators-registry"] + } + } + }, + "stakingRouter": { + "AccessControlEnumerable": { + "roles": { + "DEFAULT_ADMIN_ROLE": ["app:aragon-agent"], + "MANAGE_WITHDRAWAL_CREDENTIALS_ROLE": [], + "STAKING_MODULE_PAUSE_ROLE": ["depositSecurityModule"], + "STAKING_MODULE_RESUME_ROLE": ["depositSecurityModule"], + "STAKING_MODULE_MANAGE_ROLE": [], + "REPORT_EXITED_VALIDATORS_ROLE": ["accountingOracle"], + "UNSAFE_SET_EXITED_VALIDATORS_ROLE": [], + "REPORT_REWARDS_MINTED_ROLE": ["app:lido"] + } + }, + "OssifiableProxy": { + "admin": "app:aragon-agent" + } + }, + "withdrawalQueueERC721": { + "AccessControlEnumerable": { + "roles": { + "DEFAULT_ADMIN_ROLE": ["app:aragon-agent"], + "PAUSE_ROLE": ["gateSeal"], + "RESUME_ROLE": [], + "FINALIZE_ROLE": ["app:lido"], + "ORACLE_ROLE": ["accountingOracle"], + "MANAGE_TOKEN_URI_ROLE": [] + } + }, + "OssifiableProxy": { + "admin": "app:aragon-agent" + } + }, + "accountingOracle": { + "AccessControlEnumerable": { + "roles": { + "DEFAULT_ADMIN_ROLE": ["app:aragon-agent"], + "SUBMIT_DATA_ROLE": [], + "MANAGE_CONSENSUS_CONTRACT_ROLE": [], + "MANAGE_CONSENSUS_VERSION_ROLE": [] + } + }, + "OssifiableProxy": { + "admin": "app:aragon-agent" + } + }, + "validatorsExitBusOracle": { + "AccessControlEnumerable": { + "roles": { + "DEFAULT_ADMIN_ROLE": ["app:aragon-agent"], + "SUBMIT_DATA_ROLE": [], + "PAUSE_ROLE": ["gateSeal"], + "RESUME_ROLE": [], + "MANAGE_CONSENSUS_CONTRACT_ROLE": [], + "MANAGE_CONSENSUS_VERSION_ROLE": [] + } + }, + "OssifiableProxy": { + "admin": "app:aragon-agent" + } + }, + "hashConsensusForAccountingOracle": { + "AccessControlEnumerable": { + "roles": { + "DEFAULT_ADMIN_ROLE": ["app:aragon-agent"], + "MANAGE_MEMBERS_AND_QUORUM_ROLE": [], + "DISABLE_CONSENSUS_ROLE": [], + "MANAGE_FRAME_CONFIG_ROLE": [], + "MANAGE_FAST_LANE_CONFIG_ROLE": [], + "MANAGE_REPORT_PROCESSOR_ROLE": [] + } + } + }, + "hashConsensusForValidatorsExitBusOracle": { + "AccessControlEnumerable": { + "roles": { + "DEFAULT_ADMIN_ROLE": ["app:aragon-agent"], + "MANAGE_MEMBERS_AND_QUORUM_ROLE": [], + "DISABLE_CONSENSUS_ROLE": [], + "MANAGE_FRAME_CONFIG_ROLE": [], + "MANAGE_FAST_LANE_CONFIG_ROLE": [], + "MANAGE_REPORT_PROCESSOR_ROLE": [] + } + } + }, + "oracleReportSanityChecker": { + "AccessControlEnumerable": { + "roles": { + "DEFAULT_ADMIN_ROLE": ["app:aragon-agent"], + "ALL_LIMITS_MANAGER_ROLE": [], + "CHURN_VALIDATORS_PER_DAY_LIMIT_MANAGER_ROLE": [], + "ONE_OFF_CL_BALANCE_DECREASE_LIMIT_MANAGER_ROLE": [], + "ANNUAL_BALANCE_INCREASE_LIMIT_MANAGER_ROLE": [], + "SHARE_RATE_DEVIATION_LIMIT_MANAGER_ROLE": [], + "MAX_VALIDATOR_EXIT_REQUESTS_PER_REPORT_ROLE": [], + "MAX_ACCOUNTING_EXTRA_DATA_LIST_ITEMS_COUNT_ROLE": [], + "MAX_NODE_OPERATORS_PER_EXTRA_DATA_ITEM_COUNT_ROLE": [], + "REQUEST_TIMESTAMP_MARGIN_MANAGER_ROLE": [], + "MAX_POSITIVE_TOKEN_REBASE_MANAGER_ROLE": [] + } + } + }, + "depositSecurityModule": { + "DepositSecurityModule": { + "specificViews": { + "getOwner": "app:aragon-agent" + } + } + }, + "oracleDaemonConfig": { + "AccessControlEnumerable": { + "roles": { + "DEFAULT_ADMIN_ROLE": ["app:aragon-agent"], + "CONFIG_MANAGER_ROLE": [] + } + } + } +} \ No newline at end of file diff --git a/scripts/scratch/dao-deploy.sh b/scripts/scratch/dao-deploy.sh new file mode 100755 index 000000000..4b3b5bf44 --- /dev/null +++ b/scripts/scratch/dao-deploy.sh @@ -0,0 +1,94 @@ +#!/bin/bash +set -e +u +set -o pipefail + +ARAGON_APPS_REPO_REF=import-shared-minime + +if [[ -z "${DEPLOYER}" ]]; then + echo "Env variable DEPLOYER must be set" + exit 1 +fi +echo "DEPLOYER is $DEPLOYER" + +if [[ -z "${NETWORK}" ]]; then + echo "Env variable NETWORK must be set" + exit 1 +fi +echo "NETWORK is $NETWORK" + +function msg() { + MSG=$1 + if [ ! -z "$MSG" ]; then + echo ">>> =============================" + echo ">>> $MSG" + echo ">>> =============================" + fi +} + + +# yarn install --immutable +yarn compile + +rm -f ${NETWORK_STATE_FILE} +cp ${NETWORK_STATE_DEFAULTS_FILE} ${NETWORK_STATE_FILE} + + +# Fill in deployer, chainId, etc from env to the deploy artifact +yarn hardhat --network $NETWORK run ./scripts/scratch/00-populate-deploy-artifact-from-env.js --no-compile + +# It does not deploy DepositContract if it is specified in deployed-${NETWORK}-defaults.json +yarn hardhat --network $NETWORK run ./scripts/scratch/01-deploy-deposit-contract.js --no-compile +msg "Deposit contract deployed or is specified." + +yarn hardhat --network $NETWORK run --no-compile ./scripts/scratch/02-deploy-aragon-env.js +msg "Aragon ENV deployed." + +yarn hardhat run --no-compile ./scripts/scratch/03-deploy-aragon-std-apps.js --network $NETWORK +msg "Aragon STD apps deployed." + +yarn hardhat --network $NETWORK run ./scripts/scratch/04-deploy-lido-template-and-bases.js --no-compile +yarn hardhat --network $NETWORK run ./scripts/scratch/05-obtain-deployed-instances.js --no-compile +msg "Apps instances deployed" + +yarn hardhat --network $NETWORK run ./scripts/scratch/06-register-ens-domain.js --no-compile +msg "ENS registered" + +yarn hardhat --network $NETWORK run ./scripts/scratch/07-deploy-apm.js --no-compile +yarn hardhat --network $NETWORK run ./scripts/scratch/08-obtain-deployed-apm.js --no-compile +msg "APM deployed" + +yarn hardhat --network $NETWORK run ./scripts/scratch/09-create-app-repos.js --no-compile +msg "App repos created" + +yarn hardhat --network $NETWORK run ./scripts/scratch/10-deploy-dao.js --no-compile + +yarn hardhat --network $NETWORK run ./scripts/scratch/11-obtain-deployed-dao.js --no-compile +msg "DAO deploy started" + +# Do it at the end, because might need the contracts initialized +yarn hardhat --network $NETWORK run ./scripts/scratch/12-issue-tokens.js --no-compile +msg "Tokens issued" + +# Deploy the contracts before finalizing DAO, because the template might set permissions on some of them +yarn hardhat --network $NETWORK run ./scripts/scratch/13-deploy-non-aragon-contracts.js --no-compile +msg "Non-aragon contracts deployed" + +yarn hardhat --network $NETWORK run ./scripts/scratch/14-gate-seal.js --no-compile + +yarn hardhat --network $NETWORK run ./scripts/scratch/15-finalize-dao.js --no-compile +msg "DAO deploy finalized" + +yarn hardhat --network $NETWORK run ./scripts/scratch/16-initialize-non-aragon-contracts.js --no-compile +msg "Non-aragon contracts initialized" + +yarn hardhat --network $NETWORK run ./scripts/scratch/17-grant-roles.js --no-compile +msg "Roles granted" + +yarn hardhat --network $NETWORK run ./scripts/scratch/18-plug-curated-staking-module.js --no-compile +msg "Plugged NodeOperatorsRegistry as Curated staking module" + +yarn hardhat --network $NETWORK run ./scripts/scratch/19-transfer-roles.js --no-compile +msg "Role admin permissions transferred to Agent" + +yarn hardhat --network $NETWORK run ./scripts/scratch/20-check-dao.js --no-compile +msg "The deployed protocol state checked" diff --git a/scripts/scratch/dao-holesky-deploy.sh b/scripts/scratch/dao-holesky-deploy.sh new file mode 100755 index 000000000..63b823b17 --- /dev/null +++ b/scripts/scratch/dao-holesky-deploy.sh @@ -0,0 +1,29 @@ +#!/bin/bash +set -e +u +set -o pipefail + +if [[ -z "$DEPLOYER" ]]; then + echo "Must set DEPLOYER env variable" 1>&2 + exit 1 +fi +if [[ -z "$RPC_URL" ]]; then + echo "Must set RPC_URL env variable" 1>&2 + exit 1 +fi +if [[ -z "$GATE_SEAL_FACTORY" ]]; then + echo "Must set GATE_SEAL_FACTORY env variable" 1>&2 + exit 1 +fi + +export NETWORK=holesky +export NETWORK_STATE_FILE="deployed-${NETWORK}.json" +export NETWORK_STATE_DEFAULTS_FILE="deployed-testnet-defaults.json" + +# Holesky params: https://github.com/eth-clients/holesky/blob/main/README.md +export GENESIS_TIME=1695902400 +export DEPOSIT_CONTRACT=0x4242424242424242424242424242424242424242 + +export GAS_PRIORITY_FEE="${GAS_PRIORITY_FEE:=1}" +export GAS_MAX_FEE="${GAS_MAX_FEE:=100}" + +bash scripts/scratch/dao-deploy.sh diff --git a/scripts/scratch/dao-local-deploy.sh b/scripts/scratch/dao-local-deploy.sh new file mode 100755 index 000000000..b4f0602f1 --- /dev/null +++ b/scripts/scratch/dao-local-deploy.sh @@ -0,0 +1,18 @@ +#!/bin/bash +set -e +u +set -o pipefail + +# +export NETWORK=local +export RPC_URL="http://127.0.0.1:8545" +export GATE_SEAL_FACTORY=0x0000000000000000000000000000000000000000 +export GENESIS_TIME=1639659600 # just some time +# +export DEPLOYER=0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266 +export GAS_PRIORITY_FEE=1 +export GAS_MAX_FEE=100 +# +export NETWORK_STATE_FILE="deployed-${NETWORK}.json" +export NETWORK_STATE_DEFAULTS_FILE="deployed-testnet-defaults.json" + +bash scripts/scratch/dao-deploy.sh diff --git a/scripts/scratch/deploy-aragon-std-apps.js b/scripts/scratch/deploy-aragon-std-apps.js deleted file mode 100644 index b836519d8..000000000 --- a/scripts/scratch/deploy-aragon-std-apps.js +++ /dev/null @@ -1,175 +0,0 @@ -const path = require('path') -const fs = require('fs') -const namehash = require('eth-ens-namehash').hash - -const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log, logSplitter, logWideSplitter, logHeader, yl, gr } = require('../helpers/log') -const { readNetworkState, persistNetworkState, updateNetworkState } = require('../helpers/persisted-network-state') -const { readJSON, directoryExists } = require('../helpers/fs') -const { execLive } = require('../helpers/exec') -const { filterObject } = require('../helpers/collections') -const { readAppName } = require('../helpers/aragon') -const { gitCloneRepo } = require('../helpers/git') - -const { uploadDirToIpfs } = require('@aragon/buidler-aragon/dist/src/utils/ipfs') -const { toContentUri } = require('@aragon/buidler-aragon/dist/src/utils/apm/utils') - -const APPS = ['agent', 'finance', 'token-manager', 'vault', 'voting'] - -const NETWORK_STATE_FILE = process.env.NETWORK_STATE_FILE || 'deployed.json' -const ARAGON_APPS_REPO = process.env.ARAGON_APPS_REPO || 'https://github.com/lidofinance/aragon-apps.git' -const ARAGON_APPS_REPO_REF = process.env.ARAGON_APPS_REPO_REF || 'shapella-update' -const RELEASE_TYPE = 'major' - -const SKIP_APPS_LONG_BUILD_STEPS = process.env.SKIP_APPS_LONG_BUILD_STEPS || false - -function writeNetworkStateFile(fileName, state) { - const data = JSON.stringify(state, null, ' ') - fs.writeFileSync(fileName, data + '\n', 'utf8') -} - -async function deployAragonStdApps({ - web3, - artifacts, - networkStateFile = NETWORK_STATE_FILE, - aragonAppsRepo = ARAGON_APPS_REPO, - aragonAppsRepoRef = ARAGON_APPS_REPO_REF, - releaseType = RELEASE_TYPE -}) { - const hardhatConfig = path.resolve(hardhatArguments.config || 'hardhat.config.js') - const netId = await web3.eth.net.getId() - const netName = network.name - - logWideSplitter() - log(`Network ID: ${yl(netId)}`) - - const netState = readNetworkState(network.name, netId) - if (!netState.ensAddress) { - throw new Error(`ensAddress for network ${netId} is missing from network state file ${networkStateFile}`) - } - - if (!network.config.ensAddress) { - throw new Error(`ensAddress is not defined for network ${netName} in Hardhat config file ${hardhatConfig}`) - } - - if (network.config.ensAddress.toLowerCase() !== netState.ensAddress.toLowerCase()) { - throw new Error( - `ensAddress for network ${netId} is different in Hardhat config file ${hardhatConfig} ` + `and network state file ${networkStateFile}` - ) - } - - logHeader(`Checking out aragon-apps repo...`) - const appsRepoPath = './aragon-apps' - await gitCloneRepo(appsRepoPath, aragonAppsRepo, aragonAppsRepoRef) - - // prevent Hardhat from passing the config to subprocesses - const env = filterObject(process.env, (key) => key.substr(0, 8) !== 'HARDHAT_') - - for (const appName of APPS) { - const app = await publishApp(appName, appsRepoPath, hardhatConfig, env, netName, releaseType, netState) - persistNetworkState(network.name, netId, netState, { - [`app:${app.name}`]: { - ...netState[`app:${app.name}`], - ...app - } - }) - } -} - -async function publishApp(appName, appsRepoPath, hardhatConfig, env, netName, releaseType, netState) { - logHeader(`Publishing new ${releaseType} release of app '${appName}'`) - - let result = {} - const appRootPath = path.resolve(appsRepoPath, 'apps', appName) - const appFullName = await readAppName(appRootPath, netName) - const appId = namehash(appFullName) - - log(`App name: ${yl(appFullName)}`) - log(`App ID: ${yl(appId)}`) - logSplitter() - - const appFrontendPath = path.join(appRootPath, 'app') - const hasFrontend = await directoryExists(appFrontendPath) - - logSplitter(`Change registry app ${appName}`) - const arappFile = `${appRootPath}/arapp.json` - const arapp = await readJSON(arappFile) - if (!arapp.environments[network.name]) { - arapp.environments[network.name] = { ...arapp.environments.default } - arapp.environments[network.name].registry = network.config.ensAddress - } - - logSplitter(`Write state app ${appName}`) - writeNetworkStateFile(arappFile, arapp) - - if (hasFrontend) { - if (!SKIP_APPS_LONG_BUILD_STEPS) { - logSplitter(`Installing frontend deps for app ${appName}`) - await execLive('yarn', { - cwd: appFrontendPath - }) - logSplitter() - - logSplitter(`Build app ${appName}`) - await execLive('yarn', { - args: ['build'], - cwd: appFrontendPath - }) - logSplitter() - } else { - logSplitter(`Dependencies installing and app build steps are skipped for ${appName}`) - } - } else { - log(`The app has no frontend`) - } - - const childEnv = { - ...env, - STD_APPS_DEPLOY: '1', - APP_FRONTEND_PATH: `aragon-apps/apps/${appName}/app`, - APP_FRONTEND_DIST_PATH: `aragon-apps/apps/${appName}/dist` - } - - // if (hasFrontend) { - // const distPath = path.join(appRootPath, 'dist') - - // // Upload release directory to IPFS - // log('Uploading release assets to IPFS...') - - // const contentHash = await uploadDirToIpfs({ - // dirPath: distPath, - // apiUrl: netState.ipfsAPI - // }) - // log(`Release assets uploaded to IPFS: ${yl(contentHash)}`) - - // result.ipfsCid = contentHash - // result.contentURI = toContentUri('ipfs', contentHash) - // } - - await execLive('hardhat', { - args: [ - 'publish', - 'major', - '--config', - hardhatConfig, - '--network', - netName, - '--skip-validation', - // '--skip-app-build', - // workaround: force to read URL from Hardhat config - '--ipfs-api-url', - '' - ], - cwd: appRootPath, - env: childEnv - }) - - return { - ...result, - fullName: appFullName, - name: appName, - id: appId - } -} - -module.exports = runOrWrapScript(deployAragonStdApps, module) diff --git a/scripts/scratch/deploy-lido-apm-and-template.js b/scripts/scratch/deploy-lido-apm-and-template.js deleted file mode 100644 index 81d494f9e..000000000 --- a/scripts/scratch/deploy-lido-apm-and-template.js +++ /dev/null @@ -1,147 +0,0 @@ -const chalk = require('chalk') -const namehash = require('eth-ens-namehash').hash - -const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log, logSplitter, logWideSplitter, logHeader, logTx } = require('../helpers/log') -const { deploy, withArgs } = require('../helpers/deploy') -const { readNetworkState, persistNetworkState, updateNetworkState } = require('../helpers/persisted-network-state') - -const { deployAPM, resolveLatestVersion } = require('./components/apm') -const { getENSNodeOwner } = require('./components/ens') - -const LIDO_ENS_LABEL = process.env.LIDO_ENS_LABEL || 'lidopm' -const DAO_TEMPLATE_ENS_LABEL = process.env.DAO_TEMPLATE_ENS_LABEL || 'template' -const NETWORK_STATE_FILE = process.env.NETWORK_STATE_FILE || 'deployed.json' - -const REQUIRED_NET_STATE = [ - 'owner', - 'multisigAddress', - 'ensAddress', - 'apmRegistryFactoryAddress', - 'daoFactoryAddress', - 'miniMeTokenFactoryAddress' -] - -async function deployApmAndTemplate({ - web3, - artifacts, - networkStateFile = NETWORK_STATE_FILE, - defaultLidoEnsLabelName = LIDO_ENS_LABEL, - defaultDaoTemplateEnsLabel = DAO_TEMPLATE_ENS_LABEL -}) { - const netId = await web3.eth.net.getId() - - logWideSplitter() - log(`Network ID: ${chalk.yellow(netId)}`) - - const state = readNetworkState(network.name, netId) - - const missingState = REQUIRED_NET_STATE.filter((key) => !state[key]) - if (missingState.length) { - const missingDesc = missingState.join(', ') - throw new Error(`missing following fields from network state file, make sure you've run previous deployment steps: ${missingDesc}`) - } - - log(`Owner: ${chalk.yellow(state.owner)}`) - - const ens = await artifacts.require('ENS').at(state.ensAddress) - log(`Using ENS: ${chalk.yellow(ens.address)}`) - - if (!state.lidoEnsLabelName) { - state.lidoEnsLabelName = defaultLidoEnsLabelName - log(`Using default Lido ENS label: ${chalk.yellow(state.lidoEnsLabelName)}`) - } - - if (!state.daoTemplateEnsLabel) { - state.daoTemplateEnsLabel = defaultDaoTemplateEnsLabel - log(`Using default DAO template ENS label: ${chalk.yellow(state.daoTemplateEnsLabel)}`) - } - - logHeader(`Lido APM`) - const apmRegistryFactory = await artifacts.require('APMRegistryFactory').at(state.apmRegistryFactoryAddress) - const apmResults = await deployAPM({ - web3, - artifacts, - ens, - apmRegistryFactory, - owner: state.owner, - labelName: state.lidoEnsLabelName, - apmRegistryAddress: state.lidoApmAddress - }) - updateNetworkState(state, { - lidoApmRegistry: apmResults.apmRegistry, - lidoEnsNodeName: apmResults.ensNodeName, - lidoEnsNode: apmResults.ensNode - }) - persistNetworkState(network.name, netId, state) - - logHeader(`DAO template`) - const daoTemplateResults = await deployDaoTemplate({ - artifacts, - ens, - owner: state.owner, - lidoEnsNodeName: state.lidoEnsNodeName, - lidoApmAddress: state.lidoApmAddress, - daoFactoryAddress: state.daoFactoryAddress, - miniMeTokenFactoryAddress: state.miniMeTokenFactoryAddress, - daoTemplateEnsLabel: state.daoTemplateEnsLabel, - daoTemplateAddress: state.daoTemplateAddress - }) - updateNetworkState(state, daoTemplateResults) - persistNetworkState(network.name, netId, state) -} - -async function deployDaoTemplate({ - artifacts, - owner, - ens, - lidoApmAddress, - daoFactoryAddress, - miniMeTokenFactoryAddress, - lidoEnsNodeName, - daoTemplateEnsLabel, - daoTemplateAddress -}) { - if (daoTemplateAddress) { - log(`Using DAO template: ${chalk.yellow(daoTemplateAddress)}`) - const daoTemplate = await artifacts.require('LidoTemplate').at(daoTemplateAddress) - return { daoTemplate } - } - - const daoTemplateNodeName = `${daoTemplateEnsLabel}.${lidoEnsNodeName}` - const daoTemplateNode = namehash(daoTemplateNodeName) - log(`DAO template name: ${chalk.yellow(daoTemplateNodeName)} (${daoTemplateNode})`) - - const latestDaoTemplateVersion = await resolveLatestVersion(daoTemplateNode, ens, artifacts) - if (latestDaoTemplateVersion) { - log(`Using DAO template resolved from ENS: ${chalk.yellow(latestDaoTemplateVersion.contractAddress)}`) - const daoTemplate = await artifacts.require('LidoTemplate').at(latestDaoTemplateVersion.contractAddress) - return { daoTemplate, daoTemplateNodeName, daoTemplateNode } - } - - log(`Using Lido APM registry: ${chalk.yellow(lidoApmAddress)}`) - const lidoApmRegistry = await artifacts.require('APMRegistry').at(lidoApmAddress) - - const aragonIdAddress = await getENSNodeOwner(ens, namehash('aragonid.eth')) - if (aragonIdAddress) { - log(`Using AragonID: ${chalk.yellow(aragonIdAddress)}`) - } else { - throw new Error(`failed to resolve AragonID (aragonid.eth)`) - } - - const daoTemplate = await deploy( - 'LidoTemplate', - artifacts, - withArgs(owner, daoFactoryAddress, ens.address, miniMeTokenFactoryAddress, aragonIdAddress, { from: owner, gas: 6000000 }) - ) - logSplitter() - - await logTx( - `Registering package for DAO template as '${daoTemplateNodeName}'...`, - lidoApmRegistry.newRepoWithVersion(daoTemplateEnsLabel, owner, [1, 0, 0], daoTemplate.address, '0x0', { from: owner }) - ) - - return { daoTemplate, daoTemplateNodeName, daoTemplateNode } -} - -module.exports = runOrWrapScript(deployApmAndTemplate, module) diff --git a/scripts/scratch/deploy-lido-apps.js b/scripts/scratch/deploy-lido-apps.js deleted file mode 100644 index 9ca02de23..000000000 --- a/scripts/scratch/deploy-lido-apps.js +++ /dev/null @@ -1,136 +0,0 @@ -const fs = require('fs') -const path = require('path') -const namehash = require('eth-ens-namehash').hash - -const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log, logSplitter, logWideSplitter, logHeader, yl } = require('../helpers/log') -const { readNetworkState, persistNetworkState, updateNetworkState } = require('../helpers/persisted-network-state') -const { execLive } = require('../helpers/exec') -const { filterObject } = require('../helpers/collections') -const { readAppName } = require('../helpers/aragon') -const { readJSON } = require('../helpers/fs') - - -const NETWORK_STATE_FILE = process.env.NETWORK_STATE_FILE || 'deployed.json' -const RELEASE_TYPE = process.env.RELEASE_TYPE || 'major' -const APPS = process.env.APPS || '*' -const APPS_DIR_PATH = process.env.APPS_DIR_PATH || path.resolve(__dirname, '..', 'apps') -const SKIP_APPS_LONG_BUILD_STEPS = process.env.SKIP_APPS_LONG_BUILD_STEPS || false - -function writeNetworkStateFile(fileName, state) { - const data = JSON.stringify(state, null, ' ') - fs.writeFileSync(fileName, data + '\n', 'utf8') -} - -async function deployLidoApps({ - web3, - artifacts, - networkStateFile = NETWORK_STATE_FILE, - appsDirPath = APPS_DIR_PATH, - appNames = APPS, - releaseType = RELEASE_TYPE -}) { - const hardhatConfig = path.resolve(hardhatArguments.config || 'hardhat.config.js') - const netId = await web3.eth.net.getId() - const netName = network.name - - logWideSplitter() - log(`Network ID: ${yl(netId)}`) - - const netState = readNetworkState(network.name, netId) - if (!netState.ensAddress) { - throw new Error(`ensAddress for network ${netId} is missing from network state file ${networkStateFile}`) - } - - if (!network.config.ensAddress) { - throw new Error(`ensAddress is not defined for network ${netName} in Hardhat config file ${hardhatConfig}`) - } - - if (network.config.ensAddress.toLowerCase() !== netState.ensAddress.toLowerCase()) { - throw new Error( - `ensAddress for network ${netId} is different in Hardhat config file ${hardhatConfig} ` + `and network state file ${networkStateFile}` - ) - } - - // prevent Hardhat from passing the config to subprocesses - const env = filterObject(process.env, (key) => key.substr(0, 8) !== 'HARDHAT_') - - if (appNames && appNames !== '*') { - appNames = appNames.split(',') - } else { - appNames = fs.readdirSync(appsDirPath) - } - - for (const appName of appNames) { - const app = await publishApp(appName, env, netName, appsDirPath, releaseType) - persistNetworkState(network.name, netId, netState, { - [`app:${app.name}`]: { - ...netState[`app:${app.name}`], - ...app - } - }) - } -} - -async function publishApp(appName, env, netName, appsDirPath, releaseType) { - logHeader(`Publishing new ${releaseType} release of app '${appName}'`) - - let result = {} - const appRootPath = path.resolve(appsDirPath, appName) - const appFullName = await readAppName(appRootPath, netName) - const appId = namehash(appFullName) - const appFrontendPath = path.join(appRootPath, 'app') - - log(`App name: ${yl(appFullName)}`) - log(`App ID: ${yl(appId)}`) - logSplitter() - - logSplitter(`Change registry app ${appName}`) - const arappFile = `${appRootPath}/arapp.json` - const arapp = await readJSON(arappFile) - - if (!arapp.environments[network.name]) { - arapp.environments[network.name] = { ...arapp.environments.default } - } - arapp.environments[network.name].registry = network.config.ensAddress - - logSplitter(`Write state app ${appName}`) - writeNetworkStateFile(arappFile, arapp) - - if (!SKIP_APPS_LONG_BUILD_STEPS) { - log(`Installing frontend deps for app ${appName}`) - await execLive('yarn', { - cwd: appFrontendPath - }) - - log(`Build app ${appName}`) - await execLive('yarn', { - args: ['build'], - cwd: appFrontendPath - }) - } - - await execLive('hardhat', { - args: [ - 'publish', - releaseType, - '--network', - netName, - // '--skip-app-build', - // workaround: force to read URL from Hardhat config - '--ipfs-api-url', - '' - ], - cwd: appRootPath, - env - }) - - return { - ...result, - fullName: appFullName, - name: appName, - id: appId - } -} - -module.exports = runOrWrapScript(deployLidoApps, module) diff --git a/scripts/scratch/verify-contracts-code.sh b/scripts/scratch/verify-contracts-code.sh new file mode 100644 index 000000000..267f35b4f --- /dev/null +++ b/scripts/scratch/verify-contracts-code.sh @@ -0,0 +1,89 @@ +#!/bin/bash +set -e +SCRIPT_DIR=$( cd -- "$( dirname -- "${BASH_SOURCE[0]}" )" &> /dev/null && pwd ) + + +if [[ -z "$NETWORK" ]]; then + echo "Must set NETWORK env variable" 1>&2 + exit 1 +fi + +NETWORK_STATE_FILE="deployed-${NETWORK}.json" +if [ ! -f $NETWORK_STATE_FILE ]; then + echo "Cannot find network state file ${NETWORK_STATE_FILE}" + exit 1 +fi +echo "Using network state file ${NETWORK_STATE_FILE}" + +function jsonGet { + node -e "const fs = require('fs'); const obj = JSON.parse(fs.readFileSync('${NETWORK_STATE_FILE}', 'utf8')); const path='$1'; let res = path.split('.').reduce(function(o, k) {return o && o[k] }, obj); console.log(res)" +} + +function verify { + contractPath="$(jsonGet ${1}.contract)" + contractName="${contractPath##*/}" + contractName="${contractName%.*}" + argsJson=$(jsonGet ${1}.constructorArgs) + echo "module.exports = $argsJson" > contract-args.js + yarn hardhat --network $NETWORK verify --no-compile --contract "$contractPath:$contractName" --constructor-args contract-args.js $(jsonGet ${1}.address) +} + +# NB: Although most of the contracts listed below would be verified by running +# this bash script as it is, some might require some manual tweaking. +# Sometimes first attempt to verify fails without observable reason. +# Part of the contract require a workaround see SCRATCH_DEPLOY.md section +# "Issues with verification of part of the contracts deployed from factories". + +verify dummyEmptyContract +verify burner +verify hashConsensusForAccountingOracle +verify hashConsensusForValidatorsExitBusOracle +verify accountingOracle.implementation +verify accountingOracle.proxy +verify validatorsExitBusOracle.implementation +verify validatorsExitBusOracle.proxy +verify stakingRouter.implementation +verify stakingRouter.proxy +verify withdrawalQueueERC721.proxy +verify wstETH +verify executionLayerRewardsVault +verify eip712StETH +verify lidoTemplate +verify withdrawalVault.proxy +verify withdrawalVault.implementation +verify lidoLocator.proxy +verify lidoLocator.implementation +verify app:lido.implementation +verify app:oracle.implementation +verify app:node-operators-registry.implementation +verify app:aragon-voting.implementation +verify app:aragon-token-manager.implementation +verify app:aragon-finance.implementation +verify app:aragon-agent.implementation +verify oracleDaemonConfig +verify oracleReportSanityChecker +verify fakeAppProxyPinned +verify app:lido.proxy +verify depositSecurityModule +verify withdrawalQueueERC721.implementation +verify aragon-kernel.implementation +verify aragon-acl.implementation +verify aragon-kernel.proxy +verify ldo +verify callsScript +verify aragon-evm-script-registry.proxy +verify aragon-apm-registry.implementation +verify aragon-apm-registry.factory +verify aragon-app-repo-lido.implementation +verify aragon-app-repo-node-operators-registry.implementation +# NB: App Repos of lido, oracle, node-operators-registry, finance, agent, token-manager, voting +# share same implementation of Repo contract +verify aragon-evm-script-registry.proxy +verify aragon-evm-script-registry.implementation +verify app:simple-dvt.proxy +verify app:aragon-token-manager.proxy +verify app:oracle.proxy +verify app:node-operators-registry.proxy +verify app:aragon-voting.proxy +verify app:aragon-finance.proxy +verify app:aragon-agent.proxy diff --git a/scripts/staking-router.js b/scripts/staking-router.js index 32abf88b7..c43de26b6 100644 --- a/scripts/staking-router.js +++ b/scripts/staking-router.js @@ -62,7 +62,7 @@ let StakingRouter = { last_distribute: 0, // - // fucntions + // functions // getTotalKeys: function() { @@ -262,7 +262,7 @@ function main() { let allocation = StakingRouter.allocation console.log('allocation1', allocation) - console.log('last distibute', StakingRouter.last_distribute) + console.log('last distribute', StakingRouter.last_distribute) //community deposited communityModule.deposit(3) @@ -283,7 +283,7 @@ function main() { //try to get next deposit by por from solo - console.log('try depoist curated again 150 keys') + console.log('try deposit curated again 150 keys') curatedModule.deposit(150) allocation = StakingRouter.allocation diff --git a/scripts/start-aragon.js b/scripts/start-aragon.js deleted file mode 100644 index 5db4e0ace..000000000 --- a/scripts/start-aragon.js +++ /dev/null @@ -1,119 +0,0 @@ -const path = require('path') -const fs = require('fs') -const namehash = require('eth-ens-namehash').hash - -const { execLive } = require('./helpers/exec') -const { log, logHeader, yl, gr } = require('./helpers/log') -const { gitCloneRepo } = require('./helpers/git') -const { readJSON, fileExists } = require('./helpers/fs') - -const runOrWrapScript = require('./helpers/run-or-wrap-script') - -const APPS = process.env.APPS || '*' -const APPS_DIR_PATH = process.env.APPS_DIR_PATH || path.resolve(__dirname, '..', 'apps') -const ARAGON_APPS_REPO = process.env.ARAGON_APPS_REPO || 'https://github.com/lidofinance/aragon-client' -const ARAGON_APPS_REPO_REF = process.env.ARAGON_APPS_REPO_REF || 'master' -const RUN_CMD = process.env.RUN_CMD || 'local' -const NETWORK_NAME = process.env.NETWORK_NAME || 'localhost' - -const appsRepoPath = './aragon-client' - -const CMD_LOCAL = 'local' -const CMD_MAINNET = 'mainnet' -const CMD_RINKEBY = 'rinkeby' -const CMD_STAGING = 'staging' -const CMD_ROPSTEN = 'ropsten' -const CMD_XDAI = 'xdai' -const AVAILIABLE_RUN_CMDS = [CMD_LOCAL, CMD_MAINNET, CMD_RINKEBY, CMD_STAGING, CMD_ROPSTEN, CMD_XDAI] - -async function startAragonClient( - aragonAppsRepoRef = ARAGON_APPS_REPO_REF, - appsDirPath = APPS_DIR_PATH, - appNames = APPS, - runCmd = RUN_CMD -) { - - assertRequiredRunCmd(runCmd, AVAILIABLE_RUN_CMDS) - - var deployedFileName = `deployed-${NETWORK_NAME}.json` - var deployedFileExists = await fileExists(deployedFileName) - if (!deployedFileExists) { - log(`File ${deployedFileName} doesn't exists`) - return - } - - const deployedFile = await readJSON(deployedFileName) - const ensAddress = process.env.ENS_ADDRESS || deployedFile.ensAddress || '0x00000000000C2E074eC69A0dFb2997BA6C7d2e1e' - - - logHeader(`Checking out aragon-client repo...`) - await gitCloneRepo(appsRepoPath, ARAGON_APPS_REPO, aragonAppsRepoRef) - - await execLive('yarn', { - cwd: appsRepoPath - }) - - if (appNames && appNames !== '*') { - appNames = appNames.split(',') - } else { - appNames = fs.readdirSync(appsDirPath) - } - - const appLocations = [] - let port = 3000 - for (const appName of appNames) { - port++ - const appRootPath = path.resolve(appsDirPath, appName) - const arappFile = `${appRootPath}/arapp.json` - const arapp = await readJSON(arappFile) - const appPackageJson = await readJSON(`${appRootPath}/app/package.json`) - - const res = appPackageJson.scripts.dev && appPackageJson.scripts.dev.match(/--port\s+(\d+)/) - if (res) { - port = res[1] - } - - const network = 'mainnet' - const appId = arapp.environments[network].appName - const appHash = namehash(appId) - const location = `${appHash}:http://localhost:${port}/` - - appLocations.push(location) - - log(`${yl(appId)}: ${location}`) - } - const appLocator = process.env.ARAGON_APP_LOCATOR || appLocations.join(',') - - const aragonEnv = { - ...process.env, - ARAGON_APP_LOCATOR: appLocator, - ARAGON_ENS_REGISTRY_ADDRESS: ensAddress, - ARAGON_DEFAULT_ETH_NODE: 'ws://localhost:8545', - ARAGON_IPFS_GATEWAY: 'http://localhost:8080' - } - - if (runCmd === CMD_MAINNET) { - aragonEnv.ARAGON_DEFAULT_ETH_NODE = 'ws://localhost:8545' - aragonEnv.ARAGON_IPFS_GATEWAY = 'https://mainnet.lido.fi/ipfs' - } - - console.log(`ARAGON_APP_LOCATOR=${appLocator}`) - console.log(`ARAGON_ENS_REGISTRY_ADDRESS=${ensAddress}`) - console.log(`ARAGON_DEFAULT_ETH_NODE=${aragonEnv.ARAGON_DEFAULT_ETH_NODE}`) - console.log(`ARAGON_IPFS_GATEWAY=${aragonEnv.ARAGON_IPFS_GATEWAY}`) - - await execLive('yarn', { - args: [`start:${runCmd}`], - cwd: appsRepoPath, - env: aragonEnv - }) -} - -function assertRequiredRunCmd(runCmd, availableCmd) { - let state = availableCmd.filter((key) => runCmd == key) - if (!state.length) { - throw new Error(`Invalid RUN_CMD env, ` + `available options: ${availableCmd.join(', ')}`) - } -} - -startAragonClient() diff --git a/scripts/upgrade/deploy-new-app-instance.js b/scripts/upgrade/deploy-new-app-instance.js index 23e36ad27..08d5107a7 100644 --- a/scripts/upgrade/deploy-new-app-instance.js +++ b/scripts/upgrade/deploy-new-app-instance.js @@ -26,7 +26,7 @@ async function upgradeApp({ web3, artifacts, appName = APP }) { logSplitter() await saveDeployTx(appArtifact, `tx-13-1-deploy-${appName}-base.json`, { - from: DEPLOYER || state.multisigAddress + from: DEPLOYER || state.deployer }) logSplitter() diff --git a/scripts/upgrade/obtain-deployed-new-app-instance.js b/scripts/upgrade/obtain-deployed-new-app-instance.js index f04333b4a..b2a2d0139 100644 --- a/scripts/upgrade/obtain-deployed-new-app-instance.js +++ b/scripts/upgrade/obtain-deployed-new-app-instance.js @@ -8,7 +8,7 @@ const { APP_NAMES, APP_ARTIFACTS } = require('./constants') const VALID_APP_NAMES = Object.entries(APP_NAMES).map((e) => e[1]) const APP = process.env.APP || '' -const REQUIRED_NET_STATE = ['ensAddress', 'multisigAddress', 'lidoBaseDeployTx', 'oracleBaseDeployTx', 'nodeOperatorsRegistryBaseDeployTx'] +const REQUIRED_NET_STATE = ['ensAddress', 'deployer', 'lidoBaseDeployTx', 'oracleBaseDeployTx', 'nodeOperatorsRegistryBaseDeployTx'] async function obtainInstance({ web3, artifacts, appName = APP }) { if (!appName || !VALID_APP_NAMES.includes(appName)) { diff --git a/scripts/upgrade/upgrade-app.js b/scripts/upgrade/upgrade-app.js index b593336c2..d0f35ad5d 100644 --- a/scripts/upgrade/upgrade-app.js +++ b/scripts/upgrade/upgrade-app.js @@ -37,8 +37,8 @@ async function upgradeAppImpl({ web3, artifacts, appName = APP }) { logSplitter() - log(`Using ENS:`, yl(state.ensAddress)) - const ens = await artifacts.require('ENS').at(state.ensAddress) + log(`Using ENS:`, yl(state.ens.address)) + const ens = await artifacts.require('ENS').at(state.ens.address) log.splitter() const appId = namehash(`${appName}.${state.lidoApmEnsName}`) @@ -73,7 +73,7 @@ async function upgradeAppImpl({ web3, artifacts, appName = APP }) { } const versionTo = currentVersion.map((n) => n.toNumber()) - const txSender = HOLDER || state.multisigAddress + const txSender = HOLDER || state.deployer log(`Upgrading app:`, yl(appName)) log(`App ID:`, yl(appId)) diff --git a/scripts/upgrade/vote-new-oracle-impl.js b/scripts/upgrade/vote-new-oracle-impl.js index bc51c3fe5..b4b12bf87 100644 --- a/scripts/upgrade/vote-new-oracle-impl.js +++ b/scripts/upgrade/vote-new-oracle-impl.js @@ -42,7 +42,7 @@ async function upgradeAppImpl({ web3, artifacts, appName = APP }) { logSplitter() log(`Using ENS:`, yl(state.ensAddress)) - const ens = await artifacts.require('ENS').at(state.ensAddress) + const ens = await artifacts.require('ENS').at(state.ens.address) log.splitter() const appId = namehash(`${appName}.${state.lidoApmEnsName}`) diff --git a/scripts/utils/31-start-protocol.js b/scripts/utils/31-start-protocol.js deleted file mode 100644 index b71d79a87..000000000 --- a/scripts/utils/31-start-protocol.js +++ /dev/null @@ -1,106 +0,0 @@ -const { hash: namehash } = require('eth-ens-namehash') -const { encodeCallScript } = require('@aragon/contract-helpers-test/src/aragon-os') -const { BN } = require('bn.js') - -const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log, logSplitter, logWideSplitter, yl, gr } = require('../helpers/log') -const { saveCallTxData } = require('../helpers/tx-data') -const { readNetworkState, assertRequiredNetworkState } = require('../helpers/persisted-network-state') - -const { APP_NAMES } = require('./constants') - -const ETH = (value) => web3.utils.toWei(value + '', 'ether') - -const DEPLOYER = process.env.DEPLOYER || '' -const REQUIRED_NET_STATE = [ - 'daoAddress', - `app:${APP_NAMES.ARAGON_VOTING}`, - `app:${APP_NAMES.ARAGON_TOKEN_MANAGER}`, - `app:${APP_NAMES.LIDO}`, -] - -async function createVoting({ web3, artifacts }) { - const netId = await web3.eth.net.getId() - - logWideSplitter() - log(`Network ID:`, yl(netId)) - - const state = readNetworkState(network.name, netId) - assertRequiredNetworkState(state, REQUIRED_NET_STATE) - - logSplitter() - - const votingAddress = state[`app:${APP_NAMES.ARAGON_VOTING}`].proxyAddress - const voting = await artifacts.require('Voting').at(votingAddress) - const tokenManager = await artifacts.require('TokenManager') - .at(state[`app:${APP_NAMES.ARAGON_TOKEN_MANAGER}`].proxyAddress) - const lidoAddress = state[`app:lido`].proxyAddress - const lido = await artifacts.require('Lido').at(lidoAddress) - - - const dailyStakingLimit = ETH(150000) - const stakeLimitIncreasePerBlock = calcStakeLimitIncreasePerBlock(dailyStakingLimit) - - log(`dailyStakeLimit: `, yl(dailyStakingLimit)) - log(`stakeLimitIncreasePerBlock: `, yl(stakeLimitIncreasePerBlock)) - - log.splitter() - - const resumeProtocolCallData = { - to: lidoAddress, - calldata: await lido.contract.methods.resume().encodeABI() - } - - const resumeStakingCallData = { - to: lidoAddress, - calldata: await lido.contract.methods.resumeStaking().encodeABI() - } - - const setStakingLimitCallData = { - to: lidoAddress, - calldata: await lido.contract.methods.setStakingLimit(dailyStakingLimit, stakeLimitIncreasePerBlock).encodeABI() - } - - - const encodedUpgradeCallData = encodeCallScript([ - resumeProtocolCallData, - resumeStakingCallData, - setStakingLimitCallData, - ]) - - log(`encodedUpgradeCallData:`, yl(encodedUpgradeCallData)) - const votingCallData = encodeCallScript([ - { - to: votingAddress, - calldata: await voting.contract.methods.forward(encodedUpgradeCallData).encodeABI() - } - ]) - - const txName = `tx-31-start-protocol.json` - const votingDesc = `1) Unpause protocol -2) Unpause staking -3) Set daily staking limit to ${fromE18ToString(dailyStakingLimit)}` - - await saveCallTxData(votingDesc, tokenManager, 'forward', txName, { - arguments: [votingCallData], - from: DEPLOYER || state.multisigAddress - }) - - log.splitter() - log(gr(`Before continuing the deployment, please send all transactions listed above.`)) - log(gr(`You must complete it positively and execute before continuing with the deployment!`)) - log.splitter() -} - -function calcStakeLimitIncreasePerBlock(dailyLimit) { - const secondsPerBlock = 12 - const secondsPerDay = 24 * 60 * 60 - const blocksPerDay = secondsPerDay / secondsPerBlock - return Math.floor(dailyLimit / blocksPerDay).toString() -} - -function fromE18ToString(x) { - return `${(x / 1e18).toFixed(3)} ETH (${x} wei)` -} - -module.exports = runOrWrapScript(createVoting, module) diff --git a/scripts/utils/32-change-voting-time.js b/scripts/utils/32-change-voting-time.js deleted file mode 100644 index dcaa078f7..000000000 --- a/scripts/utils/32-change-voting-time.js +++ /dev/null @@ -1,139 +0,0 @@ -const { hash: namehash } = require('eth-ens-namehash') -const { encodeCallScript } = require('@aragon/contract-helpers-test/src/aragon-os') -const { BN } = require('bn.js') - -const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log, logSplitter, logWideSplitter, yl, gr } = require('../helpers/log') -const { saveCallTxData } = require('../helpers/tx-data') -const { readNetworkState, assertRequiredNetworkState, persistNetworkState } = require('../helpers/persisted-network-state') -const { resolveEnsAddress } = require('../components/ens') - -const { APP_NAMES } = require('./constants') - -const ETH = (value) => web3.utils.toWei(value + '', 'ether') - -const DEPLOYER = process.env.DEPLOYER || '' -const REQUIRED_NET_STATE = [ - 'lidoApmEnsName', - 'ensAddress', - 'daoAddress', - `app:${APP_NAMES.ARAGON_VOTING}`, - `app:${APP_NAMES.ARAGON_TOKEN_MANAGER}` -] - -async function createVoting({ web3, artifacts }) { - const netId = await web3.eth.net.getId() - - logWideSplitter() - log(`Network ID:`, yl(netId)) - - const state = readNetworkState(network.name, netId) - assertRequiredNetworkState(state, REQUIRED_NET_STATE) - - logSplitter() - - const votingAddress = state[`app:${APP_NAMES.ARAGON_VOTING}`].proxyAddress - const tokenManagerAddress = state[`app:${APP_NAMES.ARAGON_TOKEN_MANAGER}`].proxyAddress - const voting = await artifacts.require('Voting').at(votingAddress) - const tokenManager = await artifacts.require('TokenManager').at(tokenManagerAddress) - const kernel = await artifacts.require('Kernel').at(state.daoAddress) - const aclAddress = await kernel.acl() - const acl = await artifacts.require('ACL').at(aclAddress) - - const objectionPhaseTimeSec = 5 - const voteTimeSec = 60 - - log(`Using ENS:`, yl(state.ensAddress)) - log(`TokenManager address:`, yl(tokenManagerAddress)) - log(`Voting address:`, yl(votingAddress)) - log(`Kernel:`, yl(kernel.address)) - log(`ACL:`, yl(acl.address)) - - log.splitter() - - const voteRoleId = '0x068ca51c9d69625c7add396c98ca4f3b27d894c3b973051ad3ee53017d7094ea' // keccak256(UNSAFELY_MODIFY_VOTE_TIME_ROLE) - - - // This script grants and revokes the role so checking it is absent at the beginning - // assert.isFalse(await acl.hasPermission(votingAddress, votingAddress, voteRoleId)) - - const createChangeVoteTimePermission = { - to: acl.address, - calldata: await acl.contract.methods - .createPermission( - votingAddress, - votingAddress, - voteRoleId, - votingAddress - ) - .encodeABI() - } - - const grantChangeVoteTimePermission = { - to: acl.address, - calldata: await acl.contract.methods - .grantPermission( - votingAddress, - votingAddress, - voteRoleId - ) - .encodeABI() - } - - const revokeChangeVoteTimePermission = { - to: acl.address, - calldata: await acl.contract.methods - .revokePermission( - votingAddress, - votingAddress, - voteRoleId - ) - .encodeABI() - } - - const changeObjectionTime = { - to: votingAddress, - calldata: await voting.contract.methods.unsafelyChangeObjectionPhaseTime(objectionPhaseTimeSec).encodeABI() - } - - const changeVoteTime = { - to: votingAddress, - calldata: await voting.contract.methods.unsafelyChangeVoteTime(voteTimeSec).encodeABI() - } - - const encodedUpgradeCallData = encodeCallScript([ - createChangeVoteTimePermission, - grantChangeVoteTimePermission, - changeObjectionTime, - changeVoteTime, - revokeChangeVoteTimePermission, - ]) - - log(`encodedUpgradeCallData:`, yl(encodedUpgradeCallData)) - const votingCallData = encodeCallScript([ - { - to: votingAddress, - calldata: await voting.contract.methods.forward(encodedUpgradeCallData).encodeABI() - } - ]) - - const txName = `tx-32-change-voting-time.json` - const votingDesc = -`1) Grant permission UNSAFELY_MODIFY_VOTE_TIME_ROLE to Voting -2) Set objection phase time to ${objectionPhaseTimeSec} seconds -3) Set total vote time to ${voteTimeSec} seconds -4) Revoke permission UNSAFELY_MODIFY_VOTE_TIME_ROLE from Voting` - - await saveCallTxData(votingDesc, tokenManager, 'forward', txName, { - arguments: [votingCallData], - from: DEPLOYER || state.multisigAddress - }) - - log.splitter() - log(gr(`Before continuing the deployment, please send all transactions listed above.`)) - log(gr(`You must complete it positively and execute before continuing with the deployment!`)) - log.splitter() -} - - -module.exports = runOrWrapScript(createVoting, module) diff --git a/scripts/utils/verify-vote-tx.js b/scripts/utils/verify-vote-tx.js index 081e44e71..c1be3807e 100644 --- a/scripts/utils/verify-vote-tx.js +++ b/scripts/utils/verify-vote-tx.js @@ -7,7 +7,6 @@ const { readNetworkState, assertRequiredNetworkState } = require('../helpers/per const TX = process.env.TX const REQUIRED_NET_STATE = [ 'ensAddress', - 'multisigAddress', 'lidoBaseDeployTx', 'oracleBaseDeployTx', 'nodeOperatorsRegistryBaseDeployTx', diff --git a/scripts/utils/vote-and-enact.js b/scripts/utils/vote-and-enact.js deleted file mode 100644 index 92bee4957..000000000 --- a/scripts/utils/vote-and-enact.js +++ /dev/null @@ -1,60 +0,0 @@ -const runOrWrapScript = require('../helpers/run-or-wrap-script') -const { log, logSplitter, logWideSplitter, yl, gr } = require('../helpers/log') -const { saveCallTxData } = require('../helpers/tx-data') -const { readNetworkState, assertRequiredNetworkState, persistNetworkState } = require('../helpers/persisted-network-state') - -const { APP_NAMES } = require('./constants') - -const REQUIRED_NET_STATE = ['daoInitialSettings', `app:${APP_NAMES.ARAGON_VOTING}`, 'owner'] - -const maxWaitVoteTimeSecs = 60 - -const timer = ms => new Promise( res => setTimeout(res, ms)) - -async function voteAndEnact({ web3, artifacts }) { - const netId = await web3.eth.net.getId() - - logWideSplitter() - log(`Network ID:`, yl(netId)) - - const state = readNetworkState(network.name, netId) - assertRequiredNetworkState(state, REQUIRED_NET_STATE) - - logSplitter() - - log(`Using ENS:`, yl(state.ensAddress)) - log.splitter() - - const votingAddress = state[`app:${APP_NAMES.ARAGON_VOTING}`].proxyAddress - const voting = await artifacts.require('Voting').at(votingAddress) - - // Get voteId from env or use the latest vote id - let voteId = process.env.VOTE_ID || '' - if (voteId === '') { - voteId = ((await voting.votesLength()).toString() - 1).toString() - } - - const ldoMegaHolder = state['owner'] - log.splitter() - - log(`Voting for vote ${voteId}...`) - await voting.vote(voteId, true, false, { from: ldoMegaHolder }) - - const voteTimeSecs = +(await voting.voteTime()).toString() - if (voteTimeSecs <= maxWaitVoteTimeSecs) { - console.log(`Wait ${voteTimeSecs} seconds till the voting is over`) - await timer(voteTimeSecs * 1000) - - // NB: the vote won't execute till the voting time passes - // for a local hardhat node we can trick the time with - // `await ethers.provider.send("evm_mine", [newTimestampInSeconds]);` - // but cannot for a geth node - await voting.executeVote(voteId, { from: ldoMegaHolder }) - log(`Vote ${voteId} executed!`) - } else { - console.log(`The voting time ${voteTimeSecs} secs for vote ${voteId} is too long to wait in this script. -You'd need to enact it manually`) - } -} - -module.exports = runOrWrapScript(voteAndEnact, module) diff --git a/test/0.8.9/unstructured-storage.test.sol b/test/0.8.9/unstructured-storage.test.sol new file mode 100644 index 000000000..0e8089f1b --- /dev/null +++ b/test/0.8.9/unstructured-storage.test.sol @@ -0,0 +1,150 @@ +// SPDX-License-Identifier: MIT +pragma solidity 0.8.9; + +import "forge-std/Test.sol"; +import { UnstructuredStorage } from "contracts/0.8.9/lib/UnstructuredStorage.sol"; + +contract ExposedUnstructuredStorage { + function getStorageBool(bytes32 position) public view returns (bool) { + return UnstructuredStorage.getStorageBool(position); + } + + function getStorageAddress(bytes32 position) public view returns (address) { + return UnstructuredStorage.getStorageAddress(position); + } + + function getStorageBytes32(bytes32 position) public view returns (bytes32) { + return UnstructuredStorage.getStorageBytes32(position); + } + + function getStorageUint256(bytes32 position) public view returns (uint256) { + return UnstructuredStorage.getStorageUint256(position); + } + + function setStorageBool(bytes32 position, bool data) public { + return UnstructuredStorage.setStorageBool(position, data); + } + + function setStorageAddress(bytes32 position, address data) public { + return UnstructuredStorage.setStorageAddress(position, data); + } + + function setStorageBytes32(bytes32 position, bytes32 data) public { + return UnstructuredStorage.setStorageBytes32(position, data); + } + + function setStorageUint256(bytes32 position, uint256 data) public { + return UnstructuredStorage.setStorageUint256(position, data); + } +} + +contract ExposedUnstructuredStorageTest is Test { + ExposedUnstructuredStorage public unstructedStorage; + + function setUp() public { + unstructedStorage = new ExposedUnstructuredStorage(); + } + + function testGetStorageBoolUnitialized() public { + bytes32 position = keccak256("FOO"); + assertEq(unstructedStorage.getStorageBool(position), false); + } + + function testGetStorageBoolUnitializedFuzz(bytes32 position) public { + assertEq(unstructedStorage.getStorageBool(position), false); + } + + function testGetStorageAddressUnitialized() public { + bytes32 position = keccak256("FOO"); + assertEq(unstructedStorage.getStorageAddress(position), address(0)); + } + + function testGetStorageAddressUnitializedFuzz(bytes32 position ) public { + assertEq(unstructedStorage.getStorageAddress(position), address(0)); + } + + function testGetStorageBytes32Unitialized() public { + bytes32 position = keccak256("FOO"); + bytes32 data; + assertEq(unstructedStorage.getStorageBytes32(position), data); + } + + function testGetStorageBytes32UnitializedFuzz(bytes32 position) public { + bytes32 data; + assertEq(unstructedStorage.getStorageBytes32(position), data); + } + + function testGetStorageUint256Unitialized() public { + bytes32 position = keccak256("FOO"); + uint256 data; + assertEq(unstructedStorage.getStorageUint256(position), data); + } + + function testGetStorageUint256UnitializedFuzz(bytes32 position) public { + uint256 data; + assertEq(unstructedStorage.getStorageUint256(position), data); + } + + function testSetStorageBool() public { + bytes32 position = keccak256("FOO"); + assertEq(unstructedStorage.getStorageBool(position), false); + + unstructedStorage.setStorageBool(position, true); + assertEq(unstructedStorage.getStorageBool(position), true); + + unstructedStorage.setStorageBool(position, false); + assertEq(unstructedStorage.getStorageBool(position), false); + } + + function testSetStorageAddress() public { + bytes32 position = keccak256("FOO"); + address data = vm.addr(1); + + assertEq(unstructedStorage.getStorageAddress(position), address(0)); + unstructedStorage.setStorageAddress(position, data); + assertEq(unstructedStorage.getStorageAddress(position), data); + } + + function testSetStorageAddressFuzz(address data, bytes32 position) public { + assertEq(unstructedStorage.getStorageAddress(position), address(0)); + unstructedStorage.setStorageAddress(position, data); + assertEq(unstructedStorage.getStorageAddress(position), data); + } + + function testSetStorageBytes32() public { + bytes32 position = keccak256("FOO"); + bytes32 data = keccak256("BAR"); + bytes32 unintializedData; + + assertEq(unstructedStorage.getStorageBytes32(position), unintializedData); + unstructedStorage.setStorageBytes32(position, data); + assertEq(unstructedStorage.getStorageBytes32(position), data); + } + + function testSetStorageBytes32Fuzz(bytes32 data, bytes32 position) public { + bytes32 unintializedData; + + assertEq(unstructedStorage.getStorageBytes32(position), unintializedData); + unstructedStorage.setStorageBytes32(position, data); + assertEq(unstructedStorage.getStorageBytes32(position), data); + } + + function testSetStorageUint256() public { + bytes32 position = keccak256("FOO"); + uint256 data = 1; + uint256 unintializedData; + + assertEq(unstructedStorage.getStorageUint256(position), unintializedData); + unstructedStorage.setStorageUint256(position, data); + assertEq(unstructedStorage.getStorageUint256(position), data); + } + + function testSetStorageUint256Fuzz(uint256 data, bytes32 position) public { + uint256 unintializedData; + + assertEq(unstructedStorage.getStorageUint256(position), unintializedData); + unstructedStorage.setStorageUint256(position, data); + assertEq(unstructedStorage.getStorageUint256(position), data); + } + +} \ No newline at end of file diff --git a/test/common/lib/math-256.test.sol b/test/common/lib/math-256.test.sol new file mode 100644 index 000000000..9a82b5e34 --- /dev/null +++ b/test/common/lib/math-256.test.sol @@ -0,0 +1,300 @@ +// SPDX-License-Identifier: MIT +pragma solidity 0.8.9; + +import "forge-std/Test.sol"; +import { Math256 } from "contracts/common/lib/Math256.sol"; + +contract Math256Test is Test { + + /// uint256 tests for max/min + + function testMaxUint256_a_b() public { + uint256 a = 1; + uint256 b = 2; + + assertEq(Math256.max(a, b), b); + } + + function testMaxUint256_b_a() public { + uint256 a = 1; + uint256 b = 2; + + assertEq(Math256.max(b, a), b); + } + + function testMaxUint256_a_b_equal() public { + uint256 a = 1; + uint256 b = 1; + + assertEq(Math256.max(b, a), b); + } + + function testMaxUint256Fuzz(uint256 a, uint256 b) public { + uint256 expected; + + if (a > b) { + expected = a; + } else { + expected = b; + } + + // Must not crash + Math256.min(b, a); + + // Must be commutative + assertEq(Math256.min(b, a), Math256.min(a, b)); + + // Must be expected + assertEq(Math256.max(b, a), expected); + } + + function testMinUint256_a_b() public { + uint256 a = 2; + uint256 b = 1; + + assertEq(Math256.min(a, b), b); + } + + function testMinUint256_b_a() public { + uint256 a = 1; + uint256 b = 2; + + assertEq(Math256.min(b, a), a); + } + + function testMinUint256_a_b_equal() public { + uint256 a = 1; + uint256 b = 1; + + assertEq(Math256.max(b, a), b); + } + + function testUint256MinFuzz(uint256 a, uint256 b) public { + uint256 expected; + + if (a < b) { + expected = a; + } else { + expected = b; + } + + // Must be commutative + assertEq(Math256.min(b, a), Math256.min(a, b)); + + // Must be expected + assertEq(Math256.min(b, a), expected); + } + + /// int256 tests for max/min + + function testMaxInt256_a_b() public { + int256 a = 1; + int256 b = 2; + + assertEq(Math256.max(a, b), b); + } + + function testMaxInt256_b_a() public { + int256 a = 1; + int256 b = 2; + + assertEq(Math256.max(b, a), b); + } + + function testMaxInt256_a_b_equal() public { + int256 a = 1; + int256 b = 1; + + assertEq(Math256.max(b, a), b); + } + + function testMaxInt256_a_b_negative() public { + int256 a = -1; + int256 b = -2; + + assertEq(Math256.max(a, b), a); + } + + function testMaxInt256_a_b_positive_negative() public { + int256 a = 1; + int256 b = -2; + + assertEq(Math256.max(a, b), a); + } + + function testMaxInt256_b_a_negative() public { + int256 a = -1; + int256 b = -2; + + assertEq(Math256.max(b, a), a); + } + + function testMaxInt256_b_a_postive_negative() public { + int256 a = 1; + int256 b = -2; + + assertEq(Math256.max(b, a), a); + } + + function testMaxInt256_a_b_equal_negative() public { + int256 a = -1; + int256 b = -1; + + assertEq(Math256.max(b, a), b); + } + + function testMaxInt256Fuzz(int256 a, int256 b) public { + int256 expected; + + if (a > b) { + expected = a; + } else { + expected = b; + } + + // Must be commutative + assertEq(Math256.max(b, a), Math256.max(a, b)); + + // Must be exepcted + assertEq(Math256.max(b, a), expected); + } + + function testMinInt256_a_b() public { + int256 a = 2; + int256 b = 1; + + assertEq(Math256.min(a, b), b); + } + + function testMinInt256_b_a() public { + int256 a = 1; + int256 b = 2; + + assertEq(Math256.min(b, a), a); + } + + function testMinInt256_b_a_equal() public { + int256 a = 1; + int256 b = 1; + + assertEq(Math256.max(b, a), b); + } + + function testInt256MinFuzz(int256 a, int256 b) public { + int256 expected; + + if (a < b) { + expected = a; + } else { + expected = b; + } + + // Must not crash + Math256.min(b, a); + + // Must be commutative + assertEq(Math256.min(b, a), Math256.min(a, b)); + + // Must be expected + assertEq(Math256.min(b, a), expected); + } + + /// tests for ceilDiv + + // Commenting this out, as the implementation doesn't solve for this case + // function testCeilDivByZero() public { + // uint256 a = 1; + // uint256 b = 0; + + // vm.expectRevert("Division or modulo by 0"); + // Math256.ceilDiv(a, b); + // } + + function testCeilDivZeroFromFour() public { + uint256 a = 0; + uint256 b = 4; + assertEq(Math256.ceilDiv(a, b), 0); + } + + function testCeilDivByOne() public { + uint256 a = 2; + uint256 b = 1; + + assertEq(Math256.ceilDiv(a, b), a); + } + + function testCeilDivByTwo() public { + uint256 a = 4; + uint256 b = 2; + + assertEq(Math256.ceilDiv(a, b), b); + } + + function testCeilDivByThree() public { + uint256 a = 4; + uint256 b = 3; + + assertEq(Math256.ceilDiv(a, b), 2); + } + + function testCeilDivFuzz(uint256 a, uint256 b) public { + // Skip zero, implementation is safe against division by zero + vm.assume(b != 0); + + // This case should always be zero + if (a == 0) { + assertEq(Math256.ceilDiv(a, b), 0); + } + + // When they are both equal, the orientation shouldn't matter, it should be 1 + if (a == b) { + assertEq(Math256.ceilDiv(a, b), 1); + assertEq(Math256.ceilDiv(b, a), 1); + } + + uint256 expected = (a == 0 ? 0 : (a - 1) / b + 1); + assertEq(Math256.ceilDiv(a, b), expected); + } + + /// tests for absDiff + + function testAbsDiffZeros() public { + uint256 a = 0; + uint256 b = 0; + + assertEq(Math256.absDiff(b, a), 0); + } + + function testAbsDiffOnes() public { + uint256 a = 1; + uint256 b = 1; + + assertEq(Math256.absDiff(b, a), 0); + } + + function testAbsDiffFuzz(uint256 a, uint256 b) public { + + // It shouldn't unexpectedly crash + Math256.absDiff(b, a); + + // If they are the same, it's always zero + if (a == b) { + assertEq(Math256.absDiff(b, a), 0); + } + + // They are different + if (b > a) { + assertEq(Math256.absDiff(b, a), b - a); + } else { + assertEq(Math256.absDiff(a, b), a - b); + } + + // If one is zero, the difference should always be the other + if (a == 0) { + assertEq(Math256.absDiff(b, a), b); + } + + // Must be commutative + assertEq(Math256.absDiff(b, a), Math256.absDiff(a, b)); + } +} \ No newline at end of file diff --git a/test/common/lib/signature-utils.test.sol b/test/common/lib/signature-utils.test.sol new file mode 100644 index 000000000..54f4e8fbf --- /dev/null +++ b/test/common/lib/signature-utils.test.sol @@ -0,0 +1,151 @@ +// SPDX-License-Identifier: MIT +pragma solidity >=0.4.24 <0.9.0; + +import "forge-std/Test.sol"; +import {ECDSA} from "contracts/common/lib/ECDSA.sol"; +import { SignatureUtils } from "contracts/common/lib/SignatureUtils.sol"; + +contract ExposedSignatureUtils { + function isValidSignature( + address signer, + bytes32 msgHash, + uint8 v, + bytes32 r, + bytes32 s + ) public view returns (bool) { + return SignatureUtils.isValidSignature(signer, msgHash, v, r, s); + } + + function hasCode(address addr) public view returns (bool) { + return SignatureUtils._hasCode(addr); + } +} + +contract SignatureUtilsTest is Test { + ExposedSignatureUtils public sigUtil; + + function ethMessageHash(string memory message) internal pure returns (bytes32) { + return keccak256( + abi.encodePacked("\x19Ethereum Signed Message:\n32", message) + ); + } + + function setUp() public { + sigUtil = new ExposedSignatureUtils(); + } + + function testHasCodeFalse() public { + address eoa = 0xbeFA429d57cD18b7F8A4d91A2da9AB4AF05d0FBe; + address con = address(new ExposedSignatureUtils()); + + assertEq(sigUtil.hasCode(eoa), false); + assertEq(sigUtil.hasCode(con), true); + } + + function testEoaIsValidSignature() public { + uint256 eoaPk = 1; + address eoa = vm.addr(eoaPk); + bytes32 hash = keccak256("TEST"); + (uint8 v, bytes32 r, bytes32 s) = vm.sign(eoaPk, hash); + + assertEq(sigUtil.isValidSignature(eoa, hash, v, r, s), true); + } + + function testEoaIsValidSignatureFuzz(uint256 eoa_num) public { + // Private key must be less than the secp256k1 curve order + vm.assume(eoa_num < 115792089237316195423570985008687907852837564279074904382605163141518161494337); + + //Private key cannot be zero + vm.assume(eoa_num > 0); + + bytes32 hash = keccak256("TEST"); + (uint8 v, bytes32 r, bytes32 s) = vm.sign(eoa_num, hash); + + assertEq(sigUtil.isValidSignature(vm.addr(eoa_num), hash, v, r, s), true); + } + + function testIsValidSignatureFuzzMessage(bytes memory data) public { + uint256 eoaPk = 1; + address eoa = vm.addr(eoaPk); + bytes32 hash = keccak256(data); + (uint8 v, bytes32 r, bytes32 s) = vm.sign(eoaPk, hash); + + // Regardless of the message, it should always validate + assertEq(sigUtil.isValidSignature(eoa, hash, v, r, s), true); + } + + function testIsValidSignatureFuzzV(uint8 _v) public { + uint256 eoaPk = 1; + address eoa = vm.addr(eoaPk); + bytes32 hash = keccak256("TEST"); + (uint8 v, bytes32 r, bytes32 s) = vm.sign(eoaPk, hash); + + // Test to see if we can get valid signatures without a valid V + if (v == _v) { + assertEq(sigUtil.isValidSignature(eoa, hash, _v, r, s), true); + } else if (27 == _v) { + assertEq(sigUtil.isValidSignature(eoa, hash, _v, r, s), false); + } else { + vm.expectRevert(bytes("ECDSA: invalid signature")); + sigUtil.isValidSignature(eoa, hash, _v, r, s); + } + } + + function testIsValidSignatureFuzzR(bytes32 _r) public { + uint256 eoaPk = 1; + address eoa = vm.addr(eoaPk); + bytes32 hash = keccak256("TEST"); + (uint8 v, bytes32 r, bytes32 s) = vm.sign(eoaPk, hash); + + // Test to see if we can get valid signatures regardless of what R is + if (r == _r) { + assertEq(sigUtil.isValidSignature(eoa, hash, v, _r, s), true); + } else if (ecrecover(hash, v, _r, s) == address(0)) { + vm.expectRevert(bytes("ECDSA: invalid signature")); + sigUtil.isValidSignature(eoa, hash, v, _r, s); + } else { + assertEq(sigUtil.isValidSignature(eoa, hash, v, _r, s), false); + } + } + + function testIsValidSignatureFuzzS(bytes32 _s) public { + uint256 eoaPk = 1; + address eoa = vm.addr(eoaPk); + bytes32 hash = keccak256("TEST"); + (uint8 v, bytes32 r, bytes32 s) = vm.sign(eoaPk, hash); + + // Test to see if we can get valid signatures regardless of what S is + if (s == _s) { + assertEq(sigUtil.isValidSignature(eoa, hash, v, r, _s), true); + } else if (uint256(_s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) { + vm.expectRevert(bytes("ECDSA: invalid signature 's' value")); + sigUtil.isValidSignature(eoa, hash, v, r, _s); + } else if (ecrecover(hash, v, r, _s) == address(0)) { + vm.expectRevert(bytes("ECDSA: invalid signature")); + sigUtil.isValidSignature(eoa, hash, v, r, _s); + } else { + assertEq(sigUtil.isValidSignature(eoa, hash, v, r, _s), false); + } + } + + function testIsValidSignatureWrongSigner(uint256 rogueSigner) public { + // Ignore the 0 case for rogue signer + vm.assume(rogueSigner != 0); + + // Ignore signers above secp256k1 curve order + vm.assume(rogueSigner < 115792089237316195423570985008687907852837564279074904382605163141518161494337); + + uint256 eoaPk = 1; + address eoa = vm.addr(eoaPk); + address eoa2 = vm.addr(rogueSigner); + + bytes32 hash = keccak256("TEST"); + (uint8 v, bytes32 r, bytes32 s) = vm.sign(eoaPk, hash); + + if (eoa == eoa2) { + assertEq(sigUtil.isValidSignature(eoa2, hash, v, r, s), true); + } else { + assertEq(sigUtil.isValidSignature(eoa2, hash, v, r, s), false); + } + } +} \ No newline at end of file diff --git a/test/helpers/utils.js b/test/helpers/utils.js index c4906fd20..736cdf7da 100644 --- a/test/helpers/utils.js +++ b/test/helpers/utils.js @@ -68,6 +68,12 @@ function hex(n, byteLen = undefined) { return byteLen === undefined ? s : s.padStart(byteLen * 2, '0') } +function hexPaddedToByte(n) { + const s = n.toString(16) + const prefix = s.length % 2 ? '0x0' : '0x' + return prefix + s +} + function strip0x(s) { return s.substr(0, 2) === '0x' ? s.substr(2) : s } @@ -185,6 +191,7 @@ module.exports = { hexSplit, toBN, hex, + hexPaddedToByte, strip0x, bufferFromHexString, hexStringFromBuffer, diff --git a/yarn.lock b/yarn.lock index ae2e949a9..70a6c45de 100644 --- a/yarn.lock +++ b/yarn.lock @@ -3739,7 +3739,7 @@ __metadata: "@babel/preset-env": ^7.11.5 "@babel/register": ^7.12.1 "@nomiclabs/hardhat-ethers": ^2.0.4 - "@nomiclabs/hardhat-etherscan": ^2.1.8 + "@nomiclabs/hardhat-etherscan": ^3.1.7 "@nomiclabs/hardhat-ganache": ^2.0.1 "@nomiclabs/hardhat-truffle5": ^2.0.2 "@nomiclabs/hardhat-web3": ^2.0.0 @@ -4266,20 +4266,23 @@ __metadata: languageName: node linkType: hard -"@nomiclabs/hardhat-etherscan@npm:^2.1.8": - version: 2.1.8 - resolution: "@nomiclabs/hardhat-etherscan@npm:2.1.8" +"@nomiclabs/hardhat-etherscan@npm:^3.1.7": + version: 3.1.7 + resolution: "@nomiclabs/hardhat-etherscan@npm:3.1.7" dependencies: "@ethersproject/abi": ^5.1.2 "@ethersproject/address": ^5.0.2 - cbor: ^5.0.2 + cbor: ^8.1.0 + chalk: ^2.4.2 debug: ^4.1.1 fs-extra: ^7.0.1 - node-fetch: ^2.6.0 + lodash: ^4.17.11 semver: ^6.3.0 + table: ^6.8.0 + undici: ^5.14.0 peerDependencies: hardhat: ^2.0.4 - checksum: 9f674b9353a248472f5cda9e3b76f5e12add3c0e680444da460279d74e9110fc7d6878a3b41d7a5e5acc7638398edc0b53ecb6625d2e4576b6a4f48f817a67cf + checksum: 5e1bfb4bdfbdc3a67e103c1bc5232f91c76372f835c6993f4d7099710f56f13f3fee59eeda6b46016dd2ccc1e73dbe9970db4f363c77e4a294fdc970c92aa5f0 languageName: node linkType: hard @@ -5914,6 +5917,18 @@ __metadata: languageName: node linkType: hard +"ajv@npm:^8.0.1": + version: 8.12.0 + resolution: "ajv@npm:8.12.0" + dependencies: + fast-deep-equal: ^3.1.1 + json-schema-traverse: ^1.0.0 + require-from-string: ^2.0.2 + uri-js: ^4.2.2 + checksum: 3bdee674686025195d85ab010d40b1834cad24ad128e457371ec6e597bba66674ee3ad3e5673c3ffbdfef24e616ae06a5e078006b1c20f49726788b0e1631db3 + languageName: node + linkType: hard + "alphanum-sort@npm:^1.0.0": version: 1.0.2 resolution: "alphanum-sort@npm:1.0.2" @@ -8547,13 +8562,12 @@ __metadata: languageName: node linkType: hard -"cbor@npm:^5.0.2": - version: 5.1.0 - resolution: "cbor@npm:5.1.0" +"cbor@npm:^8.1.0": + version: 8.1.0 + resolution: "cbor@npm:8.1.0" dependencies: - bignumber.js: ^9.0.0 - nofilter: ^1.0.4 - checksum: b8257e15ba263440af093d5e4577bf7bdc6e65e952426d42161983866687cbd92ac5a36246cc95c48605b4d199d3343a8802994cc862ff4a2c609bb885796219 + nofilter: ^3.1.0 + checksum: 25ff71bd10e6f81e14ddaba17e7c1f58190b098eab2900f90806d9a7cfe6a43a0b3eb27b674b4507920a243e21307e587c17a190f2adb11c884ca3dce19fe923 languageName: node linkType: hard @@ -17980,6 +17994,13 @@ fsevents@~2.3.2: languageName: node linkType: hard +"json-schema-traverse@npm:^1.0.0": + version: 1.0.0 + resolution: "json-schema-traverse@npm:1.0.0" + checksum: 7a230bcd927f5bf41b33a822121730a225ac287e14d7e8abc94f4cbc36743f6e09455549abaada7029844f7e88a9fd693a023ec76296df17488746acb1e5a388 + languageName: node + linkType: hard + "json-schema@npm:0.2.3": version: 0.2.3 resolution: "json-schema@npm:0.2.3" @@ -18978,6 +18999,13 @@ fsevents@~2.3.2: languageName: node linkType: hard +"lodash.truncate@npm:^4.4.2": + version: 4.4.2 + resolution: "lodash.truncate@npm:4.4.2" + checksum: b1b0d7d993bb73d0032fe909d4523a836b6aa91566fa88ff78c3eac008bd3d3b2ba0f2e8381d7f906b1d6913a64982f34bea95dd556355c0d418bfddf3ab7b06 + languageName: node + linkType: hard + "lodash.uniq@npm:^4.5.0": version: 4.5.0 resolution: "lodash.uniq@npm:4.5.0" @@ -20896,10 +20924,10 @@ fsevents@~2.3.2: languageName: node linkType: hard -"nofilter@npm:^1.0.4": - version: 1.0.4 - resolution: "nofilter@npm:1.0.4" - checksum: af46b9255190250595702f75623d3de4bc5e07822d16d2b3b2e47dc892825273efedc326710e884689e4a74bfc9863c61ca8d6b4115c7bcac40a4e787ae5a8c9 +"nofilter@npm:^3.1.0": + version: 3.1.0 + resolution: "nofilter@npm:3.1.0" + checksum: d36f688b878b4a38305cfe66b33b129ef6d1737b55ad40cc917259a1d44c6e80b2b4ab81c3e86504faa0a3bcb1dad40b265026765140823c496210ee8fa1c91e languageName: node linkType: hard @@ -24280,7 +24308,7 @@ fsevents@~2.3.2: languageName: node linkType: hard -"require-from-string@npm:^2.0.0": +"require-from-string@npm:^2.0.0, require-from-string@npm:^2.0.2": version: 2.0.2 resolution: "require-from-string@npm:2.0.2" checksum: 74fc30353e5d526879b28d480c3f25ca95e9c22dfe7ac10ca0650e03407b3aeed352ff8ca706ea145617b6482a582e4a3bd65a884fc50133ebe586d47fa085c6 @@ -26634,6 +26662,19 @@ resolve@^1.22.1: languageName: node linkType: hard +"table@npm:^6.8.0": + version: 6.8.1 + resolution: "table@npm:6.8.1" + 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 + checksum: de5a4f44c7b4fac264b0ca9e916aefdabdbef50d350cfeab2338c75c86d95ad7481ffd1599467cbed50c31a1af2a9f7c0d0eee222db11d0f0c4165e88169bdba + languageName: node + linkType: hard + "tape@npm:^4.4.0, tape@npm:^4.6.3": version: 4.13.3 resolution: "tape@npm:4.13.3"