| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162 |
- // Copyright 2018 The Go Authors. All rights reserved.
- // Use of this source code is governed by a BSD-style
- // license that can be found in the LICENSE file.
- package impl
- import (
- "fmt"
- "math"
- "reflect"
- "sync"
- descriptorV1 "github.com/golang/protobuf/protoc-gen-go/descriptor"
- pvalue "github.com/golang/protobuf/v2/internal/value"
- pref "github.com/golang/protobuf/v2/reflect/protoreflect"
- ptype "github.com/golang/protobuf/v2/reflect/prototype"
- )
- // legacyWrapEnum wraps v as a protoreflect.ProtoEnum,
- // where v must be a *struct kind and not implement the v2 API already.
- func legacyWrapEnum(v reflect.Value) pref.ProtoEnum {
- et := legacyLoadEnumType(v.Type())
- return et.New(pref.EnumNumber(v.Int()))
- }
- var enumTypeCache sync.Map // map[reflect.Type]protoreflect.EnumType
- // legacyLoadEnumType dynamically loads a protoreflect.EnumType for t,
- // where t must be an int32 kind and not implement the v2 API already.
- func legacyLoadEnumType(t reflect.Type) pref.EnumType {
- // Fast-path: check if a EnumType is cached for this concrete type.
- if et, ok := enumTypeCache.Load(t); ok {
- return et.(pref.EnumType)
- }
- // Slow-path: derive enum descriptor and initialize EnumType.
- var m sync.Map // map[protoreflect.EnumNumber]proto.Enum
- ed := legacyLoadEnumDesc(t)
- et := ptype.GoEnum(ed, func(et pref.EnumType, n pref.EnumNumber) pref.ProtoEnum {
- if e, ok := m.Load(n); ok {
- return e.(pref.ProtoEnum)
- }
- e := &legacyEnumWrapper{num: n, pbTyp: et, goTyp: t}
- m.Store(n, e)
- return e
- })
- enumTypeCache.Store(t, et)
- return et.(pref.EnumType)
- }
- type legacyEnumWrapper struct {
- num pref.EnumNumber
- pbTyp pref.EnumType
- goTyp reflect.Type
- }
- func (e *legacyEnumWrapper) Number() pref.EnumNumber {
- return e.num
- }
- func (e *legacyEnumWrapper) Type() pref.EnumType {
- return e.pbTyp
- }
- func (e *legacyEnumWrapper) ProtoReflect() pref.Enum {
- return e
- }
- func (e *legacyEnumWrapper) Unwrap() interface{} {
- v := reflect.New(e.goTyp).Elem()
- v.SetInt(int64(e.num))
- return v.Interface()
- }
- var (
- _ pref.Enum = (*legacyEnumWrapper)(nil)
- _ pref.ProtoEnum = (*legacyEnumWrapper)(nil)
- _ pvalue.Unwrapper = (*legacyEnumWrapper)(nil)
- )
- var enumDescCache sync.Map // map[reflect.Type]protoreflect.EnumDescriptor
- var enumNumberType = reflect.TypeOf(pref.EnumNumber(0))
- // legacyLoadEnumDesc returns an EnumDescriptor derived from the Go type,
- // which must be an int32 kind and not implement the v2 API already.
- func legacyLoadEnumDesc(t reflect.Type) pref.EnumDescriptor {
- // Fast-path: check if an EnumDescriptor is cached for this concrete type.
- if v, ok := enumDescCache.Load(t); ok {
- return v.(pref.EnumDescriptor)
- }
- // Slow-path: initialize EnumDescriptor from the proto descriptor.
- if t.Kind() != reflect.Int32 || t.PkgPath() == "" {
- panic(fmt.Sprintf("got %v, want named int32 kind", t))
- }
- if t == enumNumberType {
- panic(fmt.Sprintf("cannot be %v", t))
- }
- // Derive the enum descriptor from the raw descriptor proto.
- e := new(ptype.StandaloneEnum)
- ev := reflect.Zero(t).Interface()
- if _, ok := ev.(pref.ProtoEnum); ok {
- panic(fmt.Sprintf("%v already implements proto.Enum", t))
- }
- if ed, ok := ev.(legacyEnum); ok {
- b, idxs := ed.EnumDescriptor()
- fd := legacyLoadFileDesc(b)
- // Derive syntax.
- switch fd.GetSyntax() {
- case "proto2", "":
- e.Syntax = pref.Proto2
- case "proto3":
- e.Syntax = pref.Proto3
- }
- // Derive the full name and correct enum descriptor.
- var ed *descriptorV1.EnumDescriptorProto
- e.FullName = pref.FullName(fd.GetPackage())
- if len(idxs) == 1 {
- ed = fd.EnumType[idxs[0]]
- e.FullName = e.FullName.Append(pref.Name(ed.GetName()))
- } else {
- md := fd.MessageType[idxs[0]]
- e.FullName = e.FullName.Append(pref.Name(md.GetName()))
- for _, i := range idxs[1 : len(idxs)-1] {
- md = md.NestedType[i]
- e.FullName = e.FullName.Append(pref.Name(md.GetName()))
- }
- ed = md.EnumType[idxs[len(idxs)-1]]
- e.FullName = e.FullName.Append(pref.Name(ed.GetName()))
- }
- // Derive the enum values.
- for _, vd := range ed.GetValue() {
- e.Values = append(e.Values, ptype.EnumValue{
- Name: pref.Name(vd.GetName()),
- Number: pref.EnumNumber(vd.GetNumber()),
- })
- }
- } else {
- // If the type does not implement legacyEnum, then there is no reliable
- // way to derive the original protobuf type information.
- // We are unable to use the global enum registry since it is
- // unfortunately keyed by the full name, which we do not know.
- // Furthermore, some generated enums register with a fork of
- // golang/protobuf so the enum may not even be found in the registry.
- //
- // Instead, create a bogus enum descriptor to ensure that
- // most operations continue to work. For example, textpb and jsonpb
- // will be unable to parse a message with an enum value by name.
- e.Syntax = pref.Proto2
- e.FullName = deriveFullName(t)
- e.Values = []ptype.EnumValue{{Name: "INVALID", Number: math.MinInt32}}
- }
- ed, err := ptype.NewEnum(e)
- if err != nil {
- panic(err)
- }
- enumDescCache.Store(t, ed)
- return ed
- }
|