#include "lightningd.h" #include "peer_control.h" #include "subd.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include static void destroy_peer(struct peer *peer) { list_del_from(&peer->ld->peers, &peer->list); } /* We copy per-peer entries above --log-level into the main log. */ static void copy_to_parent_log(const char *prefix, enum log_level level, bool continued, const struct timeabs *time UNUSED, const char *str, const u8 *io, struct log *parent_log) { if (level == LOG_IO_IN || level == LOG_IO_OUT) log_io(parent_log, level, prefix, io, tal_len(io)); else if (continued) log_add(parent_log, "%s ... %s", prefix, str); else log_(parent_log, level, "%s %s", prefix, str); } struct peer *new_peer(struct lightningd *ld, u64 dbid, const struct pubkey *id, const struct wireaddr *addr) { /* We are owned by our channels, and freed manually by destroy_channel */ struct peer *peer = tal(NULL, struct peer); peer->ld = ld; peer->dbid = dbid; peer->id = *id; peer->uncommitted_channel = NULL; if (addr) peer->addr = *addr; else peer->addr.type = ADDR_TYPE_PADDING; list_head_init(&peer->channels); peer->direction = get_channel_direction(&peer->ld->id, &peer->id); /* Max 128k per peer. */ peer->log_book = new_log_book(128*1024, get_log_level(ld->log_book)); set_log_outfn(peer->log_book, copy_to_parent_log, ld->log); list_add_tail(&ld->peers, &peer->list); tal_add_destructor(peer, destroy_peer); return peer; } void delete_peer(struct peer *peer) { assert(list_empty(&peer->channels)); assert(!peer->uncommitted_channel); /* If it only ever existed because of uncommitted channel, it won't * be in the database */ if (peer->dbid != 0) wallet_peer_delete(peer->ld->wallet, peer->dbid); tal_free(peer); } struct peer *find_peer_by_dbid(struct lightningd *ld, u64 dbid) { struct peer *p; list_for_each(&ld->peers, p, list) if (p->dbid == dbid) return p; return NULL; } struct peer *peer_by_id(struct lightningd *ld, const struct pubkey *id) { struct peer *p; list_for_each(&ld->peers, p, list) if (pubkey_eq(&p->id, id)) return p; return NULL; } struct peer *peer_from_json(struct lightningd *ld, const char *buffer, jsmntok_t *peeridtok) { struct pubkey peerid; if (!json_tok_pubkey(buffer, peeridtok, &peerid)) return NULL; return peer_by_id(ld, &peerid); } u8 *p2wpkh_for_keyidx(const tal_t *ctx, struct lightningd *ld, u64 keyidx) { struct pubkey shutdownkey; if (!bip32_pubkey(ld->wallet->bip32_base, &shutdownkey, keyidx)) return NULL; return scriptpubkey_p2wpkh(ctx, &shutdownkey); } u32 feerate_min(struct lightningd *ld) { if (ld->config.ignore_fee_limits) return 1; /* Set this to average of slow and normal.*/ return (get_feerate(ld->topology, FEERATE_SLOW) + get_feerate(ld->topology, FEERATE_NORMAL)) / 2; } /* BOLT #2: * * Given the variance in fees, and the fact that the transaction may * be spent in the future, it's a good idea for the fee payer to keep * a good margin, say 5x the expected fee requirement */ u32 feerate_max(struct lightningd *ld) { if (ld->config.ignore_fee_limits) return UINT_MAX; return get_feerate(ld->topology, FEERATE_IMMEDIATE) * 5; } static void sign_last_tx(struct channel *channel) { const tal_t *tmpctx = tal_tmpctx(channel); u8 *funding_wscript; struct pubkey local_funding_pubkey; struct secrets secrets; secp256k1_ecdsa_signature sig; assert(!channel->last_tx->input[0].witness); derive_basepoints(&channel->seed, &local_funding_pubkey, NULL, &secrets, NULL); funding_wscript = bitcoin_redeem_2of2(tmpctx, &local_funding_pubkey, &channel->channel_info.remote_fundingkey); /* Need input amount for signing */ channel->last_tx->input[0].amount = tal_dup(channel->last_tx->input, u64, &channel->funding_satoshi); sign_tx_input(channel->last_tx, 0, NULL, funding_wscript, &secrets.funding_privkey, &local_funding_pubkey, &sig); channel->last_tx->input[0].witness = bitcoin_witness_2of2(channel->last_tx->input, &channel->last_sig, &sig, &channel->channel_info.remote_fundingkey, &local_funding_pubkey); tal_free(tmpctx); } static void remove_sig(struct bitcoin_tx *signed_tx) { signed_tx->input[0].amount = tal_free(signed_tx->input[0].amount); signed_tx->input[0].witness = tal_free(signed_tx->input[0].witness); } void drop_to_chain(struct lightningd *ld, struct channel *channel) { sign_last_tx(channel); /* Keep broadcasting until we say stop (can fail due to dup, * if they beat us to the broadcast). */ broadcast_tx(ld->topology, channel, channel->last_tx, NULL); remove_sig(channel->last_tx); } void channel_errmsg(struct channel *channel, int peer_fd, int gossip_fd, const struct crypto_state *cs, u64 gossip_index, const struct channel_id *channel_id UNUSED, const char *desc, const u8 *err_for_them) { struct lightningd *ld = channel->peer->ld; u8 *msg; /* No peer fd means a subd crash or disconnection. */ if (peer_fd == -1) { channel_fail_transient(channel, "%s: %s", channel->owner->name, desc); return; } /* Do we have an error to send? */ if (err_for_them && !channel->error) channel->error = tal_dup_arr(channel, u8, err_for_them, tal_len(err_for_them), 0); /* BOLT #1: * * A sending node: *... * - when `channel_id` is 0: * - MUST fail all channels. * - MUST close the connection. */ /* FIXME: Gossipd closes connection, but doesn't fail channels. */ /* BOLT #1: * * A sending node: * - when sending `error`: * - MUST fail the channel referred to by the error message. *... * The receiving node: * - upon receiving `error`: * - MUST fail the channel referred to by the error message. */ channel_fail_permanent(channel, "%s: %s ERROR %s", channel->owner->name, err_for_them ? "sent" : "received", desc); /* Hand back to gossipd, with any error packet. */ msg = towire_gossipctl_hand_back_peer(NULL, &channel->peer->id, cs, gossip_index, err_for_them); subd_send_msg(ld->gossip, take(msg)); subd_send_fd(ld->gossip, peer_fd); subd_send_fd(ld->gossip, gossip_fd); } /* Gossipd tells us a peer has connected */ void peer_connected(struct lightningd *ld, const u8 *msg, int peer_fd, int gossip_fd) { struct pubkey id; struct crypto_state cs; u8 *gfeatures, *lfeatures; u8 *error; u8 *supported_global_features; u8 *supported_local_features; struct channel *channel; struct wireaddr addr; u64 gossip_index; struct uncommitted_channel *uc; if (!fromwire_gossip_peer_connected(msg, msg, &id, &addr, &cs, &gossip_index, &gfeatures, &lfeatures)) fatal("Gossip gave bad GOSSIP_PEER_CONNECTED message %s", tal_hex(msg, msg)); if (unsupported_features(gfeatures, lfeatures)) { log_unusual(ld->log, "peer %s offers unsupported features %s/%s", type_to_string(msg, struct pubkey, &id), tal_hex(msg, gfeatures), tal_hex(msg, lfeatures)); supported_global_features = get_supported_global_features(msg); supported_local_features = get_supported_local_features(msg); error = towire_errorfmt(msg, NULL, "We only support globalfeatures %s" " and localfeatures %s", tal_hexstr(msg, supported_global_features, tal_len(supported_global_features)), tal_hexstr(msg, supported_local_features, tal_len(supported_local_features))); goto send_error; } /* If we're already dealing with this peer, hand off to correct * subdaemon. Otherwise, we'll respond iff they ask about an inactive * channel. */ channel = active_channel_by_id(ld, &id, &uc); /* Opening now? Kill it */ if (uc) { kill_uncommitted_channel(uc, "Peer reconnected"); goto return_to_gossipd; } if (channel) { log_debug(channel->log, "Peer has reconnected, state %s", channel_state_name(channel)); /* If we have a canned error, deliver it now. */ if (channel->error) { error = channel->error; goto send_error; } #if DEVELOPER if (dev_disconnect_permanent(ld)) { channel_internal_error(channel, "dev_disconnect permfail"); error = channel->error; goto send_error; } #endif switch (channel->state) { case ONCHAIN: case FUNDING_SPEND_SEEN: case CLOSINGD_COMPLETE: /* Channel is active! */ abort(); case CHANNELD_AWAITING_LOCKIN: case CHANNELD_NORMAL: case CHANNELD_SHUTTING_DOWN: /* Stop any existing daemon, without triggering error * on this peer. */ channel_set_owner(channel, NULL); channel->peer->addr = addr; peer_start_channeld(channel, &cs, gossip_index, peer_fd, gossip_fd, NULL, true); goto connected; case CLOSINGD_SIGEXCHANGE: /* Stop any existing daemon, without triggering error * on this peer. */ channel_set_owner(channel, NULL); channel->peer->addr = addr; peer_start_closingd(channel, &cs, gossip_index, peer_fd, gossip_fd, true); goto connected; } abort(); } return_to_gossipd: /* Otherwise, we hand back to gossipd, to continue. */ msg = towire_gossipctl_hand_back_peer(msg, &id, &cs, gossip_index, NULL); subd_send_msg(ld->gossip, take(msg)); subd_send_fd(ld->gossip, peer_fd); subd_send_fd(ld->gossip, gossip_fd); connected: /* If we were waiting for connection, we succeeded. */ connect_succeeded(ld, &id); return; send_error: /* Hand back to gossipd, with an error packet. */ connect_failed(ld, &id, sanitize_error(msg, error, NULL)); msg = towire_gossipctl_hand_back_peer(msg, &id, &cs, gossip_index, error); subd_send_msg(ld->gossip, take(msg)); subd_send_fd(ld->gossip, peer_fd); subd_send_fd(ld->gossip, gossip_fd); } static struct channel *channel_by_channel_id(struct peer *peer, const struct channel_id *channel_id) { struct channel *channel; list_for_each(&peer->channels, channel, list) { struct channel_id cid; derive_channel_id(&cid, &channel->funding_txid, channel->funding_outnum); if (structeq(&cid, channel_id)) return channel; } return NULL; } /* We only get here IF we weren't trying to connect to it. */ void peer_sent_nongossip(struct lightningd *ld, const struct pubkey *id, const struct wireaddr *addr, const struct crypto_state *cs, u64 gossip_index, const u8 *gfeatures, const u8 *lfeatures, int peer_fd, int gossip_fd, const u8 *in_msg) { struct channel_id *channel_id, extracted_channel_id; struct peer *peer; u8 *error, *msg; if (!extract_channel_id(in_msg, &extracted_channel_id)) channel_id = NULL; else channel_id = &extracted_channel_id; peer = peer_by_id(ld, id); /* Open request? */ if (fromwire_peektype(in_msg) == WIRE_OPEN_CHANNEL) { error = peer_accept_channel(ld, id, addr, cs, gossip_index, gfeatures, lfeatures, peer_fd, gossip_fd, channel_id, in_msg); if (error) goto send_error; return; } /* If they are talking about a specific channel id, we may have an * error for them. */ if (peer && channel_id) { struct channel *channel; channel = channel_by_channel_id(peer, channel_id); if (channel && channel->error) { error = channel->error; goto send_error; } } /* Weird request. */ error = towire_errorfmt(ld, channel_id, "Unexpected message %i for peer", fromwire_peektype(in_msg)); send_error: /* Hand back to gossipd, with an error packet. */ connect_failed(ld, id, sanitize_error(error, error, NULL)); msg = towire_gossipctl_hand_back_peer(ld, id, cs, gossip_index, error); subd_send_msg(ld->gossip, take(msg)); subd_send_fd(ld->gossip, peer_fd); subd_send_fd(ld->gossip, gossip_fd); tal_free(error); } static enum watch_result funding_announce_cb(struct channel *channel, const struct bitcoin_tx *tx UNUSED, unsigned int depth) { if (depth < ANNOUNCE_MIN_DEPTH) { return KEEP_WATCHING; } if (!channel->owner || !streq(channel->owner->name, "lightning_channeld")) { log_debug(channel->log, "Funding tx announce ready, but channel state %s" " owned by %s", channel_state_name(channel), channel->owner ? channel->owner->name : "none"); return KEEP_WATCHING; } subd_send_msg(channel->owner, take(towire_channel_funding_announce_depth(channel))); return DELETE_WATCH; } static enum watch_result funding_lockin_cb(struct channel *channel, const struct bitcoin_tx *tx, unsigned int depth) { struct bitcoin_txid txid; const char *txidstr; struct txlocator *loc; bool channel_ready; struct lightningd *ld = channel->peer->ld; bitcoin_txid(tx, &txid); txidstr = type_to_string(channel, struct bitcoin_txid, &txid); log_debug(channel->log, "Funding tx %s depth %u of %u", txidstr, depth, channel->minimum_depth); tal_free(txidstr); if (depth < channel->minimum_depth) return KEEP_WATCHING; loc = locate_tx(channel, ld->topology, &txid); /* If we restart, we could already have peer->scid from database */ if (!channel->scid) { channel->scid = tal(channel, struct short_channel_id); channel->scid->blocknum = loc->blkheight; channel->scid->txnum = loc->index; channel->scid->outnum = channel->funding_outnum; } tal_free(loc); /* In theory, it could have been buried before we got back * from accepting openingd or disconnected: just wait for next one. */ channel_ready = (channel->owner && channel->state == CHANNELD_AWAITING_LOCKIN); if (!channel_ready) { log_debug(channel->log, "Funding tx confirmed, but channel state %s %s", channel_state_name(channel), channel->owner ? channel->owner->name : "unowned"); } else { subd_send_msg(channel->owner, take(towire_channel_funding_locked(channel, channel->scid))); } /* BOLT #7: * * If the `open_channel` message had the `announce_channel` bit set, * then both nodes must send the `announcement_signatures` message, * otherwise they MUST NOT. */ if (!(channel->channel_flags & CHANNEL_FLAGS_ANNOUNCE_CHANNEL)) return DELETE_WATCH; /* Tell channeld that we have reached the announce_depth and * that it may send the announcement_signatures upon receiving * funding_locked, or right now if it already received it * before. If we are at the right depth, call the callback * directly, otherwise schedule a callback */ if (depth >= ANNOUNCE_MIN_DEPTH) funding_announce_cb(channel, tx, depth); else watch_txid(channel, ld->topology, channel, &txid, funding_announce_cb); return DELETE_WATCH; } void channel_watch_funding(struct lightningd *ld, struct channel *channel) { /* FIXME: Remove arg from cb? */ watch_txid(channel, ld->topology, channel, &channel->funding_txid, funding_lockin_cb); watch_txo(channel, ld->topology, channel, &channel->funding_txid, channel->funding_outnum, funding_spent); } struct getpeers_args { struct command *cmd; /* If non-NULL, they want logs too */ enum log_level *ll; /* If set, only report on a specific id. */ struct pubkey *specific_id; }; static void gossipd_getpeers_complete(struct subd *gossip, const u8 *msg, const int *fds UNUSED, struct getpeers_args *gpa) { /* This is a little sneaky... */ struct pubkey *ids; struct wireaddr *addrs; struct json_result *response = new_json_result(gpa->cmd); struct peer *p; if (!fromwire_gossip_getpeers_reply(msg, msg, &ids, &addrs)) { command_fail(gpa->cmd, "Bad response from gossipd"); return; } /* First the peers not just gossiping. */ json_object_start(response, NULL); json_array_start(response, "peers"); list_for_each(&gpa->cmd->ld->peers, p, list) { bool connected; struct channel *channel; if (gpa->specific_id && !pubkey_eq(gpa->specific_id, &p->id)) continue; json_object_start(response, NULL); json_add_pubkey(response, "id", &p->id); channel = peer_active_channel(p); connected = (channel && channel->owner != NULL); json_add_bool(response, "connected", connected); if (connected) { json_array_start(response, "netaddr"); if (p->addr.type != ADDR_TYPE_PADDING) json_add_string(response, NULL, type_to_string(response, struct wireaddr, &p->addr)); json_array_end(response); } json_array_start(response, "channels"); json_add_uncommitted_channel(response, p->uncommitted_channel); list_for_each(&p->channels, channel, list) { json_object_start(response, NULL); json_add_string(response, "state", channel_state_name(channel)); if (channel->owner) json_add_string(response, "owner", channel->owner->name); if (channel->scid) json_add_short_channel_id(response, "short_channel_id", channel->scid); json_add_txid(response, "funding_txid", &channel->funding_txid); json_add_u64(response, "msatoshi_to_us", channel->our_msatoshi); json_add_u64(response, "msatoshi_total", channel->funding_satoshi * 1000); /* channel config */ json_add_u64(response, "dust_limit_satoshis", channel->our_config.dust_limit_satoshis); json_add_u64(response, "max_htlc_value_in_flight_msat", channel->our_config.max_htlc_value_in_flight_msat); json_add_u64(response, "channel_reserve_satoshis", channel->our_config.channel_reserve_satoshis); json_add_u64(response, "htlc_minimum_msat", channel->our_config.htlc_minimum_msat); json_add_num(response, "to_self_delay", channel->our_config.to_self_delay); json_add_num(response, "max_accepted_htlcs", channel->our_config.max_accepted_htlcs); json_array_start(response, "status"); for (size_t i = 0; i < ARRAY_SIZE(channel->billboard.permanent); i++) { if (!channel->billboard.permanent[i]) continue; json_add_string(response, NULL, channel->billboard.permanent[i]); } if (channel->billboard.transient) json_add_string(response, NULL, channel->billboard.transient); json_array_end(response); json_object_end(response); } json_array_end(response); if (gpa->ll) json_add_log(response, p->log_book, *gpa->ll); json_object_end(response); } for (size_t i = 0; i < tal_count(ids); i++) { /* Don't report peers in both, which can happen if they're * reconnecting */ if (peer_by_id(gpa->cmd->ld, ids + i)) continue; json_object_start(response, NULL); /* Fake state. */ json_add_string(response, "state", "GOSSIPING"); json_add_pubkey(response, "id", ids+i); json_array_start(response, "netaddr"); if (addrs[i].type != ADDR_TYPE_PADDING) json_add_string(response, NULL, type_to_string(response, struct wireaddr, addrs + i)); json_array_end(response); json_add_bool(response, "connected", true); json_add_string(response, "owner", gossip->name); json_object_end(response); } json_array_end(response); json_object_end(response); command_success(gpa->cmd, response); } static void json_listpeers(struct command *cmd, const char *buffer, const jsmntok_t *params) { jsmntok_t *leveltok; struct getpeers_args *gpa = tal(cmd, struct getpeers_args); jsmntok_t *idtok; gpa->cmd = cmd; gpa->specific_id = NULL; if (!json_get_params(cmd, buffer, params, "?id", &idtok, "?level", &leveltok, NULL)) { return; } if (idtok) { gpa->specific_id = tal_arr(cmd, struct pubkey, 1); if (!json_tok_pubkey(buffer, idtok, gpa->specific_id)) { command_fail(cmd, "id %.*s not valid", idtok->end - idtok->start, buffer + idtok->start); return; } } if (leveltok) { gpa->ll = tal(gpa, enum log_level); if (!json_tok_loglevel(buffer, leveltok, gpa->ll)) { command_fail(cmd, "Invalid level param"); return; } } else gpa->ll = NULL; /* Get peers from gossipd. */ subd_req(cmd, cmd->ld->gossip, take(towire_gossip_getpeers_request(cmd, gpa->specific_id)), -1, 0, gossipd_getpeers_complete, gpa); command_still_pending(cmd); } static const struct json_command listpeers_command = { "listpeers", json_listpeers, "Show current peers, if {level} is set, include logs for {id}" }; AUTODATA(json_command, &listpeers_command); static void json_close(struct command *cmd, const char *buffer, const jsmntok_t *params) { jsmntok_t *peertok; struct peer *peer; struct channel *channel; if (!json_get_params(cmd, buffer, params, "id", &peertok, NULL)) { return; } peer = peer_from_json(cmd->ld, buffer, peertok); if (!peer) { command_fail(cmd, "Could not find peer with that id"); return; } channel = peer_active_channel(peer); if (!channel) { struct uncommitted_channel *uc = peer->uncommitted_channel; if (uc) { /* Easy case: peer can simply be forgotten. */ kill_uncommitted_channel(uc, "close command called"); command_success(cmd, null_response(cmd)); return; } command_fail(cmd, "Peer has no active channel"); return; } /* Normal case. */ if (channel->state == CHANNELD_NORMAL) { u8 *shutdown_scriptpubkey; channel->local_shutdown_idx = wallet_get_newindex(cmd->ld); if (channel->local_shutdown_idx == -1) { command_fail(cmd, "Failed to get new key for shutdown"); return; } shutdown_scriptpubkey = p2wpkh_for_keyidx(cmd, cmd->ld, channel->local_shutdown_idx); if (!shutdown_scriptpubkey) { command_fail(cmd, "Failed to get script for shutdown"); return; } channel_set_state(channel, CHANNELD_NORMAL, CHANNELD_SHUTTING_DOWN); txfilter_add_scriptpubkey(cmd->ld->owned_txfilter, shutdown_scriptpubkey); if (channel->owner) subd_send_msg(channel->owner, take(towire_channel_send_shutdown(channel, shutdown_scriptpubkey))); command_success(cmd, null_response(cmd)); } else command_fail(cmd, "Peer is in state %s", channel_state_name(channel)); } static const struct json_command close_command = { "close", json_close, "Close the channel with peer {id}" }; AUTODATA(json_command, &close_command); static void activate_peer(struct peer *peer) { u8 *msg; struct channel *channel; struct lightningd *ld = peer->ld; /* Pass gossipd any addrhints we currently have */ msg = towire_gossipctl_peer_addrhint(peer, &peer->id, &peer->addr); subd_send_msg(peer->ld->gossip, take(msg)); /* We can only have one active channel: reconnect if not already. */ channel = peer_active_channel(peer); if (channel && !channel->owner) { msg = towire_gossipctl_reach_peer(peer, &peer->id); subd_send_msg(peer->ld->gossip, take(msg)); } list_for_each(&peer->channels, channel, list) { /* Watching lockin may be unnecessary, but it's harmless. */ channel_watch_funding(ld, channel); } } void activate_peers(struct lightningd *ld) { struct peer *p; list_for_each(&ld->peers, p, list) activate_peer(p); } #if DEVELOPER static void json_sign_last_tx(struct command *cmd, const char *buffer, const jsmntok_t *params) { jsmntok_t *peertok; struct peer *peer; struct json_result *response = new_json_result(cmd); u8 *linear; struct channel *channel; if (!json_get_params(cmd, buffer, params, "id", &peertok, NULL)) { return; } peer = peer_from_json(cmd->ld, buffer, peertok); if (!peer) { command_fail(cmd, "Could not find peer with that id"); return; } channel = peer_active_channel(peer); if (!channel) { command_fail(cmd, "Could has not active channel"); return; } log_debug(channel->log, "dev-sign-last-tx: signing tx with %zu outputs", tal_count(channel->last_tx->output)); sign_last_tx(channel); linear = linearize_tx(cmd, channel->last_tx); remove_sig(channel->last_tx); json_object_start(response, NULL); json_add_hex(response, "tx", linear, tal_len(linear)); json_object_end(response); command_success(cmd, response); } static const struct json_command dev_sign_last_tx = { "dev-sign-last-tx", json_sign_last_tx, "Sign and show the last commitment transaction with peer {id}" }; AUTODATA(json_command, &dev_sign_last_tx); static void json_dev_fail(struct command *cmd, const char *buffer, const jsmntok_t *params) { jsmntok_t *peertok; struct peer *peer; struct channel *channel; if (!json_get_params(cmd, buffer, params, "id", &peertok, NULL)) { return; } peer = peer_from_json(cmd->ld, buffer, peertok); if (!peer) { command_fail(cmd, "Could not find peer with that id"); return; } channel = peer_active_channel(peer); if (!channel) { command_fail(cmd, "Could not find active channel with peer"); return; } channel_internal_error(channel, "Failing due to dev-fail command"); command_success(cmd, null_response(cmd)); } static const struct json_command dev_fail_command = { "dev-fail", json_dev_fail, "Fail with peer {id}" }; AUTODATA(json_command, &dev_fail_command); static void dev_reenable_commit_finished(struct subd *channeld UNUSED, const u8 *resp UNUSED, const int *fds UNUSED, struct command *cmd) { command_success(cmd, null_response(cmd)); } static void json_dev_reenable_commit(struct command *cmd, const char *buffer, const jsmntok_t *params) { jsmntok_t *peertok; struct peer *peer; u8 *msg; struct channel *channel; if (!json_get_params(cmd, buffer, params, "id", &peertok, NULL)) { return; } peer = peer_from_json(cmd->ld, buffer, peertok); if (!peer) { command_fail(cmd, "Could not find peer with that id"); return; } channel = peer_active_channel(peer); if (!channel) { command_fail(cmd, "Peer has no active channel"); return; } if (!channel->owner) { command_fail(cmd, "Peer has no owner"); return; } if (!streq(channel->owner->name, "lightning_channeld")) { command_fail(cmd, "Peer owned by %s", channel->owner->name); return; } msg = towire_channel_dev_reenable_commit(channel); subd_req(peer, channel->owner, take(msg), -1, 0, dev_reenable_commit_finished, cmd); command_still_pending(cmd); } static const struct json_command dev_reenable_commit = { "dev-reenable-commit", json_dev_reenable_commit, "Re-enable the commit timer on peer {id}" }; AUTODATA(json_command, &dev_reenable_commit); struct dev_forget_channel_cmd { struct short_channel_id scid; struct channel *channel; bool force; struct command *cmd; }; static void process_dev_forget_channel(struct bitcoind *bitcoind UNUSED, const struct bitcoin_tx_output *txout, void *arg) { struct json_result *response; struct dev_forget_channel_cmd *forget = arg; if (txout != NULL && !forget->force) { command_fail(forget->cmd, "Cowardly refusing to forget channel with an " "unspent funding output, if you know what " "you're doing you can override with " "`force=true`, otherwise consider `close` or " "`dev-fail`! If you force and the channel " "confirms we will not track the funds in the " "channel"); return; } response = new_json_result(forget->cmd); json_object_start(response, NULL); json_add_bool(response, "forced", forget->force); json_add_bool(response, "funding_unspent", txout != NULL); json_add_txid(response, "funding_txid", &forget->channel->funding_txid); json_object_end(response); delete_channel(forget->channel); command_success(forget->cmd, response); } static void json_dev_forget_channel(struct command *cmd, const char *buffer, const jsmntok_t *params) { jsmntok_t *nodeidtok, *forcetok, *scidtok; struct peer *peer; struct channel *channel; struct short_channel_id scid; struct dev_forget_channel_cmd *forget = tal(cmd, struct dev_forget_channel_cmd); forget->cmd = cmd; if (!json_get_params(cmd, buffer, params, "id", &nodeidtok, "?short_channel_id", &scidtok, "?force", &forcetok, NULL)) { return; } if (scidtok && !json_tok_short_channel_id(buffer, scidtok, &scid)) { command_fail(cmd, "Invalid short_channel_id '%.*s'", scidtok->end - scidtok->start, buffer + scidtok->start); return; } forget->force = false; if (forcetok) json_tok_bool(buffer, forcetok, &forget->force); peer = peer_from_json(cmd->ld, buffer, nodeidtok); if (!peer) { command_fail(cmd, "Could not find channel with that peer"); return; } forget->channel = NULL; list_for_each(&peer->channels, channel, list) { if (scidtok) { if (!channel->scid) continue; if (!short_channel_id_eq(channel->scid, &scid)) continue; } if (forget->channel) { command_fail(cmd, "Multiple channels:" " please specify short_channel_id"); return; } forget->channel = channel; } if (!forget->channel) { command_fail(cmd, "No channels matching that short_channel_id"); return; } if (channel_has_htlc_out(forget->channel) || channel_has_htlc_in(forget->channel)) { command_fail(cmd, "This channel has HTLCs attached and it is " "not safe to forget it. Please use `close` " "or `dev-fail` instead."); return; } bitcoind_gettxout(cmd->ld->topology->bitcoind, &forget->channel->funding_txid, forget->channel->funding_outnum, process_dev_forget_channel, forget); command_still_pending(cmd); } static const struct json_command dev_forget_channel_command = { "dev-forget-channel", json_dev_forget_channel, "Forget the channel with peer {id}, ignore UTXO check with {force}='true'.", false, "Forget the channel with peer {id}. Checks if the channel is still active by checking its funding transaction. Check can be ignored by setting {force} to 'true'" }; AUTODATA(json_command, &dev_forget_channel_command); #endif /* DEVELOPER */