From 94615df801b27ccaf18ae5b326d1108d3f3b69bf Mon Sep 17 00:00:00 2001 From: Ino Murko <2582555+InoMurko@users.noreply.github.com> Date: Wed, 28 Aug 2024 16:48:59 +0200 Subject: [PATCH] fw steps tidy things cleanup --- .../1-create_multisig.ts | 79 ++++++++++ .../setup-fast-withdrawal/2-add_validators.ts | 117 ++++++++++++++ .../3-set-any-trust-fast-confirmer.ts | 90 +++++++++++ .../4-configure-minimum-assertion-period.ts | 149 ++++++++++++++++++ examples/setup-fast-withdrawal/README.md | 19 +++ examples/setup-fast-withdrawal/common.ts | 53 +++++++ examples/setup-fast-withdrawal/tsconfig.json | 6 +- package.json | 5 + src/createSafePrepareTransactionRequest.ts | 4 +- ...tFastConfirmerPrepareTransactionRequest.ts | 4 +- yarn.lock | 70 +++++++- 11 files changed, 590 insertions(+), 6 deletions(-) create mode 100644 examples/setup-fast-withdrawal/1-create_multisig.ts create mode 100644 examples/setup-fast-withdrawal/2-add_validators.ts create mode 100644 examples/setup-fast-withdrawal/3-set-any-trust-fast-confirmer.ts create mode 100644 examples/setup-fast-withdrawal/4-configure-minimum-assertion-period.ts create mode 100644 examples/setup-fast-withdrawal/common.ts diff --git a/examples/setup-fast-withdrawal/1-create_multisig.ts b/examples/setup-fast-withdrawal/1-create_multisig.ts new file mode 100644 index 00000000..fbcc6ed4 --- /dev/null +++ b/examples/setup-fast-withdrawal/1-create_multisig.ts @@ -0,0 +1,79 @@ +import { createPublicClient, http, isAddress } from 'viem'; +import { + createSafePrepareTransactionRequest, +} from '@arbitrum/orbit-sdk'; +import { getParentChainFromId } from '@arbitrum/orbit-sdk/utils'; +import { config } from 'dotenv'; +import { propose } from './common.js'; + +config(); + +//check environment variables +if (typeof process.env.OWNER_1_ADDRESS_PRIVATE_KEY === 'undefined') { + throw new Error(`Please provide the "OWNER_1_ADDRESS_PRIVATE_KEY" environment variable`); +} + +if (typeof process.env.PARENT_CHAIN_ID === 'undefined') { + throw new Error(`Please provide the "PARENT_CHAIN_ID" environment variable`); +} + +if (typeof process.env.SAFE_ADDRESS === 'undefined') { + throw new Error(`Please provide the "SAFE_ADDRESS" environment variable`); +} + +if (typeof process.env.FC_VALIDATORS === 'undefined') { + throw new Error(`Please provide the "FC_VALIDATORS" environment variable`); +} + +const rollupOwnerSafeAddress = process.env.SAFE_ADDRESS as `0x${string}`; +// // set the parent chain and create a public client for it +const parentChainId = Number(process.env.PARENT_CHAIN_ID); +const parentChain = getParentChainFromId(parentChainId); +const parentChainPublicClient = createPublicClient({ + chain: parentChain, + transport: http(process.env.RPC), +}); +// sanitize validator addresses +const fcValidators = JSON.parse(process.env.FC_VALIDATORS); +const safeWalletThreshold = fcValidators.length; +if (!fcValidators) { + throw new Error(`The "FC_VALIDATORS" environment variable must be a valid array`); +} + +const sanitizedFcValidators = [ + ...new Set( + fcValidators.filter((validator: `0x${string}`) => + isAddress(validator) ? validator : undefined, + ), + ), +]; +if (sanitizedFcValidators.length !== safeWalletThreshold) { + throw new Error( + `Some of the addresses in the "FC_VALIDATORS" environment variable appear to not be valid or duplicated.`, + ); +} + +async function main() { + // + // Step 1. Create Safe multisig + // + console.log( + `Step 1: Create a new ${safeWalletThreshold}/${safeWalletThreshold} Safe wallet with the following addresses as signers:`, + fcValidators, + ); + console.log('---'); + const txRequest = await createSafePrepareTransactionRequest({ + publicClient: parentChainPublicClient, + account: rollupOwnerSafeAddress, + owners: fcValidators, + threshold: safeWalletThreshold, + saltNonce: BigInt(Date.now()) + }); + propose(txRequest.to as string, txRequest.data as string, rollupOwnerSafeAddress); + //execute the transaction + //https://help.safe.global/en/articles/40834-verify-safe-creation + //in the executed transaction find `ProxyCreation` event + //Data singleton :
is what you're looing for +} + +main(); diff --git a/examples/setup-fast-withdrawal/2-add_validators.ts b/examples/setup-fast-withdrawal/2-add_validators.ts new file mode 100644 index 00000000..30af8b25 --- /dev/null +++ b/examples/setup-fast-withdrawal/2-add_validators.ts @@ -0,0 +1,117 @@ +import { createPublicClient, http, isAddress, Address } from 'viem'; +import { + createRollupFetchTransactionHash, + createRollupPrepareTransactionReceipt, + rollupAdminLogicPublicActions, +} from '@arbitrum/orbit-sdk'; +import { getParentChainFromId } from '@arbitrum/orbit-sdk/utils'; +import { config } from 'dotenv'; +import { propose } from './common.js'; + + +config(); + +//check environment variables +if (typeof process.env.OWNER_1_ADDRESS_PRIVATE_KEY === 'undefined') { + throw new Error(`Please provide the "OWNER_1_ADDRESS_PRIVATE_KEY" environment variable`); +} + +if (typeof process.env.PARENT_CHAIN_ID === 'undefined') { + throw new Error(`Please provide the "PARENT_CHAIN_ID" environment variable`); +} + +if (typeof process.env.SAFE_ADDRESS === 'undefined') { + throw new Error(`Please provide the "SAFE_ADDRESS" environment variable`); +} + +if (typeof process.env.FC_VALIDATORS_SAFE_ADDRESS === 'undefined') { + throw new Error(`Please provide the "FC_VALIDATORS_SAFE_ADDRESS" environment variable (run step 1)`); +} + +if (typeof process.env.FC_VALIDATORS === 'undefined') { + throw new Error(`Please provide the "FC_VALIDATORS" environment variable`); +} + +if (typeof process.env.ROLLUP_ADDRESS === 'undefined') { + throw new Error(`Please provide the "ROLLUP_ADDRESS" environment variable`); +} + +if (typeof process.env.RPC === 'undefined') { + throw new Error(`Please provide an "RPC" endpoint with unlimited eth_getLogs range`); +} + +const rollupOwnerSafeAddress = process.env.SAFE_ADDRESS as `0x${string}`; +const safeAddress = process.env.FC_VALIDATORS_SAFE_ADDRESS as `0x${string}`; +const rollupAddress = process.env.ROLLUP_ADDRESS as Address; +// // set the parent chain and create a public client for it +const parentChainId = Number(process.env.PARENT_CHAIN_ID); +const parentChain = getParentChainFromId(parentChainId); +const parentChainPublicClient = createPublicClient({ + chain: parentChain, + transport: http(process.env.RPC), +}).extend( + rollupAdminLogicPublicActions({ + rollup: rollupAddress, + }), +); + +// sanitize validator addresses +const fcValidators = JSON.parse(process.env.FC_VALIDATORS); +const safeWalletThreshold = fcValidators.length; +if (!fcValidators) { + throw new Error(`The "FC_VALIDATORS" environment variable must be a valid array`); +} + +const sanitizedFcValidators = [ + ...new Set( + fcValidators.filter((validator: `0x${string}`) => + isAddress(validator) ? validator : undefined, + ), + ), +]; +if (sanitizedFcValidators.length !== safeWalletThreshold) { + throw new Error( + `Some of the addresses in the "FC_VALIDATORS" environment variable appear to not be valid or duplicated.`, + ); +} + +async function main() { + console.log('Add the new Safe address (created in step 1) as a validator'); + fcValidators.push(safeAddress); + + console.log('Gather necessary data (UpgradeExecutor address)'); + const transactionHash = await createRollupFetchTransactionHash({ + rollup: rollupAddress, + publicClient: parentChainPublicClient, + }); + const transactionReceipt = createRollupPrepareTransactionReceipt( + await parentChainPublicClient.getTransactionReceipt({ hash: transactionHash }), + ); + const coreContracts = transactionReceipt.getCoreContracts(); + const upgradeExecutorAddress = coreContracts.upgradeExecutor; + + // + // Step 2. Add validators to the Orbit chain rollup validator whitelist + // + console.log( + `Step 2: Adding the following validators to the Rollup validator whitelist:`, + fcValidators, + ); + console.log('---'); + + // prepare set validator transaction request + const fcValidatorsStatus = Array(fcValidators.length).fill(true); + const setValidatorTransactionRequest = + await parentChainPublicClient.rollupAdminLogicPrepareTransactionRequest({ + functionName: 'setValidator', + args: [ + fcValidators, // validator address list + fcValidatorsStatus, // validator status list + ], + upgradeExecutor: upgradeExecutorAddress, + account: rollupOwnerSafeAddress, + }); + propose(setValidatorTransactionRequest.to as string, setValidatorTransactionRequest.data as string, rollupOwnerSafeAddress); +} + +main(); diff --git a/examples/setup-fast-withdrawal/3-set-any-trust-fast-confirmer.ts b/examples/setup-fast-withdrawal/3-set-any-trust-fast-confirmer.ts new file mode 100644 index 00000000..63ebb297 --- /dev/null +++ b/examples/setup-fast-withdrawal/3-set-any-trust-fast-confirmer.ts @@ -0,0 +1,90 @@ +import { createPublicClient, http, Address, parseAbi } from 'viem'; +import { + createRollupFetchTransactionHash, + createRollupPrepareTransactionReceipt, + rollupAdminLogicPublicActions, + setAnyTrustFastConfirmerPrepareTransactionRequest, +} from '@arbitrum/orbit-sdk'; +import { getParentChainFromId } from '@arbitrum/orbit-sdk/utils'; +import { config } from 'dotenv'; +import { propose } from './common.js'; + + +config(); + +//check environment variables +if (typeof process.env.OWNER_1_ADDRESS_PRIVATE_KEY === 'undefined') { + throw new Error(`Please provide the "OWNER_1_ADDRESS_PRIVATE_KEY" environment variable`); +} + +if (typeof process.env.PARENT_CHAIN_ID === 'undefined') { + throw new Error(`Please provide the "PARENT_CHAIN_ID" environment variable`); +} + +if (typeof process.env.SAFE_ADDRESS === 'undefined') { + throw new Error(`Please provide the "SAFE_ADDRESS" environment variable`); +} + +if (typeof process.env.FC_VALIDATORS_SAFE_ADDRESS === 'undefined') { + throw new Error(`Please provide the "FC_VALIDATORS_SAFE_ADDRESS" environment variable (run step 1)`); +} + +if (typeof process.env.ROLLUP_ADDRESS === 'undefined') { + throw new Error(`Please provide the "ROLLUP_ADDRESS" environment variable`); +} + +if (typeof process.env.RPC === 'undefined') { + throw new Error(`Please provide an "RPC" endpoint with unlimited eth_getLogs range`); +} + +const rollupOwnerSafeAddress = process.env.SAFE_ADDRESS as `0x${string}`; +const safeAddress = process.env.FC_VALIDATORS_SAFE_ADDRESS as `0x${string}`; +const rollupAddress = process.env.ROLLUP_ADDRESS as Address; +// // set the parent chain and create a public client for it +const parentChainId = Number(process.env.PARENT_CHAIN_ID); +const parentChain = getParentChainFromId(parentChainId); +const parentChainPublicClient = createPublicClient({ + chain: parentChain, + transport: http(process.env.RPC), +}).extend( + rollupAdminLogicPublicActions({ + rollup: rollupAddress, + }), +); + +async function main() { + const currentAnyTrustFastConfirmer = await parentChainPublicClient.readContract({ + address: rollupAddress, + abi: parseAbi(['function anyTrustFastConfirmer() view returns (address)']), + functionName: 'anyTrustFastConfirmer', + }); + + if (currentAnyTrustFastConfirmer.toLowerCase() !== safeAddress.toLowerCase()) { + console.log('Gather necessary data (UpgradeExecutor address)'); + const transactionHash = await createRollupFetchTransactionHash({ + rollup: rollupAddress, + publicClient: parentChainPublicClient, + }); + const transactionReceipt = createRollupPrepareTransactionReceipt( + await parentChainPublicClient.getTransactionReceipt({ hash: transactionHash }), + ); + const coreContracts = transactionReceipt.getCoreContracts(); + const upgradeExecutorAddress = coreContracts.upgradeExecutor; + const setAnyTrustFastConfirmerTransactionRequest = + await setAnyTrustFastConfirmerPrepareTransactionRequest({ + publicClient: parentChainPublicClient, + account: rollupOwnerSafeAddress, + rollup: rollupAddress, + upgradeExecutor: upgradeExecutorAddress, + fastConfirmer: safeAddress, + }); + propose(setAnyTrustFastConfirmerTransactionRequest.to as string, setAnyTrustFastConfirmerTransactionRequest.data as string, rollupOwnerSafeAddress); + + } else { + console.log( + `AnyTrust fast confirmer is already configured to ${currentAnyTrustFastConfirmer}. Skipping.`, + ); + } +} + +main(); diff --git a/examples/setup-fast-withdrawal/4-configure-minimum-assertion-period.ts b/examples/setup-fast-withdrawal/4-configure-minimum-assertion-period.ts new file mode 100644 index 00000000..61b496c7 --- /dev/null +++ b/examples/setup-fast-withdrawal/4-configure-minimum-assertion-period.ts @@ -0,0 +1,149 @@ +import { createPublicClient, http, Address } from 'viem'; +import { + createRollupFetchTransactionHash, + createRollupPrepareTransactionReceipt, + rollupAdminLogicPublicActions, +} from '@arbitrum/orbit-sdk'; +import { base, baseSepolia } from '@arbitrum/orbit-sdk/chains'; +import { getParentChainFromId } from '@arbitrum/orbit-sdk/utils'; +import { config } from 'dotenv'; +import { propose } from './common.js'; + + +config(); + +function getTimeDelayFromNumberOfBlocks(chainId: number, blocks: bigint): string { + // For Arbitrum L2s built on top of Ethereum, or Arbitrum L3s built on top of an Arbitrum L2, `block.number` always returns the L1 block number. + // L1 blocks are produced every 12 seconds. + // + // For Arbitrum L3s built on top of an OP Stack L2, `block.number` will return the L2 block number. + // L2 blocks in OP Stack chains are produced every 2 seconds. + const seconds = Number( + chainId === base.id || chainId === baseSepolia.id ? blocks * 2n : blocks * 12n, + ); + + const h = Math.floor(seconds / 3600); + const m = Math.floor((seconds % 3600) / 60); + const s = seconds % 60; + return `${h}h${m}m${s}s`; +} + + +//check environment variables +if (typeof process.env.OWNER_1_ADDRESS_PRIVATE_KEY === 'undefined') { + throw new Error(`Please provide the "OWNER_1_ADDRESS_PRIVATE_KEY" environment variable`); +} + +if (typeof process.env.PARENT_CHAIN_ID === 'undefined') { + throw new Error(`Please provide the "PARENT_CHAIN_ID" environment variable`); +} + +if (typeof process.env.SAFE_ADDRESS === 'undefined') { + throw new Error(`Please provide the "SAFE_ADDRESS" environment variable`); +} + +if (typeof process.env.ROLLUP_ADDRESS === 'undefined') { + throw new Error(`Please provide the "ROLLUP_ADDRESS" environment variable`); +} + +if (typeof process.env.RPC === 'undefined') { + throw new Error(`Please provide an "RPC" endpoint with unlimited eth_getLogs range`); +} + +if (typeof process.env.MINIMUM_ASSERTION_PERIOD === 'undefined') { + console.log('MinimumAssertionPeriod set to 75') +} + +if (typeof process.env.FC_VALIDATORS_SAFE_ADDRESS === 'undefined') { + throw new Error(`Please provide the "FC_VALIDATORS_SAFE_ADDRESS" environment variable (run step 1)`); +} + +const minimumAssertionPeriod = BigInt(process.env.MINIMUM_ASSERTION_PERIOD || 75); +const rollupOwnerSafeAddress = process.env.SAFE_ADDRESS as `0x${string}`; +const rollupAddress = process.env.ROLLUP_ADDRESS as Address; +// // set the parent chain and create a public client for it +const parentChainId = Number(process.env.PARENT_CHAIN_ID); +const parentChain = getParentChainFromId(parentChainId); +const parentChainPublicClient = createPublicClient({ + chain: parentChain, + transport: http(process.env.RPC), +}).extend( + rollupAdminLogicPublicActions({ + rollup: rollupAddress, + }), +); + +async function main() { + // + // Step 0. Gather necessary data (UpgradeExecutor address) + // + const transactionHash = await createRollupFetchTransactionHash({ + rollup: rollupAddress, + publicClient: parentChainPublicClient, + }); + const transactionReceipt = createRollupPrepareTransactionReceipt( + await parentChainPublicClient.getTransactionReceipt({ hash: transactionHash }), + ); + const coreContracts = transactionReceipt.getCoreContracts(); + const upgradeExecutorAddress = coreContracts.upgradeExecutor; + + console.log( + `Step 4: Configure the minimumAssertionPeriod in the Rollup contract to: ${Number( + minimumAssertionPeriod, + )}`, + ); + console.log('---'); + + // get current minimumAssertionPeriod + const currentMinimumAssertionPeriod = await parentChainPublicClient.rollupAdminLogicReadContract({ + functionName: 'minimumAssertionPeriod', + }); + + if (currentMinimumAssertionPeriod !== minimumAssertionPeriod) { + // prepare setMinimumAssertionPeriod transaction request + const setMinimumAssertionPeriodTransactionRequest = + await parentChainPublicClient.rollupAdminLogicPrepareTransactionRequest({ + functionName: 'setMinimumAssertionPeriod', + args: [minimumAssertionPeriod], + upgradeExecutor: upgradeExecutorAddress, + account: rollupOwnerSafeAddress, + }); + propose(setMinimumAssertionPeriodTransactionRequest.to as string, setMinimumAssertionPeriodTransactionRequest.data as string, rollupOwnerSafeAddress); + console.log('Transaction proposed'); + } else { + console.log( + `Minimum assertion period is already configured to ${currentMinimumAssertionPeriod}. Skipping.`, + ); + } + console.log(''); + + // + // Step 5. Show a confirmation message with the next steps + // + console.log(`Step 5: Configure your batch poster and your validator nodes`); + console.log(`---`); + console.log( + 'Your chain is now configured with a fast-withdrawal committee. The following instructions show how to configure your batch poster and your validators to start using this feature.', + ); + console.log(''); + + // Batch poster configuration + const timeDelay = getTimeDelayFromNumberOfBlocks(parentChain.id, minimumAssertionPeriod); + console.log('Your batch poster has to run at least nitro v3.1.1'); + console.log('Add the following parameter:'); + console.log(`--node.batch-poster.max-delay=${timeDelay}`); + console.log(''); + + // Validator configuration + console.log('Your validators have to run at least nitro v3.1.1'); + console.log('Add the following parameters:'); + console.log(`--node.staker.enable-fast-confirmation=true`); + console.log(`--node.staker.fast-confirm-safe-address=${process.env.FC_VALIDATORS_SAFE_ADDRESS}`); + console.log(`--node.staker.make-assertion-interval=${timeDelay}`); + console.log(''); + + // Final recommendation + console.log('Finally, restart your batch poster and validator nodes.'); +} + +main(); diff --git a/examples/setup-fast-withdrawal/README.md b/examples/setup-fast-withdrawal/README.md index d7e8663c..f3672c58 100644 --- a/examples/setup-fast-withdrawal/README.md +++ b/examples/setup-fast-withdrawal/README.md @@ -52,3 +52,22 @@ You need to set the following environment variables in an .env file: ```bash yarn dev ``` + + + +## Multisig ownership +Beware: At least one of the signers needs to be an EOA account so that it can propose transactions through this script. + +1. Build this example: rm -rf dist/ && tsc --outDir dist && ls dist +2. OWNER_1_ADDRESS_PRIVATE_KEY= PARENT_CHAIN_ID= SAFE_ADDRESS= FC_VALIDATORS='["0x1234567890123456789012345678901234567890"]' node ./dist/1-create_multisig.js +This step will create a new Safe on the parent chain and add fast confirmation validators as owners. + +If you want to use this script then you need to make sure the owner of the Rollup has been transfered to a Multisig. + +3. OWNER_1_ADDRESS_PRIVATE_KEY= PARENT_CHAIN_ID= SAFE_ADDRESS= FC_VALIDATORS_SAFE_ADDRESS= FC_VALIDATORS='["0x1234567890123456789012345678901234567890"]' ROLLUP_ADDRESS= RPC= node ./dist/2-add_validators.js +the validators list is expanded with the Safe created with step 1 (1-create_multisig). That's why you need to provide FC_VALIDATORS_SAFE_ADDRESS. + +4. OWNER_1_ADDRESS_PRIVATE_KEY= PARENT_CHAIN_ID= SAFE_ADDRESS= FC_VALIDATORS_SAFE_ADDRESS= ROLLUP_ADDRESS= RPC= node ./dist/3-set-any-trust-fast-confirmer.js +We also add this Safe as `fast confirmer`. + +5. OWNER_1_ADDRESS_PRIVATE_KEY= PARENT_CHAIN_ID= SAFE_ADDRESS= MINIMUM_ASSERTION_PERIOD=1 ROLLUP_ADDRESS= RPC= node ./dist/4-configure-minimum-assertion-period.js diff --git a/examples/setup-fast-withdrawal/common.ts b/examples/setup-fast-withdrawal/common.ts new file mode 100644 index 00000000..d5ee67a7 --- /dev/null +++ b/examples/setup-fast-withdrawal/common.ts @@ -0,0 +1,53 @@ +import { createPublicClient, http } from 'viem'; +import { getParentChainFromId, sanitizePrivateKey } from '@arbitrum/orbit-sdk/utils'; +import SafeApiKit from '@safe-global/api-kit' +import Safe from '@safe-global/protocol-kit' +import { privateKeyToAccount } from 'viem/accounts'; +import { + MetaTransactionData, + OperationType +} from '@safe-global/safe-core-sdk-types' + +export async function propose(to: string, data: string, rollupOwnerSafeAddress: string): Promise