mirror of
https://github.com/lightningnetwork/lnd.git
synced 2025-01-19 05:45:21 +01:00
218 lines
6.2 KiB
Go
218 lines
6.2 KiB
Go
package contractcourt
|
|
|
|
import (
|
|
"errors"
|
|
"io"
|
|
"sync"
|
|
|
|
"github.com/btcsuite/btcd/btcutil"
|
|
"github.com/btcsuite/btcd/chaincfg/chainhash"
|
|
"github.com/btcsuite/btcd/wire"
|
|
"github.com/lightningnetwork/lnd/channeldb"
|
|
"github.com/lightningnetwork/lnd/fn"
|
|
"github.com/lightningnetwork/lnd/input"
|
|
"github.com/lightningnetwork/lnd/sweep"
|
|
)
|
|
|
|
// anchorResolver is a resolver that will attempt to sweep our anchor output.
|
|
type anchorResolver struct {
|
|
// anchorSignDescriptor contains the information that is required to
|
|
// sweep the anchor.
|
|
anchorSignDescriptor input.SignDescriptor
|
|
|
|
// anchor is the outpoint on the commitment transaction.
|
|
anchor wire.OutPoint
|
|
|
|
// resolved reflects if the contract has been fully resolved or not.
|
|
resolved bool
|
|
|
|
// broadcastHeight is the height that the original contract was
|
|
// broadcast to the main-chain at. We'll use this value to bound any
|
|
// historical queries to the chain for spends/confirmations.
|
|
broadcastHeight uint32
|
|
|
|
// chanPoint is the channel point of the original contract.
|
|
chanPoint wire.OutPoint
|
|
|
|
// chanType denotes the type of channel the contract belongs to.
|
|
chanType channeldb.ChannelType
|
|
|
|
// currentReport stores the current state of the resolver for reporting
|
|
// over the rpc interface.
|
|
currentReport ContractReport
|
|
|
|
// reportLock prevents concurrent access to the resolver report.
|
|
reportLock sync.Mutex
|
|
|
|
contractResolverKit
|
|
}
|
|
|
|
// newAnchorResolver instantiates a new anchor resolver.
|
|
func newAnchorResolver(anchorSignDescriptor input.SignDescriptor,
|
|
anchor wire.OutPoint, broadcastHeight uint32,
|
|
chanPoint wire.OutPoint, resCfg ResolverConfig) *anchorResolver {
|
|
|
|
amt := btcutil.Amount(anchorSignDescriptor.Output.Value)
|
|
|
|
report := ContractReport{
|
|
Outpoint: anchor,
|
|
Type: ReportOutputAnchor,
|
|
Amount: amt,
|
|
LimboBalance: amt,
|
|
RecoveredBalance: 0,
|
|
}
|
|
|
|
r := &anchorResolver{
|
|
contractResolverKit: *newContractResolverKit(resCfg),
|
|
anchorSignDescriptor: anchorSignDescriptor,
|
|
anchor: anchor,
|
|
broadcastHeight: broadcastHeight,
|
|
chanPoint: chanPoint,
|
|
currentReport: report,
|
|
}
|
|
|
|
r.initLogger(r)
|
|
|
|
return r
|
|
}
|
|
|
|
// ResolverKey returns an identifier which should be globally unique for this
|
|
// particular resolver within the chain the original contract resides within.
|
|
func (c *anchorResolver) ResolverKey() []byte {
|
|
// The anchor resolver is stateless and doesn't need a database key.
|
|
return nil
|
|
}
|
|
|
|
// Resolve offers the anchor output to the sweeper and waits for it to be swept.
|
|
func (c *anchorResolver) Resolve(_ bool) (ContractResolver, error) {
|
|
// Attempt to update the sweep parameters to the post-confirmation
|
|
// situation. We don't want to force sweep anymore, because the anchor
|
|
// lost its special purpose to get the commitment confirmed. It is just
|
|
// an output that we want to sweep only if it is economical to do so.
|
|
//
|
|
// An exclusive group is not necessary anymore, because we know that
|
|
// this is the only anchor that can be swept.
|
|
//
|
|
// We also clear the parent tx information for cpfp, because the
|
|
// commitment tx is confirmed.
|
|
//
|
|
// After a restart or when the remote force closes, the sweeper is not
|
|
// yet aware of the anchor. In that case, it will be added as new input
|
|
// to the sweeper.
|
|
witnessType := input.CommitmentAnchor
|
|
|
|
// For taproot channels, we need to use the proper witness type.
|
|
if c.chanType.IsTaproot() {
|
|
witnessType = input.TaprootAnchorSweepSpend
|
|
}
|
|
|
|
anchorInput := input.MakeBaseInput(
|
|
&c.anchor, witnessType, &c.anchorSignDescriptor,
|
|
c.broadcastHeight, nil,
|
|
)
|
|
|
|
resultChan, err := c.Sweeper.SweepInput(
|
|
&anchorInput,
|
|
sweep.Params{
|
|
// For normal anchor sweeping, the budget is 330 sats.
|
|
Budget: btcutil.Amount(
|
|
anchorInput.SignDesc().Output.Value,
|
|
),
|
|
|
|
// There's no rush to sweep the anchor, so we use a nil
|
|
// deadline here.
|
|
DeadlineHeight: fn.None[int32](),
|
|
},
|
|
)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var (
|
|
outcome channeldb.ResolverOutcome
|
|
spendTx *chainhash.Hash
|
|
)
|
|
|
|
select {
|
|
case sweepRes := <-resultChan:
|
|
switch sweepRes.Err {
|
|
// Anchor was swept successfully.
|
|
case nil:
|
|
sweepTxID := sweepRes.Tx.TxHash()
|
|
|
|
spendTx = &sweepTxID
|
|
outcome = channeldb.ResolverOutcomeClaimed
|
|
|
|
// Anchor was swept by someone else. This is possible after the
|
|
// 16 block csv lock.
|
|
case sweep.ErrRemoteSpend:
|
|
c.log.Warnf("our anchor spent by someone else")
|
|
outcome = channeldb.ResolverOutcomeUnclaimed
|
|
|
|
// An unexpected error occurred.
|
|
default:
|
|
c.log.Errorf("unable to sweep anchor: %v", sweepRes.Err)
|
|
|
|
return nil, sweepRes.Err
|
|
}
|
|
|
|
case <-c.quit:
|
|
return nil, errResolverShuttingDown
|
|
}
|
|
|
|
// Update report to reflect that funds are no longer in limbo.
|
|
c.reportLock.Lock()
|
|
if outcome == channeldb.ResolverOutcomeClaimed {
|
|
c.currentReport.RecoveredBalance = c.currentReport.LimboBalance
|
|
}
|
|
c.currentReport.LimboBalance = 0
|
|
report := c.currentReport.resolverReport(
|
|
spendTx, channeldb.ResolverTypeAnchor, outcome,
|
|
)
|
|
c.reportLock.Unlock()
|
|
|
|
c.resolved = true
|
|
return nil, c.PutResolverReport(nil, report)
|
|
}
|
|
|
|
// Stop signals the resolver to cancel any current resolution processes, and
|
|
// suspend.
|
|
//
|
|
// NOTE: Part of the ContractResolver interface.
|
|
func (c *anchorResolver) Stop() {
|
|
close(c.quit)
|
|
}
|
|
|
|
// IsResolved returns true if the stored state in the resolve is fully
|
|
// resolved. In this case the target output can be forgotten.
|
|
//
|
|
// NOTE: Part of the ContractResolver interface.
|
|
func (c *anchorResolver) IsResolved() bool {
|
|
return c.resolved
|
|
}
|
|
|
|
// SupplementState allows the user of a ContractResolver to supplement it with
|
|
// state required for the proper resolution of a contract.
|
|
//
|
|
// NOTE: Part of the ContractResolver interface.
|
|
func (c *anchorResolver) SupplementState(state *channeldb.OpenChannel) {
|
|
c.chanType = state.ChanType
|
|
}
|
|
|
|
// report returns a report on the resolution state of the contract.
|
|
func (c *anchorResolver) report() *ContractReport {
|
|
c.reportLock.Lock()
|
|
defer c.reportLock.Unlock()
|
|
|
|
reportCopy := c.currentReport
|
|
return &reportCopy
|
|
}
|
|
|
|
func (c *anchorResolver) Encode(w io.Writer) error {
|
|
return errors.New("serialization not supported")
|
|
}
|
|
|
|
// A compile time assertion to ensure anchorResolver meets the
|
|
// ContractResolver interface.
|
|
var _ ContractResolver = (*anchorResolver)(nil)
|