lnwire/features: add EncodeBase32 and DecodeBase32 w/ generic helpers

This commit is contained in:
Conner Fromknecht 2019-08-26 13:07:21 -07:00
parent 2953f3532a
commit 8c2176fbf8
No known key found for this signature in database
GPG Key ID: E7D737B67FA592C7

@ -169,12 +169,25 @@ func (fv *RawFeatureVector) Encode(w io.Writer) error {
return err return err
} }
return fv.encode(w, length, 8)
}
// EncodeBase32 writes the feature vector in base32 representation. Every feature
// encoded as a bit, and the bit vector is serialized using the least number of
// bytes.
func (fv *RawFeatureVector) EncodeBase32(w io.Writer) error {
length := fv.SerializeSize32()
return fv.encode(w, length, 5)
}
// encode writes the feature vector
func (fv *RawFeatureVector) encode(w io.Writer, length, width int) error {
// Generate the data and write it. // Generate the data and write it.
data := make([]byte, length) data := make([]byte, length)
for feature := range fv.features { for feature := range fv.features {
byteIndex := int(feature / 8) byteIndex := int(feature) / width
bitIndex := feature % 8 bitIndex := int(feature) % width
data[length-byteIndex-1] |= 1 << bitIndex data[length-byteIndex-1] |= 1 << uint(bitIndex)
} }
_, err := w.Write(data) _, err := w.Write(data)
@ -193,6 +206,19 @@ func (fv *RawFeatureVector) Decode(r io.Reader) error {
} }
length := binary.BigEndian.Uint16(l[:]) length := binary.BigEndian.Uint16(l[:])
return fv.decode(r, int(length), 8)
}
// DecodeBase32 reads the feature vector from its base32 representation. Every
// feature encoded as a bit, and the bit vector is serialized using the least
// number of bytes.
func (fv *RawFeatureVector) DecodeBase32(r io.Reader, length int) error {
return fv.decode(r, length, 5)
}
// decode reads a feature vector from the next length bytes of the io.Reader,
// assuming each byte has width feature bits encoded per byte.
func (fv *RawFeatureVector) decode(r io.Reader, length, width int) error {
// Read the feature vector data. // Read the feature vector data.
data := make([]byte, length) data := make([]byte, length)
if _, err := io.ReadFull(r, data); err != nil { if _, err := io.ReadFull(r, data); err != nil {
@ -200,10 +226,10 @@ func (fv *RawFeatureVector) Decode(r io.Reader) error {
} }
// Set feature bits from parsed data. // Set feature bits from parsed data.
bitsNumber := len(data) * 8 bitsNumber := len(data) * width
for i := 0; i < bitsNumber; i++ { for i := 0; i < bitsNumber; i++ {
byteIndex := uint16(i / 8) byteIndex := int(i / width)
bitIndex := uint(i % 8) bitIndex := uint(i % width)
if (data[length-byteIndex-1]>>bitIndex)&1 == 1 { if (data[length-byteIndex-1]>>bitIndex)&1 == 1 {
fv.Set(FeatureBit(i)) fv.Set(FeatureBit(i))
} }