mirror of
https://github.com/mjl-/mox.git
synced 2025-01-15 18:06:27 +03:00
561 lines
13 KiB
Go
561 lines
13 KiB
Go
|
// Copyright 2010 The Go Authors. All rights reserved.
|
||
|
// Use of this source code is governed by a BSD-style
|
||
|
// license that can be found in the LICENSE file.
|
||
|
|
||
|
package proto
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"encoding"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"math"
|
||
|
"sort"
|
||
|
"strings"
|
||
|
|
||
|
"google.golang.org/protobuf/encoding/prototext"
|
||
|
"google.golang.org/protobuf/encoding/protowire"
|
||
|
"google.golang.org/protobuf/proto"
|
||
|
"google.golang.org/protobuf/reflect/protoreflect"
|
||
|
"google.golang.org/protobuf/reflect/protoregistry"
|
||
|
)
|
||
|
|
||
|
const wrapTextMarshalV2 = false
|
||
|
|
||
|
// TextMarshaler is a configurable text format marshaler.
|
||
|
type TextMarshaler struct {
|
||
|
Compact bool // use compact text format (one line)
|
||
|
ExpandAny bool // expand google.protobuf.Any messages of known types
|
||
|
}
|
||
|
|
||
|
// Marshal writes the proto text format of m to w.
|
||
|
func (tm *TextMarshaler) Marshal(w io.Writer, m Message) error {
|
||
|
b, err := tm.marshal(m)
|
||
|
if len(b) > 0 {
|
||
|
if _, err := w.Write(b); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
// Text returns a proto text formatted string of m.
|
||
|
func (tm *TextMarshaler) Text(m Message) string {
|
||
|
b, _ := tm.marshal(m)
|
||
|
return string(b)
|
||
|
}
|
||
|
|
||
|
func (tm *TextMarshaler) marshal(m Message) ([]byte, error) {
|
||
|
mr := MessageReflect(m)
|
||
|
if mr == nil || !mr.IsValid() {
|
||
|
return []byte("<nil>"), nil
|
||
|
}
|
||
|
|
||
|
if wrapTextMarshalV2 {
|
||
|
if m, ok := m.(encoding.TextMarshaler); ok {
|
||
|
return m.MarshalText()
|
||
|
}
|
||
|
|
||
|
opts := prototext.MarshalOptions{
|
||
|
AllowPartial: true,
|
||
|
EmitUnknown: true,
|
||
|
}
|
||
|
if !tm.Compact {
|
||
|
opts.Indent = " "
|
||
|
}
|
||
|
if !tm.ExpandAny {
|
||
|
opts.Resolver = (*protoregistry.Types)(nil)
|
||
|
}
|
||
|
return opts.Marshal(mr.Interface())
|
||
|
} else {
|
||
|
w := &textWriter{
|
||
|
compact: tm.Compact,
|
||
|
expandAny: tm.ExpandAny,
|
||
|
complete: true,
|
||
|
}
|
||
|
|
||
|
if m, ok := m.(encoding.TextMarshaler); ok {
|
||
|
b, err := m.MarshalText()
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
w.Write(b)
|
||
|
return w.buf, nil
|
||
|
}
|
||
|
|
||
|
err := w.writeMessage(mr)
|
||
|
return w.buf, err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var (
|
||
|
defaultTextMarshaler = TextMarshaler{}
|
||
|
compactTextMarshaler = TextMarshaler{Compact: true}
|
||
|
)
|
||
|
|
||
|
// MarshalText writes the proto text format of m to w.
|
||
|
func MarshalText(w io.Writer, m Message) error { return defaultTextMarshaler.Marshal(w, m) }
|
||
|
|
||
|
// MarshalTextString returns a proto text formatted string of m.
|
||
|
func MarshalTextString(m Message) string { return defaultTextMarshaler.Text(m) }
|
||
|
|
||
|
// CompactText writes the compact proto text format of m to w.
|
||
|
func CompactText(w io.Writer, m Message) error { return compactTextMarshaler.Marshal(w, m) }
|
||
|
|
||
|
// CompactTextString returns a compact proto text formatted string of m.
|
||
|
func CompactTextString(m Message) string { return compactTextMarshaler.Text(m) }
|
||
|
|
||
|
var (
|
||
|
newline = []byte("\n")
|
||
|
endBraceNewline = []byte("}\n")
|
||
|
posInf = []byte("inf")
|
||
|
negInf = []byte("-inf")
|
||
|
nan = []byte("nan")
|
||
|
)
|
||
|
|
||
|
// textWriter is an io.Writer that tracks its indentation level.
|
||
|
type textWriter struct {
|
||
|
compact bool // same as TextMarshaler.Compact
|
||
|
expandAny bool // same as TextMarshaler.ExpandAny
|
||
|
complete bool // whether the current position is a complete line
|
||
|
indent int // indentation level; never negative
|
||
|
buf []byte
|
||
|
}
|
||
|
|
||
|
func (w *textWriter) Write(p []byte) (n int, _ error) {
|
||
|
newlines := bytes.Count(p, newline)
|
||
|
if newlines == 0 {
|
||
|
if !w.compact && w.complete {
|
||
|
w.writeIndent()
|
||
|
}
|
||
|
w.buf = append(w.buf, p...)
|
||
|
w.complete = false
|
||
|
return len(p), nil
|
||
|
}
|
||
|
|
||
|
frags := bytes.SplitN(p, newline, newlines+1)
|
||
|
if w.compact {
|
||
|
for i, frag := range frags {
|
||
|
if i > 0 {
|
||
|
w.buf = append(w.buf, ' ')
|
||
|
n++
|
||
|
}
|
||
|
w.buf = append(w.buf, frag...)
|
||
|
n += len(frag)
|
||
|
}
|
||
|
return n, nil
|
||
|
}
|
||
|
|
||
|
for i, frag := range frags {
|
||
|
if w.complete {
|
||
|
w.writeIndent()
|
||
|
}
|
||
|
w.buf = append(w.buf, frag...)
|
||
|
n += len(frag)
|
||
|
if i+1 < len(frags) {
|
||
|
w.buf = append(w.buf, '\n')
|
||
|
n++
|
||
|
}
|
||
|
}
|
||
|
w.complete = len(frags[len(frags)-1]) == 0
|
||
|
return n, nil
|
||
|
}
|
||
|
|
||
|
func (w *textWriter) WriteByte(c byte) error {
|
||
|
if w.compact && c == '\n' {
|
||
|
c = ' '
|
||
|
}
|
||
|
if !w.compact && w.complete {
|
||
|
w.writeIndent()
|
||
|
}
|
||
|
w.buf = append(w.buf, c)
|
||
|
w.complete = c == '\n'
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (w *textWriter) writeName(fd protoreflect.FieldDescriptor) {
|
||
|
if !w.compact && w.complete {
|
||
|
w.writeIndent()
|
||
|
}
|
||
|
w.complete = false
|
||
|
|
||
|
if fd.Kind() != protoreflect.GroupKind {
|
||
|
w.buf = append(w.buf, fd.Name()...)
|
||
|
w.WriteByte(':')
|
||
|
} else {
|
||
|
// Use message type name for group field name.
|
||
|
w.buf = append(w.buf, fd.Message().Name()...)
|
||
|
}
|
||
|
|
||
|
if !w.compact {
|
||
|
w.WriteByte(' ')
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func requiresQuotes(u string) bool {
|
||
|
// When type URL contains any characters except [0-9A-Za-z./\-]*, it must be quoted.
|
||
|
for _, ch := range u {
|
||
|
switch {
|
||
|
case ch == '.' || ch == '/' || ch == '_':
|
||
|
continue
|
||
|
case '0' <= ch && ch <= '9':
|
||
|
continue
|
||
|
case 'A' <= ch && ch <= 'Z':
|
||
|
continue
|
||
|
case 'a' <= ch && ch <= 'z':
|
||
|
continue
|
||
|
default:
|
||
|
return true
|
||
|
}
|
||
|
}
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
// writeProto3Any writes an expanded google.protobuf.Any message.
|
||
|
//
|
||
|
// It returns (false, nil) if sv value can't be unmarshaled (e.g. because
|
||
|
// required messages are not linked in).
|
||
|
//
|
||
|
// It returns (true, error) when sv was written in expanded format or an error
|
||
|
// was encountered.
|
||
|
func (w *textWriter) writeProto3Any(m protoreflect.Message) (bool, error) {
|
||
|
md := m.Descriptor()
|
||
|
fdURL := md.Fields().ByName("type_url")
|
||
|
fdVal := md.Fields().ByName("value")
|
||
|
|
||
|
url := m.Get(fdURL).String()
|
||
|
mt, err := protoregistry.GlobalTypes.FindMessageByURL(url)
|
||
|
if err != nil {
|
||
|
return false, nil
|
||
|
}
|
||
|
|
||
|
b := m.Get(fdVal).Bytes()
|
||
|
m2 := mt.New()
|
||
|
if err := proto.Unmarshal(b, m2.Interface()); err != nil {
|
||
|
return false, nil
|
||
|
}
|
||
|
w.Write([]byte("["))
|
||
|
if requiresQuotes(url) {
|
||
|
w.writeQuotedString(url)
|
||
|
} else {
|
||
|
w.Write([]byte(url))
|
||
|
}
|
||
|
if w.compact {
|
||
|
w.Write([]byte("]:<"))
|
||
|
} else {
|
||
|
w.Write([]byte("]: <\n"))
|
||
|
w.indent++
|
||
|
}
|
||
|
if err := w.writeMessage(m2); err != nil {
|
||
|
return true, err
|
||
|
}
|
||
|
if w.compact {
|
||
|
w.Write([]byte("> "))
|
||
|
} else {
|
||
|
w.indent--
|
||
|
w.Write([]byte(">\n"))
|
||
|
}
|
||
|
return true, nil
|
||
|
}
|
||
|
|
||
|
func (w *textWriter) writeMessage(m protoreflect.Message) error {
|
||
|
md := m.Descriptor()
|
||
|
if w.expandAny && md.FullName() == "google.protobuf.Any" {
|
||
|
if canExpand, err := w.writeProto3Any(m); canExpand {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
fds := md.Fields()
|
||
|
for i := 0; i < fds.Len(); {
|
||
|
fd := fds.Get(i)
|
||
|
if od := fd.ContainingOneof(); od != nil {
|
||
|
fd = m.WhichOneof(od)
|
||
|
i += od.Fields().Len()
|
||
|
} else {
|
||
|
i++
|
||
|
}
|
||
|
if fd == nil || !m.Has(fd) {
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
switch {
|
||
|
case fd.IsList():
|
||
|
lv := m.Get(fd).List()
|
||
|
for j := 0; j < lv.Len(); j++ {
|
||
|
w.writeName(fd)
|
||
|
v := lv.Get(j)
|
||
|
if err := w.writeSingularValue(v, fd); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
w.WriteByte('\n')
|
||
|
}
|
||
|
case fd.IsMap():
|
||
|
kfd := fd.MapKey()
|
||
|
vfd := fd.MapValue()
|
||
|
mv := m.Get(fd).Map()
|
||
|
|
||
|
type entry struct{ key, val protoreflect.Value }
|
||
|
var entries []entry
|
||
|
mv.Range(func(k protoreflect.MapKey, v protoreflect.Value) bool {
|
||
|
entries = append(entries, entry{k.Value(), v})
|
||
|
return true
|
||
|
})
|
||
|
sort.Slice(entries, func(i, j int) bool {
|
||
|
switch kfd.Kind() {
|
||
|
case protoreflect.BoolKind:
|
||
|
return !entries[i].key.Bool() && entries[j].key.Bool()
|
||
|
case protoreflect.Int32Kind, protoreflect.Sint32Kind, protoreflect.Sfixed32Kind, protoreflect.Int64Kind, protoreflect.Sint64Kind, protoreflect.Sfixed64Kind:
|
||
|
return entries[i].key.Int() < entries[j].key.Int()
|
||
|
case protoreflect.Uint32Kind, protoreflect.Fixed32Kind, protoreflect.Uint64Kind, protoreflect.Fixed64Kind:
|
||
|
return entries[i].key.Uint() < entries[j].key.Uint()
|
||
|
case protoreflect.StringKind:
|
||
|
return entries[i].key.String() < entries[j].key.String()
|
||
|
default:
|
||
|
panic("invalid kind")
|
||
|
}
|
||
|
})
|
||
|
for _, entry := range entries {
|
||
|
w.writeName(fd)
|
||
|
w.WriteByte('<')
|
||
|
if !w.compact {
|
||
|
w.WriteByte('\n')
|
||
|
}
|
||
|
w.indent++
|
||
|
w.writeName(kfd)
|
||
|
if err := w.writeSingularValue(entry.key, kfd); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
w.WriteByte('\n')
|
||
|
w.writeName(vfd)
|
||
|
if err := w.writeSingularValue(entry.val, vfd); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
w.WriteByte('\n')
|
||
|
w.indent--
|
||
|
w.WriteByte('>')
|
||
|
w.WriteByte('\n')
|
||
|
}
|
||
|
default:
|
||
|
w.writeName(fd)
|
||
|
if err := w.writeSingularValue(m.Get(fd), fd); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
w.WriteByte('\n')
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if b := m.GetUnknown(); len(b) > 0 {
|
||
|
w.writeUnknownFields(b)
|
||
|
}
|
||
|
return w.writeExtensions(m)
|
||
|
}
|
||
|
|
||
|
func (w *textWriter) writeSingularValue(v protoreflect.Value, fd protoreflect.FieldDescriptor) error {
|
||
|
switch fd.Kind() {
|
||
|
case protoreflect.FloatKind, protoreflect.DoubleKind:
|
||
|
switch vf := v.Float(); {
|
||
|
case math.IsInf(vf, +1):
|
||
|
w.Write(posInf)
|
||
|
case math.IsInf(vf, -1):
|
||
|
w.Write(negInf)
|
||
|
case math.IsNaN(vf):
|
||
|
w.Write(nan)
|
||
|
default:
|
||
|
fmt.Fprint(w, v.Interface())
|
||
|
}
|
||
|
case protoreflect.StringKind:
|
||
|
// NOTE: This does not validate UTF-8 for historical reasons.
|
||
|
w.writeQuotedString(string(v.String()))
|
||
|
case protoreflect.BytesKind:
|
||
|
w.writeQuotedString(string(v.Bytes()))
|
||
|
case protoreflect.MessageKind, protoreflect.GroupKind:
|
||
|
var bra, ket byte = '<', '>'
|
||
|
if fd.Kind() == protoreflect.GroupKind {
|
||
|
bra, ket = '{', '}'
|
||
|
}
|
||
|
w.WriteByte(bra)
|
||
|
if !w.compact {
|
||
|
w.WriteByte('\n')
|
||
|
}
|
||
|
w.indent++
|
||
|
m := v.Message()
|
||
|
if m2, ok := m.Interface().(encoding.TextMarshaler); ok {
|
||
|
b, err := m2.MarshalText()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
w.Write(b)
|
||
|
} else {
|
||
|
w.writeMessage(m)
|
||
|
}
|
||
|
w.indent--
|
||
|
w.WriteByte(ket)
|
||
|
case protoreflect.EnumKind:
|
||
|
if ev := fd.Enum().Values().ByNumber(v.Enum()); ev != nil {
|
||
|
fmt.Fprint(w, ev.Name())
|
||
|
} else {
|
||
|
fmt.Fprint(w, v.Enum())
|
||
|
}
|
||
|
default:
|
||
|
fmt.Fprint(w, v.Interface())
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// writeQuotedString writes a quoted string in the protocol buffer text format.
|
||
|
func (w *textWriter) writeQuotedString(s string) {
|
||
|
w.WriteByte('"')
|
||
|
for i := 0; i < len(s); i++ {
|
||
|
switch c := s[i]; c {
|
||
|
case '\n':
|
||
|
w.buf = append(w.buf, `\n`...)
|
||
|
case '\r':
|
||
|
w.buf = append(w.buf, `\r`...)
|
||
|
case '\t':
|
||
|
w.buf = append(w.buf, `\t`...)
|
||
|
case '"':
|
||
|
w.buf = append(w.buf, `\"`...)
|
||
|
case '\\':
|
||
|
w.buf = append(w.buf, `\\`...)
|
||
|
default:
|
||
|
if isPrint := c >= 0x20 && c < 0x7f; isPrint {
|
||
|
w.buf = append(w.buf, c)
|
||
|
} else {
|
||
|
w.buf = append(w.buf, fmt.Sprintf(`\%03o`, c)...)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
w.WriteByte('"')
|
||
|
}
|
||
|
|
||
|
func (w *textWriter) writeUnknownFields(b []byte) {
|
||
|
if !w.compact {
|
||
|
fmt.Fprintf(w, "/* %d unknown bytes */\n", len(b))
|
||
|
}
|
||
|
|
||
|
for len(b) > 0 {
|
||
|
num, wtyp, n := protowire.ConsumeTag(b)
|
||
|
if n < 0 {
|
||
|
return
|
||
|
}
|
||
|
b = b[n:]
|
||
|
|
||
|
if wtyp == protowire.EndGroupType {
|
||
|
w.indent--
|
||
|
w.Write(endBraceNewline)
|
||
|
continue
|
||
|
}
|
||
|
fmt.Fprint(w, num)
|
||
|
if wtyp != protowire.StartGroupType {
|
||
|
w.WriteByte(':')
|
||
|
}
|
||
|
if !w.compact || wtyp == protowire.StartGroupType {
|
||
|
w.WriteByte(' ')
|
||
|
}
|
||
|
switch wtyp {
|
||
|
case protowire.VarintType:
|
||
|
v, n := protowire.ConsumeVarint(b)
|
||
|
if n < 0 {
|
||
|
return
|
||
|
}
|
||
|
b = b[n:]
|
||
|
fmt.Fprint(w, v)
|
||
|
case protowire.Fixed32Type:
|
||
|
v, n := protowire.ConsumeFixed32(b)
|
||
|
if n < 0 {
|
||
|
return
|
||
|
}
|
||
|
b = b[n:]
|
||
|
fmt.Fprint(w, v)
|
||
|
case protowire.Fixed64Type:
|
||
|
v, n := protowire.ConsumeFixed64(b)
|
||
|
if n < 0 {
|
||
|
return
|
||
|
}
|
||
|
b = b[n:]
|
||
|
fmt.Fprint(w, v)
|
||
|
case protowire.BytesType:
|
||
|
v, n := protowire.ConsumeBytes(b)
|
||
|
if n < 0 {
|
||
|
return
|
||
|
}
|
||
|
b = b[n:]
|
||
|
fmt.Fprintf(w, "%q", v)
|
||
|
case protowire.StartGroupType:
|
||
|
w.WriteByte('{')
|
||
|
w.indent++
|
||
|
default:
|
||
|
fmt.Fprintf(w, "/* unknown wire type %d */", wtyp)
|
||
|
}
|
||
|
w.WriteByte('\n')
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// writeExtensions writes all the extensions in m.
|
||
|
func (w *textWriter) writeExtensions(m protoreflect.Message) error {
|
||
|
md := m.Descriptor()
|
||
|
if md.ExtensionRanges().Len() == 0 {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
type ext struct {
|
||
|
desc protoreflect.FieldDescriptor
|
||
|
val protoreflect.Value
|
||
|
}
|
||
|
var exts []ext
|
||
|
m.Range(func(fd protoreflect.FieldDescriptor, v protoreflect.Value) bool {
|
||
|
if fd.IsExtension() {
|
||
|
exts = append(exts, ext{fd, v})
|
||
|
}
|
||
|
return true
|
||
|
})
|
||
|
sort.Slice(exts, func(i, j int) bool {
|
||
|
return exts[i].desc.Number() < exts[j].desc.Number()
|
||
|
})
|
||
|
|
||
|
for _, ext := range exts {
|
||
|
// For message set, use the name of the message as the extension name.
|
||
|
name := string(ext.desc.FullName())
|
||
|
if isMessageSet(ext.desc.ContainingMessage()) {
|
||
|
name = strings.TrimSuffix(name, ".message_set_extension")
|
||
|
}
|
||
|
|
||
|
if !ext.desc.IsList() {
|
||
|
if err := w.writeSingularExtension(name, ext.val, ext.desc); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
} else {
|
||
|
lv := ext.val.List()
|
||
|
for i := 0; i < lv.Len(); i++ {
|
||
|
if err := w.writeSingularExtension(name, lv.Get(i), ext.desc); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (w *textWriter) writeSingularExtension(name string, v protoreflect.Value, fd protoreflect.FieldDescriptor) error {
|
||
|
fmt.Fprintf(w, "[%s]:", name)
|
||
|
if !w.compact {
|
||
|
w.WriteByte(' ')
|
||
|
}
|
||
|
if err := w.writeSingularValue(v, fd); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
w.WriteByte('\n')
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (w *textWriter) writeIndent() {
|
||
|
if !w.complete {
|
||
|
return
|
||
|
}
|
||
|
for i := 0; i < w.indent*2; i++ {
|
||
|
w.buf = append(w.buf, ' ')
|
||
|
}
|
||
|
w.complete = false
|
||
|
}
|