blob: 6758e7409701c341b8be3d94fd3fcaebc91ac935 [file] [log] [blame]
Joe Tsai90fe9962018-10-18 11:06:29 -07001// Copyright 2018 The Go Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
Joe Tsai21ade492019-05-22 13:42:54 -04005package impl
Joe Tsai90fe9962018-10-18 11:06:29 -07006
7import (
8 "fmt"
9 "math"
10 "reflect"
11 "sync"
12
Damien Neile89e6242019-05-13 23:55:40 -070013 ptype "google.golang.org/protobuf/internal/prototype"
14 pvalue "google.golang.org/protobuf/internal/value"
15 pref "google.golang.org/protobuf/reflect/protoreflect"
Joe Tsaib2f66be2019-05-22 00:42:45 -040016 "google.golang.org/protobuf/reflect/prototype"
Joe Tsai90fe9962018-10-18 11:06:29 -070017)
18
Joe Tsai21ade492019-05-22 13:42:54 -040019// legacyWrapEnum wraps v as a protoreflect.Enum,
Joe Tsai08e00302018-11-26 22:32:06 -080020// where v must be a int32 kind and not implement the v2 API already.
Joe Tsai21ade492019-05-22 13:42:54 -040021func legacyWrapEnum(v reflect.Value) pref.Enum {
22 et := legacyLoadEnumType(v.Type())
Joe Tsaif0c01e42018-11-06 13:05:20 -080023 return et.New(pref.EnumNumber(v.Int()))
24}
25
Joe Tsai21ade492019-05-22 13:42:54 -040026var legacyEnumTypeCache sync.Map // map[reflect.Type]protoreflect.EnumType
Joe Tsai6f9095c2018-11-10 14:12:21 -080027
Joe Tsai21ade492019-05-22 13:42:54 -040028// legacyLoadEnumType dynamically loads a protoreflect.EnumType for t,
Joe Tsaif0c01e42018-11-06 13:05:20 -080029// where t must be an int32 kind and not implement the v2 API already.
Joe Tsai21ade492019-05-22 13:42:54 -040030func legacyLoadEnumType(t reflect.Type) pref.EnumType {
Joe Tsai6f9095c2018-11-10 14:12:21 -080031 // Fast-path: check if a EnumType is cached for this concrete type.
Joe Tsai21ade492019-05-22 13:42:54 -040032 if et, ok := legacyEnumTypeCache.Load(t); ok {
Joe Tsaif0c01e42018-11-06 13:05:20 -080033 return et.(pref.EnumType)
Joe Tsai6f9095c2018-11-10 14:12:21 -080034 }
35
36 // Slow-path: derive enum descriptor and initialize EnumType.
Joe Tsaib2f66be2019-05-22 00:42:45 -040037 var et pref.EnumType
Joe Tsai6f9095c2018-11-10 14:12:21 -080038 var m sync.Map // map[protoreflect.EnumNumber]proto.Enum
Joe Tsai21ade492019-05-22 13:42:54 -040039 ed := LegacyLoadEnumDesc(t)
Joe Tsaib2f66be2019-05-22 00:42:45 -040040 et = &prototype.Enum{
41 EnumDescriptor: ed,
42 NewEnum: func(n pref.EnumNumber) pref.Enum {
43 if e, ok := m.Load(n); ok {
44 return e.(pref.Enum)
45 }
Joe Tsai21ade492019-05-22 13:42:54 -040046 e := &legacyEnumWrapper{num: n, pbTyp: et, goTyp: t}
Joe Tsaib2f66be2019-05-22 00:42:45 -040047 m.Store(n, e)
48 return e
49 },
50 }
Joe Tsai21ade492019-05-22 13:42:54 -040051 if et, ok := legacyEnumTypeCache.LoadOrStore(t, et); ok {
Joe Tsaib9365042019-03-19 14:14:29 -070052 return et.(pref.EnumType)
53 }
54 return et
Joe Tsai6f9095c2018-11-10 14:12:21 -080055}
56
Joe Tsai21ade492019-05-22 13:42:54 -040057type legacyEnumWrapper struct {
Joe Tsai6f9095c2018-11-10 14:12:21 -080058 num pref.EnumNumber
59 pbTyp pref.EnumType
60 goTyp reflect.Type
61}
62
Joe Tsai21ade492019-05-22 13:42:54 -040063func (e *legacyEnumWrapper) Descriptor() pref.EnumDescriptor {
Joe Tsai0fc49f82019-05-01 12:29:25 -070064 return e.pbTyp.Descriptor()
65}
Joe Tsai21ade492019-05-22 13:42:54 -040066func (e *legacyEnumWrapper) Number() pref.EnumNumber {
Joe Tsai0fc49f82019-05-01 12:29:25 -070067 return e.num
68}
Joe Tsai21ade492019-05-22 13:42:54 -040069func (e *legacyEnumWrapper) ProtoReflect() pref.Enum {
Joe Tsai6f9095c2018-11-10 14:12:21 -080070 return e
71}
Joe Tsai21ade492019-05-22 13:42:54 -040072func (e *legacyEnumWrapper) ProtoUnwrap() interface{} {
Joe Tsai6f9095c2018-11-10 14:12:21 -080073 v := reflect.New(e.goTyp).Elem()
74 v.SetInt(int64(e.num))
75 return v.Interface()
76}
77
78var (
Joe Tsai21ade492019-05-22 13:42:54 -040079 _ pref.Enum = (*legacyEnumWrapper)(nil)
80 _ pvalue.Unwrapper = (*legacyEnumWrapper)(nil)
Joe Tsai6f9095c2018-11-10 14:12:21 -080081)
82
Joe Tsai21ade492019-05-22 13:42:54 -040083var legacyEnumDescCache sync.Map // map[reflect.Type]protoreflect.EnumDescriptor
Joe Tsai90fe9962018-10-18 11:06:29 -070084
Joe Tsai21ade492019-05-22 13:42:54 -040085var legacyEnumNumberType = reflect.TypeOf(pref.EnumNumber(0))
Joe Tsaif0c01e42018-11-06 13:05:20 -080086
Joe Tsai21ade492019-05-22 13:42:54 -040087// LegacyLoadEnumDesc returns an EnumDescriptor derived from the Go type,
Joe Tsai90fe9962018-10-18 11:06:29 -070088// which must be an int32 kind and not implement the v2 API already.
Joe Tsai35ec98f2019-03-25 14:41:32 -070089//
90// This is exported for testing purposes.
Joe Tsai21ade492019-05-22 13:42:54 -040091func LegacyLoadEnumDesc(t reflect.Type) pref.EnumDescriptor {
Joe Tsai90fe9962018-10-18 11:06:29 -070092 // Fast-path: check if an EnumDescriptor is cached for this concrete type.
Joe Tsai21ade492019-05-22 13:42:54 -040093 if ed, ok := legacyEnumDescCache.Load(t); ok {
Joe Tsaib9365042019-03-19 14:14:29 -070094 return ed.(pref.EnumDescriptor)
Joe Tsai90fe9962018-10-18 11:06:29 -070095 }
96
97 // Slow-path: initialize EnumDescriptor from the proto descriptor.
Joe Tsai6f9095c2018-11-10 14:12:21 -080098 if t.Kind() != reflect.Int32 || t.PkgPath() == "" {
99 panic(fmt.Sprintf("got %v, want named int32 kind", t))
Joe Tsai90fe9962018-10-18 11:06:29 -0700100 }
Joe Tsai21ade492019-05-22 13:42:54 -0400101 if t == legacyEnumNumberType {
Joe Tsaif0c01e42018-11-06 13:05:20 -0800102 panic(fmt.Sprintf("cannot be %v", t))
103 }
Joe Tsai90fe9962018-10-18 11:06:29 -0700104
105 // Derive the enum descriptor from the raw descriptor proto.
106 e := new(ptype.StandaloneEnum)
107 ev := reflect.Zero(t).Interface()
Damien Neila8593ba2019-01-08 16:18:07 -0800108 if _, ok := ev.(pref.Enum); ok {
Joe Tsai90fe9962018-10-18 11:06:29 -0700109 panic(fmt.Sprintf("%v already implements proto.Enum", t))
110 }
Joe Tsai6dbffb72018-12-04 14:06:19 -0800111 if ed, ok := ev.(enumV1); ok {
Joe Tsai90fe9962018-10-18 11:06:29 -0700112 b, idxs := ed.EnumDescriptor()
Joe Tsai21ade492019-05-22 13:42:54 -0400113 fd := legacyLoadFileDesc(b)
Joe Tsai90fe9962018-10-18 11:06:29 -0700114
115 // Derive syntax.
116 switch fd.GetSyntax() {
117 case "proto2", "":
118 e.Syntax = pref.Proto2
119 case "proto3":
120 e.Syntax = pref.Proto3
121 }
122
123 // Derive the full name and correct enum descriptor.
Joe Tsai21ade492019-05-22 13:42:54 -0400124 var ed *legacyEnumDescriptorProto
Joe Tsai90fe9962018-10-18 11:06:29 -0700125 e.FullName = pref.FullName(fd.GetPackage())
126 if len(idxs) == 1 {
127 ed = fd.EnumType[idxs[0]]
128 e.FullName = e.FullName.Append(pref.Name(ed.GetName()))
129 } else {
130 md := fd.MessageType[idxs[0]]
131 e.FullName = e.FullName.Append(pref.Name(md.GetName()))
132 for _, i := range idxs[1 : len(idxs)-1] {
133 md = md.NestedType[i]
134 e.FullName = e.FullName.Append(pref.Name(md.GetName()))
135 }
136 ed = md.EnumType[idxs[len(idxs)-1]]
137 e.FullName = e.FullName.Append(pref.Name(ed.GetName()))
138 }
139
140 // Derive the enum values.
Damien Neil987d5702019-04-10 11:06:53 -0700141 for _, vd := range ed.Value {
Joe Tsai90fe9962018-10-18 11:06:29 -0700142 e.Values = append(e.Values, ptype.EnumValue{
143 Name: pref.Name(vd.GetName()),
144 Number: pref.EnumNumber(vd.GetNumber()),
145 })
146 }
147 } else {
Joe Tsai6dbffb72018-12-04 14:06:19 -0800148 // If the type does not implement enumV1, then there is no reliable
Joe Tsai90fe9962018-10-18 11:06:29 -0700149 // way to derive the original protobuf type information.
150 // We are unable to use the global enum registry since it is
151 // unfortunately keyed by the full name, which we do not know.
152 // Furthermore, some generated enums register with a fork of
153 // golang/protobuf so the enum may not even be found in the registry.
154 //
155 // Instead, create a bogus enum descriptor to ensure that
Damien Neil5c5b5312019-05-14 12:44:37 -0700156 // most operations continue to work. For example, prototext and protojson
Joe Tsai90fe9962018-10-18 11:06:29 -0700157 // will be unable to parse a message with an enum value by name.
158 e.Syntax = pref.Proto2
Joe Tsai21ade492019-05-22 13:42:54 -0400159 e.FullName = legacyDeriveFullName(t)
Joe Tsai90fe9962018-10-18 11:06:29 -0700160 e.Values = []ptype.EnumValue{{Name: "INVALID", Number: math.MinInt32}}
161 }
162
163 ed, err := ptype.NewEnum(e)
164 if err != nil {
165 panic(err)
166 }
Joe Tsai21ade492019-05-22 13:42:54 -0400167 if ed, ok := legacyEnumDescCache.LoadOrStore(t, ed); ok {
Joe Tsaib9365042019-03-19 14:14:29 -0700168 return ed.(pref.EnumDescriptor)
169 }
Joe Tsai90fe9962018-10-18 11:06:29 -0700170 return ed
171}