2015-12-20 07:00:50 +01:00
|
|
|
package lnwallet
|
2015-10-27 23:50:30 +01:00
|
|
|
|
|
|
|
import (
|
2017-07-30 04:20:08 +02:00
|
|
|
"bytes"
|
2017-03-16 02:56:25 +01:00
|
|
|
"crypto/sha256"
|
2018-08-10 04:16:14 +02:00
|
|
|
"errors"
|
2015-11-05 21:36:19 +01:00
|
|
|
"fmt"
|
2021-01-11 11:03:00 +01:00
|
|
|
"math"
|
2016-10-26 23:56:48 +02:00
|
|
|
"net"
|
2015-10-27 23:50:30 +01:00
|
|
|
"sync"
|
|
|
|
"sync/atomic"
|
|
|
|
|
2018-07-18 04:20:05 +02:00
|
|
|
"github.com/btcsuite/btcd/blockchain"
|
2022-02-23 14:48:00 +01:00
|
|
|
"github.com/btcsuite/btcd/btcec/v2"
|
2023-01-20 02:15:12 +01:00
|
|
|
"github.com/btcsuite/btcd/btcec/v2/schnorr/musig2"
|
2022-02-23 14:48:00 +01:00
|
|
|
"github.com/btcsuite/btcd/btcutil"
|
|
|
|
"github.com/btcsuite/btcd/btcutil/psbt"
|
|
|
|
"github.com/btcsuite/btcd/btcutil/txsort"
|
2021-06-09 22:23:35 +02:00
|
|
|
"github.com/btcsuite/btcd/chaincfg"
|
2018-07-18 04:20:05 +02:00
|
|
|
"github.com/btcsuite/btcd/chaincfg/chainhash"
|
2018-08-10 04:16:14 +02:00
|
|
|
"github.com/btcsuite/btcd/txscript"
|
|
|
|
"github.com/btcsuite/btcd/wire"
|
2024-02-06 12:25:47 +01:00
|
|
|
"github.com/btcsuite/btcwallet/wallet"
|
2016-06-21 19:37:55 +02:00
|
|
|
"github.com/davecgh/go-spew/spew"
|
2016-01-16 19:45:54 +01:00
|
|
|
"github.com/lightningnetwork/lnd/channeldb"
|
2024-03-13 15:46:33 +01:00
|
|
|
"github.com/lightningnetwork/lnd/fn"
|
2019-01-16 15:47:43 +01:00
|
|
|
"github.com/lightningnetwork/lnd/input"
|
2018-02-18 00:20:41 +01:00
|
|
|
"github.com/lightningnetwork/lnd/keychain"
|
2024-07-31 01:44:18 +02:00
|
|
|
"github.com/lightningnetwork/lnd/lntypes"
|
2019-10-31 03:43:05 +01:00
|
|
|
"github.com/lightningnetwork/lnd/lnwallet/chainfee"
|
2019-11-01 05:14:02 +01:00
|
|
|
"github.com/lightningnetwork/lnd/lnwallet/chanfunding"
|
2019-10-01 04:59:10 +02:00
|
|
|
"github.com/lightningnetwork/lnd/lnwallet/chanvalidate"
|
2017-08-22 07:49:56 +02:00
|
|
|
"github.com/lightningnetwork/lnd/lnwire"
|
2018-07-18 04:20:05 +02:00
|
|
|
"github.com/lightningnetwork/lnd/shachain"
|
2024-04-05 02:25:09 +02:00
|
|
|
"github.com/lightningnetwork/lnd/tlv"
|
2015-10-27 23:50:30 +01:00
|
|
|
)
|
|
|
|
|
2015-11-13 04:34:35 +01:00
|
|
|
const (
|
2016-05-04 04:49:58 +02:00
|
|
|
// The size of the buffered queue of requests to the wallet from the
|
2015-12-28 21:14:00 +01:00
|
|
|
// outside word.
|
2015-11-13 04:34:35 +01:00
|
|
|
msgBufferSize = 100
|
2021-01-11 11:03:00 +01:00
|
|
|
|
2022-06-17 17:24:29 +02:00
|
|
|
// AnchorChanReservedValue is the amount we'll keep around in the
|
2021-01-11 11:03:00 +01:00
|
|
|
// wallet in case we have to fee bump anchor channels on force close.
|
|
|
|
// TODO(halseth): update constant to target a specific commit size at
|
|
|
|
// set fee rate.
|
2022-06-17 17:24:29 +02:00
|
|
|
AnchorChanReservedValue = btcutil.Amount(10_000)
|
2021-05-06 13:24:43 +02:00
|
|
|
|
2022-06-17 17:24:29 +02:00
|
|
|
// MaxAnchorChanReservedValue is the maximum value we'll reserve for
|
2021-05-06 13:24:43 +02:00
|
|
|
// anchor channel fee bumping. We cap it at 10 times the per-channel
|
|
|
|
// amount such that nodes with a high number of channels don't have to
|
|
|
|
// keep around a very large amount for the unlikely scenario that they
|
|
|
|
// all close at the same time.
|
2022-06-17 17:24:29 +02:00
|
|
|
MaxAnchorChanReservedValue = 10 * AnchorChanReservedValue
|
2015-11-05 21:36:19 +01:00
|
|
|
)
|
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
var (
|
|
|
|
// ErrPsbtFundingRequired is the error that is returned during the
|
|
|
|
// contribution handling process if the process should be paused for
|
|
|
|
// the construction of a PSBT outside of lnd's wallet.
|
|
|
|
ErrPsbtFundingRequired = errors.New("PSBT funding required")
|
2021-01-11 11:03:00 +01:00
|
|
|
|
|
|
|
// ErrReservedValueInvalidated is returned if we try to publish a
|
|
|
|
// transaction that would take the walletbalance below what we require
|
|
|
|
// to keep around to fee bump our open anchor channels.
|
|
|
|
ErrReservedValueInvalidated = errors.New("reserved wallet balance " +
|
2021-07-27 10:48:27 +02:00
|
|
|
"invalidated: transaction would leave insufficient funds for " +
|
|
|
|
"fee bumping anchor channel closings (see debug log for details)")
|
2023-10-09 10:58:18 +02:00
|
|
|
|
|
|
|
// ErrEmptyPendingChanID is returned when an empty value is used for
|
|
|
|
// the pending channel ID.
|
|
|
|
ErrEmptyPendingChanID = errors.New("pending channel ID is empty")
|
|
|
|
|
|
|
|
// ErrDuplicatePendingChanID is returned when an existing pending
|
|
|
|
// channel ID is registered again.
|
|
|
|
ErrDuplicatePendingChanID = errors.New("duplicate pending channel ID")
|
2020-03-31 09:13:16 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// PsbtFundingRequired is a type that implements the error interface and
|
|
|
|
// contains the information needed to construct a PSBT.
|
|
|
|
type PsbtFundingRequired struct {
|
|
|
|
// Intent is the pending PSBT funding intent that needs to be funded
|
|
|
|
// if the wrapping error is returned.
|
|
|
|
Intent *chanfunding.PsbtIntent
|
|
|
|
}
|
|
|
|
|
|
|
|
// Error returns the underlying error.
|
|
|
|
//
|
|
|
|
// NOTE: This method is part of the error interface.
|
|
|
|
func (p *PsbtFundingRequired) Error() string {
|
|
|
|
return ErrPsbtFundingRequired.Error()
|
|
|
|
}
|
|
|
|
|
2024-04-05 02:25:09 +02:00
|
|
|
// AuxFundingDesc stores a series of attributes that may be used to modify the
|
|
|
|
// way the channel funding occurs. This struct contains information that can
|
|
|
|
// only be derived once both sides have received and sent their contributions
|
|
|
|
// to the channel (keys, etc.).
|
|
|
|
type AuxFundingDesc struct {
|
|
|
|
// CustomFundingBlob is a custom blob that'll be stored in the database
|
|
|
|
// within the OpenChannel struct. This should represent information
|
|
|
|
// static to the channel lifetime.
|
|
|
|
CustomFundingBlob tlv.Blob
|
|
|
|
|
|
|
|
// CustomLocalCommitBlob is a custom blob that'll be stored in the
|
|
|
|
// first commitment entry for the local party.
|
|
|
|
CustomLocalCommitBlob tlv.Blob
|
|
|
|
|
|
|
|
// CustomRemoteCommitBlob is a custom blob that'll be stored in the
|
|
|
|
// first commitment entry for the remote party.
|
|
|
|
CustomRemoteCommitBlob tlv.Blob
|
|
|
|
|
|
|
|
// LocalInitAuxLeaves is the set of aux leaves that'll be used for our
|
|
|
|
// very first commitment state.
|
|
|
|
LocalInitAuxLeaves CommitAuxLeaves
|
|
|
|
|
|
|
|
// RemoteInitAuxLeaves is the set of aux leaves that'll be used for the
|
|
|
|
// very first commitment state for the remote party.
|
|
|
|
RemoteInitAuxLeaves CommitAuxLeaves
|
|
|
|
}
|
|
|
|
|
2018-08-10 04:15:18 +02:00
|
|
|
// InitFundingReserveMsg is the first message sent to initiate the workflow
|
2015-12-28 21:14:00 +01:00
|
|
|
// required to open a payment channel with a remote peer. The initial required
|
2016-10-15 15:18:38 +02:00
|
|
|
// parameters are configurable across channels. These parameters are to be
|
2017-07-30 04:20:08 +02:00
|
|
|
// chosen depending on the fee climate within the network, and time value of
|
|
|
|
// funds to be locked up within the channel. Upon success a ChannelReservation
|
|
|
|
// will be created in order to track the lifetime of this pending channel.
|
|
|
|
// Outputs selected will be 'locked', making them unavailable, for any other
|
|
|
|
// pending reservations. Therefore, all channels in reservation limbo will be
|
2018-03-10 16:21:10 +01:00
|
|
|
// periodically timed out after an idle period in order to avoid "exhaustion"
|
|
|
|
// attacks.
|
2018-08-10 04:15:18 +02:00
|
|
|
type InitFundingReserveMsg struct {
|
|
|
|
// ChainHash denotes that chain to be used to ultimately open the
|
2017-07-30 04:20:08 +02:00
|
|
|
// target channel.
|
2018-08-10 04:15:18 +02:00
|
|
|
ChainHash *chainhash.Hash
|
2017-07-30 04:20:08 +02:00
|
|
|
|
2019-11-01 05:45:44 +01:00
|
|
|
// PendingChanID is the pending channel ID for this funding flow as
|
|
|
|
// used in the wire protocol.
|
|
|
|
PendingChanID [32]byte
|
|
|
|
|
2018-08-10 04:15:18 +02:00
|
|
|
// NodeID is the ID of the remote node we would like to open a channel
|
2017-05-17 04:00:19 +02:00
|
|
|
// with.
|
2018-08-10 04:15:18 +02:00
|
|
|
NodeID *btcec.PublicKey
|
2016-10-26 23:56:48 +02:00
|
|
|
|
2018-08-10 04:15:18 +02:00
|
|
|
// NodeAddr is the address port that we used to either establish or
|
2018-02-03 07:24:43 +01:00
|
|
|
// accept the connection which led to the negotiation of this funding
|
|
|
|
// workflow.
|
2018-08-10 04:15:18 +02:00
|
|
|
NodeAddr net.Addr
|
2016-10-26 23:56:48 +02:00
|
|
|
|
2019-07-11 13:14:37 +02:00
|
|
|
// SubtractFees should be set if we intend to spend exactly
|
|
|
|
// LocalFundingAmt when opening the channel, subtracting the fees from
|
|
|
|
// the funding output. This can be used for instance to use all our
|
|
|
|
// remaining funds to open the channel, since it will take fees into
|
|
|
|
// account.
|
|
|
|
SubtractFees bool
|
|
|
|
|
2019-07-11 13:14:36 +02:00
|
|
|
// LocalFundingAmt is the amount of funds requested from us for this
|
|
|
|
// channel.
|
|
|
|
LocalFundingAmt btcutil.Amount
|
2015-11-05 21:36:19 +01:00
|
|
|
|
2019-07-11 13:14:36 +02:00
|
|
|
// RemoteFundingAmnt is the amount of funds the remote will contribute
|
|
|
|
// to this channel.
|
|
|
|
RemoteFundingAmt btcutil.Amount
|
2016-06-21 19:37:55 +02:00
|
|
|
|
2023-01-10 04:12:22 +01:00
|
|
|
// FundUpToMaxAmt defines if channel funding should try to add as many
|
|
|
|
// funds to the channel opening as possible up to this amount. If used,
|
|
|
|
// then MinFundAmt is treated as the minimum amount of funds that must
|
|
|
|
// be available to open the channel. If set to zero it is ignored.
|
|
|
|
FundUpToMaxAmt btcutil.Amount
|
|
|
|
|
|
|
|
// MinFundAmt denotes the minimum channel capacity that has to be
|
|
|
|
// allocated iff the FundUpToMaxAmt is set.
|
|
|
|
MinFundAmt btcutil.Amount
|
|
|
|
|
2023-06-10 21:27:34 +02:00
|
|
|
// Outpoints is a list of client-selected outpoints that should be used
|
|
|
|
// for funding a channel. If LocalFundingAmt is specified then this
|
|
|
|
// amount is allocated from the sum of outpoints towards funding. If the
|
|
|
|
// FundUpToMaxAmt is specified the entirety of selected funds is
|
|
|
|
// allocated towards channel funding.
|
|
|
|
Outpoints []wire.OutPoint
|
|
|
|
|
2023-01-10 04:12:22 +01:00
|
|
|
// RemoteChanReserve is the channel reserve we required for the remote
|
|
|
|
// peer.
|
|
|
|
RemoteChanReserve btcutil.Amount
|
|
|
|
|
2018-08-10 04:15:18 +02:00
|
|
|
// CommitFeePerKw is the starting accepted satoshis/Kw fee for the set
|
2017-11-23 07:30:57 +01:00
|
|
|
// of initial commitment transactions. In order to ensure timely
|
|
|
|
// confirmation, it is recommended that this fee should be generous,
|
|
|
|
// paying some multiple of the accepted base fee rate of the network.
|
2019-10-31 03:43:05 +01:00
|
|
|
CommitFeePerKw chainfee.SatPerKWeight
|
2017-11-23 07:30:57 +01:00
|
|
|
|
2018-08-10 04:15:18 +02:00
|
|
|
// FundingFeePerKw is the fee rate in sat/kw to use for the initial
|
2018-07-28 03:27:51 +02:00
|
|
|
// funding transaction.
|
2019-10-31 03:43:05 +01:00
|
|
|
FundingFeePerKw chainfee.SatPerKWeight
|
2015-12-28 21:14:00 +01:00
|
|
|
|
2018-08-10 04:15:18 +02:00
|
|
|
// PushMSat is the number of milli-satoshis that should be pushed over
|
2017-08-22 07:49:56 +02:00
|
|
|
// the responder as part of the initial channel creation.
|
2018-08-10 04:15:18 +02:00
|
|
|
PushMSat lnwire.MilliSatoshi
|
2017-01-10 02:24:13 +01:00
|
|
|
|
2018-08-10 04:15:18 +02:00
|
|
|
// Flags are the channel flags specified by the initiator in the
|
2017-11-14 02:15:27 +01:00
|
|
|
// open_channel message.
|
2018-08-10 04:15:18 +02:00
|
|
|
Flags lnwire.FundingFlag
|
2017-11-14 02:15:27 +01:00
|
|
|
|
2018-08-10 04:24:12 +02:00
|
|
|
// MinConfs indicates the minimum number of confirmations that each
|
|
|
|
// output selected to fund the channel should satisfy.
|
|
|
|
MinConfs int32
|
|
|
|
|
2020-03-06 16:11:48 +01:00
|
|
|
// CommitType indicates what type of commitment type the channel should
|
|
|
|
// be using, like tweakless or anchors.
|
|
|
|
CommitType CommitmentType
|
2019-08-01 05:16:52 +02:00
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
// ChanFunder is an optional channel funder that allows the caller to
|
|
|
|
// control exactly how the channel funding is carried out. If not
|
|
|
|
// specified, then the default chanfunding.WalletAssembler will be
|
|
|
|
// used.
|
|
|
|
ChanFunder chanfunding.Assembler
|
|
|
|
|
2024-03-30 17:55:25 +01:00
|
|
|
// AllowUtxoForFunding enables the channel funding workflow to restrict
|
|
|
|
// the selection of utxos when selecting the inputs for the channel
|
|
|
|
// opening. This does ONLY apply for the internal wallet backed channel
|
|
|
|
// opening case.
|
|
|
|
//
|
|
|
|
// NOTE: This is very useful when opening channels with unconfirmed
|
|
|
|
// inputs to make sure stable non-replaceable inputs are used.
|
|
|
|
AllowUtxoForFunding func(Utxo) bool
|
|
|
|
|
2022-04-04 22:15:02 +02:00
|
|
|
// ZeroConf is a boolean that is true if a zero-conf channel was
|
|
|
|
// negotiated.
|
|
|
|
ZeroConf bool
|
|
|
|
|
|
|
|
// OptionScidAlias is a boolean that is true if an option-scid-alias
|
|
|
|
// channel type was explicitly negotiated.
|
|
|
|
OptionScidAlias bool
|
|
|
|
|
|
|
|
// ScidAliasFeature is true if the option-scid-alias feature bit was
|
|
|
|
// negotiated.
|
|
|
|
ScidAliasFeature bool
|
|
|
|
|
2023-05-03 22:30:11 +02:00
|
|
|
// Memo is any arbitrary information we wish to store locally about the
|
|
|
|
// channel that will be useful to our future selves.
|
|
|
|
Memo []byte
|
|
|
|
|
2024-04-05 02:26:03 +02:00
|
|
|
// TapscriptRoot is an optional tapscript root that if provided, will
|
|
|
|
// be used to create the combined key for musig2 based channels.
|
2024-03-13 15:53:41 +01:00
|
|
|
TapscriptRoot fn.Option[chainhash.Hash]
|
|
|
|
|
2017-05-17 04:00:19 +02:00
|
|
|
// err is a channel in which all errors will be sent across. Will be
|
|
|
|
// nil if this initial set is successful.
|
|
|
|
//
|
2015-12-28 21:14:00 +01:00
|
|
|
// NOTE: In order to avoid deadlocks, this channel MUST be buffered.
|
|
|
|
err chan error
|
2015-10-27 23:50:30 +01:00
|
|
|
|
2017-05-17 04:00:19 +02:00
|
|
|
// resp is channel in which a ChannelReservation with our contributions
|
|
|
|
// filled in will be sent across this channel in the case of a
|
|
|
|
// successfully reservation initiation. In the case of an error, this
|
|
|
|
// will read a nil pointer.
|
|
|
|
//
|
2015-12-28 21:14:00 +01:00
|
|
|
// NOTE: In order to avoid deadlocks, this channel MUST be buffered.
|
|
|
|
resp chan *ChannelReservation
|
2015-10-27 23:50:30 +01:00
|
|
|
}
|
|
|
|
|
2015-12-28 21:14:00 +01:00
|
|
|
// fundingReserveCancelMsg is a message reserved for cancelling an existing
|
|
|
|
// channel reservation identified by its reservation ID. Cancelling a reservation
|
|
|
|
// frees its locked outputs up, for inclusion within further reservations.
|
2015-11-05 21:36:19 +01:00
|
|
|
type fundingReserveCancelMsg struct {
|
|
|
|
pendingFundingID uint64
|
|
|
|
|
2015-12-28 21:14:00 +01:00
|
|
|
// NOTE: In order to avoid deadlocks, this channel MUST be buffered.
|
2015-12-16 21:40:44 +01:00
|
|
|
err chan error // Buffered
|
2015-10-27 23:50:30 +01:00
|
|
|
}
|
|
|
|
|
2015-12-28 21:14:00 +01:00
|
|
|
// addContributionMsg represents a message executing the second phase of the
|
|
|
|
// channel reservation workflow. This message carries the counterparty's
|
|
|
|
// "contribution" to the payment channel. In the case that this message is
|
|
|
|
// processed without generating any errors, then channel reservation will then
|
|
|
|
// be able to construct the funding tx, both commitment transactions, and
|
|
|
|
// finally generate signatures for all our inputs to the funding transaction,
|
|
|
|
// and for the remote node's version of the commitment transaction.
|
2015-12-21 22:53:34 +01:00
|
|
|
type addContributionMsg struct {
|
2015-11-05 21:36:19 +01:00
|
|
|
pendingFundingID uint64
|
2015-10-27 23:50:30 +01:00
|
|
|
|
2015-12-23 05:31:17 +01:00
|
|
|
contribution *ChannelContribution
|
2015-10-27 23:50:30 +01:00
|
|
|
|
2015-12-28 21:14:00 +01:00
|
|
|
// NOTE: In order to avoid deadlocks, this channel MUST be buffered.
|
|
|
|
err chan error
|
2015-11-05 21:36:19 +01:00
|
|
|
}
|
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
// continueContributionMsg represents a message that signals that the
|
|
|
|
// interrupted funding process involving a PSBT can now be continued because the
|
|
|
|
// finalized transaction is now available.
|
|
|
|
type continueContributionMsg struct {
|
|
|
|
pendingFundingID uint64
|
|
|
|
|
2024-04-05 02:26:03 +02:00
|
|
|
// auxFundingDesc is an optional descriptor that contains information
|
|
|
|
// about the custom channel funding flow.
|
|
|
|
auxFundingDesc fn.Option[AuxFundingDesc]
|
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
// NOTE: In order to avoid deadlocks, this channel MUST be buffered.
|
|
|
|
err chan error
|
|
|
|
}
|
|
|
|
|
2016-06-21 19:37:55 +02:00
|
|
|
// addSingleContributionMsg represents a message executing the second phase of
|
|
|
|
// a single funder channel reservation workflow. This messages carries the
|
|
|
|
// counterparty's "contribution" to the payment channel. As this message is
|
|
|
|
// sent when on the responding side to a single funder workflow, no further
|
|
|
|
// action apart from storing the provided contribution is carried out.
|
|
|
|
type addSingleContributionMsg struct {
|
|
|
|
pendingFundingID uint64
|
|
|
|
|
|
|
|
contribution *ChannelContribution
|
|
|
|
|
|
|
|
// NOTE: In order to avoid deadlocks, this channel MUST be buffered.
|
|
|
|
err chan error
|
|
|
|
}
|
|
|
|
|
2015-12-28 21:14:00 +01:00
|
|
|
// addCounterPartySigsMsg represents the final message required to complete,
|
|
|
|
// and 'open' a payment channel. This message carries the counterparty's
|
|
|
|
// signatures for each of their inputs to the funding transaction, and also a
|
|
|
|
// signature allowing us to spend our version of the commitment transaction.
|
|
|
|
// If we're able to verify all the signatures are valid, the funding transaction
|
|
|
|
// will be broadcast to the network. After the funding transaction gains a
|
|
|
|
// configurable number of confirmations, the channel is officially considered
|
|
|
|
// 'open'.
|
2015-11-05 21:36:19 +01:00
|
|
|
type addCounterPartySigsMsg struct {
|
|
|
|
pendingFundingID uint64
|
2015-10-27 23:50:30 +01:00
|
|
|
|
2015-12-28 21:14:00 +01:00
|
|
|
// Should be order of sorted inputs that are theirs. Sorting is done
|
|
|
|
// in accordance to BIP-69:
|
|
|
|
// https://github.com/bitcoin/bips/blob/master/bip-0069.mediawiki.
|
2019-01-16 15:47:43 +01:00
|
|
|
theirFundingInputScripts []*input.Script
|
2015-12-21 22:54:33 +01:00
|
|
|
|
2018-02-07 04:11:11 +01:00
|
|
|
// This should be 1/2 of the signatures needed to successfully spend our
|
2015-12-21 22:54:33 +01:00
|
|
|
// version of the commitment transaction.
|
2020-04-06 02:06:38 +02:00
|
|
|
theirCommitmentSig input.Signature
|
2015-11-05 21:36:19 +01:00
|
|
|
|
2017-01-24 03:19:54 +01:00
|
|
|
// This channel is used to return the completed channel after the wallet
|
|
|
|
// has completed all of its stages in the funding process.
|
|
|
|
completeChan chan *channeldb.OpenChannel
|
|
|
|
|
2015-12-28 21:14:00 +01:00
|
|
|
// NOTE: In order to avoid deadlocks, this channel MUST be buffered.
|
|
|
|
err chan error
|
2015-10-27 23:50:30 +01:00
|
|
|
}
|
|
|
|
|
2016-06-21 19:37:55 +02:00
|
|
|
// addSingleFunderSigsMsg represents the next-to-last message required to
|
|
|
|
// complete a single-funder channel workflow. Once the initiator is able to
|
|
|
|
// construct the funding transaction, they send both the outpoint and a
|
|
|
|
// signature for our version of the commitment transaction. Once this message
|
|
|
|
// is processed we (the responder) are able to construct both commitment
|
|
|
|
// transactions, signing the remote party's version.
|
|
|
|
type addSingleFunderSigsMsg struct {
|
|
|
|
pendingFundingID uint64
|
|
|
|
|
2024-04-05 02:26:03 +02:00
|
|
|
// auxFundingDesc is an optional descriptor that contains information
|
|
|
|
// about the custom channel funding flow.
|
|
|
|
auxFundingDesc fn.Option[AuxFundingDesc]
|
|
|
|
|
2016-06-30 21:13:46 +02:00
|
|
|
// fundingOutpoint is the outpoint of the completed funding
|
2016-06-21 19:37:55 +02:00
|
|
|
// transaction as assembled by the workflow initiator.
|
|
|
|
fundingOutpoint *wire.OutPoint
|
|
|
|
|
2016-11-16 21:54:27 +01:00
|
|
|
// theirCommitmentSig are the 1/2 of the signatures needed to
|
2018-02-07 04:11:11 +01:00
|
|
|
// successfully spend our version of the commitment transaction.
|
2020-04-06 02:06:38 +02:00
|
|
|
theirCommitmentSig input.Signature
|
2016-06-21 19:37:55 +02:00
|
|
|
|
2017-01-24 03:19:54 +01:00
|
|
|
// This channel is used to return the completed channel after the wallet
|
|
|
|
// has completed all of its stages in the funding process.
|
|
|
|
completeChan chan *channeldb.OpenChannel
|
2016-06-21 19:37:55 +02:00
|
|
|
|
|
|
|
// NOTE: In order to avoid deadlocks, this channel MUST be buffered.
|
|
|
|
err chan error
|
|
|
|
}
|
|
|
|
|
2021-09-30 13:02:35 +02:00
|
|
|
// CheckReservedValueTxReq is the request struct used to call
|
|
|
|
// CheckReservedValueTx with. It contains the transaction to check as well as
|
|
|
|
// an optional explicitly defined index to denote a change output that is not
|
|
|
|
// watched by the wallet.
|
|
|
|
type CheckReservedValueTxReq struct {
|
|
|
|
// Tx is the transaction to check the outputs for.
|
|
|
|
Tx *wire.MsgTx
|
|
|
|
|
|
|
|
// ChangeIndex denotes an optional output index that can be explicitly
|
|
|
|
// set for a change that is not being watched by the wallet and would
|
|
|
|
// otherwise not be recognized as a change output.
|
|
|
|
ChangeIndex *int
|
|
|
|
}
|
|
|
|
|
2015-12-28 21:14:00 +01:00
|
|
|
// LightningWallet is a domain specific, yet general Bitcoin wallet capable of
|
|
|
|
// executing workflow required to interact with the Lightning Network. It is
|
|
|
|
// domain specific in the sense that it understands all the fancy scripts used
|
2017-11-28 00:24:45 +01:00
|
|
|
// within the Lightning Network, channel lifetimes, etc. However, it embeds a
|
2017-07-30 04:02:38 +02:00
|
|
|
// general purpose Bitcoin wallet within it. Therefore, it is also able to
|
|
|
|
// serve as a regular Bitcoin wallet which uses HD keys. The wallet is highly
|
|
|
|
// concurrent internally. All communication, and requests towards the wallet
|
|
|
|
// are dispatched as messages over channels, ensuring thread safety across all
|
2017-01-13 06:01:50 +01:00
|
|
|
// operations. Interaction has been designed independent of any peer-to-peer
|
2017-07-30 04:02:38 +02:00
|
|
|
// communication protocol, allowing the wallet to be self-contained and
|
|
|
|
// embeddable within future projects interacting with the Lightning Network.
|
|
|
|
//
|
2017-01-13 06:01:50 +01:00
|
|
|
// NOTE: At the moment the wallet requires a btcd full node, as it's dependent
|
2017-11-28 00:24:45 +01:00
|
|
|
// on btcd's websockets notifications as event triggers during the lifetime of a
|
2017-07-30 04:02:38 +02:00
|
|
|
// channel. However, once the chainntnfs package is complete, the wallet will
|
|
|
|
// be compatible with multiple RPC/notification services such as Electrum,
|
2015-12-28 21:14:00 +01:00
|
|
|
// Bitcoin Core + ZeroMQ, etc. Eventually, the wallet won't require a full-node
|
2017-07-30 04:02:38 +02:00
|
|
|
// at all, as SPV support is integrated into btcwallet.
|
2015-10-27 23:50:30 +01:00
|
|
|
type LightningWallet struct {
|
2018-06-01 00:41:41 +02:00
|
|
|
started int32 // To be used atomically.
|
|
|
|
shutdown int32 // To be used atomically.
|
|
|
|
|
|
|
|
nextFundingID uint64 // To be used atomically.
|
|
|
|
|
2017-07-30 04:02:38 +02:00
|
|
|
// Cfg is the configuration struct that will be used by the wallet to
|
|
|
|
// access the necessary interfaces and default it needs to carry on its
|
|
|
|
// duties.
|
|
|
|
Cfg Config
|
|
|
|
|
|
|
|
// WalletController is the core wallet, all non Lightning Network
|
|
|
|
// specific interaction is proxied to the internal wallet.
|
|
|
|
WalletController
|
|
|
|
|
2018-02-18 00:20:41 +01:00
|
|
|
// SecretKeyRing is the interface we'll use to derive any keys related
|
|
|
|
// to our purpose within the network including: multi-sig keys, node
|
|
|
|
// keys, revocation keys, etc.
|
|
|
|
keychain.SecretKeyRing
|
|
|
|
|
2015-12-28 21:12:18 +01:00
|
|
|
// This mutex MUST be held when performing coin selection in order to
|
|
|
|
// avoid inadvertently creating multiple funding transaction which
|
2016-10-15 15:18:38 +02:00
|
|
|
// double spend inputs across each other.
|
2015-12-28 21:12:18 +01:00
|
|
|
coinSelectMtx sync.RWMutex
|
2015-10-27 23:50:30 +01:00
|
|
|
|
2016-10-15 15:18:38 +02:00
|
|
|
// All messages to the wallet are to be sent across this channel.
|
2015-10-27 23:50:30 +01:00
|
|
|
msgChan chan interface{}
|
|
|
|
|
2015-12-28 21:14:00 +01:00
|
|
|
// Incomplete payment channels are stored in the map below. An intent
|
|
|
|
// to create a payment channel is tracked as a "reservation" within
|
|
|
|
// limbo. Once the final signatures have been exchanged, a reservation
|
|
|
|
// is removed from limbo. Each reservation is tracked by a unique
|
|
|
|
// monotonically integer. All requests concerning the channel MUST
|
|
|
|
// carry a valid, active funding ID.
|
2018-07-18 04:20:05 +02:00
|
|
|
fundingLimbo map[uint64]*ChannelReservation
|
2021-01-11 08:47:38 +01:00
|
|
|
|
|
|
|
// reservationIDs maps a pending channel ID to the reservation ID used
|
|
|
|
// as key in the fundingLimbo map. Used to easily look up a channel
|
|
|
|
// reservation given a pending channel ID.
|
|
|
|
reservationIDs map[[32]byte]uint64
|
|
|
|
limboMtx sync.RWMutex
|
2015-11-18 23:38:57 +01:00
|
|
|
|
2016-08-13 00:50:47 +02:00
|
|
|
// lockedOutPoints is a set of the currently locked outpoint. This
|
|
|
|
// information is kept in order to provide an easy way to unlock all
|
|
|
|
// the currently locked outpoints.
|
|
|
|
lockedOutPoints map[wire.OutPoint]struct{}
|
|
|
|
|
2019-11-01 05:47:27 +01:00
|
|
|
// fundingIntents houses all the "interception" registered by a caller
|
|
|
|
// using the RegisterFundingIntent method.
|
|
|
|
intentMtx sync.RWMutex
|
|
|
|
fundingIntents map[[32]byte]chanfunding.Intent
|
|
|
|
|
2018-07-18 04:20:05 +02:00
|
|
|
quit chan struct{}
|
2015-10-27 23:50:30 +01:00
|
|
|
|
|
|
|
wg sync.WaitGroup
|
|
|
|
}
|
|
|
|
|
2015-12-28 21:14:00 +01:00
|
|
|
// NewLightningWallet creates/opens and initializes a LightningWallet instance.
|
|
|
|
// If the wallet has never been created (according to the passed dataDir), first-time
|
|
|
|
// setup is executed.
|
2017-07-30 04:02:38 +02:00
|
|
|
func NewLightningWallet(Cfg Config) (*LightningWallet, error) {
|
2016-08-04 07:31:20 +02:00
|
|
|
|
2015-11-13 04:34:35 +01:00
|
|
|
return &LightningWallet{
|
2017-07-30 04:02:38 +02:00
|
|
|
Cfg: Cfg,
|
2018-02-18 00:20:41 +01:00
|
|
|
SecretKeyRing: Cfg.SecretKeyRing,
|
2017-07-30 04:02:38 +02:00
|
|
|
WalletController: Cfg.WalletController,
|
2016-08-13 00:50:47 +02:00
|
|
|
msgChan: make(chan interface{}, msgBufferSize),
|
|
|
|
nextFundingID: 0,
|
|
|
|
fundingLimbo: make(map[uint64]*ChannelReservation),
|
2021-01-11 08:47:38 +01:00
|
|
|
reservationIDs: make(map[[32]byte]uint64),
|
2016-08-13 00:50:47 +02:00
|
|
|
lockedOutPoints: make(map[wire.OutPoint]struct{}),
|
2019-11-01 05:47:27 +01:00
|
|
|
fundingIntents: make(map[[32]byte]chanfunding.Intent),
|
2016-08-13 00:50:47 +02:00
|
|
|
quit: make(chan struct{}),
|
2016-03-23 02:48:18 +01:00
|
|
|
}, nil
|
2015-11-05 21:34:11 +01:00
|
|
|
}
|
|
|
|
|
2015-12-29 19:44:59 +01:00
|
|
|
// Startup establishes a connection to the RPC source, and spins up all
|
2015-12-28 21:14:00 +01:00
|
|
|
// goroutines required to handle incoming messages.
|
2015-12-29 19:44:59 +01:00
|
|
|
func (l *LightningWallet) Startup() error {
|
2015-10-27 23:50:30 +01:00
|
|
|
// Already started?
|
|
|
|
if atomic.AddInt32(&l.started, 1) != 1 {
|
|
|
|
return nil
|
|
|
|
}
|
2015-11-27 07:48:42 +01:00
|
|
|
|
2016-08-13 00:50:47 +02:00
|
|
|
// Start the underlying wallet controller.
|
|
|
|
if err := l.Start(); err != nil {
|
2015-11-27 07:48:42 +01:00
|
|
|
return err
|
|
|
|
}
|
2015-10-27 23:50:30 +01:00
|
|
|
|
2023-02-24 02:59:06 +01:00
|
|
|
if l.Cfg.Rebroadcaster != nil {
|
|
|
|
go func() {
|
|
|
|
if err := l.Cfg.Rebroadcaster.Start(); err != nil {
|
|
|
|
walletLog.Errorf("unable to start "+
|
|
|
|
"rebroadcaster: %v", err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2015-10-27 23:50:30 +01:00
|
|
|
l.wg.Add(1)
|
|
|
|
go l.requestHandler()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-12-29 19:44:59 +01:00
|
|
|
// Shutdown gracefully stops the wallet, and all active goroutines.
|
|
|
|
func (l *LightningWallet) Shutdown() error {
|
2015-10-27 23:50:30 +01:00
|
|
|
if atomic.AddInt32(&l.shutdown, 1) != 1 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-04-24 21:39:43 +02:00
|
|
|
// Signal the underlying wallet controller to shutdown, waiting until
|
|
|
|
// all active goroutines have been shutdown.
|
2016-08-13 00:50:47 +02:00
|
|
|
if err := l.Stop(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-11-28 01:07:22 +01:00
|
|
|
|
2023-02-24 02:59:06 +01:00
|
|
|
if l.Cfg.Rebroadcaster != nil && l.Cfg.Rebroadcaster.Started() {
|
|
|
|
l.Cfg.Rebroadcaster.Stop()
|
|
|
|
}
|
|
|
|
|
2015-10-27 23:50:30 +01:00
|
|
|
close(l.quit)
|
|
|
|
l.wg.Wait()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-02-24 02:59:06 +01:00
|
|
|
// PublishTransaction wraps the wallet controller tx publish method with an
|
|
|
|
// extra rebroadcaster layer if the sub-system is configured.
|
|
|
|
func (l *LightningWallet) PublishTransaction(tx *wire.MsgTx,
|
|
|
|
label string) error {
|
|
|
|
|
|
|
|
sendTxToWallet := func() error {
|
|
|
|
return l.WalletController.PublishTransaction(tx, label)
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we don't have rebroadcaster then we can exit early (and send only
|
|
|
|
// to the wallet).
|
|
|
|
if l.Cfg.Rebroadcaster == nil || !l.Cfg.Rebroadcaster.Started() {
|
|
|
|
return sendTxToWallet()
|
|
|
|
}
|
|
|
|
|
|
|
|
// We pass this into the rebroadcaster first, so the initial attempt
|
|
|
|
// will succeed if the transaction isn't yet in the mempool. However we
|
|
|
|
// ignore the error here as this might be resent on start up and the
|
|
|
|
// transaction already exists.
|
|
|
|
_ = l.Cfg.Rebroadcaster.Broadcast(tx)
|
|
|
|
|
|
|
|
// Then we pass things into the wallet as normal, which'll add the
|
|
|
|
// transaction label on disk.
|
|
|
|
if err := sendTxToWallet(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO(roasbeef): want diff height actually? no context though
|
|
|
|
_, bestHeight, err := l.Cfg.ChainIO.GetBestBlock()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
txHash := tx.TxHash()
|
|
|
|
go func() {
|
|
|
|
const numConfs = 6
|
|
|
|
|
|
|
|
txConf, err := l.Cfg.Notifier.RegisterConfirmationsNtfn(
|
2023-03-12 08:01:34 +01:00
|
|
|
&txHash, tx.TxOut[0].PkScript, numConfs,
|
|
|
|
uint32(bestHeight),
|
2023-02-24 02:59:06 +01:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
select {
|
|
|
|
case <-txConf.Confirmed:
|
|
|
|
// TODO(roasbeef): also want to remove from
|
|
|
|
// rebroadcaster if conflict happens...deeper wallet
|
|
|
|
// integration?
|
|
|
|
l.Cfg.Rebroadcaster.MarkAsConfirmed(tx.TxHash())
|
|
|
|
|
|
|
|
case <-l.quit:
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-02-20 02:41:50 +01:00
|
|
|
// ConfirmedBalance returns the current confirmed balance of a wallet account.
|
|
|
|
// This methods wraps the internal WalletController method so we're able to
|
|
|
|
// properly hold the coin select mutex while we compute the balance.
|
|
|
|
func (l *LightningWallet) ConfirmedBalance(confs int32,
|
|
|
|
account string) (btcutil.Amount, error) {
|
|
|
|
|
2020-08-13 22:01:00 +02:00
|
|
|
l.coinSelectMtx.Lock()
|
|
|
|
defer l.coinSelectMtx.Unlock()
|
|
|
|
|
2021-02-20 02:41:50 +01:00
|
|
|
return l.WalletController.ConfirmedBalance(confs, account)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ListUnspentWitnessFromDefaultAccount returns all unspent outputs from the
|
|
|
|
// default wallet account which are version 0 witness programs. The 'minConfs'
|
|
|
|
// and 'maxConfs' parameters indicate the minimum and maximum number of
|
|
|
|
// confirmations an output needs in order to be returned by this method. Passing
|
|
|
|
// -1 as 'minConfs' indicates that even unconfirmed outputs should be returned.
|
|
|
|
// Using MaxInt32 as 'maxConfs' implies returning all outputs with at least
|
|
|
|
// 'minConfs'.
|
|
|
|
//
|
|
|
|
// NOTE: This method requires the global coin selection lock to be held.
|
|
|
|
func (l *LightningWallet) ListUnspentWitnessFromDefaultAccount(
|
|
|
|
minConfs, maxConfs int32) ([]*Utxo, error) {
|
|
|
|
|
|
|
|
return l.WalletController.ListUnspentWitness(
|
|
|
|
minConfs, maxConfs, DefaultAccountName,
|
|
|
|
)
|
2020-08-13 22:01:00 +02:00
|
|
|
}
|
|
|
|
|
2017-02-23 20:56:47 +01:00
|
|
|
// LockedOutpoints returns a list of all currently locked outpoint.
|
2016-08-13 00:50:47 +02:00
|
|
|
func (l *LightningWallet) LockedOutpoints() []*wire.OutPoint {
|
|
|
|
outPoints := make([]*wire.OutPoint, 0, len(l.lockedOutPoints))
|
|
|
|
for outPoint := range l.lockedOutPoints {
|
2019-11-07 11:44:58 +01:00
|
|
|
outPoint := outPoint
|
|
|
|
|
2016-08-13 00:50:47 +02:00
|
|
|
outPoints = append(outPoints, &outPoint)
|
|
|
|
}
|
|
|
|
|
|
|
|
return outPoints
|
|
|
|
}
|
|
|
|
|
2017-12-18 03:40:05 +01:00
|
|
|
// ResetReservations reset the volatile wallet state which tracks all currently
|
2016-08-13 00:50:47 +02:00
|
|
|
// active reservations.
|
|
|
|
func (l *LightningWallet) ResetReservations() {
|
|
|
|
l.nextFundingID = 0
|
|
|
|
l.fundingLimbo = make(map[uint64]*ChannelReservation)
|
2021-01-11 08:47:38 +01:00
|
|
|
l.reservationIDs = make(map[[32]byte]uint64)
|
2016-08-13 00:50:47 +02:00
|
|
|
|
|
|
|
for outpoint := range l.lockedOutPoints {
|
2024-02-22 00:04:56 +01:00
|
|
|
_ = l.ReleaseOutput(chanfunding.LndInternalLockID, outpoint)
|
2016-08-13 00:50:47 +02:00
|
|
|
}
|
|
|
|
l.lockedOutPoints = make(map[wire.OutPoint]struct{})
|
|
|
|
}
|
|
|
|
|
|
|
|
// ActiveReservations returns a slice of all the currently active
|
2019-10-03 17:22:43 +02:00
|
|
|
// (non-canceled) reservations.
|
2016-08-13 00:50:47 +02:00
|
|
|
func (l *LightningWallet) ActiveReservations() []*ChannelReservation {
|
|
|
|
reservations := make([]*ChannelReservation, 0, len(l.fundingLimbo))
|
|
|
|
for _, reservation := range l.fundingLimbo {
|
|
|
|
reservations = append(reservations, reservation)
|
|
|
|
}
|
|
|
|
|
|
|
|
return reservations
|
|
|
|
}
|
|
|
|
|
2016-10-15 15:18:38 +02:00
|
|
|
// requestHandler is the primary goroutine(s) responsible for handling, and
|
2018-03-10 16:21:10 +01:00
|
|
|
// dispatching replies to all messages.
|
2015-10-27 23:50:30 +01:00
|
|
|
func (l *LightningWallet) requestHandler() {
|
2019-07-09 17:26:27 +02:00
|
|
|
defer l.wg.Done()
|
|
|
|
|
2015-10-27 23:50:30 +01:00
|
|
|
out:
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case m := <-l.msgChan:
|
|
|
|
switch msg := m.(type) {
|
2018-08-10 04:15:18 +02:00
|
|
|
case *InitFundingReserveMsg:
|
2015-10-27 23:50:30 +01:00
|
|
|
l.handleFundingReserveRequest(msg)
|
2015-11-05 21:36:19 +01:00
|
|
|
case *fundingReserveCancelMsg:
|
2015-10-27 23:50:30 +01:00
|
|
|
l.handleFundingCancelRequest(msg)
|
2016-06-21 19:37:55 +02:00
|
|
|
case *addSingleContributionMsg:
|
|
|
|
l.handleSingleContribution(msg)
|
2015-12-21 22:53:34 +01:00
|
|
|
case *addContributionMsg:
|
|
|
|
l.handleContributionMsg(msg)
|
2020-03-31 09:13:16 +02:00
|
|
|
case *continueContributionMsg:
|
|
|
|
l.handleChanPointReady(msg)
|
2016-06-21 19:37:55 +02:00
|
|
|
case *addSingleFunderSigsMsg:
|
|
|
|
l.handleSingleFunderSigs(msg)
|
2015-11-05 21:36:19 +01:00
|
|
|
case *addCounterPartySigsMsg:
|
|
|
|
l.handleFundingCounterPartySigs(msg)
|
2015-10-27 23:50:30 +01:00
|
|
|
}
|
|
|
|
case <-l.quit:
|
|
|
|
// TODO: do some clean up
|
|
|
|
break out
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-15 15:18:38 +02:00
|
|
|
// InitChannelReservation kicks off the 3-step workflow required to successfully
|
2015-12-28 21:14:00 +01:00
|
|
|
// open a payment channel with a remote node. As part of the funding
|
|
|
|
// reservation, the inputs selected for the funding transaction are 'locked'.
|
|
|
|
// This ensures that multiple channel reservations aren't double spending the
|
|
|
|
// same inputs in the funding transaction. If reservation initialization is
|
2016-10-15 15:18:38 +02:00
|
|
|
// successful, a ChannelReservation containing our completed contribution is
|
|
|
|
// returned. Our contribution contains all the items necessary to allow the
|
2017-01-13 06:01:50 +01:00
|
|
|
// counterparty to build the funding transaction, and both versions of the
|
2018-03-10 16:21:10 +01:00
|
|
|
// commitment transaction. Otherwise, an error occurred and a nil pointer along
|
|
|
|
// with an error are returned.
|
2015-12-28 21:14:00 +01:00
|
|
|
//
|
2016-06-21 19:37:55 +02:00
|
|
|
// Once a ChannelReservation has been obtained, two additional steps must be
|
|
|
|
// processed before a payment channel can be considered 'open'. The second step
|
|
|
|
// validates, and processes the counterparty's channel contribution. The third,
|
|
|
|
// and final step verifies all signatures for the inputs of the funding
|
2018-03-10 16:21:10 +01:00
|
|
|
// transaction, and that the signature we record for our version of the
|
2016-06-21 19:37:55 +02:00
|
|
|
// commitment transaction is valid.
|
2017-07-30 04:20:08 +02:00
|
|
|
func (l *LightningWallet) InitChannelReservation(
|
2018-08-10 04:16:14 +02:00
|
|
|
req *InitFundingReserveMsg) (*ChannelReservation, error) {
|
|
|
|
|
|
|
|
req.resp = make(chan *ChannelReservation, 1)
|
|
|
|
req.err = make(chan error, 1)
|
|
|
|
|
|
|
|
select {
|
|
|
|
case l.msgChan <- req:
|
|
|
|
case <-l.quit:
|
|
|
|
return nil, errors.New("wallet shutting down")
|
2015-11-05 21:36:19 +01:00
|
|
|
}
|
|
|
|
|
2018-08-10 04:16:14 +02:00
|
|
|
return <-req.resp, <-req.err
|
2015-11-05 21:36:19 +01:00
|
|
|
}
|
|
|
|
|
2019-11-01 05:47:27 +01:00
|
|
|
// RegisterFundingIntent allows a caller to signal to the wallet that if a
|
|
|
|
// pending channel ID of expectedID is found, then it can skip constructing a
|
|
|
|
// new chanfunding.Assembler, and instead use the specified chanfunding.Intent.
|
|
|
|
// As an example, this lets some of the parameters for funding transaction to
|
|
|
|
// be negotiated outside the regular funding protocol.
|
|
|
|
func (l *LightningWallet) RegisterFundingIntent(expectedID [32]byte,
|
|
|
|
shimIntent chanfunding.Intent) error {
|
|
|
|
|
|
|
|
l.intentMtx.Lock()
|
2019-11-14 06:00:30 +01:00
|
|
|
defer l.intentMtx.Unlock()
|
|
|
|
|
2023-10-09 10:58:18 +02:00
|
|
|
// Sanity check the pending channel ID is not empty.
|
|
|
|
var zeroID [32]byte
|
|
|
|
if expectedID == zeroID {
|
|
|
|
return ErrEmptyPendingChanID
|
|
|
|
}
|
|
|
|
|
2019-11-14 06:00:30 +01:00
|
|
|
if _, ok := l.fundingIntents[expectedID]; ok {
|
2023-10-09 10:58:18 +02:00
|
|
|
return fmt.Errorf("%w: already has intent registered: %v",
|
|
|
|
ErrDuplicatePendingChanID, expectedID[:])
|
2019-11-14 06:00:30 +01:00
|
|
|
}
|
|
|
|
|
2019-11-01 05:47:27 +01:00
|
|
|
l.fundingIntents[expectedID] = shimIntent
|
2019-11-14 06:00:30 +01:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
// PsbtFundingVerify looks up a previously registered funding intent by its
|
|
|
|
// pending channel ID and tries to advance the state machine by verifying the
|
|
|
|
// passed PSBT.
|
2021-01-11 08:47:38 +01:00
|
|
|
func (l *LightningWallet) PsbtFundingVerify(pendingChanID [32]byte,
|
2021-06-07 11:16:38 +02:00
|
|
|
packet *psbt.Packet, skipFinalize bool) error {
|
2020-03-31 09:13:16 +02:00
|
|
|
|
|
|
|
l.intentMtx.Lock()
|
|
|
|
defer l.intentMtx.Unlock()
|
|
|
|
|
2021-01-11 08:47:38 +01:00
|
|
|
intent, ok := l.fundingIntents[pendingChanID]
|
2020-03-31 09:13:16 +02:00
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("no funding intent found for "+
|
2021-01-11 08:47:38 +01:00
|
|
|
"pendingChannelID(%x)", pendingChanID[:])
|
2020-03-31 09:13:16 +02:00
|
|
|
}
|
|
|
|
psbtIntent, ok := intent.(*chanfunding.PsbtIntent)
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("incompatible funding intent")
|
|
|
|
}
|
2021-06-07 11:16:38 +02:00
|
|
|
|
|
|
|
if skipFinalize && psbtIntent.ShouldPublishFundingTX() {
|
|
|
|
return fmt.Errorf("cannot set skip_finalize for channel that " +
|
|
|
|
"did not set no_publish")
|
|
|
|
}
|
|
|
|
|
|
|
|
err := psbtIntent.Verify(packet, skipFinalize)
|
2020-03-31 09:13:16 +02:00
|
|
|
if err != nil {
|
2024-02-26 12:19:38 +01:00
|
|
|
return fmt.Errorf("error verifying PSBT: %w", err)
|
2020-03-31 09:13:16 +02:00
|
|
|
}
|
|
|
|
|
2021-01-11 08:47:38 +01:00
|
|
|
// Get the channel reservation for that corresponds to this pending
|
|
|
|
// channel ID.
|
|
|
|
l.limboMtx.Lock()
|
|
|
|
pid, ok := l.reservationIDs[pendingChanID]
|
|
|
|
if !ok {
|
|
|
|
l.limboMtx.Unlock()
|
|
|
|
return fmt.Errorf("no channel reservation found for "+
|
|
|
|
"pendingChannelID(%x)", pendingChanID[:])
|
|
|
|
}
|
|
|
|
|
|
|
|
pendingReservation, ok := l.fundingLimbo[pid]
|
|
|
|
l.limboMtx.Unlock()
|
|
|
|
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("no channel reservation found for "+
|
|
|
|
"reservation ID %v", pid)
|
|
|
|
}
|
|
|
|
|
2024-01-03 21:47:03 +01:00
|
|
|
// Now the PSBT has been populated and verified, we can again check
|
2021-06-24 02:40:36 +02:00
|
|
|
// whether the value reserved for anchor fee bumping is respected.
|
2021-05-10 10:06:23 +02:00
|
|
|
isPublic := pendingReservation.partialState.ChannelFlags&lnwire.FFAnnounceChannel != 0
|
2021-06-24 02:40:36 +02:00
|
|
|
hasAnchors := pendingReservation.partialState.ChanType.HasAnchors()
|
|
|
|
return l.enforceNewReservedValue(intent, isPublic, hasAnchors)
|
2020-03-31 09:13:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// PsbtFundingFinalize looks up a previously registered funding intent by its
|
|
|
|
// pending channel ID and tries to advance the state machine by finalizing the
|
|
|
|
// passed PSBT.
|
2020-09-07 18:02:46 +02:00
|
|
|
func (l *LightningWallet) PsbtFundingFinalize(pid [32]byte, packet *psbt.Packet,
|
|
|
|
rawTx *wire.MsgTx) error {
|
2020-03-31 09:13:16 +02:00
|
|
|
|
|
|
|
l.intentMtx.Lock()
|
|
|
|
defer l.intentMtx.Unlock()
|
|
|
|
|
|
|
|
intent, ok := l.fundingIntents[pid]
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("no funding intent found for "+
|
|
|
|
"pendingChannelID(%x)", pid[:])
|
|
|
|
}
|
|
|
|
psbtIntent, ok := intent.(*chanfunding.PsbtIntent)
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("incompatible funding intent")
|
|
|
|
}
|
2020-09-07 18:02:46 +02:00
|
|
|
|
|
|
|
// Either the PSBT or the raw TX must be set.
|
|
|
|
switch {
|
|
|
|
case packet != nil && rawTx == nil:
|
|
|
|
err := psbtIntent.Finalize(packet)
|
|
|
|
if err != nil {
|
2024-02-26 12:19:38 +01:00
|
|
|
return fmt.Errorf("error finalizing PSBT: %w", err)
|
2020-09-07 18:02:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
case rawTx != nil && packet == nil:
|
|
|
|
err := psbtIntent.FinalizeRawTX(rawTx)
|
|
|
|
if err != nil {
|
2024-02-26 12:19:38 +01:00
|
|
|
return fmt.Errorf("error finalizing raw TX: %w", err)
|
2020-09-07 18:02:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
return fmt.Errorf("either a PSBT or raw TX must be specified")
|
2020-03-31 09:13:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-11-14 06:00:30 +01:00
|
|
|
// CancelFundingIntent allows a caller to cancel a previously registered
|
|
|
|
// funding intent. If no intent was found, then an error will be returned.
|
|
|
|
func (l *LightningWallet) CancelFundingIntent(pid [32]byte) error {
|
|
|
|
l.intentMtx.Lock()
|
|
|
|
defer l.intentMtx.Unlock()
|
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
intent, ok := l.fundingIntents[pid]
|
|
|
|
if !ok {
|
2019-11-14 06:00:30 +01:00
|
|
|
return fmt.Errorf("no funding intent found for "+
|
|
|
|
"pendingChannelID(%x)", pid[:])
|
|
|
|
}
|
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
// Give the intent a chance to clean up after itself, removing coin
|
|
|
|
// locks or similar reserved resources.
|
|
|
|
intent.Cancel()
|
|
|
|
|
2019-11-14 06:00:30 +01:00
|
|
|
delete(l.fundingIntents, pid)
|
2019-11-01 05:47:27 +01:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-12-28 21:14:00 +01:00
|
|
|
// handleFundingReserveRequest processes a message intending to create, and
|
|
|
|
// validate a funding reservation request.
|
2018-08-10 04:15:18 +02:00
|
|
|
func (l *LightningWallet) handleFundingReserveRequest(req *InitFundingReserveMsg) {
|
2023-01-10 04:12:22 +01:00
|
|
|
|
|
|
|
noFundsCommitted := req.LocalFundingAmt == 0 &&
|
|
|
|
req.RemoteFundingAmt == 0 && req.FundUpToMaxAmt == 0
|
|
|
|
|
2016-10-24 03:42:03 +02:00
|
|
|
// It isn't possible to create a channel with zero funds committed.
|
2023-01-10 04:12:22 +01:00
|
|
|
if noFundsCommitted {
|
2018-02-27 18:35:56 +01:00
|
|
|
err := ErrZeroCapacity()
|
|
|
|
req.err <- err
|
2016-10-24 03:42:03 +02:00
|
|
|
req.resp <- nil
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
// If the funding request is for a different chain than the one the
|
|
|
|
// wallet is aware of, then we'll reject the request.
|
2018-08-10 04:15:18 +02:00
|
|
|
if !bytes.Equal(l.Cfg.NetParams.GenesisHash[:], req.ChainHash[:]) {
|
|
|
|
err := ErrChainMismatch(
|
|
|
|
l.Cfg.NetParams.GenesisHash, req.ChainHash,
|
|
|
|
)
|
2018-02-27 18:35:56 +01:00
|
|
|
req.err <- err
|
2017-07-30 04:20:08 +02:00
|
|
|
req.resp <- nil
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-07-17 20:26:29 +02:00
|
|
|
// We need to avoid enforcing reserved value in the middle of PSBT
|
|
|
|
// funding because some of the following steps may add UTXOs funding
|
|
|
|
// the on-chain wallet.
|
|
|
|
// The enforcement still happens at the last step - in PsbtFundingVerify
|
|
|
|
enforceNewReservedValue := true
|
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
// If no chanFunder was provided, then we'll assume the default
|
|
|
|
// assembler, which is backed by the wallet's internal coin selection.
|
|
|
|
if req.ChanFunder == nil {
|
2021-09-23 21:40:37 +02:00
|
|
|
// We use the P2WSH dust limit since it is larger than the
|
|
|
|
// P2WPKH dust limit and to avoid threading through two
|
|
|
|
// different dust limits.
|
2019-11-01 05:39:17 +01:00
|
|
|
cfg := chanfunding.WalletConfig{
|
2024-03-30 17:55:25 +01:00
|
|
|
CoinSource: NewCoinSource(
|
|
|
|
l, req.AllowUtxoForFunding,
|
|
|
|
),
|
2019-11-01 05:39:17 +01:00
|
|
|
CoinSelectLocker: l,
|
2024-02-21 23:15:45 +01:00
|
|
|
CoinLeaser: l,
|
2019-11-01 05:39:17 +01:00
|
|
|
Signer: l.Cfg.Signer,
|
2024-02-06 12:25:45 +01:00
|
|
|
DustLimit: DustLimitForSize(
|
|
|
|
input.P2WSHSize,
|
|
|
|
),
|
|
|
|
CoinSelectionStrategy: l.Cfg.CoinSelectionStrategy,
|
2019-11-01 05:39:17 +01:00
|
|
|
}
|
|
|
|
req.ChanFunder = chanfunding.NewWalletAssembler(cfg)
|
2021-07-17 20:26:29 +02:00
|
|
|
} else {
|
|
|
|
_, isPsbtFunder := req.ChanFunder.(*chanfunding.PsbtAssembler)
|
|
|
|
enforceNewReservedValue = !isPsbtFunder
|
2019-11-01 05:39:17 +01:00
|
|
|
}
|
|
|
|
|
2019-07-11 13:14:36 +02:00
|
|
|
localFundingAmt := req.LocalFundingAmt
|
2019-11-01 05:39:17 +01:00
|
|
|
remoteFundingAmt := req.RemoteFundingAmt
|
2021-12-22 11:47:13 +01:00
|
|
|
hasAnchors := req.CommitType.HasAnchors()
|
2019-07-11 13:14:36 +02:00
|
|
|
|
2019-07-11 13:14:36 +02:00
|
|
|
var (
|
2019-11-01 05:39:17 +01:00
|
|
|
fundingIntent chanfunding.Intent
|
|
|
|
err error
|
2019-07-11 13:14:36 +02:00
|
|
|
)
|
|
|
|
|
2019-11-01 05:47:27 +01:00
|
|
|
// If we've just received an inbound funding request that we have a
|
|
|
|
// registered shim intent to, then we'll obtain the backing intent now.
|
|
|
|
// In this case, we're doing a special funding workflow that allows
|
|
|
|
// more advanced constructions such as channel factories to be
|
|
|
|
// instantiated.
|
|
|
|
l.intentMtx.Lock()
|
|
|
|
fundingIntent, ok := l.fundingIntents[req.PendingChanID]
|
|
|
|
l.intentMtx.Unlock()
|
|
|
|
|
|
|
|
// Otherwise, this is a normal funding flow, so we'll use the chan
|
|
|
|
// funder in the attached request to provision the inputs/outputs
|
|
|
|
// that'll ultimately be used to construct the funding transaction.
|
|
|
|
if !ok {
|
2021-12-22 11:47:13 +01:00
|
|
|
var err error
|
|
|
|
var numAnchorChans int
|
|
|
|
|
|
|
|
// Get the number of anchor channels to determine if there is a
|
|
|
|
// reserved value that must be respected when funding up to the
|
|
|
|
// maximum amount. Since private channels (most likely) won't be
|
|
|
|
// used for routing other than the last hop, they bear a smaller
|
|
|
|
// risk that we must force close them in order to resolve a HTLC
|
|
|
|
// up/downstream. Hence we exclude them from the count of anchor
|
|
|
|
// channels in order to attribute the respective anchor amount
|
|
|
|
// to the channel capacity.
|
|
|
|
if req.FundUpToMaxAmt > 0 && req.MinFundAmt > 0 {
|
|
|
|
numAnchorChans, err = l.CurrentNumAnchorChans()
|
|
|
|
if err != nil {
|
|
|
|
req.err <- err
|
|
|
|
req.resp <- nil
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
isPublic := req.Flags&lnwire.FFAnnounceChannel != 0
|
|
|
|
if hasAnchors && isPublic {
|
|
|
|
numAnchorChans++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-11 13:14:36 +02:00
|
|
|
// Coin selection is done on the basis of sat/kw, so we'll use
|
|
|
|
// the fee rate passed in to perform coin selection.
|
2019-11-01 05:39:17 +01:00
|
|
|
fundingReq := &chanfunding.Request{
|
2023-01-10 04:12:22 +01:00
|
|
|
RemoteAmt: req.RemoteFundingAmt,
|
|
|
|
LocalAmt: req.LocalFundingAmt,
|
|
|
|
FundUpToMaxAmt: req.FundUpToMaxAmt,
|
|
|
|
MinFundAmt: req.MinFundAmt,
|
|
|
|
RemoteChanReserve: req.RemoteChanReserve,
|
|
|
|
PushAmt: lnwire.MilliSatoshi.ToSatoshis(
|
|
|
|
req.PushMSat,
|
|
|
|
),
|
2021-12-22 11:47:13 +01:00
|
|
|
WalletReserve: l.RequiredReserve(
|
|
|
|
uint32(numAnchorChans),
|
|
|
|
),
|
2023-06-10 21:27:34 +02:00
|
|
|
Outpoints: req.Outpoints,
|
2019-11-01 05:39:17 +01:00
|
|
|
MinConfs: req.MinConfs,
|
|
|
|
SubtractFees: req.SubtractFees,
|
|
|
|
FeeRate: req.FundingFeePerKw,
|
|
|
|
ChangeAddr: func() (btcutil.Address, error) {
|
2021-02-20 02:41:45 +01:00
|
|
|
return l.NewAddress(
|
2022-08-11 03:33:19 +02:00
|
|
|
TaprootPubkey, true, DefaultAccountName,
|
2021-02-20 02:41:45 +01:00
|
|
|
)
|
2019-11-01 05:39:17 +01:00
|
|
|
},
|
2024-06-26 00:31:32 +02:00
|
|
|
Musig2: req.CommitType.IsTaproot(),
|
2019-11-01 05:39:17 +01:00
|
|
|
}
|
|
|
|
fundingIntent, err = req.ChanFunder.ProvisionChannel(
|
|
|
|
fundingReq,
|
2019-07-11 13:14:36 +02:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
req.err <- err
|
|
|
|
req.resp <- nil
|
|
|
|
return
|
|
|
|
}
|
2019-07-11 13:14:37 +02:00
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
// Register the funding intent now in case we need to access it
|
|
|
|
// again later, as it's the case for the PSBT state machine for
|
|
|
|
// example.
|
|
|
|
err = l.RegisterFundingIntent(req.PendingChanID, fundingIntent)
|
|
|
|
if err != nil {
|
|
|
|
req.err <- err
|
|
|
|
req.resp <- nil
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-09-01 07:53:06 +02:00
|
|
|
walletLog.Debugf("Registered funding intent for "+
|
|
|
|
"PendingChanID: %x", req.PendingChanID)
|
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
localFundingAmt = fundingIntent.LocalFundingAmt()
|
|
|
|
remoteFundingAmt = fundingIntent.RemoteFundingAmt()
|
2019-07-11 13:14:36 +02:00
|
|
|
}
|
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
// At this point there _has_ to be a funding intent, otherwise something
|
|
|
|
// went really wrong.
|
|
|
|
if fundingIntent == nil {
|
|
|
|
req.err <- fmt.Errorf("no funding intent present")
|
|
|
|
req.resp <- nil
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-03-14 00:57:48 +01:00
|
|
|
// If this is a shim intent, then it may be attempting to use an
|
|
|
|
// existing set of keys for the funding workflow. In this case, we'll
|
|
|
|
// make a simple wrapper keychain.KeyRing that will proxy certain
|
|
|
|
// derivation calls to future callers.
|
|
|
|
var (
|
|
|
|
keyRing keychain.KeyRing = l.SecretKeyRing
|
|
|
|
thawHeight uint32
|
|
|
|
)
|
|
|
|
if shimIntent, ok := fundingIntent.(*chanfunding.ShimIntent); ok {
|
|
|
|
keyRing = &shimKeyRing{
|
|
|
|
KeyRing: keyRing,
|
|
|
|
ShimIntent: shimIntent,
|
|
|
|
}
|
|
|
|
|
|
|
|
// As this was a registered shim intent, we'll obtain the thaw
|
|
|
|
// height of the intent, if present at all. If this is
|
|
|
|
// non-zero, then we'll mark this as the proper channel type.
|
|
|
|
thawHeight = shimIntent.ThawHeight()
|
|
|
|
}
|
|
|
|
|
2021-01-11 08:47:38 +01:00
|
|
|
// Now that we have a funding intent, we'll check whether funding a
|
|
|
|
// channel using it would violate our reserved value for anchor channel
|
2021-06-24 02:40:36 +02:00
|
|
|
// fee bumping.
|
|
|
|
//
|
2021-01-11 08:47:38 +01:00
|
|
|
// Check the reserved value using the inputs and outputs given by the
|
2021-06-24 02:40:36 +02:00
|
|
|
// intent. Note that for the PSBT intent type we don't yet have the
|
|
|
|
// funding tx ready, so this will always pass. We'll do another check
|
2021-01-11 08:47:38 +01:00
|
|
|
// when the PSBT has been verified.
|
2021-06-24 02:40:36 +02:00
|
|
|
isPublic := req.Flags&lnwire.FFAnnounceChannel != 0
|
2021-07-17 20:26:29 +02:00
|
|
|
if enforceNewReservedValue {
|
|
|
|
err = l.enforceNewReservedValue(fundingIntent, isPublic, hasAnchors)
|
|
|
|
if err != nil {
|
|
|
|
fundingIntent.Cancel()
|
2021-01-11 08:47:38 +01:00
|
|
|
|
2021-07-17 20:26:29 +02:00
|
|
|
req.err <- err
|
|
|
|
req.resp <- nil
|
|
|
|
return
|
|
|
|
}
|
2021-01-11 08:47:38 +01:00
|
|
|
}
|
|
|
|
|
2019-07-11 13:14:37 +02:00
|
|
|
// The total channel capacity will be the size of the funding output we
|
|
|
|
// created plus the remote contribution.
|
2019-11-01 05:39:17 +01:00
|
|
|
capacity := localFundingAmt + remoteFundingAmt
|
2019-07-11 13:14:37 +02:00
|
|
|
|
2016-08-13 00:50:47 +02:00
|
|
|
id := atomic.AddUint64(&l.nextFundingID, 1)
|
2018-08-10 04:15:18 +02:00
|
|
|
reservation, err := NewChannelReservation(
|
2022-04-04 22:15:02 +02:00
|
|
|
capacity, localFundingAmt, l, id, l.Cfg.NetParams.GenesisHash,
|
|
|
|
thawHeight, req,
|
2018-08-10 04:15:18 +02:00
|
|
|
)
|
2017-11-26 20:32:57 +01:00
|
|
|
if err != nil {
|
2020-03-31 09:13:16 +02:00
|
|
|
fundingIntent.Cancel()
|
2019-11-01 05:39:17 +01:00
|
|
|
|
2017-11-26 20:32:57 +01:00
|
|
|
req.err <- err
|
|
|
|
req.resp <- nil
|
|
|
|
return
|
|
|
|
}
|
2015-10-27 23:50:30 +01:00
|
|
|
|
2019-07-11 13:14:37 +02:00
|
|
|
err = l.initOurContribution(
|
2019-11-01 05:44:16 +01:00
|
|
|
reservation, fundingIntent, req.NodeAddr, req.NodeID, keyRing,
|
2019-07-11 13:14:37 +02:00
|
|
|
)
|
|
|
|
if err != nil {
|
2020-03-31 09:13:16 +02:00
|
|
|
fundingIntent.Cancel()
|
2019-11-01 05:39:17 +01:00
|
|
|
|
2019-07-11 13:14:37 +02:00
|
|
|
req.err <- err
|
|
|
|
req.resp <- nil
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a limbo and record entry for this newly pending funding
|
|
|
|
// request.
|
|
|
|
l.limboMtx.Lock()
|
|
|
|
l.fundingLimbo[id] = reservation
|
2021-01-11 08:47:38 +01:00
|
|
|
l.reservationIDs[req.PendingChanID] = id
|
2019-07-11 13:14:37 +02:00
|
|
|
l.limboMtx.Unlock()
|
|
|
|
|
|
|
|
// Funding reservation request successfully handled. The funding inputs
|
|
|
|
// will be marked as unavailable until the reservation is either
|
2019-10-03 17:22:43 +02:00
|
|
|
// completed, or canceled.
|
2019-07-11 13:14:37 +02:00
|
|
|
req.resp <- reservation
|
|
|
|
req.err <- nil
|
2022-09-01 07:53:06 +02:00
|
|
|
|
|
|
|
walletLog.Debugf("Successfully handled funding reservation with "+
|
|
|
|
"pendingChanID: %x, reservationID: %v",
|
|
|
|
reservation.pendingChanID, reservation.reservationID)
|
2019-07-11 13:14:37 +02:00
|
|
|
}
|
|
|
|
|
2021-06-24 02:40:36 +02:00
|
|
|
// enforceReservedValue enforces that the wallet, upon a new channel being
|
|
|
|
// opened, meets the minimum amount of funds required for each advertised anchor
|
|
|
|
// channel.
|
|
|
|
//
|
|
|
|
// We only enforce the reserve if we are contributing funds to the channel. This
|
|
|
|
// is done to still allow incoming channels even though we have no UTXOs
|
|
|
|
// available, as in bootstrapping phases.
|
|
|
|
func (l *LightningWallet) enforceNewReservedValue(fundingIntent chanfunding.Intent,
|
|
|
|
isPublic, hasAnchors bool) error {
|
|
|
|
|
|
|
|
// Only enforce the reserve when an advertised channel is being opened
|
|
|
|
// in which we are contributing funds to. This ensures we never dip
|
|
|
|
// below the reserve.
|
|
|
|
if !isPublic || fundingIntent.LocalFundingAmt() == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-06-17 17:24:29 +02:00
|
|
|
numAnchors, err := l.CurrentNumAnchorChans()
|
2021-06-24 02:40:36 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the to-be-opened channel.
|
|
|
|
if hasAnchors {
|
|
|
|
numAnchors++
|
|
|
|
}
|
|
|
|
|
|
|
|
return l.WithCoinSelectLock(func() error {
|
|
|
|
_, err := l.CheckReservedValue(
|
|
|
|
fundingIntent.Inputs(), fundingIntent.Outputs(),
|
|
|
|
numAnchors,
|
|
|
|
)
|
|
|
|
return err
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-06-17 17:24:29 +02:00
|
|
|
// CurrentNumAnchorChans returns the current number of non-private anchor
|
2021-05-10 10:06:23 +02:00
|
|
|
// channels the wallet should be ready to fee bump if needed.
|
2022-06-17 17:24:29 +02:00
|
|
|
func (l *LightningWallet) CurrentNumAnchorChans() (int, error) {
|
2021-01-11 08:47:38 +01:00
|
|
|
// Count all anchor channels that are open or pending
|
|
|
|
// open, or waiting close.
|
|
|
|
chans, err := l.Cfg.Database.FetchAllChannels()
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var numAnchors int
|
2021-05-10 10:06:23 +02:00
|
|
|
cntChannel := func(c *channeldb.OpenChannel) {
|
|
|
|
// We skip private channels, as we assume they won't be used
|
|
|
|
// for routing.
|
|
|
|
if c.ChannelFlags&lnwire.FFAnnounceChannel == 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Count anchor channels.
|
2021-01-11 08:47:38 +01:00
|
|
|
if c.ChanType.HasAnchors() {
|
|
|
|
numAnchors++
|
|
|
|
}
|
2021-05-10 10:06:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, c := range chans {
|
|
|
|
cntChannel(c)
|
2021-01-11 08:47:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// We also count pending close channels.
|
|
|
|
pendingClosed, err := l.Cfg.Database.FetchClosedChannels(
|
|
|
|
true,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, c := range pendingClosed {
|
|
|
|
c, err := l.Cfg.Database.FetchHistoricalChannel(
|
|
|
|
&c.ChanPoint,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
// We don't have a guarantee that all channels re found
|
|
|
|
// in the historical channels bucket, so we continue.
|
|
|
|
walletLog.Warnf("Unable to fetch historical "+
|
|
|
|
"channel: %v", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2021-05-10 10:06:23 +02:00
|
|
|
cntChannel(c)
|
2021-01-11 08:47:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return numAnchors, nil
|
|
|
|
}
|
|
|
|
|
2021-01-11 11:03:00 +01:00
|
|
|
// CheckReservedValue checks whether publishing a transaction with the given
|
|
|
|
// inputs and outputs would violate the value we reserve in the wallet for
|
|
|
|
// bumping the fee of anchor channels. The numAnchorChans argument should be
|
2024-01-03 21:47:03 +01:00
|
|
|
// set the number of open anchor channels controlled by the wallet after
|
2021-01-11 11:03:00 +01:00
|
|
|
// the transaction has been published.
|
|
|
|
//
|
|
|
|
// If the reserved value is violated, the returned error will be
|
|
|
|
// ErrReservedValueInvalidated. The method will also return the current
|
|
|
|
// reserved value, both in case of success and in case of
|
|
|
|
// ErrReservedValueInvalidated.
|
|
|
|
//
|
|
|
|
// NOTE: This method should only be run with the CoinSelectLock held.
|
|
|
|
func (l *LightningWallet) CheckReservedValue(in []wire.OutPoint,
|
|
|
|
out []*wire.TxOut, numAnchorChans int) (btcutil.Amount, error) {
|
|
|
|
|
2021-02-20 02:41:50 +01:00
|
|
|
// Get all unspent coins in the wallet. We only care about those part of
|
|
|
|
// the wallet's default account as we know we can readily sign for those
|
|
|
|
// at any time.
|
|
|
|
witnessOutputs, err := l.ListUnspentWitnessFromDefaultAccount(
|
|
|
|
0, math.MaxInt32,
|
|
|
|
)
|
2021-01-11 11:03:00 +01:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ourInput := make(map[wire.OutPoint]struct{})
|
|
|
|
for _, op := range in {
|
|
|
|
ourInput[op] = struct{}{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// When crafting a transaction with inputs from the wallet, these coins
|
|
|
|
// will usually be locked in the process, and not be returned when
|
|
|
|
// listing unspents. In this case they have already been deducted from
|
|
|
|
// the wallet balance. In case they haven't been properly locked, we
|
|
|
|
// check whether they are still listed among our unspents and deduct
|
|
|
|
// them.
|
|
|
|
var walletBalance btcutil.Amount
|
|
|
|
for _, in := range witnessOutputs {
|
|
|
|
// Spending an unlocked wallet UTXO, don't add it to the
|
|
|
|
// balance.
|
|
|
|
if _, ok := ourInput[in.OutPoint]; ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
walletBalance += in.Value
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now we go through the outputs of the transaction, if any of the
|
|
|
|
// outputs are paying into the wallet (likely a change output), we add
|
|
|
|
// it to our final balance.
|
|
|
|
for _, txOut := range out {
|
|
|
|
_, addrs, _, err := txscript.ExtractPkScriptAddrs(
|
|
|
|
txOut.PkScript, &l.Cfg.NetParams,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
// Non-standard outputs can safely be skipped because
|
|
|
|
// they're not supported by the wallet.
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, addr := range addrs {
|
|
|
|
if !l.IsOurAddress(addr) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
walletBalance += btcutil.Amount(txOut.Value)
|
|
|
|
|
|
|
|
// We break since we don't want to double count the output.
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We reserve a given amount for each anchor channel.
|
2021-12-22 11:47:13 +01:00
|
|
|
reserved := l.RequiredReserve(uint32(numAnchorChans))
|
2021-01-11 11:03:00 +01:00
|
|
|
|
|
|
|
if walletBalance < reserved {
|
|
|
|
walletLog.Debugf("Reserved value=%v above final "+
|
|
|
|
"walletbalance=%v with %d anchor channels open",
|
|
|
|
reserved, walletBalance, numAnchorChans)
|
|
|
|
return reserved, ErrReservedValueInvalidated
|
|
|
|
}
|
|
|
|
|
|
|
|
return reserved, nil
|
|
|
|
}
|
|
|
|
|
2021-01-10 19:54:49 +01:00
|
|
|
// CheckReservedValueTx calls CheckReservedValue with the inputs and outputs
|
|
|
|
// from the given tx, with the number of anchor channels currently open in the
|
|
|
|
// database.
|
|
|
|
//
|
|
|
|
// NOTE: This method should only be run with the CoinSelectLock held.
|
2021-09-30 13:02:35 +02:00
|
|
|
func (l *LightningWallet) CheckReservedValueTx(req CheckReservedValueTxReq) (
|
|
|
|
btcutil.Amount, error) {
|
2021-01-10 19:54:49 +01:00
|
|
|
|
2022-06-17 17:24:29 +02:00
|
|
|
numAnchors, err := l.CurrentNumAnchorChans()
|
2021-01-10 19:54:49 +01:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var inputs []wire.OutPoint
|
2021-09-30 13:02:35 +02:00
|
|
|
for _, txIn := range req.Tx.TxIn {
|
2021-01-10 19:54:49 +01:00
|
|
|
inputs = append(inputs, txIn.PreviousOutPoint)
|
|
|
|
}
|
|
|
|
|
2021-09-30 13:02:35 +02:00
|
|
|
reservedVal, err := l.CheckReservedValue(
|
|
|
|
inputs, req.Tx.TxOut, numAnchors,
|
|
|
|
)
|
|
|
|
switch {
|
|
|
|
// If the error returned from CheckReservedValue is
|
|
|
|
// ErrReservedValueInvalidated, then it did nonetheless return
|
|
|
|
// the required reserved value and we check for the optional
|
|
|
|
// change index.
|
|
|
|
case errors.Is(err, ErrReservedValueInvalidated):
|
|
|
|
// Without a change index provided there is nothing more to
|
|
|
|
// check and the error is returned.
|
|
|
|
if req.ChangeIndex == nil {
|
|
|
|
return reservedVal, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// If a change index was provided we make only sure that it
|
|
|
|
// would leave sufficient funds for the reserved balance value.
|
|
|
|
//
|
|
|
|
// Note: This is used if a change output index is explicitly set
|
|
|
|
// but that may not be watched by the wallet and therefore is
|
|
|
|
// not picked up by the call to CheckReservedValue above.
|
|
|
|
chIdx := *req.ChangeIndex
|
|
|
|
if chIdx < 0 || chIdx >= len(req.Tx.TxOut) ||
|
|
|
|
req.Tx.TxOut[chIdx].Value < int64(reservedVal) {
|
|
|
|
|
|
|
|
return reservedVal, err
|
|
|
|
}
|
|
|
|
|
|
|
|
case err != nil:
|
|
|
|
return reservedVal, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return reservedVal, nil
|
2021-01-10 19:54:49 +01:00
|
|
|
}
|
|
|
|
|
2019-07-11 13:14:37 +02:00
|
|
|
// initOurContribution initializes the given ChannelReservation with our coins
|
|
|
|
// and change reserved for the channel, and derives the keys to use for this
|
|
|
|
// channel.
|
|
|
|
func (l *LightningWallet) initOurContribution(reservation *ChannelReservation,
|
2019-11-01 05:39:17 +01:00
|
|
|
fundingIntent chanfunding.Intent, nodeAddr net.Addr,
|
2019-11-01 05:44:16 +01:00
|
|
|
nodeID *btcec.PublicKey, keyRing keychain.KeyRing) error {
|
2019-07-11 13:14:37 +02:00
|
|
|
|
2016-10-15 15:18:38 +02:00
|
|
|
// Grab the mutex on the ChannelReservation to ensure thread-safety
|
2015-12-19 04:39:51 +01:00
|
|
|
reservation.Lock()
|
|
|
|
defer reservation.Unlock()
|
2015-11-14 20:52:07 +01:00
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
// At this point, if we have a funding intent, we'll use it to populate
|
|
|
|
// the existing reservation state entries for our coin selection.
|
|
|
|
if fundingIntent != nil {
|
|
|
|
if intent, ok := fundingIntent.(*chanfunding.FullIntent); ok {
|
|
|
|
for _, coin := range intent.InputCoins {
|
|
|
|
reservation.ourContribution.Inputs = append(
|
|
|
|
reservation.ourContribution.Inputs,
|
|
|
|
&wire.TxIn{
|
|
|
|
PreviousOutPoint: coin.OutPoint,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
reservation.ourContribution.ChangeOutputs = intent.ChangeOutputs
|
|
|
|
}
|
|
|
|
|
|
|
|
reservation.fundingIntent = fundingIntent
|
2015-10-27 23:50:30 +01:00
|
|
|
}
|
|
|
|
|
2019-07-11 13:14:37 +02:00
|
|
|
reservation.nodeAddr = nodeAddr
|
|
|
|
reservation.partialState.IdentityPub = nodeID
|
2019-07-11 13:14:36 +02:00
|
|
|
|
2019-07-11 13:14:37 +02:00
|
|
|
var err error
|
2019-11-01 05:44:16 +01:00
|
|
|
reservation.ourContribution.MultiSigKey, err = keyRing.DeriveNextKey(
|
2018-02-18 00:20:41 +01:00
|
|
|
keychain.KeyFamilyMultiSig,
|
|
|
|
)
|
2017-07-30 04:20:08 +02:00
|
|
|
if err != nil {
|
2019-07-11 13:14:37 +02:00
|
|
|
return err
|
2017-07-30 04:20:08 +02:00
|
|
|
}
|
2019-11-01 05:44:16 +01:00
|
|
|
reservation.ourContribution.RevocationBasePoint, err = keyRing.DeriveNextKey(
|
2018-02-18 00:20:41 +01:00
|
|
|
keychain.KeyFamilyRevocationBase,
|
|
|
|
)
|
2017-07-30 04:20:08 +02:00
|
|
|
if err != nil {
|
2019-07-11 13:14:37 +02:00
|
|
|
return err
|
2017-07-30 04:20:08 +02:00
|
|
|
}
|
2019-11-01 05:44:16 +01:00
|
|
|
reservation.ourContribution.HtlcBasePoint, err = keyRing.DeriveNextKey(
|
2018-02-18 00:20:41 +01:00
|
|
|
keychain.KeyFamilyHtlcBase,
|
|
|
|
)
|
2017-11-15 05:32:39 +01:00
|
|
|
if err != nil {
|
2019-07-11 13:14:37 +02:00
|
|
|
return err
|
2017-11-15 05:32:39 +01:00
|
|
|
}
|
2019-11-01 05:44:16 +01:00
|
|
|
reservation.ourContribution.PaymentBasePoint, err = keyRing.DeriveNextKey(
|
2018-02-18 00:20:41 +01:00
|
|
|
keychain.KeyFamilyPaymentBase,
|
|
|
|
)
|
2015-12-21 00:13:14 +01:00
|
|
|
if err != nil {
|
2019-07-11 13:14:37 +02:00
|
|
|
return err
|
2015-12-21 00:13:14 +01:00
|
|
|
}
|
2019-11-01 05:44:16 +01:00
|
|
|
reservation.ourContribution.DelayBasePoint, err = keyRing.DeriveNextKey(
|
2018-02-18 00:20:41 +01:00
|
|
|
keychain.KeyFamilyDelayBase,
|
|
|
|
)
|
2015-11-05 21:36:19 +01:00
|
|
|
if err != nil {
|
2019-07-11 13:14:37 +02:00
|
|
|
return err
|
2015-11-05 21:36:19 +01:00
|
|
|
}
|
2017-07-30 04:20:08 +02:00
|
|
|
|
2020-10-21 19:34:05 +02:00
|
|
|
// With the above keys created, we'll also need to initialize our
|
2023-01-20 02:15:12 +01:00
|
|
|
// revocation tree state, and from that generate the per-commitment
|
|
|
|
// point.
|
|
|
|
producer, taprootNonceProducer, err := l.nextRevocationProducer(
|
|
|
|
reservation, keyRing,
|
|
|
|
)
|
2016-03-24 08:01:35 +01:00
|
|
|
if err != nil {
|
2019-07-11 13:14:37 +02:00
|
|
|
return err
|
2016-03-24 08:01:35 +01:00
|
|
|
}
|
2017-07-30 04:20:08 +02:00
|
|
|
|
|
|
|
firstPreimage, err := producer.AtIndex(0)
|
|
|
|
if err != nil {
|
2019-07-11 13:14:37 +02:00
|
|
|
return err
|
2017-07-30 04:20:08 +02:00
|
|
|
}
|
2019-01-16 15:47:43 +01:00
|
|
|
reservation.ourContribution.FirstCommitmentPoint = input.ComputeCommitmentPoint(
|
2017-07-30 04:20:08 +02:00
|
|
|
firstPreimage[:],
|
|
|
|
)
|
|
|
|
|
|
|
|
reservation.partialState.RevocationProducer = producer
|
2024-05-09 23:32:45 +02:00
|
|
|
reservation.ourContribution.CommitmentParams.DustLimit =
|
2024-02-05 20:40:36 +01:00
|
|
|
DustLimitUnknownWitness()
|
2017-07-30 04:20:08 +02:00
|
|
|
|
2023-01-20 02:15:12 +01:00
|
|
|
// If taproot channels are active, then we'll generate our verification
|
|
|
|
// nonce here. We'll use this nonce to verify the signature for our
|
|
|
|
// local commitment transaction. If we need to force close, then this
|
|
|
|
// is also what'll be used to sign that transaction.
|
|
|
|
if reservation.partialState.ChanType.IsTaproot() {
|
|
|
|
firstNoncePreimage, err := taprootNonceProducer.AtIndex(0)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// As we'd like the local nonce we send over to be generated
|
|
|
|
// deterministically, we'll provide a custom reader that
|
|
|
|
// actually just uses our sha-chain pre-image as the primary
|
|
|
|
// randomness source.
|
|
|
|
shaChainRand := musig2.WithCustomRand(
|
|
|
|
bytes.NewBuffer(firstNoncePreimage[:]),
|
|
|
|
)
|
|
|
|
pubKeyOpt := musig2.WithPublicKey(
|
|
|
|
reservation.ourContribution.MultiSigKey.PubKey,
|
|
|
|
)
|
|
|
|
reservation.ourContribution.LocalNonce, err = musig2.GenNonces(
|
|
|
|
pubKeyOpt, shaChainRand,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-11 13:14:37 +02:00
|
|
|
return nil
|
2015-11-05 21:36:19 +01:00
|
|
|
}
|
|
|
|
|
2015-12-28 21:14:00 +01:00
|
|
|
// handleFundingReserveCancel cancels an existing channel reservation. As part
|
|
|
|
// of the cancellation, outputs previously selected as inputs for the funding
|
|
|
|
// transaction via coin selection are freed allowing future reservations to
|
|
|
|
// include them.
|
2015-11-05 21:36:19 +01:00
|
|
|
func (l *LightningWallet) handleFundingCancelRequest(req *fundingReserveCancelMsg) {
|
|
|
|
l.limboMtx.Lock()
|
|
|
|
defer l.limboMtx.Unlock()
|
|
|
|
|
|
|
|
pendingReservation, ok := l.fundingLimbo[req.pendingFundingID]
|
|
|
|
if !ok {
|
2017-12-14 02:15:36 +01:00
|
|
|
// TODO(roasbeef): make new error, "unknown funding state" or something
|
2017-09-25 20:25:58 +02:00
|
|
|
req.err <- fmt.Errorf("attempted to cancel non-existent funding state")
|
2015-11-05 21:36:19 +01:00
|
|
|
return
|
2015-10-27 23:50:30 +01:00
|
|
|
}
|
2015-11-05 21:36:19 +01:00
|
|
|
|
2017-12-14 02:15:36 +01:00
|
|
|
// Grab the mutex on the ChannelReservation to ensure thread-safety
|
2015-11-14 20:52:07 +01:00
|
|
|
pendingReservation.Lock()
|
|
|
|
defer pendingReservation.Unlock()
|
|
|
|
|
2022-01-13 17:29:43 +01:00
|
|
|
// Mark all previously locked outpoints as usable for future funding
|
2015-11-05 21:36:19 +01:00
|
|
|
// requests.
|
2015-12-23 05:31:17 +01:00
|
|
|
for _, unusedInput := range pendingReservation.ourContribution.Inputs {
|
2016-08-13 00:50:47 +02:00
|
|
|
delete(l.lockedOutPoints, unusedInput.PreviousOutPoint)
|
2024-02-22 00:04:56 +01:00
|
|
|
_ = l.ReleaseOutput(
|
|
|
|
chanfunding.LndInternalLockID,
|
|
|
|
unusedInput.PreviousOutPoint,
|
|
|
|
)
|
2015-11-05 21:36:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
delete(l.fundingLimbo, req.pendingFundingID)
|
|
|
|
|
2019-11-01 05:47:27 +01:00
|
|
|
pid := pendingReservation.pendingChanID
|
2021-01-11 08:47:38 +01:00
|
|
|
delete(l.reservationIDs, pid)
|
2019-11-01 05:47:27 +01:00
|
|
|
|
|
|
|
l.intentMtx.Lock()
|
|
|
|
if intent, ok := l.fundingIntents[pid]; ok {
|
|
|
|
intent.Cancel()
|
|
|
|
|
|
|
|
delete(l.fundingIntents, pendingReservation.pendingChanID)
|
|
|
|
}
|
|
|
|
l.intentMtx.Unlock()
|
|
|
|
|
2015-11-05 21:36:19 +01:00
|
|
|
req.err <- nil
|
2015-10-27 23:50:30 +01:00
|
|
|
}
|
|
|
|
|
2024-03-17 21:53:38 +01:00
|
|
|
// createCommitOpts is a struct that holds the options for creating a new
|
|
|
|
// commitment transaction.
|
|
|
|
type createCommitOpts struct {
|
2024-04-05 02:26:03 +02:00
|
|
|
localAuxLeaves fn.Option[CommitAuxLeaves]
|
|
|
|
remoteAuxLeaves fn.Option[CommitAuxLeaves]
|
2024-03-17 21:53:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// defaultCommitOpts returns a new createCommitOpts with default values.
|
|
|
|
func defaultCommitOpts() createCommitOpts {
|
|
|
|
return createCommitOpts{}
|
|
|
|
}
|
|
|
|
|
2024-04-05 02:26:03 +02:00
|
|
|
// WithAuxLeaves is a functional option that can be used to set the aux leaves
|
|
|
|
// for a new commitment transaction.
|
|
|
|
func WithAuxLeaves(localLeaves,
|
|
|
|
remoteLeaves fn.Option[CommitAuxLeaves]) CreateCommitOpt {
|
|
|
|
|
|
|
|
return func(o *createCommitOpts) {
|
|
|
|
o.localAuxLeaves = localLeaves
|
|
|
|
o.remoteAuxLeaves = remoteLeaves
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-17 21:53:38 +01:00
|
|
|
// CreateCommitOpt is a functional option that can be used to modify the way a
|
|
|
|
// new commitment transaction is created.
|
|
|
|
type CreateCommitOpt func(*createCommitOpts)
|
|
|
|
|
2017-07-30 04:18:46 +02:00
|
|
|
// CreateCommitmentTxns is a helper function that creates the initial
|
|
|
|
// commitment transaction for both parties. This function is used during the
|
|
|
|
// initial funding workflow as both sides must generate a signature for the
|
|
|
|
// remote party's commitment transaction, and verify the signature for their
|
|
|
|
// version of the commitment transaction.
|
|
|
|
func CreateCommitmentTxns(localBalance, remoteBalance btcutil.Amount,
|
|
|
|
ourChanCfg, theirChanCfg *channeldb.ChannelConfig,
|
|
|
|
localCommitPoint, remoteCommitPoint *btcec.PublicKey,
|
2021-07-15 02:16:13 +02:00
|
|
|
fundingTxIn wire.TxIn, chanType channeldb.ChannelType, initiator bool,
|
2024-03-17 21:53:38 +01:00
|
|
|
leaseExpiry uint32, opts ...CreateCommitOpt) (*wire.MsgTx, *wire.MsgTx,
|
|
|
|
error) {
|
|
|
|
|
|
|
|
options := defaultCommitOpts()
|
|
|
|
for _, optFunc := range opts {
|
|
|
|
optFunc(&options)
|
|
|
|
}
|
2017-07-30 04:18:46 +02:00
|
|
|
|
2019-09-17 04:06:19 +02:00
|
|
|
localCommitmentKeys := DeriveCommitmentKeys(
|
2024-07-31 01:44:18 +02:00
|
|
|
localCommitPoint, lntypes.Local, chanType, ourChanCfg,
|
|
|
|
theirChanCfg,
|
2019-08-01 05:10:45 +02:00
|
|
|
)
|
2019-09-17 04:06:19 +02:00
|
|
|
remoteCommitmentKeys := DeriveCommitmentKeys(
|
2024-07-31 01:44:18 +02:00
|
|
|
remoteCommitPoint, lntypes.Remote, chanType, ourChanCfg,
|
|
|
|
theirChanCfg,
|
2019-08-01 05:10:45 +02:00
|
|
|
)
|
2017-07-30 04:18:46 +02:00
|
|
|
|
2020-01-06 11:42:03 +01:00
|
|
|
ourCommitTx, err := CreateCommitTx(
|
2020-01-06 11:42:04 +01:00
|
|
|
chanType, fundingTxIn, localCommitmentKeys, ourChanCfg,
|
2021-07-15 02:16:13 +02:00
|
|
|
theirChanCfg, localBalance, remoteBalance, 0, initiator,
|
2024-04-05 02:26:03 +02:00
|
|
|
leaseExpiry, options.localAuxLeaves,
|
2020-01-06 11:42:03 +01:00
|
|
|
)
|
2017-07-30 04:18:46 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
otxn := btcutil.NewTx(ourCommitTx)
|
|
|
|
if err := blockchain.CheckTransactionSanity(otxn); err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
|
2020-01-06 11:42:03 +01:00
|
|
|
theirCommitTx, err := CreateCommitTx(
|
2020-01-06 11:42:04 +01:00
|
|
|
chanType, fundingTxIn, remoteCommitmentKeys, theirChanCfg,
|
2021-07-15 02:16:13 +02:00
|
|
|
ourChanCfg, remoteBalance, localBalance, 0, !initiator,
|
2024-04-05 02:26:03 +02:00
|
|
|
leaseExpiry, options.remoteAuxLeaves,
|
2020-01-06 11:42:03 +01:00
|
|
|
)
|
2017-07-30 04:18:46 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ttxn := btcutil.NewTx(theirCommitTx)
|
|
|
|
if err := blockchain.CheckTransactionSanity(ttxn); err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return ourCommitTx, theirCommitTx, nil
|
|
|
|
}
|
|
|
|
|
2016-10-24 03:42:03 +02:00
|
|
|
// handleContributionMsg processes the second workflow step for the lifetime of
|
|
|
|
// a channel reservation. Upon completion, the reservation will carry a
|
|
|
|
// completed funding transaction (minus the counterparty's input signatures),
|
|
|
|
// both versions of the commitment transaction, and our signature for their
|
|
|
|
// version of the commitment transaction.
|
2015-12-21 22:53:34 +01:00
|
|
|
func (l *LightningWallet) handleContributionMsg(req *addContributionMsg) {
|
2017-07-30 04:20:08 +02:00
|
|
|
|
2015-11-27 07:48:20 +01:00
|
|
|
l.limboMtx.Lock()
|
2015-11-05 21:36:19 +01:00
|
|
|
pendingReservation, ok := l.fundingLimbo[req.pendingFundingID]
|
2015-11-27 07:48:20 +01:00
|
|
|
l.limboMtx.Unlock()
|
2015-11-05 21:36:19 +01:00
|
|
|
if !ok {
|
2017-09-25 20:25:58 +02:00
|
|
|
req.err <- fmt.Errorf("attempted to update non-existent funding state")
|
2015-11-05 21:36:19 +01:00
|
|
|
return
|
|
|
|
}
|
2015-11-05 21:32:15 +01:00
|
|
|
|
2018-05-22 01:52:51 +02:00
|
|
|
// Grab the mutex on the ChannelReservation to ensure thread-safety
|
2015-11-14 20:52:07 +01:00
|
|
|
pendingReservation.Lock()
|
|
|
|
defer pendingReservation.Unlock()
|
|
|
|
|
2021-06-09 22:23:35 +02:00
|
|
|
// If UpfrontShutdownScript is set, validate that it is a valid script.
|
|
|
|
shutdown := req.contribution.UpfrontShutdown
|
|
|
|
if len(shutdown) > 0 {
|
|
|
|
// Validate the shutdown script.
|
2022-06-10 20:16:02 +02:00
|
|
|
if !ValidateUpfrontShutdown(shutdown, &l.Cfg.NetParams) {
|
2021-06-09 22:23:35 +02:00
|
|
|
req.err <- fmt.Errorf("invalid shutdown script")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-28 21:14:00 +01:00
|
|
|
// Some temporary variables to cut down on the resolution verbosity.
|
2015-12-23 05:31:17 +01:00
|
|
|
pendingReservation.theirContribution = req.contribution
|
|
|
|
theirContribution := req.contribution
|
|
|
|
ourContribution := pendingReservation.ourContribution
|
2015-11-05 21:32:15 +01:00
|
|
|
|
2021-09-23 22:37:38 +02:00
|
|
|
// Perform bounds-checking on both ChannelReserve and DustLimit
|
|
|
|
// parameters.
|
|
|
|
if !pendingReservation.validateReserveBounds() {
|
|
|
|
req.err <- fmt.Errorf("invalid reserve and dust bounds")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
var (
|
|
|
|
chanPoint *wire.OutPoint
|
|
|
|
err error
|
2018-02-18 00:20:41 +01:00
|
|
|
)
|
2015-12-24 19:42:29 +01:00
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
// At this point, we can now construct our channel point. Depending on
|
|
|
|
// which type of intent we obtained from our chanfunding.Assembler,
|
|
|
|
// we'll carry out a distinct set of steps.
|
|
|
|
switch fundingIntent := pendingReservation.fundingIntent.(type) {
|
2020-03-31 09:13:16 +02:00
|
|
|
// The transaction was created outside of the wallet and might already
|
|
|
|
// be published. Nothing left to do other than using the correct
|
|
|
|
// outpoint.
|
2019-11-01 05:44:16 +01:00
|
|
|
case *chanfunding.ShimIntent:
|
|
|
|
chanPoint, err = fundingIntent.ChanPoint()
|
|
|
|
if err != nil {
|
2024-02-26 12:19:38 +01:00
|
|
|
req.err <- fmt.Errorf("unable to obtain chan point: %w",
|
|
|
|
err)
|
2019-11-01 05:44:16 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
pendingReservation.partialState.FundingOutpoint = *chanPoint
|
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
// The user has signaled that they want to use a PSBT to construct the
|
|
|
|
// funding transaction. Because we now have the multisig keys from both
|
|
|
|
// parties, we can create the multisig script that needs to be funded
|
|
|
|
// and then pause the process until the user supplies the PSBT
|
|
|
|
// containing the eventual funding transaction.
|
|
|
|
case *chanfunding.PsbtIntent:
|
|
|
|
if fundingIntent.PendingPsbt != nil {
|
|
|
|
req.err <- fmt.Errorf("PSBT funding already in" +
|
|
|
|
"progress")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now that we know our contribution, we can bind both the local
|
|
|
|
// and remote key which will be needed to calculate the multisig
|
|
|
|
// funding output in a next step.
|
|
|
|
pendingChanID := pendingReservation.pendingChanID
|
2024-04-18 03:43:01 +02:00
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
walletLog.Debugf("Advancing PSBT funding flow for "+
|
|
|
|
"pending_id(%x), binding keys local_key=%v, "+
|
|
|
|
"remote_key=%x", pendingChanID,
|
|
|
|
&ourContribution.MultiSigKey,
|
|
|
|
theirContribution.MultiSigKey.PubKey.SerializeCompressed())
|
2024-04-18 03:43:01 +02:00
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
fundingIntent.BindKeys(
|
|
|
|
&ourContribution.MultiSigKey,
|
|
|
|
theirContribution.MultiSigKey.PubKey,
|
|
|
|
)
|
|
|
|
|
2024-04-18 03:43:01 +02:00
|
|
|
// We might have a tapscript root, so we'll bind that now to
|
|
|
|
// ensure we make the proper funding output.
|
|
|
|
fundingIntent.BindTapscriptRoot(
|
|
|
|
pendingReservation.partialState.TapscriptRoot,
|
|
|
|
)
|
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
// Exit early because we can't continue the funding flow yet.
|
|
|
|
req.err <- &PsbtFundingRequired{
|
|
|
|
Intent: fundingIntent,
|
|
|
|
}
|
|
|
|
return
|
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
case *chanfunding.FullIntent:
|
|
|
|
// Now that we know their public key, we can bind theirs as
|
|
|
|
// well as ours to the funding intent.
|
|
|
|
fundingIntent.BindKeys(
|
|
|
|
&pendingReservation.ourContribution.MultiSigKey,
|
|
|
|
theirContribution.MultiSigKey.PubKey,
|
|
|
|
)
|
2015-11-05 21:36:19 +01:00
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
// With our keys bound, we can now construct+sign the final
|
|
|
|
// funding transaction and also obtain the chanPoint that
|
|
|
|
// creates the channel.
|
|
|
|
fundingTx, err := fundingIntent.CompileFundingTx(
|
|
|
|
theirContribution.Inputs,
|
|
|
|
theirContribution.ChangeOutputs,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
req.err <- fmt.Errorf("unable to construct funding "+
|
|
|
|
"tx: %v", err)
|
2015-11-05 21:36:19 +01:00
|
|
|
return
|
|
|
|
}
|
2019-11-01 05:39:17 +01:00
|
|
|
chanPoint, err = fundingIntent.ChanPoint()
|
2015-10-27 23:50:30 +01:00
|
|
|
if err != nil {
|
2019-11-01 05:39:17 +01:00
|
|
|
req.err <- fmt.Errorf("unable to obtain chan "+
|
|
|
|
"point: %v", err)
|
2015-11-05 21:36:19 +01:00
|
|
|
return
|
2015-10-27 23:50:30 +01:00
|
|
|
}
|
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
// Finally, we'll populate the relevant information in our
|
|
|
|
// pendingReservation so the rest of the funding flow can
|
|
|
|
// continue as normal.
|
|
|
|
pendingReservation.fundingTx = fundingTx
|
|
|
|
pendingReservation.partialState.FundingOutpoint = *chanPoint
|
|
|
|
pendingReservation.ourFundingInputScripts = make(
|
|
|
|
[]*input.Script, 0, len(ourContribution.Inputs),
|
2016-05-04 04:49:58 +02:00
|
|
|
)
|
2019-11-01 05:39:17 +01:00
|
|
|
for _, txIn := range fundingTx.TxIn {
|
2024-08-08 15:53:27 +02:00
|
|
|
_, err := l.FetchOutpointInfo(&txIn.PreviousOutPoint)
|
2019-11-01 05:39:17 +01:00
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
2015-10-27 23:50:30 +01:00
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
pendingReservation.ourFundingInputScripts = append(
|
|
|
|
pendingReservation.ourFundingInputScripts,
|
|
|
|
&input.Script{
|
|
|
|
Witness: txIn.Witness,
|
|
|
|
SigScript: txIn.SignatureScript,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
2016-06-21 19:37:55 +02:00
|
|
|
|
2021-12-12 20:56:20 +01:00
|
|
|
walletLog.Tracef("Funding tx for ChannelPoint(%v) "+
|
2019-11-01 05:39:17 +01:00
|
|
|
"generated: %v", chanPoint, spew.Sdump(fundingTx))
|
|
|
|
}
|
2018-01-31 23:00:01 +01:00
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
// If we landed here and didn't exit early, it means we already have
|
|
|
|
// the channel point ready. We can jump directly to the next step.
|
|
|
|
l.handleChanPointReady(&continueContributionMsg{
|
|
|
|
pendingFundingID: req.pendingFundingID,
|
|
|
|
err: req.err,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-01-20 02:15:12 +01:00
|
|
|
// genMusigSession generates a new musig2 pair session that we can use to sign
|
|
|
|
// the commitment transaction for the remote party, and verify their incoming
|
|
|
|
// partial signature.
|
|
|
|
func genMusigSession(ourContribution, theirContribution *ChannelContribution,
|
2024-04-18 03:43:01 +02:00
|
|
|
signer input.MuSig2Signer, fundingOutput *wire.TxOut,
|
|
|
|
tapscriptRoot fn.Option[chainhash.Hash]) *MusigPairSession {
|
2023-01-20 02:15:12 +01:00
|
|
|
|
|
|
|
return NewMusigPairSession(&MusigSessionCfg{
|
2024-04-18 03:43:01 +02:00
|
|
|
LocalKey: ourContribution.MultiSigKey,
|
|
|
|
RemoteKey: theirContribution.MultiSigKey,
|
|
|
|
LocalNonce: *ourContribution.LocalNonce,
|
|
|
|
RemoteNonce: *theirContribution.LocalNonce,
|
|
|
|
Signer: signer,
|
|
|
|
InputTxOut: fundingOutput,
|
|
|
|
TapscriptTweak: tapscriptRoot,
|
2023-01-20 02:15:12 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// signCommitTx generates a valid input.Signature to send to the remote party
|
|
|
|
// for their version of the commitment transaction. For regular channels, this
|
|
|
|
// will be a normal ECDSA signature. For taproot channels, this will instead be
|
|
|
|
// a musig2 partial signature that also includes the nonce used to generate it.
|
|
|
|
func (l *LightningWallet) signCommitTx(pendingReservation *ChannelReservation,
|
|
|
|
commitTx *wire.MsgTx, fundingOutput *wire.TxOut,
|
|
|
|
fundingWitnessScript []byte) (input.Signature, error) {
|
|
|
|
|
|
|
|
ourContribution := pendingReservation.ourContribution
|
|
|
|
theirContribution := pendingReservation.theirContribution
|
|
|
|
|
|
|
|
var (
|
|
|
|
sigTheirCommit input.Signature
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
switch {
|
|
|
|
// For regular channels, we can just send over a normal ECDSA signature
|
|
|
|
// w/o any extra steps.
|
|
|
|
case !pendingReservation.partialState.ChanType.IsTaproot():
|
|
|
|
ourKey := ourContribution.MultiSigKey
|
|
|
|
signDesc := input.SignDescriptor{
|
|
|
|
WitnessScript: fundingWitnessScript,
|
|
|
|
KeyDesc: ourKey,
|
|
|
|
Output: fundingOutput,
|
|
|
|
HashType: txscript.SigHashAll,
|
|
|
|
SigHashes: input.NewTxSigHashesV0Only(
|
|
|
|
commitTx,
|
|
|
|
),
|
|
|
|
InputIndex: 0,
|
|
|
|
}
|
|
|
|
sigTheirCommit, err = l.Cfg.Signer.SignOutputRaw(
|
|
|
|
commitTx, &signDesc,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// If this is a taproot channel, then we'll need to create an initial
|
|
|
|
// musig2 session here as we'll be sending over a _partial_ signature.
|
|
|
|
default:
|
|
|
|
// We're now ready to sign the first commitment. However, we'll
|
|
|
|
// only create the session if that hasn't been done already.
|
|
|
|
if pendingReservation.musigSessions == nil {
|
|
|
|
musigSessions := genMusigSession(
|
|
|
|
ourContribution, theirContribution,
|
|
|
|
l.Cfg.Signer, fundingOutput,
|
2024-04-18 03:43:01 +02:00
|
|
|
pendingReservation.partialState.TapscriptRoot,
|
2023-01-20 02:15:12 +01:00
|
|
|
)
|
|
|
|
pendingReservation.musigSessions = musigSessions
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now that we have the funding outpoint, we'll generate a
|
|
|
|
// musig2 signature for their version of the commitment
|
|
|
|
// transaction. We use the remote session as this is for the
|
|
|
|
// remote commitment transaction.
|
|
|
|
musigSessions := pendingReservation.musigSessions
|
|
|
|
partialSig, err := musigSessions.RemoteSession.SignCommit(
|
|
|
|
commitTx,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("unable to sign "+
|
|
|
|
"commitment: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
sigTheirCommit = partialSig
|
|
|
|
}
|
|
|
|
|
|
|
|
return sigTheirCommit, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// handleChanPointReady continues the funding process once the channel point is
|
|
|
|
// known and the funding transaction can be completed.
|
2020-03-31 09:13:16 +02:00
|
|
|
func (l *LightningWallet) handleChanPointReady(req *continueContributionMsg) {
|
|
|
|
l.limboMtx.Lock()
|
|
|
|
pendingReservation, ok := l.fundingLimbo[req.pendingFundingID]
|
|
|
|
l.limboMtx.Unlock()
|
|
|
|
if !ok {
|
|
|
|
req.err <- fmt.Errorf("attempted to update non-existent " +
|
|
|
|
"funding state")
|
|
|
|
return
|
|
|
|
}
|
2023-01-20 02:15:12 +01:00
|
|
|
|
2024-04-18 08:35:44 +02:00
|
|
|
chanState := pendingReservation.partialState
|
|
|
|
|
|
|
|
// If we have an aux funding desc, then we can use it to populate some
|
|
|
|
// of the optional, but opaque TLV blobs we'll carry for the channel.
|
|
|
|
chanState.CustomBlob = fn.MapOption(func(desc AuxFundingDesc) tlv.Blob {
|
|
|
|
return desc.CustomFundingBlob
|
|
|
|
})(req.auxFundingDesc)
|
|
|
|
|
|
|
|
chanState.LocalCommitment.CustomBlob = fn.MapOption(
|
|
|
|
func(desc AuxFundingDesc) tlv.Blob {
|
|
|
|
return desc.CustomLocalCommitBlob
|
|
|
|
},
|
|
|
|
)(req.auxFundingDesc)
|
|
|
|
|
|
|
|
chanState.RemoteCommitment.CustomBlob = fn.MapOption(
|
|
|
|
func(desc AuxFundingDesc) tlv.Blob {
|
|
|
|
return desc.CustomRemoteCommitBlob
|
|
|
|
},
|
|
|
|
)(req.auxFundingDesc)
|
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
ourContribution := pendingReservation.ourContribution
|
|
|
|
theirContribution := pendingReservation.theirContribution
|
|
|
|
chanPoint := pendingReservation.partialState.FundingOutpoint
|
|
|
|
|
|
|
|
// If we're in the PSBT funding flow, we now should have everything that
|
|
|
|
// is needed to construct and publish the full funding transaction.
|
|
|
|
intent := pendingReservation.fundingIntent
|
|
|
|
if psbtIntent, ok := intent.(*chanfunding.PsbtIntent); ok {
|
2021-06-07 11:16:36 +02:00
|
|
|
// With our keys bound, we can now construct and possibly sign
|
|
|
|
// the final funding transaction and also obtain the chanPoint
|
|
|
|
// that creates the channel. We _have_ to call CompileFundingTx
|
|
|
|
// even if we don't publish ourselves as that sets the actual
|
|
|
|
// funding outpoint in stone for this channel.
|
2020-03-31 09:13:16 +02:00
|
|
|
fundingTx, err := psbtIntent.CompileFundingTx()
|
|
|
|
if err != nil {
|
|
|
|
req.err <- fmt.Errorf("unable to construct funding "+
|
|
|
|
"tx: %v", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
chanPointPtr, err := psbtIntent.ChanPoint()
|
|
|
|
if err != nil {
|
|
|
|
req.err <- fmt.Errorf("unable to obtain chan "+
|
|
|
|
"point: %v", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
pendingReservation.partialState.FundingOutpoint = *chanPointPtr
|
|
|
|
chanPoint = *chanPointPtr
|
2021-06-07 11:16:36 +02:00
|
|
|
|
|
|
|
// Finally, we'll populate the relevant information in our
|
|
|
|
// pendingReservation so the rest of the funding flow can
|
|
|
|
// continue as normal in case we are going to publish ourselves.
|
|
|
|
if psbtIntent.ShouldPublishFundingTX() {
|
|
|
|
pendingReservation.fundingTx = fundingTx
|
|
|
|
pendingReservation.ourFundingInputScripts = make(
|
|
|
|
[]*input.Script, 0, len(ourContribution.Inputs),
|
2020-03-31 09:13:16 +02:00
|
|
|
)
|
2021-06-07 11:16:36 +02:00
|
|
|
for _, txIn := range fundingTx.TxIn {
|
|
|
|
pendingReservation.ourFundingInputScripts = append(
|
|
|
|
pendingReservation.ourFundingInputScripts,
|
|
|
|
&input.Script{
|
|
|
|
Witness: txIn.Witness,
|
|
|
|
SigScript: txIn.SignatureScript,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
2020-03-31 09:13:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-21 22:53:34 +01:00
|
|
|
// Initialize an empty sha-chain for them, tracking the current pending
|
2017-01-13 06:01:50 +01:00
|
|
|
// revocation hash (we don't yet know the preimage so we can't add it
|
2015-12-21 22:53:34 +01:00
|
|
|
// to the chain).
|
2016-12-14 15:01:48 +01:00
|
|
|
s := shachain.NewRevocationStore()
|
|
|
|
pendingReservation.partialState.RevocationStore = s
|
2016-08-13 00:43:16 +02:00
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
// Store their current commitment point. We'll need this after the
|
|
|
|
// first state transition in order to verify the authenticity of the
|
|
|
|
// revocation.
|
|
|
|
chanState.RemoteCurrentRevocation = theirContribution.FirstCommitmentPoint
|
2015-12-21 22:53:34 +01:00
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
// Create the txin to our commitment transaction; required to construct
|
2016-06-21 19:37:55 +02:00
|
|
|
// the commitment transactions.
|
2017-12-22 19:26:16 +01:00
|
|
|
fundingTxIn := wire.TxIn{
|
2020-03-31 09:13:16 +02:00
|
|
|
PreviousOutPoint: chanPoint,
|
2017-07-30 04:20:08 +02:00
|
|
|
}
|
2015-12-21 22:53:34 +01:00
|
|
|
|
|
|
|
// With the funding tx complete, create both commitment transactions.
|
2017-11-10 23:18:41 +01:00
|
|
|
localBalance := pendingReservation.partialState.LocalCommitment.LocalBalance.ToSatoshis()
|
|
|
|
remoteBalance := pendingReservation.partialState.LocalCommitment.RemoteBalance.ToSatoshis()
|
2021-07-15 02:16:13 +02:00
|
|
|
var leaseExpiry uint32
|
|
|
|
if pendingReservation.partialState.ChanType.HasLeaseExpiration() {
|
|
|
|
leaseExpiry = pendingReservation.partialState.ThawHeight
|
|
|
|
}
|
2024-04-05 02:26:03 +02:00
|
|
|
|
|
|
|
localAuxLeaves := fn.MapOption(
|
|
|
|
func(desc AuxFundingDesc) CommitAuxLeaves {
|
|
|
|
return desc.LocalInitAuxLeaves
|
|
|
|
},
|
|
|
|
)(req.auxFundingDesc)
|
|
|
|
remoteAuxLeaves := fn.MapOption(
|
|
|
|
func(desc AuxFundingDesc) CommitAuxLeaves {
|
|
|
|
return desc.RemoteInitAuxLeaves
|
|
|
|
},
|
|
|
|
)(req.auxFundingDesc)
|
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
ourCommitTx, theirCommitTx, err := CreateCommitmentTxns(
|
|
|
|
localBalance, remoteBalance, ourContribution.ChannelConfig,
|
|
|
|
theirContribution.ChannelConfig,
|
|
|
|
ourContribution.FirstCommitmentPoint,
|
|
|
|
theirContribution.FirstCommitmentPoint, fundingTxIn,
|
2020-01-06 11:42:04 +01:00
|
|
|
pendingReservation.partialState.ChanType,
|
2021-07-15 02:16:13 +02:00
|
|
|
pendingReservation.partialState.IsInitiator, leaseExpiry,
|
2024-04-05 02:26:03 +02:00
|
|
|
WithAuxLeaves(localAuxLeaves, remoteAuxLeaves),
|
2017-07-30 04:20:08 +02:00
|
|
|
)
|
2015-12-21 22:53:34 +01:00
|
|
|
if err != nil {
|
|
|
|
req.err <- err
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-11-16 21:54:27 +01:00
|
|
|
// With both commitment transactions constructed, generate the state
|
2017-09-25 20:25:58 +02:00
|
|
|
// obfuscator then use it to encode the current state number within
|
2017-07-30 04:20:08 +02:00
|
|
|
// both commitment transactions.
|
2017-09-25 20:25:58 +02:00
|
|
|
var stateObfuscator [StateHintSize]byte
|
2019-08-01 05:16:52 +02:00
|
|
|
if chanState.ChanType.IsSingleFunder() {
|
2018-01-18 22:45:30 +01:00
|
|
|
stateObfuscator = DeriveStateHintObfuscator(
|
2018-02-18 00:20:41 +01:00
|
|
|
ourContribution.PaymentBasePoint.PubKey,
|
|
|
|
theirContribution.PaymentBasePoint.PubKey,
|
2017-07-30 04:20:08 +02:00
|
|
|
)
|
|
|
|
} else {
|
2018-02-18 00:20:41 +01:00
|
|
|
ourSer := ourContribution.PaymentBasePoint.PubKey.SerializeCompressed()
|
|
|
|
theirSer := theirContribution.PaymentBasePoint.PubKey.SerializeCompressed()
|
2017-07-30 04:20:08 +02:00
|
|
|
switch bytes.Compare(ourSer, theirSer) {
|
|
|
|
case -1:
|
2018-01-18 22:45:30 +01:00
|
|
|
stateObfuscator = DeriveStateHintObfuscator(
|
2018-02-18 00:20:41 +01:00
|
|
|
ourContribution.PaymentBasePoint.PubKey,
|
|
|
|
theirContribution.PaymentBasePoint.PubKey,
|
2017-07-30 04:20:08 +02:00
|
|
|
)
|
|
|
|
default:
|
2018-01-18 22:45:30 +01:00
|
|
|
stateObfuscator = DeriveStateHintObfuscator(
|
2018-02-18 00:20:41 +01:00
|
|
|
theirContribution.PaymentBasePoint.PubKey,
|
|
|
|
ourContribution.PaymentBasePoint.PubKey,
|
2017-07-30 04:20:08 +02:00
|
|
|
)
|
2016-11-16 21:54:27 +01:00
|
|
|
}
|
|
|
|
}
|
2017-09-25 20:25:58 +02:00
|
|
|
err = initStateHints(ourCommitTx, theirCommitTx, stateObfuscator)
|
2016-11-16 21:54:27 +01:00
|
|
|
if err != nil {
|
|
|
|
req.err <- err
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
// Sort both transactions according to the agreed upon canonical
|
2015-12-31 07:36:01 +01:00
|
|
|
// ordering. This lets us skip sending the entire transaction over,
|
|
|
|
// instead we'll just send signatures.
|
|
|
|
txsort.InPlaceSort(ourCommitTx)
|
|
|
|
txsort.InPlaceSort(theirCommitTx)
|
|
|
|
|
2021-12-12 20:56:20 +01:00
|
|
|
walletLog.Tracef("Local commit tx for ChannelPoint(%v): %v",
|
2019-11-01 05:39:17 +01:00
|
|
|
chanPoint, spew.Sdump(ourCommitTx))
|
2021-12-12 20:56:20 +01:00
|
|
|
walletLog.Tracef("Remote commit tx for ChannelPoint(%v): %v",
|
2019-11-01 05:39:17 +01:00
|
|
|
chanPoint, spew.Sdump(theirCommitTx))
|
2018-01-31 23:00:01 +01:00
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
// Record newly available information within the open channel state.
|
2020-03-31 09:13:16 +02:00
|
|
|
chanState.FundingOutpoint = chanPoint
|
2017-11-10 23:18:41 +01:00
|
|
|
chanState.LocalCommitment.CommitTx = ourCommitTx
|
|
|
|
chanState.RemoteCommitment.CommitTx = theirCommitTx
|
2015-12-21 22:53:34 +01:00
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
// Next, we'll obtain the funding witness script, and the funding
|
|
|
|
// output itself so we can generate a valid signature for the remote
|
|
|
|
// party.
|
|
|
|
fundingIntent := pendingReservation.fundingIntent
|
|
|
|
fundingWitnessScript, fundingOutput, err := fundingIntent.FundingOutput()
|
|
|
|
if err != nil {
|
2023-01-20 02:15:12 +01:00
|
|
|
req.err <- fmt.Errorf("unable to obtain funding "+
|
|
|
|
"output: %w", err)
|
2019-11-01 05:39:17 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-12-21 22:53:34 +01:00
|
|
|
// Generate a signature for their version of the initial commitment
|
|
|
|
// transaction.
|
2023-01-20 02:15:12 +01:00
|
|
|
sigTheirCommit, err := l.signCommitTx(
|
|
|
|
pendingReservation, theirCommitTx, fundingOutput,
|
|
|
|
fundingWitnessScript,
|
|
|
|
)
|
2015-12-21 22:53:34 +01:00
|
|
|
if err != nil {
|
|
|
|
req.err <- err
|
|
|
|
return
|
|
|
|
}
|
2023-01-20 02:15:12 +01:00
|
|
|
|
2015-12-23 05:31:17 +01:00
|
|
|
pendingReservation.ourCommitmentSig = sigTheirCommit
|
2015-12-21 22:53:34 +01:00
|
|
|
|
2015-11-05 21:36:19 +01:00
|
|
|
req.err <- nil
|
|
|
|
}
|
|
|
|
|
2016-06-21 19:37:55 +02:00
|
|
|
// handleSingleContribution is called as the second step to a single funder
|
|
|
|
// workflow to which we are the responder. It simply saves the remote peer's
|
|
|
|
// contribution to the channel, as solely the remote peer will contribute any
|
|
|
|
// funds to the channel.
|
|
|
|
func (l *LightningWallet) handleSingleContribution(req *addSingleContributionMsg) {
|
|
|
|
l.limboMtx.Lock()
|
|
|
|
pendingReservation, ok := l.fundingLimbo[req.pendingFundingID]
|
|
|
|
l.limboMtx.Unlock()
|
|
|
|
if !ok {
|
2017-07-30 04:20:08 +02:00
|
|
|
req.err <- fmt.Errorf("attempted to update non-existent funding state")
|
2016-06-21 19:37:55 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
// Grab the mutex on the channelReservation to ensure thread-safety.
|
2016-06-21 19:37:55 +02:00
|
|
|
pendingReservation.Lock()
|
|
|
|
defer pendingReservation.Unlock()
|
|
|
|
|
2021-06-09 22:23:35 +02:00
|
|
|
// Validate that the remote's UpfrontShutdownScript is a valid script
|
|
|
|
// if it's set.
|
|
|
|
shutdown := req.contribution.UpfrontShutdown
|
|
|
|
if len(shutdown) > 0 {
|
|
|
|
// Validate the shutdown script.
|
2022-06-10 20:16:02 +02:00
|
|
|
if !ValidateUpfrontShutdown(shutdown, &l.Cfg.NetParams) {
|
2021-06-09 22:23:35 +02:00
|
|
|
req.err <- fmt.Errorf("invalid shutdown script")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-21 19:37:55 +02:00
|
|
|
// Simply record the counterparty's contribution into the pending
|
|
|
|
// reservation data as they'll be solely funding the channel entirely.
|
|
|
|
pendingReservation.theirContribution = req.contribution
|
|
|
|
theirContribution := pendingReservation.theirContribution
|
2017-07-30 04:20:08 +02:00
|
|
|
chanState := pendingReservation.partialState
|
2016-06-30 21:13:46 +02:00
|
|
|
|
2021-09-23 22:37:38 +02:00
|
|
|
// Perform bounds checking on both ChannelReserve and DustLimit
|
|
|
|
// parameters. The ChannelReserve may have been changed by the
|
|
|
|
// ChannelAcceptor RPC, so this is necessary.
|
|
|
|
if !pendingReservation.validateReserveBounds() {
|
|
|
|
req.err <- fmt.Errorf("invalid reserve and dust bounds")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-06-30 21:13:46 +02:00
|
|
|
// Initialize an empty sha-chain for them, tracking the current pending
|
2017-01-13 06:01:50 +01:00
|
|
|
// revocation hash (we don't yet know the preimage so we can't add it
|
2016-06-30 21:13:46 +02:00
|
|
|
// to the chain).
|
2016-12-14 15:01:48 +01:00
|
|
|
remotePreimageStore := shachain.NewRevocationStore()
|
2017-07-30 04:20:08 +02:00
|
|
|
chanState.RevocationStore = remotePreimageStore
|
2016-06-21 19:37:55 +02:00
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
// Now that we've received their first commitment point, we'll store it
|
|
|
|
// within the channel state so we can sync it to disk once the funding
|
|
|
|
// process is complete.
|
|
|
|
chanState.RemoteCurrentRevocation = theirContribution.FirstCommitmentPoint
|
2016-06-21 19:37:55 +02:00
|
|
|
|
|
|
|
req.err <- nil
|
|
|
|
}
|
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
// verifyFundingInputs attempts to verify all remote inputs to the funding
|
|
|
|
// transaction.
|
|
|
|
func (l *LightningWallet) verifyFundingInputs(fundingTx *wire.MsgTx,
|
|
|
|
remoteInputScripts []*input.Script) error {
|
2016-12-25 01:47:09 +01:00
|
|
|
|
2016-02-03 08:59:27 +01:00
|
|
|
sigIndex := 0
|
2022-03-18 18:37:43 +01:00
|
|
|
fundingHashCache := input.NewTxSigHashesV0Only(fundingTx)
|
2019-11-01 05:39:17 +01:00
|
|
|
inputScripts := remoteInputScripts
|
2015-12-03 01:51:46 +01:00
|
|
|
for i, txin := range fundingTx.TxIn {
|
2016-06-21 19:37:55 +02:00
|
|
|
if len(inputScripts) != 0 && len(txin.Witness) == 0 {
|
2016-05-04 04:49:58 +02:00
|
|
|
// Attach the input scripts so we can verify it below.
|
|
|
|
txin.Witness = inputScripts[sigIndex].Witness
|
2018-11-18 05:46:51 +01:00
|
|
|
txin.SignatureScript = inputScripts[sigIndex].SigScript
|
2016-02-03 08:43:12 +01:00
|
|
|
|
2016-01-07 01:15:49 +01:00
|
|
|
// Fetch the alleged previous output along with the
|
2015-12-03 01:51:46 +01:00
|
|
|
// pkscript referenced by this input.
|
2018-07-18 04:20:05 +02:00
|
|
|
//
|
|
|
|
// TODO(roasbeef): when dual funder pass actual
|
|
|
|
// height-hint
|
2019-11-21 04:54:47 +01:00
|
|
|
//
|
|
|
|
// TODO(roasbeef): this fails for neutrino always as it
|
|
|
|
// treats the height hint as an exact birthday of the
|
|
|
|
// utxo rather than a lower bound
|
|
|
|
pkScript, err := txscript.ComputePkScript(
|
|
|
|
txin.SignatureScript, txin.Witness,
|
2018-07-18 04:20:05 +02:00
|
|
|
)
|
|
|
|
if err != nil {
|
2024-02-26 12:19:38 +01:00
|
|
|
return fmt.Errorf("cannot create script: %w",
|
|
|
|
err)
|
2018-07-18 04:20:05 +02:00
|
|
|
}
|
|
|
|
output, err := l.Cfg.ChainIO.GetUtxo(
|
|
|
|
&txin.PreviousOutPoint,
|
2019-11-21 04:54:47 +01:00
|
|
|
pkScript.Script(), 0, l.quit,
|
2018-07-18 04:20:05 +02:00
|
|
|
)
|
2016-01-07 01:15:49 +01:00
|
|
|
if output == nil {
|
2019-11-01 05:39:17 +01:00
|
|
|
return fmt.Errorf("input to funding tx does "+
|
|
|
|
"not exist: %v", err)
|
2015-12-03 01:51:46 +01:00
|
|
|
}
|
2016-05-04 04:49:58 +02:00
|
|
|
|
|
|
|
// Ensure that the witness+sigScript combo is valid.
|
2019-11-01 05:39:17 +01:00
|
|
|
vm, err := txscript.NewEngine(
|
|
|
|
output.PkScript, fundingTx, i,
|
|
|
|
txscript.StandardVerifyFlags, nil,
|
|
|
|
fundingHashCache, output.Value,
|
2022-03-18 18:37:43 +01:00
|
|
|
txscript.NewCannedPrevOutputFetcher(
|
|
|
|
output.PkScript, output.Value,
|
|
|
|
),
|
2019-11-01 05:39:17 +01:00
|
|
|
)
|
2015-12-03 01:51:46 +01:00
|
|
|
if err != nil {
|
2019-11-01 05:39:17 +01:00
|
|
|
return fmt.Errorf("cannot create script "+
|
2017-04-14 09:46:31 +02:00
|
|
|
"engine: %s", err)
|
2015-12-03 01:51:46 +01:00
|
|
|
}
|
|
|
|
if err = vm.Execute(); err != nil {
|
2019-11-01 05:39:17 +01:00
|
|
|
return fmt.Errorf("cannot validate "+
|
2017-04-14 09:46:31 +02:00
|
|
|
"transaction: %s", err)
|
2016-01-07 01:15:49 +01:00
|
|
|
}
|
2016-02-03 08:59:27 +01:00
|
|
|
|
|
|
|
sigIndex++
|
2015-11-05 21:36:19 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-01-20 02:15:12 +01:00
|
|
|
// verifyCommitSig verifies an incoming signature for our version of the
|
|
|
|
// commitment transaction. For normal channels, this will verify that the ECDSA
|
|
|
|
// signature is valid. For taproot channels, we'll verify that their partial
|
|
|
|
// signature is valid, so it can properly be combined with our eventual
|
|
|
|
// signature when we need to broadcast.
|
|
|
|
func (l *LightningWallet) verifyCommitSig(res *ChannelReservation,
|
|
|
|
commitSig input.Signature, commitTx *wire.MsgTx) error {
|
|
|
|
|
|
|
|
localKey := res.ourContribution.MultiSigKey.PubKey
|
|
|
|
remoteKey := res.theirContribution.MultiSigKey.PubKey
|
|
|
|
channelValue := int64(res.partialState.Capacity)
|
|
|
|
|
|
|
|
switch {
|
|
|
|
// If this isn't a taproot channel, then we'll construct a segwit v0
|
|
|
|
// p2wsh sighash.
|
|
|
|
case !res.partialState.ChanType.IsTaproot():
|
|
|
|
hashCache := input.NewTxSigHashesV0Only(commitTx)
|
|
|
|
witnessScript, _, err := input.GenFundingPkScript(
|
|
|
|
localKey.SerializeCompressed(),
|
|
|
|
remoteKey.SerializeCompressed(), channelValue,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
sigHash, err := txscript.CalcWitnessSigHash(
|
|
|
|
witnessScript, hashCache, txscript.SigHashAll,
|
|
|
|
commitTx, 0, channelValue,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Verify that we've received a valid signature from the remote
|
|
|
|
// party for our version of the commitment transaction.
|
|
|
|
if !commitSig.Verify(sigHash, remoteKey) {
|
|
|
|
return fmt.Errorf("counterparty's commitment " +
|
|
|
|
"signature is invalid")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
|
|
// Otherwise for taproot channels, we'll compute the segwit v1 sighash,
|
|
|
|
// which is slightly different.
|
|
|
|
default:
|
|
|
|
// First, check to see if we've generated the musig session
|
|
|
|
// already. If we're the responder in the funding flow, we may
|
|
|
|
// not have generated it already.
|
|
|
|
if res.musigSessions == nil {
|
|
|
|
_, fundingOutput, err := input.GenTaprootFundingScript(
|
|
|
|
localKey, remoteKey, channelValue,
|
2024-03-13 15:53:41 +01:00
|
|
|
res.partialState.TapscriptRoot,
|
2023-01-20 02:15:12 +01:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
res.musigSessions = genMusigSession(
|
|
|
|
res.ourContribution, res.theirContribution,
|
|
|
|
l.Cfg.Signer, fundingOutput,
|
2024-04-18 03:43:01 +02:00
|
|
|
res.partialState.TapscriptRoot,
|
2023-01-20 02:15:12 +01:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// For the musig2 based channels, we'll use the generated local
|
|
|
|
// musig2 session to verify the signature.
|
|
|
|
localSession := res.musigSessions.LocalSession
|
|
|
|
|
|
|
|
// At this point, the commitment signature passed in should
|
|
|
|
// actually be a wrapped musig2 signature, so we'll do a type
|
|
|
|
// asset to the get the signature we actually need.
|
|
|
|
partialSig, ok := commitSig.(*MusigPartialSig)
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("expected *musig2.PartialSignature, "+
|
|
|
|
"got: %T", commitSig)
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err := localSession.VerifyCommitSig(
|
|
|
|
commitTx, partialSig.ToWireSig(),
|
|
|
|
)
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
// handleFundingCounterPartySigs is the final step in the channel reservation
|
|
|
|
// workflow. During this step, we validate *all* the received signatures for
|
|
|
|
// inputs to the funding transaction. If any of these are invalid, we bail,
|
|
|
|
// and forcibly cancel this funding request. Additionally, we ensure that the
|
|
|
|
// signature we received from the counterparty for our version of the commitment
|
|
|
|
// transaction allows us to spend from the funding output with the addition of
|
|
|
|
// our signature.
|
|
|
|
func (l *LightningWallet) handleFundingCounterPartySigs(msg *addCounterPartySigsMsg) {
|
|
|
|
l.limboMtx.RLock()
|
|
|
|
res, ok := l.fundingLimbo[msg.pendingFundingID]
|
|
|
|
l.limboMtx.RUnlock()
|
|
|
|
if !ok {
|
|
|
|
msg.err <- fmt.Errorf("attempted to update non-existent funding state")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Grab the mutex on the ChannelReservation to ensure thread-safety
|
|
|
|
res.Lock()
|
|
|
|
defer res.Unlock()
|
|
|
|
|
|
|
|
// Now we can complete the funding transaction by adding their
|
|
|
|
// signatures to their inputs.
|
|
|
|
res.theirFundingInputScripts = msg.theirFundingInputScripts
|
|
|
|
inputScripts := msg.theirFundingInputScripts
|
|
|
|
|
|
|
|
// Only if we have the final funding transaction do we need to verify
|
|
|
|
// the final set of inputs. Otherwise, it may be the case that the
|
|
|
|
// channel was funded via an external wallet.
|
|
|
|
fundingTx := res.fundingTx
|
|
|
|
if res.partialState.ChanType.HasFundingTx() {
|
|
|
|
err := l.verifyFundingInputs(fundingTx, inputScripts)
|
|
|
|
if err != nil {
|
|
|
|
msg.err <- err
|
|
|
|
msg.completeChan <- nil
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-28 21:13:41 +01:00
|
|
|
// At this point, we can also record and verify their signature for our
|
2015-12-21 22:54:33 +01:00
|
|
|
// commitment transaction.
|
2016-10-26 23:56:48 +02:00
|
|
|
res.theirCommitmentSig = msg.theirCommitmentSig
|
2017-11-10 23:18:41 +01:00
|
|
|
commitTx := res.partialState.LocalCommitment.CommitTx
|
2015-12-28 21:13:41 +01:00
|
|
|
|
2023-01-20 02:15:12 +01:00
|
|
|
err := l.verifyCommitSig(res, msg.theirCommitmentSig, commitTx)
|
2015-12-28 21:13:41 +01:00
|
|
|
if err != nil {
|
2023-01-20 02:15:12 +01:00
|
|
|
msg.err <- fmt.Errorf("counterparty's commitment signature is "+
|
|
|
|
"invalid: %w", err)
|
2017-04-14 09:46:31 +02:00
|
|
|
msg.completeChan <- nil
|
2015-12-28 21:13:41 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-04-06 02:06:38 +02:00
|
|
|
theirCommitSigBytes := msg.theirCommitmentSig.Serialize()
|
|
|
|
res.partialState.LocalCommitment.CommitSig = theirCommitSigBytes
|
2016-07-06 01:53:55 +02:00
|
|
|
|
2015-12-19 04:39:51 +01:00
|
|
|
// Funding complete, this entry can be removed from limbo.
|
2015-11-13 03:43:32 +01:00
|
|
|
l.limboMtx.Lock()
|
2016-10-26 23:56:48 +02:00
|
|
|
delete(l.fundingLimbo, res.reservationID)
|
2021-01-11 08:47:38 +01:00
|
|
|
delete(l.reservationIDs, res.pendingChanID)
|
2015-11-13 03:43:32 +01:00
|
|
|
l.limboMtx.Unlock()
|
2015-11-05 21:36:19 +01:00
|
|
|
|
2019-11-01 05:47:27 +01:00
|
|
|
l.intentMtx.Lock()
|
|
|
|
delete(l.fundingIntents, res.pendingChanID)
|
|
|
|
l.intentMtx.Unlock()
|
|
|
|
|
2017-06-06 00:05:35 +02:00
|
|
|
// As we're about to broadcast the funding transaction, we'll take note
|
|
|
|
// of the current height for record keeping purposes.
|
2017-07-30 04:02:38 +02:00
|
|
|
_, bestHeight, err := l.Cfg.ChainIO.GetBestBlock()
|
2017-06-06 00:05:35 +02:00
|
|
|
if err != nil {
|
|
|
|
msg.err <- err
|
|
|
|
msg.completeChan <- nil
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
// As we've completed the funding process, we'll no convert the
|
|
|
|
// contribution structs into their underlying channel config objects to
|
|
|
|
// he stored within the database.
|
|
|
|
res.partialState.LocalChanCfg = res.ourContribution.toChanConfig()
|
|
|
|
res.partialState.RemoteChanCfg = res.theirContribution.toChanConfig()
|
2016-06-21 19:37:55 +02:00
|
|
|
|
2018-03-11 02:28:17 +01:00
|
|
|
// We'll also record the finalized funding txn, which will allow us to
|
|
|
|
// rebroadcast on startup in case we fail.
|
|
|
|
res.partialState.FundingTxn = fundingTx
|
|
|
|
|
2019-12-03 10:38:29 +01:00
|
|
|
// Set optional upfront shutdown scripts on the channel state so that they
|
|
|
|
// are persisted. These values may be nil.
|
|
|
|
res.partialState.LocalShutdownScript =
|
|
|
|
res.ourContribution.UpfrontShutdown
|
|
|
|
res.partialState.RemoteShutdownScript =
|
|
|
|
res.theirContribution.UpfrontShutdown
|
|
|
|
|
2020-10-21 19:34:05 +02:00
|
|
|
res.partialState.RevocationKeyLocator = res.nextRevocationKeyLoc
|
|
|
|
|
2018-02-07 04:13:07 +01:00
|
|
|
// Add the complete funding transaction to the DB, in its open bucket
|
2015-12-03 01:51:21 +01:00
|
|
|
// which will be used for the lifetime of this channel.
|
2016-10-26 23:56:48 +02:00
|
|
|
nodeAddr := res.nodeAddr
|
2017-06-06 00:05:35 +02:00
|
|
|
err = res.partialState.SyncPending(nodeAddr, uint32(bestHeight))
|
|
|
|
if err != nil {
|
2016-03-24 08:01:35 +01:00
|
|
|
msg.err <- err
|
2017-04-14 09:46:31 +02:00
|
|
|
msg.completeChan <- nil
|
2016-03-24 08:01:35 +01:00
|
|
|
return
|
|
|
|
}
|
2015-12-03 01:51:21 +01:00
|
|
|
|
2017-01-24 03:19:54 +01:00
|
|
|
msg.completeChan <- res.partialState
|
2016-03-24 08:01:35 +01:00
|
|
|
msg.err <- nil
|
2015-11-13 03:43:32 +01:00
|
|
|
}
|
|
|
|
|
2016-06-21 19:37:55 +02:00
|
|
|
// handleSingleFunderSigs is called once the remote peer who initiated the
|
|
|
|
// single funder workflow has assembled the funding transaction, and generated
|
|
|
|
// a signature for our version of the commitment transaction. This method
|
|
|
|
// progresses the workflow by generating a signature for the remote peer's
|
|
|
|
// version of the commitment transaction.
|
|
|
|
func (l *LightningWallet) handleSingleFunderSigs(req *addSingleFunderSigsMsg) {
|
|
|
|
l.limboMtx.RLock()
|
|
|
|
pendingReservation, ok := l.fundingLimbo[req.pendingFundingID]
|
|
|
|
l.limboMtx.RUnlock()
|
|
|
|
if !ok {
|
2017-09-25 20:25:58 +02:00
|
|
|
req.err <- fmt.Errorf("attempted to update non-existent funding state")
|
2017-04-14 09:46:31 +02:00
|
|
|
req.completeChan <- nil
|
2016-06-21 19:37:55 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-07-30 04:02:38 +02:00
|
|
|
// Grab the mutex on the ChannelReservation to ensure thread-safety
|
2016-06-21 19:37:55 +02:00
|
|
|
pendingReservation.Lock()
|
|
|
|
defer pendingReservation.Unlock()
|
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
chanState := pendingReservation.partialState
|
2024-04-18 08:35:44 +02:00
|
|
|
|
|
|
|
// If we have an aux funding desc, then we can use it to populate some
|
|
|
|
// of the optional, but opaque TLV blobs we'll carry for the channel.
|
|
|
|
chanState.CustomBlob = fn.MapOption(func(desc AuxFundingDesc) tlv.Blob {
|
|
|
|
return desc.CustomFundingBlob
|
|
|
|
})(req.auxFundingDesc)
|
|
|
|
chanState.LocalCommitment.CustomBlob = fn.MapOption(
|
|
|
|
func(desc AuxFundingDesc) tlv.Blob {
|
|
|
|
return desc.CustomLocalCommitBlob
|
|
|
|
},
|
|
|
|
)(req.auxFundingDesc)
|
|
|
|
chanState.RemoteCommitment.CustomBlob = fn.MapOption(
|
|
|
|
func(desc AuxFundingDesc) tlv.Blob {
|
|
|
|
return desc.CustomRemoteCommitBlob
|
|
|
|
},
|
|
|
|
)(req.auxFundingDesc)
|
|
|
|
|
2023-01-20 02:15:12 +01:00
|
|
|
chanType := pendingReservation.partialState.ChanType
|
2017-07-30 04:20:08 +02:00
|
|
|
chanState.FundingOutpoint = *req.fundingOutpoint
|
2016-06-21 19:37:55 +02:00
|
|
|
fundingTxIn := wire.NewTxIn(req.fundingOutpoint, nil, nil)
|
|
|
|
|
|
|
|
// Now that we have the funding outpoint, we can generate both versions
|
|
|
|
// of the commitment transaction, and generate a signature for the
|
|
|
|
// remote node's commitment transactions.
|
2017-11-10 23:18:41 +01:00
|
|
|
localBalance := pendingReservation.partialState.LocalCommitment.LocalBalance.ToSatoshis()
|
|
|
|
remoteBalance := pendingReservation.partialState.LocalCommitment.RemoteBalance.ToSatoshis()
|
2021-07-15 02:16:13 +02:00
|
|
|
var leaseExpiry uint32
|
|
|
|
if pendingReservation.partialState.ChanType.HasLeaseExpiration() {
|
|
|
|
leaseExpiry = pendingReservation.partialState.ThawHeight
|
|
|
|
}
|
2024-04-05 02:26:03 +02:00
|
|
|
|
|
|
|
localAuxLeaves := fn.MapOption(
|
|
|
|
func(desc AuxFundingDesc) CommitAuxLeaves {
|
|
|
|
return desc.LocalInitAuxLeaves
|
|
|
|
},
|
|
|
|
)(req.auxFundingDesc)
|
|
|
|
remoteAuxLeaves := fn.MapOption(
|
|
|
|
func(desc AuxFundingDesc) CommitAuxLeaves {
|
|
|
|
return desc.RemoteInitAuxLeaves
|
|
|
|
},
|
|
|
|
)(req.auxFundingDesc)
|
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
ourCommitTx, theirCommitTx, err := CreateCommitmentTxns(
|
|
|
|
localBalance, remoteBalance,
|
|
|
|
pendingReservation.ourContribution.ChannelConfig,
|
|
|
|
pendingReservation.theirContribution.ChannelConfig,
|
|
|
|
pendingReservation.ourContribution.FirstCommitmentPoint,
|
|
|
|
pendingReservation.theirContribution.FirstCommitmentPoint,
|
2023-01-20 02:15:12 +01:00
|
|
|
*fundingTxIn, chanType,
|
2021-07-15 02:16:13 +02:00
|
|
|
pendingReservation.partialState.IsInitiator, leaseExpiry,
|
2024-04-05 02:26:03 +02:00
|
|
|
WithAuxLeaves(localAuxLeaves, remoteAuxLeaves),
|
2017-07-30 04:20:08 +02:00
|
|
|
)
|
2017-12-14 23:04:30 +01:00
|
|
|
if err != nil {
|
|
|
|
req.err <- err
|
|
|
|
req.completeChan <- nil
|
|
|
|
return
|
|
|
|
}
|
2016-06-21 19:37:55 +02:00
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
// With both commitment transactions constructed, we can now use the
|
|
|
|
// generator state obfuscator to encode the current state number within
|
2016-11-16 21:54:27 +01:00
|
|
|
// both commitment transactions.
|
2018-01-18 22:45:30 +01:00
|
|
|
stateObfuscator := DeriveStateHintObfuscator(
|
2018-02-18 00:20:41 +01:00
|
|
|
pendingReservation.theirContribution.PaymentBasePoint.PubKey,
|
|
|
|
pendingReservation.ourContribution.PaymentBasePoint.PubKey,
|
|
|
|
)
|
2017-09-25 20:25:58 +02:00
|
|
|
err = initStateHints(ourCommitTx, theirCommitTx, stateObfuscator)
|
2016-11-16 21:54:27 +01:00
|
|
|
if err != nil {
|
|
|
|
req.err <- err
|
2017-04-14 09:46:31 +02:00
|
|
|
req.completeChan <- nil
|
2016-11-16 21:54:27 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
// Sort both transactions according to the agreed upon canonical
|
2016-06-21 19:37:55 +02:00
|
|
|
// ordering. This ensures that both parties sign the same sighash
|
|
|
|
// without further synchronization.
|
|
|
|
txsort.InPlaceSort(ourCommitTx)
|
|
|
|
txsort.InPlaceSort(theirCommitTx)
|
2017-11-10 23:18:41 +01:00
|
|
|
chanState.LocalCommitment.CommitTx = ourCommitTx
|
|
|
|
chanState.RemoteCommitment.CommitTx = theirCommitTx
|
2016-06-21 19:37:55 +02:00
|
|
|
|
2018-01-31 23:00:01 +01:00
|
|
|
walletLog.Debugf("Local commit tx for ChannelPoint(%v): %v",
|
|
|
|
req.fundingOutpoint, spew.Sdump(ourCommitTx))
|
|
|
|
walletLog.Debugf("Remote commit tx for ChannelPoint(%v): %v",
|
|
|
|
req.fundingOutpoint, spew.Sdump(theirCommitTx))
|
|
|
|
|
2023-01-20 02:15:12 +01:00
|
|
|
// With both commitment transactions created, we'll now verify their
|
|
|
|
// signature on our commitment.
|
|
|
|
err = l.verifyCommitSig(
|
|
|
|
pendingReservation, req.theirCommitmentSig, ourCommitTx,
|
2018-02-18 00:20:41 +01:00
|
|
|
)
|
2017-07-30 04:20:08 +02:00
|
|
|
if err != nil {
|
|
|
|
req.err <- err
|
|
|
|
req.completeChan <- nil
|
|
|
|
return
|
|
|
|
}
|
2016-08-13 00:50:47 +02:00
|
|
|
|
2023-01-20 02:15:12 +01:00
|
|
|
theirCommitSigBytes := req.theirCommitmentSig.Serialize()
|
|
|
|
chanState.LocalCommitment.CommitSig = theirCommitSigBytes
|
|
|
|
|
|
|
|
channelValue := int64(pendingReservation.partialState.Capacity)
|
|
|
|
theirKey := pendingReservation.theirContribution.MultiSigKey
|
|
|
|
ourKey := pendingReservation.ourContribution.MultiSigKey
|
|
|
|
|
|
|
|
var (
|
|
|
|
fundingWitnessScript []byte
|
|
|
|
fundingTxOut *wire.TxOut
|
2019-11-01 05:39:17 +01:00
|
|
|
)
|
2023-01-20 02:15:12 +01:00
|
|
|
if chanType.IsTaproot() {
|
2024-11-29 10:16:13 +01:00
|
|
|
//nolint:ll
|
2024-03-13 15:46:33 +01:00
|
|
|
fundingWitnessScript, fundingTxOut, err = input.GenTaprootFundingScript(
|
2023-01-20 02:15:12 +01:00
|
|
|
ourKey.PubKey, theirKey.PubKey, channelValue,
|
2024-03-13 15:53:41 +01:00
|
|
|
pendingReservation.partialState.TapscriptRoot,
|
2023-01-20 02:15:12 +01:00
|
|
|
)
|
|
|
|
} else {
|
2024-11-29 10:16:13 +01:00
|
|
|
//nolint:ll
|
2024-03-13 15:46:33 +01:00
|
|
|
fundingWitnessScript, fundingTxOut, err = input.GenFundingPkScript(
|
2023-01-20 02:15:12 +01:00
|
|
|
ourKey.PubKey.SerializeCompressed(),
|
|
|
|
theirKey.PubKey.SerializeCompressed(), channelValue,
|
|
|
|
)
|
|
|
|
}
|
2016-06-21 19:37:55 +02:00
|
|
|
if err != nil {
|
|
|
|
req.err <- err
|
2017-04-14 09:46:31 +02:00
|
|
|
req.completeChan <- nil
|
2016-06-21 19:37:55 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// With their signature for our version of the commitment transactions
|
|
|
|
// verified, we can now generate a signature for their version,
|
|
|
|
// allowing the funding transaction to be safely broadcast.
|
2023-01-20 02:15:12 +01:00
|
|
|
sigTheirCommit, err := l.signCommitTx(
|
|
|
|
pendingReservation, theirCommitTx, fundingTxOut,
|
|
|
|
fundingWitnessScript,
|
|
|
|
)
|
2016-06-21 19:37:55 +02:00
|
|
|
if err != nil {
|
|
|
|
req.err <- err
|
2017-04-14 09:46:31 +02:00
|
|
|
req.completeChan <- nil
|
2016-06-21 19:37:55 +02:00
|
|
|
return
|
|
|
|
}
|
2023-01-20 02:15:12 +01:00
|
|
|
|
2016-06-21 19:37:55 +02:00
|
|
|
pendingReservation.ourCommitmentSig = sigTheirCommit
|
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
_, bestHeight, err := l.Cfg.ChainIO.GetBestBlock()
|
2017-06-06 00:05:35 +02:00
|
|
|
if err != nil {
|
|
|
|
req.err <- err
|
|
|
|
req.completeChan <- nil
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-12-03 10:38:29 +01:00
|
|
|
// Set optional upfront shutdown scripts on the channel state so that they
|
|
|
|
// are persisted. These values may be nil.
|
|
|
|
chanState.LocalShutdownScript =
|
|
|
|
pendingReservation.ourContribution.UpfrontShutdown
|
|
|
|
chanState.RemoteShutdownScript =
|
|
|
|
pendingReservation.theirContribution.UpfrontShutdown
|
|
|
|
|
2016-06-21 19:37:55 +02:00
|
|
|
// Add the complete funding transaction to the DB, in it's open bucket
|
|
|
|
// which will be used for the lifetime of this channel.
|
2017-07-30 04:20:08 +02:00
|
|
|
chanState.LocalChanCfg = pendingReservation.ourContribution.toChanConfig()
|
|
|
|
chanState.RemoteChanCfg = pendingReservation.theirContribution.toChanConfig()
|
2020-10-21 19:34:05 +02:00
|
|
|
|
|
|
|
chanState.RevocationKeyLocator = pendingReservation.nextRevocationKeyLoc
|
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
err = chanState.SyncPending(pendingReservation.nodeAddr, uint32(bestHeight))
|
2017-05-17 04:00:19 +02:00
|
|
|
if err != nil {
|
2017-01-23 00:06:28 +01:00
|
|
|
req.err <- err
|
2017-04-14 09:46:31 +02:00
|
|
|
req.completeChan <- nil
|
2017-01-23 00:06:28 +01:00
|
|
|
return
|
|
|
|
}
|
2016-06-21 19:37:55 +02:00
|
|
|
|
2017-07-30 04:20:08 +02:00
|
|
|
req.completeChan <- chanState
|
2017-01-23 00:06:28 +01:00
|
|
|
req.err <- nil
|
2017-01-31 05:21:52 +01:00
|
|
|
|
|
|
|
l.limboMtx.Lock()
|
|
|
|
delete(l.fundingLimbo, req.pendingFundingID)
|
2023-02-20 18:47:45 +01:00
|
|
|
delete(l.reservationIDs, pendingReservation.pendingChanID)
|
2017-01-31 05:21:52 +01:00
|
|
|
l.limboMtx.Unlock()
|
2019-11-01 05:47:27 +01:00
|
|
|
|
|
|
|
l.intentMtx.Lock()
|
|
|
|
delete(l.fundingIntents, pendingReservation.pendingChanID)
|
|
|
|
l.intentMtx.Unlock()
|
2016-01-07 01:17:18 +01:00
|
|
|
}
|
|
|
|
|
2018-11-18 06:06:59 +01:00
|
|
|
// WithCoinSelectLock will execute the passed function closure in a
|
|
|
|
// synchronized manner preventing any coin selection operations from proceeding
|
2020-03-19 05:43:49 +01:00
|
|
|
// while the closure is executing. This can be seen as the ability to execute a
|
2018-11-18 06:06:59 +01:00
|
|
|
// function closure under an exclusive coin selection lock.
|
|
|
|
func (l *LightningWallet) WithCoinSelectLock(f func() error) error {
|
|
|
|
l.coinSelectMtx.Lock()
|
|
|
|
defer l.coinSelectMtx.Unlock()
|
|
|
|
|
|
|
|
return f()
|
|
|
|
}
|
|
|
|
|
2018-01-18 22:45:30 +01:00
|
|
|
// DeriveStateHintObfuscator derives the bytes to be used for obfuscating the
|
2018-02-07 04:11:11 +01:00
|
|
|
// state hints from the root to be used for a new channel. The obfuscator is
|
2017-07-30 03:39:58 +02:00
|
|
|
// generated via the following computation:
|
|
|
|
//
|
2022-06-10 20:16:02 +02:00
|
|
|
// - sha256(initiatorKey || responderKey)[26:]
|
|
|
|
// -- where both keys are the multi-sig keys of the respective parties
|
2017-07-30 03:39:58 +02:00
|
|
|
//
|
|
|
|
// The first 6 bytes of the resulting hash are used as the state hint.
|
2018-01-18 22:45:30 +01:00
|
|
|
func DeriveStateHintObfuscator(key1, key2 *btcec.PublicKey) [StateHintSize]byte {
|
2017-07-30 03:39:58 +02:00
|
|
|
h := sha256.New()
|
|
|
|
h.Write(key1.SerializeCompressed())
|
|
|
|
h.Write(key2.SerializeCompressed())
|
2016-11-16 21:54:27 +01:00
|
|
|
|
2017-07-30 03:39:58 +02:00
|
|
|
sha := h.Sum(nil)
|
2016-11-16 21:54:27 +01:00
|
|
|
|
2017-07-30 03:39:58 +02:00
|
|
|
var obfuscator [StateHintSize]byte
|
2017-09-12 17:38:26 +02:00
|
|
|
copy(obfuscator[:], sha[26:])
|
2016-11-16 21:54:27 +01:00
|
|
|
|
2017-07-30 03:39:58 +02:00
|
|
|
return obfuscator
|
2016-11-16 21:54:27 +01:00
|
|
|
}
|
|
|
|
|
2018-02-07 04:11:11 +01:00
|
|
|
// initStateHints properly sets the obfuscated state hints on both commitment
|
2017-09-25 20:25:58 +02:00
|
|
|
// transactions using the passed obfuscator.
|
2016-11-16 21:54:27 +01:00
|
|
|
func initStateHints(commit1, commit2 *wire.MsgTx,
|
2016-12-14 15:01:48 +01:00
|
|
|
obfuscator [StateHintSize]byte) error {
|
2016-11-16 21:54:27 +01:00
|
|
|
|
2016-12-14 15:01:48 +01:00
|
|
|
if err := SetStateNumHint(commit1, 0, obfuscator); err != nil {
|
2016-11-16 21:54:27 +01:00
|
|
|
return err
|
|
|
|
}
|
2016-12-14 15:01:48 +01:00
|
|
|
if err := SetStateNumHint(commit2, 0, obfuscator); err != nil {
|
2016-11-16 21:54:27 +01:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-10-01 04:59:10 +02:00
|
|
|
// ValidateChannel will attempt to fully validate a newly mined channel, given
|
|
|
|
// its funding transaction and existing channel state. If this method returns
|
|
|
|
// an error, then the mined channel is invalid, and shouldn't be used.
|
|
|
|
func (l *LightningWallet) ValidateChannel(channelState *channeldb.OpenChannel,
|
|
|
|
fundingTx *wire.MsgTx) error {
|
|
|
|
|
2024-04-25 19:00:42 +02:00
|
|
|
var chanOpts []ChannelOpt
|
|
|
|
l.Cfg.AuxLeafStore.WhenSome(func(s AuxLeafStore) {
|
|
|
|
chanOpts = append(chanOpts, WithLeafStore(s))
|
|
|
|
})
|
2024-04-09 04:48:36 +02:00
|
|
|
l.Cfg.AuxSigner.WhenSome(func(s AuxSigner) {
|
|
|
|
chanOpts = append(chanOpts, WithAuxSigner(s))
|
|
|
|
})
|
2024-04-25 19:00:42 +02:00
|
|
|
|
2019-10-01 04:59:10 +02:00
|
|
|
// First, we'll obtain a fully signed commitment transaction so we can
|
|
|
|
// pass into it on the chanvalidate package for verification.
|
2024-04-25 19:00:42 +02:00
|
|
|
channel, err := NewLightningChannel(
|
|
|
|
l.Cfg.Signer, channelState, nil, chanOpts...,
|
|
|
|
)
|
2019-10-01 04:59:10 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-01-20 04:35:11 +01:00
|
|
|
|
|
|
|
localKey := channelState.LocalChanCfg.MultiSigKey.PubKey
|
|
|
|
remoteKey := channelState.RemoteChanCfg.MultiSigKey.PubKey
|
2019-10-01 04:59:10 +02:00
|
|
|
|
|
|
|
// We'll also need the multi-sig witness script itself so the
|
|
|
|
// chanvalidate package can check it for correctness against the
|
|
|
|
// funding transaction, and also commitment validity.
|
2023-01-20 04:35:11 +01:00
|
|
|
var fundingScript []byte
|
|
|
|
if channelState.ChanType.IsTaproot() {
|
|
|
|
fundingScript, _, err = input.GenTaprootFundingScript(
|
|
|
|
localKey, remoteKey, int64(channel.Capacity),
|
2024-03-13 15:53:41 +01:00
|
|
|
channelState.TapscriptRoot,
|
2023-01-20 04:35:11 +01:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
witnessScript, err := input.GenMultiSigScript(
|
|
|
|
localKey.SerializeCompressed(),
|
|
|
|
remoteKey.SerializeCompressed(),
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fundingScript, err = input.WitnessScriptHash(witnessScript)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-10-01 04:59:10 +02:00
|
|
|
}
|
2023-01-20 04:35:11 +01:00
|
|
|
|
|
|
|
signedCommitTx, err := channel.getSignedCommitTx()
|
2019-10-01 04:59:10 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-01-20 04:35:11 +01:00
|
|
|
commitCtx := &chanvalidate.CommitmentContext{
|
|
|
|
Value: channel.Capacity,
|
|
|
|
FullySignedCommitTx: signedCommitTx,
|
|
|
|
}
|
2019-10-01 04:59:10 +02:00
|
|
|
|
|
|
|
// Finally, we'll pass in all the necessary context needed to fully
|
|
|
|
// validate that this channel is indeed what we expect, and can be
|
|
|
|
// used.
|
|
|
|
_, err = chanvalidate.Validate(&chanvalidate.Context{
|
|
|
|
Locator: &chanvalidate.OutPointChanLocator{
|
|
|
|
ChanPoint: channelState.FundingOutpoint,
|
|
|
|
},
|
2023-01-20 04:35:11 +01:00
|
|
|
MultiSigPkScript: fundingScript,
|
2019-10-01 04:59:10 +02:00
|
|
|
FundingTx: fundingTx,
|
2023-01-20 04:35:11 +01:00
|
|
|
CommitCtx: commitCtx,
|
2019-10-01 04:59:10 +02:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2019-11-01 05:39:17 +01:00
|
|
|
|
2023-07-11 09:05:22 +02:00
|
|
|
// CancelRebroadcast cancels the rebroadcast of the given transaction.
|
|
|
|
func (l *LightningWallet) CancelRebroadcast(txid chainhash.Hash) {
|
|
|
|
// For neutrino, we don't config the rebroadcaster for the wallet as it
|
|
|
|
// manages the rebroadcasting logic in neutrino itself.
|
|
|
|
if l.Cfg.Rebroadcaster != nil {
|
|
|
|
l.Cfg.Rebroadcaster.MarkAsConfirmed(txid)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
// CoinSource is a wrapper around the wallet that implements the
|
|
|
|
// chanfunding.CoinSource interface.
|
|
|
|
type CoinSource struct {
|
2024-03-30 17:55:25 +01:00
|
|
|
wallet *LightningWallet
|
|
|
|
allowUtxo func(Utxo) bool
|
2019-11-01 05:39:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewCoinSource creates a new instance of the CoinSource wrapper struct.
|
2024-03-30 17:55:25 +01:00
|
|
|
func NewCoinSource(w *LightningWallet, allowUtxo func(Utxo) bool) *CoinSource {
|
|
|
|
return &CoinSource{
|
|
|
|
wallet: w,
|
|
|
|
allowUtxo: allowUtxo,
|
|
|
|
}
|
2019-11-01 05:39:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// ListCoins returns all UTXOs from the source that have between
|
|
|
|
// minConfs and maxConfs number of confirmations.
|
|
|
|
func (c *CoinSource) ListCoins(minConfs int32,
|
2024-02-06 12:25:47 +01:00
|
|
|
maxConfs int32) ([]wallet.Coin, error) {
|
2019-11-01 05:39:17 +01:00
|
|
|
|
2021-02-20 02:41:50 +01:00
|
|
|
utxos, err := c.wallet.ListUnspentWitnessFromDefaultAccount(
|
|
|
|
minConfs, maxConfs,
|
|
|
|
)
|
2019-11-01 05:39:17 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2024-02-06 12:25:47 +01:00
|
|
|
var coins []wallet.Coin
|
2024-03-30 17:55:25 +01:00
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
for _, utxo := range utxos {
|
2024-03-30 17:55:25 +01:00
|
|
|
// If there is a filter function supplied all utxos not adhering
|
2024-05-31 09:16:33 +02:00
|
|
|
// to these conditions will be discarded.
|
2024-03-30 17:55:25 +01:00
|
|
|
if c.allowUtxo != nil && !c.allowUtxo(*utxo) {
|
|
|
|
walletLog.Infof("Cannot use unconfirmed "+
|
|
|
|
"utxo=%v because it is unstable and could be "+
|
|
|
|
"replaced", utxo.OutPoint)
|
|
|
|
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2024-02-06 12:25:47 +01:00
|
|
|
coins = append(coins, wallet.Coin{
|
2019-11-01 05:39:17 +01:00
|
|
|
TxOut: wire.TxOut{
|
|
|
|
Value: int64(utxo.Value),
|
|
|
|
PkScript: utxo.PkScript,
|
|
|
|
},
|
|
|
|
OutPoint: utxo.OutPoint,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return coins, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// CoinFromOutPoint attempts to locate details pertaining to a coin based on
|
|
|
|
// its outpoint. If the coin isn't under the control of the backing CoinSource,
|
|
|
|
// then an error should be returned.
|
2024-02-06 12:25:47 +01:00
|
|
|
func (c *CoinSource) CoinFromOutPoint(op wire.OutPoint) (*wallet.Coin, error) {
|
2024-08-08 15:53:27 +02:00
|
|
|
inputInfo, err := c.wallet.FetchOutpointInfo(&op)
|
2019-11-01 05:39:17 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2024-02-06 12:25:47 +01:00
|
|
|
return &wallet.Coin{
|
2019-11-01 05:39:17 +01:00
|
|
|
TxOut: wire.TxOut{
|
|
|
|
Value: int64(inputInfo.Value),
|
|
|
|
PkScript: inputInfo.PkScript,
|
|
|
|
},
|
|
|
|
OutPoint: inputInfo.OutPoint,
|
|
|
|
}, nil
|
|
|
|
}
|
2019-11-01 05:44:16 +01:00
|
|
|
|
|
|
|
// shimKeyRing is a wrapper struct that's used to provide the proper multi-sig
|
|
|
|
// key for an initiated external funding flow.
|
|
|
|
type shimKeyRing struct {
|
|
|
|
keychain.KeyRing
|
|
|
|
|
|
|
|
*chanfunding.ShimIntent
|
|
|
|
}
|
|
|
|
|
|
|
|
// DeriveNextKey intercepts the normal DeriveNextKey call to a keychain.KeyRing
|
|
|
|
// instance, and supplies the multi-sig key specified by the ShimIntent. This
|
|
|
|
// allows us to transparently insert new keys into the existing funding flow,
|
|
|
|
// as these keys may not come from the wallet itself.
|
|
|
|
func (s *shimKeyRing) DeriveNextKey(keyFam keychain.KeyFamily) (keychain.KeyDescriptor, error) {
|
|
|
|
if keyFam != keychain.KeyFamilyMultiSig {
|
|
|
|
return s.KeyRing.DeriveNextKey(keyFam)
|
|
|
|
}
|
|
|
|
|
|
|
|
fundingKeys, err := s.ShimIntent.MultiSigKeys()
|
|
|
|
if err != nil {
|
|
|
|
return keychain.KeyDescriptor{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return *fundingKeys.LocalKey, nil
|
|
|
|
}
|
2021-06-09 22:23:35 +02:00
|
|
|
|
2022-06-10 20:16:02 +02:00
|
|
|
// ValidateUpfrontShutdown checks whether the provided upfront_shutdown_script
|
2021-06-09 22:23:35 +02:00
|
|
|
// is of a valid type that we accept.
|
2022-06-10 20:16:02 +02:00
|
|
|
func ValidateUpfrontShutdown(shutdown lnwire.DeliveryAddress,
|
2021-06-09 22:23:35 +02:00
|
|
|
params *chaincfg.Params) bool {
|
|
|
|
|
|
|
|
// We don't need to worry about a large UpfrontShutdownScript since it
|
|
|
|
// was already checked in lnwire when decoding from the wire.
|
|
|
|
scriptClass, _, _, _ := txscript.ExtractPkScriptAddrs(shutdown, params)
|
|
|
|
|
2022-06-10 20:16:02 +02:00
|
|
|
switch {
|
|
|
|
case scriptClass == txscript.WitnessV0PubKeyHashTy,
|
|
|
|
scriptClass == txscript.WitnessV0ScriptHashTy,
|
|
|
|
scriptClass == txscript.WitnessV1TaprootTy:
|
|
|
|
|
|
|
|
// The above three types are permitted according to BOLT#02 and
|
|
|
|
// BOLT#05. Everything else is disallowed.
|
2021-06-09 22:23:35 +02:00
|
|
|
return true
|
|
|
|
|
2022-06-10 20:16:02 +02:00
|
|
|
// In this case, we don't know about the actual script template, but it
|
|
|
|
// might be a witness program with versions 2-16. So we'll check that
|
|
|
|
// now
|
|
|
|
case txscript.IsWitnessProgram(shutdown):
|
|
|
|
version, _, err := txscript.ExtractWitnessProgramInfo(shutdown)
|
|
|
|
if err != nil {
|
|
|
|
walletLog.Warnf("unable to extract witness program "+
|
|
|
|
"version (script=%x): %v", shutdown, err)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return version >= 1 && version <= 16
|
|
|
|
|
2021-06-09 22:23:35 +02:00
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
2022-03-18 18:37:44 +01:00
|
|
|
|
|
|
|
// WalletPrevOutputFetcher is a txscript.PrevOutputFetcher that can fetch
|
|
|
|
// outputs from a given wallet controller.
|
|
|
|
type WalletPrevOutputFetcher struct {
|
|
|
|
wc WalletController
|
|
|
|
}
|
|
|
|
|
|
|
|
// A compile time assertion that WalletPrevOutputFetcher implements the
|
|
|
|
// txscript.PrevOutputFetcher interface.
|
|
|
|
var _ txscript.PrevOutputFetcher = (*WalletPrevOutputFetcher)(nil)
|
|
|
|
|
|
|
|
// NewWalletPrevOutputFetcher creates a new WalletPrevOutputFetcher that fetches
|
|
|
|
// previous outputs from the given wallet controller.
|
|
|
|
func NewWalletPrevOutputFetcher(wc WalletController) *WalletPrevOutputFetcher {
|
|
|
|
return &WalletPrevOutputFetcher{
|
|
|
|
wc: wc,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// FetchPrevOutput attempts to fetch the previous output referenced by the
|
|
|
|
// passed outpoint. A nil value will be returned if the passed outpoint doesn't
|
|
|
|
// exist.
|
|
|
|
func (w *WalletPrevOutputFetcher) FetchPrevOutput(op wire.OutPoint) *wire.TxOut {
|
2024-08-08 15:53:27 +02:00
|
|
|
utxo, err := w.wc.FetchOutpointInfo(&op)
|
2022-03-18 18:37:44 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return &wire.TxOut{
|
|
|
|
Value: int64(utxo.Value),
|
|
|
|
PkScript: utxo.PkScript,
|
|
|
|
}
|
|
|
|
}
|