{
config,
lib,
pkgs,
...
}:
let
eachGeth = config.services.geth;
gethOpts =
name,
options = {
enable = lib.mkEnableOption "Go Ethereum Node";
port = lib.mkOption {
type = lib.types.port;
default = 30303;
description = "Port number Go Ethereum will be listening on, both TCP and UDP.";
};
http = {
enable = lib.mkEnableOption "Go Ethereum HTTP API";
address = lib.mkOption {
type = lib.types.str;
default = "127.0.0.1";
description = "Listen address of Go Ethereum HTTP API.";
default = 8545;
description = "Port number of Go Ethereum HTTP API.";
apis = lib.mkOption {
type = lib.types.nullOr (lib.types.listOf lib.types.str);
default = null;
description = "APIs to enable over WebSocket";
example = [
"net"
"eth"
];
websocket = {
enable = lib.mkEnableOption "Go Ethereum WebSocket API";
description = "Listen address of Go Ethereum WebSocket API.";
default = 8546;
description = "Port number of Go Ethereum WebSocket API.";
authrpc = {
enable = lib.mkEnableOption "Go Ethereum Auth RPC API";
description = "Listen address of Go Ethereum Auth RPC API.";
default = 8551;
description = "Port number of Go Ethereum Auth RPC API.";
vhosts = lib.mkOption {
default = [ "localhost" ];
description = "List of virtual hostnames from which to accept requests.";
"localhost"
"geth.example.org"
jwtsecret = lib.mkOption {
default = "";
description = "Path to a JWT secret for authenticated RPC endpoint.";
example = "/var/run/geth/jwtsecret";
metrics = {
enable = lib.mkEnableOption "Go Ethereum prometheus metrics";
description = "Listen address of Go Ethereum metrics service.";
default = 6060;
description = "Port number of Go Ethereum metrics service.";
network = lib.mkOption {
type = lib.types.nullOr (
lib.types.enum [
"goerli"
"holesky"
"rinkeby"
"yolov2"
"ropsten"
]
);
description = "The network to connect to. Mainnet (null) is the default ethereum network.";
syncmode = lib.mkOption {
type = lib.types.enum [
"snap"
"fast"
"full"
"light"
default = "snap";
description = "Blockchain sync mode.";
gcmode = lib.mkOption {
"archive"
default = "full";
description = "Blockchain garbage collection mode.";
maxpeers = lib.mkOption {
type = lib.types.int;
default = 50;
description = "Maximum peers to connect to.";
extraArgs = lib.mkOption {
type = lib.types.listOf lib.types.str;
description = "Additional arguments passed to Go Ethereum.";
default = [ ];
package = lib.mkPackageOption pkgs [ "go-ethereum" "geth" ] { };
in
###### interface
services.geth = lib.mkOption {
type = lib.types.attrsOf (lib.types.submodule gethOpts);
default = { };
description = "Specification of one or more geth instances.";
###### implementation
config = lib.mkIf (eachGeth != { }) {
environment.systemPackages = lib.flatten (
lib.mapAttrsToList (gethName: cfg: [
cfg.package
]) eachGeth
systemd.services = lib.mapAttrs' (
gethName: cfg:
stateDir = "goethereum/${gethName}/${if (cfg.network == null) then "mainnet" else cfg.network}";
dataDir = "/var/lib/${stateDir}";
(lib.nameValuePair "geth-${gethName}" (
lib.mkIf cfg.enable {
description = "Go Ethereum node (${gethName})";
wantedBy = [ "multi-user.target" ];
after = [ "network.target" ];
serviceConfig = {
DynamicUser = true;
Restart = "always";
StateDirectory = stateDir;
# Hardening measures
PrivateTmp = "true";
ProtectSystem = "full";
NoNewPrivileges = "true";
PrivateDevices = "true";
MemoryDenyWriteExecute = "true";
script = ''
${cfg.package}/bin/geth \
--nousb \
--ipcdisable \
${lib.optionalString (cfg.network != null) ''--${cfg.network}''} \
--syncmode ${cfg.syncmode} \
--gcmode ${cfg.gcmode} \
--port ${toString cfg.port} \
--maxpeers ${toString cfg.maxpeers} \
${lib.optionalString cfg.http.enable ''--http --http.addr ${cfg.http.address} --http.port ${toString cfg.http.port}''} \
${
lib.optionalString (cfg.http.apis != null) ''--http.api ${lib.concatStringsSep "," cfg.http.apis}''
} \
${lib.optionalString cfg.websocket.enable ''--ws --ws.addr ${cfg.websocket.address} --ws.port ${toString cfg.websocket.port}''} \
lib.optionalString (
cfg.websocket.apis != null
) ''--ws.api ${lib.concatStringsSep "," cfg.websocket.apis}''
${lib.optionalString cfg.metrics.enable ''--metrics --metrics.addr ${cfg.metrics.address} --metrics.port ${toString cfg.metrics.port}''} \
--authrpc.addr ${cfg.authrpc.address} --authrpc.port ${toString cfg.authrpc.port} --authrpc.vhosts ${lib.concatStringsSep "," cfg.authrpc.vhosts} \
if (cfg.authrpc.jwtsecret != "") then
''--authrpc.jwtsecret ${cfg.authrpc.jwtsecret}''
else
''--authrpc.jwtsecret ${dataDir}/geth/jwtsecret''
${lib.escapeShellArgs cfg.extraArgs} \
--datadir ${dataDir}
'';
}
))
) eachGeth;