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
|
// +build gofuzz
package qpack
import (
"bytes"
"fmt"
"reflect"
"github.com/marten-seemann/qpack"
)
func Fuzz(data []byte) int {
if len(data) < 1 {
return 0
}
chunkLen := int(data[0]) + 1
data = data[1:]
fields, err := qpack.NewDecoder(nil).DecodeFull(data)
if err != nil {
return 0
}
if len(fields) == 0 {
return 0
}
var writtenFields []qpack.HeaderField
decoder := qpack.NewDecoder(func(hf qpack.HeaderField) {
writtenFields = append(writtenFields, hf)
})
for len(data) > 0 {
var chunk []byte
if chunkLen <= len(data) {
chunk = data[:chunkLen]
data = data[chunkLen:]
} else {
chunk = data
data = nil
}
n, err := decoder.Write(chunk)
if err != nil {
return 0
}
if n != len(chunk) {
panic("len error")
}
}
if !reflect.DeepEqual(fields, writtenFields) {
fmt.Printf("%#v vs %#v", fields, writtenFields)
panic("Write() and DecodeFull() produced different results")
}
buf := &bytes.Buffer{}
encoder := qpack.NewEncoder(buf)
for _, hf := range fields {
if err := encoder.WriteField(hf); err != nil {
panic(err)
}
}
if err := encoder.Close(); err != nil {
panic(err)
}
encodedFields, err := qpack.NewDecoder(nil).DecodeFull(buf.Bytes())
if err != nil {
fmt.Printf("Fields: %#v\n", fields)
panic(err)
}
if !reflect.DeepEqual(fields, encodedFields) {
fmt.Printf("%#v vs %#v", fields, encodedFields)
panic("unequal")
}
return 0
}
|