-
Notifications
You must be signed in to change notification settings - Fork 4
/
varint.go
167 lines (133 loc) · 2.77 KB
/
varint.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
package protocol
import (
"bytes"
"errors"
"fmt"
"io"
"math"
"github.com/vulpemventures/neutrino-elements/pkg/binary"
)
var errInvalidVarIntValue = errors.New("invalid varint value")
// VarInt is variable length integer.
type VarInt struct {
Value interface{}
}
var _ binary.Unmarshaler = (*VarInt)(nil)
var _ binary.Marshaler = (*VarInt)(nil)
func newFromInt(i int) VarInt {
if i < 0 {
return VarInt{Value: uint8(0)}
}
if i <= 0xFC {
return VarInt{Value: uint8(i)}
}
if i <= 0xFFFF {
return VarInt{Value: uint16(i)}
}
if i <= 0xFFFFFFFF {
return VarInt{Value: uint32(i)}
}
return VarInt{Value: uint64(i)}
}
func NewVarint(u interface{}) (VarInt, error) {
switch v := u.(type) {
case int:
return newFromInt(v), nil
case uint8, uint16, uint32:
return VarInt{Value: v}, nil
case uint64:
var err error = nil
if v > math.MaxInt64 {
v = math.MaxInt64
}
return VarInt{Value: v}, err
}
return VarInt{Value: nil}, errInvalidVarIntValue
}
// Int returns returns value as 'int'.
func (vi VarInt) Int() (int, error) {
switch v := vi.Value.(type) {
case uint8:
return int(v), nil
case uint16:
return int(v), nil
case uint32:
return int(v), nil
case uint64:
// Assume we'll never get value more than MaxInt64.
if v > math.MaxInt64 {
return math.MaxInt64, nil
}
return int(v), nil
}
return 0, errInvalidVarIntValue
}
func (vi *VarInt) MarshalBinary() ([]byte, error) {
buf := bytes.NewBuffer([]byte{})
switch vi.Value.(type) {
case uint16:
if err := buf.WriteByte(0xFD); err != nil {
return nil, err
}
case uint32:
if err := buf.WriteByte(0xFE); err != nil {
return nil, err
}
case uint64:
if err := buf.WriteByte(0xFF); err != nil {
return nil, err
}
case uint8:
break
default:
return nil, fmt.Errorf("unknown type of varint value: %+v", vi.Value)
}
b, err := binary.MarshalForVarint(vi.Value)
if err != nil {
return nil, err
}
if _, err := buf.Write(b); err != nil {
return nil, err
}
return buf.Bytes(), nil
}
// UnmarshalBinary implements binary.Unmarshaler interface.
func (vi *VarInt) UnmarshalBinary(r io.Reader) error {
var b uint8
lr := io.LimitReader(r, 1)
if err := binary.NewDecoder(lr).Decode(&b); err != nil {
return err
}
if b < 0xFD {
vi.Value = b
return nil
}
if b == 0xFD {
lr := io.LimitReader(r, 2)
v, err := binary.NewDecoder(lr).DecodeUint16ForVarint()
if err != nil {
return err
}
vi.Value = v
return nil
}
if b == 0xFE {
var v uint32
lr := io.LimitReader(r, 4)
if err := binary.NewDecoder(lr).Decode(&v); err != nil {
return err
}
vi.Value = v
return nil
}
if b == 0xFF {
var v uint64
lr := io.LimitReader(r, 8)
if err := binary.NewDecoder(lr).Decode(&v); err != nil {
return err
}
vi.Value = v
return nil
}
return nil
}