mirror of
https://github.com/lightningnetwork/lnd.git
synced 2025-08-25 13:12:11 +02:00
240 lines
7.0 KiB
Go
240 lines
7.0 KiB
Go
package lnwire
|
|
|
|
import (
|
|
"bytes"
|
|
"io"
|
|
|
|
"github.com/btcsuite/btcd/btcutil"
|
|
"github.com/lightningnetwork/lnd/tlv"
|
|
)
|
|
|
|
// DynPropose is a message that is sent during a dynamic commitments negotiation
|
|
// process. It is sent by both parties to propose new channel parameters.
|
|
type DynPropose struct {
|
|
// ChanID identifies the channel whose parameters we are trying to
|
|
// re-negotiate.
|
|
ChanID ChannelID
|
|
|
|
// DustLimit, if not nil, proposes a change to the dust_limit_satoshis
|
|
// for the sender's commitment transaction.
|
|
DustLimit tlv.OptionalRecordT[
|
|
tlv.TlvType0, tlv.BigSizeT[btcutil.Amount],
|
|
]
|
|
|
|
// MaxValueInFlight, if not nil, proposes a change to the
|
|
// max_htlc_value_in_flight_msat limit of the sender.
|
|
MaxValueInFlight tlv.OptionalRecordT[tlv.TlvType2, MilliSatoshi]
|
|
|
|
// HtlcMinimum, if not nil, proposes a change to the htlc_minimum_msat
|
|
// floor of the sender.
|
|
HtlcMinimum tlv.OptionalRecordT[tlv.TlvType4, MilliSatoshi]
|
|
|
|
// ChannelReserve, if not nil, proposes a change to the
|
|
// channel_reserve_satoshis requirement of the recipient.
|
|
ChannelReserve tlv.OptionalRecordT[
|
|
tlv.TlvType6, tlv.BigSizeT[btcutil.Amount],
|
|
]
|
|
|
|
// CsvDelay, if not nil, proposes a change to the to_self_delay
|
|
// requirement of the recipient.
|
|
CsvDelay tlv.OptionalRecordT[tlv.TlvType8, uint16]
|
|
|
|
// MaxAcceptedHTLCs, if not nil, proposes a change to the
|
|
// max_accepted_htlcs limit of the sender.
|
|
MaxAcceptedHTLCs tlv.OptionalRecordT[tlv.TlvType10, uint16]
|
|
|
|
// ChannelType, if not nil, proposes a change to the channel_type
|
|
// parameter.
|
|
ChannelType tlv.OptionalRecordT[tlv.TlvType12, ChannelType]
|
|
|
|
// ExtraData is the set of data that was appended to this message to
|
|
// fill out the full maximum transport message size. These fields can
|
|
// be used to specify optional data such as custom TLV fields.
|
|
//
|
|
// NOTE: Since the fields in this structure are part of the TLV stream,
|
|
// ExtraData will contain all TLV records _except_ the ones that are
|
|
// present in earlier parts of this structure.
|
|
ExtraData ExtraOpaqueData
|
|
}
|
|
|
|
// A compile time check to ensure DynPropose implements the lnwire.Message
|
|
// interface.
|
|
var _ Message = (*DynPropose)(nil)
|
|
|
|
// A compile time check to ensure DynPropose implements the
|
|
// lnwire.SizeableMessage interface.
|
|
var _ SizeableMessage = (*DynPropose)(nil)
|
|
|
|
// Encode serializes the target DynPropose into the passed io.Writer.
|
|
// Serialization will observe the rules defined by the passed protocol version.
|
|
//
|
|
// This is a part of the lnwire.Message interface.
|
|
func (dp *DynPropose) Encode(w *bytes.Buffer, _ uint32) error {
|
|
if err := WriteChannelID(w, dp.ChanID); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Create extra data records.
|
|
producers, err := dp.ExtraData.RecordProducers()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Append the known records.
|
|
producers = append(producers, dynProposeRecords(dp)...)
|
|
|
|
// Encode all records.
|
|
var tlvData ExtraOpaqueData
|
|
err = tlvData.PackRecords(producers...)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return WriteBytes(w, tlvData)
|
|
}
|
|
|
|
// Decode deserializes the serialized DynPropose stored in the passed io.Reader
|
|
// into the target DynPropose using the deserialization rules defined by the
|
|
// passed protocol version.
|
|
//
|
|
// This is a part of the lnwire.Message interface.
|
|
func (dp *DynPropose) Decode(r io.Reader, _ uint32) error {
|
|
// Parse out the only required field.
|
|
if err := ReadElements(r, &dp.ChanID); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Parse out TLV stream.
|
|
var tlvRecords ExtraOpaqueData
|
|
if err := ReadElements(r, &tlvRecords); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Prepare receiving buffers to be filled by TLV extraction.
|
|
var dustLimit tlv.RecordT[tlv.TlvType0, tlv.BigSizeT[btcutil.Amount]]
|
|
var maxValue tlv.RecordT[tlv.TlvType2, MilliSatoshi]
|
|
var htlcMin tlv.RecordT[tlv.TlvType4, MilliSatoshi]
|
|
var reserve tlv.RecordT[tlv.TlvType6, tlv.BigSizeT[btcutil.Amount]]
|
|
csvDelay := dp.CsvDelay.Zero()
|
|
maxHtlcs := dp.MaxAcceptedHTLCs.Zero()
|
|
chanType := dp.ChannelType.Zero()
|
|
|
|
knownRecords, extraData, err := ParseAndExtractExtraData(
|
|
tlvRecords, &dustLimit, &maxValue, &htlcMin, &reserve,
|
|
&csvDelay, &maxHtlcs, &chanType,
|
|
)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Check the results of the TLV Stream decoding and appropriately set
|
|
// message fields.
|
|
if _, ok := knownRecords[dp.DustLimit.TlvType()]; ok {
|
|
dp.DustLimit = tlv.SomeRecordT(dustLimit)
|
|
}
|
|
|
|
if _, ok := knownRecords[dp.MaxValueInFlight.TlvType()]; ok {
|
|
dp.MaxValueInFlight = tlv.SomeRecordT(maxValue)
|
|
}
|
|
|
|
if _, ok := knownRecords[dp.HtlcMinimum.TlvType()]; ok {
|
|
dp.HtlcMinimum = tlv.SomeRecordT(htlcMin)
|
|
}
|
|
|
|
if _, ok := knownRecords[dp.ChannelReserve.TlvType()]; ok {
|
|
dp.ChannelReserve = tlv.SomeRecordT(reserve)
|
|
}
|
|
|
|
if _, ok := knownRecords[dp.CsvDelay.TlvType()]; ok {
|
|
dp.CsvDelay = tlv.SomeRecordT(csvDelay)
|
|
}
|
|
|
|
if _, ok := knownRecords[dp.MaxAcceptedHTLCs.TlvType()]; ok {
|
|
dp.MaxAcceptedHTLCs = tlv.SomeRecordT(maxHtlcs)
|
|
}
|
|
|
|
if _, ok := knownRecords[dp.ChannelType.TlvType()]; ok {
|
|
dp.ChannelType = tlv.SomeRecordT(chanType)
|
|
}
|
|
|
|
dp.ExtraData = extraData
|
|
|
|
return nil
|
|
}
|
|
|
|
// MsgType returns the MessageType code which uniquely identifies this message
|
|
// as a DynPropose on the wire.
|
|
//
|
|
// This is part of the lnwire.Message interface.
|
|
func (dp *DynPropose) MsgType() MessageType {
|
|
return MsgDynPropose
|
|
}
|
|
|
|
// SerializedSize returns the serialized size of the message in bytes.
|
|
//
|
|
// This is part of the lnwire.SizeableMessage interface.
|
|
func (dp *DynPropose) SerializedSize() (uint32, error) {
|
|
return MessageSerializedSize(dp)
|
|
}
|
|
|
|
// SerializeTlvData takes just the TLV data of DynPropose (which covers all of
|
|
// the parameters on deck for changing) and serializes just this component. The
|
|
// main purpose of this is to make it easier to validate the DynAck signature.
|
|
func (dp *DynPropose) SerializeTlvData() ([]byte, error) {
|
|
producers := dynProposeRecords(dp)
|
|
|
|
var extra ExtraOpaqueData
|
|
err := extra.PackRecords(producers...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return extra, nil
|
|
}
|
|
|
|
func dynProposeRecords(dp *DynPropose) []tlv.RecordProducer {
|
|
recordProducers := make([]tlv.RecordProducer, 0, 7)
|
|
|
|
dp.DustLimit.WhenSome(
|
|
func(dl tlv.RecordT[tlv.TlvType0,
|
|
tlv.BigSizeT[btcutil.Amount]]) {
|
|
|
|
recordProducers = append(recordProducers, &dl)
|
|
},
|
|
)
|
|
dp.MaxValueInFlight.WhenSome(
|
|
func(mvif tlv.RecordT[tlv.TlvType2, MilliSatoshi]) {
|
|
recordProducers = append(recordProducers, &mvif)
|
|
},
|
|
)
|
|
dp.HtlcMinimum.WhenSome(
|
|
func(hm tlv.RecordT[tlv.TlvType4, MilliSatoshi]) {
|
|
recordProducers = append(recordProducers, &hm)
|
|
},
|
|
)
|
|
dp.ChannelReserve.WhenSome(
|
|
func(reserve tlv.RecordT[tlv.TlvType6,
|
|
tlv.BigSizeT[btcutil.Amount]]) {
|
|
|
|
recordProducers = append(recordProducers, &reserve)
|
|
},
|
|
)
|
|
dp.CsvDelay.WhenSome(
|
|
func(wait tlv.RecordT[tlv.TlvType8, uint16]) {
|
|
recordProducers = append(recordProducers, &wait)
|
|
},
|
|
)
|
|
dp.MaxAcceptedHTLCs.WhenSome(
|
|
func(mah tlv.RecordT[tlv.TlvType10, uint16]) {
|
|
recordProducers = append(recordProducers, &mah)
|
|
},
|
|
)
|
|
dp.ChannelType.WhenSome(
|
|
func(ty tlv.RecordT[tlv.TlvType12, ChannelType]) {
|
|
recordProducers = append(recordProducers, &ty)
|
|
},
|
|
)
|
|
|
|
return recordProducers
|
|
}
|