Ethereum Execution (Eth 1.0) Client built in TypeScript/JavaScript. |
---|
The EthereumJS Client is an Ethereum Execution Client (similar to go-ethereum or Nethermind) written in TypeScript
/JavaScript
, the non-Smart-Contract language Ethereum dApp developers are most familiar with. It is targeted to be a client for research and development and not meant to be used in production on mainnet
for the foreseeable future (out of resource and security considerations).
Here are some use cases:
- Sync the main Ethereum networks (
mainnet
,goerli
,rinkeby
,sepolia
...) - Set up your own local development networks (PoA Clique or PoW with CPU miner)
- Run a network with your own custom EthereumJS VM
- Analyze what's in the Ethereum
mainnet
transaction pool - Run experiments with Ethereum browser sync (see example)
The client has an extremely modular design by building upon central other libraries in the EthereumJS monorepo (VM, Merkle Patricia Tree, Blockchain, Block, tx, devp2p and Common) and is therefore extremely well suited for a deep dive into Ethereum protocol development.
We invite you to explore and would be delighted if you give us feedback on your journey! 🙂 ❤️
npm install -g @ethereumjs/client
For development purposes the client can also be set up like this:
- Clone the monorepo with
git clone https://github.com/ethereumjs/ethereumjs-monorepo.git
- Set things up and install dependencies (see monorepo docs)
- Run the client with
npm run client:start
from theclient
root directory (e.g.packages/client
if working on a local copy of the monorepo)
You can get the client up and running by going to the shell and run:
# npm installation
ethereumjs
# GitHub installation
npm run client:start
And pass in CLI paramters like:
# npm installation
ethereumjs --network=goerli
# GitHub installation
npm run client:start -- --network=goerli
To see a help page with the full list of client options available run:
ethereumjs --help
The EthereumJS client is tightly integrated with the EthereumJS Common library and gets its network state and information from this library and supports all networks support by Common
.
The main supported networks are:
mainnet
rinkeby
ropsten
goerli
sepolia
(v0.3.0
+)
Use the CLI --network
option to switch the network:
ethereumjs --network=rinkeby
The client currently supports full
sync being set as a default and has experimental support for light
sync.
The EthereumJS client is participating actively in the different testnets preparing for the Merge hardfork and the team is following up on implementing the Merge specs and updating along new versions of the specification.
There are dedicated instructions on how to setup for running the client in conjunction with a consensus (Eth 2.0) client (e.g. Lodestar).
Testnet Instructions:
When running an Execution Layer (EL) client (so here: the EthereumJS
client) together with a Consensus Layer (CL) client (e.g. Lodestar or Lighthouse) it is generally recommended to first start and last stop the EL client, since this client acts as a service worker post-Merge and the CL client is depending on a working EL client connection.
Currently the following sync scenarios are supported:
- EL and CL Live Transition: both clients follow the head of the respective chains pre Merge and go through the Merge transition live and in sync
- EL synced to Merge block, CL Merge transition: the EL client has been synced to the Merge transitition block and is waiting for CL requests, the CL is syncing through the transition and starts EL requests along
- Resume syncing with equally synced clients / CL client behind: sync can be resumed with EL client and CL client synced to the respective counterparts of the beacon and embedded execution chain, CL client can also be somewhat behind the EL client
- CL client ahead of EL client: if the CL client has been synced without the EL client syncing in lockstep on/after merge (e.g. using weak subjectivity sync) and is therefore ahead of the EL client post-merge, the EL backfills blocks and brings the chain up-to-date with the head fcUs being issued by the CL via optimistic (beacon) sync (since
v0.5.0
)
The EthereumJS client supports running custom chains based on a custom chain configuration. There are two ways of reading in custom chain configuration parameters:
We have got our own flexible chain configuration and genesis state configuration format applied in the Common
library, see the Common
chain JSON files as well as corresponding Genesis JSON files for inspiration.
Custom chain files following this format can be passed in to the client with the following options:
ethereumjs --customChain=[COMMON_FORMAT_CHAIN_FILE] --customGenesisState=[COMMON_FORMAT_GENESIS_STATE]
It is also possible to use a chain configuration as defined by the Go-Ethereum genesis.json file format.
Use the following option to pass in to the client:
ethereumjs --gethGenesis=[GETH_GENESIS_JSON_FILE]
The client supports private custom network mining by using the --mine
option together with passing in a comma separated list of accounts with the --unlock
option:
ethereumjs --mine --unlock=[ADDRESS1],[ADDRESS2],...
Note that this feature is in beta
and shouldn't be used with accounts holding a substantial amount of Ether
on mainnet (or other valuable assets) for security reasons.
The client provides a quick way to get a local instance of a blockchain up and running using the --dev
command. This will start up a private PoA clique network with a prefunded account that mines block on 10 second intervals. The prefunded account and its private key are printed to the screen when the client starts. When paired with the --rpc
command, you have a ready-made environment for local development.
ethereumjs --dev --rpc
==================================================
Account generated for mining blocks:
Address: 0xd8066d5822138e7c76d1565deb249f5f7ae370fa
Private key: 0x6239e36ab8b27212868a1aa3f9c3b88b084075ea56aa4979d206371f065d3feb
WARNING: Do not use this account for mainnet funds
==================================================
Please heed the warning and do not use the provided account/private key for mainnet funds.
This can also be paired with the --unlock
command if you would like to specify the miner/prefunded account:
ethereumjs --dev --rpc --unlock=0xd8066d5822138e7c76d1565deb249f5f7ae370fa
Note: If the --dev
command is used in conjunction with --unlock
to use a predefined account, the blockchain's state will be preserved between consecutive runs. If you try to use a different predefined account, you may see errors related to incompatible genesis blocks. Simply run the client with the --dev
flag by itself and use the new prefunded account provided by the client in further rounds of execution.
To explicitly set the miner coinbase (etherbase) specify --minerCoinbase=[ADDRESS]
- otherwise this will default to the primary account.
The --dev
command defaults to --dev=poa
. If you would like to use PoW ethash with CPU miner (warning: slow) then pass --dev=pow
.
See also this diagram for an overview of the client structure with the initialization and message flow.
You can expose a JSON-RPC interface along a client run with:
ethereumjs --rpc
To run just the server without syncing:
ethereumjs --rpc --maxPeers=0
Currently only a small subset of RPC
methods are implemented.(*) You can have a look at the
./lib/rpc/modules/ source folder or the tracking issue
#1114 for an overview.
(*) Side note: implementing RPC methods is actually an extremely thankful task for a first-time contribution on the project *hint* hint. 😄
You can use cURL
to request data from an API endpoint. Here is a simple example for
web3_clientVersion:
curl -X POST -H "Content-Type: application/json" --data '{"jsonrpc":"2.0","id":1,"method":"web3_clientVersion", "params": []}' http://localhost:8545
Note that "params": []
can also be omitted in this case.
Or - somewhat more convenient and with formatted output - with a tool like httpie:
http POST http://localhost:8545 jsonrpc=2.0 id=1 method=web3_clientVersion params:=[]
Note the :=
separator for the params
parameter to
indicate raw JSON
as an input.
This will give you an output like the following:
{
"id": "1",
"jsonrpc": "2.0",
"result": "EthereumJS/0.0.5/darwin/node12.15.0"
}
Here's an example for a call on an endpoint with the need for parameters. The following call uses
the eth_getBlockByNumer endpoint
to request data for block number 436 (you can use a tool like
RapidTables for conversion to hex
):
curl -X POST -H "Content-Type: application/json" --data '{"jsonrpc":"2.0","method":"eth_getBlockByNumber","params":["0x1b4", true],"id":1}' http://127.0.0.1:8545
Same with httpie
:
http POST http://localhost:8545 jsonrpc=2.0 id=1 method=eth_getBlockByNumber params:='["0x1b4",true]'
Output:
{
"jsonrpc": "2.0",
"id": "1",
"result": {
"number": "0x1b4",
"hash": "0xdc0818cf78f21a8e70579cb46a43643f78291264dda342ae31049421c82d21ae",
"parentHash": "0xe99e022112df268087ea7eafaf4790497fd21dbeeb6bd7a1721df161a6657a54",
"nonce": "0x689056015818adbe",
"sha3Uncles": "0x1dcc4de8dec75d7aab85b567b6ccd41ad312451b948a7413f0a142fd40d49347",
"logsBloom": "0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000",
"transactionsRoot": "0x56e81f171bcc55a6ff8345e692c0f86e5b48e01b996cadc001622fb5e363b421",
"stateRoot": "0xddc8b0234c2e0cad087c8b389aa7ef01f7d79b2570bccb77ce48648aa61c904d",
"receiptsRoot": "0x56e81f171bcc55a6ff8345e692c0f86e5b48e01b996cadc001622fb5e363b421",
"miner": "0xbb7b8287f3f0a933474a79eae42cbca977791171",
"difficulty": "0x4ea3f27bc",
"totalDifficulty": "0x78ed983323d",
"extraData": "0x476574682f4c5649562f76312e302e302f6c696e75782f676f312e342e32",
"size": "0x747",
"gasLimit": "0x1388",
"gasUsed": "0x0",
"timestamp": "0x55ba467c",
"transactions": [],
"uncles": []
}
}
For an overview on the design goals which served as a guideline on design decisions as well as some structural client overview see the dedicated DESIGN.md document.
To get a start on customizing the client and using it programmatically see the code from ./bin/cli.ts to get an idea of how an EthereumClient instance is invoked programmatically.
We would love to hear feedback from you on what you are planning and exchange on ideas how a programmatic exposure of the client API can be achieved more systematically and useful for third-party development use.
For some guidance on how to setup local testnetworks see the examples on local debugging and setting up a private network with Geth.
The client's logging verbosity level can be set with --loglevel
. Available levels are
error
, warn
, info
, debug
.
ethereumjs --loglevel=debug
For more in-depth debugging on networking the underlying devp2p library integrates with the debug package and can also be used from within a client execution context:
DEBUG=*,-babel [CLIENT_START_COMMAND]
The above command outputs the log messages from all devp2p
debug loggers available. For a more targeted logging the different loggers can also be activated separately, e.g.:
DEBUG=devp2p:rlpx,devp2p:eth,-babel [CLIENT_START_COMMAND]
To update the structure diagram files in the root folder open the client.drawio
file in draw.io, make your changes, and open a PR with the updated files. Export svg
and png
with border
width=20
and transparency=false
. For png
go to "Advanced" and select 300 DPI
.
See our organizational documentation for an introduction to EthereumJS
as well as information on current standards and best practices.
If you want to join for work or do improvements on the libraries have a look at our contribution guidelines.