Metal Network Runner
The Metal Network Runner (ANR) allows a user to define, create and interact with a network of Metal nodes. It can be used for development and testing.
Developing P2P systems is hard, and blockchains are no different. A developer can't just focus on the functionality of a node, but needs to consider the dynamics of the network, the interaction of nodes and emergent system properties. A lot of testing can't be addressed by unit testing, but needs a special kind of integration testing, where the code runs in interaction with other nodes, attempting to simulate real network scenarios.
In the context of metal, Subnets are a special focus which requires new tooling and support for playing, working and testing with this unique feature of the Metal ecosystem.
The ANR aims at being a tool for developers and system integrators alike, offering functionality to
run networks of MetalGo nodes with support for custom node, Subnet and network configurations,
allowing to locally test code before deploying to Mainnet or even public testnets like tahoe
.
Note that this tool is not for running production nodes, and that because it is being heavily developed right now, documentation might differ slightly from the actual code.
Installation
curl -sSfL https://raw.githubusercontent.com/MetalBlockchain/metal-network-runner/main/scripts/install.sh | sh -s
The script installs the binary inside the ~/bin
directory. If the directory doesn't exist,
it will be created.
Please make sure that ~/bin
is in your $PATH
:
export PATH=~/bin:$PATH
To add it to your path permanently, add an export command to your shell initialization script. If
you run bash
, use .bashrc
. If you run zsh
, use .zshrc
.
Furthermore, METALGO_EXEC_PATH
should be set properly in all shells you run commands related
to Metal Network Runner. We strongly recommend that you put the following in to your shell's
configuration file.
# replace execPath with the path to MetalGo on your machine
# e.g., ${HOME}/go/src/github.com/!metal!blockchain/metalgo/build/metalgo
METALGO_EXEC_PATH="${HOME}/go/src/github.com/!metal!blockchain/metalgo/build/metalgo"
Unless otherwise specified, file paths given below are relative to the root of this repository.
Usage
There are two main ways to use the network-runner:
Run ANR as a binary
This is the recommended approach for most use cases. Doesn't require Golang installation and provides a RPC server with an HTTP API and a client library for easy interaction.
Import this repository into your go program
This allows for custom network scenarios and high flexibility, but requires more code to be written.
Running the binary, the user can send requests to the RPC server in order to start a network, create
Subnets, add nodes to the network, remove nodes from the network, restart nodes, etc. You can make
requests through the metal-network-runner
command or by making API calls. Requests are
"translated" into gRPC and sent to the server.
Each node can then also be reached via API endpoints which each node exposes.
Examples
When running with the binary, ANR runs a server process as an RPC server which then waits for API calls and handles them. Therefore we run one shell with the RPC server, and another one for issuing calls.
Start the Server
metal-network-runner server \
--log-level debug \
--port=":8080" \
--grpc-gateway-port=":8081"
Note that the above command will run until you stop it with CTRL + C
. Further commands will have
to be run in a separate terminal.
The RPC server listens to two ports:
port
: the main gRPC port (see gRPC).grpc-gateway-port
: the gRPC gateway port (see gRPC-gateway), which allows for HTTP requests.
When using the binary to issue calls, the main port will be hit. In this mode, the binary executes
compiled code to issue calls. Alternatively, plain HTTP can be used to issue calls, without the need
to use the binary. In this mode, the grpc-gateway-port
should be queried.
Each of the examples below will show both modes, clarifying its usage.
Run Queries
Ping the Server
curl -X POST -k http://localhost:8081/v1/ping -d ''
or
metal-network-runner ping \
--log-level debug \
--endpoint="0.0.0.0:8080"
Start a New Metal Network with Five Nodes
curl -X POST -k http://localhost:8081/v1/control/start -d '{"execPath":"'${METALGO_EXEC_PATH}'","numNodes":5}'
or
metal-network-runner control start \
--log-level debug \
--endpoint="0.0.0.0:8080" \
--number-of-nodes=5 \
--metalgo-path ${METALGO_EXEC_PATH}
Additional optional parameters which can be passed to the start command:
--plugin-dir ${METALGO_PLUGIN_PATH} \
--blockchain-specs '[{"vm_name":"subnetevm","genesis":"/tmp/subnet-evm.genesis.json"}]' \
--global-node-config '{"index-enabled":false, "api-admin-enabled":true,"network-peer-list-gossip-frequency":"300ms"}' \
--custom-node-configs" '{"node1":{"log-level":"debug","api-admin-enabled":false},"node2":{...},...}'
--plugin-dir
and --blockchain-specs
are parameters relevant to Subnet operation.
--plugin-dir
can be used to indicate to ANR where it will find plugin binaries for your own VMs.
It is optional. If not set, ANR will assume a default location which is relative to the
metalgo-path
given.
--blockchain-specs
specifies details about how to create your own blockchains. It takes a JSON
array for each blockchain, with the following possible fields:
"vm_name": human readable name for the VM
"genesis": path to a file containing the genesis for your blockchain (must be a valid path)
The network-runner supports MetalGo node configuration at different levels.
If neither
--global-node-config
nor--custom-node-configs
is supplied, all nodes get a standard set of config options. Currently this set contains:{
"network-peer-list-gossip-frequency": "250ms",
"network-max-reconnect-delay": "1s",
"public-ip": "127.0.0.1",
"health-check-frequency": "2s",
"api-admin-enabled": true,
"api-ipcs-enabled": true,
"index-enabled": true
}--global-node-config
is a JSON string representing a single MetalGo config, which will be applied to all nodes. This makes it easy to define common properties to all nodes. Whatever is set here will be combined with the standard set above.--custom-node-configs
is a map of JSON strings representing the complete network with individual configs. This allows to configure each node independently. If set,--number-of-nodes
will be ignored to avoid conflicts.The configs can be combined and will be merged, that is one could set global
--global-node-config
entries applied to each node, and also set--custom-node-configs
for additional entries.Common
--custom-node-configs
entries override--global-node-config
entries which override the standard set.The following entries will be ignored in all cases because the network-runner needs to set them internally to function properly:
--log-dir
--db-dir
--http-port
--staking-port
--public-ipc
Wait for All the Nodes in the Cluster to Become Healthy
curl -X POST -k http://localhost:8081/v1/control/health -d ''
or
metal-network-runner control health \
--log-level debug \
--endpoint="0.0.0.0:8080"
The response to this call is actually pretty large, as it contains the state of the whole cluster.
At the very end of it there should be a text saying healthy:true
(it would say false
if it
wasn't healthy).
Get API Endpoints of All Nodes in the Cluster
curl -X POST -k http://localhost:8081/v1/control/uris -d ''
or
metal-network-runner control uris \
--log-level debug \
--endpoint="0.0.0.0:8080"
Query Cluster Status from the Server
curl -X POST -k http://localhost:8081/v1/control/status -d ''
or
metal-network-runner control status \
--log-level debug \
--endpoint="0.0.0.0:8080"
Stream Cluster Status
metal-network-runner control \
--request-timeout=3m \
stream-status \
--push-interval=5s \
--log-level debug \
--endpoint="0.0.0.0:8080"
Remove (Stop) a Node
curl -X POST -k http://localhost:8081/v1/control/removenode -d '{"name":"node5"}'
or
metal-network-runner control remove-node node5 \
--request-timeout=3m \
--log-level debug \
--endpoint="0.0.0.0:8080" \
Restart a Node
In this example we are stopping the node named node1
.
Note: By convention all node names start with node
and a number. We suggest to stick to this
convention to avoid issues.
# e.g., ${HOME}/go/src/github.com/!metal!blockchain/metalgo/build/metalgo
METALGO_EXEC_PATH="metalgo"
Note that you can restart the node with a different binary by providing
curl -X POST -k http://localhost:8081/v1/control/restartnode -d '{"name":"node1","execPath":"'${METALGO_EXEC_PATH}'"}'
or
metal-network-runner control restart-node node1 \
--request-timeout=3m \
--log-level debug \
--endpoint="0.0.0.0:8080" \
--metalgo-path ${METALGO_EXEC_PATH}
Add a Node
In this example we are adding a node named node99
.
# e.g., ${HOME}/go/src/github.com/!metal!blockchain/metalgo/build/metalgo
METALGO_EXEC_PATH="metalgo"
Note that you can add the new node with a different binary by providing
curl -X POST -k http://localhost:8081/v1/control/addnode -d '{"name":"node99","execPath":"'${METALGO_EXEC_PATH}'"}'
or
metal-network-runner control add-node node99 \
--request-timeout=3m \
--endpoint="0.0.0.0:8080" \
--metalgo-path ${METALGO_EXEC_PATH}
It's also possible to provide individual node config parameters:
--node-config '{"index-enabled":false, "api-admin-enabled":true,"network-peer-list-gossip-frequency":"300ms"}'
--node-config
allows to specify specific MetalGo config parameters to the new node.
See here for the reference of supported flags.
Note: The following parameters will be ignored if set in --node-config
, because the network
runner needs to set its own in order to function properly: --log-dir
--db-dir
Note: The following Subnet parameters will be set from the global network configuration to this node:
--whitelisted-subnets
--plugin-dir
Terminate the Cluster
Note that this will still require to stop your RPC server process with Ctrl-C
to free the shell.
curl -X POST -k http://localhost:8081/v1/control/stop -d ''
or
metal-network-runner control stop \
--log-level debug \
--endpoint="0.0.0.0:8080"
Subnets
For general Subnet documentation, please refer to Subnets. ANR can be a great helper working with Subnets, and can be used to develop and test new Subnets before deploying them in public networks.
RPC Server Subnet-EVM Example
The Subnet-EVM is a simplified version of Coreth VM (C-Chain). This chain implements the Ethereum Virtual Machine and supports Solidity smart-contracts as well as most other Ethereum client functionality. It can be used to create your own fully Ethereum-compatible Subnet running on Metal. This means you can run your Ethereum-compatible dApps in custom Subnets, defining your own gas limits and fees, and deploying solidity smart-contracts while taking advantage of Metal's validator network, fast finality, consensus mechanism and other features. Essentially, think of it as your own Ethereum where you can concentrate on your business case rather than the infrastructure. See Subnet-EVM for further information.
Subnet-CLI
ANR requires an additional tool, such as subnet-cli
, to be able to
create the necessary configuration to deploy a Subnet in a local custom test-network. For a smoother
experience, we recommend using Metal-CLI though, as it
hides all this complexity away!
Install and start the RPC server just as in start the server Make sure the server is up:
curl -X POST -k http://localhost:8081/v1/ping -d ''
Start the Cluster with Custom VMs
First, download/install subnet-cli
:
# or download from https://github.com/MetalBlockchain/subnet-cli/releases
cd ${HOME}/go/src/github.com/!metal!blockchain/subnet-cli
go install -v .
Create a VM ID:
subnet-cli create VMID subnetevm
# srEXiWaHuhNyGwPUi444Tu47ZEDwxTWrbQiuD7FmgSAQ6X7Dy
Build or...
rm -rf ${HOME}/go/src/github.com/!metal!blockchain/metalgo/build
cd ${HOME}/go/src/github.com/!metal!blockchain/metalgo
./scripts/build.sh
...download metalgo
(if not done already)
Clone and build the subnet-evm
plugin (requires golang
installation):
git clone https://github.com/MetalBlockchain/subnet-evm ${HOME}/go/src/github.com/!metal!blockchain/subnet-evm
cd ${HOME}/go/src/github.com/!metal!blockchain/subnet-evm
go build -v \
-o ${HOME}/go/src/github.com/!metal!blockchain/metalgo/build/plugins/srEXiWaHuhNyGwPUi444Tu47ZEDwxTWrbQiuD7FmgSAQ6X7Dy \
./plugin
Verify everything has been built correctly:
find ${HOME}/go/src/github.com/!metal!blockchain/metalgo/build
# should yield something like:
# .../build
# .../build/plugins
# .../build/plugins/srEXiWaHuhNyGwPUi444Tu47ZEDwxTWrbQiuD7FmgSAQ6X7Dy
# .../build/plugins/evm
# .../build/metalgo
Generate the genesis for the custom VM
export CHAIN_ID=99999
export GENESIS_ADDRESS="0x8db97C7cEcE249c2b98bDC0226Cc4C2A57BF52FC"
cat <<EOF > /tmp/subnet-evm.genesis.json
{
"config": {
"chainId": $CHAIN_ID,
"homesteadBlock": 0,
"eip150Block": 0,
"eip150Hash": "0x2086799aeebeae135c246c65021c82b4e15a2c451340993aacfd2751886514f0",
"eip155Block": 0,
"eip158Block": 0,
"byzantiumBlock": 0,
"constantinopleBlock": 0,
"petersburgBlock": 0,
"istanbulBlock": 0,
"muirGlacierBlock": 0,
"subnetEVMTimestamp": 0,
"feeConfig": {
"gasLimit": 20000000,
"minBaseFee": 1000000000,
"targetGas": 100000000,
"baseFeeChangeDenominator": 48,
"minBlockGasCost": 0,
"maxBlockGasCost": 10000000,
"targetBlockRate": 2,
"blockGasCostStep": 500000
}
},
"alloc": {
"${GENESIS_ADDRESS}": {
"balance": "0x52B7D2DCC80CD2E4000000"
}
},
"nonce": "0x0",
"timestamp": "0x0",
"extraData": "0x00",
"gasLimit": "0x1312D00",
"difficulty": "0x0",
"mixHash": "0x0000000000000000000000000000000000000000000000000000000000000000",
"coinbase": "0x0000000000000000000000000000000000000000",
"number": "0x0",
"gasUsed": "0x0",
"parentHash": "0x0000000000000000000000000000000000000000000000000000000000000000"
}
EOF
cat /tmp/subnet-evm.genesis.json
Set some environment variables for convenience:
# replace execPath with the path to MetalGo on your machine
METALGO_EXEC_PATH="${HOME}/go/src/github.com/!metal!blockchain/metalgo/build/metalgo"
METALGO_PLUGIN_PATH="${HOME}/go/src/github.com/!metal!blockchain/metalgo/build/plugins"
Now start the nodes with custom VM support.
For this, we need to point set the custom-vms
parameter (a map) to contain the genesis and the VM name:
curl -X POST -k http://localhost:8081/v1/control/start -d '{"execPath":"'${METALGO_EXEC_PATH}'","numNodes":5,"logLevel":"INFO","pluginDir":"'${METALGO_PLUGIN_PATH}'","blockchainSpecs":{"vm_name":"subnetevm","genesis":"/tmp/subnet-evm.genesis.json"}}'
or
metal-network-runner control start \
--log-level debug \
--endpoint="0.0.0.0:8080" \
--metalgo-path ${METALGO_EXEC_PATH} \
--blockchain-specs '[{"vm_name":"subnetevm","genesis":"/tmp/subnet-evm.genesis.json"}]' \
--plugin-dir ${METALGO_PLUGIN_PATH} \
Check it all up:
# to get cluster information including blockchain ID
curl -X POST -k http://localhost:8081/v1/control/status -d ''
DONE! You are now running your very own Ethereum blockchain on Metal!
RPC Server blobvm
Example
While above we configured and deployed an Ethereum compatible Subnet, Metal supports deploying your completely custom blockchain, still taking advantage of existing Metal infrastructure and its consensus protocol.
A custom blockchain requires your custom VM. In this tutorial we are going to deploy a custom VM
with a Subnet. The process is very similar to the subnet-evm
one, except that the VM is different.
We will also need the subnet-cli
tool for this tutorial, check this for some
context.
The VM we are going to use here is the BlobVM. This is a simple VM which enables content-addressable storage of arbitrary keys/values using any EIP-712 compatible wallet.
Install and start the RPC server just as in start the server Make sure the server is up:
curl -X POST -k http://localhost:8081/v1/ping -d ''
First, download/install subnet-cli
:
# or download from https://github.com/MetalBlockchain/subnet-cli/releases
cd ${HOME}/go/src/github.com/!metal!blockchain/subnet-cli
go install -v .
Create a VM ID:
subnet-cli create VMID blobvm
# kM6h4LYe3AcEU1MB2UNg6ubzAiDAALZzpVrbX8zn3hXF6Avd8
Build or...
rm -rf ${HOME}/go/src/github.com/!metal!blockchain/metalgo/build
cd ${HOME}/go/src/github.com/!metal!blockchain/metalgo
./scripts/build.sh
...download metalgo
(if not done already)
Clone and build the blobvm
plugin (requires golang
installation):
git clone https://github.com/MetalBlockchain/blobvm ${HOME}/go/src/github.com/!metal!blockchain/blobvm
cd ${HOME}/go/src/github.com/!metal!blockchain/blobvm
go build -v \
-o ${HOME}/go/src/github.com/!metal!blockchain/metalgo/build/plugins/kM6h4LYe3AcEU1MB2UNg6ubzAiDAALZzpVrbX8zn3hXF6Avd8 \
./cmd/blobvm
Verify everything has been built correctly:
find ${HOME}/go/src/github.com/!metal!blockchain/metalgo/build
# should yield something like:
# .../build
# .../build/plugins
# .../build/plugins/kM6h4LYe3AcEU1MB2UNg6ubzAiDAALZzpVrbX8zn3hXF6Avd8
# .../build/plugins/evm
# .../build/metalgo
Every VM needs a genesis file in order to be deployed on Metal. For the BlobVM, the file is very simple, and that repository contains a helper tool to create it:
# generate the genesis for the custom VM
cd ${HOME}/go/src/github.com/!metal!blockchain/blobvm
go install -v ./cmd/blob-cli
echo "[]" > /tmp/alloc.json
blob-cli genesis 1 /tmp/alloc.json --genesis-file /tmp/blobvm.genesis.json
# print contents
cat /tmp/blobvm.genesis.json
Set some environment variables for convenience:
# replace execPath with the path to MetalGo on your machine
METALGO_EXEC_PATH="${HOME}/go/src/github.com/!metal!blockchain/metalgo/build/metalgo"
METALGO_PLUGIN_PATH="${HOME}/go/src/github.com/!metal!blockchain/metalgo/build/plugins"
Now start the nodes with custom VM support.
For this, we need to point set the custom-vms
parameter (a map) to contain the genesis and the VM name:
curl -X POST -k http://localhost:8081/v1/control/start -d '{"execPath":"'${METALGO_EXEC_PATH}'","numNodes":5,"logLevel":"INFO","pluginDir":"'${METALGO_PLUGIN_PATH}'","customVms":{"blobvm":"/tmp/blobvm.genesis.json"}}'
or
metal-network-runner control start \
--log-level debug \
--endpoint="0.0.0.0:8080" \
--metalgo-path ${METALGO_EXEC_PATH} \
--plugin-dir ${METALGO_PLUGIN_PATH} \
--blockchain-specs '[{"vm_name":"blobvm","genesis":"/tmp/blobvm.genesis.json"}]'
Check it all up:
# to get cluster information including blockchain ID
curl -X POST -k http://localhost:8081/v1/control/status -d ''
DONE! You are now running your very own custom blockchain!
Using Metal Network as a Library
The Metal Network Runner can also be imported as a library into your programs so that you can use it to programmatically start, interact with and stop Metal networks. For an example of using the Network Runner in a program, see an example.
Creating a network is as simple as:
network, err := local.NewDefaultNetwork(log, binaryPath)
where log
is a logger of type
logging.Logger
and binaryPath
is the path of the MetalGo binary that each node that exists on network startup
will run.
For example, the below snippet creates a new network using default configurations, and each node in
the network runs the binaries at /home/user/go/src/github.com/!metal!blockchain/metalgo/build
:
network, err := local.NewDefaultNetwork(log,"/home/user/go/src/github.com/!metal!blockchain/metalgo/build")
Once you create a network, you must eventually call Stop()
on it to make sure all of the nodes
in the network stop. Calling this method kills all of the Metal nodes in the network. You
probably want to call this method in a defer
statement to make sure it runs.
To wait until the network is ready to use, use the network's Healthy
method. It returns a channel
which will be notified when all nodes are healthy.
Each node has a unique name. Use the network's GetNodeNames()
method to get the names of all
nodes.
Use the network's method GetNode(string)
to get a node by its name. For example:
names, _ := network.GetNodeNames()
node, _ := network.GetNode(names[0])
Then you can make API calls to the node:
id, _ := node.GetAPIClient().InfoAPI().GetNodeID() // Gets the node's node ID
balance, _ := node.GetAPIClient().XChainAPI().GetBalance(address,assetID,false) // Pretend these arguments are defined
After a network has been created and is healthy, you can add or remove nodes to/from the network:
newNode, _ := network.AddNode(nodeConfig)
err := network.RemoveNode(names[0])
Where nodeConfig
is a struct which contains information about the new node to be created. For a
local node, the most important elements are its name, its binary path and its identity, given by a
TLS key/cert.
You can create a network where nodes are running different binaries -- just provide different binary paths to each:
stakingCert, stakingKey, err := staking.NewCertAndKeyBytes()
if err != nil {
return err
}
nodeConfig := node.Config{
Name: "New Node",
ImplSpecificConfig: local.NodeConfig{
BinaryPath: "/tmp/my-metalgo/build",
},
StakingKey: stakingKey,
StakingCert: stakingCert,
}
After adding a node, you may want to call the network's Healthy
method again and wait until the
new node is healthy before making API calls to it.
Creating Custom Networks
To create custom networks, pass a custom config (the second parameter) to the
local.NewNetwork(logging.Logger, network.Config)
function. The config defines the number of nodes
when the network starts, the genesis state of the network, and the configs for each node.
Please refer to NetworkConfig for more details.