2015-12-20 07:00:50 +01:00
|
|
|
package lnwallet
|
2015-11-13 03:43:32 +01:00
|
|
|
|
|
|
|
import (
|
2021-07-15 01:29:29 +02:00
|
|
|
"errors"
|
2016-10-26 23:56:48 +02:00
|
|
|
"net"
|
2015-11-14 20:52:07 +01:00
|
|
|
"sync"
|
|
|
|
|
2022-02-23 14:48:00 +01:00
|
|
|
"github.com/btcsuite/btcd/btcec/v2"
|
2023-01-20 02:02:32 +01:00
|
|
|
"github.com/btcsuite/btcd/btcec/v2/schnorr/musig2"
|
2022-02-23 14:48:00 +01:00
|
|
|
"github.com/btcsuite/btcd/btcutil"
|
2018-06-05 03:34:16 +02:00
|
|
|
"github.com/btcsuite/btcd/chaincfg/chainhash"
|
|
|
|
"github.com/btcsuite/btcd/wire"
|
2018-07-26 15:16:47 +02:00
|
|
|
"github.com/lightningnetwork/lnd/channeldb"
|
2019-01-16 15:47:43 +01:00
|
|
|
"github.com/lightningnetwork/lnd/input"
|
2020-10-21 19:34:05 +02:00
|
|
|
"github.com/lightningnetwork/lnd/keychain"
|
2019-11-01 05:39:17 +01:00
|
|
|
"github.com/lightningnetwork/lnd/lnwallet/chanfunding"
|
2018-07-26 15:16:47 +02:00
|
|
|
"github.com/lightningnetwork/lnd/lnwire"
|
2015-11-13 03:43:32 +01:00
|
|
|
)
|
|
|
|
|
2020-03-06 16:11:48 +01:00
|
|
|
// CommitmentType is an enum indicating the commitment type we should use for
|
|
|
|
// the channel we are opening.
|
|
|
|
type CommitmentType int
|
|
|
|
|
|
|
|
const (
|
|
|
|
// CommitmentTypeLegacy is the legacy commitment format with a tweaked
|
|
|
|
// to_remote key.
|
|
|
|
CommitmentTypeLegacy = iota
|
|
|
|
|
|
|
|
// CommitmentTypeTweakless is a newer commitment format where the
|
|
|
|
// to_remote key is static.
|
|
|
|
CommitmentTypeTweakless
|
2020-03-06 16:11:49 +01:00
|
|
|
|
2020-12-07 14:22:07 +01:00
|
|
|
// CommitmentTypeAnchorsZeroFeeHtlcTx is a commitment type that is an
|
|
|
|
// extension of the outdated CommitmentTypeAnchors, which in addition
|
|
|
|
// requires second-level HTLC transactions to be signed using a
|
|
|
|
// zero-fee.
|
|
|
|
CommitmentTypeAnchorsZeroFeeHtlcTx
|
2021-07-15 01:29:29 +02:00
|
|
|
|
|
|
|
// CommitmentTypeScriptEnforcedLease is a commitment type that builds
|
|
|
|
// upon CommitmentTypeTweakless and CommitmentTypeAnchorsZeroFeeHtlcTx,
|
|
|
|
// which in addition requires a CLTV clause to spend outputs paying to
|
|
|
|
// the channel initiator. This is intended for use on leased channels to
|
|
|
|
// guarantee that the channel initiator has no incentives to close a
|
|
|
|
// leased channel before its maturity date.
|
|
|
|
CommitmentTypeScriptEnforcedLease
|
2023-01-20 02:02:32 +01:00
|
|
|
|
|
|
|
// CommitmentTypeSimpleTaproot is the base commitment type for the
|
|
|
|
// channels that use a musig2 funding output and the tapscript tree
|
2023-01-20 02:15:12 +01:00
|
|
|
// where relevant for the commitment transaction pk scripts.
|
2023-01-20 02:02:32 +01:00
|
|
|
CommitmentTypeSimpleTaproot
|
2020-03-06 16:11:48 +01:00
|
|
|
)
|
|
|
|
|
2021-07-15 01:29:29 +02:00
|
|
|
// HasStaticRemoteKey returns whether the commitment type supports remote
|
|
|
|
// outputs backed by static keys.
|
|
|
|
func (c CommitmentType) HasStaticRemoteKey() bool {
|
|
|
|
switch c {
|
|
|
|
case CommitmentTypeTweakless,
|
|
|
|
CommitmentTypeAnchorsZeroFeeHtlcTx,
|
2023-01-20 02:02:32 +01:00
|
|
|
CommitmentTypeScriptEnforcedLease,
|
|
|
|
CommitmentTypeSimpleTaproot:
|
2021-07-15 01:29:29 +02:00
|
|
|
return true
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// HasAnchors returns whether the commitment type supports anchor outputs.
|
|
|
|
func (c CommitmentType) HasAnchors() bool {
|
|
|
|
switch c {
|
|
|
|
case CommitmentTypeAnchorsZeroFeeHtlcTx,
|
2023-01-20 02:02:32 +01:00
|
|
|
CommitmentTypeScriptEnforcedLease,
|
|
|
|
CommitmentTypeSimpleTaproot:
|
2021-07-15 01:29:29 +02:00
|
|
|
return true
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-20 02:02:32 +01:00
|
|
|
// IsTaproot returns true if the channel type is a taproot channel.
|
|
|
|
func (c CommitmentType) IsTaproot() bool {
|
|
|
|
return c == CommitmentTypeSimpleTaproot
|
|
|
|
}
|
|
|
|
|
2020-03-06 16:11:48 +01:00
|
|
|
// String returns the name of the CommitmentType.
|
|
|
|
func (c CommitmentType) String() string {
|
|
|
|
switch c {
|
|
|
|
case CommitmentTypeLegacy:
|
|
|
|
return "legacy"
|
|
|
|
case CommitmentTypeTweakless:
|
|
|
|
return "tweakless"
|
2020-12-07 14:22:07 +01:00
|
|
|
case CommitmentTypeAnchorsZeroFeeHtlcTx:
|
|
|
|
return "anchors-zero-fee-second-level"
|
2021-07-15 01:29:29 +02:00
|
|
|
case CommitmentTypeScriptEnforcedLease:
|
|
|
|
return "script-enforced-lease"
|
2023-01-20 02:02:32 +01:00
|
|
|
case CommitmentTypeSimpleTaproot:
|
|
|
|
return "simple-taproot"
|
2020-03-06 16:11:48 +01:00
|
|
|
default:
|
|
|
|
return "invalid"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-30 03:58:55 +02:00
|
|
|
// ChannelContribution is the primary constituent of the funding workflow
|
|
|
|
// within lnwallet. Each side first exchanges their respective contributions
|
|
|
|
// along with channel specific parameters like the min fee/KB. Once
|
|
|
|
// contributions have been exchanged, each side will then produce signatures
|
|
|
|
// for all their inputs to the funding transactions, and finally a signature
|
|
|
|
// for the other party's version of the commitment transaction.
|
2015-12-23 05:30:11 +01:00
|
|
|
type ChannelContribution struct {
|
2016-06-30 21:02:45 +02:00
|
|
|
// FundingOutpoint is the amount of funds contributed to the funding
|
|
|
|
// transaction.
|
2015-12-23 05:30:11 +01:00
|
|
|
FundingAmount btcutil.Amount
|
2015-12-18 20:29:35 +01:00
|
|
|
|
2015-12-23 05:30:11 +01:00
|
|
|
// Inputs to the funding transaction.
|
|
|
|
Inputs []*wire.TxIn
|
|
|
|
|
2016-06-30 21:02:45 +02:00
|
|
|
// ChangeOutputs are the Outputs to be used in the case that the total
|
2016-10-16 01:02:09 +02:00
|
|
|
// value of the funding inputs is greater than the total potential
|
2016-06-30 21:02:45 +02:00
|
|
|
// channel capacity.
|
2015-12-23 05:30:11 +01:00
|
|
|
ChangeOutputs []*wire.TxOut
|
|
|
|
|
2017-07-30 03:58:55 +02:00
|
|
|
// FirstCommitmentPoint is the first commitment point that will be used
|
|
|
|
// to create the revocation key in the first commitment transaction we
|
|
|
|
// send to the remote party.
|
|
|
|
FirstCommitmentPoint *btcec.PublicKey
|
2015-11-14 20:52:07 +01:00
|
|
|
|
2017-07-30 03:58:55 +02:00
|
|
|
// ChannelConfig is the concrete contribution that this node is
|
|
|
|
// offering to the channel. This includes all the various constraints
|
|
|
|
// such as the min HTLC, and also all the keys which will be used for
|
|
|
|
// the duration of the channel.
|
|
|
|
*channeldb.ChannelConfig
|
2019-12-03 10:38:29 +01:00
|
|
|
|
|
|
|
// UpfrontShutdown is an optional address to which the channel should be
|
|
|
|
// paid out to on cooperative close.
|
|
|
|
UpfrontShutdown lnwire.DeliveryAddress
|
2023-01-20 02:02:32 +01:00
|
|
|
|
|
|
|
// LocalNonce is populated if the channel type is a simple taproot
|
|
|
|
// channel. This stores the public (and secret) nonce that will be used
|
|
|
|
// to generate commitments for the local party.
|
|
|
|
LocalNonce *musig2.Nonces
|
2017-07-30 03:58:55 +02:00
|
|
|
}
|
2017-04-13 04:20:36 +02:00
|
|
|
|
2017-07-30 03:58:55 +02:00
|
|
|
// toChanConfig returns the raw channel configuration generated by a node's
|
|
|
|
// contribution to the channel.
|
|
|
|
func (c *ChannelContribution) toChanConfig() channeldb.ChannelConfig {
|
|
|
|
return *c.ChannelConfig
|
2015-12-23 05:30:11 +01:00
|
|
|
}
|
|
|
|
|
2016-01-02 21:09:31 +01:00
|
|
|
// ChannelReservation represents an intent to open a lightning payment channel
|
2018-03-10 16:21:10 +01:00
|
|
|
// with a counterparty. The funding processes from reservation to channel opening
|
|
|
|
// is a 3-step process. In order to allow for full concurrency during the
|
2016-10-16 01:02:09 +02:00
|
|
|
// reservation workflow, resources consumed by a contribution are "locked"
|
|
|
|
// themselves. This prevents a number of race conditions such as two funding
|
|
|
|
// transactions double-spending the same input. A reservation can also be
|
2019-10-03 17:22:43 +02:00
|
|
|
// canceled, which removes the resources from limbo, allowing another
|
2016-10-16 01:02:09 +02:00
|
|
|
// reservation to claim them.
|
2016-01-02 21:09:31 +01:00
|
|
|
//
|
|
|
|
// The reservation workflow consists of the following three steps:
|
|
|
|
// 1. lnwallet.InitChannelReservation
|
2016-10-16 01:02:09 +02:00
|
|
|
// * One requests the wallet to allocate the necessary resources for a
|
2022-08-22 20:58:42 +02:00
|
|
|
// channel reservation. These resources are put in limbo for the lifetime
|
|
|
|
// of a reservation.
|
2018-03-10 16:21:10 +01:00
|
|
|
// * Once completed the reservation will have the wallet's contribution
|
2022-08-22 20:58:42 +02:00
|
|
|
// accessible via the .OurContribution() method. This contribution
|
|
|
|
// contains the necessary items to allow the remote party to build both
|
|
|
|
// the funding, and commitment transactions.
|
2016-06-21 07:31:57 +02:00
|
|
|
// 2. ChannelReservation.ProcessContribution/ChannelReservation.ProcessSingleContribution
|
2016-01-02 21:09:31 +01:00
|
|
|
// * The counterparty presents their contribution to the payment channel.
|
2022-08-22 20:58:42 +02:00
|
|
|
// This allows us to build the funding, and commitment transactions
|
|
|
|
// ourselves.
|
2016-01-02 21:09:31 +01:00
|
|
|
// * We're now able to sign our inputs to the funding transactions, and
|
2022-08-22 20:58:42 +02:00
|
|
|
// the counterparty's version of the commitment transaction.
|
2016-01-02 21:09:31 +01:00
|
|
|
// * All signatures crafted by us, are now available via .OurSignatures().
|
2016-06-21 07:31:57 +02:00
|
|
|
// 3. ChannelReservation.CompleteReservation/ChannelReservation.CompleteReservationSingle
|
2016-01-02 21:09:31 +01:00
|
|
|
// * The final step in the workflow. The counterparty presents the
|
2022-08-22 20:58:42 +02:00
|
|
|
// signatures for all their inputs to the funding transaction, as well
|
|
|
|
// as a signature to our version of the commitment transaction.
|
2016-01-02 21:09:31 +01:00
|
|
|
// * We then verify the validity of all signatures before considering the
|
2022-08-22 20:58:42 +02:00
|
|
|
// channel "open".
|
2015-12-23 05:30:11 +01:00
|
|
|
type ChannelReservation struct {
|
|
|
|
// This mutex MUST be held when either reading or modifying any of the
|
|
|
|
// fields below.
|
|
|
|
sync.RWMutex
|
2015-11-13 03:43:32 +01:00
|
|
|
|
2016-06-21 07:31:57 +02:00
|
|
|
// fundingTx is the funding transaction for this pending channel.
|
|
|
|
fundingTx *wire.MsgTx
|
|
|
|
|
2015-11-13 03:43:32 +01:00
|
|
|
// In order of sorted inputs. 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
|
|
|
ourFundingInputScripts []*input.Script
|
|
|
|
theirFundingInputScripts []*input.Script
|
2015-11-13 03:43:32 +01:00
|
|
|
|
2015-12-23 05:30:11 +01:00
|
|
|
// Our signature for their version of the commitment transaction.
|
2020-04-06 02:06:38 +02:00
|
|
|
ourCommitmentSig input.Signature
|
|
|
|
theirCommitmentSig input.Signature
|
2015-11-13 03:43:32 +01:00
|
|
|
|
2015-12-23 05:30:11 +01:00
|
|
|
ourContribution *ChannelContribution
|
|
|
|
theirContribution *ChannelContribution
|
|
|
|
|
2015-12-26 19:35:15 +01:00
|
|
|
partialState *channeldb.OpenChannel
|
2018-02-03 07:24:43 +01:00
|
|
|
nodeAddr net.Addr
|
2015-11-13 03:43:32 +01:00
|
|
|
|
2015-12-23 05:30:11 +01:00
|
|
|
// The ID of this reservation, used to uniquely track the reservation
|
|
|
|
// throughout its lifetime.
|
2015-11-13 03:43:32 +01:00
|
|
|
reservationID uint64
|
2015-12-15 22:22:18 +01:00
|
|
|
|
2019-11-01 05:45:44 +01:00
|
|
|
// pendingChanID is the pending channel ID for this channel as
|
|
|
|
// identified within the wire protocol.
|
|
|
|
pendingChanID [32]byte
|
|
|
|
|
2017-08-22 08:20:57 +02:00
|
|
|
// pushMSat the amount of milli-satoshis that should be pushed to the
|
2017-01-10 02:24:13 +01:00
|
|
|
// responder of a single funding channel as part of the initial
|
|
|
|
// commitment state.
|
2017-08-22 08:20:57 +02:00
|
|
|
pushMSat lnwire.MilliSatoshi
|
2017-01-10 02:24:13 +01:00
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
wallet *LightningWallet
|
|
|
|
chanFunder chanfunding.Assembler
|
|
|
|
|
|
|
|
fundingIntent chanfunding.Intent
|
2020-10-21 19:34:05 +02:00
|
|
|
|
|
|
|
// nextRevocationKeyLoc stores the key locator information for this
|
|
|
|
// channel.
|
|
|
|
nextRevocationKeyLoc keychain.KeyLocator
|
2023-01-20 02:02:32 +01:00
|
|
|
|
|
|
|
musigSessions *MusigPairSession
|
2015-11-13 03:43:32 +01:00
|
|
|
}
|
|
|
|
|
2016-08-13 00:50:47 +02:00
|
|
|
// NewChannelReservation creates a new channel reservation. This function is
|
2017-01-10 02:24:13 +01:00
|
|
|
// used only internally by lnwallet. In order to concurrent safety, the
|
|
|
|
// creation of all channel reservations should be carried out via the
|
2016-01-02 21:09:31 +01:00
|
|
|
// lnwallet.InitChannelReservation interface.
|
2019-07-11 13:14:36 +02:00
|
|
|
func NewChannelReservation(capacity, localFundingAmt btcutil.Amount,
|
2022-04-04 22:15:02 +02:00
|
|
|
wallet *LightningWallet, id uint64, chainHash *chainhash.Hash,
|
|
|
|
thawHeight uint32, req *InitFundingReserveMsg) (*ChannelReservation,
|
|
|
|
error) {
|
2016-11-16 21:51:18 +01:00
|
|
|
|
|
|
|
var (
|
2017-08-22 08:20:57 +02:00
|
|
|
ourBalance lnwire.MilliSatoshi
|
|
|
|
theirBalance lnwire.MilliSatoshi
|
2017-01-17 05:28:30 +01:00
|
|
|
initiator bool
|
2016-11-16 21:51:18 +01:00
|
|
|
)
|
2016-06-21 07:31:57 +02:00
|
|
|
|
2020-03-06 16:11:49 +01:00
|
|
|
// Based on the channel type, we determine the initial commit weight
|
|
|
|
// and fee.
|
|
|
|
commitWeight := int64(input.CommitWeight)
|
2023-03-02 06:48:28 +01:00
|
|
|
if req.CommitType.IsTaproot() {
|
|
|
|
commitWeight = input.TaprootCommitWeight
|
|
|
|
} else if req.CommitType.HasAnchors() {
|
2021-07-15 01:29:29 +02:00
|
|
|
commitWeight = int64(input.AnchorCommitWeight)
|
2020-03-06 16:11:49 +01:00
|
|
|
}
|
2022-04-04 22:15:02 +02:00
|
|
|
commitFee := req.CommitFeePerKw.FeeForWeight(commitWeight)
|
2020-03-06 16:11:49 +01:00
|
|
|
|
2019-07-11 13:14:36 +02:00
|
|
|
localFundingMSat := lnwire.NewMSatFromSatoshis(localFundingAmt)
|
2019-08-01 05:16:52 +02:00
|
|
|
// TODO(halseth): make method take remote funding amount directly
|
2019-07-11 13:14:36 +02:00
|
|
|
// instead of inferring it from capacity and local amt.
|
2017-08-22 08:20:57 +02:00
|
|
|
capacityMSat := lnwire.NewMSatFromSatoshis(capacity)
|
2020-03-06 16:11:49 +01:00
|
|
|
|
|
|
|
// The total fee paid by the initiator will be the commitment fee in
|
|
|
|
// addition to the two anchor outputs.
|
2017-08-22 08:20:57 +02:00
|
|
|
feeMSat := lnwire.NewMSatFromSatoshis(commitFee)
|
2022-04-04 22:15:02 +02:00
|
|
|
if req.CommitType.HasAnchors() {
|
2020-03-06 16:11:49 +01:00
|
|
|
feeMSat += 2 * lnwire.NewMSatFromSatoshis(anchorSize)
|
|
|
|
}
|
2017-08-22 08:20:57 +02:00
|
|
|
|
2021-09-23 21:40:37 +02:00
|
|
|
// Used to cut down on verbosity.
|
|
|
|
defaultDust := wallet.Cfg.DefaultConstraints.DustLimit
|
|
|
|
|
2016-10-24 03:42:03 +02:00
|
|
|
// If we're the responder to a single-funder reservation, then we have
|
2017-01-10 02:24:13 +01:00
|
|
|
// no initial balance in the channel unless the remote party is pushing
|
|
|
|
// some funds to us within the first commitment state.
|
2019-07-11 13:14:36 +02:00
|
|
|
if localFundingAmt == 0 {
|
2022-04-04 22:15:02 +02:00
|
|
|
ourBalance = req.PushMSat
|
|
|
|
theirBalance = capacityMSat - feeMSat - req.PushMSat
|
2017-01-17 05:28:30 +01:00
|
|
|
initiator = false
|
2018-02-25 04:11:14 +01:00
|
|
|
|
|
|
|
// If the responder doesn't have enough funds to actually pay
|
|
|
|
// the fees, then we'll bail our early.
|
|
|
|
if int64(theirBalance) < 0 {
|
|
|
|
return nil, ErrFunderBalanceDust(
|
|
|
|
int64(commitFee), int64(theirBalance.ToSatoshis()),
|
2021-09-23 21:40:37 +02:00
|
|
|
int64(2*defaultDust),
|
2018-02-25 04:11:14 +01:00
|
|
|
)
|
|
|
|
}
|
2016-06-21 07:31:57 +02:00
|
|
|
} else {
|
2016-09-13 04:05:56 +02:00
|
|
|
// TODO(roasbeef): need to rework fee structure in general and
|
|
|
|
// also when we "unlock" dual funder within the daemon
|
2016-10-24 03:42:03 +02:00
|
|
|
|
2019-07-11 13:14:36 +02:00
|
|
|
if capacity == localFundingAmt {
|
2016-11-16 21:51:18 +01:00
|
|
|
// If we're initiating a single funder workflow, then
|
|
|
|
// we pay all the initial fees within the commitment
|
2017-01-10 02:24:13 +01:00
|
|
|
// transaction. We also deduct our balance by the
|
|
|
|
// amount pushed as part of the initial state.
|
2022-04-04 22:15:02 +02:00
|
|
|
ourBalance = capacityMSat - feeMSat - req.PushMSat
|
|
|
|
theirBalance = req.PushMSat
|
2016-09-13 04:05:56 +02:00
|
|
|
} else {
|
2016-11-16 21:51:18 +01:00
|
|
|
// Otherwise, this is a dual funder workflow where both
|
|
|
|
// slides split the amount funded and the commitment
|
|
|
|
// fee.
|
2019-07-11 13:14:36 +02:00
|
|
|
ourBalance = localFundingMSat - (feeMSat / 2)
|
2022-04-04 22:15:02 +02:00
|
|
|
theirBalance = capacityMSat - localFundingMSat - (feeMSat / 2) + req.PushMSat
|
2016-09-13 04:05:56 +02:00
|
|
|
}
|
2016-10-24 03:42:03 +02:00
|
|
|
|
2017-01-17 05:28:30 +01:00
|
|
|
initiator = true
|
2018-02-25 04:11:14 +01:00
|
|
|
|
|
|
|
// If we, the initiator don't have enough funds to actually pay
|
|
|
|
// the fees, then we'll exit with an error.
|
|
|
|
if int64(ourBalance) < 0 {
|
|
|
|
return nil, ErrFunderBalanceDust(
|
|
|
|
int64(commitFee), int64(ourBalance),
|
2021-09-23 21:40:37 +02:00
|
|
|
int64(2*defaultDust),
|
2018-02-25 04:11:14 +01:00
|
|
|
)
|
|
|
|
}
|
2016-06-21 07:31:57 +02:00
|
|
|
}
|
|
|
|
|
2017-11-26 20:32:57 +01:00
|
|
|
// If we're the initiator and our starting balance within the channel
|
2018-01-29 00:11:13 +01:00
|
|
|
// after we take account of fees is below 2x the dust limit, then we'll
|
|
|
|
// reject this channel creation request.
|
2017-11-26 20:32:57 +01:00
|
|
|
//
|
|
|
|
// TODO(roasbeef): reject if 30% goes to fees? dust channel
|
2021-09-23 21:40:37 +02:00
|
|
|
if initiator && ourBalance.ToSatoshis() <= 2*defaultDust {
|
2018-02-27 18:32:54 +01:00
|
|
|
return nil, ErrFunderBalanceDust(
|
|
|
|
int64(commitFee),
|
|
|
|
int64(ourBalance.ToSatoshis()),
|
2021-09-23 21:40:37 +02:00
|
|
|
int64(2*defaultDust),
|
2018-02-27 18:32:54 +01:00
|
|
|
)
|
2017-11-26 20:32:57 +01:00
|
|
|
}
|
|
|
|
|
2020-03-06 16:11:48 +01:00
|
|
|
// Similarly we ensure their balance is reasonable if we are not the
|
|
|
|
// initiator.
|
2021-09-23 21:40:37 +02:00
|
|
|
if !initiator && theirBalance.ToSatoshis() <= 2*defaultDust {
|
2020-03-06 16:11:48 +01:00
|
|
|
return nil, ErrFunderBalanceDust(
|
|
|
|
int64(commitFee),
|
|
|
|
int64(theirBalance.ToSatoshis()),
|
2021-09-23 21:40:37 +02:00
|
|
|
int64(2*defaultDust),
|
2020-03-06 16:11:48 +01:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2017-01-17 05:28:30 +01:00
|
|
|
// Next we'll set the channel type based on what we can ascertain about
|
|
|
|
// the balances/push amount within the channel.
|
|
|
|
var chanType channeldb.ChannelType
|
2016-11-16 21:51:18 +01:00
|
|
|
|
2017-01-17 05:28:30 +01:00
|
|
|
// If either of the balances are zero at this point, or we have a
|
|
|
|
// non-zero push amt (there's no pushing for dual funder), then this is
|
|
|
|
// a single-funder channel.
|
2022-04-04 22:15:02 +02:00
|
|
|
if ourBalance == 0 || theirBalance == 0 || req.PushMSat != 0 {
|
2020-03-06 16:11:49 +01:00
|
|
|
// Both the tweakless type and the anchor type is tweakless,
|
|
|
|
// hence set the bit.
|
2022-04-04 22:15:02 +02:00
|
|
|
if req.CommitType.HasStaticRemoteKey() {
|
2019-10-31 03:24:49 +01:00
|
|
|
chanType |= channeldb.SingleFunderTweaklessBit
|
2019-08-01 05:16:52 +02:00
|
|
|
} else {
|
2019-10-31 03:24:49 +01:00
|
|
|
chanType |= channeldb.SingleFunderBit
|
2019-08-01 05:16:52 +02:00
|
|
|
}
|
2019-11-01 05:39:17 +01:00
|
|
|
|
2022-04-04 22:15:02 +02:00
|
|
|
switch a := req.ChanFunder.(type) {
|
2020-07-09 02:13:14 +02:00
|
|
|
// The first channels of a batch shouldn't publish the batch TX
|
|
|
|
// to avoid problems if some of the funding flows can't be
|
|
|
|
// completed. Only the last channel of a batch should publish.
|
|
|
|
case chanfunding.ConditionalPublishAssembler:
|
|
|
|
if !a.ShouldPublishFundingTx() {
|
|
|
|
chanType |= channeldb.NoFundingTxBit
|
|
|
|
}
|
|
|
|
|
|
|
|
// Normal funding flow, the assembler creates a TX from the
|
|
|
|
// internal wallet.
|
|
|
|
case chanfunding.FundingTxAssembler:
|
|
|
|
// Do nothing, a FundingTxAssembler has the transaction.
|
|
|
|
|
2019-11-01 05:39:17 +01:00
|
|
|
// If this intent isn't one that's able to provide us with a
|
|
|
|
// funding transaction, then we'll set the chanType bit to
|
|
|
|
// signal that we don't have access to one.
|
2020-07-09 02:13:14 +02:00
|
|
|
default:
|
2019-11-01 05:39:17 +01:00
|
|
|
chanType |= channeldb.NoFundingTxBit
|
|
|
|
}
|
2017-01-17 05:28:30 +01:00
|
|
|
} else {
|
|
|
|
// Otherwise, this is a dual funder channel, and no side is
|
|
|
|
// technically the "initiator"
|
2016-11-16 21:51:18 +01:00
|
|
|
initiator = false
|
2019-10-31 03:24:49 +01:00
|
|
|
chanType |= channeldb.DualFunderBit
|
2016-11-16 21:51:18 +01:00
|
|
|
}
|
|
|
|
|
2020-12-07 14:22:07 +01:00
|
|
|
// We are adding anchor outputs to our commitment. We only support this
|
|
|
|
// in combination with zero-fee second-levels HTLCs.
|
2022-04-04 22:15:02 +02:00
|
|
|
if req.CommitType.HasAnchors() {
|
2020-03-06 16:11:49 +01:00
|
|
|
chanType |= channeldb.AnchorOutputsBit
|
2020-12-07 14:22:07 +01:00
|
|
|
chanType |= channeldb.ZeroHtlcTxFeeBit
|
2020-03-06 16:11:49 +01:00
|
|
|
}
|
|
|
|
|
2021-07-15 01:29:29 +02:00
|
|
|
// Set the appropriate LeaseExpiration/Frozen bit based on the
|
|
|
|
// reservation parameters.
|
2022-04-04 22:15:02 +02:00
|
|
|
if req.CommitType == CommitmentTypeScriptEnforcedLease {
|
2021-07-15 01:29:29 +02:00
|
|
|
if thawHeight == 0 {
|
|
|
|
return nil, errors.New("missing absolute expiration " +
|
|
|
|
"for script enforced lease commitment type")
|
|
|
|
}
|
|
|
|
chanType |= channeldb.LeaseExpirationBit
|
|
|
|
} else if thawHeight > 0 {
|
2020-03-14 00:57:48 +01:00
|
|
|
chanType |= channeldb.FrozenBit
|
|
|
|
}
|
|
|
|
|
2023-01-20 02:02:32 +01:00
|
|
|
if req.CommitType == CommitmentTypeSimpleTaproot {
|
|
|
|
chanType |= channeldb.SimpleTaprootFeatureBit
|
|
|
|
}
|
|
|
|
|
2022-04-04 22:15:02 +02:00
|
|
|
if req.ZeroConf {
|
|
|
|
chanType |= channeldb.ZeroConfBit
|
|
|
|
}
|
|
|
|
|
|
|
|
if req.OptionScidAlias {
|
|
|
|
chanType |= channeldb.ScidAliasChanBit
|
|
|
|
}
|
|
|
|
|
|
|
|
if req.ScidAliasFeature {
|
|
|
|
chanType |= channeldb.ScidAliasFeatureBit
|
|
|
|
}
|
|
|
|
|
2015-12-19 04:42:49 +01:00
|
|
|
return &ChannelReservation{
|
2015-12-23 05:30:11 +01:00
|
|
|
ourContribution: &ChannelContribution{
|
2017-08-22 08:20:57 +02:00
|
|
|
FundingAmount: ourBalance.ToSatoshis(),
|
2017-07-30 03:58:55 +02:00
|
|
|
ChannelConfig: &channeldb.ChannelConfig{},
|
2015-12-23 05:30:11 +01:00
|
|
|
},
|
|
|
|
theirContribution: &ChannelContribution{
|
2017-08-22 08:20:57 +02:00
|
|
|
FundingAmount: theirBalance.ToSatoshis(),
|
2017-07-30 03:58:55 +02:00
|
|
|
ChannelConfig: &channeldb.ChannelConfig{},
|
2015-12-23 05:30:11 +01:00
|
|
|
},
|
2015-12-26 19:35:15 +01:00
|
|
|
partialState: &channeldb.OpenChannel{
|
2017-11-14 02:15:27 +01:00
|
|
|
ChanType: chanType,
|
|
|
|
ChainHash: *chainHash,
|
|
|
|
IsPending: true,
|
|
|
|
IsInitiator: initiator,
|
2022-04-04 22:15:02 +02:00
|
|
|
ChannelFlags: req.Flags,
|
2017-11-14 02:15:27 +01:00
|
|
|
Capacity: capacity,
|
2017-11-10 23:18:41 +01:00
|
|
|
LocalCommitment: channeldb.ChannelCommitment{
|
|
|
|
LocalBalance: ourBalance,
|
|
|
|
RemoteBalance: theirBalance,
|
2022-04-04 22:15:02 +02:00
|
|
|
FeePerKw: btcutil.Amount(req.CommitFeePerKw),
|
2017-11-10 23:18:41 +01:00
|
|
|
CommitFee: commitFee,
|
|
|
|
},
|
|
|
|
RemoteCommitment: channeldb.ChannelCommitment{
|
|
|
|
LocalBalance: ourBalance,
|
|
|
|
RemoteBalance: theirBalance,
|
2022-04-04 22:15:02 +02:00
|
|
|
FeePerKw: btcutil.Amount(req.CommitFeePerKw),
|
2017-11-10 23:18:41 +01:00
|
|
|
CommitFee: commitFee,
|
|
|
|
},
|
2022-04-07 20:04:50 +02:00
|
|
|
ThawHeight: thawHeight,
|
|
|
|
Db: wallet.Cfg.Database,
|
|
|
|
InitialLocalBalance: ourBalance,
|
|
|
|
InitialRemoteBalance: theirBalance,
|
2023-05-03 22:50:07 +02:00
|
|
|
Memo: req.Memo,
|
2015-12-19 04:42:49 +01:00
|
|
|
},
|
2022-04-04 22:15:02 +02:00
|
|
|
pushMSat: req.PushMSat,
|
|
|
|
pendingChanID: req.PendingChanID,
|
2017-07-30 03:58:55 +02:00
|
|
|
reservationID: id,
|
|
|
|
wallet: wallet,
|
2022-04-04 22:15:02 +02:00
|
|
|
chanFunder: req.ChanFunder,
|
2017-11-26 20:32:57 +01:00
|
|
|
}, nil
|
2015-12-23 05:30:11 +01:00
|
|
|
}
|
|
|
|
|
2022-04-04 22:15:02 +02:00
|
|
|
// AddAlias stores the first alias for zero-conf channels.
|
|
|
|
func (r *ChannelReservation) AddAlias(scid lnwire.ShortChannelID) {
|
|
|
|
r.Lock()
|
|
|
|
defer r.Unlock()
|
|
|
|
|
|
|
|
r.partialState.ShortChannelID = scid
|
|
|
|
}
|
|
|
|
|
2017-07-30 03:58:55 +02:00
|
|
|
// SetNumConfsRequired sets the number of confirmations that are required for
|
|
|
|
// the ultimate funding transaction before the channel can be considered open.
|
|
|
|
// This is distinct from the main reservation workflow as it allows
|
|
|
|
// implementations a bit more flexibility w.r.t to if the responder of the
|
|
|
|
// initiator sets decides the number of confirmations needed.
|
|
|
|
func (r *ChannelReservation) SetNumConfsRequired(numConfs uint16) {
|
|
|
|
r.Lock()
|
|
|
|
defer r.Unlock()
|
|
|
|
|
|
|
|
r.partialState.NumConfsRequired = numConfs
|
|
|
|
}
|
|
|
|
|
2022-04-04 22:15:02 +02:00
|
|
|
// IsZeroConf returns if the reservation's underlying partial channel state is
|
|
|
|
// a zero-conf channel.
|
|
|
|
func (r *ChannelReservation) IsZeroConf() bool {
|
|
|
|
r.RLock()
|
|
|
|
defer r.RUnlock()
|
|
|
|
|
|
|
|
return r.partialState.IsZeroConf()
|
|
|
|
}
|
|
|
|
|
2023-01-20 02:02:32 +01:00
|
|
|
// IsTaproot returns if the reservation's underlying partial channel state is a
|
|
|
|
// taproot channel.
|
|
|
|
func (r *ChannelReservation) IsTaproot() bool {
|
|
|
|
r.RLock()
|
|
|
|
defer r.RUnlock()
|
|
|
|
|
|
|
|
return r.partialState.ChanType.IsTaproot()
|
|
|
|
}
|
|
|
|
|
2017-09-12 17:41:40 +02:00
|
|
|
// CommitConstraints takes the constraints that the remote party specifies for
|
|
|
|
// the type of commitments that we can generate for them. These constraints
|
|
|
|
// include several parameters that serve as flow control restricting the amount
|
|
|
|
// of satoshis that can be transferred in a single commitment. This function
|
|
|
|
// will also attempt to verify the constraints for sanity, returning an error
|
|
|
|
// if the parameters are seemed unsound.
|
2020-10-29 10:29:53 +01:00
|
|
|
func (r *ChannelReservation) CommitConstraints(c *channeldb.ChannelConstraints,
|
2021-09-23 22:37:38 +02:00
|
|
|
maxLocalCSVDelay uint16, responder bool) error {
|
2022-02-07 13:58:28 +01:00
|
|
|
|
2017-07-30 03:58:55 +02:00
|
|
|
r.Lock()
|
|
|
|
defer r.Unlock()
|
|
|
|
|
2022-09-29 10:56:44 +02:00
|
|
|
// First, verify the sanity of the channel constraints.
|
|
|
|
err := VerifyConstraints(c, maxLocalCSVDelay, r.partialState.Capacity)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2017-11-29 14:16:36 +01:00
|
|
|
}
|
|
|
|
|
2019-01-30 02:56:28 +01:00
|
|
|
// Our dust limit should always be less than or equal to our proposed
|
2018-05-14 20:04:34 +02:00
|
|
|
// channel reserve.
|
2021-09-23 22:37:38 +02:00
|
|
|
if responder && r.ourContribution.DustLimit > c.ChanReserve {
|
2018-12-10 22:58:34 +01:00
|
|
|
r.ourContribution.DustLimit = c.ChanReserve
|
2018-05-14 20:04:34 +02:00
|
|
|
}
|
|
|
|
|
2018-12-10 22:58:34 +01:00
|
|
|
r.ourContribution.ChanReserve = c.ChanReserve
|
|
|
|
r.ourContribution.MaxPendingAmount = c.MaxPendingAmount
|
|
|
|
r.ourContribution.MinHTLC = c.MinHTLC
|
|
|
|
r.ourContribution.MaxAcceptedHtlcs = c.MaxAcceptedHtlcs
|
|
|
|
r.ourContribution.CsvDelay = c.CsvDelay
|
2017-09-12 17:41:40 +02:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-09-23 22:37:38 +02:00
|
|
|
// validateReserveBounds checks that both ChannelReserve values are above both
|
|
|
|
// DustLimit values. This not only avoids stuck channels, but is also mandated
|
|
|
|
// by BOLT#02 even if it's not explicit. This returns true if the bounds are
|
|
|
|
// valid. This function should be called with the lock held.
|
|
|
|
func (r *ChannelReservation) validateReserveBounds() bool {
|
|
|
|
ourDustLimit := r.ourContribution.DustLimit
|
|
|
|
ourRequiredReserve := r.ourContribution.ChanReserve
|
|
|
|
theirDustLimit := r.theirContribution.DustLimit
|
|
|
|
theirRequiredReserve := r.theirContribution.ChanReserve
|
|
|
|
|
|
|
|
// We take the smaller of the two ChannelReserves and compare it
|
|
|
|
// against the larger of the two DustLimits.
|
|
|
|
minChanReserve := ourRequiredReserve
|
|
|
|
if minChanReserve > theirRequiredReserve {
|
|
|
|
minChanReserve = theirRequiredReserve
|
|
|
|
}
|
|
|
|
|
|
|
|
maxDustLimit := ourDustLimit
|
|
|
|
if maxDustLimit < theirDustLimit {
|
|
|
|
maxDustLimit = theirDustLimit
|
|
|
|
}
|
|
|
|
|
|
|
|
return minChanReserve >= maxDustLimit
|
|
|
|
}
|
|
|
|
|
2016-01-02 21:09:31 +01:00
|
|
|
// OurContribution returns the wallet's fully populated contribution to the
|
|
|
|
// pending payment channel. See 'ChannelContribution' for further details
|
|
|
|
// regarding the contents of a contribution.
|
2017-07-30 03:58:55 +02:00
|
|
|
//
|
2015-12-23 05:30:11 +01:00
|
|
|
// NOTE: This SHOULD NOT be modified.
|
2016-01-02 21:09:31 +01:00
|
|
|
// TODO(roasbeef): make copy?
|
2015-12-23 05:30:11 +01:00
|
|
|
func (r *ChannelReservation) OurContribution() *ChannelContribution {
|
|
|
|
r.RLock()
|
|
|
|
defer r.RUnlock()
|
2017-07-30 03:58:55 +02:00
|
|
|
|
2015-12-23 05:30:11 +01:00
|
|
|
return r.ourContribution
|
|
|
|
}
|
|
|
|
|
2016-11-24 09:49:18 +01:00
|
|
|
// ProcessContribution verifies the counterparty's contribution to the pending
|
2016-01-02 21:09:31 +01:00
|
|
|
// payment channel. As a result of this incoming message, lnwallet is able to
|
|
|
|
// build the funding transaction, and both commitment transactions. Once this
|
|
|
|
// message has been processed, all signatures to inputs to the funding
|
|
|
|
// transaction belonging to the wallet are available. Additionally, the wallet
|
|
|
|
// will generate a signature to the counterparty's version of the commitment
|
|
|
|
// transaction.
|
2015-12-23 05:30:11 +01:00
|
|
|
func (r *ChannelReservation) ProcessContribution(theirContribution *ChannelContribution) error {
|
|
|
|
errChan := make(chan error, 1)
|
|
|
|
|
|
|
|
r.wallet.msgChan <- &addContributionMsg{
|
|
|
|
pendingFundingID: r.reservationID,
|
|
|
|
contribution: theirContribution,
|
|
|
|
err: errChan,
|
|
|
|
}
|
|
|
|
|
|
|
|
return <-errChan
|
|
|
|
}
|
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
// IsPsbt returns true if there is a PSBT funding intent mapped to this
|
|
|
|
// reservation.
|
|
|
|
func (r *ChannelReservation) IsPsbt() bool {
|
|
|
|
_, ok := r.fundingIntent.(*chanfunding.PsbtIntent)
|
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
2020-08-21 12:47:52 +02:00
|
|
|
// IsCannedShim returns true if there is a canned shim funding intent mapped to
|
|
|
|
// this reservation.
|
|
|
|
func (r *ChannelReservation) IsCannedShim() bool {
|
|
|
|
_, ok := r.fundingIntent.(*chanfunding.ShimIntent)
|
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
2020-03-31 09:13:16 +02:00
|
|
|
// ProcessPsbt continues a previously paused funding flow that involves PSBT to
|
|
|
|
// construct the funding transaction. This method can be called once the PSBT is
|
|
|
|
// finalized and the signed transaction is available.
|
|
|
|
func (r *ChannelReservation) ProcessPsbt() error {
|
|
|
|
errChan := make(chan error, 1)
|
|
|
|
|
|
|
|
r.wallet.msgChan <- &continueContributionMsg{
|
|
|
|
pendingFundingID: r.reservationID,
|
|
|
|
err: errChan,
|
|
|
|
}
|
|
|
|
|
|
|
|
return <-errChan
|
|
|
|
}
|
|
|
|
|
|
|
|
// RemoteCanceled informs the PSBT funding state machine that the remote peer
|
|
|
|
// has canceled the pending reservation, likely due to a timeout.
|
|
|
|
func (r *ChannelReservation) RemoteCanceled() {
|
|
|
|
psbtIntent, ok := r.fundingIntent.(*chanfunding.PsbtIntent)
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
psbtIntent.RemoteCanceled()
|
|
|
|
}
|
|
|
|
|
2016-06-21 07:31:57 +02:00
|
|
|
// ProcessSingleContribution verifies, and records the initiator's contribution
|
|
|
|
// to this pending single funder channel. Internally, no further action is
|
|
|
|
// taken other than recording the initiator's contribution to the single funder
|
|
|
|
// channel.
|
|
|
|
func (r *ChannelReservation) ProcessSingleContribution(theirContribution *ChannelContribution) error {
|
|
|
|
errChan := make(chan error, 1)
|
|
|
|
|
|
|
|
r.wallet.msgChan <- &addSingleContributionMsg{
|
|
|
|
pendingFundingID: r.reservationID,
|
|
|
|
contribution: theirContribution,
|
|
|
|
err: errChan,
|
|
|
|
}
|
|
|
|
|
|
|
|
return <-errChan
|
|
|
|
}
|
|
|
|
|
2016-01-02 21:09:31 +01:00
|
|
|
// TheirContribution returns the counterparty's pending contribution to the
|
2017-07-30 03:58:55 +02:00
|
|
|
// payment channel. See 'ChannelContribution' for further details regarding the
|
|
|
|
// contents of a contribution. This attribute will ONLY be available after a
|
2018-09-28 05:59:59 +02:00
|
|
|
// call to .ProcessContribution().
|
2017-07-30 03:58:55 +02:00
|
|
|
//
|
2016-01-02 21:09:31 +01:00
|
|
|
// NOTE: This SHOULD NOT be modified.
|
2015-12-23 05:30:11 +01:00
|
|
|
func (r *ChannelReservation) TheirContribution() *ChannelContribution {
|
|
|
|
r.RLock()
|
|
|
|
defer r.RUnlock()
|
|
|
|
return r.theirContribution
|
|
|
|
}
|
|
|
|
|
2016-01-02 21:09:31 +01:00
|
|
|
// OurSignatures retrieves the wallet's signatures to all inputs to the funding
|
|
|
|
// transaction belonging to itself, and also a signature for the counterparty's
|
|
|
|
// version of the commitment transaction. The signatures for the wallet's
|
|
|
|
// inputs to the funding transaction are returned in sorted order according to
|
|
|
|
// BIP-69: https://github.com/bitcoin/bips/blob/master/bip-0069.mediawiki.
|
2017-07-30 03:58:55 +02:00
|
|
|
//
|
2016-01-02 21:09:31 +01:00
|
|
|
// NOTE: These signatures will only be populated after a call to
|
2018-09-28 05:59:59 +02:00
|
|
|
// .ProcessContribution()
|
2020-04-06 02:06:38 +02:00
|
|
|
func (r *ChannelReservation) OurSignatures() ([]*input.Script,
|
|
|
|
input.Signature) {
|
|
|
|
|
2015-12-23 05:30:11 +01:00
|
|
|
r.RLock()
|
|
|
|
defer r.RUnlock()
|
2016-05-04 04:46:53 +02:00
|
|
|
return r.ourFundingInputScripts, r.ourCommitmentSig
|
2015-12-23 05:30:11 +01:00
|
|
|
}
|
|
|
|
|
2017-07-30 03:58:55 +02:00
|
|
|
// CompleteReservation finalizes the pending channel reservation, transitioning
|
|
|
|
// from a pending payment channel, to an open payment channel. All passed
|
|
|
|
// signatures to the counterparty's inputs to the funding transaction will be
|
|
|
|
// fully verified. Signatures are expected to be passed in sorted order
|
|
|
|
// according to BIP-69:
|
|
|
|
// https://github.com/bitcoin/bips/blob/master/bip-0069.mediawiki.
|
|
|
|
// Additionally, verification is performed in order to ensure that the
|
|
|
|
// counterparty supplied a valid signature to our version of the commitment
|
2019-05-10 17:14:19 +02:00
|
|
|
// transaction. Once this method returns, callers should broadcast the
|
2018-07-26 15:16:47 +02:00
|
|
|
// created funding transaction, then call .WaitForChannelOpen() which will
|
|
|
|
// block until the funding transaction obtains the configured number of
|
|
|
|
// confirmations. Once the method unblocks, a LightningChannel instance is
|
|
|
|
// returned, marking the channel available for updates.
|
2019-01-16 15:47:43 +01:00
|
|
|
func (r *ChannelReservation) CompleteReservation(fundingInputScripts []*input.Script,
|
2020-04-06 02:06:38 +02:00
|
|
|
commitmentSig input.Signature) (*channeldb.OpenChannel, error) {
|
2015-12-29 06:58:06 +01:00
|
|
|
|
2016-06-21 07:31:57 +02:00
|
|
|
// TODO(roasbeef): add flag for watch or not?
|
2015-12-23 05:30:11 +01:00
|
|
|
errChan := make(chan error, 1)
|
2017-01-24 03:19:54 +01:00
|
|
|
completeChan := make(chan *channeldb.OpenChannel, 1)
|
2015-12-23 05:30:11 +01:00
|
|
|
|
|
|
|
r.wallet.msgChan <- &addCounterPartySigsMsg{
|
2016-05-04 04:46:53 +02:00
|
|
|
pendingFundingID: r.reservationID,
|
|
|
|
theirFundingInputScripts: fundingInputScripts,
|
|
|
|
theirCommitmentSig: commitmentSig,
|
2017-01-24 03:19:54 +01:00
|
|
|
completeChan: completeChan,
|
2016-05-04 04:46:53 +02:00
|
|
|
err: errChan,
|
2015-12-19 04:42:49 +01:00
|
|
|
}
|
2015-12-23 05:30:11 +01:00
|
|
|
|
2017-01-24 03:19:54 +01:00
|
|
|
return <-completeChan, <-errChan
|
2015-12-23 05:30:11 +01:00
|
|
|
}
|
|
|
|
|
2016-06-21 07:31:57 +02:00
|
|
|
// CompleteReservationSingle finalizes the pending single funder channel
|
2017-07-30 03:58:55 +02:00
|
|
|
// reservation. Using the funding outpoint of the constructed funding
|
|
|
|
// transaction, and the initiator's signature for our version of the commitment
|
|
|
|
// transaction, we are able to verify the correctness of our commitment
|
|
|
|
// transaction as crafted by the initiator. Once this method returns, our
|
|
|
|
// signature for the initiator's version of the commitment transaction is
|
|
|
|
// available via the .OurSignatures() method. As this method should only be
|
|
|
|
// called as a response to a single funder channel, only a commitment signature
|
|
|
|
// will be populated.
|
|
|
|
func (r *ChannelReservation) CompleteReservationSingle(fundingPoint *wire.OutPoint,
|
2020-04-06 02:06:38 +02:00
|
|
|
commitSig input.Signature) (*channeldb.OpenChannel, error) {
|
2016-11-16 21:54:27 +01:00
|
|
|
|
2016-06-21 07:31:57 +02:00
|
|
|
errChan := make(chan error, 1)
|
2017-01-24 03:19:54 +01:00
|
|
|
completeChan := make(chan *channeldb.OpenChannel, 1)
|
2016-06-21 07:31:57 +02:00
|
|
|
|
|
|
|
r.wallet.msgChan <- &addSingleFunderSigsMsg{
|
|
|
|
pendingFundingID: r.reservationID,
|
|
|
|
fundingOutpoint: fundingPoint,
|
|
|
|
theirCommitmentSig: commitSig,
|
2017-01-24 03:19:54 +01:00
|
|
|
completeChan: completeChan,
|
2016-06-21 07:31:57 +02:00
|
|
|
err: errChan,
|
|
|
|
}
|
|
|
|
|
2017-01-24 03:19:54 +01:00
|
|
|
return <-completeChan, <-errChan
|
2016-06-21 07:31:57 +02:00
|
|
|
}
|
|
|
|
|
2017-02-23 20:56:47 +01:00
|
|
|
// TheirSignatures returns the counterparty's signatures to all inputs to the
|
2016-01-02 21:09:31 +01:00
|
|
|
// funding transaction belonging to them, as well as their signature for the
|
|
|
|
// wallet's version of the commitment transaction. This methods is provided for
|
|
|
|
// additional verification, such as needed by tests.
|
2017-07-30 03:58:55 +02:00
|
|
|
//
|
2016-01-02 21:09:31 +01:00
|
|
|
// NOTE: These attributes will be unpopulated before a call to
|
|
|
|
// .CompleteReservation().
|
2020-04-06 02:06:38 +02:00
|
|
|
func (r *ChannelReservation) TheirSignatures() ([]*input.Script,
|
|
|
|
input.Signature) {
|
|
|
|
|
2015-12-23 05:30:11 +01:00
|
|
|
r.RLock()
|
|
|
|
defer r.RUnlock()
|
2016-05-04 04:46:53 +02:00
|
|
|
return r.theirFundingInputScripts, r.theirCommitmentSig
|
2015-12-23 05:30:11 +01:00
|
|
|
}
|
|
|
|
|
2016-01-02 21:09:31 +01:00
|
|
|
// FinalFundingTx returns the finalized, fully signed funding transaction for
|
|
|
|
// this reservation.
|
2016-06-21 07:31:57 +02:00
|
|
|
//
|
|
|
|
// NOTE: If this reservation was created as the non-initiator to a single
|
|
|
|
// funding workflow, then the full funding transaction will not be available.
|
|
|
|
// Instead we will only have the final outpoint of the funding transaction.
|
2015-12-23 05:30:11 +01:00
|
|
|
func (r *ChannelReservation) FinalFundingTx() *wire.MsgTx {
|
|
|
|
r.RLock()
|
|
|
|
defer r.RUnlock()
|
2016-06-21 07:31:57 +02:00
|
|
|
return r.fundingTx
|
|
|
|
}
|
|
|
|
|
|
|
|
// FundingOutpoint returns the outpoint of the funding transaction.
|
|
|
|
//
|
2017-09-25 20:25:58 +02:00
|
|
|
// NOTE: The pointer returned will only be set once the .ProcessContribution()
|
2016-06-21 07:31:57 +02:00
|
|
|
// method is called in the case of the initiator of a single funder workflow,
|
|
|
|
// and after the .CompleteReservationSingle() method is called in the case of
|
|
|
|
// a responder to a single funder workflow.
|
|
|
|
func (r *ChannelReservation) FundingOutpoint() *wire.OutPoint {
|
|
|
|
r.RLock()
|
|
|
|
defer r.RUnlock()
|
2017-07-30 03:58:55 +02:00
|
|
|
return &r.partialState.FundingOutpoint
|
2016-11-16 21:54:27 +01:00
|
|
|
}
|
|
|
|
|
2019-12-03 10:38:29 +01:00
|
|
|
// SetOurUpfrontShutdown sets the upfront shutdown address on our contribution.
|
|
|
|
func (r *ChannelReservation) SetOurUpfrontShutdown(shutdown lnwire.DeliveryAddress) {
|
|
|
|
r.Lock()
|
|
|
|
defer r.Unlock()
|
|
|
|
|
|
|
|
r.ourContribution.UpfrontShutdown = shutdown
|
|
|
|
}
|
|
|
|
|
2019-07-11 13:14:38 +02:00
|
|
|
// Capacity returns the channel capacity for this reservation.
|
|
|
|
func (r *ChannelReservation) Capacity() btcutil.Amount {
|
|
|
|
r.RLock()
|
|
|
|
defer r.RUnlock()
|
|
|
|
return r.partialState.Capacity
|
|
|
|
}
|
|
|
|
|
2021-07-15 01:29:29 +02:00
|
|
|
// LeaseExpiry returns the absolute expiration height for a leased channel using
|
|
|
|
// the script enforced commitment type. A zero value is returned when the
|
|
|
|
// channel is not using a script enforced lease commitment type.
|
|
|
|
func (r *ChannelReservation) LeaseExpiry() uint32 {
|
|
|
|
if !r.partialState.ChanType.HasLeaseExpiration() {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
return r.partialState.ThawHeight
|
|
|
|
}
|
|
|
|
|
2016-01-02 21:09:31 +01:00
|
|
|
// Cancel abandons this channel reservation. This method should be called in
|
|
|
|
// the scenario that communications with the counterparty break down. Upon
|
|
|
|
// cancellation, all resources previously reserved for this pending payment
|
|
|
|
// channel are returned to the free pool, allowing subsequent reservations to
|
|
|
|
// utilize the now freed resources.
|
2015-12-23 05:30:11 +01:00
|
|
|
func (r *ChannelReservation) Cancel() error {
|
|
|
|
errChan := make(chan error, 1)
|
|
|
|
r.wallet.msgChan <- &fundingReserveCancelMsg{
|
|
|
|
pendingFundingID: r.reservationID,
|
|
|
|
err: errChan,
|
|
|
|
}
|
|
|
|
|
|
|
|
return <-errChan
|
|
|
|
}
|
|
|
|
|
2022-09-29 10:56:44 +02:00
|
|
|
// VerifyConstraints is a helper function that can be used to check the sanity
|
|
|
|
// of various channel constraints.
|
|
|
|
func VerifyConstraints(c *channeldb.ChannelConstraints,
|
|
|
|
maxLocalCSVDelay uint16, channelCapacity btcutil.Amount) error {
|
|
|
|
|
|
|
|
// Fail if the csv delay for our funds exceeds our maximum.
|
|
|
|
if c.CsvDelay > maxLocalCSVDelay {
|
|
|
|
return ErrCsvDelayTooLarge(c.CsvDelay, maxLocalCSVDelay)
|
|
|
|
}
|
|
|
|
|
|
|
|
// The channel reserve should always be greater or equal to the dust
|
|
|
|
// limit. The reservation request should be denied if otherwise.
|
|
|
|
if c.DustLimit > c.ChanReserve {
|
|
|
|
return ErrChanReserveTooSmall(c.ChanReserve, c.DustLimit)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Validate against the maximum-sized witness script dust limit, and
|
|
|
|
// also ensure that the DustLimit is not too large.
|
|
|
|
maxWitnessLimit := DustLimitForSize(input.UnknownWitnessSize)
|
|
|
|
if c.DustLimit < maxWitnessLimit || c.DustLimit > 3*maxWitnessLimit {
|
|
|
|
return ErrInvalidDustLimit(c.DustLimit)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fail if we consider the channel reserve to be too large. We
|
|
|
|
// currently fail if it is greater than 20% of the channel capacity.
|
|
|
|
maxChanReserve := channelCapacity / 5
|
|
|
|
if c.ChanReserve > maxChanReserve {
|
|
|
|
return ErrChanReserveTooLarge(c.ChanReserve, maxChanReserve)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fail if the minimum HTLC value is too large. If this is too large,
|
|
|
|
// the channel won't be useful for sending small payments. This limit
|
|
|
|
// is currently set to maxValueInFlight, effectively letting the remote
|
|
|
|
// setting this as large as it wants.
|
|
|
|
if c.MinHTLC > c.MaxPendingAmount {
|
|
|
|
return ErrMinHtlcTooLarge(c.MinHTLC, c.MaxPendingAmount)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fail if maxHtlcs is above the maximum allowed number of 483. This
|
|
|
|
// number is specified in BOLT-02.
|
|
|
|
if c.MaxAcceptedHtlcs > uint16(input.MaxHTLCNumber/2) {
|
|
|
|
return ErrMaxHtlcNumTooLarge(
|
|
|
|
c.MaxAcceptedHtlcs, uint16(input.MaxHTLCNumber/2),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fail if we consider maxHtlcs too small. If this is too small we
|
|
|
|
// cannot offer many HTLCs to the remote.
|
|
|
|
const minNumHtlc = 5
|
|
|
|
if c.MaxAcceptedHtlcs < minNumHtlc {
|
|
|
|
return ErrMaxHtlcNumTooSmall(c.MaxAcceptedHtlcs, minNumHtlc)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fail if we consider maxValueInFlight too small. We currently require
|
|
|
|
// the remote to at least allow minNumHtlc * minHtlc in flight.
|
|
|
|
if c.MaxPendingAmount < minNumHtlc*c.MinHTLC {
|
|
|
|
return ErrMaxValueInFlightTooSmall(
|
|
|
|
c.MaxPendingAmount, minNumHtlc*c.MinHTLC,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-01-23 00:06:28 +01:00
|
|
|
// OpenChannelDetails wraps the finalized fully confirmed channel which
|
|
|
|
// resulted from a ChannelReservation instance with details concerning exactly
|
|
|
|
// _where_ in the chain the channel was ultimately opened.
|
|
|
|
type OpenChannelDetails struct {
|
|
|
|
// Channel is the active channel created by an instance of a
|
|
|
|
// ChannelReservation and the required funding workflow.
|
|
|
|
Channel *LightningChannel
|
|
|
|
|
2017-07-30 03:58:55 +02:00
|
|
|
// ConfirmationHeight is the block height within the chain that
|
|
|
|
// included the channel.
|
2017-01-23 00:06:28 +01:00
|
|
|
ConfirmationHeight uint32
|
|
|
|
|
|
|
|
// TransactionIndex is the index within the confirming block that the
|
|
|
|
// transaction resides.
|
|
|
|
TransactionIndex uint32
|
|
|
|
}
|