2019-08-05 11:44:38 +03:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
|
let
|
|
|
|
cfg = config.services.lnd;
|
2019-11-27 16:04:23 +03:00
|
|
|
inherit (config) nix-bitcoin-services;
|
2020-01-12 22:52:39 +03:00
|
|
|
secretsDir = config.nix-bitcoin.secretsDir;
|
2019-08-05 11:44:38 +03:00
|
|
|
configFile = pkgs.writeText "lnd.conf" ''
|
|
|
|
datadir=${cfg.dataDir}
|
2019-08-23 02:38:27 +03:00
|
|
|
logdir=${cfg.dataDir}/logs
|
2019-08-05 11:44:38 +03:00
|
|
|
bitcoin.mainnet=1
|
2020-01-12 22:52:39 +03:00
|
|
|
tlscertpath=${secretsDir}/lnd-cert
|
|
|
|
tlskeypath=${secretsDir}/lnd-key
|
2019-08-05 11:44:38 +03:00
|
|
|
|
2019-11-27 16:04:43 +03:00
|
|
|
rpclisten=localhost:${toString cfg.rpcPort}
|
|
|
|
|
2019-08-05 11:44:38 +03:00
|
|
|
bitcoin.active=1
|
|
|
|
bitcoin.node=bitcoind
|
|
|
|
|
|
|
|
tor.active=true
|
|
|
|
tor.v3=true
|
|
|
|
tor.streamisolation=true
|
2019-08-23 02:38:27 +03:00
|
|
|
tor.privatekeypath=${cfg.dataDir}/v3_onion_private_key
|
2019-08-05 11:44:38 +03:00
|
|
|
|
|
|
|
bitcoind.rpcuser=${config.services.bitcoind.rpcuser}
|
|
|
|
bitcoind.zmqpubrawblock=${config.services.bitcoind.zmqpubrawblock}
|
|
|
|
bitcoind.zmqpubrawtx=${config.services.bitcoind.zmqpubrawtx}
|
|
|
|
|
|
|
|
${cfg.extraConfig}
|
|
|
|
'';
|
|
|
|
in {
|
|
|
|
|
|
|
|
options.services.lnd = {
|
|
|
|
enable = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
|
|
|
description = ''
|
|
|
|
If enabled, the LND service will be installed.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
dataDir = mkOption {
|
|
|
|
type = types.path;
|
|
|
|
default = "/var/lib/lnd";
|
|
|
|
description = "The data directory for LND.";
|
|
|
|
};
|
2019-11-27 16:04:43 +03:00
|
|
|
rpcPort = mkOption {
|
|
|
|
type = types.ints.u16;
|
|
|
|
default = 10009;
|
|
|
|
description = "Port on which to listen for gRPC connections.";
|
|
|
|
};
|
2019-08-05 11:44:38 +03:00
|
|
|
extraConfig = mkOption {
|
2020-03-09 11:22:00 +03:00
|
|
|
type = types.lines;
|
|
|
|
default = "";
|
|
|
|
example = ''
|
|
|
|
autopilot.active=1
|
|
|
|
'';
|
|
|
|
description = "Additional configurations to be appended to <filename>lnd.conf</filename>.";
|
|
|
|
};
|
|
|
|
package = mkOption {
|
|
|
|
type = types.package;
|
|
|
|
default = pkgs.nix-bitcoin.lnd;
|
|
|
|
defaultText = "pkgs.nix-bitcoin.lnd";
|
|
|
|
description = "The package providing lnd binaries.";
|
|
|
|
};
|
2019-11-27 16:04:34 +03:00
|
|
|
cli = mkOption {
|
|
|
|
readOnly = true;
|
|
|
|
default = pkgs.writeScriptBin "lncli"
|
|
|
|
# Switch user because lnd makes datadir contents readable by user only
|
|
|
|
''
|
2020-03-09 11:22:00 +03:00
|
|
|
exec sudo -u lnd ${cfg.package}/bin/lncli --tlscertpath ${secretsDir}/lnd-cert \
|
2019-11-27 16:04:34 +03:00
|
|
|
--macaroonpath '${cfg.dataDir}/chain/bitcoin/mainnet/admin.macaroon' "$@"
|
|
|
|
'';
|
|
|
|
description = "Binary to connect with the lnd instance.";
|
|
|
|
};
|
2019-08-05 11:44:38 +03:00
|
|
|
enforceTor = nix-bitcoin-services.enforceTor;
|
|
|
|
};
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
2020-03-09 11:22:00 +03:00
|
|
|
environment.systemPackages = [ cfg.package ];
|
2019-08-05 11:44:38 +03:00
|
|
|
systemd.services.lnd = {
|
|
|
|
description = "Run LND";
|
2019-11-27 16:04:21 +03:00
|
|
|
path = [ pkgs.nix-bitcoin.bitcoind ];
|
2019-08-05 11:44:38 +03:00
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
requires = [ "bitcoind.service" ];
|
|
|
|
after = [ "bitcoind.service" ];
|
|
|
|
preStart = ''
|
|
|
|
mkdir -m 0770 -p ${cfg.dataDir}
|
|
|
|
cp ${configFile} ${cfg.dataDir}/lnd.conf
|
|
|
|
chown -R 'lnd:lnd' '${cfg.dataDir}'
|
|
|
|
chmod u=rw,g=r,o= ${cfg.dataDir}/lnd.conf
|
2020-01-12 22:52:39 +03:00
|
|
|
echo "bitcoind.rpcpass=$(cat ${secretsDir}/bitcoin-rpcpassword)" >> '${cfg.dataDir}/lnd.conf'
|
2019-08-05 11:44:38 +03:00
|
|
|
'';
|
|
|
|
serviceConfig = {
|
|
|
|
PermissionsStartOnly = "true";
|
2020-03-09 11:22:00 +03:00
|
|
|
ExecStart = "${cfg.package}/bin/lnd --configfile=${cfg.dataDir}/lnd.conf";
|
2019-08-05 11:44:38 +03:00
|
|
|
User = "lnd";
|
|
|
|
Restart = "on-failure";
|
|
|
|
RestartSec = "10s";
|
|
|
|
} // nix-bitcoin-services.defaultHardening
|
|
|
|
// (if cfg.enforceTor
|
|
|
|
then nix-bitcoin-services.allowTor
|
|
|
|
else nix-bitcoin-services.allowAnyIP
|
|
|
|
) // nix-bitcoin-services.allowAnyProtocol; # For ZMQ
|
2019-11-27 16:04:45 +03:00
|
|
|
postStart = let
|
|
|
|
mainnetDir = "${cfg.dataDir}/chain/bitcoin/mainnet";
|
|
|
|
in ''
|
2019-11-27 16:04:39 +03:00
|
|
|
umask 377
|
|
|
|
|
2019-11-27 16:04:41 +03:00
|
|
|
attempts=50
|
|
|
|
while ! { exec 3>/dev/tcp/127.0.0.1/8080 && exec 3>&-; } &>/dev/null; do
|
|
|
|
((attempts-- == 0)) && { echo "lnd REST service unreachable"; exit 1; }
|
|
|
|
sleep 0.1
|
|
|
|
done
|
2019-11-27 16:04:39 +03:00
|
|
|
|
2020-01-12 22:52:39 +03:00
|
|
|
if [[ ! -f ${secretsDir}/lnd-seed-mnemonic ]]; then
|
2019-11-27 16:04:45 +03:00
|
|
|
echo Create lnd seed
|
2019-11-27 16:04:39 +03:00
|
|
|
|
|
|
|
${pkgs.curl}/bin/curl -s \
|
2020-01-12 22:52:39 +03:00
|
|
|
--cacert ${secretsDir}/lnd-cert \
|
|
|
|
-X GET https://127.0.0.1:8080/v1/genseed | ${pkgs.jq}/bin/jq -c '.cipher_seed_mnemonic' > ${secretsDir}/lnd-seed-mnemonic
|
2019-11-27 16:04:39 +03:00
|
|
|
fi
|
|
|
|
|
2019-11-27 16:04:45 +03:00
|
|
|
if [[ ! -f ${mainnetDir}/wallet.db ]]; then
|
|
|
|
echo Create lnd wallet
|
2019-11-27 16:04:39 +03:00
|
|
|
|
2019-11-27 16:04:42 +03:00
|
|
|
${pkgs.curl}/bin/curl -s --output /dev/null --show-error \
|
2020-01-12 22:52:39 +03:00
|
|
|
--cacert ${secretsDir}/lnd-cert \
|
|
|
|
-X POST -d "{\"wallet_password\": \"$(cat ${secretsDir}/lnd-wallet-password | tr -d '\n' | base64 -w0)\", \
|
|
|
|
\"cipher_seed_mnemonic\": $(cat ${secretsDir}/lnd-seed-mnemonic | tr -d '\n')}" \
|
2019-11-27 16:04:40 +03:00
|
|
|
https://127.0.0.1:8080/v1/initwallet
|
2019-11-27 16:04:45 +03:00
|
|
|
|
|
|
|
# Guarantees that RPC calls with cfg.cli succeed after the service is started
|
|
|
|
echo Wait until wallet is created
|
|
|
|
while [[ ! -f ${mainnetDir}/admin.macaroon ]]; do
|
|
|
|
sleep 0.1
|
|
|
|
done
|
2019-11-27 16:04:39 +03:00
|
|
|
else
|
2019-11-27 16:04:45 +03:00
|
|
|
echo Unlock lnd wallet
|
2019-11-27 16:04:39 +03:00
|
|
|
|
|
|
|
${pkgs.curl}/bin/curl -s \
|
2019-11-27 16:04:45 +03:00
|
|
|
-H "Grpc-Metadata-macaroon: $(${pkgs.xxd}/bin/xxd -ps -u -c 99999 '${mainnetDir}/admin.macaroon')" \
|
2020-01-12 22:52:39 +03:00
|
|
|
--cacert ${secretsDir}/lnd-cert \
|
2019-11-27 16:04:40 +03:00
|
|
|
-X POST \
|
2020-01-12 22:52:39 +03:00
|
|
|
-d "{\"wallet_password\": \"$(cat ${secretsDir}/lnd-wallet-password | tr -d '\n' | base64 -w0)\"}" \
|
2019-11-27 16:04:40 +03:00
|
|
|
https://127.0.0.1:8080/v1/unlockwallet
|
2019-11-27 16:04:39 +03:00
|
|
|
fi
|
2019-11-27 16:04:44 +03:00
|
|
|
|
|
|
|
# Wait until the RPC port is open
|
|
|
|
while ! { exec 3>/dev/tcp/127.0.0.1/${toString cfg.rpcPort}; } &>/dev/null; do
|
|
|
|
sleep 0.1
|
|
|
|
done
|
2019-11-27 16:04:39 +03:00
|
|
|
'';
|
2019-08-05 11:44:38 +03:00
|
|
|
};
|
2020-01-12 22:52:37 +03:00
|
|
|
users.users.lnd = {
|
|
|
|
description = "LND User";
|
|
|
|
group = "lnd";
|
|
|
|
extraGroups = [ "bitcoinrpc" ];
|
|
|
|
home = cfg.dataDir;
|
|
|
|
};
|
|
|
|
users.groups.lnd = {};
|
2020-01-12 22:52:38 +03:00
|
|
|
nix-bitcoin.secrets = {
|
|
|
|
lnd-wallet-password.user = "lnd";
|
|
|
|
lnd-key.user = "lnd";
|
|
|
|
lnd-cert.user = "lnd";
|
|
|
|
};
|
2019-08-05 11:44:38 +03:00
|
|
|
};
|
|
|
|
}
|