- Recommended Setup for Consensus Nodes
- SSH Authentication Keys
- Service Provider Specific Configuration
- Linux Server Configuration
Remote access to machines should be restricted to SSH using public-key with physical Yubico 4/NEO keys.
Nodes should not be run in shared machines for security concerns. Preferred form is to run the nodes with colocation in high security datacenters like Deltalis, Equinix and A1 Arsenal. The hardware configurations for the IaaS providers below can serve as a guideline of expectations for 2018.
Since consensus nodes are the ones computing, the network hardware requirements need to be reviewed periodically to guarantee that performance is optimal.
If colocation is not possible, the recommended IaaS providers and minimum configurations are listed below:
-
- CPU: Intel Xeon E3-1270v6 - 4c/8t - 3.8GHz
- RAM: 32GB DDR4 ECC 2133 MHz
- SSD: softraid-1 2x450GB NVMe
- NET: 1 Gbps
-
- CPU: Intel® Xeon E3-1240v5 - 4c/8t - 3.5GHz
- RAM: 32 GB DDR3 ECC 1333 MHz
- SSD: softraid-1 2x120GB Enterprise SSD
- NET: 2 x 1 Gbps Bonded
-
- CPU: Intel Xeon E3-1275v6 - 4c/8t - 3.8GHz
- RAM: 32 GB DDR4 ECC
- SSD: softraid-1 2x240GB Enterprise SSD
- NET: 1 Gbps
Use a password manager to hold every password on this setup (Lastpass and Dashlane are recommended). All services should be setup with 2FA enabled and with physical keys when the option is available.
All passwords should be considered strong (use this lastpass solution).
Restricting SSH login to public keys makes it as good as the keys protection, so we require the use of a physical OpenPGP SmartCard for SSH authentication. We recommend Yubikey 4. For more about the abilities of PGP on Yubikey, refer to official documentation.
- Full Windows guide
- Full Linux/MacOS
Change the defaults Admin PIN from 12345678
and PIN from 123456
to secure memorized combinations.
After adding your key to the SmartCard enabled authentication agent, gpg-agent comes bundled in gpg2 - we recommend that.
On the provider firewall (outside OS settings), setup for block all and add exceptions to ports 22, 20333 and 10333 only. If other services share the account, please be sure to place the nodes in an anti-affinity group.
Every node needs 2 managers, located in different jurisdictions. Each should have a dedicated user able to login into the system and a third consensus user (not acessible by SSH) should be the only with access to the private keys for the consensus node (careful with the Ubuntu version, this guide uses 16.04 LTS).
On first login, give a strong password to root, you will only need this if sudo password is lost (or to revoke those):
su (sudo su on Ubuntu)
passwd
Update packages, Ubuntu:
apt-get update
apt-get upgrade
CentOS:
yum update
Add manager users (repeat for both managers):
useradd -m node
mkdir /home/node/.ssh
chmod 700 /home/node/.ssh
This guide is based on bash, so set that as preferred shell for the manager:
usermod -s /bin/bash node
Copy the public key of Yubikey public key from ssh-add -L
of managers into authorized_keys:
vim /home/node/.ssh/authorized_keys
Setup permissions:
chmod 400 /home/node/.ssh/authorized_keys
chown node:node /home/node -R
Setup a password for managers (this will be sudo passwords):
passwd node
Now we will setup sudo for managers, add %sudo
goupd and comment with #
any other group that is not root
.
visudo
The file should look like:
# This file MUST be edited with the 'visudo' command as root.
#
# Please consider adding local content in /etc/sudoers.d/ instead of
# directly modifying this file.
#
# See the man page for details on how to write a sudoers file.
#
Defaults env_reset
Defaults mail_badpass
Defaults secure_path="/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/snap/bin"
# Host alias specification
# User alias specification
# Cmnd alias specification
# User privilege specification
root ALL=(ALL:ALL) ALL
# Members of the admin group may gain root privileges
#%admin ALL=(ALL) ALL
# Allow members of group sudo to execute any command
%sudo ALL=(ALL:ALL) ALL
# See sudoers(5) for more information on "#include" directives:
#includedir /etc/sudoers.d
Add managers to sudo group:
usermod -aG sudo node
You will need to login and logout from manager account to update it:
su -l node
exit
Back as su, now lock SSH login to public key and managers only:
vim /etc/ssh/sshd_config
These lines should be added/modified, add all managers separated by space in AllowUsers:
X11Forwarding no
PermitRootLogin no
PasswordAuthentication no
AllowUsers node
LoginGraceTime 30
AllowTcpForwarding no
TCPKeepAlive no
AllowAgentForwarding no
DebianBanner no
Banner /etc/ssh/sshd-banner
Setup a legal banner for SSH:
echo "WARNING: Unauthorized access to this system is forbidden and will be
prosecuted by law. By accessing this system, you agree that your actions
may be monitored if unauthorized usage is suspected." >> /etc/ssh/sshd-banner
Add user to run consensus node:
useradd consensus
mkdir /home/consensus
chown consensus:consensus /home/consensus -R
Create a very strong password for consensus, this should be shared in a secure way between managers:
passwd consensus
Last step as su is to reboot ssh, after that login back as a manager.
sudo systemctl restart sshd.service
First thing is to install firewall and lockdown the node, on Debian (Ubuntu):
sudo apt-get install ufw
On CentOS:
sudo yum install epel-release
sudo yum install ufw
Set IPV6 to yes in vim /etc/default/ufw
and allow only the ports used:
sudo ufw default deny incoming
sudo ufw allow ssh
sudo ufw allow 10333
sudo ufw allow 20333
sudo ufw disable
sudo ufw enable
sudo apt-get install unattended-upgrades
sudo vim /etc/apt/apt.conf.d/10periodic
Update to match:
APT::Periodic::Update-Package-Lists "1";
APT::Periodic::Download-Upgradeable-Packages "1";
APT::Periodic::AutocleanInterval "7";
APT::Periodic::Unattended-Upgrade "1";
Now disable non security automatic updates:
sudo vim /etc/apt/apt.conf.d/50unattended-upgrades
Update so that the only uncommented lines are:
Unattended-Upgrade::Allowed-Origins {
"${distro_id}:${distro_codename}";
"${distro_id}:${distro_codename}-security";
"${distro_id}ESM:${distro_codename}";
};
Install yum-cron and enable it for security packages:
sudo yum -y install yum-cron
sudo systemctl start yum-cron
sudo systemctl enable yum-cron
sudo nano /etc/yum/yum-cron.conf
Now change it to match:
update_cmd = security
apply_updates = yes
emit_via = email
email_to = YOUR_EMAIL_TO_RECEIVE_UPDATE_NOTIFICATIONS
On email_to, insert the e-mail you want to be warned when updates occur.
Now we will install fail2ban, a tool that bans IPs in iptables that have suspicious behaviour. It has good defaults, so just installing is enough. On Ubuntu:
sudo apt-get install fail2ban
On CentOS:
sudo yum install fail2ban
2FA combined with physical OpenPGP keys for SSH makes for a very strong auth setup. On Ubuntu install it using:
sudo apt-get install libpam-google-authenticator
On CentOS (with epel enabled - see above):
sudo yum install google-authenticator
After installing, follow the instructions when running the command as manager, (anwser as y/y/y/n/y), do this as both managers first, as we will update PAM to require the 2FA after this step:
google-authenticator
Now edit the SSH configuration to require the 2FA setup:
sudo vim /etc/pam.d/sshd
Add the following line at the end of the file:
auth required pam_google_authenticator.so
Comment out the line about password prompt, on Ubuntu:
# Standard Un*x authentication.
#@include common-auth
On CentOS:
#auth substack password-auth
Edit the sshd_config file to require the 2FA challenge:
sudo vim /etc/ssh/sshd_config
Edit file to allow challenge and add line with auth methods:
ChallengeResponseAuthentication yes
AuthenticationMethods publickey,password publickey,keyboard-interactive
Now restart SSHD service:
sudo systemctl restart sshd.service
Without closing your current SSH session, open another window and try to login to confirm that the setup is working correctly.
Monitoring a consensus node is very important to spot issues and help improve the NEO project. We will install netdata and access it only by a SSH tunnel (we keep the firewall blocking setup).
Install the pre-built static version (to reduce attack surface and not have unneeded dependencies):
bash <(curl -Ss https://my-netdata.io/kickstart-static64.sh)
To access it, create a SHH tunnel and open a browser to localhost:19999
ssh -f node@SERVERIP -L 19999:SERVERIP:19999 -N
sudo apt-get install mailutils
On CentOS:
sudo yum install mailx
Edit the default bash profile:
sudo vim /etc/profile
Add the following lines at the end of the file, edit with your e-mail to receive login notifications:
SIP="$(echo $SSH_CONNECTION | cut -d " " -f 1)"
SHOSTNAME=$(hostname)
SNOW=$(date +"%e %b %Y, %a %r")
echo 'Someone from '$SIP' logged into '$SHOSTNAME' on '$SNOW'.' | mail -s 'SSH Login Notification' '[email protected]'
Configure logwatch to send you a daily summary of the actions on the node (that should be none usally), on Ubuntu:
sudo apt-get install logwatch
On CentOS:
sudo yum install logwatch
Now add a cron job to send your e-mail the summary:
sudo vim /etc/cron.daily/00logwatch
Change the default execution to:
/usr/sbin/logwatch --output mail --mailto [email protected] --detail high
We will reduce attack surface by blacklisting unneeded modules, normally WiFi and Bluetooth are already off (verify!) server kernels so we need to disable only USB storage.
sudo vi /etc/modprobe.d/blacklist.conf
Add the lines:
blacklist usb-storage
blacklist firewire-core
Install pre-requisites, on Ubuntu:
sudo apt-get install unzip sqlite3 libsqlite3-dev libleveldb-dev libunwind-dev
On CentOS:
sudo yum install unzip leveldb-devel libunwind-devel
Login as the consensus user:
su consensus
cd ~
Download, verify the checksum and unzip the latests neo-cli client for your distro:
wget https://github.com/neo-project/neo-cli/releases/download/v2.5.2/neo-cli-YOURDISTRIBUTION.zip
sha256sum neo-cli-YOURDISTRIBUTION.zip
unzip neo-cli-YOURDISTRIBUTION.zip
cd neo-cli
chmod u+x neo-cli
Now copy the setup you are running the node for (testnet or mainnet):
mv protocol.json protocol.json.back
cp protocol.testnet.json protocol.json
If this is the first time, you will need to create a wallet for consensus:
./neo-cli
neo> create wallet /home/consensus/cn_wallet.json
password: SOMESTRONGPASSWORD
password: SOMESTRONGPASSWORD
Now copy the start_consensus script:
cd ~
wget https://raw.githubusercontent.com/CityOfZion/standards/master/assets/nodes/start_consensus.sh
chmod u+x start_consensus.sh
Edit the contents to match your wallet file location and password. Now let's run it under control of supervisord, on Ubuntu:
sudo apt-get install supervisor
On CentOS:
sudo yum install supervisor
Now configure supervisord to execute start_consensus (edit the file if needed):
wget https://raw.githubusercontent.com/CityOfZion/standards/master/assets/nodes/supervisord.conf
chmod 700 supervisord.conf
cp supervisord.conf /etc/supervisord.conf
sudo supervisord
To make supervisord run automatically on system restart, you need to add its init script.
That is all, now logout from the server and only login if any update must be deployed or ill-behaviour is detected.
- GRUB Password
- Disk Encryption
- Enable TCP SYN Cookie (net.ipv4.tcp_syncookies = 1 -> /etc/sysctl.conf)