2017-06-20 17:15:25 +02:00
# include <bitcoin/address.h>
# include <bitcoin/base58.h>
# include <bitcoin/script.h>
# include <ccan/tal/str/str.h>
2017-12-11 04:43:11 +01:00
# include <common/bech32.h>
2017-08-28 18:05:01 +02:00
# include <common/key_derive.h>
# include <common/status.h>
# include <common/utxo.h>
2017-08-28 18:02:01 +02:00
# include <common/withdraw_tx.h>
2017-06-24 08:50:14 +02:00
# include <errno.h>
2017-11-30 17:07:38 +01:00
# include <hsmd/gen_hsm_client_wire.h>
2017-12-19 21:10:19 +01:00
# include <inttypes.h>
2017-08-28 18:04:01 +02:00
# include <lightningd/bitcoind.h>
# include <lightningd/chaintopology.h>
2017-06-24 08:50:23 +02:00
# include <lightningd/hsm_control.h>
2017-08-28 18:04:01 +02:00
# include <lightningd/jsonrpc.h>
2017-06-20 17:15:25 +02:00
# include <lightningd/lightningd.h>
2017-08-28 18:04:01 +02:00
# include <lightningd/log.h>
2018-02-09 14:24:33 +01:00
# include <lightningd/peer_control.h>
2017-06-20 17:15:25 +02:00
# include <lightningd/subd.h>
# include <wally_bip32.h>
2017-06-24 08:50:14 +02:00
# include <wire/wire_sync.h>
2017-06-20 17:15:25 +02:00
struct withdrawal {
u64 amount , changesatoshi ;
2017-12-10 06:48:18 +01:00
u8 * destination ;
2017-06-20 17:15:25 +02:00
const struct utxo * * utxos ;
u64 change_key_index ;
struct command * cmd ;
const char * hextx ;
} ;
/**
* wallet_withdrawal_broadcast - The tx has been broadcast ( or it failed )
*
* This is the final step in the withdrawal . We either successfully
* broadcast the withdrawal transaction or it failed somehow . So we
* report success or a broadcast failure . Upon success we also mark
* the used outputs as spent , and add the change output to our pool of
* available outputs .
*/
2018-02-06 14:00:12 +01:00
static void wallet_withdrawal_broadcast ( struct bitcoind * bitcoind UNUSED ,
2017-06-20 17:15:25 +02:00
int exitstatus , const char * msg ,
struct withdrawal * withdraw )
{
struct command * cmd = withdraw - > cmd ;
2017-08-28 18:09:01 +02:00
struct lightningd * ld = withdraw - > cmd - > ld ;
2017-06-22 14:20:41 +02:00
struct bitcoin_tx * tx ;
u64 change_satoshi = 0 ;
2017-06-20 17:15:25 +02:00
/* Massage output into shape so it doesn't kill the JSON serialization */
char * output = tal_strjoin ( cmd , tal_strsplit ( cmd , msg , " \n " , STR_NO_EMPTY ) , " " , STR_NO_TRAIL ) ;
if ( exitstatus = = 0 ) {
2017-06-22 14:20:41 +02:00
/* Mark used outputs as spent */
2017-06-20 17:15:25 +02:00
wallet_confirm_utxos ( ld - > wallet , withdraw - > utxos ) ;
2017-06-22 14:20:41 +02:00
/* Parse the tx and extract the change output. We
* generated the hex tx , so this should always work */
tx = bitcoin_tx_from_hex ( withdraw , withdraw - > hextx , strlen ( withdraw - > hextx ) ) ;
assert ( tx ! = NULL ) ;
wallet_extract_owned_outputs ( ld - > wallet , tx , & change_satoshi ) ;
2018-01-29 06:59:21 +01:00
/* Note normally, change_satoshi == withdraw->changesatoshi, but
* not if we ' re actually making a payment to ourselves ! */
2018-02-05 18:48:57 +01:00
assert ( change_satoshi > = withdraw - > changesatoshi ) ;
2017-06-22 14:20:41 +02:00
2017-06-20 17:15:25 +02:00
struct json_result * response = new_json_result ( cmd ) ;
json_object_start ( response , NULL ) ;
json_add_string ( response , " tx " , withdraw - > hextx ) ;
json_add_string ( response , " txid " , output ) ;
json_object_end ( response ) ;
command_success ( cmd , response ) ;
} else {
command_fail ( cmd , " Error broadcasting transaction: %s " , output ) ;
}
}
/**
* json_withdraw - Entrypoint for the withdrawal flow
*
* A user has requested a withdrawal over the JSON - RPC , parse the
* request , select coins and a change key . Then send the request to
* the HSM to generate the signatures .
*/
static void json_withdraw ( struct command * cmd ,
const char * buffer , const jsmntok_t * params )
{
jsmntok_t * desttok , * sattok ;
struct withdrawal * withdraw ;
2017-12-19 21:12:41 +01:00
u32 feerate_per_kw = get_feerate ( cmd - > ld - > topology , FEERATE_NORMAL ) ;
2017-06-20 17:15:25 +02:00
u64 fee_estimate ;
2017-06-24 08:50:14 +02:00
struct bitcoin_tx * tx ;
2017-12-19 21:10:19 +01:00
bool withdraw_all = false ;
2018-02-24 15:59:39 +01:00
enum address_parse_result addr_parse ;
2017-06-24 08:50:14 +02:00
2018-01-29 01:34:28 +01:00
if ( ! json_get_params ( cmd , buffer , params ,
2017-06-20 17:15:25 +02:00
" destination " , & desttok ,
" satoshi " , & sattok ,
NULL ) ) {
return ;
}
withdraw = tal ( cmd , struct withdrawal ) ;
withdraw - > cmd = cmd ;
2017-12-19 21:10:19 +01:00
if ( json_tok_streq ( buffer , sattok , " all " ) )
withdraw_all = true ;
else if ( ! json_tok_u64 ( buffer , sattok , & withdraw - > amount ) ) {
2017-06-20 17:15:25 +02:00
command_fail ( cmd , " Invalid satoshis " ) ;
return ;
}
2017-12-10 07:49:51 +01:00
2017-12-10 12:49:46 +01:00
/* Parse address. */
2018-02-24 15:59:39 +01:00
addr_parse = json_tok_address_scriptpubkey ( cmd ,
get_chainparams ( cmd - > ld ) ,
buffer , desttok ,
( const u8 * * ) ( & withdraw - > destination ) ) ;
2017-12-10 07:49:51 +01:00
/* Check that destination address could be understood. */
2018-02-24 15:59:39 +01:00
if ( addr_parse = = ADDRESS_PARSE_UNRECOGNIZED ) {
2017-06-20 17:15:25 +02:00
command_fail ( cmd , " Could not parse destination address " ) ;
return ;
}
2017-12-09 14:08:40 +01:00
/* Check address given is compatible with the chain we are on. */
2018-02-24 15:59:39 +01:00
if ( addr_parse = = ADDRESS_PARSE_WRONG_NETWORK ) {
2018-02-15 00:45:04 +01:00
command_fail ( cmd ,
2018-02-24 15:59:39 +01:00
" Destination address is not on network %s " ,
2018-02-15 00:45:04 +01:00
get_chainparams ( cmd - > ld ) - > network_name ) ;
2017-12-09 14:08:40 +01:00
return ;
}
2017-06-20 17:15:25 +02:00
/* Select the coins */
2017-12-19 21:10:19 +01:00
if ( withdraw_all ) {
withdraw - > utxos = wallet_select_all ( cmd , cmd - > ld - > wallet ,
feerate_per_kw ,
2017-12-20 07:05:01 +01:00
tal_len ( withdraw - > destination ) ,
2017-12-19 21:10:19 +01:00
& withdraw - > amount ,
& fee_estimate ) ;
/* FIXME Pull dust amount from the daemon config */
if ( ! withdraw - > utxos | | withdraw - > amount < 546 ) {
command_fail ( cmd , " Cannot afford fee % " PRIu64 ,
fee_estimate ) ;
return ;
}
withdraw - > changesatoshi = 0 ;
} else {
withdraw - > utxos = wallet_select_coins ( cmd , cmd - > ld - > wallet ,
withdraw - > amount ,
feerate_per_kw ,
2017-12-20 07:05:01 +01:00
tal_len ( withdraw - > destination ) ,
2017-12-19 21:10:19 +01:00
& fee_estimate ,
& withdraw - > changesatoshi ) ;
if ( ! withdraw - > utxos ) {
command_fail ( cmd , " Not enough funds available " ) ;
return ;
}
2017-06-20 17:15:25 +02:00
}
2017-06-22 15:19:20 +02:00
/* FIXME(cdecker) Pull this from the daemon config */
if ( withdraw - > changesatoshi < = 546 )
withdraw - > changesatoshi = 0 ;
2017-12-20 06:56:40 +01:00
if ( withdraw - > changesatoshi )
withdraw - > change_key_index = wallet_get_newindex ( cmd - > ld ) ;
else
withdraw - > change_key_index = 0 ;
2017-06-20 17:15:25 +02:00
2017-11-30 17:07:38 +01:00
u8 * msg = towire_hsm_sign_withdrawal ( cmd ,
withdraw - > amount ,
withdraw - > changesatoshi ,
withdraw - > change_key_index ,
2017-12-10 07:24:38 +01:00
withdraw - > destination ,
2018-02-08 02:24:46 +01:00
withdraw - > utxos ) ;
2017-06-24 08:50:14 +02:00
2017-08-28 18:09:01 +02:00
if ( ! wire_sync_write ( cmd - > ld - > hsm_fd , take ( msg ) ) )
2017-06-24 08:50:14 +02:00
fatal ( " Could not write sign_withdrawal to HSM: %s " ,
strerror ( errno ) ) ;
2017-08-28 18:09:01 +02:00
msg = hsm_sync_read ( cmd , cmd - > ld ) ;
2017-06-24 08:50:14 +02:00
2018-02-20 21:59:09 +01:00
if ( ! fromwire_hsm_sign_withdrawal_reply ( msg , msg , & tx ) )
2017-06-24 08:50:14 +02:00
fatal ( " HSM gave bad sign_withdrawal_reply %s " ,
tal_hex ( withdraw , msg ) ) ;
/* Now broadcast the transaction */
withdraw - > hextx = tal_hex ( withdraw , linearize_tx ( cmd , tx ) ) ;
2017-08-28 18:09:01 +02:00
bitcoind_sendrawtx ( cmd - > ld - > topology - > bitcoind , withdraw - > hextx ,
2017-06-24 08:50:14 +02:00
wallet_withdrawal_broadcast , withdraw ) ;
2017-12-15 11:15:54 +01:00
command_still_pending ( cmd ) ;
2017-06-20 17:15:25 +02:00
}
static const struct json_command withdraw_command = {
" withdraw " ,
json_withdraw ,
2018-02-06 14:00:12 +01:00
" Send to {destination} address {satoshi} (or 'all') amount via Bitcoin transaction " ,
false , " Send funds from the internal wallet to the specified address. Either specify a number of satoshis to send or 'all' to sweep all funds in the internal wallet to the address. "
2017-06-20 17:15:25 +02:00
} ;
AUTODATA ( json_command , & withdraw_command ) ;
2017-06-21 17:18:55 +02:00
2018-02-06 14:00:12 +01:00
static void json_newaddr ( struct command * cmd , const char * buffer UNUSED ,
const jsmntok_t * params UNUSED )
2017-06-21 17:18:55 +02:00
{
struct json_result * response = new_json_result ( cmd ) ;
struct ext_key ext ;
struct sha256 h ;
2018-01-24 08:01:21 +01:00
struct ripemd160 h160 ;
2017-06-21 17:18:55 +02:00
struct pubkey pubkey ;
2018-01-24 08:01:21 +01:00
jsmntok_t * addrtype ;
2017-06-21 17:18:55 +02:00
u8 * redeemscript ;
2018-01-24 08:01:21 +01:00
bool is_p2wpkh , ok ;
2017-06-25 04:33:13 +02:00
s64 keyidx ;
2018-01-24 08:01:21 +01:00
char * out ;
const char * hrp ;
if ( ! json_get_params ( cmd , buffer , params ,
" ?addresstype " , & addrtype , NULL ) ) {
return ;
}
if ( ! addrtype | | json_tok_streq ( buffer , addrtype , " p2sh-segwit " ) )
is_p2wpkh = false ;
else if ( json_tok_streq ( buffer , addrtype , " bech32 " ) )
is_p2wpkh = true ;
else {
command_fail ( cmd ,
" Invalid address type "
" (expected bech32 or p2sh-segwit) " ) ;
return ;
}
2017-06-21 17:18:55 +02:00
2017-08-28 18:09:01 +02:00
keyidx = wallet_get_newindex ( cmd - > ld ) ;
2017-06-25 04:33:13 +02:00
if ( keyidx < 0 ) {
2017-06-21 17:18:55 +02:00
command_fail ( cmd , " Keys exhausted " ) ;
return ;
}
2017-08-28 18:09:01 +02:00
if ( bip32_key_from_parent ( cmd - > ld - > wallet - > bip32_base , keyidx ,
2017-06-21 17:18:55 +02:00
BIP32_FLAG_KEY_PUBLIC , & ext ) ! = WALLY_OK ) {
command_fail ( cmd , " Keys generation failure " ) ;
return ;
}
if ( ! secp256k1_ec_pubkey_parse ( secp256k1_ctx , & pubkey . pubkey ,
ext . pub_key , sizeof ( ext . pub_key ) ) ) {
command_fail ( cmd , " Key parsing failure " ) ;
return ;
}
2017-11-27 16:20:10 +01:00
txfilter_add_derkey ( cmd - > ld - > owned_txfilter , ext . pub_key ) ;
2018-01-24 08:01:21 +01:00
if ( is_p2wpkh ) {
hrp = get_chainparams ( cmd - > ld ) - > bip173_name ;
/* out buffer is 73 + strlen(human readable part). see bech32.h */
out = tal_arr ( cmd , char , 73 + strlen ( hrp ) ) ;
pubkey_to_hash160 ( & pubkey , & h160 ) ;
ok = segwit_addr_encode ( out , hrp , 0 , h160 . u . u8 , sizeof ( h160 . u . u8 ) ) ;
if ( ! ok ) {
command_fail ( cmd , " p2wpkh address encoding failure. " ) ;
return ;
}
}
else {
redeemscript = bitcoin_redeem_p2sh_p2wpkh ( cmd , & pubkey ) ;
sha256 ( & h , redeemscript , tal_count ( redeemscript ) ) ;
ripemd160 ( & h160 , h . u . u8 , sizeof ( h ) ) ;
out = p2sh_to_base58 ( cmd ,
get_chainparams ( cmd - > ld ) - > testnet , & h160 ) ;
}
2017-06-21 17:18:55 +02:00
json_object_start ( response , NULL ) ;
2018-01-24 08:01:21 +01:00
json_add_string ( response , " address " , out ) ;
2017-06-21 17:18:55 +02:00
json_object_end ( response ) ;
command_success ( cmd , response ) ;
}
static const struct json_command newaddr_command = {
" newaddr " ,
json_newaddr ,
2018-02-06 14:13:57 +01:00
" Get a new {bech32, p2sh-segwit} address to fund a channel " , false ,
" Generates a new address that belongs to the internal wallet. Funds sent to these addresses will be managed by lightningd. Use `withdraw` to withdraw funds to an external wallet. "
2017-06-21 17:18:55 +02:00
} ;
AUTODATA ( json_command , & newaddr_command ) ;
2017-06-21 17:40:42 +02:00
2018-02-18 13:52:46 +01:00
static void json_listaddrs ( struct command * cmd ,
const char * buffer , const jsmntok_t * params )
{
struct json_result * response = new_json_result ( cmd ) ;
struct ext_key ext ;
struct sha256 h ;
struct ripemd160 h160 ;
struct pubkey pubkey ;
jsmntok_t * bip32tok ;
u8 * redeemscript ;
bool ok ;
char * out_p2sh ;
char * out_p2wpkh ;
const char * hrp ;
u64 bip32_max_index ;
if ( ! json_get_params ( cmd , buffer , params ,
" ?bip32_max_index " , & bip32tok ,
NULL ) ) {
return ;
}
if ( ! bip32tok | | ! json_tok_u64 ( buffer , bip32tok , & bip32_max_index ) ) {
bip32_max_index = db_get_intvar ( cmd - > ld - > wallet - > db , " bip32_max_index " , 0 ) ;
}
json_object_start ( response , NULL ) ;
json_array_start ( response , " addresses " ) ;
2018-02-23 18:56:18 +01:00
for ( s64 keyidx = 0 ; keyidx < = bip32_max_index ; keyidx + + ) {
2018-02-18 13:52:46 +01:00
if ( keyidx = = BIP32_INITIAL_HARDENED_CHILD ) {
break ;
}
if ( bip32_key_from_parent ( cmd - > ld - > wallet - > bip32_base , keyidx ,
BIP32_FLAG_KEY_PUBLIC , & ext ) ! = WALLY_OK ) {
command_fail ( cmd , " Keys generation failure " ) ;
return ;
}
if ( ! secp256k1_ec_pubkey_parse ( secp256k1_ctx , & pubkey . pubkey ,
ext . pub_key , sizeof ( ext . pub_key ) ) ) {
command_fail ( cmd , " Key parsing failure " ) ;
return ;
}
// p2sh
redeemscript = bitcoin_redeem_p2sh_p2wpkh ( cmd , & pubkey ) ;
sha256 ( & h , redeemscript , tal_count ( redeemscript ) ) ;
ripemd160 ( & h160 , h . u . u8 , sizeof ( h ) ) ;
out_p2sh = p2sh_to_base58 ( cmd ,
get_chainparams ( cmd - > ld ) - > testnet , & h160 ) ;
// bech32 : p2wpkh
hrp = get_chainparams ( cmd - > ld ) - > bip173_name ;
/* out buffer is 73 + strlen(human readable part). see bech32.h */
out_p2wpkh = tal_arr ( cmd , char , 73 + strlen ( hrp ) ) ;
pubkey_to_hash160 ( & pubkey , & h160 ) ;
ok = segwit_addr_encode ( out_p2wpkh , hrp , 0 , h160 . u . u8 , sizeof ( h160 . u . u8 ) ) ;
if ( ! ok ) {
command_fail ( cmd , " p2wpkh address encoding failure. " ) ;
return ;
}
// outputs
json_object_start ( response , NULL ) ;
json_add_u64 ( response , " keyidx " , keyidx ) ;
json_add_pubkey ( response , " pubkey " , & pubkey ) ;
json_add_string ( response , " p2sh " , out_p2sh ) ;
json_add_hex ( response , " p2sh_redeemscript " , redeemscript , tal_count ( redeemscript ) ) ;
json_add_string ( response , " bech32 " , out_p2wpkh ) ;
json_add_hex ( response , " bech32_redeemscript " , & h160 . u . u8 , sizeof ( struct ripemd160 ) ) ;
json_object_end ( response ) ;
}
json_array_end ( response ) ;
json_object_end ( response ) ;
command_success ( cmd , response ) ;
}
static const struct json_command listaddrs_command = {
" dev-listaddrs " ,
json_listaddrs ,
" Show addresses list up to derivation {index} (default is the last bip32 index) " , false ,
" Show addresses of your internal wallet. Use `newaddr` to generate a new address. "
} ;
AUTODATA ( json_command , & listaddrs_command ) ;
2018-02-06 14:00:12 +01:00
static void json_listfunds ( struct command * cmd , const char * buffer UNUSED ,
const jsmntok_t * params UNUSED )
2017-09-05 02:11:53 +02:00
{
struct json_result * response = new_json_result ( cmd ) ;
2018-02-09 14:24:33 +01:00
struct peer * p ;
2017-09-05 02:11:53 +02:00
struct utxo * * utxos =
wallet_get_utxos ( cmd , cmd - > ld - > wallet , output_state_available ) ;
json_object_start ( response , NULL ) ;
json_array_start ( response , " outputs " ) ;
2018-02-06 14:14:43 +01:00
for ( size_t i = 0 ; i < tal_count ( utxos ) ; i + + ) {
2017-09-05 02:11:53 +02:00
json_object_start ( response , NULL ) ;
2017-12-18 07:20:24 +01:00
json_add_txid ( response , " txid " , & utxos [ i ] - > txid ) ;
2017-09-05 02:11:53 +02:00
json_add_num ( response , " output " , utxos [ i ] - > outnum ) ;
json_add_u64 ( response , " value " , utxos [ i ] - > amount ) ;
json_object_end ( response ) ;
}
json_array_end ( response ) ;
2018-02-09 14:24:33 +01:00
/* Add funds that are allocated to channels */
json_array_start ( response , " channels " ) ;
list_for_each ( & cmd - > ld - > peers , p , list ) {
2018-02-12 11:12:55 +01:00
struct channel * c ;
list_for_each ( & p - > channels , c , list ) {
json_object_start ( response , NULL ) ;
json_add_pubkey ( response , " peer_id " , & p - > id ) ;
if ( c - > scid )
json_add_short_channel_id ( response ,
" short_channel_id " ,
c - > scid ) ;
/* Poor man's rounding to satoshis to match the unit for outputs */
json_add_u64 ( response , " channel_sat " ,
2018-02-19 02:06:12 +01:00
( c - > our_msatoshi + 500 ) / 1000 ) ;
2018-02-12 11:12:55 +01:00
json_add_u64 ( response , " channel_total_sat " ,
c - > funding_satoshi ) ;
2018-02-19 02:06:12 +01:00
json_add_txid ( response , " funding_txid " ,
& c - > funding_txid ) ;
2018-02-12 11:12:55 +01:00
json_object_end ( response ) ;
}
2018-02-09 14:24:33 +01:00
}
json_array_end ( response ) ;
2017-09-05 02:11:53 +02:00
json_object_end ( response ) ;
2018-02-09 14:24:33 +01:00
2017-09-05 02:11:53 +02:00
command_success ( cmd , response ) ;
}
static const struct json_command listfunds_command = {
2018-01-22 09:55:07 +01:00
" listfunds " ,
json_listfunds ,
2018-02-06 14:13:57 +01:00
" Show available funds from the internal wallet " , false ,
" Returns a list of funds (outputs) that can be used by the internal wallet to open new channels or can be withdrawn, using the `withdraw` command, to another wallet. "
2018-01-22 09:55:07 +01:00
} ;
2017-09-05 02:11:53 +02:00
AUTODATA ( json_command , & listfunds_command ) ;
2018-01-31 23:38:27 +01:00
struct txo_rescan {
struct command * cmd ;
struct utxo * * utxos ;
struct json_result * response ;
} ;
static void process_utxo_result ( struct bitcoind * bitcoind ,
const struct bitcoin_tx_output * txout ,
void * arg )
{
struct txo_rescan * rescan = arg ;
struct json_result * response = rescan - > response ;
struct utxo * u = rescan - > utxos [ 0 ] ;
enum output_status newstate =
txout = = NULL ? output_state_spent : output_state_available ;
json_object_start ( rescan - > response , NULL ) ;
json_add_txid ( response , " txid " , & u - > txid ) ;
json_add_num ( response , " output " , u - > outnum ) ;
json_add_num ( response , " oldstate " , u - > status ) ;
json_add_num ( response , " newstate " , newstate ) ;
json_object_end ( rescan - > response ) ;
wallet_update_output_status ( bitcoind - > ld - > wallet , & u - > txid , u - > outnum ,
u - > status , newstate ) ;
/* Remove the utxo we just resolved */
rescan - > utxos [ 0 ] = rescan - > utxos [ tal_count ( rescan - > utxos ) - 1 ] ;
tal_resize ( & rescan - > utxos , tal_count ( rescan - > utxos ) - 1 ) ;
if ( tal_count ( rescan - > utxos ) = = 0 ) {
/* Complete the response */
json_array_end ( rescan - > response ) ;
json_object_end ( rescan - > response ) ;
command_success ( rescan - > cmd , rescan - > response ) ;
} else {
bitcoind_gettxout (
bitcoind - > ld - > topology - > bitcoind , & rescan - > utxos [ 0 ] - > txid ,
rescan - > utxos [ 0 ] - > outnum , process_utxo_result , rescan ) ;
}
}
2018-02-06 14:00:12 +01:00
static void json_dev_rescan_outputs ( struct command * cmd ,
const char * buffer UNUSED ,
const jsmntok_t * params UNUSED )
2018-01-31 23:38:27 +01:00
{
struct txo_rescan * rescan = tal ( cmd , struct txo_rescan ) ;
rescan - > response = new_json_result ( cmd ) ;
rescan - > cmd = cmd ;
/* Open the result structure so we can incrementally add results */
json_object_start ( rescan - > response , NULL ) ;
json_array_start ( rescan - > response , " outputs " ) ;
2018-02-08 02:06:52 +01:00
rescan - > utxos = wallet_get_utxos ( rescan , cmd - > ld - > wallet , output_state_any ) ;
if ( tal_count ( rescan - > utxos ) = = 0 ) {
json_array_end ( rescan - > response ) ;
json_object_end ( rescan - > response ) ;
command_success ( cmd , rescan - > response ) ;
return ;
}
2018-01-31 23:38:27 +01:00
bitcoind_gettxout ( cmd - > ld - > topology - > bitcoind , & rescan - > utxos [ 0 ] - > txid ,
rescan - > utxos [ 0 ] - > outnum , process_utxo_result ,
rescan ) ;
command_still_pending ( cmd ) ;
}
static const struct json_command dev_rescan_output_command = {
" dev-rescan-outputs " , json_dev_rescan_outputs ,
2018-02-06 14:13:57 +01:00
" Synchronize the state of our funds with bitcoind " , false ,
" For each output stored in the internal wallet ask `bitcoind` whether we are in sync with its state (spent vs. unspent) "
2018-01-31 23:38:27 +01:00
} ;
AUTODATA ( json_command , & dev_rescan_output_command ) ;