mirror of
https://github.com/gusaul/grpcox.git
synced 2025-01-26 06:14:40 +00:00
2746 lines
92 KiB
Go
2746 lines
92 KiB
Go
package dynamic
|
|
|
|
import (
|
|
"bytes"
|
|
"compress/gzip"
|
|
"errors"
|
|
"fmt"
|
|
"reflect"
|
|
"sort"
|
|
"strings"
|
|
|
|
"github.com/golang/protobuf/proto"
|
|
"github.com/golang/protobuf/protoc-gen-go/descriptor"
|
|
|
|
"github.com/jhump/protoreflect/desc"
|
|
)
|
|
|
|
// ErrUnknownTagNumber is an error that is returned when an operation refers
|
|
// to an unknown tag number.
|
|
var ErrUnknownTagNumber = errors.New("unknown tag number")
|
|
|
|
// UnknownTagNumberError is the same as ErrUnknownTagNumber.
|
|
// Deprecated: use ErrUnknownTagNumber
|
|
var UnknownTagNumberError = ErrUnknownTagNumber
|
|
|
|
// ErrUnknownFieldName is an error that is returned when an operation refers
|
|
// to an unknown field name.
|
|
var ErrUnknownFieldName = errors.New("unknown field name")
|
|
|
|
// UnknownFieldNameError is the same as ErrUnknownFieldName.
|
|
// Deprecated: use ErrUnknownFieldName
|
|
var UnknownFieldNameError = ErrUnknownFieldName
|
|
|
|
// ErrFieldIsNotMap is an error that is returned when map-related operations
|
|
// are attempted with fields that are not maps.
|
|
var ErrFieldIsNotMap = errors.New("field is not a map type")
|
|
|
|
// FieldIsNotMapError is the same as ErrFieldIsNotMap.
|
|
// Deprecated: use ErrFieldIsNotMap
|
|
var FieldIsNotMapError = ErrFieldIsNotMap
|
|
|
|
// ErrFieldIsNotRepeated is an error that is returned when repeated field
|
|
// operations are attempted with fields that are not repeated.
|
|
var ErrFieldIsNotRepeated = errors.New("field is not repeated")
|
|
|
|
// FieldIsNotRepeatedError is the same as ErrFieldIsNotRepeated.
|
|
// Deprecated: use ErrFieldIsNotRepeated
|
|
var FieldIsNotRepeatedError = ErrFieldIsNotRepeated
|
|
|
|
// ErrIndexOutOfRange is an error that is returned when an invalid index is
|
|
// provided when access a single element of a repeated field.
|
|
var ErrIndexOutOfRange = errors.New("index is out of range")
|
|
|
|
// IndexOutOfRangeError is the same as ErrIndexOutOfRange.
|
|
// Deprecated: use ErrIndexOutOfRange
|
|
var IndexOutOfRangeError = ErrIndexOutOfRange
|
|
|
|
// ErrNumericOverflow is an error returned by operations that encounter a
|
|
// numeric value that is too large, for example de-serializing a value into an
|
|
// int32 field when the value is larger that can fit into a 32-bit value.
|
|
var ErrNumericOverflow = errors.New("numeric value is out of range")
|
|
|
|
// NumericOverflowError is the same as ErrNumericOverflow.
|
|
// Deprecated: use ErrNumericOverflow
|
|
var NumericOverflowError = ErrNumericOverflow
|
|
|
|
var typeOfProtoMessage = reflect.TypeOf((*proto.Message)(nil)).Elem()
|
|
var typeOfDynamicMessage = reflect.TypeOf((*Message)(nil))
|
|
var typeOfBytes = reflect.TypeOf(([]byte)(nil))
|
|
|
|
var varintTypes = map[descriptor.FieldDescriptorProto_Type]bool{}
|
|
var fixed32Types = map[descriptor.FieldDescriptorProto_Type]bool{}
|
|
var fixed64Types = map[descriptor.FieldDescriptorProto_Type]bool{}
|
|
|
|
func init() {
|
|
varintTypes[descriptor.FieldDescriptorProto_TYPE_BOOL] = true
|
|
varintTypes[descriptor.FieldDescriptorProto_TYPE_INT32] = true
|
|
varintTypes[descriptor.FieldDescriptorProto_TYPE_INT64] = true
|
|
varintTypes[descriptor.FieldDescriptorProto_TYPE_UINT32] = true
|
|
varintTypes[descriptor.FieldDescriptorProto_TYPE_UINT64] = true
|
|
varintTypes[descriptor.FieldDescriptorProto_TYPE_SINT32] = true
|
|
varintTypes[descriptor.FieldDescriptorProto_TYPE_SINT64] = true
|
|
varintTypes[descriptor.FieldDescriptorProto_TYPE_ENUM] = true
|
|
|
|
fixed32Types[descriptor.FieldDescriptorProto_TYPE_FIXED32] = true
|
|
fixed32Types[descriptor.FieldDescriptorProto_TYPE_SFIXED32] = true
|
|
fixed32Types[descriptor.FieldDescriptorProto_TYPE_FLOAT] = true
|
|
|
|
fixed64Types[descriptor.FieldDescriptorProto_TYPE_FIXED64] = true
|
|
fixed64Types[descriptor.FieldDescriptorProto_TYPE_SFIXED64] = true
|
|
fixed64Types[descriptor.FieldDescriptorProto_TYPE_DOUBLE] = true
|
|
}
|
|
|
|
// Message is a dynamic protobuf message. Instead of a generated struct,
|
|
// like most protobuf messages, this is a map of field number to values and
|
|
// a message descriptor, which is used to validate the field values and
|
|
// also to de-serialize messages (from the standard binary format, as well
|
|
// as from the text format and from JSON).
|
|
type Message struct {
|
|
md *desc.MessageDescriptor
|
|
er *ExtensionRegistry
|
|
mf *MessageFactory
|
|
extraFields map[int32]*desc.FieldDescriptor
|
|
values map[int32]interface{}
|
|
unknownFields map[int32][]UnknownField
|
|
}
|
|
|
|
// UnknownField represents a field that was parsed from the binary wire
|
|
// format for a message, but was not a recognized field number. Enough
|
|
// information is preserved so that re-serializing the message won't lose
|
|
// any of the unrecognized data.
|
|
type UnknownField struct {
|
|
// Encoding indicates how the unknown field was encoded on the wire. If it
|
|
// is proto.WireBytes or proto.WireGroupStart then Contents will be set to
|
|
// the raw bytes. If it is proto.WireTypeFixed32 then the data is in the least
|
|
// significant 32 bits of Value. Otherwise, the data is in all 64 bits of
|
|
// Value.
|
|
Encoding int8
|
|
Contents []byte
|
|
Value uint64
|
|
}
|
|
|
|
// NewMessage creates a new dynamic message for the type represented by the given
|
|
// message descriptor. During de-serialization, a default MessageFactory is used to
|
|
// instantiate any nested message fields and no extension fields will be parsed. To
|
|
// use a custom MessageFactory or ExtensionRegistry, use MessageFactory.NewMessage.
|
|
func NewMessage(md *desc.MessageDescriptor) *Message {
|
|
return NewMessageWithMessageFactory(md, nil)
|
|
}
|
|
|
|
// NewMessageWithExtensionRegistry creates a new dynamic message for the type
|
|
// represented by the given message descriptor. During de-serialization, the given
|
|
// ExtensionRegistry is used to parse extension fields and nested messages will be
|
|
// instantiated using dynamic.NewMessageFactoryWithExtensionRegistry(er).
|
|
func NewMessageWithExtensionRegistry(md *desc.MessageDescriptor, er *ExtensionRegistry) *Message {
|
|
mf := NewMessageFactoryWithExtensionRegistry(er)
|
|
return NewMessageWithMessageFactory(md, mf)
|
|
}
|
|
|
|
// NewMessageWithMessageFactory creates a new dynamic message for the type
|
|
// represented by the given message descriptor. During de-serialization, the given
|
|
// MessageFactory is used to instantiate nested messages.
|
|
func NewMessageWithMessageFactory(md *desc.MessageDescriptor, mf *MessageFactory) *Message {
|
|
var er *ExtensionRegistry
|
|
if mf != nil {
|
|
er = mf.er
|
|
}
|
|
return &Message{
|
|
md: md,
|
|
mf: mf,
|
|
er: er,
|
|
}
|
|
}
|
|
|
|
// AsDynamicMessage converts the given message to a dynamic message. If the
|
|
// given message is dynamic, it is returned. Otherwise, a dynamic message is
|
|
// created using NewMessage.
|
|
func AsDynamicMessage(msg proto.Message) (*Message, error) {
|
|
return AsDynamicMessageWithMessageFactory(msg, nil)
|
|
}
|
|
|
|
// AsDynamicMessageWithExtensionRegistry converts the given message to a dynamic
|
|
// message. If the given message is dynamic, it is returned. Otherwise, a
|
|
// dynamic message is created using NewMessageWithExtensionRegistry.
|
|
func AsDynamicMessageWithExtensionRegistry(msg proto.Message, er *ExtensionRegistry) (*Message, error) {
|
|
mf := NewMessageFactoryWithExtensionRegistry(er)
|
|
return AsDynamicMessageWithMessageFactory(msg, mf)
|
|
}
|
|
|
|
// AsDynamicMessageWithMessageFactory converts the given message to a dynamic
|
|
// message. If the given message is dynamic, it is returned. Otherwise, a
|
|
// dynamic message is created using NewMessageWithMessageFactory.
|
|
func AsDynamicMessageWithMessageFactory(msg proto.Message, mf *MessageFactory) (*Message, error) {
|
|
if dm, ok := msg.(*Message); ok {
|
|
return dm, nil
|
|
}
|
|
md, err := desc.LoadMessageDescriptorForMessage(msg)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
dm := NewMessageWithMessageFactory(md, mf)
|
|
err = dm.mergeFrom(msg)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return dm, nil
|
|
}
|
|
|
|
// GetMessageDescriptor returns a descriptor for this message's type.
|
|
func (m *Message) GetMessageDescriptor() *desc.MessageDescriptor {
|
|
return m.md
|
|
}
|
|
|
|
// GetKnownFields returns a slice of descriptors for all known fields. The
|
|
// fields will not be in any defined order.
|
|
func (m *Message) GetKnownFields() []*desc.FieldDescriptor {
|
|
if len(m.extraFields) == 0 {
|
|
return m.md.GetFields()
|
|
}
|
|
flds := make([]*desc.FieldDescriptor, len(m.md.GetFields()), len(m.md.GetFields())+len(m.extraFields))
|
|
copy(flds, m.md.GetFields())
|
|
for _, fld := range m.extraFields {
|
|
if !fld.IsExtension() {
|
|
flds = append(flds, fld)
|
|
}
|
|
}
|
|
return flds
|
|
}
|
|
|
|
// GetKnownExtensions returns a slice of descriptors for all extensions known by
|
|
// the message's extension registry. The fields will not be in any defined order.
|
|
func (m *Message) GetKnownExtensions() []*desc.FieldDescriptor {
|
|
if !m.md.IsExtendable() {
|
|
return nil
|
|
}
|
|
exts := m.er.AllExtensionsForType(m.md.GetFullyQualifiedName())
|
|
for _, fld := range m.extraFields {
|
|
if fld.IsExtension() {
|
|
exts = append(exts, fld)
|
|
}
|
|
}
|
|
return exts
|
|
}
|
|
|
|
// GetUnknownFields returns a slice of tag numbers for all unknown fields that
|
|
// this message contains. The tags will not be in any defined order.
|
|
func (m *Message) GetUnknownFields() []int32 {
|
|
flds := make([]int32, 0, len(m.unknownFields))
|
|
for tag := range m.unknownFields {
|
|
flds = append(flds, tag)
|
|
}
|
|
return flds
|
|
}
|
|
|
|
// Descriptor returns the serialized form of the file descriptor in which the
|
|
// message was defined and a path to the message type therein. This mimics the
|
|
// method of the same name on message types generated by protoc.
|
|
func (m *Message) Descriptor() ([]byte, []int) {
|
|
// get encoded file descriptor
|
|
b, err := proto.Marshal(m.md.GetFile().AsProto())
|
|
if err != nil {
|
|
panic(fmt.Sprintf("failed to get encoded descriptor for %s: %v", m.md.GetFile().GetName(), err))
|
|
}
|
|
var zippedBytes bytes.Buffer
|
|
w := gzip.NewWriter(&zippedBytes)
|
|
if _, err := w.Write(b); err != nil {
|
|
panic(fmt.Sprintf("failed to get encoded descriptor for %s: %v", m.md.GetFile().GetName(), err))
|
|
}
|
|
if err := w.Close(); err != nil {
|
|
panic(fmt.Sprintf("failed to get an encoded descriptor for %s: %v", m.md.GetFile().GetName(), err))
|
|
}
|
|
|
|
// and path to message
|
|
path := []int{}
|
|
var d desc.Descriptor
|
|
name := m.md.GetFullyQualifiedName()
|
|
for d = m.md.GetParent(); d != nil; name, d = d.GetFullyQualifiedName(), d.GetParent() {
|
|
found := false
|
|
switch d := d.(type) {
|
|
case (*desc.FileDescriptor):
|
|
for i, md := range d.GetMessageTypes() {
|
|
if md.GetFullyQualifiedName() == name {
|
|
found = true
|
|
path = append(path, i)
|
|
}
|
|
}
|
|
case (*desc.MessageDescriptor):
|
|
for i, md := range d.GetNestedMessageTypes() {
|
|
if md.GetFullyQualifiedName() == name {
|
|
found = true
|
|
path = append(path, i)
|
|
}
|
|
}
|
|
}
|
|
if !found {
|
|
panic(fmt.Sprintf("failed to compute descriptor path for %s", m.md.GetFullyQualifiedName()))
|
|
}
|
|
}
|
|
// reverse the path
|
|
i := 0
|
|
j := len(path) - 1
|
|
for i < j {
|
|
path[i], path[j] = path[j], path[i]
|
|
i++
|
|
j--
|
|
}
|
|
|
|
return zippedBytes.Bytes(), path
|
|
}
|
|
|
|
// XXX_MessageName returns the fully qualified name of this message's type. This
|
|
// allows dynamic messages to be used with proto.MessageName.
|
|
func (m *Message) XXX_MessageName() string {
|
|
return m.md.GetFullyQualifiedName()
|
|
}
|
|
|
|
// FindFieldDescriptor returns a field descriptor for the given tag number. This
|
|
// searches known fields in the descriptor, known fields discovered during calls
|
|
// to GetField or SetField, and extension fields known by the message's extension
|
|
// registry. It returns nil if the tag is unknown.
|
|
func (m *Message) FindFieldDescriptor(tagNumber int32) *desc.FieldDescriptor {
|
|
fd := m.md.FindFieldByNumber(tagNumber)
|
|
if fd != nil {
|
|
return fd
|
|
}
|
|
fd = m.er.FindExtension(m.md.GetFullyQualifiedName(), tagNumber)
|
|
if fd != nil {
|
|
return fd
|
|
}
|
|
return m.extraFields[tagNumber]
|
|
}
|
|
|
|
// FindFieldDescriptorByName returns a field descriptor for the given field
|
|
// name. This searches known fields in the descriptor, known fields discovered
|
|
// during calls to GetField or SetField, and extension fields known by the
|
|
// message's extension registry. It returns nil if the name is unknown. If the
|
|
// given name refers to an extension, it should be fully qualified and may be
|
|
// optionally enclosed in parentheses or brackets.
|
|
func (m *Message) FindFieldDescriptorByName(name string) *desc.FieldDescriptor {
|
|
if name == "" {
|
|
return nil
|
|
}
|
|
fd := m.md.FindFieldByName(name)
|
|
if fd != nil {
|
|
return fd
|
|
}
|
|
mustBeExt := false
|
|
if name[0] == '(' {
|
|
if name[len(name)-1] != ')' {
|
|
// malformed name
|
|
return nil
|
|
}
|
|
mustBeExt = true
|
|
name = name[1 : len(name)-1]
|
|
} else if name[0] == '[' {
|
|
if name[len(name)-1] != ']' {
|
|
// malformed name
|
|
return nil
|
|
}
|
|
mustBeExt = true
|
|
name = name[1 : len(name)-1]
|
|
}
|
|
fd = m.er.FindExtensionByName(m.md.GetFullyQualifiedName(), name)
|
|
if fd != nil {
|
|
return fd
|
|
}
|
|
for _, fd := range m.extraFields {
|
|
if fd.IsExtension() && name == fd.GetFullyQualifiedName() {
|
|
return fd
|
|
} else if !mustBeExt && !fd.IsExtension() && name == fd.GetName() {
|
|
return fd
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// FindFieldDescriptorByJSONName returns a field descriptor for the given JSON
|
|
// name. This searches known fields in the descriptor, known fields discovered
|
|
// during calls to GetField or SetField, and extension fields known by the
|
|
// message's extension registry. If no field matches the given JSON name, it
|
|
// will fall back to searching field names (e.g. FindFieldDescriptorByName). If
|
|
// this also yields no match, nil is returned.
|
|
func (m *Message) FindFieldDescriptorByJSONName(name string) *desc.FieldDescriptor {
|
|
if name == "" {
|
|
return nil
|
|
}
|
|
fd := m.md.FindFieldByJSONName(name)
|
|
if fd != nil {
|
|
return fd
|
|
}
|
|
mustBeExt := false
|
|
if name[0] == '(' {
|
|
if name[len(name)-1] != ')' {
|
|
// malformed name
|
|
return nil
|
|
}
|
|
mustBeExt = true
|
|
name = name[1 : len(name)-1]
|
|
} else if name[0] == '[' {
|
|
if name[len(name)-1] != ']' {
|
|
// malformed name
|
|
return nil
|
|
}
|
|
mustBeExt = true
|
|
name = name[1 : len(name)-1]
|
|
}
|
|
fd = m.er.FindExtensionByJSONName(m.md.GetFullyQualifiedName(), name)
|
|
if fd != nil {
|
|
return fd
|
|
}
|
|
for _, fd := range m.extraFields {
|
|
if fd.IsExtension() && name == fd.GetFullyQualifiedJSONName() {
|
|
return fd
|
|
} else if !mustBeExt && !fd.IsExtension() && name == fd.GetJSONName() {
|
|
return fd
|
|
}
|
|
}
|
|
|
|
// try non-JSON names
|
|
return m.FindFieldDescriptorByName(name)
|
|
}
|
|
|
|
func (m *Message) checkField(fd *desc.FieldDescriptor) error {
|
|
return checkField(fd, m.md)
|
|
}
|
|
|
|
func checkField(fd *desc.FieldDescriptor, md *desc.MessageDescriptor) error {
|
|
if fd.GetOwner().GetFullyQualifiedName() != md.GetFullyQualifiedName() {
|
|
return fmt.Errorf("given field, %s, is for wrong message type: %s; expecting %s", fd.GetName(), fd.GetOwner().GetFullyQualifiedName(), md.GetFullyQualifiedName())
|
|
}
|
|
if fd.IsExtension() && !md.IsExtension(fd.GetNumber()) {
|
|
return fmt.Errorf("given field, %s, is an extension but is not in message extension range: %v", fd.GetFullyQualifiedName(), md.GetExtensionRanges())
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// GetField returns the value for the given field descriptor. It panics if an
|
|
// error is encountered. See TryGetField.
|
|
func (m *Message) GetField(fd *desc.FieldDescriptor) interface{} {
|
|
if v, err := m.TryGetField(fd); err != nil {
|
|
panic(err.Error())
|
|
} else {
|
|
return v
|
|
}
|
|
}
|
|
|
|
// TryGetField returns the value for the given field descriptor. An error is
|
|
// returned if the given field descriptor does not belong to the right message
|
|
// type.
|
|
//
|
|
// The Go type of the returned value, for scalar fields, is the same as protoc
|
|
// would generate for the field (in a non-dynamic message). The table below
|
|
// lists the scalar types and the corresponding Go types.
|
|
// +-------------------------+-----------+
|
|
// | Declared Type | Go Type |
|
|
// +-------------------------+-----------+
|
|
// | int32, sint32, sfixed32 | int32 |
|
|
// | int64, sint64, sfixed64 | int64 |
|
|
// | uint32, fixed32 | uint32 |
|
|
// | uint64, fixed64 | uint64 |
|
|
// | float | float32 |
|
|
// | double | double32 |
|
|
// | bool | bool |
|
|
// | string | string |
|
|
// | bytes | []byte |
|
|
// +-------------------------+-----------+
|
|
//
|
|
// Values for enum fields will always be int32 values. You can use the enum
|
|
// descriptor associated with the field to lookup value names with those values.
|
|
// Values for message type fields may be an instance of the generated type *or*
|
|
// may be another *dynamic.Message that represents the type.
|
|
//
|
|
// If the given field is a map field, the returned type will be
|
|
// map[interface{}]interface{}. The actual concrete types of keys and values is
|
|
// as described above. If the given field is a (non-map) repeated field, the
|
|
// returned type is always []interface{}; the type of the actual elements is as
|
|
// described above.
|
|
//
|
|
// If this message has no value for the given field, its default value is
|
|
// returned. If the message is defined in a file with "proto3" syntax, the
|
|
// default is always the zero value for the field. The default value for map and
|
|
// repeated fields is a nil map or slice (respectively). For field's whose types
|
|
// is a message, the default value is an empty message for "proto2" syntax or a
|
|
// nil message for "proto3" syntax. Note that the in the latter case, a non-nil
|
|
// interface with a nil pointer is returned, not a nil interface. Also note that
|
|
// whether the returned value is an empty message or nil depends on if *this*
|
|
// message was defined as "proto3" syntax, not the message type referred to by
|
|
// the field's type.
|
|
//
|
|
// If the given field descriptor is not known (e.g. not present in the message
|
|
// descriptor) but corresponds to an unknown field, the unknown value will be
|
|
// parsed and become known. The parsed value will be returned, or an error will
|
|
// be returned if the unknown value cannot be parsed according to the field
|
|
// descriptor's type information.
|
|
func (m *Message) TryGetField(fd *desc.FieldDescriptor) (interface{}, error) {
|
|
if err := m.checkField(fd); err != nil {
|
|
return nil, err
|
|
}
|
|
return m.getField(fd)
|
|
}
|
|
|
|
// GetFieldByName returns the value for the field with the given name. It panics
|
|
// if an error is encountered. See TryGetFieldByName.
|
|
func (m *Message) GetFieldByName(name string) interface{} {
|
|
if v, err := m.TryGetFieldByName(name); err != nil {
|
|
panic(err.Error())
|
|
} else {
|
|
return v
|
|
}
|
|
}
|
|
|
|
// TryGetFieldByName returns the value for the field with the given name. An
|
|
// error is returned if the given name is unknown. If the given name refers to
|
|
// an extension field, it should be fully qualified and optionally enclosed in
|
|
// parenthesis or brackets.
|
|
//
|
|
// If this message has no value for the given field, its default value is
|
|
// returned. (See TryGetField for more info on types and default field values.)
|
|
func (m *Message) TryGetFieldByName(name string) (interface{}, error) {
|
|
fd := m.FindFieldDescriptorByName(name)
|
|
if fd == nil {
|
|
return nil, UnknownFieldNameError
|
|
}
|
|
return m.getField(fd)
|
|
}
|
|
|
|
// GetFieldByNumber returns the value for the field with the given tag number.
|
|
// It panics if an error is encountered. See TryGetFieldByNumber.
|
|
func (m *Message) GetFieldByNumber(tagNumber int) interface{} {
|
|
if v, err := m.TryGetFieldByNumber(tagNumber); err != nil {
|
|
panic(err.Error())
|
|
} else {
|
|
return v
|
|
}
|
|
}
|
|
|
|
// TryGetFieldByNumber returns the value for the field with the given tag
|
|
// number. An error is returned if the given tag is unknown.
|
|
//
|
|
// If this message has no value for the given field, its default value is
|
|
// returned. (See TryGetField for more info on types and default field values.)
|
|
func (m *Message) TryGetFieldByNumber(tagNumber int) (interface{}, error) {
|
|
fd := m.FindFieldDescriptor(int32(tagNumber))
|
|
if fd == nil {
|
|
return nil, UnknownTagNumberError
|
|
}
|
|
return m.getField(fd)
|
|
}
|
|
|
|
func (m *Message) getField(fd *desc.FieldDescriptor) (interface{}, error) {
|
|
return m.doGetField(fd, false)
|
|
}
|
|
|
|
func (m *Message) doGetField(fd *desc.FieldDescriptor, nilIfAbsent bool) (interface{}, error) {
|
|
res := m.values[fd.GetNumber()]
|
|
if res == nil {
|
|
var err error
|
|
if res, err = m.parseUnknownField(fd); err != nil {
|
|
return nil, err
|
|
}
|
|
if res == nil {
|
|
if nilIfAbsent {
|
|
return nil, nil
|
|
} else {
|
|
def := fd.GetDefaultValue()
|
|
if def != nil {
|
|
return def, nil
|
|
}
|
|
// GetDefaultValue only returns nil for message types
|
|
md := fd.GetMessageType()
|
|
if md.IsProto3() {
|
|
// try to return a proper nil pointer
|
|
msgType := proto.MessageType(md.GetFullyQualifiedName())
|
|
if msgType != nil && msgType.Implements(typeOfProtoMessage) {
|
|
return reflect.Zero(msgType).Interface().(proto.Message), nil
|
|
}
|
|
// fallback to nil dynamic message pointer
|
|
return (*Message)(nil), nil
|
|
} else {
|
|
// for proto2, return default instance of message
|
|
return m.mf.NewMessage(md), nil
|
|
}
|
|
}
|
|
}
|
|
}
|
|
rt := reflect.TypeOf(res)
|
|
if rt.Kind() == reflect.Map {
|
|
// make defensive copies to prevent caller from storing illegal keys and values
|
|
m := res.(map[interface{}]interface{})
|
|
res := map[interface{}]interface{}{}
|
|
for k, v := range m {
|
|
res[k] = v
|
|
}
|
|
return res, nil
|
|
} else if rt.Kind() == reflect.Slice && rt != typeOfBytes {
|
|
// make defensive copies to prevent caller from storing illegal elements
|
|
sl := res.([]interface{})
|
|
res := make([]interface{}, len(sl))
|
|
copy(res, sl)
|
|
return res, nil
|
|
}
|
|
return res, nil
|
|
}
|
|
|
|
// HasField returns true if this message has a value for the given field. If the
|
|
// given field is not valid (e.g. belongs to a different message type), false is
|
|
// returned. If this message is defined in a file with "proto3" syntax, this
|
|
// will return false even if a field was explicitly assigned its zero value (the
|
|
// zero values for a field are intentionally indistinguishable from absent).
|
|
func (m *Message) HasField(fd *desc.FieldDescriptor) bool {
|
|
if err := m.checkField(fd); err != nil {
|
|
return false
|
|
}
|
|
return m.HasFieldNumber(int(fd.GetNumber()))
|
|
}
|
|
|
|
// HasFieldName returns true if this message has a value for a field with the
|
|
// given name. If the given name is unknown, this returns false.
|
|
func (m *Message) HasFieldName(name string) bool {
|
|
fd := m.FindFieldDescriptorByName(name)
|
|
if fd == nil {
|
|
return false
|
|
}
|
|
return m.HasFieldNumber(int(fd.GetNumber()))
|
|
}
|
|
|
|
// HasFieldNumber returns true if this message has a value for a field with the
|
|
// given tag number. If the given tag is unknown, this returns false.
|
|
func (m *Message) HasFieldNumber(tagNumber int) bool {
|
|
if _, ok := m.values[int32(tagNumber)]; ok {
|
|
return true
|
|
}
|
|
_, ok := m.unknownFields[int32(tagNumber)]
|
|
return ok
|
|
}
|
|
|
|
// SetField sets the value for the given field descriptor to the given value. It
|
|
// panics if an error is encountered. See TrySetField.
|
|
func (m *Message) SetField(fd *desc.FieldDescriptor, val interface{}) {
|
|
if err := m.TrySetField(fd, val); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TrySetField sets the value for the given field descriptor to the given value.
|
|
// An error is returned if the given field descriptor does not belong to the
|
|
// right message type or if the given value is not a correct/compatible type for
|
|
// the given field.
|
|
//
|
|
// The Go type expected for a field is the same as TryGetField would return for
|
|
// the field. So message values can be supplied as either the correct generated
|
|
// message type or as a *dynamic.Message.
|
|
//
|
|
// Since it is cumbersome to work with dynamic messages, some concessions are
|
|
// made to simplify usage regarding types:
|
|
//
|
|
// 1. If a numeric type is provided that can be converted *without loss or
|
|
// overflow*, it is accepted. This allows for setting int64 fields using int
|
|
// or int32 values. Similarly for uint64 with uint and uint32 values and for
|
|
// float64 fields with float32 values.
|
|
// 2. The value can be a named type, as long as its underlying type is correct.
|
|
// 3. Map and repeated fields can be set using any kind of concrete map or
|
|
// slice type, as long as the values within are all of the correct type. So
|
|
// a field defined as a 'map<string, int32>` can be set using a
|
|
// map[string]int32, a map[string]interface{}, or even a
|
|
// map[interface{}]interface{}.
|
|
// 4. Finally, dynamic code that chooses to not treat maps as a special-case
|
|
// find that they can set map fields using a slice where each element is a
|
|
// message that matches the implicit map-entry field message type.
|
|
//
|
|
// If the given field descriptor is not known (e.g. not present in the message
|
|
// descriptor) it will become known. Subsequent operations using tag numbers or
|
|
// names will be able to resolve the newly-known type. If the message has a
|
|
// value for the unknown value, it is cleared, replaced by the given known
|
|
// value.
|
|
func (m *Message) TrySetField(fd *desc.FieldDescriptor, val interface{}) error {
|
|
if err := m.checkField(fd); err != nil {
|
|
return err
|
|
}
|
|
return m.setField(fd, val)
|
|
}
|
|
|
|
// SetFieldByName sets the value for the field with the given name to the given
|
|
// value. It panics if an error is encountered. See TrySetFieldByName.
|
|
func (m *Message) SetFieldByName(name string, val interface{}) {
|
|
if err := m.TrySetFieldByName(name, val); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TrySetFieldByName sets the value for the field with the given name to the
|
|
// given value. An error is returned if the given name is unknown or if the
|
|
// given value has an incorrect type. If the given name refers to an extension
|
|
// field, it should be fully qualified and optionally enclosed in parenthesis or
|
|
// brackets.
|
|
//
|
|
// (See TrySetField for more info on types.)
|
|
func (m *Message) TrySetFieldByName(name string, val interface{}) error {
|
|
fd := m.FindFieldDescriptorByName(name)
|
|
if fd == nil {
|
|
return UnknownFieldNameError
|
|
}
|
|
return m.setField(fd, val)
|
|
}
|
|
|
|
// SetFieldByNumber sets the value for the field with the given tag number to
|
|
// the given value. It panics if an error is encountered. See
|
|
// TrySetFieldByNumber.
|
|
func (m *Message) SetFieldByNumber(tagNumber int, val interface{}) {
|
|
if err := m.TrySetFieldByNumber(tagNumber, val); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TrySetFieldByNumber sets the value for the field with the given tag number to
|
|
// the given value. An error is returned if the given tag is unknown or if the
|
|
// given value has an incorrect type.
|
|
//
|
|
// (See TrySetField for more info on types.)
|
|
func (m *Message) TrySetFieldByNumber(tagNumber int, val interface{}) error {
|
|
fd := m.FindFieldDescriptor(int32(tagNumber))
|
|
if fd == nil {
|
|
return UnknownTagNumberError
|
|
}
|
|
return m.setField(fd, val)
|
|
}
|
|
|
|
func (m *Message) setField(fd *desc.FieldDescriptor, val interface{}) error {
|
|
var err error
|
|
if val, err = validFieldValue(fd, val); err != nil {
|
|
return err
|
|
}
|
|
m.internalSetField(fd, val)
|
|
return nil
|
|
}
|
|
|
|
func (m *Message) internalSetField(fd *desc.FieldDescriptor, val interface{}) {
|
|
if m.md.IsProto3() && fd.GetOneOf() == nil {
|
|
// proto3 considers fields that are set to their zero value as unset
|
|
if fd.IsRepeated() {
|
|
// can't use == comparison below for map and slices, so just test length
|
|
// (zero length is same as default)
|
|
if reflect.ValueOf(val).Len() == 0 {
|
|
if m.values != nil {
|
|
delete(m.values, fd.GetNumber())
|
|
}
|
|
return
|
|
}
|
|
} else {
|
|
// can't compare slices, so we have to special-case []byte values
|
|
var equal bool
|
|
if b, ok := val.([]byte); ok {
|
|
equal = ok && bytes.Equal(b, fd.GetDefaultValue().([]byte))
|
|
} else {
|
|
defVal := fd.GetDefaultValue()
|
|
equal = defVal == val
|
|
if !equal && defVal == nil {
|
|
// above just checks if value is the nil interface,
|
|
// but we should also test if the given value is a
|
|
// nil pointer
|
|
rv := reflect.ValueOf(val)
|
|
if rv.Kind() == reflect.Ptr && rv.IsNil() {
|
|
equal = true
|
|
}
|
|
}
|
|
}
|
|
if equal {
|
|
if m.values != nil {
|
|
delete(m.values, fd.GetNumber())
|
|
}
|
|
return
|
|
}
|
|
}
|
|
}
|
|
if m.values == nil {
|
|
m.values = map[int32]interface{}{}
|
|
}
|
|
m.values[fd.GetNumber()] = val
|
|
// if this field is part of a one-of, make sure all other one-of choices are cleared
|
|
od := fd.GetOneOf()
|
|
if od != nil {
|
|
for _, other := range od.GetChoices() {
|
|
if other.GetNumber() != fd.GetNumber() {
|
|
delete(m.values, other.GetNumber())
|
|
}
|
|
}
|
|
}
|
|
// also clear any unknown fields
|
|
if m.unknownFields != nil {
|
|
delete(m.unknownFields, fd.GetNumber())
|
|
}
|
|
// and add this field if it was previously unknown
|
|
if existing := m.FindFieldDescriptor(fd.GetNumber()); existing == nil {
|
|
m.addField(fd)
|
|
}
|
|
}
|
|
|
|
func (m *Message) addField(fd *desc.FieldDescriptor) {
|
|
if m.extraFields == nil {
|
|
m.extraFields = map[int32]*desc.FieldDescriptor{}
|
|
}
|
|
m.extraFields[fd.GetNumber()] = fd
|
|
}
|
|
|
|
// ClearField removes any value for the given field. It panics if an error is
|
|
// encountered. See TryClearField.
|
|
func (m *Message) ClearField(fd *desc.FieldDescriptor) {
|
|
if err := m.TryClearField(fd); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryClearField removes any value for the given field. An error is returned if
|
|
// the given field descriptor does not belong to the right message type.
|
|
func (m *Message) TryClearField(fd *desc.FieldDescriptor) error {
|
|
if err := m.checkField(fd); err != nil {
|
|
return err
|
|
}
|
|
m.clearField(fd)
|
|
return nil
|
|
}
|
|
|
|
// ClearFieldByName removes any value for the field with the given name. It
|
|
// panics if an error is encountered. See TryClearFieldByName.
|
|
func (m *Message) ClearFieldByName(name string) {
|
|
if err := m.TryClearFieldByName(name); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryClearFieldByName removes any value for the field with the given name. An
|
|
// error is returned if the given name is unknown. If the given name refers to
|
|
// an extension field, it should be fully qualified and optionally enclosed in
|
|
// parenthesis or brackets.
|
|
func (m *Message) TryClearFieldByName(name string) error {
|
|
fd := m.FindFieldDescriptorByName(name)
|
|
if fd == nil {
|
|
return UnknownFieldNameError
|
|
}
|
|
m.clearField(fd)
|
|
return nil
|
|
}
|
|
|
|
// ClearFieldByNumber removes any value for the field with the given tag number.
|
|
// It panics if an error is encountered. See TryClearFieldByNumber.
|
|
func (m *Message) ClearFieldByNumber(tagNumber int) {
|
|
if err := m.TryClearFieldByNumber(tagNumber); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryClearFieldByNumber removes any value for the field with the given tag
|
|
// number. An error is returned if the given tag is unknown.
|
|
func (m *Message) TryClearFieldByNumber(tagNumber int) error {
|
|
fd := m.FindFieldDescriptor(int32(tagNumber))
|
|
if fd == nil {
|
|
return UnknownTagNumberError
|
|
}
|
|
m.clearField(fd)
|
|
return nil
|
|
}
|
|
|
|
func (m *Message) clearField(fd *desc.FieldDescriptor) {
|
|
// clear value
|
|
if m.values != nil {
|
|
delete(m.values, fd.GetNumber())
|
|
}
|
|
// also clear any unknown fields
|
|
if m.unknownFields != nil {
|
|
delete(m.unknownFields, fd.GetNumber())
|
|
}
|
|
// and add this field if it was previously unknown
|
|
if existing := m.FindFieldDescriptor(fd.GetNumber()); existing == nil {
|
|
m.addField(fd)
|
|
}
|
|
}
|
|
|
|
// GetOneOfField returns which of the given one-of's fields is set and the
|
|
// corresponding value. It panics if an error is encountered. See
|
|
// TryGetOneOfField.
|
|
func (m *Message) GetOneOfField(od *desc.OneOfDescriptor) (*desc.FieldDescriptor, interface{}) {
|
|
if fd, val, err := m.TryGetOneOfField(od); err != nil {
|
|
panic(err.Error())
|
|
} else {
|
|
return fd, val
|
|
}
|
|
}
|
|
|
|
// TryGetOneOfField returns which of the given one-of's fields is set and the
|
|
// corresponding value. An error is returned if the given one-of belongs to the
|
|
// wrong message type. If the given one-of has no field set, this method will
|
|
// return nil, nil.
|
|
//
|
|
// The type of the value, if one is set, is the same as would be returned by
|
|
// TryGetField using the returned field descriptor.
|
|
//
|
|
// Like with TryGetField, if the given one-of contains any fields that are not
|
|
// known (e.g. not present in this message's descriptor), they will become known
|
|
// and any unknown value will be parsed (and become a known value on success).
|
|
func (m *Message) TryGetOneOfField(od *desc.OneOfDescriptor) (*desc.FieldDescriptor, interface{}, error) {
|
|
if od.GetOwner().GetFullyQualifiedName() != m.md.GetFullyQualifiedName() {
|
|
return nil, nil, fmt.Errorf("given one-of, %s, is for wrong message type: %s; expecting %s", od.GetName(), od.GetOwner().GetFullyQualifiedName(), m.md.GetFullyQualifiedName())
|
|
}
|
|
for _, fd := range od.GetChoices() {
|
|
val, err := m.doGetField(fd, true)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
if val != nil {
|
|
return fd, val, nil
|
|
}
|
|
}
|
|
return nil, nil, nil
|
|
}
|
|
|
|
// ClearOneOfField removes any value for any of the given one-of's fields. It
|
|
// panics if an error is encountered. See TryClearOneOfField.
|
|
func (m *Message) ClearOneOfField(od *desc.OneOfDescriptor) {
|
|
if err := m.TryClearOneOfField(od); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryClearOneOfField removes any value for any of the given one-of's fields. An
|
|
// error is returned if the given one-of descriptor does not belong to the right
|
|
// message type.
|
|
func (m *Message) TryClearOneOfField(od *desc.OneOfDescriptor) error {
|
|
if od.GetOwner().GetFullyQualifiedName() != m.md.GetFullyQualifiedName() {
|
|
return fmt.Errorf("given one-of, %s, is for wrong message type: %s; expecting %s", od.GetName(), od.GetOwner().GetFullyQualifiedName(), m.md.GetFullyQualifiedName())
|
|
}
|
|
for _, fd := range od.GetChoices() {
|
|
m.clearField(fd)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// GetMapField returns the value for the given map field descriptor and given
|
|
// key. It panics if an error is encountered. See TryGetMapField.
|
|
func (m *Message) GetMapField(fd *desc.FieldDescriptor, key interface{}) interface{} {
|
|
if v, err := m.TryGetMapField(fd, key); err != nil {
|
|
panic(err.Error())
|
|
} else {
|
|
return v
|
|
}
|
|
}
|
|
|
|
// TryGetMapField returns the value for the given map field descriptor and given
|
|
// key. An error is returned if the given field descriptor does not belong to
|
|
// the right message type or if it is not a map field.
|
|
//
|
|
// If the map field does not contain the requested key, this method returns
|
|
// nil, nil. The Go type of the value returned mirrors the type that protoc
|
|
// would generate for the field. (See TryGetField for more details on types).
|
|
//
|
|
// If the given field descriptor is not known (e.g. not present in the message
|
|
// descriptor) but corresponds to an unknown field, the unknown value will be
|
|
// parsed and become known. The parsed value will be searched for the requested
|
|
// key and any value returned. An error will be returned if the unknown value
|
|
// cannot be parsed according to the field descriptor's type information.
|
|
func (m *Message) TryGetMapField(fd *desc.FieldDescriptor, key interface{}) (interface{}, error) {
|
|
if err := m.checkField(fd); err != nil {
|
|
return nil, err
|
|
}
|
|
return m.getMapField(fd, key)
|
|
}
|
|
|
|
// GetMapFieldByName returns the value for the map field with the given name and
|
|
// given key. It panics if an error is encountered. See TryGetMapFieldByName.
|
|
func (m *Message) GetMapFieldByName(name string, key interface{}) interface{} {
|
|
if v, err := m.TryGetMapFieldByName(name, key); err != nil {
|
|
panic(err.Error())
|
|
} else {
|
|
return v
|
|
}
|
|
}
|
|
|
|
// TryGetMapFieldByName returns the value for the map field with the given name
|
|
// and given key. An error is returned if the given name is unknown or if it
|
|
// names a field that is not a map field.
|
|
//
|
|
// If this message has no value for the given field or the value has no value
|
|
// for the requested key, then this method returns nil, nil.
|
|
//
|
|
// (See TryGetField for more info on types.)
|
|
func (m *Message) TryGetMapFieldByName(name string, key interface{}) (interface{}, error) {
|
|
fd := m.FindFieldDescriptorByName(name)
|
|
if fd == nil {
|
|
return nil, UnknownFieldNameError
|
|
}
|
|
return m.getMapField(fd, key)
|
|
}
|
|
|
|
// GetMapFieldByNumber returns the value for the map field with the given tag
|
|
// number and given key. It panics if an error is encountered. See
|
|
// TryGetMapFieldByNumber.
|
|
func (m *Message) GetMapFieldByNumber(tagNumber int, key interface{}) interface{} {
|
|
if v, err := m.TryGetMapFieldByNumber(tagNumber, key); err != nil {
|
|
panic(err.Error())
|
|
} else {
|
|
return v
|
|
}
|
|
}
|
|
|
|
// TryGetMapFieldByNumber returns the value for the map field with the given tag
|
|
// number and given key. An error is returned if the given tag is unknown or if
|
|
// it indicates a field that is not a map field.
|
|
//
|
|
// If this message has no value for the given field or the value has no value
|
|
// for the requested key, then this method returns nil, nil.
|
|
//
|
|
// (See TryGetField for more info on types.)
|
|
func (m *Message) TryGetMapFieldByNumber(tagNumber int, key interface{}) (interface{}, error) {
|
|
fd := m.FindFieldDescriptor(int32(tagNumber))
|
|
if fd == nil {
|
|
return nil, UnknownTagNumberError
|
|
}
|
|
return m.getMapField(fd, key)
|
|
}
|
|
|
|
func (m *Message) getMapField(fd *desc.FieldDescriptor, key interface{}) (interface{}, error) {
|
|
if !fd.IsMap() {
|
|
return nil, FieldIsNotMapError
|
|
}
|
|
kfd := fd.GetMessageType().GetFields()[0]
|
|
ki, err := validElementFieldValue(kfd, key)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
mp := m.values[fd.GetNumber()]
|
|
if mp == nil {
|
|
if mp, err = m.parseUnknownField(fd); err != nil {
|
|
return nil, err
|
|
} else if mp == nil {
|
|
return nil, nil
|
|
}
|
|
}
|
|
return mp.(map[interface{}]interface{})[ki], nil
|
|
}
|
|
|
|
// ForEachMapFieldEntry executes the given function for each entry in the map
|
|
// value for the given field descriptor. It stops iteration if the function
|
|
// returns false. It panics if an error is encountered. See
|
|
// TryForEachMapFieldEntry.
|
|
func (m *Message) ForEachMapFieldEntry(fd *desc.FieldDescriptor, fn func(key, val interface{}) bool) {
|
|
if err := m.TryForEachMapFieldEntry(fd, fn); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryForEachMapFieldEntry executes the given function for each entry in the map
|
|
// value for the given field descriptor. An error is returned if the given field
|
|
// descriptor does not belong to the right message type or if it is not a map
|
|
// field.
|
|
//
|
|
// Iteration ends either when all entries have been examined or when the given
|
|
// function returns false. So the function is expected to return true for normal
|
|
// iteration and false to break out. If this message has no value for the given
|
|
// field, it returns without invoking the given function.
|
|
//
|
|
// The Go type of the key and value supplied to the function mirrors the type
|
|
// that protoc would generate for the field. (See TryGetField for more details
|
|
// on types).
|
|
//
|
|
// If the given field descriptor is not known (e.g. not present in the message
|
|
// descriptor) but corresponds to an unknown field, the unknown value will be
|
|
// parsed and become known. The parsed value will be searched for the requested
|
|
// key and any value returned. An error will be returned if the unknown value
|
|
// cannot be parsed according to the field descriptor's type information.
|
|
func (m *Message) TryForEachMapFieldEntry(fd *desc.FieldDescriptor, fn func(key, val interface{}) bool) error {
|
|
if err := m.checkField(fd); err != nil {
|
|
return err
|
|
}
|
|
return m.forEachMapFieldEntry(fd, fn)
|
|
}
|
|
|
|
// ForEachMapFieldEntryByName executes the given function for each entry in the
|
|
// map value for the field with the given name. It stops iteration if the
|
|
// function returns false. It panics if an error is encountered. See
|
|
// TryForEachMapFieldEntryByName.
|
|
func (m *Message) ForEachMapFieldEntryByName(name string, fn func(key, val interface{}) bool) {
|
|
if err := m.TryForEachMapFieldEntryByName(name, fn); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryForEachMapFieldEntryByName executes the given function for each entry in
|
|
// the map value for the field with the given name. It stops iteration if the
|
|
// function returns false. An error is returned if the given name is unknown or
|
|
// if it names a field that is not a map field.
|
|
//
|
|
// If this message has no value for the given field, it returns without ever
|
|
// invoking the given function.
|
|
//
|
|
// (See TryGetField for more info on types supplied to the function.)
|
|
func (m *Message) TryForEachMapFieldEntryByName(name string, fn func(key, val interface{}) bool) error {
|
|
fd := m.FindFieldDescriptorByName(name)
|
|
if fd == nil {
|
|
return UnknownFieldNameError
|
|
}
|
|
return m.forEachMapFieldEntry(fd, fn)
|
|
}
|
|
|
|
// ForEachMapFieldEntryByNumber executes the given function for each entry in
|
|
// the map value for the field with the given tag number. It stops iteration if
|
|
// the function returns false. It panics if an error is encountered. See
|
|
// TryForEachMapFieldEntryByNumber.
|
|
func (m *Message) ForEachMapFieldEntryByNumber(tagNumber int, fn func(key, val interface{}) bool) {
|
|
if err := m.TryForEachMapFieldEntryByNumber(tagNumber, fn); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryForEachMapFieldEntryByNumber executes the given function for each entry in
|
|
// the map value for the field with the given tag number. It stops iteration if
|
|
// the function returns false. An error is returned if the given tag is unknown
|
|
// or if it indicates a field that is not a map field.
|
|
//
|
|
// If this message has no value for the given field, it returns without ever
|
|
// invoking the given function.
|
|
//
|
|
// (See TryGetField for more info on types supplied to the function.)
|
|
func (m *Message) TryForEachMapFieldEntryByNumber(tagNumber int, fn func(key, val interface{}) bool) error {
|
|
fd := m.FindFieldDescriptor(int32(tagNumber))
|
|
if fd == nil {
|
|
return UnknownTagNumberError
|
|
}
|
|
return m.forEachMapFieldEntry(fd, fn)
|
|
}
|
|
|
|
func (m *Message) forEachMapFieldEntry(fd *desc.FieldDescriptor, fn func(key, val interface{}) bool) error {
|
|
if !fd.IsMap() {
|
|
return FieldIsNotMapError
|
|
}
|
|
mp := m.values[fd.GetNumber()]
|
|
if mp == nil {
|
|
if mp, err := m.parseUnknownField(fd); err != nil {
|
|
return err
|
|
} else if mp == nil {
|
|
return nil
|
|
}
|
|
}
|
|
for k, v := range mp.(map[interface{}]interface{}) {
|
|
if !fn(k, v) {
|
|
break
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// PutMapField sets the value for the given map field descriptor and given key
|
|
// to the given value. It panics if an error is encountered. See TryPutMapField.
|
|
func (m *Message) PutMapField(fd *desc.FieldDescriptor, key interface{}, val interface{}) {
|
|
if err := m.TryPutMapField(fd, key, val); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryPutMapField sets the value for the given map field descriptor and given
|
|
// key to the given value. An error is returned if the given field descriptor
|
|
// does not belong to the right message type, if the given field is not a map
|
|
// field, or if the given value is not a correct/compatible type for the given
|
|
// field.
|
|
//
|
|
// The Go type expected for a field is the same as required by TrySetField for
|
|
// a field with the same type as the map's value type.
|
|
//
|
|
// If the given field descriptor is not known (e.g. not present in the message
|
|
// descriptor) it will become known. Subsequent operations using tag numbers or
|
|
// names will be able to resolve the newly-known type. If the message has a
|
|
// value for the unknown value, it is cleared, replaced by the given known
|
|
// value.
|
|
func (m *Message) TryPutMapField(fd *desc.FieldDescriptor, key interface{}, val interface{}) error {
|
|
if err := m.checkField(fd); err != nil {
|
|
return err
|
|
}
|
|
return m.putMapField(fd, key, val)
|
|
}
|
|
|
|
// PutMapFieldByName sets the value for the map field with the given name and
|
|
// given key to the given value. It panics if an error is encountered. See
|
|
// TryPutMapFieldByName.
|
|
func (m *Message) PutMapFieldByName(name string, key interface{}, val interface{}) {
|
|
if err := m.TryPutMapFieldByName(name, key, val); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryPutMapFieldByName sets the value for the map field with the given name and
|
|
// the given key to the given value. An error is returned if the given name is
|
|
// unknown, if it names a field that is not a map, or if the given value has an
|
|
// incorrect type.
|
|
//
|
|
// (See TrySetField for more info on types.)
|
|
func (m *Message) TryPutMapFieldByName(name string, key interface{}, val interface{}) error {
|
|
fd := m.FindFieldDescriptorByName(name)
|
|
if fd == nil {
|
|
return UnknownFieldNameError
|
|
}
|
|
return m.putMapField(fd, key, val)
|
|
}
|
|
|
|
// PutMapFieldByNumber sets the value for the map field with the given tag
|
|
// number and given key to the given value. It panics if an error is
|
|
// encountered. See TryPutMapFieldByNumber.
|
|
func (m *Message) PutMapFieldByNumber(tagNumber int, key interface{}, val interface{}) {
|
|
if err := m.TryPutMapFieldByNumber(tagNumber, key, val); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryPutMapFieldByNumber sets the value for the map field with the given tag
|
|
// number and the given key to the given value. An error is returned if the
|
|
// given tag is unknown, if it indicates a field that is not a map, or if the
|
|
// given value has an incorrect type.
|
|
//
|
|
// (See TrySetField for more info on types.)
|
|
func (m *Message) TryPutMapFieldByNumber(tagNumber int, key interface{}, val interface{}) error {
|
|
fd := m.FindFieldDescriptor(int32(tagNumber))
|
|
if fd == nil {
|
|
return UnknownTagNumberError
|
|
}
|
|
return m.putMapField(fd, key, val)
|
|
}
|
|
|
|
func (m *Message) putMapField(fd *desc.FieldDescriptor, key interface{}, val interface{}) error {
|
|
if !fd.IsMap() {
|
|
return FieldIsNotMapError
|
|
}
|
|
kfd := fd.GetMessageType().GetFields()[0]
|
|
ki, err := validElementFieldValue(kfd, key)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
vfd := fd.GetMessageType().GetFields()[1]
|
|
vi, err := validElementFieldValue(vfd, val)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
mp := m.values[fd.GetNumber()]
|
|
if mp == nil {
|
|
if mp, err = m.parseUnknownField(fd); err != nil {
|
|
return err
|
|
} else if mp == nil {
|
|
mp = map[interface{}]interface{}{}
|
|
m.internalSetField(fd, map[interface{}]interface{}{ki: vi})
|
|
return nil
|
|
}
|
|
}
|
|
mp.(map[interface{}]interface{})[ki] = vi
|
|
return nil
|
|
}
|
|
|
|
// RemoveMapField changes the value for the given field descriptor by removing
|
|
// any value associated with the given key. It panics if an error is
|
|
// encountered. See TryRemoveMapField.
|
|
func (m *Message) RemoveMapField(fd *desc.FieldDescriptor, key interface{}) {
|
|
if err := m.TryRemoveMapField(fd, key); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryRemoveMapField changes the value for the given field descriptor by
|
|
// removing any value associated with the given key. An error is returned if the
|
|
// given field descriptor does not belong to the right message type or if the
|
|
// given field is not a map field.
|
|
//
|
|
// If the given field descriptor is not known (e.g. not present in the message
|
|
// descriptor) it will become known. Subsequent operations using tag numbers or
|
|
// names will be able to resolve the newly-known type. If the message has a
|
|
// value for the unknown value, it is parsed and any value for the given key
|
|
// removed.
|
|
func (m *Message) TryRemoveMapField(fd *desc.FieldDescriptor, key interface{}) error {
|
|
if err := m.checkField(fd); err != nil {
|
|
return err
|
|
}
|
|
return m.removeMapField(fd, key)
|
|
}
|
|
|
|
// RemoveMapFieldByName changes the value for the field with the given name by
|
|
// removing any value associated with the given key. It panics if an error is
|
|
// encountered. See TryRemoveMapFieldByName.
|
|
func (m *Message) RemoveMapFieldByName(name string, key interface{}) {
|
|
if err := m.TryRemoveMapFieldByName(name, key); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryRemoveMapFieldByName changes the value for the field with the given name
|
|
// by removing any value associated with the given key. An error is returned if
|
|
// the given name is unknown or if it names a field that is not a map.
|
|
func (m *Message) TryRemoveMapFieldByName(name string, key interface{}) error {
|
|
fd := m.FindFieldDescriptorByName(name)
|
|
if fd == nil {
|
|
return UnknownFieldNameError
|
|
}
|
|
return m.removeMapField(fd, key)
|
|
}
|
|
|
|
// RemoveMapFieldByNumber changes the value for the field with the given tag
|
|
// number by removing any value associated with the given key. It panics if an
|
|
// error is encountered. See TryRemoveMapFieldByNumber.
|
|
func (m *Message) RemoveMapFieldByNumber(tagNumber int, key interface{}) {
|
|
if err := m.TryRemoveMapFieldByNumber(tagNumber, key); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryRemoveMapFieldByNumber changes the value for the field with the given tag
|
|
// number by removing any value associated with the given key. An error is
|
|
// returned if the given tag is unknown or if it indicates a field that is not
|
|
// a map.
|
|
func (m *Message) TryRemoveMapFieldByNumber(tagNumber int, key interface{}) error {
|
|
fd := m.FindFieldDescriptor(int32(tagNumber))
|
|
if fd == nil {
|
|
return UnknownTagNumberError
|
|
}
|
|
return m.removeMapField(fd, key)
|
|
}
|
|
|
|
func (m *Message) removeMapField(fd *desc.FieldDescriptor, key interface{}) error {
|
|
if !fd.IsMap() {
|
|
return FieldIsNotMapError
|
|
}
|
|
kfd := fd.GetMessageType().GetFields()[0]
|
|
ki, err := validElementFieldValue(kfd, key)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
mp := m.values[fd.GetNumber()]
|
|
if mp == nil {
|
|
if mp, err = m.parseUnknownField(fd); err != nil {
|
|
return err
|
|
} else if mp == nil {
|
|
return nil
|
|
}
|
|
}
|
|
res := mp.(map[interface{}]interface{})
|
|
delete(res, ki)
|
|
if len(res) == 0 {
|
|
delete(m.values, fd.GetNumber())
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// FieldLength returns the number of elements in this message for the given
|
|
// field descriptor. It panics if an error is encountered. See TryFieldLength.
|
|
func (m *Message) FieldLength(fd *desc.FieldDescriptor) int {
|
|
l, err := m.TryFieldLength(fd)
|
|
if err != nil {
|
|
panic(err.Error())
|
|
}
|
|
return l
|
|
}
|
|
|
|
// TryFieldLength returns the number of elements in this message for the given
|
|
// field descriptor. An error is returned if the given field descriptor does not
|
|
// belong to the right message type or if it is neither a map field nor a
|
|
// repeated field.
|
|
func (m *Message) TryFieldLength(fd *desc.FieldDescriptor) (int, error) {
|
|
if err := m.checkField(fd); err != nil {
|
|
return 0, err
|
|
}
|
|
return m.fieldLength(fd)
|
|
}
|
|
|
|
// FieldLengthByName returns the number of elements in this message for the
|
|
// field with the given name. It panics if an error is encountered. See
|
|
// TryFieldLengthByName.
|
|
func (m *Message) FieldLengthByName(name string) int {
|
|
l, err := m.TryFieldLengthByName(name)
|
|
if err != nil {
|
|
panic(err.Error())
|
|
}
|
|
return l
|
|
}
|
|
|
|
// TryFieldLengthByName returns the number of elements in this message for the
|
|
// field with the given name. An error is returned if the given name is unknown
|
|
// or if the named field is neither a map field nor a repeated field.
|
|
func (m *Message) TryFieldLengthByName(name string) (int, error) {
|
|
fd := m.FindFieldDescriptorByName(name)
|
|
if fd == nil {
|
|
return 0, UnknownFieldNameError
|
|
}
|
|
return m.fieldLength(fd)
|
|
}
|
|
|
|
// FieldLengthByNumber returns the number of elements in this message for the
|
|
// field with the given tag number. It panics if an error is encountered. See
|
|
// TryFieldLengthByNumber.
|
|
func (m *Message) FieldLengthByNumber(tagNumber int32) int {
|
|
l, err := m.TryFieldLengthByNumber(tagNumber)
|
|
if err != nil {
|
|
panic(err.Error())
|
|
}
|
|
return l
|
|
}
|
|
|
|
// TryFieldLengthByNumber returns the number of elements in this message for the
|
|
// field with the given tag number. An error is returned if the given tag is
|
|
// unknown or if the named field is neither a map field nor a repeated field.
|
|
func (m *Message) TryFieldLengthByNumber(tagNumber int32) (int, error) {
|
|
fd := m.FindFieldDescriptor(int32(tagNumber))
|
|
if fd == nil {
|
|
return 0, UnknownTagNumberError
|
|
}
|
|
return m.fieldLength(fd)
|
|
}
|
|
|
|
func (m *Message) fieldLength(fd *desc.FieldDescriptor) (int, error) {
|
|
if !fd.IsRepeated() {
|
|
return 0, FieldIsNotRepeatedError
|
|
}
|
|
val := m.values[fd.GetNumber()]
|
|
if val == nil {
|
|
var err error
|
|
if val, err = m.parseUnknownField(fd); err != nil {
|
|
return 0, err
|
|
} else if val == nil {
|
|
return 0, nil
|
|
}
|
|
}
|
|
if sl, ok := val.([]interface{}); ok {
|
|
return len(sl), nil
|
|
} else if mp, ok := val.(map[interface{}]interface{}); ok {
|
|
return len(mp), nil
|
|
}
|
|
return 0, nil
|
|
}
|
|
|
|
// GetRepeatedField returns the value for the given repeated field descriptor at
|
|
// the given index. It panics if an error is encountered. See
|
|
// TryGetRepeatedField.
|
|
func (m *Message) GetRepeatedField(fd *desc.FieldDescriptor, index int) interface{} {
|
|
if v, err := m.TryGetRepeatedField(fd, index); err != nil {
|
|
panic(err.Error())
|
|
} else {
|
|
return v
|
|
}
|
|
}
|
|
|
|
// TryGetRepeatedField returns the value for the given repeated field descriptor
|
|
// at the given index. An error is returned if the given field descriptor does
|
|
// not belong to the right message type, if it is not a repeated field, or if
|
|
// the given index is out of range (less than zero or greater than or equal to
|
|
// the length of the repeated field). Also, even though map fields technically
|
|
// are repeated fields, if the given field is a map field an error will result:
|
|
// map representation does not lend itself to random access by index.
|
|
//
|
|
// The Go type of the value returned mirrors the type that protoc would generate
|
|
// for the field's element type. (See TryGetField for more details on types).
|
|
//
|
|
// If the given field descriptor is not known (e.g. not present in the message
|
|
// descriptor) but corresponds to an unknown field, the unknown value will be
|
|
// parsed and become known. The value at the given index in the parsed value
|
|
// will be returned. An error will be returned if the unknown value cannot be
|
|
// parsed according to the field descriptor's type information.
|
|
func (m *Message) TryGetRepeatedField(fd *desc.FieldDescriptor, index int) (interface{}, error) {
|
|
if index < 0 {
|
|
return nil, IndexOutOfRangeError
|
|
}
|
|
if err := m.checkField(fd); err != nil {
|
|
return nil, err
|
|
}
|
|
return m.getRepeatedField(fd, index)
|
|
}
|
|
|
|
// GetRepeatedFieldByName returns the value for the repeated field with the
|
|
// given name at the given index. It panics if an error is encountered. See
|
|
// TryGetRepeatedFieldByName.
|
|
func (m *Message) GetRepeatedFieldByName(name string, index int) interface{} {
|
|
if v, err := m.TryGetRepeatedFieldByName(name, index); err != nil {
|
|
panic(err.Error())
|
|
} else {
|
|
return v
|
|
}
|
|
}
|
|
|
|
// TryGetRepeatedFieldByName returns the value for the repeated field with the
|
|
// given name at the given index. An error is returned if the given name is
|
|
// unknown, if it names a field that is not a repeated field (or is a map
|
|
// field), or if the given index is out of range (less than zero or greater
|
|
// than or equal to the length of the repeated field).
|
|
//
|
|
// (See TryGetField for more info on types.)
|
|
func (m *Message) TryGetRepeatedFieldByName(name string, index int) (interface{}, error) {
|
|
if index < 0 {
|
|
return nil, IndexOutOfRangeError
|
|
}
|
|
fd := m.FindFieldDescriptorByName(name)
|
|
if fd == nil {
|
|
return nil, UnknownFieldNameError
|
|
}
|
|
return m.getRepeatedField(fd, index)
|
|
}
|
|
|
|
// GetRepeatedFieldByNumber returns the value for the repeated field with the
|
|
// given tag number at the given index. It panics if an error is encountered.
|
|
// See TryGetRepeatedFieldByNumber.
|
|
func (m *Message) GetRepeatedFieldByNumber(tagNumber int, index int) interface{} {
|
|
if v, err := m.TryGetRepeatedFieldByNumber(tagNumber, index); err != nil {
|
|
panic(err.Error())
|
|
} else {
|
|
return v
|
|
}
|
|
}
|
|
|
|
// TryGetRepeatedFieldByNumber returns the value for the repeated field with the
|
|
// given tag number at the given index. An error is returned if the given tag is
|
|
// unknown, if it indicates a field that is not a repeated field (or is a map
|
|
// field), or if the given index is out of range (less than zero or greater than
|
|
// or equal to the length of the repeated field).
|
|
//
|
|
// (See TryGetField for more info on types.)
|
|
func (m *Message) TryGetRepeatedFieldByNumber(tagNumber int, index int) (interface{}, error) {
|
|
if index < 0 {
|
|
return nil, IndexOutOfRangeError
|
|
}
|
|
fd := m.FindFieldDescriptor(int32(tagNumber))
|
|
if fd == nil {
|
|
return nil, UnknownTagNumberError
|
|
}
|
|
return m.getRepeatedField(fd, index)
|
|
}
|
|
|
|
func (m *Message) getRepeatedField(fd *desc.FieldDescriptor, index int) (interface{}, error) {
|
|
if fd.IsMap() || !fd.IsRepeated() {
|
|
return nil, FieldIsNotRepeatedError
|
|
}
|
|
sl := m.values[fd.GetNumber()]
|
|
if sl == nil {
|
|
var err error
|
|
if sl, err = m.parseUnknownField(fd); err != nil {
|
|
return nil, err
|
|
} else if sl == nil {
|
|
return nil, IndexOutOfRangeError
|
|
}
|
|
}
|
|
res := sl.([]interface{})
|
|
if index >= len(res) {
|
|
return nil, IndexOutOfRangeError
|
|
}
|
|
return res[index], nil
|
|
}
|
|
|
|
// AddRepeatedField appends the given value to the given repeated field. It
|
|
// panics if an error is encountered. See TryAddRepeatedField.
|
|
func (m *Message) AddRepeatedField(fd *desc.FieldDescriptor, val interface{}) {
|
|
if err := m.TryAddRepeatedField(fd, val); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryAddRepeatedField appends the given value to the given repeated field. An
|
|
// error is returned if the given field descriptor does not belong to the right
|
|
// message type, if the given field is not repeated, or if the given value is
|
|
// not a correct/compatible type for the given field. If the given field is a
|
|
// map field, the call will succeed if the given value is an instance of the
|
|
// map's entry message type.
|
|
//
|
|
// The Go type expected for a field is the same as required by TrySetField for
|
|
// a non-repeated field of the same type.
|
|
//
|
|
// If the given field descriptor is not known (e.g. not present in the message
|
|
// descriptor) it will become known. Subsequent operations using tag numbers or
|
|
// names will be able to resolve the newly-known type. If the message has a
|
|
// value for the unknown value, it is parsed and the given value is appended to
|
|
// it.
|
|
func (m *Message) TryAddRepeatedField(fd *desc.FieldDescriptor, val interface{}) error {
|
|
if err := m.checkField(fd); err != nil {
|
|
return err
|
|
}
|
|
return m.addRepeatedField(fd, val)
|
|
}
|
|
|
|
// AddRepeatedFieldByName appends the given value to the repeated field with the
|
|
// given name. It panics if an error is encountered. See
|
|
// TryAddRepeatedFieldByName.
|
|
func (m *Message) AddRepeatedFieldByName(name string, val interface{}) {
|
|
if err := m.TryAddRepeatedFieldByName(name, val); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryAddRepeatedFieldByName appends the given value to the repeated field with
|
|
// the given name. An error is returned if the given name is unknown, if it
|
|
// names a field that is not repeated, or if the given value has an incorrect
|
|
// type.
|
|
//
|
|
// (See TrySetField for more info on types.)
|
|
func (m *Message) TryAddRepeatedFieldByName(name string, val interface{}) error {
|
|
fd := m.FindFieldDescriptorByName(name)
|
|
if fd == nil {
|
|
return UnknownFieldNameError
|
|
}
|
|
return m.addRepeatedField(fd, val)
|
|
}
|
|
|
|
// AddRepeatedFieldByNumber appends the given value to the repeated field with
|
|
// the given tag number. It panics if an error is encountered. See
|
|
// TryAddRepeatedFieldByNumber.
|
|
func (m *Message) AddRepeatedFieldByNumber(tagNumber int, val interface{}) {
|
|
if err := m.TryAddRepeatedFieldByNumber(tagNumber, val); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TryAddRepeatedFieldByNumber appends the given value to the repeated field
|
|
// with the given tag number. An error is returned if the given tag is unknown,
|
|
// if it indicates a field that is not repeated, or if the given value has an
|
|
// incorrect type.
|
|
//
|
|
// (See TrySetField for more info on types.)
|
|
func (m *Message) TryAddRepeatedFieldByNumber(tagNumber int, val interface{}) error {
|
|
fd := m.FindFieldDescriptor(int32(tagNumber))
|
|
if fd == nil {
|
|
return UnknownTagNumberError
|
|
}
|
|
return m.addRepeatedField(fd, val)
|
|
}
|
|
|
|
func (m *Message) addRepeatedField(fd *desc.FieldDescriptor, val interface{}) error {
|
|
if !fd.IsRepeated() {
|
|
return FieldIsNotRepeatedError
|
|
}
|
|
val, err := validElementFieldValue(fd, val)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if fd.IsMap() {
|
|
// We're lenient. Just as we allow setting a map field to a slice of entry messages, we also allow
|
|
// adding entries one at a time (as if the field were a normal repeated field).
|
|
msg := val.(proto.Message)
|
|
dm, err := asDynamicMessage(msg, fd.GetMessageType(), m.mf)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
k, err := dm.TryGetFieldByNumber(1)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
v, err := dm.TryGetFieldByNumber(2)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return m.putMapField(fd, k, v)
|
|
}
|
|
|
|
sl := m.values[fd.GetNumber()]
|
|
if sl == nil {
|
|
if sl, err = m.parseUnknownField(fd); err != nil {
|
|
return err
|
|
} else if sl == nil {
|
|
sl = []interface{}{}
|
|
}
|
|
}
|
|
res := sl.([]interface{})
|
|
res = append(res, val)
|
|
m.internalSetField(fd, res)
|
|
return nil
|
|
}
|
|
|
|
// SetRepeatedField sets the value for the given repeated field descriptor and
|
|
// given index to the given value. It panics if an error is encountered. See
|
|
// SetRepeatedField.
|
|
func (m *Message) SetRepeatedField(fd *desc.FieldDescriptor, index int, val interface{}) {
|
|
if err := m.TrySetRepeatedField(fd, index, val); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TrySetRepeatedField sets the value for the given repeated field descriptor
|
|
// and given index to the given value. An error is returned if the given field
|
|
// descriptor does not belong to the right message type, if the given field is
|
|
// not repeated, or if the given value is not a correct/compatible type for the
|
|
// given field. Also, even though map fields technically are repeated fields, if
|
|
// the given field is a map field an error will result: map representation does
|
|
// not lend itself to random access by index.
|
|
//
|
|
// The Go type expected for a field is the same as required by TrySetField for
|
|
// a non-repeated field of the same type.
|
|
//
|
|
// If the given field descriptor is not known (e.g. not present in the message
|
|
// descriptor) it will become known. Subsequent operations using tag numbers or
|
|
// names will be able to resolve the newly-known type. If the message has a
|
|
// value for the unknown value, it is parsed and the element at the given index
|
|
// is replaced with the given value.
|
|
func (m *Message) TrySetRepeatedField(fd *desc.FieldDescriptor, index int, val interface{}) error {
|
|
if index < 0 {
|
|
return IndexOutOfRangeError
|
|
}
|
|
if err := m.checkField(fd); err != nil {
|
|
return err
|
|
}
|
|
return m.setRepeatedField(fd, index, val)
|
|
}
|
|
|
|
// SetRepeatedFieldByName sets the value for the repeated field with the given
|
|
// name and given index to the given value. It panics if an error is
|
|
// encountered. See TrySetRepeatedFieldByName.
|
|
func (m *Message) SetRepeatedFieldByName(name string, index int, val interface{}) {
|
|
if err := m.TrySetRepeatedFieldByName(name, index, val); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TrySetRepeatedFieldByName sets the value for the repeated field with the
|
|
// given name and the given index to the given value. An error is returned if
|
|
// the given name is unknown, if it names a field that is not repeated (or is a
|
|
// map field), or if the given value has an incorrect type.
|
|
//
|
|
// (See TrySetField for more info on types.)
|
|
func (m *Message) TrySetRepeatedFieldByName(name string, index int, val interface{}) error {
|
|
if index < 0 {
|
|
return IndexOutOfRangeError
|
|
}
|
|
fd := m.FindFieldDescriptorByName(name)
|
|
if fd == nil {
|
|
return UnknownFieldNameError
|
|
}
|
|
return m.setRepeatedField(fd, index, val)
|
|
}
|
|
|
|
// SetRepeatedFieldByNumber sets the value for the repeated field with the given
|
|
// tag number and given index to the given value. It panics if an error is
|
|
// encountered. See TrySetRepeatedFieldByNumber.
|
|
func (m *Message) SetRepeatedFieldByNumber(tagNumber int, index int, val interface{}) {
|
|
if err := m.TrySetRepeatedFieldByNumber(tagNumber, index, val); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
// TrySetRepeatedFieldByNumber sets the value for the repeated field with the
|
|
// given tag number and the given index to the given value. An error is returned
|
|
// if the given tag is unknown, if it indicates a field that is not repeated (or
|
|
// is a map field), or if the given value has an incorrect type.
|
|
//
|
|
// (See TrySetField for more info on types.)
|
|
func (m *Message) TrySetRepeatedFieldByNumber(tagNumber int, index int, val interface{}) error {
|
|
if index < 0 {
|
|
return IndexOutOfRangeError
|
|
}
|
|
fd := m.FindFieldDescriptor(int32(tagNumber))
|
|
if fd == nil {
|
|
return UnknownTagNumberError
|
|
}
|
|
return m.setRepeatedField(fd, index, val)
|
|
}
|
|
|
|
func (m *Message) setRepeatedField(fd *desc.FieldDescriptor, index int, val interface{}) error {
|
|
if fd.IsMap() || !fd.IsRepeated() {
|
|
return FieldIsNotRepeatedError
|
|
}
|
|
val, err := validElementFieldValue(fd, val)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
sl := m.values[fd.GetNumber()]
|
|
if sl == nil {
|
|
if sl, err = m.parseUnknownField(fd); err != nil {
|
|
return err
|
|
} else if sl == nil {
|
|
return IndexOutOfRangeError
|
|
}
|
|
}
|
|
res := sl.([]interface{})
|
|
if index >= len(res) {
|
|
return IndexOutOfRangeError
|
|
}
|
|
res[index] = val
|
|
return nil
|
|
}
|
|
|
|
// GetUnknownField gets the value(s) for the given unknown tag number. If this
|
|
// message has no unknown fields with the given tag, nil is returned.
|
|
func (m *Message) GetUnknownField(tagNumber int32) []UnknownField {
|
|
if u, ok := m.unknownFields[tagNumber]; ok {
|
|
return u
|
|
} else {
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func (m *Message) parseUnknownField(fd *desc.FieldDescriptor) (interface{}, error) {
|
|
unks, ok := m.unknownFields[fd.GetNumber()]
|
|
if !ok {
|
|
return nil, nil
|
|
}
|
|
var v interface{}
|
|
var sl []interface{}
|
|
var mp map[interface{}]interface{}
|
|
if fd.IsMap() {
|
|
mp = map[interface{}]interface{}{}
|
|
}
|
|
var err error
|
|
for _, unk := range unks {
|
|
var val interface{}
|
|
if unk.Encoding == proto.WireBytes || unk.Encoding == proto.WireStartGroup {
|
|
val, err = unmarshalLengthDelimitedField(fd, unk.Contents, m.mf)
|
|
} else {
|
|
val, err = unmarshalSimpleField(fd, unk.Value)
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if fd.IsMap() {
|
|
newEntry := val.(*Message)
|
|
kk, err := newEntry.TryGetFieldByNumber(1)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
vv, err := newEntry.TryGetFieldByNumber(2)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
mp[kk] = vv
|
|
v = mp
|
|
} else if fd.IsRepeated() {
|
|
t := reflect.TypeOf(val)
|
|
if t.Kind() == reflect.Slice && t != typeOfBytes {
|
|
// append slices if we unmarshalled a packed repeated field
|
|
newVals := val.([]interface{})
|
|
sl = append(sl, newVals...)
|
|
} else {
|
|
sl = append(sl, val)
|
|
}
|
|
v = sl
|
|
} else {
|
|
v = val
|
|
}
|
|
}
|
|
m.internalSetField(fd, v)
|
|
return v, nil
|
|
}
|
|
|
|
func validFieldValue(fd *desc.FieldDescriptor, val interface{}) (interface{}, error) {
|
|
return validFieldValueForRv(fd, reflect.ValueOf(val))
|
|
}
|
|
|
|
func validFieldValueForRv(fd *desc.FieldDescriptor, val reflect.Value) (interface{}, error) {
|
|
if fd.IsMap() && val.Kind() == reflect.Map {
|
|
// make a defensive copy while we check the contents
|
|
// (also converts to map[interface{}]interface{} if it's some other type)
|
|
keyField := fd.GetMessageType().GetFields()[0]
|
|
valField := fd.GetMessageType().GetFields()[1]
|
|
m := map[interface{}]interface{}{}
|
|
for _, k := range val.MapKeys() {
|
|
if k.Kind() == reflect.Interface {
|
|
// unwrap it
|
|
k = reflect.ValueOf(k.Interface())
|
|
}
|
|
kk, err := validFieldValueForRv(keyField, k)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
v := val.MapIndex(k)
|
|
if v.Kind() == reflect.Interface {
|
|
// unwrap it
|
|
v = reflect.ValueOf(v.Interface())
|
|
}
|
|
vv, err := validFieldValueForRv(valField, v)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
m[kk] = vv
|
|
}
|
|
return m, nil
|
|
}
|
|
|
|
if fd.IsRepeated() { // this will also catch map fields where given value was not a map
|
|
if val.Kind() != reflect.Array && val.Kind() != reflect.Slice {
|
|
if fd.IsMap() {
|
|
return nil, fmt.Errorf("value for map field must be a map; instead was %v", val.Type())
|
|
} else {
|
|
return nil, fmt.Errorf("value for repeated field must be a slice; instead was %v", val.Type())
|
|
}
|
|
}
|
|
|
|
if fd.IsMap() {
|
|
// value should be a slice of entry messages that we need convert into a map[interface{}]interface{}
|
|
m := map[interface{}]interface{}{}
|
|
for i := 0; i < val.Len(); i++ {
|
|
e, err := validElementFieldValue(fd, val.Index(i).Interface())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
msg := e.(proto.Message)
|
|
dm, err := asDynamicMessage(msg, fd.GetMessageType(), nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
k, err := dm.TryGetFieldByNumber(1)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
v, err := dm.TryGetFieldByNumber(2)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
m[k] = v
|
|
}
|
|
return m, nil
|
|
}
|
|
|
|
// make a defensive copy while checking contents (also converts to []interface{})
|
|
s := make([]interface{}, val.Len())
|
|
for i := 0; i < val.Len(); i++ {
|
|
ev := val.Index(i)
|
|
if ev.Kind() == reflect.Interface {
|
|
// unwrap it
|
|
ev = reflect.ValueOf(ev.Interface())
|
|
}
|
|
e, err := validElementFieldValueForRv(fd, ev)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
s[i] = e
|
|
}
|
|
|
|
return s, nil
|
|
}
|
|
|
|
return validElementFieldValueForRv(fd, val)
|
|
}
|
|
|
|
func asDynamicMessage(m proto.Message, md *desc.MessageDescriptor, mf *MessageFactory) (*Message, error) {
|
|
if dm, ok := m.(*Message); ok {
|
|
return dm, nil
|
|
}
|
|
dm := NewMessageWithMessageFactory(md, mf)
|
|
if err := dm.mergeFrom(m); err != nil {
|
|
return nil, err
|
|
}
|
|
return dm, nil
|
|
}
|
|
|
|
func validElementFieldValue(fd *desc.FieldDescriptor, val interface{}) (interface{}, error) {
|
|
return validElementFieldValueForRv(fd, reflect.ValueOf(val))
|
|
}
|
|
|
|
func validElementFieldValueForRv(fd *desc.FieldDescriptor, val reflect.Value) (interface{}, error) {
|
|
t := fd.GetType()
|
|
typeName := strings.ToLower(t.String())
|
|
if !val.IsValid() {
|
|
return nil, fmt.Errorf("%s field %s is not compatible with nil value", typeName, fd.GetFullyQualifiedName())
|
|
}
|
|
switch t {
|
|
case descriptor.FieldDescriptorProto_TYPE_SFIXED32,
|
|
descriptor.FieldDescriptorProto_TYPE_INT32,
|
|
descriptor.FieldDescriptorProto_TYPE_SINT32,
|
|
descriptor.FieldDescriptorProto_TYPE_ENUM:
|
|
return toInt32(reflect.Indirect(val), typeName, fd.GetFullyQualifiedName())
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_SFIXED64,
|
|
descriptor.FieldDescriptorProto_TYPE_INT64,
|
|
descriptor.FieldDescriptorProto_TYPE_SINT64:
|
|
return toInt64(reflect.Indirect(val), typeName, fd.GetFullyQualifiedName())
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_FIXED32,
|
|
descriptor.FieldDescriptorProto_TYPE_UINT32:
|
|
return toUint32(reflect.Indirect(val), typeName, fd.GetFullyQualifiedName())
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_FIXED64,
|
|
descriptor.FieldDescriptorProto_TYPE_UINT64:
|
|
return toUint64(reflect.Indirect(val), typeName, fd.GetFullyQualifiedName())
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_FLOAT:
|
|
return toFloat32(reflect.Indirect(val), typeName, fd.GetFullyQualifiedName())
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_DOUBLE:
|
|
return toFloat64(reflect.Indirect(val), typeName, fd.GetFullyQualifiedName())
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_BOOL:
|
|
return toBool(reflect.Indirect(val), typeName, fd.GetFullyQualifiedName())
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_BYTES:
|
|
return toBytes(reflect.Indirect(val), typeName, fd.GetFullyQualifiedName())
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_STRING:
|
|
return toString(reflect.Indirect(val), typeName, fd.GetFullyQualifiedName())
|
|
|
|
case descriptor.FieldDescriptorProto_TYPE_MESSAGE,
|
|
descriptor.FieldDescriptorProto_TYPE_GROUP:
|
|
m, err := asMessage(val, fd.GetFullyQualifiedName())
|
|
// check that message is correct type
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var msgType string
|
|
if dm, ok := m.(*Message); ok {
|
|
msgType = dm.GetMessageDescriptor().GetFullyQualifiedName()
|
|
} else {
|
|
msgType = proto.MessageName(m)
|
|
}
|
|
if msgType != fd.GetMessageType().GetFullyQualifiedName() {
|
|
return nil, fmt.Errorf("message field %s requires value of type %s; received %s", fd.GetFullyQualifiedName(), fd.GetMessageType().GetFullyQualifiedName(), msgType)
|
|
}
|
|
return m, nil
|
|
|
|
default:
|
|
return nil, fmt.Errorf("unable to handle unrecognized field type: %v", fd.GetType())
|
|
}
|
|
}
|
|
|
|
func toInt32(v reflect.Value, what string, fieldName string) (int32, error) {
|
|
if v.Kind() == reflect.Int32 {
|
|
return int32(v.Int()), nil
|
|
}
|
|
return 0, fmt.Errorf("%s field %s is not compatible with value of type %v", what, fieldName, v.Type())
|
|
}
|
|
|
|
func toUint32(v reflect.Value, what string, fieldName string) (uint32, error) {
|
|
if v.Kind() == reflect.Uint32 {
|
|
return uint32(v.Uint()), nil
|
|
}
|
|
return 0, fmt.Errorf("%s field %s is not compatible with value of type %v", what, fieldName, v.Type())
|
|
}
|
|
|
|
func toFloat32(v reflect.Value, what string, fieldName string) (float32, error) {
|
|
if v.Kind() == reflect.Float32 {
|
|
return float32(v.Float()), nil
|
|
}
|
|
return 0, fmt.Errorf("%s field %s is not compatible with value of type %v", what, fieldName, v.Type())
|
|
}
|
|
|
|
func toInt64(v reflect.Value, what string, fieldName string) (int64, error) {
|
|
if v.Kind() == reflect.Int64 || v.Kind() == reflect.Int || v.Kind() == reflect.Int32 {
|
|
return v.Int(), nil
|
|
}
|
|
return 0, fmt.Errorf("%s field %s is not compatible with value of type %v", what, fieldName, v.Type())
|
|
}
|
|
|
|
func toUint64(v reflect.Value, what string, fieldName string) (uint64, error) {
|
|
if v.Kind() == reflect.Uint64 || v.Kind() == reflect.Uint || v.Kind() == reflect.Uint32 {
|
|
return v.Uint(), nil
|
|
}
|
|
return 0, fmt.Errorf("%s field %s is not compatible with value of type %v", what, fieldName, v.Type())
|
|
}
|
|
|
|
func toFloat64(v reflect.Value, what string, fieldName string) (float64, error) {
|
|
if v.Kind() == reflect.Float64 || v.Kind() == reflect.Float32 {
|
|
return v.Float(), nil
|
|
}
|
|
return 0, fmt.Errorf("%s field %s is not compatible with value of type %v", what, fieldName, v.Type())
|
|
}
|
|
|
|
func toBool(v reflect.Value, what string, fieldName string) (bool, error) {
|
|
if v.Kind() == reflect.Bool {
|
|
return v.Bool(), nil
|
|
}
|
|
return false, fmt.Errorf("%s field %s is not compatible with value of type %v", what, fieldName, v.Type())
|
|
}
|
|
|
|
func toBytes(v reflect.Value, what string, fieldName string) ([]byte, error) {
|
|
if v.Kind() == reflect.Slice && v.Type().Elem().Kind() == reflect.Uint8 {
|
|
return v.Bytes(), nil
|
|
}
|
|
return nil, fmt.Errorf("%s field %s is not compatible with value of type %v", what, fieldName, v.Type())
|
|
}
|
|
|
|
func toString(v reflect.Value, what string, fieldName string) (string, error) {
|
|
if v.Kind() == reflect.String {
|
|
return v.String(), nil
|
|
}
|
|
return "", fmt.Errorf("%s field %s is not compatible with value of type %v", what, fieldName, v.Type())
|
|
}
|
|
|
|
func asMessage(v reflect.Value, fieldName string) (proto.Message, error) {
|
|
t := v.Type()
|
|
// we need a pointer to a struct that implements proto.Message
|
|
if t.Kind() != reflect.Ptr || t.Elem().Kind() != reflect.Struct || !t.Implements(typeOfProtoMessage) {
|
|
return nil, fmt.Errorf("message field %s requires is not compatible with value of type %v", fieldName, v.Type())
|
|
}
|
|
return v.Interface().(proto.Message), nil
|
|
}
|
|
|
|
// Reset resets this message to an empty message. It removes all values set in
|
|
// the message.
|
|
func (m *Message) Reset() {
|
|
m.values = nil
|
|
m.unknownFields = nil
|
|
}
|
|
|
|
// String returns this message rendered in compact text format.
|
|
func (m *Message) String() string {
|
|
b, err := m.MarshalText()
|
|
if err != nil {
|
|
panic(fmt.Sprintf("Failed to create string representation of message: %s", err.Error()))
|
|
}
|
|
return string(b)
|
|
}
|
|
|
|
// ProtoMessage is present to satisfy the proto.Message interface.
|
|
func (m *Message) ProtoMessage() {
|
|
}
|
|
|
|
// ConvertTo converts this dynamic message into the given message. This is
|
|
// shorthand for resetting then merging:
|
|
// target.Reset()
|
|
// m.MergeInto(target)
|
|
func (m *Message) ConvertTo(target proto.Message) error {
|
|
if err := m.checkType(target); err != nil {
|
|
return err
|
|
}
|
|
|
|
target.Reset()
|
|
return m.mergeInto(target)
|
|
}
|
|
|
|
// ConvertFrom converts the given message into this dynamic message. This is
|
|
// shorthand for resetting then merging:
|
|
// m.Reset()
|
|
// m.MergeFrom(target)
|
|
func (m *Message) ConvertFrom(target proto.Message) error {
|
|
if err := m.checkType(target); err != nil {
|
|
return err
|
|
}
|
|
|
|
m.Reset()
|
|
return m.mergeFrom(target)
|
|
}
|
|
|
|
// MergeInto merges this dynamic message into the given message. All field
|
|
// values in this message will be set on the given message. For map fields,
|
|
// entries are added to the given message (if the given message has existing
|
|
// values for like keys, they are overwritten). For slice fields, elements are
|
|
// added.
|
|
//
|
|
// If the given message has a different set of known fields, it is possible for
|
|
// some known fields in this message to be represented as unknown fields in the
|
|
// given message after merging, and vice versa.
|
|
func (m *Message) MergeInto(target proto.Message) error {
|
|
if err := m.checkType(target); err != nil {
|
|
return err
|
|
}
|
|
return m.mergeInto(target)
|
|
}
|
|
|
|
// MergeFrom merges the given message into this dynamic message. All field
|
|
// values in the given message will be set on this message. For map fields,
|
|
// entries are added to this message (if this message has existing values for
|
|
// like keys, they are overwritten). For slice fields, elements are added.
|
|
//
|
|
// If the given message has a different set of known fields, it is possible for
|
|
// some known fields in that message to be represented as unknown fields in this
|
|
// message after merging, and vice versa.
|
|
func (m *Message) MergeFrom(source proto.Message) error {
|
|
if err := m.checkType(source); err != nil {
|
|
return err
|
|
}
|
|
return m.mergeFrom(source)
|
|
}
|
|
|
|
// Merge implements the proto.Merger interface so that dynamic messages are
|
|
// compatible with the proto.Merge function. It delegates to MergeFrom but will
|
|
// panic on error as the proto.Merger interface doesn't allow for returning an
|
|
// error.
|
|
//
|
|
// Unlike nearly all other methods, this method can work if this message's type
|
|
// is not defined (such as instantiating the message without using NewMessage).
|
|
// This is strictly so that dynamic message's are compatible with the
|
|
// proto.Clone function, which instantiates a new message via reflection (thus
|
|
// its message descriptor will not be set) and than calls Merge.
|
|
func (m *Message) Merge(source proto.Message) {
|
|
if m.md == nil {
|
|
// To support proto.Clone, initialize the descriptor from the source.
|
|
if dm, ok := source.(*Message); ok {
|
|
m.md = dm.md
|
|
m.er = dm.er
|
|
m.extraFields = dm.extraFields
|
|
} else if md, err := desc.LoadMessageDescriptorForMessage(source); err != nil {
|
|
panic(err.Error())
|
|
} else {
|
|
m.md = md
|
|
}
|
|
}
|
|
|
|
if err := m.MergeFrom(source); err != nil {
|
|
panic(err.Error())
|
|
}
|
|
}
|
|
|
|
func (m *Message) checkType(target proto.Message) error {
|
|
if dm, ok := target.(*Message); ok {
|
|
if dm.md.GetFullyQualifiedName() != m.md.GetFullyQualifiedName() {
|
|
return fmt.Errorf("given message has wrong type: %q; expecting %q", dm.md.GetFullyQualifiedName(), m.md.GetFullyQualifiedName())
|
|
}
|
|
return nil
|
|
}
|
|
|
|
msgName := proto.MessageName(target)
|
|
if msgName != m.md.GetFullyQualifiedName() {
|
|
return fmt.Errorf("given message has wrong type: %q; expecting %q", msgName, m.md.GetFullyQualifiedName())
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (m *Message) mergeInto(pm proto.Message) error {
|
|
if dm, ok := pm.(*Message); ok {
|
|
return dm.mergeFrom(m)
|
|
}
|
|
|
|
target := reflect.ValueOf(pm)
|
|
if target.Kind() == reflect.Ptr {
|
|
target = target.Elem()
|
|
}
|
|
|
|
// track tags for which the dynamic message has data but the given
|
|
// message doesn't know about it
|
|
u := target.FieldByName("XXX_unrecognized")
|
|
var unknownTags map[int32]struct{}
|
|
if u.IsValid() && u.Type() == typeOfBytes {
|
|
unknownTags = map[int32]struct{}{}
|
|
for tag := range m.values {
|
|
unknownTags[tag] = struct{}{}
|
|
}
|
|
}
|
|
|
|
// check that we can successfully do the merge
|
|
structProps := proto.GetProperties(reflect.TypeOf(pm).Elem())
|
|
for _, prop := range structProps.Prop {
|
|
if prop.Tag == 0 {
|
|
continue // one-of or special field (such as XXX_unrecognized, etc.)
|
|
}
|
|
tag := int32(prop.Tag)
|
|
v, ok := m.values[tag]
|
|
if !ok {
|
|
continue
|
|
}
|
|
if unknownTags != nil {
|
|
delete(unknownTags, tag)
|
|
}
|
|
f := target.FieldByName(prop.Name)
|
|
ft := f.Type()
|
|
val := reflect.ValueOf(v)
|
|
if !canConvert(val, ft) {
|
|
return fmt.Errorf("cannot convert %v to %v", val.Type(), ft)
|
|
}
|
|
}
|
|
// check one-of fields
|
|
for _, oop := range structProps.OneofTypes {
|
|
prop := oop.Prop
|
|
tag := int32(prop.Tag)
|
|
v, ok := m.values[tag]
|
|
if !ok {
|
|
continue
|
|
}
|
|
if unknownTags != nil {
|
|
delete(unknownTags, tag)
|
|
}
|
|
stf, ok := oop.Type.Elem().FieldByName(prop.Name)
|
|
if !ok {
|
|
return fmt.Errorf("one-of field indicates struct field name %s, but type %v has no such field", prop.Name, oop.Type.Elem())
|
|
}
|
|
ft := stf.Type
|
|
val := reflect.ValueOf(v)
|
|
if !canConvert(val, ft) {
|
|
return fmt.Errorf("cannot convert %v to %v", val.Type(), ft)
|
|
}
|
|
}
|
|
// and check extensions, too
|
|
for tag, ext := range proto.RegisteredExtensions(pm) {
|
|
v, ok := m.values[tag]
|
|
if !ok {
|
|
continue
|
|
}
|
|
if unknownTags != nil {
|
|
delete(unknownTags, tag)
|
|
}
|
|
ft := reflect.TypeOf(ext.ExtensionType)
|
|
val := reflect.ValueOf(v)
|
|
if !canConvert(val, ft) {
|
|
return fmt.Errorf("cannot convert %v to %v", val.Type(), ft)
|
|
}
|
|
}
|
|
|
|
// now actually perform the merge
|
|
for _, prop := range structProps.Prop {
|
|
v, ok := m.values[int32(prop.Tag)]
|
|
if !ok {
|
|
continue
|
|
}
|
|
f := target.FieldByName(prop.Name)
|
|
mergeVal(reflect.ValueOf(v), f)
|
|
}
|
|
// merge one-ofs
|
|
for _, oop := range structProps.OneofTypes {
|
|
prop := oop.Prop
|
|
tag := int32(prop.Tag)
|
|
v, ok := m.values[tag]
|
|
if !ok {
|
|
continue
|
|
}
|
|
oov := reflect.New(oop.Type.Elem())
|
|
f := oov.Elem().FieldByName(prop.Name)
|
|
mergeVal(reflect.ValueOf(v), f)
|
|
target.Field(oop.Field).Set(oov)
|
|
}
|
|
// merge extensions, too
|
|
for tag, ext := range proto.RegisteredExtensions(pm) {
|
|
v, ok := m.values[tag]
|
|
if !ok {
|
|
continue
|
|
}
|
|
e := reflect.New(reflect.TypeOf(ext.ExtensionType)).Elem()
|
|
mergeVal(reflect.ValueOf(v), e)
|
|
if err := proto.SetExtension(pm, ext, e.Interface()); err != nil {
|
|
// shouldn't happen since we already checked that the extension type was compatible above
|
|
return err
|
|
}
|
|
}
|
|
|
|
// if we have fields that the given message doesn't know about, add to its unknown fields
|
|
if len(unknownTags) > 0 {
|
|
ub := u.Interface().([]byte)
|
|
var b codedBuffer
|
|
for tag := range unknownTags {
|
|
fd := m.FindFieldDescriptor(tag)
|
|
if err := marshalField(tag, fd, m.values[tag], &b, false); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
ub = append(ub, b.buf...)
|
|
u.Set(reflect.ValueOf(ub))
|
|
}
|
|
|
|
// finally, convey unknown fields into the given message by letting it unmarshal them
|
|
// (this will append to its unknown fields if not known; if somehow the given message recognizes
|
|
// a field even though the dynamic message did not, it will get correctly unmarshalled)
|
|
if unknownTags != nil && len(m.unknownFields) > 0 {
|
|
var b codedBuffer
|
|
m.marshalUnknownFields(&b)
|
|
proto.UnmarshalMerge(b.buf, pm)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func canConvert(src reflect.Value, target reflect.Type) bool {
|
|
if src.Kind() == reflect.Interface {
|
|
src = reflect.ValueOf(src.Interface())
|
|
}
|
|
srcType := src.Type()
|
|
// we allow convertible types instead of requiring exact types so that calling
|
|
// code can, for example, assign an enum constant to an enum field. In that case,
|
|
// one type is the enum type (a sub-type of int32) and the other may be the int32
|
|
// type. So we automatically do the conversion in that case.
|
|
if srcType.ConvertibleTo(target) {
|
|
return true
|
|
} else if target.Kind() == reflect.Ptr && srcType.ConvertibleTo(target.Elem()) {
|
|
return true
|
|
} else if target.Kind() == reflect.Slice {
|
|
if srcType.Kind() != reflect.Slice {
|
|
return false
|
|
}
|
|
et := target.Elem()
|
|
for i := 0; i < src.Len(); i++ {
|
|
if !canConvert(src.Index(i), et) {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
} else if target.Kind() == reflect.Map {
|
|
if srcType.Kind() != reflect.Map {
|
|
return false
|
|
}
|
|
kt := target.Key()
|
|
vt := target.Elem()
|
|
for _, k := range src.MapKeys() {
|
|
if !canConvert(k, kt) {
|
|
return false
|
|
}
|
|
if !canConvert(src.MapIndex(k), vt) {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
} else if srcType == typeOfDynamicMessage && target.Implements(typeOfProtoMessage) {
|
|
z := reflect.Zero(target).Interface()
|
|
msgType := proto.MessageName(z.(proto.Message))
|
|
return msgType == src.Interface().(*Message).GetMessageDescriptor().GetFullyQualifiedName()
|
|
} else {
|
|
return false
|
|
}
|
|
}
|
|
|
|
func mergeVal(src, target reflect.Value) {
|
|
if src.Kind() == reflect.Interface && !src.IsNil() {
|
|
src = src.Elem()
|
|
}
|
|
srcType := src.Type()
|
|
targetType := target.Type()
|
|
if srcType.ConvertibleTo(targetType) {
|
|
if targetType.Implements(typeOfProtoMessage) && !target.IsNil() {
|
|
Merge(target.Interface().(proto.Message), src.Convert(targetType).Interface().(proto.Message))
|
|
} else {
|
|
target.Set(src.Convert(targetType))
|
|
}
|
|
} else if targetType.Kind() == reflect.Ptr && srcType.ConvertibleTo(targetType.Elem()) {
|
|
if !src.CanAddr() {
|
|
target.Set(reflect.New(targetType.Elem()))
|
|
target.Elem().Set(src.Convert(targetType.Elem()))
|
|
} else {
|
|
target.Set(src.Addr().Convert(targetType))
|
|
}
|
|
} else if targetType.Kind() == reflect.Slice {
|
|
l := target.Len()
|
|
newL := l + src.Len()
|
|
if target.Cap() < newL {
|
|
// expand capacity of the slice and copy
|
|
newSl := reflect.MakeSlice(targetType, newL, newL)
|
|
for i := 0; i < target.Len(); i++ {
|
|
newSl.Index(i).Set(target.Index(i))
|
|
}
|
|
target.Set(newSl)
|
|
} else {
|
|
target.SetLen(newL)
|
|
}
|
|
for i := 0; i < src.Len(); i++ {
|
|
dest := target.Index(l + i)
|
|
if dest.Kind() == reflect.Ptr {
|
|
dest.Set(reflect.New(dest.Type().Elem()))
|
|
}
|
|
mergeVal(src.Index(i), dest)
|
|
}
|
|
} else if targetType.Kind() == reflect.Map {
|
|
tkt := targetType.Key()
|
|
tvt := targetType.Elem()
|
|
for _, k := range src.MapKeys() {
|
|
v := src.MapIndex(k)
|
|
skt := k.Type()
|
|
svt := v.Type()
|
|
var nk, nv reflect.Value
|
|
if tkt == skt {
|
|
nk = k
|
|
} else if tkt.Kind() == reflect.Ptr && tkt.Elem() == skt {
|
|
nk = k.Addr()
|
|
} else {
|
|
nk = reflect.New(tkt).Elem()
|
|
mergeVal(k, nk)
|
|
}
|
|
if tvt == svt {
|
|
nv = v
|
|
} else if tvt.Kind() == reflect.Ptr && tvt.Elem() == svt {
|
|
nv = v.Addr()
|
|
} else {
|
|
nv = reflect.New(tvt).Elem()
|
|
mergeVal(v, nv)
|
|
}
|
|
if target.IsNil() {
|
|
target.Set(reflect.MakeMap(targetType))
|
|
}
|
|
target.SetMapIndex(nk, nv)
|
|
}
|
|
} else if srcType == typeOfDynamicMessage && targetType.Implements(typeOfProtoMessage) {
|
|
dm := src.Interface().(*Message)
|
|
if target.IsNil() {
|
|
target.Set(reflect.New(targetType.Elem()))
|
|
}
|
|
m := target.Interface().(proto.Message)
|
|
dm.mergeInto(m)
|
|
} else {
|
|
panic(fmt.Sprintf("cannot convert %v to %v", srcType, targetType))
|
|
}
|
|
}
|
|
|
|
func (m *Message) mergeFrom(pm proto.Message) error {
|
|
if dm, ok := pm.(*Message); ok {
|
|
// if given message is also a dynamic message, we merge differently
|
|
for tag, v := range dm.values {
|
|
fd := m.FindFieldDescriptor(tag)
|
|
if fd == nil {
|
|
fd = dm.FindFieldDescriptor(tag)
|
|
}
|
|
if err := mergeField(m, fd, v); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
pmrv := reflect.ValueOf(pm)
|
|
if pmrv.IsNil() {
|
|
// nil is an empty message, so nothing to do
|
|
return nil
|
|
}
|
|
|
|
// check that we can successfully do the merge
|
|
src := pmrv.Elem()
|
|
values := map[*desc.FieldDescriptor]interface{}{}
|
|
props := proto.GetProperties(reflect.TypeOf(pm).Elem())
|
|
if props == nil {
|
|
return fmt.Errorf("could not determine message properties to merge for %v", reflect.TypeOf(pm).Elem())
|
|
}
|
|
|
|
// regular fields
|
|
for _, prop := range props.Prop {
|
|
if prop.Tag == 0 {
|
|
continue // one-of or special field (such as XXX_unrecognized, etc.)
|
|
}
|
|
fd := m.FindFieldDescriptor(int32(prop.Tag))
|
|
if fd == nil {
|
|
// Our descriptor has different fields than this message object. So
|
|
// try to reflect on the message object's fields.
|
|
md, err := desc.LoadMessageDescriptorForMessage(pm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
fd = md.FindFieldByNumber(int32(prop.Tag))
|
|
if fd == nil {
|
|
return fmt.Errorf("message descriptor %q did not contain field for tag %d (%q)", md.GetFullyQualifiedName(), prop.Tag, prop.Name)
|
|
}
|
|
}
|
|
rv := src.FieldByName(prop.Name)
|
|
if (rv.Kind() == reflect.Ptr || rv.Kind() == reflect.Slice) && rv.IsNil() {
|
|
continue
|
|
}
|
|
if v, err := validFieldValueForRv(fd, rv); err != nil {
|
|
return err
|
|
} else {
|
|
values[fd] = v
|
|
}
|
|
}
|
|
|
|
// one-of fields
|
|
for _, oop := range props.OneofTypes {
|
|
oov := src.Field(oop.Field).Elem()
|
|
if !oov.IsValid() || oov.Type() != oop.Type {
|
|
// this field is unset (in other words, one-of message field is not currently set to this option)
|
|
continue
|
|
}
|
|
prop := oop.Prop
|
|
rv := oov.Elem().FieldByName(prop.Name)
|
|
fd := m.FindFieldDescriptor(int32(prop.Tag))
|
|
if fd == nil {
|
|
// Our descriptor has different fields than this message object. So
|
|
// try to reflect on the message object's fields.
|
|
md, err := desc.LoadMessageDescriptorForMessage(pm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
fd = md.FindFieldByNumber(int32(prop.Tag))
|
|
if fd == nil {
|
|
return fmt.Errorf("message descriptor %q did not contain field for tag %d (%q in one-of %q)", md.GetFullyQualifiedName(), prop.Tag, prop.Name, src.Type().Field(oop.Field).Name)
|
|
}
|
|
}
|
|
if v, err := validFieldValueForRv(fd, rv); err != nil {
|
|
return err
|
|
} else {
|
|
values[fd] = v
|
|
}
|
|
}
|
|
|
|
// extension fields
|
|
rexts, _ := proto.ExtensionDescs(pm)
|
|
var unknownExtensions []byte
|
|
for _, ed := range rexts {
|
|
v, _ := proto.GetExtension(pm, ed)
|
|
if v == nil {
|
|
continue
|
|
}
|
|
if ed.ExtensionType == nil {
|
|
extBytes, _ := v.([]byte)
|
|
if len(extBytes) > 0 {
|
|
unknownExtensions = append(unknownExtensions, extBytes...)
|
|
}
|
|
continue
|
|
}
|
|
fd := m.er.FindExtension(m.md.GetFullyQualifiedName(), ed.Field)
|
|
if fd == nil {
|
|
var err error
|
|
if fd, err = desc.LoadFieldDescriptorForExtension(ed); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if v, err := validFieldValue(fd, v); err != nil {
|
|
return err
|
|
} else {
|
|
values[fd] = v
|
|
}
|
|
}
|
|
|
|
// now actually perform the merge
|
|
for fd, v := range values {
|
|
mergeField(m, fd, v)
|
|
}
|
|
|
|
u := src.FieldByName("XXX_unrecognized")
|
|
if u.IsValid() && u.Type() == typeOfBytes {
|
|
// ignore any error returned: pulling in unknown fields is best-effort
|
|
m.UnmarshalMerge(u.Interface().([]byte))
|
|
}
|
|
|
|
// lastly, also extract any unknown extensions the message may have (unknown extensions
|
|
// are stored with other extensions, not in the XXX_unrecognized field, so we have to do
|
|
// more than just the step above...)
|
|
if len(unknownExtensions) > 0 {
|
|
// pulling in unknown fields is best-effort, so we just ignore errors
|
|
m.UnmarshalMerge(unknownExtensions)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Validate checks that all required fields are present. It returns an error if any are absent.
|
|
func (m *Message) Validate() error {
|
|
missingFields := m.findMissingFields()
|
|
if len(missingFields) == 0 {
|
|
return nil
|
|
}
|
|
return fmt.Errorf("some required fields missing: %v", strings.Join(missingFields, ", "))
|
|
}
|
|
|
|
func (m *Message) findMissingFields() []string {
|
|
if m.md.IsProto3() {
|
|
// proto3 does not allow required fields
|
|
return nil
|
|
}
|
|
var missingFields []string
|
|
for _, fd := range m.md.GetFields() {
|
|
if fd.IsRequired() {
|
|
if _, ok := m.values[fd.GetNumber()]; !ok {
|
|
missingFields = append(missingFields, fd.GetName())
|
|
}
|
|
}
|
|
}
|
|
return missingFields
|
|
}
|
|
|
|
// ValidateRecursive checks that all required fields are present and also
|
|
// recursively validates all fields who are also messages. It returns an error
|
|
// if any required fields, in this message or nested within, are absent.
|
|
func (m *Message) ValidateRecursive() error {
|
|
return m.validateRecursive("")
|
|
}
|
|
|
|
func (m *Message) validateRecursive(prefix string) error {
|
|
if missingFields := m.findMissingFields(); len(missingFields) > 0 {
|
|
for i := range missingFields {
|
|
missingFields[i] = fmt.Sprintf("%s%s", prefix, missingFields[i])
|
|
}
|
|
return fmt.Errorf("some required fields missing: %v", strings.Join(missingFields, ", "))
|
|
}
|
|
|
|
for tag, fld := range m.values {
|
|
fd := m.FindFieldDescriptor(tag)
|
|
var chprefix string
|
|
var md *desc.MessageDescriptor
|
|
checkMsg := func(pm proto.Message) error {
|
|
var dm *Message
|
|
if d, ok := pm.(*Message); ok {
|
|
dm = d
|
|
} else {
|
|
dm = m.mf.NewDynamicMessage(md)
|
|
if err := dm.ConvertFrom(pm); err != nil {
|
|
return nil
|
|
}
|
|
}
|
|
if err := dm.validateRecursive(chprefix); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
isMap := fd.IsMap()
|
|
if isMap && fd.GetMapValueType().GetMessageType() != nil {
|
|
md = fd.GetMapValueType().GetMessageType()
|
|
mp := fld.(map[interface{}]interface{})
|
|
for k, v := range mp {
|
|
chprefix = fmt.Sprintf("%s%s[%v].", prefix, getName(fd), k)
|
|
if err := checkMsg(v.(proto.Message)); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
} else if !isMap && fd.GetMessageType() != nil {
|
|
md = fd.GetMessageType()
|
|
if fd.IsRepeated() {
|
|
sl := fld.([]interface{})
|
|
for i, v := range sl {
|
|
chprefix = fmt.Sprintf("%s%s[%d].", prefix, getName(fd), i)
|
|
if err := checkMsg(v.(proto.Message)); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
} else {
|
|
chprefix = fmt.Sprintf("%s%s.", prefix, getName(fd))
|
|
if err := checkMsg(fld.(proto.Message)); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func getName(fd *desc.FieldDescriptor) string {
|
|
if fd.IsExtension() {
|
|
return fmt.Sprintf("(%s)", fd.GetFullyQualifiedName())
|
|
} else {
|
|
return fd.GetName()
|
|
}
|
|
}
|
|
|
|
// knownFieldTags return tags of present and recognized fields, in sorted order.
|
|
func (m *Message) knownFieldTags() []int {
|
|
if len(m.values) == 0 {
|
|
return []int(nil)
|
|
}
|
|
|
|
keys := make([]int, len(m.values))
|
|
i := 0
|
|
for k := range m.values {
|
|
keys[i] = int(k)
|
|
i++
|
|
}
|
|
|
|
sort.Ints(keys)
|
|
return keys
|
|
}
|
|
|
|
// allKnownFieldTags return tags of present and recognized fields, including
|
|
// those that are unset, in sorted order. This only includes extensions that are
|
|
// present. Known but not-present extensions are not included in the returned
|
|
// set of tags.
|
|
func (m *Message) allKnownFieldTags() []int {
|
|
fds := m.md.GetFields()
|
|
keys := make([]int, 0, len(fds)+len(m.extraFields))
|
|
|
|
for k := range m.values {
|
|
keys = append(keys, int(k))
|
|
}
|
|
|
|
// also include known fields that are not present
|
|
for _, fd := range fds {
|
|
if _, ok := m.values[fd.GetNumber()]; !ok {
|
|
keys = append(keys, int(fd.GetNumber()))
|
|
}
|
|
}
|
|
for _, fd := range m.extraFields {
|
|
if !fd.IsExtension() { // skip extensions that are not present
|
|
if _, ok := m.values[fd.GetNumber()]; !ok {
|
|
keys = append(keys, int(fd.GetNumber()))
|
|
}
|
|
}
|
|
}
|
|
|
|
sort.Ints(keys)
|
|
return keys
|
|
}
|
|
|
|
// unknownFieldTags return tags of present but unrecognized fields, in sorted order.
|
|
func (m *Message) unknownFieldTags() []int {
|
|
if len(m.unknownFields) == 0 {
|
|
return []int(nil)
|
|
}
|
|
keys := make([]int, len(m.unknownFields))
|
|
i := 0
|
|
for k := range m.unknownFields {
|
|
keys[i] = int(k)
|
|
i++
|
|
}
|
|
sort.Ints(keys)
|
|
return keys
|
|
}
|