forked from rosspeoples/go-snmplib
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathber.go
404 lines (363 loc) · 11.2 KB
/
ber.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
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
package snmplib
/* This file implements BER ASN1 encoding and decoding.
References : http://rane.com/note161.html
This package was made due to the inability of the encoding/asn1 library to
parse SNMP packets received from actual network devices. In order to fix
encoding/asn1 I would need to make deep changes in that core library file.
First difference is that this file works differently from the standard
libary one : this will convert between []interface{} and ASN1, whereas
encoding/asn1 converts between structs and ASN1.
Furthermore encoding/asn1 is an implementation of DER, whereas this does BER
(DER is a subset of BER). They're different like xml and html are different.
In theory html should be valid xml, in practice it's not. This means you can't
use an existing xml parser to parse html if you communicate with external
devices, because it wouldn't parse. Likewise you can't use a DER parser to
parse BER.
*/
import (
"errors"
"fmt"
"strconv"
"time"
)
// BERType constants for the Type of the TLV field.
type BERType uint8
// Constants for the different types of the TLV fields.
const (
AsnBoolean BERType = 0x01
AsnInteger BERType = 0x02
AsnBitStr BERType = 0x03
AsnOctetStr BERType = 0x04
AsnNull BERType = 0x05
AsnObjectID BERType = 0x06
AsnSequence BERType = 0x10
AsnSet BERType = 0x11
AsnUniversal BERType = 0x00
AsnApplication BERType = 0x40
AsnContext BERType = 0x80
AsnPrivate BERType = 0xC0
AsnPrimitive BERType = 0x00
AsnConstructor BERType = 0x20
AsnLongLen BERType = 0x80
AsnExtensionID BERType = 0x1F
AsnBit8 BERType = 0x80
Integer BERType = AsnUniversal | 0x02
Integer32 BERType = AsnUniversal | 0x02
Bitstring BERType = AsnUniversal | 0x03
Octetstring BERType = AsnUniversal | 0x04
Null BERType = AsnUniversal | 0x05
UOid BERType = AsnUniversal | 0x06
Sequence BERType = AsnConstructor | 0x10
Ipaddress BERType = AsnApplication | 0x00
Counter BERType = AsnApplication | 0x01
Counter32 BERType = AsnApplication | 0x01
Gauge BERType = AsnApplication | 0x02
Gauge32 BERType = AsnApplication | 0x02
Timeticks BERType = AsnApplication | 0x03
Opaque BERType = AsnApplication | 0x04
Counter64 BERType = AsnApplication | 0x06
AsnGetRequest BERType = 0xa0
AsnGetNextRequest BERType = 0xa1
AsnGetResponse BERType = 0xa2
AsnSetRequest BERType = 0xa3
AsnTrap BERType = 0xa4
AsnGetBulkRequest BERType = 0xa5
AsnInform BERType = 0xa6
AsnTrap2 BERType = 0xa7
AsnReport BERType = 0xa8
noSuchObject BERType = 0x80
noSuchInstance BERType = 0x81
)
// SNMPVersion indicates which SNMP version is in use.
type SNMPVersion uint8
// List the supported snmp versions.
const (
SNMPv1 SNMPVersion = 0
SNMPv2c SNMPVersion = 1
SNMPv3 SNMPVersion = 3
)
// EncodeLength encodes an integer value as a BER compliant length value.
func EncodeLength(length int) []byte {
// The first bit is used to indicate whether this is the final byte
// encoding the length. So, if the first bit is 0, just return a one
// byte response containing the byte-encoded length.
if length <= 0x7f {
return []byte{byte(length)}
}
// If the length is bigger the format is, first bit 1 + the rest of the
// bits in the first byte encode the length of the length, then follows
// the actual length.
// Technically the SNMP spec allows for packet lengths longer than can be
// specified in a 127-byte encoded integer, however, going out on a limb
// here, I don't think I'm going to support a use case that insane.
r := EncodeInteger(length)
numOctets := len(r)
result := make([]byte, 1+numOctets)
result[0] = 0x80 | byte(numOctets)
for i, b := range r {
result[1+i] = b
}
return result
}
// DecodeLength returns the length and the length of the length or an error.
// Caveats: Does not support indefinite length. Couldn't find any
// SNMP packet dump actually using that.
func DecodeLength(toparse []byte) (int, int, error) {
// If the first bit is zero, the rest of the first byte indicates the length. Values up to 127 are encoded this way (unless you're using indefinite length, but we don't support that)
if toparse[0] == 0x80 {
return 0, 0, fmt.Errorf("we don't support indefinite length encoding")
}
if toparse[0]&0x80 == 0 {
return int(toparse[0]), 1, nil
}
// If the first bit is one, the rest of the first byte encodes the length of then encoded length. So read how many bytes are part of the length.
numOctets := int(toparse[0] & 0x7f)
if len(toparse) < 1+numOctets {
return 0, 0, fmt.Errorf("invalid length")
}
// Decode the specified number of bytes as a BER Integer encoded
// value.
val, err := DecodeInteger(toparse[1 : numOctets+1])
if err != nil {
return 0, 0, err
}
return val, 1 + numOctets, nil
}
// DecodeCounter64 decodes a counter64.
func DecodeCounter64(toparse []byte) (uint64, error) {
if len(toparse) > 8 {
return 0, fmt.Errorf("don't support more than 64 bits")
}
var val uint64
val = 0
for _, b := range toparse {
val = val*256 + uint64(b)
}
return val, nil
}
// DecodeInteger decodes an integer. Will error out if it's longer than 64 bits.
func DecodeInteger(toparse []byte) (int, error) {
if len(toparse) > 8 {
return 0, fmt.Errorf("don't support more than 64 bits")
}
val := 0
for _, b := range toparse {
val = val*256 + int(b)
}
return val, nil
}
// DecodeIntegerSigned decodes a signed integer. Will error out if it's longer than 64 bits.
func DecodeIntegerSigned(toparse []byte) (int, error) {
if len(toparse) > 8 {
return 0, fmt.Errorf("don't support more than 64 bits")
}
val := 0
for _, b := range toparse {
val = val*256 + int(b)
}
// If highest order bit is 1, number is negative: decode as 2's complement.
if toparse[0]&0x80 != 0 {
nbits := len(toparse) * 8
twotonbits := uint(1) << uint(nbits)
val = val - int(twotonbits)
}
return val, nil
}
// DecodeIPAddress decodes an IP address.
func DecodeIPAddress(toparse []byte) (string, error) {
if len(toparse) != 4 {
return "", fmt.Errorf("need 4 bytes for IP address")
}
return fmt.Sprintf("%d.%d.%d.%d", toparse[0], toparse[1], toparse[2], toparse[3]), nil
}
// EncodeInteger encodes an integer to BER format.
func EncodeInteger(toEncode int) []byte {
if toEncode == 0 {
return []byte{0}
}
result := make([]byte, 8)
pos := 7
i := toEncode
for i > 0 {
result[pos] = byte(i % 256)
i = i >> 8
pos--
}
if result[pos+1] >= 0x80 {
result[pos] = 0x00
pos--
}
return result[pos+1 : 8]
}
// DecodeSequence decodes BER binary data into into *[]interface{}.
func DecodeSequence(toparse []byte) ([]interface{}, error) {
var result []interface{}
if len(toparse) < 2 {
return nil, fmt.Errorf("sequence cannot be shorter than 2 bytes")
}
sqType := BERType(toparse[0])
result = append(result, sqType)
// Bit 6 is the P/C primitive/constructed bit. Which means it's a set, essentially.
if sqType != Sequence && (toparse[0]&0x20 == 0) {
return nil, fmt.Errorf("byte array parsed in is not a sequence")
}
seqLength, seqLenLen, err := DecodeLength(toparse[1:])
if err != nil {
return nil, errors.New("failed to parse sequence length" + strconv.Itoa(seqLenLen))
}
if seqLength == 0 {
return result, nil
}
lidx := 0
idx := 1 + seqLenLen
if 1+seqLenLen+seqLength > len(toparse) {
return nil, errors.New("sequence does not contain the amount of bytes reported in its length")
}
toparse = toparse[:(1 + seqLenLen + seqLength)]
// Let's guarantee progress.
for idx < len(toparse) && idx > lidx {
berType := toparse[idx]
berLength, berLenLen, err := DecodeLength(toparse[idx+1:])
if err != nil {
return nil, fmt.Errorf("length parse error @ idx %v", idx)
}
berValue := toparse[idx+1+berLenLen : idx+1+berLenLen+berLength]
berAll := toparse[idx : idx+1+berLenLen+berLength]
switch BERType(berType) {
case AsnBoolean:
if berLength != 1 {
return nil, fmt.Errorf("boolean length != 1 @ idx %v", idx)
}
result = append(result, berValue[0] == 0)
case AsnInteger:
decodedValue, err := DecodeIntegerSigned(berValue)
if err != nil {
return nil, err
}
result = append(result, decodedValue)
case AsnOctetStr:
result = append(result, string(berValue))
case AsnNull:
result = append(result, nil)
case AsnObjectID:
oid, err := DecodeOid(berValue)
if err != nil {
return nil, err
}
result = append(result, *oid)
case Gauge32, Counter32:
val, err := DecodeInteger(berValue)
if err != nil {
return nil, err
}
result = append(result, val)
case Counter64:
val, err := DecodeCounter64(berValue)
if err != nil {
return nil, err
}
result = append(result, val)
case Timeticks:
val, err := DecodeInteger(berValue)
if err != nil {
return nil, err
}
result = append(result, time.Duration(val)*10*time.Millisecond)
case Ipaddress:
val, err := DecodeIPAddress(berValue)
if err != nil {
return nil, err
}
result = append(result, val)
case Sequence:
pdu, err := DecodeSequence(berAll)
if err != nil {
return nil, err
}
result = append(result, pdu)
case AsnGetNextRequest, AsnGetRequest, AsnGetResponse, AsnReport, AsnTrap2, AsnTrap:
pdu, err := DecodeSequence(berAll)
if err != nil {
return nil, err
}
result = append(result, pdu)
case noSuchObject:
return nil, fmt.Errorf("No Such Object")
case noSuchInstance:
return nil, fmt.Errorf("No Such Instance currently exists at this OID")
default:
return nil, fmt.Errorf("did not understand type %v", berType)
}
lidx = idx
idx = idx + 1 + berLenLen + berLength
}
return result, nil
}
// EncodeSequence will encode an []interface{} into an SNMP bytestream.
func EncodeSequence(toEncode []interface{}) ([]byte, error) {
switch toEncode[0].(type) {
default:
return nil, fmt.Errorf("first element of sequence to encode should be sequence type")
case BERType:
// OK
}
seqType := toEncode[0].(BERType)
var toEncap []byte
for _, val := range toEncode[1:] {
switch val := val.(type) {
default:
return nil, fmt.Errorf("couldn't handle type %T", val)
case nil:
toEncap = append(toEncap, byte(AsnNull))
toEncap = append(toEncap, 0)
case int:
enc := EncodeInteger(val)
// TODO encode length ?
toEncap = append(toEncap, byte(AsnInteger))
toEncap = append(toEncap, byte(len(enc)))
for _, b := range enc {
toEncap = append(toEncap, b)
}
case string:
enc := []byte(val)
toEncap = append(toEncap, byte(AsnOctetStr))
for _, b := range EncodeLength(len(enc)) {
toEncap = append(toEncap, b)
}
for _, b := range enc {
toEncap = append(toEncap, b)
}
case Oid:
enc, err := val.Encode()
if err != nil {
return nil, err
}
toEncap = append(toEncap, byte(AsnObjectID))
encLen := EncodeLength(len(enc))
for _, b := range encLen {
toEncap = append(toEncap, b)
}
for _, b := range enc {
toEncap = append(toEncap, b)
}
case []interface{}:
enc, err := EncodeSequence(val)
if err != nil {
return nil, err
}
for _, b := range enc {
toEncap = append(toEncap, b)
}
}
}
l := EncodeLength(len(toEncap))
// Encode length ...
result := []byte{byte(seqType)}
for _, b := range l {
result = append(result, b)
}
for _, b := range toEncap {
result = append(result, b)
}
return result, nil
}