Skip to content

Commit 6b4828a

Browse files
iwdgoFiloSottile
authored andcommitted
encoding/asn1: fix returned type for an Object Identifier
Unmarshal/Marshal/Unmarshal was not idempotent as the Object Identifier type was not returned through the interface. The limit case OID = 0 returns an error. The zero OID is 0.0 A test is fixed to use the Object Identifier type. Other related test are added. Fixes #11130 Change-Id: I15483a3126066c9b99cf5bd9c4b0cc15ec1d61ca Reviewed-on: https://go-review.googlesource.com/113837 Run-TryBot: Brad Fitzpatrick <[email protected]> TryBot-Result: Gobot Gobot <[email protected]> Reviewed-by: Filippo Valsorda <[email protected]>
1 parent c4f9fa1 commit 6b4828a

File tree

3 files changed

+59
-2
lines changed

3 files changed

+59
-2
lines changed

src/encoding/asn1/asn1.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -250,7 +250,7 @@ func (oi ObjectIdentifier) String() string {
250250
// parseObjectIdentifier parses an OBJECT IDENTIFIER from the given bytes and
251251
// returns it. An object identifier is a sequence of variable length integers
252252
// that are assigned in a hierarchy.
253-
func parseObjectIdentifier(bytes []byte) (s []int, err error) {
253+
func parseObjectIdentifier(bytes []byte) (s ObjectIdentifier, err error) {
254254
if len(bytes) == 0 {
255255
err = SyntaxError{"zero length OBJECT IDENTIFIER"}
256256
return

src/encoding/asn1/asn1_test.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -227,7 +227,7 @@ func TestBitStringRightAlign(t *testing.T) {
227227
type objectIdentifierTest struct {
228228
in []byte
229229
ok bool
230-
out []int
230+
out ObjectIdentifier // has base type[]int
231231
}
232232

233233
var objectIdentifierTestData = []objectIdentifierTest{

src/encoding/asn1/marshal_test.go

+57
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ import (
1111
"strings"
1212
"testing"
1313
"time"
14+
"reflect"
1415
)
1516

1617
type intStruct struct {
@@ -253,6 +254,62 @@ func TestInvalidUTF8(t *testing.T) {
253254
}
254255
}
255256

257+
func TestMarshalOID(t *testing.T) {
258+
var marshalTestsOID = []marshalTest{
259+
{[]byte("\x06\x01\x30"), "0403060130"}, // bytes format returns a byte sequence \x04
260+
// {ObjectIdentifier([]int{0}), "060100"}, // returns an error as OID 0.0 has the same encoding
261+
{[]byte("\x06\x010"), "0403060130"}, // same as above "\x06\x010" = "\x06\x01" + "0"
262+
{ObjectIdentifier([]int{2,999,3}), "0603883703"}, // Example of ITU-T X.690
263+
{ObjectIdentifier([]int{0,0}), "060100"}, // zero OID
264+
}
265+
for i, test := range marshalTestsOID {
266+
data, err := Marshal(test.in)
267+
if err != nil {
268+
t.Errorf("#%d failed: %s", i, err)
269+
}
270+
out, _ := hex.DecodeString(test.out)
271+
if !bytes.Equal(out, data) {
272+
t.Errorf("#%d got: %x want %x\n\t%q\n\t%q", i, data, out, data, out)
273+
}
274+
}
275+
}
276+
277+
func TestIssue11130(t *testing.T) {
278+
data := []byte("\x06\x010") // == \x06\x01\x30 == OID = 0 (the figure)
279+
var v interface{}
280+
// v has Zero value here and Elem() would panic
281+
_, err := Unmarshal(data, &v)
282+
if err != nil {
283+
t.Errorf("%v", err)
284+
return
285+
}
286+
if reflect.TypeOf(v).String() != reflect.TypeOf(ObjectIdentifier{}).String() {
287+
t.Errorf("marshal OID returned an invalid type")
288+
return
289+
}
290+
291+
data1, err := Marshal(v)
292+
if err != nil {
293+
t.Errorf("%v", err)
294+
return
295+
}
296+
297+
if !bytes.Equal(data,data1) {
298+
t.Errorf("got: %q, want: %q \n", data1, data)
299+
return
300+
}
301+
302+
var v1 interface{}
303+
_, err = Unmarshal(data1, &v1)
304+
if err != nil {
305+
t.Errorf("%v", err)
306+
return
307+
}
308+
if !reflect.DeepEqual(v, v1) {
309+
t.Errorf("got: %#v data=%q , want : %#v data=%q\n ", v1, data1, v, data)
310+
}
311+
}
312+
256313
func BenchmarkMarshal(b *testing.B) {
257314
b.ReportAllocs()
258315

0 commit comments

Comments
 (0)