2016-08-13 01:29:38 +03:00
|
|
|
package btcwallet
|
|
|
|
|
|
|
|
import (
|
2016-10-15 06:15:50 +03:00
|
|
|
"bytes"
|
2016-08-13 01:29:38 +03:00
|
|
|
"encoding/hex"
|
|
|
|
"fmt"
|
|
|
|
"math"
|
|
|
|
"sync"
|
2016-12-09 05:29:55 +03:00
|
|
|
"time"
|
2016-08-13 01:29:38 +03:00
|
|
|
|
2018-06-05 04:34:16 +03:00
|
|
|
"github.com/btcsuite/btcd/chaincfg"
|
|
|
|
"github.com/btcsuite/btcd/chaincfg/chainhash"
|
|
|
|
"github.com/btcsuite/btcd/txscript"
|
|
|
|
"github.com/btcsuite/btcd/wire"
|
|
|
|
"github.com/btcsuite/btcutil"
|
|
|
|
"github.com/btcsuite/btcwallet/chain"
|
|
|
|
"github.com/btcsuite/btcwallet/waddrmgr"
|
|
|
|
base "github.com/btcsuite/btcwallet/wallet"
|
2019-03-05 16:22:30 +03:00
|
|
|
"github.com/btcsuite/btcwallet/wallet/txauthor"
|
|
|
|
"github.com/btcsuite/btcwallet/wallet/txrules"
|
2018-06-05 04:34:16 +03:00
|
|
|
"github.com/btcsuite/btcwallet/walletdb"
|
2020-05-21 01:33:56 +03:00
|
|
|
"github.com/btcsuite/btcwallet/wtxmgr"
|
2018-07-26 05:33:46 +03:00
|
|
|
"github.com/lightningnetwork/lnd/keychain"
|
|
|
|
"github.com/lightningnetwork/lnd/lnwallet"
|
2019-10-31 05:43:05 +03:00
|
|
|
"github.com/lightningnetwork/lnd/lnwallet/chainfee"
|
2016-08-13 01:29:38 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
defaultAccount = uint32(waddrmgr.DefaultAccountNum)
|
2020-05-05 22:10:06 +03:00
|
|
|
|
|
|
|
// UnconfirmedHeight is the special case end height that is used to
|
|
|
|
// obtain unconfirmed transactions from ListTransactionDetails.
|
|
|
|
UnconfirmedHeight int32 = -1
|
2016-08-13 01:29:38 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2018-02-18 02:03:59 +03:00
|
|
|
// waddrmgrNamespaceKey is the namespace key that the waddrmgr state is
|
|
|
|
// stored within the top-level waleltdb buckets of btcwallet.
|
2017-04-24 05:19:17 +03:00
|
|
|
waddrmgrNamespaceKey = []byte("waddrmgr")
|
2018-02-18 02:03:59 +03:00
|
|
|
|
|
|
|
// lightningAddrSchema is the scope addr schema for all keys that we
|
|
|
|
// derive. We'll treat them all as p2wkh addresses, as atm we must
|
|
|
|
// specify a particular type.
|
|
|
|
lightningAddrSchema = waddrmgr.ScopeAddrSchema{
|
|
|
|
ExternalAddrType: waddrmgr.WitnessPubKey,
|
|
|
|
InternalAddrType: waddrmgr.WitnessPubKey,
|
|
|
|
}
|
2016-08-13 01:29:38 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
// BtcWallet is an implementation of the lnwallet.WalletController interface
|
|
|
|
// backed by an active instance of btcwallet. At the time of the writing of
|
|
|
|
// this documentation, this implementation requires a full btcd node to
|
|
|
|
// operate.
|
|
|
|
type BtcWallet struct {
|
|
|
|
// wallet is an active instance of btcwallet.
|
|
|
|
wallet *base.Wallet
|
|
|
|
|
2017-05-25 03:34:03 +03:00
|
|
|
chain chain.Interface
|
2016-08-13 01:29:38 +03:00
|
|
|
|
2017-04-24 05:19:17 +03:00
|
|
|
db walletdb.DB
|
|
|
|
|
|
|
|
cfg *Config
|
2016-08-13 01:29:38 +03:00
|
|
|
|
|
|
|
netParams *chaincfg.Params
|
|
|
|
|
2018-03-13 03:33:11 +03:00
|
|
|
chainKeyScope waddrmgr.KeyScope
|
2016-08-13 01:29:38 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// A compile time check to ensure that BtcWallet implements the
|
2018-08-24 16:30:23 +03:00
|
|
|
// WalletController and BlockChainIO interfaces.
|
2016-08-13 01:29:38 +03:00
|
|
|
var _ lnwallet.WalletController = (*BtcWallet)(nil)
|
2018-08-24 16:30:23 +03:00
|
|
|
var _ lnwallet.BlockChainIO = (*BtcWallet)(nil)
|
2016-08-13 01:29:38 +03:00
|
|
|
|
|
|
|
// New returns a new fully initialized instance of BtcWallet given a valid
|
2016-10-16 02:02:09 +03:00
|
|
|
// configuration struct.
|
2017-04-24 05:19:17 +03:00
|
|
|
func New(cfg Config) (*BtcWallet, error) {
|
2016-08-13 01:29:38 +03:00
|
|
|
// Ensure the wallet exists or create it when the create flag is set.
|
2017-10-12 12:07:58 +03:00
|
|
|
netDir := NetworkDir(cfg.DataDir, cfg.NetParams)
|
2016-08-13 01:29:38 +03:00
|
|
|
|
2018-03-13 03:33:11 +03:00
|
|
|
// Create the key scope for the coin type being managed by this wallet.
|
|
|
|
chainKeyScope := waddrmgr.KeyScope{
|
|
|
|
Purpose: keychain.BIP0043Purpose,
|
|
|
|
Coin: cfg.CoinType,
|
|
|
|
}
|
|
|
|
|
2018-05-22 10:28:51 +03:00
|
|
|
// Maybe the wallet has already been opened and unlocked by the
|
|
|
|
// WalletUnlocker. So if we get a non-nil value from the config,
|
|
|
|
// we assume everything is in order.
|
|
|
|
var wallet = cfg.Wallet
|
|
|
|
if wallet == nil {
|
|
|
|
// No ready wallet was passed, so try to open an existing one.
|
|
|
|
var pubPass []byte
|
|
|
|
if cfg.PublicPass == nil {
|
|
|
|
pubPass = defaultPubPassphrase
|
|
|
|
} else {
|
|
|
|
pubPass = cfg.PublicPass
|
2016-08-13 01:29:38 +03:00
|
|
|
}
|
2019-10-04 18:10:49 +03:00
|
|
|
loader := base.NewLoader(
|
|
|
|
cfg.NetParams, netDir, cfg.NoFreelistSync,
|
|
|
|
cfg.RecoveryWindow,
|
|
|
|
)
|
2018-05-22 10:28:51 +03:00
|
|
|
walletExists, err := loader.WalletExists()
|
2018-02-18 02:03:59 +03:00
|
|
|
if err != nil {
|
2016-08-13 01:29:38 +03:00
|
|
|
return nil, err
|
|
|
|
}
|
2018-05-22 10:28:51 +03:00
|
|
|
|
|
|
|
if !walletExists {
|
|
|
|
// Wallet has never been created, perform initial
|
|
|
|
// set up.
|
|
|
|
wallet, err = loader.CreateNewWallet(
|
|
|
|
pubPass, cfg.PrivatePass, cfg.HdSeed,
|
|
|
|
cfg.Birthday,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Wallet has been created and been initialized at
|
|
|
|
// this point, open it along with all the required DB
|
|
|
|
// namespaces, and the DB itself.
|
|
|
|
wallet, err = loader.OpenExistingWallet(pubPass, false)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
2016-08-13 01:29:38 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return &BtcWallet{
|
2018-03-13 03:33:11 +03:00
|
|
|
cfg: &cfg,
|
|
|
|
wallet: wallet,
|
|
|
|
db: wallet.Database(),
|
|
|
|
chain: cfg.ChainSource,
|
|
|
|
netParams: cfg.NetParams,
|
|
|
|
chainKeyScope: chainKeyScope,
|
2016-08-13 01:29:38 +03:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2017-11-10 03:30:20 +03:00
|
|
|
// BackEnd returns the underlying ChainService's name as a string.
|
|
|
|
//
|
|
|
|
// This is a part of the WalletController interface.
|
|
|
|
func (b *BtcWallet) BackEnd() string {
|
|
|
|
if b.chain != nil {
|
|
|
|
return b.chain.BackEnd()
|
|
|
|
}
|
|
|
|
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2018-02-18 02:03:59 +03:00
|
|
|
// InternalWallet returns a pointer to the internal base wallet which is the
|
|
|
|
// core of btcwallet.
|
|
|
|
func (b *BtcWallet) InternalWallet() *base.Wallet {
|
|
|
|
return b.wallet
|
|
|
|
}
|
|
|
|
|
2016-08-13 01:29:38 +03:00
|
|
|
// Start initializes the underlying rpc connection, the wallet itself, and
|
|
|
|
// begins syncing to the current available blockchain state.
|
|
|
|
//
|
|
|
|
// This is a part of the WalletController interface.
|
|
|
|
func (b *BtcWallet) Start() error {
|
2018-11-14 07:07:54 +03:00
|
|
|
// We'll start by unlocking the wallet and ensuring that the KeyScope:
|
|
|
|
// (1017, 1) exists within the internal waddrmgr. We'll need this in
|
|
|
|
// order to properly generate the keys required for signing various
|
|
|
|
// contracts.
|
2017-04-24 05:17:54 +03:00
|
|
|
if err := b.wallet.Unlock(b.cfg.PrivatePass, nil); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-03-13 03:33:11 +03:00
|
|
|
_, err := b.wallet.Manager.FetchScopedKeyManager(b.chainKeyScope)
|
2018-02-18 02:03:59 +03:00
|
|
|
if err != nil {
|
|
|
|
// If the scope hasn't yet been created (it wouldn't been
|
|
|
|
// loaded by default if it was), then we'll manually create the
|
|
|
|
// scope for the first time ourselves.
|
|
|
|
err := walletdb.Update(b.db, func(tx walletdb.ReadWriteTx) error {
|
|
|
|
addrmgrNs := tx.ReadWriteBucket(waddrmgrNamespaceKey)
|
|
|
|
|
|
|
|
_, err := b.wallet.Manager.NewScopedKeyManager(
|
2018-03-13 03:33:11 +03:00
|
|
|
addrmgrNs, b.chainKeyScope, lightningAddrSchema,
|
2018-02-18 02:03:59 +03:00
|
|
|
)
|
|
|
|
return err
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-14 07:07:54 +03:00
|
|
|
// Establish an RPC connection in addition to starting the goroutines
|
|
|
|
// in the underlying wallet.
|
|
|
|
if err := b.chain.Start(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Start the underlying btcwallet core.
|
|
|
|
b.wallet.Start()
|
|
|
|
|
|
|
|
// Pass the rpc client into the wallet so it can sync up to the
|
|
|
|
// current main chain.
|
|
|
|
b.wallet.SynchronizeRPC(b.chain)
|
|
|
|
|
2016-08-13 01:29:38 +03:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Stop signals the wallet for shutdown. Shutdown may entail closing
|
|
|
|
// any active sockets, database handles, stopping goroutines, etc.
|
|
|
|
//
|
|
|
|
// This is a part of the WalletController interface.
|
|
|
|
func (b *BtcWallet) Stop() error {
|
|
|
|
b.wallet.Stop()
|
|
|
|
|
|
|
|
b.wallet.WaitForShutdown()
|
|
|
|
|
2017-05-25 03:34:03 +03:00
|
|
|
b.chain.Stop()
|
2016-08-13 01:29:38 +03:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ConfirmedBalance returns the sum of all the wallet's unspent outputs that
|
|
|
|
// have at least confs confirmations. If confs is set to zero, then all unspent
|
|
|
|
// outputs, including those currently in the mempool will be included in the
|
|
|
|
// final sum.
|
|
|
|
//
|
|
|
|
// This is a part of the WalletController interface.
|
2018-02-18 02:36:53 +03:00
|
|
|
func (b *BtcWallet) ConfirmedBalance(confs int32) (btcutil.Amount, error) {
|
2016-08-13 01:29:38 +03:00
|
|
|
var balance btcutil.Amount
|
|
|
|
|
2018-10-28 17:55:18 +03:00
|
|
|
witnessOutputs, err := b.ListUnspentWitness(confs, math.MaxInt32)
|
2018-02-18 02:36:53 +03:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
2016-08-13 01:29:38 +03:00
|
|
|
}
|
|
|
|
|
2018-02-18 02:36:53 +03:00
|
|
|
for _, witnessOutput := range witnessOutputs {
|
|
|
|
balance += witnessOutput.Value
|
|
|
|
}
|
2018-02-18 02:03:59 +03:00
|
|
|
|
2016-08-13 01:29:38 +03:00
|
|
|
return balance, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewAddress returns the next external or internal address for the wallet
|
2017-11-24 07:02:33 +03:00
|
|
|
// dictated by the value of the `change` parameter. If change is true, then an
|
2016-08-13 01:29:38 +03:00
|
|
|
// internal address will be returned, otherwise an external address should be
|
|
|
|
// returned.
|
|
|
|
//
|
|
|
|
// This is a part of the WalletController interface.
|
|
|
|
func (b *BtcWallet) NewAddress(t lnwallet.AddressType, change bool) (btcutil.Address, error) {
|
2018-02-18 02:03:59 +03:00
|
|
|
var keyScope waddrmgr.KeyScope
|
2016-08-13 01:29:38 +03:00
|
|
|
|
|
|
|
switch t {
|
|
|
|
case lnwallet.WitnessPubKey:
|
2018-02-18 02:03:59 +03:00
|
|
|
keyScope = waddrmgr.KeyScopeBIP0084
|
2016-08-13 01:29:38 +03:00
|
|
|
case lnwallet.NestedWitnessPubKey:
|
2018-02-18 02:03:59 +03:00
|
|
|
keyScope = waddrmgr.KeyScopeBIP0049Plus
|
2016-08-13 01:29:38 +03:00
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("unknown address type")
|
|
|
|
}
|
|
|
|
|
|
|
|
if change {
|
2018-02-18 02:03:59 +03:00
|
|
|
return b.wallet.NewChangeAddress(defaultAccount, keyScope)
|
2016-08-13 01:29:38 +03:00
|
|
|
}
|
2017-02-23 22:56:47 +03:00
|
|
|
|
2018-02-18 02:03:59 +03:00
|
|
|
return b.wallet.NewAddress(defaultAccount, keyScope)
|
2019-02-20 06:16:39 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// LastUnusedAddress returns the last *unused* address known by the wallet. An
|
|
|
|
// address is unused if it hasn't received any payments. This can be useful in
|
|
|
|
// UIs in order to continually show the "freshest" address without having to
|
|
|
|
// worry about "address inflation" caused by continual refreshing. Similar to
|
|
|
|
// NewAddress it can derive a specified address type, and also optionally a
|
|
|
|
// change address.
|
|
|
|
func (b *BtcWallet) LastUnusedAddress(addrType lnwallet.AddressType) (
|
|
|
|
btcutil.Address, error) {
|
|
|
|
|
|
|
|
var keyScope waddrmgr.KeyScope
|
|
|
|
|
|
|
|
switch addrType {
|
|
|
|
case lnwallet.WitnessPubKey:
|
|
|
|
keyScope = waddrmgr.KeyScopeBIP0084
|
|
|
|
case lnwallet.NestedWitnessPubKey:
|
|
|
|
keyScope = waddrmgr.KeyScopeBIP0049Plus
|
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("unknown address type")
|
|
|
|
}
|
|
|
|
|
|
|
|
return b.wallet.CurrentAddress(defaultAccount, keyScope)
|
2016-08-13 01:29:38 +03:00
|
|
|
}
|
|
|
|
|
2018-09-28 06:58:46 +03:00
|
|
|
// IsOurAddress checks if the passed address belongs to this wallet
|
2016-08-13 01:29:38 +03:00
|
|
|
//
|
|
|
|
// This is a part of the WalletController interface.
|
2018-09-28 06:58:46 +03:00
|
|
|
func (b *BtcWallet) IsOurAddress(a btcutil.Address) bool {
|
|
|
|
result, err := b.wallet.HaveAddress(a)
|
|
|
|
return result && (err == nil)
|
2016-08-13 01:29:38 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// SendOutputs funds, signs, and broadcasts a Bitcoin transaction paying out to
|
|
|
|
// the specified outputs. In the case the wallet has insufficient funds, or the
|
2018-04-18 05:03:27 +03:00
|
|
|
// outputs are non-standard, a non-nil error will be returned.
|
2016-08-13 01:29:38 +03:00
|
|
|
//
|
2020-06-12 21:22:00 +03:00
|
|
|
// NOTE: This method requires the global coin selection lock to be held.
|
|
|
|
//
|
2016-08-13 01:29:38 +03:00
|
|
|
// This is a part of the WalletController interface.
|
2017-11-23 09:29:32 +03:00
|
|
|
func (b *BtcWallet) SendOutputs(outputs []*wire.TxOut,
|
2020-09-27 18:48:15 +03:00
|
|
|
feeRate chainfee.SatPerKWeight, minconf int32, label string) (*wire.MsgTx, error) {
|
2017-11-23 09:29:32 +03:00
|
|
|
|
2018-07-28 04:20:58 +03:00
|
|
|
// Convert our fee rate from sat/kw to sat/kb since it's required by
|
|
|
|
// SendOutputs.
|
|
|
|
feeSatPerKB := btcutil.Amount(feeRate.FeePerKVByte())
|
2017-11-23 09:29:32 +03:00
|
|
|
|
2019-03-05 16:22:30 +03:00
|
|
|
// Sanity check outputs.
|
|
|
|
if len(outputs) < 1 {
|
|
|
|
return nil, lnwallet.ErrNoOutputs
|
|
|
|
}
|
2020-05-18 15:13:24 +03:00
|
|
|
|
2020-09-27 18:48:15 +03:00
|
|
|
// Sanity check minconf.
|
|
|
|
if minconf < 0 {
|
|
|
|
return nil, lnwallet.ErrInvalidMinconf
|
|
|
|
}
|
|
|
|
|
2020-05-18 15:13:24 +03:00
|
|
|
return b.wallet.SendOutputs(
|
2020-09-27 18:48:15 +03:00
|
|
|
outputs, defaultAccount, minconf, feeSatPerKB, label,
|
2020-05-18 15:13:24 +03:00
|
|
|
)
|
2016-08-13 01:29:38 +03:00
|
|
|
}
|
|
|
|
|
2019-03-05 16:22:30 +03:00
|
|
|
// CreateSimpleTx creates a Bitcoin transaction paying to the specified
|
|
|
|
// outputs. The transaction is not broadcasted to the network, but a new change
|
|
|
|
// address might be created in the wallet database. In the case the wallet has
|
|
|
|
// insufficient funds, or the outputs are non-standard, an error should be
|
|
|
|
// returned. This method also takes the target fee expressed in sat/kw that
|
|
|
|
// should be used when crafting the transaction.
|
|
|
|
//
|
|
|
|
// NOTE: The dryRun argument can be set true to create a tx that doesn't alter
|
|
|
|
// the database. A tx created with this set to true SHOULD NOT be broadcasted.
|
|
|
|
//
|
2020-06-12 21:22:00 +03:00
|
|
|
// NOTE: This method requires the global coin selection lock to be held.
|
|
|
|
//
|
2019-03-05 16:22:30 +03:00
|
|
|
// This is a part of the WalletController interface.
|
|
|
|
func (b *BtcWallet) CreateSimpleTx(outputs []*wire.TxOut,
|
2019-10-31 05:43:05 +03:00
|
|
|
feeRate chainfee.SatPerKWeight, dryRun bool) (*txauthor.AuthoredTx, error) {
|
2019-03-05 16:22:30 +03:00
|
|
|
|
|
|
|
// The fee rate is passed in using units of sat/kw, so we'll convert
|
|
|
|
// this to sat/KB as the CreateSimpleTx method requires this unit.
|
|
|
|
feeSatPerKB := btcutil.Amount(feeRate.FeePerKVByte())
|
|
|
|
|
|
|
|
// Sanity check outputs.
|
2019-03-05 16:22:30 +03:00
|
|
|
if len(outputs) < 1 {
|
|
|
|
return nil, lnwallet.ErrNoOutputs
|
|
|
|
}
|
2019-03-05 16:22:30 +03:00
|
|
|
for _, output := range outputs {
|
2019-06-19 03:04:59 +03:00
|
|
|
// When checking an output for things like dusty-ness, we'll
|
|
|
|
// use the default mempool relay fee rather than the target
|
|
|
|
// effective fee rate to ensure accuracy. Otherwise, we may
|
|
|
|
// mistakenly mark small-ish, but not quite dust output as
|
|
|
|
// dust.
|
|
|
|
err := txrules.CheckOutput(
|
|
|
|
output, txrules.DefaultRelayFeePerKb,
|
|
|
|
)
|
2019-03-05 16:22:30 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return b.wallet.CreateSimpleTx(defaultAccount, outputs, 1, feeSatPerKB, dryRun)
|
|
|
|
}
|
|
|
|
|
2016-08-13 01:29:38 +03:00
|
|
|
// LockOutpoint marks an outpoint as locked meaning it will no longer be deemed
|
|
|
|
// as eligible for coin selection. Locking outputs are utilized in order to
|
|
|
|
// avoid race conditions when selecting inputs for usage when funding a
|
|
|
|
// channel.
|
|
|
|
//
|
2020-06-12 21:22:00 +03:00
|
|
|
// NOTE: This method requires the global coin selection lock to be held.
|
|
|
|
//
|
2016-08-13 01:29:38 +03:00
|
|
|
// This is a part of the WalletController interface.
|
|
|
|
func (b *BtcWallet) LockOutpoint(o wire.OutPoint) {
|
|
|
|
b.wallet.LockOutpoint(o)
|
|
|
|
}
|
|
|
|
|
2018-04-18 05:02:04 +03:00
|
|
|
// UnlockOutpoint unlocks a previously locked output, marking it eligible for
|
2017-12-18 05:40:05 +03:00
|
|
|
// coin selection.
|
2016-08-13 01:29:38 +03:00
|
|
|
//
|
2020-06-12 21:22:00 +03:00
|
|
|
// NOTE: This method requires the global coin selection lock to be held.
|
|
|
|
//
|
2016-08-13 01:29:38 +03:00
|
|
|
// This is a part of the WalletController interface.
|
|
|
|
func (b *BtcWallet) UnlockOutpoint(o wire.OutPoint) {
|
|
|
|
b.wallet.UnlockOutpoint(o)
|
|
|
|
}
|
|
|
|
|
2020-05-21 01:33:56 +03:00
|
|
|
// LeaseOutput locks an output to the given ID, preventing it from being
|
|
|
|
// available for any future coin selection attempts. The absolute time of the
|
|
|
|
// lock's expiration is returned. The expiration of the lock can be extended by
|
|
|
|
// successive invocations of this call. Outputs can be unlocked before their
|
|
|
|
// expiration through `ReleaseOutput`.
|
|
|
|
//
|
|
|
|
// If the output is not known, wtxmgr.ErrUnknownOutput is returned. If the
|
|
|
|
// output has already been locked to a different ID, then
|
|
|
|
// wtxmgr.ErrOutputAlreadyLocked is returned.
|
2020-06-12 21:22:00 +03:00
|
|
|
//
|
|
|
|
// NOTE: This method requires the global coin selection lock to be held.
|
2020-05-21 01:33:56 +03:00
|
|
|
func (b *BtcWallet) LeaseOutput(id wtxmgr.LockID, op wire.OutPoint) (time.Time,
|
|
|
|
error) {
|
2020-06-10 23:15:38 +03:00
|
|
|
|
|
|
|
// Make sure we don't attempt to double lock an output that's been
|
|
|
|
// locked by the in-memory implementation.
|
|
|
|
if b.wallet.LockedOutpoint(op) {
|
|
|
|
return time.Time{}, wtxmgr.ErrOutputAlreadyLocked
|
|
|
|
}
|
|
|
|
|
2020-05-21 01:33:56 +03:00
|
|
|
return b.wallet.LeaseOutput(id, op)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ReleaseOutput unlocks an output, allowing it to be available for coin
|
|
|
|
// selection if it remains unspent. The ID should match the one used to
|
|
|
|
// originally lock the output.
|
2020-06-12 21:22:00 +03:00
|
|
|
//
|
|
|
|
// NOTE: This method requires the global coin selection lock to be held.
|
2020-05-21 01:33:56 +03:00
|
|
|
func (b *BtcWallet) ReleaseOutput(id wtxmgr.LockID, op wire.OutPoint) error {
|
|
|
|
return b.wallet.ReleaseOutput(id, op)
|
|
|
|
}
|
|
|
|
|
2016-08-13 01:29:38 +03:00
|
|
|
// ListUnspentWitness returns a slice of all the unspent outputs the wallet
|
|
|
|
// controls which pay to witness programs either directly or indirectly.
|
|
|
|
//
|
2020-06-12 21:22:00 +03:00
|
|
|
// NOTE: This method requires the global coin selection lock to be held.
|
|
|
|
//
|
2016-08-13 01:29:38 +03:00
|
|
|
// This is a part of the WalletController interface.
|
2018-10-28 17:55:18 +03:00
|
|
|
func (b *BtcWallet) ListUnspentWitness(minConfs, maxConfs int32) (
|
|
|
|
[]*lnwallet.Utxo, error) {
|
2016-08-13 01:29:38 +03:00
|
|
|
// First, grab all the unfiltered currently unspent outputs.
|
|
|
|
unspentOutputs, err := b.wallet.ListUnspent(minConfs, maxConfs, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Next, we'll run through all the regular outputs, only saving those
|
|
|
|
// which are p2wkh outputs or a p2wsh output nested within a p2sh output.
|
|
|
|
witnessOutputs := make([]*lnwallet.Utxo, 0, len(unspentOutputs))
|
|
|
|
for _, output := range unspentOutputs {
|
|
|
|
pkScript, err := hex.DecodeString(output.ScriptPubKey)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-06-10 15:22:17 +03:00
|
|
|
addressType := lnwallet.UnknownAddressType
|
2017-10-03 04:52:45 +03:00
|
|
|
if txscript.IsPayToWitnessPubKeyHash(pkScript) {
|
|
|
|
addressType = lnwallet.WitnessPubKey
|
|
|
|
} else if txscript.IsPayToScriptHash(pkScript) {
|
|
|
|
// TODO(roasbeef): This assumes all p2sh outputs returned by the
|
|
|
|
// wallet are nested p2pkh. We can't check the redeem script because
|
|
|
|
// the btcwallet service does not include it.
|
|
|
|
addressType = lnwallet.NestedWitnessPubKey
|
|
|
|
}
|
|
|
|
|
|
|
|
if addressType == lnwallet.WitnessPubKey ||
|
|
|
|
addressType == lnwallet.NestedWitnessPubKey {
|
|
|
|
|
2017-01-06 00:56:27 +03:00
|
|
|
txid, err := chainhash.NewHashFromStr(output.TxID)
|
2016-08-13 01:29:38 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-03-26 05:15:05 +03:00
|
|
|
// We'll ensure we properly convert the amount given in
|
|
|
|
// BTC to satoshis.
|
|
|
|
amt, err := btcutil.NewAmount(output.Amount)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2016-08-13 01:29:38 +03:00
|
|
|
utxo := &lnwallet.Utxo{
|
2017-10-03 04:52:45 +03:00
|
|
|
AddressType: addressType,
|
2018-03-26 05:15:05 +03:00
|
|
|
Value: amt,
|
2017-10-03 04:52:45 +03:00
|
|
|
PkScript: pkScript,
|
2016-08-13 01:29:38 +03:00
|
|
|
OutPoint: wire.OutPoint{
|
|
|
|
Hash: *txid,
|
|
|
|
Index: output.Vout,
|
|
|
|
},
|
2018-09-27 16:49:44 +03:00
|
|
|
Confirmations: output.Confirmations,
|
2016-08-13 01:29:38 +03:00
|
|
|
}
|
|
|
|
witnessOutputs = append(witnessOutputs, utxo)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return witnessOutputs, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// PublishTransaction performs cursory validation (dust checks, etc), then
|
2018-01-16 01:27:27 +03:00
|
|
|
// finally broadcasts the passed transaction to the Bitcoin network. If
|
2019-02-21 02:06:21 +03:00
|
|
|
// publishing the transaction fails, an error describing the reason is returned
|
|
|
|
// (currently ErrDoubleSpend). If the transaction is already published to the
|
|
|
|
// network (either in the mempool or chain) no error will be returned.
|
2020-05-18 15:13:23 +03:00
|
|
|
func (b *BtcWallet) PublishTransaction(tx *wire.MsgTx, label string) error {
|
|
|
|
if err := b.wallet.PublishTransaction(tx, label); err != nil {
|
2018-01-16 01:27:27 +03:00
|
|
|
|
2019-09-19 15:59:07 +03:00
|
|
|
// If we failed to publish the transaction, check whether we
|
|
|
|
// got an error of known type.
|
|
|
|
switch err.(type) {
|
2018-01-16 01:27:27 +03:00
|
|
|
|
2019-09-19 15:59:07 +03:00
|
|
|
// If the wallet reports a double spend, convert it to our
|
|
|
|
// internal ErrDoubleSpend and return.
|
|
|
|
case *base.ErrDoubleSpend:
|
|
|
|
return lnwallet.ErrDoubleSpend
|
|
|
|
|
|
|
|
// If the wallet reports a replacement error, return
|
|
|
|
// ErrDoubleSpend, as we currently are never attempting to
|
|
|
|
// replace transactions.
|
|
|
|
case *base.ErrReplacement:
|
|
|
|
return lnwallet.ErrDoubleSpend
|
2018-01-16 01:27:27 +03:00
|
|
|
|
|
|
|
default:
|
2019-09-19 15:59:07 +03:00
|
|
|
return err
|
2018-01-16 01:27:27 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
2016-08-13 01:29:38 +03:00
|
|
|
}
|
2016-10-15 06:15:50 +03:00
|
|
|
|
2020-05-25 09:38:05 +03:00
|
|
|
// LabelTransaction adds a label to a transaction. If the tx already
|
|
|
|
// has a label, this call will fail unless the overwrite parameter
|
|
|
|
// is set. Labels must not be empty, and they are limited to 500 chars.
|
|
|
|
//
|
|
|
|
// Note: it is part of the WalletController interface.
|
|
|
|
func (b *BtcWallet) LabelTransaction(hash chainhash.Hash, label string,
|
|
|
|
overwrite bool) error {
|
|
|
|
|
|
|
|
return b.wallet.LabelTransaction(hash, label, overwrite)
|
|
|
|
}
|
|
|
|
|
2016-10-15 06:15:50 +03:00
|
|
|
// extractBalanceDelta extracts the net balance delta from the PoV of the
|
|
|
|
// wallet given a TransactionSummary.
|
2017-12-06 20:19:37 +03:00
|
|
|
func extractBalanceDelta(
|
|
|
|
txSummary base.TransactionSummary,
|
|
|
|
tx *wire.MsgTx,
|
|
|
|
) (btcutil.Amount, error) {
|
2016-10-15 06:15:50 +03:00
|
|
|
// For each input we debit the wallet's outflow for this transaction,
|
|
|
|
// and for each output we credit the wallet's inflow for this
|
|
|
|
// transaction.
|
|
|
|
var balanceDelta btcutil.Amount
|
|
|
|
for _, input := range txSummary.MyInputs {
|
|
|
|
balanceDelta -= input.PreviousAmount
|
|
|
|
}
|
|
|
|
for _, output := range txSummary.MyOutputs {
|
|
|
|
balanceDelta += btcutil.Amount(tx.TxOut[output.Index].Value)
|
|
|
|
}
|
|
|
|
|
|
|
|
return balanceDelta, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// minedTransactionsToDetails is a helper function which converts a summary
|
|
|
|
// information about mined transactions to a TransactionDetail.
|
2017-12-06 20:19:37 +03:00
|
|
|
func minedTransactionsToDetails(
|
|
|
|
currentHeight int32,
|
|
|
|
block base.Block,
|
|
|
|
chainParams *chaincfg.Params,
|
|
|
|
) ([]*lnwallet.TransactionDetail, error) {
|
2016-10-15 06:15:50 +03:00
|
|
|
|
|
|
|
details := make([]*lnwallet.TransactionDetail, 0, len(block.Transactions))
|
|
|
|
for _, tx := range block.Transactions {
|
2017-12-06 20:19:37 +03:00
|
|
|
wireTx := &wire.MsgTx{}
|
|
|
|
txReader := bytes.NewReader(tx.Transaction)
|
|
|
|
|
|
|
|
if err := wireTx.Deserialize(txReader); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var destAddresses []btcutil.Address
|
|
|
|
for _, txOut := range wireTx.TxOut {
|
2019-06-07 17:36:20 +03:00
|
|
|
_, outAddresses, _, err := txscript.ExtractPkScriptAddrs(
|
|
|
|
txOut.PkScript, chainParams,
|
|
|
|
)
|
2017-12-06 20:19:37 +03:00
|
|
|
if err != nil {
|
2020-08-25 21:21:29 +03:00
|
|
|
// Skip any unsupported addresses to prevent
|
|
|
|
// other transactions from not being returned.
|
|
|
|
continue
|
2017-12-06 20:19:37 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
destAddresses = append(destAddresses, outAddresses...)
|
|
|
|
}
|
|
|
|
|
2016-10-15 06:15:50 +03:00
|
|
|
txDetail := &lnwallet.TransactionDetail{
|
|
|
|
Hash: *tx.Hash,
|
|
|
|
NumConfirmations: currentHeight - block.Height + 1,
|
|
|
|
BlockHash: block.Hash,
|
|
|
|
BlockHeight: block.Height,
|
|
|
|
Timestamp: block.Timestamp,
|
|
|
|
TotalFees: int64(tx.Fee),
|
2017-12-06 20:19:37 +03:00
|
|
|
DestAddresses: destAddresses,
|
2019-06-07 17:36:20 +03:00
|
|
|
RawTx: tx.Transaction,
|
2020-05-18 15:13:24 +03:00
|
|
|
Label: tx.Label,
|
2016-10-15 06:15:50 +03:00
|
|
|
}
|
|
|
|
|
2017-12-06 20:19:37 +03:00
|
|
|
balanceDelta, err := extractBalanceDelta(tx, wireTx)
|
2016-10-15 06:15:50 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
txDetail.Value = balanceDelta
|
|
|
|
|
|
|
|
details = append(details, txDetail)
|
|
|
|
}
|
|
|
|
|
|
|
|
return details, nil
|
|
|
|
}
|
|
|
|
|
2017-12-06 20:19:37 +03:00
|
|
|
// unminedTransactionsToDetail is a helper function which converts a summary
|
2018-04-18 05:02:04 +03:00
|
|
|
// for an unconfirmed transaction to a transaction detail.
|
2017-12-06 20:19:37 +03:00
|
|
|
func unminedTransactionsToDetail(
|
|
|
|
summary base.TransactionSummary,
|
2019-07-10 09:38:27 +03:00
|
|
|
chainParams *chaincfg.Params,
|
2017-12-06 20:19:37 +03:00
|
|
|
) (*lnwallet.TransactionDetail, error) {
|
2019-06-13 23:54:33 +03:00
|
|
|
|
2017-12-06 20:19:37 +03:00
|
|
|
wireTx := &wire.MsgTx{}
|
|
|
|
txReader := bytes.NewReader(summary.Transaction)
|
|
|
|
|
|
|
|
if err := wireTx.Deserialize(txReader); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-07-10 09:38:27 +03:00
|
|
|
var destAddresses []btcutil.Address
|
|
|
|
for _, txOut := range wireTx.TxOut {
|
|
|
|
_, outAddresses, _, err :=
|
|
|
|
txscript.ExtractPkScriptAddrs(txOut.PkScript, chainParams)
|
|
|
|
if err != nil {
|
2020-08-25 21:21:29 +03:00
|
|
|
// Skip any unsupported addresses to prevent other
|
|
|
|
// transactions from not being returned.
|
|
|
|
continue
|
2019-07-10 09:38:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
destAddresses = append(destAddresses, outAddresses...)
|
|
|
|
}
|
|
|
|
|
2016-10-15 06:15:50 +03:00
|
|
|
txDetail := &lnwallet.TransactionDetail{
|
2019-07-10 09:38:27 +03:00
|
|
|
Hash: *summary.Hash,
|
|
|
|
TotalFees: int64(summary.Fee),
|
|
|
|
Timestamp: summary.Timestamp,
|
|
|
|
DestAddresses: destAddresses,
|
|
|
|
RawTx: summary.Transaction,
|
2020-05-18 15:13:24 +03:00
|
|
|
Label: summary.Label,
|
2016-10-15 06:15:50 +03:00
|
|
|
}
|
|
|
|
|
2017-12-06 20:19:37 +03:00
|
|
|
balanceDelta, err := extractBalanceDelta(summary, wireTx)
|
2016-10-15 06:15:50 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
txDetail.Value = balanceDelta
|
|
|
|
|
|
|
|
return txDetail, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ListTransactionDetails returns a list of all transactions which are
|
2020-05-05 22:10:06 +03:00
|
|
|
// relevant to the wallet. It takes inclusive start and end height to allow
|
|
|
|
// paginated queries. Unconfirmed transactions can be included in the query
|
|
|
|
// by providing endHeight = UnconfirmedHeight (= -1).
|
2016-10-15 06:15:50 +03:00
|
|
|
//
|
|
|
|
// This is a part of the WalletController interface.
|
2020-05-05 22:10:06 +03:00
|
|
|
func (b *BtcWallet) ListTransactionDetails(startHeight,
|
|
|
|
endHeight int32) ([]*lnwallet.TransactionDetail, error) {
|
|
|
|
|
2016-10-15 06:15:50 +03:00
|
|
|
// Grab the best block the wallet knows of, we'll use this to calculate
|
|
|
|
// # of confirmations shortly below.
|
|
|
|
bestBlock := b.wallet.Manager.SyncedTo()
|
|
|
|
currentHeight := bestBlock.Height
|
|
|
|
|
2020-05-05 22:10:06 +03:00
|
|
|
// We'll attempt to find all transactions from start to end height.
|
|
|
|
start := base.NewBlockIdentifierFromHeight(startHeight)
|
|
|
|
stop := base.NewBlockIdentifierFromHeight(endHeight)
|
2016-10-15 06:15:50 +03:00
|
|
|
txns, err := b.wallet.GetTransactions(start, stop, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
txDetails := make([]*lnwallet.TransactionDetail, 0,
|
|
|
|
len(txns.MinedTransactions)+len(txns.UnminedTransactions))
|
|
|
|
|
2017-06-08 03:01:17 +03:00
|
|
|
// For both confirmed and unconfirmed transactions, create a
|
2016-10-15 06:15:50 +03:00
|
|
|
// TransactionDetail which re-packages the data returned by the base
|
|
|
|
// wallet.
|
|
|
|
for _, blockPackage := range txns.MinedTransactions {
|
2019-06-07 17:36:20 +03:00
|
|
|
details, err := minedTransactionsToDetails(
|
|
|
|
currentHeight, blockPackage, b.netParams,
|
|
|
|
)
|
2016-10-15 06:15:50 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
txDetails = append(txDetails, details...)
|
|
|
|
}
|
|
|
|
for _, tx := range txns.UnminedTransactions {
|
2019-07-10 09:38:27 +03:00
|
|
|
detail, err := unminedTransactionsToDetail(tx, b.netParams)
|
2016-10-15 06:15:50 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
txDetails = append(txDetails, detail)
|
|
|
|
}
|
|
|
|
|
|
|
|
return txDetails, nil
|
|
|
|
}
|
2016-10-16 00:11:18 +03:00
|
|
|
|
|
|
|
// txSubscriptionClient encapsulates the transaction notification client from
|
|
|
|
// the base wallet. Notifications received from the client will be proxied over
|
|
|
|
// two distinct channels.
|
|
|
|
type txSubscriptionClient struct {
|
|
|
|
txClient base.TransactionNotificationsClient
|
|
|
|
|
|
|
|
confirmed chan *lnwallet.TransactionDetail
|
|
|
|
unconfirmed chan *lnwallet.TransactionDetail
|
|
|
|
|
|
|
|
w *base.Wallet
|
|
|
|
|
|
|
|
wg sync.WaitGroup
|
|
|
|
quit chan struct{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ConfirmedTransactions returns a channel which will be sent on as new
|
|
|
|
// relevant transactions are confirmed.
|
|
|
|
//
|
|
|
|
// This is part of the TransactionSubscription interface.
|
|
|
|
func (t *txSubscriptionClient) ConfirmedTransactions() chan *lnwallet.TransactionDetail {
|
|
|
|
return t.confirmed
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnconfirmedTransactions returns a channel which will be sent on as
|
|
|
|
// new relevant transactions are seen within the network.
|
|
|
|
//
|
|
|
|
// This is part of the TransactionSubscription interface.
|
|
|
|
func (t *txSubscriptionClient) UnconfirmedTransactions() chan *lnwallet.TransactionDetail {
|
|
|
|
return t.unconfirmed
|
|
|
|
}
|
|
|
|
|
|
|
|
// Cancel finalizes the subscription, cleaning up any resources allocated.
|
|
|
|
//
|
|
|
|
// This is part of the TransactionSubscription interface.
|
|
|
|
func (t *txSubscriptionClient) Cancel() {
|
|
|
|
close(t.quit)
|
|
|
|
t.wg.Wait()
|
|
|
|
|
|
|
|
t.txClient.Done()
|
|
|
|
}
|
|
|
|
|
|
|
|
// notificationProxier proxies the notifications received by the underlying
|
|
|
|
// wallet's notification client to a higher-level TransactionSubscription
|
|
|
|
// client.
|
|
|
|
func (t *txSubscriptionClient) notificationProxier() {
|
|
|
|
out:
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case txNtfn := <-t.txClient.C:
|
|
|
|
// TODO(roasbeef): handle detached blocks
|
|
|
|
currentHeight := t.w.Manager.SyncedTo().Height
|
|
|
|
|
|
|
|
// Launch a goroutine to re-package and send
|
|
|
|
// notifications for any newly confirmed transactions.
|
|
|
|
go func() {
|
|
|
|
for _, block := range txNtfn.AttachedBlocks {
|
2017-12-06 20:19:37 +03:00
|
|
|
details, err := minedTransactionsToDetails(currentHeight, block, t.w.ChainParams())
|
2016-10-16 00:11:18 +03:00
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, d := range details {
|
|
|
|
select {
|
|
|
|
case t.confirmed <- d:
|
|
|
|
case <-t.quit:
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Launch a goroutine to re-package and send
|
|
|
|
// notifications for any newly unconfirmed transactions.
|
|
|
|
go func() {
|
|
|
|
for _, tx := range txNtfn.UnminedTransactions {
|
2019-07-10 09:38:27 +03:00
|
|
|
detail, err := unminedTransactionsToDetail(
|
|
|
|
tx, t.w.ChainParams(),
|
|
|
|
)
|
2016-10-16 00:11:18 +03:00
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
select {
|
|
|
|
case t.unconfirmed <- detail:
|
|
|
|
case <-t.quit:
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
case <-t.quit:
|
|
|
|
break out
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
t.wg.Done()
|
|
|
|
}
|
|
|
|
|
|
|
|
// SubscribeTransactions returns a TransactionSubscription client which
|
|
|
|
// is capable of receiving async notifications as new transactions
|
|
|
|
// related to the wallet are seen within the network, or found in
|
|
|
|
// blocks.
|
|
|
|
//
|
|
|
|
// This is a part of the WalletController interface.
|
|
|
|
func (b *BtcWallet) SubscribeTransactions() (lnwallet.TransactionSubscription, error) {
|
|
|
|
walletClient := b.wallet.NtfnServer.TransactionNotifications()
|
|
|
|
|
|
|
|
txClient := &txSubscriptionClient{
|
|
|
|
txClient: walletClient,
|
|
|
|
confirmed: make(chan *lnwallet.TransactionDetail),
|
|
|
|
unconfirmed: make(chan *lnwallet.TransactionDetail),
|
|
|
|
w: b.wallet,
|
|
|
|
quit: make(chan struct{}),
|
|
|
|
}
|
|
|
|
txClient.wg.Add(1)
|
|
|
|
go txClient.notificationProxier()
|
|
|
|
|
|
|
|
return txClient, nil
|
|
|
|
}
|
2016-12-09 05:29:55 +03:00
|
|
|
|
2019-06-13 23:54:33 +03:00
|
|
|
// IsSynced returns a boolean indicating if from the PoV of the wallet, it has
|
|
|
|
// fully synced to the current best block in the main chain.
|
2016-12-09 05:29:55 +03:00
|
|
|
//
|
|
|
|
// This is a part of the WalletController interface.
|
2017-12-10 10:42:46 +03:00
|
|
|
func (b *BtcWallet) IsSynced() (bool, int64, error) {
|
2018-03-06 21:17:09 +03:00
|
|
|
// Grab the best chain state the wallet is currently aware of.
|
2016-12-09 05:29:55 +03:00
|
|
|
syncState := b.wallet.Manager.SyncedTo()
|
|
|
|
|
2018-03-06 21:17:09 +03:00
|
|
|
// We'll also extract the current best wallet timestamp so the caller
|
|
|
|
// can get an idea of where we are in the sync timeline.
|
|
|
|
bestTimestamp := syncState.Timestamp.Unix()
|
2017-05-25 03:34:31 +03:00
|
|
|
|
|
|
|
// Next, query the chain backend to grab the info about the tip of the
|
|
|
|
// main chain.
|
2018-03-06 21:17:09 +03:00
|
|
|
bestHash, bestHeight, err := b.cfg.ChainSource.GetBestBlock()
|
2017-11-10 03:30:20 +03:00
|
|
|
if err != nil {
|
2017-12-10 10:42:46 +03:00
|
|
|
return false, 0, err
|
2016-12-09 05:29:55 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// If the wallet hasn't yet fully synced to the node's best chain tip,
|
|
|
|
// then we're not yet fully synced.
|
2018-11-21 05:14:14 +03:00
|
|
|
if syncState.Height < bestHeight || !b.wallet.ChainSynced() {
|
2018-03-06 21:17:09 +03:00
|
|
|
return false, bestTimestamp, nil
|
2016-12-09 05:29:55 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// If the wallet is on par with the current best chain tip, then we
|
2017-05-25 03:34:31 +03:00
|
|
|
// still may not yet be synced as the chain backend may still be
|
|
|
|
// catching up to the main chain. So we'll grab the block header in
|
|
|
|
// order to make a guess based on the current time stamp.
|
2017-11-10 03:30:20 +03:00
|
|
|
blockHeader, err := b.cfg.ChainSource.GetBlockHeader(bestHash)
|
|
|
|
if err != nil {
|
2017-12-10 10:42:46 +03:00
|
|
|
return false, 0, err
|
2016-12-09 05:29:55 +03:00
|
|
|
}
|
|
|
|
|
2018-07-26 05:33:46 +03:00
|
|
|
// If the timestamp on the best header is more than 2 hours in the
|
2016-12-09 05:29:55 +03:00
|
|
|
// past, then we're not yet synced.
|
|
|
|
minus24Hours := time.Now().Add(-2 * time.Hour)
|
2018-07-26 05:33:46 +03:00
|
|
|
if blockHeader.Timestamp.Before(minus24Hours) {
|
|
|
|
return false, bestTimestamp, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return true, bestTimestamp, nil
|
2016-12-09 05:29:55 +03:00
|
|
|
}
|
2020-06-09 09:47:17 +03:00
|
|
|
|
|
|
|
// GetRecoveryInfo returns a boolean indicating whether the wallet is started
|
|
|
|
// in recovery mode. It also returns a float64, ranging from 0 to 1,
|
|
|
|
// representing the recovery progress made so far.
|
|
|
|
//
|
|
|
|
// This is a part of the WalletController interface.
|
|
|
|
func (b *BtcWallet) GetRecoveryInfo() (bool, float64, error) {
|
|
|
|
isRecoveryMode := true
|
|
|
|
progress := float64(0)
|
|
|
|
|
|
|
|
// A zero value in RecoveryWindow indicates there is no trigger of
|
|
|
|
// recovery mode.
|
|
|
|
if b.cfg.RecoveryWindow == 0 {
|
|
|
|
isRecoveryMode = false
|
|
|
|
return isRecoveryMode, progress, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Query the wallet's birthday block height from db.
|
|
|
|
var birthdayBlock waddrmgr.BlockStamp
|
|
|
|
err := walletdb.View(b.db, func(tx walletdb.ReadTx) error {
|
|
|
|
var err error
|
|
|
|
addrmgrNs := tx.ReadBucket(waddrmgrNamespaceKey)
|
|
|
|
birthdayBlock, _, err = b.wallet.Manager.BirthdayBlock(addrmgrNs)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
// The wallet won't start until the backend is synced, thus the birthday
|
|
|
|
// block won't be set and this particular error will be returned. We'll
|
|
|
|
// catch this error and return a progress of 0 instead.
|
|
|
|
if waddrmgr.IsError(err, waddrmgr.ErrBirthdayBlockNotSet) {
|
|
|
|
return isRecoveryMode, progress, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return isRecoveryMode, progress, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Grab the best chain state the wallet is currently aware of.
|
|
|
|
syncState := b.wallet.Manager.SyncedTo()
|
|
|
|
|
|
|
|
// Next, query the chain backend to grab the info about the tip of the
|
|
|
|
// main chain.
|
|
|
|
//
|
|
|
|
// NOTE: The actual recovery process is handled by the btcsuite/btcwallet.
|
|
|
|
// The process purposefully doesn't update the best height. It might create
|
|
|
|
// a small difference between the height queried here and the height used
|
|
|
|
// in the recovery process, ie, the bestHeight used here might be greater,
|
|
|
|
// showing the recovery being unfinished while it's actually done. However,
|
|
|
|
// during a wallet rescan after the recovery, the wallet's synced height
|
|
|
|
// will catch up and this won't be an issue.
|
|
|
|
_, bestHeight, err := b.cfg.ChainSource.GetBestBlock()
|
|
|
|
if err != nil {
|
|
|
|
return isRecoveryMode, progress, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// The birthday block height might be greater than the current synced height
|
|
|
|
// in a newly restored wallet, and might be greater than the chain tip if a
|
|
|
|
// rollback happens. In that case, we will return zero progress here.
|
|
|
|
if syncState.Height < birthdayBlock.Height ||
|
|
|
|
bestHeight < birthdayBlock.Height {
|
|
|
|
return isRecoveryMode, progress, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// progress is the ratio of the [number of blocks processed] over the [total
|
|
|
|
// number of blocks] needed in a recovery mode, ranging from 0 to 1, in
|
|
|
|
// which,
|
|
|
|
// - total number of blocks is the current chain's best height minus the
|
|
|
|
// wallet's birthday height plus 1.
|
|
|
|
// - number of blocks processed is the wallet's synced height minus its
|
|
|
|
// birthday height plus 1.
|
|
|
|
// - If the wallet is born very recently, the bestHeight can be equal to
|
|
|
|
// the birthdayBlock.Height, and it will recovery instantly.
|
|
|
|
progress = float64(syncState.Height-birthdayBlock.Height+1) /
|
|
|
|
float64(bestHeight-birthdayBlock.Height+1)
|
|
|
|
|
|
|
|
return isRecoveryMode, progress, nil
|
|
|
|
}
|