diff --git a/go.mod b/go.mod index 8f12f9df6d..0f9e423f8f 100644 --- a/go.mod +++ b/go.mod @@ -2,8 +2,6 @@ module github.com/algorand/go-algorand go 1.20 -replace github.com/algorand/msgp => ./msgp - require ( github.com/DataDog/zstd v1.5.2 github.com/algorand/avm-abi v0.2.0 @@ -12,7 +10,7 @@ require ( github.com/algorand/go-deadlock v0.2.3 github.com/algorand/go-sumhash v0.1.0 github.com/algorand/graphtrace v0.1.0 - github.com/algorand/msgp v1.1.55 + github.com/algorand/msgp v1.1.59 github.com/algorand/oapi-codegen v1.12.0-algorand.0 github.com/algorand/sortition v1.0.0 github.com/algorand/websocket v1.4.6 diff --git a/go.sum b/go.sum index 6c9d5f8827..2a0b6c8a23 100644 --- a/go.sum +++ b/go.sum @@ -67,8 +67,8 @@ github.com/algorand/go-sumhash v0.1.0 h1:b/QRhyLuF//vOcicBIxBXYW8bERNoeLxieht/dU github.com/algorand/go-sumhash v0.1.0/go.mod h1:OOe7jdDWUhLkuP1XytkK5gnLu9entAviN5DfDZh6XAc= github.com/algorand/graphtrace v0.1.0 h1:QemP1iT0W56SExD0NfiU6rsG34/v0Je6bg5UZnptEUM= github.com/algorand/graphtrace v0.1.0/go.mod h1:HscLQrzBdH1BH+5oehs3ICd8SYcXvnSL9BjfTu8WHCc= -github.com/algorand/msgp v1.1.55 h1:kWc9Xc08xtxCTWUiq1cRW5XGF+DFcfSGihYf0IZ/ivs= -github.com/algorand/msgp v1.1.55/go.mod h1:RqZQBzAFDWpwh5TlabzZkWy+6kwL9cvXfLbU0gD99EA= +github.com/algorand/msgp v1.1.59 h1:c7cAE5iynWJ3HysI2peIOtfUKIg/nOAMIJGoGUowGYY= +github.com/algorand/msgp v1.1.59/go.mod h1:RqZQBzAFDWpwh5TlabzZkWy+6kwL9cvXfLbU0gD99EA= github.com/algorand/oapi-codegen v1.12.0-algorand.0 h1:W9PvED+wAJc+9EeXPONnA+0zE9UhynEqoDs4OgAxKhk= github.com/algorand/oapi-codegen v1.12.0-algorand.0/go.mod h1:tIWJ9K/qrLDVDt5A1p82UmxZIEGxv2X+uoujdhEAL48= github.com/algorand/sortition v1.0.0 h1:PJiZtdSTBm4nArQrZXBnhlljHXhuyAXRJBqVWowQu3E= diff --git a/msgp/.gitignore b/msgp/.gitignore deleted file mode 100644 index 2d19289705..0000000000 --- a/msgp/.gitignore +++ /dev/null @@ -1,13 +0,0 @@ -_generated/generated.go -_generated/generated_test.go -_generated/*_gen.go -_generated/*_gen_test.go -_generated/embeddedStruct/*_gen.go -_generated/embeddedStruct/*_gen_test.go -msgp/defgen_test.go -msgp/cover.out -*~ -*.coverprofile -.idea/ -.vscode/ -cover.out diff --git a/msgp/.travis.yml b/msgp/.travis.yml deleted file mode 100644 index 5f574c093b..0000000000 --- a/msgp/.travis.yml +++ /dev/null @@ -1,11 +0,0 @@ -language: go - -go: - - 1.12.x - - tip - -env: - - GIMME_ARCH=amd64 - - GIMME_ARCH=386 - -script: "make travis" diff --git a/msgp/LICENSE b/msgp/LICENSE deleted file mode 100644 index 14d60424e8..0000000000 --- a/msgp/LICENSE +++ /dev/null @@ -1,8 +0,0 @@ -Copyright (c) 2014 Philip Hofer -Portions Copyright (c) 2009 The Go Authors (license at http://golang.org) where indicated - -Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. \ No newline at end of file diff --git a/msgp/Makefile b/msgp/Makefile deleted file mode 100644 index a77d20d5ce..0000000000 --- a/msgp/Makefile +++ /dev/null @@ -1,47 +0,0 @@ - -# NOTE: This Makefile is only necessary if you -# plan on developing the msgp tool and library. -# Installation can still be performed with a -# normal `go install`. - -# generated unit test files -MGEN = ./msgp/defgen_test.go - -SHELL := /bin/bash - -BIN = $(GOBIN)/msgp - -.PHONY: clean wipe install get-deps bench all - -$(BIN): */*.go - @go install ./... - -install: $(BIN) - -$(MGEN): ./msgp/defs_test.go - go generate ./msgp - -test: all - go test -covermode=atomic -coverprofile=cover.out ./... - -bench: all - go test -bench ./... - -clean: - $(RM) $(MGEN) - -wipe: clean - $(RM) $(BIN) - -get-deps: - go get -d -t ./... - -all: install $(MGEN) - -# travis CI enters here -travis: - go get -d -t ./... - go build -o "$${GOPATH%%:*}/bin/msgp" . - go generate ./msgp - go generate ./_generated - go test -v ./... ./_generated diff --git a/msgp/README.md b/msgp/README.md deleted file mode 100644 index f5d3528a03..0000000000 --- a/msgp/README.md +++ /dev/null @@ -1,4 +0,0 @@ -MessagePack Code Generator -======= - -Based on [github.com/tinylib/msgp](https://github.com/tinylib/msgp) diff --git a/msgp/gen/elem.go b/msgp/gen/elem.go deleted file mode 100644 index 707b56bd94..0000000000 --- a/msgp/gen/elem.go +++ /dev/null @@ -1,992 +0,0 @@ -package gen - -import ( - "fmt" - "go/ast" - "strconv" - "strings" -) - -var ( - identNext = 0 - identPrefix = "za" -) - -func resetIdent(prefix string) { - identPrefix = prefix - identNext = 0 -} - -// generate a random identifier name -func randIdent() string { - identNext++ - return fmt.Sprintf("%s%04d", identPrefix, identNext) -} - -// This code defines the type declaration tree. -// -// Consider the following: -// -// type Marshaler struct { -// Thing1 *float64 `msg:"thing1"` -// Body []byte `msg:"body"` -// } -// -// A parser using this generator as a backend -// should parse the above into: -// -// var val Elem = &Ptr{ -// name: "z", -// Value: &Struct{ -// Name: "Marshaler", -// Fields: []StructField{ -// { -// FieldTag: "thing1", -// FieldElem: &Ptr{ -// name: "z.Thing1", -// Value: &BaseElem{ -// name: "*z.Thing1", -// Value: Float64, -// Convert: false, -// }, -// }, -// }, -// { -// FieldTag: "body", -// FieldElem: &BaseElem{ -// name: "z.Body", -// Value: Bytes, -// Convert: false, -// }, -// }, -// }, -// }, -// } - -// Base is one of the -// base types -type Primitive uint8 - -// this is effectively the -// list of currently available -// ReadXxxx / WriteXxxx methods. -const ( - Invalid Primitive = iota - Bytes - String - Float32 - Float64 - Complex64 - Complex128 - Uint - Uint8 - Uint16 - Uint32 - Uint64 - Uintptr - Byte - Int - Int8 - Int16 - Int32 - Int64 - Bool - Intf // interface{} - Time // time.Time - Ext // extension - Error // error - Duration // time.Duration - - IDENT // IDENT means an unrecognized identifier -) - -// all of the recognized identities -// that map to primitive types -var primitives = map[string]Primitive{ - "[]byte": Bytes, - "string": String, - "float32": Float32, - "float64": Float64, - "complex64": Complex64, - "complex128": Complex128, - "uint": Uint, - "uint8": Uint8, - "uint16": Uint16, - "uint32": Uint32, - "uint64": Uint64, - "uintptr": Uintptr, - "byte": Byte, - "rune": Int32, - "int": Int, - "int8": Int8, - "int16": Int16, - "int32": Int32, - "int64": Int64, - "bool": Bool, - "interface{}": Intf, - "time.Time": Time, - "msgp.Extension": Ext, - "error": Error, - "time.Duration": Duration, -} - -// types built into the library -// that satisfy all of the -// interfaces. -var builtins = map[string]struct{}{ - "msgp.Raw": struct{}{}, - "msgp.Number": struct{}{}, -} - -// Callback represents a function that can is expected to be printed into the generated code. -// for example, at the end of a successful unmarshalling. -type Callback struct { - Fname string - CallbackType CallbackType -} - -type CallbackType uint64 - -// UnmarshalCallBack represents a type callback that should run over the generated code. -const UnmarshalCallBack CallbackType = 1 - -func (c Callback) IsUnmarshallCallback() bool { return c.CallbackType == UnmarshalCallBack } -func (c Callback) GetName() string { return c.Fname } - -// common data/methods for every Elem -type common struct { - vname, alias string - allocbound string - maxtotalbytes string - callbacks []Callback -} - -func (c *common) SetVarname(s string) { c.vname = s } -func (c *common) Varname() string { return c.vname } -func (c *common) Alias(typ string) { c.alias = typ } -func (c *common) SortInterface() string { return "" } -func (c *common) SetAllocBound(s string) { c.allocbound = s } -func (c *common) AllocBound() string { return c.allocbound } -func (c *common) SetMaxTotalBytes(s string) { c.maxtotalbytes = s } -func (c *common) MaxTotalBytes() string { return c.maxtotalbytes } -func (c *common) GetCallbacks() []Callback { return c.callbacks } -func (c *common) AddCallback(cb Callback) { c.callbacks = append(c.callbacks, cb) } -func (c *common) hidden() {} - -func IsDangling(e Elem) bool { - if be, ok := e.(*BaseElem); ok && be.Dangling() { - return true - } - return false -} - -// Elem is a go type capable of being -// serialized into MessagePack. It is -// implemented by *Ptr, *Struct, *Array, -// *Slice, *Map, and *BaseElem. -type Elem interface { - // SetVarname sets this nodes - // variable name and recursively - // sets the names of all its children. - // In general, this should only be - // called on the parent of the tree. - SetVarname(s string) - - // Varname returns the variable - // name of the element. - Varname() string - - // TypeName is the canonical - // go type name of the node - // e.g. "string", "int", "map[string]float64" - // OR the alias name, if it has been set. - TypeName() string - - // Alias sets a type (alias) name - Alias(typ string) - - // Copy should perform a deep copy of the object - Copy() Elem - - // Complexity returns a measure of the - // complexity of element (greater than - // or equal to 1.) - Complexity() int - - // ZeroExpr returns the expression for the correct zero/empty - // value. Can be used for assignment. - // Returns "" if zero/empty not supported for this Elem. - ZeroExpr() string - - // IfZeroExpr returns the expression to compare to zero/empty - // for this type. It is meant to be used in an if statement - // and may include the simple statement form followed by - // semicolon and then the expression. - // Returns "" if zero/empty not supported for this Elem. - IfZeroExpr() string - - // SortInterface returns the sort.Interface for sorting a - // slice of this type. - SortInterface() string - - // Comparable returns whether the type is comparable, along the lines - // of the Go spec (https://golang.org/ref/spec#Comparison_operators), - // used to determine whether we can compare to a zero value to determine - // zeroness. - Comparable() bool - - // SetAllocBound specifies the maximum number of elements to allocate - // when decoding this type. Meaningful for slices and maps. - // Blank means unspecified bound. "-" means no bound. - SetAllocBound(bound string) - - // AllocBound returns the maximum number of elements to allocate - // when decoding this type. Meaningful for slices and maps. - AllocBound() string - - // SetMaxTotalBytes specifies the maximum number of bytes to allocate when - // decoding this type. - // Blank means unspecified bound. "-" means no bound. - SetMaxTotalBytes(bound string) - - // MaxTotalBytes specifies the maximum number of bytes to allocate when - // decoding this type. Meaningful for slices of strings or byteslices. - MaxTotalBytes() string - - // AddCallback adds to the elem a Callback it should call at the end of marshaling - AddCallback(Callback) - - // GetCallbacks fetches all callbacks this Elem stored. - GetCallbacks() []Callback - - hidden() -} - -// Ident returns the *BaseElem that corresponds -// to the provided identity. -func Ident(importPrefix string, id string) *BaseElem { - p, ok := primitives[id] - if ok { - return &BaseElem{Value: p} - } - id = importPrefix + id - be := &BaseElem{Value: IDENT, IdentName: id} - be.Alias(id) - return be -} - -type Array struct { - common - Index string // index variable name - Size string // array size - SizeHint string // const object referred to by Size - Els Elem // child -} - -func (a *Array) SetVarname(s string) { - a.common.SetVarname(s) -ridx: - a.Index = randIdent() - - // try to avoid using the same - // index as a parent slice - if strings.Contains(a.Varname(), a.Index) { - goto ridx - } - - a.Els.SetVarname(fmt.Sprintf("%s[%s]", a.Varname(), a.Index)) -} - -func (a *Array) TypeName() string { - if a.common.alias != "" { - return a.common.alias - } - a.common.Alias(fmt.Sprintf("[%s]%s", a.Size, a.Els.TypeName())) - return a.common.alias -} - -func (a *Array) Copy() Elem { - b := *a - b.Els = a.Els.Copy() - return &b -} - -func (a *Array) Complexity() int { return 1 + a.Els.Complexity() } - -func (a *Array) sz() int { - szString := a.SizeHint - if szString == "" { - szString = a.Size - } - - s, err := strconv.Atoi(szString) - if err != nil { - panic(err) - } - - return s -} - -// ZeroExpr returns the zero/empty expression or empty string if not supported. -func (a *Array) ZeroExpr() string { - zeroElem := a.Els.ZeroExpr() - if zeroElem == "" { - return "" - } - - sz := a.sz() - - res := fmt.Sprintf("%s{", a.TypeName()) - for i := 0; i < sz; i++ { - res += fmt.Sprintf("%s, ", zeroElem) - } - res += "}" - return res -} - -// IfZeroExpr returns the expression to compare to zero/empty. -func (a *Array) IfZeroExpr() string { - // Special case for arrays of comparable elements: Go generates - // faster code if we just compare to a zero value. - if a.Els.Comparable() { - return fmt.Sprintf("%s == (%s{})", a.Varname(), a.TypeName()) - } - - sz := a.sz() - - var res string - for i := 0; i < sz; i++ { - el := a.Els.Copy() - el.SetVarname(fmt.Sprintf("%s[%d]", a.Varname(), i)) - if res != "" { - res += " && " - } - res += "(" + el.IfZeroExpr() + ")" - } - return res -} - -// Comparable returns whether this elem's type is comparable. -func (a *Array) Comparable() bool { - return a.Els.Comparable() -} - -// Map is a map[string]Elem -type Map struct { - common - Keyidx string // key variable name - Key Elem // type of map key - Validx string // value variable name - Value Elem // value element -} - -func (m *Map) SetVarname(s string) { - m.common.SetVarname(s) -ridx: - m.Keyidx = randIdent() - m.Validx = randIdent() - - // just in case - if m.Keyidx == m.Validx { - goto ridx - } - - m.Key.SetVarname(m.Keyidx) - m.Value.SetVarname(m.Validx) -} - -func (m *Map) TypeName() string { - if m.common.alias != "" { - return m.common.alias - } - m.common.Alias("map[" + m.Key.TypeName() + "]" + m.Value.TypeName()) - return m.common.alias -} - -func (m *Map) Copy() Elem { - g := *m - g.Key = m.Key.Copy() - g.Value = m.Value.Copy() - return &g -} - -func (m *Map) Complexity() int { return 2 + m.Value.Complexity() } - -// ZeroExpr returns the zero/empty expression or empty string if not supported. Always "nil" for this case. -func (m *Map) ZeroExpr() string { return "nil" } - -// IfZeroExpr returns the expression to compare to zero/empty. -func (m *Map) IfZeroExpr() string { return "len(" + m.Varname() + ") == 0" } - -// Comparable returns whether this elem's type is comparable. -func (m *Map) Comparable() bool { - return false -} - -type Slice struct { - common - Index string - Els Elem // The type of each element -} - -func (s *Slice) SetVarname(a string) { - s.common.SetVarname(a) - s.Index = randIdent() - varName := s.Varname() - if varName[0] == '*' { - // Pointer-to-slice requires parenthesis for slicing. - varName = "(" + varName + ")" - } - s.Els.SetVarname(fmt.Sprintf("%s[%s]", varName, s.Index)) -} - -func (s *Slice) TypeName() string { - if s.common.alias != "" { - return s.common.alias - } - s.common.Alias("[]" + s.Els.TypeName()) - return s.common.alias -} - -func (s *Slice) Copy() Elem { - z := *s - z.Els = s.Els.Copy() - return &z -} - -func (s *Slice) Complexity() int { - return 1 + s.Els.Complexity() -} - -// ZeroExpr returns the zero/empty expression or empty string if not supported. Always "nil" for this case. -func (s *Slice) ZeroExpr() string { return "nil" } - -// IfZeroExpr returns the expression to compare to zero/empty. -func (s *Slice) IfZeroExpr() string { return "len(" + s.Varname() + ") == 0" } - -// Comparable returns whether this elem's type is comparable. -func (s *Slice) Comparable() bool { - return false -} - -type Ptr struct { - common - Value Elem -} - -func (s *Ptr) SetVarname(a string) { - s.common.SetVarname(a) - - // struct fields are dereferenced - // automatically... - switch x := s.Value.(type) { - case *Struct: - // struct fields are automatically dereferenced - x.SetVarname(a) - return - - case *BaseElem: - // identities have pointer receivers - if x.Value == IDENT { - x.SetVarname(a) - } else { - x.SetVarname("*" + a) - } - return - - default: - s.Value.SetVarname("*" + a) - return - } -} - -func (s *Ptr) TypeName() string { - if s.common.alias != "" { - return s.common.alias - } - s.common.Alias("*" + s.Value.TypeName()) - return s.common.alias -} - -func (s *Ptr) Copy() Elem { - v := *s - v.Value = s.Value.Copy() - return &v -} - -func (s *Ptr) Complexity() int { return 1 + s.Value.Complexity() } - -func (s *Ptr) Needsinit() bool { - if be, ok := s.Value.(*BaseElem); ok && be.needsref { - return false - } - return true -} - -// ZeroExpr returns the zero/empty expression or empty string if not supported. Always "nil" for this case. -func (s *Ptr) ZeroExpr() string { return "nil" } - -// IfZeroExpr returns the expression to compare to zero/empty. -func (s *Ptr) IfZeroExpr() string { return s.Varname() + " == nil" } - -// Comparable returns whether this elem's type is comparable. -func (s *Ptr) Comparable() bool { - return false -} - -type Struct struct { - common - Fields []StructField // field list - AsTuple bool // write as an array instead of a map -} - -func (s *Struct) TypeName() string { - if s.common.alias != "" { - return s.common.alias - } - str := "struct{\n" - for i := range s.Fields { - str += s.Fields[i].FieldName + - " " + s.Fields[i].FieldElem.TypeName() + - " " + s.Fields[i].RawTag + ";\n" - } - str += "}" - s.common.Alias(str) - return s.common.alias -} - -func (s *Struct) SetVarname(a string) { - s.common.SetVarname(a) - writeStructFields(s.Fields, a) -} - -func (s *Struct) Copy() Elem { - g := *s - g.Fields = make([]StructField, len(s.Fields)) - copy(g.Fields, s.Fields) - for i := range s.Fields { - g.Fields[i].FieldElem = s.Fields[i].FieldElem.Copy() - } - return &g -} - -func (s *Struct) Complexity() int { - c := 1 - for i := range s.Fields { - c += s.Fields[i].FieldElem.Complexity() - } - return c -} - -// ZeroExpr returns the zero/empty expression or empty string if not supported. -func (s *Struct) ZeroExpr() string { - if s.alias == "" { - return "" // structs with no names not supported (for now) - } - return "(" + s.TypeName() + "{})" -} - -// IfZeroExpr returns the expression to compare to zero/empty. -func (s *Struct) IfZeroExpr() string { - if s.alias == "" { - return "" // structs with no names not supported (for now) - } - - var res string - for i := range s.Fields { - if !ast.IsExported(s.Fields[i].FieldName) { - continue - } - - fieldZero := s.Fields[i].FieldElem.IfZeroExpr() - if fieldZero != "" { - if res != "" { - res += " && " - } - res += "(" + fieldZero + ")" - } - } - return res -} - -// Comparable returns whether this elem's type is comparable. -func (s *Struct) Comparable() bool { - for _, sf := range s.Fields { - if !sf.FieldElem.Comparable() { - return false - } - } - return true -} - -// AnyHasTagPart returns true if HasTagPart(p) is true for any field. -func (s *Struct) AnyHasTagPart(pname string) bool { - for _, sf := range s.Fields { - if sf.HasTagPart(pname) { - return true - } - } - return false -} - -// UnderscoreStructHasTagPart returns true if HasTagPart(p) is true for the _struct field. -func (s *Struct) UnderscoreStructHasTagPart(pname string) bool { - for _, sf := range s.Fields { - if sf.FieldName == "_struct" && sf.HasTagPart(pname) { - return true - } - } - return false -} - -// HasAnyStructTag returns true if any of the fields in the struct have -// a codec: tag. This is used to determine which structs we can skip -// because they are not intended for encoding/decoding. -func (s *Struct) HasAnyStructTag() bool { - for _, sf := range s.Fields { - if sf.HasCodecTag { - return true - } - } - return false -} - -// HasUnderscoreStructTag returns true if there is a field named _struct -// with a codec: tag. This is used to ensure developers don't forget to -// annotate their structs with omitempty (unless explicitly opted out). -func (s *Struct) HasUnderscoreStructTag() bool { - for _, sf := range s.Fields { - if sf.FieldName == "_struct" && sf.HasCodecTag { - return true - } - } - return false -} - -type StructField struct { - FieldTag string // the string inside the `codec:""` tag up to the first comma - FieldTagParts []string // the string inside the `codec:""` tag split by commas - RawTag string // the full struct tag - HasCodecTag bool // has a `codec:` tag - FieldName string // the name of the struct field - FieldElem Elem // the field type - FieldPath []string // set of embedded struct names for accessing FieldName -} - -type byFieldTag []StructField - -func (a byFieldTag) Len() int { return len(a) } -func (a byFieldTag) Less(i, j int) bool { return a[i].FieldTag < a[j].FieldTag } -func (a byFieldTag) Swap(i, j int) { a[i], a[j] = a[j], a[i] } - -// HasTagPart returns true if the specified tag part (option) is present. -func (sf *StructField) HasTagPart(pname string) bool { - if len(sf.FieldTagParts) < 2 { - return false - } - for _, p := range sf.FieldTagParts[1:] { - if p == pname { - return true - } - } - return false -} - -type ShimMode int - -const ( - Cast ShimMode = iota - Convert -) - -// BaseElem is an element that -// can be represented by a primitive -// MessagePack type. -type BaseElem struct { - common - ShimMode ShimMode // Method used to shim - ShimToBase string // shim to base type, or empty - ShimFromBase string // shim from base type, or empty - Value Primitive // Type of element - IdentName string // name, for Value == IDENT - Convert bool // should we do an explicit conversion? - mustinline bool // must inline; not printable - needsref bool // needs reference for shim -} - -func (s *BaseElem) Dangling() bool { return s.mustinline } - -func (s *BaseElem) Alias(typ string) { - s.common.Alias(typ) - if s.Value != IDENT { - s.Convert = true - } - if strings.Contains(typ, ".") { - s.mustinline = true - } -} - -func (s *BaseElem) SetVarname(a string) { - // extensions whose parents - // are not pointers need to - // be explicitly referenced - if s.Value == Ext || s.needsref { - if strings.HasPrefix(a, "*") { - s.common.SetVarname(a[1:]) - return - } - s.common.SetVarname("&" + a) - return - } - - s.common.SetVarname(a) -} - -// TypeName returns the syntactically correct Go -// type name for the base element. -func (s *BaseElem) TypeName() string { - if s.common.alias != "" { - return s.common.alias - } - s.common.Alias(s.BaseType()) - return s.common.alias -} - -// ToBase, used if Convert==true, is used as tmp = {{ToBase}}({{Varname}}) -func (s *BaseElem) ToBase() string { - if s.ShimToBase != "" { - return s.ShimToBase - } - return s.BaseType() -} - -// FromBase, used if Convert==true, is used as {{Varname}} = {{FromBase}}(tmp) -func (s *BaseElem) FromBase() string { - if s.ShimFromBase != "" { - return s.ShimFromBase - } - return s.TypeName() -} - -// BaseName returns the string form of the -// base type (e.g. Float64, Ident, etc) -func (s *BaseElem) BaseName() string { - // time and duration are special cases; - // we strip the package prefix - if s.Value == Time { - return "Time" - } - if s.Value == Duration { - return "Duration" - } - return s.Value.String() -} - -func (s *BaseElem) BaseType() string { - switch s.Value { - case IDENT: - return s.TypeName() - - // exceptions to the naming/capitalization - // rule: - case Intf: - return "interface{}" - case Bytes: - return "[]byte" - case Time: - return "time.Time" - case Ext: - return "msgp.Extension" - - // everything else is base.String() with - // the first letter as lowercase - default: - return strings.ToLower(s.BaseName()) - } -} - -func (s *BaseElem) Needsref(b bool) { - s.needsref = b -} - -func (s *BaseElem) Copy() Elem { - g := *s - return &g -} - -func (s *BaseElem) Complexity() int { - if s.Convert && !s.mustinline { - return 2 - } - // we need to return 1 if !printable(), - // in order to make sure that stuff gets - // inlined appropriately - return 1 -} - -// Resolved returns whether or not -// the type of the element is -// a primitive or a builtin provided -// by the package. -func (s *BaseElem) Resolved() bool { - if s.Value == IDENT { - _, ok := builtins[s.TypeName()] - return ok - } - return true -} - -// ZeroExpr returns the zero/empty expression or empty string if not supported. -func (s *BaseElem) ZeroExpr() string { - - switch s.Value { - case Bytes: - return "nil" - case String: - return "\"\"" - case Complex64, Complex128: - return "complex(0,0)" - case Float32, - Float64, - Uint, - Uint8, - Uint16, - Uint32, - Uint64, - Byte, - Int, - Int8, - Int16, - Int32, - Int64, - Duration: - return "0" - case Bool: - return "false" - - case Time: - return "(time.Time{})" - } - - return "" -} - -// IfZeroExpr returns the expression to compare to zero/empty. -func (s *BaseElem) IfZeroExpr() string { - // Byte slices are special: we treat both nil and empty as - // zero for encoding purposes. - if s.Value == Bytes { - return "len(" + s.Varname() + ") == 0" - } - - z := s.ZeroExpr() - if z == "" { - // Assume this is an identifier from another package, - // and that it has generated code for MsgIsZero. - return s.Varname() + ".MsgIsZero()" - } - return s.Varname() + " == " + z -} - -// Comparable returns whether this elem's type is comparable. -func (s *BaseElem) Comparable() bool { - switch s.Value { - case String, Float32, Float64, Complex64, Complex128, - Uint, Uint8, Uint16, Uint32, Uint64, Byte, - Int, Int8, Int16, Int32, Int64, Bool, Time: - return true - default: - return false - } -} - -// SortInterface returns a sort.Interface for sorting a slice of this type. -func (s *BaseElem) SortInterface() string { - sortIntf, ok := sortInterface[s.TypeName()] - if ok { - return sortIntf - } - return "" -} - -func (k Primitive) String() string { - switch k { - case String: - return "String" - case Bytes: - return "Bytes" - case Float32: - return "Float32" - case Float64: - return "Float64" - case Complex64: - return "Complex64" - case Complex128: - return "Complex128" - case Uint: - return "Uint" - case Uint8: - return "Uint8" - case Uint16: - return "Uint16" - case Uint32: - return "Uint32" - case Uint64: - return "Uint64" - case Byte: - return "Byte" - case Int: - return "Int" - case Int8: - return "Int8" - case Int16: - return "Int16" - case Int32: - return "Int32" - case Int64: - return "Int64" - case Bool: - return "Bool" - case Intf: - return "Intf" - case Time: - return "time.Time" - case Duration: - return "time.Duration" - case Ext: - return "Extension" - case IDENT: - return "Ident" - default: - return "INVALID" - } -} - -// writeStructFields is a trampoline for writeBase for -// all of the fields in a struct -func writeStructFields(s []StructField, name string) { - for i := range s { - var path string - for _, pathelem := range s[i].FieldPath { - path += "." + pathelem - } - s[i].FieldElem.SetVarname(fmt.Sprintf("%s%s.%s", name, path, s[i].FieldName)) - } -} - -// SetSortInterface registers sort.Interface types from -// the msgp:sort directive. It would have been nice to -// register it inside the Elem, but unfortunately that -// only affects the type definition; call sites that -// refer to that type (e.g., map keys) have a different -// Elem that does not inherit (get copied) from the type -// definition in f.Identities. -var sortInterface map[string]string - -func SetSortInterface(sorttype string, sortintf string) { - if sortInterface == nil { - sortInterface = make(map[string]string) - } - - sortInterface[sorttype] = sortintf -} diff --git a/msgp/gen/iszero.go b/msgp/gen/iszero.go deleted file mode 100644 index 2f7ed2c306..0000000000 --- a/msgp/gen/iszero.go +++ /dev/null @@ -1,68 +0,0 @@ -package gen - -import ( - "io" -) - -func isZeros(w io.Writer, topics *Topics) *isZeroGen { - return &isZeroGen{ - p: printer{w: w}, - topics: topics, - } -} - -type isZeroGen struct { - passes - p printer - ctx *Context - topics *Topics -} - -func (s *isZeroGen) Method() Method { return IsZero } - -func (s *isZeroGen) Apply(dirs []string) error { - return nil -} - -func (s *isZeroGen) Execute(p Elem) ([]string, error) { - if !s.p.ok() { - return nil, s.p.err - } - p = s.applyall(p) - if p == nil { - return nil, nil - } - - // We might change p.Varname in methodReceiver(); make a copy - // to not affect other code that will use p. - p = p.Copy() - - s.ctx = &Context{} - s.ctx.PushString(p.TypeName()) - - s.p.comment("MsgIsZero returns whether this is a zero value") - - if IsDangling(p) { - baseType := p.(*BaseElem).IdentName - ptrName := p.Varname() - receiver := methodReceiver(p) - s.p.printf("\nfunc (%s %s) MsgIsZero() bool {", ptrName, receiver) - s.p.printf("\n return ((*(%s))(%s)).MsgIsZero()", baseType, ptrName) - s.p.printf("\n}") - s.topics.Add(receiver, "MsgIsZero") - return nil, s.p.err - } - - ptrName := p.Varname() - receiver := imutMethodReceiver(p) - s.p.printf("\nfunc (%s %s) MsgIsZero() bool {", ptrName, receiver) - ize := p.IfZeroExpr() - if ize == "" { - ize = "true" - } - s.p.printf("\nreturn %s", ize) - s.p.printf("\n}") - - s.topics.Add(receiver, "MsgIsZero") - return nil, s.p.err -} diff --git a/msgp/gen/marshal.go b/msgp/gen/marshal.go deleted file mode 100644 index 56340d66fa..0000000000 --- a/msgp/gen/marshal.go +++ /dev/null @@ -1,393 +0,0 @@ -package gen - -import ( - "fmt" - "go/ast" - "io" - "sort" - "strings" - - "github.com/algorand/msgp/msgp" -) - -func marshal(w io.Writer, topics *Topics) *marshalGen { - return &marshalGen{ - p: printer{w: w}, - topics: topics, - } -} - -type marshalGen struct { - passes - p printer - fuse []byte - ctx *Context - msgs []string - topics *Topics -} - -func (m *marshalGen) Method() Method { return Marshal } - -func (m *marshalGen) Apply(dirs []string) error { - return nil -} - -func (m *marshalGen) Execute(p Elem) ([]string, error) { - m.msgs = nil - if !m.p.ok() { - return m.msgs, m.p.err - } - p = m.applyall(p) - if p == nil { - return m.msgs, nil - } - - // We might change p.Varname in methodReceiver(); make a copy - // to not affect other code that will use p. - p = p.Copy() - - m.ctx = &Context{} - - m.p.comment("MarshalMsg implements msgp.Marshaler") - - if IsDangling(p) { - baseType := p.(*BaseElem).IdentName - c := p.Varname() - methodRecv := methodReceiver(p) - m.p.printf("\nfunc (%s %s) MarshalMsg(b []byte) []byte {", c, methodRecv) - m.p.printf("\n return ((*(%s))(%s)).MarshalMsg(b)", baseType, c) - m.p.printf("\n}") - - m.p.printf("\nfunc (_ %[2]s) CanMarshalMsg(%[1]s interface{}) bool {", c, methodRecv) - m.p.printf("\n _, ok := (%s).(%s)", c, methodRecv) - m.p.printf("\n return ok") - m.p.printf("\n}") - - m.topics.Add(methodRecv, "MarshalMsg") - m.topics.Add(methodRecv, "CanMarshalMsg") - - return m.msgs, m.p.err - } - - // save the vname before - // calling methodReceiver so - // that z.Msgsize() is printed correctly - c := p.Varname() - methodRecv := imutMethodReceiver(p) - - m.p.printf("\nfunc (%s %s) MarshalMsg(b []byte) (o []byte) {", c, methodRecv) - m.p.printf("\no = msgp.Require(b, %s.Msgsize())", c) - next(m, p) - m.p.nakedReturn() - - m.p.printf("\nfunc (_ %[2]s) CanMarshalMsg(%[1]s interface{}) bool {", c, methodRecv) - m.p.printf("\n _, ok := (%s).(%s)", c, methodRecv) - - // If this is a value receiver, check for a pointer type too - if methodRecv == p.TypeName() { - m.p.printf("\n if !ok {") - m.p.printf("\n _, ok = (%s).(*%s)", c, methodRecv) - m.p.printf("\n }") - } - - m.p.printf("\n return ok") - m.p.printf("\n}") - - m.topics.Add(methodRecv, "MarshalMsg") - m.topics.Add(methodRecv, "CanMarshalMsg") - - return m.msgs, m.p.err -} - -func (m *marshalGen) rawAppend(typ string, argfmt string, arg interface{}) { - m.p.printf("\no = msgp.Append%s(o, %s)", typ, fmt.Sprintf(argfmt, arg)) -} - -func (m *marshalGen) fuseHook() { - if len(m.fuse) > 0 { - m.rawbytes(m.fuse) - m.fuse = m.fuse[:0] - } -} - -func (m *marshalGen) Fuse(b []byte) { - if len(m.fuse) == 0 { - m.fuse = b - } else { - m.fuse = append(m.fuse, b...) - } -} - -func (m *marshalGen) gStruct(s *Struct) { - if !m.p.ok() { - return - } - - if s.AsTuple { - m.tuple(s) - } else { - m.mapstruct(s) - } - return -} - -func (m *marshalGen) tuple(s *Struct) { - data := make([]byte, 0, 5) - data = msgp.AppendArrayHeader(data, uint32(len(s.Fields))) - m.p.printf("\n// array header, size %d", len(s.Fields)) - m.Fuse(data) - if len(s.Fields) == 0 { - m.fuseHook() - } - for i := range s.Fields { - if !m.p.ok() { - return - } - m.ctx.PushString(s.Fields[i].FieldName) - next(m, s.Fields[i].FieldElem) - m.ctx.Pop() - } -} - -func isFieldOmitEmpty(sf StructField, s *Struct) bool { - tagName := "omitempty" - - // go-codec distinguished between omitempty and omitemptyarray - e := sf.FieldElem - _, isArray := e.(*Array) - if isArray { - tagName = "omitemptyarray" - } - - return sf.HasTagPart(tagName) || s.UnderscoreStructHasTagPart(tagName) -} - -func (m *marshalGen) mapstruct(s *Struct) { - - // Every struct must have a _struct annotation with a codec: tag. - // In the common case, the tag would contain omitempty, but it could - // also be blank, if for some reason omitempty is not desired. This - // check guards against developers forgetting to specify omitempty. - if !s.HasUnderscoreStructTag() { - m.msgs = append(m.msgs, fmt.Sprintf("Missing _struct annotation on struct %v", s)) - return - } - - sortedFields := append([]StructField(nil), s.Fields...) - sort.Sort(byFieldTag(sortedFields)) - - oeIdentPrefix := randIdent() - - var data []byte - nfields := len(sortedFields) - bm := bmask{ - bitlen: nfields, - varname: oeIdentPrefix + "Mask", - } - - exportedFields := 0 - for _, sf := range sortedFields { - if !ast.IsExported(sf.FieldName) { - continue - } - exportedFields++ - } - - omitempty := s.AnyHasTagPart("omitempty") - var fieldNVar string - needCloseBrace := false - needBmDecl := true - if omitempty { - - fieldNVar = oeIdentPrefix + "Len" - - m.p.printf("\n// omitempty: check for empty values") - m.p.printf("\n%s := uint32(%d)", fieldNVar, exportedFields) - for i, sf := range sortedFields { - if !m.p.ok() { - return - } - - if !ast.IsExported(sf.FieldName) { - continue - } - - fieldOmitEmpty := isFieldOmitEmpty(sf, s) - if ize := sf.FieldElem.IfZeroExpr(); ize != "" && fieldOmitEmpty { - if needBmDecl { - m.p.printf("\n%s", bm.typeDecl()) - needBmDecl = false - } - - m.p.printf("\nif %s {", ize) - m.p.printf("\n%s--", fieldNVar) - m.p.printf("\n%s", bm.setStmt(i)) - m.p.printf("\n}") - } - } - - m.p.printf("\n// variable map header, size %s", fieldNVar) - m.p.varAppendMapHeader("o", fieldNVar, exportedFields) - if !m.p.ok() { - return - } - - // quick check for the case where the entire thing is empty, but only at the top level - if !strings.Contains(s.Varname(), ".") { - m.p.printf("\nif %s != 0 {", fieldNVar) - needCloseBrace = true - } - - } else { - - // non-omitempty version - data = make([]byte, 0, 64) - data = msgp.AppendMapHeader(data, uint32(exportedFields)) - m.p.printf("\n// map header, size %d", exportedFields) - m.Fuse(data) - if exportedFields == 0 { - m.fuseHook() - } - - } - - for i, sf := range sortedFields { - if !ast.IsExported(sf.FieldName) { - continue - } - - if !m.p.ok() { - return - } - - fieldOmitEmpty := isFieldOmitEmpty(sf, s) - - // if field is omitempty, wrap with if statement based on the emptymask - oeField := fieldOmitEmpty && sf.FieldElem.IfZeroExpr() != "" - if oeField { - m.p.printf("\nif %s == 0 { // if not empty", bm.readExpr(i)) - } - - data = msgp.AppendString(nil, sf.FieldTag) - - m.p.printf("\n// string %q", sf.FieldTag) - m.Fuse(data) - m.fuseHook() - - m.ctx.PushString(sf.FieldName) - next(m, sf.FieldElem) - m.ctx.Pop() - - if oeField { - m.p.printf("\n}") // close if statement - } - - } - - if needCloseBrace { - m.p.printf("\n}") - } -} - -// append raw data -func (m *marshalGen) rawbytes(bts []byte) { - m.p.print("\no = append(o, ") - for _, b := range bts { - m.p.printf("0x%x,", b) - } - m.p.print(")") -} - -func (m *marshalGen) gMap(s *Map) { - if !m.p.ok() { - return - } - m.fuseHook() - vname := s.Varname() - m.p.printf("\nif %s == nil {", vname) - m.p.printf("\n o = msgp.AppendNil(o)") - m.p.printf("\n} else {") - m.rawAppend(mapHeader, lenAsUint32, vname) - m.p.printf("\n}") - - m.p.printf("\n%s_keys := make([]%s, 0, len(%s))", s.Keyidx, s.Key.TypeName(), vname) - m.p.printf("\nfor %s := range %s {", s.Keyidx, vname) - m.p.printf("\n%s_keys = append(%s_keys, %s)", s.Keyidx, s.Keyidx, s.Keyidx) - m.p.closeblock() - - m.p.printf("\nsort.Sort(%s(%s_keys))", s.Key.SortInterface(), s.Keyidx) - - m.p.printf("\nfor _, %s := range %s_keys {", s.Keyidx, s.Keyidx) - m.p.printf("\n%s := %s[%s]", s.Validx, vname, s.Keyidx) - m.p.printf("\n_ = %s", s.Validx) // we may not use the value, if it's a struct{} - m.ctx.PushVar(s.Keyidx) - next(m, s.Key) - next(m, s.Value) - m.ctx.Pop() - m.p.closeblock() -} - -func (m *marshalGen) gSlice(s *Slice) { - if !m.p.ok() { - return - } - m.fuseHook() - vname := s.Varname() - m.p.printf("\nif %s == nil {", vname) - m.p.printf("\n o = msgp.AppendNil(o)") - m.p.printf("\n} else {") - m.rawAppend(arrayHeader, lenAsUint32, vname) - m.p.printf("\n}") - m.p.rangeBlock(m.ctx, s.Index, vname, m, s.Els) -} - -func (m *marshalGen) gArray(a *Array) { - if !m.p.ok() { - return - } - m.fuseHook() - if be, ok := a.Els.(*BaseElem); ok && be.Value == Byte { - m.rawAppend("Bytes", "(%s)[:]", a.Varname()) - return - } - - m.rawAppend(arrayHeader, literalFmt, a.Size) - m.p.rangeBlock(m.ctx, a.Index, a.Varname(), m, a.Els) -} - -func (m *marshalGen) gPtr(p *Ptr) { - if !m.p.ok() { - return - } - m.fuseHook() - m.p.printf("\nif %s == nil {\no = msgp.AppendNil(o)\n} else {", p.Varname()) - next(m, p.Value) - m.p.closeblock() -} - -func (m *marshalGen) gBase(b *BaseElem) { - if !m.p.ok() { - return - } - m.fuseHook() - vname := b.Varname() - - if b.Convert { - if b.ShimMode == Cast { - vname = tobaseConvert(b) - } else { - vname = randIdent() - m.p.printf("\nvar %s %s", vname, b.BaseType()) - m.p.printf("\n%s = %s", vname, tobaseConvert(b)) - } - } - - switch b.Value { - case IDENT: - m.p.printf("\no = %s.MarshalMsg(o)", vname) - case Intf, Ext: - m.p.printf("\no = msgp.Append%s(o, %s)", b.BaseName(), vname) - default: - m.rawAppend(b.BaseName(), literalFmt, vname) - } -} diff --git a/msgp/gen/maxsize.go b/msgp/gen/maxsize.go deleted file mode 100644 index 50522492cc..0000000000 --- a/msgp/gen/maxsize.go +++ /dev/null @@ -1,379 +0,0 @@ -package gen - -import ( - "bytes" - "fmt" - "go/ast" - "io" - "reflect" - "strconv" - "strings" - - "github.com/algorand/msgp/msgp" -) - -type maxSizeState uint8 - -const ( - // need to write "s = ..." - assignM maxSizeState = iota - - // need to write "s += ..." - addM - - // can just append "+ ..." - exprM - - multM - // the result is multiplied by whatever is preceeding it -) - -func maxSizes(w io.Writer, topics *Topics) *maxSizeGen { - return &maxSizeGen{ - p: printer{w: w}, - state: assignM, - topics: topics, - } -} - -type maxSizeGen struct { - passes - p printer - state maxSizeState - ctx *Context - topics *Topics -} - -func (s *maxSizeGen) Method() Method { return MaxSize } - -func (s *maxSizeGen) Apply(dirs []string) error { - return nil -} - -// this lets us chain together addition -// operations where possible -func (s *maxSizeGen) addConstant(sz string) { - if !s.p.ok() { - return - } - - switch s.state { - case assignM: - s.p.print("\ns = " + sz) - s.state = exprM - return - case addM: - s.p.print("\ns += " + sz) - s.state = exprM - return - case exprM: - s.p.print(" + " + sz) - return - case multM: - s.p.print(" * ( " + sz + ")") - s.state = addM - return - } - - panic("unknown size state") -} - -func (s *maxSizeGen) Execute(p Elem) ([]string, error) { - if !s.p.ok() { - return nil, s.p.err - } - p = s.applyall(p) - if p == nil { - return nil, nil - } - - // We might change p.Varname in methodReceiver(); make a copy - // to not affect other code that will use p. - p = p.Copy() - - s.p.comment("MaxSize returns a maximum valid message size for this message type") - - if IsDangling(p) { - baseType := p.(*BaseElem).IdentName - s.p.printf("\nfunc %s int{", getMaxSizeMethod(p.TypeName())) - s.p.printf("\n return %s", getMaxSizeMethod(baseType)) - s.p.printf("\n}") - s.topics.Add(baseType, getMaxSizeMethod(baseType)) - return nil, s.p.err - } - - s.ctx = &Context{} - s.ctx.PushString(p.TypeName()) - - // receiver := imutMethodReceiver(p) - s.p.printf("\nfunc %s (s int) {", getMaxSizeMethod(p.TypeName())) - s.state = assignM - next(s, p) - s.p.nakedReturn() - s.topics.Add(p.TypeName(), getMaxSizeMethod(p.TypeName())) - return nil, s.p.err -} - -func (s *maxSizeGen) gStruct(st *Struct) { - if !s.p.ok() { - return - } - - nfields := uint32(0) - for i := range st.Fields { - if ast.IsExported(st.Fields[i].FieldName) { - nfields += 1 - } - } - - if st.AsTuple { - data := msgp.AppendArrayHeader(nil, nfields) - s.addConstant(strconv.Itoa(len(data))) - for i := range st.Fields { - if !ast.IsExported(st.Fields[i].FieldName) { - continue - } - - if !s.p.ok() { - return - } - next(s, st.Fields[i].FieldElem) - } - } else { - data := msgp.AppendMapHeader(nil, nfields) - s.addConstant(strconv.Itoa(len(data))) - for i := range st.Fields { - if !ast.IsExported(st.Fields[i].FieldName) { - continue - } - - data = data[:0] - data = msgp.AppendString(data, st.Fields[i].FieldTag) - s.addConstant(strconv.Itoa(len(data))) - next(s, st.Fields[i].FieldElem) - } - } -} - -func (s *maxSizeGen) gPtr(p *Ptr) { - s.state = addM // inner must use add - next(s, p.Value) - s.state = addM // closing block; reset to add -} - -func (s *maxSizeGen) gSlice(sl *Slice) { - if !s.p.ok() { - return - } - s.state = addM - s.p.comment("Calculating size of slice: " + sl.Varname()) - if (sl.AllocBound() == "" || sl.AllocBound() == "-") && (sl.MaxTotalBytes() == "" || sl.MaxTotalBytes() == "-") { - s.p.printf("\npanic(\"Slice %s is unbounded\")", sl.Varname()) - s.state = addM // reset the add to prevent further + expressions from being added to the end the panic statement - return - } - - s.addConstant(builtinSize(arrayHeader)) - - // use maxtotalbytes if it's available - if sl.common.MaxTotalBytes() != "" && sl.common.MaxTotalBytes() != "-" { - s.addConstant(sl.common.MaxTotalBytes()) - return - } - - topLevelAllocBound := sl.AllocBound() - childElement := sl.Els - if sl.Els.AllocBound() == "" && len(strings.Split(sl.AllocBound(), ",")) > 1 { - splitIndex := strings.Index(sl.AllocBound(), ",") - childElement = sl.Els.Copy() - childElement.SetAllocBound(sl.AllocBound()[splitIndex+1:]) - topLevelAllocBound = sl.AllocBound()[:splitIndex] - } - - if str, err := maxSizeExpr(childElement); err == nil { - s.addConstant(fmt.Sprintf("((%s) * (%s))", topLevelAllocBound, str)) - } else { - s.p.printf("\npanic(\"Unable to determine max size: %s\")", err) - } - s.state = addM - return -} - -func (s *maxSizeGen) gArray(a *Array) { - if !s.p.ok() { - return - } - // If this is not the first line where we define s = ... then we need to reset the state - // to addM so that the comment is printed correctly on a newline - if s.state != assignM { - s.state = addM - } - s.p.comment("Calculating size of array: " + a.Varname()) - - s.addConstant(builtinSize(arrayHeader)) - - if str, err := maxSizeExpr(a.Els); err == nil { - s.addConstant(fmt.Sprintf("((%s) * (%s))", a.Size, str)) - } else { - s.p.printf("\npanic(\"Unable to determine max size: %s\")", err) - - } - s.state = addM - return -} - -func (s *maxSizeGen) gMap(m *Map) { - vn := m.Varname() - s.state = addM - s.addConstant(builtinSize(mapHeader)) - topLevelAllocBound := m.AllocBound() - if topLevelAllocBound != "" && topLevelAllocBound == "-" { - s.p.printf("\npanic(\"Map %s is unbounded\")", m.Varname()) - s.state = addM // reset the add to prevent further + expressions from being added to the end the panic statement - return - } - splitBounds := strings.Split(m.AllocBound(), ",") - if len(splitBounds) > 1 { - topLevelAllocBound = splitBounds[0] - m.Key.SetAllocBound(splitBounds[1]) - if len(splitBounds) > 2 { - m.Value.SetAllocBound(splitBounds[2]) - } - } - - s.p.comment("Adding size of map keys for " + vn) - s.p.printf("\ns += %s", topLevelAllocBound) - s.state = multM - next(s, m.Key) - - s.p.comment("Adding size of map values for " + vn) - s.p.printf("\ns += %s", topLevelAllocBound) - s.state = multM - next(s, m.Value) - - s.state = addM -} - -func (s *maxSizeGen) gBase(b *BaseElem) { - if !s.p.ok() { - return - } - if b.MaxTotalBytes() != "" { - s.p.comment("Using maxtotalbytes for: " + b.Varname()) - s.state = addM - s.addConstant(b.MaxTotalBytes()) - s.state = addM - return - } - if b.Convert && b.ShimMode == Convert { - s.state = addM - vname := randIdent() - s.p.printf("\nvar %s %s", vname, b.BaseType()) - - // ensure we don't get "unused variable" warnings from outer slice iterations - s.p.printf("\n_ = %s", b.Varname()) - - value, err := baseMaxSizeExpr(b.Value, vname, b.BaseName(), b.TypeName(), b.common.AllocBound()) - if err != nil { - s.p.printf("\npanic(\"Unable to determine max size: %s\")", err) - s.state = addM // reset the add to prevent further + expressions from being added to the end the panic statement - return - } - s.p.printf("\ns += %s", value) - s.state = exprM - - } else { - vname := b.Varname() - if b.Convert { - vname = tobaseConvert(b) - } - value, err := baseMaxSizeExpr(b.Value, vname, b.BaseName(), b.TypeName(), b.common.AllocBound()) - if err != nil { - s.p.printf("\npanic(\"Unable to determine max size: %s\")", err) - s.state = addM // reset the add to prevent further + expressions from being added to the end the panic statement - return - } - s.addConstant(value) - } -} - -func baseMaxSizeExpr(value Primitive, vname, basename, typename string, allocbound string) (string, error) { - if typename == "msgp.Raw" { - return "", fmt.Errorf("MaxSize() not implemented for Raw type") - } - switch value { - case Ext: - return "", fmt.Errorf("MaxSize() not implemented for Ext type") - case Intf: - return "", fmt.Errorf("MaxSize() not implemented for Interfaces") - case IDENT: - return getMaxSizeMethod(typename), nil - case Bytes: - if allocbound == "" || allocbound == "-" { - return "", fmt.Errorf("Byteslice type %s is unbounded", vname) - } - return "msgp.BytesPrefixSize + " + allocbound, nil - case String: - if allocbound == "" || allocbound == "-" { - return "", fmt.Errorf("String type %s is unbounded", vname) - } - return "msgp.StringPrefixSize + " + allocbound, nil - default: - return builtinSize(basename), nil - } -} - -// return a fixed-size expression, if possible. -// only possible for *BaseElem, *Array and Struct. -// returns (expr, err) -func maxSizeExpr(e Elem) (string, error) { - switch e := e.(type) { - case *Array: - if str, err := maxSizeExpr(e.Els); err == nil { - return fmt.Sprintf("(%s * (%s))", e.Size, str), nil - } else { - return "", err - } - case *BaseElem: - if fixedSize(e.Value) { - return builtinSize(e.BaseName()), nil - } else if (e.TypeName()) == "msgp.Raw" { - return "", fmt.Errorf("Raw type is unbounded") - } else if (e.Value) == String { - if e.AllocBound() == "" || e.AllocBound() == "-" { - return "", fmt.Errorf("String type is unbounded for %s", e.Varname()) - } - return fmt.Sprintf("(msgp.StringPrefixSize + %s)", e.AllocBound()), nil - } else if (e.Value) == IDENT { - return fmt.Sprintf("(%s)", getMaxSizeMethod(e.TypeName())), nil - } else if (e.Value) == Bytes { - if e.AllocBound() == "" || e.AllocBound() == "-" { - return "", fmt.Errorf("Inner byteslice type is unbounded") - } - return fmt.Sprintf("(msgp.BytesPrefixSize + %s)", e.AllocBound()), nil - } - case *Struct: - return fmt.Sprintf("(%s)", getMaxSizeMethod(e.TypeName())), nil - case *Slice: - if e.AllocBound() == "" || e.AllocBound() == "-" { - return "", fmt.Errorf("Slice %s is unbounded", e.Varname()) - } - if str, err := maxSizeExpr(e.Els); err == nil { - return fmt.Sprintf("(%s * (%s))", e.AllocBound(), str), nil - } else { - return "", err - } - } - return fmt.Sprintf("%s, %s", e.TypeName(), reflect.TypeOf(e)), nil -} - -func getMaxSizeMethod(typeName string) (s string) { - var pos int - dotIndex := strings.Index(typeName, ".") - if dotIndex != -1 { - pos = dotIndex + 1 - } - b := []byte(typeName) - b[pos] = bytes.ToUpper(b)[pos] - return string(b) + "MaxSize()" -} diff --git a/msgp/gen/size.go b/msgp/gen/size.go deleted file mode 100644 index d3cbb3d63d..0000000000 --- a/msgp/gen/size.go +++ /dev/null @@ -1,325 +0,0 @@ -package gen - -import ( - "fmt" - "go/ast" - "io" - "strconv" - - "github.com/algorand/msgp/msgp" -) - -type sizeState uint8 - -const ( - // need to write "s = ..." - assign sizeState = iota - - // need to write "s += ..." - add - - // can just append "+ ..." - expr -) - -func sizes(w io.Writer, topics *Topics) *sizeGen { - return &sizeGen{ - p: printer{w: w}, - state: assign, - topics: topics, - } -} - -type sizeGen struct { - passes - p printer - state sizeState - ctx *Context - topics *Topics -} - -func (s *sizeGen) Method() Method { return Size } - -func (s *sizeGen) Apply(dirs []string) error { - return nil -} - -func builtinSize(typ string) string { - return "msgp." + typ + "Size" -} - -// this lets us chain together addition -// operations where possible -func (s *sizeGen) addConstant(sz string) { - if !s.p.ok() { - return - } - - switch s.state { - case assign: - s.p.print("\ns = " + sz) - s.state = expr - return - case add: - s.p.print("\ns += " + sz) - s.state = expr - return - case expr: - s.p.print(" + " + sz) - return - } - - panic("unknown size state") -} - -func (s *sizeGen) Execute(p Elem) ([]string, error) { - if !s.p.ok() { - return nil, s.p.err - } - p = s.applyall(p) - if p == nil { - return nil, nil - } - - // We might change p.Varname in methodReceiver(); make a copy - // to not affect other code that will use p. - p = p.Copy() - - s.p.comment("Msgsize returns an upper bound estimate of the number of bytes occupied by the serialized message") - - if IsDangling(p) { - baseType := p.(*BaseElem).IdentName - ptrName := p.Varname() - receiver := methodReceiver(p) - s.p.printf("\nfunc (%s %s) Msgsize() int {", ptrName, receiver) - s.p.printf("\n return ((*(%s))(%s)).Msgsize()", baseType, ptrName) - s.p.printf("\n}") - s.topics.Add(receiver, "Msgsize") - return nil, s.p.err - } - - s.ctx = &Context{} - s.ctx.PushString(p.TypeName()) - - ptrName := p.Varname() - receiver := imutMethodReceiver(p) - s.p.printf("\nfunc (%s %s) Msgsize() (s int) {", ptrName, receiver) - s.state = assign - next(s, p) - s.p.nakedReturn() - s.topics.Add(receiver, "Msgsize") - return nil, s.p.err -} - -func (s *sizeGen) gStruct(st *Struct) { - if !s.p.ok() { - return - } - - nfields := uint32(0) - for i := range st.Fields { - if ast.IsExported(st.Fields[i].FieldName) { - nfields += 1 - } - } - - if st.AsTuple { - data := msgp.AppendArrayHeader(nil, nfields) - s.addConstant(strconv.Itoa(len(data))) - for i := range st.Fields { - if !ast.IsExported(st.Fields[i].FieldName) { - continue - } - - if !s.p.ok() { - return - } - next(s, st.Fields[i].FieldElem) - } - } else { - data := msgp.AppendMapHeader(nil, nfields) - s.addConstant(strconv.Itoa(len(data))) - for i := range st.Fields { - if !ast.IsExported(st.Fields[i].FieldName) { - continue - } - - data = data[:0] - data = msgp.AppendString(data, st.Fields[i].FieldTag) - s.addConstant(strconv.Itoa(len(data))) - next(s, st.Fields[i].FieldElem) - } - } -} - -func (s *sizeGen) gPtr(p *Ptr) { - s.state = add // inner must use add - s.p.printf("\nif %s == nil {\ns += msgp.NilSize\n} else {", p.Varname()) - next(s, p.Value) - s.state = add // closing block; reset to add - s.p.closeblock() -} - -func (s *sizeGen) gSlice(sl *Slice) { - if !s.p.ok() { - return - } - - s.addConstant(builtinSize(arrayHeader)) - - // if the slice's element is a fixed size - // (e.g. float64, [32]int, etc.), then - // print the length times the element size directly - if str, ok := fixedsizeExpr(sl.Els); ok { - s.addConstant(fmt.Sprintf("(%s * (%s))", lenExpr(sl), str)) - return - } - - // add inside the range block, and immediately after - s.state = add - s.p.rangeBlock(s.ctx, sl.Index, sl.Varname(), s, sl.Els) - s.state = add -} - -func (s *sizeGen) gArray(a *Array) { - if !s.p.ok() { - return - } - - s.addConstant(builtinSize(arrayHeader)) - - // if the array's children are a fixed - // size, we can compile an expression - // that always represents the array's wire size - if str, ok := fixedsizeExpr(a); ok { - s.addConstant(str) - return - } - - s.state = add - s.p.rangeBlock(s.ctx, a.Index, a.Varname(), s, a.Els) - s.state = add -} - -func (s *sizeGen) gMap(m *Map) { - s.addConstant(builtinSize(mapHeader)) - vn := m.Varname() - s.p.printf("\nif %s != nil {", vn) - s.p.printf("\nfor %s, %s := range %s {", m.Keyidx, m.Validx, vn) - s.p.printf("\n_ = %s", m.Keyidx) // we may not use the key - s.p.printf("\n_ = %s", m.Validx) // we may not use the value - s.p.printf("\ns += 0") - s.state = expr - s.ctx.PushVar(m.Keyidx) - next(s, m.Key) - next(s, m.Value) - s.ctx.Pop() - s.p.closeblock() - s.p.closeblock() - s.state = add -} - -func (s *sizeGen) gBase(b *BaseElem) { - if !s.p.ok() { - return - } - if b.Convert && b.ShimMode == Convert { - s.state = add - vname := randIdent() - s.p.printf("\nvar %s %s", vname, b.BaseType()) - - // ensure we don't get "unused variable" warnings from outer slice iterations - s.p.printf("\n_ = %s", b.Varname()) - - s.p.printf("\ns += %s", basesizeExpr(b.Value, vname, b.BaseName())) - s.state = expr - - } else { - vname := b.Varname() - if b.Convert { - vname = tobaseConvert(b) - } - s.addConstant(basesizeExpr(b.Value, vname, b.BaseName())) - } -} - -// returns "len(slice)" -func lenExpr(sl *Slice) string { - return "len(" + sl.Varname() + ")" -} - -// is a given primitive always the same (max) -// size on the wire? -func fixedSize(p Primitive) bool { - switch p { - case Intf, Ext, IDENT, Bytes, String: - return false - default: - return true - } -} - -// strip reference from string -func stripRef(s string) string { - if s[0] == '&' { - return s[1:] - } - return s -} - -// return a fixed-size expression, if possible. -// only possible for *BaseElem and *Array. -// returns (expr, ok) -func fixedsizeExpr(e Elem) (string, bool) { - switch e := e.(type) { - case *Array: - if str, ok := fixedsizeExpr(e.Els); ok { - return fmt.Sprintf("(%s * (%s))", e.Size, str), true - } - case *BaseElem: - if fixedSize(e.Value) { - return builtinSize(e.BaseName()), true - } - case *Struct: - var str string - for _, f := range e.Fields { - if fs, ok := fixedsizeExpr(f.FieldElem); ok { - if str == "" { - str = fs - } else { - str += "+" + fs - } - } else { - return "", false - } - } - var hdrlen int - mhdr := msgp.AppendMapHeader(nil, uint32(len(e.Fields))) - hdrlen += len(mhdr) - var strbody []byte - for _, f := range e.Fields { - strbody = msgp.AppendString(strbody[:0], f.FieldTag) - hdrlen += len(strbody) - } - return fmt.Sprintf("%d + %s", hdrlen, str), true - } - return "", false -} - -// print size expression of a variable name -func basesizeExpr(value Primitive, vname, basename string) string { - switch value { - case Ext: - return "msgp.ExtensionPrefixSize + " + stripRef(vname) + ".Len()" - case Intf: - return "msgp.GuessSize(" + vname + ")" - case IDENT: - return vname + ".Msgsize()" - case Bytes: - return "msgp.BytesPrefixSize + len(" + vname + ")" - case String: - return "msgp.StringPrefixSize + len(" + vname + ")" - default: - return builtinSize(basename) - } -} diff --git a/msgp/gen/spec.go b/msgp/gen/spec.go deleted file mode 100644 index 5567b72c5c..0000000000 --- a/msgp/gen/spec.go +++ /dev/null @@ -1,554 +0,0 @@ -package gen - -import ( - "bytes" - "fmt" - "io" - "strings" -) - -const ( - lenAsUint32 = "uint32(len(%s))" - literalFmt = "%s" - intFmt = "%d" - quotedFmt = `"%s"` - mapHeader = "MapHeader" - arrayHeader = "ArrayHeader" - mapKey = "MapKeyPtr" - stringTyp = "String" - u32 = "uint32" -) - -// Method is a bitfield representing something that the -// generator knows how to print. -type Method uint8 - -// are the bits in 'f' set in 'm'? -func (m Method) isset(f Method) bool { return (m&f == f) } - -// String implements fmt.Stringer -func (m Method) String() string { - switch m { - case 0, invalidmeth: - return "" - case Marshal: - return "marshal" - case Unmarshal: - return "unmarshal" - case Size: - return "size" - case IsZero: - return "iszero" - case MaxSize: - return "maxsize" - case Test: - return "test" - default: - // return e.g. "marshal+unmarshal+test" - modes := [...]Method{Marshal, Unmarshal, Size, IsZero, MaxSize, Test} - any := false - nm := "" - for _, mm := range modes { - if m.isset(mm) { - if any { - nm += "+" + mm.String() - } else { - nm += mm.String() - any = true - } - } - } - return nm - - } -} - -func strtoMeth(s string) Method { - switch s { - case "marshal": - return Marshal - case "unmarshal": - return Unmarshal - case "size": - return Size - case "iszero": - return IsZero - case "maxsize": - return MaxSize - case "test": - return Test - default: - return 0 - } -} - -const ( - Marshal Method = 1 << iota // msgp.Marshaler - Unmarshal // msgp.Unmarshaler - Size // msgp.Sizer - IsZero // implement MsgIsZero() - Test // generate tests - MaxSize // msgp.MaxSize - invalidmeth // this isn't a method - marshaltest = Marshal | Unmarshal | Test // tests for Marshaler and Unmarshaler -) - -type Printer struct { - gens []generator -} - -func NewPrinter(m Method, topics *Topics, out io.Writer, tests io.Writer) *Printer { - if m.isset(Test) && tests == nil { - panic("cannot print tests with 'nil' tests argument!") - } - gens := make([]generator, 0, 7) - if m.isset(Marshal) { - gens = append(gens, marshal(out, topics)) - } - if m.isset(Unmarshal) { - gens = append(gens, unmarshal(out, topics)) - } - if m.isset(Size) { - gens = append(gens, sizes(out, topics)) - } - if m.isset(IsZero) { - gens = append(gens, isZeros(out, topics)) - } - if m.isset(MaxSize) { - gens = append(gens, maxSizes(out, topics)) - } - if m.isset(marshaltest) { - gens = append(gens, mtest(tests)) - } - if len(gens) == 0 { - panic("NewPrinter called with invalid method flags") - } - return &Printer{gens: gens} -} - -// TransformPass is a pass that transforms individual -// elements. (Note that if the returned is different from -// the argument, it should not point to the same objects.) -type TransformPass func(Elem) Elem - -// IgnoreTypename is a pass that just ignores -// types of a given name. -func IgnoreTypename(name string) TransformPass { - return func(e Elem) Elem { - if e.TypeName() == name { - return nil - } - return e - } -} - -// ApplyDirective applies a directive to a named pass -// and all of its dependents. -func (p *Printer) ApplyDirective(pass Method, t TransformPass) { - for _, g := range p.gens { - if g.Method().isset(pass) { - g.Add(t) - } - } -} - -// Print prints an Elem. -func (p *Printer) Print(e Elem) ([]string, error) { - // If the elem is a struct and has no _struct annotations, skip it. - es, ok := e.(*Struct) - if ok && !es.HasAnyStructTag() { - return nil, nil - } - - var msgs []string - - for _, g := range p.gens { - // Elem.SetVarname() is called before the Print() step in parse.FileSet.PrintTo(). - // Elem.SetVarname() generates identifiers as it walks the Elem. This can cause - // collisions between idents created during SetVarname and idents created during Print, - // hence the separate prefixes. - resetIdent("zb") - m, err := g.Execute(e) - resetIdent("za") - - if err != nil { - return nil, err - } - - msgs = append(msgs, m...) - } - return msgs, nil -} - -type contextItem interface { - Arg() string -} - -type contextString string - -func (c contextString) Arg() string { - return fmt.Sprintf("%q", c) -} - -type contextVar string - -func (c contextVar) Arg() string { - return string(c) -} - -type Context struct { - path []contextItem -} - -func (c *Context) PushString(s string) { - c.path = append(c.path, contextString(s)) -} - -func (c *Context) PushVar(s string) { - c.path = append(c.path, contextVar(s)) -} - -func (c *Context) Pop() { - c.path = c.path[:len(c.path)-1] -} - -func (c *Context) ArgsStr() string { - var out string - for idx, p := range c.path { - if idx > 0 { - out += ", " - } - out += p.Arg() - } - return out -} - -// generator is the interface through -// which code is generated. -type generator interface { - Method() Method - Add(p TransformPass) - Execute(Elem) ([]string, error) // execute writes the method for the provided object. -} - -type passes []TransformPass - -func (p *passes) Add(t TransformPass) { - *p = append(*p, t) -} - -func (p *passes) applyall(e Elem) Elem { - for _, t := range *p { - e = t(e) - if e == nil { - return nil - } - } - return e -} - -type traversal interface { - gMap(*Map) - gSlice(*Slice) - gArray(*Array) - gPtr(*Ptr) - gBase(*BaseElem) - gStruct(*Struct) -} - -// type-switch dispatch to the correct -// method given the type of 'e' -func next(t traversal, e Elem) { - switch e := e.(type) { - case *Map: - t.gMap(e) - case *Struct: - t.gStruct(e) - case *Slice: - t.gSlice(e) - case *Array: - t.gArray(e) - case *Ptr: - t.gPtr(e) - case *BaseElem: - t.gBase(e) - default: - panic("bad element type") - } -} - -// possibly-immutable method receiver -func imutMethodReceiver(p Elem) string { - switch e := p.(type) { - case *Struct: - // TODO(HACK): actually do real math here. - if len(e.Fields) <= 3 { - for i := range e.Fields { - if be, ok := e.Fields[i].FieldElem.(*BaseElem); !ok || (be.Value == IDENT || be.Value == Bytes) { - goto nope - } - } - return p.TypeName() - } - nope: - p.SetVarname("(*" + p.Varname() + ")") - return "*" + p.TypeName() - - // gets dereferenced automatically - case *Array: - p.SetVarname("(*" + p.Varname() + ")") - return "*" + p.TypeName() - - // everything else can be - // by-value. - default: - return p.TypeName() - } -} - -// if necessary, wraps a type -// so that its method receiver -// is of the write type. -func methodReceiver(p Elem) string { - p.SetVarname("(*" + p.Varname() + ")") - return "*" + p.TypeName() -} - -// shared utility for generators -type printer struct { - w io.Writer - err error -} - -// writes "var {{name}} {{typ}};" -func (p *printer) declare(name string, typ string) { - p.printf("\nvar %s %s", name, typ) -} - -// does: -// -// if m == nil { -// m = make(type, size) -// } else if len(m) > 0 { -// for key := range m { delete(m, key) } -// } -// -func (p *printer) resizeMap(size string, isnil string, m *Map, ctx string) []string { - vn := m.Varname() - if !p.ok() { - return nil - } - - allocbound := m.AllocBound() - if allocbound == "" { - return []string{fmt.Sprintf("Missing allocbound on map %v", m)} - } - allocbound = strings.Split(allocbound, ",")[0] - if allocbound != "-" { - p.printf("\nif %s > %s {", size, allocbound) - p.printf("\nerr = msgp.ErrOverflow(uint64(%s), uint64(%s))", size, allocbound) - p.printf("\nerr = msgp.WrapError(err, %s)", ctx) - p.printf("\nreturn") - p.printf("\n}") - } - - // go-codec compat: nil clears map, but if a map already exists - // (e.g., because we are decoding the same key twice), then keep - // the map as-is. - - p.printf("\nif %s {", isnil) - p.printf("\n %s = nil", vn) - p.printf("\n} else if %s == nil {", vn) - p.printf("\n %s = make(%s, %s)", vn, m.TypeName(), size) - p.closeblock() - - return nil -} - -// assign key to value based on varnames -func (p *printer) mapAssign(m *Map) { - if !p.ok() { - return - } - p.printf("\n%s[%s] = %s", m.Varname(), m.Keyidx, m.Validx) -} - -// clear map keys -func (p *printer) clearMap(name string) { - p.printf("\nfor key := range %[1]s { delete(%[1]s, key) }", name) -} - -func (p *printer) wrapErrCheck(ctx string) { - p.print("\nif err != nil {") - p.printf("\nerr = msgp.WrapError(err, %s)", ctx) - p.printf("\nreturn") - p.print("\n}") -} - -func (p *printer) resizeSlice(size string, isnil string, s *Slice, ctx string) []string { - allocbound := s.AllocBound() - if allocbound == "" { - return []string{fmt.Sprintf("Missing allocbound on slice %v", s)} - } - allocbound = strings.Split(allocbound, ",")[0] - if allocbound != "-" { - p.printf("\nif %s > %s {", size, allocbound) - p.printf("\nerr = msgp.ErrOverflow(uint64(%s), uint64(%s))", size, allocbound) - p.printf("\nerr = msgp.WrapError(err, %s)", ctx) - p.printf("\nreturn") - p.printf("\n}") - } - - p.printf("\nif %s {", isnil) - p.printf("\n %s = nil", s.Varname()) - p.printf("\n} else if %[1]s != nil && cap(%[1]s) >= %[2]s {", s.Varname(), size) - p.printf("\n %[1]s = (%[1]s)[:%[2]s]", s.Varname(), size) - p.printf("\n} else {") - p.printf("\n %[1]s = make(%[3]s, %[2]s)", s.Varname(), size, s.TypeName()) - p.printf("\n}") - - return nil -} - -func (p *printer) arrayCheck(want string, got string) { - p.printf("\nif %[1]s != %[2]s { err = msgp.ArrayError{Wanted: %[2]s, Got: %[1]s}; return }", got, want) -} - -func (p *printer) arrayCheckBound(want string, got string) { - p.printf("\nif %[1]s > %[2]s { err = msgp.ArrayError{Wanted: %[2]s, Got: %[1]s}; return }", got, want) -} - -func (p *printer) closeblock() { p.print("\n}") } - -// does: -// -// for idx := range iter { -// {{generate inner}} -// } -// -func (p *printer) rangeBlock(ctx *Context, idx string, iter string, t traversal, inner Elem) { - ctx.PushVar(idx) - p.printf("\n for %s := range %s {", idx, iter) - next(t, inner) - p.closeblock() - ctx.Pop() -} - -func (p *printer) nakedReturn() { - if p.ok() { - p.print("\nreturn\n}\n") - } -} - -func (p *printer) comment(s string) { - p.print("\n// " + s) -} - -func (p *printer) printf(format string, args ...interface{}) { - if p.err == nil { - _, p.err = fmt.Fprintf(p.w, format, args...) - } -} - -func (p *printer) print(format string) { - if p.err == nil { - _, p.err = io.WriteString(p.w, format) - } -} - -func (p *printer) initPtr(pt *Ptr) { - if pt.Needsinit() { - vname := pt.Varname() - p.printf("\nif %s == nil { %s = new(%s); }", vname, vname, pt.Value.TypeName()) - } -} - -func (p *printer) ok() bool { return p.err == nil } - -func tobaseConvert(b *BaseElem) string { - return b.ToBase() + "(" + b.Varname() + ")" -} - -func (p *printer) varWriteMapHeader(receiver string, sizeVarname string, maxSize int) { - if maxSize <= 15 { - p.printf("\nerr = %s.Append(0x80 | uint8(%s))", receiver, sizeVarname) - } else { - p.printf("\nerr = %s.WriteMapHeader(%s)", receiver, sizeVarname) - } -} - -func (p *printer) varAppendMapHeader(sliceVarname string, sizeVarname string, maxSize int) { - if maxSize <= 15 { - p.printf("\n%s = append(%s, 0x80 | uint8(%s))", sliceVarname, sliceVarname, sizeVarname) - } else { - p.printf("\n%s = msgp.AppendMapHeader(%s, %s)", sliceVarname, sliceVarname, sizeVarname) - } -} - -// bmask is a bitmask of a the specified number of bits -type bmask struct { - bitlen int - varname string -} - -// typeDecl returns the variable declaration as a var statement -func (b *bmask) typeDecl() string { - return fmt.Sprintf("var %s %s /* %d bits */", b.varname, b.typeName(), b.bitlen) -} - -// typeName returns the type, e.g. "uint8" or "[2]uint64" -func (b *bmask) typeName() string { - - if b.bitlen <= 8 { - return "uint8" - } - if b.bitlen <= 16 { - return "uint16" - } - if b.bitlen <= 32 { - return "uint32" - } - if b.bitlen <= 64 { - return "uint64" - } - - return fmt.Sprintf("[%d]uint64", (b.bitlen+64-1)/64) -} - -// readExpr returns the expression to read from a position in the bitmask. -// Compare ==0 for false or !=0 for true. -func (b *bmask) readExpr(bitoffset int) string { - - if bitoffset < 0 || bitoffset >= b.bitlen { - panic(fmt.Errorf("bitoffset %d out of range for bitlen %d", bitoffset, b.bitlen)) - } - - var buf bytes.Buffer - buf.Grow(len(b.varname) + 16) - buf.WriteByte('(') - buf.WriteString(b.varname) - if b.bitlen > 64 { - fmt.Fprintf(&buf, "[%d]", (bitoffset / 64)) - } - buf.WriteByte('&') - fmt.Fprintf(&buf, "0x%X", (uint64(1) << (uint64(bitoffset) % 64))) - buf.WriteByte(')') - - return buf.String() - -} - -// setStmt returns the statement to set the specified bit in the bitmask. -func (b *bmask) setStmt(bitoffset int) string { - - var buf bytes.Buffer - buf.Grow(len(b.varname) + 16) - buf.WriteString(b.varname) - if b.bitlen > 64 { - fmt.Fprintf(&buf, "[%d]", (bitoffset / 64)) - } - fmt.Fprintf(&buf, " |= 0x%X", (uint64(1) << (uint64(bitoffset) % 64))) - - return buf.String() - -} diff --git a/msgp/gen/testgen.go b/msgp/gen/testgen.go deleted file mode 100644 index f046702ff6..0000000000 --- a/msgp/gen/testgen.go +++ /dev/null @@ -1,104 +0,0 @@ -package gen - -import ( - "io" - "text/template" -) - -var ( - marshalTestTempl = template.New("MarshalTest") -) - -// TODO(philhofer): -// for simplicity's sake, right now -// we can only generate tests for types -// that can be initialized with the -// "Type{}" syntax. -// we should support all the types. - -func mtest(w io.Writer) *mtestGen { - return &mtestGen{w: w} -} - -type mtestGen struct { - passes - w io.Writer -} - -func (m *mtestGen) Execute(p Elem) ([]string, error) { - p = m.applyall(p) - if p != nil && !IsDangling(p) { - switch p.(type) { - case *Struct, *Array, *Slice, *Map: - return nil, marshalTestTempl.Execute(m.w, p) - } - } - return nil, nil -} - -func (m *mtestGen) Method() Method { return marshaltest } - -func init() { - template.Must(marshalTestTempl.Parse(`func TestMarshalUnmarshal{{.TypeName}}(t *testing.T) { - partitiontest.PartitionTest(t) - v := {{.TypeName}}{} - bts := v.MarshalMsg(nil) - left, err := v.UnmarshalMsg(bts) - if err != nil { - t.Fatal(err) - } - if len(left) > 0 { - t.Errorf("%d bytes left over after UnmarshalMsg(): %q", len(left), left) - } - - left, err = msgp.Skip(bts) - if err != nil { - t.Fatal(err) - } - if len(left) > 0 { - t.Errorf("%d bytes left over after Skip(): %q", len(left), left) - } -} - -func TestRandomizedEncoding{{.TypeName}}(t *testing.T) { - protocol.RunEncodingTest(t, &{{.TypeName}}{}) -} - -func BenchmarkMarshalMsg{{.TypeName}}(b *testing.B) { - v := {{.TypeName}}{} - b.ReportAllocs() - b.ResetTimer() - for i:=0; i %s\n", value)) - } - outbuf.WriteString("//\n") - } - outbuf.WriteString("\n") - return outbuf.Bytes() -} - -func (t *Topics) Add(key, value string) { - if t.structs == nil { - t.structs = make(map[string][]string) - } - if key[0] == '*' { - key = key[1:] - value = "(*) " + value - } - t.structs[key] = append(t.structs[key], value) -} diff --git a/msgp/gen/unmarshal.go b/msgp/gen/unmarshal.go deleted file mode 100644 index 509d9bece9..0000000000 --- a/msgp/gen/unmarshal.go +++ /dev/null @@ -1,358 +0,0 @@ -package gen - -import ( - "go/ast" - "io" - "strconv" - "strings" -) - -func unmarshal(w io.Writer, topics *Topics) *unmarshalGen { - return &unmarshalGen{ - p: printer{w: w}, - topics: topics, - } -} - -type unmarshalGen struct { - passes - p printer - hasfield bool - ctx *Context - msgs []string - topics *Topics -} - -func (u *unmarshalGen) Method() Method { return Unmarshal } - -func (u *unmarshalGen) needsField() { - if u.hasfield { - return - } - u.p.print("\nvar field []byte; _ = field") - u.hasfield = true -} - -func (u *unmarshalGen) Execute(p Elem) ([]string, error) { - u.msgs = nil - u.hasfield = false - if !u.p.ok() { - return u.msgs, u.p.err - } - p = u.applyall(p) - if p == nil { - return u.msgs, nil - } - - // We might change p.Varname in methodReceiver(); make a copy - // to not affect other code that will use p. - p = p.Copy() - - u.ctx = &Context{} - - u.p.comment("UnmarshalMsg implements msgp.Unmarshaler") - - if IsDangling(p) { - baseType := p.(*BaseElem).IdentName - c := p.Varname() - methodRecv := methodReceiver(p) - u.p.printf("\nfunc (%s %s) UnmarshalMsg(bts []byte) ([]byte, error) {", c, methodRecv) - u.p.printf("\n return ((*(%s))(%s)).UnmarshalMsg(bts)", baseType, c) - u.p.printf("\n}") - - u.p.printf("\nfunc (%s %s) UnmarshalMsgWithState(bts []byte, st msgp.UnmarshalState) ([]byte, error) {", c, methodRecv) - u.p.printf("\n return ((*(%s))(%s)).UnmarshalMsgWithState(bts, st)", baseType, c) - u.p.printf("\n}") - - u.p.printf("\nfunc (_ %[2]s) CanUnmarshalMsg(%[1]s interface{}) bool {", c, methodRecv) - u.p.printf("\n _, ok := (%s).(%s)", c, methodRecv) - u.p.printf("\n return ok") - u.p.printf("\n}") - - u.topics.Add(methodRecv, "UnmarshalMsg") - u.topics.Add(methodRecv, "UnmarshalMsgWithState") - u.topics.Add(methodRecv, "CanUnmarshalMsg") - - return u.msgs, u.p.err - } - - // save the vname before calling methodReceiver - c := p.Varname() - methodRecv := methodReceiver(p) - - u.p.printf("\nfunc (%s %s) UnmarshalMsgWithState(bts []byte, st msgp.UnmarshalState) (o []byte, err error) {", c, methodRecv) - u.p.printf("\n if st.Depth == 0 {") - u.p.printf("\n err = msgp.ErrMaxDepthExceeded{}") - u.p.printf("\n return") - u.p.printf("\n }") - u.p.printf("\n st.Depth--") - next(u, p) - u.p.print("\no = bts") - - // right before the return: attempt to inspect well formed: - for _, callback := range p.GetCallbacks() { - if !callback.IsUnmarshallCallback() { - continue - } - - u.p.printf("\nif err = %s.%s(); err != nil {", c, callback.GetName()) - u.p.printf("\n return") - u.p.printf("\n}") - } - u.p.nakedReturn() - - u.p.printf("\nfunc (%s %s) UnmarshalMsg(bts []byte) (o []byte, err error) {", c, methodRecv) - u.p.printf("\n return %s.UnmarshalMsgWithState(bts, msgp.DefaultUnmarshalState)", c) - u.p.printf("\n}") - - u.p.printf("\nfunc (_ %[2]s) CanUnmarshalMsg(%[1]s interface{}) bool {", c, methodRecv) - u.p.printf("\n _, ok := (%s).(%s)", c, methodRecv) - u.p.printf("\n return ok") - u.p.printf("\n}") - - u.topics.Add(methodRecv, "UnmarshalMsg") - u.topics.Add(methodRecv, "UnmarshalMsgWithState") - u.topics.Add(methodRecv, "CanUnmarshalMsg") - - return u.msgs, u.p.err -} - -// does assignment to the variable "name" with the type "base" -func (u *unmarshalGen) assignAndCheck(name string, isnil string, base string) { - if !u.p.ok() { - return - } - u.p.printf("\n%s, %s, bts, err = msgp.Read%sBytes(bts)", name, isnil, base) - u.p.wrapErrCheck(u.ctx.ArgsStr()) -} - -func (u *unmarshalGen) gStruct(s *Struct) { - if !u.p.ok() { - return - } - if s.AsTuple { - u.tuple(s) - } else { - u.mapstruct(s) - } - return -} - -func (u *unmarshalGen) tuple(s *Struct) { - - // open block - sz := randIdent() - u.p.declare(sz, "int") - u.assignAndCheck(sz, "_", arrayHeader) - u.p.arrayCheck(strconv.Itoa(len(s.Fields)), sz) - for i := range s.Fields { - if !u.p.ok() { - return - } - u.ctx.PushString(s.Fields[i].FieldName) - next(u, s.Fields[i].FieldElem) - u.ctx.Pop() - } -} - -func (u *unmarshalGen) mapstruct(s *Struct) { - u.needsField() - sz := randIdent() - isnil := randIdent() - u.p.declare(sz, "int") - u.p.declare(isnil, "bool") - - // go-codec compat: decode an array as sequential elements from this struct, - // in the order they are defined in the Go type (as opposed to canonical - // order by sorted tag). - u.p.printf("\n%s, %s, bts, err = msgp.Read%sBytes(bts)", sz, isnil, mapHeader) - u.p.printf("\nif _, ok := err.(msgp.TypeError); ok {") - - u.assignAndCheck(sz, isnil, arrayHeader) - - u.ctx.PushString("struct-from-array") - for i := range s.Fields { - if !ast.IsExported(s.Fields[i].FieldName) { - continue - } - - u.p.printf("\nif %s > 0 {", sz) - u.p.printf("\n%s--", sz) - u.ctx.PushString(s.Fields[i].FieldName) - next(u, s.Fields[i].FieldElem) - u.ctx.Pop() - u.p.printf("\n}") - } - - u.p.printf("\nif %s > 0 {", sz) - u.p.printf("\nerr = msgp.ErrTooManyArrayFields(%s)", sz) - u.p.wrapErrCheck(u.ctx.ArgsStr()) - u.p.printf("\n}") - u.ctx.Pop() - - u.p.printf("\n} else {") - u.p.wrapErrCheck(u.ctx.ArgsStr()) - - u.p.printf("\nif %s {", isnil) - u.p.printf("\n %s = %s{}", s.Varname(), s.TypeName()) - u.p.printf("\n}") - - u.p.printf("\nfor %s > 0 {", sz) - u.p.printf("\n%s--; field, bts, err = msgp.ReadMapKeyZC(bts)", sz) - u.p.wrapErrCheck(u.ctx.ArgsStr()) - u.p.print("\nswitch string(field) {") - for i := range s.Fields { - if !ast.IsExported(s.Fields[i].FieldName) { - continue - } - - if !u.p.ok() { - return - } - u.p.printf("\ncase \"%s\":", s.Fields[i].FieldTag) - u.ctx.PushString(s.Fields[i].FieldName) - next(u, s.Fields[i].FieldElem) - u.ctx.Pop() - } - u.p.print("\ndefault:\nerr = msgp.ErrNoField(string(field))") - u.p.wrapErrCheck(u.ctx.ArgsStr()) - u.p.print("\n}") // close switch - u.p.print("\n}") // close for loop - u.p.print("\n}") // close else statement for array decode -} - -func (u *unmarshalGen) gBase(b *BaseElem) { - if !u.p.ok() { - return - } - - refname := b.Varname() // assigned to - lowered := b.Varname() // passed as argument - if b.Convert { - // begin 'tmp' block - refname = randIdent() - lowered = b.ToBase() + "(" + lowered + ")" - u.p.printf("\n{\nvar %s %s", refname, b.BaseType()) - } - - switch b.Value { - case Bytes: - if b.common.AllocBound() != "" { - sz := randIdent() - u.p.printf("\nvar %s int", sz) - u.p.printf("\n%s, err = msgp.ReadBytesBytesHeader(bts)", sz) - u.p.wrapErrCheck(u.ctx.ArgsStr()) - u.p.printf("\nif %s > %s {", sz, b.common.AllocBound()) - u.p.printf("\nerr = msgp.ErrOverflow(uint64(%s), uint64(%s))", sz, b.common.AllocBound()) - u.p.printf("\nreturn") - u.p.printf("\n}") - } - u.p.printf("\n%s, bts, err = msgp.ReadBytesBytes(bts, %s)", refname, lowered) - case Ext: - u.p.printf("\nbts, err = msgp.ReadExtensionBytes(bts, %s)", lowered) - case IDENT: - u.p.printf("\nbts, err = %s.UnmarshalMsgWithState(bts, st)", lowered) - case String: - if b.common.AllocBound() != "" { - sz := randIdent() - u.p.printf("\nvar %s int", sz) - u.p.printf("\n%s, err = msgp.ReadBytesBytesHeader(bts)", sz) - u.p.wrapErrCheck(u.ctx.ArgsStr()) - u.p.printf("\nif %s > %s {", sz, b.common.AllocBound()) - u.p.printf("\nerr = msgp.ErrOverflow(uint64(%s), uint64(%s))", sz, b.common.AllocBound()) - u.p.printf("\nreturn") - u.p.printf("\n}") - } - u.p.printf("\n%s, bts, err = msgp.ReadStringBytes(bts)", refname) - default: - u.p.printf("\n%s, bts, err = msgp.Read%sBytes(bts)", refname, b.BaseName()) - } - u.p.wrapErrCheck(u.ctx.ArgsStr()) - - if b.Convert { - // close 'tmp' block - if b.ShimMode == Cast { - u.p.printf("\n%s = %s(%s)\n", b.Varname(), b.FromBase(), refname) - } else { - u.p.printf("\n%s, err = %s(%s)", b.Varname(), b.FromBase(), refname) - u.p.wrapErrCheck(u.ctx.ArgsStr()) - } - u.p.printf("}") - } -} - -func (u *unmarshalGen) gArray(a *Array) { - if !u.p.ok() { - return - } - - // special case for [const]byte objects - // see decode.go for symmetry - if be, ok := a.Els.(*BaseElem); ok && be.Value == Byte { - u.p.printf("\nbts, err = msgp.ReadExactBytes(bts, (%s)[:])", a.Varname()) - u.p.wrapErrCheck(u.ctx.ArgsStr()) - return - } - - sz := randIdent() - u.p.declare(sz, "int") - u.assignAndCheck(sz, "_", arrayHeader) - u.p.arrayCheckBound(a.Size, sz) - - u.ctx.PushVar(a.Index) - u.p.printf("\nfor %[1]s := 0; %[1]s < %[2]s; %[1]s++ {", a.Index, sz) - next(u, a.Els) - u.p.closeblock() - u.ctx.Pop() -} - -func (u *unmarshalGen) gSlice(s *Slice) { - if !u.p.ok() { - return - } - sz := randIdent() - isnil := randIdent() - u.p.declare(sz, "int") - u.p.declare(isnil, "bool") - u.assignAndCheck(sz, isnil, arrayHeader) - resizemsgs := u.p.resizeSlice(sz, isnil, s, u.ctx.ArgsStr()) - u.msgs = append(u.msgs, resizemsgs...) - childElement := s.Els - if s.Els.AllocBound() == "" && len(strings.Split(s.AllocBound(), ",")) > 1 { - childElement = s.Els.Copy() - childElement.SetAllocBound(s.AllocBound()[strings.Index(s.AllocBound(), ",")+1:]) - } - u.p.rangeBlock(u.ctx, s.Index, s.Varname(), u, childElement) -} - -func (u *unmarshalGen) gMap(m *Map) { - if !u.p.ok() { - return - } - sz := randIdent() - isnil := randIdent() - u.p.declare(sz, "int") - u.p.declare(isnil, "bool") - u.assignAndCheck(sz, isnil, mapHeader) - - // allocate or clear map - resizemsgs := u.p.resizeMap(sz, isnil, m, u.ctx.ArgsStr()) - u.msgs = append(u.msgs, resizemsgs...) - - // loop and get key,value - u.p.printf("\nfor %s > 0 {", sz) - u.p.printf("\nvar %s %s; var %s %s; %s--", m.Keyidx, m.Key.TypeName(), m.Validx, m.Value.TypeName(), sz) - next(u, m.Key) - u.ctx.PushVar(m.Keyidx) - next(u, m.Value) - u.ctx.Pop() - u.p.mapAssign(m) - u.p.closeblock() -} - -func (u *unmarshalGen) gPtr(p *Ptr) { - u.p.printf("\nif msgp.IsNil(bts) { bts, err = msgp.ReadNilBytes(bts); if err != nil { return }; %s = nil; } else { ", p.Varname()) - u.p.initPtr(p) - next(u, p.Value) - u.p.closeblock() -} diff --git a/msgp/go.mod b/msgp/go.mod deleted file mode 100644 index c5f3bfa33b..0000000000 --- a/msgp/go.mod +++ /dev/null @@ -1,19 +0,0 @@ -module github.com/algorand/msgp - -go 1.20 - -require ( - github.com/daixiang0/gci v0.3.2 - github.com/ttacon/chalk v0.0.0-20160626202418-22c06c80ed31 - golang.org/x/tools v0.1.5 -) - -require ( - github.com/hexops/gotextdiff v1.0.3 // indirect - github.com/kr/text v0.2.0 // indirect - golang.org/x/mod v0.5.0 // indirect - golang.org/x/sync v0.0.0-20210220032951-036812b2e83c // indirect - golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e // indirect - golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 // indirect - gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b // indirect -) diff --git a/msgp/go.sum b/msgp/go.sum deleted file mode 100644 index 6b653c5230..0000000000 --- a/msgp/go.sum +++ /dev/null @@ -1,27 +0,0 @@ -github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= -github.com/daixiang0/gci v0.3.2 h1:MDBsgEJGSJ++/N6Je/b2yK5x5WqmKYlzmILLVP41nl8= -github.com/daixiang0/gci v0.3.2/go.mod h1:jaASoJmv/ykO9dAAPy31iJnreV19248qKDdVWf3QgC4= -github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= -github.com/hexops/gotextdiff v1.0.3 h1:gitA9+qJrrTCsiCl7+kh75nPqQt1cx4ZkudSTLoUqJM= -github.com/hexops/gotextdiff v1.0.3/go.mod h1:pSWU5MAI3yDq+fZBTazCSJysOMbxWL1BSow5/V2vxeg= -github.com/kr/pretty v0.2.0 h1:s5hAObm+yFO5uHYt5dYjxi2rXrsnmRpJx4OYvIWUaQs= -github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= -github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= -github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= -github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= -github.com/ttacon/chalk v0.0.0-20160626202418-22c06c80ed31 h1:OXcKh35JaYsGMRzpvFkLv/MEyPuL49CThT1pZ8aSml4= -github.com/ttacon/chalk v0.0.0-20160626202418-22c06c80ed31/go.mod h1:onvgF043R+lC5RZ8IT9rBXDaEDnpnw/Cl+HFiw+v/7Q= -golang.org/x/mod v0.5.0 h1:UG21uOlmZabA4fW5i7ZX6bjw1xELEGg/ZLgZq9auk/Q= -golang.org/x/mod v0.5.0/go.mod h1:5OXOZSfqPIIbmVBIIKWRFfZjPR0E5r58TLhUjH0a2Ro= -golang.org/x/sync v0.0.0-20210220032951-036812b2e83c h1:5KslGYwFpkhGh+Q16bwMP3cOontH8FOep7tGV86Y7SQ= -golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e h1:fLOSk5Q00efkSvAm+4xcoXD+RRmLmmulPn5I3Y9F2EM= -golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/tools v0.1.5 h1:ouewzE6p+/VEB31YYnTbEJdi8pFqKp4P4n85vwo3DHA= -golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= -golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 h1:go1bK/D/BFZV2I8cIQd1NKEZ+0owSTG1fDTci4IqFcE= -golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= -gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= -gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo= -gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b h1:h8qDotaEPuJATrMmW04NCwg7v22aHH28wwpauUhK9Oo= -gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= diff --git a/msgp/issue185_test.go b/msgp/issue185_test.go deleted file mode 100644 index c888b4158a..0000000000 --- a/msgp/issue185_test.go +++ /dev/null @@ -1,312 +0,0 @@ -package main - -import ( - "fmt" - "go/ast" - "go/parser" - "go/token" - "io/ioutil" - "os" - "path/filepath" - "reflect" - "sort" - "testing" - "text/template" - - "github.com/algorand/msgp/gen" -) - -// When stuff's going wrong, you'll be glad this is here! -const debugTemp = false - -// Ensure that consistent identifiers are generated on a per-method basis by msgp. -// -// Also ensure that no duplicate identifiers appear in a method. -// -// structs are currently processed alphabetically by msgp. this test relies on -// that property. -// -func TestIssue185Idents(t *testing.T) { - t.SkipNow() - - var identCases = []struct { - tpl *template.Template - expectedChanged []string - }{ - {tpl: issue185IdentsTpl, expectedChanged: []string{"Test1"}}, - {tpl: issue185ComplexIdentsTpl, expectedChanged: []string{"Test2"}}, - } - - methods := []string{"Msgsize", "MarshalMsg", "UnmarshalMsg"} - - for idx, identCase := range identCases { - // generate the code, extract the generated variable names, mapped to function name - var tplData issue185TplData - varsBefore, err := loadVars(identCase.tpl, tplData) - if err != nil { - t.Fatalf("%d: could not extract before vars: %v", idx, err) - } - - // regenerate the code with extra field(s), extract the generated variable - // names, mapped to function name - tplData.Extra = true - varsAfter, err := loadVars(identCase.tpl, tplData) - if err != nil { - t.Fatalf("%d: could not extract after vars: %v", idx, err) - } - - // ensure that all declared variable names inside each of the methods we - // expect to change have actually changed - for _, stct := range identCase.expectedChanged { - for _, method := range methods { - fn := fmt.Sprintf("%s.%s", stct, method) - - bv, av := varsBefore.Value(fn), varsAfter.Value(fn) - if len(bv) > 0 && len(av) > 0 && reflect.DeepEqual(bv, av) { - t.Fatalf("%d vars identical! expected vars to change for %s", idx, fn) - } - delete(varsBefore, fn) - delete(varsAfter, fn) - } - } - - // all of the remaining keys should not have changed - for bmethod, bvars := range varsBefore { - avars := varsAfter.Value(bmethod) - - if !reflect.DeepEqual(bvars, avars) { - t.Fatalf("%d: vars changed! expected vars identical for %s", idx, bmethod) - } - delete(varsBefore, bmethod) - delete(varsAfter, bmethod) - } - - if len(varsBefore) > 0 || len(varsAfter) > 0 { - t.Fatalf("%d: unexpected methods remaining", idx) - } - } -} - -type issue185TplData struct { - Extra bool -} - -func TestIssue185Overlap(t *testing.T) { - t.SkipNow() - - var overlapCases = []struct { - tpl *template.Template - data issue185TplData - }{ - {tpl: issue185IdentsTpl, data: issue185TplData{Extra: false}}, - {tpl: issue185IdentsTpl, data: issue185TplData{Extra: true}}, - {tpl: issue185ComplexIdentsTpl, data: issue185TplData{Extra: false}}, - {tpl: issue185ComplexIdentsTpl, data: issue185TplData{Extra: true}}, - } - - for idx, o := range overlapCases { - // regenerate the code with extra field(s), extract the generated variable - // names, mapped to function name - mvars, err := loadVars(o.tpl, o.data) - if err != nil { - t.Fatalf("%d: could not extract after vars: %v", idx, err) - } - - identCnt := 0 - for fn, vars := range mvars { - sort.Strings(vars) - - // Loose sanity check to make sure the tests expectations aren't broken. - // If the prefix ever changes, this needs to change. - for _, v := range vars { - if v[0] == 'z' { - identCnt++ - } - } - - for i := 0; i < len(vars)-1; i++ { - if vars[i] == vars[i+1] { - t.Fatalf("%d: duplicate var %s in function %s", idx, vars[i], fn) - } - } - } - - // one last sanity check: if there aren't any vars that start with 'z', - // this test's expectations are unsatisfiable. - if identCnt == 0 { - t.Fatalf("%d: no generated identifiers found", idx) - } - } -} - -func loadVars(tpl *template.Template, tplData interface{}) (vars extractedVars, err error) { - tempDir, err := ioutil.TempDir("", "msgp-") - if err != nil { - err = fmt.Errorf("could not create temp dir: %v", err) - return - } - - if !debugTemp { - defer os.RemoveAll(tempDir) - } else { - fmt.Println(tempDir) - } - tfile := filepath.Join(tempDir, "msg.go") - genFile := newFilename(tfile, "") - - if err = goGenerateTpl(tempDir, tfile, tpl, tplData); err != nil { - err = fmt.Errorf("could not generate code: %v", err) - return - } - - vars, err = extractVars(genFile) - if err != nil { - err = fmt.Errorf("could not extract after vars: %v", err) - return - } - - return -} - -type varVisitor struct { - vars []string - fset *token.FileSet -} - -func (v *varVisitor) Visit(node ast.Node) (w ast.Visitor) { - gen, ok := node.(*ast.GenDecl) - if !ok { - return v - } - for _, spec := range gen.Specs { - if vspec, ok := spec.(*ast.ValueSpec); ok { - for _, n := range vspec.Names { - v.vars = append(v.vars, n.Name) - } - } - } - return v -} - -type extractedVars map[string][]string - -func (e extractedVars) Value(key string) []string { - if v, ok := e[key]; ok { - return v - } - panic(fmt.Errorf("unknown key %s", key)) -} - -func extractVars(file string) (extractedVars, error) { - fset := token.NewFileSet() - - f, err := parser.ParseFile(fset, file, nil, 0) - if err != nil { - return nil, err - } - - vars := make(map[string][]string) - for _, d := range f.Decls { - switch d := d.(type) { - case *ast.FuncDecl: - sn := "" - switch rt := d.Recv.List[0].Type.(type) { - case *ast.Ident: - sn = rt.Name - case *ast.StarExpr: - sn = rt.X.(*ast.Ident).Name - default: - panic("unknown receiver type") - } - - key := fmt.Sprintf("%s.%s", sn, d.Name.Name) - vis := &varVisitor{fset: fset} - ast.Walk(vis, d.Body) - vars[key] = vis.vars - } - } - return vars, nil -} - -func goGenerateTpl(cwd, tfile string, tpl *template.Template, tplData interface{}) error { - outf, err := os.OpenFile(tfile, os.O_CREATE|os.O_RDWR|os.O_TRUNC, 0600) - if err != nil { - return err - } - defer outf.Close() - - if err := tpl.Execute(outf, tplData); err != nil { - return err - } - - mode := gen.Size | gen.Marshal | gen.Unmarshal - - return Run(tfile, mode, false, "") -} - -var issue185IdentsTpl = template.Must(template.New("").Parse(` -package issue185 - -//go:generate msgp - -type Test1 struct { - Foo string - Bar string - {{ if .Extra }}Baz []string{{ end }} - Qux string -} - -type Test2 struct { - Foo string - Bar string - Baz string -} -`)) - -var issue185ComplexIdentsTpl = template.Must(template.New("").Parse(` -package issue185 - -//go:generate msgp - -type Test1 struct { - Foo string - Bar string - Baz string -} - -type Test2 struct { - Foo string - Bar string - Baz []string - Qux map[string]string - Yep map[string]map[string]string - Quack struct { - Quack struct { - Quack struct { - {{ if .Extra }}Extra []string{{ end }} - Quack string - } - } - } - Nup struct { - Foo string - Bar string - Baz []string - Qux map[string]string - Yep map[string]map[string]string - } - Ding struct { - Dong struct { - Dung struct { - Thing string - } - } - } -} - -type Test3 struct { - Foo string - Bar string - Baz string -} -`)) diff --git a/msgp/main.go b/msgp/main.go deleted file mode 100644 index 8a157d28f4..0000000000 --- a/msgp/main.go +++ /dev/null @@ -1,117 +0,0 @@ -// msgp is a code generation tool for -// creating methods to serialize and de-serialize -// Go data structures to and from MessagePack. -// -// This package is targeted at the `go generate` tool. -// To use it, include the following directive in a -// go source file with types requiring source generation: -// -// //go:generate msgp -// -// The go generate tool should set the proper environment variables for -// the generator to execute without any command-line flags. However, the -// following options are supported, if you need them: -// -// -o = output file name (default is {input}_gen.go) -// -file = input file name (or directory; default is $GOFILE, which is set by the `go generate` command) -// -io = satisfy the `msgp.Decodable` and `msgp.Encodable` interfaces (default is true) -// -marshal = satisfy the `msgp.Marshaler` and `msgp.Unmarshaler` interfaces (default is true) -// -tests = generate tests and benchmarks (default is true) -// -// For more information, please read README.md, and the wiki at github.com/tinylib/msgp -// -package main - -import ( - "flag" - "fmt" - "os" - "path/filepath" - "strings" - - "github.com/algorand/msgp/gen" - "github.com/algorand/msgp/parse" - "github.com/algorand/msgp/printer" - "github.com/ttacon/chalk" -) - -var ( - out = flag.String("o", "", "output file") - file = flag.String("file", "", "input file") - marshal = flag.Bool("marshal", true, "create Marshal and Unmarshal methods") - tests = flag.Bool("tests", true, "create tests and benchmarks") - unexported = flag.Bool("unexported", true, "also process unexported types") - skipFormat = flag.Bool("skip-format", false, "skip formatting the generated code (for debug)") - warnPkgMask = flag.String("warnmask", "", "skip generating warnings on datatypes outside given package") -) - -func main() { - flag.Parse() - - // GOFILE is set by go generate - if *file == "" { - *file = os.Getenv("GOFILE") - if *file == "" { - fmt.Println(chalk.Red.Color("No file to parse.")) - os.Exit(1) - } - } - - var mode gen.Method - if *marshal { - mode |= (gen.Marshal | gen.Unmarshal | gen.Size | gen.IsZero | gen.MaxSize) - } - if *tests { - mode |= gen.Test - } - - if mode&^gen.Test == 0 { - fmt.Println(chalk.Red.Color("No methods to generate; -marshal=false")) - os.Exit(1) - } - - if err := Run(*file, mode, *unexported, *warnPkgMask); err != nil { - fmt.Println(chalk.Red.Color(err.Error())) - os.Exit(1) - } -} - -// Run writes all methods using the associated file or path, e.g. -// -// err := msgp.Run("path/to/myfile.go", gen.Size|gen.Marshal|gen.Unmarshal|gen.Test, false) -// -func Run(gofile string, mode gen.Method, unexported bool, warnPkgMask string) error { - if mode&^gen.Test == 0 { - return nil - } - fmt.Println(chalk.Magenta.Color("======== MessagePack Code Generator =======")) - fmt.Printf(chalk.Magenta.Color(">>> Input: \"%s\"\n"), gofile) - fs, err := parse.File(gofile, unexported, warnPkgMask) - if err != nil { - return err - } - - if len(fs.Identities) == 0 { - fmt.Println(chalk.Magenta.Color("No types requiring code generation were found!")) - return nil - } - - return printer.PrintFile(newFilename(gofile, fs.Package), fs, mode, *skipFormat) -} - -// picks a new file name based on input flags and input filename(s). -func newFilename(old string, pkg string) string { - if *out != "" { - if pre := strings.TrimPrefix(*out, old); len(pre) > 0 && - !strings.HasSuffix(*out, ".go") { - return filepath.Join(old, *out) - } - return *out - } - - if fi, err := os.Stat(old); err == nil && fi.IsDir() { - old = filepath.Join(old, pkg) - } - // new file name is old file name + _gen.go - return strings.TrimSuffix(old, ".go") + "_gen.go" -} diff --git a/msgp/msgp/defs.go b/msgp/msgp/defs.go deleted file mode 100644 index b1188b041d..0000000000 --- a/msgp/msgp/defs.go +++ /dev/null @@ -1,142 +0,0 @@ -// This package is the support library for the msgp code generator (http://github.com/algorand/msgp). -// -// This package defines the utilites used by the msgp code generator for encoding and decoding MessagePack -// from []byte and io.Reader/io.Writer types. Much of this package is devoted to helping the msgp code -// generator implement the Marshaler/Unmarshaler and Encodable/Decodable interfaces. -// -// This package defines four "families" of functions: -// - AppendXxxx() appends an object to a []byte in MessagePack encoding. -// - ReadXxxxBytes() reads an object from a []byte and returns the remaining bytes. -// - (*Writer).WriteXxxx() writes an object to the buffered *Writer type. -// - (*Reader).ReadXxxx() reads an object from a buffered *Reader type. -// -// Once a type has satisfied the `Encodable` and `Decodable` interfaces, -// it can be written and read from arbitrary `io.Writer`s and `io.Reader`s using -// msgp.Encode(io.Writer, msgp.Encodable) -// and -// msgp.Decode(io.Reader, msgp.Decodable) -// -// There are also methods for converting MessagePack to JSON without -// an explicit de-serialization step. -// -// For additional tips, tricks, and gotchas, please visit -// the wiki at http://github.com/tinylib/msgp -package msgp - -const last4 = 0x0f -const first4 = 0xf0 -const last5 = 0x1f -const first3 = 0xe0 -const last7 = 0x7f - -func isfixint(b byte) bool { - return b>>7 == 0 -} - -func isnfixint(b byte) bool { - return b&first3 == mnfixint -} - -func isfixmap(b byte) bool { - return b&first4 == mfixmap -} - -func isfixarray(b byte) bool { - return b&first4 == mfixarray -} - -func isfixstr(b byte) bool { - return b&first3 == mfixstr -} - -func wfixint(u uint8) byte { - return u & last7 -} - -func rfixint(b byte) uint8 { - return b -} - -func wnfixint(i int8) byte { - return byte(i) | mnfixint -} - -func rnfixint(b byte) int8 { - return int8(b) -} - -func rfixmap(b byte) uint8 { - return b & last4 -} - -func wfixmap(u uint8) byte { - return mfixmap | (u & last4) -} - -func rfixstr(b byte) uint8 { - return b & last5 -} - -func wfixstr(u uint8) byte { - return (u & last5) | mfixstr -} - -func rfixarray(b byte) uint8 { - return (b & last4) -} - -func wfixarray(u uint8) byte { - return (u & last4) | mfixarray -} - -// These are all the byte -// prefixes defined by the -// msgpack standard -const ( - // 0XXXXXXX - mfixint uint8 = 0x00 - - // 111XXXXX - mnfixint uint8 = 0xe0 - - // 1000XXXX - mfixmap uint8 = 0x80 - - // 1001XXXX - mfixarray uint8 = 0x90 - - // 101XXXXX - mfixstr uint8 = 0xa0 - - mnil uint8 = 0xc0 - mfalse uint8 = 0xc2 - mtrue uint8 = 0xc3 - mbin8 uint8 = 0xc4 - mbin16 uint8 = 0xc5 - mbin32 uint8 = 0xc6 - mext8 uint8 = 0xc7 - mext16 uint8 = 0xc8 - mext32 uint8 = 0xc9 - mfloat32 uint8 = 0xca - mfloat64 uint8 = 0xcb - muint8 uint8 = 0xcc - muint16 uint8 = 0xcd - muint32 uint8 = 0xce - muint64 uint8 = 0xcf - mint8 uint8 = 0xd0 - mint16 uint8 = 0xd1 - mint32 uint8 = 0xd2 - mint64 uint8 = 0xd3 - mfixext1 uint8 = 0xd4 - mfixext2 uint8 = 0xd5 - mfixext4 uint8 = 0xd6 - mfixext8 uint8 = 0xd7 - mfixext16 uint8 = 0xd8 - mstr8 uint8 = 0xd9 - mstr16 uint8 = 0xda - mstr32 uint8 = 0xdb - marray16 uint8 = 0xdc - marray32 uint8 = 0xdd - mmap16 uint8 = 0xde - mmap32 uint8 = 0xdf -) diff --git a/msgp/msgp/defs_test.go b/msgp/msgp/defs_test.go deleted file mode 100644 index 667dfd6012..0000000000 --- a/msgp/msgp/defs_test.go +++ /dev/null @@ -1,12 +0,0 @@ -package msgp_test - -//go:generate msgp -o=defgen_test.go -tests=false - -type Blobs []Blob - -type Blob struct { - Name string `msg:"name"` - Float float64 `msg:"float"` - Bytes []byte `msg:"bytes"` - Amount int64 `msg:"amount"` -} diff --git a/msgp/msgp/elsize.go b/msgp/msgp/elsize.go deleted file mode 100644 index 95762e7eeb..0000000000 --- a/msgp/msgp/elsize.go +++ /dev/null @@ -1,99 +0,0 @@ -package msgp - -// size of every object on the wire, -// plus type information. gives us -// constant-time type information -// for traversing composite objects. -// -var sizes = [256]bytespec{ - mnil: {size: 1, extra: constsize, typ: NilType}, - mfalse: {size: 1, extra: constsize, typ: BoolType}, - mtrue: {size: 1, extra: constsize, typ: BoolType}, - mbin8: {size: 2, extra: extra8, typ: BinType}, - mbin16: {size: 3, extra: extra16, typ: BinType}, - mbin32: {size: 5, extra: extra32, typ: BinType}, - mext8: {size: 3, extra: extra8, typ: ExtensionType}, - mext16: {size: 4, extra: extra16, typ: ExtensionType}, - mext32: {size: 6, extra: extra32, typ: ExtensionType}, - mfloat32: {size: 5, extra: constsize, typ: Float32Type}, - mfloat64: {size: 9, extra: constsize, typ: Float64Type}, - muint8: {size: 2, extra: constsize, typ: UintType}, - muint16: {size: 3, extra: constsize, typ: UintType}, - muint32: {size: 5, extra: constsize, typ: UintType}, - muint64: {size: 9, extra: constsize, typ: UintType}, - mint8: {size: 2, extra: constsize, typ: IntType}, - mint16: {size: 3, extra: constsize, typ: IntType}, - mint32: {size: 5, extra: constsize, typ: IntType}, - mint64: {size: 9, extra: constsize, typ: IntType}, - mfixext1: {size: 3, extra: constsize, typ: ExtensionType}, - mfixext2: {size: 4, extra: constsize, typ: ExtensionType}, - mfixext4: {size: 6, extra: constsize, typ: ExtensionType}, - mfixext8: {size: 10, extra: constsize, typ: ExtensionType}, - mfixext16: {size: 18, extra: constsize, typ: ExtensionType}, - mstr8: {size: 2, extra: extra8, typ: StrType}, - mstr16: {size: 3, extra: extra16, typ: StrType}, - mstr32: {size: 5, extra: extra32, typ: StrType}, - marray16: {size: 3, extra: array16v, typ: ArrayType}, - marray32: {size: 5, extra: array32v, typ: ArrayType}, - mmap16: {size: 3, extra: map16v, typ: MapType}, - mmap32: {size: 5, extra: map32v, typ: MapType}, -} - -func init() { - // set up fixed fields - - // fixint - for i := mfixint; i < 0x80; i++ { - sizes[i] = bytespec{size: 1, extra: constsize, typ: IntType} - } - - // nfixint - for i := uint16(mnfixint); i < 0x100; i++ { - sizes[uint8(i)] = bytespec{size: 1, extra: constsize, typ: IntType} - } - - // fixstr gets constsize, - // since the prefix yields the size - for i := mfixstr; i < 0xc0; i++ { - sizes[i] = bytespec{size: 1 + rfixstr(i), extra: constsize, typ: StrType} - } - - // fixmap - for i := mfixmap; i < 0x90; i++ { - sizes[i] = bytespec{size: 1, extra: varmode(2 * rfixmap(i)), typ: MapType} - } - - // fixarray - for i := mfixarray; i < 0xa0; i++ { - sizes[i] = bytespec{size: 1, extra: varmode(rfixarray(i)), typ: ArrayType} - } -} - -// a valid bytespsec has -// non-zero 'size' and -// non-zero 'typ' -type bytespec struct { - size uint8 // prefix size information - extra varmode // extra size information - typ Type // type - _ byte // makes bytespec 4 bytes (yes, this matters) -} - -// size mode -// if positive, # elements for composites -type varmode int8 - -const ( - constsize varmode = 0 // constant size (size bytes + uint8(varmode) objects) - extra8 = -1 // has uint8(p[1]) extra bytes - extra16 = -2 // has be16(p[1:]) extra bytes - extra32 = -3 // has be32(p[1:]) extra bytes - map16v = -4 // use map16 - map32v = -5 // use map32 - array16v = -6 // use array16 - array32v = -7 // use array32 -) - -func getType(v byte) Type { - return sizes[v].typ -} diff --git a/msgp/msgp/errors.go b/msgp/msgp/errors.go deleted file mode 100644 index fd2618855f..0000000000 --- a/msgp/msgp/errors.go +++ /dev/null @@ -1,354 +0,0 @@ -package msgp - -import ( - "fmt" - "reflect" -) - -const resumableDefault = false - -var ( - // ErrShortBytes is returned when the - // slice being decoded is too short to - // contain the contents of the message - ErrShortBytes error = errShort{} - - // this error is only returned - // if we reach code that should - // be unreachable - fatal error = errFatal{} -) - -type ErrNoField string - -func (e ErrNoField) Error() string { - return fmt.Sprintf("Unknown field: %s", string(e)) -} - -type ErrTooManyArrayFields int - -func (e ErrTooManyArrayFields) Error() string { - return fmt.Sprintf("Too many array fields when decoding into struct: %d left", int(e)) -} - -// Error is the interface satisfied -// by all of the errors that originate -// from this package. -type Error interface { - error - - // Resumable returns whether - // or not the error means that - // the stream of data is malformed - // and the information is unrecoverable. - Resumable() bool -} - -// contextError allows msgp Error instances to be enhanced with additional -// context about their origin. -type contextError interface { - Error - - // withContext must not modify the error instance - it must clone and - // return a new error with the context added. - withContext(ctx string) error -} - -// Cause returns the underlying cause of an error that has been wrapped -// with additional context. -func Cause(e error) error { - out := e - if e, ok := e.(errWrapped); ok && e.cause != nil { - out = e.cause - } - return out -} - -// Resumable returns whether or not the error means that the stream of data is -// malformed and the information is unrecoverable. -func Resumable(e error) bool { - if e, ok := e.(Error); ok { - return e.Resumable() - } - return resumableDefault -} - -// WrapError wraps an error with additional context that allows the part of the -// serialized type that caused the problem to be identified. Underlying errors -// can be retrieved using Cause() -// -// The input error is not modified - a new error should be returned. -// -// ErrShortBytes is not wrapped with any context due to backward compatibility -// issues with the public API. -func WrapError(err error, ctx ...interface{}) error { - switch e := err.(type) { - case errShort, ErrMaxDepthExceeded: - return e - case contextError: - return e.withContext(ctxString(ctx)) - default: - return errWrapped{cause: err, ctx: ctxString(ctx)} - } -} - -// ctxString converts the incoming interface{} slice into a single string. -func ctxString(ctx []interface{}) string { - out := "" - for idx, cv := range ctx { - if idx > 0 { - out += "/" - } - out += fmt.Sprintf("%v", cv) - } - return out -} - -func addCtx(ctx, add string) string { - if ctx != "" { - return add + "/" + ctx - } else { - return add - } -} - -// errWrapped allows arbitrary errors passed to WrapError to be enhanced with -// context and unwrapped with Cause() -type errWrapped struct { - cause error - ctx string -} - -func (e errWrapped) Error() string { - if e.ctx != "" { - return fmt.Sprintf("%s at %s", e.cause, e.ctx) - } else { - return e.cause.Error() - } -} - -func (e errWrapped) Resumable() bool { - if e, ok := e.cause.(Error); ok { - return e.Resumable() - } - return resumableDefault -} - -type errShort struct{} - -func (e errShort) Error() string { return "msgp: too few bytes left to read object" } -func (e errShort) Resumable() bool { return false } - -// errOverflow is returned when the message -// being decoded has some length field that -// exceeds the maximum allowed length. -type errOverflow struct { - l uint64 - bound uint64 -} - -func (e errOverflow) Error() string { - return fmt.Sprintf("msgp: length overflow: %d > %d", e.l, e.bound) -} - -func (e errOverflow) Resumable() bool { - return false -} - -func ErrOverflow(l uint64, bound uint64) error { - return errOverflow{l, bound} -} - -type errFatal struct { - ctx string -} - -func (f errFatal) Error() string { - out := "msgp: fatal decoding error (unreachable code)" - if f.ctx != "" { - out += " at " + f.ctx - } - return out -} - -func (f errFatal) Resumable() bool { return false } - -func (f errFatal) withContext(ctx string) error { f.ctx = addCtx(f.ctx, ctx); return f } - -// ArrayError is an error returned -// when decoding a fix-sized array -// of the wrong size -type ArrayError struct { - Wanted int - Got int - ctx string -} - -// Error implements the error interface -func (a ArrayError) Error() string { - out := fmt.Sprintf("msgp: wanted array of size %d; got %d", a.Wanted, a.Got) - if a.ctx != "" { - out += " at " + a.ctx - } - return out -} - -// Resumable is always 'true' for ArrayErrors -func (a ArrayError) Resumable() bool { return true } - -func (a ArrayError) withContext(ctx string) error { a.ctx = addCtx(a.ctx, ctx); return a } - -// IntOverflow is returned when a call -// would downcast an integer to a type -// with too few bits to hold its value. -type IntOverflow struct { - Value int64 // the value of the integer - FailedBitsize int // the bit size that the int64 could not fit into - ctx string -} - -// Error implements the error interface -func (i IntOverflow) Error() string { - str := fmt.Sprintf("msgp: %d overflows int%d", i.Value, i.FailedBitsize) - if i.ctx != "" { - str += " at " + i.ctx - } - return str -} - -// Resumable is always 'true' for overflows -func (i IntOverflow) Resumable() bool { return true } - -func (i IntOverflow) withContext(ctx string) error { i.ctx = addCtx(i.ctx, ctx); return i } - -// UintOverflow is returned when a call -// would downcast an unsigned integer to a type -// with too few bits to hold its value -type UintOverflow struct { - Value uint64 // value of the uint - FailedBitsize int // the bit size that couldn't fit the value - ctx string -} - -// Error implements the error interface -func (u UintOverflow) Error() string { - str := fmt.Sprintf("msgp: %d overflows uint%d", u.Value, u.FailedBitsize) - if u.ctx != "" { - str += " at " + u.ctx - } - return str -} - -// Resumable is always 'true' for overflows -func (u UintOverflow) Resumable() bool { return true } - -func (u UintOverflow) withContext(ctx string) error { u.ctx = addCtx(u.ctx, ctx); return u } - -// UintBelowZero is returned when a call -// would cast a signed integer below zero -// to an unsigned integer. -type UintBelowZero struct { - Value int64 // value of the incoming int - ctx string -} - -// Error implements the error interface -func (u UintBelowZero) Error() string { - str := fmt.Sprintf("msgp: attempted to cast int %d to unsigned", u.Value) - if u.ctx != "" { - str += " at " + u.ctx - } - return str -} - -// Resumable is always 'true' for overflows -func (u UintBelowZero) Resumable() bool { return true } - -func (u UintBelowZero) withContext(ctx string) error { - u.ctx = ctx - return u -} - -// A TypeError is returned when a particular -// decoding method is unsuitable for decoding -// a particular MessagePack value. -type TypeError struct { - Method Type // Type expected by method - Encoded Type // Type actually encoded - - ctx string -} - -// Error implements the error interface -func (t TypeError) Error() string { - out := fmt.Sprintf("msgp: attempted to decode type %q with method for %q", t.Encoded, t.Method) - if t.ctx != "" { - out += " at " + t.ctx - } - return out -} - -// Resumable returns 'true' for TypeErrors -func (t TypeError) Resumable() bool { return true } - -func (t TypeError) withContext(ctx string) error { t.ctx = addCtx(t.ctx, ctx); return t } - -// returns either InvalidPrefixError or -// TypeError depending on whether or not -// the prefix is recognized -func badPrefix(want Type, lead byte) error { - t := sizes[lead].typ - if t == InvalidType { - return InvalidPrefixError(lead) - } - return TypeError{Method: want, Encoded: t} -} - -// InvalidPrefixError is returned when a bad encoding -// uses a prefix that is not recognized in the MessagePack standard. -// This kind of error is unrecoverable. -type InvalidPrefixError byte - -// Error implements the error interface -func (i InvalidPrefixError) Error() string { - return fmt.Sprintf("msgp: unrecognized type prefix 0x%x", byte(i)) -} - -// Resumable returns 'false' for InvalidPrefixErrors -func (i InvalidPrefixError) Resumable() bool { return false } - -// ErrUnsupportedType is returned -// when a bad argument is supplied -// to a function that takes `interface{}`. -type ErrUnsupportedType struct { - T reflect.Type - - ctx string -} - -// Error implements error -func (e *ErrUnsupportedType) Error() string { - out := fmt.Sprintf("msgp: type %q not supported", e.T) - if e.ctx != "" { - out += " at " + e.ctx - } - return out -} - -// Resumable returns 'true' for ErrUnsupportedType -func (e *ErrUnsupportedType) Resumable() bool { return true } - -func (e *ErrUnsupportedType) withContext(ctx string) error { - o := *e - o.ctx = addCtx(o.ctx, ctx) - return &o -} - -// ErrMaxDepthExceeded is returned if the maximum traversal depth is exceeded. -type ErrMaxDepthExceeded struct{} - -// Error implements error -func (e ErrMaxDepthExceeded) Error() string { return "Max depth exceeded" } - -// Resumable implements Error -func (e ErrMaxDepthExceeded) Resumable() bool { return false } diff --git a/msgp/msgp/errors_test.go b/msgp/msgp/errors_test.go deleted file mode 100644 index 308bec64d7..0000000000 --- a/msgp/msgp/errors_test.go +++ /dev/null @@ -1,89 +0,0 @@ -package msgp - -import ( - "errors" - "fmt" - "strings" - "testing" -) - -func TestWrapVanillaErrorWithNoAdditionalContext(t *testing.T) { - err := errors.New("test") - w := WrapError(err) - if w == err { - t.Fatal() - } - if w.Error() != err.Error() { - t.Fatal() - } - if w.(errWrapped).Resumable() { - t.Fatal() - } -} - -func TestWrapVanillaErrorWithAdditionalContext(t *testing.T) { - err := errors.New("test") - w := WrapError(err, "foo", "bar") - if w == err { - t.Fatal() - } - if w.Error() == err.Error() { - t.Fatal() - } - if w.(Error).Resumable() { - t.Fatal() - } - if !strings.HasPrefix(w.Error(), err.Error()) { - t.Fatal() - } - rest := w.Error()[len(err.Error()):] - if rest != " at foo/bar" { - t.Fatal() - } -} - -func TestWrapResumableError(t *testing.T) { - err := ArrayError{} - w := WrapError(err) - if !w.(Error).Resumable() { - t.Fatal() - } -} - -func TestWrapMultiple(t *testing.T) { - err := &TypeError{} - w := WrapError(WrapError(err, "b"), "a") - expected := `msgp: attempted to decode type "" with method for "" at a/b` - if expected != w.Error() { - t.Fatal() - } -} - -func TestCause(t *testing.T) { - for idx, err := range []error{ - errors.New("test"), - ArrayError{}, - &ErrUnsupportedType{}, - } { - t.Run(fmt.Sprintf("%d", idx), func(t *testing.T) { - cerr := WrapError(err, "test") - if cerr == err { - t.Fatal() - } - if Cause(err) != err { - t.Fatal() - } - }) - } -} - -func TestCauseShortByte(t *testing.T) { - err := ErrShortBytes - cerr := WrapError(err, "test") - if cerr != err { - t.Fatal() - } - if Cause(err) != err { - t.Fatal() - } -} diff --git a/msgp/msgp/extension.go b/msgp/msgp/extension.go deleted file mode 100644 index 4f279baa18..0000000000 --- a/msgp/msgp/extension.go +++ /dev/null @@ -1,287 +0,0 @@ -package msgp - -import ( - "fmt" - "math" -) - -const ( - // Complex64Extension is the extension number used for complex64 - Complex64Extension = 3 - - // Complex128Extension is the extension number used for complex128 - Complex128Extension = 4 - - // TimeExtension is the extension number used for time.Time - TimeExtension = 5 -) - -// our extensions live here -var extensionReg = make(map[int8]func() Extension) - -// RegisterExtension registers extensions so that they -// can be initialized and returned by methods that -// decode `interface{}` values. This should only -// be called during initialization. f() should return -// a newly-initialized zero value of the extension. Keep in -// mind that extensions 3, 4, and 5 are reserved for -// complex64, complex128, and time.Time, respectively, -// and that MessagePack reserves extension types from -127 to -1. -// -// For example, if you wanted to register a user-defined struct: -// -// msgp.RegisterExtension(10, func() msgp.Extension { &MyExtension{} }) -// -// RegisterExtension will panic if you call it multiple times -// with the same 'typ' argument, or if you use a reserved -// type (3, 4, or 5). -func RegisterExtension(typ int8, f func() Extension) { - switch typ { - case Complex64Extension, Complex128Extension, TimeExtension: - panic(fmt.Sprint("msgp: forbidden extension type:", typ)) - } - if _, ok := extensionReg[typ]; ok { - panic(fmt.Sprint("msgp: RegisterExtension() called with typ", typ, "more than once")) - } - extensionReg[typ] = f -} - -// ExtensionTypeError is an error type returned -// when there is a mis-match between an extension type -// and the type encoded on the wire -type ExtensionTypeError struct { - Got int8 - Want int8 -} - -// Error implements the error interface -func (e ExtensionTypeError) Error() string { - return fmt.Sprintf("msgp: error decoding extension: wanted type %d; got type %d", e.Want, e.Got) -} - -// Resumable returns 'true' for ExtensionTypeErrors -func (e ExtensionTypeError) Resumable() bool { return true } - -func errExt(got int8, wanted int8) error { - return ExtensionTypeError{Got: got, Want: wanted} -} - -// Extension is the interface fulfilled -// by types that want to define their -// own binary encoding. -type Extension interface { - // ExtensionType should return - // a int8 that identifies the concrete - // type of the extension. (Types <0 are - // officially reserved by the MessagePack - // specifications.) - ExtensionType() int8 - - // Len should return the length - // of the data to be encoded - Len() int - - // MarshalBinaryTo should copy - // the data into the supplied slice, - // assuming that the slice has length Len() - MarshalBinaryTo([]byte) error - - UnmarshalBinary([]byte) error -} - -// RawExtension implements the Extension interface -type RawExtension struct { - Data []byte - Type int8 -} - -// ExtensionType implements Extension.ExtensionType, and returns r.Type -func (r *RawExtension) ExtensionType() int8 { return r.Type } - -// Len implements Extension.Len, and returns len(r.Data) -func (r *RawExtension) Len() int { return len(r.Data) } - -// MarshalBinaryTo implements Extension.MarshalBinaryTo, -// and returns a copy of r.Data -func (r *RawExtension) MarshalBinaryTo(d []byte) error { - copy(d, r.Data) - return nil -} - -// UnmarshalBinary implements Extension.UnmarshalBinary, -// and sets r.Data to the contents of the provided slice -func (r *RawExtension) UnmarshalBinary(b []byte) error { - if cap(r.Data) >= len(b) { - r.Data = r.Data[0:len(b)] - } else { - r.Data = make([]byte, len(b)) - } - copy(r.Data, b) - return nil -} - -// peekExtension peeks at the extension encoding type -// (must guarantee at least 1 byte in 'b') -func peekExtension(b []byte) (int8, error) { - spec := sizes[b[0]] - size := spec.size - if spec.typ != ExtensionType { - return 0, badPrefix(ExtensionType, b[0]) - } - if len(b) < int(size) { - return 0, ErrShortBytes - } - // for fixed extensions, - // the type information is in - // the second byte - if spec.extra == constsize { - return int8(b[1]), nil - } - // otherwise, it's in the last - // part of the prefix - return int8(b[size-1]), nil -} - -// AppendExtension appends a MessagePack extension to the provided slice -func AppendExtension(b []byte, e Extension) ([]byte, error) { - l := e.Len() - var o []byte - var n int - switch l { - case 0: - o, n = ensure(b, 3) - o[n] = mext8 - o[n+1] = 0 - o[n+2] = byte(e.ExtensionType()) - return o[:n+3], nil - case 1: - o, n = ensure(b, 3) - o[n] = mfixext1 - o[n+1] = byte(e.ExtensionType()) - n += 2 - case 2: - o, n = ensure(b, 4) - o[n] = mfixext2 - o[n+1] = byte(e.ExtensionType()) - n += 2 - case 4: - o, n = ensure(b, 6) - o[n] = mfixext4 - o[n+1] = byte(e.ExtensionType()) - n += 2 - case 8: - o, n = ensure(b, 10) - o[n] = mfixext8 - o[n+1] = byte(e.ExtensionType()) - n += 2 - case 16: - o, n = ensure(b, 18) - o[n] = mfixext16 - o[n+1] = byte(e.ExtensionType()) - n += 2 - default: - switch { - case l < math.MaxUint8: - o, n = ensure(b, l+3) - o[n] = mext8 - o[n+1] = byte(uint8(l)) - o[n+2] = byte(e.ExtensionType()) - n += 3 - case l < math.MaxUint16: - o, n = ensure(b, l+4) - o[n] = mext16 - big.PutUint16(o[n+1:], uint16(l)) - o[n+3] = byte(e.ExtensionType()) - n += 4 - default: - o, n = ensure(b, l+6) - o[n] = mext32 - big.PutUint32(o[n+1:], uint32(l)) - o[n+5] = byte(e.ExtensionType()) - n += 6 - } - } - return o, e.MarshalBinaryTo(o[n:]) -} - -// ReadExtensionBytes reads an extension from 'b' into 'e' -// and returns any remaining bytes. -// Possible errors: -// - ErrShortBytes ('b' not long enough) -// - ExtensionTypeError{} (wire type not the same as e.Type()) -// - TypeError{} (next object not an extension) -// - InvalidPrefixError -// - An umarshal error returned from e.UnmarshalBinary -func ReadExtensionBytes(b []byte, e Extension) ([]byte, error) { - l := len(b) - if l < 3 { - return b, ErrShortBytes - } - lead := b[0] - var ( - sz int // size of 'data' - off int // offset of 'data' - typ int8 - ) - switch lead { - case mfixext1: - typ = int8(b[1]) - sz = 1 - off = 2 - case mfixext2: - typ = int8(b[1]) - sz = 2 - off = 2 - case mfixext4: - typ = int8(b[1]) - sz = 4 - off = 2 - case mfixext8: - typ = int8(b[1]) - sz = 8 - off = 2 - case mfixext16: - typ = int8(b[1]) - sz = 16 - off = 2 - case mext8: - sz = int(uint8(b[1])) - typ = int8(b[2]) - off = 3 - if sz == 0 { - return b[3:], e.UnmarshalBinary(b[3:3]) - } - case mext16: - if l < 4 { - return b, ErrShortBytes - } - sz = int(big.Uint16(b[1:])) - typ = int8(b[3]) - off = 4 - case mext32: - if l < 6 { - return b, ErrShortBytes - } - var err error - sz, err = u32int(big.Uint32(b[1:])) - if err != nil { - return b, err - } - typ = int8(b[5]) - off = 6 - default: - return b, badPrefix(ExtensionType, lead) - } - - if typ != e.ExtensionType() { - return b, errExt(typ, e.ExtensionType()) - } - - // the data of the extension starts - // at 'off' and is 'sz' bytes long - if len(b[off:]) < sz { - return b, ErrShortBytes - } - tot := off + sz - return b[tot:], e.UnmarshalBinary(b[off:tot]) -} diff --git a/msgp/msgp/extension_test.go b/msgp/msgp/extension_test.go deleted file mode 100644 index a977ea4667..0000000000 --- a/msgp/msgp/extension_test.go +++ /dev/null @@ -1,30 +0,0 @@ -package msgp - -import ( - "math/rand" - "testing" - "time" -) - -var extSizes = [...]int{0, 1, 2, 4, 8, 16, int(tint8), int(tuint16), int(tuint32)} - -func randomExt() RawExtension { - e := RawExtension{} - e.Type = int8(rand.Int()) - e.Data = RandBytes(extSizes[rand.Intn(len(extSizes))]) - return e -} - -func TestReadWriteExtensionBytes(t *testing.T) { - var bts []byte - rand.Seed(time.Now().Unix()) - - for i := 0; i < 24; i++ { - e := randomExt() - bts, _ = AppendExtension(bts[0:0], &e) - _, err := ReadExtensionBytes(bts, &e) - if err != nil { - t.Errorf("error with extension (length %d): %s", len(bts), err) - } - } -} diff --git a/msgp/msgp/floatbench_test.go b/msgp/msgp/floatbench_test.go deleted file mode 100644 index 575b081bb5..0000000000 --- a/msgp/msgp/floatbench_test.go +++ /dev/null @@ -1,25 +0,0 @@ -package msgp - -import ( - "testing" -) - -func BenchmarkReadWriteFloat32(b *testing.B) { - var f float32 = 3.9081 - bts := AppendFloat32([]byte{}, f) - b.ResetTimer() - for i := 0; i < b.N; i++ { - bts = AppendFloat32(bts[0:0], f) - f, bts, _ = ReadFloat32Bytes(bts) - } -} - -func BenchmarkReadWriteFloat64(b *testing.B) { - var f float64 = 3.9081 - bts := AppendFloat64([]byte{}, f) - b.ResetTimer() - for i := 0; i < b.N; i++ { - bts = AppendFloat64(bts[0:0], f) - f, bts, _ = ReadFloat64Bytes(bts) - } -} diff --git a/msgp/msgp/int.go b/msgp/msgp/int.go deleted file mode 100644 index 75bfa1500d..0000000000 --- a/msgp/msgp/int.go +++ /dev/null @@ -1,20 +0,0 @@ -package msgp - -// MaxInt is the maximum int, which might be int32 or int64 -const MaxInt = int((^uint(0)) >> 1) - -func u32int(x uint32) (int, error) { - if uint64(x) > uint64(MaxInt) { - return 0, ErrOverflow(uint64(x), uint64(MaxInt)) - } - - return int(x), nil -} - -func u64int(x uint64) (int, error) { - if x > uint64(MaxInt) { - return 0, ErrOverflow(x, uint64(MaxInt)) - } - - return int(x), nil -} diff --git a/msgp/msgp/integers.go b/msgp/msgp/integers.go deleted file mode 100644 index f817d77598..0000000000 --- a/msgp/msgp/integers.go +++ /dev/null @@ -1,174 +0,0 @@ -package msgp - -/* ---------------------------------- - integer encoding utilities - (inline-able) - - TODO(tinylib): there are faster, - albeit non-portable solutions - to the code below. implement - byteswap? - ---------------------------------- */ - -func putMint64(b []byte, i int64) { - b[0] = mint64 - b[1] = byte(i >> 56) - b[2] = byte(i >> 48) - b[3] = byte(i >> 40) - b[4] = byte(i >> 32) - b[5] = byte(i >> 24) - b[6] = byte(i >> 16) - b[7] = byte(i >> 8) - b[8] = byte(i) -} - -func getMint64(b []byte) int64 { - return (int64(b[1]) << 56) | (int64(b[2]) << 48) | - (int64(b[3]) << 40) | (int64(b[4]) << 32) | - (int64(b[5]) << 24) | (int64(b[6]) << 16) | - (int64(b[7]) << 8) | (int64(b[8])) -} - -func putMint32(b []byte, i int32) { - b[0] = mint32 - b[1] = byte(i >> 24) - b[2] = byte(i >> 16) - b[3] = byte(i >> 8) - b[4] = byte(i) -} - -func getMint32(b []byte) int32 { - return (int32(b[1]) << 24) | (int32(b[2]) << 16) | (int32(b[3]) << 8) | (int32(b[4])) -} - -func putMint16(b []byte, i int16) { - b[0] = mint16 - b[1] = byte(i >> 8) - b[2] = byte(i) -} - -func getMint16(b []byte) (i int16) { - return (int16(b[1]) << 8) | int16(b[2]) -} - -func putMint8(b []byte, i int8) { - b[0] = mint8 - b[1] = byte(i) -} - -func getMint8(b []byte) (i int8) { - return int8(b[1]) -} - -func putMuint64(b []byte, u uint64) { - b[0] = muint64 - b[1] = byte(u >> 56) - b[2] = byte(u >> 48) - b[3] = byte(u >> 40) - b[4] = byte(u >> 32) - b[5] = byte(u >> 24) - b[6] = byte(u >> 16) - b[7] = byte(u >> 8) - b[8] = byte(u) -} - -func getMuint64(b []byte) uint64 { - return (uint64(b[1]) << 56) | (uint64(b[2]) << 48) | - (uint64(b[3]) << 40) | (uint64(b[4]) << 32) | - (uint64(b[5]) << 24) | (uint64(b[6]) << 16) | - (uint64(b[7]) << 8) | (uint64(b[8])) -} - -func putMuint32(b []byte, u uint32) { - b[0] = muint32 - b[1] = byte(u >> 24) - b[2] = byte(u >> 16) - b[3] = byte(u >> 8) - b[4] = byte(u) -} - -func getMuint32(b []byte) uint32 { - return (uint32(b[1]) << 24) | (uint32(b[2]) << 16) | (uint32(b[3]) << 8) | (uint32(b[4])) -} - -func putMuint16(b []byte, u uint16) { - b[0] = muint16 - b[1] = byte(u >> 8) - b[2] = byte(u) -} - -func getMuint16(b []byte) uint16 { - return (uint16(b[1]) << 8) | uint16(b[2]) -} - -func putMuint8(b []byte, u uint8) { - b[0] = muint8 - b[1] = byte(u) -} - -func getMuint8(b []byte) uint8 { - return uint8(b[1]) -} - -func getUnix(b []byte) (sec int64, nsec int32) { - sec = (int64(b[0]) << 56) | (int64(b[1]) << 48) | - (int64(b[2]) << 40) | (int64(b[3]) << 32) | - (int64(b[4]) << 24) | (int64(b[5]) << 16) | - (int64(b[6]) << 8) | (int64(b[7])) - - nsec = (int32(b[8]) << 24) | (int32(b[9]) << 16) | (int32(b[10]) << 8) | (int32(b[11])) - return -} - -func putUnix(b []byte, sec int64, nsec int32) { - b[0] = byte(sec >> 56) - b[1] = byte(sec >> 48) - b[2] = byte(sec >> 40) - b[3] = byte(sec >> 32) - b[4] = byte(sec >> 24) - b[5] = byte(sec >> 16) - b[6] = byte(sec >> 8) - b[7] = byte(sec) - b[8] = byte(nsec >> 24) - b[9] = byte(nsec >> 16) - b[10] = byte(nsec >> 8) - b[11] = byte(nsec) -} - -/* ----------------------------- - prefix utilities - ----------------------------- */ - -// write prefix and uint8 -func prefixu8(b []byte, pre byte, sz uint8) { - b[0] = pre - b[1] = byte(sz) -} - -// write prefix and big-endian uint16 -func prefixu16(b []byte, pre byte, sz uint16) { - b[0] = pre - b[1] = byte(sz >> 8) - b[2] = byte(sz) -} - -// write prefix and big-endian uint32 -func prefixu32(b []byte, pre byte, sz uint32) { - b[0] = pre - b[1] = byte(sz >> 24) - b[2] = byte(sz >> 16) - b[3] = byte(sz >> 8) - b[4] = byte(sz) -} - -func prefixu64(b []byte, pre byte, sz uint64) { - b[0] = pre - b[1] = byte(sz >> 56) - b[2] = byte(sz >> 48) - b[3] = byte(sz >> 40) - b[4] = byte(sz >> 32) - b[5] = byte(sz >> 24) - b[6] = byte(sz >> 16) - b[7] = byte(sz >> 8) - b[8] = byte(sz) -} diff --git a/msgp/msgp/read.go b/msgp/msgp/read.go deleted file mode 100644 index f214e07451..0000000000 --- a/msgp/msgp/read.go +++ /dev/null @@ -1,91 +0,0 @@ -package msgp - -// Type is a MessagePack wire type, -// including this package's built-in -// extension types. -type Type byte - -// MessagePack Types -// -// The zero value of Type -// is InvalidType. -const ( - InvalidType Type = iota - - // MessagePack built-in types - - StrType - BinType - MapType - ArrayType - Float64Type - Float32Type - BoolType - IntType - UintType - NilType - ExtensionType - - // pseudo-types provided - // by extensions - - Complex64Type - Complex128Type - TimeType - - _maxtype -) - -// String implements fmt.Stringer -func (t Type) String() string { - switch t { - case StrType: - return "str" - case BinType: - return "bin" - case MapType: - return "map" - case ArrayType: - return "array" - case Float64Type: - return "float64" - case Float32Type: - return "float32" - case BoolType: - return "bool" - case UintType: - return "uint" - case IntType: - return "int" - case ExtensionType: - return "ext" - case NilType: - return "nil" - default: - return "" - } -} - -// Unmarshaler is the interface fulfilled -// by objects that know how to unmarshal -// themselves from MessagePack. -// UnmarshalMsg unmarshals the object -// from binary, returing any leftover -// bytes and any errors encountered. -// CanUnmarshalMsg checks that o is of the same type as -// was used to generate the UnmarshalMsg code; it can be -// used to guard against UnmarshalMsg() going to an embedded -// field in a struct rather than unmarshaling the entire struct. -type Unmarshaler interface { - UnmarshalMsg([]byte) ([]byte, error) - UnmarshalMsgWithState([]byte, UnmarshalState) ([]byte, error) - CanUnmarshalMsg(o interface{}) bool -} - -// UnmarshalState holds state while running UnmarshalMsg. -type UnmarshalState struct { - Depth uint64 -} - -// DefaultUnmarshalState defines the default state. -var DefaultUnmarshalState = UnmarshalState{Depth: 10000} diff --git a/msgp/msgp/read_bytes.go b/msgp/msgp/read_bytes.go deleted file mode 100644 index 0ade0fb1ca..0000000000 --- a/msgp/msgp/read_bytes.go +++ /dev/null @@ -1,1359 +0,0 @@ -package msgp - -import ( - "encoding/binary" - "math" - "time" -) - -var big = binary.BigEndian - -// NextType returns the type of the next -// object in the slice. If the length -// of the input is zero, it returns -// InvalidType. -func NextType(b []byte) Type { - if len(b) == 0 { - return InvalidType - } - spec := sizes[b[0]] - t := spec.typ - if t == ExtensionType && len(b) > int(spec.size) { - var tp int8 - if spec.extra == constsize { - tp = int8(b[1]) - } else { - tp = int8(b[spec.size-1]) - } - switch tp { - case TimeExtension: - return TimeType - case Complex128Extension: - return Complex128Type - case Complex64Extension: - return Complex64Type - default: - return ExtensionType - } - } - return t -} - -// IsNil returns true if len(b)>0 and -// the leading byte is a 'nil' MessagePack -// byte; false otherwise -func IsNil(b []byte) bool { - if len(b) != 0 && b[0] == mnil { - return true - } - return false -} - -// Raw is raw MessagePack. -// Raw allows you to read and write -// data without interpreting its contents. -type Raw []byte - -// CanMarshalMsg returns true if the z interface is a Raw object ( part of the Marshaler interface ) -func (Raw) CanMarshalMsg(z interface{}) bool { - _, ok := (z).(Raw) - if !ok { - _, ok = (z).(*Raw) - } - return ok -} - -// MarshalMsg implements msgp.Marshaler. -// It appends the raw contents of 'raw' -// to the provided byte slice. If 'raw' -// is 0 bytes, 'nil' will be appended instead. -func (r Raw) MarshalMsg(b []byte) []byte { - i := len(r) - if i == 0 { - return AppendNil(b) - } - o, l := ensure(b, i) - copy(o[l:], []byte(r)) - return o -} - -// CanUnmarshalMsg returns true if the z interface is a Raw object ( part of the Unmarshaler interface ) -func (*Raw) CanUnmarshalMsg(z interface{}) bool { - _, ok := (z).(*Raw) - return ok -} - -// UnmarshalMsg implements msgp.Unmarshaler. -// It sets the contents of *Raw to be the next -// object in the provided byte slice. -func (r *Raw) UnmarshalMsg(b []byte) ([]byte, error) { - return r.UnmarshalMsgWithState(b, DefaultUnmarshalState) -} - -// UnmarshalMsg implements msgp.Unmarshaler. -// It sets the contents of *Raw to be the next -// object in the provided byte slice. -func (r *Raw) UnmarshalMsgWithState(b []byte, st UnmarshalState) ([]byte, error) { - if st.Depth == 0 { - return nil, ErrMaxDepthExceeded{} - } - l := len(b) - out, err := Skip(b) - if err != nil { - return b, err - } - rlen := l - len(out) - if IsNil(b[:rlen]) { - rlen = 0 - } - if cap(*r) < rlen { - *r = make(Raw, rlen) - } else { - *r = (*r)[0:rlen] - } - copy(*r, b[:rlen]) - return out, nil -} - -// Msgsize implements msgp.Sizer -func (r Raw) Msgsize() int { - l := len(r) - if l == 0 { - return 1 // for 'nil' - } - return l -} - -// MsgIsZero returns whether this is a zero value -func (r *Raw) MsgIsZero() bool { - return len(*r) == 0 -} - -// ReadMapHeaderBytes reads a map header size -// from 'b' and returns the remaining bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not a map) -func ReadMapHeaderBytes(b []byte) (sz int, isnil bool, o []byte, err error) { - l := len(b) - if l < 1 { - err = ErrShortBytes - return - } - - lead := b[0] - if isfixmap(lead) { - sz = int(rfixmap(lead)) - o = b[1:] - return - } - - switch lead { - // go-codec compatibility: mnil decodes as a nil map / empty struct - case mnil: - sz = 0 - o = b[1:] - isnil = true - return - - case mmap16: - if l < 3 { - err = ErrShortBytes - return - } - sz = int(big.Uint16(b[1:])) - o = b[3:] - return - - case mmap32: - if l < 5 { - err = ErrShortBytes - return - } - sz, err = u32int(big.Uint32(b[1:])) - if err != nil { - return - } - o = b[5:] - return - - default: - o = b - err = badPrefix(MapType, lead) - return - } -} - -// ReadMapKeyZC attempts to read a map key -// from 'b' and returns the key bytes and the remaining bytes -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not a str or bin) -func ReadMapKeyZC(b []byte) ([]byte, []byte, error) { - o, x, err := ReadStringZC(b) - if err != nil { - if tperr, ok := err.(TypeError); ok && tperr.Encoded == BinType { - return ReadBytesZC(b) - } - return nil, b, err - } - return o, x, nil -} - -// ReadArrayHeaderBytes attempts to read -// the array header size off of 'b' and return -// the size and remaining bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not an array) -func ReadArrayHeaderBytes(b []byte) (sz int, isnil bool, o []byte, err error) { - return readArrayHeaderBytes(b, true) -} - -func readArrayHeaderBytes(b []byte, flattenMap bool) (sz int, isnil bool, o []byte, err error) { - if len(b) < 1 { - return 0, false, nil, ErrShortBytes - } - lead := b[0] - if isfixarray(lead) { - sz = int(rfixarray(lead)) - o = b[1:] - return - } - - // go-codec compat: map can be decoded as an array, by alternating - // the map keys and values in the decoded array. - if flattenMap && isfixmap(lead) { - sz = 2 * int(rfixmap(lead)) - o = b[1:] - return - } - - switch lead { - case mnil: - // go-codec compat: nil decodes as an empty array (nil for slice) - sz = 0 - o = b[1:] - isnil = true - return - - case marray16: - if len(b) < 3 { - err = ErrShortBytes - return - } - sz = int(big.Uint16(b[1:])) - o = b[3:] - return - - case marray32: - if len(b) < 5 { - err = ErrShortBytes - return - } - sz, err = u32int(big.Uint32(b[1:])) - if err != nil { - return - } - o = b[5:] - return - - // go-codec compat: map can be decoded as an array, by alternating - // the map keys and values in the decoded array. - case mmap16: - if flattenMap { - if len(b) < 3 { - err = ErrShortBytes - return - } - sz = 2 * int(big.Uint16(b[1:])) - o = b[3:] - return - } - - case mmap32: - if flattenMap { - if len(b) < 5 { - err = ErrShortBytes - return - } - u64sz := 2 * uint64(big.Uint32(b[1:])) - sz, err = u64int(u64sz) - if err != nil { - return - } - o = b[5:] - return - } - } - - o = b - err = badPrefix(ArrayType, lead) - return -} - -// ReadNilBytes tries to read a "nil" byte -// off of 'b' and return the remaining bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not a 'nil') -// - InvalidPrefixError -func ReadNilBytes(b []byte) ([]byte, error) { - if len(b) < 1 { - return nil, ErrShortBytes - } - if b[0] != mnil { - return b, badPrefix(NilType, b[0]) - } - return b[1:], nil -} - -// ReadFloat64Bytes tries to read a float64 -// from 'b' and return the value and the remaining bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not a float64) -func ReadFloat64Bytes(b []byte) (f float64, o []byte, err error) { - if len(b) < 9 { - if len(b) >= 5 && b[0] == mfloat32 { - var tf float32 - tf, o, err = ReadFloat32Bytes(b) - f = float64(tf) - return - } - if b[0] == mnil { - o = b[1:] - return - } - err = ErrShortBytes - return - } - - if b[0] != mfloat64 { - if b[0] == mfloat32 { - var tf float32 - tf, o, err = ReadFloat32Bytes(b) - f = float64(tf) - return - } - err = badPrefix(Float64Type, b[0]) - return - } - - f = math.Float64frombits(getMuint64(b)) - o = b[9:] - return -} - -// ReadFloat32Bytes tries to read a float64 -// from 'b' and return the value and the remaining bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not a float32) -func ReadFloat32Bytes(b []byte) (f float32, o []byte, err error) { - if len(b) < 1 { - err = ErrShortBytes - return - } - - if b[0] == mnil { - o = b[1:] - return - } - - if b[0] != mfloat32 { - err = TypeError{Method: Float32Type, Encoded: getType(b[0])} - return - } - - if len(b) < 5 { - err = ErrShortBytes - return - } - - f = math.Float32frombits(getMuint32(b)) - o = b[5:] - return -} - -// ReadBoolBytes tries to read a float64 -// from 'b' and return the value and the remaining bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not a bool) -func ReadBoolBytes(b []byte) (bool, []byte, error) { - if len(b) < 1 { - return false, b, ErrShortBytes - } - switch b[0] { - case mtrue: - return true, b[1:], nil - case mfalse: - return false, b[1:], nil - case mnil: - return false, b[1:], nil - default: - return false, b, badPrefix(BoolType, b[0]) - } -} - -// ReadDurationBytes tries to read a time.Duration -// from 'b' and return the value and the remaining bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError (not a int) -func ReadDurationBytes(b []byte) (d time.Duration, o []byte, err error) { - i, o, err := ReadInt64Bytes(b) - return time.Duration(i), o, err -} - -// ReadInt64Bytes tries to read an int64 -// from 'b' and return the value and the remaining bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError (not a int) -func ReadInt64Bytes(b []byte) (i int64, o []byte, err error) { - l := len(b) - if l < 1 { - return 0, nil, ErrShortBytes - } - - lead := b[0] - if isfixint(lead) { - i = int64(rfixint(lead)) - o = b[1:] - return - } - if isnfixint(lead) { - i = int64(rnfixint(lead)) - o = b[1:] - return - } - - switch lead { - case mnil: - i = 0 - o = b[1:] - return - - case mint8: - if l < 2 { - err = ErrShortBytes - return - } - i = int64(getMint8(b)) - o = b[2:] - return - - case muint8: - if l < 2 { - err = ErrShortBytes - return - } - i = int64(getMuint8(b)) - o = b[2:] - return - - case mint16: - if l < 3 { - err = ErrShortBytes - return - } - i = int64(getMint16(b)) - o = b[3:] - return - - case muint16: - if l < 3 { - err = ErrShortBytes - return - } - i = int64(getMuint16(b)) - o = b[3:] - return - - case mint32: - if l < 5 { - err = ErrShortBytes - return - } - i = int64(getMint32(b)) - o = b[5:] - return - - case muint32: - if l < 5 { - err = ErrShortBytes - return - } - i = int64(getMuint32(b)) - o = b[5:] - return - - case mint64: - if l < 9 { - err = ErrShortBytes - return - } - i = int64(getMint64(b)) - o = b[9:] - return - - case muint64: - if l < 9 { - err = ErrShortBytes - return - } - u := getMuint64(b) - // go-codec compat: uint64 encodings that exceed MaxInt64 - // just overflow when parsed as int64. - // - // if u > math.MaxInt64 { - // err = UintOverflow{Value: u, FailedBitsize: 64} - // return - // } - i = int64(u) - o = b[9:] - return - - default: - err = badPrefix(IntType, lead) - return - } -} - -// ReadInt32Bytes tries to read an int32 -// from 'b' and return the value and the remaining bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not a int) -// - IntOverflow{} (value doesn't fit in int32) -func ReadInt32Bytes(b []byte) (int32, []byte, error) { - i, o, err := ReadInt64Bytes(b) - if i > math.MaxInt32 || i < math.MinInt32 { - return 0, o, IntOverflow{Value: i, FailedBitsize: 32} - } - return int32(i), o, err -} - -// ReadInt16Bytes tries to read an int16 -// from 'b' and return the value and the remaining bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not a int) -// - IntOverflow{} (value doesn't fit in int16) -func ReadInt16Bytes(b []byte) (int16, []byte, error) { - i, o, err := ReadInt64Bytes(b) - if i > math.MaxInt16 || i < math.MinInt16 { - return 0, o, IntOverflow{Value: i, FailedBitsize: 16} - } - return int16(i), o, err -} - -// ReadInt8Bytes tries to read an int16 -// from 'b' and return the value and the remaining bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not a int) -// - IntOverflow{} (value doesn't fit in int8) -func ReadInt8Bytes(b []byte) (int8, []byte, error) { - i, o, err := ReadInt64Bytes(b) - if i > math.MaxInt8 || i < math.MinInt8 { - return 0, o, IntOverflow{Value: i, FailedBitsize: 8} - } - return int8(i), o, err -} - -// ReadUint64Bytes tries to read a uint64 -// from 'b' and return the value and the remaining bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not a uint) -func ReadUint64Bytes(b []byte) (u uint64, o []byte, err error) { - l := len(b) - if l < 1 { - return 0, nil, ErrShortBytes - } - - lead := b[0] - if isfixint(lead) { - u = uint64(rfixint(lead)) - o = b[1:] - return - } - - switch lead { - case mnil: - u = 0 - o = b[1:] - return - - case mint8: - if l < 2 { - err = ErrShortBytes - return - } - v := int64(getMint8(b)) - if v < 0 { - err = UintBelowZero{Value: v} - return - } - u = uint64(v) - o = b[2:] - return - - case muint8: - if l < 2 { - err = ErrShortBytes - return - } - u = uint64(getMuint8(b)) - o = b[2:] - return - - case mint16: - if l < 3 { - err = ErrShortBytes - return - } - v := int64(getMint16(b)) - if v < 0 { - err = UintBelowZero{Value: v} - return - } - u = uint64(v) - o = b[3:] - return - - case muint16: - if l < 3 { - err = ErrShortBytes - return - } - u = uint64(getMuint16(b)) - o = b[3:] - return - - case mint32: - if l < 5 { - err = ErrShortBytes - return - } - v := int64(getMint32(b)) - if v < 0 { - err = UintBelowZero{Value: v} - return - } - u = uint64(v) - o = b[5:] - return - - case muint32: - if l < 5 { - err = ErrShortBytes - return - } - u = uint64(getMuint32(b)) - o = b[5:] - return - - case mint64: - if l < 9 { - err = ErrShortBytes - return - } - v := int64(getMint64(b)) - if v < 0 { - err = UintBelowZero{Value: v} - return - } - u = uint64(v) - o = b[9:] - return - - case muint64: - if l < 9 { - err = ErrShortBytes - return - } - u = getMuint64(b) - o = b[9:] - return - - default: - if isnfixint(lead) { - err = UintBelowZero{Value: int64(rnfixint(lead))} - } else { - err = badPrefix(UintType, lead) - } - return - } -} - -// ReadUint32Bytes tries to read a uint32 -// from 'b' and return the value and the remaining bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not a uint) -// - UintOverflow{} (value too large for uint32) -func ReadUint32Bytes(b []byte) (uint32, []byte, error) { - v, o, err := ReadUint64Bytes(b) - if v > math.MaxUint32 { - return 0, nil, UintOverflow{Value: v, FailedBitsize: 32} - } - return uint32(v), o, err -} - -// ReadUint16Bytes tries to read a uint16 -// from 'b' and return the value and the remaining bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not a uint) -// - UintOverflow{} (value too large for uint16) -func ReadUint16Bytes(b []byte) (uint16, []byte, error) { - v, o, err := ReadUint64Bytes(b) - if v > math.MaxUint16 { - return 0, nil, UintOverflow{Value: v, FailedBitsize: 16} - } - return uint16(v), o, err -} - -// ReadUint8Bytes tries to read a uint8 -// from 'b' and return the value and the remaining bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not a uint) -// - UintOverflow{} (value too large for uint8) -func ReadUint8Bytes(b []byte) (uint8, []byte, error) { - v, o, err := ReadUint64Bytes(b) - if v > math.MaxUint8 { - return 0, nil, UintOverflow{Value: v, FailedBitsize: 8} - } - return uint8(v), o, err -} - -// ReadByteBytes is analogous to ReadUint8Bytes -func ReadByteBytes(b []byte) (byte, []byte, error) { - return ReadUint8Bytes(b) -} - -// ReadBytesBytes reads a 'bin' object -// from 'b' and returns its value and -// the remaining bytes in 'b'. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not a 'bin' object) -func ReadBytesBytes(b []byte, scratch []byte) (v []byte, o []byte, err error) { - return readBytesBytes(b, scratch, false, true) -} - -func readBytesBytesSlow(b []byte, flattenMap bool) (v []byte, o []byte, err error) { - var count int - count, _, o, err = readArrayHeaderBytes(b, flattenMap) - if err != nil { - return - } - - if len(o) < count { - err = ErrShortBytes - return - } - - v = make([]byte, count) - for idx := range v { - v[idx], o, err = ReadByteBytes(o) - if err != nil { - return - } - } - - return -} - -// ReadBytesBytesHeader reads the header of a 'bin' object -// from 'b' and return it's length, in bytes. -// Possible errors: -// - ErrShortBytes (too few bytes) -// - TypeError{} (not a 'bin' object) -func ReadBytesBytesHeader(b []byte) (sz int, err error) { - l := len(b) - if l < 1 { - return 0, ErrShortBytes - } - - lead := b[0] - - // go-codec compat: decode string encodings into byte arrays - - if isfixstr(lead) { - sz = int(rfixstr(lead)) - return - } - - switch lead { - case mstr8: - if l < 2 { - err = ErrShortBytes - return - } - sz = int(b[1]) - return - - case mstr16: - if l < 3 { - err = ErrShortBytes - return - } - sz = int(big.Uint16(b[1:])) - return - - case mstr32: - if l < 5 { - err = ErrShortBytes - return - } - sz, err = u32int(big.Uint32(b[1:])) - return - - case mnil: - sz = 0 - return - - case mbin8: - if l < 2 { - err = ErrShortBytes - return - } - sz = int(b[1]) - return - - case mbin16: - if l < 3 { - err = ErrShortBytes - return - } - sz = int(big.Uint16(b[1:])) - return - - case mbin32: - if l < 5 { - err = ErrShortBytes - return - } - sz, err = u32int(big.Uint32(b[1:])) - return - - default: - sz, _, _, err = readArrayHeaderBytes(b, true) - return - } -} - -func readBytesBytes(b []byte, scratch []byte, zc bool, flattenMap bool) (v []byte, o []byte, err error) { - l := len(b) - if l < 1 { - return nil, nil, ErrShortBytes - } - - lead := b[0] - var read int - - // go-codec compat: decode string encodings into byte arrays - - if isfixstr(lead) { - read = int(rfixstr(lead)) - b = b[1:] - } else { - switch lead { - case mstr8: - if l < 2 { - err = ErrShortBytes - return - } - read = int(b[1]) - b = b[2:] - - case mstr16: - if l < 3 { - err = ErrShortBytes - return - } - read = int(big.Uint16(b[1:])) - b = b[3:] - - case mstr32: - if l < 5 { - err = ErrShortBytes - return - } - read, err = u32int(big.Uint32(b[1:])) - if err != nil { - return - } - b = b[5:] - - case mnil: - v = nil - o = b[1:] - return - - case mbin8: - if l < 2 { - err = ErrShortBytes - return - } - read = int(b[1]) - b = b[2:] - - case mbin16: - if l < 3 { - err = ErrShortBytes - return - } - read = int(big.Uint16(b[1:])) - b = b[3:] - - case mbin32: - if l < 5 { - err = ErrShortBytes - return - } - read, err = u32int(big.Uint32(b[1:])) - if err != nil { - return - } - b = b[5:] - - default: - // go-codec compat: decode into byte array/slice from - // explicit array encodings (including the weird case - // of decoding a map as a key-value interleaved array). - v, o, err = readBytesBytesSlow(b, flattenMap) - if err != nil { - // If that doesn't work, return the original error code. - err = badPrefix(BinType, lead) - } - return - } - } - - if len(b) < read { - err = ErrShortBytes - return - } - - // zero-copy - if zc { - v = b[0:read] - o = b[read:] - return - } - - // The "scratch != nil" check is to match go-codec behavior: - // decode zero-length byte slices as a non-nil byte slice. - if scratch != nil && cap(scratch) >= read { - v = scratch[0:read] - } else { - v = make([]byte, read) - } - - o = b[copy(v, b):] - return -} - -// ReadBytesZC extracts the messagepack-encoded -// binary field without copying. The returned []byte -// points to the same memory as the input slice. -// Possible errors: -// - ErrShortBytes (b not long enough) -// - TypeError{} (object not 'bin') -func ReadBytesZC(b []byte) (v []byte, o []byte, err error) { - return readBytesBytes(b, nil, true, true) -} - -func readExactBytesSlow(b []byte, into []byte) (o []byte, err error) { - var count int - count, _, o, err = ReadArrayHeaderBytes(b) - if err != nil { - return - } - - if count > len(into) { - err = ArrayError{Wanted: len(into), Got: count} - return - } - - for idx := 0; idx < count; idx++ { - into[idx], o, err = ReadByteBytes(o) - if err != nil { - return - } - } - - return -} - -func ReadExactBytes(b []byte, into []byte) (o []byte, err error) { - l := len(b) - if l < 1 { - err = ErrShortBytes - return - } - - lead := b[0] - var read int - var skip int - - // go-codec compat: decode string encodings into byte arrays - - if isfixstr(lead) { - read = int(rfixstr(lead)) - skip = 1 - } else { - switch lead { - case mstr8: - if l < 2 { - err = ErrShortBytes - return - } - read = int(b[1]) - skip = 2 - - case mstr16: - if l < 3 { - err = ErrShortBytes - return - } - read = int(big.Uint16(b[1:])) - skip = 3 - - case mstr32: - if l < 5 { - err = ErrShortBytes - return - } - read, err = u32int(big.Uint32(b[1:])) - if err != nil { - return - } - skip = 5 - - case mnil: - // go-codec compat: decoding nil into an array clears the array; - // different from decoding a zero-length array (which updates - // in-place). - for i := range into { - into[i] = 0 - } - read = 0 - skip = 1 - - case mbin8: - if l < 2 { - err = ErrShortBytes - return - } - read = int(b[1]) - skip = 2 - - case mbin16: - if l < 3 { - err = ErrShortBytes - return - } - read = int(big.Uint16(b[1:])) - skip = 3 - - case mbin32: - if l < 5 { - err = ErrShortBytes - return - } - read, err = u32int(big.Uint32(b[1:])) - if err != nil { - return - } - skip = 5 - - default: - // go-codec compat: decode into byte array from - // explicit array encodings (including the weird case - // of decoding a map as a key-value interleaved array). - o, err = readExactBytesSlow(b, into) - if err != nil { - // If that doesn't work, return the original error code. - err = badPrefix(BinType, lead) - } - return - } - } - - // go-codec compat: allow decoding a different number of bytes than the - // size of the fixed array; take the min of the size of the Go type and - // the encoded array size. - // - // if read != len(into) { - // err = ArrayError{Wanted: uint32(len(into)), Got: uint32(read)} - // return - // } - - if read > len(b[skip:]) { - err = ErrShortBytes - return - } - - copy(into, b[skip:skip+read]) - o = b[skip+read:] - return -} - -// ReadStringZC reads a messagepack string field -// without copying. The returned []byte points -// to the same memory as the input slice. -// Possible errors: -// - ErrShortBytes (b not long enough) -// - TypeError{} (object not 'str') -func ReadStringZC(b []byte) (v []byte, o []byte, err error) { - l := len(b) - if l < 1 { - return nil, nil, ErrShortBytes - } - - lead := b[0] - var read int - - if isfixstr(lead) { - read = int(rfixstr(lead)) - b = b[1:] - } else { - switch lead { - case mnil: - read = 0 - b = b[1:] - - case mstr8: - if l < 2 { - err = ErrShortBytes - return - } - read = int(b[1]) - b = b[2:] - - case mstr16: - if l < 3 { - err = ErrShortBytes - return - } - read = int(big.Uint16(b[1:])) - b = b[3:] - - case mstr32: - if l < 5 { - err = ErrShortBytes - return - } - read, err = u32int(big.Uint32(b[1:])) - if err != nil { - return - } - b = b[5:] - - default: - // go-codec compat: decode bin types into string - v, o, err = readBytesBytes(b, nil, true, false) - if err != nil { - // If the fallback fails, return original error code - err = TypeError{Method: StrType, Encoded: getType(lead)} - } - return - } - } - - if len(b) < read { - err = ErrShortBytes - return - } - - v = b[0:read] - o = b[read:] - return -} - -// ReadStringBytes reads a 'str' object -// from 'b' and returns its value and the -// remaining bytes in 'b'. -// Possible errors: -// - ErrShortBytes (b not long enough) -// - TypeError{} (not 'str' type) -// - InvalidPrefixError -func ReadStringBytes(b []byte) (string, []byte, error) { - v, o, err := ReadStringZC(b) - return string(v), o, err -} - -// ReadStringAsBytes reads a 'str' object -// into a slice of bytes. 'v' is the value of -// the 'str' object, which may reside in memory -// pointed to by 'scratch.' 'o' is the remaining bytes -// in 'b'. -// Possible errors: -// - ErrShortBytes (b not long enough) -// - TypeError{} (not 'str' type) -// - InvalidPrefixError (unknown type marker) -func ReadStringAsBytes(b []byte, scratch []byte) (v []byte, o []byte, err error) { - var tmp []byte - tmp, o, err = ReadStringZC(b) - v = append(scratch[:0], tmp...) - return -} - -// ReadComplex128Bytes reads a complex128 -// extension object from 'b' and returns the -// remaining bytes. -// Possible errors: -// - ErrShortBytes (not enough bytes in 'b') -// - TypeError{} (object not a complex128) -// - InvalidPrefixError -// - ExtensionTypeError{} (object an extension of the correct size, but not a complex128) -func ReadComplex128Bytes(b []byte) (c complex128, o []byte, err error) { - if len(b) < 18 { - err = ErrShortBytes - return - } - if b[0] != mfixext16 { - err = badPrefix(Complex128Type, b[0]) - return - } - if int8(b[1]) != Complex128Extension { - err = errExt(int8(b[1]), Complex128Extension) - return - } - c = complex(math.Float64frombits(big.Uint64(b[2:])), - math.Float64frombits(big.Uint64(b[10:]))) - o = b[18:] - return -} - -// ReadComplex64Bytes reads a complex64 -// extension object from 'b' and returns the -// remaining bytes. -// Possible errors: -// - ErrShortBytes (not enough bytes in 'b') -// - TypeError{} (object not a complex64) -// - ExtensionTypeError{} (object an extension of the correct size, but not a complex64) -func ReadComplex64Bytes(b []byte) (c complex64, o []byte, err error) { - if len(b) < 10 { - err = ErrShortBytes - return - } - if b[0] != mfixext8 { - err = badPrefix(Complex64Type, b[0]) - return - } - if b[1] != Complex64Extension { - err = errExt(int8(b[1]), Complex64Extension) - return - } - c = complex(math.Float32frombits(big.Uint32(b[2:])), - math.Float32frombits(big.Uint32(b[6:]))) - o = b[10:] - return -} - -// ReadTimeBytes reads a time.Time -// extension object from 'b' and returns the -// remaining bytes. -// Possible errors: -// - ErrShortBytes (not enough bytes in 'b') -// - TypeError{} (object not a complex64) -// - ExtensionTypeError{} (object an extension of the correct size, but not a time.Time) -func ReadTimeBytes(b []byte) (t time.Time, o []byte, err error) { - if len(b) < 1 { - err = ErrShortBytes - return - } - if b[0] == mnil { - o = b[1:] - return - } - if len(b) < 15 { - err = ErrShortBytes - return - } - if b[0] != mext8 || b[1] != 12 { - err = badPrefix(TimeType, b[0]) - return - } - if int8(b[2]) != TimeExtension { - err = errExt(int8(b[2]), TimeExtension) - return - } - sec, nsec := getUnix(b[3:]) - t = time.Unix(sec, int64(nsec)).Local() - o = b[15:] - return -} - -// Skip skips the next object in 'b' and -// returns the remaining bytes. If the object -// is a map or array, all of its elements -// will be skipped. -// Possible Errors: -// - ErrShortBytes (not enough bytes in b) -// - InvalidPrefixError (bad encoding) -func Skip(b []byte) ([]byte, error) { - sz, asz, err := getSize(b) - if err != nil { - return b, err - } - if uintptr(len(b)) < sz { - return b, ErrShortBytes - } - b = b[sz:] - for asz > 0 { - b, err = Skip(b) - if err != nil { - return b, err - } - asz-- - } - return b, nil -} - -// returns (skip N bytes, skip M objects, error) -func getSize(b []byte) (uintptr, uintptr, error) { - l := len(b) - if l == 0 { - return 0, 0, ErrShortBytes - } - lead := b[0] - spec := &sizes[lead] // get type information - size, mode := spec.size, spec.extra - if size == 0 { - return 0, 0, InvalidPrefixError(lead) - } - if mode >= 0 { // fixed composites - return uintptr(size), uintptr(mode), nil - } - if l < int(size) { - return 0, 0, ErrShortBytes - } - switch mode { - case extra8: - return uintptr(size) + uintptr(b[1]), 0, nil - case extra16: - return uintptr(size) + uintptr(big.Uint16(b[1:])), 0, nil - case extra32: - return uintptr(size) + uintptr(big.Uint32(b[1:])), 0, nil - case map16v: - return uintptr(size), 2 * uintptr(big.Uint16(b[1:])), nil - case map32v: - return uintptr(size), 2 * uintptr(big.Uint32(b[1:])), nil - case array16v: - return uintptr(size), uintptr(big.Uint16(b[1:])), nil - case array32v: - return uintptr(size), uintptr(big.Uint32(b[1:])), nil - default: - return 0, 0, fatal - } -} diff --git a/msgp/msgp/read_bytes_test.go b/msgp/msgp/read_bytes_test.go deleted file mode 100644 index 73a4a16b02..0000000000 --- a/msgp/msgp/read_bytes_test.go +++ /dev/null @@ -1,100 +0,0 @@ -package msgp - -import ( - "testing" - "time" -) - -func BenchmarkReadMapHeaderBytes(b *testing.B) { - sizes := []uint32{1, 100, tuint16, tuint32} - buf := make([]byte, 0, 5*len(sizes)) - for _, sz := range sizes { - buf = AppendMapHeader(buf, sz) - } - b.SetBytes(int64(len(buf) / len(sizes))) - b.ReportAllocs() - b.ResetTimer() - o := buf - for i := 0; i < b.N; i++ { - _, _, buf, _ = ReadMapHeaderBytes(buf) - if len(buf) == 0 { - buf = o - } - } -} - -func BenchmarkReadArrayHeaderBytes(b *testing.B) { - sizes := []uint32{1, 100, tuint16, tuint32} - buf := make([]byte, 0, 5*len(sizes)) - for _, sz := range sizes { - buf = AppendArrayHeader(buf, sz) - } - b.SetBytes(int64(len(buf) / len(sizes))) - b.ReportAllocs() - b.ResetTimer() - o := buf - for i := 0; i < b.N; i++ { - _, _, buf, _ = ReadArrayHeaderBytes(buf) - if len(buf) == 0 { - buf = o - } - } -} - -func BenchmarkReadNilByte(b *testing.B) { - buf := []byte{mnil} - b.SetBytes(1) - b.ReportAllocs() - b.ResetTimer() - for i := 0; i < b.N; i++ { - ReadNilBytes(buf) - } -} - -func BenchmarkReadFloat64Bytes(b *testing.B) { - f := float64(3.14159) - buf := make([]byte, 0, 9) - buf = AppendFloat64(buf, f) - b.SetBytes(int64(len(buf))) - b.ReportAllocs() - b.ResetTimer() - for i := 0; i < b.N; i++ { - ReadFloat64Bytes(buf) - } -} - -func BenchmarkReadFloat32Bytes(b *testing.B) { - f := float32(3.14159) - buf := make([]byte, 0, 5) - buf = AppendFloat32(buf, f) - b.SetBytes(int64(len(buf))) - b.ReportAllocs() - b.ResetTimer() - for i := 0; i < b.N; i++ { - ReadFloat32Bytes(buf) - } -} - -func BenchmarkReadBoolBytes(b *testing.B) { - buf := []byte{mtrue, mfalse, mtrue, mfalse} - b.SetBytes(1) - b.ReportAllocs() - b.ResetTimer() - o := buf - for i := 0; i < b.N; i++ { - _, buf, _ = ReadBoolBytes(buf) - if len(buf) == 0 { - buf = o - } - } -} - -func BenchmarkReadTimeBytes(b *testing.B) { - data := AppendTime(nil, time.Now()) - b.SetBytes(15) - b.ReportAllocs() - b.ResetTimer() - for i := 0; i < b.N; i++ { - ReadTimeBytes(data) - } -} diff --git a/msgp/msgp/size.go b/msgp/msgp/size.go deleted file mode 100644 index e3a613b248..0000000000 --- a/msgp/msgp/size.go +++ /dev/null @@ -1,39 +0,0 @@ -package msgp - -// The sizes provided -// are the worst-case -// encoded sizes for -// each type. For variable- -// length types ([]byte, string), -// the total encoded size is -// the prefix size plus the -// length of the object. -const ( - Int64Size = 9 - IntSize = Int64Size - UintSize = Int64Size - Int8Size = 2 - Int16Size = 3 - Int32Size = 5 - Uint8Size = 2 - ByteSize = Uint8Size - Uint16Size = 3 - Uint32Size = 5 - Uint64Size = Int64Size - Float64Size = 9 - Float32Size = 5 - Complex64Size = 10 - Complex128Size = 18 - - DurationSize = Int64Size - TimeSize = 15 - BoolSize = 1 - NilSize = 1 - - MapHeaderSize = 5 - ArrayHeaderSize = 5 - - BytesPrefixSize = 5 - StringPrefixSize = 5 - ExtensionPrefixSize = 6 -) diff --git a/msgp/msgp/write.go b/msgp/msgp/write.go deleted file mode 100644 index da23cf7010..0000000000 --- a/msgp/msgp/write.go +++ /dev/null @@ -1,65 +0,0 @@ -package msgp - -// Sizer is an interface implemented -// by types that can estimate their -// size when MessagePack encoded. -// This interface is optional, but -// encoding/marshaling implementations -// may use this as a way to pre-allocate -// memory for serialization. -type Sizer interface { - Msgsize() int -} - -// MaxSizer is an interface implemented -// by types that can determine their max -// when implemented. -// This interface is optional, but -// implementations may use this as a way to limit -// number of bytes read during deserialization -type MaxSizer interface { - MaxSize() int -} - -// Require ensures that cap(old)-len(old) >= extra. -// It might be that this is impossible because len(old)+extra -// overflows int. If so, Require will not grow the slice, -// but at this point, we have run out of memory, and panic -// (from subsequent out-of-bounds access) is as good of an -// outcome as any. -func Require(old []byte, extra int) []byte { - l := len(old) - c := cap(old) - r := l + extra - if c >= r { - return old - } else if l == 0 { - return make([]byte, 0, extra) - } - // the new size is the greater - // of double the old capacity - // and the sum of the old length - // and the number of new bytes - // necessary. - c <<= 1 - if c < r { - c = r - } - n := make([]byte, l, c) - copy(n, old) - return n -} - -// Marshaler is the interface implemented -// by types that know how to marshal themselves -// as MessagePack. MarshalMsg appends the marshalled -// form of the object to the provided -// byte slice, returning the extended slice. -// CanMarshalMsg checks that o is of the same type as -// was used to generate the MarshalMsg code; it can be -// used to guard against MarshalMsg() going to an embedded -// field in a struct rather than marshaling the entire struct. -type Marshaler interface { - MarshalMsg([]byte) []byte - CanMarshalMsg(o interface{}) bool -} diff --git a/msgp/msgp/write_bytes.go b/msgp/msgp/write_bytes.go deleted file mode 100644 index b96715ccf3..0000000000 --- a/msgp/msgp/write_bytes.go +++ /dev/null @@ -1,288 +0,0 @@ -package msgp - -import ( - "math" - "time" -) - -// ensure 'sz' extra bytes in 'b' btw len(b) and cap(b) -// If the growth length overflows, we are anyway running -// out of memory, so panic (on a subsequent out-of-bounds -// slice reference) seems like as good of a result as any. -func ensure(b []byte, sz int) ([]byte, int) { - l := len(b) - c := cap(b) - if c-l < sz { - o := make([]byte, (2*c)+sz) // exponential growth - n := copy(o, b) - return o[:n+sz], n - } - return b[:l+sz], l -} - -// AppendMapHeader appends a map header with the -// given size to the slice -func AppendMapHeader(b []byte, sz uint32) []byte { - switch { - case sz <= 15: - return append(b, wfixmap(uint8(sz))) - - case sz <= math.MaxUint16: - o, n := ensure(b, 3) - prefixu16(o[n:], mmap16, uint16(sz)) - return o - - default: - o, n := ensure(b, 5) - prefixu32(o[n:], mmap32, sz) - return o - } -} - -// AppendArrayHeader appends an array header with -// the given size to the slice -func AppendArrayHeader(b []byte, sz uint32) []byte { - switch { - case sz <= 15: - return append(b, wfixarray(uint8(sz))) - - case sz <= math.MaxUint16: - o, n := ensure(b, 3) - prefixu16(o[n:], marray16, uint16(sz)) - return o - - default: - o, n := ensure(b, 5) - prefixu32(o[n:], marray32, sz) - return o - } -} - -// AppendNil appends a 'nil' byte to the slice -func AppendNil(b []byte) []byte { return append(b, mnil) } - -// AppendFloat64 appends a float64 to the slice -func AppendFloat64(b []byte, f float64) []byte { - o, n := ensure(b, Float64Size) - prefixu64(o[n:], mfloat64, math.Float64bits(f)) - return o -} - -// AppendFloat32 appends a float32 to the slice -func AppendFloat32(b []byte, f float32) []byte { - o, n := ensure(b, Float32Size) - prefixu32(o[n:], mfloat32, math.Float32bits(f)) - return o -} - -// AppendDuration appends a time.Duration to the slice -func AppendDuration(b []byte, d time.Duration) []byte { - return AppendInt64(b, int64(d)) -} - -// AppendInt64 appends an int64 to the slice -func AppendInt64(b []byte, i int64) []byte { - if i >= 0 { - return AppendUint64(b, uint64(i)) - } - switch { - case i >= -32: - return append(b, wnfixint(int8(i))) - case i >= math.MinInt8: - o, n := ensure(b, 2) - putMint8(o[n:], int8(i)) - return o - case i >= math.MinInt16: - o, n := ensure(b, 3) - putMint16(o[n:], int16(i)) - return o - case i >= math.MinInt32: - o, n := ensure(b, 5) - putMint32(o[n:], int32(i)) - return o - default: - o, n := ensure(b, 9) - putMint64(o[n:], i) - return o - } -} - -// AppendInt8 appends an int8 to the slice -func AppendInt8(b []byte, i int8) []byte { return AppendInt64(b, int64(i)) } - -// AppendInt16 appends an int16 to the slice -func AppendInt16(b []byte, i int16) []byte { return AppendInt64(b, int64(i)) } - -// AppendInt32 appends an int32 to the slice -func AppendInt32(b []byte, i int32) []byte { return AppendInt64(b, int64(i)) } - -// AppendUint64 appends a uint64 to the slice -func AppendUint64(b []byte, u uint64) []byte { - switch { - case u <= (1<<7)-1: - return append(b, wfixint(uint8(u))) - - case u <= math.MaxUint8: - o, n := ensure(b, 2) - putMuint8(o[n:], uint8(u)) - return o - - case u <= math.MaxUint16: - o, n := ensure(b, 3) - putMuint16(o[n:], uint16(u)) - return o - - case u <= math.MaxUint32: - o, n := ensure(b, 5) - putMuint32(o[n:], uint32(u)) - return o - - default: - o, n := ensure(b, 9) - putMuint64(o[n:], u) - return o - - } -} - -// AppendUint8 appends a uint8 to the slice -func AppendUint8(b []byte, u uint8) []byte { return AppendUint64(b, uint64(u)) } - -// AppendByte is analogous to AppendUint8 -func AppendByte(b []byte, u byte) []byte { return AppendUint8(b, uint8(u)) } - -// AppendUint16 appends a uint16 to the slice -func AppendUint16(b []byte, u uint16) []byte { return AppendUint64(b, uint64(u)) } - -// AppendUint32 appends a uint32 to the slice -func AppendUint32(b []byte, u uint32) []byte { return AppendUint64(b, uint64(u)) } - -// AppendBytes appends bytes to the slice as MessagePack 'bin' data -func AppendBytes(b []byte, bts []byte) []byte { - sz := len(bts) - var o []byte - var n int - switch { - case bts == nil: - o, n = ensure(b, 1) - o[n] = mnil - n += 1 - case sz <= math.MaxUint8: - o, n = ensure(b, 2+sz) - prefixu8(o[n:], mbin8, uint8(sz)) - n += 2 - case sz <= math.MaxUint16: - o, n = ensure(b, 3+sz) - prefixu16(o[n:], mbin16, uint16(sz)) - n += 3 - default: - o, n = ensure(b, 5+sz) - prefixu32(o[n:], mbin32, uint32(sz)) - n += 5 - } - return o[:n+copy(o[n:], bts)] -} - -// AppendBool appends a bool to the slice -func AppendBool(b []byte, t bool) []byte { - if t { - return append(b, mtrue) - } - return append(b, mfalse) -} - -// AppendString appends a string as a MessagePack 'str' to the slice -func AppendString(b []byte, s string) []byte { - sz := len(s) - var n int - var o []byte - switch { - case sz <= 31: - o, n = ensure(b, 1+sz) - o[n] = wfixstr(uint8(sz)) - n++ - case sz <= math.MaxUint8: - o, n = ensure(b, 2+sz) - prefixu8(o[n:], mstr8, uint8(sz)) - n += 2 - case sz <= math.MaxUint16: - o, n = ensure(b, 3+sz) - prefixu16(o[n:], mstr16, uint16(sz)) - n += 3 - default: - o, n = ensure(b, 5+sz) - prefixu32(o[n:], mstr32, uint32(sz)) - n += 5 - } - return o[:n+copy(o[n:], s)] -} - -// AppendStringFromBytes appends a []byte -// as a MessagePack 'str' to the slice 'b.' -func AppendStringFromBytes(b []byte, str []byte) []byte { - sz := len(str) - var n int - var o []byte - switch { - case sz <= 31: - o, n = ensure(b, 1+sz) - o[n] = wfixstr(uint8(sz)) - n++ - case sz <= math.MaxUint8: - o, n = ensure(b, 2+sz) - prefixu8(o[n:], mstr8, uint8(sz)) - n += 2 - case sz <= math.MaxUint16: - o, n = ensure(b, 3+sz) - prefixu16(o[n:], mstr16, uint16(sz)) - n += 3 - default: - o, n = ensure(b, 5+sz) - prefixu32(o[n:], mstr32, uint32(sz)) - n += 5 - } - return o[:n+copy(o[n:], str)] -} - -// AppendComplex64 appends a complex64 to the slice as a MessagePack extension -func AppendComplex64(b []byte, c complex64) []byte { - o, n := ensure(b, Complex64Size) - o[n] = mfixext8 - o[n+1] = Complex64Extension - big.PutUint32(o[n+2:], math.Float32bits(real(c))) - big.PutUint32(o[n+6:], math.Float32bits(imag(c))) - return o -} - -// AppendComplex128 appends a complex128 to the slice as a MessagePack extension -func AppendComplex128(b []byte, c complex128) []byte { - o, n := ensure(b, Complex128Size) - o[n] = mfixext16 - o[n+1] = Complex128Extension - big.PutUint64(o[n+2:], math.Float64bits(real(c))) - big.PutUint64(o[n+10:], math.Float64bits(imag(c))) - return o -} - -// AppendTime appends a time.Time to the slice as a MessagePack extension -func AppendTime(b []byte, t time.Time) []byte { - o, n := ensure(b, TimeSize) - t = t.UTC() - o[n] = mext8 - o[n+1] = 12 - o[n+2] = TimeExtension - putUnix(o[n+3:], t.Unix(), int32(t.Nanosecond())) - return o -} - -// AppendMapStrStr appends a map[string]string to the slice -// as a MessagePack map with 'str'-type keys and values -func AppendMapStrStr(b []byte, m map[string]string) []byte { - sz := uint32(len(m)) - b = AppendMapHeader(b, sz) - for key, val := range m { - b = AppendString(b, key) - b = AppendString(b, val) - } - return b -} diff --git a/msgp/msgp/write_bytes_test.go b/msgp/msgp/write_bytes_test.go deleted file mode 100644 index d7e88738cf..0000000000 --- a/msgp/msgp/write_bytes_test.go +++ /dev/null @@ -1,140 +0,0 @@ -package msgp - -import ( - "testing" - "time" -) - -func BenchmarkAppendMapHeader(b *testing.B) { - buf := make([]byte, 0, 9) - N := b.N / 4 - b.ReportAllocs() - b.ResetTimer() - for i := 0; i < N; i++ { - AppendMapHeader(buf[:0], 0) - AppendMapHeader(buf[:0], uint32(tint8)) - AppendMapHeader(buf[:0], tuint16) - AppendMapHeader(buf[:0], tuint32) - } -} - -func BenchmarkAppendArrayHeader(b *testing.B) { - buf := make([]byte, 0, 9) - N := b.N / 4 - b.ReportAllocs() - b.ResetTimer() - for i := 0; i < N; i++ { - AppendArrayHeader(buf[:0], 0) - AppendArrayHeader(buf[:0], uint32(tint8)) - AppendArrayHeader(buf[:0], tuint16) - AppendArrayHeader(buf[:0], tuint32) - } -} - -func TestAppendNil(t *testing.T) { - var bts []byte - bts = AppendNil(bts[0:0]) - if bts[0] != mnil { - t.Fatal("bts[0] is not 'nil'") - } -} - -func BenchmarkAppendFloat64(b *testing.B) { - f := float64(3.14159) - buf := make([]byte, 0, 9) - b.SetBytes(9) - b.ReportAllocs() - b.ResetTimer() - for i := 0; i < b.N; i++ { - AppendFloat64(buf[0:0], f) - } -} - -func BenchmarkAppendFloat32(b *testing.B) { - f := float32(3.14159) - buf := make([]byte, 0, 5) - b.SetBytes(5) - b.ReportAllocs() - b.ResetTimer() - for i := 0; i < b.N; i++ { - AppendFloat32(buf[0:0], f) - } -} - -func BenchmarkAppendInt64(b *testing.B) { - is := []int64{0, 1, -5, -50, int64(tint16), int64(tint32), int64(tint64)} - l := len(is) - buf := make([]byte, 0, 9) - b.ReportAllocs() - b.ResetTimer() - for i := 0; i < b.N; i++ { - AppendInt64(buf[0:0], is[i%l]) - } -} - -func BenchmarkAppendUint64(b *testing.B) { - us := []uint64{0, 1, 15, uint64(tuint16), uint64(tuint32), tuint64} - buf := make([]byte, 0, 9) - b.ReportAllocs() - b.ResetTimer() - l := len(us) - for i := 0; i < b.N; i++ { - AppendUint64(buf[0:0], us[i%l]) - } -} - -func benchappendBytes(size uint32, b *testing.B) { - bts := RandBytes(int(size)) - buf := make([]byte, 0, len(bts)+5) - b.SetBytes(int64(len(bts) + 5)) - b.ReportAllocs() - b.ResetTimer() - for i := 0; i < b.N; i++ { - AppendBytes(buf[0:0], bts) - } -} - -func BenchmarkAppend16Bytes(b *testing.B) { benchappendBytes(16, b) } - -func BenchmarkAppend256Bytes(b *testing.B) { benchappendBytes(256, b) } - -func BenchmarkAppend2048Bytes(b *testing.B) { benchappendBytes(2048, b) } - -func benchappendString(size uint32, b *testing.B) { - str := string(RandBytes(int(size))) - buf := make([]byte, 0, len(str)+5) - b.SetBytes(int64(len(str) + 5)) - b.ReportAllocs() - b.ResetTimer() - for i := 0; i < b.N; i++ { - AppendString(buf[0:0], str) - } -} - -func BenchmarkAppend16String(b *testing.B) { benchappendString(16, b) } - -func BenchmarkAppend256String(b *testing.B) { benchappendString(256, b) } - -func BenchmarkAppend2048String(b *testing.B) { benchappendString(2048, b) } - -func BenchmarkAppendBool(b *testing.B) { - vs := []bool{true, false} - buf := make([]byte, 0, 1) - b.SetBytes(1) - b.ReportAllocs() - b.ResetTimer() - for i := 0; i < b.N; i++ { - AppendBool(buf[0:0], vs[i%2]) - } -} - -func BenchmarkAppendTime(b *testing.B) { - t := time.Now() - b.SetBytes(15) - buf := make([]byte, 0, 15) - b.ReportAllocs() - b.ResetTimer() - for i := 0; i < b.N; i++ { - AppendTime(buf[0:0], t) - } -} diff --git a/msgp/msgp/write_test.go b/msgp/msgp/write_test.go deleted file mode 100644 index 25b65a091e..0000000000 --- a/msgp/msgp/write_test.go +++ /dev/null @@ -1,24 +0,0 @@ -package msgp - -import ( - "math" - "math/rand" -) - -var ( - tint8 int8 = 126 // cannot be most fix* types - tint16 int16 = 150 // cannot be int8 - tint32 int32 = math.MaxInt16 + 100 // cannot be int16 - tint64 int64 = math.MaxInt32 + 100 // cannot be int32 - tuint16 uint32 = 300 // cannot be uint8 - tuint32 uint32 = math.MaxUint16 + 100 // cannot be uint16 - tuint64 uint64 = math.MaxUint32 + 100 // cannot be uint32 -) - -func RandBytes(sz int) []byte { - out := make([]byte, sz) - for i := range out { - out[i] = byte(rand.Int63n(math.MaxInt64) % 256) - } - return out -} diff --git a/msgp/parse/directives.go b/msgp/parse/directives.go deleted file mode 100644 index 75d00111f8..0000000000 --- a/msgp/parse/directives.go +++ /dev/null @@ -1,194 +0,0 @@ -package parse - -import ( - "errors" - "fmt" - "go/ast" - "strings" - - "github.com/algorand/msgp/gen" -) - -const linePrefix = "//msgp:" - -// func(args, fileset) -type directive func([]string, *FileSet) error - -// func(passName, args, printer) -type passDirective func(gen.Method, []string, *gen.Printer) error - -// map of all recognized directives -// -// to add a directive, define a func([]string, *FileSet) error -// and then add it to this list. -var directives = map[string]directive{ - "shim": applyShim, - "ignore": ignore, - "tuple": astuple, - "sort": sortintf, - "allocbound": allocbound, - // _postunmarshalcheck is used to add callbacks to the end of un-marshalling that are tied to a specific Element. - _postunmarshalcheck: postunmarshalcheck, -} - -const _postunmarshalcheck = "postunmarshalcheck" - -var errNotEnoughArguments = errors.New("postunmarshalcheck did not receive enough arguments. expected at least 3") - -//msgp:postunmarshalcheck {Type} {funcName} {funcName} ... -// the functions should have no params, and output zero. -func postunmarshalcheck(text []string, f *FileSet) error { - if len(text) < 3 { - return errNotEnoughArguments - } - // not error but doesn't do anything - if text[0] != _postunmarshalcheck { - return nil - } - text = text[1:] - - elemType := text[0] - elem, ok := f.Identities[elemType] - if !ok { - return errors.New(fmt.Sprintf("postunmarshalcheck error: type %v does not exist", elemType)) - } - for _, fName := range text[1:] { - elem.AddCallback(gen.Callback{ - Fname: fName, - CallbackType: gen.UnmarshalCallBack, - }) - } - return nil -} - -var passDirectives = map[string]passDirective{ - "ignore": passignore, -} - -func passignore(m gen.Method, text []string, p *gen.Printer) error { - pushstate(m.String()) - for _, a := range text { - p.ApplyDirective(m, gen.IgnoreTypename(a)) - infof("ignoring %s\n", a) - } - popstate() - return nil -} - -// find all comment lines that begin with //msgp: -func yieldComments(c []*ast.CommentGroup) []string { - var out []string - for _, cg := range c { - for _, line := range cg.List { - if strings.HasPrefix(line.Text, linePrefix) { - out = append(out, strings.TrimPrefix(line.Text, linePrefix)) - } - } - } - return out -} - -//msgp:shim {Type} as:{Newtype} using:{toFunc/fromFunc} mode:{Mode} -func applyShim(text []string, f *FileSet) error { - if len(text) < 4 || len(text) > 5 { - return fmt.Errorf("shim directive should have 3 or 4 arguments; found %d", len(text)-1) - } - - name := text[1] - be := gen.Ident("", strings.TrimPrefix(strings.TrimSpace(text[2]), "as:")) // parse as::{base} - if name[0] == '*' { - name = name[1:] - be.Needsref(true) - } - be.Alias(name) - - usestr := strings.TrimPrefix(strings.TrimSpace(text[3]), "using:") // parse using::{method/method} - - methods := strings.Split(usestr, "/") - if len(methods) != 2 { - return fmt.Errorf("expected 2 using::{} methods; found %d (%q)", len(methods), text[3]) - } - - be.ShimToBase = methods[0] - be.ShimFromBase = methods[1] - - if len(text) == 5 { - modestr := strings.TrimPrefix(strings.TrimSpace(text[4]), "mode:") // parse mode::{mode} - switch modestr { - case "cast": - be.ShimMode = gen.Cast - case "convert": - be.ShimMode = gen.Convert - default: - return fmt.Errorf("invalid shim mode; found %s, expected 'cast' or 'convert", modestr) - } - } - - infof("%s -> %s\n", name, be.Value.String()) - f.findShim(name, be) - - return nil -} - -//msgp:ignore {TypeA} {TypeB}... -func ignore(text []string, f *FileSet) error { - if len(text) < 2 { - return nil - } - for _, item := range text[1:] { - name := strings.TrimSpace(item) - if _, ok := f.Identities[name]; ok { - delete(f.Identities, name) - infof("ignoring %s\n", name) - } - } - return nil -} - -//msgp:tuple {TypeA} {TypeB}... -func astuple(text []string, f *FileSet) error { - if len(text) < 2 { - return nil - } - for _, item := range text[1:] { - name := strings.TrimSpace(item) - if el, ok := f.Identities[name]; ok { - if st, ok := el.(*gen.Struct); ok { - st.AsTuple = true - infoln(name) - } else { - warnf("%s: only structs can be tuples\n", name) - } - } - } - return nil -} - -//msgp:sort {Type} {SortInterface} -func sortintf(text []string, f *FileSet) error { - if len(text) != 3 { - return nil - } - sortType := strings.TrimSpace(text[1]) - sortIntf := strings.TrimSpace(text[2]) - gen.SetSortInterface(sortType, sortIntf) - infof("sorting %s using %s\n", sortType, sortIntf) - return nil -} - -//msgp:allocbound {Type} {Bound} -func allocbound(text []string, f *FileSet) error { - if len(text) != 3 { - return nil - } - allocBoundType := strings.TrimSpace(text[1]) - allocBound := strings.TrimSpace(text[2]) - t, ok := f.Identities[allocBoundType] - if !ok { - warnf("allocbound: cannot find type %s\n", allocBoundType) - } else { - t.SetAllocBound(allocBound) - infof("allocbound(%s): setting to %s\n", allocBoundType, allocBound) - } - return nil -} diff --git a/msgp/parse/directives_test.go b/msgp/parse/directives_test.go deleted file mode 100644 index e237290f1b..0000000000 --- a/msgp/parse/directives_test.go +++ /dev/null @@ -1,53 +0,0 @@ -package parse - -import ( - "testing" - - "github.com/algorand/msgp/gen" -) - -const ( - testStructName = "TestStruct" - testFuncName = "callback" -) - -func TestPostunmarshalcheck(t *testing.T) { - st := gen.Struct{ - Fields: nil, - AsTuple: false, - } - - fl := FileSet{ - Identities: map[string]gen.Elem{testStructName: &st}, - Directives: []string{"postunmarshalcheck"}, // raw preprocessor directives - } - if err := postunmarshalcheck([]string{"postunmarshalcheck", testStructName, testFuncName}, &fl); err != nil { - t.Fatal() - } - if testFuncName != st.GetCallbacks()[0].GetName() { - t.Fatal() - } - if !st.GetCallbacks()[0].IsUnmarshallCallback() { - t.Fatal() - } -} - -func TestPostunmarshalcheckFailures(t *testing.T) { - - st := gen.Struct{ - Fields: nil, - AsTuple: false, - } - - fl := FileSet{ - Identities: map[string]gen.Elem{testStructName: &st}, - Directives: []string{"postunmarshalcheck"}, // raw preprocessor directives - } - if err := postunmarshalcheck([]string{"postunmarshalcheck", testFuncName}, &fl); err == nil { - t.Fatal() - } - - if err := postunmarshalcheck([]string{"postunmarshalcheck", "non-existing-type", testFuncName}, &fl); err == nil { - t.Fatal() - } -} diff --git a/msgp/parse/getast.go b/msgp/parse/getast.go deleted file mode 100644 index d0ff38091f..0000000000 --- a/msgp/parse/getast.go +++ /dev/null @@ -1,818 +0,0 @@ -package parse - -import ( - "fmt" - "go/ast" - "reflect" - "sort" - "strings" - - "github.com/algorand/msgp/gen" - "github.com/ttacon/chalk" - "golang.org/x/tools/go/packages" -) - -// A FileSet is the in-memory representation of a -// parsed file. -type FileSet struct { - Package string // package name - PkgPath string // package path - Specs map[string]ast.Expr // type specs in file - Aliases map[string]ast.Expr // type aliases in file - Interfaces map[string]ast.Expr // type interfaces in file - Consts map[string]ast.Expr // consts - Identities map[string]gen.Elem // processed from specs - Directives []string // raw preprocessor directives - Imports []*ast.ImportSpec // imports - ImportSet ImportSet - ImportName map[string]string -} - -// An ImportSet describes the FileSets for a group of imported packages -type ImportSet map[string]*FileSet - -// File parses a file at the relative path -// provided and produces a new *FileSet. -// If you pass in a path to a directory, the entire -// directory will be parsed. -// If unexport is false, only exported identifiers are included in the FileSet. -// If the resulting FileSet would be empty, an error is returned. -func File(name string, unexported bool, warnPkgMask string) (*FileSet, error) { - pushstate(name) - defer popstate() - - cfg := &packages.Config{ - Mode: packages.NeedName | packages.NeedImports | packages.NeedDeps | packages.NeedSyntax | packages.NeedFiles | packages.NeedExportsFile | packages.NeedTypesInfo, - } - - pkgs, err := packages.Load(cfg, name) - if err != nil { - return nil, err - } - - if len(pkgs) != 1 { - return nil, fmt.Errorf("multiple packages in directory: %s", name) - } - - var one *packages.Package - for _, nm := range pkgs { - one = nm - break - } - - imps := make(map[string]*FileSet) - - fs := packageToFileSet(one, imps, unexported) - for _, ifs := range imps { - ifs.process(warnPkgMask) - ifs.applyDirectives() - ifs.propInline() - } - fs.process(warnPkgMask) - fs.applyDirectives() - fs.propInline() - return fs, nil -} - -func packageToFileSet(p *packages.Package, imps map[string]*FileSet, unexported bool) *FileSet { - fs := &FileSet{ - Package: p.Name, - PkgPath: p.PkgPath, - Specs: make(map[string]ast.Expr), - Aliases: make(map[string]ast.Expr), - Interfaces: make(map[string]ast.Expr), - Consts: make(map[string]ast.Expr), - Identities: make(map[string]gen.Elem), - ImportSet: imps, - ImportName: make(map[string]string), - } - - for name, importpkg := range p.Imports { - _, ok := imps[name] - if ok { - continue - } - - imps[name] = packageToFileSet(importpkg, imps, unexported) - } - - for _, fl := range p.Syntax { - pushstate(fl.Name.Name) - fs.Directives = append(fs.Directives, yieldComments(fl.Comments)...) - if !unexported { - ast.FileExports(fl) - } - - for _, importspec := range fl.Imports { - pkgpath := importspec.Path.Value[1 : len(importspec.Path.Value)-1] - if pkgpath == "C" { - continue - } - - var importname string - if importspec.Name != nil { - importname = importspec.Name.Name - } else { - p, ok := imps[pkgpath] - if !ok { - fmt.Printf("missing import %s\n", pkgpath) - } else { - importname = p.Package - } - } - fs.ImportName[importname] = pkgpath - } - - fs.getTypeSpecs(fl) - popstate() - } - - return fs -} - -// applyDirectives applies all of the directives that -// are known to the parser. additional method-specific -// directives remain in f.Directives -func (f *FileSet) applyDirectives() { - newdirs := make([]string, 0, len(f.Directives)) - for _, d := range f.Directives { - chunks := strings.Split(d, " ") - if len(chunks) > 0 { - if fn, ok := directives[chunks[0]]; ok { - pushstate(chunks[0]) - err := fn(chunks, f) - if err != nil { - warnln(err.Error()) - } - popstate() - } else { - newdirs = append(newdirs, d) - } - } - } - f.Directives = newdirs -} - -// A linkset is a graph of unresolved -// identities. -// -// Since gen.Ident can only represent -// one level of type indirection (e.g. Foo -> uint8), -// type declarations like `type Foo Bar` -// aren't resolve-able until we've processed -// everything else. -// -// The goal of this dependency resolution -// is to distill the type declaration -// into just one level of indirection. -// In other words, if we have: -// -// type A uint64 -// type B A -// type C B -// type D C -// -// ... then we want to end up -// figuring out that D is just a uint64. -type linkset map[string]*gen.BaseElem - -func (f *FileSet) resolve(ls linkset) { - progress := true - for progress && len(ls) > 0 { - progress = false - for name, elem := range ls { - real, ok := f.Identities[elem.TypeName()] - if ok { - // copy the old type descriptor, - // alias it to the new value, - // and insert it into the resolved - // identities list - progress = true - nt := real.Copy() - nt.Alias(name) - f.Identities[name] = nt - delete(ls, name) - } - } - } - - // what's left can't be resolved - for name, elem := range ls { - // warnf("couldn't resolve type %s (%s)\n", name, elem.TypeName()) - nt := elem.Copy() - nt.Alias(name) - f.Identities[name] = nt - } -} - -// process takes the contents of f.Specs and -// uses them to populate f.Identities -func (f *FileSet) process(warnPkgMask string) { - if warnPkgMask != "" && !strings.HasPrefix(f.PkgPath, warnPkgMask) { - increasePrintLevel() - defer decreasePrintLevel() - } - deferred := make(linkset) -parse: - for name, def := range f.Specs { - pushstate(name) - - el := f.parseExpr("", def) - - if el == nil { - warnln("failed to parse") - popstate() - continue parse - } - // push unresolved identities into - // the graph of links and resolve after - // we've handled every possible named type. - if be, ok := el.(*gen.BaseElem); ok && be.Value == gen.IDENT { - deferred[name] = be - popstate() - continue parse - } - el.Alias(name) - f.Identities[name] = el - popstate() - } - - if len(deferred) > 0 { - f.resolve(deferred) - } -} - -func strToMethod(s string) gen.Method { - switch s { - case "test": - return gen.Test - case "size": - return gen.Size - case "marshal": - return gen.Marshal - case "unmarshal": - return gen.Unmarshal - case "maxsize": - return gen.MaxSize - default: - return 0 - } -} - -func (f *FileSet) applyDirs(p *gen.Printer) { - // apply directives of the form - // - // //msgp:encode ignore {{TypeName}} - // -loop: - for _, d := range f.Directives { - chunks := strings.Split(d, " ") - if len(chunks) > 1 { - for i := range chunks { - chunks[i] = strings.TrimSpace(chunks[i]) - } - m := strToMethod(chunks[0]) - if m == 0 { - warnf("unknown pass name: %q\n", chunks[0]) - continue loop - } - if fn, ok := passDirectives[chunks[1]]; ok { - pushstate(chunks[1]) - err := fn(m, chunks[2:], p) - if err != nil { - warnf("error applying directive: %s\n", err) - } - popstate() - } else { - warnf("unrecognized directive %q\n", chunks[1]) - } - } else { - warnf("empty directive: %q\n", d) - } - } -} - -func (f *FileSet) PrintTo(p *gen.Printer) error { - var msgs []string - - f.applyDirs(p) - names := make([]string, 0, len(f.Identities)) - for name := range f.Identities { - names = append(names, name) - } - sort.Strings(names) - for _, name := range names { - el := f.Identities[name] - el.SetVarname("z") - pushstate(el.TypeName()) - m, err := p.Print(el) - popstate() - if err != nil { - return err - } - msgs = append(msgs, m...) - } - for _, msg := range msgs { - warnln(msg) - } - if len(msgs) > 0 { - return fmt.Errorf("Errors encountered, exiting") - } - return nil -} - -// getTypeSpecs extracts all of the *ast.TypeSpecs in the file -// into fs.Identities, but does not set the actual element -func (fs *FileSet) getTypeSpecs(f *ast.File) { - - // collect all imports... - fs.Imports = append(fs.Imports, f.Imports...) - - // check all declarations... - for i := range f.Decls { - - // for GenDecls... - if g, ok := f.Decls[i].(*ast.GenDecl); ok { - - // and check the specs... - for _, s := range g.Specs { - - // for ast.TypeSpecs.... - switch s := s.(type) { - case *ast.TypeSpec: - switch s.Type.(type) { - - // this is the list of parse-able - // type specs - case *ast.StructType, - *ast.ArrayType, - *ast.StarExpr, - *ast.SelectorExpr, - *ast.MapType, - *ast.Ident: - - if strings.HasPrefix(s.Name.Name, "_Ctype_") || s.Name.Name == "_" { - continue - } - - if s.Assign == 0 { - fs.Specs[s.Name.Name] = s.Type - } else { - fs.Aliases[s.Name.Name] = s.Type - } - case *ast.InterfaceType: - fs.Interfaces[s.Name.Name] = s.Type - } - - case *ast.ValueSpec: - if len(s.Names) == 1 && len(s.Values) == 1 { - fs.Consts[s.Names[0].Name] = s.Values[0] - } - } - } - } - } -} - -func fieldName(f *ast.Field) string { - switch len(f.Names) { - case 0: - return stringify(f.Type) - case 1: - return f.Names[0].Name - default: - return f.Names[0].Name + " (and others)" - } -} - -func (fs *FileSet) parseFieldList(importPrefix string, fl *ast.FieldList) []gen.StructField { - if fl == nil || fl.NumFields() == 0 { - return nil - } - out := make([]gen.StructField, 0, fl.NumFields()) - for _, field := range fl.List { - pushstate(fieldName(field)) - fds := fs.getField(importPrefix, field) - if len(fds) > 0 { - out = append(out, fds...) - } else { - warnln("ignored.") - } - popstate() - } - return out -} - -// translate *ast.Field into []gen.StructField -func (fs *FileSet) getField(importPrefix string, f *ast.Field) []gen.StructField { - sf := make([]gen.StructField, 1) - var extension, flatten bool - var allocbound string - var allocbounds []string - var maxtotalbytes string - - // always flatten embedded structs - flatten = true - - // parse tag; otherwise field name is field tag - if f.Tag != nil { - var body string - body, sf[0].HasCodecTag = reflect.StructTag(strings.Trim(f.Tag.Value, "`")).Lookup("codec") - tags := strings.Split(body, ",") - for _, tag := range tags[1:] { - if tag == "extension" { - extension = true - } - if strings.HasPrefix(tag, "allocbound=") { - allocbounds = append(allocbounds, strings.Split(tag, "=")[1]) - } - if strings.HasPrefix(tag, "maxtotalbytes=") { - maxtotalbytes = strings.Split(tag, "=")[1] - } - } - // ignore "-" fields - if tags[0] == "-" { - return nil - } - sf[0].FieldTag = tags[0] - sf[0].FieldTagParts = tags - sf[0].RawTag = f.Tag.Value - } - allocbound = strings.Join(allocbounds, ",") - ex := fs.parseExpr(importPrefix, f.Type) - if ex == nil { - return nil - } - - // parse field name - switch len(f.Names) { - case 0: - if flatten { - maybe := fs.getFieldsFromEmbeddedStruct(importPrefix, f.Type) - if maybe != nil { - // Prefix all field names with the explicit - // embedded struct selector, to avoid ambiguity. - for i := range maybe { - maybe[i].FieldPath = append([]string{embedded(f.Type)}, maybe[i].FieldPath...) - } - - return maybe - } - } - - // If not flattening, or the embedded type wasn't a struct, - // embed it under the type name. - sf[0].FieldName = embedded(f.Type) - case 1: - sf[0].FieldName = f.Names[0].Name - default: - // this is for a multiple in-line declaration, - // e.g. type A struct { One, Two int } - sf = sf[0:0] - for _, nm := range f.Names { - sf = append(sf, gen.StructField{ - FieldTag: nm.Name, - FieldName: nm.Name, - FieldElem: ex.Copy(), - }) - } - return sf - } - - // resolve local package type aliases that referenced in this package structs - resolveAlias := func(el gen.Elem) { - if a, ok := fs.Aliases[el.TypeName()]; ok { - if b, ok := a.(*ast.SelectorExpr); ok { - if c, ok := b.X.(*ast.Ident); ok { - el.Alias(c.Name + "." + b.Sel.Name) - } - } else if b, ok := a.(*ast.Ident); ok { - el.Alias(b.Name) - } - } - } - // resolve field alias type - resolveAlias(ex) - // resolve field map type that have alias type key or value - if m, ok := ex.(*gen.Map); ok { - resolveAlias(m.Key) - resolveAlias(m.Value) - } - // resolve field slice type that have alias type element - if m, ok := ex.(*gen.Slice); ok { - resolveAlias(m.Els) - } - - sf[0].FieldElem = ex - if sf[0].FieldTag == "" { - sf[0].FieldTag = sf[0].FieldName - } - if sf[0].FieldTagParts == nil { - sf[0].FieldTagParts = []string{sf[0].FieldName} - } - sf[0].FieldElem.SetAllocBound(allocbound) - sf[0].FieldElem.SetMaxTotalBytes(maxtotalbytes) - - // validate extension - if extension { - switch ex := ex.(type) { - case *gen.Ptr: - if b, ok := ex.Value.(*gen.BaseElem); ok { - b.Value = gen.Ext - } else { - warnln("couldn't cast to extension.") - return nil - } - case *gen.BaseElem: - ex.Value = gen.Ext - default: - warnln("couldn't cast to extension.") - return nil - } - } - return sf -} - -func (fs *FileSet) getFieldsFromEmbeddedStruct(importPrefix string, f ast.Expr) []gen.StructField { - switch f := f.(type) { - case *ast.Ident: - s, ok := fs.Specs[f.Name] - if !ok { - s = fs.Aliases[f.Name] - } - - switch s := s.(type) { - case *ast.StructType: - return fs.parseFieldList(importPrefix, s.Fields) - default: - return nil - } - case *ast.SelectorExpr: - pkg := f.X - pkgid, ok := pkg.(*ast.Ident) - if !ok { - return nil - } - - pkgname, ok := fs.ImportName[pkgid.Name] - if !ok { - return nil - } - - pkgfs, ok := fs.ImportSet[pkgname] - if !ok { - return nil - } - - return pkgfs.getFieldsFromEmbeddedStruct(pkgid.Name+".", f.Sel) - default: - // other possibilities are disallowed - return nil - } -} - -// extract embedded field name -// -// so, for a struct like -// -// type A struct { -// io.Writer -// } -// -// we want "Writer" -func embedded(f ast.Expr) string { - switch f := f.(type) { - case *ast.Ident: - return f.Name - case *ast.StarExpr: - return embedded(f.X) - case *ast.SelectorExpr: - return f.Sel.Name - default: - // other possibilities are disallowed - return "" - } -} - -// stringify a field type name -func stringify(e ast.Expr) string { - switch e := e.(type) { - case *ast.Ident: - return e.Name - case *ast.StarExpr: - return "*" + stringify(e.X) - case *ast.SelectorExpr: - return stringify(e.X) + "." + e.Sel.Name - case *ast.ArrayType: - if e.Len == nil { - return "[]" + stringify(e.Elt) - } - return fmt.Sprintf("[%s]%s", stringify(e.Len), stringify(e.Elt)) - case *ast.InterfaceType: - if e.Methods == nil || e.Methods.NumFields() == 0 { - return "interface{}" - } - } - return "" -} - -// recursively translate ast.Expr to gen.Elem; nil means type not supported -// expected input types: -// - *ast.MapType (map[T]J) -// - *ast.Ident (name) -// - *ast.ArrayType ([(sz)]T) -// - *ast.StarExpr (*T) -// - *ast.StructType (struct {}) -// - *ast.SelectorExpr (a.B) -// - *ast.InterfaceType (interface {}) -func (fs *FileSet) parseExpr(importPrefix string, e ast.Expr) gen.Elem { - switch e := e.(type) { - - case *ast.MapType: - kt := fs.parseExpr(importPrefix, e.Key) - if kt == nil { - return nil - } - - vt := fs.parseExpr(importPrefix, e.Value) - if vt == nil { - return nil - } - - return &gen.Map{Key: kt, Value: vt} - - case *ast.Ident: - b := gen.Ident(importPrefix, e.Name) - - // work to resove this expression - // can be done later, once we've resolved - // everything else. - if b.Value == gen.IDENT { - _, specOK := fs.Specs[e.Name] - _, aliasOK := fs.Aliases[e.Name] - _, interfaceOK := fs.Interfaces[e.Name] - if !specOK && !aliasOK && !interfaceOK { - warnf("non-local identifier: %s\n", e.Name) - } - } - return b - - case *ast.ArrayType: - - // special case for []byte - if e.Len == nil { - if i, ok := e.Elt.(*ast.Ident); ok && i.Name == "byte" { - return &gen.BaseElem{Value: gen.Bytes} - } - } - - // return early if we don't know - // what the slice element type is - els := fs.parseExpr(importPrefix, e.Elt) - if els == nil { - return nil - } - - // array and not a slice - if e.Len != nil { - switch s := e.Len.(type) { - case *ast.BasicLit: - return &gen.Array{ - Size: s.Value, - Els: els, - } - - case *ast.Ident: - sizeHint := "" - if s.Obj != nil && s.Obj.Kind == ast.Con { - switch d := s.Obj.Decl.(type) { - case *ast.ValueSpec: - if len(d.Names) == 1 && len(d.Values) == 1 { - v := d.Values[0] - // Keep trying to resolve this value - repeat := true - for repeat { - switch vv := v.(type) { - case *ast.BasicLit: - sizeHint = vv.Value - repeat = false - case *ast.SelectorExpr: - switch xv := vv.X.(type) { - case *ast.Ident: - pkgpath := fs.ImportName[xv.Name] - pkg := fs.ImportSet[pkgpath] - v = pkg.Consts[vv.Sel.Name] - } - default: - repeat = false - } - } - } - } - } - return &gen.Array{ - Size: s.String(), - SizeHint: sizeHint, - Els: els, - } - - case *ast.SelectorExpr: - return &gen.Array{ - Size: stringify(s), - Els: els, - } - - default: - return nil - } - } - return &gen.Slice{Els: els} - - case *ast.StarExpr: - if v := fs.parseExpr(importPrefix, e.X); v != nil { - return &gen.Ptr{Value: v} - } - return nil - - case *ast.StructType: - return &gen.Struct{Fields: fs.parseFieldList(importPrefix, e.Fields)} - - case *ast.SelectorExpr: - return gen.Ident("", stringify(e)) - - case *ast.InterfaceType: - // support `interface{}` - if len(e.Methods.List) == 0 { - return &gen.BaseElem{Value: gen.Intf} - } - return nil - - default: // other types not supported - return nil - } -} - -func infof(s string, v ...interface{}) { - pushstate(s) - if print(0) { - fmt.Printf(chalk.Green.Color(strings.Join(logctx, ": ")), v...) - } - popstate() -} - -func infoln(s string) { - pushstate(s) - if print(0) { - fmt.Println(chalk.Green.Color(strings.Join(logctx, ": "))) - } - popstate() -} - -func warnf(s string, v ...interface{}) { - pushstate(s) - if print(1) { - fmt.Printf(chalk.Yellow.Color(strings.Join(logctx, ": ")), v...) - } - popstate() -} - -func warnln(s string) { - pushstate(s) - if print(1) { - fmt.Println(chalk.Yellow.Color(strings.Join(logctx, ": "))) - } - popstate() -} - -func fatalf(s string, v ...interface{}) { - pushstate(s) - if print(2) { - fmt.Printf(chalk.Red.Color(strings.Join(logctx, ": ")), v...) - } - popstate() -} - -var logctx []string -var printlevel int - -func increasePrintLevel() { - printlevel++ -} - -func decreasePrintLevel() { - printlevel-- -} - -func print(level int) bool { - return printlevel < level -} - -// push logging state -func pushstate(s string) { - logctx = append(logctx, s) -} - -// pop logging state -func popstate() { - logctx = logctx[:len(logctx)-1] -} diff --git a/msgp/parse/inline.go b/msgp/parse/inline.go deleted file mode 100644 index 0fba91dbf4..0000000000 --- a/msgp/parse/inline.go +++ /dev/null @@ -1,175 +0,0 @@ -package parse - -import ( - "sort" - - "github.com/algorand/msgp/gen" -) - -// This file defines when and how we -// propagate type information from -// one type declaration to another. -// After the processing pass, every -// non-primitive type is marshalled/unmarshalled/etc. -// through a function call. Here, we propagate -// the type information into the caller's type -// tree *if* the child type is simple enough. -// -// For example, types like -// -// type A [4]int -// -// will get pushed into parent methods, -// whereas types like -// -// type B [3]map[string]struct{A, B [4]string} -// -// will not. - -// this is an approximate measure -// of the number of children in a node -const maxComplex = 5 - -// begin recursive search for identities with the -// given name and replace them with be -func (f *FileSet) findShim(id string, be *gen.BaseElem) { - for name, el := range f.Identities { - pushstate(name) - switch el := el.(type) { - case *gen.Struct: - for i := range el.Fields { - f.nextShim(&el.Fields[i].FieldElem, id, be) - } - case *gen.Array: - f.nextShim(&el.Els, id, be) - case *gen.Slice: - f.nextShim(&el.Els, id, be) - case *gen.Map: - f.nextShim(&el.Value, id, be) - case *gen.Ptr: - f.nextShim(&el.Value, id, be) - } - popstate() - } - // we'll need this at the top level as well - f.Identities[id] = be -} - -func (f *FileSet) nextShim(ref *gen.Elem, id string, be *gen.BaseElem) { - if (*ref).TypeName() == id { - vn := (*ref).Varname() - *ref = be.Copy() - (*ref).SetVarname(vn) - } else { - switch el := (*ref).(type) { - case *gen.Struct: - for i := range el.Fields { - f.nextShim(&el.Fields[i].FieldElem, id, be) - } - case *gen.Array: - f.nextShim(&el.Els, id, be) - case *gen.Slice: - f.nextShim(&el.Els, id, be) - case *gen.Map: - f.nextShim(&el.Value, id, be) - case *gen.Ptr: - f.nextShim(&el.Value, id, be) - } - } -} - -// propInline identifies and inlines candidates -func (f *FileSet) propInline() { - type gelem struct { - name string - el gen.Elem - } - - all := make([]gelem, 0, len(f.Identities)) - - for name, el := range f.Identities { - all = append(all, gelem{name: name, el: el}) - } - - // make sure we process inlining determinstically: - // start with the least-complex elems; - // use identifier names as a tie-breaker - sort.Slice(all, func(i, j int) bool { - ig, jg := &all[i], &all[j] - ic, jc := ig.el.Complexity(), jg.el.Complexity() - return ic < jc || (ic == jc && ig.name < jg.name) - }) - - for i := range all { - name := all[i].name - pushstate(name) - switch el := all[i].el.(type) { - case *gen.Struct: - for i := range el.Fields { - f.nextInline(&el.Fields[i].FieldElem, name) - } - case *gen.Array: - f.nextInline(&el.Els, name) - case *gen.Slice: - f.nextInline(&el.Els, name) - case *gen.Map: - f.nextInline(&el.Value, name) - case *gen.Ptr: - f.nextInline(&el.Value, name) - } - popstate() - } -} - -const fatalloop = `detected infinite recursion in inlining loop! -Please file a bug at github.com/tinylib/msgp/issues! -Thanks! -` - -func (f *FileSet) nextInline(ref *gen.Elem, root string) { - switch el := (*ref).(type) { - case *gen.BaseElem: - // ensure that we're not inlining - // a type into itself - typ := el.TypeName() - if el.Value == gen.IDENT && typ != root { - if node, ok := f.Identities[typ]; ok && node.Complexity() < maxComplex { - // infof("inlining %s\n", typ) - - // This should never happen; it will cause - // infinite recursion. - if node == *ref { - panic(fatalloop) - } - - *ref = node.Copy() - f.nextInline(ref, node.TypeName()) - } else if !ok && !el.Resolved() { - // this is the point at which we're sure that - // we've got a type that isn't a primitive, - // a library builtin, or a processed type - - // we correctly handle this case by forwarding - // methods to the unresolved identifier; the - // code will fail to compile if the requisite - // methods aren't available for that type. - - // warnf("unresolved identifier: %s\n", typ) - } - } - case *gen.Struct: - for i := range el.Fields { - f.nextInline(&el.Fields[i].FieldElem, root) - } - case *gen.Array: - f.nextInline(&el.Els, root) - case *gen.Slice: - f.nextInline(&el.Els, root) - case *gen.Map: - f.nextInline(&el.Value, root) - case *gen.Ptr: - f.nextInline(&el.Value, root) - default: - panic("bad elem type") - } -} diff --git a/msgp/printer/print.go b/msgp/printer/print.go deleted file mode 100644 index fb45e1ed41..0000000000 --- a/msgp/printer/print.go +++ /dev/null @@ -1,168 +0,0 @@ -package printer - -import ( - "bytes" - "fmt" - "io" - "io/ioutil" - "strings" - - "github.com/algorand/msgp/gen" - "github.com/algorand/msgp/parse" - "github.com/daixiang0/gci/pkg/gci" - "github.com/daixiang0/gci/pkg/gci/sections" - "github.com/ttacon/chalk" - "golang.org/x/tools/imports" -) - -func infof(s string, v ...interface{}) { - fmt.Printf(chalk.Magenta.Color(s), v...) -} - -// PrintFile prints the methods for the provided list -// of elements to the given file name and canonical -// package path. -func PrintFile(file string, f *parse.FileSet, mode gen.Method, skipFormat bool) error { - out, tests, err := generate(f, mode) - if err != nil { - return err - } - - // we'll run goimports on the main file - // in another goroutine, and run it here - // for the test file. empirically, this - // takes about the same amount of time as - // doing them in serial when GOMAXPROCS=1, - // and faster otherwise. - res := goformat(file, out.Bytes(), skipFormat) - if tests != nil { - testfile := strings.TrimSuffix(file, ".go") + "_test.go" - err = format(testfile, tests.Bytes(), skipFormat) - if err != nil { - return err - } - infof(">>> Wrote and formatted \"%s\"\n", testfile) - } - err = <-res - if err != nil { - return err - } - return nil -} - -func format(file string, data []byte, skipFormat bool) error { - if skipFormat { - return ioutil.WriteFile(file, data, 0600) - } - // first run through goimports (which cleans up unused deps & does gofmt) - out, err := imports.Process(file, data, nil) - if err != nil { - return err - } - if err := ioutil.WriteFile(file, out, 0600); err != nil { - return err - } - // then run through gci to arrange import order - if err := gci.WriteFormattedFiles([]string{file}, gci.GciConfiguration{ - Sections: gci.SectionList{ - sections.StandardPackage{}, - sections.DefaultSection{}, - sections.Prefix{ImportPrefix: "github.com/algorand"}, - sections.Prefix{ImportPrefix: "github.com/algorand/go-algorand"}, - }, - SectionSeparators: gci.SectionList{sections.NewLine{}}, - }); err != nil { - return err - } - return nil -} - -func goformat(file string, data []byte, skipFormat bool) <-chan error { - out := make(chan error, 1) - go func(file string, data []byte, end chan error) { - end <- format(file, data, skipFormat) - infof(">>> Wrote and formatted \"%s\"\n", file) - }(file, data, out) - return out -} - -func dedupImports(imp []string) []string { - m := make(map[string]struct{}) - for i := range imp { - m[imp[i]] = struct{}{} - } - r := []string{} - for k := range m { - r = append(r, k) - } - return r -} - -func generate(f *parse.FileSet, mode gen.Method) (*bytes.Buffer, *bytes.Buffer, error) { - outbuf := bytes.NewBuffer(make([]byte, 0, 4096)) - writePkgHeader(outbuf, f.Package) - - myImports := []string{"github.com/algorand/msgp/msgp"} - for _, imp := range f.Imports { - if imp.Name != nil { - // have an alias, include it. - myImports = append(myImports, imp.Name.Name+` `+imp.Path.Value) - } else { - myImports = append(myImports, imp.Path.Value) - } - } - dedup := dedupImports(myImports) - writeImportHeader(outbuf, dedup...) - - var testbuf *bytes.Buffer - var testwr io.Writer - if mode&gen.Test == gen.Test { - testbuf = bytes.NewBuffer(make([]byte, 0, 4096)) - writeBuildHeader(testbuf, []string{"!skip_msgp_testing"}) - writePkgHeader(testbuf, f.Package) - writeImportHeader( - testbuf, - "github.com/algorand/msgp/msgp", - "github.com/algorand/go-algorand/protocol", - "github.com/algorand/go-algorand/test/partitiontest", - "testing") - testwr = testbuf - } - funcbuf := bytes.NewBuffer(make([]byte, 0, 4096)) - var topics gen.Topics - - err := f.PrintTo(gen.NewPrinter(mode, &topics, funcbuf, testwr)) - if err == nil { - outbuf.Write(topics.Bytes()) - outbuf.Write(funcbuf.Bytes()) - } - return outbuf, testbuf, err -} - -func writePkgHeader(b *bytes.Buffer, name string) { - b.WriteString("package ") - b.WriteString(name) - b.WriteByte('\n') - // write generated code marker - // https://github.com/tinylib/msgp/issues/229 - // https://golang.org/s/generatedcode - b.WriteString("// Code generated by github.com/algorand/msgp DO NOT EDIT.\n\n") -} - -func writeImportHeader(b *bytes.Buffer, imports ...string) { - b.WriteString("import (\n") - for _, im := range imports { - if im[len(im)-1] == '"' { - // support aliased imports - fmt.Fprintf(b, "\t%s\n", im) - } else { - fmt.Fprintf(b, "\t%q\n", im) - } - } - b.WriteString(")\n\n") -} - -func writeBuildHeader(b *bytes.Buffer, buildHeaders []string) { - headers := fmt.Sprintf("//go:build %s\n// +build %s\n\n", strings.Join(buildHeaders, " "), strings.Join(buildHeaders, " ")) - b.WriteString(headers) -} diff --git a/msgp/printer/print_test.go b/msgp/printer/print_test.go deleted file mode 100644 index 112fa10a74..0000000000 --- a/msgp/printer/print_test.go +++ /dev/null @@ -1,19 +0,0 @@ -package printer - -import ( - "bytes" - "testing" -) - -func TestWriteBuildHeader(t *testing.T) { - testBuf := bytes.NewBuffer(make([]byte, 0, 4096)) - buildHeaders := []string{"foobar"} - expectedBuf := bytes.NewBuffer(make([]byte, 0, 4096)) - expectedBuf.WriteString("//go:build foobar\n// +build foobar\n\n") - - writeBuildHeader(testBuf, buildHeaders) - - if testBuf.String() != expectedBuf.String() { - t.Errorf("testBuf:\n%s not equal to expectedBuf:\n%s", testBuf, expectedBuf) - } -} diff --git a/scripts/check_license.sh b/scripts/check_license.sh index 9e40983f93..51f83d7d4f 100755 --- a/scripts/check_license.sh +++ b/scripts/check_license.sh @@ -4,7 +4,7 @@ PROJECT_ROOT=$(git rev-parse --show-toplevel) LICENSE_LOCATION="$PROJECT_ROOT"/scripts/LICENSE_HEADER NUMLINES=$(< "$LICENSE_LOCATION" wc -l | tr -d ' ') LICENSE=$(sed "s/{DATE_Y}/$(date +"%Y")/" "$LICENSE_LOCATION") -VERSIONED_GO_FILES=$(git ls-tree --full-tree --name-only -r HEAD | grep "\.go$" | grep -v "^msgp/") +VERSIONED_GO_FILES=$(git ls-tree --full-tree --name-only -r HEAD | grep "\.go$") EXCLUDE=( "Code generated by" "David Lazar" diff --git a/tools/block-generator/go.mod b/tools/block-generator/go.mod index 6508fea45a..139c8fa164 100644 --- a/tools/block-generator/go.mod +++ b/tools/block-generator/go.mod @@ -1,7 +1,6 @@ module github.com/algorand/go-algorand/tools/block-generator replace github.com/algorand/go-algorand => ../.. -replace github.com/algorand/msgp => ../../msgp go 1.20 diff --git a/tools/x-repo-types/go.mod b/tools/x-repo-types/go.mod index 7427bc676e..df9ddf0114 100644 --- a/tools/x-repo-types/go.mod +++ b/tools/x-repo-types/go.mod @@ -3,7 +3,6 @@ module github.com/algorand/go-algorand/tools/x-repo-types go 1.20 replace github.com/algorand/go-algorand => ../.. -replace github.com/algorand/msgp => ../../msgp require ( github.com/algorand/go-algorand v0.0.0