You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
660 lines
20 KiB
660 lines
20 KiB
#include <arpa/inet.h>
|
|
#include <bitcoin/chainparams.h>
|
|
#include <ccan/array_size/array_size.h>
|
|
#include <ccan/err/err.h>
|
|
#include <ccan/opt/opt.h>
|
|
#include <ccan/read_write_all/read_write_all.h>
|
|
#include <ccan/short_types/short_types.h>
|
|
#include <ccan/tal/grab_file/grab_file.h>
|
|
#include <ccan/tal/str/str.h>
|
|
#include <common/configdir.h>
|
|
#include <common/version.h>
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <inttypes.h>
|
|
#include <lightningd/bitcoind.h>
|
|
#include <lightningd/chaintopology.h>
|
|
#include <lightningd/lightningd.h>
|
|
#include <lightningd/log.h>
|
|
#include <lightningd/netaddress.h>
|
|
#include <lightningd/opt_time.h>
|
|
#include <lightningd/options.h>
|
|
#include <lightningd/subd.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
#include <sys/types.h>
|
|
#include <unistd.h>
|
|
#include <wire/wire.h>
|
|
|
|
/* Tal wrappers for opt. */
|
|
static void *opt_allocfn(size_t size)
|
|
{
|
|
return tal_alloc_(NULL, size, false, false, TAL_LABEL("opt_allocfn", ""));
|
|
}
|
|
|
|
static void *tal_reallocfn(void *ptr, size_t size)
|
|
{
|
|
if (!ptr)
|
|
return opt_allocfn(size);
|
|
tal_resize_(&ptr, 1, size, false);
|
|
return ptr;
|
|
}
|
|
|
|
static void tal_freefn(void *ptr)
|
|
{
|
|
tal_free(ptr);
|
|
}
|
|
|
|
/* FIXME: Put into ccan/time. */
|
|
#define TIME_FROM_SEC(sec) { { .tv_nsec = 0, .tv_sec = sec } }
|
|
#define TIME_FROM_MSEC(msec) \
|
|
{ { .tv_nsec = ((msec) % 1000) * 1000000, .tv_sec = (msec) / 1000 } }
|
|
|
|
static char *opt_set_u64(const char *arg, u64 *u)
|
|
{
|
|
char *endp;
|
|
unsigned long long l;
|
|
|
|
/* This is how the manpage says to do it. Yech. */
|
|
errno = 0;
|
|
l = strtoull(arg, &endp, 0);
|
|
if (*endp || !arg[0])
|
|
return tal_fmt(NULL, "'%s' is not a number", arg);
|
|
*u = l;
|
|
if (errno || *u != l)
|
|
return tal_fmt(NULL, "'%s' is out of range", arg);
|
|
return NULL;
|
|
}
|
|
|
|
static char *opt_set_u32(const char *arg, u32 *u)
|
|
{
|
|
char *endp;
|
|
unsigned long l;
|
|
|
|
/* This is how the manpage says to do it. Yech. */
|
|
errno = 0;
|
|
l = strtoul(arg, &endp, 0);
|
|
if (*endp || !arg[0])
|
|
return tal_fmt(NULL, "'%s' is not a number", arg);
|
|
*u = l;
|
|
if (errno || *u != l)
|
|
return tal_fmt(NULL, "'%s' is out of range", arg);
|
|
return NULL;
|
|
}
|
|
|
|
static char *opt_set_u16(const char *arg, u16 *u)
|
|
{
|
|
char *endp;
|
|
unsigned long l;
|
|
|
|
/* This is how the manpage says to do it. Yech. */
|
|
errno = 0;
|
|
l = strtoul(arg, &endp, 0);
|
|
if (*endp || !arg[0])
|
|
return tal_fmt(NULL, "'%s' is not a number", arg);
|
|
*u = l;
|
|
if (errno || *u != l)
|
|
return tal_fmt(NULL, "'%s' is out of range", arg);
|
|
return NULL;
|
|
}
|
|
|
|
static char *opt_set_s32(const char *arg, s32 *u)
|
|
{
|
|
char *endp;
|
|
long l;
|
|
|
|
/* This is how the manpage says to do it. Yech. */
|
|
errno = 0;
|
|
l = strtol(arg, &endp, 0);
|
|
if (*endp || !arg[0])
|
|
return tal_fmt(NULL, "'%s' is not a number", arg);
|
|
*u = l;
|
|
if (errno || *u != l)
|
|
return tal_fmt(NULL, "'%s' is out of range", arg);
|
|
return NULL;
|
|
}
|
|
|
|
/* FIXME: Rename ipaddr and hoist up */
|
|
bool parse_ipaddr(const char *arg, struct ipaddr *addr, u16 port)
|
|
{
|
|
struct in6_addr v6;
|
|
struct in_addr v4;
|
|
|
|
/* FIXME: change arg to addr[:port] and use getaddrinfo? */
|
|
if (streq(arg, "localhost"))
|
|
arg = "127.0.0.1";
|
|
else if (streq(arg, "ip6-localhost"))
|
|
arg = "::1";
|
|
|
|
memset(&addr->addr, 0, sizeof(addr->addr));
|
|
if (inet_pton(AF_INET, arg, &v4) == 1) {
|
|
addr->type = ADDR_TYPE_IPV4;
|
|
addr->addrlen = 4;
|
|
addr->port = port;
|
|
memcpy(&addr->addr, &v4, addr->addrlen);
|
|
return true;
|
|
} else if (inet_pton(AF_INET6, arg, &v6) == 1) {
|
|
addr->type = ADDR_TYPE_IPV6;
|
|
addr->addrlen = 16;
|
|
addr->port = port;
|
|
memcpy(&addr->addr, &v6, addr->addrlen);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static char *opt_add_ipaddr(const char *arg, struct lightningd *ld)
|
|
{
|
|
size_t n = tal_count(ld->wireaddrs);
|
|
|
|
tal_resize(&ld->wireaddrs, n+1);
|
|
|
|
if (parse_ipaddr(arg, &ld->wireaddrs[n], ld->portnum))
|
|
return NULL;
|
|
|
|
return tal_fmt(NULL, "Unable to parse IP address '%s'", arg);
|
|
}
|
|
|
|
static void opt_show_u64(char buf[OPT_SHOW_LEN], const u64 *u)
|
|
{
|
|
snprintf(buf, OPT_SHOW_LEN, "%"PRIu64, *u);
|
|
}
|
|
|
|
static void opt_show_u32(char buf[OPT_SHOW_LEN], const u32 *u)
|
|
{
|
|
snprintf(buf, OPT_SHOW_LEN, "%"PRIu32, *u);
|
|
}
|
|
|
|
static void opt_show_s32(char buf[OPT_SHOW_LEN], const s32 *u)
|
|
{
|
|
snprintf(buf, OPT_SHOW_LEN, "%"PRIi32, *u);
|
|
}
|
|
|
|
static void opt_show_u16(char buf[OPT_SHOW_LEN], const u16 *u)
|
|
{
|
|
snprintf(buf, OPT_SHOW_LEN, "%u", *u);
|
|
}
|
|
|
|
static char *opt_set_network(const char *arg, struct lightningd *ld)
|
|
{
|
|
ld->topology->bitcoind->chainparams = chainparams_for_network(arg);
|
|
if (!ld->topology->bitcoind->chainparams)
|
|
return tal_fmt(NULL, "Unknown network name '%s'", arg);
|
|
return NULL;
|
|
}
|
|
|
|
static void opt_show_network(char buf[OPT_SHOW_LEN],
|
|
const struct lightningd *ld)
|
|
{
|
|
snprintf(buf, OPT_SHOW_LEN, "%s", get_chainparams(ld)->network_name);
|
|
}
|
|
|
|
static char *opt_set_rgb(const char *arg, struct lightningd *ld)
|
|
{
|
|
ld->rgb = tal_free(ld->rgb);
|
|
/* BOLT #7:
|
|
*
|
|
* the first byte of `rgb` is the red value, the second byte is the
|
|
* green value and the last byte is the blue value */
|
|
ld->rgb = tal_hexdata(ld, arg, strlen(arg));
|
|
if (!ld->rgb || tal_len(ld->rgb) != 3)
|
|
return tal_fmt(NULL, "rgb '%s' is not six hex digits", arg);
|
|
return NULL;
|
|
}
|
|
|
|
static char *opt_set_alias(const char *arg, struct lightningd *ld)
|
|
{
|
|
ld->alias = tal_free(ld->alias);
|
|
/* BOLT #7:
|
|
*
|
|
* * [`32`:`alias`]
|
|
*...
|
|
* It MUST set `alias` to a valid UTF-8 string, with any `alias` bytes
|
|
* following equal to zero.
|
|
*/
|
|
if (strlen(arg) > 32)
|
|
return tal_fmt(NULL, "Alias '%s' is over 32 characters", arg);
|
|
ld->alias = tal_arrz(ld, char, 33);
|
|
strncpy(ld->alias, arg, 32);
|
|
return NULL;
|
|
}
|
|
|
|
static void config_register_opts(struct lightningd *ld)
|
|
{
|
|
opt_register_arg("--locktime-blocks", opt_set_u32, opt_show_u32,
|
|
&ld->config.locktime_blocks,
|
|
"Blocks before peer can unilaterally spend funds");
|
|
opt_register_arg("--max-locktime-blocks", opt_set_u32, opt_show_u32,
|
|
&ld->config.locktime_max,
|
|
"Maximum seconds peer can lock up our funds");
|
|
opt_register_arg("--anchor-onchain", opt_set_u32, opt_show_u32,
|
|
&ld->config.anchor_onchain_wait,
|
|
"Blocks before we give up on pending anchor transaction");
|
|
opt_register_arg("--anchor-confirms", opt_set_u32, opt_show_u32,
|
|
&ld->config.anchor_confirms,
|
|
"Confirmations required for anchor transaction");
|
|
opt_register_arg("--max-anchor-confirms", opt_set_u32, opt_show_u32,
|
|
&ld->config.anchor_confirms_max,
|
|
"Maximum confirmations other side can wait for anchor transaction");
|
|
opt_register_arg("--forever-confirms", opt_set_u32, opt_show_u32,
|
|
&ld->config.forever_confirms,
|
|
"Confirmations after which we consider a reorg impossible");
|
|
opt_register_arg("--commit-fee-min=<percent>", opt_set_u32, opt_show_u32,
|
|
&ld->config.commitment_fee_min_percent,
|
|
"Minimum percentage of fee to accept for commitment");
|
|
opt_register_arg("--commit-fee-max=<percent>", opt_set_u32, opt_show_u32,
|
|
&ld->config.commitment_fee_max_percent,
|
|
"Maximum percentage of fee to accept for commitment (0 for unlimited)");
|
|
opt_register_arg("--commit-fee=<percent>", opt_set_u32, opt_show_u32,
|
|
&ld->config.commitment_fee_percent,
|
|
"Percentage of fee to request for their commitment");
|
|
opt_register_arg("--override-fee-rate", opt_set_u64, opt_show_u64,
|
|
&ld->topology->override_fee_rate,
|
|
"Force a specific rate in satoshis per kb regardless of estimated fees");
|
|
opt_register_arg("--default-fee-rate", opt_set_u64, opt_show_u64,
|
|
&ld->topology->default_fee_rate,
|
|
"Satoshis per kb if can't estimate fees");
|
|
opt_register_arg("--min-htlc-expiry", opt_set_u32, opt_show_u32,
|
|
&ld->config.min_htlc_expiry,
|
|
"Minimum number of blocks to accept an HTLC before expiry");
|
|
opt_register_arg("--max-htlc-expiry", opt_set_u32, opt_show_u32,
|
|
&ld->config.max_htlc_expiry,
|
|
"Maximum number of blocks to accept an HTLC before expiry");
|
|
opt_register_arg("--deadline-blocks", opt_set_u32, opt_show_u32,
|
|
&ld->config.deadline_blocks,
|
|
"Number of blocks before HTLC timeout before we drop connection");
|
|
opt_register_arg("--bitcoind-poll", opt_set_time, opt_show_time,
|
|
&ld->config.poll_time,
|
|
"Time between polling for new transactions");
|
|
opt_register_arg("--commit-time", opt_set_time, opt_show_time,
|
|
&ld->config.commit_time,
|
|
"Time after changes before sending out COMMIT");
|
|
opt_register_arg("--fee-base", opt_set_u32, opt_show_u32,
|
|
&ld->config.fee_base,
|
|
"Millisatoshi minimum to charge for HTLC");
|
|
opt_register_arg("--fee-per-satoshi", opt_set_s32, opt_show_s32,
|
|
&ld->config.fee_per_satoshi,
|
|
"Microsatoshi fee for every satoshi in HTLC");
|
|
opt_register_noarg("--no-reconnect", opt_set_bool,
|
|
&ld->config.no_reconnect, "Disable automatic reconnect attempts");
|
|
|
|
opt_register_arg("--ipaddr", opt_add_ipaddr, NULL,
|
|
ld,
|
|
"Set the IP address (v4 or v6) to announce to the network for incoming connections");
|
|
|
|
opt_register_early_arg("--network", opt_set_network, opt_show_network,
|
|
ld,
|
|
"Select the network parameters (bitcoin, testnet,"
|
|
" regtest, or litecoin)");
|
|
}
|
|
|
|
#if DEVELOPER
|
|
static char *opt_set_hsm_seed(const char *arg, struct lightningd *ld)
|
|
{
|
|
ld->dev_hsm_seed = tal_hexdata(ld, arg, strlen(arg));
|
|
if (ld->dev_hsm_seed)
|
|
return NULL;
|
|
|
|
return tal_fmt(NULL, "bad hex string '%s'", arg);
|
|
}
|
|
|
|
static void dev_register_opts(struct lightningd *ld)
|
|
{
|
|
opt_register_noarg("--dev-no-broadcast", opt_set_bool,
|
|
&ld->topology->dev_no_broadcast, opt_hidden);
|
|
opt_register_noarg("--dev-fail-on-subdaemon-fail", opt_set_bool,
|
|
&ld->dev_subdaemon_fail, opt_hidden);
|
|
opt_register_arg("--dev-debugger=<subdaemon>", opt_subd_debug, NULL,
|
|
ld, "Wait for gdb attach at start of <subdaemon>");
|
|
opt_register_arg("--dev-broadcast-interval=<ms>", opt_set_uintval,
|
|
opt_show_uintval, &ld->broadcast_interval,
|
|
"Time between gossip broadcasts in milliseconds (default: 30000)");
|
|
opt_register_arg("--dev-disconnect=<filename>", opt_subd_dev_disconnect,
|
|
NULL, ld, "File containing disconnection points");
|
|
opt_register_arg("--dev-hsm-seed=<seed>", opt_set_hsm_seed,
|
|
NULL, ld, "Hex-encoded seed for HSM");
|
|
}
|
|
#endif
|
|
|
|
static const struct config testnet_config = {
|
|
/* 6 blocks to catch cheating attempts. */
|
|
.locktime_blocks = 6,
|
|
|
|
/* They can have up to 3 days. */
|
|
.locktime_max = 3 * 6 * 24,
|
|
|
|
/* Testnet can have long runs of empty blocks. */
|
|
.anchor_onchain_wait = 100,
|
|
|
|
/* We're fairly trusting, under normal circumstances. */
|
|
.anchor_confirms = 1,
|
|
|
|
/* More than 10 confirms seems overkill. */
|
|
.anchor_confirms_max = 10,
|
|
|
|
/* At some point, you've got to let it go... */
|
|
/* FIXME-OLD #onchain:
|
|
*
|
|
* Outputs... are considered *irrevocably resolved* once they
|
|
* are included in a block at least 100 deep on the most-work
|
|
* blockchain. 100 blocks is far greater than the longest
|
|
* known bitcoin fork, and the same value used to wait for
|
|
* confirmations of miner's rewards[1].
|
|
*/
|
|
.forever_confirms = 10,
|
|
|
|
/* Testnet fees are crazy, allow infinite feerange. */
|
|
.commitment_fee_min_percent = 0,
|
|
.commitment_fee_max_percent = 0,
|
|
|
|
/* We offer to pay 5 times 2-block fee */
|
|
.commitment_fee_percent = 500,
|
|
|
|
/* Don't bother me unless I have 6 hours to collect. */
|
|
.min_htlc_expiry = 6 * 6,
|
|
/* Don't lock up channel for more than 5 days. */
|
|
.max_htlc_expiry = 5 * 6 * 24,
|
|
|
|
/* If we're closing on HTLC expiry, and you're unresponsive, we abort. */
|
|
.deadline_blocks = 4,
|
|
|
|
/* How often to bother bitcoind. */
|
|
.poll_time = TIME_FROM_SEC(10),
|
|
|
|
/* Send commit 10msec after receiving; almost immediately. */
|
|
.commit_time = TIME_FROM_MSEC(10),
|
|
|
|
/* Allow dust payments */
|
|
.fee_base = 1,
|
|
/* Take 0.001% */
|
|
.fee_per_satoshi = 10,
|
|
|
|
/* Automatically reconnect */
|
|
.no_reconnect = false,
|
|
};
|
|
|
|
/* aka. "Dude, where's my coins?" */
|
|
static const struct config mainnet_config = {
|
|
/* ~one day to catch cheating attempts. */
|
|
.locktime_blocks = 6 * 24,
|
|
|
|
/* They can have up to 3 days. */
|
|
.locktime_max = 3 * 6 * 24,
|
|
|
|
/* You should get in within 10 blocks. */
|
|
.anchor_onchain_wait = 10,
|
|
|
|
/* We're fairly trusting, under normal circumstances. */
|
|
.anchor_confirms = 3,
|
|
|
|
/* More than 10 confirms seems overkill. */
|
|
.anchor_confirms_max = 10,
|
|
|
|
/* At some point, you've got to let it go... */
|
|
/* FIXME-OLD #onchain:
|
|
*
|
|
* Outputs... are considered *irrevocably resolved* once they
|
|
* are included in a block at least 100 deep on the most-work
|
|
* blockchain. 100 blocks is far greater than the longest
|
|
* known bitcoin fork, and the same value used to wait for
|
|
* confirmations of miner's rewards[1].
|
|
*/
|
|
.forever_confirms = 100,
|
|
|
|
/* Insist between 2 and 20 times the 2-block fee. */
|
|
.commitment_fee_min_percent = 200,
|
|
.commitment_fee_max_percent = 2000,
|
|
|
|
/* We offer to pay 5 times 2-block fee */
|
|
.commitment_fee_percent = 500,
|
|
|
|
/* Don't bother me unless I have 6 hours to collect. */
|
|
.min_htlc_expiry = 6 * 6,
|
|
/* Don't lock up channel for more than 5 days. */
|
|
.max_htlc_expiry = 5 * 6 * 24,
|
|
|
|
/* If we're closing on HTLC expiry, and you're unresponsive, we abort. */
|
|
.deadline_blocks = 10,
|
|
|
|
/* How often to bother bitcoind. */
|
|
.poll_time = TIME_FROM_SEC(30),
|
|
|
|
/* Send commit 10msec after receiving; almost immediately. */
|
|
.commit_time = TIME_FROM_MSEC(10),
|
|
|
|
/* Discourage dust payments */
|
|
.fee_base = 546000,
|
|
/* Take 0.001% */
|
|
.fee_per_satoshi = 10,
|
|
|
|
/* Automatically reconnect */
|
|
.no_reconnect = false,
|
|
};
|
|
|
|
static void check_config(struct lightningd *ld)
|
|
{
|
|
/* We do this by ensuring it's less than the minimum we would accept. */
|
|
if (ld->config.commitment_fee_max_percent != 0
|
|
&& ld->config.commitment_fee_max_percent
|
|
< ld->config.commitment_fee_min_percent)
|
|
fatal("Commitment fee invalid min-max %u-%u",
|
|
ld->config.commitment_fee_min_percent,
|
|
ld->config.commitment_fee_max_percent);
|
|
|
|
if (ld->config.forever_confirms < 100 && !get_chainparams(ld)->testnet)
|
|
log_unusual(ld->log,
|
|
"Warning: forever-confirms of %u is less than 100!",
|
|
ld->config.forever_confirms);
|
|
|
|
if (ld->config.anchor_confirms == 0)
|
|
fatal("anchor-confirms must be greater than zero");
|
|
|
|
/* FIXME-OLD #2:
|
|
*
|
|
* a node MUST estimate the deadline for successful redemption
|
|
* for each HTLC it offers. A node MUST NOT offer a HTLC
|
|
* after this deadline */
|
|
if (ld->config.deadline_blocks >= ld->config.min_htlc_expiry)
|
|
fatal("Deadline %u can't be more than minimum expiry %u",
|
|
ld->config.deadline_blocks,
|
|
ld->config.min_htlc_expiry);
|
|
}
|
|
|
|
static void setup_default_config(struct lightningd *ld)
|
|
{
|
|
if (get_chainparams(ld)->testnet)
|
|
ld->config = testnet_config;
|
|
else
|
|
ld->config = mainnet_config;
|
|
}
|
|
|
|
|
|
/* FIXME: make this nicer! */
|
|
static void config_log_stderr_exit(const char *fmt, ...)
|
|
{
|
|
char *msg;
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
|
|
/* This is the format we expect: mangle it to remove '--'. */
|
|
if (streq(fmt, "%s: %.*s: %s")) {
|
|
const char *argv0 = va_arg(ap, const char *);
|
|
unsigned int len = va_arg(ap, unsigned int);
|
|
const char *arg = va_arg(ap, const char *);
|
|
const char *problem = va_arg(ap, const char *);
|
|
|
|
msg = tal_fmt(NULL, "%s line %s: %.*s: %s",
|
|
argv0, arg+strlen(arg)+1, len-2, arg+2, problem);
|
|
} else {
|
|
msg = tal_vfmt(NULL, fmt, ap);
|
|
}
|
|
va_end(ap);
|
|
|
|
fatal("%s", msg);
|
|
}
|
|
|
|
/* We turn the config file into cmdline arguments. */
|
|
static void opt_parse_from_config(struct lightningd *ld)
|
|
{
|
|
char *contents, **lines;
|
|
char **argv;
|
|
int i, argc;
|
|
|
|
contents = grab_file(ld, "config");
|
|
/* Doesn't have to exist. */
|
|
if (!contents) {
|
|
if (errno != ENOENT)
|
|
fatal("Opening and reading config: %s",
|
|
strerror(errno));
|
|
/* Now we can set up defaults, since no config file. */
|
|
setup_default_config(ld);
|
|
return;
|
|
}
|
|
|
|
lines = tal_strsplit(contents, contents, "\r\n", STR_NO_EMPTY);
|
|
|
|
/* We have to keep argv around, since opt will point into it */
|
|
argv = tal_arr(ld, char *, argc = 1);
|
|
argv[0] = "lightning config file";
|
|
|
|
for (i = 0; i < tal_count(lines) - 1; i++) {
|
|
if (strstarts(lines[i], "#"))
|
|
continue;
|
|
/* Only valid forms are "foo" and "foo=bar" */
|
|
tal_resize(&argv, argc+1);
|
|
/* Stash line number after nul. */
|
|
argv[argc++] = tal_fmt(argv, "--%s%c%u", lines[i], 0, i+1);
|
|
}
|
|
tal_resize(&argv, argc+1);
|
|
argv[argc] = NULL;
|
|
|
|
opt_early_parse(argc, argv, config_log_stderr_exit);
|
|
/* Now we can set up defaults, depending on whether testnet or not */
|
|
setup_default_config(ld);
|
|
|
|
opt_parse(&argc, argv, config_log_stderr_exit);
|
|
|
|
if (ld->portnum && tal_count(ld->wireaddrs) == 0)
|
|
guess_addresses(ld);
|
|
|
|
tal_free(contents);
|
|
}
|
|
|
|
void register_opts(struct lightningd *ld)
|
|
{
|
|
opt_set_alloc(opt_allocfn, tal_reallocfn, tal_freefn);
|
|
|
|
opt_register_early_noarg("--help|-h", opt_usage_and_exit,
|
|
"\n"
|
|
"A bitcoin lightning daemon.",
|
|
"Print this message.");
|
|
opt_register_arg("--port", opt_set_u16, opt_show_u16, &ld->portnum,
|
|
"Port to bind to (0 means don't listen)");
|
|
opt_register_arg("--bitcoin-datadir", opt_set_charp, NULL,
|
|
&ld->topology->bitcoind->datadir,
|
|
"-datadir arg for bitcoin-cli");
|
|
opt_register_arg("--rgb", opt_set_rgb, NULL, ld,
|
|
"RRGGBB hex color for node");
|
|
opt_register_arg("--alias", opt_set_alias, NULL, ld,
|
|
"Up to 32-byte alias for node");
|
|
opt_register_logging(ld->log);
|
|
opt_register_version();
|
|
|
|
configdir_register_opts(ld, &ld->config_dir, &ld->rpc_filename);
|
|
config_register_opts(ld);
|
|
#if DEVELOPER
|
|
dev_register_opts(ld);
|
|
#endif
|
|
}
|
|
|
|
/* Names stolen from https://github.com/ternus/nsaproductgenerator/blob/master/nsa.js */
|
|
static const char *codename_adjective[]
|
|
= { "LOUD", "RED", "BLUE", "GREEN", "YELLOW", "IRATE", "ANGRY", "PEEVED",
|
|
"HAPPY", "SLIMY", "SLEEPY", "JUNIOR", "SLICKER", "UNITED", "SOMBER",
|
|
"BIZARRE", "ODD", "WEIRD", "WRONG", "LATENT", "CHILLY", "STRANGE", "LOUD",
|
|
"SILENT", "HOPPING", "ORANGE", "VIOLET", "VIOLENT", "LIGHTNING" };
|
|
|
|
static const char *codename_noun[]
|
|
= { "WHISPER", "FELONY", "MOON", "SUCKER", "PENGUIN", "WAFFLE", "MAESTRO",
|
|
"NIGHT", "TRINITY", "DEITY", "MONKEY", "ARK", "SQUIRREL", "IRON", "BOUNCE",
|
|
"FARM", "CHEF", "TROUGH", "NET", "TRAWL", "GLEE", "WATER", "SPORK", "PLOW",
|
|
"FEED", "SOUFFLE", "ROUTE", "BAGEL", "MONTANA", "ANALYST", "AUTO", "WATCH",
|
|
"PHOTO", "YARD", "SOURCE", "MONKEY", "SEAGULL", "TOLL", "SPAWN", "GOPHER",
|
|
"CHIPMUNK", "SET", "CALENDAR", "ARTIST", "CHASER", "SCAN", "TOTE", "BEAM",
|
|
"ENTOURAGE", "GENESIS", "WALK", "SPATULA", "RAGE", "FIRE", "MASTER" };
|
|
|
|
void setup_color_and_alias(struct lightningd *ld)
|
|
{
|
|
u8 der[PUBKEY_DER_LEN];
|
|
pubkey_to_der(der, &ld->id);
|
|
|
|
if (!ld->rgb)
|
|
/* You can't get much red by default */
|
|
ld->rgb = tal_dup_arr(ld, u8, der, 3, 0);
|
|
|
|
if (!ld->alias) {
|
|
u64 adjective, noun;
|
|
|
|
memcpy(&adjective, der+3, sizeof(adjective));
|
|
memcpy(&noun, der+3+sizeof(adjective), sizeof(noun));
|
|
noun %= ARRAY_SIZE(codename_noun);
|
|
adjective %= ARRAY_SIZE(codename_adjective);
|
|
ld->alias = tal_arrz(ld, char, 33);
|
|
assert(strlen(codename_adjective[adjective])
|
|
+ strlen(codename_noun[noun]) < 33);
|
|
strcpy(ld->alias, codename_adjective[adjective]);
|
|
strcat(ld->alias, codename_noun[noun]);
|
|
}
|
|
}
|
|
|
|
bool handle_opts(struct lightningd *ld, int argc, char *argv[])
|
|
{
|
|
bool newdir = false;
|
|
|
|
/* Get any configdir/testnet options first. */
|
|
opt_early_parse(argc, argv, opt_log_stderr_exit);
|
|
|
|
/* Move to config dir, to save ourselves the hassle of path manip. */
|
|
if (chdir(ld->config_dir) != 0) {
|
|
log_unusual(ld->log, "Creating lightningd dir %s"
|
|
" (because chdir gave %s)",
|
|
ld->config_dir, strerror(errno));
|
|
if (mkdir(ld->config_dir, 0700) != 0)
|
|
fatal("Could not make directory %s: %s",
|
|
ld->config_dir, strerror(errno));
|
|
if (chdir(ld->config_dir) != 0)
|
|
fatal("Could not change directory %s: %s",
|
|
ld->config_dir, strerror(errno));
|
|
newdir = true;
|
|
}
|
|
|
|
/* Now look for config file */
|
|
opt_parse_from_config(ld);
|
|
|
|
opt_parse(&argc, argv, opt_log_stderr_exit);
|
|
if (argc != 1)
|
|
errx(1, "no arguments accepted");
|
|
|
|
check_config(ld);
|
|
|
|
#if DEVELOPER
|
|
if (ld->dev_hsm_seed) {
|
|
int fd;
|
|
unlink("hsm_secret");
|
|
fd = open("hsm_secret", O_CREAT|O_WRONLY, 0400);
|
|
if (fd < 0 ||
|
|
!write_all(fd, ld->dev_hsm_seed, tal_len(ld->dev_hsm_seed))
|
|
|| fsync(fd) != 0)
|
|
fatal("dev-hsm-seed: Could not write file: %s",
|
|
strerror(errno));
|
|
close(fd);
|
|
}
|
|
#endif
|
|
|
|
return newdir;
|
|
}
|
|
|