lnwire: New API for feature vectors.
This is a rewrite of feature vectors in lnwire. This has a few benefits: 1) a simpler interface 2) separate structs for a plain set of feature bits and a feature vector with associated feature names their respective feature sets 3) loosened requirements that bits MUST be assigned in pairs 4) fix endianness of encoding/decoding
This commit is contained in:
parent
b4273d1eaa
commit
1633ab180f
@ -2,49 +2,24 @@ package lnwire
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"math"
|
|
||||||
|
|
||||||
"github.com/go-errors/errors"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// featureFlag represent the status of the feature optional/required and needed
|
// FeatureBit represents a feature that can be enabled in either a local or
|
||||||
// to allow future incompatible changes, or backward compatible changes.
|
// global feature vector at a specific bit position. Feature bits follow the
|
||||||
type featureFlag uint8
|
// "it's OK to be odd" rule, where features at even bit positions must be known
|
||||||
|
// to a node receiving them from a peer while odd bits do not. In accordance,
|
||||||
// String returns the string representation for the featureFlag.
|
// feature bits are usually assigned in pairs, first being assigned an odd bit
|
||||||
func (f featureFlag) String() string {
|
// position which may later be changed to the preceding even position once
|
||||||
switch f {
|
// knowledge of the feature becomes required on the network.
|
||||||
case OptionalFlag:
|
type FeatureBit uint16
|
||||||
return "optional"
|
|
||||||
case RequiredFlag:
|
|
||||||
return "required"
|
|
||||||
default:
|
|
||||||
return "<unknown>"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// featureName represent the name of the feature and needed in order to have
|
|
||||||
// the compile errors if we specify wrong feature name.
|
|
||||||
type featureName string
|
|
||||||
|
|
||||||
const (
|
const (
|
||||||
// OptionalFlag represent the feature which we already have but it
|
// InitialRoutingSync is a local feature bit meaning that the receiving node
|
||||||
// isn't required yet, and if remote peer doesn't have this feature we
|
// should send a complete dump of routing information when a new connection
|
||||||
// may turn it off without disconnecting with peer.
|
// is established.
|
||||||
OptionalFlag featureFlag = 2 // 0b10
|
InitialRoutingSync FeatureBit = 3
|
||||||
|
|
||||||
// RequiredFlag represent the features which is required for proper
|
|
||||||
// peer interaction, we disconnect with peer if it doesn't have this
|
|
||||||
// particular feature.
|
|
||||||
RequiredFlag featureFlag = 1 // 0b01
|
|
||||||
|
|
||||||
// flagMask is a mask which is needed to extract feature flag value.
|
|
||||||
flagMask = 3 // 0b11
|
|
||||||
|
|
||||||
// flagBitsSize represent the size of the feature flag in bits. For
|
|
||||||
// more information read the init message specification.
|
|
||||||
flagBitsSize = 2
|
|
||||||
|
|
||||||
// maxAllowedSize is a maximum allowed size of feature vector.
|
// maxAllowedSize is a maximum allowed size of feature vector.
|
||||||
//
|
//
|
||||||
@ -61,61 +36,62 @@ const (
|
|||||||
maxAllowedSize = 32781
|
maxAllowedSize = 32781
|
||||||
)
|
)
|
||||||
|
|
||||||
// Feature represent the feature which is used on stage of initialization of
|
// LocalFeatures is a mapping of known connection-local feature bits to a
|
||||||
// feature vector. Initial feature flags might be changed dynamically later.
|
// descriptive name. All known local feature bits must be assigned a name in
|
||||||
type Feature struct {
|
// this mapping. Local features are those which are only sent to the peer and
|
||||||
Name featureName
|
// not advertised to the entire network. A full description of these feature
|
||||||
Flag featureFlag
|
// bits is provided in the BOLT-09 specification.
|
||||||
|
var LocalFeatures = map[FeatureBit]string{
|
||||||
|
InitialRoutingSync: "initial-routing-sync",
|
||||||
}
|
}
|
||||||
|
|
||||||
// FeatureVector represents the global/local feature vector. With this
|
// GlobalFeatures is a mapping of known global feature bits to a descriptive
|
||||||
// structure you may set/get the feature by name and compare feature vector
|
// name. All known global feature bits must be assigned a name in this mapping.
|
||||||
// with remote one.
|
// Global features are those which are advertised to the entire network. A full
|
||||||
type FeatureVector struct {
|
// description of these feature bits is provided in the BOLT-09 specification.
|
||||||
// featuresMap is the map which stores the correspondence between
|
var GlobalFeatures map[FeatureBit]string
|
||||||
// feature name and its index within feature vector. Index within
|
|
||||||
// feature vector and actual binary position of feature are different
|
|
||||||
// things)
|
|
||||||
featuresMap map[featureName]int // name -> index
|
|
||||||
|
|
||||||
// flags is the map which stores the correspondence between feature
|
// RawFeatureVector represents a set of feature bits as defined in BOLT-09.
|
||||||
// index and its flag.
|
// A RawFeatureVector itself just stores a set of bit flags but can be used to
|
||||||
flags map[int]featureFlag // index -> flag
|
// construct a FeatureVector which binds meaning to each bit. Feature vectors
|
||||||
|
//can be serialized and deserialized to/from a byte representation that is
|
||||||
|
// transmitted in Lightning network messages.
|
||||||
|
type RawFeatureVector struct {
|
||||||
|
features map[FeatureBit]bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewFeatureVector creates new instance of feature vector.
|
// NewRawFeatureVector creates a feature vector with all of the feature bits
|
||||||
func NewFeatureVector(features []Feature) *FeatureVector {
|
// given as arguments enabled.
|
||||||
featuresMap := make(map[featureName]int)
|
func NewRawFeatureVector(bits ...FeatureBit) *RawFeatureVector {
|
||||||
flags := make(map[int]featureFlag)
|
fv := &RawFeatureVector{features: make(map[FeatureBit]bool)}
|
||||||
|
for _, bit := range bits {
|
||||||
for index, feature := range features {
|
fv.Set(bit)
|
||||||
featuresMap[feature.Name] = index
|
|
||||||
flags[index] = feature.Flag
|
|
||||||
}
|
|
||||||
|
|
||||||
return &FeatureVector{
|
|
||||||
featuresMap: featuresMap,
|
|
||||||
flags: flags,
|
|
||||||
}
|
}
|
||||||
|
return fv
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetFeatureFlag assign flag to the feature.
|
// IsSet returns whether a particular feature bit is enabled in the vector.
|
||||||
func (f *FeatureVector) SetFeatureFlag(name featureName, flag featureFlag) error {
|
func (fv *RawFeatureVector) IsSet(feature FeatureBit) bool {
|
||||||
position, ok := f.featuresMap[name]
|
return fv.features[feature]
|
||||||
if !ok {
|
|
||||||
return errors.Errorf("can't find feature with name: %v", name)
|
|
||||||
}
|
|
||||||
|
|
||||||
f.flags[position] = flag
|
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// serializedSize returns the number of bytes which is needed to represent
|
// Set marks a feature as enabled in the vector.
|
||||||
// feature vector in byte format.
|
func (fv *RawFeatureVector) Set(feature FeatureBit) {
|
||||||
func (f *FeatureVector) serializedSize() uint16 {
|
fv.features[feature] = true
|
||||||
// Find the largest index in f.flags
|
}
|
||||||
|
|
||||||
|
// Unset marks a feature as disabled in the vector.
|
||||||
|
func (fv *RawFeatureVector) Unset(feature FeatureBit) {
|
||||||
|
delete(fv.features, feature)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SerializeSize returns the number of bytes needed to represent feature vector
|
||||||
|
// in byte format.
|
||||||
|
func (fv *RawFeatureVector) SerializeSize() int {
|
||||||
|
// Find the largest feature bit index
|
||||||
max := -1
|
max := -1
|
||||||
for index := range f.flags {
|
for feature := range fv.features {
|
||||||
|
index := int(feature)
|
||||||
if index > max {
|
if index > max {
|
||||||
max = index
|
max = index
|
||||||
}
|
}
|
||||||
@ -123,181 +99,134 @@ func (f *FeatureVector) serializedSize() uint16 {
|
|||||||
if max == -1 {
|
if max == -1 {
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
// We calculate length via the largest index in f.flags so as to not
|
|
||||||
// get an index out of bounds in Encode's setFlag function.
|
// We calculate byte-length via the largest bit index
|
||||||
return uint16(math.Ceil(float64(flagBitsSize*(max+1)) / 8))
|
return max/8 + 1
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewFeatureVectorFromReader decodes the feature vector from binary
|
// Encode writes the feature vector in byte representation. Every feature
|
||||||
// representation and creates the instance of it. Every feature decoded as 2
|
// encoded as a bit, and the bit vector is serialized using the least number of
|
||||||
// bits where odd bit determine whether the feature is "optional" and even bit
|
// bytes. Since the bit vector length is variable, the first two bytes of the
|
||||||
// told us whether the feature is "required". The even/odd semantic allows
|
// serialization represent the length.
|
||||||
// future incompatible changes, or backward compatible changes. Bits generally
|
func (fv *RawFeatureVector) Encode(w io.Writer) error {
|
||||||
// assigned in pairs, so that optional features can later become compulsory.
|
|
||||||
func NewFeatureVectorFromReader(r io.Reader) (*FeatureVector, error) {
|
|
||||||
f := &FeatureVector{
|
|
||||||
flags: make(map[int]featureFlag),
|
|
||||||
}
|
|
||||||
|
|
||||||
getFlag := func(data []byte, position int) featureFlag {
|
|
||||||
byteNumber := uint(position / 8)
|
|
||||||
bitNumber := uint(position % 8)
|
|
||||||
|
|
||||||
return featureFlag((data[byteNumber] >> bitNumber) & flagMask)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Read the length of the feature vector.
|
|
||||||
var l [2]byte
|
|
||||||
if _, err := io.ReadFull(r, l[:]); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
length := binary.BigEndian.Uint16(l[:])
|
|
||||||
|
|
||||||
// Read the feature vector data.
|
|
||||||
data := make([]byte, length)
|
|
||||||
if _, err := io.ReadFull(r, data[:]); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Initialize feature vector.
|
|
||||||
bitsNumber := len(data) * 8
|
|
||||||
for position := 0; position <= bitsNumber-flagBitsSize; position += flagBitsSize {
|
|
||||||
flag := getFlag(data, position)
|
|
||||||
switch flag {
|
|
||||||
case OptionalFlag, RequiredFlag:
|
|
||||||
// Every feature/flag takes 2 bits, so in order to get
|
|
||||||
// the feature/flag index we should divide position
|
|
||||||
// on 2.
|
|
||||||
index := position / flagBitsSize
|
|
||||||
f.flags[index] = flag
|
|
||||||
default:
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return f, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Encode encodes the features vector into bytes representation, every feature
|
|
||||||
// encoded as 2 bits where odd bit determine whether the feature is "optional"
|
|
||||||
// and even bit told us whether the feature is "required". The even/odd
|
|
||||||
// semantic allows future incompatible changes, or backward compatible changes.
|
|
||||||
// Bits generally assigned in pairs, so that optional features can later become
|
|
||||||
// compulsory.
|
|
||||||
func (f *FeatureVector) Encode(w io.Writer) error {
|
|
||||||
setFlag := func(data []byte, position int, flag featureFlag) {
|
|
||||||
byteNumber := uint(position / 8)
|
|
||||||
bitNumber := uint(position % 8)
|
|
||||||
|
|
||||||
data[byteNumber] |= (byte(flag) << bitNumber)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write length of feature vector.
|
// Write length of feature vector.
|
||||||
var l [2]byte
|
var l [2]byte
|
||||||
length := f.serializedSize()
|
length := fv.SerializeSize()
|
||||||
binary.BigEndian.PutUint16(l[:], length)
|
binary.BigEndian.PutUint16(l[:], uint16(length))
|
||||||
if _, err := w.Write(l[:]); err != nil {
|
if _, err := w.Write(l[:]); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Generate the data and write it.
|
// Generate the data and write it.
|
||||||
data := make([]byte, length)
|
data := make([]byte, length)
|
||||||
for index, flag := range f.flags {
|
for feature := range fv.features {
|
||||||
// Every feature takes 2 bits, so in order to get the feature
|
byteIndex := int(feature / 8)
|
||||||
// bits position we should multiply index by 2.
|
bitIndex := feature % 8
|
||||||
position := index * flagBitsSize
|
data[length-byteIndex-1] |= 1 << bitIndex
|
||||||
setFlag(data, position, flag)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err := w.Write(data)
|
_, err := w.Write(data)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Compare checks that features are compatible and returns the features which
|
// Decode reads the feature vector from its byte representation. Every feature
|
||||||
// were present in both remote and local feature vectors. If remote/local node
|
// encoded as a bit, and the bit vector is serialized using the least number of
|
||||||
// doesn't have the feature and local/remote node require it than such vectors
|
// bytes. Since the bit vector length is variable, the first two bytes of the
|
||||||
// are incompatible.
|
// serialization represent the length.
|
||||||
func (f *FeatureVector) Compare(f2 *FeatureVector) (*SharedFeatures, error) {
|
func (fv *RawFeatureVector) Decode(r io.Reader) error {
|
||||||
shared := newSharedFeatures(f.Copy())
|
// Read the length of the feature vector.
|
||||||
|
var l [2]byte
|
||||||
|
if _, err := io.ReadFull(r, l[:]); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
length := binary.BigEndian.Uint16(l[:])
|
||||||
|
|
||||||
for index, flag := range f.flags {
|
// Read the feature vector data.
|
||||||
if _, exist := f2.flags[index]; !exist {
|
data := make([]byte, length)
|
||||||
switch flag {
|
if _, err := io.ReadFull(r, data); err != nil {
|
||||||
case RequiredFlag:
|
return err
|
||||||
return nil, errors.New("Remote node hasn't " +
|
}
|
||||||
"locally required feature")
|
|
||||||
case OptionalFlag:
|
// Set feature bits from parsed data.
|
||||||
// If feature is optional and remote side
|
bitsNumber := len(data) * 8
|
||||||
// haven't it than it might be safely disabled.
|
for i := 0; i < bitsNumber; i++ {
|
||||||
delete(shared.flags, index)
|
byteIndex := uint16(i / 8)
|
||||||
continue
|
bitIndex := uint(i % 8)
|
||||||
|
if (data[length-byteIndex-1]>>bitIndex)&1 == 1 {
|
||||||
|
fv.Set(FeatureBit(i))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// If feature exists on both sides than such feature might be
|
return nil
|
||||||
// considered as active.
|
|
||||||
shared.flags[index] = flag
|
|
||||||
}
|
|
||||||
|
|
||||||
for index, flag := range f2.flags {
|
|
||||||
if _, exist := f.flags[index]; !exist {
|
|
||||||
switch flag {
|
|
||||||
case RequiredFlag:
|
|
||||||
return nil, errors.New("Local node hasn't " +
|
|
||||||
"locally required feature")
|
|
||||||
case OptionalFlag:
|
|
||||||
// If feature is optional and local side
|
|
||||||
// haven't it than it might be safely disabled.
|
|
||||||
delete(shared.flags, index)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If feature exists on both sides than such feature might be
|
|
||||||
// considered as active.
|
|
||||||
shared.flags[index] = flag
|
|
||||||
}
|
|
||||||
|
|
||||||
return shared, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copy generate new distinct instance of the feature vector.
|
// FeatureVector represents a set of enabled features. The set stores
|
||||||
func (f *FeatureVector) Copy() *FeatureVector {
|
// information on enabled flags and metadata about the feature names. A feature
|
||||||
features := make([]Feature, len(f.featuresMap))
|
// vector is serializable to a compact byte representation that is included in
|
||||||
|
// Lightning network messages.
|
||||||
|
type FeatureVector struct {
|
||||||
|
*RawFeatureVector
|
||||||
|
featureNames map[FeatureBit]string
|
||||||
|
}
|
||||||
|
|
||||||
for name, index := range f.featuresMap {
|
// NewFeatureVector constructs a new FeatureVector from a raw feature vector and
|
||||||
features[index] = Feature{
|
// mapping of feature definitions.
|
||||||
Name: name,
|
func NewFeatureVector(featureVector *RawFeatureVector,
|
||||||
Flag: f.flags[index],
|
featureNames map[FeatureBit]string) *FeatureVector {
|
||||||
|
|
||||||
|
return &FeatureVector{
|
||||||
|
RawFeatureVector: featureVector,
|
||||||
|
featureNames: featureNames,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HasFeature returns whether a particular feature is included in the set. The
|
||||||
|
// feature can be seen as set either if the bit is set directly OR the queried
|
||||||
|
// bit has the same meaning as its corresponding even/odd bit, which is set
|
||||||
|
// instead. The second case is because feature bits are generally assigned in
|
||||||
|
// pairs where both the even and odd position represent the same feature.
|
||||||
|
func (fv *FeatureVector) HasFeature(feature FeatureBit) bool {
|
||||||
|
return fv.IsSet(feature) ||
|
||||||
|
(fv.isFeatureBitPair(feature) && fv.IsSet(feature^1))
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnknownRequiredFeatures returns a list of feature bits set in the vector that
|
||||||
|
// are unknown and in an even bit position. Feature bits with an even index must
|
||||||
|
// be known to a node receiving the feature vector in a message.
|
||||||
|
func (fv *FeatureVector) UnknownRequiredFeatures() []FeatureBit {
|
||||||
|
var unknown []FeatureBit
|
||||||
|
for feature := range fv.features {
|
||||||
|
if feature%2 == 0 && !fv.IsKnown(feature) {
|
||||||
|
unknown = append(unknown, feature)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return unknown
|
||||||
return NewFeatureVector(features)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// SharedFeatures is a product of comparison of two features vector which
|
// Name returns a string identifier for the feature represented by this bit. If
|
||||||
// consist of features which are present in both local and remote features
|
// the bit does not represent a known feature, this returns a string indicating
|
||||||
// vectors.
|
// as much.
|
||||||
type SharedFeatures struct {
|
func (fv *FeatureVector) Name(bit FeatureBit) string {
|
||||||
*FeatureVector
|
name, known := fv.featureNames[bit]
|
||||||
}
|
if !known {
|
||||||
|
name = "unknown"
|
||||||
// newSharedFeatures creates new shared features instance.
|
|
||||||
func newSharedFeatures(f *FeatureVector) *SharedFeatures {
|
|
||||||
return &SharedFeatures{f}
|
|
||||||
}
|
|
||||||
|
|
||||||
// IsActive checks is feature active or not, it might be disabled during
|
|
||||||
// comparision with remote feature vector if it was optional and remote peer
|
|
||||||
// doesn't support it.
|
|
||||||
func (f *SharedFeatures) IsActive(name featureName) bool {
|
|
||||||
index, ok := f.featuresMap[name]
|
|
||||||
if !ok {
|
|
||||||
// If we even have no such feature in feature map, than it
|
|
||||||
// can't be active in any circumstances.
|
|
||||||
return false
|
|
||||||
}
|
}
|
||||||
|
return fmt.Sprintf("%s(%d)", name, bit)
|
||||||
_, exist := f.flags[index]
|
}
|
||||||
return exist
|
|
||||||
|
// IsKnown returns whether this feature bit represents a known feature.
|
||||||
|
func (fv *FeatureVector) IsKnown(bit FeatureBit) bool {
|
||||||
|
_, known := fv.featureNames[bit]
|
||||||
|
return known
|
||||||
|
}
|
||||||
|
|
||||||
|
// isFeatureBitPair returns whether this feature bit and its corresponding
|
||||||
|
// even/odd bit both represent the same feature. This may often be the case as
|
||||||
|
// bits are generally assigned in pairs, first being assigned an odd bit
|
||||||
|
// position then being promoted to an even bit position once the network is
|
||||||
|
// ready.
|
||||||
|
func (fv *FeatureVector) isFeatureBitPair(bit FeatureBit) bool {
|
||||||
|
name1, known1 := fv.featureNames[bit]
|
||||||
|
name2, known2 := fv.featureNames[bit^1]
|
||||||
|
return known1 && known2 && name1 == name2
|
||||||
}
|
}
|
||||||
|
@ -3,152 +3,260 @@ package lnwire
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"reflect"
|
"reflect"
|
||||||
|
"sort"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/davecgh/go-spew/spew"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// TestFeaturesRemoteRequireError checks that we throw an error if remote peer
|
var testFeatureNames = map[FeatureBit]string{
|
||||||
// has required feature which we don't support.
|
0: "feature1",
|
||||||
func TestFeaturesRemoteRequireError(t *testing.T) {
|
3: "feature2",
|
||||||
|
4: "feature3",
|
||||||
|
5: "feature3",
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFeatureVectorSetUnset(t *testing.T) {
|
||||||
t.Parallel()
|
t.Parallel()
|
||||||
|
|
||||||
const (
|
tests := []struct {
|
||||||
first = "first"
|
bits []FeatureBit
|
||||||
second = "second"
|
expectedFeatures []bool
|
||||||
)
|
}{
|
||||||
|
// No features are enabled if no bits are set.
|
||||||
|
{
|
||||||
|
bits: nil,
|
||||||
|
expectedFeatures: []bool{false, false, false, false, false, false, false, false},
|
||||||
|
},
|
||||||
|
// Test setting an even bit for an even-only bit feature. The
|
||||||
|
// corresponding odd bit should not be seen as set.
|
||||||
|
{
|
||||||
|
bits: []FeatureBit{0},
|
||||||
|
expectedFeatures: []bool{true, false, false, false, false, false, false, false},
|
||||||
|
},
|
||||||
|
// Test setting an odd bit for an even-only bit feature. The
|
||||||
|
// corresponding even bit should not be seen as set.
|
||||||
|
{
|
||||||
|
bits: []FeatureBit{1},
|
||||||
|
expectedFeatures: []bool{false, true, false, false, false, false, false, false},
|
||||||
|
},
|
||||||
|
// Test setting an even bit for an odd-only bit feature. The bit should
|
||||||
|
// be seen as set and the odd bit should not.
|
||||||
|
{
|
||||||
|
bits: []FeatureBit{2},
|
||||||
|
expectedFeatures: []bool{false, false, true, false, false, false, false, false},
|
||||||
|
},
|
||||||
|
// Test setting an odd bit for an odd-only bit feature. The bit should
|
||||||
|
// be seen as set and the even bit should not.
|
||||||
|
{
|
||||||
|
bits: []FeatureBit{3},
|
||||||
|
expectedFeatures: []bool{false, false, false, true, false, false, false, false},
|
||||||
|
},
|
||||||
|
// Test setting an even bit for even-odd pair feature. Both bits in the
|
||||||
|
// pair should be seen as set.
|
||||||
|
{
|
||||||
|
bits: []FeatureBit{4},
|
||||||
|
expectedFeatures: []bool{false, false, false, false, true, true, false, false},
|
||||||
|
},
|
||||||
|
// Test setting an odd bit for even-odd pair feature. Both bits in the
|
||||||
|
// pair should be seen as set.
|
||||||
|
{
|
||||||
|
bits: []FeatureBit{5},
|
||||||
|
expectedFeatures: []bool{false, false, false, false, true, true, false, false},
|
||||||
|
},
|
||||||
|
// Test setting an even bit for an unknown feature. The bit should be
|
||||||
|
// seen as set and the odd bit should not.
|
||||||
|
{
|
||||||
|
bits: []FeatureBit{6},
|
||||||
|
expectedFeatures: []bool{false, false, false, false, false, false, true, false},
|
||||||
|
},
|
||||||
|
// Test setting an odd bit for an unknown feature. The bit should be
|
||||||
|
// seen as set and the odd bit should not.
|
||||||
|
{
|
||||||
|
bits: []FeatureBit{7},
|
||||||
|
expectedFeatures: []bool{false, false, false, false, false, false, false, true},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
localFeatures := NewFeatureVector([]Feature{
|
fv := NewFeatureVector(nil, testFeatureNames)
|
||||||
{first, OptionalFlag},
|
for i, test := range tests {
|
||||||
})
|
for _, bit := range test.bits {
|
||||||
|
fv.Set(bit)
|
||||||
|
}
|
||||||
|
|
||||||
remoteFeatures := NewFeatureVector([]Feature{
|
for j, expectedSet := range test.expectedFeatures {
|
||||||
{first, OptionalFlag},
|
if fv.HasFeature(FeatureBit(j)) != expectedSet {
|
||||||
{second, RequiredFlag},
|
t.Errorf("Expection failed in case %d, bit %d", i, j)
|
||||||
})
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if _, err := localFeatures.Compare(remoteFeatures); err == nil {
|
for _, bit := range test.bits {
|
||||||
t.Fatal("error wasn't received")
|
fv.Unset(bit)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TestFeaturesLocalRequireError checks that we throw an error if local peer has
|
func TestFeatureVectorEncodeDecode(t *testing.T) {
|
||||||
// required feature which remote peer don't support.
|
|
||||||
func TestFeaturesLocalRequireError(t *testing.T) {
|
|
||||||
t.Parallel()
|
t.Parallel()
|
||||||
|
|
||||||
const (
|
tests := []struct {
|
||||||
first = "first"
|
bits []FeatureBit
|
||||||
second = "second"
|
expectedEncoded []byte
|
||||||
)
|
}{
|
||||||
|
{
|
||||||
localFeatures := NewFeatureVector([]Feature{
|
bits: nil,
|
||||||
{first, OptionalFlag},
|
expectedEncoded: []byte{0x00, 0x00},
|
||||||
{second, RequiredFlag},
|
},
|
||||||
})
|
{
|
||||||
|
bits: []FeatureBit{2, 3, 7},
|
||||||
remoteFeatures := NewFeatureVector([]Feature{
|
expectedEncoded: []byte{0x00, 0x01, 0x8C},
|
||||||
{first, OptionalFlag},
|
},
|
||||||
})
|
{
|
||||||
|
bits: []FeatureBit{2, 3, 8},
|
||||||
if _, err := localFeatures.Compare(remoteFeatures); err == nil {
|
expectedEncoded: []byte{0x00, 0x02, 0x01, 0x0C},
|
||||||
t.Fatal("error wasn't received")
|
},
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// TestOptionalFeature checks that if remote peer don't have the feature but
|
for i, test := range tests {
|
||||||
// on our side this feature is optional than we mark this feature as disabled.
|
fv := NewRawFeatureVector(test.bits...)
|
||||||
func TestOptionalFeature(t *testing.T) {
|
|
||||||
t.Parallel()
|
|
||||||
|
|
||||||
const first = "first"
|
// Test that Encode produces the correct serialization.
|
||||||
|
buffer := new(bytes.Buffer)
|
||||||
localFeatures := NewFeatureVector([]Feature{
|
err := fv.Encode(buffer)
|
||||||
{first, OptionalFlag},
|
|
||||||
})
|
|
||||||
|
|
||||||
remoteFeatures := NewFeatureVector([]Feature{})
|
|
||||||
|
|
||||||
shared, err := localFeatures.Compare(remoteFeatures)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("error while feature vector compare: %v", err)
|
t.Errorf("Failed to encode feature vector in case %d: %v", i, err)
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
if shared.IsActive(first) {
|
encoded := buffer.Bytes()
|
||||||
t.Fatal("locally feature was set but remote peer notified us" +
|
if !bytes.Equal(encoded, test.expectedEncoded) {
|
||||||
" that it don't have it")
|
t.Errorf("Wrong encoding in case %d: got %v, expected %v",
|
||||||
|
i, encoded, test.expectedEncoded)
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
// A feature with a non-existent name shouldn't be active.
|
// Test that decoding then re-encoding produces the same result.
|
||||||
if shared.IsActive("nothere") {
|
fv2 := NewRawFeatureVector()
|
||||||
t.Fatal("non-existent feature shouldn't be active")
|
err = fv2.Decode(bytes.NewReader(encoded))
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// TestSetRequireAfterInit checks that we can change the feature flag after
|
|
||||||
// initialization.
|
|
||||||
func TestSetRequireAfterInit(t *testing.T) {
|
|
||||||
t.Parallel()
|
|
||||||
|
|
||||||
const first = "first"
|
|
||||||
|
|
||||||
localFeatures := NewFeatureVector([]Feature{
|
|
||||||
{first, OptionalFlag},
|
|
||||||
})
|
|
||||||
localFeatures.SetFeatureFlag(first, RequiredFlag)
|
|
||||||
remoteFeatures := NewFeatureVector([]Feature{})
|
|
||||||
|
|
||||||
_, err := localFeatures.Compare(remoteFeatures)
|
|
||||||
if err == nil {
|
|
||||||
t.Fatalf("feature was set as required but error wasn't "+
|
|
||||||
"returned: %v", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// TestDecodeEncodeFeaturesVector checks that feature vector might be
|
|
||||||
// successfully encoded and decoded.
|
|
||||||
func TestDecodeEncodeFeaturesVector(t *testing.T) {
|
|
||||||
t.Parallel()
|
|
||||||
|
|
||||||
const first = "first"
|
|
||||||
|
|
||||||
f := NewFeatureVector([]Feature{
|
|
||||||
{first, OptionalFlag},
|
|
||||||
})
|
|
||||||
|
|
||||||
var b bytes.Buffer
|
|
||||||
if err := f.Encode(&b); err != nil {
|
|
||||||
t.Fatalf("error while encoding feature vector: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
nf, err := NewFeatureVectorFromReader(&b)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("error while decoding feature vector: %v", err)
|
t.Errorf("Failed to decode feature vector in case %d: %v", i, err)
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
// Assert equality of the two instances.
|
buffer2 := new(bytes.Buffer)
|
||||||
if !reflect.DeepEqual(f.flags, nf.flags) {
|
err = fv2.Encode(buffer2)
|
||||||
t.Fatalf("encode/decode feature vector don't match %v vs "+
|
if err != nil {
|
||||||
"%v", spew.Sdump(f), spew.Sdump(nf))
|
t.Errorf("Failed to re-encode feature vector in case %d: %v",
|
||||||
|
i, err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
reencoded := buffer2.Bytes()
|
||||||
|
if !bytes.Equal(reencoded, test.expectedEncoded) {
|
||||||
|
t.Errorf("Wrong re-encoding in case %d: got %v, expected %v",
|
||||||
|
i, reencoded, test.expectedEncoded)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestFeatureFlagString(t *testing.T) {
|
func TestFeatureVectorUnknownFeatures(t *testing.T) {
|
||||||
t.Parallel()
|
t.Parallel()
|
||||||
|
|
||||||
if OptionalFlag.String() != "optional" {
|
tests := []struct {
|
||||||
t.Fatalf("incorrect string, expected optional got %v",
|
bits []FeatureBit
|
||||||
OptionalFlag.String())
|
expectedUnknown []FeatureBit
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
bits: nil,
|
||||||
|
expectedUnknown: nil,
|
||||||
|
},
|
||||||
|
// Since bits {0, 3, 4, 5} are known, and only even bits are considered
|
||||||
|
// required (according to the "it's OK to be odd rule"), that leaves
|
||||||
|
// {2, 6} as both unknown and required.
|
||||||
|
{
|
||||||
|
bits: []FeatureBit{0, 1, 2, 3, 4, 5, 6, 7},
|
||||||
|
expectedUnknown: []FeatureBit{2, 6},
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
if RequiredFlag.String() != "required" {
|
for i, test := range tests {
|
||||||
t.Fatalf("incorrect string, expected required got %v",
|
rawVector := NewRawFeatureVector(test.bits...)
|
||||||
OptionalFlag.String())
|
fv := NewFeatureVector(rawVector, testFeatureNames)
|
||||||
}
|
|
||||||
|
|
||||||
fakeFlag := featureFlag(9)
|
unknown := fv.UnknownRequiredFeatures()
|
||||||
if fakeFlag.String() != "<unknown>" {
|
|
||||||
t.Fatalf("incorrect string, expected <unknown> got %v",
|
// Sort to make comparison independent of order
|
||||||
fakeFlag.String())
|
sort.Slice(unknown, func(i, j int) bool {
|
||||||
|
return unknown[i] < unknown[j]
|
||||||
|
})
|
||||||
|
if !reflect.DeepEqual(unknown, test.expectedUnknown) {
|
||||||
|
t.Errorf("Wrong unknown features in case %d: got %v, expected %v",
|
||||||
|
i, unknown, test.expectedUnknown)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFeatureNames(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
bit FeatureBit
|
||||||
|
expectedName string
|
||||||
|
expectedKnown bool
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
bit: 0,
|
||||||
|
expectedName: "feature1(0)",
|
||||||
|
expectedKnown: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
bit: 1,
|
||||||
|
expectedName: "unknown(1)",
|
||||||
|
expectedKnown: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
bit: 2,
|
||||||
|
expectedName: "unknown(2)",
|
||||||
|
expectedKnown: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
bit: 3,
|
||||||
|
expectedName: "feature2(3)",
|
||||||
|
expectedKnown: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
bit: 4,
|
||||||
|
expectedName: "feature3(4)",
|
||||||
|
expectedKnown: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
bit: 5,
|
||||||
|
expectedName: "feature3(5)",
|
||||||
|
expectedKnown: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
bit: 6,
|
||||||
|
expectedName: "unknown(6)",
|
||||||
|
expectedKnown: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
bit: 7,
|
||||||
|
expectedName: "unknown(7)",
|
||||||
|
expectedKnown: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
fv := NewFeatureVector(nil, testFeatureNames)
|
||||||
|
for _, test := range tests {
|
||||||
|
name := fv.Name(test.bit)
|
||||||
|
if name != test.expectedName {
|
||||||
|
t.Errorf("Name for feature bit %d is incorrect: "+
|
||||||
|
"expected %s, got %s", test.bit, name, test.expectedName)
|
||||||
|
}
|
||||||
|
|
||||||
|
known := fv.IsKnown(test.bit)
|
||||||
|
if known != test.expectedKnown {
|
||||||
|
t.Errorf("IsKnown for feature bit %d is incorrect: "+
|
||||||
|
"expected %v, got %v", test.bit, known, test.expectedKnown)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user