Lisk provides a cost-efficient, fast, and scalable Layer 2 (L2) network based on Optimism (OP) that is secured by Ethereum.
This repository contains information on how to run your own node on the Lisk network.
We recommend you the following hardware configuration to run a Lisk L2 node:
- a modern multi-core CPU with good single-core performance
- a minimum of 16 GB RAM (32 GB recommended)
- a locally attached NVMe SSD drive
- adequate storage capacity to accommodate both the snapshot restoration process (if restoring from snapshot) and chain data, ensuring a minimum of (2 * current_chain_size) + snapshot_size + 20%_buffer
- if running with docker, please install Docker Engine version 27.0.1 or higher
Note: If utilizing Amazon Elastic Block Store (EBS), ensure timing buffered disk reads are fast enough in order to avoid latency issues alongside the rate of new blocks added to Base during the initial synchronization process; io2 block express
is recommended.
Network | Status |
---|---|
Lisk Sepolia | ✅ |
Lisk Mainnet | ✅ |
Note:
- It is now possible to run the Lisk nodes with the
--op-network
flag on the op-geth execution client.- It is still not possible to run the Lisk nodes with the
--chain
flag on the op-reth execution client.
git clone https://github.com/LiskHQ/lisk-node.git
cd lisk-node
-
Ensure you have an Ethereum L1 full node RPC available (not Lisk), and set the
OP_NODE_L1_ETH_RPC
and theOP_NODE_L1_BEACON
variables (within the.env.*
files, if using docker-compose). If running your own L1 node, it needs to be synced before the Lisk node will be able to fully sync. -
Please ensure that the environment file relevant to your network (
.env.sepolia
, or.env.mainnet
) is set for theenv_file
properties at two places withindocker-compose.yml
. By default, it is set to.env.mainnet
. -
We currently support running either the
op-geth
or theop-reth
nodes alongside theop-node
. By default, we run theop-geth
node. If you would like to run theop-reth
client, please set theCLIENT
environment variable toreth
before starting the node.Note:
- The
op-reth
client can be built in either themaxperf
(default) orrelease
profile. To learn more about them, please check reth's documentation on Optimizations. Please set theRETH_BUILD_PROFILE
environment variable accordingly. - Unless you are building the
op-reth
client inrelease
profile, please ensure that you have a machine with 32 GB RAM. - Additionally, if you have the Docker Desktop installed on your system, please make sure to set
Memory limit
to a minimum of16 GB
.
It can be set underSettings -> Resources -> Resource Allocation -> Memory limit
.
- The
-
Run:
IMPORTANT: To run the node on Lisk Sepolia, first patch the Dockerfile(s) with:git apply dockerfile-lisk-sepolia.patch
withop-geth
execution client:docker compose up --build --detach
or, with
op-reth
execution client:CLIENT=reth RETH_BUILD_PROFILE=maxperf docker compose up --build --detach
-
You should now be able to
curl
your Lisk node:curl -s -d '{"id":0,"jsonrpc":"2.0","method":"eth_getBlockByNumber","params":["latest",false]}' \ -H "Content-Type: application/json" http://localhost:8545
-
Before proceeding, please make sure to install the following dependency (this information is missing in the OP documentations linked below):
-
To build
op-node
andop-geth
from source, follow OP documentation on Building a Node from Source.- Before building the
op-node
, please patch the code withop-node-lisk-sepolia.patch
for an unhandledSystemConfig
event emitted on Lisk Sepolia, resulting in errors on the Lisk nodes.git apply <path-to-op-node-lisk-sepolia.patch>
- Before building the
-
To build
op-reth
from source, follow the reth official documentation.
Set the following environment variable:
export DATADIR_PATH=... # Path to the folder where geth data will be stored
op-geth
and op-node
communicate over the engine API authrpc. This communication can be secured with a shared secret which can be provided to both when starting the applications. In this case, the secret takes the form of a random 32-byte hex string and can be generated with:
openssl rand -hex 32 > jwt.txt
For more information refer to the OP documentation.
Navigate to your op-geth
directory and start service by running the command:
./build/bin/geth \
--op-network=$OP_NODE_NETWORK \
--datadir=$DATADIR_PATH \
--verbosity=3 \
--authrpc.addr=0.0.0.0 \
--authrpc.port=8551 \
--authrpc.vhosts="*" \
--authrpc.jwtsecret=PATH_TO_JWT_TEXT_FILE \
--ws \
--ws.addr=0.0.0.0 \
--ws.port=8546 \
--ws.origins="*" \
--ws.api=debug,eth,net,engine \
--http \
--http.corsdomain="*" \
--http.vhosts="*" \
--http.addr=0.0.0.0 \
--http.port=8545 \
--http.api=web3,debug,eth,net,engine \
--metrics \
--metrics.addr=0.0.0.0 \
--metrics.port=6060 \
--syncmode=full \
--gcmode=full \
--port=30303 \
--maxpeers=100 \
--rollup.sequencerhttp=SEQUENCER_HTTP \
--rollup.halt=major \
--rollup.disabletxpoolgossip=true \
--nat=extip:0.0.0.0
Refer to the op-geth
configuration documentation for detailed information about available options.
Navigate to your reth
directory and start service by running the command:
./target/release/op-reth node \
-vvv \
--chain=PATH_TO_NETWORK_GENESIS_FILE \
--datadir="$DATADIR_PATH" \
--log.stdout.format log-fmt \
--authrpc.addr=0.0.0.0 \
--authrpc.port=8551 \
--authrpc.jwtsecret=PATH_TO_JWT_TEXT_FILE \
--ws \
--ws.origins="*" \
--ws.addr=0.0.0.0 \
--ws.port=8546 \
--ws.api=debug,eth,net,txpool \
--http \
--http.corsdomain="*" \
--http.addr=0.0.0.0 \
--http.port=8545 \
--http.api=web3,debug,eth,net,txpool \
--metrics=0.0.0.0:6060 \
--disable-discovery \
--port=30303 \
--rollup.sequencer-http=SEQUENCER_HTTP \
--rollup.disable-tx-pool-gossip
Refer to the reth
configuration documentation for detailed information about available options.
Note:
Official Lisk Sequencer HTTP RPC:
- Lisk Sepolia: https://rpc.sepolia-api.lisk.com
- Lisk Mainnet: https://rpc.api.lisk.com
Navigate to your op-node
directory and start service by running the command:
./bin/op-node \
--network="$OP_NODE_NETWORK" \
--l1=$OP_NODE_L1_ETH_RPC \
--l1.rpckind=$OP_NODE_L1_RPC_KIND \
--l1.beacon=$OP_NODE_L1_BEACON \
--l2=ws://localhost:8551 \
--l2.jwt-secret=PATH_TO_JWT_TEXT_FILE
The above command starts op-node
in full sync mode. Depending on the chain length, the initial sync process could take significant time; varying from days to weeks.
INFO [06-26|13:31:20.389] Advancing bq origin origin=17171d..1bc69b:8300332 originBehind=false
For more information refer to the OP documentation.
Note:
- In case you had skipped the
op-geth
initialization step above, you can start the node with the--network=OP_NODE_NETWORK
flag.- When specifying the
--network
flag, kindly make sure to remove the--rollup.config
flag.
Refer to the op-node
configuration documentation for detailed information about available options.
Note:
Some L1 nodes (e.g. Erigon) do not support fetching storage proofs. You can work around this by specifying--l1.trustrpc
when starting op-node (add it inop-node-entrypoint
and rebuild the docker image withdocker compose build
.) Do not do this unless you fully trust the L1 node provider.
Note:
- Snapshots are only available for the
op-geth
client and are from an archival node. They are of two types:
export
: small download size, slow to restore from, data is verified during restoredatadir
: large download size, fast to restore from, no data verification during restore
To enable auto-snapshot download and application, please set the APPLY_SNAPSHOT
environment variable to true
, when starting the node.
APPLY_SNAPSHOT=true docker compose up --build --detach
To choose the snapshot type, please set the SNAPSHOT_TYPE
flag to either export
(default) or datadir
, when starting the node.
APPLY_SNAPSHOT=true SNAPSHOT_TYPE=export docker compose up --build --detach
You can also download and apply a snapshot from a custom URL by setting the SNAPSHOT_URL
environment variable.
Please make sure the snapshot file ends with *.tar.gz
.
APPLY_SNAPSHOT=true SNAPSHOT_URL=<custom-snapshot-url> docker compose up --build --detach
Please follow the steps below:
-
Download the snapshot and the corresponding checksum from. The latest snapshot name is always listed in the
latest-<export|datadir>
file:- Sepolia: https://snapshots.lisk.com/sepolia
- Mainnet: https://snapshots.lisk.com/mainnet
-
Verify the integrity of the downloaded snapshot with:
sha256sum -c <checksum-file-name>
-
Import the snapshot:
export
:tar -xf <path-to-downloaded-export-snapshot-tarball> ./build/bin/geth import --datadir=$GETH_DATA_DIR <path-to-extracted-export-snapshot>
datadir
:tar --directory $GETH_DATA_DIR -xf <path-to-datadir-snapshot>
Sync speed depends on your L1 node, as the majority of the chain is derived from data submitted to the L1. You can check your syncing status using the optimism_syncStatus
RPC on the op-node
container. Example:
command -v jq &> /dev/null || { echo "jq is not installed" 1>&2 ; }
echo Latest synced block behind by: \
$((($( date +%s )-\
$( curl -s -d '{"id":0,"jsonrpc":"2.0","method":"optimism_syncStatus"}' -H "Content-Type: application/json" http://localhost:9545 |
jq -r .result.unsafe_l2.timestamp))/60)) minutes