How to Use the OP Stack with Avail
Introduction
Embark on setting up your own Avail-Optimism chain. This guide targets Ethereum's Goerli testnet and Avail Goldberg testnet. For a detailed understanding, review the Optimism Documentation.
In this guide, you will conduct the following:
- Introduction
- Prerequisites
- Compile the Core Codebase
- Generate and Secure Keys
- Network Configuration and Setup
- Core Contract Deployment
- Setting Up L2 Configuration
- Initialize and Configure Geth
- Launch and Monitor Nodes
- Acquire Goerli ETH for Layer 2
- Conduct Test Transactions
Prerequisites
Ensure you have installed the following software.
Installation commands are based on Ubuntu 20.04 LTS:
|
|
Compile the Core Codebase
Setting up the EVM Rollup requires compiling code from two critical repositories: the avail-op-stack-adapter monorepo and the op-geth repository.
Build the Adapter Source
Clone and navigate to the Avail adapter:
git clone https://github.com/availproject/avail-op-stack-adapter.git
cd avail-op-stack-adapterInstall modules (simultaneously, you may start building
op-geth
):pnpm install
Compile the necessary packages:
make op-node op-batcher op-proposer
pnpm build
Build the Optimism Geth Source
Clone and navigate to
op-geth
:git clone https://github.com/ethereum-optimism/op-geth.git
cd op-gethCompile
op-geth
:make geth
Get Access to a Goerli Node
For deploying to Goerli, access an L1 node using a provider like Alchemy or run your own Goerli node.
Generate and Secure Keys
Create four essential accounts with private keys:
Admin
(contract upgrade authority)Batcher
(publishes Sequencer data to L1)Proposer
(publishes L2 results to L1)Sequencer
(signs blocks on the p2p network)
You can use cast wallet
in the contracts-bedrock
package for key generation:
In the Avail adapter repo, navigate to the contracts-bedrock package:
cd ~/avail-op-stack-adapter/packages/contracts-bedrock
Generate accounts:
echo "Admin:"
cast wallet new
echo "Proposer:"
cast wallet new
echo "Batcher:"
cast wallet new
echo "Sequencer:"
cast wallet newYou should see an output similar to:
Admin:
Successfully created new keypair.
Address: 0xc4A01194958DE0D90A876e8A5fc9D7B530072148
Private key: 0xb8e39bd94a210e410c4024e1cc91014de45a5eb1e42f3aa99a368b5a5ac19b45
Proposer:
Successfully created new keypair.
Address: 0xFC0374Ae658e46cA4022acA179d3cb6D8e1A4934
Private key: 0xa9bc1b3f5deb1e00251df68bf86e3493b25bc5430665433546f2f9aacc748d1a
Batcher:
Successfully created new keypair.
Address: 0xD6857B5BE9468Be67d64ABaB48459378d5329b96
Private key: 0xe9cd8960fc7984a301d567b819e0c62871eb2c7239c2e66b8f319eaa45c3cbd5
Sequencer:
Successfully created new keypair.
Address: 0x33348817E4B1192D576C4f157e9a5EC93dc5392D
Private key: 0xd98b49e11e4e0be9931017831395e6644a50c36285d08e14d1a479af5ee08675Record and securely store these key details. You'll need to fund
Admin
,Proposer
, andBatcher
with Goerli ETH (2 ETH forAdmin
, 5 ETH forProposer
, 10 ETH forBatcher
).Note for ProductionUse secure hardware for key management in production environments.
cast wallet
is not designed for production deployments.
Network Configuration and Setup
After building the repositories, configure your chain settings in the contracts-bedrock package.
Ensure you are in the
contracts-bedrock
sub-directory:cd ~/avail-op-stack-adapter/packages/contracts-bedrock
Copy the environment file:
cp .envrc.example .envrc
Edit
.envrc
with necessary values:ETH_RPC_URL
: URL for your L1 node.PRIVATE_KEY
: Private key of the Admin account.DEPLOYMENT_CONTEXT
: Name of the network; should be "avail-optimism"
Activate the environment with
direnv
:If you need to install
direnv
, ensure you also modify the shell configuration.direnv allow .
Choose an L1 block as a starting point using
cast
command:Using a finalized L1 block as our starting block is best.
cast block finalized --rpc-url $ETH_RPC_URL | grep -E "(timestamp|hash|number)"
You should get a response that looks like:
hash 0x784d8e7f0e90969e375c7d12dac7a3df6879450d41b4cb04d4f8f209ff0c4cd9
number 8482289
timestamp 1676253324Next, create and modify
deploy-config/avail-optimism.json
based ondeploy-config/getting-started.json
. Retain the default settings provided in the configuration and apply the following modifications:- Replace
ADMIN
with the address of the Admin account you generated earlier. - Replace
PROPOSER
with the address of the Proposer account you generated earlier. - Replace
BATCHER
with the address of the Batcher account you generated earlier. - Replace
SEQUENCER
with the address of the Sequencer account you generated earlier. - Replace
BLOCKHASH
with the blockhash you got from the cast command. - Replace
TIMESTAMP
with the timestamp you got from the cast command. Note that although all the other fields are strings, this field is a number! Don’t include the quotation marks.
- Replace
Configure
enableDA
inavail-optimism.json
(true
for Avail chain as DA,false
for Ethereum).Enter
op-avail
module:cd ~/avail-op-stack-adapter/op-avail
Create
config.json
with necessary variables (seed
,api_url
,app_id
).{
"seed": "test test test test test test test test test test test test",
"api_url": "wss://goldberg.avail.tools:443/ws",
"app_id": 1
}
Core Contract Deployment
Deploy essential L1 contracts for the chain’s functionality:
- Navigate to
/avail-op-stack-adapter/packages/contracts-bedrock/deployments
, and createavail-optimism
directory:
cd ~/avail-op-stack-adapter/packages/contracts-bedrock/deployments
mkdir avail-optimism
- Navigate to
/avail-op-stack-adapter/packages/contracts-bedrock/
and the deploy contracts (this can take up to 15 minutes):forge script scripts/Deploy.s.sol:Deploy --private-key $PRIVATE_KEY --broadcast --rpc-url $ETH_RPC_URL
forge script scripts/Deploy.s.sol:Deploy --sig 'sync()' --private-key $PRIVATE_KEY --broadcast --rpc-url $ETH_RPC_URL
Setting Up L2 Configuration
After configuring the L1 layer, focus shifts to establishing the L2 infrastructure. This involves generating three key files:
genesis.json
for the genesis blockrollup.json
for rollup configurationsjwt.txt
for secure communication betweenop-node
andop-geth
Navigate to the
op-node
directory:cd ~/avail-op-stack-adapter/op-node
Run the following command, ensuring you replace
<RPC>
with your specific L1 RPC URL. This generates thegenesis.json
androllup.json
files:go run cmd/main.go genesis l2 \
--deploy-config ../packages/contracts-bedrock/deploy-config/avail-optimism.json \
--deployment-dir ../packages/contracts-bedrock/deployments/avail-optimism/ \
--outfile.l2 genesis.json \
--outfile.rollup rollup.json \
--l1-rpc=$L1_RPCYou'll find the newly created
genesis.json
androllup.json
in theop-node
package.Generate a
jwt.txt
file, which is crucial for the secure interaction between nodes:openssl rand -hex 32 > jwt.txt
To get
op-geth
ready, move thegenesis.json
andjwt.txt
files into its directory:cp genesis.json ~/op-geth
cp jwt.txt ~/op-geth
These steps ensure the L2 layer is correctly configured and ready for integration with the L1 components, paving the way for a fully functional EVM Rollup on the Avail-OP Stack.
Initialize and Configure Geth
Prepare op-geth
for running the chain:
Navigate to
op-geth
:cd ~/op-geth
Create a data directory:
mkdir datadir
Initialize with the genesis file:
build/bin/geth init --datadir=datadir genesis.json
Launch and Monitor Nodes
Running op-geth
and op-node
is essential for every node. op-batcher
and op-proposer
are exclusive to the sequencer.
Set the following environment variables:
Variable | Value |
---|---|
SEQ_KEY | Sequencer private key |
BATCHER_KEY | Batcher private key (minimum 1 ETH) |
PROPOSER_KEY | Proposer private key |
L1_RPC | L1 node URL |
RPC_KIND | L1 server type (e.g., alchemy, quicknode) |
L2OO_ADDR | L2OutputOracleProxy address |
Running op-geth
To initiate op-geth
, navigate to its directory and execute the following commands:
cd ~/op-geth
./build/bin/geth \
--datadir ./datadir \
--http \
--http.corsdomain="*" \
--http.vhosts="*" \
--http.addr=0.0.0.0 \
--http.port=9545 \
--http.api=web3,debug,eth,txpool,net,engine \
--ws \
--ws.addr=0.0.0.0 \
--ws.port=9546 \
--ws.origins="*" \
--ws.api=debug,eth,txpool,net,engine \
--syncmode=full \
--gcmode=archive \
--nodiscover \
--maxpeers=0 \
--networkid=42069 \
--authrpc.vhosts="*" \
--authrpc.addr=0.0.0.0 \
--authrpc.port=9551 \
--authrpc.jwtsecret=./jwt.txt \
--rollup.disabletxpoolgossip=true
op-geth
is now active, but block creation will begin once op-node
is operational.
Why Archive Mode?
Archive mode, requiring more disk space than full mode, is essential for:
op-proposer
to access the full state history.- The explorer's functionality.
Reinitializing op-geth
In cases of database corruption indicated by op-node
errors or failure to find L2 heads, follow these steps:
- Stop
op-geth
. - Remove the existing data:
cd ~/op-geth
rm -rf datadir/geth - Reinitialize:
build/bin/geth init --datadir=datadir genesis.json
- Restart
op-geth
and thenop-node
.
Running op-node
To launch op-node
, which acts as a consensus client, run:
cd ~/avail-op-stack-adapter/op-node
./bin/op-node \
--l2=http://localhost:9551 \
--l2.jwt-secret=./jwt.txt \
--sequencer.enabled \
--sequencer.l1-confs=3 \
--verifier.l1-confs=3 \
--rollup.config=./rollup.json \
--rpc.addr=0.0.0.0 \
--rpc.port=9547 \
--p2p.disable \
--rpc.enable-admin \
--p2p.sequencer.key=$SEQ_KEY \
--l1=$L1_RPC \
--l1.rpckind=$RPC_KIND
Block creation will commence once op-node
starts processing L1 information and interfaces with op-geth
.
P2P Synchronization
To optimize synchronization and avoid network resource waste:
- Disable p2p sync (
--p2p.disable
) by default. - Use specific command line parameters for synchronization among multiple nodes.
Running op-batcher
op-batcher
is crucial in publishing transactions from the Sequencer to L1. Ensure it has at least 1 Goerli ETH for operational continuity.
cd ~/avail-op-stack-adapter/op-batcher
./bin/op-batcher \
--l2-eth-rpc=http://localhost:9545 \
--rollup-rpc=http://localhost:9547 \
--poll-interval=10s \
--sub-safety-margin=6 \
--num-confirmations=1 \
--safe-abort-nonce-too-low-count=3 \
--resubmission-timeout=30s \
--rpc.addr=0.0.0.0 \
--rpc.port=9548 \
--rpc.enable-admin \
--max-channel-duration=1 \
--l1-eth-rpc=$L1_RPC \
--private-key=$BATCHER_KEY
Controlling Batcher Costs
Adjust the --max-channel-duration=n
setting to balance transaction frequency on L1 and the operational costs of the batcher.
Running op-proposer
Finally, start op-proposer
to propose new state roots:
cd ~/avail-op-stack-adapter/op-proposer
./bin/op-proposer \
--poll-interval=12s \
--rpc.port=9560 \
--rollup-rpc=http://localhost:9547 \
--l2oo-address=$L2OO_ADDR \
--private-key=$PROPOSER_KEY \
--l1-eth-rpc=$L1_RPC
Acquire Goerli ETH for Layer 2
To obtain ETH on your Rollup:
Go to
contracts-bedrock
:cd ~/avail-op-stack-adapter/packages/contracts-bedrock
Find the L1 standard bridge contract address:
cat deployments/avail-optimism/L1StandardBridgeProxy.json | jq -r .address
Send Goerli ETH to the bridge contract address.
Conduct Test Transactions
You now have a fully operational Avail-Powered Optimism-based EVM Rollup. Experiment with it as you would with any other test blockchain.
Congratulations on setting up your chain!