mirror of
https://github.com/lightningnetwork/lnd.git
synced 2025-01-19 05:45:21 +01:00
9192c165ff
We need a new feature bit for BOLT11 invoices in order to indicate that they contain the new blinded path tagged field. Tagged fields pre-date TLV and so nodes who dont understand them will simply skip them. Therefore the feature bit helps them to fail fast.
320 lines
9.6 KiB
Go
320 lines
9.6 KiB
Go
package feature
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
|
|
"github.com/lightningnetwork/lnd/lnwire"
|
|
)
|
|
|
|
var (
|
|
// ErrUnknownSet is returned if a proposed feature vector contains a
|
|
// set that is unknown to LND.
|
|
ErrUnknownSet = errors.New("unknown feature bit set")
|
|
|
|
// ErrFeatureConfigured is returned if an attempt is made to unset a
|
|
// feature that was configured at startup.
|
|
ErrFeatureConfigured = errors.New("can't unset configured feature")
|
|
)
|
|
|
|
// Config houses any runtime modifications to the default set descriptors. For
|
|
// our purposes, this typically means disabling certain features to test legacy
|
|
// protocol interoperability or functionality.
|
|
type Config struct {
|
|
// NoTLVOnion unsets any optional or required TLVOnionPaylod bits from
|
|
// all feature sets.
|
|
NoTLVOnion bool
|
|
|
|
// NoStaticRemoteKey unsets any optional or required StaticRemoteKey
|
|
// bits from all feature sets.
|
|
NoStaticRemoteKey bool
|
|
|
|
// NoAnchors unsets any bits signaling support for anchor outputs.
|
|
NoAnchors bool
|
|
|
|
// NoWumbo unsets any bits signalling support for wumbo channels.
|
|
NoWumbo bool
|
|
|
|
// NoTaprootChans unsets any bits signaling support for taproot
|
|
// channels.
|
|
NoTaprootChans bool
|
|
|
|
// NoScriptEnforcementLease unsets any bits signaling support for script
|
|
// enforced leases.
|
|
NoScriptEnforcementLease bool
|
|
|
|
// NoKeysend unsets any bits signaling support for accepting keysend
|
|
// payments.
|
|
NoKeysend bool
|
|
|
|
// NoOptionScidAlias unsets any bits signalling support for
|
|
// option_scid_alias. This also implicitly disables zero-conf channels.
|
|
NoOptionScidAlias bool
|
|
|
|
// NoZeroConf unsets any bits signalling support for zero-conf
|
|
// channels. This should be used instead of NoOptionScidAlias to still
|
|
// keep option-scid-alias support.
|
|
NoZeroConf bool
|
|
|
|
// NoAnySegwit unsets any bits that signal support for using other
|
|
// segwit witness versions for co-op closes.
|
|
NoAnySegwit bool
|
|
|
|
// NoRouteBlinding unsets route blinding feature bits.
|
|
NoRouteBlinding bool
|
|
|
|
// CustomFeatures is a set of custom features to advertise in each
|
|
// set.
|
|
CustomFeatures map[Set][]lnwire.FeatureBit
|
|
}
|
|
|
|
// Manager is responsible for generating feature vectors for different requested
|
|
// feature sets.
|
|
type Manager struct {
|
|
// fsets is a static map of feature set to raw feature vectors. Requests
|
|
// are fulfilled by cloning these internal feature vectors.
|
|
fsets map[Set]*lnwire.RawFeatureVector
|
|
|
|
// configFeatures is a set of custom features that were "hard set" in
|
|
// lnd's config that cannot be updated at runtime (as is the case with
|
|
// our "standard" features that are defined in LND).
|
|
configFeatures map[Set]*lnwire.FeatureVector
|
|
}
|
|
|
|
// NewManager creates a new feature Manager, applying any custom modifications
|
|
// to its feature sets before returning.
|
|
func NewManager(cfg Config) (*Manager, error) {
|
|
return newManager(cfg, defaultSetDesc)
|
|
}
|
|
|
|
// newManager creates a new feature Manager, applying any custom modifications
|
|
// to its feature sets before returning. This method accepts the setDesc as its
|
|
// own parameter so that it can be unit tested.
|
|
func newManager(cfg Config, desc setDesc) (*Manager, error) {
|
|
// First build the default feature vector for all known sets.
|
|
fsets := make(map[Set]*lnwire.RawFeatureVector)
|
|
for bit, sets := range desc {
|
|
for set := range sets {
|
|
// Fetch the feature vector for this set, allocating a
|
|
// new one if it doesn't exist.
|
|
fv, ok := fsets[set]
|
|
if !ok {
|
|
fv = lnwire.NewRawFeatureVector()
|
|
}
|
|
|
|
// Set the configured bit on the feature vector,
|
|
// ensuring that we don't set two feature bits for the
|
|
// same pair.
|
|
err := fv.SafeSet(bit)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unable to set "+
|
|
"%v in %v: %v", bit, set, err)
|
|
}
|
|
|
|
// Write the updated feature vector under its set.
|
|
fsets[set] = fv
|
|
}
|
|
}
|
|
|
|
// Now, remove any features as directed by the config.
|
|
configFeatures := make(map[Set]*lnwire.FeatureVector)
|
|
for set, raw := range fsets {
|
|
if cfg.NoTLVOnion {
|
|
raw.Unset(lnwire.TLVOnionPayloadOptional)
|
|
raw.Unset(lnwire.TLVOnionPayloadRequired)
|
|
raw.Unset(lnwire.PaymentAddrOptional)
|
|
raw.Unset(lnwire.PaymentAddrRequired)
|
|
raw.Unset(lnwire.MPPOptional)
|
|
raw.Unset(lnwire.MPPRequired)
|
|
raw.Unset(lnwire.RouteBlindingOptional)
|
|
raw.Unset(lnwire.RouteBlindingRequired)
|
|
raw.Unset(lnwire.Bolt11BlindedPathsOptional)
|
|
raw.Unset(lnwire.Bolt11BlindedPathsRequired)
|
|
raw.Unset(lnwire.AMPOptional)
|
|
raw.Unset(lnwire.AMPRequired)
|
|
raw.Unset(lnwire.KeysendOptional)
|
|
raw.Unset(lnwire.KeysendRequired)
|
|
}
|
|
if cfg.NoStaticRemoteKey {
|
|
raw.Unset(lnwire.StaticRemoteKeyOptional)
|
|
raw.Unset(lnwire.StaticRemoteKeyRequired)
|
|
}
|
|
if cfg.NoAnchors {
|
|
raw.Unset(lnwire.AnchorsZeroFeeHtlcTxOptional)
|
|
raw.Unset(lnwire.AnchorsZeroFeeHtlcTxRequired)
|
|
|
|
// If anchors are disabled, then we also need to
|
|
// disable all other features that depend on it as
|
|
// well, as otherwise we may create an invalid feature
|
|
// bit set.
|
|
for bit, depFeatures := range deps {
|
|
for depFeature := range depFeatures {
|
|
switch {
|
|
case depFeature == lnwire.AnchorsZeroFeeHtlcTxRequired:
|
|
fallthrough
|
|
case depFeature == lnwire.AnchorsZeroFeeHtlcTxOptional:
|
|
raw.Unset(bit)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if cfg.NoWumbo {
|
|
raw.Unset(lnwire.WumboChannelsOptional)
|
|
raw.Unset(lnwire.WumboChannelsRequired)
|
|
}
|
|
if cfg.NoScriptEnforcementLease {
|
|
raw.Unset(lnwire.ScriptEnforcedLeaseOptional)
|
|
raw.Unset(lnwire.ScriptEnforcedLeaseRequired)
|
|
}
|
|
if cfg.NoKeysend {
|
|
raw.Unset(lnwire.KeysendOptional)
|
|
raw.Unset(lnwire.KeysendRequired)
|
|
}
|
|
if cfg.NoOptionScidAlias {
|
|
raw.Unset(lnwire.ScidAliasOptional)
|
|
raw.Unset(lnwire.ScidAliasRequired)
|
|
}
|
|
if cfg.NoZeroConf {
|
|
raw.Unset(lnwire.ZeroConfOptional)
|
|
raw.Unset(lnwire.ZeroConfRequired)
|
|
}
|
|
if cfg.NoAnySegwit {
|
|
raw.Unset(lnwire.ShutdownAnySegwitOptional)
|
|
raw.Unset(lnwire.ShutdownAnySegwitRequired)
|
|
}
|
|
if cfg.NoTaprootChans {
|
|
raw.Unset(lnwire.SimpleTaprootChannelsOptionalStaging)
|
|
raw.Unset(lnwire.SimpleTaprootChannelsRequiredStaging)
|
|
}
|
|
if cfg.NoRouteBlinding {
|
|
raw.Unset(lnwire.RouteBlindingOptional)
|
|
raw.Unset(lnwire.RouteBlindingRequired)
|
|
raw.Unset(lnwire.Bolt11BlindedPathsOptional)
|
|
raw.Unset(lnwire.Bolt11BlindedPathsRequired)
|
|
}
|
|
for _, custom := range cfg.CustomFeatures[set] {
|
|
if custom > set.Maximum() {
|
|
return nil, fmt.Errorf("feature bit: %v "+
|
|
"exceeds set: %v maximum: %v", custom,
|
|
set, set.Maximum())
|
|
}
|
|
|
|
if raw.IsSet(custom) {
|
|
return nil, fmt.Errorf("feature bit: %v "+
|
|
"already set", custom)
|
|
}
|
|
|
|
if err := raw.SafeSet(custom); err != nil {
|
|
return nil, fmt.Errorf("%w: could not set "+
|
|
"feature: %d", err, custom)
|
|
}
|
|
}
|
|
|
|
// Track custom features separately so that we can check that
|
|
// they aren't unset in subsequent updates. If there is no
|
|
// entry for the set, the vector will just be empty.
|
|
configFeatures[set] = lnwire.NewFeatureVector(
|
|
lnwire.NewRawFeatureVector(cfg.CustomFeatures[set]...),
|
|
lnwire.Features,
|
|
)
|
|
|
|
// Ensure that all of our feature sets properly set any
|
|
// dependent features.
|
|
fv := lnwire.NewFeatureVector(raw, lnwire.Features)
|
|
err := ValidateDeps(fv)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("invalid feature set %v: %w",
|
|
set, err)
|
|
}
|
|
}
|
|
|
|
return &Manager{
|
|
fsets: fsets,
|
|
configFeatures: configFeatures,
|
|
}, nil
|
|
}
|
|
|
|
// GetRaw returns a raw feature vector for the passed set. If no set is known,
|
|
// an empty raw feature vector is returned.
|
|
func (m *Manager) GetRaw(set Set) *lnwire.RawFeatureVector {
|
|
if fv, ok := m.fsets[set]; ok {
|
|
return fv.Clone()
|
|
}
|
|
|
|
return lnwire.NewRawFeatureVector()
|
|
}
|
|
|
|
// setRaw sets a new raw feature vector for the given set.
|
|
func (m *Manager) setRaw(set Set, raw *lnwire.RawFeatureVector) {
|
|
m.fsets[set] = raw
|
|
}
|
|
|
|
// Get returns a feature vector for the passed set. If no set is known, an empty
|
|
// feature vector is returned.
|
|
func (m *Manager) Get(set Set) *lnwire.FeatureVector {
|
|
raw := m.GetRaw(set)
|
|
return lnwire.NewFeatureVector(raw, lnwire.Features)
|
|
}
|
|
|
|
// ListSets returns a list of the feature sets that our node supports.
|
|
func (m *Manager) ListSets() []Set {
|
|
var sets []Set
|
|
|
|
for set := range m.fsets {
|
|
sets = append(sets, set)
|
|
}
|
|
|
|
return sets
|
|
}
|
|
|
|
// UpdateFeatureSets accepts a map of new feature vectors for each of the
|
|
// manager's known sets, validates that the update can be applied and modifies
|
|
// the feature manager's internal state. If a set is not included in the update
|
|
// map, it is left unchanged. The feature vectors provided are expected to
|
|
// include the current set of features, updated with desired bits added/removed.
|
|
func (m *Manager) UpdateFeatureSets(
|
|
updates map[Set]*lnwire.RawFeatureVector) error {
|
|
|
|
for set, newFeatures := range updates {
|
|
if !set.valid() {
|
|
return fmt.Errorf("%w: set: %d", ErrUnknownSet, set)
|
|
}
|
|
|
|
if err := newFeatures.ValidatePairs(); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := m.Get(set).ValidateUpdate(
|
|
newFeatures, set.Maximum(),
|
|
); err != nil {
|
|
return err
|
|
}
|
|
|
|
// If any features were configured for this set, ensure that
|
|
// they are still set in the new feature vector.
|
|
if cfgFeat, haveCfgFeat := m.configFeatures[set]; haveCfgFeat {
|
|
for feature := range cfgFeat.Features() {
|
|
if !newFeatures.IsSet(feature) {
|
|
return fmt.Errorf("%w: can't unset: "+
|
|
"%d", ErrFeatureConfigured,
|
|
feature)
|
|
}
|
|
}
|
|
}
|
|
|
|
fv := lnwire.NewFeatureVector(newFeatures, lnwire.Features)
|
|
if err := ValidateDeps(fv); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// Only update the current feature sets once every proposed set has
|
|
// passed validation so that we don't partially update any sets then
|
|
// fail out on a later set's validation.
|
|
for set, features := range updates {
|
|
m.setRaw(set, features.Clone())
|
|
}
|
|
|
|
return nil
|
|
}
|