The Nolus Pirin (Mainnet) launched on May 23rd, 2023, at 15:00 UTC. Please, check out the instructions below on how to run a service on mainnet
Full Node
Full Node
System Configuration
Usually, you should be able to run a node on an operating system of your choice. You will be able to compile the Nolus daemon on most modern Linux distributions and recent versions of macOS. However, this tutorial uses commands available in the Ubuntu LTS release. If you are using a different operating system, you might need to adjust the commands you will see. Running a full node on the Nolus blockchain is a resource-intensive process that requires a persistent server since you would need to store all the information on the blockchain and constantly be in sync with the other network participants. It is also a prerequisite for being a validator
Hardware Requirements
The recommended hardware to run a Nolus node will vary depending on the node's use case and desired functionalities. For example, if the node acts as an archive, meaning that it stores all the historical data of the blockchain dating back to the genesis block, it would naturally require a lot of storage (disk space). The recommended by us minimum requirements to run a full node are the following:
4+ vCPU
16+ GB RAM
240+ GB SSD
Prerequisites: Golang go1.21.5 linux/amd64 Linux users
sudo apt-get install -y build-essential
Nolus core is the official Golang reference implementation used for a node. Follow this guide to install Nolus core which will allow you to use nolusd. nolusd stands for Nolus Daemon and is the command-line interface (CLI) and daemon that enables you to interact with the Nolus blockchain
Build Nolus Core
1. Use Git to Clone the Nolus Core Repository
git clone https://github.com/Nolus-Protocol/nolus-core
2. Make Sure That You Are in the “Core” Directory
cd nolus-core
3. Check Out the Main Branch Containing the Latest Stable Release
⚠️ If you want to sync from block 1, then feel free to checkout the initial version of the pirin-1 mainnet which is v0.3.0 and upgrade the binary when you reach an upgrade height. Should you decide to checkout the latest version (v0.6.2), you would be able to sync from a recent snapshot.
git checkout [latest version]
ex., git checkout v0.3.0
4. Compile the Nolus Core and Install It
make install
5. Verify That Nolus Core Is Installed Correctly by Checking the Version
nolusd version --long
You should see a similar output as the one shown in the example below:
name: nolus server_name: nolusd version: [version_of_nolusd] commit: [commit_hash] build_tags: netgo ledger go: go version go1.21.5 linux/amd64
⚠️ If the nolusd: command not found error message is returned, run the following command to confirm whether the Go binary path has been successfully defined: export PATH=$PATH:$(go env GOPATH)/bin
Configuration and Execution
1. Initialize a Nolus Node
nolusd init [custom_moniker]
ex., nolusd init “My First Nolus Node”
This command initializes your node in a .nolus directory using the $HOME path by default. It adds a number of configuration files such as config.toml
and app.toml
which are to be modified in the next steps.
⚠️ Make sure to use ASCII characters for your node’s name (moniker) and not Unicode. The name is important if you are running a validator since in this case, it would be visible to the other participants. You can update your node’s moniker by editing the moniker field in ~/.nolus/config/config.toml
2. Obtain the Genesis File and Set Persistent Peers
Retrieve the genesis state file from the Nolus Repository or another trusted source. This is a file that contains details with regard to the initial state of the network. There are details about governance, staking, account balances, slashing, etc.
wget https://raw.githubusercontent.com/nolus-protocol/nolus-networks/main/mainnet/pirin-1/genesis.json
Override the default genesis file in your Nolus node directory with the correct one you retrieved via the above command.
mv ./genesis.json ~/.nolus/config/genesis.json
ℹ️ You can download the addrbook.json file in order to synchronize with the rest of the node participants: https://github.com/nolus-protocol/nolus-networks/blob/main/mainnet/pirin-1/addrbook.json
In case your node is not syncing when you start it, feel free to use the live peers provided by kjnodes: https://services.kjnodes.com/mainnet/nolus/
You would need to connect to at least one of them to sync with the current state of the network. They are specified in the form <NODE-ID>@<IP-ADDRESS>:<PORT>
. Export them in a variable:
PEERS="[email protected]:26656,[email protected]:26656,[email protected]:26656,[email protected]:26656,[email protected]:60756,[email protected]:26656,[email protected]:26656,[email protected]:26656,[email protected]:2640,[email protected]:26656"
Set them in the ~/.nolus/config/config.toml
file:
sed -i -e "s|^persistent_peers *=.*|persistent_peers = \"$PEERS\"|" $HOME/.nolus/config/config.toml
3. Edit Configuration Files
3.1 Adjust the minimum gas price:
Open ~/.nolus/config/app.toml
.
Modify the minimum-gas-prices field to set a minimum gas price threshold that your validator would be willing to accept in order to prevent spam attacks:
minimum-gas-prices = "0.0025unls”
ℹ️ Bear in mind that a portion of the fees paid by the user gets diverted to the Nolus Protocol. With that in mind, you would need to take into account the fee rate that is applied as tax. On genesis, it would be 40%.
3.2 Configure the application state pruning (Optional):
Open ~/.nolus/config/app.toml
. By default, the application will keep the application data for a time horizon that matches the unbonding period (21 days in blocks) and prune the remaining application state.
If you want to prune the entire application state, set:
pruning = "everything”
If you want to keep the entire application state (relevant for archive nodes), set:
pruning = "nothing”
4. Set Up Cosmovisor (Optional)
⚠️ Using Cosmovisor is optional. It helps you automate downloading binaries for chain upgrades by monitoring the governance module. This significantly reduces the downtime of your node in such cases. Otherwise, you would need to manually download the new binary, stop the current binary, switch from the old binary to the new one, and finally restart the node with the new binary.
4.1 Install Cosmovisor
You can get the latest version of Cosmovisor from the official GitHub repository of the Cosmos SDK
go install cosmossdk.io/tools/cosmovisor/cmd/cosmovisor@latest
4.2 Include environment variables
In the .profile
file, usually located at ~/.profile
, add:
export DAEMON_NAME=nolusd export DAEMON_HOME=$HOME/.nolus export MONIKER_NAME=<the-name-of-your-node>
Source your profile to have access to this variable via:
source ~/.profile
4.3 Adjust folder layout
$DAEMON_HOME/cosmovisor is expected to belong completely to Cosmovisor and the subprocesses that are controlled by it. The folder content is organized as follows:
Do not worry about the current directory. It is a symbolic link to the currently active directory (i.e. genesis or upgrades/<name>). The other folders can be set up using:
mkdir -p $DAEMON_HOME/cosmovisor/genesis/bin mkdir -p $DAEMON_HOME/cosmovisor/upgrades
4.4 Set up genesis binary
Cosmovisor would need to know which binary to choose at genesis. You need to copy your Nolus node binary and paste it into the directory Cosmovisor expects:
cp /home/<your-user>/go/bin/nolusd $DAEMON_HOME/cosmovisor/genesis/bin
4.5 Set up a service
If an error or reboot happens, we want to make sure that Cosmovisor would automatically restart and not have to manually do it ourselves, since commands sent to the Cosmovisor are automatically directed at the underlying binary. For this reason, we set up a service by creating first a .service file:
sudo nano /etc/systemd/system/cosmovisor.service
Change the contents below to match your user. Double check if the paths below match yours:
[Unit] Description=cosmovisor After=network-online.target [Service] User=<your-user> ExecStart=/home/<your-user>/go/bin/cosmovisor run start Restart=always RestartSec=3 LimitNOFILE=4096 Environment="DAEMON_NAME=nolusd" Environment="DAEMON_HOME=/home/<your-user>/.nolus" Environment="DAEMON_ALLOW_DOWNLOAD_BINARIES=false" Environment="DAEMON_RESTART_AFTER_UPGRADE=true" Environment="DAEMON_LOG_BUFFER_SIZE=512" [Install] WantedBy=multi-user.target
5. Run Your Node
5.1 If you have NOT set up Cosmovisor you can simply run the binary using the command below:
nolusd start
5.2 If you have set up Cosmovisor, you need first to enable the Cosmovisor service and then run it:
sudo -S systemctl daemon-reload sudo -S systemctl enable cosmovisor sudo systemctl start cosmovisor
Check whether it is running by entering:
sudo systemctl status cosmovisor
If you need to monitor the service after launch, you can view the logs using:
journalctl -u cosmovisor -f
After starting the nolusd daemon, the chain will begin to sync to the network. The time to sync to the network will vary depending on your setup and the current size of the blockchain, but it could take a very long time. Use the following command to query the status of your node:
curl http://localhost:26657/status | jq .result.sync_info.catching_up
If it returns true, then your node is still syncing. If it returns false, your node has caught up to the current state of the network, and you are safe to upgrade your node to a validator.
6. Run Your Node From A Snapshot (Optional)
Alternatively, you can bootstrap your Nolus node by utilizing a snapshot service provided by a third-party node provider. In general, make sure to double-check the source and contents of the data that you are downloading.
ℹ️ Snapshots allow a new node to join the network relatively quickly by recovering the application state from a backup file instead of having to sync from the start. A snapshot contains a compressed copy of the Nolus chain data and wasm directories.
You can start your node from a pruned snapshot provided by kjnodes. Bear in mind that this snapshot is suitable for running a validator node but not for an RPC one: https://services.kjnodes.com/mainnet/nolus/snapshot/
Validator
Validator
Create a Validator
1. Create (Or Restore) A Local Key Pair
Prior to creating your validator, you must first create your application key. Note that this is not your consensus key and will not be used for signing consensus votes. Instead, it is used to sign transactions.
Create a new key pair
nolusd keys add <key-name>
Restore an existing Nolus wallet by providing a BIP39 mnemonic
nolusd keys add <key-name> --recover
Get your public address from the keystore
nolusd keys show <key-name> -a
Feel free to replace <key-name> with a name of your choice.
⚠️ After creating a new key, its information will be shown alongside the seed phrase. Make sure to write it down, as it is the only way to restore your keys.
2. Upgrade to a Validator
⚠️ Do not attempt to upgrade your node to a validator until the node is fully in sync after you have started the nolusd binary, as shown in the previous section.
Since you need to send a transaction to be part of the validator set, you need to have a small amount of NLS tokens in the wallet address you are using on your keyring. Once you have a positive balance, you can send the create-validator transaction.
nolusd tx staking create-validator \\ --amount 1000000unls \\ --commission-rate "0.05" \\ --commission-max-rate "0.10" \\ --commission-max-change-rate "0.01" \\ --min-self-delegation "1" \\ --pubkey=$(nolusd tendermint show-validator) \\ --moniker <the-name-of-your-node> \\ --chain-id "pirin-1" \\ --fees 500unls \\ --from <key-name>
Аn example empty command:
nolusd tx staking create-validator \\ --amount=[staking_amount_unls] \\ --commission-rate="[commission_rate]" \\ --commission-max-rate="[maximum_commission_rate]" \\ --commission-max-change-rate="[maximum_rate_of_change_of_commission]" \\ --min-self-delegation="[min_self_delegation_amount]" \\ --pubkey=$(nolusd tendermint show-validator) \\ --moniker="[moniker_id_of_your_node]" \\ --chain-id="[chain-id]" \\ --gas-prices="[gas_price]" \\ --from=[KEY_NAME] \\
If you need further explanation for each of these command flags:
the amount flag is the amount you will place in your own validator in unls (in the example, 1000000unls is 1 NLS)
the commission rate is the rate you will charge your delegates (in the example above, 5 percent)
the commission-max-rate is the most you are allowed to charge your delegates (in the example above, 10 percent)
the commission-max-change-rate is how much you can increase your commission rate in a 24-hour period (in the example above, 1 percent per day until reaching the max rate)
min-self-delegation is a strictly positive integer that represents the minimum amount of self-delegated voting power your validator must always have. In the example above, this is equal to 1 NLS or 1000000unls
the pubkey is the validator public key found earlier
the chain-id is whatever chain-id you are working with (in the Nolus mainnet case, it is pirin-1)
the gas price is the amount of gas used to send this create-validator transaction
the from flag is the KEY_NAME you created when initializing the key on your keyring
3. Confirm That Your Validator Is Active
If running the following command returns something, your validator is active:
nolusd query tendermint-validator-set | grep "$(nolusd tendermint show-address)”
You are looking for the bech32 encoded address in the ~/.nolusd/config/priv_validator.json
file with a nolusvalcons prefix.
4. Secure Your Keys
In general, a Nolus validator needs to do two things:
Sign and commit blocks (using the Tendermint Consensus key)
Conduct on-chain operations such as voting on Governance proposals (using an Application Operator Key)
⚠️ Take good care of those two keys to mitigate hardware or software failures.
Restore a Validator
A validator can be completely restored on a new Nolus node with the following set of keys:
The Consensus key, stored in
~/.nolus/config/priv_validator.json
The mnemonic to the validator wallet (application key)
⚠️ Danger! Before proceeding, ensure that the existing validator is not active. Double voting has severe slashing consequences.
To restore a validator:
Set up a full Nolus node synced up to the latest block. Check the previous section.
Replace the
~/.nolus/config/priv_validator.json
file of the new node with the associated file from the old node, then restart your node.
Unjail a Validator
If your validator fails to sign at least 5% of the last 10000 blocks, it will get jailed. To unjail it so that it can continue to sign blocks and respectively earn staking rewards, you can submit the following transaction:
nolusd tx slashing unjail --from <yourKey> --chain-id pirin-1 --fees 500unls
Relayer
Relayer
An IBC (Inter-Blockchain Communication) relayer is an off-chain process responsible for relaying messages between two chains. It is a mandatory actor in the IBC network architecture. This is because blockchains are not passing messages directly to one another over the network. Instead, they create and store the data to be retrieved and used by a relayer to build the IBC messages and subsequently pass them to the destination chain via a dedicated channel.=
ℹ️ This page contains instructions on running a Hermes relayer, a Rust-based implementation. The other two popular but less preferred implementations are TS-based and go-based
Hardware Requirements
8-core (4 physical core), x86_64 architecture processor
32 GB RAM (or equivalent swap file set up)
1TB+ nVME drives
The relayer service itself is quite lightweight, the nodes are the ones requiring the above-specified hardware requirements
Prerequisites: Before beginning, ensure you have a Nolus full node running in the background. You could relay on the same machine or connect to the Osmosis and Nolus full nodes via a network. You will need build-essential and git installed to follow these instructions.
Download the latest Osmosis binary. Check out https://docs.osmosis.zone/networks/join-mainnet/ for a detailed setup guideline.
Hermes
1. Install Rust Dependencies
Since we are relying on a Rust-based implementation, we would need to install some Rust dependencies first:
curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh
source "$HOME/.cargo/env"
sudo apt-get install pkg-config libssl-dev
sudo apt install librust-openssl-dev build-essential git
2. Configure Hermes
Create the directory where you'll place the binary, clone the hermes source repository and build it using the latest release:
cargo install ibc-relayer-cli --bin hermes --locked mkdir -p $HOME/hermes git clone https://github.com/informalsystems/ibc-rs.git hermes cd hermes git checkout v1.7.1
Make a hermes “keys” directory, copy config.toml from the cloned repo to the .hermes directory:
mkdir -p $HOME/.hermes mkdir -p $HOME/.hermes/keys cp config.toml $HOME/.hermes
Check if hermes is installed properly by running:
hermes version
Edit the hermes config.toml file by including configurations for the two chains that you want to relay between, namely Nolus Pirin mainnet and Osmosis mainnet:
nano $HOME/.hermes/config.toml
You would need to specify the hermes configurations for your two running nodes. Do not forget to remove the generated by default “chains” section with the example at the end of the file. You can adjust the ports and instances as they are defined on the instance(s) which you are using. Here it is assumed that you’re running both a Nolus and an Osmosis mainnet full nodes on the same instance or in the same container:
[mode.clients]
enabled = true
refresh = true
misbehaviour = false
[mode.connections]
enabled = true
[mode.channels]
enabled = true
[mode.packets]
enabled = true
[[chains]]
id = 'pirin-1'
rpc_addr = 'http://127.0.0.1:26657'
event_source = { mode = 'push', url = 'ws://127.0.0.1:26657/websocket', batch_delay = '500ms' }
grpc_addr = 'http://127.0.0.1:9090'
rpc_timeout = '10s'
account_prefix = 'nolus'
key_name = 'hermes-nolus'
store_prefix = 'ibc'
default_gas = 5000000
max_gas = 15000000
gas_multiplier = 1.1
max_msg_num = 30
max_tx_size = 2097152
clock_drift = '5s'
max_block_time = '30s'
trusting_period = '14days'
memo_prefix = ''
[chains.trust_threshold]
numerator = '1'
denominator = '3'
[chains.gas_price]
price = 0.0025
denom = 'unls'
[chains.packet_filter]
policy = 'allow'
list = [
['transfer', 'channel-0'],
['transfer', 'channel-3839'],
['ica*', '*'], # allow relaying on all channels whose port starts with `ica
]
[chains.address_type]
derivation = 'cosmos'
[[chains]]
id = 'osmosis-1'
rpc_addr = 'http://127.0.0.1:26557'
event_source = { mode = 'push', url = 'ws://127.0.0.1:26557/websocket', batch_delay = '500ms' }
grpc_addr = 'http://127.0.0.1:9091'
rpc_timeout = '10s'
account_prefix = 'osmo'
key_name = 'hermes-osmosis'
store_prefix = 'ibc'
default_gas = 5000000
max_gas = 15000000
gas_multiplier = 1.1
max_msg_num = 20
max_tx_size = 209715
clock_drift = '20s'
max_block_time = '10s'
trusting_period = '10days'
memo_prefix = ''
[chains.trust_threshold]
numerator = '1'
denominator = '3'
[chains.gas_price]
price = 0.025
denom = 'uosmo'
[chains.packet_filter]
policy = 'allow'
list = [
['transfer', 'channel-783'],
['ica*', '*'], # allow relaying on all channels whose port starts with `ica
]
[chains.address_type]
derivation = 'cosmos'
[[chains]]
id = 'neutron-1'
rpc_addr = '<rpc-address-of-your-neutron-mainnet-node>'
grpc_addr = '<grpc-address-of-your-neutron-mainnet-node>'
event_source = { mode = 'push', url = 'ws://<rpc-address-of-your-neutron-mainnet-node>/websocket', batch_delay = '500ms' }
rpc_timeout = '10s'
account_prefix = 'neutron'
key_name = 'hermes-neutron'
store_prefix = 'ibc'
default_gas = 5000000
max_gas = 15000000
gas_multiplier = 1.2
max_msg_num = 20
max_tx_size = 209715
clock_drift = '20s'
max_block_time = '10s'
trusting_period = '1209600s'
memo_prefix = ""
ccv_consumer_chain = true
[chains.trust_threshold]
numerator = "1"
denominator = "3"
[chains.gas_price]
price = 0.075
denom = 'untrn'
[chains.packet_filter]
policy = 'allow'
list = [
['transfer', 'channel-44'],
['ica*', '*'], # allow relaying on all channels whose port starts with `ica
]
[chains.address_type]
derivation = "cosmos"
Add your relayer wallet to hermes' keyring (located in $HOME/.hermes/keys
). The wallet should have a positive balance on all the networks since the relayer would need to pay gas fees to submit IBC transactions
ℹ️ The best practice is to use the same mnemonic over all networks. Do not use your relaying addresses for anything else because it will lead to account sequence errors
hermes keys add --chain pirin-1 --mnemonic-file <hermes-seed-file> hermes keys add --chain osmosis-1 --mnemonic-file <hermes-seed-file> hermes keys add --chain neutron-1 --mnemonic-file <hermes-seed-file>
3. Validate Configuration
Validate your ~/.hermes/config.toml
file by running:
hermes config validate
Perform a health check:
hermes health-check
You should see a similar output as the one below:
... INFO ThreadId(01) [pirin-1] chain is healthy INFO ThreadId(01) [osmosis-1] chain is healthy INFO ThreadId(01) [neutron-1] chain is healthy
4. Run Hermes
If your nodes are fully synced, feel free to start the hermes daemon:
hermes start
IBC Transfer Channels for Nolus
Source | Source Chain ID | Source Channel | Destination | Destination Chain ID | Destination Channel |
Nolus Pirin Mainnet | pirin-1 | channel-0 | Osmosis Mainnet | osmosis-1 | channel-783 |
Nolus Pirin Mainnet | pirin-1 | channel-3839 | Neutron Mainnet | neutron-1 | channel-44 |
Nolus Pirin Mainnet | pirin-1 | channel-10177 | Axelar Mainnet | axelar-dojo-1 | channel-143 |
In addition, by design, Nolus opens new ICA channels for each lease (borrow) position opened.