Monero Daemon Configuration
Introduction and Scope
This page documents the configuration of the Monero XMR Daemon (monerod
) as a system service. Configuring
monerod
as a service ensures that it starts automatically whenever the host computer boots up. The Monero
daemon supports interactive commands and to take advantage of this interactive aspect of the daemon I have
configured a named pipe or Unix socket that allows for this. With this configuration you can send commands
to the Monero daemon by echoing commands into the named pipe.
Pre-Requisites
Linux Operating System
I am running Debian, but these instructions apply to basically any machine running Linux.
Disk Space Requirements
The disk space required to house the full Monero blockchain is significant. As of the writing of this article, September 26, 2024, the full monero blockchain is 206 Gb. It’s recommended that you use a SSD hard-drive, but I’m a traditional magnetic spinning disk. I migrated from an SSD to this spinning disk and didn’t notice any degradation of mining performance.
bzip2
The Monero GUI Wallet maintainers use bzip2
to compress their software. This isn’t part of the
default Debian install, but can be easily downloaded and installed using sudo apt install bzip2
.
Download and Install the Monero Daemon Software
The Monero GUI Wallet includes the Monero Daemon. It’s very important that you download this software from the official source, the GetMonero.org. You can navigate to their Downloads page and download the software from there. I image you could download the Monero CLI Wallet instead, but since I use the Monero GUI Wallet as my primary Wallet for Monero XMR, that’s the software I chose.
Once you’ve naviaged to the Downloads page, click on the
Linux 64-bit link to start the download. When the
download finishes you’ll end up with a compressed tarball. As of the writing of this article the
actual file was called monero-gui-linux-x64-v0.18.3.4.tar.bz2
. As newer versions of the
Monero GUI Wallet are released, the version number will change, but I’ll use that filename in this
documentation.
I run custom software out of the /opt
directory on my computers and I use /opt/prod
, /opt/qa
,
/opt/dev
to house production, test and development versions of software. As this is production
software, I will install it into /opt/prod
. The commands below show me uncompressing and then
untarring the software. I use gzip
as my preferred compression software and I keep copies of the
original software in /opt/src
, even if it’s not source code. The commands below assume that
the software was downloaded in the sally user’s Downloads directory.
sudo mkdir -p /opt/src/monero-gui-wallet
sudo mv ~sally/Downloads/monero-gui-linux-x64-v0.18.3.4.tar.bz2 /opt/src/monero-gui-wallet
sudo bzip -d /opt/src/monero-gui-wallet/monero-gui-linux-x64-v0.18.3.4.tar.bz2
sudo mkdir /opt/prod
cd /opt/prod
sudo tar xvf /opt/src/monero-gui-wallet/monero-gui-linux-x64-v0.18.3.4.tar
sudo gzip /opt/src/monero-gui-wallet/monero-gui-linux-x64-v0.18.3.4.tar
sudo ln -s monero-gui-v0.18.3.4 monerod
When you untar the software, it creates a directory called monero-gui-v0.18.3.4
. The last command,
above, creates a symbolic link, or shortcut, to the monero-gui-v0.18.3.4
called monerod
. This
shortcut is referenced by the system service file definition (described below). By creating and
using this shortcut you can easily upgrade your Monero GUI Wallet, delete the shortcut, and then
re-create it without needing to update your system service definition file.
The start-monerod.sh Script
The Monero Daemon has a lot of command switches. I’ve created a simple script that stores all
the switches and their values. You’ll want to create this script using your favorite editor.
As the root user create the script in the /opt/prod/monerod
directory. A complete listing is
shown below.
A few things to note about this script.
- I run two full monero nodes on my network which are in the same subnet and are behind the same router. The router supports upnp for routing inbound traffic. In order for the router to differentiate between the two nodes each node needs to use unique port numbers. The port numbers I use in the script below are non-standard, the standard port numbers start with 18 (e.g. ZMQ_PUB_PORT=”18083”). This isn’t really relevant, but is worth mentioning in case you encounter documentation that talks about slightly different port numbers. You can alter these ports again, but I don’t recommend it.
- You’ll note that I have p2pmd.xmrvsbeast and nodes.hashvault.pro configured with the
--add-priority-hosts
switch. I trust both of these machines and the people who run the Monero nodes on these machines. You can safely remove both of these lines if you prefer. - You may also notice that the
DATA_DIR
is set to/opt/prod/monero-blockchain
, an explanation for this is covered in the next section.
#!/bin/bash
### User defined variables
# Bind to all available network interfaces
IP_ALL="0.0.0.0"
# Port numbers
ZMQ_PUB_PORT="20083"
ZMQ_RPC_PORT="20082"
P2P_BIND_PORT="20080"
RPC_BIND_PORT="20081"
PRIORITY_NODE_PORT="18080"
# Peer limits
OUT_PEERS="20"
IN_PEERS="20"
# Log settings
LOG_LEVEL="0"
MAX_LOG_FILES="5"
MAX_LOG_SIZE="100000"
LOG_NAME="monerod.log"
# Not sure
SHOW_TIME_STATS="1"
# Where the blockchain (lmdb directory) is stored
DATA_DIR="/opt/prod/monero-blockchain"
# Trusted Monero nodes to sync off
PRIORITY_NODE_1="p2pmd.xmrvsbeast.com"
PRIORITY_NODE_2="nodes.hashvault.pro"
### End of user defined variables
MONEROD_DIR=/opt/prod/monerod
# Set the log name for monerod
LOG_NAME="${MONERO_DIR}/${LOG_NAME}"
# Make sure the daemon is being run as root
USER=$(whoami)
if [ "$USER" != "root" ]; then
echo "ERROR: You must run the monerod daemon as root, exiting.."
exit 1
fi
# Launch the monerod daemon
$MONERO_DIR/monerod \
--zmq-pub tcp://${IP_ALL}:${ZMQ_PUB_PORT} \
--zmq-rpc-bind-ip ${IP_ALL} --zmq-rpc-bind-port ${ZMQ_RPC_PORT} \
--p2p-bind-ip ${IP_ALL} --p2p-bind-port ${P2P_BIND_PORT} \
--add-priority-node=${PRIORITY_NODE_1}:${P2P_BIND_PORT} \
--add-priority-node=${PRIORITY_NODE_2}:${P2P_BIND_PORT} \
--rpc-bind-ip ${IP_ALL} --rpc-bind-port ${RPC_BIND_PORT} --restricted-rpc \
--confirm-external-bind \
--data-dir ${DATA_DIR} \
--out-peers ${OUT_PEERS} --in-peers ${IN_PEERS} \
--disable-dns-checkpoints --enable-dns-blocklist \
--log-file ${MONERO_DIR}/monerod.txt \
--log-level ${LOG_LEVEL} --max-log-files ${MAX_LOG_FILES} \
--max-log-file-size ${MAX_LOG_SIZE} \
--show-time-stats ${SHOW_TIME_STATS} \
--igd enabled \
--max-connections-per-ip 1 \
--db-sync-mode safe
Once you’ve created this shell script you’ll need to make it executible:
sudo chmod a+x /opt/prod/monerod/start-monerod.sh
The Actual Blockchain
I keep the actual blockchain in a different directory than the Monero Daemon. This
is referenced by the --data-dir
switch and the DATA_DIR
definition in the
start-monerod.sh
script described in the previous section. By keeping the blockchain
in a different directory I can download and install a newer version of the
Monero GUI Wallet, copy in the start-monerod.sh
script and fire it up without
needing to move the 202 Gb file. It also allows you to store that file on another disk
mounted on another directory in case you’re using a dedicated SSD drive to house the
blockchain.
You’ll need to create this directory:
sudo mkdir /opt/prod/monero-blockchain
Configuring the Monero Daemon to Run as a Service
As mentioned in the introduction of this page, the Monero Daemon is an interactive process that accepts commands. In order to take advantage of this feature while still running it as a service this implementation uses two distinct services. The primary service is the actual Monero Daemon. The secondary service sets up a named pipe which allows the user to send commands to the Monero Daemon by echoing commands into the named pipe.
Configuring the Monero Daemon Service
The Monero Daemon Service is configured as a standard systemd service. To do this
you need to create a systemd service description file in the /etc/systemd/system
directory. Name this file monerod.service
and create it as the root user. A complete
listing of this file is shown below.
[Unit]
Description=Monero Daemon Full Blockchain Node
After=network.target monerod.socket
BindsTo=monerod.socket
[Service]
StandardInput=socket
Sockets=monerod.socket
WorkingDirectory=/opt/prod/monerod/
Type=simple
Restart=always
ExecStart=/opt/prod/monerod/start-monerod.sh
TimeoutStopSec=60
StandardOutput=file:/opt/prod/monerod/monerod.log
StandardError=file:/opt/prod/monerod/monerod.err
[Install]
WantedBy=multi-user.target
A few things to note about the systemd service definition file, above, are:
- The systemd service runs a shell script called
start-monerod.sh
. A complete listing of this file’s contents is shown in a previous section of this page. - The service definition relies on a
monerod.socket
service. A description of thismonerod.socket
service and a complete service definition is shown in the next section.
Configuring the Monero Socket Service
This service creates a named pipe which allows you to send commands to the Monero Daemon
service. Like the Monero Daemon service definition, this service definition file should
also be created in the /etc/systemd/system
directory. A complete listing of the service
definition file is shown below. Create this file as the root user and name it monerod.socket
.
[Unit]
Description=Monerod Stdin Socket
[Socket]
ListenFIFO=/opt/prod/monerod/monerod.stdin
RemoveOnStop=true
[Install]
WantedBy=sockets.target
Note that the named pipe is called /opt/prod/monerod/monerod.stdin
. To send commands to the
Monero Daemon simply echo the command and direct the output into this named pipe. The
shell command below shows an example:
echo status > /opt/prod/monerod/monerod.stdin
If you look at the previous section where we defined the Monero Daemon service, you will
see that it includes a StandardOutput
directive. The results of the command (e.g. status)
will show up in this file.
Refreshing systemd and Configuring systemd to run the Monero Daemon at Boot Time
In order to let systemd know about these newly created services i.e. monerod.service and monerod.socket you need to issue the command below:
sudo systemd daemon-reload
To have systemd automatically start the Monero Daemon whenever your system boots, simply execute the command below:
sudo systemd enable monerod.service
That’s it! You’re done!
art P2Pool, but I found the solution to using a named pipe and P2Pool service definition in this Reddit post by Krewlar. Kudos to Krewlar for doing the heavy lifting for this solution.
Systemd Service Definitions to run P2Pool as a service
This solution uses two distinct systemd files to start P2Pool.
A socket service that creates a named pipe that connects the the P2Pool daemon’s standard input. This pipe is used to interact with the daemon once it’s running. By using this architecture, you can run processes (e.g. a cron script) to send commands to the P2Pool daemon while still running it as a service. The actual P2Pool service definition. This definition does not start P2Pool directly, instead it calls a shell script (shown below) that passes in all the options used to start the P2Pool daemon.
The systemd p2pool.socket definition
This file is installed in /etc/systemd/system
and is called p2pool.socket. A full listing of the file is shown below:
[Unit]
Description=P2Pool Socket
[Socket]
ListenFIFO=/opt/prod/p2pool/p2pool.stdin
RemoveOnStop=true
[Install]
WantedBy=sockets.target
The systemd p2pool.service Definition
This file is installed in /etc/systemd/system
and is called p2pool.service. A full listing of the file is shown below:
[Unit]
Description=P2Pool Full Node
After=network.target p2pool.socket
#Requires=monerod.service
BindsTo=p2pool.socket
[Service]
StandardInput=socket
Sockets=p2pool.socket
WorkingDirectory=/opt/prod/p2pool/
Type=simple
Restart=always
ExecStartPre=sysctl vm.nr_hugepages=3072
ExecStart=/opt/prod/p2pool/start-p2pool-mini.sh
TimeoutStopSec=60
StandardOutput=file:/opt/prod/p2pool/p2pool.log
StandardError=file:/opt/prod/p2pool/p2pool.err
[Unit]
Description=P2Pool Full Node
After=network.target p2pool.socket
#Requires=monerod.service
BindsTo=p2pool.socket
[Service]
StandardInput=socket
Sockets=p2pool.socket
WorkingDirectory=/opt/prod/p2pool/
Type=simple
Restart=always
ExecStartPre=sysctl vm.nr_hugepages=3072
ExecStart=/opt/prod/p2pool/start-p2pool-mini.sh
TimeoutStopSec=60
StandardOutput=file:/opt/prod/p2pool/p2pool.log
StandardError=file:/opt/prod/p2pool/p2pool.err
[Install]
WantedBy=multi-user.target
[Install]
WantedBy=multi-user.target
Activing and Enabling the P2Pool Service
To refresh systemd’s configuration after creating the service and socket definitions use the command below:
sudo systemd daemon-reload
To have the P2Pool daemon automatically start at boot time use the command below:
sudo systemd enable p2pool.service
To actually start the service without rebooting use the command below:
sudo systemd start p2pool.service
The P2Pool Daemon Startup Script
I use a shell script to start the P2Pool Daemon. A few points about the startup script:
- The script is not run directly, it’s used by a systemd service
- P2Pool is configured to mine on the mini sidechain (the
--mini
switch) - Be sure to substitute your own Monero wallet address for the WALLET variable
- The
MONERO_NODE
is the IP of a machine that hosts the Monero Blockchain i.e. runs the monerod daemon - The
ZMQ_PORT
andRPC_PORT
need to match what the monerod daemon’s configuration - The
P2P_DIR
is the directory where you have the P2Pool software installed
This section shows the start script used to launch P2Pool daemon.
#!/bin/bash
# kermit
MONERO_NODE="192.168.0.176"
ANY_IP="0.0.0.0"
STRATUM_PORT=3333
P2P_PORT=38889
ZMQ_PORT=20083
RPC_PORT=20081
P2P_DIR="/opt/prod/p2pool"
WALLET="******************************************************************"
LOG_LEVEL=0
IN_PEERS=10
OUT_PEERS=10
DATA_API_DIR="${P2P_DIR}/json"
P2P_LOG="${P2P_DIR}/p2pool.log"
USER=$(whoami)
if [ "$USER" != "root" ]; then
echo "ERROR: Run the p2pool daemon as root, exiting.."
exit 1
fi
./p2pool \
--host ${MONERO_NODE} \
--wallet ${WALLET} \
--mini \
--stratum ${ANY_IP}:${STRATUM_PORT} \
--p2p ${ANY_IP}:${P2P_PORT} \
--rpc-port ${RPC_PORT} \
--zmq-port ${ZMQ_PORT} \
--loglevel ${LOG_LEVEL} \
--in-peers ${IN_PEERS} \
--out-peers ${OUT_PEERS} \
| tee -a ${P2P_LOG}
Monero Daemon Startup Script on Kermit
This section documents the startup script used to launch the Monero daemon on Kermit.
#!/bin/bash
### User defined variables
# Bind to all available network interfaces
IP_ALL="0.0.0.0"
# Port numbers
ZMQ_PUB_PORT="20083"
ZMQ_RPC_PORT="20082"
P2P_BIND_PORT="20080"
RPC_BIND_PORT="20081"
PRIORITY_NODE_PORT="18080"
# Peer limits
OUT_PEERS="10"
IN_PEERS="10"
# Log settings
LOG_LEVEL="0"
MAX_LOG_FILES="5"
MAX_LOG_SIZE="100000"
LOG_NAME="monerod.log"
# Not sure
SHOW_TIME_STATS="1"
# Where the blockchain (lmdb directory) is stored
DATA_DIR="/opt/prod/monero-blockchain"
# Trusted Monero nodes to sync off
PRIORITY_NODE_1="p2pmd.xmrvsbeast.com"
PRIORITY_NODE_2="nodes.hashvault.pro"
### End of user defined variables
MONEROD_DIR=/opt/prod/monerod
MONERO_GUI_DIR=/opt/prod/monero-gui
MONERO_CLI_DIR=/opt/prod/monero-cli
# Find the monerod daemon
if [ -d $MONEROD_DIR ]; then
MONERO_DIR=$MONEROD_DIR
elif [ -d $MONERO_GUI_DIR ]; then
MONERO_DIR=$MONERO_GUI_DIR
elif [ -d $MONERO_CLI_DIR ]; then
MONERO_DIR=$MONERO_CLI_DIR
else
echo "ERROR: Unable to locate the monerod daemon, exiting..."
exit 1
fi
# Set the log name for monerod
LOG_NAME="${MONERO_DIR}/${LOG_NAME}"
# Make sure the daemon is being run as root
USER=$(whoami)
if [ "$USER" != "root" ]; then
echo "ERROR: Run the monerod daemon as root, exiting.."
exit 1
fi
# Launch the monerod daemon
$MONERO_DIR/monerod \
--zmq-pub tcp://${IP_ALL}:${ZMQ_PUB_PORT} \
--zmq-rpc-bind-ip ${IP_ALL} --zmq-rpc-bind-port ${ZMQ_RPC_PORT} \
--p2p-bind-ip ${IP_ALL} --p2p-bind-port ${P2P_BIND_PORT} \
--add-priority-node=${PRIORITY_NODE_1}:${P2P_BIND_PORT} \
--add-priority-node=${PRIORITY_NODE_2}:${P2P_BIND_PORT} \
--rpc-bind-ip ${IP_ALL} --rpc-bind-port ${RPC_BIND_PORT} --restricted-rpc \
--confirm-external-bind \
--data-dir ${DATA_DIR} \
--out-peers ${OUT_PEERS} --in-peers ${IN_PEERS} \
--disable-dns-checkpoints --enable-dns-blocklist \
--log-file ${MONERO_DIR}/monerod.txt \
--log-level ${LOG_LEVEL} --max-log-files ${MAX_LOG_FILES} \
--max-log-file-size ${MAX_LOG_SIZE} \
--show-time-stats ${SHOW_TIME_STATS} \
--igd enabled \
--max-connections-per-ip 1 \
--db-sync-mode safe