forked from lightningnetwork/lightning-onion
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpath.go
258 lines (209 loc) · 8.28 KB
/
path.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
package sphinx
import (
"errors"
"fmt"
"github.com/btcsuite/btcd/btcec/v2"
)
const (
// NumMaxHops is the maximum path length. There is a maximum of 1300
// bytes in the routing info block. Legacy hop payloads are always 65
// bytes, while tlv payloads are at least 47 bytes (tlvlen 1, amt 2,
// timelock 2, nextchan 10, hmac 32) for the intermediate hops and 37
// bytes (tlvlen 1, amt 2, timelock 2, hmac 32) for the exit hop. The
// maximum path length can therefore only be reached by using tlv
// payloads only. With that, the maximum number of intermediate hops
// is: Floor((1300 - 37) / 47) = 26. Including the exit hop, the
// maximum path length is 27 hops.
NumMaxHops = 27
routeBlindingHMACKey = "blinded_node_id"
)
// PaymentPath represents a series of hops within the Lightning Network
// starting at a sender and terminating at a receiver. Each hop contains a set
// of mandatory data which contains forwarding instructions for that hop.
// Additionally, we can also transmit additional data to each hop by utilizing
// the un-used hops (see TrueRouteLength()) to pack in additional data. In
// order to do this, we encrypt the several hops with the same node public key,
// and unroll the extra data into the space used for route forwarding
// information.
type PaymentPath [NumMaxHops]OnionHop
// OnionHop represents an abstract hop (a link between two nodes) within the
// Lightning Network. A hop is composed of the incoming node (able to decrypt
// the encrypted routing information), and the routing information itself.
// Optionally, the crafter of a route can indicate that additional data aside
// from the routing information is be delivered, which will manifest as
// additional hops to pack the data.
type OnionHop struct {
// NodePub is the target node for this hop. The payload will enter this
// hop, it'll decrypt the routing information, and hand off the
// internal packet to the next hop.
NodePub btcec.PublicKey
// HopPayload is the opaque payload provided to this node. If the
// HopData above is specified, then it'll be packed into this payload.
HopPayload HopPayload
}
// IsEmpty returns true if the hop isn't populated.
func (o OnionHop) IsEmpty() bool {
return o.NodePub.X().BitLen() == 0 || o.NodePub.Y().BitLen() == 0
}
// NodeKeys returns a slice pointing to node keys that this route comprises of.
// The size of the returned slice will be TrueRouteLength().
func (p *PaymentPath) NodeKeys() []*btcec.PublicKey {
var nodeKeys [NumMaxHops]*btcec.PublicKey
routeLen := p.TrueRouteLength()
for i := 0; i < routeLen; i++ {
nodeKeys[i] = &p[i].NodePub
}
return nodeKeys[:routeLen]
}
// TrueRouteLength returns the "true" length of the PaymentPath. The max
// payment path is NumMaxHops size, but in practice routes are much smaller.
// This method will return the number of actual hops (nodes) involved in this
// route. For references, a direct path has a length of 1, path through an
// intermediate node has a length of 2 (3 nodes involved).
func (p *PaymentPath) TrueRouteLength() int {
var routeLength int
for _, hop := range p {
// When we hit the first empty hop, we know we're now in the
// zero'd out portion of the array.
if hop.IsEmpty() {
return routeLength
}
routeLength++
}
return routeLength
}
// TotalPayloadSize returns the sum of the size of each payload in the "true"
// route.
func (p *PaymentPath) TotalPayloadSize() int {
var totalSize int
for _, hop := range p {
if hop.IsEmpty() {
continue
}
totalSize += hop.HopPayload.NumBytes()
}
return totalSize
}
// BlindedPath represents all the data that the creator of a blinded path must
// transmit to the builder of route that will send to this path.
type BlindedPath struct {
// IntroductionPoint is the real node ID of the first hop in the blinded
// path. The sender should be able to find this node in the network
// graph and route to it.
IntroductionPoint *btcec.PublicKey
// BlindingPoint is the first ephemeral blinding point. This is the
// point that the introduction node will use in order to create a shared
// secret with the builder of the blinded route. This point will need
// to be communicated to the introduction node by the sender in some
// way.
BlindingPoint *btcec.PublicKey
// BlindedHops is a list of ordered BlindedHopInfo. Each entry
// represents a hop in the blinded path along with the encrypted data to
// be sent to that node. Note that the first entry in the list
// represents the introduction point of the path and so the node ID of
// this point does not strictly need to be transmitted to the sender
// since they will be able to derive the point using the BlindingPoint.
BlindedHops []*BlindedHopInfo
}
// BlindedHopInfo represents a blinded node pub key along with the encrypted
// data for a node in a blinded route.
type BlindedHopInfo struct {
// BlindedNodePub is the blinded public key of the node in the blinded
// route.
BlindedNodePub *btcec.PublicKey
// CipherText is the encrypted payload to be transported to the hop in
// the blinded route.
CipherText []byte
}
// HopInfo represents a real node pub key along with the plaintext data for a
// node in a blinded route.
type HopInfo struct {
// NodePub is the real public key of the node in the blinded route.
NodePub *btcec.PublicKey
// PlainText is the un-encrypted payload to be transported to the hop
// the blinded route.
PlainText []byte
}
// Encrypt uses the given sharedSecret to blind the public key of the node and
// encrypt the payload and returns the resulting BlindedHopInfo.
func (i *HopInfo) Encrypt(sharedSecret Hash256) (*BlindedHopInfo, error) {
blindedData, err := encryptBlindedHopData(sharedSecret, i.PlainText)
if err != nil {
return nil, err
}
return &BlindedHopInfo{
BlindedNodePub: blindNodeID(sharedSecret, i.NodePub),
CipherText: blindedData,
}, nil
}
// BuildBlindedPath creates a new BlindedPath from a session key along with a
// list of HopInfo representing the nodes in the blinded path. The first hop in
// paymentPath is expected to be the introduction node.
func BuildBlindedPath(sessionKey *btcec.PrivateKey,
paymentPath []*HopInfo) (*BlindedPath, error) {
if len(paymentPath) < 1 {
return nil, errors.New("at least 1 hop is required to create " +
"a blinded path")
}
bp := &BlindedPath{
IntroductionPoint: paymentPath[0].NodePub,
BlindingPoint: sessionKey.PubKey(),
BlindedHops: make([]*BlindedHopInfo, len(paymentPath)),
}
keys := make([]*btcec.PublicKey, len(paymentPath))
for i, p := range paymentPath {
keys[i] = p.NodePub
}
hopSharedSecrets, err := generateSharedSecrets(keys, sessionKey)
if err != nil {
return nil, fmt.Errorf("error generating shared secret: %v",
err)
}
for i, hop := range paymentPath {
blindedInfo, err := hop.Encrypt(hopSharedSecrets[i])
if err != nil {
return nil, err
}
bp.BlindedHops[i] = blindedInfo
}
return bp, nil
}
// blindNodeID blinds the given public key using the provided shared secret.
func blindNodeID(sharedSecret Hash256,
pubKey *btcec.PublicKey) *btcec.PublicKey {
blindingFactorBytes := generateKey(routeBlindingHMACKey, &sharedSecret)
var blindingFactor btcec.ModNScalar
blindingFactor.SetBytes(&blindingFactorBytes)
return blindGroupElement(pubKey, blindingFactor)
}
// encryptBlindedHopData blinds/encrypts the given plain text data using the
// provided shared secret.
func encryptBlindedHopData(sharedSecret Hash256, plainTxt []byte) ([]byte,
error) {
rhoKey := generateKey("rho", &sharedSecret)
return chacha20polyEncrypt(rhoKey[:], plainTxt)
}
// decryptBlindedHopData decrypts the data encrypted by the creator of the
// blinded route.
func decryptBlindedHopData(privKey SingleKeyECDH, ephemPub *btcec.PublicKey,
encryptedData []byte) ([]byte, error) {
ss, err := privKey.ECDH(ephemPub)
if err != nil {
return nil, err
}
ssHash := Hash256(ss)
rho := generateKey("rho", &ssHash)
return chacha20polyDecrypt(rho[:], encryptedData)
}
// NextEphemeral computes the next ephemeral key given the current ephemeral
// key and this node's private key.
func NextEphemeral(privKey SingleKeyECDH,
ephemPub *btcec.PublicKey) (*btcec.PublicKey, error) {
ss, err := privKey.ECDH(ephemPub)
if err != nil {
return nil, err
}
blindingFactor := computeBlindingFactor(ephemPub, ss[:])
nextEphem := blindGroupElement(ephemPub, blindingFactor)
return nextEphem, nil
}