@@ -20,6 +20,7 @@ import (
20
20
"github.com/btcsuite/btcd/chaincfg/chainhash"
21
21
"github.com/btcsuite/btcd/wire"
22
22
"github.com/davecgh/go-spew/spew"
23
+ "github.com/lightninglabs/lndclient"
23
24
"github.com/lightninglabs/taproot-assets/address"
24
25
"github.com/lightninglabs/taproot-assets/asset"
25
26
"github.com/lightninglabs/taproot-assets/commitment"
@@ -52,6 +53,28 @@ const (
52
53
// ackTimeout is the amount of time we'll wait to receive the protocol
53
54
// level ACK from the remote party before timing out.
54
55
ackTimeout = time .Second * 30
56
+
57
+ // maxNumAssetIDs is the maximum number of fungible asset pieces (asset
58
+ // IDs) that can be committed to a single channel. The number needs to
59
+ // be limited to prevent the number of required HTLC signatures to be
60
+ // too large for a single CommitSig wire message to carry them. This
61
+ // value is tightly coupled with the number of HTLCs that can be added
62
+ // to a channel at the same time (maxNumHTLCs). The values were
63
+ // determined with the TestMaxCommitSigMsgSize test in
64
+ // aux_leaf_signer_test.go then a set was chosen that would allow for
65
+ // a decent number of HTLCs (and also a number that is divisible by two
66
+ // because each side will only be allowed to add half of the total).
67
+ maxNumAssetIDs = 3
68
+
69
+ // maxNumHTLCs is the maximum number of HTLCs there can be in an asset
70
+ // channel to avoid the number of signatures exceeding the maximum
71
+ // message size of a CommitSig message. See maxNumAssetIDs for more
72
+ // information.
73
+ maxNumHTLCs = 166
74
+
75
+ // maxNumHTLCsPerParty is the maximum number of HTLCs that can be added
76
+ // by a single party to a channel.
77
+ maxNumHTLCsPerParty = maxNumHTLCs / 2
55
78
)
56
79
57
80
// ErrorReporter is used to report an error back to the caller and/or peer that
@@ -138,6 +161,11 @@ type PsbtChannelFunder interface {
138
161
// process. Afterward, the funding transaction should be signed and
139
162
// broadcast.
140
163
OpenChannel (context.Context , OpenChanReq ) (AssetChanIntent , error )
164
+
165
+ // ChannelAcceptor is used to accept and potentially influence
166
+ // parameters of incoming channels.
167
+ ChannelAcceptor (ctx context.Context ,
168
+ acceptor lndclient.AcceptorFunction ) (chan error , error )
141
169
}
142
170
143
171
// TxPublisher is an interface used to publish transactions.
@@ -222,6 +250,9 @@ type FundingControllerCfg struct {
222
250
// FeatureBits is used to verify that the peer has the required feature
223
251
// to fund asset channels.
224
252
FeatureBits FeatureBitVerifer
253
+
254
+ // ErrChan is used to report errors back to the main server.
255
+ ErrChan chan <- error
225
256
}
226
257
227
258
// bindFundingReq is a request to bind a pending channel ID to a complete aux
@@ -298,6 +329,36 @@ func (f *FundingController) Start() error {
298
329
f .Wg .Add (1 )
299
330
go f .chanFunder ()
300
331
332
+ f .Wg .Add (1 )
333
+ go func () {
334
+ defer f .Wg .Done ()
335
+
336
+ ctx , cancel := f .WithCtxQuitNoTimeout ()
337
+ defer cancel ()
338
+
339
+ errChan , err := f .cfg .ChannelFunder .ChannelAcceptor (
340
+ ctx , f .channelAcceptor ,
341
+ )
342
+ if err != nil {
343
+ err = fmt .Errorf ("unable to start channel acceptor: %w" ,
344
+ err )
345
+ f .cfg .ErrChan <- err
346
+ return
347
+ }
348
+
349
+ // We'll accept channels for as long as the funding controller
350
+ // is running or until we receive an error.
351
+ select {
352
+ case err := <- errChan :
353
+ err = fmt .Errorf ("channel acceptor error: %w" , err )
354
+ f .cfg .ErrChan <- err
355
+
356
+ case <- f .Quit :
357
+ log .Infof ("Stopping channel acceptor, funding " +
358
+ "controller shutting down" )
359
+ }
360
+ }()
361
+
301
362
return nil
302
363
}
303
364
@@ -1008,10 +1069,11 @@ func (f *FundingController) completeChannelFunding(ctx context.Context,
1008
1069
// Now that we have the initial PSBT template, we can start the funding
1009
1070
// flow with lnd.
1010
1071
fundingReq := OpenChanReq {
1011
- ChanAmt : 100_000 ,
1012
- PushAmt : fundingState .pushAmt ,
1013
- PeerPub : fundingState .peerPub ,
1014
- TempPID : fundingState .pid ,
1072
+ ChanAmt : 100_000 ,
1073
+ PushAmt : fundingState .pushAmt ,
1074
+ PeerPub : fundingState .peerPub ,
1075
+ TempPID : fundingState .pid ,
1076
+ RemoteMaxHtlc : maxNumHTLCsPerParty ,
1015
1077
}
1016
1078
assetChanIntent , err := f .cfg .ChannelFunder .OpenChannel (ctx , fundingReq )
1017
1079
if err != nil {
@@ -1682,6 +1744,43 @@ func (f *FundingController) chanFunder() {
1682
1744
}
1683
1745
}
1684
1746
1747
+ // channelAcceptor is a callback that's called by the lnd client when a new
1748
+ // channel is proposed. This function is responsible for deciding whether to
1749
+ // accept the channel based on the channel parameters, and to also set some
1750
+ // channel parameters for our own side.
1751
+ func (f * FundingController ) channelAcceptor (_ context.Context ,
1752
+ req * lndclient.AcceptorRequest ) (* lndclient.AcceptorResponse , error ) {
1753
+
1754
+ // Avoid nil pointer dereference.
1755
+ if req .CommitmentType == nil {
1756
+ return nil , fmt .Errorf ("commitment type is required" )
1757
+ }
1758
+
1759
+ // Ignore any non-asset channels, just accept them.
1760
+ if * req .CommitmentType != lnwallet .CommitmentTypeSimpleTaprootOverlay {
1761
+ return & lndclient.AcceptorResponse {
1762
+ Accept : true ,
1763
+ }, nil
1764
+ }
1765
+
1766
+ // Reject custom channels that don't observe the max HTLC limit.
1767
+ if req .MaxAcceptedHtlcs > maxNumHTLCsPerParty {
1768
+ return & lndclient.AcceptorResponse {
1769
+ Accept : false ,
1770
+ Error : fmt .Sprintf ("max accepted HTLCs must be at " +
1771
+ "most %d, got %d" , maxNumHTLCsPerParty ,
1772
+ req .MaxAcceptedHtlcs ),
1773
+ }, nil
1774
+ }
1775
+
1776
+ // Everything looks good, we can now set our own max HTLC limit we'll
1777
+ // observe for this channel.
1778
+ return & lndclient.AcceptorResponse {
1779
+ Accept : true ,
1780
+ MaxHtlcCount : maxNumHTLCsPerParty ,
1781
+ }, nil
1782
+ }
1783
+
1685
1784
// validateProofs validates the inclusion/exclusion/split proofs and the
1686
1785
// transfer witness of the given proofs.
1687
1786
func (f * FundingController ) validateProofs (proofs []* proof.Proof ) error {
0 commit comments