2020-01-10 12:12:20 +01:00
|
|
|
/* Code for talking to bitcoind. We use a plugin as the Bitcoin backend.
|
|
|
|
* The default one shipped with C-lightning is a plugin which talks to bitcoind
|
|
|
|
* by using bitcoin-cli, but the interface we use to gather Bitcoin data is
|
|
|
|
* standardized and you can use another plugin as the Bitcoin backend, or
|
|
|
|
* even make your own! */
|
2021-12-04 12:23:56 +01:00
|
|
|
#include "config.h"
|
|
|
|
#include <bitcoin/base58.h>
|
|
|
|
#include <bitcoin/block.h>
|
|
|
|
#include <bitcoin/feerate.h>
|
|
|
|
#include <bitcoin/script.h>
|
|
|
|
#include <bitcoin/shadouble.h>
|
2020-01-07 11:59:18 +01:00
|
|
|
#include <ccan/array_size/array_size.h>
|
2016-01-21 21:11:49 +01:00
|
|
|
#include <ccan/io/io.h>
|
|
|
|
#include <ccan/tal/str/str.h>
|
2019-01-15 04:54:27 +01:00
|
|
|
#include <common/json_helpers.h>
|
2017-12-15 11:22:57 +01:00
|
|
|
#include <common/memleak.h>
|
2021-12-04 12:23:56 +01:00
|
|
|
#include <lightningd/bitcoind.h>
|
2018-01-04 12:40:58 +01:00
|
|
|
#include <lightningd/chaintopology.h>
|
2021-09-16 07:00:42 +02:00
|
|
|
#include <lightningd/io_loop_with_timers.h>
|
2021-12-04 12:23:56 +01:00
|
|
|
#include <lightningd/lightningd.h>
|
|
|
|
#include <lightningd/log.h>
|
2020-01-07 11:59:18 +01:00
|
|
|
#include <lightningd/plugin.h>
|
2016-01-21 21:11:49 +01:00
|
|
|
|
2020-03-04 11:44:34 +01:00
|
|
|
/* The names of the requests we can make to our Bitcoin backend. */
|
2020-01-07 11:59:18 +01:00
|
|
|
static const char *methods[] = {"getchaininfo", "getrawblockbyheight",
|
|
|
|
"sendrawtransaction", "getutxout",
|
2020-03-05 10:58:34 +01:00
|
|
|
"estimatefees"};
|
2020-01-07 11:59:18 +01:00
|
|
|
|
2020-04-28 11:17:40 +02:00
|
|
|
static void bitcoin_destructor(struct plugin *p)
|
|
|
|
{
|
|
|
|
if (p->plugins->ld->state == LD_STATE_SHUTDOWN)
|
|
|
|
return;
|
|
|
|
fatal("The Bitcoin backend died.");
|
|
|
|
}
|
|
|
|
|
2020-01-07 18:08:03 +01:00
|
|
|
static void plugin_config_cb(const char *buffer,
|
|
|
|
const jsmntok_t *toks,
|
|
|
|
const jsmntok_t *idtok,
|
|
|
|
struct plugin *plugin)
|
|
|
|
{
|
2020-05-04 19:19:05 +02:00
|
|
|
plugin->plugin_state = INIT_COMPLETE;
|
2020-01-07 18:08:03 +01:00
|
|
|
io_break(plugin);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void config_plugin(struct plugin *plugin)
|
|
|
|
{
|
|
|
|
struct jsonrpc_request *req;
|
|
|
|
|
|
|
|
req = jsonrpc_request_start(plugin, "init", plugin->log,
|
2020-10-12 07:33:50 +02:00
|
|
|
NULL, plugin_config_cb, plugin);
|
2020-01-07 18:08:03 +01:00
|
|
|
plugin_populate_init_request(plugin, req);
|
|
|
|
jsonrpc_request_end(req);
|
|
|
|
plugin_request_send(plugin, req);
|
2020-03-04 11:44:34 +01:00
|
|
|
|
2020-04-28 11:17:40 +02:00
|
|
|
tal_add_destructor(plugin, bitcoin_destructor);
|
2020-03-04 11:44:34 +01:00
|
|
|
|
2020-01-07 18:08:03 +01:00
|
|
|
io_loop_with_timers(plugin->plugins->ld);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void wait_plugin(struct bitcoind *bitcoind, const char *method,
|
|
|
|
struct plugin *p)
|
|
|
|
{
|
|
|
|
/* We need our Bitcoin backend to be initialized, but the plugins have
|
|
|
|
* not yet been started at this point.
|
|
|
|
* So send `init` to each plugin which registered for a Bitcoin method
|
|
|
|
* and wait for its response, which we take as an ACK that it is
|
|
|
|
* operational (i.e. bcli will wait for `bitcoind` to be warmed up
|
|
|
|
* before responding to `init`).
|
|
|
|
* Note that lightningd/plugin will not send `init` to an already
|
|
|
|
* configured plugin. */
|
2020-05-04 19:19:05 +02:00
|
|
|
if (p->plugin_state == NEEDS_INIT)
|
2020-01-07 18:08:03 +01:00
|
|
|
config_plugin(p);
|
2020-05-04 19:19:05 +02:00
|
|
|
|
2020-01-07 18:08:03 +01:00
|
|
|
strmap_add(&bitcoind->pluginsmap, method, p);
|
|
|
|
}
|
|
|
|
|
2020-01-07 11:59:18 +01:00
|
|
|
void bitcoind_check_commands(struct bitcoind *bitcoind)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
struct plugin *p;
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(methods); i++) {
|
|
|
|
p = find_plugin_for_command(bitcoind->ld, methods[i]);
|
|
|
|
if (p == NULL) {
|
2020-02-06 19:00:14 +01:00
|
|
|
/* For testing .. */
|
|
|
|
log_debug(bitcoind->ld->log, "Missing a Bitcoin plugin"
|
|
|
|
" command");
|
2020-01-07 11:59:18 +01:00
|
|
|
fatal("Could not access the plugin for %s, is a "
|
|
|
|
"Bitcoin plugin (by default plugins/bcli) "
|
|
|
|
"registered ?", methods[i]);
|
|
|
|
}
|
2020-01-07 18:08:03 +01:00
|
|
|
wait_plugin(bitcoind, methods[i], p);
|
2020-01-07 11:59:18 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-09 18:41:07 +01:00
|
|
|
/* Our Bitcoin backend plugin gave us a bad response. We can't recover. */
|
|
|
|
static void bitcoin_plugin_error(struct bitcoind *bitcoind, const char *buf,
|
|
|
|
const jsmntok_t *toks, const char *method,
|
2020-03-05 10:58:34 +01:00
|
|
|
const char *fmt, ...)
|
2016-01-21 21:11:49 +01:00
|
|
|
{
|
2020-03-05 10:58:34 +01:00
|
|
|
va_list ap;
|
|
|
|
char *reason;
|
|
|
|
struct plugin *p;
|
|
|
|
|
|
|
|
va_start(ap, fmt);
|
|
|
|
reason = tal_vfmt(NULL, fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
p = strmap_get(&bitcoind->pluginsmap, method);
|
2020-01-09 18:41:07 +01:00
|
|
|
fatal("%s error: bad response to %s (%s), response was %.*s",
|
|
|
|
p->cmd, method, reason,
|
|
|
|
toks->end - toks->start, buf + toks->start);
|
2018-01-31 04:11:36 +01:00
|
|
|
}
|
2016-01-21 21:11:49 +01:00
|
|
|
|
2020-03-10 00:03:34 +01:00
|
|
|
/* Send a request to the Bitcoin plugin which registered that method,
|
|
|
|
* if it's still alive. */
|
|
|
|
static void bitcoin_plugin_send(struct bitcoind *bitcoind,
|
|
|
|
struct jsonrpc_request *req)
|
|
|
|
{
|
|
|
|
struct plugin *plugin = strmap_get(&bitcoind->pluginsmap, req->method);
|
|
|
|
if (!plugin)
|
|
|
|
fatal("Bitcoin backend plugin for %s died.", req->method);
|
|
|
|
|
|
|
|
plugin_request_send(plugin, req);
|
|
|
|
}
|
|
|
|
|
2020-03-05 10:58:34 +01:00
|
|
|
/* `estimatefees`
|
2020-01-09 18:41:07 +01:00
|
|
|
*
|
|
|
|
* Gather feerate from our Bitcoin backend. Will set the feerate to `null`
|
|
|
|
* if estimation failed.
|
|
|
|
*
|
2020-03-05 10:58:34 +01:00
|
|
|
* - `opening` is used for funding and also misc transactions
|
|
|
|
* - `mutual_close` is used for the mutual close transaction
|
|
|
|
* - `unilateral_close` is used for unilateral close (commitment transactions)
|
|
|
|
* - `delayed_to_us` is used for resolving our output from our unilateral close
|
|
|
|
* - `htlc_resolution` is used for resolving onchain HTLCs
|
|
|
|
* - `penalty` is used for resolving revoked transactions
|
|
|
|
* - `min` is the minimum acceptable feerate
|
2020-03-11 13:37:24 +01:00
|
|
|
* - `max` is the maximum acceptable feerate
|
2020-03-05 10:58:34 +01:00
|
|
|
*
|
2020-01-09 18:41:07 +01:00
|
|
|
* Plugin response:
|
|
|
|
* {
|
2020-03-05 10:58:34 +01:00
|
|
|
* "opening": <sat per kVB>,
|
|
|
|
* "mutual_close": <sat per kVB>,
|
|
|
|
* "unilateral_close": <sat per kVB>,
|
|
|
|
* "delayed_to_us": <sat per kVB>,
|
|
|
|
* "htlc_resolution": <sat per kVB>,
|
|
|
|
* "penalty": <sat per kVB>,
|
|
|
|
* "min_acceptable": <sat per kVB>,
|
|
|
|
* "max_acceptable": <sat per kVB>,
|
2020-01-09 18:41:07 +01:00
|
|
|
* }
|
|
|
|
*/
|
2016-01-21 21:11:49 +01:00
|
|
|
|
2020-01-09 18:41:07 +01:00
|
|
|
struct estimatefee_call {
|
2017-03-02 13:21:49 +01:00
|
|
|
struct bitcoind *bitcoind;
|
2017-11-21 04:33:22 +01:00
|
|
|
void (*cb)(struct bitcoind *bitcoind, const u32 satoshi_per_kw[],
|
2017-11-21 04:30:29 +01:00
|
|
|
void *);
|
|
|
|
void *arg;
|
|
|
|
};
|
|
|
|
|
2020-03-05 10:58:34 +01:00
|
|
|
static void estimatefees_callback(const char *buf, const jsmntok_t *toks,
|
|
|
|
const jsmntok_t *idtok,
|
|
|
|
struct estimatefee_call *call)
|
2016-07-19 05:22:18 +02:00
|
|
|
{
|
2020-01-09 18:41:07 +01:00
|
|
|
const jsmntok_t *resulttok, *feeratetok;
|
2020-03-05 10:58:34 +01:00
|
|
|
u32 *feerates = tal_arr(call, u32, NUM_FEERATES);
|
2020-01-09 18:41:07 +01:00
|
|
|
|
|
|
|
resulttok = json_get_member(buf, toks, "result");
|
|
|
|
if (!resulttok)
|
|
|
|
bitcoin_plugin_error(call->bitcoind, buf, toks,
|
2020-03-05 10:58:34 +01:00
|
|
|
"estimatefees",
|
2020-01-09 18:41:07 +01:00
|
|
|
"bad 'result' field");
|
|
|
|
|
2020-09-08 14:42:30 +02:00
|
|
|
for (int f = 0; f < NUM_FEERATES; f++) {
|
2020-03-05 10:58:34 +01:00
|
|
|
feeratetok = json_get_member(buf, resulttok, feerate_name(f));
|
|
|
|
if (!feeratetok)
|
|
|
|
bitcoin_plugin_error(call->bitcoind, buf, toks,
|
|
|
|
"estimatefees",
|
|
|
|
"missing '%s' field", feerate_name(f));
|
2021-07-08 04:47:03 +02:00
|
|
|
/* We still use the bcli plugin for min and max, even with
|
|
|
|
* force_feerates */
|
|
|
|
if (f < tal_count(call->bitcoind->ld->force_feerates)) {
|
|
|
|
feerates[f] = call->bitcoind->ld->force_feerates[f];
|
|
|
|
continue;
|
|
|
|
}
|
2016-07-19 05:22:18 +02:00
|
|
|
|
2020-03-05 10:58:34 +01:00
|
|
|
/* FIXME: We could trawl recent blocks for median fee... */
|
|
|
|
if (!json_to_u32(buf, feeratetok, &feerates[f])) {
|
2021-07-08 04:58:38 +02:00
|
|
|
if (chainparams->testnet)
|
|
|
|
log_debug(call->bitcoind->log,
|
|
|
|
"Unable to estimate %s fees",
|
|
|
|
feerate_name(f));
|
|
|
|
else
|
|
|
|
log_unusual(call->bitcoind->log,
|
|
|
|
"Unable to estimate %s fees",
|
|
|
|
feerate_name(f));
|
2018-10-22 21:36:08 +02:00
|
|
|
|
|
|
|
#if DEVELOPER
|
2020-03-05 10:58:34 +01:00
|
|
|
/* This is needed to test for failed feerate estimates
|
|
|
|
* in DEVELOPER mode */
|
|
|
|
feerates[f] = 0;
|
2018-10-22 21:36:08 +02:00
|
|
|
#else
|
2020-03-05 10:58:34 +01:00
|
|
|
/* If we are in testnet mode we want to allow payments
|
|
|
|
* with the minimal fee even if the estimate didn't
|
|
|
|
* work out. This is less disruptive than erring out
|
|
|
|
* all the time. */
|
|
|
|
if (chainparams->testnet)
|
|
|
|
feerates[f] = FEERATE_FLOOR;
|
|
|
|
else
|
|
|
|
feerates[f] = 0;
|
2018-10-22 21:36:08 +02:00
|
|
|
#endif
|
2020-03-05 10:58:34 +01:00
|
|
|
} else
|
|
|
|
/* Rate in satoshi per kw. */
|
|
|
|
feerates[f] = feerate_from_style(feerates[f],
|
|
|
|
FEERATE_PER_KBYTE);
|
2017-11-21 04:30:29 +01:00
|
|
|
}
|
2017-11-21 04:26:39 +01:00
|
|
|
|
2020-03-05 10:58:34 +01:00
|
|
|
call->cb(call->bitcoind, feerates, call->arg);
|
|
|
|
tal_free(call);
|
2016-07-19 05:22:18 +02:00
|
|
|
}
|
|
|
|
|
2017-11-21 04:30:29 +01:00
|
|
|
void bitcoind_estimate_fees_(struct bitcoind *bitcoind,
|
|
|
|
size_t num_estimates,
|
|
|
|
void (*cb)(struct bitcoind *bitcoind,
|
2017-11-21 04:33:22 +01:00
|
|
|
const u32 satoshi_per_kw[], void *),
|
2017-11-21 04:30:29 +01:00
|
|
|
void *arg)
|
2016-04-12 05:37:03 +02:00
|
|
|
{
|
2020-03-05 10:58:34 +01:00
|
|
|
struct jsonrpc_request *req;
|
|
|
|
struct estimatefee_call *call = tal(bitcoind, struct estimatefee_call);
|
2017-11-21 04:30:29 +01:00
|
|
|
|
2020-03-05 10:58:34 +01:00
|
|
|
call->bitcoind = bitcoind;
|
|
|
|
call->cb = cb;
|
|
|
|
call->arg = arg;
|
2017-11-21 04:30:29 +01:00
|
|
|
|
2020-03-05 10:58:34 +01:00
|
|
|
req = jsonrpc_request_start(bitcoind, "estimatefees", bitcoind->log,
|
2020-10-12 07:33:50 +02:00
|
|
|
NULL, estimatefees_callback, call);
|
2020-03-05 10:58:34 +01:00
|
|
|
jsonrpc_request_end(req);
|
|
|
|
plugin_request_send(strmap_get(&bitcoind->pluginsmap,
|
|
|
|
"estimatefees"), req);
|
2016-04-12 05:37:03 +02:00
|
|
|
}
|
2020-01-08 17:53:31 +01:00
|
|
|
|
2020-01-09 12:25:45 +01:00
|
|
|
/* `sendrawtransaction`
|
|
|
|
*
|
|
|
|
* Send a transaction to the Bitcoin backend plugin. If the broadcast was
|
|
|
|
* not successful on its end, the plugin will populate the `errmsg` with
|
|
|
|
* the reason.
|
|
|
|
*
|
|
|
|
* Plugin response:
|
|
|
|
* {
|
|
|
|
* "success": <true|false>,
|
|
|
|
* "errmsg": "<not empty if !success>"
|
|
|
|
* }
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct sendrawtx_call {
|
|
|
|
struct bitcoind *bitcoind;
|
|
|
|
void (*cb)(struct bitcoind *bitcoind,
|
|
|
|
bool success,
|
|
|
|
const char *err_msg,
|
|
|
|
void *);
|
|
|
|
void *cb_arg;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void sendrawtx_callback(const char *buf, const jsmntok_t *toks,
|
|
|
|
const jsmntok_t *idtok,
|
|
|
|
struct sendrawtx_call *call)
|
|
|
|
{
|
2021-01-07 06:34:43 +01:00
|
|
|
const char *err;
|
2021-01-06 07:03:04 +01:00
|
|
|
const char *errmsg = NULL;
|
2020-01-09 12:25:45 +01:00
|
|
|
bool success = false;
|
|
|
|
|
2021-01-07 06:34:43 +01:00
|
|
|
err = json_scan(tmpctx, buf, toks, "{result:{success:%}}",
|
|
|
|
JSON_SCAN(json_to_bool, &success));
|
|
|
|
if (err) {
|
2020-01-09 12:25:45 +01:00
|
|
|
bitcoin_plugin_error(call->bitcoind, buf, toks,
|
|
|
|
"sendrawtransaction",
|
2021-01-07 06:34:43 +01:00
|
|
|
"bad 'result' field: %s", err);
|
2021-01-06 07:03:04 +01:00
|
|
|
} else if (!success) {
|
2021-01-07 06:34:43 +01:00
|
|
|
err = json_scan(tmpctx, buf, toks, "{result:{errmsg:%}}",
|
|
|
|
JSON_SCAN_TAL(tmpctx, json_strdup, &errmsg));
|
|
|
|
if (err)
|
2021-01-06 07:03:04 +01:00
|
|
|
bitcoin_plugin_error(call->bitcoind, buf, toks,
|
|
|
|
"sendrawtransaction",
|
2021-01-07 06:34:43 +01:00
|
|
|
"bad 'errmsg' field: %s",
|
|
|
|
err);
|
2021-01-06 07:03:04 +01:00
|
|
|
}
|
2020-01-09 12:25:45 +01:00
|
|
|
|
|
|
|
db_begin_transaction(call->bitcoind->ld->wallet->db);
|
2021-01-06 07:03:04 +01:00
|
|
|
call->cb(call->bitcoind, success, errmsg, call->cb_arg);
|
2020-01-09 12:25:45 +01:00
|
|
|
db_commit_transaction(call->bitcoind->ld->wallet->db);
|
|
|
|
|
|
|
|
tal_free(call);
|
|
|
|
}
|
|
|
|
|
2020-09-08 05:22:41 +02:00
|
|
|
void bitcoind_sendrawtx_ahf_(struct bitcoind *bitcoind,
|
|
|
|
const char *hextx,
|
|
|
|
bool allowhighfees,
|
|
|
|
void (*cb)(struct bitcoind *bitcoind,
|
|
|
|
bool success, const char *msg, void *),
|
|
|
|
void *cb_arg)
|
2020-01-09 12:25:45 +01:00
|
|
|
{
|
|
|
|
struct jsonrpc_request *req;
|
|
|
|
struct sendrawtx_call *call = tal(bitcoind, struct sendrawtx_call);
|
|
|
|
|
|
|
|
call->bitcoind = bitcoind;
|
|
|
|
call->cb = cb;
|
|
|
|
call->cb_arg = cb_arg;
|
|
|
|
log_debug(bitcoind->log, "sendrawtransaction: %s", hextx);
|
|
|
|
|
|
|
|
req = jsonrpc_request_start(bitcoind, "sendrawtransaction",
|
2020-09-08 05:22:41 +02:00
|
|
|
bitcoind->log,
|
2021-01-29 01:01:09 +01:00
|
|
|
NULL, sendrawtx_callback,
|
2020-01-09 12:25:45 +01:00
|
|
|
call);
|
|
|
|
json_add_string(req->stream, "tx", hextx);
|
2020-09-08 05:22:41 +02:00
|
|
|
json_add_bool(req->stream, "allowhighfees", allowhighfees);
|
2020-01-09 12:25:45 +01:00
|
|
|
jsonrpc_request_end(req);
|
2020-03-10 00:03:34 +01:00
|
|
|
bitcoin_plugin_send(bitcoind, req);
|
2020-01-09 12:25:45 +01:00
|
|
|
}
|
|
|
|
|
2020-09-08 05:22:41 +02:00
|
|
|
void bitcoind_sendrawtx_(struct bitcoind *bitcoind,
|
|
|
|
const char *hextx,
|
|
|
|
void (*cb)(struct bitcoind *bitcoind,
|
|
|
|
bool success, const char *msg, void *),
|
|
|
|
void *arg)
|
|
|
|
{
|
|
|
|
return bitcoind_sendrawtx_ahf_(bitcoind, hextx, false, cb, arg);
|
|
|
|
}
|
|
|
|
|
2020-01-08 17:53:31 +01:00
|
|
|
/* `getrawblockbyheight`
|
|
|
|
*
|
|
|
|
* If no block were found at that height, will set each field to `null`.
|
|
|
|
* Plugin response:
|
|
|
|
* {
|
|
|
|
* "blockhash": "<blkid>",
|
|
|
|
* "block": "rawblock"
|
|
|
|
* }
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct getrawblockbyheight_call {
|
|
|
|
struct bitcoind *bitcoind;
|
2017-03-02 13:21:49 +01:00
|
|
|
void (*cb)(struct bitcoind *bitcoind,
|
2020-01-08 17:53:31 +01:00
|
|
|
struct bitcoin_blkid *blkid,
|
|
|
|
struct bitcoin_block *block,
|
|
|
|
void *);
|
|
|
|
void *cb_arg;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
getrawblockbyheight_callback(const char *buf, const jsmntok_t *toks,
|
|
|
|
const jsmntok_t *idtok,
|
|
|
|
struct getrawblockbyheight_call *call)
|
|
|
|
{
|
2021-01-07 06:34:43 +01:00
|
|
|
const char *block_str, *err;
|
2020-01-08 17:53:31 +01:00
|
|
|
struct bitcoin_blkid blkid;
|
|
|
|
struct bitcoin_block *blk;
|
2016-04-24 12:06:13 +02:00
|
|
|
|
2020-01-08 17:53:31 +01:00
|
|
|
/* If block hash is `null`, this means not found! Call the callback
|
|
|
|
* with NULL values. */
|
2021-01-07 06:34:43 +01:00
|
|
|
err = json_scan(tmpctx, buf, toks, "{result:{blockhash:null}}");
|
|
|
|
if (!err) {
|
2020-01-08 17:53:31 +01:00
|
|
|
db_begin_transaction(call->bitcoind->ld->wallet->db);
|
|
|
|
call->cb(call->bitcoind, NULL, NULL, call->cb_arg);
|
|
|
|
db_commit_transaction(call->bitcoind->ld->wallet->db);
|
|
|
|
goto clean;
|
|
|
|
}
|
|
|
|
|
2021-01-07 06:34:43 +01:00
|
|
|
err = json_scan(tmpctx, buf, toks, "{result:{blockhash:%,block:%}}",
|
|
|
|
JSON_SCAN(json_to_sha256, &blkid.shad.sha),
|
|
|
|
JSON_SCAN_TAL(tmpctx, json_strdup, &block_str));
|
|
|
|
if (err)
|
2020-01-08 17:53:31 +01:00
|
|
|
bitcoin_plugin_error(call->bitcoind, buf, toks,
|
|
|
|
"getrawblockbyheight",
|
2021-01-07 06:34:43 +01:00
|
|
|
"bad 'result' field: %s", err);
|
2021-01-06 07:03:04 +01:00
|
|
|
|
2020-01-08 17:53:31 +01:00
|
|
|
blk = bitcoin_block_from_hex(tmpctx, chainparams, block_str,
|
|
|
|
strlen(block_str));
|
|
|
|
if (!blk)
|
|
|
|
bitcoin_plugin_error(call->bitcoind, buf, toks,
|
|
|
|
"getrawblockbyheight",
|
|
|
|
"bad block");
|
2016-04-24 12:06:13 +02:00
|
|
|
|
2020-01-08 17:53:31 +01:00
|
|
|
db_begin_transaction(call->bitcoind->ld->wallet->db);
|
|
|
|
call->cb(call->bitcoind, &blkid, blk, call->cb_arg);
|
|
|
|
db_commit_transaction(call->bitcoind->ld->wallet->db);
|
|
|
|
|
|
|
|
clean:
|
|
|
|
tal_free(call);
|
2016-04-24 12:06:13 +02:00
|
|
|
}
|
|
|
|
|
2020-01-08 17:53:31 +01:00
|
|
|
void bitcoind_getrawblockbyheight_(struct bitcoind *bitcoind,
|
|
|
|
u32 height,
|
|
|
|
void (*cb)(struct bitcoind *bitcoind,
|
|
|
|
struct bitcoin_blkid *blkid,
|
|
|
|
struct bitcoin_block *blk,
|
|
|
|
void *arg),
|
|
|
|
void *cb_arg)
|
2016-04-24 12:06:13 +02:00
|
|
|
{
|
2020-01-08 17:53:31 +01:00
|
|
|
struct jsonrpc_request *req;
|
|
|
|
struct getrawblockbyheight_call *call = tal(NULL,
|
|
|
|
struct getrawblockbyheight_call);
|
|
|
|
|
|
|
|
call->bitcoind = bitcoind;
|
|
|
|
call->cb = cb;
|
|
|
|
call->cb_arg = cb_arg;
|
|
|
|
|
|
|
|
req = jsonrpc_request_start(bitcoind, "getrawblockbyheight",
|
2020-10-12 07:33:50 +02:00
|
|
|
bitcoind->log,
|
|
|
|
NULL, getrawblockbyheight_callback,
|
2021-11-24 05:06:05 +01:00
|
|
|
call);
|
2020-01-08 17:53:31 +01:00
|
|
|
json_add_num(req->stream, "height", height);
|
|
|
|
jsonrpc_request_end(req);
|
2020-03-10 00:03:34 +01:00
|
|
|
bitcoin_plugin_send(bitcoind, req);
|
2020-01-08 17:53:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* `getchaininfo`
|
|
|
|
*
|
|
|
|
* Called at startup to check the network we are operating on, and to check
|
|
|
|
* if the Bitcoin backend is synced to the network tip. This also allows to
|
|
|
|
* get the current block count.
|
|
|
|
* {
|
|
|
|
* "chain": "<bip70_chainid>",
|
|
|
|
* "headercount": <number of fetched headers>,
|
|
|
|
* "blockcount": <number of fetched block>,
|
|
|
|
* "ibd": <synced?>
|
|
|
|
* }
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct getchaininfo_call {
|
|
|
|
struct bitcoind *bitcoind;
|
|
|
|
/* Should we log verbosely? */
|
|
|
|
bool first_call;
|
|
|
|
void (*cb)(struct bitcoind *bitcoind,
|
|
|
|
const char *chain,
|
|
|
|
u32 headercount,
|
|
|
|
u32 blockcount,
|
|
|
|
const bool ibd,
|
|
|
|
const bool first_call,
|
|
|
|
void *);
|
|
|
|
void *cb_arg;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void getchaininfo_callback(const char *buf, const jsmntok_t *toks,
|
|
|
|
const jsmntok_t *idtok,
|
|
|
|
struct getchaininfo_call *call)
|
|
|
|
{
|
2021-01-07 06:34:43 +01:00
|
|
|
const char *err, *chain;
|
2021-01-06 07:03:04 +01:00
|
|
|
u32 headers, blocks;
|
|
|
|
bool ibd;
|
|
|
|
|
2021-01-07 06:34:43 +01:00
|
|
|
err = json_scan(tmpctx, buf, toks,
|
|
|
|
"{result:{chain:%,headercount:%,blockcount:%,ibd:%}}",
|
|
|
|
JSON_SCAN_TAL(tmpctx, json_strdup, &chain),
|
|
|
|
JSON_SCAN(json_to_number, &headers),
|
|
|
|
JSON_SCAN(json_to_number, &blocks),
|
|
|
|
JSON_SCAN(json_to_bool, &ibd));
|
|
|
|
if (err)
|
2020-01-08 17:53:31 +01:00
|
|
|
bitcoin_plugin_error(call->bitcoind, buf, toks, "getchaininfo",
|
2021-01-07 06:34:43 +01:00
|
|
|
"bad 'result' field: %s", err);
|
2020-01-08 17:53:31 +01:00
|
|
|
|
|
|
|
db_begin_transaction(call->bitcoind->ld->wallet->db);
|
2021-01-06 07:03:04 +01:00
|
|
|
call->cb(call->bitcoind, chain, headers, blocks, ibd,
|
|
|
|
call->first_call, call->cb_arg);
|
2020-01-08 17:53:31 +01:00
|
|
|
db_commit_transaction(call->bitcoind->ld->wallet->db);
|
|
|
|
|
|
|
|
tal_free(call);
|
|
|
|
}
|
|
|
|
|
|
|
|
void bitcoind_getchaininfo_(struct bitcoind *bitcoind,
|
|
|
|
const bool first_call,
|
|
|
|
void (*cb)(struct bitcoind *bitcoind,
|
|
|
|
const char *chain,
|
|
|
|
u32 headercount,
|
|
|
|
u32 blockcount,
|
|
|
|
const bool ibd,
|
|
|
|
const bool first_call,
|
|
|
|
void *),
|
|
|
|
void *cb_arg)
|
|
|
|
{
|
|
|
|
struct jsonrpc_request *req;
|
|
|
|
struct getchaininfo_call *call = tal(bitcoind, struct getchaininfo_call);
|
|
|
|
|
|
|
|
call->bitcoind = bitcoind;
|
|
|
|
call->cb = cb;
|
|
|
|
call->cb_arg = cb_arg;
|
|
|
|
call->first_call = first_call;
|
|
|
|
|
|
|
|
req = jsonrpc_request_start(bitcoind, "getchaininfo", bitcoind->log,
|
2020-10-12 07:33:50 +02:00
|
|
|
NULL, getchaininfo_callback, call);
|
2020-01-08 17:53:31 +01:00
|
|
|
jsonrpc_request_end(req);
|
2020-03-10 00:03:34 +01:00
|
|
|
bitcoin_plugin_send(bitcoind, req);
|
2016-04-24 12:06:13 +02:00
|
|
|
}
|
|
|
|
|
2020-01-09 16:38:12 +01:00
|
|
|
/* `getutxout`
|
|
|
|
*
|
2021-03-14 08:50:40 +01:00
|
|
|
* Get information about an UTXO. If the TXO is spent, the plugin will set
|
2020-01-09 16:38:12 +01:00
|
|
|
* all fields to `null`.
|
|
|
|
* {
|
|
|
|
* "amount": <The output's amount in *sats*>,
|
|
|
|
* "script": "The output's scriptPubKey",
|
|
|
|
* }
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct getutxout_call {
|
|
|
|
struct bitcoind *bitcoind;
|
2018-01-04 12:40:58 +01:00
|
|
|
unsigned int blocknum, txnum, outnum;
|
|
|
|
|
2018-01-31 22:26:25 +01:00
|
|
|
/* The real callback */
|
2020-01-09 16:38:12 +01:00
|
|
|
void (*cb)(struct bitcoind *bitcoind,
|
|
|
|
const struct bitcoin_tx_output *txout, void *arg);
|
2018-01-04 12:40:58 +01:00
|
|
|
/* The real callback arg */
|
2020-01-09 16:38:12 +01:00
|
|
|
void *cb_arg;
|
2018-01-04 12:40:58 +01:00
|
|
|
};
|
|
|
|
|
2020-01-09 16:38:12 +01:00
|
|
|
static void getutxout_callback(const char *buf, const jsmntok_t *toks,
|
|
|
|
const jsmntok_t *idtok,
|
|
|
|
struct getutxout_call *call)
|
2018-01-04 12:40:58 +01:00
|
|
|
{
|
2021-01-07 06:34:43 +01:00
|
|
|
const char *err;
|
2020-01-09 16:38:12 +01:00
|
|
|
struct bitcoin_tx_output txout;
|
2018-01-04 12:40:58 +01:00
|
|
|
|
2021-01-07 06:34:43 +01:00
|
|
|
err = json_scan(tmpctx, buf, toks, "{result:{script:null}}");
|
|
|
|
if (!err) {
|
2020-01-09 16:38:12 +01:00
|
|
|
db_begin_transaction(call->bitcoind->ld->wallet->db);
|
|
|
|
call->cb(call->bitcoind, NULL, call->cb_arg);
|
|
|
|
db_commit_transaction(call->bitcoind->ld->wallet->db);
|
|
|
|
goto clean;
|
|
|
|
}
|
2018-01-04 12:40:58 +01:00
|
|
|
|
2021-01-07 06:34:43 +01:00
|
|
|
err = json_scan(tmpctx, buf, toks, "{result:{script:%,amount:%}}",
|
|
|
|
JSON_SCAN_TAL(tmpctx, json_tok_bin_from_hex,
|
|
|
|
&txout.script),
|
|
|
|
JSON_SCAN(json_to_sat, &txout.amount));
|
|
|
|
if (err)
|
2020-01-09 16:38:12 +01:00
|
|
|
bitcoin_plugin_error(call->bitcoind, buf, toks, "getutxout",
|
2021-01-07 06:34:43 +01:00
|
|
|
"bad 'result' field: %s", err);
|
2018-01-04 12:40:58 +01:00
|
|
|
|
2020-01-09 16:38:12 +01:00
|
|
|
db_begin_transaction(call->bitcoind->ld->wallet->db);
|
|
|
|
call->cb(call->bitcoind, &txout, call->cb_arg);
|
|
|
|
db_commit_transaction(call->bitcoind->ld->wallet->db);
|
2018-01-04 12:40:58 +01:00
|
|
|
|
2020-01-09 16:38:12 +01:00
|
|
|
clean:
|
|
|
|
tal_free(call);
|
2018-01-04 12:40:58 +01:00
|
|
|
}
|
|
|
|
|
2020-01-09 16:38:12 +01:00
|
|
|
void bitcoind_getutxout_(struct bitcoind *bitcoind,
|
2021-10-13 05:45:36 +02:00
|
|
|
const struct bitcoin_outpoint *outpoint,
|
2020-01-09 16:38:12 +01:00
|
|
|
void (*cb)(struct bitcoind *bitcoind,
|
|
|
|
const struct bitcoin_tx_output *txout,
|
|
|
|
void *arg),
|
|
|
|
void *cb_arg)
|
2018-01-31 22:26:25 +01:00
|
|
|
{
|
2020-01-09 16:38:12 +01:00
|
|
|
struct jsonrpc_request *req;
|
|
|
|
struct getutxout_call *call = tal(bitcoind, struct getutxout_call);
|
|
|
|
|
|
|
|
call->bitcoind = bitcoind;
|
|
|
|
call->cb = cb;
|
|
|
|
call->cb_arg = cb_arg;
|
|
|
|
|
|
|
|
req = jsonrpc_request_start(bitcoind, "getutxout", bitcoind->log,
|
2020-10-12 07:33:50 +02:00
|
|
|
NULL, getutxout_callback, call);
|
2021-10-13 05:45:36 +02:00
|
|
|
json_add_txid(req->stream, "txid", &outpoint->txid);
|
|
|
|
json_add_num(req->stream, "vout", outpoint->n);
|
2020-01-09 16:38:12 +01:00
|
|
|
jsonrpc_request_end(req);
|
2020-03-10 00:03:34 +01:00
|
|
|
bitcoin_plugin_send(bitcoind, req);
|
2018-01-31 22:26:25 +01:00
|
|
|
}
|
|
|
|
|
2019-08-05 18:33:08 +02:00
|
|
|
/* Context for the getfilteredblock call. Wraps the actual arguments while we
|
|
|
|
* process the various steps. */
|
|
|
|
struct filteredblock_call {
|
2019-08-06 16:26:42 +02:00
|
|
|
struct list_node list;
|
2019-08-08 06:24:33 +02:00
|
|
|
void (*cb)(struct bitcoind *bitcoind, const struct filteredblock *fb,
|
2019-08-05 18:33:08 +02:00
|
|
|
void *arg);
|
|
|
|
void *arg;
|
|
|
|
|
|
|
|
struct filteredblock *result;
|
|
|
|
struct filteredblock_outpoint **outpoints;
|
|
|
|
size_t current_outpoint;
|
|
|
|
struct timeabs start_time;
|
2019-08-19 19:49:09 +02:00
|
|
|
u32 height;
|
2019-08-05 18:33:08 +02:00
|
|
|
};
|
|
|
|
|
2019-08-06 16:26:42 +02:00
|
|
|
/* Declaration for recursion in process_getfilteredblock_step1 */
|
|
|
|
static void
|
|
|
|
process_getfiltered_block_final(struct bitcoind *bitcoind,
|
|
|
|
const struct filteredblock_call *call);
|
|
|
|
|
2019-08-05 18:33:08 +02:00
|
|
|
static void
|
2020-01-09 14:14:15 +01:00
|
|
|
process_getfilteredblock_step2(struct bitcoind *bitcoind,
|
2019-08-05 18:33:08 +02:00
|
|
|
const struct bitcoin_tx_output *output,
|
|
|
|
void *arg)
|
|
|
|
{
|
|
|
|
struct filteredblock_call *call = (struct filteredblock_call *)arg;
|
|
|
|
struct filteredblock_outpoint *o = call->outpoints[call->current_outpoint];
|
|
|
|
|
|
|
|
/* If this output is unspent, add it to the filteredblock result. */
|
|
|
|
if (output)
|
|
|
|
tal_arr_expand(&call->result->outpoints, tal_steal(call->result, o));
|
|
|
|
|
|
|
|
call->current_outpoint++;
|
|
|
|
if (call->current_outpoint < tal_count(call->outpoints)) {
|
|
|
|
o = call->outpoints[call->current_outpoint];
|
2021-10-13 05:45:36 +02:00
|
|
|
bitcoind_getutxout(bitcoind, &o->outpoint,
|
2020-01-09 14:14:15 +01:00
|
|
|
process_getfilteredblock_step2, call);
|
2019-08-05 18:33:08 +02:00
|
|
|
} else {
|
|
|
|
/* If there were no more outpoints to check, we call the callback. */
|
2019-08-06 16:26:42 +02:00
|
|
|
process_getfiltered_block_final(bitcoind, call);
|
2019-08-05 18:33:08 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-09 14:14:15 +01:00
|
|
|
static void process_getfilteredblock_step1(struct bitcoind *bitcoind,
|
|
|
|
struct bitcoin_blkid *blkid,
|
2019-08-05 18:33:08 +02:00
|
|
|
struct bitcoin_block *block,
|
|
|
|
struct filteredblock_call *call)
|
|
|
|
{
|
|
|
|
struct filteredblock_outpoint *o;
|
|
|
|
struct bitcoin_tx *tx;
|
2019-08-19 19:49:09 +02:00
|
|
|
|
2020-01-09 14:14:15 +01:00
|
|
|
/* If we were unable to fetch the block hash (bitcoind doesn't know
|
|
|
|
* about a block at that height), we can short-circuit and just call
|
|
|
|
* the callback. */
|
|
|
|
if (!blkid)
|
2019-08-19 19:49:09 +02:00
|
|
|
return process_getfiltered_block_final(bitcoind, call);
|
|
|
|
|
2020-01-09 14:14:15 +01:00
|
|
|
/* So we have the first piece of the puzzle, the block hash */
|
|
|
|
call->result = tal(call, struct filteredblock);
|
|
|
|
call->result->height = call->height;
|
|
|
|
call->result->outpoints = tal_arr(call->result, struct filteredblock_outpoint *, 0);
|
|
|
|
call->result->id = *blkid;
|
|
|
|
|
|
|
|
/* If the plugin gave us a block id, they MUST send us a block. */
|
|
|
|
assert(block != NULL);
|
|
|
|
|
2019-08-05 18:33:08 +02:00
|
|
|
call->result->prev_hash = block->hdr.prev_hash;
|
|
|
|
|
|
|
|
/* Allocate an array containing all the potentially interesting
|
|
|
|
* outpoints. We will later copy the ones we're interested in into the
|
|
|
|
* call->result if they are unspent. */
|
|
|
|
|
|
|
|
call->outpoints = tal_arr(call, struct filteredblock_outpoint *, 0);
|
|
|
|
for (size_t i = 0; i < tal_count(block->tx); i++) {
|
|
|
|
tx = block->tx[i];
|
|
|
|
for (size_t j = 0; j < tx->wtx->num_outputs; j++) {
|
|
|
|
const u8 *script = bitcoin_tx_output_get_script(NULL, tx, j);
|
2019-09-26 21:07:20 +02:00
|
|
|
struct amount_asset amount = bitcoin_tx_output_get_amount(tx, j);
|
|
|
|
if (amount_asset_is_main(&amount) && is_p2wsh(script, NULL)) {
|
2019-08-05 18:33:08 +02:00
|
|
|
/* This is an interesting output, remember it. */
|
|
|
|
o = tal(call->outpoints, struct filteredblock_outpoint);
|
2021-10-13 05:45:36 +02:00
|
|
|
bitcoin_txid(tx, &o->outpoint.txid);
|
|
|
|
o->outpoint.n = j;
|
2019-09-26 21:07:20 +02:00
|
|
|
o->amount = amount_asset_to_sat(&amount);
|
2019-08-05 18:33:08 +02:00
|
|
|
o->txindex = i;
|
|
|
|
o->scriptPubKey = tal_steal(o, script);
|
|
|
|
tal_arr_expand(&call->outpoints, o);
|
|
|
|
} else {
|
|
|
|
tal_free(script);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tal_count(call->outpoints) == 0) {
|
|
|
|
/* If there were no outpoints to check, we can short-circuit
|
|
|
|
* and just call the callback. */
|
2019-08-06 16:26:42 +02:00
|
|
|
process_getfiltered_block_final(bitcoind, call);
|
2019-08-05 18:33:08 +02:00
|
|
|
} else {
|
|
|
|
|
|
|
|
/* Otherwise we start iterating through call->outpoints and
|
|
|
|
* store the one's that are unspent in
|
|
|
|
* call->result->outpoints. */
|
|
|
|
o = call->outpoints[call->current_outpoint];
|
2021-10-13 05:45:36 +02:00
|
|
|
bitcoind_getutxout(bitcoind, &o->outpoint,
|
2020-01-09 14:14:15 +01:00
|
|
|
process_getfilteredblock_step2, call);
|
2019-08-05 18:33:08 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-06 16:26:42 +02:00
|
|
|
/* Takes a call, dispatches it to all queued requests that match the same
|
|
|
|
* height, and then kicks off the next call. */
|
|
|
|
static void
|
|
|
|
process_getfiltered_block_final(struct bitcoind *bitcoind,
|
|
|
|
const struct filteredblock_call *call)
|
|
|
|
{
|
|
|
|
struct filteredblock_call *c, *next;
|
2019-08-19 19:49:09 +02:00
|
|
|
u32 height = call->height;
|
|
|
|
|
|
|
|
if (call->result == NULL)
|
|
|
|
goto next;
|
|
|
|
|
2019-08-06 16:26:42 +02:00
|
|
|
/* Need to steal so we don't accidentally free it while iterating through the list below. */
|
|
|
|
struct filteredblock *fb = tal_steal(NULL, call->result);
|
|
|
|
list_for_each_safe(&bitcoind->pending_getfilteredblock, c, next, list) {
|
2019-08-19 19:49:09 +02:00
|
|
|
if (c->height == height) {
|
2019-08-06 16:26:42 +02:00
|
|
|
c->cb(bitcoind, fb, c->arg);
|
|
|
|
list_del(&c->list);
|
|
|
|
tal_free(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tal_free(fb);
|
|
|
|
|
2019-08-19 19:49:09 +02:00
|
|
|
next:
|
2019-08-06 16:26:42 +02:00
|
|
|
/* Nothing to free here, since `*call` was already deleted during the
|
|
|
|
* iteration above. It was also removed from the list, so no need to
|
|
|
|
* pop here. */
|
|
|
|
if (!list_empty(&bitcoind->pending_getfilteredblock)) {
|
|
|
|
c = list_top(&bitcoind->pending_getfilteredblock, struct filteredblock_call, list);
|
2020-01-09 14:14:15 +01:00
|
|
|
bitcoind_getrawblockbyheight(bitcoind, c->height,
|
|
|
|
process_getfilteredblock_step1, c);
|
2019-08-06 16:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-05 18:33:08 +02:00
|
|
|
void bitcoind_getfilteredblock_(struct bitcoind *bitcoind, u32 height,
|
|
|
|
void (*cb)(struct bitcoind *bitcoind,
|
2019-08-08 06:24:33 +02:00
|
|
|
const struct filteredblock *fb,
|
2019-08-05 18:33:08 +02:00
|
|
|
void *arg),
|
|
|
|
void *arg)
|
|
|
|
{
|
|
|
|
/* Stash the call context for when we need to call the callback after
|
|
|
|
* all the bitcoind calls we need to perform. */
|
|
|
|
struct filteredblock_call *call = tal(bitcoind, struct filteredblock_call);
|
2019-08-06 16:26:42 +02:00
|
|
|
/* If this is the first request, we should start processing it. */
|
|
|
|
bool start = list_empty(&bitcoind->pending_getfilteredblock);
|
2019-08-05 18:33:08 +02:00
|
|
|
call->cb = cb;
|
|
|
|
call->arg = arg;
|
2019-08-19 19:49:09 +02:00
|
|
|
call->height = height;
|
2019-08-05 18:33:08 +02:00
|
|
|
assert(call->cb != NULL);
|
|
|
|
call->start_time = time_now();
|
2019-08-19 19:49:09 +02:00
|
|
|
call->result = NULL;
|
2019-08-06 12:38:20 +02:00
|
|
|
call->current_outpoint = 0;
|
2019-08-05 18:33:08 +02:00
|
|
|
|
2019-08-06 16:26:42 +02:00
|
|
|
list_add_tail(&bitcoind->pending_getfilteredblock, &call->list);
|
|
|
|
if (start)
|
2020-01-09 14:14:15 +01:00
|
|
|
bitcoind_getrawblockbyheight(bitcoind, height,
|
|
|
|
process_getfilteredblock_step1, call);
|
2019-08-05 18:33:08 +02:00
|
|
|
}
|
|
|
|
|
2017-09-12 06:55:54 +02:00
|
|
|
static void destroy_bitcoind(struct bitcoind *bitcoind)
|
|
|
|
{
|
2020-01-07 16:09:59 +01:00
|
|
|
strmap_clear(&bitcoind->pluginsmap);
|
2017-09-12 06:55:54 +02:00
|
|
|
}
|
|
|
|
|
2017-11-01 11:20:40 +01:00
|
|
|
struct bitcoind *new_bitcoind(const tal_t *ctx,
|
|
|
|
struct lightningd *ld,
|
|
|
|
struct log *log)
|
2017-03-02 13:21:49 +01:00
|
|
|
{
|
|
|
|
struct bitcoind *bitcoind = tal(ctx, struct bitcoind);
|
|
|
|
|
2020-01-07 16:09:59 +01:00
|
|
|
strmap_init(&bitcoind->pluginsmap);
|
2017-11-01 11:20:40 +01:00
|
|
|
bitcoind->ld = ld;
|
2017-03-02 13:21:49 +01:00
|
|
|
bitcoind->log = log;
|
2019-08-06 16:26:42 +02:00
|
|
|
list_head_init(&bitcoind->pending_getfilteredblock);
|
2017-09-12 06:55:54 +02:00
|
|
|
tal_add_destructor(bitcoind, destroy_bitcoind);
|
2020-01-08 17:53:31 +01:00
|
|
|
bitcoind->synced = false;
|
2017-03-02 13:21:49 +01:00
|
|
|
|
|
|
|
return bitcoind;
|
|
|
|
}
|