diff --git a/.changeset/honest-shrimps-bake.md b/.changeset/honest-shrimps-bake.md new file mode 100644 index 000000000..8d73f1419 --- /dev/null +++ b/.changeset/honest-shrimps-bake.md @@ -0,0 +1,6 @@ +--- +'@chugsplash/executor': patch +'@chugsplash/plugins': patch +--- + +Integrate etherscan verification into executor diff --git a/packages/core/src/utils.ts b/packages/core/src/utils.ts index 48d7a8598..d8ac4c8f7 100644 --- a/packages/core/src/utils.ts +++ b/packages/core/src/utils.ts @@ -7,7 +7,6 @@ import { ChugSplashRegistryABI, ChugSplashManagerABI, ChugSplashManagerProxyArtifact, - // CHUGSPLASH_REGISTRY_ADDRESS, CHUGSPLASH_REGISTRY_PROXY_ADDRESS, } from '@chugsplash/contracts' @@ -69,14 +68,14 @@ export const writeDeploymentArtifact = ( export const getProxyAddress = ( projectName: string, - target: string + referenceName: string ): string => { // const chugSplashManagerAddress = getChugSplashManagerAddress(projectName) const chugSplashManagerAddress = getChugSplashManagerProxyAddress(projectName) return utils.getCreate2Address( chugSplashManagerAddress, - utils.keccak256(utils.toUtf8Bytes(target)), + utils.keccak256(utils.toUtf8Bytes(referenceName)), utils.solidityKeccak256( ['bytes', 'bytes'], [ diff --git a/packages/executor/src/index.ts b/packages/executor/src/index.ts index d9b875bef..96aeb42aa 100644 --- a/packages/executor/src/index.ts +++ b/packages/executor/src/index.ts @@ -7,8 +7,9 @@ import { CHUGSPLASH_REGISTRY_PROXY_ADDRESS, } from '@chugsplash/contracts' import { ChugSplashBundleState } from '@chugsplash/core' +import { getChainId } from '@eth-optimism/core-utils' -import { compileRemoteBundle } from './utils' +import { compileRemoteBundle, verifyChugSplashConfig } from './utils' type Options = { network: string @@ -116,6 +117,10 @@ export class ChugSplashExecutor extends BaseServiceV2 { deployer: signer, hide: false, }) + + if ((await getChainId(this.state.wallet.provider)) !== 31337) { + await verifyChugSplashConfig(hre, proposalEvent.args.configUri) + } } } } diff --git a/packages/executor/src/utils/compile.ts b/packages/executor/src/utils/compile.ts index 937900f5e..5493d9d8d 100644 --- a/packages/executor/src/utils/compile.ts +++ b/packages/executor/src/utils/compile.ts @@ -1,17 +1,12 @@ import * as dotenv from 'dotenv' -import { SolcBuild } from 'hardhat/types' -import { - TASK_COMPILE_SOLIDITY_GET_SOLC_BUILD, - TASK_COMPILE_SOLIDITY_RUN_SOLCJS, - TASK_COMPILE_SOLIDITY_RUN_SOLC, -} from 'hardhat/builtin-tasks/task-names' -import { add0x } from '@eth-optimism/core-utils' import { CanonicalChugSplashConfig, ChugSplashActionBundle, } from '@chugsplash/core' -import { create } from 'ipfs-http-client' -import { ContractArtifact } from '@chugsplash/plugins' +import { + TASK_CHUGSPLASH_FETCH, + TASK_CHUGSPLASH_BUNDLE_REMOTE, +} from '@chugsplash/plugins' // Load environment variables from .env dotenv.config() @@ -30,89 +25,11 @@ export const compileRemoteBundle = async ( bundle: ChugSplashActionBundle canonicalConfig: CanonicalChugSplashConfig }> => { - const canonicalConfig = await fetchChugSplashConfig(configUri) - const bundle = await hre.run('chugsplash-bundle-remote', { + const canonicalConfig = await hre.run(TASK_CHUGSPLASH_FETCH, { + configUri, + }) + const bundle = await hre.run(TASK_CHUGSPLASH_BUNDLE_REMOTE, { canonicalConfig, }) return { bundle, canonicalConfig } } - -export const getArtifactsFromCanonicalConfig = async ( - hre: any, - canonicalConfig: CanonicalChugSplashConfig -): Promise<{ - [contractName: string]: ContractArtifact -}> => { - const artifacts = {} - for (const source of canonicalConfig.inputs) { - const solcBuild: SolcBuild = await hre.run( - TASK_COMPILE_SOLIDITY_GET_SOLC_BUILD, - { - quiet: true, - solcVersion: source.solcVersion, - } - ) - - let output: any // TODO: Compiler output - if (solcBuild.isSolcJs) { - output = await hre.run(TASK_COMPILE_SOLIDITY_RUN_SOLCJS, { - input: source.input, - solcJsPath: solcBuild.compilerPath, - }) - } else { - output = await hre.run(TASK_COMPILE_SOLIDITY_RUN_SOLC, { - input: source.input, - solcPath: solcBuild.compilerPath, - }) - } - - for (const [sourceName, fileOutput] of Object.entries(output.contracts)) { - for (const [contractName, contractOutput] of Object.entries(fileOutput)) { - artifacts[contractName] = { - bytecode: add0x(contractOutput.evm.bytecode.object), - storageLayout: contractOutput.storageLayout, - contractName, - sourceName, - abi: contractOutput.abi, - sources: output.sources, - immutableReferences: - output.contracts[sourceName][contractName].evm.deployedBytecode - .immutableReferences, - } - } - } - } - return artifacts -} - -// TODO: change file name or add another file -export const fetchChugSplashConfig = async ( - configUri: string -): Promise => { - const projectCredentials = `${process.env.IPFS_PROJECT_ID}:${process.env.IPFS_API_KEY_SECRET}` - const ipfs = create({ - host: 'ipfs.infura.io', - port: 5001, - protocol: 'https', - headers: { - authorization: `Basic ${Buffer.from(projectCredentials).toString( - 'base64' - )}`, - }, - }) - - let config: CanonicalChugSplashConfig - if (configUri.startsWith('ipfs://')) { - const decoder = new TextDecoder() - let data = '' - const stream = await ipfs.cat(configUri.replace('ipfs://', '')) - for await (const chunk of stream) { - // Chunks of data are returned as a Uint8Array. Convert it back to a string - data += decoder.decode(chunk, { stream: true }) - } - config = JSON.parse(data) - } else { - throw new Error('unsupported URI type') - } - return config -} diff --git a/packages/executor/src/utils/etherscan.ts b/packages/executor/src/utils/etherscan.ts index ec3d7824a..65b127fd8 100644 --- a/packages/executor/src/utils/etherscan.ts +++ b/packages/executor/src/utils/etherscan.ts @@ -2,10 +2,17 @@ import assert from 'assert' import { Contract } from 'ethers' import { + CanonicalChugSplashConfig, CompilerInput, getChugSplashManagerProxyAddress, + getProxyAddress, + parseChugSplashConfig, } from '@chugsplash/core' -import { getConstructorArgs } from '@chugsplash/plugins' +import { + getConstructorArgs, + TASK_CHUGSPLASH_FETCH, + getArtifactsFromParsedCanonicalConfig, +} from '@chugsplash/plugins' import { TASK_VERIFY_GET_ETHERSCAN_ENDPOINT } from '@nomiclabs/hardhat-etherscan/dist/src/constants' import { EtherscanURLs } from '@nomiclabs/hardhat-etherscan/dist/src/types' import { @@ -28,11 +35,6 @@ import { ChugSplashManagerABI } from '@chugsplash/contracts' import { EthereumProvider } from 'hardhat/types' import { request } from 'undici' -import { - fetchChugSplashConfig, - getArtifactsFromCanonicalConfig, -} from './compile' - export interface EtherscanResponseBody { status: string message: string @@ -44,30 +46,32 @@ export const RESPONSE_OK = '1' export const verifyChugSplashConfig = async (hre: any, configUri: string) => { const { etherscanApiKey, etherscanApiEndpoints } = await getEtherscanInfo(hre) - const canonicalConfig = await fetchChugSplashConfig(configUri) - const artifacts = await getArtifactsFromCanonicalConfig(hre, canonicalConfig) - + const canonicalConfig = await hre.run(TASK_CHUGSPLASH_FETCH, { + configUri, + }) + const artifacts = await getArtifactsFromParsedCanonicalConfig( + hre, + parseChugSplashConfig(canonicalConfig) as CanonicalChugSplashConfig + ) const ChugSplashManager = new Contract( getChugSplashManagerProxyAddress(canonicalConfig.options.projectName), ChugSplashManagerABI, hre.ethers.provider ) - for (const [referenceName, contractConfig] of Object.entries( - canonicalConfig.contracts - )) { - const artifact = artifacts[contractConfig.contract] - const { abi, contractName, sourceName, fileOutput } = artifact + for (const referenceName of Object.keys(canonicalConfig.contracts)) { + const artifact = artifacts[referenceName] + const { abi, contractName, sourceName, compilerOutput } = artifact const { constructorArgValues } = getConstructorArgs( canonicalConfig, referenceName, abi, - fileOutput, + compilerOutput, sourceName, contractName ) - const contractAddress = await ChugSplashManager.implementations( + const implementationAddress = await ChugSplashManager.implementations( referenceName ) @@ -79,7 +83,7 @@ export const verifyChugSplashConfig = async (hre: any, configUri: string) => { hre.network.name, hre.network.provider, etherscanApiEndpoints, - contractAddress, + implementationAddress, sourceName, contractName, abi, @@ -88,6 +92,14 @@ export const verifyChugSplashConfig = async (hre: any, configUri: string) => { compilerInput.solcVersion, constructorArgValues ) + + await linkProxyWithImplementation( + etherscanApiEndpoints, + etherscanApiKey, + getProxyAddress(canonicalConfig.options.projectName, referenceName), + implementationAddress, + contractName + ) } } diff --git a/packages/executor/src/utils/initialize.ts b/packages/executor/src/utils/initialize.ts index 9d6578dbb..398e41cac 100644 --- a/packages/executor/src/utils/initialize.ts +++ b/packages/executor/src/utils/initialize.ts @@ -1,5 +1,5 @@ import { ethers } from 'ethers' -import { deployChugSplashContracts } from '@chugsplash/plugins' +import { deployChugSplashPredeploys } from '@chugsplash/plugins' import { CHUGSPLASH_CONSTRUCTOR_ARGS, ChugSplashBootLoaderArtifact, @@ -29,7 +29,7 @@ export const initializeChugSplashContracts = async ( hre: any, deployer: ethers.Signer ) => { - await deployChugSplashContracts(hre, deployer) + await deployChugSplashPredeploys(hre, deployer) const { etherscanApiKey, etherscanApiEndpoints } = await getEtherscanInfo(hre) diff --git a/packages/plugins/src/hardhat/artifacts.ts b/packages/plugins/src/hardhat/artifacts.ts index 8430a8696..dbc384bde 100644 --- a/packages/plugins/src/hardhat/artifacts.ts +++ b/packages/plugins/src/hardhat/artifacts.ts @@ -1,9 +1,19 @@ import path from 'path' import * as semver from 'semver' -import { SolidityStorageLayout, ChugSplashConfig } from '@chugsplash/core' -import { remove0x } from '@eth-optimism/core-utils' +import { + SolidityStorageLayout, + ChugSplashConfig, + CanonicalChugSplashConfig, +} from '@chugsplash/core' +import { add0x, remove0x } from '@eth-optimism/core-utils' import { ethers, utils } from 'ethers' +import { + TASK_COMPILE_SOLIDITY_GET_SOLC_BUILD, + TASK_COMPILE_SOLIDITY_RUN_SOLC, + TASK_COMPILE_SOLIDITY_RUN_SOLCJS, +} from 'hardhat/builtin-tasks/task-names' +import { SolcBuild } from 'hardhat/types' // TODO export type ContractArtifact = any @@ -291,3 +301,85 @@ export const getImmutableVariables = ( } return immutableVariables } + +export const getArtifactsFromParsedCanonicalConfig = async ( + hre: any, + parsedCanonicalConfig: CanonicalChugSplashConfig +): Promise<{ [referenceName: string]: any }> => { + const compilerOutputs: any[] = [] + // Get the compiler output for each compiler input. + for (const compilerInput of parsedCanonicalConfig.inputs) { + const solcBuild: SolcBuild = await hre.run( + TASK_COMPILE_SOLIDITY_GET_SOLC_BUILD, + { + quiet: true, + solcVersion: compilerInput.solcVersion, + } + ) + + let compilerOutput: any // TODO: Compiler output type + if (solcBuild.isSolcJs) { + compilerOutput = await hre.run(TASK_COMPILE_SOLIDITY_RUN_SOLCJS, { + input: compilerInput.input, + solcJsPath: solcBuild.compilerPath, + }) + } else { + compilerOutput = await hre.run(TASK_COMPILE_SOLIDITY_RUN_SOLC, { + input: compilerInput.input, + solcPath: solcBuild.compilerPath, + }) + } + compilerOutputs.push(compilerOutput) + } + + const artifacts = {} + // Generate an artifact for each contract in the ChugSplash config. + for (const [referenceName, contractConfig] of Object.entries( + parsedCanonicalConfig.contracts + )) { + let compilerOutputIndex = 0 + while (artifacts[referenceName] === undefined) { + // Iterate through the sources in the current compiler output to find the one that + // contains this contract. + const compilerOutput = compilerOutputs[compilerOutputIndex] + for (const [sourceName, sourceOutput] of Object.entries( + compilerOutput.contracts + )) { + // Check if the current source contains the contract. + if (sourceOutput.hasOwnProperty(contractConfig.contract)) { + const contractOutput = sourceOutput[contractConfig.contract] + + const creationCode = getCreationCode( + add0x(contractOutput.evm.bytecode.object), + parsedCanonicalConfig, + referenceName, + contractOutput.abi, + compilerOutput, + sourceName, + contractConfig.contract + ) + const immutableVariables = getImmutableVariables( + compilerOutput, + sourceName, + contractConfig.contract + ) + + artifacts[referenceName] = { + creationCode, + storageLayout: contractOutput.storageLayout, + immutableVariables, + abi: contractOutput.abi, + compilerOutput, + sourceName, + contractName: contractConfig.contract, + } + // We can exit the loop at this point since each contract only has a single artifact + // associated with it. + break + } + } + compilerOutputIndex += 1 + } + } + return artifacts +} diff --git a/packages/plugins/src/hardhat/deployments.ts b/packages/plugins/src/hardhat/deployments.ts index 2937662a1..8d7da9632 100644 --- a/packages/plugins/src/hardhat/deployments.ts +++ b/packages/plugins/src/hardhat/deployments.ts @@ -28,7 +28,7 @@ import { writeHardhatSnapshotId } from './utils' * @param hre Hardhat Runtime Environment. * @param contractName Name of the contract in the config file. */ -export const deployContracts = async ( +export const deployConfigs = async ( hre: any, verbose: boolean, hide: boolean, @@ -36,11 +36,11 @@ export const deployContracts = async ( ) => { const fileNames = fs.readdirSync(hre.config.paths.chugsplash) for (const fileName of fileNames) { - await deployChugSplashConfig(hre, fileName, verbose, hide, local) + await deployConfig(hre, fileName, verbose, hide, local) } } -export const deployChugSplashConfig = async ( +export const deployConfig = async ( hre: any, fileName: string, verbose: boolean, diff --git a/packages/plugins/src/hardhat/predeploys.ts b/packages/plugins/src/hardhat/predeploys.ts index 27d5806d3..93fa7ae79 100644 --- a/packages/plugins/src/hardhat/predeploys.ts +++ b/packages/plugins/src/hardhat/predeploys.ts @@ -21,7 +21,7 @@ import { CHUGSPLASH_CONSTRUCTOR_ARGS, } from '@chugsplash/contracts' -export const deployChugSplashContracts = async ( +export const deployChugSplashPredeploys = async ( hre: HardhatRuntimeEnvironment, deployer: ethers.Signer ): Promise => { diff --git a/packages/plugins/src/hardhat/tasks.ts b/packages/plugins/src/hardhat/tasks.ts index f25fde480..3d4b7e34b 100644 --- a/packages/plugins/src/hardhat/tasks.ts +++ b/packages/plugins/src/hardhat/tasks.ts @@ -1,20 +1,16 @@ import * as path from 'path' import * as fs from 'fs' -import { Contract, ethers, utils } from 'ethers' +import { Contract, ethers } from 'ethers' import { subtask, task, types } from 'hardhat/config' -import { SolcBuild } from 'hardhat/types' import { TASK_NODE, TASK_COMPILE, - TASK_COMPILE_SOLIDITY_GET_SOLC_BUILD, - TASK_COMPILE_SOLIDITY_RUN_SOLCJS, - TASK_COMPILE_SOLIDITY_RUN_SOLC, TASK_TEST, TASK_RUN, } from 'hardhat/builtin-tasks/task-names' import { create, IPFSHTTPClient } from 'ipfs-http-client' -import { add0x, getChainId, remove0x } from '@eth-optimism/core-utils' +import { getChainId } from '@eth-optimism/core-utils' import { computeBundleId, makeActionBundleFromConfig, @@ -30,8 +26,6 @@ import { isSetImplementationAction, fromRawChugSplashAction, getProxyAddress, - createDeploymentFolderForNetwork, - writeDeploymentArtifact, log as ChugSplashLog, } from '@chugsplash/core' import { @@ -45,37 +39,37 @@ import Hash from 'ipfs-only-hash' import * as dotenv from 'dotenv' import { + getArtifactsFromParsedCanonicalConfig, getBuildInfo, - getConstructorArgs, getContractArtifact, getCreationCode, getImmutableVariables, getStorageLayout, } from './artifacts' -import { deployContracts } from './deployments' -import { deployChugSplashContracts } from './predeploys' +import { deployConfigs } from './deployments' +import { deployChugSplashPredeploys } from './predeploys' import { writeHardhatSnapshotId } from './utils' // Load environment variables from .env dotenv.config() // internal tasks -const TASK_CHUGSPLASH_LOAD = 'chugsplash-load' -const TASK_CHUGSPLASH_FETCH = 'chugsplash-fetch' -const TASK_CHUGSPLASH_BUNDLE_LOCAL = 'chugsplash-bundle-local' -const TASK_CHUGSPLASH_BUNDLE_REMOTE = 'chugsplash-bundle-remote' +export const TASK_CHUGSPLASH_LOAD = 'chugsplash-load' +export const TASK_CHUGSPLASH_FETCH = 'chugsplash-fetch' +export const TASK_CHUGSPLASH_BUNDLE_LOCAL = 'chugsplash-bundle-local' +export const TASK_CHUGSPLASH_BUNDLE_REMOTE = 'chugsplash-bundle-remote' // public tasks -const TASK_CHUGSPLASH_DEPLOY = 'chugsplash-deploy' -const TASK_CHUGSPLASH_REGISTER = 'chugsplash-register' -const TASK_CHUGSPLASH_LIST_ALL_PROJECTS = 'chugsplash-list-projects' -const TASK_CHUGSPLASH_CHECK_BUNDLE = 'chugsplash-check-bundle' -const TASK_CHUGSPLASH_COMMIT = 'chugsplash-commit' -const TASK_CHUGSPLASH_PROPOSE = 'chugsplash-propose' -const TASK_CHUGSPLASH_APPROVE = 'chugsplash-approve' -const TASK_CHUGSPLASH_EXECUTE = 'chugsplash-execute' -const TASK_CHUGSPLASH_LIST_BUNDLES = 'chugsplash-list-bundles' -const TASK_CHUGSPLASH_STATUS = 'chugsplash-status' +export const TASK_CHUGSPLASH_DEPLOY = 'chugsplash-deploy' +export const TASK_CHUGSPLASH_REGISTER = 'chugsplash-register' +export const TASK_CHUGSPLASH_LIST_ALL_PROJECTS = 'chugsplash-list-projects' +export const TASK_CHUGSPLASH_CHECK_BUNDLE = 'chugsplash-check-bundle' +export const TASK_CHUGSPLASH_COMMIT = 'chugsplash-commit' +export const TASK_CHUGSPLASH_PROPOSE = 'chugsplash-propose' +export const TASK_CHUGSPLASH_APPROVE = 'chugsplash-approve' +export const TASK_CHUGSPLASH_EXECUTE = 'chugsplash-execute' +export const TASK_CHUGSPLASH_LIST_BUNDLES = 'chugsplash-list-bundles' +export const TASK_CHUGSPLASH_STATUS = 'chugsplash-status' subtask(TASK_CHUGSPLASH_LOAD) .addParam('deployConfig', undefined, undefined, types.string) @@ -151,77 +145,10 @@ subtask(TASK_CHUGSPLASH_BUNDLE_REMOTE) args.canonicalConfig ) as CanonicalChugSplashConfig - const compilerOutputs: any[] = [] - // Get the compiler output for each compiler input. - for (const compilerInput of parsedCanonicalConfig.inputs) { - const solcBuild: SolcBuild = await hre.run( - TASK_COMPILE_SOLIDITY_GET_SOLC_BUILD, - { - quiet: true, - solcVersion: compilerInput.solcVersion, - } - ) - - let compilerOutput: any // TODO: Compiler output type - if (solcBuild.isSolcJs) { - compilerOutput = await hre.run(TASK_COMPILE_SOLIDITY_RUN_SOLCJS, { - input: compilerInput.input, - solcJsPath: solcBuild.compilerPath, - }) - } else { - compilerOutput = await hre.run(TASK_COMPILE_SOLIDITY_RUN_SOLC, { - input: compilerInput.input, - solcPath: solcBuild.compilerPath, - }) - } - compilerOutputs.push(compilerOutput) - } - - const artifacts = {} - // Generate an artifact for each contract in the ChugSplash config. - for (const [referenceName, contractConfig] of Object.entries( - parsedCanonicalConfig.contracts - )) { - let compilerOutputIndex = 0 - while (artifacts[referenceName] === undefined) { - // Iterate through the sources in the current compiler output to find the one that - // contains this contract. - const compilerOutput = compilerOutputs[compilerOutputIndex] - for (const [sourceName, sourceOutput] of Object.entries( - compilerOutput.contracts - )) { - // Check if the current source contains the contract. - if (sourceOutput.hasOwnProperty(contractConfig.contract)) { - const contractOutput = sourceOutput[contractConfig.contract] - - const creationCode = getCreationCode( - add0x(contractOutput.evm.bytecode.object), - parsedCanonicalConfig, - referenceName, - contractOutput.abi, - compilerOutput, - sourceName, - contractConfig.contract - ) - const immutableVariables = getImmutableVariables( - compilerOutput, - sourceName, - contractConfig.contract - ) - - artifacts[referenceName] = { - creationCode, - storageLayout: contractOutput.storageLayout, - immutableVariables, - } - // We can exit the loop at this point since each contract only has a single artifact - // associated with it. - break - } - } - compilerOutputIndex += 1 - } - } + const artifacts = await getArtifactsFromParsedCanonicalConfig( + hre, + parsedCanonicalConfig + ) return makeActionBundleFromConfig( parsedCanonicalConfig, @@ -302,8 +229,8 @@ task(TASK_CHUGSPLASH_DEPLOY) hre: any ) => { const signer = await hre.ethers.getSigner() - await deployChugSplashContracts(hre, signer) - await deployContracts(hre, args.log, args.hide, args.local) + await deployChugSplashPredeploys(hre, signer) + await deployConfigs(hre, args.log, args.hide, args.local) } ) @@ -1175,8 +1102,8 @@ task(TASK_NODE) if (!args.disable) { if ((await getChainId(hre.ethers.provider)) === 31337) { const deployer = await hre.ethers.getSigner() - await deployChugSplashContracts(hre, deployer) - await deployContracts(hre, args.log, args.hide, args.local) + await deployChugSplashPredeploys(hre, deployer) + await deployConfigs(hre, args.log, args.hide, args.local) await writeHardhatSnapshotId(hre) } } @@ -1210,8 +1137,8 @@ task(TASK_TEST) throw new Error('Snapshot failed to be reverted.') } } catch { - await deployChugSplashContracts(hre, await hre.ethers.getSigner()) - await deployContracts(hre, false, !args.show, args.local) + await deployChugSplashPredeploys(hre, await hre.ethers.getSigner()) + await deployConfigs(hre, false, !args.show, args.local) } finally { await writeHardhatSnapshotId(hre) }