• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

lightningnetwork / lnd / 18016273007

25 Sep 2025 05:55PM UTC coverage: 54.653% (-12.0%) from 66.622%
18016273007

Pull #10248

github

web-flow
Merge 128443298 into b09b20c69
Pull Request #10248: Enforce TLV when creating a Route

25 of 30 new or added lines in 4 files covered. (83.33%)

23906 existing lines in 281 files now uncovered.

109536 of 200421 relevant lines covered (54.65%)

21816.97 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

72.83
/input/taproot.go
1
package input
2

3
import (
4
        "fmt"
5

6
        "github.com/btcsuite/btcd/btcec/v2"
7
        "github.com/btcsuite/btcd/btcec/v2/schnorr"
8
        "github.com/btcsuite/btcd/txscript"
9
        "github.com/btcsuite/btcd/wire"
10
        "github.com/btcsuite/btcwallet/waddrmgr"
11
        "github.com/lightningnetwork/lnd/fn/v2"
12
)
13

14
const (
15
        // PubKeyFormatCompressedOdd is the identifier prefix byte for a public
16
        // key whose Y coordinate is odd when serialized in the compressed
17
        // format per section 2.3.4 of
18
        // [SEC1](https://secg.org/sec1-v2.pdf#subsubsection.2.3.4).
19
        // This is copied from the github.com/decred/dcrd/dcrec/secp256k1/v4 to
20
        // avoid needing to directly reference (and by accident pull in
21
        // incompatible crypto primitives) the package.
22
        PubKeyFormatCompressedOdd byte = 0x03
23
)
24

25
// AuxTapLeaf is a type alias for an optional tapscript leaf that may be added
26
// to the tapscript tree of HTLC and commitment outputs.
27
type AuxTapLeaf = fn.Option[txscript.TapLeaf]
28

29
// NoneTapLeaf returns an empty optional tapscript leaf.
30
func NoneTapLeaf() AuxTapLeaf {
19✔
31
        return fn.None[txscript.TapLeaf]()
19✔
32
}
19✔
33

34
// HtlcIndex represents the monotonically increasing counter that is used to
35
// identify HTLCs created by a peer.
36
type HtlcIndex = uint64
37

38
// HtlcAuxLeaf is a type that represents an auxiliary leaf for an HTLC output.
39
// An HTLC may have up to two aux leaves: one for the output on the commitment
40
// transaction, and one for the second level HTLC.
41
type HtlcAuxLeaf struct {
42
        AuxTapLeaf
43

44
        // SecondLevelLeaf is the auxiliary leaf for the second level HTLC
45
        // success or timeout transaction.
46
        SecondLevelLeaf AuxTapLeaf
47
}
48

49
// HtlcAuxLeaves is a type alias for a map of optional tapscript leaves.
50
type HtlcAuxLeaves = map[HtlcIndex]HtlcAuxLeaf
51

52
// NewTxSigHashesV0Only returns a new txscript.TxSigHashes instance that will
53
// only calculate the sighash midstate values for segwit v0 inputs and can
54
// therefore never be used for transactions that want to spend segwit v1
55
// (taproot) inputs.
56
func NewTxSigHashesV0Only(tx *wire.MsgTx) *txscript.TxSigHashes {
6,269✔
57
        // The canned output fetcher returns a wire.TxOut instance with the
6,269✔
58
        // given pk script and amount. We can get away with nil since the first
6,269✔
59
        // thing the TxSigHashes constructor checks is the length of the pk
6,269✔
60
        // script and whether it matches taproot output script length. If the
6,269✔
61
        // length doesn't match it assumes v0 inputs only.
6,269✔
62
        nilFetcher := txscript.NewCannedPrevOutputFetcher(nil, 0)
6,269✔
63
        return txscript.NewTxSigHashes(tx, nilFetcher)
6,269✔
64
}
6,269✔
65

66
// MultiPrevOutFetcher returns a txscript.MultiPrevOutFetcher for the given set
67
// of inputs.
68
func MultiPrevOutFetcher(inputs []Input) (*txscript.MultiPrevOutFetcher, error) {
63✔
69
        fetcher := txscript.NewMultiPrevOutFetcher(nil)
63✔
70
        for _, inp := range inputs {
157✔
71
                op := inp.OutPoint()
94✔
72
                desc := inp.SignDesc()
94✔
73

94✔
74
                if op == EmptyOutPoint {
94✔
75
                        return nil, fmt.Errorf("missing input outpoint")
×
76
                }
×
77

78
                if desc == nil || desc.Output == nil {
94✔
79
                        return nil, fmt.Errorf("missing input utxo information")
×
80
                }
×
81

82
                fetcher.AddPrevOut(op, desc.Output)
94✔
83
        }
84

85
        return fetcher, nil
63✔
86
}
87

88
// TapscriptFullTree creates a waddrmgr.Tapscript for the given internal key and
89
// tree leaves.
90
func TapscriptFullTree(internalKey *btcec.PublicKey,
91
        allTreeLeaves ...txscript.TapLeaf) *waddrmgr.Tapscript {
1✔
92

1✔
93
        tree := txscript.AssembleTaprootScriptTree(allTreeLeaves...)
1✔
94
        rootHash := tree.RootNode.TapHash()
1✔
95
        tapKey := txscript.ComputeTaprootOutputKey(internalKey, rootHash[:])
1✔
96

1✔
97
        var outputKeyYIsOdd bool
1✔
98
        if tapKey.SerializeCompressed()[0] == PubKeyFormatCompressedOdd {
2✔
99
                outputKeyYIsOdd = true
1✔
100
        }
1✔
101

102
        return &waddrmgr.Tapscript{
1✔
103
                Type: waddrmgr.TapscriptTypeFullTree,
1✔
104
                ControlBlock: &txscript.ControlBlock{
1✔
105
                        InternalKey:     internalKey,
1✔
106
                        OutputKeyYIsOdd: outputKeyYIsOdd,
1✔
107
                        LeafVersion:     txscript.BaseLeafVersion,
1✔
108
                },
1✔
109
                Leaves: allTreeLeaves,
1✔
110
        }
1✔
111
}
112

113
// TapscriptPartialReveal creates a waddrmgr.Tapscript for the given internal
114
// key and revealed script.
115
func TapscriptPartialReveal(internalKey *btcec.PublicKey,
116
        revealedLeaf txscript.TapLeaf,
117
        inclusionProof []byte) *waddrmgr.Tapscript {
1✔
118

1✔
119
        controlBlock := &txscript.ControlBlock{
1✔
120
                InternalKey:    internalKey,
1✔
121
                LeafVersion:    txscript.BaseLeafVersion,
1✔
122
                InclusionProof: inclusionProof,
1✔
123
        }
1✔
124
        rootHash := controlBlock.RootHash(revealedLeaf.Script)
1✔
125
        tapKey := txscript.ComputeTaprootOutputKey(internalKey, rootHash)
1✔
126

1✔
127
        if tapKey.SerializeCompressed()[0] == PubKeyFormatCompressedOdd {
2✔
128
                controlBlock.OutputKeyYIsOdd = true
1✔
129
        }
1✔
130

131
        return &waddrmgr.Tapscript{
1✔
132
                Type:           waddrmgr.TapscriptTypePartialReveal,
1✔
133
                ControlBlock:   controlBlock,
1✔
134
                RevealedScript: revealedLeaf.Script,
1✔
135
        }
1✔
136
}
137

138
// TapscriptRootHashOnly creates a waddrmgr.Tapscript for the given internal key
139
// and root hash.
140
func TapscriptRootHashOnly(internalKey *btcec.PublicKey,
UNCOV
141
        rootHash []byte) *waddrmgr.Tapscript {
×
UNCOV
142

×
UNCOV
143
        controlBlock := &txscript.ControlBlock{
×
UNCOV
144
                InternalKey: internalKey,
×
UNCOV
145
        }
×
UNCOV
146

×
UNCOV
147
        tapKey := txscript.ComputeTaprootOutputKey(internalKey, rootHash)
×
UNCOV
148
        if tapKey.SerializeCompressed()[0] == PubKeyFormatCompressedOdd {
×
UNCOV
149
                controlBlock.OutputKeyYIsOdd = true
×
UNCOV
150
        }
×
151

UNCOV
152
        return &waddrmgr.Tapscript{
×
UNCOV
153
                Type:         waddrmgr.TaprootKeySpendRootHash,
×
UNCOV
154
                ControlBlock: controlBlock,
×
UNCOV
155
                RootHash:     rootHash,
×
UNCOV
156
        }
×
157
}
158

159
// TapscriptFullKeyOnly creates a waddrmgr.Tapscript for the given full Taproot
160
// key.
UNCOV
161
func TapscriptFullKeyOnly(taprootKey *btcec.PublicKey) *waddrmgr.Tapscript {
×
UNCOV
162
        return &waddrmgr.Tapscript{
×
UNCOV
163
                Type:          waddrmgr.TaprootFullKeyOnly,
×
UNCOV
164
                FullOutputKey: taprootKey,
×
UNCOV
165
        }
×
UNCOV
166
}
×
167

168
// PayToTaprootScript creates a new script to pay to a version 1 (taproot)
169
// witness program. The passed public key will be serialized as an x-only key
170
// to create the witness program.
171
func PayToTaprootScript(taprootKey *btcec.PublicKey) ([]byte, error) {
4,448✔
172
        builder := txscript.NewScriptBuilder()
4,448✔
173

4,448✔
174
        builder.AddOp(txscript.OP_1)
4,448✔
175
        builder.AddData(schnorr.SerializePubKey(taprootKey))
4,448✔
176

4,448✔
177
        return builder.Script()
4,448✔
178
}
4,448✔
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2025 Coveralls, Inc