Run Ethereum multi-client devnets using Kurtosis and the ethpandaops/ethereum-package. Use for spinning up local testnets, validating cross-client interop, testing fork transitions, running assertoor checks, debugging CL/EL client interactions, or verifying new feature implementations across multiple consensus and execution clients.
Resources
1Install
npx skillscat add lodekeeper/dotfiles/kurtosis-devnet Install via the SkillsCat registry.
Kurtosis Devnet
Run Ethereum consensus/execution client devnets via kurtosis + ethereum-package.
Quick Start
# Start devnet from config
# Always use --image-download always to ensure external images are up to date
kurtosis run github.com/ethpandaops/ethereum-package \
--enclave <name> \
--args-file network_params.yaml \
--image-download always
# List enclaves
kurtosis enclave ls
# Inspect services
kurtosis enclave inspect <name>
# View logs
kurtosis service logs <enclave> <service-name>
kurtosis service logs <enclave> <service-name> --follow
# Cleanup
kurtosis enclave rm -f <name>Config File (network_params.yaml)
See references/config-reference.md for full config structure and examples.
Key sections:
participants: list of CL+EL client pairs with images, flags, validator countsnetwork_params: fork epochs, slot time, network-level settingsadditional_services: dora (explorer), assertoor (testing), prometheus, grafana, etc.assertoor_params: automated chain health checksport_publisher: expose CL/EL ports to host
Building Custom Client Images
When testing local branches, build a Docker image first:
# Lodestar (fast build)
cd ~/lodestar && docker build -t lodestar:custom -f Dockerfile.dev .
# Then reference in config
# cl_image: lodestar:customUse Dockerfile.dev over Dockerfile for faster builds (skips production optimizations).
Service Naming Convention
Kurtosis names services as: {role}-{index}-{cl_type}-{el_type}
Examples:
cl-1-lodestar-reth— first CL node (Lodestar with Reth EL)el-1-reth-lodestar— corresponding EL nodevc-1-lodestar-reth— validator client
Accessing Services
After kurtosis enclave inspect <name>, find mapped ports:
# CL beacon API (find actual port from inspect output)
curl http://127.0.0.1:<mapped-port>/eth/v1/node/syncing
# Or use port_publisher for predictable ports:
port_publisher:
cl:
enabled: true
public_port_start: 33000 # cl-1=33000, cl-2=33005, etc.
el:
enabled: true
public_port_start: 32000Port publisher assigns sequential ports (step of 5 per service).
Assertoor (Automated Testing)
Add to config:
additional_services:
- assertoor
assertoor_params:
run_stability_check: true # chain stability, finality, no reorgs
run_block_proposal_check: true # every client pair proposes a blockCheck results via assertoor API or Dora dashboard.
Supernode Mode
Set supernode: true on participants to run beacon+validator in a single process (faster startup, simpler topology). Each supernode handles its own validators without separate VC.
Mixed-Client Topologies
For cross-client interop testing, mix CL and EL types:
participants:
- cl_type: lodestar
el_type: reth
...
- cl_type: lighthouse
el_type: geth
...
- cl_type: prysm
el_type: geth
...Log Levels
For debugging, set global_log_level: "debug" in the config to get debug logs from all nodes:
global_log_level: "debug"This applies to all participants. For per-participant overrides, use cl_log_level / el_log_level on individual entries.
Monitoring & Debugging
# Stream logs from a specific service
kurtosis service logs <enclave> cl-1-lodestar-reth --follow
# Save all logs for analysis
for svc in $(kurtosis enclave inspect <enclave> 2>/dev/null | grep -oP 'cl-\d+-\S+'); do
kurtosis service logs <enclave> $svc > "/tmp/${svc}.log" 2>&1
done
# Dora explorer (if enabled)
# Find port via: kurtosis enclave inspect <enclave> | grep dora
# Check chain finality
curl -s http://127.0.0.1:<port>/eth/v1/beacon/states/head/finality_checkpoints | jq
# Check peer count
curl -s http://127.0.0.1:<port>/eth/v1/node/peers | jq '.data | length'Common Patterns
Fork Transition Testing
network_params:
electra_fork_epoch: 0
fulu_fork_epoch: 1 # fork at epoch 1 (slot 32)
seconds_per_slot: 6 # faster for testingNodes Without Validators (Observer Nodes)
- cl_type: lodestar
cl_image: lodestar:custom
el_type: reth
count: 1
validator_count: 0 # observer-onlyExtra CL/VC Params
cl_extra_params:
- --targetPeers=8
- --activateZkvm
vc_extra_params:
- --suggestedFeeRecipient=0x...Troubleshooting
| Issue | Fix |
|---|---|
| Peers not connecting | Increase --targetPeers, check directPeers config |
| No finality | Need ≥2/3 validators attesting; check VC logs |
| "discv5 has no boot enr" | Harmless startup warning, ignore |
| Port conflicts | Change public_port_start or stop conflicting enclaves |
| Image not found | Ensure Docker image is built locally or available in registry |
| Slow startup | Use Dockerfile.dev for local builds; reduce validator count |
Wait for Finality
Finality typically takes 2-3 epochs after genesis. With seconds_per_slot: 6 and 32 slots/epoch:
- 1 epoch ≈ 192s (3.2 min)
- First finalization ≈ epoch 3-4 boundary (≈10-13 min)
Monitor: curl -s http://<port>/eth/v1/beacon/states/head/finality_checkpoints | jq '.data.finalized.epoch'
Self-Maintenance
If any commands, file paths, URLs, or configurations in this skill are outdated or no longer work, update this SKILL.md with the correct information after completing your current task. Skills should stay accurate and self-healing — fix what you find broken.