2016-01-21 21:11:49 +01:00
|
|
|
#include "bitcoind.h"
|
2016-04-24 12:07:13 +02:00
|
|
|
#include "chaintopology.h"
|
2016-08-18 06:55:13 +02:00
|
|
|
#include "db.h"
|
2017-01-04 03:48:47 +01:00
|
|
|
#include "invoice.h"
|
2016-09-07 23:47:27 +02:00
|
|
|
#include "irc_announce.h"
|
2016-01-21 21:11:48 +01:00
|
|
|
#include "jsonrpc.h"
|
2016-01-21 21:11:47 +01:00
|
|
|
#include "lightningd.h"
|
|
|
|
#include "log.h"
|
2017-01-04 03:52:29 +01:00
|
|
|
#include "options.h"
|
2016-12-21 17:15:35 +01:00
|
|
|
#include "p2p_announce.h"
|
2016-01-21 21:11:48 +01:00
|
|
|
#include "peer.h"
|
2016-06-28 23:19:21 +02:00
|
|
|
#include "routing.h"
|
2016-01-21 21:11:48 +01:00
|
|
|
#include "secrets.h"
|
2016-01-21 21:11:48 +01:00
|
|
|
#include "timeout.h"
|
2016-12-02 08:41:06 +01:00
|
|
|
#include "utils.h"
|
2017-07-10 10:59:15 +02:00
|
|
|
#include <bitcoin/chainparams.h>
|
2016-01-21 21:11:48 +01:00
|
|
|
#include <ccan/container_of/container_of.h>
|
2016-01-21 21:11:47 +01:00
|
|
|
#include <ccan/err/err.h>
|
2016-01-21 21:11:48 +01:00
|
|
|
#include <ccan/io/io.h>
|
2016-01-21 21:11:47 +01:00
|
|
|
#include <ccan/opt/opt.h>
|
2016-10-07 05:30:18 +02:00
|
|
|
#include <ccan/tal/grab_file/grab_file.h>
|
2016-01-21 21:11:47 +01:00
|
|
|
#include <ccan/tal/str/str.h>
|
2016-01-21 21:11:47 +01:00
|
|
|
#include <ccan/tal/tal.h>
|
2016-05-09 22:59:12 +02:00
|
|
|
#include <ccan/time/time.h>
|
2016-01-21 21:11:48 +01:00
|
|
|
#include <ccan/timer/timer.h>
|
2016-01-21 21:11:47 +01:00
|
|
|
#include <errno.h>
|
2016-01-21 21:11:49 +01:00
|
|
|
#include <inttypes.h>
|
2016-11-10 13:59:35 +01:00
|
|
|
#include <signal.h>
|
2016-01-21 21:11:47 +01:00
|
|
|
#include <unistd.h>
|
2016-01-21 21:11:47 +01:00
|
|
|
#include <version.h>
|
2016-01-21 21:11:47 +01:00
|
|
|
|
|
|
|
static struct lightningd_state *lightningd_state(void)
|
|
|
|
{
|
2016-01-21 21:11:49 +01:00
|
|
|
struct lightningd_state *dstate = tal(NULL, struct lightningd_state);
|
|
|
|
|
2017-01-10 05:48:26 +01:00
|
|
|
dstate->log_book = new_log_book(dstate, 20*1024*1024, LOG_INFORM);
|
|
|
|
dstate->base_log = new_log(dstate, dstate->log_book,
|
2016-01-21 21:11:49 +01:00
|
|
|
"lightningd(%u):", (int)getpid());
|
|
|
|
|
|
|
|
list_head_init(&dstate->peers);
|
2016-08-31 08:34:59 +02:00
|
|
|
list_head_init(&dstate->pay_commands);
|
2016-09-12 20:07:07 +02:00
|
|
|
dstate->portnum = 0;
|
2016-10-07 05:30:18 +02:00
|
|
|
dstate->testnet = true;
|
2016-11-09 09:26:15 +01:00
|
|
|
timers_init(&dstate->timers, time_mono());
|
2016-04-12 05:37:03 +02:00
|
|
|
list_head_init(&dstate->wallet);
|
2016-08-18 06:55:14 +02:00
|
|
|
list_head_init(&dstate->addresses);
|
2016-06-30 01:38:10 +02:00
|
|
|
dstate->dev_never_routefail = false;
|
2017-01-19 23:46:07 +01:00
|
|
|
dstate->rstate = new_routing_state(dstate, dstate->base_log);
|
2016-08-18 06:55:14 +02:00
|
|
|
dstate->reexec = NULL;
|
2016-10-21 03:34:41 +02:00
|
|
|
dstate->external_ip = NULL;
|
|
|
|
dstate->announce = NULL;
|
2017-01-04 03:48:47 +01:00
|
|
|
dstate->invoices = invoices_init(dstate);
|
2016-01-21 21:11:49 +01:00
|
|
|
return dstate;
|
2016-01-21 21:11:47 +01:00
|
|
|
}
|
|
|
|
|
2017-03-02 13:21:49 +01:00
|
|
|
static void json_lightningd_dev_broadcast(struct command *cmd,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *params)
|
|
|
|
{
|
|
|
|
json_dev_broadcast(cmd, cmd->dstate->topology, buffer, params);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct json_command dev_broadcast_command = {
|
|
|
|
"dev-broadcast",
|
|
|
|
json_lightningd_dev_broadcast,
|
|
|
|
"Pretend we broadcast txs, but don't send to bitcoind",
|
|
|
|
"Returns an empty result on success (waits for flush if enabled)"
|
|
|
|
};
|
|
|
|
AUTODATA(json_command, &dev_broadcast_command);
|
|
|
|
|
2016-01-21 21:11:47 +01:00
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
2016-01-21 21:11:49 +01:00
|
|
|
struct lightningd_state *dstate = lightningd_state();
|
2016-01-21 21:11:47 +01:00
|
|
|
|
2016-01-21 21:11:47 +01:00
|
|
|
err_set_progname(argv[0]);
|
|
|
|
|
2016-03-15 07:39:42 +01:00
|
|
|
if (!streq(protobuf_c_version(), PROTOBUF_C_VERSION))
|
|
|
|
errx(1, "Compiled against protobuf %s, but have %s",
|
|
|
|
PROTOBUF_C_VERSION, protobuf_c_version());
|
2016-11-11 00:02:04 +01:00
|
|
|
|
2016-12-02 08:43:27 +01:00
|
|
|
secp256k1_ctx = secp256k1_context_create(SECP256K1_CONTEXT_VERIFY
|
|
|
|
| SECP256K1_CONTEXT_SIGN);
|
2016-12-05 04:33:06 +01:00
|
|
|
|
2017-03-02 13:21:49 +01:00
|
|
|
dstate->topology = new_topology(dstate, dstate->base_log);
|
2017-03-02 13:21:49 +01:00
|
|
|
dstate->bitcoind = new_bitcoind(dstate, dstate->base_log);
|
2017-07-10 10:59:15 +02:00
|
|
|
dstate->bitcoind->chainparams = chainparams_for_network("regtest");
|
2017-03-02 13:21:49 +01:00
|
|
|
|
2017-01-04 03:52:29 +01:00
|
|
|
/* Handle options and config; move to .lightningd */
|
2017-02-24 06:52:56 +01:00
|
|
|
register_opts(dstate);
|
2017-01-04 03:52:29 +01:00
|
|
|
handle_opts(dstate, argc, argv);
|
2016-01-21 21:11:48 +01:00
|
|
|
|
|
|
|
/* Activate crash log now we're in the right place. */
|
2016-01-21 21:11:49 +01:00
|
|
|
crashlog_activate(dstate->base_log);
|
2016-01-21 21:11:48 +01:00
|
|
|
|
2016-11-10 13:59:35 +01:00
|
|
|
/* Ignore SIGPIPE: we look at our write return values*/
|
|
|
|
signal(SIGPIPE, SIG_IGN);
|
|
|
|
|
2016-01-21 21:11:48 +01:00
|
|
|
/* Set up node ID and private key. */
|
2016-01-21 21:11:49 +01:00
|
|
|
secrets_init(dstate);
|
2017-01-19 23:46:07 +01:00
|
|
|
new_node(dstate->rstate, &dstate->id);
|
2016-08-18 06:55:13 +02:00
|
|
|
|
|
|
|
/* Read or create database. */
|
|
|
|
db_init(dstate);
|
|
|
|
|
2016-04-24 12:07:13 +02:00
|
|
|
/* Initialize block topology. */
|
2017-03-02 13:21:49 +01:00
|
|
|
setup_topology(dstate->topology, dstate->bitcoind, &dstate->timers,
|
|
|
|
dstate->config.poll_time,
|
|
|
|
get_peer_min_block(dstate));
|
2016-04-24 12:07:13 +02:00
|
|
|
|
2016-08-09 05:41:24 +02:00
|
|
|
/* Create RPC socket (if any) */
|
|
|
|
setup_jsonrpc(dstate, dstate->rpc_filename);
|
|
|
|
|
2017-01-04 04:39:15 +01:00
|
|
|
/* Set up connections from peers (if dstate->portnum is set) */
|
|
|
|
setup_listeners(dstate);
|
2016-08-09 05:41:24 +02:00
|
|
|
|
2016-09-07 23:47:27 +02:00
|
|
|
/* set up IRC peer discovery */
|
|
|
|
if (dstate->config.use_irc)
|
|
|
|
setup_irc_connection(dstate);
|
|
|
|
|
2016-12-21 17:15:35 +01:00
|
|
|
/* set up P2P gossip protocol */
|
|
|
|
setup_p2p_announce(dstate);
|
|
|
|
|
2016-01-21 21:11:49 +01:00
|
|
|
log_info(dstate->base_log, "Hello world!");
|
2016-01-21 21:11:48 +01:00
|
|
|
|
2016-08-18 06:55:08 +02:00
|
|
|
/* If we loaded peers from database, reconnect now. */
|
|
|
|
reconnect_peers(dstate);
|
2016-11-09 09:22:15 +01:00
|
|
|
|
|
|
|
/* And send out anchors again if we're waiting. */
|
|
|
|
rebroadcast_anchors(dstate);
|
2016-11-11 00:02:04 +01:00
|
|
|
|
2016-01-21 21:15:28 +01:00
|
|
|
for (;;) {
|
|
|
|
struct timer *expired;
|
|
|
|
void *v = io_loop(&dstate->timers, &expired);
|
2016-01-21 21:11:48 +01:00
|
|
|
|
2016-01-21 21:15:28 +01:00
|
|
|
/* We use io_break(dstate) to shut down. */
|
|
|
|
if (v == dstate)
|
|
|
|
break;
|
|
|
|
|
2016-05-09 22:56:09 +02:00
|
|
|
if (expired)
|
|
|
|
timer_expired(dstate, expired);
|
2016-06-30 01:38:11 +02:00
|
|
|
else
|
|
|
|
cleanup_peers(dstate);
|
2016-01-21 21:11:48 +01:00
|
|
|
}
|
|
|
|
|
2016-08-18 06:55:14 +02:00
|
|
|
if (dstate->reexec) {
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
log_unusual(dstate->base_log, "Restart at user request");
|
|
|
|
fflush(stdout);
|
|
|
|
fflush(stderr);
|
|
|
|
|
|
|
|
/* Manually close all fds (or near enough!) */
|
|
|
|
for (fd = 3; fd < 1024; fd++)
|
|
|
|
close(fd);
|
|
|
|
|
|
|
|
if (dstate->dev_never_routefail) {
|
|
|
|
size_t n = tal_count(dstate->reexec);
|
|
|
|
tal_resizez(&dstate->reexec, n+1);
|
|
|
|
dstate->reexec[n-1] = "--dev-no-routefail";
|
|
|
|
}
|
|
|
|
execvp(dstate->reexec[0], dstate->reexec);
|
|
|
|
fatal("Exec '%s' failed: %s",
|
|
|
|
dstate->reexec[0], strerror(errno));
|
|
|
|
}
|
|
|
|
|
2016-01-21 21:11:49 +01:00
|
|
|
tal_free(dstate);
|
2016-01-21 21:11:47 +01:00
|
|
|
opt_free_table();
|
2016-01-21 21:11:47 +01:00
|
|
|
return 0;
|
|
|
|
}
|