You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
454 lines
14 KiB
454 lines
14 KiB
#ifndef WALLET_WALLET_H
|
|
#define WALLET_WALLET_H
|
|
|
|
#include "config.h"
|
|
#include "db.h"
|
|
#include <bitcoin/tx.h>
|
|
#include <ccan/crypto/shachain/shachain.h>
|
|
#include <ccan/list/list.h>
|
|
#include <ccan/tal/tal.h>
|
|
#include <common/channel_config.h>
|
|
#include <common/utxo.h>
|
|
#include <lightningd/htlc_end.h>
|
|
#include <lightningd/invoice.h>
|
|
#include <onchaind/onchain_wire.h>
|
|
#include <wally_bip32.h>
|
|
|
|
struct lightningd;
|
|
struct pubkey;
|
|
|
|
struct wallet {
|
|
struct db *db;
|
|
struct log *log;
|
|
struct ext_key *bip32_base;
|
|
};
|
|
|
|
/* Possible states for tracked outputs in the database. Not sure yet
|
|
* whether we really want to have reservations reflected in the
|
|
* database, it would simplify queries at the cost of some IO ops */
|
|
enum output_status {
|
|
output_state_available= 0,
|
|
output_state_reserved = 1,
|
|
output_state_spent = 2,
|
|
/* Special status used to express that we don't care in
|
|
* queries */
|
|
output_state_any = 255
|
|
};
|
|
|
|
/* Enumeration of all known output types. These include all types that
|
|
* could ever end up on-chain and we may need to react upon. Notice
|
|
* that `to_local`, `htlc_offer`, and `htlc_recv` may need immediate
|
|
* action since they are encumbered with a CSV. */
|
|
enum wallet_output_type {
|
|
p2sh_wpkh = 0,
|
|
to_local = 1,
|
|
htlc_offer = 3,
|
|
htlc_recv = 4,
|
|
our_change = 5
|
|
};
|
|
|
|
/* A database backed shachain struct. The datastructure is
|
|
* writethrough, reads are performed from an in-memory version, all
|
|
* writes are passed through to the DB. */
|
|
struct wallet_shachain {
|
|
u64 id;
|
|
struct shachain chain;
|
|
};
|
|
|
|
/* A database backed peer struct. Like wallet_shachain, it is writethrough. */
|
|
/* TODO(cdecker) Separate peer from channel */
|
|
struct wallet_channel {
|
|
u64 id;
|
|
struct peer *peer;
|
|
};
|
|
|
|
/* Possible states for a wallet_payment. Payments start in
|
|
* `PENDING`. Outgoing payments are set to `PAYMENT_COMPLETE` once we
|
|
* get the preimage matching the rhash, or to
|
|
* `PAYMENT_FAILED`. Incoming payments are set to `PAYMENT_COMPLETE`
|
|
* once the matching invoice is marked as complete, or `FAILED`
|
|
* otherwise. */
|
|
/* /!\ This is a DB ENUM, please do not change the numbering of any
|
|
* already defined elements (adding is ok) /!\ */
|
|
enum wallet_payment_status {
|
|
PAYMENT_PENDING = 0,
|
|
PAYMENT_COMPLETE = 1,
|
|
PAYMENT_FAILED = 2
|
|
};
|
|
|
|
/* Incoming and outgoing payments. A simple persisted representation
|
|
* of a payment we either initiated or received. This can be used by
|
|
* a UI to display the balance history. We explicitly exclude
|
|
* forwarded payments.
|
|
*/
|
|
struct wallet_payment {
|
|
u64 id;
|
|
u32 timestamp;
|
|
bool incoming;
|
|
struct sha256 payment_hash;
|
|
enum wallet_payment_status status;
|
|
struct pubkey *destination;
|
|
u64 msatoshi;
|
|
};
|
|
|
|
/**
|
|
* wallet_new - Constructor for a new sqlite3 based wallet
|
|
*
|
|
* This is guaranteed to either return a valid wallet, or abort with
|
|
* `fatal` if it cannot be initialized.
|
|
*/
|
|
struct wallet *wallet_new(const tal_t *ctx, struct log *log);
|
|
|
|
/**
|
|
* wallet_add_utxo - Register a UTXO which we (partially) own
|
|
*
|
|
* Add a UTXO to the set of outputs we care about.
|
|
*/
|
|
bool wallet_add_utxo(struct wallet *w, struct utxo *utxo,
|
|
enum wallet_output_type type);
|
|
|
|
/**
|
|
* wallet_update_output_status - Perform an output state transition
|
|
*
|
|
* Change the current status of an output we are tracking in the
|
|
* database. Returns true if the output exists with the @oldstatus and
|
|
* was successfully updated to @newstatus. May fail if either the
|
|
* output does not exist, or it does not have the expected
|
|
* @oldstatus. In case we don't care about the previous state use
|
|
* `output_state_any` as @oldstatus.
|
|
*/
|
|
bool wallet_update_output_status(struct wallet *w,
|
|
const struct bitcoin_txid *txid,
|
|
const u32 outnum, enum output_status oldstatus,
|
|
enum output_status newstatus);
|
|
|
|
/**
|
|
* wallet_get_utxos - Retrieve all utxos matching a given state
|
|
*
|
|
* Returns a `tal_arr` of `utxo` structs. Double indirection in order
|
|
* to be able to steal individual elements onto something else.
|
|
*/
|
|
struct utxo **wallet_get_utxos(const tal_t *ctx, struct wallet *w,
|
|
const enum output_status state);
|
|
|
|
const struct utxo **wallet_select_coins(const tal_t *ctx, struct wallet *w,
|
|
const u64 value,
|
|
const u32 feerate_per_kw,
|
|
size_t outscriptlen,
|
|
u64 *fee_estimate,
|
|
u64 *change_satoshi);
|
|
|
|
const struct utxo **wallet_select_all(const tal_t *ctx, struct wallet *w,
|
|
const u32 feerate_per_kw,
|
|
size_t outscriptlen,
|
|
u64 *value,
|
|
u64 *fee_estimate);
|
|
|
|
/**
|
|
* wallet_confirm_utxos - Once we've spent a set of utxos, mark them confirmed.
|
|
*
|
|
* May be called once the transaction spending these UTXOs has been
|
|
* broadcast. If something fails use `tal_free(utxos)` instead to undo
|
|
* the reservation.
|
|
*/
|
|
void wallet_confirm_utxos(struct wallet *w, const struct utxo **utxos);
|
|
|
|
/**
|
|
* wallet_can_spend - Do we have the private key matching this scriptpubkey?
|
|
*
|
|
* FIXME: This is very slow with lots of inputs!
|
|
*
|
|
* @w: (in) allet holding the pubkeys to check against (privkeys are on HSM)
|
|
* @script: (in) the script to check
|
|
* @index: (out) the bip32 derivation index that matched the script
|
|
* @output_is_p2sh: (out) whether the script is a p2sh, or p2wpkh
|
|
*/
|
|
bool wallet_can_spend(struct wallet *w, const u8 *script,
|
|
u32 *index, bool *output_is_p2sh);
|
|
|
|
/**
|
|
* wallet_get_newindex - get a new index from the wallet.
|
|
* @ld: (in) lightning daemon
|
|
*
|
|
* Returns -1 on error (key exhaustion).
|
|
*/
|
|
s64 wallet_get_newindex(struct lightningd *ld);
|
|
|
|
/**
|
|
* wallet_shachain_init -- wallet wrapper around shachain_init
|
|
*/
|
|
void wallet_shachain_init(struct wallet *wallet, struct wallet_shachain *chain);
|
|
|
|
/**
|
|
* wallet_shachain_add_hash -- wallet wrapper around shachain_add_hash
|
|
*/
|
|
bool wallet_shachain_add_hash(struct wallet *wallet,
|
|
struct wallet_shachain *chain,
|
|
uint64_t index,
|
|
const struct sha256 *hash);
|
|
|
|
/**
|
|
* wallet_shachain_load -- Load an existing shachain from the wallet.
|
|
*
|
|
* @wallet: the wallet to load from
|
|
* @id: the shachain id to load
|
|
* @chain: where to load the shachain into
|
|
*/
|
|
bool wallet_shachain_load(struct wallet *wallet, u64 id,
|
|
struct wallet_shachain *chain);
|
|
|
|
|
|
/**
|
|
* wallet_channel_save -- Upsert the channel into the database
|
|
*
|
|
* @wallet: the wallet to save into
|
|
* @chan: the instance to store (not const so we can update the unique_id upon
|
|
* insert)
|
|
* @current_block_height: current height, minimum block this funding tx could
|
|
* be in (only used on initial insert).
|
|
*/
|
|
void wallet_channel_save(struct wallet *w, struct wallet_channel *chan,
|
|
u32 current_block_height);
|
|
|
|
/**
|
|
* wallet_channel_config_save -- Upsert a channel_config into the database
|
|
*/
|
|
void wallet_channel_config_save(struct wallet *w, struct channel_config *cc);
|
|
|
|
/**
|
|
* wallet_channel_config_load -- Load channel_config from database into cc
|
|
*/
|
|
bool wallet_channel_config_load(struct wallet *w, const u64 id,
|
|
struct channel_config *cc);
|
|
|
|
/**
|
|
* wallet_peer_by_nodeid -- Given a node_id/pubkey, load the peer from DB
|
|
*
|
|
* @w: the wallet to load from
|
|
* @nodeid: the node_id to search for
|
|
* @peer(out): the destination where to store the peer
|
|
*
|
|
* Returns true on success, or false if we were unable to find a peer
|
|
* with the given node_id.
|
|
*/
|
|
bool wallet_peer_by_nodeid(struct wallet *w, const struct pubkey *nodeid,
|
|
struct peer *peer);
|
|
|
|
/**
|
|
* wlalet_channels_load_active -- Load persisted active channels into the peers
|
|
*
|
|
* @ctx: context to allocate peers from
|
|
* @w: wallet to load from
|
|
* @peers: list_head to load channels/peers into
|
|
*
|
|
* Be sure to call this only once on startup since it'll append peers
|
|
* loaded from the database to the list without checking.
|
|
*/
|
|
bool wallet_channels_load_active(const tal_t *ctx,
|
|
struct wallet *w, struct list_head *peers);
|
|
|
|
/**
|
|
* wallet_channels_first_blocknum - get first block we're interested in.
|
|
*
|
|
* @w: wallet to load from.
|
|
*
|
|
* Returns UINT32_MAX if nothing interesting.
|
|
*/
|
|
u32 wallet_channels_first_blocknum(struct wallet *w);
|
|
|
|
/**
|
|
* wallet_extract_owned_outputs - given a tx, extract all of our outputs
|
|
*/
|
|
int wallet_extract_owned_outputs(struct wallet *w, const struct bitcoin_tx *tx,
|
|
u64 *total_satoshi);
|
|
|
|
/**
|
|
* wallet_htlc_save_in - store a htlc_in in the database
|
|
*
|
|
* @wallet: wallet to store the htlc into
|
|
* @chan: the `wallet_channel` this HTLC is associated with
|
|
* @in: the htlc_in to store
|
|
*
|
|
* This will store the contents of the `struct htlc_in` in the
|
|
* database. Since `struct htlc_in` commonly only change state after
|
|
* being created we do not support updating arbitrary fields and this
|
|
* function will fail when attempting to call it multiple times for
|
|
* the same `struct htlc_in`. Instead `wallet_htlc_update` may be used
|
|
* for state transitions or to set the `payment_key` for completed
|
|
* HTLCs.
|
|
*/
|
|
void wallet_htlc_save_in(struct wallet *wallet,
|
|
const struct wallet_channel *chan, struct htlc_in *in);
|
|
|
|
/**
|
|
* wallet_htlc_save_out - store a htlc_out in the database
|
|
*
|
|
* See comment for wallet_htlc_save_in.
|
|
*/
|
|
void wallet_htlc_save_out(struct wallet *wallet,
|
|
const struct wallet_channel *chan,
|
|
struct htlc_out *out);
|
|
|
|
/**
|
|
* wallet_htlc_update - perform state transition or add payment_key
|
|
*
|
|
* @wallet: the wallet containing the HTLC to update
|
|
* @htlc_dbid: the database ID used to identify the HTLC
|
|
* @new_state: the state we should transition to
|
|
* @payment_key: the `payment_key` which hashes to the `payment_hash`,
|
|
* or NULL if unknown.
|
|
*
|
|
* Used to update the state of an HTLC, either a `struct htlc_in` or a
|
|
* `struct htlc_out` and optionally set the `payment_key` should the
|
|
* HTLC have been settled.
|
|
*/
|
|
void wallet_htlc_update(struct wallet *wallet, const u64 htlc_dbid,
|
|
const enum htlc_state new_state,
|
|
const struct preimage *payment_key);
|
|
|
|
/**
|
|
* wallet_htlcs_load_for_channel - Load HTLCs associated with chan from DB.
|
|
*
|
|
* @wallet: wallet to load from
|
|
* @chan: load HTLCs associated with this channel
|
|
* @htlcs_in: htlc_in_map to store loaded htlc_in in
|
|
* @htlcs_out: htlc_out_map to store loaded htlc_out in
|
|
*
|
|
* This function looks for HTLCs that are associated with the given
|
|
* channel and loads them into the provided maps. One caveat is that
|
|
* the `struct htlc_out` instances are not wired up with the
|
|
* corresponding `struct htlc_in` in the forwarding case nor are they
|
|
* associated with a `struct pay_command` in the case we originated
|
|
* the payment. In the former case the corresponding `struct htlc_in`
|
|
* may not have been loaded yet. In the latter case the pay_command
|
|
* does not exist anymore since we restarted.
|
|
*
|
|
* Use `wallet_htlcs_reconnect` to wire htlc_out instances to the
|
|
* corresponding htlc_in after loading all channels.
|
|
*/
|
|
bool wallet_htlcs_load_for_channel(struct wallet *wallet,
|
|
struct wallet_channel *chan,
|
|
struct htlc_in_map *htlcs_in,
|
|
struct htlc_out_map *htlcs_out);
|
|
|
|
/**
|
|
* wallet_htlcs_reconnect -- Link outgoing HTLCs to their origins
|
|
*
|
|
* For each outgoing HTLC find the incoming HTLC that triggered it. If
|
|
* we are the origin of the transfer then we cannot resolve the
|
|
* incoming HTLC in which case we just leave it `NULL`.
|
|
*/
|
|
bool wallet_htlcs_reconnect(struct wallet *wallet,
|
|
struct htlc_in_map *htlcs_in,
|
|
struct htlc_out_map *htlcs_out);
|
|
|
|
/**
|
|
* wallet_invoice_nextpaid -- Find a paid invoice.
|
|
*
|
|
* Get the details (label, rhash, msatoshi, pay_index) of the first paid
|
|
* invoice greater than the given pay_index. Return false if no paid
|
|
* invoice found, return true if found. The first ever paid invoice will
|
|
* have a pay_index of 1 or greater, so giving a pay_index of 0 will get
|
|
* the first ever paid invoice if there is one.
|
|
*
|
|
* @ctx: Context to create the returned label.
|
|
* @wallet: Wallet to query
|
|
* @pay_index: The paid invoice returned will have pay_index greater
|
|
* than this argument.
|
|
* @outlabel: Pointer to label of found paid invoice. Caller
|
|
* must free if this function returns true.
|
|
* @outrhash: Pointer to struct rhash to be filled.
|
|
* @outmsatoshi: Pointer to number of millisatoshis value to pay.
|
|
* @outpay_index: Pointer to pay_index of found paid invoice.
|
|
*/
|
|
bool wallet_invoice_nextpaid(const tal_t *cxt,
|
|
const struct wallet *wallet,
|
|
u64 pay_index,
|
|
char **outlabel,
|
|
struct sha256 *outrhash,
|
|
u64 *outmsatoshi,
|
|
u64 *outpay_index);
|
|
|
|
/**
|
|
* wallet_invoice_save -- Save/update an invoice to the wallet
|
|
*
|
|
* Save or update the invoice in the wallet. If `inv->id` is 0 this
|
|
* invoice will be considered a new invoice and result in an intert
|
|
* into the database, otherwise it'll be updated.
|
|
*
|
|
* @wallet: Wallet to store in
|
|
* @inv: Invoice to save
|
|
*/
|
|
void wallet_invoice_save(struct wallet *wallet, struct invoice *inv);
|
|
|
|
/**
|
|
* wallet_invoices_load -- Load all invoices into memory
|
|
*
|
|
* Load all invoices into the given `invoices` struct.
|
|
*
|
|
* @wallet: Wallet to load invoices from
|
|
* @invs: invoices container to load into
|
|
*/
|
|
bool wallet_invoices_load(struct wallet *wallet, struct invoices *invs);
|
|
|
|
/**
|
|
* wallet_invoice_remove -- Remove the specified invoice from the wallet
|
|
*
|
|
* Remove the invoice from the underlying database. The invoice is
|
|
* identified by `inv->id` so if the caller does not have the full
|
|
* invoice, it may just instantiate a new one and set the `id` to
|
|
* match the desired invoice.
|
|
*
|
|
* @wallet: Wallet to remove from
|
|
* @inv: Invoice to remove.
|
|
*/
|
|
bool wallet_invoice_remove(struct wallet *wallet, struct invoice *inv);
|
|
|
|
/**
|
|
* wallet_htlc_stubs - Retrieve HTLC stubs for the given channel
|
|
*
|
|
* Load minimal necessary information about HTLCs for the on-chain
|
|
* settlement. This returns a `tal_arr` allocated off of @ctx with the
|
|
* necessary size to hold all HTLCs.
|
|
*
|
|
* @ctx: Allocation context for the return value
|
|
* @wallet: Wallet to load from
|
|
* @chan: Channel to fetch stubs for
|
|
*/
|
|
struct htlc_stub *wallet_htlc_stubs(const tal_t *ctx, struct wallet *wallet,
|
|
struct wallet_channel *chan);
|
|
|
|
/**
|
|
* wallet_payment_add - Record a new incoming/outgoing payment
|
|
*
|
|
* Stores the payment in the database.
|
|
*/
|
|
bool wallet_payment_add(struct wallet *wallet,
|
|
struct wallet_payment *payment);
|
|
|
|
/**
|
|
* wallet_payment_by_hash - Retrieve a specific payment
|
|
*
|
|
* Given the `payment_hash` retrieve the matching payment.
|
|
*/
|
|
struct wallet_payment *
|
|
wallet_payment_by_hash(const tal_t *ctx, struct wallet *wallet,
|
|
const struct sha256 *payment_hash);
|
|
|
|
/**
|
|
* wallet_payment_set_status - Update the status of the payment
|
|
*
|
|
* Search for the payment with the given `payment_hash` and update
|
|
* its state.
|
|
*/
|
|
void wallet_payment_set_status(struct wallet *wallet,
|
|
const struct sha256 *payment_hash,
|
|
const enum wallet_payment_status newstatus);
|
|
|
|
/**
|
|
* wallet_payment_list - Retrieve a list of payments
|
|
*/
|
|
const struct wallet_payment **wallet_payment_list(const tal_t *ctx,
|
|
struct wallet *wallet);
|
|
|
|
#endif /* WALLET_WALLET_H */
|
|
|