EthereumJS - Developer Docs
May 29, 2026 · View on GitHub
This guide provides an overview of the monorepo, development tools used, shared configuration and additionally covers some advanced topics.
It is intended to be both an entrypoint for external contributors as well as a reference point for team members.
Contents
Monorepo
Structure
The EthereumJS project uses npm workspaces to manage all the packages in our monorepo and link packages together.
Key Directories
/packages- Contains all EthereumJS packages/config- Shared configuration files and scriptspackages/ethereum-tests- Git submodule with Ethereum test vectors (legacy)packages/execution-spec-tests- Git submodule with selected execution-spec-tests fixtures
Scripts
The ./config/cli directory contains helper scripts referenced in package.json files:
coverage.sh- Runs test coveragets-build.sh- Builds TypeScript for productionts-compile.sh- Compiles TypeScript for development
Workflow
Common Commands
- Clean the workspace:
npm run clean- Removes build artifacts and node_modules - Lint code:
npm run lint- Check code style with ESLint v9 and Biome - Fix linting issues:
npm run lint:fix- Automatically fix style issues - Build all packages:
npm run build --workspaces- Build all packages in the monorepo - Build documentation:
npm run docs:build- Generate documentation for all packages
Working on a Specific Package
To focus on a single package (e.g., VM):
- Navigate to the package directory:
cd packages/vm - Run tests:
npm run test - Run a specific test:
npx vitest test/path/to/test.spec.ts - Build just that package:
npm run build --workspace=@ethereumjs/vm
Releases
Overview
Releases are done in sync for all active packages and all libraries are always bumped to a same new version number. Library combinations with matching versions are CI tested and ensured to be compatible with each other.
Most release rounds are done as bugfix releases, including releases of non-finalized EIP versions. Minor releases are done for hardfork finalization and otherwise outstanding selected features. Major release rounds are rarely done and are reserved to bundle structural breaking changes which come along significant changes to the API.
Process
Version/Dependency Update & Publish Script
We have a release script that handles version bumping and publishing for all packages. It supports both regular releases and lightweight in-between releases (nightly, alpha).
tsx scripts/release-npm.ts [--bump-version=<version>] [--publish=<tag>] [--scope=<scope>] [--otp=<code>]
Options:
--bump-version=<version>- Bump package versions to the specified version (skips publish unless--publishis also set)--publish=<tag>- Publish packages with the specified npm tag (default:latest)--scope=<scope>- Publish under a different npm scope (default:ethereumjs, see Fork Releases)--otp=<code>- One-time password when npm 2FA is enabled
With no flags, the script publishes the current package versions to npm under the latest tag (typical flow after a manual version bump and CHANGELOG prep).
npm authentication (required for publish):
- Interactive (maintainer laptop):
npm login— stores a token in~/.npmrc. Use a granular access token with publish access to the@ethereumjsscope (recommended over classic tokens). - Token in config: add a registry
_authTokenentry to~/.npmrc(see npm registry auth; never commit tokens). Same granular publish token as above. - 2FA: pass
--otp=<code>when your npm account requires it.
The script runs npm whoami before publishing and exits if you are not authenticated.
What the script does:
- Active packages: Updates version numbers and
@ethereumjs/*dependency references (with--bump-version), then publishes in dependency order (rlp→ … →vm) - Deprecated packages + testdata: Only updates (active)
@ethereumjs/*dependency references when bumping (keeps their own version unchanged, not published) prepublishOnlyper package: clean, build, and test run automatically vianpm publish(can take a while for the full round)
Examples:
# Publish current versions (after bump + CHANGELOG prep) — most common
tsx scripts/release-npm.ts
# Same, explicit tag
tsx scripts/release-npm.ts --publish=latest
# Bump versions only (no publish) - for preparing a release
tsx scripts/release-npm.ts --bump-version=10.1.0
# Bump versions and publish - full release in one step
tsx scripts/release-npm.ts --bump-version=10.1.0 --publish=latest
# Lightweight nightly release
tsx scripts/release-npm.ts --bump-version=10.1.1-nightly.1 --publish=nightly
# Publish with 2FA one-time password
tsx scripts/release-npm.ts --otp=123456
Fork Releases (Feel Your Protocol)
The release script supports publishing all packages under a different npm scope via the --scope flag. This is used by Feel Your Protocol to publish fork releases from feature branches (e.g. EIP prototype implementations) that can be integrated as separate dependencies alongside the official @ethereumjs/* packages.
# Publish all packages under @feelyourprotocol scope
tsx scripts/release-npm.ts --scope=feelyourprotocol --bump-version=8141.0.0 --publish=latest
When --scope is set to a value other than ethereumjs, the script:
- Rewrites package names:
@ethereumjs/evm→@<scope>/evm - Rewrites inter-package dependency references to match the new scope
- Rewrites
@ethereumjs/import paths in all source files undersrc/ - Skips deps-only packages (not published, rewriting would break local dev)
- Publishes with
--access=public(required for new scoped npm orgs)
The --scope flag is fully generic and not tied to any specific npm org.
CHANGELOG Preparation
The following prompt has been tested with Cursor IDE to work well for CHANGELOG updates (please update placeholders in the first paragraph accordingly):
I want to do a new release round for all active packages listed in @README.md. Version bump has already been done, see an exemplary [ REFERENCE package.json FILE ] file, release is target for today. Last release round has been done on [ ENTER DATE IN FORMAT: April 29 2025 ] along commit [ ENTER COMMIT HASH, e.g.: 9e461f54312bf20c710b43ab73f7d3ad753f8765 ]. An exemplary CHANGELOG.md file is [ REFERENCE e.g. block CHANGELOG.md file ].
Can you please add new sections in the CHANGELOG files and add one-line summaries for the user-facing changes? For this please go for the commits since last release, one commit represents one PR due to our (squash) merge policy. You can leave out PRs only updating documentation, code in the examples folder or tests. Also tooling infrastructure (linting,...) and CI updating PRs can be left out. New support for new and deprecation for older Node.js as well as TypeScript versions should be added. Version updates for external dependencies - so not from within the monorepo - should be added as well.
Here is an example for the format of a change/PR entry:
- New default hardfork: `Shanghai` -> `Cancun`, see PR [#3566](https://github.com/ethereumjs/ethereumjs-monorepo/pull/3566)
For the CHANGELOG files you have not added lines in this step please nevertheless add a CHANGELOG entry (we do releases for all active packages no matter the changeset) and enter the following sentence: Maintenance release, no active changes.
Windows Users Note
Windows users might encounter errors with script paths. To fix, configure Git bash as the script shell:
npm config set script-shell "C:\\Program Files (x86)\\git\\bin\\bash.exe"
To reset this setting:
npm config delete script-shell
Development Tools
TypeScript
All packages use TypeScript with a shared base configuration.
Configuration Files
Each package should have:
tsconfig.json- For development and testingtsconfig.prod.json- For building production releases
Example tsconfig.json:
{
"extends": "../../config/tsconfig.json",
"include": ["src/**/*.ts", "test/**/*.ts"]
}
Example tsconfig.prod.json:
{
"extends": "../../config/tsconfig.prod.json",
"include": ["src/**/*.ts"],
"compilerOptions": {
"outDir": "./dist"
}
}
Build Commands
Use these commands in your package scripts:
{
"scripts": {
"tsc": "../../config/cli/ts-compile.sh",
"build": "../../config/cli/ts-build.sh"
}
}
Linting
We use ESLint v9 and Biome for code style enforcement and linting.
Configuration Files
Each package includes:
eslint.config.mjs- package specific ESLint configuration that extends the repository wide config
Commands
Commands area available on both root and package levels.
Run npm run lint to find lint issues and npm run lint:fix to fix fixable lint issues.
Spellcheck
We use cspell to do spellchecking.
Configuration Files
The following two configuration files include a list of allowed words (add yours if you have one necessary) as well as some additional configuration, separate for docs and code.
config/cspell-md.json|Markdownconfig/cspell-ts.json|TypeScript
Commands
Commands area available on both root and package levels.
{
"scripts": {
"sc": "npm run spellcheck",
"spellcheck": "npm run spellcheck:ts && npm run spellcheck:md",
"spellcheck:ts": "npx cspell --gitignore -c ../../config/cspell-ts.json ...",
"spellcheck:md": "npx cspell --gitignore -c ../../config/cspell-md.json ..."
}
}
Testing
The project uses Vitest for testing with c8 for code coverage.
General
Each package includes one or more test scripts. To run all tests in any package, use npm run test. Refer to the package.json for more specifics.
To run a specific test and watch for changes:
npx vitest test/path/to/test.spec.ts
Browser
We use vitest with playwright to run browser tests in real Chromium (headless).
Local: browser tests are optional unless you are working on bundling or browser-specific behaviour. Install Chromium once (Chromium only — not the full Playwright browser set):
npm run install-browser-deps
# equivalent to: npx playwright install chromium
Then run npm run test:browser in a package, or npm run test:browser from the monorepo root.
CI: deps are restored on the host runner (same cache as other jobs). Browser tests run in the official Playwright Docker image (mcr.microsoft.com/playwright:v1.60.0-noble) via docker run — preinstalled browsers, no Chromium download per run. Keep the image tag in .github/workflows/browser.yml in sync with the playwright version in package-lock.json.
Advanced Topics
Linking to an External Library
Quick Summary
To test packages with an external project locally, use npm link:
- Build the package you want to test:
cd packages/package-name
npm run build
- Link the package globally:
npm link
- In your test project, link to the local package:
cd path/to/your/project
npm link @ethereumjs/package-name
-
When you make changes to your package, rebuild it for the changes to be reflected.
-
When done testing, unlink:
# In your test project
npm unlink --no-save @ethereumjs/package-name
# In the package directory
npm unlink
When making changes to the linked package, rebuild it for the changes to be reflected in your test project.
Shared Dependencies
Common development dependencies (e.g. eslint, biome) are defined in the root package.json.
Additional Docs
There are selected additional developer docs available to get more deep on certain topics. The following is an overview.
VM
VM Docs for testing, debugging and VM/EVM profiling.
Client
Client Docs for running Hive tests.