2018-01-14 15:15:30 +01:00
|
|
|
#include "invoices.h"
|
2017-05-23 22:07:20 +02:00
|
|
|
#include "wallet.h"
|
|
|
|
|
2020-05-21 23:32:53 +02:00
|
|
|
#include <bitcoin/psbt.h>
|
2017-06-21 17:40:42 +02:00
|
|
|
#include <bitcoin/script.h>
|
2019-12-12 18:18:25 +01:00
|
|
|
#include <ccan/array_size/array_size.h>
|
2019-01-08 01:19:50 +01:00
|
|
|
#include <ccan/mem/mem.h>
|
2017-08-05 12:26:18 +02:00
|
|
|
#include <ccan/tal/str/str.h>
|
2019-12-12 18:18:25 +01:00
|
|
|
#include <common/fee_states.h>
|
2018-03-07 01:06:07 +01:00
|
|
|
#include <common/key_derive.h>
|
2019-06-05 09:00:05 +02:00
|
|
|
#include <common/memleak.h>
|
2020-01-23 00:38:04 +01:00
|
|
|
#include <common/onionreply.h>
|
2018-03-07 01:06:07 +01:00
|
|
|
#include <common/wireaddr.h>
|
2017-07-18 19:01:26 +02:00
|
|
|
#include <inttypes.h>
|
2020-04-02 04:55:44 +02:00
|
|
|
#include <lightningd/coin_mvts.h>
|
2017-06-25 04:33:13 +02:00
|
|
|
#include <lightningd/lightningd.h>
|
2019-06-15 18:50:07 +02:00
|
|
|
#include <lightningd/notification.h>
|
2017-07-20 16:42:55 +02:00
|
|
|
#include <lightningd/peer_control.h>
|
|
|
|
#include <lightningd/peer_htlcs.h>
|
2020-08-25 04:15:48 +02:00
|
|
|
#include <onchaind/onchaind_wiregen.h>
|
2018-04-16 15:29:40 +02:00
|
|
|
#include <string.h>
|
2019-07-24 18:29:03 +02:00
|
|
|
#include <wallet/db_common.h>
|
2017-06-05 13:59:51 +02:00
|
|
|
|
2017-07-18 19:01:26 +02:00
|
|
|
#define SQLITE_MAX_UINT 0x7FFFFFFFFFFFFFFF
|
2017-09-14 21:27:41 +02:00
|
|
|
#define DIRECTION_INCOMING 0
|
|
|
|
#define DIRECTION_OUTGOING 1
|
2018-03-04 03:42:19 +01:00
|
|
|
/* How many blocks must a UTXO entry be buried under to be considered old enough
|
|
|
|
* to prune? */
|
|
|
|
#define UTXO_PRUNE_DEPTH 144
|
2017-07-18 19:01:26 +02:00
|
|
|
|
2020-07-14 21:30:26 +02:00
|
|
|
/* 12 hours is usually enough reservation time */
|
|
|
|
#define RESERVATION_INC (6 * 12)
|
|
|
|
|
2018-02-26 14:57:13 +01:00
|
|
|
static void outpointfilters_init(struct wallet *w)
|
|
|
|
{
|
2021-01-07 14:27:54 +01:00
|
|
|
struct db_stmt *stmt;
|
2020-08-28 05:56:34 +02:00
|
|
|
struct utxo **utxos = wallet_get_utxos(NULL, w, OUTPUT_STATE_ANY);
|
2021-01-07 14:27:54 +01:00
|
|
|
struct bitcoin_txid txid;
|
|
|
|
u32 outnum;
|
2018-02-26 14:57:13 +01:00
|
|
|
|
|
|
|
w->owned_outpoints = outpointfilter_new(w);
|
|
|
|
for (size_t i = 0; i < tal_count(utxos); i++)
|
|
|
|
outpointfilter_add(w->owned_outpoints, &utxos[i]->txid, utxos[i]->outnum);
|
|
|
|
|
|
|
|
tal_free(utxos);
|
2021-01-07 14:27:54 +01:00
|
|
|
|
|
|
|
w->utxoset_outpoints = outpointfilter_new(w);
|
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db,
|
|
|
|
SQL("SELECT txid, outnum FROM utxoset WHERE spendheight is NULL"));
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
while (db_step(stmt)) {
|
|
|
|
db_column_sha256d(stmt, 0, &txid.shad);
|
|
|
|
outnum = db_column_int(stmt, 1);
|
|
|
|
outpointfilter_add(w->utxoset_outpoints, &txid, outnum);
|
|
|
|
}
|
|
|
|
tal_free(stmt);
|
2018-02-26 14:57:13 +01:00
|
|
|
}
|
|
|
|
|
2020-07-29 03:20:30 +02:00
|
|
|
struct wallet *wallet_new(struct lightningd *ld, struct timers *timers,
|
|
|
|
struct ext_key *bip32_base STEALS)
|
2017-05-23 22:07:20 +02:00
|
|
|
{
|
2018-02-12 11:10:17 +01:00
|
|
|
struct wallet *wallet = tal(ld, struct wallet);
|
|
|
|
wallet->ld = ld;
|
2019-11-18 01:27:17 +01:00
|
|
|
wallet->log = new_log(wallet, ld->log_book, NULL, "wallet");
|
2020-07-29 03:20:30 +02:00
|
|
|
wallet->bip32_base = tal_steal(wallet, bip32_base);
|
2020-04-14 15:36:58 +02:00
|
|
|
wallet->keyscan_gap = 50;
|
2018-01-17 21:29:49 +01:00
|
|
|
list_head_init(&wallet->unstored_payments);
|
2020-07-29 03:20:30 +02:00
|
|
|
wallet->db = db_setup(wallet, ld, wallet->bip32_base);
|
2018-02-26 14:57:13 +01:00
|
|
|
|
|
|
|
db_begin_transaction(wallet->db);
|
2019-11-18 01:27:17 +01:00
|
|
|
wallet->invoices = invoices_new(wallet, wallet->db, timers);
|
2018-02-26 14:57:13 +01:00
|
|
|
outpointfilters_init(wallet);
|
|
|
|
db_commit_transaction(wallet->db);
|
2017-05-23 22:07:20 +02:00
|
|
|
return wallet;
|
|
|
|
}
|
|
|
|
|
2020-07-06 07:28:43 +02:00
|
|
|
/**
|
|
|
|
* wallet_add_utxo - Register an UTXO which we (partially) own
|
|
|
|
*
|
|
|
|
* Add an UTXO to the set of outputs we care about.
|
|
|
|
*
|
|
|
|
* This can fail if we've already seen UTXO.
|
|
|
|
*/
|
|
|
|
static bool wallet_add_utxo(struct wallet *w, struct utxo *utxo,
|
|
|
|
enum wallet_output_type type)
|
2017-05-23 22:07:20 +02:00
|
|
|
{
|
2019-08-19 12:59:48 +02:00
|
|
|
struct db_stmt *stmt;
|
2017-10-18 14:54:33 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT * from outputs WHERE "
|
|
|
|
"prev_out_tx=? AND prev_out_index=?"));
|
|
|
|
db_bind_txid(stmt, 0, &utxo->txid);
|
|
|
|
db_bind_int(stmt, 1, utxo->outnum);
|
|
|
|
db_query_prepared(stmt);
|
2019-03-15 03:49:18 +01:00
|
|
|
|
|
|
|
/* If we get a result, that means a clash. */
|
2019-08-20 09:54:38 +02:00
|
|
|
if (db_step(stmt)) {
|
|
|
|
tal_free(stmt);
|
2019-03-15 03:49:18 +01:00
|
|
|
return false;
|
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2019-03-15 03:49:18 +01:00
|
|
|
|
2019-08-19 12:59:48 +02:00
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("INSERT INTO outputs ("
|
|
|
|
" prev_out_tx"
|
|
|
|
", prev_out_index"
|
|
|
|
", value"
|
|
|
|
", type"
|
|
|
|
", status"
|
|
|
|
", keyindex"
|
|
|
|
", channel_id"
|
|
|
|
", peer_id"
|
|
|
|
", commitment_point"
|
2020-08-14 03:30:42 +02:00
|
|
|
", option_anchor_outputs"
|
2019-08-19 12:59:48 +02:00
|
|
|
", confirmation_height"
|
|
|
|
", spend_height"
|
|
|
|
", scriptpubkey"
|
2020-08-14 03:30:42 +02:00
|
|
|
") VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?);"));
|
2019-08-19 12:59:48 +02:00
|
|
|
db_bind_txid(stmt, 0, &utxo->txid);
|
|
|
|
db_bind_int(stmt, 1, utxo->outnum);
|
|
|
|
db_bind_amount_sat(stmt, 2, &utxo->amount);
|
|
|
|
db_bind_int(stmt, 3, wallet_output_type_in_db(type));
|
2020-08-28 05:56:34 +02:00
|
|
|
db_bind_int(stmt, 4, OUTPUT_STATE_AVAILABLE);
|
2019-08-19 12:59:48 +02:00
|
|
|
db_bind_int(stmt, 5, utxo->keyindex);
|
2017-12-20 12:44:00 +01:00
|
|
|
if (utxo->close_info) {
|
2019-08-19 12:59:48 +02:00
|
|
|
db_bind_u64(stmt, 6, utxo->close_info->channel_id);
|
|
|
|
db_bind_node_id(stmt, 7, &utxo->close_info->peer_id);
|
2019-09-10 04:24:27 +02:00
|
|
|
if (utxo->close_info->commitment_point)
|
|
|
|
db_bind_pubkey(stmt, 8, utxo->close_info->commitment_point);
|
|
|
|
else
|
|
|
|
db_bind_null(stmt, 8);
|
2020-08-14 03:30:42 +02:00
|
|
|
db_bind_int(stmt, 9, utxo->close_info->option_anchor_outputs);
|
2017-12-20 12:44:00 +01:00
|
|
|
} else {
|
2019-08-19 12:59:48 +02:00
|
|
|
db_bind_null(stmt, 6);
|
|
|
|
db_bind_null(stmt, 7);
|
|
|
|
db_bind_null(stmt, 8);
|
2020-08-14 03:30:42 +02:00
|
|
|
db_bind_null(stmt, 9);
|
2017-12-20 12:44:00 +01:00
|
|
|
}
|
2018-02-26 11:36:48 +01:00
|
|
|
|
|
|
|
if (utxo->blockheight) {
|
2020-08-14 03:30:42 +02:00
|
|
|
db_bind_int(stmt, 10, *utxo->blockheight);
|
2018-02-26 11:36:48 +01:00
|
|
|
} else
|
2020-08-14 03:30:42 +02:00
|
|
|
db_bind_null(stmt, 10);
|
2018-02-26 11:36:48 +01:00
|
|
|
|
|
|
|
if (utxo->spendheight)
|
2020-08-14 03:30:42 +02:00
|
|
|
db_bind_int(stmt, 11, *utxo->spendheight);
|
2018-02-26 11:36:48 +01:00
|
|
|
else
|
2020-08-14 03:30:42 +02:00
|
|
|
db_bind_null(stmt, 11);
|
2018-02-26 11:36:48 +01:00
|
|
|
|
2020-08-14 03:30:42 +02:00
|
|
|
db_bind_blob(stmt, 12, utxo->scriptPubkey,
|
2020-07-29 04:08:17 +02:00
|
|
|
tal_bytelen(utxo->scriptPubkey));
|
2019-02-22 15:47:30 +01:00
|
|
|
|
2019-08-19 12:59:48 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2019-03-15 03:49:18 +01:00
|
|
|
return true;
|
2017-05-23 22:07:20 +02:00
|
|
|
}
|
2017-05-31 16:16:59 +02:00
|
|
|
|
2017-06-05 13:59:51 +02:00
|
|
|
/**
|
2018-02-08 22:43:01 +01:00
|
|
|
* wallet_stmt2output - Extract data from stmt and fill an UTXO
|
2017-06-05 13:59:51 +02:00
|
|
|
*/
|
2019-08-20 09:54:38 +02:00
|
|
|
static struct utxo *wallet_stmt2output(const tal_t *ctx, struct db_stmt *stmt)
|
2017-06-05 13:59:51 +02:00
|
|
|
{
|
2018-09-27 02:19:24 +02:00
|
|
|
struct utxo *utxo = tal(ctx, struct utxo);
|
2020-08-28 05:56:32 +02:00
|
|
|
u32 *blockheight, *spendheight;
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_txid(stmt, 0, &utxo->txid);
|
|
|
|
utxo->outnum = db_column_int(stmt, 1);
|
|
|
|
db_column_amount_sat(stmt, 2, &utxo->amount);
|
|
|
|
utxo->is_p2sh = db_column_int(stmt, 3) == p2sh_wpkh;
|
|
|
|
utxo->status = db_column_int(stmt, 4);
|
|
|
|
utxo->keyindex = db_column_int(stmt, 5);
|
|
|
|
if (!db_column_is_null(stmt, 6)) {
|
2017-12-20 12:44:00 +01:00
|
|
|
utxo->close_info = tal(utxo, struct unilateral_close_info);
|
2019-08-20 09:54:38 +02:00
|
|
|
utxo->close_info->channel_id = db_column_u64(stmt, 6);
|
|
|
|
db_column_node_id(stmt, 7, &utxo->close_info->peer_id);
|
2019-09-10 04:24:27 +02:00
|
|
|
if (!db_column_is_null(stmt, 8)) {
|
|
|
|
utxo->close_info->commitment_point
|
|
|
|
= tal(utxo->close_info, struct pubkey);
|
|
|
|
db_column_pubkey(stmt, 8,
|
|
|
|
utxo->close_info->commitment_point);
|
|
|
|
} else
|
|
|
|
utxo->close_info->commitment_point = NULL;
|
2020-08-14 03:30:42 +02:00
|
|
|
utxo->close_info->option_anchor_outputs
|
|
|
|
= db_column_int(stmt, 9);
|
2017-12-20 12:44:00 +01:00
|
|
|
} else {
|
|
|
|
utxo->close_info = NULL;
|
|
|
|
}
|
|
|
|
|
2020-07-29 04:08:17 +02:00
|
|
|
utxo->scriptPubkey =
|
2020-08-14 03:30:42 +02:00
|
|
|
tal_dup_arr(utxo, u8, db_column_blob(stmt, 12),
|
|
|
|
db_column_bytes(stmt, 12), 0);
|
2020-07-29 04:08:17 +02:00
|
|
|
|
2018-02-26 11:36:48 +01:00
|
|
|
utxo->blockheight = NULL;
|
|
|
|
utxo->spendheight = NULL;
|
|
|
|
|
2020-08-14 03:30:42 +02:00
|
|
|
if (!db_column_is_null(stmt, 10)) {
|
2018-03-22 00:13:16 +01:00
|
|
|
blockheight = tal(utxo, u32);
|
2020-08-14 03:30:42 +02:00
|
|
|
*blockheight = db_column_int(stmt, 10);
|
2018-02-26 11:36:48 +01:00
|
|
|
utxo->blockheight = blockheight;
|
|
|
|
}
|
|
|
|
|
2020-08-14 03:30:42 +02:00
|
|
|
if (!db_column_is_null(stmt, 11)) {
|
2018-03-22 00:13:16 +01:00
|
|
|
spendheight = tal(utxo, u32);
|
2020-08-14 03:30:42 +02:00
|
|
|
*spendheight = db_column_int(stmt, 11);
|
2018-02-26 11:36:48 +01:00
|
|
|
utxo->spendheight = spendheight;
|
|
|
|
}
|
|
|
|
|
2020-08-28 05:56:32 +02:00
|
|
|
/* This column can be null if 0.9.1 db or below. */
|
2020-08-28 05:56:34 +02:00
|
|
|
utxo->reserved_til = db_column_int_or_default(stmt, 13, 0);
|
2019-02-22 15:47:30 +01:00
|
|
|
|
2018-09-27 02:19:24 +02:00
|
|
|
return utxo;
|
2017-06-05 13:59:51 +02:00
|
|
|
}
|
|
|
|
|
2017-05-31 16:16:59 +02:00
|
|
|
bool wallet_update_output_status(struct wallet *w,
|
2017-12-18 07:41:52 +01:00
|
|
|
const struct bitcoin_txid *txid,
|
2017-05-31 16:16:59 +02:00
|
|
|
const u32 outnum, enum output_status oldstatus,
|
|
|
|
enum output_status newstatus)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
|
|
|
size_t changes;
|
2020-08-28 05:56:34 +02:00
|
|
|
if (oldstatus != OUTPUT_STATE_ANY) {
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("UPDATE outputs SET status=? WHERE status=? AND "
|
|
|
|
"prev_out_tx=? AND prev_out_index=?"));
|
|
|
|
db_bind_int(stmt, 0, output_status_in_db(newstatus));
|
|
|
|
db_bind_int(stmt, 1, output_status_in_db(oldstatus));
|
|
|
|
db_bind_txid(stmt, 2, txid);
|
|
|
|
db_bind_int(stmt, 3, outnum);
|
2017-05-31 16:16:59 +02:00
|
|
|
} else {
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db,
|
|
|
|
SQL("UPDATE outputs SET status=? WHERE "
|
|
|
|
"prev_out_tx=? AND prev_out_index=?"));
|
|
|
|
db_bind_int(stmt, 0, output_status_in_db(newstatus));
|
|
|
|
db_bind_txid(stmt, 1, txid);
|
|
|
|
db_bind_int(stmt, 2, outnum);
|
2017-05-31 16:16:59 +02:00
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(stmt);
|
|
|
|
changes = db_count_changes(stmt);
|
|
|
|
tal_free(stmt);
|
|
|
|
return changes > 0;
|
2017-05-31 16:16:59 +02:00
|
|
|
}
|
2017-06-05 13:59:51 +02:00
|
|
|
|
|
|
|
struct utxo **wallet_get_utxos(const tal_t *ctx, struct wallet *w, const enum output_status state)
|
|
|
|
{
|
|
|
|
struct utxo **results;
|
|
|
|
int i;
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
|
|
|
|
2020-08-28 05:56:34 +02:00
|
|
|
if (state == OUTPUT_STATE_ANY) {
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT"
|
|
|
|
" prev_out_tx"
|
|
|
|
", prev_out_index"
|
|
|
|
", value"
|
|
|
|
", type"
|
|
|
|
", status"
|
|
|
|
", keyindex"
|
|
|
|
", channel_id"
|
|
|
|
", peer_id"
|
|
|
|
", commitment_point"
|
2020-08-14 03:30:42 +02:00
|
|
|
", option_anchor_outputs"
|
2019-08-20 09:54:38 +02:00
|
|
|
", confirmation_height"
|
|
|
|
", spend_height"
|
|
|
|
", scriptpubkey "
|
2020-07-14 21:29:26 +02:00
|
|
|
", reserved_til "
|
2019-08-20 09:54:38 +02:00
|
|
|
"FROM outputs"));
|
|
|
|
} else {
|
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT"
|
|
|
|
" prev_out_tx"
|
|
|
|
", prev_out_index"
|
|
|
|
", value"
|
|
|
|
", type"
|
|
|
|
", status"
|
|
|
|
", keyindex"
|
|
|
|
", channel_id"
|
|
|
|
", peer_id"
|
|
|
|
", commitment_point"
|
2020-08-14 03:30:42 +02:00
|
|
|
", option_anchor_outputs"
|
2019-08-20 09:54:38 +02:00
|
|
|
", confirmation_height"
|
|
|
|
", spend_height"
|
|
|
|
", scriptpubkey "
|
2020-07-14 21:29:26 +02:00
|
|
|
", reserved_til "
|
2019-08-20 09:54:38 +02:00
|
|
|
"FROM outputs "
|
|
|
|
"WHERE status= ? "));
|
|
|
|
db_bind_int(stmt, 0, output_status_in_db(state));
|
|
|
|
}
|
|
|
|
db_query_prepared(stmt);
|
2017-06-05 13:59:51 +02:00
|
|
|
|
2018-08-09 04:27:25 +02:00
|
|
|
results = tal_arr(ctx, struct utxo*, 0);
|
2019-08-20 09:54:38 +02:00
|
|
|
for (i=0; db_step(stmt); i++) {
|
2018-09-27 02:19:24 +02:00
|
|
|
struct utxo *u = wallet_stmt2output(results, stmt);
|
2019-01-15 04:51:27 +01:00
|
|
|
tal_arr_expand(&results, u);
|
2017-06-05 13:59:51 +02:00
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2018-09-27 02:19:24 +02:00
|
|
|
|
2018-08-13 05:02:18 +02:00
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
struct utxo **wallet_get_unconfirmed_closeinfo_utxos(const tal_t *ctx,
|
|
|
|
struct wallet *w)
|
2018-08-13 05:02:18 +02:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-08-13 05:02:18 +02:00
|
|
|
struct utxo **results;
|
|
|
|
int i;
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT"
|
|
|
|
" prev_out_tx"
|
|
|
|
", prev_out_index"
|
|
|
|
", value"
|
|
|
|
", type"
|
|
|
|
", status"
|
|
|
|
", keyindex"
|
|
|
|
", channel_id"
|
|
|
|
", peer_id"
|
|
|
|
", commitment_point"
|
2020-08-14 03:30:42 +02:00
|
|
|
", option_anchor_outputs"
|
2019-08-20 09:54:38 +02:00
|
|
|
", confirmation_height"
|
|
|
|
", spend_height"
|
|
|
|
", scriptpubkey"
|
2020-07-14 21:29:26 +02:00
|
|
|
", reserved_til"
|
2019-08-20 09:54:38 +02:00
|
|
|
" FROM outputs"
|
|
|
|
" WHERE channel_id IS NOT NULL AND "
|
|
|
|
"confirmation_height IS NULL"));
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
results = tal_arr(ctx, struct utxo *, 0);
|
|
|
|
for (i = 0; db_step(stmt); i++) {
|
2018-09-27 02:19:24 +02:00
|
|
|
struct utxo *u = wallet_stmt2output(results, stmt);
|
2019-01-15 04:51:27 +01:00
|
|
|
tal_arr_expand(&results, u);
|
2018-08-13 05:02:18 +02:00
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2017-06-05 13:59:51 +02:00
|
|
|
|
|
|
|
return results;
|
|
|
|
}
|
2017-06-13 15:42:31 +02:00
|
|
|
|
2020-04-03 23:56:57 +02:00
|
|
|
struct utxo *wallet_utxo_get(const tal_t *ctx, struct wallet *w,
|
|
|
|
const struct bitcoin_txid *txid,
|
|
|
|
u32 outnum)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
struct utxo *utxo;
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT"
|
|
|
|
" prev_out_tx"
|
|
|
|
", prev_out_index"
|
|
|
|
", value"
|
|
|
|
", type"
|
|
|
|
", status"
|
|
|
|
", keyindex"
|
|
|
|
", channel_id"
|
|
|
|
", peer_id"
|
|
|
|
", commitment_point"
|
2020-08-14 03:30:42 +02:00
|
|
|
", option_anchor_outputs"
|
2020-04-03 23:56:57 +02:00
|
|
|
", confirmation_height"
|
|
|
|
", spend_height"
|
|
|
|
", scriptpubkey"
|
2020-07-14 21:29:26 +02:00
|
|
|
", reserved_til"
|
2020-04-03 23:56:57 +02:00
|
|
|
" FROM outputs"
|
|
|
|
" WHERE prev_out_tx = ?"
|
|
|
|
" AND prev_out_index = ?"));
|
|
|
|
|
|
|
|
db_bind_sha256d(stmt, 0, &txid->shad);
|
|
|
|
db_bind_int(stmt, 1, outnum);
|
|
|
|
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
if (!db_step(stmt)) {
|
|
|
|
tal_free(stmt);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
utxo = wallet_stmt2output(ctx, stmt);
|
|
|
|
tal_free(stmt);
|
|
|
|
|
|
|
|
return utxo;
|
|
|
|
}
|
|
|
|
|
2020-06-10 01:41:51 +02:00
|
|
|
bool wallet_unreserve_output(struct wallet *w,
|
|
|
|
const struct bitcoin_txid *txid,
|
|
|
|
const u32 outnum)
|
|
|
|
{
|
|
|
|
return wallet_update_output_status(w, txid, outnum,
|
2020-08-28 05:56:34 +02:00
|
|
|
OUTPUT_STATE_RESERVED,
|
|
|
|
OUTPUT_STATE_AVAILABLE);
|
2020-06-10 01:41:51 +02:00
|
|
|
}
|
|
|
|
|
2017-06-13 15:42:31 +02:00
|
|
|
/**
|
|
|
|
* unreserve_utxo - Mark a reserved UTXO as available again
|
|
|
|
*/
|
|
|
|
static void unreserve_utxo(struct wallet *w, const struct utxo *unres)
|
|
|
|
{
|
|
|
|
if (!wallet_update_output_status(w, &unres->txid, unres->outnum,
|
2020-08-28 05:56:34 +02:00
|
|
|
OUTPUT_STATE_RESERVED,
|
|
|
|
OUTPUT_STATE_AVAILABLE)) {
|
2017-11-01 02:21:56 +01:00
|
|
|
fatal("Unable to unreserve output");
|
2017-06-13 15:42:31 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* destroy_utxos - Destructor for an array of pointers to utxo
|
|
|
|
*/
|
|
|
|
static void destroy_utxos(const struct utxo **utxos, struct wallet *w)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < tal_count(utxos); i++)
|
|
|
|
unreserve_utxo(w, utxos[i]);
|
|
|
|
}
|
|
|
|
|
2020-06-10 01:41:51 +02:00
|
|
|
void wallet_persist_utxo_reservation(struct wallet *w, const struct utxo **utxos)
|
|
|
|
{
|
|
|
|
tal_del_destructor2(utxos, destroy_utxos, w);
|
|
|
|
}
|
|
|
|
|
2017-06-13 15:42:31 +02:00
|
|
|
void wallet_confirm_utxos(struct wallet *w, const struct utxo **utxos)
|
|
|
|
{
|
|
|
|
tal_del_destructor2(utxos, destroy_utxos, w);
|
|
|
|
for (size_t i = 0; i < tal_count(utxos); i++) {
|
|
|
|
if (!wallet_update_output_status(
|
|
|
|
w, &utxos[i]->txid, utxos[i]->outnum,
|
2020-08-28 05:56:34 +02:00
|
|
|
OUTPUT_STATE_RESERVED, OUTPUT_STATE_SPENT)) {
|
2017-11-01 02:21:56 +01:00
|
|
|
fatal("Unable to mark output as spent");
|
2017-06-13 15:42:31 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-14 21:30:26 +02:00
|
|
|
static void db_set_utxo(struct db *db, const struct utxo *utxo)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
|
2020-08-28 05:56:34 +02:00
|
|
|
if (utxo->status == OUTPUT_STATE_RESERVED)
|
2020-07-14 21:30:26 +02:00
|
|
|
assert(utxo->reserved_til);
|
|
|
|
else
|
|
|
|
assert(!utxo->reserved_til);
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(
|
fix: broken SQL statement in wallet db_set_utxo
I discovered this accidentally when using the `tests/plugins/dblog.py`
plugin on another testcase: tests/test_connection.py::test_fail_unconfirmed
There the plugin/hook crashes because it can't execute the statement:
```json
{
"jsonrpc": "2.0",
"id": 34,
"error": {
"code": -32600,
"message": "Error while processing db_write: unrecognized token: \"174WHERE\"",
"traceback": "Traceback (most recent call last):\n File \"/home/will/projects/lightning.git/contrib/pyln-client/pyln/client/plugin.py\", line 535, in _dispatch_request\n result = self._exec_func(method.func, request)\n File \"/home/will/projects/lightning.git/contrib/pyln-client/pyln/client/plugin.py\", line 520, in _exec_func\n return func(*ba.args, **ba.kwargs)\n File \"/home/will/projects/lightning.git/tests/plugins/dblog.py\", line 45, in db_write\n plugin.conn.execute(c)\nsqlite3.OperationalError: unrecognized token: \"174WHERE\"\n"
}
}
```
Changelog-Fixed: plugin: Regression with SQL statement expansion that could result in invalid statements being passed to the `db_write` hook.
2020-09-24 14:43:31 +02:00
|
|
|
db, SQL("UPDATE outputs SET status=?, reserved_til=? "
|
2020-07-14 21:30:26 +02:00
|
|
|
"WHERE prev_out_tx=? AND prev_out_index=?"));
|
|
|
|
db_bind_int(stmt, 0, output_status_in_db(utxo->status));
|
2020-08-28 05:56:34 +02:00
|
|
|
db_bind_int(stmt, 1, utxo->reserved_til);
|
2020-07-14 21:30:26 +02:00
|
|
|
db_bind_txid(stmt, 2, &utxo->txid);
|
|
|
|
db_bind_int(stmt, 3, utxo->outnum);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool wallet_reserve_utxo(struct wallet *w, struct utxo *utxo, u32 current_height)
|
|
|
|
{
|
|
|
|
switch (utxo->status) {
|
2020-08-28 05:56:34 +02:00
|
|
|
case OUTPUT_STATE_SPENT:
|
2020-07-14 21:30:26 +02:00
|
|
|
return false;
|
2020-08-28 05:56:34 +02:00
|
|
|
case OUTPUT_STATE_AVAILABLE:
|
|
|
|
case OUTPUT_STATE_RESERVED:
|
2020-07-14 21:30:26 +02:00
|
|
|
break;
|
2020-08-28 05:56:34 +02:00
|
|
|
case OUTPUT_STATE_ANY:
|
2020-07-14 21:30:26 +02:00
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We simple increase existing reservations, which DTRT if we unreserve */
|
2020-08-28 05:56:34 +02:00
|
|
|
if (utxo->reserved_til >= current_height)
|
|
|
|
utxo->reserved_til += RESERVATION_INC;
|
2020-07-14 21:30:26 +02:00
|
|
|
else
|
2020-08-28 05:56:34 +02:00
|
|
|
utxo->reserved_til = current_height + RESERVATION_INC;
|
2020-07-14 21:30:26 +02:00
|
|
|
|
2020-08-28 05:56:34 +02:00
|
|
|
utxo->status = OUTPUT_STATE_RESERVED;
|
2020-07-14 21:30:26 +02:00
|
|
|
|
|
|
|
db_set_utxo(w->db, utxo);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void wallet_unreserve_utxo(struct wallet *w, struct utxo *utxo, u32 current_height)
|
|
|
|
{
|
2020-08-28 05:56:34 +02:00
|
|
|
if (utxo->status != OUTPUT_STATE_RESERVED)
|
2020-07-14 21:30:26 +02:00
|
|
|
fatal("UTXO %s:%u is not reserved",
|
|
|
|
type_to_string(tmpctx, struct bitcoin_txid, &utxo->txid),
|
|
|
|
utxo->outnum);
|
|
|
|
|
2020-08-28 05:56:34 +02:00
|
|
|
if (utxo->reserved_til <= current_height + RESERVATION_INC) {
|
2020-08-28 05:56:34 +02:00
|
|
|
utxo->status = OUTPUT_STATE_AVAILABLE;
|
2020-08-28 05:56:34 +02:00
|
|
|
utxo->reserved_til = 0;
|
2020-07-14 21:30:26 +02:00
|
|
|
} else
|
2020-08-28 05:56:34 +02:00
|
|
|
utxo->reserved_til -= RESERVATION_INC;
|
2020-07-14 21:30:26 +02:00
|
|
|
|
|
|
|
db_set_utxo(w->db, utxo);
|
|
|
|
}
|
|
|
|
|
2020-07-15 07:30:49 +02:00
|
|
|
static bool excluded(const struct utxo **excludes,
|
|
|
|
const struct utxo *utxo)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < tal_count(excludes); i++) {
|
|
|
|
if (bitcoin_txid_eq(&excludes[i]->txid, &utxo->txid)
|
|
|
|
&& excludes[i]->outnum == utxo->outnum)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool deep_enough(u32 maxheight, const struct utxo *utxo)
|
|
|
|
{
|
|
|
|
/* If we require confirmations check that we have a
|
|
|
|
* confirmation height and that it is below the required
|
|
|
|
* maxheight (current_height - minconf) */
|
|
|
|
if (maxheight == 0)
|
|
|
|
return true;
|
|
|
|
if (!utxo->blockheight)
|
|
|
|
return false;
|
|
|
|
return *utxo->blockheight <= maxheight;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME: Make this wallet_find_utxos, and branch and bound and I've
|
|
|
|
* left that to @niftynei to do, who actually read the paper! */
|
|
|
|
struct utxo *wallet_find_utxo(const tal_t *ctx, struct wallet *w,
|
|
|
|
unsigned current_blockheight,
|
|
|
|
struct amount_sat *amount_hint,
|
|
|
|
unsigned feerate_per_kw,
|
|
|
|
u32 maxheight,
|
|
|
|
const struct utxo **excludes)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
struct utxo *utxo;
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT"
|
|
|
|
" prev_out_tx"
|
|
|
|
", prev_out_index"
|
|
|
|
", value"
|
|
|
|
", type"
|
|
|
|
", status"
|
|
|
|
", keyindex"
|
|
|
|
", channel_id"
|
|
|
|
", peer_id"
|
|
|
|
", commitment_point"
|
2020-08-14 03:30:42 +02:00
|
|
|
", option_anchor_outputs"
|
2020-07-15 07:30:49 +02:00
|
|
|
", confirmation_height"
|
|
|
|
", spend_height"
|
|
|
|
", scriptpubkey "
|
|
|
|
", reserved_til"
|
|
|
|
" FROM outputs"
|
|
|
|
" WHERE status = ?"
|
|
|
|
" OR (status = ? AND reserved_til <= ?)"
|
|
|
|
"ORDER BY RANDOM();"));
|
2020-08-28 05:56:34 +02:00
|
|
|
db_bind_int(stmt, 0, output_status_in_db(OUTPUT_STATE_AVAILABLE));
|
|
|
|
db_bind_int(stmt, 1, output_status_in_db(OUTPUT_STATE_RESERVED));
|
2020-07-15 07:30:49 +02:00
|
|
|
db_bind_u64(stmt, 2, current_blockheight);
|
|
|
|
|
|
|
|
/* FIXME: Use feerate + estimate of input cost to establish
|
|
|
|
* range for amount_hint */
|
|
|
|
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
utxo = NULL;
|
|
|
|
while (!utxo && db_step(stmt)) {
|
|
|
|
utxo = wallet_stmt2output(ctx, stmt);
|
|
|
|
if (excluded(excludes, utxo) || !deep_enough(maxheight, utxo))
|
|
|
|
utxo = tal_free(utxo);
|
|
|
|
|
|
|
|
}
|
|
|
|
tal_free(stmt);
|
|
|
|
return utxo;
|
|
|
|
}
|
|
|
|
|
2020-07-06 07:28:43 +02:00
|
|
|
bool wallet_add_onchaind_utxo(struct wallet *w,
|
|
|
|
const struct bitcoin_txid *txid,
|
|
|
|
u32 outnum,
|
|
|
|
const u8 *scriptpubkey,
|
|
|
|
u32 blockheight,
|
|
|
|
struct amount_sat amount,
|
|
|
|
const struct channel *channel,
|
|
|
|
/* NULL if option_static_remotekey */
|
|
|
|
const struct pubkey *commitment_point)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT * from outputs WHERE "
|
|
|
|
"prev_out_tx=? AND prev_out_index=?"));
|
|
|
|
db_bind_txid(stmt, 0, txid);
|
|
|
|
db_bind_int(stmt, 1, outnum);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
/* If we get a result, that means a clash. */
|
|
|
|
if (db_step(stmt)) {
|
|
|
|
tal_free(stmt);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
tal_free(stmt);
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("INSERT INTO outputs ("
|
|
|
|
" prev_out_tx"
|
|
|
|
", prev_out_index"
|
|
|
|
", value"
|
|
|
|
", type"
|
|
|
|
", status"
|
|
|
|
", keyindex"
|
|
|
|
", channel_id"
|
|
|
|
", peer_id"
|
|
|
|
", commitment_point"
|
2020-08-14 03:30:42 +02:00
|
|
|
", option_anchor_outputs"
|
2020-07-06 07:28:43 +02:00
|
|
|
", confirmation_height"
|
|
|
|
", spend_height"
|
|
|
|
", scriptpubkey"
|
2020-08-14 03:30:42 +02:00
|
|
|
") VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?);"));
|
2020-07-06 07:28:43 +02:00
|
|
|
db_bind_txid(stmt, 0, txid);
|
|
|
|
db_bind_int(stmt, 1, outnum);
|
|
|
|
db_bind_amount_sat(stmt, 2, &amount);
|
|
|
|
db_bind_int(stmt, 3, wallet_output_type_in_db(p2wpkh));
|
2020-08-28 05:56:34 +02:00
|
|
|
db_bind_int(stmt, 4, OUTPUT_STATE_AVAILABLE);
|
2020-07-06 07:28:43 +02:00
|
|
|
db_bind_int(stmt, 5, 0);
|
|
|
|
db_bind_u64(stmt, 6, channel->dbid);
|
|
|
|
db_bind_node_id(stmt, 7, &channel->peer->id);
|
|
|
|
if (commitment_point)
|
|
|
|
db_bind_pubkey(stmt, 8, commitment_point);
|
|
|
|
else
|
|
|
|
db_bind_null(stmt, 8);
|
|
|
|
|
2020-08-14 03:30:42 +02:00
|
|
|
db_bind_int(stmt, 9, channel->option_anchor_outputs);
|
|
|
|
db_bind_int(stmt, 10, blockheight);
|
2020-07-06 07:28:43 +02:00
|
|
|
|
|
|
|
/* spendheight */
|
2020-08-14 03:30:42 +02:00
|
|
|
db_bind_null(stmt, 11);
|
|
|
|
db_bind_blob(stmt, 12, scriptpubkey, tal_bytelen(scriptpubkey));
|
2020-07-06 07:28:43 +02:00
|
|
|
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-06-21 17:40:42 +02:00
|
|
|
bool wallet_can_spend(struct wallet *w, const u8 *script,
|
|
|
|
u32 *index, bool *output_is_p2sh)
|
|
|
|
{
|
|
|
|
struct ext_key ext;
|
|
|
|
u64 bip32_max_index = db_get_intvar(w->db, "bip32_max_index", 0);
|
|
|
|
u32 i;
|
|
|
|
|
|
|
|
/* If not one of these, can't be for us. */
|
2017-10-26 04:58:19 +02:00
|
|
|
if (is_p2sh(script, NULL))
|
2017-06-21 17:40:42 +02:00
|
|
|
*output_is_p2sh = true;
|
2017-10-26 04:58:19 +02:00
|
|
|
else if (is_p2wpkh(script, NULL))
|
2017-06-21 17:40:42 +02:00
|
|
|
*output_is_p2sh = false;
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
|
2020-04-14 15:36:58 +02:00
|
|
|
for (i = 0; i <= bip32_max_index + w->keyscan_gap; i++) {
|
2017-06-21 17:40:42 +02:00
|
|
|
u8 *s;
|
|
|
|
|
|
|
|
if (bip32_key_from_parent(w->bip32_base, i,
|
|
|
|
BIP32_FLAG_KEY_PUBLIC, &ext)
|
|
|
|
!= WALLY_OK) {
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
s = scriptpubkey_p2wpkh_derkey(w, ext.pub_key);
|
|
|
|
if (*output_is_p2sh) {
|
|
|
|
u8 *p2sh = scriptpubkey_p2sh(w, s);
|
|
|
|
tal_free(s);
|
|
|
|
s = p2sh;
|
|
|
|
}
|
|
|
|
if (scripteq(s, script)) {
|
2020-04-14 15:36:58 +02:00
|
|
|
/* If we found a used key in the keyscan_gap we should
|
|
|
|
* remember that. */
|
|
|
|
if (i > bip32_max_index)
|
|
|
|
db_set_intvar(w->db, "bip32_max_index", i);
|
2017-06-21 17:40:42 +02:00
|
|
|
tal_free(s);
|
|
|
|
*index = i;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
tal_free(s);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-06-25 04:33:13 +02:00
|
|
|
s64 wallet_get_newindex(struct lightningd *ld)
|
|
|
|
{
|
|
|
|
u64 newidx = db_get_intvar(ld->wallet->db, "bip32_max_index", 0) + 1;
|
|
|
|
|
|
|
|
if (newidx == BIP32_INITIAL_HARDENED_CHILD)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
db_set_intvar(ld->wallet->db, "bip32_max_index", newidx);
|
|
|
|
return newidx;
|
|
|
|
}
|
2017-07-18 19:01:26 +02:00
|
|
|
|
2018-02-18 13:54:46 +01:00
|
|
|
static void wallet_shachain_init(struct wallet *wallet,
|
|
|
|
struct wallet_shachain *chain)
|
2017-07-18 19:01:26 +02:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-02-18 13:54:46 +01:00
|
|
|
|
|
|
|
assert(chain->id == 0);
|
|
|
|
|
2017-07-18 19:01:26 +02:00
|
|
|
/* Create shachain */
|
|
|
|
shachain_init(&chain->chain);
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(
|
|
|
|
wallet->db,
|
|
|
|
SQL("INSERT INTO shachains (min_index, num_valid) VALUES (?, 0);"));
|
|
|
|
db_bind_u64(stmt, 0, chain->chain.min_index);
|
|
|
|
db_exec_prepared_v2(stmt);
|
2017-10-18 15:34:51 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
chain->id = db_last_insert_id_v2(stmt);
|
|
|
|
tal_free(stmt);
|
2017-07-18 19:01:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO(cdecker) Stolen from shachain, move to some appropriate location */
|
2017-08-18 06:43:53 +02:00
|
|
|
static unsigned int count_trailing_zeroes(uint64_t index)
|
2017-07-18 19:01:26 +02:00
|
|
|
{
|
|
|
|
#if HAVE_BUILTIN_CTZLL
|
|
|
|
return index ? (unsigned int)__builtin_ctzll(index) : SHACHAIN_BITS;
|
|
|
|
#else
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
for (i = 0; i < SHACHAIN_BITS; i++) {
|
|
|
|
if (index & (1ULL << i))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return i;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
bool wallet_shachain_add_hash(struct wallet *wallet,
|
|
|
|
struct wallet_shachain *chain,
|
2017-08-18 06:43:53 +02:00
|
|
|
uint64_t index,
|
2018-07-09 13:17:58 +02:00
|
|
|
const struct secret *hash)
|
2017-07-18 19:01:26 +02:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2017-07-18 19:01:26 +02:00
|
|
|
u32 pos = count_trailing_zeroes(index);
|
2018-07-09 13:17:58 +02:00
|
|
|
struct sha256 s;
|
2019-09-12 22:46:34 +02:00
|
|
|
bool updated;
|
2018-07-09 13:17:58 +02:00
|
|
|
|
|
|
|
BUILD_ASSERT(sizeof(s) == sizeof(*hash));
|
|
|
|
memcpy(&s, hash, sizeof(s));
|
|
|
|
|
2017-07-18 19:01:26 +02:00
|
|
|
assert(index < SQLITE_MAX_UINT);
|
2018-07-09 13:17:58 +02:00
|
|
|
if (!shachain_add_hash(&chain->chain, index, &s)) {
|
2017-07-18 19:01:26 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(
|
|
|
|
wallet->db,
|
|
|
|
SQL("UPDATE shachains SET num_valid=?, min_index=? WHERE id=?"));
|
|
|
|
db_bind_int(stmt, 0, chain->chain.num_valid);
|
|
|
|
db_bind_u64(stmt, 1, index);
|
|
|
|
db_bind_u64(stmt, 2, chain->id);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2017-10-18 15:34:51 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(wallet->db,
|
2019-09-12 22:46:34 +02:00
|
|
|
SQL("UPDATE shachain_known SET idx=?, hash=? "
|
|
|
|
"WHERE shachain_id=? AND pos=?"));
|
|
|
|
db_bind_u64(stmt, 0, index);
|
|
|
|
db_bind_secret(stmt, 1, hash);
|
|
|
|
db_bind_u64(stmt, 2, chain->id);
|
|
|
|
db_bind_int(stmt, 3, pos);
|
|
|
|
db_exec_prepared_v2(stmt);
|
|
|
|
updated = db_count_changes(stmt) == 1;
|
|
|
|
tal_free(stmt);
|
|
|
|
|
|
|
|
if (!updated) {
|
|
|
|
stmt = db_prepare_v2(
|
|
|
|
wallet->db, SQL("INSERT INTO shachain_known (shachain_id, "
|
|
|
|
"pos, idx, hash) VALUES (?, ?, ?, ?);"));
|
|
|
|
db_bind_u64(stmt, 0, chain->id);
|
|
|
|
db_bind_int(stmt, 1, pos);
|
|
|
|
db_bind_u64(stmt, 2, index);
|
|
|
|
db_bind_secret(stmt, 3, hash);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
|
|
|
}
|
2017-07-18 19:01:26 +02:00
|
|
|
|
2017-11-01 02:10:48 +01:00
|
|
|
return true;
|
2017-07-18 19:01:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool wallet_shachain_load(struct wallet *wallet, u64 id,
|
|
|
|
struct wallet_shachain *chain)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2017-07-18 19:01:26 +02:00
|
|
|
chain->id = id;
|
|
|
|
shachain_init(&chain->chain);
|
|
|
|
|
|
|
|
/* Load shachain metadata */
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(
|
|
|
|
wallet->db,
|
|
|
|
SQL("SELECT min_index, num_valid FROM shachains WHERE id=?"));
|
|
|
|
db_bind_u64(stmt, 0, id);
|
|
|
|
db_query_prepared(stmt);
|
2017-10-18 15:34:51 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_step(stmt)) {
|
|
|
|
tal_free(stmt);
|
2017-07-18 19:01:26 +02:00
|
|
|
return false;
|
2019-08-20 09:54:38 +02:00
|
|
|
}
|
2017-07-18 19:01:26 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
chain->chain.min_index = db_column_u64(stmt, 0);
|
|
|
|
chain->chain.num_valid = db_column_u64(stmt, 1);
|
|
|
|
tal_free(stmt);
|
2017-07-18 19:01:26 +02:00
|
|
|
|
|
|
|
/* Load shachain known entries */
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(wallet->db,
|
|
|
|
SQL("SELECT idx, hash, pos FROM shachain_known "
|
|
|
|
"WHERE shachain_id=?"));
|
|
|
|
db_bind_u64(stmt, 0, id);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
while (db_step(stmt)) {
|
|
|
|
int pos = db_column_int(stmt, 2);
|
|
|
|
chain->chain.known[pos].index = db_column_u64(stmt, 0);
|
|
|
|
db_column_sha256(stmt, 1, &chain->chain.known[pos].hash);
|
|
|
|
}
|
|
|
|
tal_free(stmt);
|
2017-07-18 19:01:26 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-02-21 16:37:08 +01:00
|
|
|
static struct peer *wallet_peer_load(struct wallet *w, const u64 dbid)
|
2017-07-20 16:42:55 +02:00
|
|
|
{
|
2017-12-18 11:17:15 +01:00
|
|
|
const unsigned char *addrstr;
|
2019-08-20 09:54:38 +02:00
|
|
|
struct peer *peer = NULL;
|
2019-04-08 11:58:32 +02:00
|
|
|
struct node_id id;
|
2018-10-26 08:01:26 +02:00
|
|
|
struct wireaddr_internal addr;
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2017-12-18 11:17:15 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("SELECT id, node_id, address FROM peers WHERE id=?;"));
|
|
|
|
db_bind_u64(stmt, 0, dbid);
|
|
|
|
db_query_prepared(stmt);
|
2017-12-18 11:17:15 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_step(stmt))
|
|
|
|
goto done;
|
2019-03-15 03:50:18 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (db_column_is_null(stmt, 1))
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
db_column_node_id(stmt, 1, &id);
|
|
|
|
|
|
|
|
addrstr = db_column_text(stmt, 2);
|
|
|
|
if (!parse_wireaddr_internal((const char*)addrstr, &addr, DEFAULT_PORT, false, false, true, NULL))
|
|
|
|
goto done;
|
2017-12-18 11:17:15 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
peer = new_peer(w->ld, db_column_u64(stmt, 0), &id, &addr);
|
2017-12-18 11:17:15 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
done:
|
|
|
|
tal_free(stmt);
|
2018-02-12 11:12:55 +01:00
|
|
|
return peer;
|
2017-07-20 16:42:55 +02:00
|
|
|
}
|
|
|
|
|
2020-08-13 19:45:02 +02:00
|
|
|
static struct bitcoin_signature *
|
|
|
|
wallet_htlc_sigs_load(const tal_t *ctx, struct wallet *w, u64 channelid,
|
|
|
|
bool option_anchor_outputs)
|
2018-02-09 16:40:59 +01:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2020-08-13 19:45:02 +02:00
|
|
|
struct bitcoin_signature *htlc_sigs = tal_arr(ctx, struct bitcoin_signature, 0);
|
2019-08-20 09:54:38 +02:00
|
|
|
|
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("SELECT signature FROM htlc_sigs WHERE channelid = ?"));
|
|
|
|
db_bind_u64(stmt, 0, channelid);
|
|
|
|
db_query_prepared(stmt);
|
2018-02-09 16:40:59 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
while (db_step(stmt)) {
|
2020-08-13 19:45:02 +02:00
|
|
|
struct bitcoin_signature sig;
|
|
|
|
db_column_signature(stmt, 0, &sig.s);
|
|
|
|
/* BOLT-a12da24dd0102c170365124782b46d9710950ac1 #3:
|
|
|
|
* ## HTLC-Timeout and HTLC-Success Transactions
|
|
|
|
*...
|
|
|
|
* * if `option_anchor_outputs` applies to this commitment
|
|
|
|
* transaction, `SIGHASH_SINGLE|SIGHASH_ANYONECANPAY` is
|
|
|
|
* used.
|
|
|
|
*/
|
|
|
|
if (option_anchor_outputs)
|
|
|
|
sig.sighash_type = SIGHASH_SINGLE|SIGHASH_ANYONECANPAY;
|
|
|
|
else
|
|
|
|
sig.sighash_type = SIGHASH_ALL;
|
2019-01-15 04:51:27 +01:00
|
|
|
tal_arr_expand(&htlc_sigs, sig);
|
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2018-09-27 02:19:24 +02:00
|
|
|
|
|
|
|
log_debug(w->log, "Loaded %zu HTLC signatures from DB",
|
|
|
|
tal_count(htlc_sigs));
|
2018-02-09 16:40:59 +01:00
|
|
|
return htlc_sigs;
|
|
|
|
}
|
|
|
|
|
2019-05-15 17:21:26 +02:00
|
|
|
bool wallet_remote_ann_sigs_load(const tal_t *ctx, struct wallet *w, u64 id,
|
|
|
|
secp256k1_ecdsa_signature **remote_ann_node_sig,
|
|
|
|
secp256k1_ecdsa_signature **remote_ann_bitcoin_sig)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
|
|
|
bool res;
|
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("SELECT remote_ann_node_sig, remote_ann_bitcoin_sig"
|
|
|
|
" FROM channels WHERE id = ?"));
|
|
|
|
db_bind_u64(stmt, 0, id);
|
|
|
|
db_query_prepared(stmt);
|
2019-05-15 17:21:26 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
res = db_step(stmt);
|
2019-05-15 17:21:26 +02:00
|
|
|
|
|
|
|
/* This must succeed, since we know the channel exists */
|
2019-08-20 09:54:38 +02:00
|
|
|
assert(res);
|
2019-05-15 17:21:26 +02:00
|
|
|
|
|
|
|
/* if only one sig exists, forget the sig and hope peer send new ones*/
|
2019-08-20 09:54:38 +02:00
|
|
|
if (db_column_is_null(stmt, 0) || db_column_is_null(stmt, 1)) {
|
2019-05-15 17:21:26 +02:00
|
|
|
*remote_ann_node_sig = *remote_ann_bitcoin_sig = NULL;
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2019-05-15 17:21:26 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* the case left over is both sigs exist */
|
|
|
|
*remote_ann_node_sig = tal(ctx, secp256k1_ecdsa_signature);
|
|
|
|
*remote_ann_bitcoin_sig = tal(ctx, secp256k1_ecdsa_signature);
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_column_signature(stmt, 0, *remote_ann_node_sig))
|
2019-05-15 17:21:26 +02:00
|
|
|
goto fail;
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_column_signature(stmt, 1, *remote_ann_bitcoin_sig))
|
2019-05-15 17:21:26 +02:00
|
|
|
goto fail;
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2019-05-15 17:21:26 +02:00
|
|
|
return true;
|
|
|
|
|
|
|
|
fail:
|
|
|
|
*remote_ann_node_sig = tal_free(*remote_ann_node_sig);
|
|
|
|
*remote_ann_bitcoin_sig = tal_free(*remote_ann_bitcoin_sig);
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2019-05-15 17:21:26 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-12-12 18:18:25 +01:00
|
|
|
static struct fee_states *wallet_channel_fee_states_load(struct wallet *w,
|
|
|
|
const u64 id,
|
2019-09-09 18:11:24 +02:00
|
|
|
enum side opener)
|
2019-12-12 18:18:25 +01:00
|
|
|
{
|
|
|
|
struct fee_states *fee_states;
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT hstate, feerate_per_kw FROM channel_feerates WHERE channel_id = ?"));
|
|
|
|
db_bind_u64(stmt, 0, id);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
/* Start with blank slate. */
|
2019-09-09 18:11:24 +02:00
|
|
|
fee_states = new_fee_states(w, opener, NULL);
|
2019-12-12 18:18:25 +01:00
|
|
|
while (db_step(stmt)) {
|
|
|
|
enum htlc_state hstate = db_column_int(stmt, 0);
|
|
|
|
u32 feerate = db_column_int(stmt, 1);
|
|
|
|
|
|
|
|
if (fee_states->feerate[hstate] != NULL) {
|
|
|
|
log_broken(w->log,
|
|
|
|
"duplicate channel_feerates for %s id %"PRIu64,
|
|
|
|
htlc_state_name(hstate), id);
|
|
|
|
fee_states = tal_free(fee_states);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
fee_states->feerate[hstate] = tal_dup(fee_states, u32, &feerate);
|
|
|
|
}
|
|
|
|
tal_free(stmt);
|
|
|
|
|
2019-09-09 18:11:24 +02:00
|
|
|
if (fee_states && !fee_states_valid(fee_states, opener)) {
|
2019-12-12 18:18:25 +01:00
|
|
|
log_broken(w->log,
|
|
|
|
"invalid channel_feerates for id %"PRIu64, id);
|
|
|
|
fee_states = tal_free(fee_states);
|
|
|
|
}
|
|
|
|
return fee_states;
|
|
|
|
}
|
|
|
|
|
2017-07-20 16:42:55 +02:00
|
|
|
/**
|
2019-08-20 09:54:38 +02:00
|
|
|
* wallet_stmt2channel - Helper to populate a wallet_channel from a `db_stmt`
|
2017-07-20 16:42:55 +02:00
|
|
|
*/
|
2019-08-20 09:54:38 +02:00
|
|
|
static struct channel *wallet_stmt2channel(struct wallet *w, struct db_stmt *stmt)
|
2017-07-20 16:42:55 +02:00
|
|
|
{
|
|
|
|
bool ok = true;
|
2018-02-19 02:06:14 +01:00
|
|
|
struct channel_info channel_info;
|
2020-09-17 03:58:59 +02:00
|
|
|
struct fee_states *fee_states;
|
2018-02-19 02:06:14 +01:00
|
|
|
struct short_channel_id *scid;
|
2020-09-09 09:20:53 +02:00
|
|
|
struct channel_id cid;
|
2018-02-12 11:12:55 +01:00
|
|
|
struct channel *chan;
|
|
|
|
u64 peer_dbid;
|
|
|
|
struct peer *peer;
|
2018-02-19 02:06:14 +01:00
|
|
|
struct wallet_shachain wshachain;
|
|
|
|
struct channel_config our_config;
|
|
|
|
struct bitcoin_txid funding_txid;
|
2018-12-03 00:15:06 +01:00
|
|
|
struct bitcoin_signature last_sig;
|
2018-02-19 02:06:14 +01:00
|
|
|
u8 *remote_shutdown_scriptpubkey;
|
2019-09-29 09:35:45 +02:00
|
|
|
u8 *local_shutdown_scriptpubkey;
|
2018-02-19 02:06:14 +01:00
|
|
|
struct changed_htlc *last_sent_commit;
|
2019-08-20 09:54:38 +02:00
|
|
|
s64 final_key_idx, channel_config_id;
|
2018-07-23 04:23:02 +02:00
|
|
|
struct basepoints local_basepoints;
|
|
|
|
struct pubkey local_funding_pubkey;
|
2018-08-17 07:06:36 +02:00
|
|
|
struct pubkey *future_per_commitment_point;
|
2019-09-30 23:47:16 +02:00
|
|
|
struct amount_sat funding_sat, our_funding_sat;
|
2019-08-20 09:54:38 +02:00
|
|
|
struct amount_msat push_msat, our_msat, msat_to_us_min, msat_to_us_max;
|
2020-09-11 22:28:15 +02:00
|
|
|
struct wally_psbt *psbt;
|
2018-02-12 11:12:55 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
peer_dbid = db_column_u64(stmt, 1);
|
2018-02-12 11:12:55 +01:00
|
|
|
peer = find_peer_by_dbid(w->ld, peer_dbid);
|
|
|
|
if (!peer) {
|
2018-02-21 16:37:08 +01:00
|
|
|
peer = wallet_peer_load(w, peer_dbid);
|
2018-02-19 02:06:14 +01:00
|
|
|
if (!peer) {
|
2018-02-12 11:12:55 +01:00
|
|
|
return NULL;
|
2018-02-19 02:06:14 +01:00
|
|
|
}
|
2017-07-20 16:42:55 +02:00
|
|
|
}
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_column_is_null(stmt, 2)) {
|
2018-02-19 02:06:14 +01:00
|
|
|
scid = tal(tmpctx, struct short_channel_id);
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_column_short_channel_id(stmt, 2, scid))
|
2019-01-20 06:43:35 +01:00
|
|
|
return NULL;
|
2017-07-20 16:42:55 +02:00
|
|
|
} else {
|
2018-02-19 02:06:14 +01:00
|
|
|
scid = NULL;
|
2017-07-20 16:42:55 +02:00
|
|
|
}
|
|
|
|
|
2020-09-09 09:20:53 +02:00
|
|
|
ok &= wallet_shachain_load(w, db_column_u64(stmt, 29), &wshachain);
|
2017-07-20 16:42:55 +02:00
|
|
|
|
2020-09-09 09:20:53 +02:00
|
|
|
remote_shutdown_scriptpubkey = db_column_arr(tmpctx, stmt, 30, u8);
|
|
|
|
local_shutdown_scriptpubkey = db_column_arr(tmpctx, stmt, 49, u8);
|
2017-07-20 16:42:55 +02:00
|
|
|
|
|
|
|
/* Do we have a last_sent_commit, if yes, populate */
|
2020-09-09 09:20:53 +02:00
|
|
|
if (!db_column_is_null(stmt, 43)) {
|
|
|
|
const u8 *cursor = db_column_blob(stmt, 43);
|
|
|
|
size_t len = db_column_bytes(stmt, 43);
|
2018-09-04 09:02:35 +02:00
|
|
|
size_t n = 0;
|
|
|
|
last_sent_commit = tal_arr(tmpctx, struct changed_htlc, n);
|
|
|
|
while (len) {
|
|
|
|
tal_resize(&last_sent_commit, n+1);
|
|
|
|
fromwire_changed_htlc(&cursor, &len,
|
|
|
|
&last_sent_commit[n++]);
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
last_sent_commit = NULL;
|
|
|
|
|
|
|
|
#ifdef COMPAT_V060
|
2020-09-09 09:20:53 +02:00
|
|
|
if (!last_sent_commit && !db_column_is_null(stmt, 32)) {
|
2018-02-19 02:06:14 +01:00
|
|
|
last_sent_commit = tal(tmpctx, struct changed_htlc);
|
2020-09-09 09:20:53 +02:00
|
|
|
last_sent_commit->newstate = db_column_u64(stmt, 32);
|
|
|
|
last_sent_commit->id = db_column_u64(stmt, 33);
|
2017-07-20 16:42:55 +02:00
|
|
|
}
|
2018-09-04 09:02:35 +02:00
|
|
|
#endif
|
2017-08-08 19:09:08 +02:00
|
|
|
|
2020-09-09 09:20:53 +02:00
|
|
|
if (!db_column_is_null(stmt, 42)) {
|
2018-08-17 07:06:36 +02:00
|
|
|
future_per_commitment_point = tal(tmpctx, struct pubkey);
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_pubkey(stmt, 42, future_per_commitment_point);
|
2018-08-17 07:06:36 +02:00
|
|
|
} else
|
|
|
|
future_per_commitment_point = NULL;
|
|
|
|
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_channel_id(stmt, 3, &cid);
|
|
|
|
channel_config_id = db_column_u64(stmt, 4);
|
2019-08-20 09:54:38 +02:00
|
|
|
ok &= wallet_channel_config_load(w, channel_config_id, &our_config);
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_sha256d(stmt, 13, &funding_txid.shad);
|
|
|
|
ok &= db_column_signature(stmt, 35, &last_sig.s);
|
2018-12-03 00:15:06 +01:00
|
|
|
last_sig.sighash_type = SIGHASH_ALL;
|
2017-08-21 11:25:15 +02:00
|
|
|
|
2018-02-19 02:06:14 +01:00
|
|
|
/* Populate channel_info */
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_pubkey(stmt, 20, &channel_info.remote_fundingkey);
|
|
|
|
db_column_pubkey(stmt, 21, &channel_info.theirbase.revocation);
|
|
|
|
db_column_pubkey(stmt, 22, &channel_info.theirbase.payment);
|
|
|
|
db_column_pubkey(stmt, 23, &channel_info.theirbase.htlc);
|
|
|
|
db_column_pubkey(stmt, 24, &channel_info.theirbase.delayed_payment);
|
|
|
|
db_column_pubkey(stmt, 25, &channel_info.remote_per_commit);
|
|
|
|
db_column_pubkey(stmt, 26, &channel_info.old_remote_per_commit);
|
|
|
|
|
|
|
|
wallet_channel_config_load(w, db_column_u64(stmt, 5),
|
2018-02-19 02:06:14 +01:00
|
|
|
&channel_info.their_config);
|
|
|
|
|
2020-09-17 03:58:59 +02:00
|
|
|
fee_states
|
2019-12-12 18:18:25 +01:00
|
|
|
= wallet_channel_fee_states_load(w,
|
|
|
|
db_column_u64(stmt, 0),
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_int(stmt, 7));
|
2020-09-17 03:58:59 +02:00
|
|
|
if (!fee_states)
|
2019-12-12 18:18:25 +01:00
|
|
|
ok = false;
|
|
|
|
|
2018-02-19 02:06:14 +01:00
|
|
|
if (!ok) {
|
2020-09-17 03:58:59 +02:00
|
|
|
tal_free(fee_states);
|
2018-02-19 02:06:14 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
2018-02-09 16:40:59 +01:00
|
|
|
|
2020-09-09 09:20:53 +02:00
|
|
|
final_key_idx = db_column_u64(stmt, 31);
|
2018-03-07 01:06:07 +01:00
|
|
|
if (final_key_idx < 0) {
|
|
|
|
log_broken(w->log, "%s: Final key < 0", __func__);
|
|
|
|
return NULL;
|
|
|
|
}
|
2018-07-23 04:23:02 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
get_channel_basepoints(w->ld, &peer->id, db_column_u64(stmt, 0),
|
2018-07-23 04:23:03 +02:00
|
|
|
&local_basepoints, &local_funding_pubkey);
|
2019-08-20 09:54:38 +02:00
|
|
|
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_amount_sat(stmt, 15, &funding_sat);
|
|
|
|
db_column_amount_sat(stmt, 16, &our_funding_sat);
|
|
|
|
db_column_amount_msat(stmt, 18, &push_msat);
|
|
|
|
db_column_amount_msat(stmt, 19, &our_msat);
|
|
|
|
db_column_amount_msat(stmt, 40, &msat_to_us_min);
|
|
|
|
db_column_amount_msat(stmt, 41, &msat_to_us_max);
|
2019-08-20 09:54:38 +02:00
|
|
|
|
2020-09-11 22:28:15 +02:00
|
|
|
if (!db_column_is_null(stmt, 50)) {
|
|
|
|
psbt = db_column_psbt(tmpctx, stmt, 50);
|
|
|
|
} else
|
|
|
|
psbt = NULL;
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
chan = new_channel(peer, db_column_u64(stmt, 0),
|
2018-02-19 02:06:14 +01:00
|
|
|
&wshachain,
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_int(stmt, 6),
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_int(stmt, 7),
|
2018-02-19 02:06:14 +01:00
|
|
|
NULL, /* Set up fresh log */
|
2018-02-23 06:53:44 +01:00
|
|
|
"Loaded from database",
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_int(stmt, 8),
|
2020-09-09 09:20:53 +02:00
|
|
|
&our_config,
|
|
|
|
db_column_int(stmt, 9),
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_u64(stmt, 10),
|
|
|
|
db_column_u64(stmt, 11),
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_u64(stmt, 12),
|
2018-02-19 02:06:14 +01:00
|
|
|
&funding_txid,
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_int(stmt, 14),
|
2019-08-20 09:54:38 +02:00
|
|
|
funding_sat,
|
|
|
|
push_msat,
|
2019-09-30 23:47:16 +02:00
|
|
|
our_funding_sat,
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_int(stmt, 17) != 0,
|
2018-02-19 02:06:14 +01:00
|
|
|
scid,
|
2020-09-09 09:20:53 +02:00
|
|
|
&cid,
|
2019-08-20 09:54:38 +02:00
|
|
|
our_msat,
|
|
|
|
msat_to_us_min, /* msatoshi_to_us_min */
|
|
|
|
msat_to_us_max, /* msatoshi_to_us_max */
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_psbt_to_tx(tmpctx, stmt, 34),
|
2018-02-19 02:06:14 +01:00
|
|
|
&last_sig,
|
|
|
|
wallet_htlc_sigs_load(tmpctx, w,
|
2020-08-13 19:45:02 +02:00
|
|
|
db_column_u64(stmt, 0),
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_int(stmt, 48)),
|
2018-02-19 02:06:14 +01:00
|
|
|
&channel_info,
|
2020-09-17 03:58:59 +02:00
|
|
|
take(fee_states),
|
2018-02-19 02:06:14 +01:00
|
|
|
remote_shutdown_scriptpubkey,
|
2019-09-29 09:35:45 +02:00
|
|
|
local_shutdown_scriptpubkey,
|
2018-03-07 01:06:07 +01:00
|
|
|
final_key_idx,
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_int(stmt, 36) != 0,
|
2018-02-19 02:06:14 +01:00
|
|
|
last_sent_commit,
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_u64(stmt, 37),
|
2019-09-30 23:47:16 +02:00
|
|
|
db_column_int(stmt, 38),
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_int(stmt, 39),
|
2018-04-26 06:51:01 +02:00
|
|
|
/* Not connected */
|
2018-07-23 04:23:02 +02:00
|
|
|
false,
|
2018-08-17 07:06:36 +02:00
|
|
|
&local_basepoints, &local_funding_pubkey,
|
2019-02-21 01:01:33 +01:00
|
|
|
future_per_commitment_point,
|
2019-09-30 23:47:16 +02:00
|
|
|
db_column_int(stmt, 44),
|
2020-09-09 09:20:53 +02:00
|
|
|
db_column_int(stmt, 45),
|
|
|
|
db_column_arr(tmpctx, stmt, 46, u8),
|
|
|
|
db_column_int(stmt, 47),
|
2020-09-11 22:28:15 +02:00
|
|
|
db_column_int(stmt, 48),
|
2020-10-28 11:46:17 +01:00
|
|
|
psbt,
|
|
|
|
db_column_int(stmt, 51),
|
|
|
|
db_column_int(stmt, 52));
|
2020-05-21 23:32:53 +02:00
|
|
|
|
2018-02-12 11:12:55 +01:00
|
|
|
return chan;
|
2017-07-20 16:42:55 +02:00
|
|
|
}
|
|
|
|
|
2019-08-09 18:01:31 +02:00
|
|
|
static void set_max_channel_dbid(struct wallet *w)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2019-08-09 18:01:31 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT id FROM channels ORDER BY id DESC LIMIT 1;"));
|
|
|
|
db_query_prepared(stmt);
|
2019-08-09 18:01:31 +02:00
|
|
|
w->max_channel_dbid = 0;
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (db_step(stmt))
|
|
|
|
w->max_channel_dbid = db_column_u64(stmt, 0);
|
2019-08-09 18:01:31 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2019-08-09 18:01:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool wallet_channels_load_active(struct wallet *w)
|
2017-08-17 15:30:24 +02:00
|
|
|
{
|
|
|
|
bool ok = true;
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2019-07-16 20:24:03 +02:00
|
|
|
int count = 0;
|
2018-02-12 11:10:37 +01:00
|
|
|
|
2019-07-16 20:24:03 +02:00
|
|
|
/* We load all channels */
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT"
|
|
|
|
" id"
|
|
|
|
", peer_id"
|
|
|
|
", short_channel_id"
|
2020-09-09 09:20:53 +02:00
|
|
|
", full_channel_id"
|
2019-08-20 09:54:38 +02:00
|
|
|
", channel_config_local"
|
|
|
|
", channel_config_remote"
|
|
|
|
", state"
|
|
|
|
", funder"
|
|
|
|
", channel_flags"
|
|
|
|
", minimum_depth"
|
|
|
|
", next_index_local"
|
|
|
|
", next_index_remote"
|
|
|
|
", next_htlc_id"
|
|
|
|
", funding_tx_id"
|
|
|
|
", funding_tx_outnum"
|
|
|
|
", funding_satoshi"
|
2019-09-30 23:47:16 +02:00
|
|
|
", our_funding_satoshi"
|
2019-08-20 09:54:38 +02:00
|
|
|
", funding_locked_remote"
|
|
|
|
", push_msatoshi"
|
|
|
|
", msatoshi_local"
|
|
|
|
", fundingkey_remote"
|
|
|
|
", revocation_basepoint_remote"
|
|
|
|
", payment_basepoint_remote"
|
|
|
|
", htlc_basepoint_remote"
|
|
|
|
", delayed_payment_basepoint_remote"
|
|
|
|
", per_commit_remote"
|
|
|
|
", old_per_commit_remote"
|
|
|
|
", local_feerate_per_kw"
|
|
|
|
", remote_feerate_per_kw"
|
|
|
|
", shachain_remote_id"
|
|
|
|
", shutdown_scriptpubkey_remote"
|
|
|
|
", shutdown_keyidx_local"
|
|
|
|
", last_sent_commit_state"
|
|
|
|
", last_sent_commit_id"
|
|
|
|
", last_tx"
|
|
|
|
", last_sig"
|
|
|
|
", last_was_revoke"
|
|
|
|
", first_blocknum"
|
|
|
|
", min_possible_feerate"
|
|
|
|
", max_possible_feerate"
|
|
|
|
", msatoshi_to_us_min"
|
|
|
|
", msatoshi_to_us_max"
|
|
|
|
", future_per_commitment_point"
|
|
|
|
", last_sent_commit"
|
|
|
|
", feerate_base"
|
|
|
|
", feerate_ppm"
|
|
|
|
", remote_upfront_shutdown_script"
|
2019-09-10 04:22:27 +02:00
|
|
|
", option_static_remotekey"
|
2020-08-13 19:40:02 +02:00
|
|
|
", option_anchor_outputs"
|
2019-09-29 09:35:45 +02:00
|
|
|
", shutdown_scriptpubkey_local"
|
2020-09-11 22:28:15 +02:00
|
|
|
", funding_psbt"
|
2020-10-28 11:46:17 +01:00
|
|
|
", closer"
|
|
|
|
", state_change_reason"
|
2019-08-20 09:54:38 +02:00
|
|
|
" FROM channels WHERE state < ?;"));
|
|
|
|
db_bind_int(stmt, 0, CLOSED);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
while (db_step(stmt)) {
|
2019-05-16 10:54:38 +02:00
|
|
|
struct channel *c = wallet_stmt2channel(w, stmt);
|
2018-02-12 11:12:55 +01:00
|
|
|
if (!c) {
|
|
|
|
ok = false;
|
|
|
|
break;
|
|
|
|
}
|
2017-08-17 15:30:24 +02:00
|
|
|
count++;
|
|
|
|
}
|
|
|
|
log_debug(w->log, "Loaded %d channels from DB", count);
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2017-08-17 15:30:24 +02:00
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
2019-08-09 18:01:31 +02:00
|
|
|
bool wallet_init_channels(struct wallet *w)
|
|
|
|
{
|
|
|
|
/* We set the max channel database id separately */
|
|
|
|
set_max_channel_dbid(w);
|
|
|
|
return wallet_channels_load_active(w);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-03-24 09:26:08 +01:00
|
|
|
static
|
|
|
|
void wallet_channel_stats_incr_x(struct wallet *w,
|
|
|
|
char const *dir,
|
|
|
|
char const *typ,
|
|
|
|
u64 cdbid,
|
2019-02-21 04:45:55 +01:00
|
|
|
struct amount_msat msat)
|
2018-03-24 09:26:08 +01:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
|
|
|
const char *query;
|
|
|
|
/* TODO These would be much better as a switch statement, leaving
|
|
|
|
* these here for now in order to keep the commit clean. */
|
|
|
|
if (streq(dir, "in") && streq(typ, "offered")) {
|
|
|
|
query = SQL("UPDATE channels"
|
|
|
|
" SET in_payments_offered = COALESCE(in_payments_offered, 0) + 1"
|
|
|
|
" , in_msatoshi_offered = COALESCE(in_msatoshi_offered, 0) + ?"
|
|
|
|
" WHERE id = ?;");
|
|
|
|
} else if (streq(dir, "in") && streq(typ, "fulfilled")) {
|
|
|
|
query = SQL("UPDATE channels"
|
|
|
|
" SET in_payments_fulfilled = COALESCE(in_payments_fulfilled, 0) + 1"
|
|
|
|
" , in_msatoshi_fulfilled = COALESCE(in_msatoshi_fulfilled, 0) + ?"
|
|
|
|
" WHERE id = ?;");
|
|
|
|
} else if (streq(dir, "out") && streq(typ, "offered")) {
|
|
|
|
query = SQL("UPDATE channels"
|
|
|
|
" SET out_payments_offered = COALESCE(out_payments_offered, 0) + 1"
|
|
|
|
" , out_msatoshi_offered = COALESCE(out_msatoshi_offered, 0) + ?"
|
|
|
|
" WHERE id = ?;");
|
|
|
|
} else if (streq(dir, "out") && streq(typ, "fulfilled")) {
|
|
|
|
query = SQL("UPDATE channels"
|
|
|
|
" SET out_payments_fulfilled = COALESCE(out_payments_fulfilled, 0) + 1"
|
|
|
|
" , out_msatoshi_fulfilled = COALESCE(out_msatoshi_fulfilled, 0) + ?"
|
|
|
|
" WHERE id = ?;");
|
|
|
|
} else {
|
|
|
|
fatal("Unknown stats key %s %s", dir, typ);
|
|
|
|
}
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(w->db, query);
|
|
|
|
db_bind_amount_msat(stmt, 0, &msat);
|
|
|
|
db_bind_u64(stmt, 1, cdbid);
|
|
|
|
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-03-24 09:26:08 +01:00
|
|
|
}
|
2019-02-21 04:45:55 +01:00
|
|
|
void wallet_channel_stats_incr_in_offered(struct wallet *w, u64 id,
|
|
|
|
struct amount_msat m)
|
2018-03-24 09:26:08 +01:00
|
|
|
{
|
|
|
|
wallet_channel_stats_incr_x(w, "in", "offered", id, m);
|
|
|
|
}
|
2019-02-21 04:45:55 +01:00
|
|
|
void wallet_channel_stats_incr_in_fulfilled(struct wallet *w, u64 id,
|
|
|
|
struct amount_msat m)
|
2018-03-24 09:26:08 +01:00
|
|
|
{
|
|
|
|
wallet_channel_stats_incr_x(w, "in", "fulfilled", id, m);
|
|
|
|
}
|
2019-02-21 04:45:55 +01:00
|
|
|
void wallet_channel_stats_incr_out_offered(struct wallet *w, u64 id,
|
|
|
|
struct amount_msat m)
|
2018-03-24 09:26:08 +01:00
|
|
|
{
|
|
|
|
wallet_channel_stats_incr_x(w, "out", "offered", id, m);
|
|
|
|
}
|
2019-02-21 04:45:55 +01:00
|
|
|
void wallet_channel_stats_incr_out_fulfilled(struct wallet *w, u64 id,
|
|
|
|
struct amount_msat m)
|
2018-03-24 09:26:08 +01:00
|
|
|
{
|
|
|
|
wallet_channel_stats_incr_x(w, "out", "fulfilled", id, m);
|
|
|
|
}
|
|
|
|
|
|
|
|
void wallet_channel_stats_load(struct wallet *w,
|
|
|
|
u64 id,
|
|
|
|
struct channel_stats *stats)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-10-12 14:58:36 +02:00
|
|
|
int res;
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL(
|
|
|
|
"SELECT"
|
|
|
|
" in_payments_offered, in_payments_fulfilled"
|
|
|
|
", in_msatoshi_offered, in_msatoshi_fulfilled"
|
|
|
|
", out_payments_offered, out_payments_fulfilled"
|
|
|
|
", out_msatoshi_offered, out_msatoshi_fulfilled"
|
|
|
|
" FROM channels"
|
|
|
|
" WHERE id = ?"));
|
|
|
|
db_bind_u64(stmt, 0, id);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
res = db_step(stmt);
|
2018-10-12 14:58:36 +02:00
|
|
|
|
|
|
|
/* This must succeed, since we know the channel exists */
|
2019-08-20 09:54:38 +02:00
|
|
|
assert(res);
|
2018-10-12 14:58:36 +02:00
|
|
|
|
2019-10-17 13:55:11 +02:00
|
|
|
stats->in_payments_offered = db_column_int_or_default(stmt, 0, 0);
|
|
|
|
stats->in_payments_fulfilled = db_column_int_or_default(stmt, 1, 0);
|
|
|
|
db_column_amount_msat_or_default(stmt, 2, &stats->in_msatoshi_offered, AMOUNT_MSAT(0));
|
|
|
|
db_column_amount_msat_or_default(stmt, 3, &stats->in_msatoshi_fulfilled, AMOUNT_MSAT(0));
|
|
|
|
stats->out_payments_offered = db_column_int_or_default(stmt, 4, 0);
|
|
|
|
stats->out_payments_fulfilled = db_column_int_or_default(stmt, 5, 0);
|
|
|
|
db_column_amount_msat_or_default(stmt, 6, &stats->out_msatoshi_offered, AMOUNT_MSAT(0));
|
|
|
|
db_column_amount_msat_or_default(stmt, 7, &stats->out_msatoshi_fulfilled, AMOUNT_MSAT(0));
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2018-03-24 09:26:08 +01:00
|
|
|
}
|
|
|
|
|
2018-06-04 14:47:32 +02:00
|
|
|
void wallet_blocks_heights(struct wallet *w, u32 def, u32 *min, u32 *max)
|
2018-01-03 06:26:30 +01:00
|
|
|
{
|
2018-06-04 14:47:32 +02:00
|
|
|
assert(min != NULL && max != NULL);
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt = db_prepare_v2(w->db, SQL("SELECT MIN(height), MAX(height) FROM blocks;"));
|
|
|
|
db_query_prepared(stmt);
|
2019-03-15 03:50:18 +01:00
|
|
|
*min = def;
|
|
|
|
*max = def;
|
2018-01-03 06:26:30 +01:00
|
|
|
|
2018-04-17 23:31:30 +02:00
|
|
|
/* If we ever processed a block we'll get the latest block in the chain */
|
2019-08-20 09:54:38 +02:00
|
|
|
if (db_step(stmt)) {
|
|
|
|
if (!db_column_is_null(stmt, 0)) {
|
|
|
|
*min = db_column_int(stmt, 0);
|
|
|
|
*max = db_column_int(stmt, 1);
|
2019-03-15 03:50:18 +01:00
|
|
|
}
|
2018-04-27 03:40:38 +02:00
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2018-01-03 06:26:30 +01:00
|
|
|
}
|
|
|
|
|
2018-02-18 13:54:46 +01:00
|
|
|
static void wallet_channel_config_insert(struct wallet *w,
|
|
|
|
struct channel_config *cc)
|
2017-08-05 14:45:42 +02:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2017-08-05 14:45:42 +02:00
|
|
|
|
2018-02-18 13:54:46 +01:00
|
|
|
assert(cc->id == 0);
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("INSERT INTO channel_configs DEFAULT VALUES;"));
|
|
|
|
db_exec_prepared_v2(stmt);
|
|
|
|
cc->id = db_last_insert_id_v2(stmt);
|
|
|
|
tal_free(stmt);
|
2018-02-18 13:54:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void wallet_channel_config_save(struct wallet *w,
|
|
|
|
const struct channel_config *cc)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-02-18 13:54:46 +01:00
|
|
|
|
|
|
|
assert(cc->id != 0);
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("UPDATE channel_configs SET"
|
|
|
|
" dust_limit_satoshis=?,"
|
|
|
|
" max_htlc_value_in_flight_msat=?,"
|
|
|
|
" channel_reserve_satoshis=?,"
|
|
|
|
" htlc_minimum_msat=?,"
|
|
|
|
" to_self_delay=?,"
|
|
|
|
" max_accepted_htlcs=?"
|
|
|
|
" WHERE id=?;"));
|
|
|
|
db_bind_amount_sat(stmt, 0, &cc->dust_limit);
|
|
|
|
db_bind_amount_msat(stmt, 1, &cc->max_htlc_value_in_flight);
|
|
|
|
db_bind_amount_sat(stmt, 2, &cc->channel_reserve);
|
|
|
|
db_bind_amount_msat(stmt, 3, &cc->htlc_minimum);
|
|
|
|
db_bind_int(stmt, 4, cc->to_self_delay);
|
|
|
|
db_bind_int(stmt, 5, cc->max_accepted_htlcs);
|
|
|
|
db_bind_u64(stmt, 6, cc->id);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2017-08-05 14:45:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool wallet_channel_config_load(struct wallet *w, const u64 id,
|
|
|
|
struct channel_config *cc)
|
|
|
|
{
|
|
|
|
bool ok = true;
|
|
|
|
int col = 1;
|
2019-08-20 09:54:38 +02:00
|
|
|
const char *query = SQL(
|
2019-07-17 15:12:34 +02:00
|
|
|
"SELECT id, dust_limit_satoshis, max_htlc_value_in_flight_msat, "
|
2017-08-05 14:45:42 +02:00
|
|
|
"channel_reserve_satoshis, htlc_minimum_msat, to_self_delay, "
|
2019-08-20 09:54:38 +02:00
|
|
|
"max_accepted_htlcs FROM channel_configs WHERE id= ? ;");
|
|
|
|
struct db_stmt *stmt = db_prepare_v2(w->db, query);
|
|
|
|
db_bind_u64(stmt, 0, id);
|
|
|
|
db_query_prepared(stmt);
|
2019-07-17 14:44:43 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_step(stmt))
|
2017-08-05 14:45:42 +02:00
|
|
|
return false;
|
2019-03-15 03:50:18 +01:00
|
|
|
|
2017-08-05 14:45:42 +02:00
|
|
|
cc->id = id;
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_amount_sat(stmt, col++, &cc->dust_limit);
|
|
|
|
db_column_amount_msat(stmt, col++, &cc->max_htlc_value_in_flight);
|
|
|
|
db_column_amount_sat(stmt, col++, &cc->channel_reserve);
|
|
|
|
db_column_amount_msat(stmt, col++, &cc->htlc_minimum);
|
|
|
|
cc->to_self_delay = db_column_int(stmt, col++);
|
|
|
|
cc->max_accepted_htlcs = db_column_int(stmt, col++);
|
2017-08-05 14:45:42 +02:00
|
|
|
assert(col == 7);
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2017-08-05 14:45:42 +02:00
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
2018-02-18 13:53:46 +01:00
|
|
|
u64 wallet_get_channel_dbid(struct wallet *wallet)
|
|
|
|
{
|
|
|
|
return ++wallet->max_channel_dbid;
|
|
|
|
}
|
|
|
|
|
2019-05-15 17:21:26 +02:00
|
|
|
/* When we receive the remote announcement message, we will also call this function */
|
|
|
|
void wallet_announcement_save(struct wallet *w, u64 id,
|
|
|
|
secp256k1_ecdsa_signature *remote_ann_node_sig,
|
|
|
|
secp256k1_ecdsa_signature *remote_ann_bitcoin_sig)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2019-05-15 17:21:26 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("UPDATE channels SET"
|
|
|
|
" remote_ann_node_sig=?,"
|
|
|
|
" remote_ann_bitcoin_sig=?"
|
|
|
|
" WHERE id=?"));
|
2019-05-15 17:21:26 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_signature(stmt, 0, remote_ann_node_sig);
|
|
|
|
db_bind_signature(stmt, 1, remote_ann_bitcoin_sig);
|
|
|
|
db_bind_u64(stmt, 2, id);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2019-05-15 17:21:26 +02:00
|
|
|
}
|
|
|
|
|
2018-02-12 11:12:55 +01:00
|
|
|
void wallet_channel_save(struct wallet *w, struct channel *chan)
|
2018-01-03 06:26:43 +01:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-09-04 09:02:35 +02:00
|
|
|
u8 *last_sent_commit;
|
2018-02-06 12:15:26 +01:00
|
|
|
assert(chan->first_blocknum);
|
2017-10-20 19:04:43 +02:00
|
|
|
|
2018-02-12 11:12:55 +01:00
|
|
|
wallet_channel_config_save(w, &chan->our_config);
|
2017-08-05 15:49:36 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("UPDATE channels SET"
|
|
|
|
" shachain_remote_id=?,"
|
|
|
|
" short_channel_id=?,"
|
2020-09-09 09:20:53 +02:00
|
|
|
" full_channel_id=?,"
|
2019-08-20 09:54:38 +02:00
|
|
|
" state=?,"
|
|
|
|
" funder=?,"
|
|
|
|
" channel_flags=?,"
|
|
|
|
" minimum_depth=?,"
|
|
|
|
" next_index_local=?,"
|
|
|
|
" next_index_remote=?,"
|
|
|
|
" next_htlc_id=?,"
|
|
|
|
" funding_tx_id=?,"
|
|
|
|
" funding_tx_outnum=?,"
|
|
|
|
" funding_satoshi=?,"
|
2019-09-30 23:47:16 +02:00
|
|
|
" our_funding_satoshi=?,"
|
2019-08-20 09:54:38 +02:00
|
|
|
" funding_locked_remote=?,"
|
|
|
|
" push_msatoshi=?,"
|
|
|
|
" msatoshi_local=?,"
|
|
|
|
" shutdown_scriptpubkey_remote=?,"
|
|
|
|
" shutdown_keyidx_local=?,"
|
|
|
|
" channel_config_local=?,"
|
|
|
|
" last_tx=?, last_sig=?,"
|
|
|
|
" last_was_revoke=?,"
|
|
|
|
" min_possible_feerate=?,"
|
|
|
|
" max_possible_feerate=?,"
|
|
|
|
" msatoshi_to_us_min=?,"
|
|
|
|
" msatoshi_to_us_max=?,"
|
|
|
|
" feerate_base=?,"
|
|
|
|
" feerate_ppm=?,"
|
2019-09-10 04:22:27 +02:00
|
|
|
" remote_upfront_shutdown_script=?,"
|
2019-09-29 09:35:45 +02:00
|
|
|
" option_static_remotekey=?,"
|
2020-08-13 19:40:02 +02:00
|
|
|
" option_anchor_outputs=?,"
|
2020-09-11 22:28:15 +02:00
|
|
|
" shutdown_scriptpubkey_local=?,"
|
2020-10-28 11:46:17 +01:00
|
|
|
" funding_psbt=?,"
|
|
|
|
" closer=?,"
|
|
|
|
" state_change_reason=?"
|
2019-08-20 09:54:38 +02:00
|
|
|
" WHERE id=?"));
|
|
|
|
db_bind_u64(stmt, 0, chan->their_shachain.id);
|
2018-02-12 11:12:55 +01:00
|
|
|
if (chan->scid)
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_short_channel_id(stmt, 1, chan->scid);
|
2018-06-15 12:42:40 +02:00
|
|
|
else
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 1);
|
2020-09-09 09:20:53 +02:00
|
|
|
|
|
|
|
db_bind_channel_id(stmt, 2, &chan->cid);
|
|
|
|
db_bind_int(stmt, 3, chan->state);
|
|
|
|
db_bind_int(stmt, 4, chan->opener);
|
|
|
|
db_bind_int(stmt, 5, chan->channel_flags);
|
|
|
|
db_bind_int(stmt, 6, chan->minimum_depth);
|
|
|
|
|
|
|
|
db_bind_u64(stmt, 7, chan->next_index[LOCAL]);
|
|
|
|
db_bind_u64(stmt, 8, chan->next_index[REMOTE]);
|
|
|
|
db_bind_u64(stmt, 9, chan->next_htlc_id);
|
|
|
|
|
|
|
|
db_bind_sha256d(stmt, 10, &chan->funding_txid.shad);
|
|
|
|
|
|
|
|
db_bind_int(stmt, 11, chan->funding_outnum);
|
|
|
|
db_bind_amount_sat(stmt, 12, &chan->funding);
|
|
|
|
db_bind_amount_sat(stmt, 13, &chan->our_funds);
|
|
|
|
db_bind_int(stmt, 14, chan->remote_funding_locked);
|
|
|
|
db_bind_amount_msat(stmt, 15, &chan->push);
|
|
|
|
db_bind_amount_msat(stmt, 16, &chan->our_msat);
|
|
|
|
|
|
|
|
db_bind_talarr(stmt, 17, chan->shutdown_scriptpubkey[REMOTE]);
|
|
|
|
db_bind_u64(stmt, 18, chan->final_key_idx);
|
|
|
|
db_bind_u64(stmt, 19, chan->our_config.id);
|
|
|
|
db_bind_psbt(stmt, 20, chan->last_tx->psbt);
|
|
|
|
db_bind_signature(stmt, 21, &chan->last_sig.s);
|
|
|
|
db_bind_int(stmt, 22, chan->last_was_revoke);
|
|
|
|
db_bind_int(stmt, 23, chan->min_possible_feerate);
|
|
|
|
db_bind_int(stmt, 24, chan->max_possible_feerate);
|
|
|
|
db_bind_amount_msat(stmt, 25, &chan->msat_to_us_min);
|
|
|
|
db_bind_amount_msat(stmt, 26, &chan->msat_to_us_max);
|
|
|
|
db_bind_int(stmt, 27, chan->feerate_base);
|
|
|
|
db_bind_int(stmt, 28, chan->feerate_ppm);
|
|
|
|
db_bind_talarr(stmt, 29, chan->remote_upfront_shutdown_script);
|
|
|
|
db_bind_int(stmt, 30, chan->option_static_remotekey);
|
|
|
|
db_bind_int(stmt, 31, chan->option_anchor_outputs);
|
|
|
|
db_bind_talarr(stmt, 32, chan->shutdown_scriptpubkey[LOCAL]);
|
2020-09-11 22:28:15 +02:00
|
|
|
if (chan->psbt)
|
|
|
|
db_bind_psbt(stmt, 33, chan->psbt);
|
|
|
|
else
|
|
|
|
db_bind_null(stmt, 33);
|
2020-10-28 11:46:17 +01:00
|
|
|
db_bind_int(stmt, 34, chan->closer);
|
|
|
|
db_bind_int(stmt, 35, chan->state_change_cause);
|
|
|
|
db_bind_u64(stmt, 36, chan->dbid);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2017-08-05 12:26:18 +02:00
|
|
|
|
2018-02-19 02:06:12 +01:00
|
|
|
wallet_channel_config_save(w, &chan->channel_info.their_config);
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("UPDATE channels SET"
|
|
|
|
" fundingkey_remote=?,"
|
|
|
|
" revocation_basepoint_remote=?,"
|
|
|
|
" payment_basepoint_remote=?,"
|
|
|
|
" htlc_basepoint_remote=?,"
|
|
|
|
" delayed_payment_basepoint_remote=?,"
|
|
|
|
" per_commit_remote=?,"
|
|
|
|
" old_per_commit_remote=?,"
|
|
|
|
" channel_config_remote=?,"
|
|
|
|
" future_per_commitment_point=?"
|
|
|
|
" WHERE id=?"));
|
|
|
|
db_bind_pubkey(stmt, 0, &chan->channel_info.remote_fundingkey);
|
|
|
|
db_bind_pubkey(stmt, 1, &chan->channel_info.theirbase.revocation);
|
|
|
|
db_bind_pubkey(stmt, 2, &chan->channel_info.theirbase.payment);
|
|
|
|
db_bind_pubkey(stmt, 3, &chan->channel_info.theirbase.htlc);
|
|
|
|
db_bind_pubkey(stmt, 4, &chan->channel_info.theirbase.delayed_payment);
|
|
|
|
db_bind_pubkey(stmt, 5, &chan->channel_info.remote_per_commit);
|
|
|
|
db_bind_pubkey(stmt, 6, &chan->channel_info.old_remote_per_commit);
|
2019-12-12 18:18:25 +01:00
|
|
|
db_bind_u64(stmt, 7, chan->channel_info.their_config.id);
|
2018-08-17 07:06:36 +02:00
|
|
|
if (chan->future_per_commitment_point)
|
2019-12-12 18:18:25 +01:00
|
|
|
db_bind_pubkey(stmt, 8, chan->future_per_commitment_point);
|
2018-08-17 07:06:36 +02:00
|
|
|
else
|
2019-12-12 18:18:25 +01:00
|
|
|
db_bind_null(stmt, 8);
|
|
|
|
db_bind_u64(stmt, 9, chan->dbid);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2017-08-05 12:26:18 +02:00
|
|
|
|
2019-12-12 18:18:25 +01:00
|
|
|
/* FIXME: Updates channel_feerates by discarding and rewriting. */
|
|
|
|
stmt = db_prepare_v2(w->db, SQL("DELETE FROM channel_feerates "
|
|
|
|
"WHERE channel_id=?"));
|
|
|
|
db_bind_u64(stmt, 0, chan->dbid);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
|
|
|
|
|
|
|
for (enum htlc_state i = 0;
|
2020-09-17 03:58:59 +02:00
|
|
|
i < ARRAY_SIZE(chan->fee_states->feerate);
|
2019-12-12 18:18:25 +01:00
|
|
|
i++) {
|
2020-09-17 03:58:59 +02:00
|
|
|
if (!chan->fee_states->feerate[i])
|
2019-12-12 18:18:25 +01:00
|
|
|
continue;
|
|
|
|
stmt = db_prepare_v2(w->db, SQL("INSERT INTO channel_feerates "
|
|
|
|
" VALUES(?, ?, ?)"));
|
|
|
|
db_bind_u64(stmt, 0, chan->dbid);
|
|
|
|
db_bind_int(stmt, 1, i);
|
2020-09-17 03:58:59 +02:00
|
|
|
db_bind_int(stmt, 2, *chan->fee_states->feerate[i]);
|
2019-12-12 18:18:25 +01:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
|
|
|
}
|
|
|
|
|
2017-08-05 12:26:18 +02:00
|
|
|
/* If we have a last_sent_commit, store it */
|
2018-09-04 09:02:35 +02:00
|
|
|
last_sent_commit = tal_arr(tmpctx, u8, 0);
|
|
|
|
for (size_t i = 0; i < tal_count(chan->last_sent_commit); i++)
|
|
|
|
towire_changed_htlc(&last_sent_commit,
|
|
|
|
&chan->last_sent_commit[i]);
|
2020-08-25 23:20:52 +02:00
|
|
|
/* Make it null in db if it's empty */
|
|
|
|
if (tal_count(last_sent_commit) == 0)
|
|
|
|
last_sent_commit = tal_free(last_sent_commit);
|
2018-09-04 09:02:35 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("UPDATE channels SET"
|
|
|
|
" last_sent_commit=?"
|
|
|
|
" WHERE id=?"));
|
2020-08-25 23:20:52 +02:00
|
|
|
db_bind_talarr(stmt, 0, last_sent_commit);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_u64(stmt, 1, chan->dbid);
|
2019-09-12 22:46:34 +02:00
|
|
|
db_exec_prepared_v2(stmt);
|
|
|
|
tal_free(stmt);
|
2017-08-05 12:26:18 +02:00
|
|
|
}
|
2017-09-01 14:33:21 +02:00
|
|
|
|
2020-10-28 11:46:22 +01:00
|
|
|
void wallet_state_change_add(struct wallet *w,
|
|
|
|
const u64 channel_id,
|
|
|
|
struct timeabs *timestamp,
|
|
|
|
enum channel_state old_state,
|
|
|
|
enum channel_state new_state,
|
|
|
|
enum state_change cause,
|
|
|
|
char *message)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
stmt = db_prepare_v2(w->db,
|
|
|
|
SQL("INSERT INTO channel_state_changes ("
|
|
|
|
" channel_id"
|
|
|
|
", timestamp"
|
|
|
|
", old_state"
|
|
|
|
", new_state"
|
|
|
|
", cause"
|
|
|
|
", message"
|
|
|
|
") VALUES (?, ?, ?, ?, ?, ?);"));
|
|
|
|
|
|
|
|
db_bind_u64(stmt, 0, channel_id);
|
|
|
|
db_bind_timeabs(stmt, 1, *timestamp);
|
|
|
|
db_bind_int(stmt, 2, old_state);
|
|
|
|
db_bind_int(stmt, 3, new_state);
|
|
|
|
db_bind_int(stmt, 4, cause);
|
|
|
|
db_bind_text(stmt, 5, message);
|
|
|
|
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
|
|
|
}
|
|
|
|
|
2020-10-28 11:46:23 +01:00
|
|
|
struct state_change_entry *wallet_state_change_get(struct wallet *w,
|
|
|
|
const tal_t *ctx,
|
|
|
|
u64 channel_id)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
struct state_change_entry tmp;
|
|
|
|
struct state_change_entry *res = tal_arr(ctx,
|
|
|
|
struct state_change_entry, 0);
|
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("SELECT"
|
|
|
|
" timestamp,"
|
|
|
|
" old_state,"
|
|
|
|
" new_state,"
|
|
|
|
" cause,"
|
|
|
|
" message "
|
|
|
|
"FROM channel_state_changes "
|
|
|
|
"WHERE channel_id = ? "
|
|
|
|
"ORDER BY timestamp ASC;"));
|
|
|
|
db_bind_int(stmt, 0, channel_id);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
while (db_step(stmt)) {
|
|
|
|
tmp.timestamp = db_column_timeabs(stmt, 0);
|
|
|
|
tmp.old_state = db_column_int(stmt, 1);
|
|
|
|
tmp.new_state = db_column_int(stmt, 2);
|
|
|
|
tmp.cause = db_column_int(stmt, 3);
|
|
|
|
tmp.message = tal_strdup(ctx, (const char *)db_column_text(stmt, 4));
|
|
|
|
tal_arr_expand(&res, tmp);
|
|
|
|
}
|
|
|
|
tal_free(stmt);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2020-06-26 17:35:57 +02:00
|
|
|
static void wallet_peer_save(struct wallet *w, struct peer *peer)
|
2018-02-18 13:53:46 +01:00
|
|
|
{
|
2020-06-26 17:35:57 +02:00
|
|
|
const char *addr =
|
|
|
|
type_to_string(tmpctx, struct wireaddr_internal, &peer->addr);
|
|
|
|
struct db_stmt *stmt =
|
|
|
|
db_prepare_v2(w->db, SQL("SELECT id FROM peers WHERE node_id = ?"));
|
|
|
|
|
|
|
|
db_bind_node_id(stmt, 0, &peer->id);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
if (db_step(stmt)) {
|
|
|
|
/* So we already knew this peer, just return its dbid */
|
|
|
|
peer->dbid = db_column_u64(stmt, 0);
|
|
|
|
tal_free(stmt);
|
|
|
|
|
|
|
|
/* Since we're at it update the wireaddr */
|
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("UPDATE peers SET address = ? WHERE id = ?"));
|
|
|
|
db_bind_text(stmt, 0, addr);
|
|
|
|
db_bind_u64(stmt, 1, peer->dbid);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-02-18 13:53:46 +01:00
|
|
|
|
2020-06-26 17:35:57 +02:00
|
|
|
} else {
|
|
|
|
/* Unknown peer, create it from scratch */
|
|
|
|
tal_free(stmt);
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db,
|
|
|
|
SQL("INSERT INTO peers (node_id, address) VALUES (?, ?);")
|
|
|
|
);
|
2020-06-26 17:35:57 +02:00
|
|
|
db_bind_node_id(stmt, 0, &peer->id);
|
|
|
|
db_bind_text(stmt, 1,addr);
|
2019-08-28 22:38:32 +02:00
|
|
|
db_exec_prepared_v2(stmt);
|
2020-06-26 17:35:57 +02:00
|
|
|
peer->dbid = db_last_insert_id_v2(take(stmt));
|
2018-02-18 13:53:46 +01:00
|
|
|
}
|
2020-06-26 17:35:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void wallet_channel_insert(struct wallet *w, struct channel *chan)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
|
|
|
|
if (chan->peer->dbid == 0)
|
|
|
|
wallet_peer_save(w, chan->peer);
|
2018-02-18 13:53:46 +01:00
|
|
|
|
|
|
|
/* Insert a stub, that we update, unifies INSERT and UPDATE paths */
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("INSERT INTO channels ("
|
|
|
|
"peer_id, first_blocknum, id) VALUES (?, ?, ?);"));
|
|
|
|
db_bind_u64(stmt, 0, chan->peer->dbid);
|
|
|
|
db_bind_int(stmt, 1, chan->first_blocknum);
|
|
|
|
db_bind_int(stmt, 2, chan->dbid);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-02-18 13:53:46 +01:00
|
|
|
|
2018-02-18 13:54:46 +01:00
|
|
|
wallet_channel_config_insert(w, &chan->our_config);
|
2018-02-19 02:06:12 +01:00
|
|
|
wallet_channel_config_insert(w, &chan->channel_info.their_config);
|
2018-02-18 13:54:46 +01:00
|
|
|
wallet_shachain_init(w, &chan->their_shachain);
|
|
|
|
|
2018-02-18 13:53:46 +01:00
|
|
|
/* Now save path as normal */
|
|
|
|
wallet_channel_save(w, chan);
|
|
|
|
}
|
|
|
|
|
2019-06-22 15:08:21 +02:00
|
|
|
void wallet_channel_close(struct wallet *w, u64 wallet_id)
|
2018-01-23 14:48:09 +01:00
|
|
|
{
|
2019-06-22 15:08:21 +02:00
|
|
|
/* We keep a couple of dependent tables around as well, such as the
|
|
|
|
* channel_configs table, since that might help us debug some issues,
|
|
|
|
* and it is rather limited in size. Tables that can grow quite
|
|
|
|
* considerably and that are of limited use after channel closure will
|
|
|
|
* be pruned as well. */
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2019-06-22 15:08:21 +02:00
|
|
|
|
|
|
|
/* Delete entries from `channel_htlcs` */
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("DELETE FROM channel_htlcs "
|
|
|
|
"WHERE channel_id=?"));
|
|
|
|
db_bind_u64(stmt, 0, wallet_id);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2019-06-22 15:08:21 +02:00
|
|
|
|
|
|
|
/* Delete entries from `htlc_sigs` */
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("DELETE FROM htlc_sigs "
|
|
|
|
"WHERE channelid=?"));
|
|
|
|
db_bind_u64(stmt, 0, wallet_id);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2019-06-22 15:08:21 +02:00
|
|
|
|
|
|
|
/* Delete entries from `htlc_sigs` */
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("DELETE FROM channeltxs "
|
|
|
|
"WHERE channel_id=?"));
|
|
|
|
db_bind_u64(stmt, 0, wallet_id);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2019-06-22 15:08:21 +02:00
|
|
|
|
|
|
|
/* Delete shachains */
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("DELETE FROM shachains "
|
|
|
|
"WHERE id IN ("
|
|
|
|
" SELECT shachain_remote_id "
|
|
|
|
" FROM channels "
|
|
|
|
" WHERE channels.id=?"
|
|
|
|
")"));
|
|
|
|
db_bind_u64(stmt, 0, wallet_id);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2019-06-22 15:08:21 +02:00
|
|
|
|
|
|
|
/* Set the channel to closed and disassociate with peer */
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("UPDATE channels "
|
2020-09-24 16:05:31 +02:00
|
|
|
"SET state=?, peer_id=? "
|
2019-08-20 09:54:38 +02:00
|
|
|
"WHERE channels.id=?"));
|
|
|
|
db_bind_u64(stmt, 0, CLOSED);
|
|
|
|
db_bind_null(stmt, 1);
|
|
|
|
db_bind_u64(stmt, 2, wallet_id);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-02-14 02:53:04 +01:00
|
|
|
}
|
2018-02-12 11:10:40 +01:00
|
|
|
|
2018-02-14 02:53:04 +01:00
|
|
|
void wallet_peer_delete(struct wallet *w, u64 peer_dbid)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-02-14 02:53:04 +01:00
|
|
|
|
|
|
|
/* Must not have any channels still using this peer */
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT * FROM channels WHERE peer_id = ?;"));
|
|
|
|
db_bind_u64(stmt, 0, peer_dbid);
|
|
|
|
db_query_prepared(stmt);
|
2019-07-17 14:44:43 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (db_step(stmt))
|
2019-03-15 03:50:18 +01:00
|
|
|
fatal("We have channels using peer %"PRIu64, peer_dbid);
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2018-02-14 02:53:04 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("DELETE FROM peers WHERE id=?"));
|
|
|
|
db_bind_u64(stmt, 0, peer_dbid);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-01-23 14:48:09 +01:00
|
|
|
}
|
|
|
|
|
2018-08-13 05:03:08 +02:00
|
|
|
void wallet_confirm_tx(struct wallet *w,
|
|
|
|
const struct bitcoin_txid *txid,
|
|
|
|
const u32 confirmation_height)
|
2018-03-02 14:33:01 +01:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-03-02 14:33:01 +01:00
|
|
|
assert(confirmation_height > 0);
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("UPDATE outputs "
|
|
|
|
"SET confirmation_height = ? "
|
|
|
|
"WHERE prev_out_tx = ?"));
|
|
|
|
db_bind_int(stmt, 0, confirmation_height);
|
|
|
|
db_bind_sha256d(stmt, 1, &txid->shad);
|
2018-03-02 14:33:01 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-03-02 14:33:01 +01:00
|
|
|
}
|
|
|
|
|
2020-06-16 20:43:51 +02:00
|
|
|
int wallet_extract_owned_outputs(struct wallet *w, const struct wally_tx *wtx,
|
2019-02-21 04:45:55 +01:00
|
|
|
const u32 *blockheight,
|
|
|
|
struct amount_sat *total)
|
2017-09-01 14:33:21 +02:00
|
|
|
{
|
|
|
|
int num_utxos = 0;
|
2019-02-21 04:45:55 +01:00
|
|
|
|
|
|
|
*total = AMOUNT_SAT(0);
|
2020-06-16 20:43:51 +02:00
|
|
|
for (size_t output = 0; output < wtx->num_outputs; output++) {
|
2017-09-01 14:33:21 +02:00
|
|
|
struct utxo *utxo;
|
|
|
|
u32 index;
|
|
|
|
bool is_p2sh;
|
2019-04-13 19:14:07 +02:00
|
|
|
const u8 *script;
|
2020-06-16 20:43:51 +02:00
|
|
|
struct amount_asset asset =
|
|
|
|
wally_tx_output_get_amount(&wtx->outputs[output]);
|
2020-04-02 04:55:44 +02:00
|
|
|
struct chain_coin_mvt *mvt;
|
2019-09-26 21:07:20 +02:00
|
|
|
|
|
|
|
if (!amount_asset_is_main(&asset))
|
|
|
|
continue;
|
2017-09-01 14:33:21 +02:00
|
|
|
|
2020-06-16 20:43:51 +02:00
|
|
|
script = wally_tx_output_get_script(tmpctx,
|
|
|
|
&wtx->outputs[output]);
|
2019-04-13 19:14:07 +02:00
|
|
|
if (!script)
|
|
|
|
continue;
|
2019-03-25 11:35:56 +01:00
|
|
|
|
2019-04-13 19:14:07 +02:00
|
|
|
if (!wallet_can_spend(w, script, &index, &is_p2sh))
|
2017-09-01 14:33:21 +02:00
|
|
|
continue;
|
|
|
|
|
|
|
|
utxo = tal(w, struct utxo);
|
|
|
|
utxo->keyindex = index;
|
|
|
|
utxo->is_p2sh = is_p2sh;
|
2019-09-26 21:07:20 +02:00
|
|
|
utxo->amount = amount_asset_to_sat(&asset);
|
2020-08-28 05:56:34 +02:00
|
|
|
utxo->status = OUTPUT_STATE_AVAILABLE;
|
2020-06-16 20:43:51 +02:00
|
|
|
wally_txid(wtx, &utxo->txid);
|
2017-09-01 14:33:21 +02:00
|
|
|
utxo->outnum = output;
|
2017-12-20 12:44:00 +01:00
|
|
|
utxo->close_info = NULL;
|
2018-02-26 11:36:48 +01:00
|
|
|
|
2019-02-15 17:18:39 +01:00
|
|
|
utxo->blockheight = blockheight ? blockheight : NULL;
|
2018-02-26 11:36:48 +01:00
|
|
|
utxo->spendheight = NULL;
|
2020-02-27 03:17:01 +01:00
|
|
|
utxo->scriptPubkey = tal_dup_talarr(utxo, u8, script);
|
2018-02-26 11:36:48 +01:00
|
|
|
|
2019-02-21 04:45:55 +01:00
|
|
|
log_debug(w->log, "Owning output %zu %s (%s) txid %s%s",
|
|
|
|
output,
|
|
|
|
type_to_string(tmpctx, struct amount_sat,
|
2019-03-25 11:35:56 +01:00
|
|
|
&utxo->amount),
|
2017-09-12 13:50:47 +02:00
|
|
|
is_p2sh ? "P2SH" : "SEGWIT",
|
2018-03-15 05:30:39 +01:00
|
|
|
type_to_string(tmpctx, struct bitcoin_txid,
|
2019-02-15 17:18:39 +01:00
|
|
|
&utxo->txid), blockheight ? " CONFIRMED" : "");
|
2017-09-12 13:50:47 +02:00
|
|
|
|
2020-04-03 23:52:35 +02:00
|
|
|
/* We only record final ledger movements */
|
|
|
|
if (blockheight) {
|
2020-04-15 05:08:06 +02:00
|
|
|
mvt = new_coin_deposit_sat(utxo, "wallet", &utxo->txid, utxo->outnum,
|
|
|
|
blockheight ? *blockheight : 0,
|
2020-04-15 05:40:28 +02:00
|
|
|
utxo->amount);
|
2020-04-03 23:52:35 +02:00
|
|
|
notify_chain_mvt(w->ld, mvt);
|
|
|
|
}
|
|
|
|
|
2017-09-12 13:50:47 +02:00
|
|
|
if (!wallet_add_utxo(w, utxo, is_p2sh ? p2sh_wpkh : our_change)) {
|
2018-03-02 14:33:01 +01:00
|
|
|
/* In case we already know the output, make
|
|
|
|
* sure we actually track its
|
|
|
|
* blockheight. This can happen when we grab
|
|
|
|
* the output from a transaction we created
|
2018-03-23 11:01:43 +01:00
|
|
|
* ourselves. */
|
2018-03-22 00:13:16 +01:00
|
|
|
if (blockheight)
|
2018-08-13 05:03:08 +02:00
|
|
|
wallet_confirm_tx(w, &utxo->txid, *blockheight);
|
2017-09-01 14:33:21 +02:00
|
|
|
tal_free(utxo);
|
2018-03-02 15:18:20 +01:00
|
|
|
continue;
|
2017-09-01 14:33:21 +02:00
|
|
|
}
|
2019-09-03 23:33:10 +02:00
|
|
|
|
|
|
|
/* This is an unconfirmed change output, we should track it */
|
|
|
|
if (!is_p2sh && !blockheight)
|
|
|
|
txfilter_add_scriptpubkey(w->ld->owned_txfilter, script);
|
|
|
|
|
2018-02-26 14:57:13 +01:00
|
|
|
outpointfilter_add(w->owned_outpoints, &utxo->txid, utxo->outnum);
|
|
|
|
|
2019-02-21 04:45:55 +01:00
|
|
|
if (!amount_sat_add(total, *total, utxo->amount))
|
|
|
|
fatal("Cannot add utxo output %zu/%zu %s + %s",
|
2020-06-16 20:43:51 +02:00
|
|
|
output, wtx->num_outputs,
|
2019-02-21 04:45:55 +01:00
|
|
|
type_to_string(tmpctx, struct amount_sat, total),
|
|
|
|
type_to_string(tmpctx, struct amount_sat,
|
|
|
|
&utxo->amount));
|
2019-09-30 22:25:00 +02:00
|
|
|
|
|
|
|
wallet_annotate_txout(w, &utxo->txid, output, TX_WALLET_DEPOSIT, 0);
|
2017-12-13 21:03:24 +01:00
|
|
|
tal_free(utxo);
|
2017-09-01 14:33:21 +02:00
|
|
|
num_utxos++;
|
|
|
|
}
|
|
|
|
return num_utxos;
|
|
|
|
}
|
|
|
|
|
2017-11-01 02:10:48 +01:00
|
|
|
void wallet_htlc_save_in(struct wallet *wallet,
|
2018-02-12 11:12:55 +01:00
|
|
|
const struct channel *chan, struct htlc_in *in)
|
2017-09-14 21:27:41 +02:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2017-09-14 21:27:41 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(wallet->db,
|
|
|
|
SQL("INSERT INTO channel_htlcs ("
|
|
|
|
" channel_id,"
|
|
|
|
" channel_htlc_id, "
|
|
|
|
" direction,"
|
|
|
|
" msatoshi,"
|
|
|
|
" cltv_expiry,"
|
|
|
|
" payment_hash, "
|
|
|
|
" payment_key,"
|
|
|
|
" hstate,"
|
|
|
|
" shared_secret,"
|
|
|
|
" routing_onion,"
|
|
|
|
" received_time) VALUES "
|
|
|
|
"(?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?);"));
|
|
|
|
|
|
|
|
db_bind_u64(stmt, 0, chan->dbid);
|
|
|
|
db_bind_u64(stmt, 1, in->key.id);
|
|
|
|
db_bind_int(stmt, 2, DIRECTION_INCOMING);
|
|
|
|
db_bind_amount_msat(stmt, 3, &in->msat);
|
|
|
|
db_bind_int(stmt, 4, in->cltv_expiry);
|
|
|
|
db_bind_sha256(stmt, 5, &in->payment_hash);
|
2017-10-30 22:44:00 +01:00
|
|
|
|
2017-10-30 16:49:25 +01:00
|
|
|
if (in->preimage)
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_preimage(stmt, 6, in->preimage);
|
2018-06-15 12:54:22 +02:00
|
|
|
else
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 6);
|
|
|
|
db_bind_int(stmt, 7, in->hstate);
|
2017-10-30 16:49:25 +01:00
|
|
|
|
2019-01-08 01:19:50 +01:00
|
|
|
if (!in->shared_secret)
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 8);
|
2019-01-08 01:19:50 +01:00
|
|
|
else
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_secret(stmt, 8, in->shared_secret);
|
2017-10-30 22:44:00 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_blob(stmt, 9, in->onion_routing_packet,
|
|
|
|
sizeof(in->onion_routing_packet));
|
2017-10-30 16:49:25 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_timeabs(stmt, 10, in->received_time);
|
2019-04-03 19:01:47 +02:00
|
|
|
|
2019-08-28 22:38:32 +02:00
|
|
|
db_exec_prepared_v2(stmt);
|
|
|
|
in->dbid = db_last_insert_id_v2(take(stmt));
|
2017-09-14 21:27:41 +02:00
|
|
|
}
|
|
|
|
|
2017-11-01 02:10:48 +01:00
|
|
|
void wallet_htlc_save_out(struct wallet *wallet,
|
2018-02-12 11:12:55 +01:00
|
|
|
const struct channel *chan,
|
2017-09-14 21:27:41 +02:00
|
|
|
struct htlc_out *out)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2017-09-14 21:27:41 +02:00
|
|
|
|
|
|
|
/* We absolutely need the incoming HTLC to be persisted before
|
|
|
|
* we can persist it's dependent */
|
|
|
|
assert(out->in == NULL || out->in->dbid != 0);
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(
|
2017-10-30 16:49:25 +01:00
|
|
|
wallet->db,
|
2019-08-20 09:54:38 +02:00
|
|
|
SQL("INSERT INTO channel_htlcs ("
|
|
|
|
" channel_id,"
|
|
|
|
" channel_htlc_id,"
|
|
|
|
" direction,"
|
|
|
|
" origin_htlc,"
|
|
|
|
" msatoshi,"
|
|
|
|
" cltv_expiry,"
|
|
|
|
" payment_hash,"
|
|
|
|
" payment_key,"
|
|
|
|
" hstate,"
|
2019-12-12 00:39:07 +01:00
|
|
|
" routing_onion,"
|
|
|
|
" partid) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?);"));
|
2019-08-20 09:54:38 +02:00
|
|
|
|
|
|
|
db_bind_u64(stmt, 0, chan->dbid);
|
|
|
|
db_bind_u64(stmt, 1, out->key.id);
|
|
|
|
db_bind_int(stmt, 2, DIRECTION_OUTGOING);
|
2017-10-30 16:49:25 +01:00
|
|
|
if (out->in)
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_u64(stmt, 3, out->in->dbid);
|
2018-06-14 15:19:02 +02:00
|
|
|
else
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 3);
|
|
|
|
db_bind_amount_msat(stmt, 4, &out->msat);
|
|
|
|
db_bind_int(stmt, 5, out->cltv_expiry);
|
|
|
|
db_bind_sha256(stmt, 6, &out->payment_hash);
|
2017-10-30 22:44:00 +01:00
|
|
|
|
2017-10-30 16:49:25 +01:00
|
|
|
if (out->preimage)
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_preimage(stmt, 7, out->preimage);
|
2018-06-15 12:56:44 +02:00
|
|
|
else
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 7);
|
|
|
|
db_bind_int(stmt, 8, out->hstate);
|
2017-10-30 16:49:25 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_blob(stmt, 9, out->onion_routing_packet,
|
|
|
|
sizeof(out->onion_routing_packet));
|
2019-12-12 00:39:07 +01:00
|
|
|
if (!out->am_origin)
|
|
|
|
db_bind_null(stmt, 10);
|
|
|
|
else
|
|
|
|
db_bind_u64(stmt, 10, out->partid);
|
2017-09-14 21:27:41 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(stmt);
|
|
|
|
out->dbid = db_last_insert_id_v2(stmt);
|
|
|
|
tal_free(stmt);
|
2017-09-14 21:27:41 +02:00
|
|
|
}
|
|
|
|
|
2020-03-19 00:28:29 +01:00
|
|
|
/* input htlcs use failcode & failonion & we_filled, output htlcs use failmsg & failonion */
|
2017-11-01 02:10:48 +01:00
|
|
|
void wallet_htlc_update(struct wallet *wallet, const u64 htlc_dbid,
|
2017-09-14 21:27:41 +02:00
|
|
|
const enum htlc_state new_state,
|
2018-10-09 10:56:52 +02:00
|
|
|
const struct preimage *payment_key,
|
2020-08-31 03:13:25 +02:00
|
|
|
enum onion_wire badonion,
|
2020-02-18 01:00:58 +01:00
|
|
|
const struct onionreply *failonion,
|
2020-03-19 00:28:29 +01:00
|
|
|
const u8 *failmsg,
|
2020-04-14 21:02:46 +02:00
|
|
|
bool *we_filled)
|
2017-09-14 21:27:41 +02:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2017-09-14 21:27:41 +02:00
|
|
|
|
2020-02-21 06:10:40 +01:00
|
|
|
/* We should only use this for badonion codes */
|
|
|
|
assert(!badonion || (badonion & BADONION));
|
|
|
|
|
2017-09-18 20:18:29 +02:00
|
|
|
/* The database ID must be set by a previous call to
|
|
|
|
* `wallet_htlc_save_*` */
|
|
|
|
assert(htlc_dbid);
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(
|
|
|
|
wallet->db, SQL("UPDATE channel_htlcs SET hstate=?, payment_key=?, "
|
2020-03-19 00:28:29 +01:00
|
|
|
"malformed_onion=?, failuremsg=?, localfailmsg=?, "
|
|
|
|
"we_filled=?"
|
2020-02-18 01:00:58 +01:00
|
|
|
" WHERE id=?"));
|
2017-10-30 22:44:00 +01:00
|
|
|
|
2018-08-09 04:27:25 +02:00
|
|
|
/* FIXME: htlc_state_in_db */
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_int(stmt, 0, new_state);
|
2020-03-19 00:28:29 +01:00
|
|
|
db_bind_u64(stmt, 6, htlc_dbid);
|
2017-09-14 21:27:41 +02:00
|
|
|
|
2017-10-30 22:44:00 +01:00
|
|
|
if (payment_key)
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_preimage(stmt, 1, payment_key);
|
2018-06-15 12:57:56 +02:00
|
|
|
else
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 1);
|
2017-10-30 22:44:00 +01:00
|
|
|
|
2020-02-21 06:10:40 +01:00
|
|
|
db_bind_int(stmt, 2, badonion);
|
2020-02-18 01:00:58 +01:00
|
|
|
|
2020-02-18 00:53:58 +01:00
|
|
|
if (failonion)
|
|
|
|
db_bind_onionreply(stmt, 3, failonion);
|
2018-10-09 10:56:52 +02:00
|
|
|
else
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 3);
|
2018-10-09 10:56:52 +02:00
|
|
|
|
2020-08-25 23:20:52 +02:00
|
|
|
db_bind_talarr(stmt, 4, failmsg);
|
2020-02-18 01:00:58 +01:00
|
|
|
|
2020-03-19 00:28:29 +01:00
|
|
|
if (we_filled)
|
2020-04-14 21:02:46 +02:00
|
|
|
db_bind_int(stmt, 5, *we_filled);
|
2020-03-19 00:28:29 +01:00
|
|
|
else
|
|
|
|
db_bind_null(stmt, 5);
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2017-09-14 21:27:41 +02:00
|
|
|
}
|
|
|
|
|
2018-02-12 11:13:04 +01:00
|
|
|
static bool wallet_stmt2htlc_in(struct channel *channel,
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt, struct htlc_in *in)
|
2017-09-14 21:27:41 +02:00
|
|
|
{
|
|
|
|
bool ok = true;
|
2019-08-20 09:54:38 +02:00
|
|
|
in->dbid = db_column_u64(stmt, 0);
|
|
|
|
in->key.id = db_column_u64(stmt, 1);
|
2018-02-12 11:13:04 +01:00
|
|
|
in->key.channel = channel;
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_amount_msat(stmt, 2, &in->msat);
|
|
|
|
in->cltv_expiry = db_column_int(stmt, 3);
|
|
|
|
in->hstate = db_column_int(stmt, 4);
|
2020-04-11 05:22:40 +02:00
|
|
|
/* FIXME: save blinding in db !*/
|
|
|
|
in->blinding = NULL;
|
2017-09-14 21:27:41 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_sha256(stmt, 5, &in->payment_hash);
|
2017-10-30 22:44:00 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_column_is_null(stmt, 6)) {
|
2017-09-14 21:27:41 +02:00
|
|
|
in->preimage = tal(in, struct preimage);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_preimage(stmt, 6, in->preimage);
|
2017-09-14 21:27:41 +02:00
|
|
|
} else {
|
|
|
|
in->preimage = NULL;
|
|
|
|
}
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
assert(db_column_bytes(stmt, 7) == sizeof(in->onion_routing_packet));
|
|
|
|
memcpy(&in->onion_routing_packet, db_column_blob(stmt, 7),
|
2017-10-30 22:44:00 +01:00
|
|
|
sizeof(in->onion_routing_packet));
|
2017-09-14 21:27:41 +02:00
|
|
|
|
2020-01-23 00:38:04 +01:00
|
|
|
if (db_column_is_null(stmt, 8))
|
2020-02-18 00:53:58 +01:00
|
|
|
in->failonion = NULL;
|
2020-01-23 00:38:04 +01:00
|
|
|
else
|
2020-02-18 00:53:58 +01:00
|
|
|
in->failonion = db_column_onionreply(in, stmt, 8);
|
2020-02-21 06:10:44 +01:00
|
|
|
in->badonion = db_column_int(stmt, 9);
|
2019-08-20 09:54:38 +02:00
|
|
|
if (db_column_is_null(stmt, 11)) {
|
2019-01-08 01:19:50 +01:00
|
|
|
in->shared_secret = NULL;
|
|
|
|
} else {
|
2019-08-20 09:54:38 +02:00
|
|
|
assert(db_column_bytes(stmt, 11) == sizeof(struct secret));
|
2019-01-08 01:19:50 +01:00
|
|
|
in->shared_secret = tal(in, struct secret);
|
2019-08-20 09:54:38 +02:00
|
|
|
memcpy(in->shared_secret, db_column_blob(stmt, 11),
|
2019-01-08 01:19:50 +01:00
|
|
|
sizeof(struct secret));
|
|
|
|
#ifdef COMPAT_V062
|
|
|
|
if (memeqzero(in->shared_secret, sizeof(*in->shared_secret)))
|
|
|
|
in->shared_secret = tal_free(in->shared_secret);
|
|
|
|
#endif
|
|
|
|
}
|
2017-09-14 21:27:41 +02:00
|
|
|
|
wallet: fix null column access for pre-0.7.3 dbs.
Added in d901304120, this column is null in old dbs like mine:
2020-02-15T00:08:41.444Z **BROKEN** database: Accessing a null column 12 in query SELECT id, channel_htlc_id, msatoshi, cltv_expiry, hstate, payment_hash, payment_key, routing_onion, failuremsg, malformed_onion, origin_htlc, shared_secret, received_time FROM channel_htlcs WHERE direction= ? AND channel_id= ? AND hstate != ?
Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
2020-02-17 00:48:49 +01:00
|
|
|
#ifdef COMPAT_V072
|
|
|
|
if (db_column_is_null(stmt, 12)) {
|
|
|
|
in->received_time.ts.tv_sec = 0;
|
|
|
|
in->received_time.ts.tv_nsec = 0;
|
|
|
|
} else
|
|
|
|
#endif /* COMPAT_V072 */
|
2019-08-20 09:54:38 +02:00
|
|
|
in->received_time = db_column_timeabs(stmt, 12);
|
2019-04-03 19:01:47 +02:00
|
|
|
|
2020-02-21 06:10:40 +01:00
|
|
|
#ifdef COMPAT_V080
|
|
|
|
/* This field is now reserved for badonion codes: the rest should
|
|
|
|
* use the failonion field. */
|
2020-02-21 06:10:44 +01:00
|
|
|
if (in->badonion && !(in->badonion & BADONION)) {
|
2020-02-21 06:10:40 +01:00
|
|
|
log_broken(channel->log,
|
|
|
|
"Replacing incoming HTLC %"PRIu64" error "
|
|
|
|
"%s with WIRE_TEMPORARY_NODE_FAILURE",
|
2020-08-31 03:13:25 +02:00
|
|
|
in->key.id, onion_wire_name(in->badonion));
|
2020-02-21 06:10:44 +01:00
|
|
|
in->badonion = 0;
|
2020-02-21 06:10:40 +01:00
|
|
|
in->failonion = create_onionreply(in,
|
|
|
|
in->shared_secret,
|
|
|
|
towire_temporary_node_failure(tmpctx));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-04-14 21:02:46 +02:00
|
|
|
if (!db_column_is_null(stmt, 13)) {
|
|
|
|
in->we_filled = tal(in, bool);
|
|
|
|
*in->we_filled = db_column_int(stmt, 13);
|
|
|
|
} else
|
|
|
|
in->we_filled = NULL;
|
2020-03-19 00:28:29 +01:00
|
|
|
|
2017-09-14 21:27:41 +02:00
|
|
|
return ok;
|
|
|
|
}
|
2018-10-09 10:55:52 +02:00
|
|
|
|
2019-12-12 00:39:10 +01:00
|
|
|
/* Removes matching htlc from unconnected_htlcs_in */
|
|
|
|
static bool wallet_stmt2htlc_out(struct wallet *wallet,
|
|
|
|
struct channel *channel,
|
|
|
|
struct db_stmt *stmt, struct htlc_out *out,
|
|
|
|
struct htlc_in_map *unconnected_htlcs_in)
|
2017-09-14 21:27:41 +02:00
|
|
|
{
|
|
|
|
bool ok = true;
|
2019-08-20 09:54:38 +02:00
|
|
|
out->dbid = db_column_u64(stmt, 0);
|
|
|
|
out->key.id = db_column_u64(stmt, 1);
|
2018-02-12 11:13:04 +01:00
|
|
|
out->key.channel = channel;
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_amount_msat(stmt, 2, &out->msat);
|
|
|
|
out->cltv_expiry = db_column_int(stmt, 3);
|
|
|
|
out->hstate = db_column_int(stmt, 4);
|
|
|
|
db_column_sha256(stmt, 5, &out->payment_hash);
|
2020-04-11 05:22:40 +02:00
|
|
|
/* FIXME: save blinding in db !*/
|
|
|
|
out->blinding = NULL;
|
2017-09-14 21:27:41 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_column_is_null(stmt, 6)) {
|
2017-09-14 21:27:41 +02:00
|
|
|
out->preimage = tal(out, struct preimage);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_preimage(stmt, 6, out->preimage);
|
2017-09-14 21:27:41 +02:00
|
|
|
} else {
|
|
|
|
out->preimage = NULL;
|
|
|
|
}
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
assert(db_column_bytes(stmt, 7) == sizeof(out->onion_routing_packet));
|
|
|
|
memcpy(&out->onion_routing_packet, db_column_blob(stmt, 7),
|
2017-10-30 22:44:00 +01:00
|
|
|
sizeof(out->onion_routing_packet));
|
2017-09-14 21:27:41 +02:00
|
|
|
|
2020-01-23 00:38:04 +01:00
|
|
|
if (db_column_is_null(stmt, 8))
|
2020-02-18 00:53:58 +01:00
|
|
|
out->failonion = NULL;
|
2020-01-23 00:38:04 +01:00
|
|
|
else
|
2020-02-18 00:53:58 +01:00
|
|
|
out->failonion = db_column_onionreply(out, stmt, 8);
|
2020-02-18 01:00:58 +01:00
|
|
|
|
2020-02-21 06:06:58 +01:00
|
|
|
if (db_column_is_null(stmt, 14))
|
|
|
|
out->failmsg = NULL;
|
|
|
|
else
|
|
|
|
out->failmsg = tal_dup_arr(out, u8, db_column_blob(stmt, 14),
|
|
|
|
db_column_bytes(stmt, 14), 0);
|
|
|
|
|
2019-12-12 00:39:10 +01:00
|
|
|
out->in = NULL;
|
2018-10-09 10:55:52 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_column_is_null(stmt, 10)) {
|
2019-12-12 00:39:10 +01:00
|
|
|
u64 in_id = db_column_u64(stmt, 10);
|
|
|
|
struct htlc_in *hin;
|
|
|
|
|
|
|
|
hin = remove_htlc_in_by_dbid(unconnected_htlcs_in, in_id);
|
|
|
|
if (hin)
|
|
|
|
htlc_out_connect_htlc_in(out, hin);
|
2018-10-09 22:21:31 +02:00
|
|
|
out->am_origin = false;
|
2019-12-12 00:39:10 +01:00
|
|
|
if (!out->in && !out->preimage) {
|
|
|
|
#ifdef COMPAT_V061
|
|
|
|
log_broken(wallet->log,
|
|
|
|
"Missing preimage for orphaned HTLC; replacing with zeros");
|
|
|
|
out->preimage = talz(out, struct preimage);
|
|
|
|
#else
|
|
|
|
fatal("Unable to find corresponding htlc_in %"PRIu64
|
|
|
|
" for unfulfilled htlc_out %"PRIu64,
|
|
|
|
in_id, out->dbid);
|
|
|
|
#endif
|
|
|
|
}
|
2018-10-09 10:55:52 +02:00
|
|
|
} else {
|
2019-12-12 00:39:07 +01:00
|
|
|
out->partid = db_column_u64(stmt, 13);
|
2018-10-09 22:21:31 +02:00
|
|
|
out->am_origin = true;
|
2018-10-09 10:55:52 +02:00
|
|
|
}
|
2017-09-14 21:27:41 +02:00
|
|
|
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
2018-10-09 10:54:52 +02:00
|
|
|
static void fixup_hin(struct wallet *wallet, struct htlc_in *hin)
|
|
|
|
{
|
2020-02-18 00:53:58 +01:00
|
|
|
/* We didn't used to save failcore, failonion... */
|
2018-10-09 10:57:52 +02:00
|
|
|
#ifdef COMPAT_V061
|
2018-10-09 10:54:52 +02:00
|
|
|
/* We care about HTLCs being removed only, not those being added. */
|
|
|
|
if (hin->hstate < SENT_REMOVE_HTLC)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* Successful ones are fine. */
|
|
|
|
if (hin->preimage)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* Failed ones (only happens after db fixed!) OK. */
|
2020-02-21 06:10:44 +01:00
|
|
|
if (hin->badonion || hin->failonion)
|
2018-10-09 10:54:52 +02:00
|
|
|
return;
|
|
|
|
|
2020-02-21 06:10:44 +01:00
|
|
|
hin->failonion = create_onionreply(hin,
|
|
|
|
hin->shared_secret,
|
|
|
|
towire_temporary_node_failure(tmpctx));
|
2018-10-09 10:54:52 +02:00
|
|
|
|
|
|
|
log_broken(wallet->log, "HTLC #%"PRIu64" (%s) "
|
2019-02-21 04:45:55 +01:00
|
|
|
" for amount %s"
|
2018-10-09 10:54:52 +02:00
|
|
|
" from %s"
|
|
|
|
" is missing a resolution:"
|
2018-10-09 10:57:52 +02:00
|
|
|
" subsituting temporary node failure",
|
2018-10-09 10:54:52 +02:00
|
|
|
hin->key.id, htlc_state_name(hin->hstate),
|
2019-02-21 04:45:55 +01:00
|
|
|
type_to_string(tmpctx, struct amount_msat, &hin->msat),
|
2019-04-08 11:58:32 +02:00
|
|
|
type_to_string(tmpctx, struct node_id,
|
2018-10-09 10:54:52 +02:00
|
|
|
&hin->key.channel->peer->id));
|
2018-10-09 10:56:52 +02:00
|
|
|
#endif
|
2018-10-09 10:57:52 +02:00
|
|
|
}
|
|
|
|
|
2019-12-12 00:39:10 +01:00
|
|
|
bool wallet_htlcs_load_in_for_channel(struct wallet *wallet,
|
|
|
|
struct channel *chan,
|
|
|
|
struct htlc_in_map *htlcs_in)
|
2017-09-14 21:27:41 +02:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2017-09-14 21:27:41 +02:00
|
|
|
bool ok = true;
|
2019-12-12 00:39:10 +01:00
|
|
|
int incount = 0;
|
2017-09-14 21:27:41 +02:00
|
|
|
|
2019-12-12 00:39:10 +01:00
|
|
|
log_debug(wallet->log, "Loading in HTLCs for channel %"PRIu64, chan->dbid);
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(wallet->db, SQL("SELECT"
|
|
|
|
" id"
|
|
|
|
", channel_htlc_id"
|
|
|
|
", msatoshi"
|
|
|
|
", cltv_expiry"
|
|
|
|
", hstate"
|
|
|
|
", payment_hash"
|
|
|
|
", payment_key"
|
|
|
|
", routing_onion"
|
|
|
|
", failuremsg"
|
|
|
|
", malformed_onion"
|
|
|
|
", origin_htlc"
|
|
|
|
", shared_secret"
|
|
|
|
", received_time"
|
2020-03-19 00:28:29 +01:00
|
|
|
", we_filled"
|
2019-08-20 09:54:38 +02:00
|
|
|
" FROM channel_htlcs"
|
|
|
|
" WHERE direction= ?"
|
|
|
|
" AND channel_id= ?"
|
|
|
|
" AND hstate != ?"));
|
|
|
|
db_bind_int(stmt, 0, DIRECTION_INCOMING);
|
|
|
|
db_bind_u64(stmt, 1, chan->dbid);
|
|
|
|
db_bind_int(stmt, 2, SENT_REMOVE_ACK_REVOCATION);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
while (db_step(stmt)) {
|
2017-09-14 21:27:41 +02:00
|
|
|
struct htlc_in *in = tal(chan, struct htlc_in);
|
|
|
|
ok &= wallet_stmt2htlc_in(chan, stmt, in);
|
|
|
|
connect_htlc_in(htlcs_in, in);
|
2018-10-09 10:54:52 +02:00
|
|
|
fixup_hin(wallet, in);
|
|
|
|
ok &= htlc_in_check(in, NULL) != NULL;
|
2017-09-14 21:27:41 +02:00
|
|
|
incount++;
|
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2017-09-14 21:27:41 +02:00
|
|
|
|
2019-12-12 00:39:10 +01:00
|
|
|
log_debug(wallet->log, "Restored %d incoming HTLCS", incount);
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool wallet_htlcs_load_out_for_channel(struct wallet *wallet,
|
|
|
|
struct channel *chan,
|
|
|
|
struct htlc_out_map *htlcs_out,
|
|
|
|
struct htlc_in_map *unconnected_htlcs_in)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
bool ok = true;
|
|
|
|
int outcount = 0;
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(wallet->db, SQL("SELECT"
|
2019-07-17 15:12:34 +02:00
|
|
|
" id"
|
2019-07-17 14:44:43 +02:00
|
|
|
", channel_htlc_id"
|
|
|
|
", msatoshi"
|
|
|
|
", cltv_expiry"
|
|
|
|
", hstate"
|
|
|
|
", payment_hash"
|
|
|
|
", payment_key"
|
|
|
|
", routing_onion"
|
|
|
|
", failuremsg"
|
|
|
|
", malformed_onion"
|
|
|
|
", origin_htlc"
|
|
|
|
", shared_secret"
|
|
|
|
", received_time"
|
2019-12-12 00:39:07 +01:00
|
|
|
", partid"
|
2020-02-18 01:00:58 +01:00
|
|
|
", localfailmsg"
|
2019-07-17 14:44:43 +02:00
|
|
|
" FROM channel_htlcs"
|
|
|
|
" WHERE direction = ?"
|
|
|
|
" AND channel_id = ?"
|
2019-08-20 09:54:38 +02:00
|
|
|
" AND hstate != ?"));
|
|
|
|
db_bind_int(stmt, 0, DIRECTION_OUTGOING);
|
|
|
|
db_bind_u64(stmt, 1, chan->dbid);
|
|
|
|
db_bind_int(stmt, 2, RCVD_REMOVE_ACK_REVOCATION);
|
|
|
|
db_query_prepared(stmt);
|
2017-09-14 21:27:41 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
while (db_step(stmt)) {
|
2017-09-14 21:27:41 +02:00
|
|
|
struct htlc_out *out = tal(chan, struct htlc_out);
|
2019-12-12 00:39:10 +01:00
|
|
|
ok &= wallet_stmt2htlc_out(wallet, chan, stmt, out,
|
|
|
|
unconnected_htlcs_in);
|
2017-09-14 21:27:41 +02:00
|
|
|
connect_htlc_out(htlcs_out, out);
|
|
|
|
/* Cannot htlc_out_check because we haven't wired the
|
|
|
|
* dependencies in yet */
|
|
|
|
outcount++;
|
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2019-03-15 03:50:18 +01:00
|
|
|
|
2019-12-12 00:39:10 +01:00
|
|
|
log_debug(wallet->log, "Restored %d outgoing HTLCS", outcount);
|
2017-09-14 21:27:41 +02:00
|
|
|
|
|
|
|
return ok;
|
|
|
|
}
|
2017-09-26 14:40:41 +02:00
|
|
|
|
2018-02-23 02:04:47 +01:00
|
|
|
bool wallet_invoice_create(struct wallet *wallet,
|
|
|
|
struct invoice *pinvoice,
|
2019-02-21 03:38:35 +01:00
|
|
|
const struct amount_msat *msat TAKES,
|
2019-06-12 02:38:54 +02:00
|
|
|
const struct json_escape *label TAKES,
|
2018-02-28 17:26:58 +01:00
|
|
|
u64 expiry,
|
|
|
|
const char *b11enc,
|
2018-07-24 20:24:18 +02:00
|
|
|
const char *description,
|
2019-11-23 01:19:23 +01:00
|
|
|
const u8 *features,
|
2018-02-28 17:26:58 +01:00
|
|
|
const struct preimage *r,
|
2020-12-14 02:18:24 +01:00
|
|
|
const struct sha256 *rhash,
|
|
|
|
const struct sha256 *local_offer_id)
|
2018-02-28 17:26:58 +01:00
|
|
|
{
|
2020-12-14 02:18:24 +01:00
|
|
|
return invoices_create(wallet->invoices, pinvoice, msat, label, expiry, b11enc, description, features, r, rhash, local_offer_id);
|
2018-01-14 15:15:30 +01:00
|
|
|
}
|
2018-02-23 02:04:47 +01:00
|
|
|
bool wallet_invoice_find_by_label(struct wallet *wallet,
|
|
|
|
struct invoice *pinvoice,
|
2019-06-12 02:38:54 +02:00
|
|
|
const struct json_escape *label)
|
2017-12-26 11:50:30 +01:00
|
|
|
{
|
2018-02-23 02:04:47 +01:00
|
|
|
return invoices_find_by_label(wallet->invoices, pinvoice, label);
|
2017-12-26 11:50:30 +01:00
|
|
|
}
|
2018-04-25 01:04:33 +02:00
|
|
|
bool wallet_invoice_find_by_rhash(struct wallet *wallet,
|
|
|
|
struct invoice *pinvoice,
|
|
|
|
const struct sha256 *rhash)
|
|
|
|
{
|
|
|
|
return invoices_find_by_rhash(wallet->invoices, pinvoice, rhash);
|
|
|
|
}
|
2018-02-23 02:04:47 +01:00
|
|
|
bool wallet_invoice_find_unpaid(struct wallet *wallet,
|
|
|
|
struct invoice *pinvoice,
|
|
|
|
const struct sha256 *rhash)
|
2017-12-19 12:52:03 +01:00
|
|
|
{
|
2018-02-23 02:04:47 +01:00
|
|
|
return invoices_find_unpaid(wallet->invoices, pinvoice, rhash);
|
2017-12-19 12:52:03 +01:00
|
|
|
}
|
2018-01-14 15:15:30 +01:00
|
|
|
bool wallet_invoice_delete(struct wallet *wallet,
|
2018-02-23 02:04:47 +01:00
|
|
|
struct invoice invoice)
|
2017-12-19 12:52:03 +01:00
|
|
|
{
|
2018-01-14 15:15:30 +01:00
|
|
|
return invoices_delete(wallet->invoices, invoice);
|
2017-12-19 12:52:03 +01:00
|
|
|
}
|
2018-02-26 13:37:53 +01:00
|
|
|
void wallet_invoice_delete_expired(struct wallet *wallet, u64 e)
|
|
|
|
{
|
|
|
|
invoices_delete_expired(wallet->invoices, e);
|
|
|
|
}
|
2018-02-24 13:11:14 +01:00
|
|
|
bool wallet_invoice_iterate(struct wallet *wallet,
|
|
|
|
struct invoice_iterator *it)
|
2017-10-04 14:43:55 +02:00
|
|
|
{
|
2018-02-24 13:11:14 +01:00
|
|
|
return invoices_iterate(wallet->invoices, it);
|
|
|
|
}
|
2018-07-27 12:57:02 +02:00
|
|
|
const struct invoice_details *
|
|
|
|
wallet_invoice_iterator_deref(const tal_t *ctx, struct wallet *wallet,
|
|
|
|
const struct invoice_iterator *it)
|
2018-02-24 13:11:14 +01:00
|
|
|
{
|
2018-07-27 12:57:02 +02:00
|
|
|
return invoices_iterator_deref(ctx, wallet->invoices, it);
|
2017-10-04 14:43:55 +02:00
|
|
|
}
|
2020-12-02 03:30:59 +01:00
|
|
|
bool wallet_invoice_resolve(struct wallet *wallet,
|
2018-02-23 02:04:47 +01:00
|
|
|
struct invoice invoice,
|
2019-02-21 04:45:55 +01:00
|
|
|
struct amount_msat msatoshi_received)
|
2017-10-04 14:43:55 +02:00
|
|
|
{
|
2020-12-02 03:30:59 +01:00
|
|
|
return invoices_resolve(wallet->invoices, invoice, msatoshi_received);
|
2017-09-26 14:40:41 +02:00
|
|
|
}
|
2018-01-14 15:15:30 +01:00
|
|
|
void wallet_invoice_waitany(const tal_t *ctx,
|
|
|
|
struct wallet *wallet,
|
|
|
|
u64 lastpay_index,
|
|
|
|
void (*cb)(const struct invoice *, void*),
|
|
|
|
void *cbarg)
|
2017-10-05 23:29:56 +02:00
|
|
|
{
|
2018-01-14 15:15:30 +01:00
|
|
|
invoices_waitany(ctx, wallet->invoices, lastpay_index, cb, cbarg);
|
2017-10-05 23:29:56 +02:00
|
|
|
}
|
2018-01-14 15:15:30 +01:00
|
|
|
void wallet_invoice_waitone(const tal_t *ctx,
|
|
|
|
struct wallet *wallet,
|
2018-02-23 02:04:47 +01:00
|
|
|
struct invoice invoice,
|
2018-01-14 15:15:30 +01:00
|
|
|
void (*cb)(const struct invoice *, void*),
|
|
|
|
void *cbarg)
|
|
|
|
{
|
|
|
|
invoices_waitone(ctx, wallet->invoices, invoice, cb, cbarg);
|
|
|
|
}
|
2018-07-27 12:57:02 +02:00
|
|
|
|
|
|
|
const struct invoice_details *wallet_invoice_details(const tal_t *ctx,
|
|
|
|
struct wallet *wallet,
|
|
|
|
struct invoice invoice)
|
2018-02-20 02:07:30 +01:00
|
|
|
{
|
2018-07-27 12:57:02 +02:00
|
|
|
return invoices_get_details(ctx, wallet->invoices, invoice);
|
2018-02-20 02:07:30 +01:00
|
|
|
}
|
2018-01-14 15:15:30 +01:00
|
|
|
|
2017-12-15 11:29:32 +01:00
|
|
|
struct htlc_stub *wallet_htlc_stubs(const tal_t *ctx, struct wallet *wallet,
|
2018-02-12 11:12:55 +01:00
|
|
|
struct channel *chan)
|
2017-10-10 13:04:32 +02:00
|
|
|
{
|
|
|
|
struct htlc_stub *stubs;
|
|
|
|
struct sha256 payment_hash;
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2017-10-10 13:04:32 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(wallet->db,
|
|
|
|
SQL("SELECT channel_id, direction, cltv_expiry, "
|
|
|
|
"channel_htlc_id, payment_hash "
|
|
|
|
"FROM channel_htlcs WHERE channel_id = ?;"));
|
|
|
|
|
|
|
|
db_bind_u64(stmt, 0, chan->dbid);
|
|
|
|
db_query_prepared(stmt);
|
2017-10-10 13:04:32 +02:00
|
|
|
|
|
|
|
stubs = tal_arr(ctx, struct htlc_stub, 0);
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
while (db_step(stmt)) {
|
2019-01-15 04:51:27 +01:00
|
|
|
struct htlc_stub stub;
|
2017-10-10 13:04:32 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
assert(db_column_u64(stmt, 0) == chan->dbid);
|
2017-10-10 13:04:32 +02:00
|
|
|
|
|
|
|
/* FIXME: merge these two enums */
|
2019-08-20 09:54:38 +02:00
|
|
|
stub.owner = db_column_int(stmt, 1)==DIRECTION_INCOMING?REMOTE:LOCAL;
|
|
|
|
stub.cltv_expiry = db_column_int(stmt, 2);
|
2019-09-12 22:46:34 +02:00
|
|
|
stub.id = db_column_u64(stmt, 3);
|
2017-10-10 13:04:32 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_sha256(stmt, 4, &payment_hash);
|
2019-01-15 04:51:27 +01:00
|
|
|
ripemd160(&stub.ripemd, payment_hash.u.u8, sizeof(payment_hash.u));
|
|
|
|
tal_arr_expand(&stubs, stub);
|
2017-10-10 13:04:32 +02:00
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2017-10-10 13:04:32 +02:00
|
|
|
return stubs;
|
|
|
|
}
|
2017-11-02 21:09:57 +01:00
|
|
|
|
2018-03-16 01:09:37 +01:00
|
|
|
void wallet_local_htlc_out_delete(struct wallet *wallet,
|
|
|
|
struct channel *chan,
|
2019-12-12 00:16:23 +01:00
|
|
|
const struct sha256 *payment_hash,
|
|
|
|
u64 partid)
|
2018-03-16 01:09:37 +01:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-03-16 01:09:37 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(wallet->db, SQL("DELETE FROM channel_htlcs"
|
|
|
|
" WHERE direction = ?"
|
|
|
|
" AND origin_htlc = ?"
|
2019-12-12 00:39:07 +01:00
|
|
|
" AND payment_hash = ?"
|
|
|
|
" AND partid = ?;"));
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_int(stmt, 0, DIRECTION_OUTGOING);
|
|
|
|
db_bind_int(stmt, 1, 0);
|
|
|
|
db_bind_sha256(stmt, 2, payment_hash);
|
2019-12-12 00:39:07 +01:00
|
|
|
db_bind_u64(stmt, 3, partid);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-03-16 01:09:37 +01:00
|
|
|
}
|
|
|
|
|
2018-01-17 21:29:49 +01:00
|
|
|
static struct wallet_payment *
|
2019-12-12 00:16:23 +01:00
|
|
|
find_unstored_payment(struct wallet *wallet,
|
|
|
|
const struct sha256 *payment_hash,
|
|
|
|
u64 partid)
|
2018-01-17 21:29:49 +01:00
|
|
|
{
|
|
|
|
struct wallet_payment *i;
|
|
|
|
|
|
|
|
list_for_each(&wallet->unstored_payments, i, list) {
|
2019-12-12 00:16:23 +01:00
|
|
|
if (sha256_eq(payment_hash, &i->payment_hash)
|
|
|
|
&& i->partid == partid)
|
2018-01-17 21:29:49 +01:00
|
|
|
return i;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-02-14 02:53:13 +01:00
|
|
|
static void destroy_unstored_payment(struct wallet_payment *payment)
|
2018-01-17 21:29:49 +01:00
|
|
|
{
|
|
|
|
list_del(&payment->list);
|
|
|
|
}
|
|
|
|
|
|
|
|
void wallet_payment_setup(struct wallet *wallet, struct wallet_payment *payment)
|
|
|
|
{
|
2019-12-12 00:16:23 +01:00
|
|
|
assert(!find_unstored_payment(wallet, &payment->payment_hash,
|
|
|
|
payment->partid));
|
2018-01-17 21:29:49 +01:00
|
|
|
|
|
|
|
list_add_tail(&wallet->unstored_payments, &payment->list);
|
2018-02-14 02:53:13 +01:00
|
|
|
tal_add_destructor(payment, destroy_unstored_payment);
|
2018-01-17 21:29:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void wallet_payment_store(struct wallet *wallet,
|
2020-01-07 21:00:51 +01:00
|
|
|
struct wallet_payment *payment TAKES)
|
2017-11-02 21:09:57 +01:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2020-01-07 21:00:51 +01:00
|
|
|
if (!find_unstored_payment(wallet, &payment->payment_hash, payment->partid)) {
|
2018-03-09 03:17:51 +01:00
|
|
|
/* Already stored on-disk */
|
|
|
|
#if DEVELOPER
|
|
|
|
/* Double-check that it is indeed stored to disk
|
|
|
|
* (catch bug, where we call this on a payment_hash
|
|
|
|
* we never paid to) */
|
2019-03-15 03:50:18 +01:00
|
|
|
bool res;
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt =
|
|
|
|
db_prepare_v2(wallet->db, SQL("SELECT status FROM payments"
|
2019-12-12 00:16:23 +01:00
|
|
|
" WHERE payment_hash=?"
|
|
|
|
" AND partid = ?;"));
|
2020-01-07 21:00:51 +01:00
|
|
|
db_bind_sha256(stmt, 0, &payment->payment_hash);
|
|
|
|
db_bind_u64(stmt, 1, payment->partid);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_query_prepared(stmt);
|
|
|
|
res = db_step(stmt);
|
2019-03-15 03:50:18 +01:00
|
|
|
assert(res);
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2018-03-09 03:17:51 +01:00
|
|
|
#endif
|
|
|
|
return;
|
|
|
|
}
|
2017-11-02 21:09:57 +01:00
|
|
|
|
|
|
|
/* Don't attempt to add the same payment twice */
|
|
|
|
assert(!payment->id);
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(
|
2017-11-02 21:09:57 +01:00
|
|
|
wallet->db,
|
2019-08-20 09:54:38 +02:00
|
|
|
SQL("INSERT INTO payments ("
|
2020-05-21 23:43:27 +02:00
|
|
|
" status,"
|
|
|
|
" payment_hash,"
|
|
|
|
" destination,"
|
|
|
|
" msatoshi,"
|
|
|
|
" timestamp,"
|
|
|
|
" path_secrets,"
|
|
|
|
" route_nodes,"
|
|
|
|
" route_channels,"
|
|
|
|
" msatoshi_sent,"
|
|
|
|
" description,"
|
|
|
|
" bolt11,"
|
|
|
|
" total_msat,"
|
2020-12-14 02:20:44 +01:00
|
|
|
" partid,"
|
|
|
|
" local_offer_id"
|
|
|
|
") VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?);"));
|
2019-08-20 09:54:38 +02:00
|
|
|
|
|
|
|
db_bind_int(stmt, 0, payment->status);
|
|
|
|
db_bind_sha256(stmt, 1, &payment->payment_hash);
|
2019-11-07 23:13:29 +01:00
|
|
|
|
|
|
|
if (payment->destination != NULL)
|
|
|
|
db_bind_node_id(stmt, 2, payment->destination);
|
|
|
|
else
|
|
|
|
db_bind_null(stmt, 2);
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_amount_msat(stmt, 3, &payment->msatoshi);
|
|
|
|
db_bind_int(stmt, 4, payment->timestamp);
|
|
|
|
|
2019-11-11 17:18:10 +01:00
|
|
|
if (payment->path_secrets != NULL)
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_secret_arr(stmt, 5, payment->path_secrets);
|
2019-11-11 17:18:10 +01:00
|
|
|
else
|
|
|
|
db_bind_null(stmt, 5);
|
|
|
|
|
|
|
|
assert((payment->route_channels == NULL) == (payment->route_nodes == NULL));
|
|
|
|
if (payment->route_nodes) {
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_node_id_arr(stmt, 6, payment->route_nodes);
|
|
|
|
db_bind_short_channel_id_arr(stmt, 7, payment->route_channels);
|
|
|
|
} else {
|
|
|
|
db_bind_null(stmt, 6);
|
|
|
|
db_bind_null(stmt, 7);
|
|
|
|
}
|
|
|
|
|
|
|
|
db_bind_amount_msat(stmt, 8, &payment->msatoshi_sent);
|
2017-11-02 21:09:57 +01:00
|
|
|
|
2019-02-23 04:11:14 +01:00
|
|
|
if (payment->label != NULL)
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_text(stmt, 9, payment->label);
|
2018-07-20 15:57:00 +02:00
|
|
|
else
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 9);
|
2018-07-20 15:57:00 +02:00
|
|
|
|
2020-12-14 02:24:37 +01:00
|
|
|
if (payment->invstring != NULL)
|
|
|
|
db_bind_text(stmt, 10, payment->invstring);
|
2019-02-23 04:11:12 +01:00
|
|
|
else
|
2020-08-19 12:38:50 +02:00
|
|
|
db_bind_null(stmt, 10);
|
2018-01-17 21:29:49 +01:00
|
|
|
|
2019-12-12 00:16:23 +01:00
|
|
|
db_bind_amount_msat(stmt, 11, &payment->total_msat);
|
|
|
|
db_bind_u64(stmt, 12, payment->partid);
|
|
|
|
|
2020-12-14 02:20:44 +01:00
|
|
|
if (payment->local_offer_id != NULL)
|
|
|
|
db_bind_sha256(stmt, 13, payment->local_offer_id);
|
|
|
|
else
|
|
|
|
db_bind_null(stmt, 13);
|
|
|
|
|
2020-01-07 21:00:51 +01:00
|
|
|
db_exec_prepared_v2(stmt);
|
|
|
|
payment->id = db_last_insert_id_v2(stmt);
|
|
|
|
assert(payment->id > 0);
|
|
|
|
tal_free(stmt);
|
|
|
|
|
|
|
|
if (taken(payment)) {
|
|
|
|
tal_free(payment);
|
|
|
|
} else {
|
|
|
|
list_del(&payment->list);
|
|
|
|
tal_del_destructor(payment, destroy_unstored_payment);
|
|
|
|
}
|
2017-11-02 21:09:57 +01:00
|
|
|
}
|
|
|
|
|
2018-01-17 21:29:49 +01:00
|
|
|
void wallet_payment_delete(struct wallet *wallet,
|
2019-12-12 00:16:23 +01:00
|
|
|
const struct sha256 *payment_hash,
|
|
|
|
u64 partid)
|
2018-01-17 21:29:49 +01:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-01-17 21:29:49 +01:00
|
|
|
struct wallet_payment *payment;
|
|
|
|
|
2019-12-12 00:16:23 +01:00
|
|
|
payment = find_unstored_payment(wallet, payment_hash, partid);
|
2018-01-17 21:29:49 +01:00
|
|
|
if (payment) {
|
|
|
|
tal_free(payment);
|
|
|
|
return;
|
|
|
|
}
|
2018-01-17 21:29:49 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(
|
2019-12-12 00:16:23 +01:00
|
|
|
wallet->db, SQL("DELETE FROM payments WHERE payment_hash = ?"
|
|
|
|
" AND partid = ?"));
|
2018-01-17 21:29:49 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_sha256(stmt, 0, payment_hash);
|
2019-12-12 00:16:23 +01:00
|
|
|
db_bind_u64(stmt, 1, partid);
|
2018-01-17 21:29:49 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-01-17 21:29:49 +01:00
|
|
|
}
|
|
|
|
|
2020-08-08 17:47:53 +02:00
|
|
|
void wallet_payment_delete_by_hash(struct wallet *wallet,
|
|
|
|
const struct sha256 *payment_hash)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
stmt = db_prepare_v2(
|
|
|
|
wallet->db, SQL("DELETE FROM payments WHERE payment_hash = ?"));
|
|
|
|
|
|
|
|
db_bind_sha256(stmt, 0, payment_hash);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
|
|
|
}
|
|
|
|
|
2017-11-02 21:09:57 +01:00
|
|
|
static struct wallet_payment *wallet_stmt2payment(const tal_t *ctx,
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt)
|
2017-11-02 21:09:57 +01:00
|
|
|
{
|
|
|
|
struct wallet_payment *payment = tal(ctx, struct wallet_payment);
|
2019-08-20 09:54:38 +02:00
|
|
|
payment->id = db_column_u64(stmt, 0);
|
|
|
|
payment->status = db_column_int(stmt, 1);
|
2017-11-02 21:09:57 +01:00
|
|
|
|
2019-11-07 23:13:29 +01:00
|
|
|
if (!db_column_is_null(stmt, 2)) {
|
|
|
|
payment->destination = tal(payment, struct node_id);
|
|
|
|
db_column_node_id(stmt, 2, payment->destination);
|
|
|
|
} else {
|
|
|
|
payment->destination = NULL;
|
|
|
|
}
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_amount_msat(stmt, 3, &payment->msatoshi);
|
|
|
|
db_column_sha256(stmt, 4, &payment->payment_hash);
|
2017-11-02 21:09:57 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
payment->timestamp = db_column_int(stmt, 5);
|
|
|
|
if (!db_column_is_null(stmt, 6)) {
|
2018-01-17 21:29:49 +01:00
|
|
|
payment->payment_preimage = tal(payment, struct preimage);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_preimage(stmt, 6, payment->payment_preimage);
|
2018-01-17 21:29:49 +01:00
|
|
|
} else
|
|
|
|
payment->payment_preimage = NULL;
|
2018-01-17 21:29:49 +01:00
|
|
|
|
2019-11-11 17:18:10 +01:00
|
|
|
/* We either used `sendpay` or `sendonion` with the `shared_secrets`
|
|
|
|
* argument. */
|
|
|
|
if (!db_column_is_null(stmt, 7))
|
2019-08-20 09:54:38 +02:00
|
|
|
payment->path_secrets = db_column_secret_arr(payment, stmt, 7);
|
2019-11-11 17:18:10 +01:00
|
|
|
else
|
|
|
|
payment->path_secrets = NULL;
|
2018-01-20 15:00:35 +01:00
|
|
|
|
2019-11-11 17:18:10 +01:00
|
|
|
/* Either none, or both are set */
|
|
|
|
assert(db_column_is_null(stmt, 8) == db_column_is_null(stmt, 9));
|
|
|
|
if (!db_column_is_null(stmt, 8)) {
|
2019-08-20 09:54:38 +02:00
|
|
|
payment->route_nodes = db_column_node_id_arr(payment, stmt, 8);
|
|
|
|
payment->route_channels =
|
|
|
|
db_column_short_channel_id_arr(payment, stmt, 9);
|
2019-11-08 17:14:53 +01:00
|
|
|
} else {
|
|
|
|
payment->route_nodes = NULL;
|
|
|
|
payment->route_channels = NULL;
|
2019-08-20 09:54:38 +02:00
|
|
|
}
|
2018-01-20 15:00:35 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_amount_msat(stmt, 10, &payment->msatoshi_sent);
|
2018-03-22 14:21:18 +01:00
|
|
|
|
2019-09-12 22:46:34 +02:00
|
|
|
if (!db_column_is_null(stmt, 11) && db_column_text(stmt, 11) != NULL)
|
2019-08-20 09:54:38 +02:00
|
|
|
payment->label =
|
|
|
|
tal_strdup(payment, (const char *)db_column_text(stmt, 11));
|
2018-07-20 15:57:00 +02:00
|
|
|
else
|
2019-02-23 04:11:14 +01:00
|
|
|
payment->label = NULL;
|
2018-07-20 15:57:00 +02:00
|
|
|
|
2019-09-12 22:46:34 +02:00
|
|
|
if (!db_column_is_null(stmt, 12) && db_column_text(stmt, 12) != NULL)
|
2020-12-14 02:24:37 +01:00
|
|
|
payment->invstring = tal_strdup(
|
2019-08-20 09:54:38 +02:00
|
|
|
payment, (const char *)db_column_text(stmt, 12));
|
2019-02-23 04:11:12 +01:00
|
|
|
else
|
2020-12-14 02:24:37 +01:00
|
|
|
payment->invstring = NULL;
|
2019-02-23 04:11:12 +01:00
|
|
|
|
2019-11-11 19:38:27 +01:00
|
|
|
if (!db_column_is_null(stmt, 13))
|
|
|
|
payment->failonion =
|
|
|
|
tal_dup_arr(payment, u8, db_column_blob(stmt, 13),
|
|
|
|
db_column_bytes(stmt, 13), 0);
|
|
|
|
else
|
|
|
|
payment->failonion = NULL;
|
|
|
|
|
2020-07-27 16:00:26 +02:00
|
|
|
if (!db_column_is_null(stmt, 14))
|
|
|
|
db_column_amount_msat(stmt, 14, &payment->total_msat);
|
|
|
|
else
|
|
|
|
payment->total_msat = AMOUNT_MSAT(0);
|
|
|
|
|
|
|
|
if (!db_column_is_null(stmt, 15))
|
|
|
|
payment->partid = db_column_u64(stmt, 15);
|
|
|
|
else
|
|
|
|
payment->partid = 0;
|
|
|
|
|
2020-12-14 02:20:44 +01:00
|
|
|
if (!db_column_is_null(stmt, 16)) {
|
|
|
|
payment->local_offer_id = tal(payment, struct sha256);
|
|
|
|
db_column_sha256(stmt, 16, payment->local_offer_id);
|
|
|
|
} else
|
|
|
|
payment->local_offer_id = NULL;
|
|
|
|
|
2017-11-02 21:09:57 +01:00
|
|
|
return payment;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct wallet_payment *
|
|
|
|
wallet_payment_by_hash(const tal_t *ctx, struct wallet *wallet,
|
2019-12-12 00:16:23 +01:00
|
|
|
const struct sha256 *payment_hash,
|
|
|
|
u64 partid)
|
2017-11-02 21:09:57 +01:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-01-17 21:29:49 +01:00
|
|
|
struct wallet_payment *payment;
|
|
|
|
|
|
|
|
/* Present the illusion that it's in the db... */
|
2019-12-12 00:16:23 +01:00
|
|
|
payment = find_unstored_payment(wallet, payment_hash, partid);
|
2018-01-17 21:29:49 +01:00
|
|
|
if (payment)
|
|
|
|
return payment;
|
2017-11-02 21:09:57 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(wallet->db, SQL("SELECT"
|
|
|
|
" id"
|
|
|
|
", status"
|
|
|
|
", destination"
|
|
|
|
", msatoshi"
|
|
|
|
", payment_hash"
|
|
|
|
", timestamp"
|
|
|
|
", payment_preimage"
|
|
|
|
", path_secrets"
|
|
|
|
", route_nodes"
|
|
|
|
", route_channels"
|
|
|
|
", msatoshi_sent"
|
|
|
|
", description"
|
|
|
|
", bolt11"
|
2019-11-11 19:38:27 +01:00
|
|
|
", failonionreply"
|
2019-12-12 00:16:23 +01:00
|
|
|
", total_msat"
|
|
|
|
", partid"
|
2020-12-14 02:20:44 +01:00
|
|
|
", local_offer_id"
|
2019-08-20 09:54:38 +02:00
|
|
|
" FROM payments"
|
2019-12-12 00:16:23 +01:00
|
|
|
" WHERE payment_hash = ?"
|
|
|
|
" AND partid = ?"));
|
2019-08-20 09:54:38 +02:00
|
|
|
|
|
|
|
db_bind_sha256(stmt, 0, payment_hash);
|
2019-12-12 00:16:23 +01:00
|
|
|
db_bind_u64(stmt, 1, partid);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_query_prepared(stmt);
|
|
|
|
if (db_step(stmt)) {
|
2017-11-02 21:09:57 +01:00
|
|
|
payment = wallet_stmt2payment(ctx, stmt);
|
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2017-11-02 21:09:57 +01:00
|
|
|
return payment;
|
|
|
|
}
|
|
|
|
|
|
|
|
void wallet_payment_set_status(struct wallet *wallet,
|
|
|
|
const struct sha256 *payment_hash,
|
2019-12-12 00:16:23 +01:00
|
|
|
u64 partid,
|
2018-01-17 21:29:49 +01:00
|
|
|
const enum wallet_payment_status newstatus,
|
|
|
|
const struct preimage *preimage)
|
2017-11-02 21:09:57 +01:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-01-17 21:29:50 +01:00
|
|
|
struct wallet_payment *payment;
|
2017-11-02 21:09:57 +01:00
|
|
|
|
2018-01-17 21:29:50 +01:00
|
|
|
/* We can only fail an unstored payment! */
|
2019-12-12 00:16:23 +01:00
|
|
|
payment = find_unstored_payment(wallet, payment_hash, partid);
|
2018-01-17 21:29:50 +01:00
|
|
|
if (payment) {
|
|
|
|
assert(newstatus == PAYMENT_FAILED);
|
|
|
|
tal_free(payment);
|
|
|
|
return;
|
|
|
|
}
|
2018-01-17 21:29:49 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(wallet->db,
|
|
|
|
SQL("UPDATE payments SET status=? "
|
2019-12-12 00:16:23 +01:00
|
|
|
"WHERE payment_hash=? AND partid=?"));
|
2017-11-02 21:09:57 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_int(stmt, 0, wallet_payment_status_in_db(newstatus));
|
|
|
|
db_bind_sha256(stmt, 1, payment_hash);
|
2019-12-12 00:16:23 +01:00
|
|
|
db_bind_u64(stmt, 2, partid);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-01-17 21:29:49 +01:00
|
|
|
|
|
|
|
if (preimage) {
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(wallet->db,
|
|
|
|
SQL("UPDATE payments SET payment_preimage=? "
|
2019-12-12 00:16:23 +01:00
|
|
|
"WHERE payment_hash=? AND partid=?"));
|
2018-01-17 21:29:49 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_preimage(stmt, 0, preimage);
|
|
|
|
db_bind_sha256(stmt, 1, payment_hash);
|
2019-12-12 00:16:23 +01:00
|
|
|
db_bind_u64(stmt, 2, partid);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-01-17 21:29:49 +01:00
|
|
|
}
|
2018-03-21 13:14:34 +01:00
|
|
|
if (newstatus != PAYMENT_PENDING) {
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt =
|
|
|
|
db_prepare_v2(wallet->db, SQL("UPDATE payments"
|
|
|
|
" SET path_secrets = NULL"
|
|
|
|
" , route_nodes = NULL"
|
|
|
|
" , route_channels = NULL"
|
2019-12-12 00:16:23 +01:00
|
|
|
" WHERE payment_hash = ?"
|
|
|
|
" AND partid = ?;"));
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_sha256(stmt, 0, payment_hash);
|
2019-12-12 00:16:23 +01:00
|
|
|
db_bind_u64(stmt, 1, partid);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-03-21 13:14:34 +01:00
|
|
|
}
|
2017-11-02 21:09:57 +01:00
|
|
|
}
|
2017-11-16 19:11:59 +01:00
|
|
|
|
2018-03-07 16:41:23 +01:00
|
|
|
void wallet_payment_get_failinfo(const tal_t *ctx,
|
|
|
|
struct wallet *wallet,
|
|
|
|
const struct sha256 *payment_hash,
|
2019-12-12 00:16:23 +01:00
|
|
|
u64 partid,
|
2018-03-07 16:41:23 +01:00
|
|
|
/* outputs */
|
2020-01-23 00:38:04 +01:00
|
|
|
struct onionreply **failonionreply,
|
2018-03-07 16:41:23 +01:00
|
|
|
bool *faildestperm,
|
|
|
|
int *failindex,
|
2020-08-31 03:13:25 +02:00
|
|
|
enum onion_wire *failcode,
|
2019-04-08 11:58:32 +02:00
|
|
|
struct node_id **failnode,
|
2018-03-07 16:41:23 +01:00
|
|
|
struct short_channel_id **failchannel,
|
2018-04-16 15:29:40 +02:00
|
|
|
u8 **failupdate,
|
2019-01-15 05:02:27 +01:00
|
|
|
char **faildetail,
|
|
|
|
int *faildirection)
|
2018-03-07 16:41:23 +01:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-03-07 16:41:23 +01:00
|
|
|
bool resb;
|
|
|
|
size_t len;
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(wallet->db,
|
|
|
|
SQL("SELECT failonionreply, faildestperm"
|
2019-03-15 03:50:18 +01:00
|
|
|
", failindex, failcode"
|
|
|
|
", failnode, failchannel"
|
|
|
|
", failupdate, faildetail, faildirection"
|
|
|
|
" FROM payments"
|
2019-12-12 00:16:23 +01:00
|
|
|
" WHERE payment_hash=? AND partid=?;"));
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_sha256(stmt, 0, payment_hash);
|
2019-12-12 00:16:23 +01:00
|
|
|
db_bind_u64(stmt, 1, partid);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_query_prepared(stmt);
|
|
|
|
resb = db_step(stmt);
|
2019-03-15 03:50:18 +01:00
|
|
|
assert(resb);
|
2019-08-20 09:54:38 +02:00
|
|
|
|
|
|
|
if (db_column_is_null(stmt, 0))
|
2018-03-07 16:41:23 +01:00
|
|
|
*failonionreply = NULL;
|
|
|
|
else {
|
2020-01-23 00:38:04 +01:00
|
|
|
*failonionreply = db_column_onionreply(ctx, stmt, 0);
|
2018-03-07 16:41:23 +01:00
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
*faildestperm = db_column_int(stmt, 1) != 0;
|
|
|
|
*failindex = db_column_int(stmt, 2);
|
2020-08-31 03:13:25 +02:00
|
|
|
*failcode = (enum onion_wire) db_column_int(stmt, 3);
|
2019-08-20 09:54:38 +02:00
|
|
|
if (db_column_is_null(stmt, 4))
|
2018-03-07 16:41:23 +01:00
|
|
|
*failnode = NULL;
|
|
|
|
else {
|
2019-04-08 11:58:32 +02:00
|
|
|
*failnode = tal(ctx, struct node_id);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_node_id(stmt, 4, *failnode);
|
2018-03-07 16:41:23 +01:00
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
if (db_column_is_null(stmt, 5))
|
2018-03-07 16:41:23 +01:00
|
|
|
*failchannel = NULL;
|
|
|
|
else {
|
|
|
|
*failchannel = tal(ctx, struct short_channel_id);
|
2019-08-20 09:54:38 +02:00
|
|
|
resb = db_column_short_channel_id(stmt, 5, *failchannel);
|
2018-03-07 16:41:23 +01:00
|
|
|
assert(resb);
|
2019-01-15 05:02:27 +01:00
|
|
|
|
|
|
|
/* For pre-0.6.2 dbs, direction will be 0 */
|
2019-08-20 09:54:38 +02:00
|
|
|
*faildirection = db_column_int(stmt, 8);
|
2018-03-07 16:41:23 +01:00
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
if (db_column_is_null(stmt, 6))
|
2018-03-07 16:41:23 +01:00
|
|
|
*failupdate = NULL;
|
|
|
|
else {
|
2019-08-20 09:54:38 +02:00
|
|
|
len = db_column_bytes(stmt, 6);
|
2018-03-07 16:41:23 +01:00
|
|
|
*failupdate = tal_arr(ctx, u8, len);
|
2019-08-20 09:54:38 +02:00
|
|
|
memcpy(*failupdate, db_column_blob(stmt, 6), len);
|
2018-03-07 16:41:23 +01:00
|
|
|
}
|
2019-11-08 17:14:53 +01:00
|
|
|
if (!db_column_is_null(stmt, 7))
|
|
|
|
*faildetail = tal_strndup(ctx, db_column_blob(stmt, 7),
|
|
|
|
db_column_bytes(stmt, 7));
|
|
|
|
else
|
|
|
|
*faildetail = NULL;
|
2018-03-07 16:41:23 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2018-03-07 16:41:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void wallet_payment_set_failinfo(struct wallet *wallet,
|
|
|
|
const struct sha256 *payment_hash,
|
2019-12-12 00:16:23 +01:00
|
|
|
u64 partid,
|
2020-01-23 00:38:04 +01:00
|
|
|
const struct onionreply *failonionreply,
|
2018-03-07 16:41:23 +01:00
|
|
|
bool faildestperm,
|
|
|
|
int failindex,
|
2020-08-31 03:13:25 +02:00
|
|
|
enum onion_wire failcode,
|
2019-04-08 11:58:32 +02:00
|
|
|
const struct node_id *failnode,
|
2018-03-07 16:41:23 +01:00
|
|
|
const struct short_channel_id *failchannel,
|
2018-04-16 15:29:40 +02:00
|
|
|
const u8 *failupdate /*tal_arr*/,
|
2019-01-15 05:02:27 +01:00
|
|
|
const char *faildetail,
|
|
|
|
int faildirection)
|
2018-03-07 16:41:23 +01:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(wallet->db, SQL("UPDATE payments"
|
|
|
|
" SET failonionreply=?"
|
|
|
|
" , faildestperm=?"
|
|
|
|
" , failindex=?"
|
|
|
|
" , failcode=?"
|
|
|
|
" , failnode=?"
|
|
|
|
" , failchannel=?"
|
|
|
|
" , failupdate=?"
|
|
|
|
" , faildetail=?"
|
|
|
|
" , faildirection=?"
|
2019-12-12 00:16:23 +01:00
|
|
|
" WHERE payment_hash=?"
|
|
|
|
" AND partid=?;"));
|
2018-03-07 16:41:23 +01:00
|
|
|
if (failonionreply)
|
2020-08-25 23:20:52 +02:00
|
|
|
db_bind_talarr(stmt, 0, failonionreply->contents);
|
2018-03-07 16:41:23 +01:00
|
|
|
else
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 0);
|
|
|
|
db_bind_int(stmt, 1, faildestperm ? 1 : 0);
|
|
|
|
db_bind_int(stmt, 2, failindex);
|
|
|
|
db_bind_int(stmt, 3, (int) failcode);
|
|
|
|
|
2018-03-07 16:41:23 +01:00
|
|
|
if (failnode)
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_node_id(stmt, 4, failnode);
|
2018-03-07 16:41:23 +01:00
|
|
|
else
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 4);
|
|
|
|
|
2018-03-07 16:41:23 +01:00
|
|
|
if (failchannel) {
|
2019-11-11 18:33:04 +01:00
|
|
|
db_bind_short_channel_id(stmt, 5, failchannel);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_int(stmt, 8, faildirection);
|
2019-01-15 05:02:27 +01:00
|
|
|
} else {
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 5);
|
|
|
|
db_bind_null(stmt, 8);
|
2019-01-15 05:02:27 +01:00
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
|
2020-08-25 23:20:52 +02:00
|
|
|
db_bind_talarr(stmt, 6, failupdate);
|
2018-03-07 16:41:23 +01:00
|
|
|
|
2019-11-08 17:14:53 +01:00
|
|
|
if (faildetail != NULL)
|
|
|
|
db_bind_text(stmt, 7, faildetail);
|
|
|
|
else
|
|
|
|
db_bind_null(stmt, 7);
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_sha256(stmt, 9, payment_hash);
|
2019-12-12 00:16:23 +01:00
|
|
|
db_bind_u64(stmt, 10, partid);
|
2018-03-07 16:41:23 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-03-07 16:41:23 +01:00
|
|
|
}
|
|
|
|
|
2018-01-16 20:44:32 +01:00
|
|
|
const struct wallet_payment **
|
|
|
|
wallet_payment_list(const tal_t *ctx,
|
|
|
|
struct wallet *wallet,
|
|
|
|
const struct sha256 *payment_hash)
|
2017-11-16 19:11:59 +01:00
|
|
|
{
|
|
|
|
const struct wallet_payment **payments;
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-01-17 21:29:49 +01:00
|
|
|
struct wallet_payment *p;
|
|
|
|
size_t i;
|
2017-11-16 19:11:59 +01:00
|
|
|
|
|
|
|
payments = tal_arr(ctx, const struct wallet_payment *, 0);
|
2018-01-16 20:44:32 +01:00
|
|
|
if (payment_hash) {
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt =
|
|
|
|
db_prepare_v2(wallet->db, SQL("SELECT"
|
|
|
|
" id"
|
|
|
|
", status"
|
|
|
|
", destination"
|
|
|
|
", msatoshi"
|
|
|
|
", payment_hash"
|
|
|
|
", timestamp"
|
|
|
|
", payment_preimage"
|
|
|
|
", path_secrets"
|
|
|
|
", route_nodes"
|
|
|
|
", route_channels"
|
|
|
|
", msatoshi_sent"
|
|
|
|
", description"
|
|
|
|
", bolt11"
|
2019-11-11 19:38:27 +01:00
|
|
|
", failonionreply"
|
2019-12-12 00:16:23 +01:00
|
|
|
", total_msat"
|
|
|
|
", partid"
|
2020-12-14 02:20:44 +01:00
|
|
|
", local_offer_id"
|
2019-08-20 09:54:38 +02:00
|
|
|
" FROM payments"
|
|
|
|
" WHERE payment_hash = ?;"));
|
|
|
|
db_bind_sha256(stmt, 0, payment_hash);
|
2018-01-16 20:44:32 +01:00
|
|
|
} else {
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(wallet->db, SQL("SELECT"
|
|
|
|
" id"
|
|
|
|
", status"
|
|
|
|
", destination"
|
|
|
|
", msatoshi"
|
|
|
|
", payment_hash"
|
|
|
|
", timestamp"
|
|
|
|
", payment_preimage"
|
|
|
|
", path_secrets"
|
|
|
|
", route_nodes"
|
|
|
|
", route_channels"
|
|
|
|
", msatoshi_sent"
|
|
|
|
", description"
|
|
|
|
", bolt11"
|
2019-11-11 19:38:27 +01:00
|
|
|
", failonionreply"
|
2019-12-12 00:16:23 +01:00
|
|
|
", total_msat"
|
|
|
|
", partid"
|
2020-12-14 02:20:44 +01:00
|
|
|
", local_offer_id"
|
2020-05-29 04:27:55 +02:00
|
|
|
" FROM payments"
|
|
|
|
" ORDER BY id;"));
|
2019-08-20 09:54:38 +02:00
|
|
|
}
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
for (i = 0; db_step(stmt); i++) {
|
2017-11-16 19:11:59 +01:00
|
|
|
tal_resize(&payments, i+1);
|
|
|
|
payments[i] = wallet_stmt2payment(payments, stmt);
|
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2017-11-16 19:11:59 +01:00
|
|
|
|
2018-01-17 21:29:49 +01:00
|
|
|
/* Now attach payments not yet in db. */
|
|
|
|
list_for_each(&wallet->unstored_payments, p, list) {
|
2018-07-04 07:30:02 +02:00
|
|
|
if (payment_hash && !sha256_eq(&p->payment_hash, payment_hash))
|
2018-01-16 20:44:32 +01:00
|
|
|
continue;
|
2018-01-17 21:29:49 +01:00
|
|
|
tal_resize(&payments, i+1);
|
|
|
|
payments[i++] = p;
|
|
|
|
}
|
|
|
|
|
2017-11-16 19:11:59 +01:00
|
|
|
return payments;
|
|
|
|
}
|
2018-02-09 15:44:39 +01:00
|
|
|
|
2020-12-14 02:20:44 +01:00
|
|
|
const struct wallet_payment **
|
|
|
|
wallet_payments_by_offer(const tal_t *ctx,
|
|
|
|
struct wallet *wallet,
|
|
|
|
const struct sha256 *local_offer_id)
|
|
|
|
{
|
|
|
|
const struct wallet_payment **payments;
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
struct wallet_payment *p;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
payments = tal_arr(ctx, const struct wallet_payment *, 0);
|
|
|
|
stmt = db_prepare_v2(wallet->db, SQL("SELECT"
|
|
|
|
" id"
|
|
|
|
", status"
|
|
|
|
", destination"
|
|
|
|
", msatoshi"
|
|
|
|
", payment_hash"
|
|
|
|
", timestamp"
|
|
|
|
", payment_preimage"
|
|
|
|
", path_secrets"
|
|
|
|
", route_nodes"
|
|
|
|
", route_channels"
|
|
|
|
", msatoshi_sent"
|
|
|
|
", description"
|
|
|
|
", bolt11"
|
|
|
|
", failonionreply"
|
|
|
|
", total_msat"
|
|
|
|
", partid"
|
|
|
|
", local_offer_id"
|
|
|
|
" FROM payments"
|
|
|
|
" WHERE local_offer_id = ?;"));
|
|
|
|
db_bind_sha256(stmt, 0, local_offer_id);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
for (i = 0; db_step(stmt); i++) {
|
|
|
|
tal_resize(&payments, i+1);
|
|
|
|
payments[i] = wallet_stmt2payment(payments, stmt);
|
|
|
|
}
|
|
|
|
tal_free(stmt);
|
|
|
|
|
|
|
|
/* Now attach payments not yet in db. */
|
|
|
|
list_for_each(&wallet->unstored_payments, p, list) {
|
|
|
|
if (!p->local_offer_id || !sha256_eq(p->local_offer_id, local_offer_id))
|
|
|
|
continue;
|
|
|
|
tal_resize(&payments, i+1);
|
|
|
|
payments[i++] = p;
|
|
|
|
}
|
|
|
|
|
|
|
|
return payments;
|
|
|
|
}
|
|
|
|
|
2018-02-09 15:44:39 +01:00
|
|
|
void wallet_htlc_sigs_save(struct wallet *w, u64 channel_id,
|
2020-08-13 19:45:02 +02:00
|
|
|
const struct bitcoin_signature *htlc_sigs)
|
2018-02-09 15:44:39 +01:00
|
|
|
{
|
|
|
|
/* Clear any existing HTLC sigs for this channel */
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("DELETE FROM htlc_sigs WHERE channelid = ?"));
|
|
|
|
db_bind_u64(stmt, 0, channel_id);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-02-09 15:44:39 +01:00
|
|
|
|
|
|
|
/* Now insert the new ones */
|
|
|
|
for (size_t i=0; i<tal_count(htlc_sigs); i++) {
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db,
|
|
|
|
SQL("INSERT INTO htlc_sigs (channelid, "
|
|
|
|
"signature) VALUES (?, ?)"));
|
|
|
|
db_bind_u64(stmt, 0, channel_id);
|
2020-08-13 19:45:02 +02:00
|
|
|
db_bind_signature(stmt, 1, &htlc_sigs[i].s);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-02-09 15:44:39 +01:00
|
|
|
}
|
|
|
|
}
|
2018-02-16 17:55:33 +01:00
|
|
|
|
2019-11-20 02:55:57 +01:00
|
|
|
bool wallet_network_check(struct wallet *w)
|
2018-02-16 17:55:33 +01:00
|
|
|
{
|
|
|
|
struct bitcoin_blkid chainhash;
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt = db_prepare_v2(
|
2019-09-10 15:45:53 +02:00
|
|
|
w->db, SQL("SELECT blobval FROM vars WHERE name='genesis_hash'"));
|
2019-08-20 09:54:38 +02:00
|
|
|
db_query_prepared(stmt);
|
2018-02-16 17:55:33 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (db_step(stmt)) {
|
|
|
|
db_column_sha256d(stmt, 0, &chainhash.shad);
|
|
|
|
tal_free(stmt);
|
2018-07-04 07:30:02 +02:00
|
|
|
if (!bitcoin_blkid_eq(&chainhash,
|
|
|
|
&chainparams->genesis_blockhash)) {
|
2018-02-16 17:55:33 +01:00
|
|
|
log_broken(w->log, "Wallet blockchain hash does not "
|
|
|
|
"match network blockchain hash: %s "
|
2018-07-03 16:37:01 +02:00
|
|
|
"!= %s. "
|
2018-10-19 06:21:00 +02:00
|
|
|
"Are you on the right network? "
|
2019-07-19 04:37:33 +02:00
|
|
|
"(--network={one of %s})",
|
2018-02-16 17:55:33 +01:00
|
|
|
type_to_string(w, struct bitcoin_blkid,
|
|
|
|
&chainhash),
|
|
|
|
type_to_string(w, struct bitcoin_blkid,
|
2019-07-19 04:37:33 +02:00
|
|
|
&chainparams->genesis_blockhash),
|
|
|
|
chainparams_get_network_names(tmpctx));
|
2018-02-16 17:55:33 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2018-02-16 17:55:33 +01:00
|
|
|
/* Still a pristine wallet, claim it for the chain
|
|
|
|
* that we are running */
|
2019-09-10 15:45:53 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("INSERT INTO vars (name, blobval) "
|
2019-08-20 09:54:38 +02:00
|
|
|
"VALUES ('genesis_hash', ?);"));
|
|
|
|
db_bind_sha256d(stmt, 0, &chainparams->genesis_blockhash.shad);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-02-16 17:55:33 +01:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2018-02-17 16:45:34 +01:00
|
|
|
|
2018-03-04 03:42:19 +01:00
|
|
|
/**
|
|
|
|
* wallet_utxoset_prune -- Remove spent UTXO entries that are old
|
|
|
|
*/
|
|
|
|
static void wallet_utxoset_prune(struct wallet *w, const u32 blockheight)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2021-01-07 14:27:54 +01:00
|
|
|
struct bitcoin_txid txid;
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db,
|
|
|
|
SQL("SELECT txid, outnum FROM utxoset WHERE spendheight < ?"));
|
|
|
|
db_bind_int(stmt, 0, blockheight - UTXO_PRUNE_DEPTH);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
while (db_step(stmt)) {
|
|
|
|
db_column_sha256d(stmt, 0, &txid.shad);
|
|
|
|
outpointfilter_remove(w->utxoset_outpoints, &txid,
|
|
|
|
db_column_int(stmt, 1));
|
|
|
|
}
|
|
|
|
tal_free(stmt);
|
2018-03-26 20:30:11 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db,
|
|
|
|
SQL("DELETE FROM utxoset WHERE spendheight < ?"));
|
|
|
|
db_bind_int(stmt, 0, blockheight - UTXO_PRUNE_DEPTH);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-03-04 03:42:19 +01:00
|
|
|
}
|
|
|
|
|
2018-02-17 16:45:34 +01:00
|
|
|
void wallet_block_add(struct wallet *w, struct block *b)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt =
|
|
|
|
db_prepare_v2(w->db, SQL("INSERT INTO blocks "
|
|
|
|
"(height, hash, prev_hash) "
|
|
|
|
"VALUES (?, ?, ?);"));
|
|
|
|
db_bind_int(stmt, 0, b->height);
|
|
|
|
db_bind_sha256d(stmt, 1, &b->blkid.shad);
|
2018-02-17 16:45:34 +01:00
|
|
|
if (b->prev) {
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_sha256d(stmt, 2, &b->prev->blkid.shad);
|
2018-02-17 16:45:34 +01:00
|
|
|
}else {
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 2);
|
2018-02-17 16:45:34 +01:00
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-03-04 03:42:19 +01:00
|
|
|
|
|
|
|
/* Now cleanup UTXOs that we don't care about anymore */
|
|
|
|
wallet_utxoset_prune(w, b->height);
|
2018-02-17 16:45:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void wallet_block_remove(struct wallet *w, struct block *b)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt =
|
|
|
|
db_prepare_v2(w->db, SQL("DELETE FROM blocks WHERE hash = ?"));
|
|
|
|
db_bind_sha256d(stmt, 0, &b->blkid.shad);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-02-17 16:45:34 +01:00
|
|
|
|
2019-07-17 15:12:34 +02:00
|
|
|
/* Make sure that all descendants of the block are also deleted */
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db,
|
|
|
|
SQL("SELECT * FROM blocks WHERE height >= ?;"));
|
|
|
|
db_bind_int(stmt, 0, b->height);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
assert(!db_step(stmt));
|
|
|
|
tal_free(stmt);
|
2018-02-17 16:45:34 +01:00
|
|
|
}
|
2018-02-19 14:01:42 +01:00
|
|
|
|
|
|
|
void wallet_blocks_rollback(struct wallet *w, u32 height)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt = db_prepare_v2(w->db, SQL("DELETE FROM blocks "
|
|
|
|
"WHERE height > ?"));
|
|
|
|
db_bind_int(stmt, 0, height);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-02-19 14:01:42 +01:00
|
|
|
}
|
2018-03-02 15:07:13 +01:00
|
|
|
|
2020-09-04 17:50:17 +02:00
|
|
|
bool wallet_outpoint_spend(struct wallet *w, const tal_t *ctx, const u32 blockheight,
|
|
|
|
const struct bitcoin_txid *txid, const u32 outnum)
|
2018-03-02 15:07:13 +01:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2020-09-04 17:50:17 +02:00
|
|
|
bool our_spend;
|
2018-03-02 15:07:13 +01:00
|
|
|
if (outpointfilter_matches(w->owned_outpoints, txid, outnum)) {
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("UPDATE outputs "
|
2020-08-28 04:40:57 +02:00
|
|
|
"SET spend_height = ?, "
|
|
|
|
" status = ? "
|
2019-08-20 09:54:38 +02:00
|
|
|
"WHERE prev_out_tx = ?"
|
|
|
|
" AND prev_out_index = ?"));
|
2018-03-02 15:07:13 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_int(stmt, 0, blockheight);
|
2020-08-28 05:56:34 +02:00
|
|
|
db_bind_int(stmt, 1, output_status_in_db(OUTPUT_STATE_SPENT));
|
2020-08-28 04:40:57 +02:00
|
|
|
db_bind_sha256d(stmt, 2, &txid->shad);
|
|
|
|
db_bind_int(stmt, 3, outnum);
|
2018-03-04 02:27:17 +01:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2020-04-03 23:56:57 +02:00
|
|
|
|
2020-09-04 17:50:17 +02:00
|
|
|
our_spend = true;
|
2020-04-03 23:56:57 +02:00
|
|
|
} else
|
2020-09-04 17:50:17 +02:00
|
|
|
our_spend = false;
|
2018-03-04 02:27:17 +01:00
|
|
|
|
2021-01-07 14:27:54 +01:00
|
|
|
if (outpointfilter_matches(w->utxoset_outpoints, txid, outnum)) {
|
|
|
|
stmt = db_prepare_v2(w->db, SQL("UPDATE utxoset "
|
|
|
|
"SET spendheight = ? "
|
|
|
|
"WHERE txid = ?"
|
|
|
|
" AND outnum = ?"));
|
2018-03-04 02:27:17 +01:00
|
|
|
|
2021-01-07 14:27:54 +01:00
|
|
|
db_bind_int(stmt, 0, blockheight);
|
|
|
|
db_bind_sha256d(stmt, 1, &txid->shad);
|
|
|
|
db_bind_int(stmt, 2, outnum);
|
|
|
|
db_exec_prepared_v2(stmt);
|
|
|
|
tal_free(stmt);
|
|
|
|
}
|
2020-09-04 17:50:17 +02:00
|
|
|
return our_spend;
|
2018-03-02 15:07:13 +01:00
|
|
|
}
|
2018-03-04 01:37:56 +01:00
|
|
|
|
|
|
|
void wallet_utxoset_add(struct wallet *w, const struct bitcoin_tx *tx,
|
|
|
|
const u32 outnum, const u32 blockheight,
|
|
|
|
const u32 txindex, const u8 *scriptpubkey,
|
2019-02-21 04:45:55 +01:00
|
|
|
struct amount_sat sat)
|
2018-03-04 01:37:56 +01:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-03-04 01:37:56 +01:00
|
|
|
struct bitcoin_txid txid;
|
|
|
|
bitcoin_txid(tx, &txid);
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("INSERT INTO utxoset ("
|
|
|
|
" txid,"
|
|
|
|
" outnum,"
|
|
|
|
" blockheight,"
|
|
|
|
" spendheight,"
|
|
|
|
" txindex,"
|
|
|
|
" scriptpubkey,"
|
|
|
|
" satoshis"
|
|
|
|
") VALUES(?, ?, ?, ?, ?, ?, ?);"));
|
|
|
|
db_bind_sha256d(stmt, 0, &txid.shad);
|
|
|
|
db_bind_int(stmt, 1, outnum);
|
|
|
|
db_bind_int(stmt, 2, blockheight);
|
|
|
|
db_bind_null(stmt, 3);
|
|
|
|
db_bind_int(stmt, 4, txindex);
|
2020-08-25 23:20:52 +02:00
|
|
|
db_bind_talarr(stmt, 5, scriptpubkey);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_amount_sat(stmt, 6, &sat);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2021-01-07 14:27:54 +01:00
|
|
|
|
|
|
|
outpointfilter_add(w->utxoset_outpoints, &txid, outnum);
|
2018-03-04 01:37:56 +01:00
|
|
|
}
|
2018-03-05 23:15:54 +01:00
|
|
|
|
2019-08-08 06:24:33 +02:00
|
|
|
void wallet_filteredblock_add(struct wallet *w, const struct filteredblock *fb)
|
2019-08-05 23:16:32 +02:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2019-08-05 23:16:32 +02:00
|
|
|
if (wallet_have_block(w, fb->height))
|
|
|
|
return;
|
2019-09-12 22:46:34 +02:00
|
|
|
|
|
|
|
stmt = db_prepare_v2(w->db, SQL("INSERT INTO blocks "
|
2019-08-20 09:54:38 +02:00
|
|
|
"(height, hash, prev_hash) "
|
|
|
|
"VALUES (?, ?, ?);"));
|
|
|
|
db_bind_int(stmt, 0, fb->height);
|
|
|
|
db_bind_sha256d(stmt, 1, &fb->id.shad);
|
|
|
|
db_bind_sha256d(stmt, 2, &fb->prev_hash.shad);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2019-08-05 23:16:32 +02:00
|
|
|
|
|
|
|
for (size_t i = 0; i < tal_count(fb->outpoints); i++) {
|
|
|
|
struct filteredblock_outpoint *o = fb->outpoints[i];
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt =
|
|
|
|
db_prepare_v2(w->db, SQL("INSERT INTO utxoset ("
|
|
|
|
" txid,"
|
|
|
|
" outnum,"
|
|
|
|
" blockheight,"
|
|
|
|
" spendheight,"
|
|
|
|
" txindex,"
|
|
|
|
" scriptpubkey,"
|
|
|
|
" satoshis"
|
|
|
|
") VALUES(?, ?, ?, ?, ?, ?, ?);"));
|
|
|
|
db_bind_sha256d(stmt, 0, &o->txid.shad);
|
|
|
|
db_bind_int(stmt, 1, o->outnum);
|
|
|
|
db_bind_int(stmt, 2, fb->height);
|
|
|
|
db_bind_null(stmt, 3);
|
|
|
|
db_bind_int(stmt, 4, o->txindex);
|
2020-08-25 23:20:52 +02:00
|
|
|
db_bind_talarr(stmt, 5, o->scriptPubKey);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_amount_sat(stmt, 6, &o->amount);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2021-01-07 14:27:54 +01:00
|
|
|
|
|
|
|
outpointfilter_add(w->utxoset_outpoints, &o->txid, o->outnum);
|
2019-08-05 23:16:32 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool wallet_have_block(struct wallet *w, u32 blockheight)
|
|
|
|
{
|
|
|
|
bool result;
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("SELECT height FROM blocks WHERE height = ?"));
|
|
|
|
db_bind_int(stmt, 0, blockheight);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
result = db_step(stmt);
|
|
|
|
tal_free(stmt);
|
2019-08-05 23:16:32 +02:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-03-05 23:15:54 +01:00
|
|
|
struct outpoint *wallet_outpoint_for_scid(struct wallet *w, tal_t *ctx,
|
|
|
|
const struct short_channel_id *scid)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-03-05 23:15:54 +01:00
|
|
|
struct outpoint *op;
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT"
|
2019-07-17 15:12:34 +02:00
|
|
|
" txid,"
|
|
|
|
" spendheight,"
|
|
|
|
" scriptpubkey,"
|
|
|
|
" satoshis "
|
|
|
|
"FROM utxoset "
|
|
|
|
"WHERE blockheight = ?"
|
|
|
|
" AND txindex = ?"
|
|
|
|
" AND outnum = ?"
|
2019-08-20 09:54:38 +02:00
|
|
|
" AND spendheight IS NULL"));
|
|
|
|
db_bind_int(stmt, 0, short_channel_id_blocknum(scid));
|
|
|
|
db_bind_int(stmt, 1, short_channel_id_txnum(scid));
|
|
|
|
db_bind_int(stmt, 2, short_channel_id_outnum(scid));
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
if (!db_step(stmt)) {
|
|
|
|
tal_free(stmt);
|
2018-03-05 23:15:54 +01:00
|
|
|
return NULL;
|
2019-08-20 09:54:38 +02:00
|
|
|
}
|
2018-03-05 23:15:54 +01:00
|
|
|
|
|
|
|
op = tal(ctx, struct outpoint);
|
|
|
|
op->blockheight = short_channel_id_blocknum(scid);
|
|
|
|
op->txindex = short_channel_id_txnum(scid);
|
|
|
|
op->outnum = short_channel_id_outnum(scid);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_sha256d(stmt, 0, &op->txid.shad);
|
2019-09-12 22:46:34 +02:00
|
|
|
if (db_column_is_null(stmt, 1))
|
|
|
|
op->spendheight = 0;
|
|
|
|
else
|
|
|
|
op->spendheight = db_column_int(stmt, 1);
|
2019-08-20 09:54:38 +02:00
|
|
|
op->scriptpubkey = tal_arr(op, u8, db_column_bytes(stmt, 2));
|
|
|
|
memcpy(op->scriptpubkey, db_column_blob(stmt, 2), db_column_bytes(stmt, 2));
|
|
|
|
db_column_amount_sat(stmt, 3, &op->sat);
|
|
|
|
tal_free(stmt);
|
2018-03-05 23:15:54 +01:00
|
|
|
|
|
|
|
return op;
|
|
|
|
}
|
2018-04-07 14:39:11 +02:00
|
|
|
|
2020-09-08 12:34:36 +02:00
|
|
|
const struct short_channel_id *
|
|
|
|
wallet_utxoset_get_spent(const tal_t *ctx, struct wallet *w, u32 blockheight)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
struct short_channel_id *res;
|
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT"
|
|
|
|
" blockheight,"
|
|
|
|
" txindex,"
|
|
|
|
" outnum "
|
|
|
|
"FROM utxoset "
|
|
|
|
"WHERE spendheight = ?"));
|
|
|
|
db_bind_int(stmt, 0, blockheight);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
res = tal_arr(ctx, struct short_channel_id, 0);
|
|
|
|
while (db_step(stmt)) {
|
|
|
|
struct short_channel_id scid;
|
|
|
|
u64 blocknum, txnum, outnum;
|
|
|
|
bool ok;
|
|
|
|
blocknum = db_column_int(stmt, 0);
|
|
|
|
txnum = db_column_int(stmt, 1);
|
|
|
|
outnum = db_column_int(stmt, 2);
|
|
|
|
ok = mk_short_channel_id(&scid, blocknum, txnum, outnum);
|
|
|
|
|
|
|
|
assert(ok);
|
|
|
|
tal_arr_expand(&res, scid);
|
|
|
|
}
|
|
|
|
tal_free(stmt);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2020-08-07 03:30:47 +02:00
|
|
|
void wallet_transaction_add(struct wallet *w, const struct wally_tx *tx,
|
2018-04-07 14:39:11 +02:00
|
|
|
const u32 blockheight, const u32 txindex)
|
|
|
|
{
|
|
|
|
struct bitcoin_txid txid;
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("SELECT blockheight FROM transactions WHERE id=?"));
|
2018-04-07 14:39:11 +02:00
|
|
|
|
2020-08-07 03:30:47 +02:00
|
|
|
wally_txid(tx, &txid);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_txid(stmt, 0, &txid);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
if (!db_step(stmt)) {
|
|
|
|
tal_free(stmt);
|
2018-04-07 14:39:11 +02:00
|
|
|
/* This transaction is still unknown, insert */
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db,
|
|
|
|
SQL("INSERT INTO transactions ("
|
2020-05-21 23:43:27 +02:00
|
|
|
" id"
|
|
|
|
", blockheight"
|
|
|
|
", txindex"
|
2019-08-20 09:54:38 +02:00
|
|
|
", rawtx) VALUES (?, ?, ?, ?);"));
|
|
|
|
db_bind_txid(stmt, 0, &txid);
|
2018-04-07 14:39:11 +02:00
|
|
|
if (blockheight) {
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_int(stmt, 1, blockheight);
|
|
|
|
db_bind_int(stmt, 2, txindex);
|
2018-04-07 14:39:11 +02:00
|
|
|
} else {
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 1);
|
|
|
|
db_bind_null(stmt, 2);
|
2018-04-07 14:39:11 +02:00
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_tx(stmt, 3, tx);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2019-03-15 03:50:18 +01:00
|
|
|
} else {
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2019-03-15 03:50:18 +01:00
|
|
|
|
|
|
|
if (blockheight) {
|
|
|
|
/* We know about the transaction, update */
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db,
|
|
|
|
SQL("UPDATE transactions "
|
|
|
|
"SET blockheight = ?, txindex = ? "
|
|
|
|
"WHERE id = ?"));
|
|
|
|
db_bind_int(stmt, 0, blockheight);
|
|
|
|
db_bind_int(stmt, 1, txindex);
|
|
|
|
db_bind_txid(stmt, 2, &txid);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2019-03-15 03:50:18 +01:00
|
|
|
}
|
2018-04-07 14:39:11 +02:00
|
|
|
}
|
|
|
|
}
|
2018-04-09 15:18:09 +02:00
|
|
|
|
2019-09-30 22:24:04 +02:00
|
|
|
static void wallet_annotation_add(struct wallet *w, const struct bitcoin_txid *txid, int num,
|
|
|
|
enum wallet_tx_annotation_type annotation_type, enum wallet_tx_type type, u64 channel)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db,SQL("INSERT INTO transaction_annotations "
|
|
|
|
"(txid, idx, location, type, channel) "
|
|
|
|
"VALUES (?, ?, ?, ?, ?) ON CONFLICT(txid,idx) DO NOTHING;"));
|
|
|
|
|
|
|
|
db_bind_txid(stmt, 0, txid);
|
|
|
|
db_bind_int(stmt, 1, num);
|
|
|
|
db_bind_int(stmt, 2, annotation_type);
|
|
|
|
db_bind_int(stmt, 3, type);
|
|
|
|
if (channel != 0)
|
|
|
|
db_bind_u64(stmt, 4, channel);
|
|
|
|
else
|
|
|
|
db_bind_null(stmt, 4);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
|
|
|
}
|
|
|
|
|
|
|
|
void wallet_annotate_txout(struct wallet *w, const struct bitcoin_txid *txid,
|
|
|
|
int outnum, enum wallet_tx_type type, u64 channel)
|
|
|
|
{
|
|
|
|
wallet_annotation_add(w, txid, outnum, OUTPUT_ANNOTATION, type, channel);
|
|
|
|
}
|
|
|
|
|
|
|
|
void wallet_annotate_txin(struct wallet *w, const struct bitcoin_txid *txid,
|
|
|
|
int innum, enum wallet_tx_type type, u64 channel)
|
|
|
|
{
|
|
|
|
wallet_annotation_add(w, txid, innum, INPUT_ANNOTATION, type, channel);
|
|
|
|
}
|
|
|
|
|
2019-05-23 22:41:27 +02:00
|
|
|
void wallet_transaction_annotate(struct wallet *w,
|
2019-06-07 11:38:20 +02:00
|
|
|
const struct bitcoin_txid *txid, enum wallet_tx_type type,
|
2019-05-23 22:41:27 +02:00
|
|
|
u64 channel_id)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("SELECT type, channel_id FROM transactions WHERE id=?"));
|
|
|
|
db_bind_txid(stmt, 0, txid);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
if (!db_step(stmt))
|
2019-05-23 22:41:27 +02:00
|
|
|
fatal("Attempting to annotate a transaction we don't have: %s",
|
|
|
|
type_to_string(tmpctx, struct bitcoin_txid, txid));
|
2019-08-20 09:54:38 +02:00
|
|
|
|
2019-09-11 00:22:32 +02:00
|
|
|
if (!db_column_is_null(stmt, 0))
|
2019-09-12 22:46:34 +02:00
|
|
|
type |= db_column_u64(stmt, 0);
|
2019-09-11 00:22:32 +02:00
|
|
|
|
|
|
|
if (channel_id == 0 && !db_column_is_null(stmt, 1))
|
2019-08-20 09:54:38 +02:00
|
|
|
channel_id = db_column_u64(stmt, 1);
|
2019-05-23 22:41:27 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2019-05-23 22:41:27 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("UPDATE transactions "
|
|
|
|
"SET type = ?"
|
|
|
|
", channel_id = ? "
|
|
|
|
"WHERE id = ?"));
|
|
|
|
|
2019-09-12 22:46:34 +02:00
|
|
|
db_bind_u64(stmt, 0, type);
|
2019-05-23 22:41:27 +02:00
|
|
|
|
|
|
|
if (channel_id)
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_int(stmt, 1, channel_id);
|
2019-05-23 22:41:27 +02:00
|
|
|
else
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 1);
|
|
|
|
|
|
|
|
db_bind_txid(stmt, 2, txid);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
2019-05-23 22:41:27 +02:00
|
|
|
}
|
|
|
|
|
2019-08-23 23:34:52 +02:00
|
|
|
bool wallet_transaction_type(struct wallet *w, const struct bitcoin_txid *txid,
|
|
|
|
enum wallet_tx_type *type)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt = db_prepare_v2(w->db, SQL("SELECT type FROM transactions WHERE id=?"));
|
|
|
|
db_bind_sha256(stmt, 0, &txid->shad.sha);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
if (!db_step(stmt)) {
|
|
|
|
tal_free(stmt);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-10-04 13:43:05 +02:00
|
|
|
if (!db_column_is_null(stmt, 0))
|
|
|
|
*type = db_column_u64(stmt, 0);
|
|
|
|
else
|
|
|
|
*type = 0;
|
|
|
|
|
2019-08-23 23:34:52 +02:00
|
|
|
tal_free(stmt);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-12-31 00:36:00 +01:00
|
|
|
struct bitcoin_tx *wallet_transaction_get(const tal_t *ctx, struct wallet *w,
|
|
|
|
const struct bitcoin_txid *txid)
|
|
|
|
{
|
|
|
|
struct bitcoin_tx *tx;
|
|
|
|
struct db_stmt *stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("SELECT rawtx FROM transactions WHERE id=?"));
|
|
|
|
db_bind_txid(stmt, 0, txid);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
if (!db_step(stmt)) {
|
|
|
|
tal_free(stmt);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!db_column_is_null(stmt, 0))
|
|
|
|
tx = db_column_tx(ctx, stmt, 0);
|
|
|
|
else
|
|
|
|
tx = NULL;
|
|
|
|
|
|
|
|
tal_free(stmt);
|
|
|
|
return tx;
|
|
|
|
}
|
|
|
|
|
2018-04-09 15:18:09 +02:00
|
|
|
u32 wallet_transaction_height(struct wallet *w, const struct bitcoin_txid *txid)
|
|
|
|
{
|
|
|
|
u32 blockheight;
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("SELECT blockheight FROM transactions WHERE id=?"));
|
|
|
|
db_bind_txid(stmt, 0, txid);
|
|
|
|
db_query_prepared(stmt);
|
2018-04-09 15:18:09 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_step(stmt)) {
|
|
|
|
tal_free(stmt);
|
2018-04-09 15:18:09 +02:00
|
|
|
return 0;
|
2019-08-20 09:54:38 +02:00
|
|
|
}
|
2018-04-09 15:18:09 +02:00
|
|
|
|
2019-09-12 22:46:34 +02:00
|
|
|
if (!db_column_is_null(stmt, 0))
|
|
|
|
blockheight = db_column_int(stmt, 0);
|
|
|
|
else
|
|
|
|
blockheight = 0;
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2018-04-09 15:18:09 +02:00
|
|
|
return blockheight;
|
|
|
|
}
|
2018-04-09 18:48:39 +02:00
|
|
|
|
|
|
|
struct txlocator *wallet_transaction_locate(const tal_t *ctx, struct wallet *w,
|
|
|
|
const struct bitcoin_txid *txid)
|
|
|
|
{
|
|
|
|
struct txlocator *loc;
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-04-09 18:48:39 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("SELECT blockheight, txindex FROM transactions WHERE id=?"));
|
|
|
|
db_bind_txid(stmt, 0, txid);
|
|
|
|
db_query_prepared(stmt);
|
2018-04-09 18:48:39 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_step(stmt)) {
|
|
|
|
tal_free(stmt);
|
2019-03-15 03:50:18 +01:00
|
|
|
return NULL;
|
2019-08-20 09:54:38 +02:00
|
|
|
}
|
2018-04-09 18:48:39 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (db_column_is_null(stmt, 0))
|
2019-03-15 03:50:18 +01:00
|
|
|
loc = NULL;
|
|
|
|
else {
|
|
|
|
loc = tal(ctx, struct txlocator);
|
2019-08-20 09:54:38 +02:00
|
|
|
loc->blkheight = db_column_int(stmt, 0);
|
|
|
|
loc->index = db_column_int(stmt, 1);
|
2019-03-15 03:50:18 +01:00
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2018-04-09 18:48:39 +02:00
|
|
|
return loc;
|
|
|
|
}
|
2018-04-10 11:14:50 +02:00
|
|
|
|
|
|
|
struct bitcoin_txid *wallet_transactions_by_height(const tal_t *ctx,
|
|
|
|
struct wallet *w,
|
|
|
|
const u32 blockheight)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-04-10 11:14:50 +02:00
|
|
|
struct bitcoin_txid *txids = tal_arr(ctx, struct bitcoin_txid, 0);
|
|
|
|
int count = 0;
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("SELECT id FROM transactions WHERE blockheight=?"));
|
|
|
|
db_bind_int(stmt, 0, blockheight);
|
|
|
|
db_query_prepared(stmt);
|
2018-04-10 11:14:50 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
while (db_step(stmt)) {
|
2018-04-10 11:14:50 +02:00
|
|
|
count++;
|
|
|
|
tal_resize(&txids, count);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_txid(stmt, 0, &txids[count-1]);
|
2018-04-10 11:14:50 +02:00
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2018-04-10 11:14:50 +02:00
|
|
|
|
|
|
|
return txids;
|
|
|
|
}
|
|
|
|
|
2018-04-03 16:57:28 +02:00
|
|
|
void wallet_channeltxs_add(struct wallet *w, struct channel *chan,
|
|
|
|
const int type, const struct bitcoin_txid *txid,
|
|
|
|
const u32 input_num, const u32 blockheight)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
|
|
|
stmt = db_prepare_v2(w->db, SQL("INSERT INTO channeltxs ("
|
|
|
|
" channel_id"
|
|
|
|
", type"
|
|
|
|
", transaction_id"
|
|
|
|
", input_num"
|
|
|
|
", blockheight"
|
|
|
|
") VALUES (?, ?, ?, ?, ?);"));
|
|
|
|
db_bind_int(stmt, 0, chan->dbid);
|
|
|
|
db_bind_int(stmt, 1, type);
|
|
|
|
db_bind_sha256(stmt, 2, &txid->shad.sha);
|
|
|
|
db_bind_int(stmt, 3, input_num);
|
|
|
|
db_bind_int(stmt, 4, blockheight);
|
2018-04-03 16:57:28 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2018-04-03 16:57:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
u32 *wallet_onchaind_channels(struct wallet *w,
|
|
|
|
const tal_t *ctx)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-04-03 16:57:28 +02:00
|
|
|
size_t count = 0;
|
|
|
|
u32 *channel_ids = tal_arr(ctx, u32, 0);
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db,
|
|
|
|
SQL("SELECT DISTINCT(channel_id) FROM channeltxs WHERE type = ?;"));
|
2020-08-25 04:15:48 +02:00
|
|
|
db_bind_int(stmt, 0, WIRE_ONCHAIND_INIT);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_query_prepared(stmt);
|
2018-04-03 16:57:28 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
while (db_step(stmt)) {
|
2018-04-03 16:57:28 +02:00
|
|
|
count++;
|
|
|
|
tal_resize(&channel_ids, count);
|
2019-08-20 09:54:38 +02:00
|
|
|
channel_ids[count-1] = db_column_u64(stmt, 0);
|
2018-04-03 16:57:28 +02:00
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2018-04-03 16:57:28 +02:00
|
|
|
|
|
|
|
return channel_ids;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct channeltx *wallet_channeltxs_get(struct wallet *w, const tal_t *ctx,
|
|
|
|
u32 channel_id)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2018-04-03 16:57:28 +02:00
|
|
|
size_t count = 0;
|
|
|
|
struct channeltx *res = tal_arr(ctx, struct channeltx, 0);
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db, SQL("SELECT"
|
|
|
|
" c.type"
|
|
|
|
", c.blockheight"
|
|
|
|
", t.rawtx"
|
|
|
|
", c.input_num"
|
|
|
|
", c.blockheight - t.blockheight + 1 AS depth"
|
|
|
|
", t.id as txid "
|
|
|
|
"FROM channeltxs c "
|
2019-09-12 22:46:34 +02:00
|
|
|
"JOIN transactions t ON t.id = c.transaction_id "
|
2019-08-20 09:54:38 +02:00
|
|
|
"WHERE c.channel_id = ? "
|
|
|
|
"ORDER BY c.id ASC;"));
|
|
|
|
db_bind_int(stmt, 0, channel_id);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
while (db_step(stmt)) {
|
2018-04-03 16:57:28 +02:00
|
|
|
count++;
|
|
|
|
tal_resize(&res, count);
|
|
|
|
|
|
|
|
res[count-1].channel_id = channel_id;
|
2019-08-20 09:54:38 +02:00
|
|
|
res[count-1].type = db_column_int(stmt, 0);
|
|
|
|
res[count-1].blockheight = db_column_int(stmt, 1);
|
|
|
|
res[count-1].tx = db_column_tx(ctx, stmt, 2);
|
|
|
|
res[count-1].input_num = db_column_int(stmt, 3);
|
|
|
|
res[count-1].depth = db_column_int(stmt, 4);
|
|
|
|
db_column_txid(stmt, 5, &res[count-1].txid);
|
|
|
|
}
|
|
|
|
tal_free(stmt);
|
2018-04-03 16:57:28 +02:00
|
|
|
return res;
|
|
|
|
}
|
2018-10-16 21:28:35 +02:00
|
|
|
|
2019-09-12 22:46:34 +02:00
|
|
|
static bool wallet_forwarded_payment_update(struct wallet *w,
|
|
|
|
const struct htlc_in *in,
|
|
|
|
const struct htlc_out *out,
|
|
|
|
enum forward_status state,
|
2020-08-31 03:13:25 +02:00
|
|
|
enum onion_wire failcode,
|
2019-09-12 22:46:34 +02:00
|
|
|
struct timeabs *resolved_time)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
bool changed;
|
|
|
|
|
|
|
|
/* We update based solely on the htlc_in since an HTLC cannot be
|
|
|
|
* associated with more than one forwarded payment. This saves us from
|
|
|
|
* having to have two versions of the update statement (one with and
|
|
|
|
* one without the htlc_out restriction).*/
|
|
|
|
stmt = db_prepare_v2(w->db,
|
|
|
|
SQL("UPDATE forwarded_payments SET"
|
|
|
|
" in_msatoshi=?"
|
|
|
|
", out_msatoshi=?"
|
|
|
|
", state=?"
|
|
|
|
", resolved_time=?"
|
|
|
|
", failcode=?"
|
|
|
|
" WHERE in_htlc_id=?"));
|
|
|
|
db_bind_amount_msat(stmt, 0, &in->msat);
|
|
|
|
|
|
|
|
if (out) {
|
|
|
|
db_bind_amount_msat(stmt, 1, &out->msat);
|
|
|
|
} else {
|
|
|
|
db_bind_null(stmt, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
db_bind_int(stmt, 2, wallet_forward_status_in_db(state));
|
|
|
|
|
|
|
|
if (resolved_time != NULL) {
|
|
|
|
db_bind_timeabs(stmt, 3, *resolved_time);
|
|
|
|
} else {
|
|
|
|
db_bind_null(stmt, 3);
|
|
|
|
}
|
|
|
|
|
2020-01-05 16:52:34 +01:00
|
|
|
if (failcode != 0) {
|
2019-09-12 22:46:34 +02:00
|
|
|
assert(state == FORWARD_FAILED || state == FORWARD_LOCAL_FAILED);
|
|
|
|
db_bind_int(stmt, 4, (int)failcode);
|
|
|
|
} else {
|
|
|
|
db_bind_null(stmt, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
db_bind_u64(stmt, 5, in->dbid);
|
|
|
|
db_exec_prepared_v2(stmt);
|
|
|
|
changed = db_count_changes(stmt) != 0;
|
|
|
|
tal_free(stmt);
|
|
|
|
|
|
|
|
return changed;
|
|
|
|
}
|
|
|
|
|
2018-10-16 21:28:35 +02:00
|
|
|
void wallet_forwarded_payment_add(struct wallet *w, const struct htlc_in *in,
|
2020-02-13 03:11:01 +01:00
|
|
|
const struct short_channel_id *scid_out,
|
2018-10-16 21:28:35 +02:00
|
|
|
const struct htlc_out *out,
|
2019-04-15 15:47:37 +02:00
|
|
|
enum forward_status state,
|
2020-08-31 03:13:25 +02:00
|
|
|
enum onion_wire failcode)
|
2018-10-16 21:28:35 +02:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2019-06-15 18:50:07 +02:00
|
|
|
struct timeabs *resolved_time;
|
2019-09-12 22:46:34 +02:00
|
|
|
|
|
|
|
if (state == FORWARD_SETTLED || state == FORWARD_FAILED) {
|
|
|
|
resolved_time = tal(tmpctx, struct timeabs);
|
|
|
|
*resolved_time = time_now();
|
|
|
|
} else {
|
|
|
|
resolved_time = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wallet_forwarded_payment_update(w, in, out, state, failcode, resolved_time))
|
|
|
|
goto notify;
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db,
|
2019-09-12 22:46:34 +02:00
|
|
|
SQL("INSERT INTO forwarded_payments ("
|
2019-08-20 09:54:38 +02:00
|
|
|
" in_htlc_id"
|
|
|
|
", out_htlc_id"
|
|
|
|
", in_channel_scid"
|
|
|
|
", out_channel_scid"
|
|
|
|
", in_msatoshi"
|
|
|
|
", out_msatoshi"
|
|
|
|
", state"
|
|
|
|
", received_time"
|
|
|
|
", resolved_time"
|
|
|
|
", failcode"
|
|
|
|
") VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?);"));
|
|
|
|
db_bind_u64(stmt, 0, in->dbid);
|
2019-04-15 15:47:37 +02:00
|
|
|
|
2020-01-05 16:52:34 +01:00
|
|
|
if (out) {
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_u64(stmt, 1, out->dbid);
|
|
|
|
db_bind_u64(stmt, 3, out->key.channel->scid->u64);
|
|
|
|
db_bind_amount_msat(stmt, 5, &out->msat);
|
2019-04-15 15:47:37 +02:00
|
|
|
} else {
|
|
|
|
/* FORWARD_LOCAL_FAILED may occur before we get htlc_out */
|
|
|
|
assert(failcode != 0);
|
|
|
|
assert(state == FORWARD_LOCAL_FAILED);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 1);
|
|
|
|
db_bind_null(stmt, 3);
|
|
|
|
db_bind_null(stmt, 5);
|
2019-04-15 15:47:37 +02:00
|
|
|
}
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_u64(stmt, 2, in->key.channel->scid->u64);
|
2019-04-15 15:47:37 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_amount_msat(stmt, 4, &in->msat);
|
2019-04-15 15:47:37 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_int(stmt, 6, wallet_forward_status_in_db(state));
|
|
|
|
db_bind_timeabs(stmt, 7, in->received_time);
|
2019-04-03 20:50:14 +02:00
|
|
|
|
2019-09-12 22:46:34 +02:00
|
|
|
if (resolved_time != NULL)
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_timeabs(stmt, 8, *resolved_time);
|
2019-09-12 22:46:34 +02:00
|
|
|
else
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 8);
|
2019-04-03 20:50:14 +02:00
|
|
|
|
2020-01-05 16:52:34 +01:00
|
|
|
if (failcode != 0) {
|
2019-04-15 15:47:37 +02:00
|
|
|
assert(state == FORWARD_FAILED || state == FORWARD_LOCAL_FAILED);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_int(stmt, 9, (int)failcode);
|
2019-04-15 15:47:37 +02:00
|
|
|
} else {
|
2019-08-20 09:54:38 +02:00
|
|
|
db_bind_null(stmt, 9);
|
2019-04-15 15:47:37 +02:00
|
|
|
}
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_exec_prepared_v2(take(stmt));
|
2019-06-15 18:50:07 +02:00
|
|
|
|
2019-09-12 22:46:34 +02:00
|
|
|
notify:
|
2020-02-13 03:11:01 +01:00
|
|
|
notify_forward_event(w->ld, in, scid_out, out ? &out->msat : NULL,
|
|
|
|
state, failcode, resolved_time);
|
2018-10-16 21:28:35 +02:00
|
|
|
}
|
2018-10-17 20:11:04 +02:00
|
|
|
|
2019-02-21 04:45:55 +01:00
|
|
|
struct amount_msat wallet_total_forward_fees(struct wallet *w)
|
2018-10-17 20:11:04 +02:00
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2019-02-21 04:45:55 +01:00
|
|
|
struct amount_msat total;
|
2019-03-15 03:50:18 +01:00
|
|
|
bool res;
|
2018-10-19 06:19:52 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT"
|
2019-09-11 00:22:32 +02:00
|
|
|
" CAST(COALESCE(SUM(in_msatoshi - out_msatoshi), 0) AS BIGINT)"
|
2019-07-17 15:12:34 +02:00
|
|
|
"FROM forwarded_payments "
|
2019-08-20 09:54:38 +02:00
|
|
|
"WHERE state = ?;"));
|
|
|
|
db_bind_int(stmt, 0, wallet_forward_status_in_db(FORWARD_SETTLED));
|
|
|
|
db_query_prepared(stmt);
|
2018-10-19 06:19:52 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
res = db_step(stmt);
|
2019-03-15 03:50:18 +01:00
|
|
|
assert(res);
|
2018-10-17 20:11:04 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_amount_msat(stmt, 0, &total);
|
|
|
|
tal_free(stmt);
|
2018-10-17 20:11:04 +02:00
|
|
|
|
2018-10-19 06:19:52 +02:00
|
|
|
return total;
|
2018-10-17 20:11:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const struct forwarding *wallet_forwarded_payments_get(struct wallet *w,
|
|
|
|
const tal_t *ctx)
|
|
|
|
{
|
|
|
|
struct forwarding *results = tal_arr(ctx, struct forwarding, 0);
|
|
|
|
size_t count = 0;
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db,
|
|
|
|
SQL("SELECT"
|
|
|
|
" f.state"
|
|
|
|
", in_msatoshi"
|
|
|
|
", out_msatoshi"
|
|
|
|
", hin.payment_hash as payment_hash"
|
|
|
|
", in_channel_scid"
|
|
|
|
", out_channel_scid"
|
|
|
|
", f.received_time"
|
|
|
|
", f.resolved_time"
|
|
|
|
", f.failcode "
|
|
|
|
"FROM forwarded_payments f "
|
2019-09-12 22:46:34 +02:00
|
|
|
"LEFT JOIN channel_htlcs hin ON (f.in_htlc_id = hin.id)"));
|
2019-08-20 09:54:38 +02:00
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
for (count=0; db_step(stmt); count++) {
|
2018-10-17 20:11:04 +02:00
|
|
|
tal_resize(&results, count+1);
|
|
|
|
struct forwarding *cur = &results[count];
|
2019-08-20 09:54:38 +02:00
|
|
|
cur->status = db_column_int(stmt, 0);
|
|
|
|
db_column_amount_msat(stmt, 1, &cur->msat_in);
|
2019-04-15 16:04:12 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_column_is_null(stmt, 2)) {
|
|
|
|
db_column_amount_msat(stmt, 2, &cur->msat_out);
|
2019-06-24 18:54:33 +02:00
|
|
|
if (!amount_msat_sub(&cur->fee, cur->msat_in, cur->msat_out)) {
|
|
|
|
log_broken(w->log, "Forwarded in %s less than out %s!",
|
|
|
|
type_to_string(tmpctx, struct amount_msat,
|
|
|
|
&cur->msat_in),
|
|
|
|
type_to_string(tmpctx, struct amount_msat,
|
|
|
|
&cur->msat_out));
|
|
|
|
cur->fee = AMOUNT_MSAT(0);
|
|
|
|
}
|
|
|
|
}
|
2019-04-15 16:04:12 +02:00
|
|
|
else {
|
|
|
|
assert(cur->status == FORWARD_LOCAL_FAILED);
|
|
|
|
cur->msat_out = AMOUNT_MSAT(0);
|
2019-06-24 18:54:33 +02:00
|
|
|
/* For this case, this forward_payment doesn't have out channel,
|
|
|
|
* so the fee should be set as 0.*/
|
|
|
|
cur->fee = AMOUNT_MSAT(0);
|
2019-02-21 04:45:55 +01:00
|
|
|
}
|
2018-10-17 23:28:43 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_column_is_null(stmt, 3)) {
|
2019-08-10 10:13:40 +02:00
|
|
|
cur->payment_hash = tal(ctx, struct sha256);
|
2019-08-20 09:54:38 +02:00
|
|
|
db_column_sha256(stmt, 3, cur->payment_hash);
|
2018-10-17 23:28:43 +02:00
|
|
|
} else {
|
|
|
|
cur->payment_hash = NULL;
|
|
|
|
}
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
cur->channel_in.u64 = db_column_u64(stmt, 4);
|
2019-04-15 16:04:12 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_column_is_null(stmt, 5)) {
|
|
|
|
cur->channel_out.u64 = db_column_u64(stmt, 5);
|
2019-04-15 16:04:12 +02:00
|
|
|
} else {
|
|
|
|
assert(cur->status == FORWARD_LOCAL_FAILED);
|
|
|
|
cur->channel_out.u64 = 0;
|
|
|
|
}
|
2019-04-03 21:28:33 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
cur->received_time = db_column_timeabs(stmt, 6);
|
2019-04-15 16:04:12 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_column_is_null(stmt, 7)) {
|
2019-04-03 21:28:33 +02:00
|
|
|
cur->resolved_time = tal(ctx, struct timeabs);
|
2019-08-20 09:54:38 +02:00
|
|
|
*cur->resolved_time = db_column_timeabs(stmt, 7);
|
2019-04-03 21:28:33 +02:00
|
|
|
} else {
|
|
|
|
cur->resolved_time = NULL;
|
|
|
|
}
|
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
if (!db_column_is_null(stmt, 8)) {
|
2019-06-24 18:54:33 +02:00
|
|
|
assert(cur->status == FORWARD_FAILED ||
|
|
|
|
cur->status == FORWARD_LOCAL_FAILED);
|
2019-08-20 09:54:38 +02:00
|
|
|
cur->failcode = db_column_int(stmt, 8);
|
2019-04-15 16:04:12 +02:00
|
|
|
} else {
|
|
|
|
cur->failcode = 0;
|
|
|
|
}
|
2018-10-17 20:11:04 +02:00
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
tal_free(stmt);
|
2018-10-17 20:11:04 +02:00
|
|
|
return results;
|
|
|
|
}
|
2019-06-05 09:00:02 +02:00
|
|
|
|
2019-05-28 22:36:15 +02:00
|
|
|
struct wallet_transaction *wallet_transactions_get(struct wallet *w, const tal_t *ctx)
|
|
|
|
{
|
2019-08-20 09:54:38 +02:00
|
|
|
struct db_stmt *stmt;
|
2019-05-28 22:36:15 +02:00
|
|
|
size_t count;
|
2019-10-02 19:36:28 +02:00
|
|
|
struct wallet_transaction *cur = NULL, *txs = tal_arr(ctx, struct wallet_transaction, 0);
|
|
|
|
struct bitcoin_txid last;
|
|
|
|
|
|
|
|
/* Make sure we can check for changing txids */
|
|
|
|
memset(&last, 0, sizeof(last));
|
2019-05-28 22:36:15 +02:00
|
|
|
|
2019-08-20 09:54:38 +02:00
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db,
|
2019-10-02 19:36:28 +02:00
|
|
|
SQL("SELECT"
|
|
|
|
" t.id"
|
|
|
|
", t.rawtx"
|
|
|
|
", t.blockheight"
|
|
|
|
", t.txindex"
|
|
|
|
", t.type as txtype"
|
|
|
|
", c2.short_channel_id as txchan"
|
|
|
|
", a.location"
|
|
|
|
", a.idx as ann_idx"
|
|
|
|
", a.type as annotation_type"
|
|
|
|
", c.short_channel_id"
|
|
|
|
" FROM"
|
|
|
|
" transactions t LEFT JOIN"
|
|
|
|
" transaction_annotations a ON (a.txid = t.id) LEFT JOIN"
|
|
|
|
" channels c ON (a.channel = c.id) LEFT JOIN"
|
|
|
|
" channels c2 ON (t.channel_id = c2.id) "
|
2019-11-14 13:33:11 +01:00
|
|
|
"ORDER BY t.blockheight, t.txindex ASC"));
|
2019-08-20 09:54:38 +02:00
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
for (count = 0; db_step(stmt); count++) {
|
2019-10-02 19:36:28 +02:00
|
|
|
struct bitcoin_txid curtxid;
|
|
|
|
db_column_txid(stmt, 0, &curtxid);
|
|
|
|
|
|
|
|
/* If this is a new entry, allocate it in the array and set
|
2019-11-14 14:14:29 +01:00
|
|
|
* the common fields (all fields from the transactions table. */
|
2019-10-02 19:36:28 +02:00
|
|
|
if (!bitcoin_txid_eq(&last, &curtxid)) {
|
|
|
|
last = curtxid;
|
2019-11-14 14:14:29 +01:00
|
|
|
tal_resize(&txs, tal_count(txs) + 1);
|
|
|
|
cur = &txs[tal_count(txs) - 1];
|
2019-10-02 19:36:28 +02:00
|
|
|
db_column_txid(stmt, 0, &cur->id);
|
|
|
|
cur->tx = db_column_tx(txs, stmt, 1);
|
|
|
|
cur->rawtx = tal_dup_arr(txs, u8, db_column_blob(stmt, 1),
|
|
|
|
db_column_bytes(stmt, 1), 0);
|
2019-11-18 09:56:12 +01:00
|
|
|
/* TX may be unconfirmed. */
|
|
|
|
if (!db_column_is_null(stmt, 2)) {
|
|
|
|
cur->blockheight = db_column_int(stmt, 2);
|
|
|
|
if (!db_column_is_null(stmt, 3)) {
|
|
|
|
cur->txindex = db_column_int(stmt, 3);
|
|
|
|
} else {
|
|
|
|
cur->txindex = 0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
cur->blockheight = 0;
|
|
|
|
cur->txindex = 0;
|
|
|
|
}
|
2019-10-02 19:36:28 +02:00
|
|
|
if (!db_column_is_null(stmt, 4))
|
|
|
|
cur->annotation.type = db_column_u64(stmt, 4);
|
|
|
|
else
|
|
|
|
cur->annotation.type = 0;
|
|
|
|
if (!db_column_is_null(stmt, 5))
|
|
|
|
db_column_short_channel_id(stmt, 5, &cur->annotation.channel);
|
|
|
|
else
|
|
|
|
cur->annotation.channel.u64 = 0;
|
|
|
|
|
|
|
|
cur->output_annotations = tal_arrz(txs, struct tx_annotation, cur->tx->wtx->num_outputs);
|
|
|
|
cur->input_annotations = tal_arrz(txs, struct tx_annotation, cur->tx->wtx->num_inputs);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* This should always be set by the above if-statement,
|
|
|
|
* otherwise we have a txid of all 0x00 bytes... */
|
|
|
|
assert(cur != NULL);
|
|
|
|
|
|
|
|
/* Check if we have any annotations. If there are none the
|
|
|
|
* fields are all set to null */
|
|
|
|
if (!db_column_is_null(stmt, 6)) {
|
|
|
|
enum wallet_tx_annotation_type loc = db_column_int(stmt, 6);
|
|
|
|
int idx = db_column_int(stmt, 7);
|
|
|
|
struct tx_annotation *ann;
|
|
|
|
|
|
|
|
/* Select annotation from array to fill in. */
|
|
|
|
if (loc == OUTPUT_ANNOTATION)
|
|
|
|
ann = &cur->output_annotations[idx];
|
|
|
|
else if (loc == INPUT_ANNOTATION)
|
|
|
|
ann = &cur->input_annotations[idx];
|
|
|
|
else
|
|
|
|
fatal("Transaction annotations are only available for inputs and outputs. Value %d", loc);
|
|
|
|
|
2020-03-02 13:15:50 +01:00
|
|
|
/* cppcheck-suppress uninitvar - false positive on fatal() above */
|
2019-10-02 19:36:28 +02:00
|
|
|
ann->type = db_column_int(stmt, 8);
|
|
|
|
if (!db_column_is_null(stmt, 9))
|
|
|
|
db_column_short_channel_id(stmt, 9, &ann->channel);
|
|
|
|
else
|
|
|
|
ann->channel.u64 = 0;
|
|
|
|
}
|
2019-08-20 09:54:38 +02:00
|
|
|
}
|
|
|
|
tal_free(stmt);
|
2019-05-28 22:36:15 +02:00
|
|
|
return txs;
|
|
|
|
}
|
2020-05-07 02:42:40 +02:00
|
|
|
|
|
|
|
void wallet_penalty_base_add(struct wallet *w, u64 chan_id,
|
|
|
|
const struct penalty_base *pb)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
stmt = db_prepare_v2(w->db,
|
|
|
|
SQL("INSERT INTO penalty_bases ("
|
|
|
|
" channel_id"
|
|
|
|
", commitnum"
|
|
|
|
", txid"
|
|
|
|
", outnum"
|
|
|
|
", amount"
|
|
|
|
") VALUES (?, ?, ?, ?, ?);"));
|
|
|
|
|
|
|
|
db_bind_u64(stmt, 0, chan_id);
|
|
|
|
db_bind_u64(stmt, 1, pb->commitment_num);
|
|
|
|
db_bind_txid(stmt, 2, &pb->txid);
|
|
|
|
db_bind_int(stmt, 3, pb->outnum);
|
|
|
|
db_bind_amount_sat(stmt, 4, &pb->amount);
|
|
|
|
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
|
|
|
}
|
|
|
|
|
|
|
|
struct penalty_base *wallet_penalty_base_load_for_channel(const tal_t *ctx,
|
|
|
|
struct wallet *w,
|
|
|
|
u64 chan_id)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
struct penalty_base *res = tal_arr(ctx, struct penalty_base, 0);
|
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db,
|
|
|
|
SQL("SELECT commitnum, txid, outnum, amount "
|
|
|
|
"FROM penalty_bases "
|
|
|
|
"WHERE channel_id = ?"));
|
|
|
|
|
|
|
|
db_bind_u64(stmt, 0, chan_id);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
while (db_step(stmt)) {
|
|
|
|
struct penalty_base pb;
|
|
|
|
pb.commitment_num = db_column_u64(stmt, 0);
|
|
|
|
db_column_txid(stmt, 1, &pb.txid);
|
|
|
|
pb.outnum = db_column_int(stmt, 2);
|
|
|
|
db_column_amount_sat(stmt, 3, &pb.amount);
|
|
|
|
tal_arr_expand(&res, pb);
|
|
|
|
}
|
|
|
|
tal_free(stmt);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
void wallet_penalty_base_delete(struct wallet *w, u64 chan_id, u64 commitnum)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
stmt = db_prepare_v2(
|
|
|
|
w->db,
|
|
|
|
SQL("DELETE FROM penalty_bases "
|
|
|
|
"WHERE channel_id = ? AND commitnum = ?"));
|
|
|
|
db_bind_u64(stmt, 0, chan_id);
|
|
|
|
db_bind_u64(stmt, 1, commitnum);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
|
|
|
}
|
2020-12-14 02:17:58 +01:00
|
|
|
|
|
|
|
bool wallet_offer_create(struct wallet *w,
|
|
|
|
const struct sha256 *offer_id,
|
|
|
|
const char *bolt12,
|
|
|
|
const struct json_escape *label,
|
|
|
|
enum offer_status status)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
|
|
|
|
assert(offer_status_active(status));
|
|
|
|
|
|
|
|
/* Test if already exists. */
|
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT 1"
|
|
|
|
" FROM offers"
|
|
|
|
" WHERE offer_id = ?;"));
|
|
|
|
db_bind_sha256(stmt, 0, offer_id);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
if (db_step(stmt)) {
|
|
|
|
tal_free(stmt);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
tal_free(stmt);
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(w->db,
|
|
|
|
SQL("INSERT INTO offers ("
|
|
|
|
" offer_id"
|
|
|
|
", bolt12"
|
|
|
|
", label"
|
|
|
|
", status"
|
|
|
|
") VALUES (?, ?, ?, ?);"));
|
|
|
|
|
|
|
|
db_bind_sha256(stmt, 0, offer_id);
|
|
|
|
db_bind_text(stmt, 1, bolt12);
|
|
|
|
if (label)
|
|
|
|
db_bind_json_escape(stmt, 2, label);
|
|
|
|
else
|
|
|
|
db_bind_null(stmt, 2);
|
|
|
|
db_bind_int(stmt, 3, offer_status_in_db(status));
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *wallet_offer_find(const tal_t *ctx,
|
|
|
|
struct wallet *w,
|
|
|
|
const struct sha256 *offer_id,
|
|
|
|
const struct json_escape **label,
|
|
|
|
enum offer_status *status)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
char *bolt12;
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT bolt12, label, status"
|
|
|
|
" FROM offers"
|
|
|
|
" WHERE offer_id = ?;"));
|
|
|
|
db_bind_sha256(stmt, 0, offer_id);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
|
|
|
if (!db_step(stmt)) {
|
|
|
|
tal_free(stmt);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
bolt12 = tal_strdup(ctx, cast_signed(const char *, db_column_text(stmt, 0)));
|
|
|
|
if (label) {
|
|
|
|
if (db_column_is_null(stmt, 1))
|
|
|
|
*label = NULL;
|
|
|
|
else
|
|
|
|
*label = db_column_json_escape(ctx, stmt, 1);
|
|
|
|
}
|
|
|
|
if (status)
|
|
|
|
*status = offer_status_in_db(db_column_int(stmt, 2));
|
|
|
|
tal_free(stmt);
|
|
|
|
return bolt12;
|
|
|
|
}
|
|
|
|
|
2020-12-15 00:43:42 +01:00
|
|
|
struct db_stmt *wallet_offer_id_first(struct wallet *w, struct sha256 *offer_id)
|
2020-12-14 02:17:58 +01:00
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(w->db, SQL("SELECT offer_id FROM offers;"));
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
|
2020-12-15 00:43:42 +01:00
|
|
|
return wallet_offer_id_next(w, stmt, offer_id);
|
2020-12-14 02:17:58 +01:00
|
|
|
}
|
|
|
|
|
2020-12-15 00:43:42 +01:00
|
|
|
struct db_stmt *wallet_offer_id_next(struct wallet *w,
|
|
|
|
struct db_stmt *stmt,
|
|
|
|
struct sha256 *offer_id)
|
2020-12-14 02:17:58 +01:00
|
|
|
{
|
|
|
|
if (!db_step(stmt))
|
|
|
|
return tal_free(stmt);
|
|
|
|
|
|
|
|
db_column_sha256(stmt, 0, offer_id);
|
|
|
|
return stmt;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If we make an offer inactive, this also expires all invoices
|
|
|
|
* which we issued for it. */
|
|
|
|
static void offer_status_update(struct db *db,
|
|
|
|
const struct sha256 *offer_id,
|
|
|
|
enum offer_status oldstatus,
|
|
|
|
enum offer_status newstatus)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(db, SQL("UPDATE offers"
|
|
|
|
" SET status=?"
|
|
|
|
" WHERE offer_id = ?;"));
|
|
|
|
db_bind_int(stmt, 0, offer_status_in_db(newstatus));
|
|
|
|
db_bind_sha256(stmt, 1, offer_id);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
|
|
|
|
|
|
|
if (!offer_status_active(oldstatus)
|
|
|
|
|| offer_status_active(newstatus))
|
|
|
|
return;
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(db, SQL("UPDATE invoices"
|
|
|
|
" SET state=?"
|
2020-12-14 02:18:24 +01:00
|
|
|
" WHERE state=? AND local_offer_id = ?;"));
|
2020-12-14 02:20:44 +01:00
|
|
|
db_bind_int(stmt, 0, invoice_status_in_db(EXPIRED));
|
|
|
|
db_bind_int(stmt, 1, invoice_status_in_db(UNPAID));
|
2020-12-14 02:17:58 +01:00
|
|
|
db_bind_sha256(stmt, 2, offer_id);
|
|
|
|
db_exec_prepared_v2(take(stmt));
|
|
|
|
}
|
|
|
|
|
|
|
|
enum offer_status wallet_offer_disable(struct wallet *w,
|
|
|
|
const struct sha256 *offer_id,
|
|
|
|
enum offer_status s)
|
|
|
|
{
|
|
|
|
enum offer_status newstatus;
|
|
|
|
|
|
|
|
assert(offer_status_active(s));
|
|
|
|
|
|
|
|
newstatus = offer_status_in_db(s &= ~OFFER_STATUS_ACTIVE_F);
|
|
|
|
offer_status_update(w->db, offer_id, s, newstatus);
|
|
|
|
|
|
|
|
return newstatus;
|
|
|
|
}
|
|
|
|
|
|
|
|
void wallet_offer_mark_used(struct db *db, const struct sha256 *offer_id)
|
|
|
|
{
|
|
|
|
struct db_stmt *stmt;
|
|
|
|
enum offer_status status;
|
|
|
|
|
|
|
|
stmt = db_prepare_v2(db, SQL("SELECT status"
|
|
|
|
" FROM offers"
|
|
|
|
" WHERE offer_id = ?;"));
|
|
|
|
db_bind_sha256(stmt, 0, offer_id);
|
|
|
|
db_query_prepared(stmt);
|
|
|
|
if (!db_step(stmt))
|
2020-12-15 00:43:42 +01:00
|
|
|
fatal("%s: unknown offer_id %s",
|
|
|
|
__func__,
|
2020-12-14 02:17:58 +01:00
|
|
|
type_to_string(tmpctx, struct sha256, offer_id));
|
|
|
|
|
|
|
|
status = offer_status_in_db(db_column_int(stmt, 0));
|
|
|
|
tal_free(stmt);
|
|
|
|
|
|
|
|
if (!offer_status_active(status))
|
2020-12-15 00:43:42 +01:00
|
|
|
fatal("%s: offer_id %s not active: status %i",
|
|
|
|
__func__,
|
2020-12-14 02:17:58 +01:00
|
|
|
type_to_string(tmpctx, struct sha256, offer_id),
|
|
|
|
status);
|
|
|
|
|
|
|
|
if (status == OFFER_SINGLE_USE)
|
|
|
|
offer_status_update(db, offer_id, status, OFFER_USED);
|
|
|
|
}
|