155 строки
3.9 KiB
Go
155 строки
3.9 KiB
Go
// Copyright 2017 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package cryptobyte_test
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
|
|
"golang.org/x/crypto/cryptobyte"
|
|
"golang.org/x/crypto/cryptobyte/asn1"
|
|
)
|
|
|
|
func ExampleString_lengthPrefixed() {
|
|
// This is an example of parsing length-prefixed data (as found in, for
|
|
// example, TLS). Imagine a 16-bit prefixed series of 8-bit prefixed
|
|
// strings.
|
|
|
|
input := cryptobyte.String([]byte{0, 12, 5, 'h', 'e', 'l', 'l', 'o', 5, 'w', 'o', 'r', 'l', 'd'})
|
|
var result []string
|
|
|
|
var values cryptobyte.String
|
|
if !input.ReadUint16LengthPrefixed(&values) ||
|
|
!input.Empty() {
|
|
panic("bad format")
|
|
}
|
|
|
|
for !values.Empty() {
|
|
var value cryptobyte.String
|
|
if !values.ReadUint8LengthPrefixed(&value) {
|
|
panic("bad format")
|
|
}
|
|
|
|
result = append(result, string(value))
|
|
}
|
|
|
|
// Output: []string{"hello", "world"}
|
|
fmt.Printf("%#v\n", result)
|
|
}
|
|
|
|
func ExampleString_aSN1() {
|
|
// This is an example of parsing ASN.1 data that looks like:
|
|
// Foo ::= SEQUENCE {
|
|
// version [6] INTEGER DEFAULT 0
|
|
// data OCTET STRING
|
|
// }
|
|
|
|
input := cryptobyte.String([]byte{0x30, 12, 0xa6, 3, 2, 1, 2, 4, 5, 'h', 'e', 'l', 'l', 'o'})
|
|
|
|
var (
|
|
version int64
|
|
data, inner, versionBytes cryptobyte.String
|
|
haveVersion bool
|
|
)
|
|
if !input.ReadASN1(&inner, asn1.SEQUENCE) ||
|
|
!input.Empty() ||
|
|
!inner.ReadOptionalASN1(&versionBytes, &haveVersion, asn1.Tag(6).Constructed().ContextSpecific()) ||
|
|
(haveVersion && !versionBytes.ReadASN1Integer(&version)) ||
|
|
(haveVersion && !versionBytes.Empty()) ||
|
|
!inner.ReadASN1(&data, asn1.OCTET_STRING) ||
|
|
!inner.Empty() {
|
|
panic("bad format")
|
|
}
|
|
|
|
// Output: haveVersion: true, version: 2, data: hello
|
|
fmt.Printf("haveVersion: %t, version: %d, data: %s\n", haveVersion, version, string(data))
|
|
}
|
|
|
|
func ExampleBuilder_aSN1() {
|
|
// This is an example of building ASN.1 data that looks like:
|
|
// Foo ::= SEQUENCE {
|
|
// version [6] INTEGER DEFAULT 0
|
|
// data OCTET STRING
|
|
// }
|
|
|
|
version := int64(2)
|
|
data := []byte("hello")
|
|
const defaultVersion = 0
|
|
|
|
var b cryptobyte.Builder
|
|
b.AddASN1(asn1.SEQUENCE, func(b *cryptobyte.Builder) {
|
|
if version != defaultVersion {
|
|
b.AddASN1(asn1.Tag(6).Constructed().ContextSpecific(), func(b *cryptobyte.Builder) {
|
|
b.AddASN1Int64(version)
|
|
})
|
|
}
|
|
b.AddASN1OctetString(data)
|
|
})
|
|
|
|
result, err := b.Bytes()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// Output: 300ca603020102040568656c6c6f
|
|
fmt.Printf("%x\n", result)
|
|
}
|
|
|
|
func ExampleBuilder_lengthPrefixed() {
|
|
// This is an example of building length-prefixed data (as found in,
|
|
// for example, TLS). Imagine a 16-bit prefixed series of 8-bit
|
|
// prefixed strings.
|
|
input := []string{"hello", "world"}
|
|
|
|
var b cryptobyte.Builder
|
|
b.AddUint16LengthPrefixed(func(b *cryptobyte.Builder) {
|
|
for _, value := range input {
|
|
b.AddUint8LengthPrefixed(func(b *cryptobyte.Builder) {
|
|
b.AddBytes([]byte(value))
|
|
})
|
|
}
|
|
})
|
|
|
|
result, err := b.Bytes()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// Output: 000c0568656c6c6f05776f726c64
|
|
fmt.Printf("%x\n", result)
|
|
}
|
|
|
|
func ExampleBuilder_lengthPrefixOverflow() {
|
|
// Writing more data that can be expressed by the length prefix results
|
|
// in an error from Bytes().
|
|
|
|
tooLarge := make([]byte, 256)
|
|
|
|
var b cryptobyte.Builder
|
|
b.AddUint8LengthPrefixed(func(b *cryptobyte.Builder) {
|
|
b.AddBytes(tooLarge)
|
|
})
|
|
|
|
result, err := b.Bytes()
|
|
fmt.Printf("len=%d err=%s\n", len(result), err)
|
|
|
|
// Output: len=0 err=cryptobyte: pending child length 256 exceeds 1-byte length prefix
|
|
}
|
|
|
|
func ExampleBuilderContinuation_errorHandling() {
|
|
var b cryptobyte.Builder
|
|
// Continuations that panic with a BuildError will cause Bytes to
|
|
// return the inner error.
|
|
b.AddUint16LengthPrefixed(func(b *cryptobyte.Builder) {
|
|
b.AddUint32(0)
|
|
panic(cryptobyte.BuildError{Err: errors.New("example error")})
|
|
})
|
|
|
|
result, err := b.Bytes()
|
|
fmt.Printf("len=%d err=%s\n", len(result), err)
|
|
|
|
// Output: len=0 err=example error
|
|
}
|