Local Setup
The below guide will instruct you on how to set up a Mind-chain Network on your local machine for testing and development purposes.
The procedure differs greatly from the way you would want to set up Mind-chain Network for a real use scenario on a cloud provider: Cloud Setup
Requirements
Refer to Installation to install Mind-chain.
Overview
In this guide, our goal is to establish a working Mind-chain
blockchain network working with IBFT consensus protocol.
The blockchain network will consist of 4 nodes of whom all 4 are validator nodes, and as such are eligible for both proposing block, and validating blocks that came from other proposers.
All 4 nodes will run on the same machine, as the idea of this guide is to give you a fully functional IBFT cluster in the least amount of time.
To achieve that, we will guide you through 4 easy steps:
- Initializing data directories will generate both the validator keys for each of the 4 nodes, and initialize empty blockchain data directories. The validator keys are important as we need to bootstrap the genesis block with the initial set of validators using these keys.
- Preparing the connection string for the bootnode will be the vital information for every node we will run as to which node to connect to when starting for the first time.
- Generating the
genesis.json
file will require as input both the validator keys generated in step 1 used for setting the initial validators of the network in the genesis block and the bootnode connection string from step 2. - Running all the nodes is the end goal of this guide and will be the last step we do, we will instruct the nodes which data directory to use and where to find the
genesis.json
which bootstraps the initial network state.
As all four nodes will be running on localhost, during the setup process it is expected that all the data directories for each of the nodes are in the same parent directory.
There is no minimum to the number of nodes in a cluster, which means clusters with only 1 validator node are possible. Keep in mind that with a single node cluster, there is no crash tolerance and no BFT guarantee.
The minimum recommended number of nodes for achieving a BFT guarantee is 4 - since in a 4 node cluster, the failure of 1 node can be tolerated, with the remaining 3 functioning normally.
Step 1: Initialize data folders for IBFT and generate validator keys
In order to get up and running with IBFT, you need to initialize the data folders, one for each node:
mind secrets init --data-dir test-chain-1
mind secrets init --data-dir test-chain-2
mind secrets init --data-dir test-chain-3
mind secrets init --data-dir test-chain-4
Each of these commands will print the validator key and the node ID. You will need the Node ID of the first node for the next step.
Step 2: Prepare the multiaddr connection string for the bootnode
For a node to successfully establish connectivity, it must know which bootnode
server to connect to in order to gain
information about all the remaining nodes on the network. The bootnode
is sometimes also known as the rendezvous
server in p2p jargon.
bootnode
is not a special instance of the Mind-chainnode. Every Mind-chainnode can serve as a bootnode
, but
every Mind-chainnode needs to have a set of bootnodes specified which will be contacted to provide information on how to connect with
all remaining nodes in the network.
To create the connection string for specifying the bootnode, we will need to conform to the multiaddr format:
/ip4/<ip_address>/tcp/<port>/p2p/<node_id>
In this guide, we will treat the first and second nodes as the bootnodes for all other nodes. What will happen in this scenario
is that nodes that connect to the node 1
or node 2
will get information on how to connect to one another through the mutually
contacted bootnode.
At least one bootnode is required, so other nodes in the network can discover each other. More bootnodes are recommended, as
they provide resilience to the network in case of outages.
In this guide we will list two nodes, but this can be changed on the fly, with no impact on the validity of the genesis.json
file.
Since we are running on localhost, it is safe to assume that the <ip_address>
is 127.0.0.1
.
For the <port>
we will use 10001
since we will configure the libp2p server for node 1
to listen on this port later.
And lastly, we need the <node_id>
which we can get from the output of the previously ran command mind secrets init --data-dir test-chain-1
command (which was used to generate keys and data directories for the node1
)
After the assembly, the multiaddr connection string to the node 1
which we will use as the bootnode will look something like this (only the <node_id>
which is at the end should be different):
/ip4/127.0.0.1/tcp/10001/p2p/16Uiu2HAmJxxH1tScDX2rLGSU9exnuvZKNM9SoK3v315azp68DLPW
Similarly, we construct the multiaddr for second bootnode as shown below
/ip4/127.0.0.1/tcp/20001/p2p/16Uiu2HAmS9Nq4QAaEiogE4ieJFUYsoH28magT7wSvJPpfUGBj3Hq
Step 3: Generate the genesis file with the 4 nodes as validators
mind genesis --consensus ibft --ibft-validators-prefix-path test-chain- --bootnode /ip4/127.0.0.1/tcp/10001/p2p/16Uiu2HAmJxxH1tScDX2rLGSU9exnuvZKNM9SoK3v315azp68DLPW --bootnode /ip4/127.0.0.1/tcp/20001/p2p/16Uiu2HAmS9Nq4QAaEiogE4ieJFUYsoH28magT7wSvJPpfUGBj3Hq
What this command does:
- The
--ibft-validators-prefix-path
sets the prefix folder path to the one specified which IBFT in Mind-chaincan use. This directory is used to house theconsensus/
folder, where the validator's private key is kept. The validator's public key is needed in order to build the genesis file - the initial list of bootstrap nodes. This flag only makes sense when setting up the network on localhost, as in a real-world scenario we cannot expect all the nodes' data directories to be on the same filesystem from where we can easily read their public keys. - The
--bootnode
sets the address of the bootnode that will enable the nodes to find each other. We will use the multiaddr string of thenode 1
, as mentioned in step 2.
The result of this command is the genesis.json
file which contains the genesis block of our new blockchain, with the predefined validator set and the configuration for which node to contact first in order to establish connectivity.
You will probably want to set up your blockchain network with some addresses having "premined" balances.
To achieve this, pass as many --premine
flags as you want per address that you want to be initialized with a certain balance
on the blockchain.
For example, if we would like to premine 1000 ETH to address 0x3956E90e632AEbBF34DEB49b71c28A83Bc029862
in our genesis block, then we would need to supply the following argument:
--premine=0x3956E90e632AEbBF34DEB49b71c28A83Bc029862:1000000000000000000000
Note that the premined amount is in WEI, not ETH.
The default gas limit for each block is 5242880
. This value is written in the genesis file, but you may want to
increase / decrease it.
const (
GenesisFileName = "./genesis.json"
DefaultChainName = "example"
DefaultChainID = 2000
DefaultPremineBalance = "0x3635C9ADC5DEA00000"
DefaultConsensus = "pow"
GenesisGasUsed = 458752
GenesisGasLimit = 5242880 // The default block gas limit
)
To do so, you can use the flag --block-gas-limit
followed by the desired value as shown below :
--block-gas-limit 1000000000
The default file descriptor limit ( maximum number of open files ) on some operating systems is pretty small. If the nodes are expected to have high throughput, you might consider increasing this limit on the OS level.
For Ubuntu distro the procedure is as follows ( if you're not using Ubuntu/Debian distro, check the official docs for your OS ) :
- Check current os limits ( open files )
ubuntu@ubuntu:~$ ulimit -a
core file size (blocks, -c) 0
data seg size (kbytes, -d) unlimited
scheduling priority (-e) 0
file size (blocks, -f) unlimited
pending signals (-i) 15391
max locked memory (kbytes, -l) 65536
max memory size (kbytes, -m) unlimited
open files (-n) 1024
pipe size (512 bytes, -p) 8
POSIX message queues (bytes, -q) 819200
real-time priority (-r) 0
stack size (kbytes, -s) 8192
cpu time (seconds, -t) unlimited
max user processes (-u) 15391
virtual memory (kbytes, -v) unlimited
file locks (-x) unlimited
Increase open files limit
- Localy - affects only current session:
ulimit -u 65535
- Globaly or per user ( add limits at the end of /etc/security/limits.conf file ) :
sudo vi /etc/security/limits.conf # we use vi, but you can use your favorite text editor
/etc/security/limits.conf# /etc/security/limits.conf
#
#Each line describes a limit for a user in the form:
#
#<domain> <type> <item> <value>
#
#Where:
#<domain> can be:
# - a user name
# - a group name, with @group syntax
# - the wildcard *, for default entry
# - the wildcard %, can be also used with %group syntax,
# for maxlogin limit
# - NOTE: group and wildcard limits are not applied to root.
# To apply a limit to the root user, <domain> must be
# the literal username root.
#
#<type> can have the two values:
# - "soft" for enforcing the soft limits
# - "hard" for enforcing hard limits
#
#<item> can be one of the following:
# - core - limits the core file size (KB)
# - data - max data size (KB)
# - fsize - maximum filesize (KB)
# - memlock - max locked-in-memory address space (KB)
# - nofile - max number of open file descriptors
# - rss - max resident set size (KB)
# - stack - max stack size (KB)
# - cpu - max CPU time (MIN)
# - nproc - max number of processes
# - as - address space limit (KB)
# - maxlogins - max number of logins for this user
# - maxsyslogins - max number of logins on the system
# - priority - the priority to run user process with
# - locks - max number of file locks the user can hold
# - sigpending - max number of pending signals
# - msgqueue - max memory used by POSIX message queues (bytes)
# - nice - max nice priority allowed to raise to values: [-20, 19]
# - rtprio - max realtime priority
# - chroot - change root to directory (Debian-specific)
#
#<domain> <type> <item> <value>
#
#* soft core 0
#root hard core 100000
#* hard rss 10000
#@student hard nproc 20
#@faculty soft nproc 20
#@faculty hard nproc 50
#ftp hard nproc 0
#ftp - chroot /ftp
#@student - maxlogins 4
* soft nofile 65535
* hard nofile 65535
# End of fileOptionaly, modify additional parameters, save the file and restart the system. After restart check file descriptor limit again. It should be set to the value you defined in limits.conf file.
Step 4: Run all the clients
Because we are attempting to run a Mind-chain Network consisting of 4 nodes all on the same machine, we need to take care to avoid port conflicts. This is why we will use the following reasoning for determining the listening ports of each server of a node:
10000
for the gRPC server ofnode 1
,20000
for the GRPC server ofnode 2
, etc.10001
for the libp2p server ofnode 1
,20001
for the libp2p server ofnode 2
, etc.10002
for the JSON-RPC server ofnode 1
,20002
for the JSON-RPC server ofnode 2
, etc.
To run the first client (note the port 10001
since it was used as a part of the libp2p multiaddr in step 2 alongside node 1's Node ID):
mind server --data-dir ./test-chain-1 --chain genesis.json --grpc :10000 --libp2p :10001 --jsonrpc :10002 --seal
To run the second client:
mind server --data-dir ./test-chain-2 --chain genesis.json --grpc :20000 --libp2p :20001 --jsonrpc :20002 --seal
To run the third client:
mind server --data-dir ./test-chain-3 --chain genesis.json --grpc :30000 --libp2p :30001 --jsonrpc :30002 --seal
To run the fourth client:
mind server --data-dir ./test-chain-4 --chain genesis.json --grpc :40000 --libp2p :40001 --jsonrpc :40002 --seal
To briefly go over what has been done so far:
- The directory for the client data has been specified to be ./test-chain-*
- The GRPC servers have been started on ports 10000, 20000, 30000 and 40000, for each node respectively
- The libp2p servers have been started on ports 10001, 20001, 30001 and 40001, for each node respectively
- The JSON-RPC servers have been started on ports 10002, 20002, 30002 and 40002, for each node respectively
- The seal flag means that the node which is being started is going to participate in block sealing
- The chain flag specifies which genesis file should be used for chain configuration
The structure of the genesis file is covered in the CLI Commands section.
After running the previous commands, you have set up a 4 node Mind-chain Network , capable of sealing blocks and recovering from node failure.
Instead of specifying all configuration parameters as CLI arguments, the Client can also be started using a config file by executing the following command:
mind server --config <config_file_path>
Example:
mind server --config ./test/config-node1.json
Currently, we only support json
based configuration file, sample config file can be found here
A Non-validator will always sync the latest blocks received from the validator node, you can start a non-validator node by running the following command.
mind server --data-dir <directory_path> --chain <genesis_filename> --grpc <portNo> --libp2p <portNo> --jsonrpc <portNo>
For example, you can add fifth Non-validator client by executing the following command :
mind server --data-dir ./test-chain --chain genesis.json --grpc :50000 --libp2p :50001 --jsonrpc :50002
A Mind-chainnode can be started with a set price limit for incoming transactions.
The unit for the price limit is wei
.
Setting a price limit means that any transaction processed by the current node will need to have a gas price higher then the set price limit, otherwise it will not be included in a block.
Having the majority of nodes respect a certain price limit enforces the rule that transactions in the network cannot be below a certain price threshold.
The default value for the price limit is 0
, meaning it is not enforced at all by default.
Example of using the --price-limit
flag:
mind server --price-limit 100000 ...
It is worth noting that price limits are enforced only on non-local transactions, meaning that the price limit does not apply to transactions added locally on the node.
Step 5: Interact with the Mind-chain Network
Now that you've set up at least 1 running client, you can go ahead and interact with the blockchain using the account you premined above and by specifying the JSON-RPC URL to any of the 4 nodes:
- Node 1:
http://localhost:10002
- Node 2:
http://localhost:20002
- Node 3:
http://localhost:30002
- Node 4:
http://localhost:40002
Follow this guide to issue operator commands to the newly built cluster: How to query operator information (the GRPC ports for the cluster we have built are 10000
/20000
/30000
/40000
for each node respectively)