diff --git a/pkg/permissions/CHANGELOG.md b/pkg/permissions/CHANGELOG.md new file mode 100644 index 0000000000..1512c42162 --- /dev/null +++ b/pkg/permissions/CHANGELOG.md @@ -0,0 +1,3 @@ +# Changelog + +## Unreleased diff --git a/pkg/permissions/README.md b/pkg/permissions/README.md new file mode 100644 index 0000000000..536ad1df64 --- /dev/null +++ b/pkg/permissions/README.md @@ -0,0 +1,14 @@ +# Balancer + +# Balancer V2 Permissions + +[![NPM Package](https://img.shields.io/npm/v/@balancer-labs/v2-permissions.svg)](https://www.npmjs.org/package/@balancer-labs/v2-permissions) +[![GitHub Repository](https://img.shields.io/badge/github-deployments-lightgrey?logo=github)](https://github.com/balancer-labs/balancer-v2-monorepo/tree/master/pkg/permissions) + +## Overview + +### Installation + +```console +$ npm install @balancer-labs/v2-permissions +``` diff --git a/pkg/permissions/labelled-accounts/mainnet.json b/pkg/permissions/labelled-accounts/mainnet.json new file mode 100644 index 0000000000..47f1a5bb43 --- /dev/null +++ b/pkg/permissions/labelled-accounts/mainnet.json @@ -0,0 +1,13 @@ +{ + "0x10a19e7ee7d7f8a52822f6817de8ea18204f2e4f": "DAO_MULTISIG", + "0x166f54f44f271407f24aa1be415a730035637325": "BALLER_OPS_MULTISIG", + "0x75a52c0e32397a3fc0c052e2ceb3479802713cf4": "LINEAR_POOL_CONTROLLER_BALLER_MULTISIG", + "0xf4a80929163c5179ca042e1b292f5efbbe3d89e6": "SWAP_FEE_CONTROLLER_BALLER_MULTISIG", + "0xc38c5f97b34e175ffd35407fc91a937300e33860": "LM_MULTISIG", + "0x7c68c42de679ffb0f16216154c996c354cf1161b": "TREASURY_MULTISIG", + "0xa29f61256e948f3fb707b4b3b138c5ccb9ef9888": "EMERGENCY_SUBDAO_MULTISIG", + "0x02f35dA6A02017154367Bc4d47bb6c7D06C7533B": "BLABS_OPS_MULTISIG", + "0xd2eb7bd802a7ca68d9acd209bec4e664a9abdd7b": "BLABS_VEBAL_MULTISIG", + "0xe4a8ed6c1d8d048bd29a00946bfcf2db10e7923b": "GAUNTLET_FEE_SETTER", + "0xc92e8bdf79f0507f65a392b0ab4667716bfe0110": "GNOSIS_PROTOCOL_RELAYER" +} diff --git a/pkg/permissions/labelled-accounts/readme.md b/pkg/permissions/labelled-accounts/readme.md new file mode 100644 index 0000000000..daf27ac714 --- /dev/null +++ b/pkg/permissions/labelled-accounts/readme.md @@ -0,0 +1,5 @@ +# Labelled Accounts + +This directory contains accounts which hold permissions over areas of the the Balancer Protocol which aren't deployed as part of the Balancer Protocol itself (as these are tracked in the `@balancer-labs/v2-deployments` package). These are generally Safe multisigs or externally developed contracts. + +Balancer DAO maintains documentation on the list of Safe multisigs it uses on [Notion](https://quark-ceres-740.notion.site/Multisig-List-Guidelines-402e18cff13e4f2fa571bc14ed007546) diff --git a/pkg/permissions/package.json b/pkg/permissions/package.json new file mode 100644 index 0000000000..7f3e6e610c --- /dev/null +++ b/pkg/permissions/package.json @@ -0,0 +1,41 @@ +{ + "name": "@balancer-labs/v2-permissions", + "version": "0.1.0", + "description": "Registry of permissions granted on Balancer's Authorizer contract", + "license": "GPL-3.0-only", + "homepage": "https://github.com/balancer-labs/balancer-v2-monorepo/tree/master/pkg/permissions#readme", + "repository": { + "type": "git", + "url": "https://github.com/balancer-labs/balancer-v2-monorepo.git", + "directory": "pkg/permissions" + }, + "bugs": { + "url": "https://github.com/balancer-labs/balancer-v2-monorepo/issues" + }, + "main": "dist/index.js", + "module": "dist/index.esm.js", + "browser": "dist/index.umd.js", + "typings": "dist/index.d.ts", + "files": [ + "dist/" + ], + "scripts": { + "permissions:update": "yarn ts-node scripts/getPermissionedAccounts.ts", + "permissions:map": "yarn ts-node scripts/getCallableFunctions.ts", + "permissions:unknown": "yarn ts-node scripts/getUnrecognisedActionIds.ts", + "lint": "eslint . --ext .ts --ignore-path ../../.eslintignore --max-warnings 0", + "test": "hardhat test ./**/test/*.ts" + }, + "devDependencies": { + "@types/node": "^14.14.31", + "@typescript-eslint/eslint-plugin": "^5.41.0", + "@typescript-eslint/parser": "^5.41.0", + "chalk": "^4.1.2", + "eslint": "^8.26.0", + "eslint-plugin-prettier": "^4.2.1", + "node-fetch": "^2.6.7", + "prettier": "^2.7.1", + "ts-node": "^10.9.1", + "typescript": "^4.0.2" + } +} diff --git a/pkg/permissions/permissions/actionIds.json b/pkg/permissions/permissions/actionIds.json new file mode 100644 index 0000000000..56532e75d6 --- /dev/null +++ b/pkg/permissions/permissions/actionIds.json @@ -0,0 +1,129 @@ +{ + "0x02f35da6a02017154367bc4d47bb6c7d06c7533b": [ + "0x5bcdcc8d471eea0c6345d3dd65ad4997a32054e1e0672b780a9b6c36df0166a3", + "0x8b2c6767a8c426408240798cd82acf7ba6091320da176d0b1ab39e99fd5c409d", + "0x3f63974a377ba4713661ede455bceda6686a0395f8b8ed8701ad1f13bb926c4d", + "0xc89b780137460c1010bc938658c3b615990dd348e27ff1d095be12e6fe617f64", + "0xbdac75576424959cffc7f91ec4674a05fd1c62bedcbcbce9dab046c58c881950" + ], + "DAO_MULTISIG": [ + "0xac0fcdc4520d7bde1c58bbefd7c8dd39aaf382a20c27991134c14fe63d2c96f3", + "0x43cd68bd7db0472f3fac100d3f402a603c8ab62e816feff20dbe3ec6c6e61b89", + "0xec1d467d9ab03a0079c22a89037209f5763aec973897ea763e2cf25d71a5f12e", + "0x4907aec017cb19a28528e722251b40fd7c5eadd4f4a0f0c6a9bca9888f8a0b7f", + "0x0000000000000000000000000000000000000000000000000000000000000000", + "0x453b670b2708db1ba5df1da1d48add0564558624efac456e43e9c9fff99d51af", + "0xb28b769768735d011b267f781c3be90bce51d5059ba015bc7a28b3e882fb2083" + ], + "0x170027069fd114bff2f57b0fc796df93290c02a6": [ + "0x3c7de1d8a207c7901ec612f9f0f50957da016911a50d5c22bbe5c9f4f3392d95", + "0xb5593fe09464f360ecf835d5b9319ce69900ae1b29d13844b73c250b1f5f92fb" + ], + "0x239e55f427d44c3cc793f49bfb507ebe76638a2b": [ + "0xdddd30813da50fda5faba482fd2937d0c6165d2faf027d3dfbd1554f3d7d47ff" + ], + "0x2536dfeecb7a0397cf98edada8486254533b1afa": [ + "0x78ad1b68d148c070372f8643c4648efbb63c6a8a338f3c24714868e791367653", + "0x7b8a1d293670124924a0f532213753b89db10bde737249d4540e9a03657d1aff", + "0x1282ab709b2b70070f829c46bc36f76b32ad4989fecb2fcb09a1b3ce00bbfc30", + "0x0014a06d322ff07fcc02b12f93eb77bb76e28cdee4fc0670b9dec98d24bbfec8", + "0xeba777d811cd36c06d540d7ff2ed18ed042fd67bbf7c9afcf88c818c7ee6b498", + "0xc149e88b59429ded7f601ab52ecd62331cac006ae07c16543439ed138dcb8d34" + ], + "0x2ffb7b215ae7f088ec2530c7aa8e1b24e398f26a": [ + "0xf49d7ffb5922642adc9f29cfb52b2214e81e0b0e54e9cd1e9f70439f0011f368" + ], + "0x5ef4c5352882b10893b70dbcaa0c000965bd23c5": [ + "0xb2b6e48fa160a7c887d9d7a68b6a9bb9d47d4953d33e07f3a39e175d75e97796" + ], + "LINEAR_POOL_CONTROLLER_BALLER_MULTISIG": [ + "0xe4814396e9db5314024c424f43d6a129829efad6c545df373b226431cbcadbd3", + "0x2256d78edacd087428321791a930d4f9fd7acf56e8862187466f1caf179c1a08", + "0x1e3ce02b9d143fb44dc00c908d6b454553cf1c8c48e54090fa1f5fdd18a8e6b9" + ], + "TREASURY_MULTISIG": [ + "0x826ac7ce861f2a54e071e6c724653757fdd1259804eb1ca7f040aa1cd09923fe" + ], + "0x886a3ec7bcc508b8795990b60fa21f85f9db7948": [ + "0x7b8a1d293670124924a0f532213753b89db10bde737249d4540e9a03657d1aff", + "0x78ad1b68d148c070372f8643c4648efbb63c6a8a338f3c24714868e791367653", + "0x1282ab709b2b70070f829c46bc36f76b32ad4989fecb2fcb09a1b3ce00bbfc30", + "0xeba777d811cd36c06d540d7ff2ed18ed042fd67bbf7c9afcf88c818c7ee6b498", + "0x0014a06d322ff07fcc02b12f93eb77bb76e28cdee4fc0670b9dec98d24bbfec8", + "0xc149e88b59429ded7f601ab52ecd62331cac006ae07c16543439ed138dcb8d34" + ], + "0x97207b095e4d5c9a6e4cfbfcd2c3358e03b90c4a": [ + "0xb28b769768735d011b267f781c3be90bce51d5059ba015bc7a28b3e882fb2083", + "0xbe2a180d5cc5d803a8eec4cea569989fc1c593d7eeadd1f262f360a68b0e842e" + ], + "EMERGENCY_SUBDAO_MULTISIG": [ + "0x8d329099a8220fbd27ff3cf304a4cb1dae32335654ec5115c3a643ac0e623418", + "0xc0d91e75884e4ce70f827133990e1c6ee501b41ad3096d25bce3c04d2976c3e7", + "0xa738fa584fff6afe4e319db36f7f5270924047e5e2c04a1712cbfc082e3fd078", + "0x367e95c6cc9f3041f3c6ee21b06ef8992a82318a6b2adbbfb6af3ee601769a30", + "0x8186826062c35b40965262f49014e5ca45b7064fba48b12107613bce22571a99", + "0xfef90c64be79cb170a20e526196e7c8f2f37f441ae85c945c18a91a64777d309", + "0x84163b5cca492497c5fa264018819677910a8022689972cc54566d8667dbce68", + "0xa5a62b55fdf9496f8e1b3feba479423a4349b385bd444f893b3cd4cf9387ce3f", + "0x0d9dbee65c669ef9d726a603957e4a610b40b2662eba759efbedfe87216ec751" + ], + "0xac9f49ef3ab0bbc929f7b1bb0a17e1fca5786251": [ + "0xeba777d811cd36c06d540d7ff2ed18ed042fd67bbf7c9afcf88c818c7ee6b498", + "0x7b8a1d293670124924a0f532213753b89db10bde737249d4540e9a03657d1aff", + "0x1282ab709b2b70070f829c46bc36f76b32ad4989fecb2fcb09a1b3ce00bbfc30", + "0x78ad1b68d148c070372f8643c4648efbb63c6a8a338f3c24714868e791367653", + "0xc149e88b59429ded7f601ab52ecd62331cac006ae07c16543439ed138dcb8d34", + "0x0014a06d322ff07fcc02b12f93eb77bb76e28cdee4fc0670b9dec98d24bbfec8" + ], + "LM_MULTISIG": [ + "0x5c62111a5fb2cd09521d2805fb5080f8db7f341691a1e38c34a5ededb8f8bfd3", + "0x590e300e371ba81baff1c912e578fdecbfa490f39994607a18ee692ab942f846", + "0x77238124388523487417c8ad8cec25726833e50ca5cab74a4924470fee49ae5d", + "0x72c4c054ad03b4f5f0ba716c30d74c6f27fafb105c850cb59e2b6fec32a42f2f", + "0x5dce9596402d216d8b1fa2b9f8e18b0dc1b5c81f96e0827c6cc83eba6e2205d4", + "0xc63b7b73283233470a85ad7ec28f772b7571c0f6ba90d506999809c2e25a7da6", + "0x82c7bc265be8c8190319e29a314f8c32e62b98bbc9c39defff06a42b34557191", + "0xeb223764963bceacbb06d72a3697801c2460ddf95b2ec410d2641d69249d466f" + ], + "GNOSIS_PROTOCOL_RELAYER": [ + "0xeba777d811cd36c06d540d7ff2ed18ed042fd67bbf7c9afcf88c818c7ee6b498", + "0x1282ab709b2b70070f829c46bc36f76b32ad4989fecb2fcb09a1b3ce00bbfc30" + ], + "BLABS_VEBAL_MULTISIG": [ + "0x802db13f34b039826402f87748c166a94c8130bf894f8af7e1144c874b36b76e", + "0xbfa133e7b0ebe7bf8b3f11a17a38c0f4492b428e4fb7fc8b509da63189247b06", + "0x79922681fd17c90b4f3409d605f5b059ffcbcef7b5440321ae93b87f3b5c1c78" + ], + "0xdcdbf71a870cc60c6f9b621e28a7d3ffd6dd4965": [ + "0x1282ab709b2b70070f829c46bc36f76b32ad4989fecb2fcb09a1b3ce00bbfc30", + "0xc149e88b59429ded7f601ab52ecd62331cac006ae07c16543439ed138dcb8d34", + "0x7b8a1d293670124924a0f532213753b89db10bde737249d4540e9a03657d1aff", + "0x78ad1b68d148c070372f8643c4648efbb63c6a8a338f3c24714868e791367653", + "0xeba777d811cd36c06d540d7ff2ed18ed042fd67bbf7c9afcf88c818c7ee6b498" + ], + "GAUNTLET_FEE_SETTER": [ + "0x36e042f590f2c5d0d8959cc373c8b1681f70f84e9656be8dd0eae652e01de4eb", + "0x78e9adfe5f05d7114a59d0870d78971192f871f57bb36e2aff2edbe75d425844", + "0xc065d550fa98abc242b6baf98e7b2063590675f1ddd81bdb9ea8d8f5c5d52f98", + "0x6c3a14f10cbcc5a3f4d0e4e8ad279e7a842735ab188e2b13fb84c6542cc3320c", + "0xcf5e03a737e4f5ba6d13e23f893a1e0255b362d8ce22e9568e1565fcf92789c7", + "0x7b09f4b61ccfe85436161b0223489b187d9f9158c542b5e6105df147afc78aca", + "0x15d3918ca8f9895d8906a780f5f402d32707bada7b1b5e7b21b7351257103a35", + "0x3697d13ee45583cf9c2c64a978ab5886bcd07ec2b851efbea2fced982b8f9596" + ], + "0xeb151668006cd04dadd098afd0a82e78f77076c3": [ + "0x96932b9555c49f1a3a7fb90d4b1ea803f16e02e14a6b942202a84e5f6b65d5c4" + ], + "SWAP_FEE_CONTROLLER_BALLER_MULTISIG": [ + "0x11562115fbcf4955e097732f59969867f1cb458a8cbd648231b0ffae14c800de", + "0xcad4ec1d64970817394bee6f75af4645fb72ba5b88902c4c155ce82aab0a3a5a", + "0xf8ab8bdb4497d157053d2f796e50c33e6fff3d586b6db6880ab12eff1d907b2b", + "0xf27148d3f1da6319bd754a52acd00b2fc3fa6474241d2398c6d58e8ac0cd9539", + "0xef008574ca41f2b6033a54a73ad6adc382165acd85b6f76f8456d9946b299a16", + "0x94611f33019f04ed070e076bbacb9ff5c5fe03d7184bef4026e1ee669d3b623e", + "0x7fad14fae895c80a37148957909942740cfbc0ddc5676b975d9893577ba7cd17", + "0xc30e3272c4933a085c95b84fca44f1a9b3d43e3e560b7b1fac0a6b2c9bbda16f", + "0xd4f0c40da2129d4b1aba541e693e03b92a323a66f649257a258fe6e4ea331b52", + "0xe5a9dede86018292d3cd547db825db489579eedbf2eebd3694ab93e912c1fae5" + ] +} \ No newline at end of file diff --git a/pkg/permissions/permissions/functions.json b/pkg/permissions/permissions/functions.json new file mode 100644 index 0000000000..55c8c7aa7a --- /dev/null +++ b/pkg/permissions/permissions/functions.json @@ -0,0 +1,612 @@ +{ + "0x02f35da6a02017154367bc4d47bb6c7d06c7533b": [ + { + "taskId": "20211208-aave-linear-pool", + "contractName": "AaveLinearPool", + "signature": "setPaused(bool)", + "useAdaptor": false + }, + { + "taskId": "20220413-arbitrum-root-gauge-factory", + "contractName": "ArbitrumRootGaugeFactory", + "signature": "setArbitrumFees(uint64,uint64,uint64)", + "useAdaptor": false + }, + { + "taskId": "20220325-gauge-controller", + "contractName": "VotingEscrow", + "signature": "checkpoint()", + "useAdaptor": true + }, + { + "taskId": "20220325-gauge-controller", + "contractName": "GaugeController", + "signature": "checkpoint()", + "useAdaptor": true + }, + { + "taskId": "20220325-single-recipient-gauge-factory", + "contractName": "SingleRecipientGauge", + "signature": "checkpoint()", + "useAdaptor": true + }, + { + "taskId": "20220413-arbitrum-root-gauge-factory", + "contractName": "ArbitrumRootGauge", + "signature": "checkpoint()", + "useAdaptor": true + }, + { + "taskId": "20220413-polygon-root-gauge-factory", + "contractName": "PolygonRootGauge", + "signature": "checkpoint()", + "useAdaptor": true + }, + { + "taskId": "20220420-fee-distributor", + "contractName": "FeeDistributor", + "signature": "checkpoint()", + "useAdaptor": true + }, + { + "taskId": "20220628-optimism-root-gauge-factory", + "contractName": "OptimismRootGauge", + "signature": "checkpoint()", + "useAdaptor": true + }, + { + "taskId": "20220714-fee-distributor-v2", + "contractName": "FeeDistributor", + "signature": "checkpoint()", + "useAdaptor": true + }, + { + "taskId": "20220823-optimism-root-gauge-factory-v2", + "contractName": "OptimismRootGauge", + "signature": "checkpoint()", + "useAdaptor": true + }, + { + "taskId": "20220823-arbitrum-root-gauge-factory-v2", + "contractName": "ArbitrumRootGauge", + "signature": "checkpoint()", + "useAdaptor": true + }, + { + "taskId": "20220823-polygon-root-gauge-factory-v2", + "contractName": "PolygonRootGauge", + "signature": "checkpoint()", + "useAdaptor": true + }, + { + "taskId": "20220628-optimism-root-gauge-factory", + "contractName": "OptimismRootGaugeFactory", + "signature": "setOptimismGasLimit(uint32)", + "useAdaptor": false + }, + { + "taskId": "20211208-stable-phantom-pool", + "contractName": "StablePhantomPool", + "signature": "setPaused(bool)", + "useAdaptor": false + } + ], + "DAO_MULTISIG": [ + { + "taskId": "20220325-ve-delegation", + "contractName": "VotingEscrowDelegationProxy", + "signature": "setDelegation(address)", + "useAdaptor": false + }, + { + "taskId": "20220420-smart-wallet-checker", + "contractName": "SmartWalletChecker", + "signature": "denylistAddress(address)", + "useAdaptor": false + }, + { + "taskId": "20220325-mainnet-gauge-factory", + "contractName": "LiquidityGaugeV5", + "signature": "killGauge()", + "useAdaptor": true + }, + { + "taskId": "20220325-single-recipient-gauge-factory", + "contractName": "SingleRecipientGauge", + "signature": "killGauge()", + "useAdaptor": true + }, + { + "taskId": "20220413-arbitrum-root-gauge-factory", + "contractName": "ArbitrumRootGauge", + "signature": "killGauge()", + "useAdaptor": true + }, + { + "taskId": "20220413-polygon-root-gauge-factory", + "contractName": "PolygonRootGauge", + "signature": "killGauge()", + "useAdaptor": true + }, + { + "taskId": "20220628-optimism-root-gauge-factory", + "contractName": "OptimismRootGauge", + "signature": "killGauge()", + "useAdaptor": true + }, + { + "taskId": "20220822-mainnet-gauge-factory-v2", + "contractName": "LiquidityGaugeV5", + "signature": "killGauge()", + "useAdaptor": true + }, + { + "taskId": "20220823-optimism-root-gauge-factory-v2", + "contractName": "OptimismRootGauge", + "signature": "killGauge()", + "useAdaptor": true + }, + { + "taskId": "20220823-arbitrum-root-gauge-factory-v2", + "contractName": "ArbitrumRootGauge", + "signature": "killGauge()", + "useAdaptor": true + }, + { + "taskId": "20220823-polygon-root-gauge-factory-v2", + "contractName": "PolygonRootGauge", + "signature": "killGauge()", + "useAdaptor": true + }, + { + "taskId": "20220725-protocol-fee-percentages-provider", + "contractName": "ProtocolFeePercentagesProvider", + "signature": "setFeeTypePercentage(uint256,uint256)", + "useAdaptor": false + }, + { + "taskId": "20220420-smart-wallet-checker", + "contractName": "SmartWalletChecker", + "signature": "allowlistAddress(address)", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "ProtocolFeesCollector", + "signature": "setSwapFeePercentage(uint256)", + "useAdaptor": false + } + ], + "0x170027069fd114bff2f57b0fc796df93290c02a6": [ + { + "taskId": "20210418-weighted-pool", + "contractName": "WeightedPool", + "signature": "setPaused(bool)", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "setPaused(bool)", + "useAdaptor": false + } + ], + "0x239e55f427d44c3cc793f49bfb507ebe76638a2b": [ + { + "taskId": "20220325-balancer-token-admin", + "contractName": "BalancerTokenAdmin", + "signature": "mint(address,uint256)", + "useAdaptor": false + } + ], + "0x2536dfeecb7a0397cf98edada8486254533b1afa": [ + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "joinPool(bytes32,address,address,(address[],uint256[],bytes,bool))", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "swap((bytes32,uint8,address,address,uint256,bytes),(address,bool,address,bool),uint256,uint256)", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "batchSwap(uint8,(bytes32,uint256,uint256,uint256,bytes)[],address[],(address,bool,address,bool),int256[],uint256)", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "setRelayerApproval(address,address,bool)", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "manageUserBalance((uint8,address,uint256,address,address)[])", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "exitPool(bytes32,address,address,(address[],uint256[],bytes,bool))", + "useAdaptor": false + } + ], + "0x2ffb7b215ae7f088ec2530c7aa8e1b24e398f26a": [ + { + "taskId": "20220325-gauge-controller", + "contractName": "GaugeController", + "signature": "add_gauge(address,int128)", + "useAdaptor": true + } + ], + "0x5ef4c5352882b10893b70dbcaa0c000965bd23c5": [ + { + "taskId": "20210418-vault", + "contractName": "ProtocolFeesCollector", + "signature": "withdrawCollectedFees(address[],uint256[],address)", + "useAdaptor": false + } + ], + "LINEAR_POOL_CONTROLLER_BALLER_MULTISIG": [ + { + "taskId": "20211208-stable-phantom-pool", + "contractName": "StablePhantomPool", + "signature": "setTokenRateCacheDuration(address,uint256)", + "useAdaptor": false + }, + { + "taskId": "20211208-aave-linear-pool", + "contractName": "AaveLinearPool", + "signature": "setSwapFeePercentage(uint256)", + "useAdaptor": false + }, + { + "taskId": "20211208-aave-linear-pool", + "contractName": "AaveLinearPool", + "signature": "setTargets(uint256,uint256)", + "useAdaptor": false + } + ], + "TREASURY_MULTISIG": [ + { + "taskId": "20220517-protocol-fee-withdrawer", + "contractName": "ProtocolFeesWithdrawer", + "signature": "withdrawCollectedFees(address[],uint256[],address)", + "useAdaptor": false + } + ], + "0x886a3ec7bcc508b8795990b60fa21f85f9db7948": [ + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "swap((bytes32,uint8,address,address,uint256,bytes),(address,bool,address,bool),uint256,uint256)", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "joinPool(bytes32,address,address,(address[],uint256[],bytes,bool))", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "batchSwap(uint8,(bytes32,uint256,uint256,uint256,bytes)[],address[],(address,bool,address,bool),int256[],uint256)", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "manageUserBalance((uint8,address,uint256,address,address)[])", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "setRelayerApproval(address,address,bool)", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "exitPool(bytes32,address,address,(address[],uint256[],bytes,bool))", + "useAdaptor": false + } + ], + "0x97207b095e4d5c9a6e4cfbfcd2c3358e03b90c4a": [ + { + "taskId": "20210418-vault", + "contractName": "ProtocolFeesCollector", + "signature": "setSwapFeePercentage(uint256)", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "ProtocolFeesCollector", + "signature": "setFlashLoanFeePercentage(uint256)", + "useAdaptor": false + } + ], + "EMERGENCY_SUBDAO_MULTISIG": [ + { + "taskId": "20220817-aave-rebalanced-linear-pool", + "contractName": "AaveLinearPool", + "signature": "pause()", + "useAdaptor": false + }, + { + "taskId": "20220906-composable-stable-pool", + "contractName": "ComposableStablePool", + "signature": "pause()", + "useAdaptor": false + }, + { + "taskId": "20220817-aave-rebalanced-linear-pool", + "contractName": "AaveLinearPool", + "signature": "unpause()", + "useAdaptor": false + }, + { + "taskId": "20220908-weighted-pool-v2", + "contractName": "WeightedPool", + "signature": "pause()", + "useAdaptor": false + }, + { + "taskId": "20220906-composable-stable-pool", + "contractName": "ComposableStablePool", + "signature": "unpause()", + "useAdaptor": false + }, + { + "taskId": "20220517-protocol-fee-withdrawer", + "contractName": "ProtocolFeesWithdrawer", + "signature": "denylistToken(address)", + "useAdaptor": false + }, + { + "taskId": "20220908-weighted-pool-v2", + "contractName": "WeightedPool", + "signature": "unpause()", + "useAdaptor": false + } + ], + "0xac9f49ef3ab0bbc929f7b1bb0a17e1fca5786251": [ + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "manageUserBalance((uint8,address,uint256,address,address)[])", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "swap((bytes32,uint8,address,address,uint256,bytes),(address,bool,address,bool),uint256,uint256)", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "batchSwap(uint8,(bytes32,uint256,uint256,uint256,bytes)[],address[],(address,bool,address,bool),int256[],uint256)", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "joinPool(bytes32,address,address,(address[],uint256[],bytes,bool))", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "exitPool(bytes32,address,address,(address[],uint256[],bytes,bool))", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "setRelayerApproval(address,address,bool)", + "useAdaptor": false + } + ], + "LM_MULTISIG": [ + { + "taskId": "20220628-gauge-adder-v2", + "contractName": "GaugeAdder", + "signature": "addPolygonGauge(address)", + "useAdaptor": false + }, + { + "taskId": "20220628-gauge-adder-v2", + "contractName": "GaugeAdder", + "signature": "addEthereumGauge(address)", + "useAdaptor": false + }, + { + "taskId": "20220325-gauge-adder", + "contractName": "GaugeAdder", + "signature": "addArbitrumGauge(address)", + "useAdaptor": false + }, + { + "taskId": "20220325-gauge-adder", + "contractName": "GaugeAdder", + "signature": "addEthereumGauge(address)", + "useAdaptor": false + }, + { + "taskId": "20220628-gauge-adder-v2", + "contractName": "GaugeAdder", + "signature": "addOptimismGauge(address)", + "useAdaptor": false + }, + { + "taskId": "20220628-gauge-adder-v2", + "contractName": "GaugeAdder", + "signature": "addArbitrumGauge(address)", + "useAdaptor": false + }, + { + "taskId": "20220325-gauge-adder", + "contractName": "GaugeAdder", + "signature": "addPolygonGauge(address)", + "useAdaptor": false + } + ], + "GNOSIS_PROTOCOL_RELAYER": [ + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "manageUserBalance((uint8,address,uint256,address,address)[])", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "batchSwap(uint8,(bytes32,uint256,uint256,uint256,bytes)[],address[],(address,bool,address,bool),int256[],uint256)", + "useAdaptor": false + } + ], + "BLABS_VEBAL_MULTISIG": [ + { + "taskId": "20220325-bal-token-holder-factory", + "contractName": "BALTokenHolder", + "signature": "function withdrawFunds(address,uint256)", + "useAdaptor": false + } + ], + "0xdcdbf71a870cc60c6f9b621e28a7d3ffd6dd4965": [ + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "batchSwap(uint8,(bytes32,uint256,uint256,uint256,bytes)[],address[],(address,bool,address,bool),int256[],uint256)", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "exitPool(bytes32,address,address,(address[],uint256[],bytes,bool))", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "swap((bytes32,uint8,address,address,uint256,bytes),(address,bool,address,bool),uint256,uint256)", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "joinPool(bytes32,address,address,(address[],uint256[],bytes,bool))", + "useAdaptor": false + }, + { + "taskId": "20210418-vault", + "contractName": "Vault", + "signature": "manageUserBalance((uint8,address,uint256,address,address)[])", + "useAdaptor": false + } + ], + "GAUNTLET_FEE_SETTER": [ + { + "taskId": "20211208-stable-phantom-pool", + "contractName": "StablePhantomPool", + "signature": "setSwapFeePercentage(uint256)", + "useAdaptor": false + }, + { + "taskId": "20220908-weighted-pool-v2", + "contractName": "WeightedPool", + "signature": "setSwapFeePercentage(uint256)", + "useAdaptor": false + }, + { + "taskId": "20210418-weighted-pool", + "contractName": "WeightedPool2Tokens", + "signature": "setSwapFeePercentage(uint256)", + "useAdaptor": false + }, + { + "taskId": "20220906-composable-stable-pool", + "contractName": "ComposableStablePool", + "signature": "setSwapFeePercentage(uint256)", + "useAdaptor": false + }, + { + "taskId": "20220609-stable-pool-v2", + "contractName": "StablePool", + "signature": "setSwapFeePercentage(uint256)", + "useAdaptor": false + }, + { + "taskId": "20210624-stable-pool", + "contractName": "StablePool", + "signature": "setSwapFeePercentage(uint256)", + "useAdaptor": false + }, + { + "taskId": "20210727-meta-stable-pool", + "contractName": "MetaStablePool", + "signature": "setSwapFeePercentage(uint256)", + "useAdaptor": false + }, + { + "taskId": "20210418-weighted-pool", + "contractName": "WeightedPool", + "signature": "setSwapFeePercentage(uint256)", + "useAdaptor": false + } + ], + "0xeb151668006cd04dadd098afd0a82e78f77076c3": [ + { + "taskId": "20220325-balancer-token-admin", + "contractName": "BalancerTokenAdmin", + "signature": "activate()", + "useAdaptor": false + } + ], + "SWAP_FEE_CONTROLLER_BALLER_MULTISIG": [ + { + "taskId": "20220609-stable-pool-v2", + "contractName": "StablePool", + "signature": "startAmplificationParameterUpdate(uint256,uint256)", + "useAdaptor": false + }, + { + "taskId": "20220906-composable-stable-pool", + "contractName": "ComposableStablePool", + "signature": "startAmplificationParameterUpdate(uint256,uint256)", + "useAdaptor": false + }, + { + "taskId": "20220817-aave-rebalanced-linear-pool", + "contractName": "AaveLinearPool", + "signature": "setTargets(uint256,uint256)", + "useAdaptor": false + }, + { + "taskId": "20220817-aave-rebalanced-linear-pool", + "contractName": "AaveLinearPool", + "signature": "setSwapFeePercentage(uint256)", + "useAdaptor": false + }, + { + "taskId": "20220906-composable-stable-pool", + "contractName": "ComposableStablePool", + "signature": "stopAmplificationParameterUpdate()", + "useAdaptor": false + }, + { + "taskId": "20220609-stable-pool-v2", + "contractName": "StablePool", + "signature": "stopAmplificationParameterUpdate()", + "useAdaptor": false + } + ] +} \ No newline at end of file diff --git a/pkg/permissions/permissions/unrecognised.json b/pkg/permissions/permissions/unrecognised.json new file mode 100644 index 0000000000..2cc1c6cb0c --- /dev/null +++ b/pkg/permissions/permissions/unrecognised.json @@ -0,0 +1,23 @@ +{ + "DAO_MULTISIG": [ + "0x0000000000000000000000000000000000000000000000000000000000000000" + ], + "EMERGENCY_SUBDAO_MULTISIG": [ + "0x367e95c6cc9f3041f3c6ee21b06ef8992a82318a6b2adbbfb6af3ee601769a30", + "0xfef90c64be79cb170a20e526196e7c8f2f37f441ae85c945c18a91a64777d309" + ], + "LM_MULTISIG": [ + "0x590e300e371ba81baff1c912e578fdecbfa490f39994607a18ee692ab942f846" + ], + "BLABS_VEBAL_MULTISIG": [ + "0x802db13f34b039826402f87748c166a94c8130bf894f8af7e1144c874b36b76e", + "0xbfa133e7b0ebe7bf8b3f11a17a38c0f4492b428e4fb7fc8b509da63189247b06", + "0x79922681fd17c90b4f3409d605f5b059ffcbcef7b5440321ae93b87f3b5c1c78" + ], + "SWAP_FEE_CONTROLLER_BALLER_MULTISIG": [ + "0x11562115fbcf4955e097732f59969867f1cb458a8cbd648231b0ffae14c800de", + "0xf8ab8bdb4497d157053d2f796e50c33e6fff3d586b6db6880ab12eff1d907b2b", + "0x94611f33019f04ed070e076bbacb9ff5c5fe03d7184bef4026e1ee669d3b623e", + "0xd4f0c40da2129d4b1aba541e693e03b92a323a66f649257a258fe6e4ea331b52" + ] +} \ No newline at end of file diff --git a/pkg/permissions/scripts/getCallableFunctions.ts b/pkg/permissions/scripts/getCallableFunctions.ts new file mode 100644 index 0000000000..190a06b0f8 --- /dev/null +++ b/pkg/permissions/scripts/getCallableFunctions.ts @@ -0,0 +1,65 @@ +import fs from 'fs'; +import path from 'path'; +import { ActionIdInfo, ContractActionIdData, getActionIdFunctions, safeReadJsonFile } from '../src/actionIds'; +import { getAccountLabel } from '../src/labelling'; + +const funcsPath = path.join(__dirname, '../../deployments/action-ids/mainnet/action-ids.json'); + +const actionIdFunctions = getActionIdFunctions(safeReadJsonFile>(funcsPath)); + +const specialCasedActionIds: Record = { + // Withdraw permission over BALTokenHolder for veBAL gauge. + '0x79922681fd17c90b4f3409d605f5b059ffcbcef7b5440321ae93b87f3b5c1c78': [ + { + taskId: '20220325-bal-token-holder-factory', + contractName: 'BALTokenHolder', + signature: 'function withdrawFunds(address,uint256)', + useAdaptor: false, + }, + ], +}; + +const ignoredActionIds: string[] = [ + // Root permission. This is being removed as part of the migration to TimelockAuthorizer. + '0x0000000000000000000000000000000000000000000000000000000000000000', + // Unknown permissions. Should be revoked/renounced before migration. + // Granted in https://forum.balancer.fi/t/bip-49-composablestable-and-aavelinearpools-permission-granting/3631 + // but doesn't correspond to anything onchain. + '0x11562115fbcf4955e097732f59969867f1cb458a8cbd648231b0ffae14c800de', + '0xf8ab8bdb4497d157053d2f796e50c33e6fff3d586b6db6880ab12eff1d907b2b', + '0x94611f33019f04ed070e076bbacb9ff5c5fe03d7184bef4026e1ee669d3b623e', + '0xd4f0c40da2129d4b1aba541e693e03b92a323a66f649257a258fe6e4ea331b52', + '0x367e95c6cc9f3041f3c6ee21b06ef8992a82318a6b2adbbfb6af3ee601769a30', + '0xfef90c64be79cb170a20e526196e7c8f2f37f441ae85c945c18a91a64777d309', + // Withdraw permission over unused BALTokenHolders + // LM committee BALTokenHolder + '0x590e300e371ba81baff1c912e578fdecbfa490f39994607a18ee692ab942f846', + // Polygon BALTokenHolder + '0x802db13f34b039826402f87748c166a94c8130bf894f8af7e1144c874b36b76e', + // Arbitrum BALTokenHolder + '0xbfa133e7b0ebe7bf8b3f11a17a38c0f4492b428e4fb7fc8b509da63189247b06', +]; + +const main = async () => { + const inputPath = path.join(__dirname, '../permissions/actionIds.json'); + const userPermissions = safeReadJsonFile(inputPath); + + const callableFunctions = Object.fromEntries( + Object.entries(userPermissions).map(([user, actionIds]) => [ + getAccountLabel(user), + actionIds + .filter((actionId) => !ignoredActionIds.includes(actionId)) + .flatMap((actionId) => { + const actionIdInfo = actionIdFunctions[actionId] ?? specialCasedActionIds[actionId]; + if (actionIdInfo === undefined) throw new Error(`Unknown action id: ${actionId}`); + return actionIdInfo; + }), + ]) + ); + + const filePath = path.join(__dirname, '../permissions/functions.json'); + + fs.writeFileSync(filePath, JSON.stringify(callableFunctions, null, 2)); +}; + +main(); diff --git a/pkg/permissions/scripts/getPermissionedAccounts.ts b/pkg/permissions/scripts/getPermissionedAccounts.ts new file mode 100644 index 0000000000..8931ca2463 --- /dev/null +++ b/pkg/permissions/scripts/getPermissionedAccounts.ts @@ -0,0 +1,21 @@ +import fs from 'fs'; +import path from 'path'; +import { getAccountsWithPermissions } from '../src/accounts'; +import { getAccountLabel } from '../src/labelling'; + +const main = async () => { + const userPermissions = await getAccountsWithPermissions(); + + const flattenedUserPermissions = Object.fromEntries( + userPermissions.map((user) => [ + getAccountLabel(user.id), + user.permissions.flatMap((permission) => permission.action.id), + ]) + ); + + const filePath = path.join(__dirname, '../permissions/actionIds.json'); + + fs.writeFileSync(filePath, JSON.stringify(flattenedUserPermissions, null, 2)); +}; + +main(); diff --git a/pkg/permissions/scripts/getUnrecognisedActionIds.ts b/pkg/permissions/scripts/getUnrecognisedActionIds.ts new file mode 100644 index 0000000000..06155b5ba7 --- /dev/null +++ b/pkg/permissions/scripts/getUnrecognisedActionIds.ts @@ -0,0 +1,28 @@ +import fs from 'fs'; +import path from 'path'; +import { ContractActionIdData, getActionIdFunctions, safeReadJsonFile } from '../src/actionIds'; +import { getAccountLabel } from '../src/labelling'; + +const funcsPath = path.join(__dirname, '../../deployments/action-ids/mainnet/action-ids.json'); + +const actionIdFunctions = getActionIdFunctions(safeReadJsonFile>(funcsPath)); + +const main = async () => { + const inputPath = path.join(__dirname, '../permissions/actionIds.json'); + const userPermissions = safeReadJsonFile(inputPath); + + const callableFunctions = Object.fromEntries( + Object.entries(userPermissions) + .map(([user, actionIds]) => [ + getAccountLabel(user), + actionIds.filter((actionId) => actionIdFunctions[actionId] == undefined), + ]) + .filter(([, actionIds]) => actionIds.length > 0) + ); + + const filePath = path.join(__dirname, '../permissions/unrecognised.json'); + + fs.writeFileSync(filePath, JSON.stringify(callableFunctions, null, 2)); +}; + +main(); diff --git a/pkg/permissions/src/accounts.ts b/pkg/permissions/src/accounts.ts new file mode 100644 index 0000000000..9ddf7ee30e --- /dev/null +++ b/pkg/permissions/src/accounts.ts @@ -0,0 +1,35 @@ +import fetch, { Response } from 'node-fetch'; +import { Account } from './types'; + +export const getAccountsWithPermissions = async (): Promise => { + const response: Response = await fetch('https://api.thegraph.com/subgraphs/name/balancer-labs/balancer-authorizer', { + method: 'POST', + + headers: { + 'Content-Type': 'application/json', + }, + + body: JSON.stringify({ + query: `{ + accounts(first: 500) { + id + permissions { + action { + id + } + } + } + }`, + }), + }); + + const { + data: { accounts }, + }: { data: { accounts: Account[] } } = await response.json(); + + // The subgraph response also includes accounts which once had permissions which have since been revoked. + // We filter these out as they're not interesting. + const accountsWithPermissions = accounts.filter((acc) => acc.permissions.length > 0); + + return accountsWithPermissions; +}; diff --git a/pkg/permissions/src/actionIds.ts b/pkg/permissions/src/actionIds.ts new file mode 100644 index 0000000000..a5beb3097f --- /dev/null +++ b/pkg/permissions/src/actionIds.ts @@ -0,0 +1,39 @@ +import fs from 'fs'; +import path from 'path'; + +export const ACTION_ID_DIRECTORY = path.join(__dirname, '../../deployments/action-ids'); + +export type ContractActionIdData = { useAdaptor: boolean; factoryOutput?: string; actionIds: Record }; +export type ActionIdInfo = { + taskId: string; + contractName: string; + signature: string; + useAdaptor: boolean; +}; + +export function safeReadJsonFile(filePath: string): Record { + const fileExists = fs.existsSync(filePath) && fs.statSync(filePath).isFile(); + + return fileExists ? JSON.parse(fs.readFileSync(filePath).toString()) : {}; +} + +export function getActionIdFunctions( + actionIdFileContents: Record> +): Record { + // Reverse the mapping of `contractName -> signature -> actionId` to be `actionId -> [contractName, signature][]`. + // This simplifies checking for duplicate actionIds to just reading the length of the arrays. + return Object.entries(actionIdFileContents) + .flatMap(([taskId, taskData]) => + Object.entries(taskData).flatMap(([contractName, contractData]) => + Object.entries(contractData.actionIds).map<[string, ActionIdInfo]>(([signature, actionId]) => [ + actionId, + { taskId, contractName, signature, useAdaptor: contractData.useAdaptor }, + ]) + ) + ) + .reduce((acc: Record, [actionId, actionIdInfo]) => { + acc[actionId] = acc[actionId] ?? []; + acc[actionId].push(actionIdInfo); + return acc; + }, {}); +} diff --git a/pkg/permissions/src/labelling.ts b/pkg/permissions/src/labelling.ts new file mode 100644 index 0000000000..5459779afd --- /dev/null +++ b/pkg/permissions/src/labelling.ts @@ -0,0 +1,5 @@ +import labelledAccounts from '../labelled-accounts/mainnet.json'; + +export const getAccountLabel = (address: string): string => { + return labelledAccounts[address as never] ?? address; +}; diff --git a/pkg/permissions/src/types.ts b/pkg/permissions/src/types.ts new file mode 100644 index 0000000000..d07731a356 --- /dev/null +++ b/pkg/permissions/src/types.ts @@ -0,0 +1,16 @@ +export interface Action { + id: string; + permissions: Permission[]; +} + +export interface Permission { + id: string; + account: Account; + action: Action; + txHash: string; +} + +export interface Account { + id: string; + permissions: Permission[]; +} diff --git a/pkg/permissions/tsconfig.json b/pkg/permissions/tsconfig.json new file mode 100644 index 0000000000..e3907a4b8f --- /dev/null +++ b/pkg/permissions/tsconfig.json @@ -0,0 +1,10 @@ +{ + "extends": "../../tsconfig.json", + "compilerOptions": { + "moduleResolution": "node", + "declaration": true, + "outDir": "dist", + "resolveJsonModule": true + }, + "include": ["src", "scripts"] +} diff --git a/yarn.lock b/yarn.lock index 94c63b7429..d203957494 100644 --- a/yarn.lock +++ b/yarn.lock @@ -367,6 +367,23 @@ __metadata: languageName: unknown linkType: soft +"@balancer-labs/v2-permissions@workspace:pkg/permissions": + version: 0.0.0-use.local + resolution: "@balancer-labs/v2-permissions@workspace:pkg/permissions" + dependencies: + "@types/node": ^14.14.31 + "@typescript-eslint/eslint-plugin": ^5.41.0 + "@typescript-eslint/parser": ^5.41.0 + chalk: ^4.1.2 + eslint: ^8.26.0 + eslint-plugin-prettier: ^4.2.1 + node-fetch: ^2.6.7 + prettier: ^2.7.1 + ts-node: ^10.9.1 + typescript: ^4.0.2 + languageName: unknown + linkType: soft + "@balancer-labs/v2-pool-linear@workspace:pkg/pool-linear": version: 0.0.0-use.local resolution: "@balancer-labs/v2-pool-linear@workspace:pkg/pool-linear"