mirror of
https://github.com/protocolbuffers/protobuf-go.git
synced 2025-01-01 11:58:21 +00:00
990b9f5919
The prototype package was initially used by generated reflection support, but has now been replaced by internal/fileinit. Eventually, this functionality should be deleted and re-written in terms of other components in the repo. Usages that prototype currently provides (but should be moved) are: * Constructing standalone messages and enums, which is behavior we should provide in reflect/protodesc. The google.protobuf.{Enum,Type} are well-known proto messages designed for this purpose. * Constructing placeholder files, enums, and messages. * Consructing protoreflect.{Message,Enum,Extension}Types, which are protobuf descriptors with associated Go type information. Change-Id: Id7dbefff952682781b439aa555508c59b2629f9e Reviewed-on: https://go-review.googlesource.com/c/protobuf/+/167383 Reviewed-by: Damien Neil <dneil@google.com>
163 lines
4.7 KiB
Go
163 lines
4.7 KiB
Go
// 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 legacy
|
|
|
|
import (
|
|
"fmt"
|
|
"math"
|
|
"reflect"
|
|
"sync"
|
|
|
|
ptype "github.com/golang/protobuf/v2/internal/prototype"
|
|
pvalue "github.com/golang/protobuf/v2/internal/value"
|
|
pref "github.com/golang/protobuf/v2/reflect/protoreflect"
|
|
|
|
descriptorpb "github.com/golang/protobuf/v2/types/descriptor"
|
|
)
|
|
|
|
// wrapEnum wraps v as a protoreflect.Enum,
|
|
// where v must be a int32 kind and not implement the v2 API already.
|
|
func wrapEnum(v reflect.Value) pref.Enum {
|
|
et := loadEnumType(v.Type())
|
|
return et.New(pref.EnumNumber(v.Int()))
|
|
}
|
|
|
|
var enumTypeCache sync.Map // map[reflect.Type]protoreflect.EnumType
|
|
|
|
// loadEnumType dynamically loads a protoreflect.EnumType for t,
|
|
// where t must be an int32 kind and not implement the v2 API already.
|
|
func loadEnumType(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 := loadEnumDesc(t)
|
|
et := ptype.GoEnum(ed, func(et pref.EnumType, n pref.EnumNumber) pref.Enum {
|
|
if e, ok := m.Load(n); ok {
|
|
return e.(pref.Enum)
|
|
}
|
|
e := &enumWrapper{num: n, pbTyp: et, goTyp: t}
|
|
m.Store(n, e)
|
|
return e
|
|
})
|
|
enumTypeCache.Store(t, et)
|
|
return et.(pref.EnumType)
|
|
}
|
|
|
|
type enumWrapper struct {
|
|
num pref.EnumNumber
|
|
pbTyp pref.EnumType
|
|
goTyp reflect.Type
|
|
}
|
|
|
|
func (e *enumWrapper) Number() pref.EnumNumber {
|
|
return e.num
|
|
}
|
|
func (e *enumWrapper) Type() pref.EnumType {
|
|
return e.pbTyp
|
|
}
|
|
func (e *enumWrapper) ProtoReflect() pref.Enum {
|
|
return e
|
|
}
|
|
func (e *enumWrapper) ProtoUnwrap() interface{} {
|
|
v := reflect.New(e.goTyp).Elem()
|
|
v.SetInt(int64(e.num))
|
|
return v.Interface()
|
|
}
|
|
|
|
var (
|
|
_ pref.Enum = (*enumWrapper)(nil)
|
|
_ pvalue.Unwrapper = (*enumWrapper)(nil)
|
|
)
|
|
|
|
var enumDescCache sync.Map // map[reflect.Type]protoreflect.EnumDescriptor
|
|
|
|
var enumNumberType = reflect.TypeOf(pref.EnumNumber(0))
|
|
|
|
// loadEnumDesc returns an EnumDescriptor derived from the Go type,
|
|
// which must be an int32 kind and not implement the v2 API already.
|
|
func loadEnumDesc(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.Enum); ok {
|
|
panic(fmt.Sprintf("%v already implements proto.Enum", t))
|
|
}
|
|
if ed, ok := ev.(enumV1); ok {
|
|
b, idxs := ed.EnumDescriptor()
|
|
fd := loadFileDesc(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 *descriptorpb.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 enumV1, 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
|
|
}
|