456 lines
12 KiB
Go
456 lines
12 KiB
Go
|
// Package text supports marshaling Cap'n Proto messages as text based on a schema.
|
||
|
package text
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"math"
|
||
|
"strconv"
|
||
|
|
||
|
"zombiezen.com/go/capnproto2"
|
||
|
"zombiezen.com/go/capnproto2/internal/nodemap"
|
||
|
"zombiezen.com/go/capnproto2/internal/schema"
|
||
|
"zombiezen.com/go/capnproto2/internal/strquote"
|
||
|
"zombiezen.com/go/capnproto2/schemas"
|
||
|
)
|
||
|
|
||
|
// Marker strings.
|
||
|
const (
|
||
|
voidMarker = "void"
|
||
|
interfaceMarker = "<external capability>"
|
||
|
anyPointerMarker = "<opaque pointer>"
|
||
|
)
|
||
|
|
||
|
// Marshal returns the text representation of a struct.
|
||
|
func Marshal(typeID uint64, s capnp.Struct) (string, error) {
|
||
|
buf := new(bytes.Buffer)
|
||
|
if err := NewEncoder(buf).Encode(typeID, s); err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
return buf.String(), nil
|
||
|
}
|
||
|
|
||
|
// MarshalList returns the text representation of a struct list.
|
||
|
func MarshalList(typeID uint64, l capnp.List) (string, error) {
|
||
|
buf := new(bytes.Buffer)
|
||
|
if err := NewEncoder(buf).EncodeList(typeID, l); err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
return buf.String(), nil
|
||
|
}
|
||
|
|
||
|
// An Encoder writes the text format of Cap'n Proto messages to an output stream.
|
||
|
type Encoder struct {
|
||
|
w errWriter
|
||
|
tmp []byte
|
||
|
nodes nodemap.Map
|
||
|
}
|
||
|
|
||
|
// NewEncoder returns a new encoder that writes to w.
|
||
|
func NewEncoder(w io.Writer) *Encoder {
|
||
|
return &Encoder{w: errWriter{w: w}}
|
||
|
}
|
||
|
|
||
|
// UseRegistry changes the registry that the encoder consults for
|
||
|
// schemas from the default registry.
|
||
|
func (enc *Encoder) UseRegistry(reg *schemas.Registry) {
|
||
|
enc.nodes.UseRegistry(reg)
|
||
|
}
|
||
|
|
||
|
// Encode writes the text representation of s to the stream.
|
||
|
func (enc *Encoder) Encode(typeID uint64, s capnp.Struct) error {
|
||
|
if enc.w.err != nil {
|
||
|
return enc.w.err
|
||
|
}
|
||
|
err := enc.marshalStruct(typeID, s)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
return enc.w.err
|
||
|
}
|
||
|
|
||
|
// EncodeList writes the text representation of struct list l to the stream.
|
||
|
func (enc *Encoder) EncodeList(typeID uint64, l capnp.List) error {
|
||
|
_, seg, _ := capnp.NewMessage(capnp.SingleSegment(nil))
|
||
|
typ, _ := schema.NewRootType(seg)
|
||
|
typ.SetStructType()
|
||
|
typ.StructType().SetTypeId(typeID)
|
||
|
return enc.marshalList(typ, l)
|
||
|
}
|
||
|
|
||
|
func (enc *Encoder) marshalBool(v bool) {
|
||
|
if v {
|
||
|
enc.w.WriteString("true")
|
||
|
} else {
|
||
|
enc.w.WriteString("false")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (enc *Encoder) marshalInt(i int64) {
|
||
|
enc.tmp = strconv.AppendInt(enc.tmp[:0], i, 10)
|
||
|
enc.w.Write(enc.tmp)
|
||
|
}
|
||
|
|
||
|
func (enc *Encoder) marshalUint(i uint64) {
|
||
|
enc.tmp = strconv.AppendUint(enc.tmp[:0], i, 10)
|
||
|
enc.w.Write(enc.tmp)
|
||
|
}
|
||
|
|
||
|
func (enc *Encoder) marshalFloat32(f float32) {
|
||
|
enc.tmp = strconv.AppendFloat(enc.tmp[:0], float64(f), 'g', -1, 32)
|
||
|
enc.w.Write(enc.tmp)
|
||
|
}
|
||
|
|
||
|
func (enc *Encoder) marshalFloat64(f float64) {
|
||
|
enc.tmp = strconv.AppendFloat(enc.tmp[:0], f, 'g', -1, 64)
|
||
|
enc.w.Write(enc.tmp)
|
||
|
}
|
||
|
|
||
|
func (enc *Encoder) marshalText(t []byte) {
|
||
|
enc.tmp = strquote.Append(enc.tmp[:0], t)
|
||
|
enc.w.Write(enc.tmp)
|
||
|
}
|
||
|
|
||
|
func needsEscape(b byte) bool {
|
||
|
return b < 0x20 || b >= 0x7f
|
||
|
}
|
||
|
|
||
|
func hexDigit(b byte) byte {
|
||
|
const digits = "0123456789abcdef"
|
||
|
return digits[b]
|
||
|
}
|
||
|
|
||
|
func (enc *Encoder) marshalStruct(typeID uint64, s capnp.Struct) error {
|
||
|
n, err := enc.nodes.Find(typeID)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
if !n.IsValid() || n.Which() != schema.Node_Which_structNode {
|
||
|
return fmt.Errorf("cannot find struct type %#x", typeID)
|
||
|
}
|
||
|
var discriminant uint16
|
||
|
if n.StructNode().DiscriminantCount() > 0 {
|
||
|
discriminant = s.Uint16(capnp.DataOffset(n.StructNode().DiscriminantOffset() * 2))
|
||
|
}
|
||
|
enc.w.WriteByte('(')
|
||
|
fields := codeOrderFields(n.StructNode())
|
||
|
first := true
|
||
|
for _, f := range fields {
|
||
|
if !(f.Which() == schema.Field_Which_slot || f.Which() == schema.Field_Which_group) {
|
||
|
continue
|
||
|
}
|
||
|
if dv := f.DiscriminantValue(); !(dv == schema.Field_noDiscriminant || dv == discriminant) {
|
||
|
continue
|
||
|
}
|
||
|
if !first {
|
||
|
enc.w.WriteString(", ")
|
||
|
}
|
||
|
first = false
|
||
|
name, err := f.NameBytes()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
enc.w.Write(name)
|
||
|
enc.w.WriteString(" = ")
|
||
|
switch f.Which() {
|
||
|
case schema.Field_Which_slot:
|
||
|
if err := enc.marshalFieldValue(s, f); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
case schema.Field_Which_group:
|
||
|
if err := enc.marshalStruct(f.Group().TypeId(), s); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
enc.w.WriteByte(')')
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (enc *Encoder) marshalFieldValue(s capnp.Struct, f schema.Field) error {
|
||
|
typ, err := f.Slot().Type()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
dv, err := f.Slot().DefaultValue()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
if dv.IsValid() && int(typ.Which()) != int(dv.Which()) {
|
||
|
name, _ := f.Name()
|
||
|
return fmt.Errorf("marshal field %s: default value is a %v, want %v", name, dv.Which(), typ.Which())
|
||
|
}
|
||
|
switch typ.Which() {
|
||
|
case schema.Type_Which_void:
|
||
|
enc.w.WriteString(voidMarker)
|
||
|
case schema.Type_Which_bool:
|
||
|
v := s.Bit(capnp.BitOffset(f.Slot().Offset()))
|
||
|
d := dv.Bool()
|
||
|
enc.marshalBool(!d && v || d && !v)
|
||
|
case schema.Type_Which_int8:
|
||
|
v := s.Uint8(capnp.DataOffset(f.Slot().Offset()))
|
||
|
d := uint8(dv.Int8())
|
||
|
enc.marshalInt(int64(int8(v ^ d)))
|
||
|
case schema.Type_Which_int16:
|
||
|
v := s.Uint16(capnp.DataOffset(f.Slot().Offset() * 2))
|
||
|
d := uint16(dv.Int16())
|
||
|
enc.marshalInt(int64(int16(v ^ d)))
|
||
|
case schema.Type_Which_int32:
|
||
|
v := s.Uint32(capnp.DataOffset(f.Slot().Offset() * 4))
|
||
|
d := uint32(dv.Int32())
|
||
|
enc.marshalInt(int64(int32(v ^ d)))
|
||
|
case schema.Type_Which_int64:
|
||
|
v := s.Uint64(capnp.DataOffset(f.Slot().Offset() * 8))
|
||
|
d := uint64(dv.Int64())
|
||
|
enc.marshalInt(int64(v ^ d))
|
||
|
case schema.Type_Which_uint8:
|
||
|
v := s.Uint8(capnp.DataOffset(f.Slot().Offset()))
|
||
|
d := dv.Uint8()
|
||
|
enc.marshalUint(uint64(v ^ d))
|
||
|
case schema.Type_Which_uint16:
|
||
|
v := s.Uint16(capnp.DataOffset(f.Slot().Offset() * 2))
|
||
|
d := dv.Uint16()
|
||
|
enc.marshalUint(uint64(v ^ d))
|
||
|
case schema.Type_Which_uint32:
|
||
|
v := s.Uint32(capnp.DataOffset(f.Slot().Offset() * 4))
|
||
|
d := dv.Uint32()
|
||
|
enc.marshalUint(uint64(v ^ d))
|
||
|
case schema.Type_Which_uint64:
|
||
|
v := s.Uint64(capnp.DataOffset(f.Slot().Offset() * 8))
|
||
|
d := dv.Uint64()
|
||
|
enc.marshalUint(v ^ d)
|
||
|
case schema.Type_Which_float32:
|
||
|
v := s.Uint32(capnp.DataOffset(f.Slot().Offset() * 4))
|
||
|
d := math.Float32bits(dv.Float32())
|
||
|
enc.marshalFloat32(math.Float32frombits(v ^ d))
|
||
|
case schema.Type_Which_float64:
|
||
|
v := s.Uint64(capnp.DataOffset(f.Slot().Offset() * 8))
|
||
|
d := math.Float64bits(dv.Float64())
|
||
|
enc.marshalFloat64(math.Float64frombits(v ^ d))
|
||
|
case schema.Type_Which_structType:
|
||
|
p, err := s.Ptr(uint16(f.Slot().Offset()))
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
if !p.IsValid() {
|
||
|
p, _ = dv.StructValuePtr()
|
||
|
}
|
||
|
return enc.marshalStruct(typ.StructType().TypeId(), p.Struct())
|
||
|
case schema.Type_Which_data:
|
||
|
p, err := s.Ptr(uint16(f.Slot().Offset()))
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
if !p.IsValid() {
|
||
|
b, _ := dv.Data()
|
||
|
enc.marshalText(b)
|
||
|
return nil
|
||
|
}
|
||
|
enc.marshalText(p.Data())
|
||
|
case schema.Type_Which_text:
|
||
|
p, err := s.Ptr(uint16(f.Slot().Offset()))
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
if !p.IsValid() {
|
||
|
b, _ := dv.TextBytes()
|
||
|
enc.marshalText(b)
|
||
|
return nil
|
||
|
}
|
||
|
enc.marshalText(p.TextBytes())
|
||
|
case schema.Type_Which_list:
|
||
|
elem, err := typ.List().ElementType()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
p, err := s.Ptr(uint16(f.Slot().Offset()))
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
if !p.IsValid() {
|
||
|
p, _ = dv.ListPtr()
|
||
|
}
|
||
|
return enc.marshalList(elem, p.List())
|
||
|
case schema.Type_Which_enum:
|
||
|
v := s.Uint16(capnp.DataOffset(f.Slot().Offset() * 2))
|
||
|
d := dv.Uint16()
|
||
|
return enc.marshalEnum(typ.Enum().TypeId(), v^d)
|
||
|
case schema.Type_Which_interface:
|
||
|
enc.w.WriteString(interfaceMarker)
|
||
|
case schema.Type_Which_anyPointer:
|
||
|
enc.w.WriteString(anyPointerMarker)
|
||
|
default:
|
||
|
return fmt.Errorf("unknown field type %v", typ.Which())
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func codeOrderFields(s schema.Node_structNode) []schema.Field {
|
||
|
list, _ := s.Fields()
|
||
|
n := list.Len()
|
||
|
fields := make([]schema.Field, n)
|
||
|
for i := 0; i < n; i++ {
|
||
|
f := list.At(i)
|
||
|
fields[f.CodeOrder()] = f
|
||
|
}
|
||
|
return fields
|
||
|
}
|
||
|
|
||
|
func (enc *Encoder) marshalList(elem schema.Type, l capnp.List) error {
|
||
|
switch elem.Which() {
|
||
|
case schema.Type_Which_void:
|
||
|
enc.w.WriteString(capnp.VoidList{List: l}.String())
|
||
|
case schema.Type_Which_bool:
|
||
|
enc.w.WriteString(capnp.BitList{List: l}.String())
|
||
|
case schema.Type_Which_int8:
|
||
|
enc.w.WriteString(capnp.Int8List{List: l}.String())
|
||
|
case schema.Type_Which_int16:
|
||
|
enc.w.WriteString(capnp.Int16List{List: l}.String())
|
||
|
case schema.Type_Which_int32:
|
||
|
enc.w.WriteString(capnp.Int32List{List: l}.String())
|
||
|
case schema.Type_Which_int64:
|
||
|
enc.w.WriteString(capnp.Int64List{List: l}.String())
|
||
|
case schema.Type_Which_uint8:
|
||
|
enc.w.WriteString(capnp.UInt8List{List: l}.String())
|
||
|
case schema.Type_Which_uint16:
|
||
|
enc.w.WriteString(capnp.UInt16List{List: l}.String())
|
||
|
case schema.Type_Which_uint32:
|
||
|
enc.w.WriteString(capnp.UInt32List{List: l}.String())
|
||
|
case schema.Type_Which_uint64:
|
||
|
enc.w.WriteString(capnp.UInt64List{List: l}.String())
|
||
|
case schema.Type_Which_float32:
|
||
|
enc.w.WriteString(capnp.Float32List{List: l}.String())
|
||
|
case schema.Type_Which_float64:
|
||
|
enc.w.WriteString(capnp.Float64List{List: l}.String())
|
||
|
case schema.Type_Which_data:
|
||
|
enc.w.WriteString(capnp.DataList{List: l}.String())
|
||
|
case schema.Type_Which_text:
|
||
|
enc.w.WriteString(capnp.TextList{List: l}.String())
|
||
|
case schema.Type_Which_structType:
|
||
|
enc.w.WriteByte('[')
|
||
|
for i := 0; i < l.Len(); i++ {
|
||
|
if i > 0 {
|
||
|
enc.w.WriteString(", ")
|
||
|
}
|
||
|
err := enc.marshalStruct(elem.StructType().TypeId(), l.Struct(i))
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
enc.w.WriteByte(']')
|
||
|
case schema.Type_Which_list:
|
||
|
enc.w.WriteByte('[')
|
||
|
ee, err := elem.List().ElementType()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
for i := 0; i < l.Len(); i++ {
|
||
|
if i > 0 {
|
||
|
enc.w.WriteString(", ")
|
||
|
}
|
||
|
p, err := capnp.PointerList{List: l}.PtrAt(i)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
err = enc.marshalList(ee, p.List())
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
enc.w.WriteByte(']')
|
||
|
case schema.Type_Which_enum:
|
||
|
enc.w.WriteByte('[')
|
||
|
il := capnp.UInt16List{List: l}
|
||
|
typ := elem.Enum().TypeId()
|
||
|
// TODO(light): only search for node once
|
||
|
for i := 0; i < il.Len(); i++ {
|
||
|
if i > 0 {
|
||
|
enc.w.WriteString(", ")
|
||
|
}
|
||
|
enc.marshalEnum(typ, il.At(i))
|
||
|
}
|
||
|
enc.w.WriteByte(']')
|
||
|
case schema.Type_Which_interface:
|
||
|
enc.w.WriteByte('[')
|
||
|
for i := 0; i < l.Len(); i++ {
|
||
|
if i > 0 {
|
||
|
enc.w.WriteString(", ")
|
||
|
}
|
||
|
enc.w.WriteString(interfaceMarker)
|
||
|
}
|
||
|
enc.w.WriteByte(']')
|
||
|
case schema.Type_Which_anyPointer:
|
||
|
enc.w.WriteByte('[')
|
||
|
for i := 0; i < l.Len(); i++ {
|
||
|
if i > 0 {
|
||
|
enc.w.WriteString(", ")
|
||
|
}
|
||
|
enc.w.WriteString(anyPointerMarker)
|
||
|
}
|
||
|
enc.w.WriteByte(']')
|
||
|
default:
|
||
|
return fmt.Errorf("unknown list type %v", elem.Which())
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (enc *Encoder) marshalEnum(typ uint64, val uint16) error {
|
||
|
n, err := enc.nodes.Find(typ)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
if n.Which() != schema.Node_Which_enum {
|
||
|
return fmt.Errorf("marshaling enum of type @%#x: type is not an enum", typ)
|
||
|
}
|
||
|
enums, err := n.Enum().Enumerants()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
if int(val) >= enums.Len() {
|
||
|
enc.marshalUint(uint64(val))
|
||
|
return nil
|
||
|
}
|
||
|
name, err := enums.At(int(val)).NameBytes()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
enc.w.Write(name)
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
type errWriter struct {
|
||
|
w io.Writer
|
||
|
err error
|
||
|
}
|
||
|
|
||
|
func (ew *errWriter) Write(p []byte) (int, error) {
|
||
|
if ew.err != nil {
|
||
|
return 0, ew.err
|
||
|
}
|
||
|
var n int
|
||
|
n, ew.err = ew.w.Write(p)
|
||
|
return n, ew.err
|
||
|
}
|
||
|
|
||
|
func (ew *errWriter) WriteString(s string) (int, error) {
|
||
|
if ew.err != nil {
|
||
|
return 0, ew.err
|
||
|
}
|
||
|
var n int
|
||
|
n, ew.err = io.WriteString(ew.w, s)
|
||
|
return n, ew.err
|
||
|
}
|
||
|
|
||
|
func (ew *errWriter) WriteByte(b byte) error {
|
||
|
if ew.err != nil {
|
||
|
return ew.err
|
||
|
}
|
||
|
if bw, ok := ew.w.(io.ByteWriter); ok {
|
||
|
ew.err = bw.WriteByte(b)
|
||
|
} else {
|
||
|
_, ew.err = ew.w.Write([]byte{b})
|
||
|
}
|
||
|
return ew.err
|
||
|
}
|