blob: 76dfdf7941e4220d2dc3b1f4c89d52132f71a5ce [file] [log] [blame]
Joe Tsai08e00302018-11-26 22:32:06 -08001// 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
5package impl
6
7import (
Joe Tsai6663f3a2019-09-03 15:33:00 -07008 "fmt"
Joe Tsai0fc49f82019-05-01 12:29:25 -07009 "reflect"
Joe Tsai8e506a82019-03-16 00:05:34 -070010 "strconv"
11
Damien Neil5c5b5312019-05-14 12:44:37 -070012 "google.golang.org/protobuf/encoding/prototext"
Joe Tsai6663f3a2019-09-03 15:33:00 -070013 "google.golang.org/protobuf/proto"
Damien Neile89e6242019-05-13 23:55:40 -070014 pref "google.golang.org/protobuf/reflect/protoreflect"
Joe Tsai6663f3a2019-09-03 15:33:00 -070015 piface "google.golang.org/protobuf/runtime/protoiface"
Joe Tsai08e00302018-11-26 22:32:06 -080016)
17
18// Export is a zero-length named type that exists only to export a set of
19// functions that we do not want to appear in godoc.
20type Export struct{}
21
Joe Tsai0fc49f82019-05-01 12:29:25 -070022// enum is any enum type generated by protoc-gen-go
23// and must be a named int32 type.
24type enum = interface{}
25
Joe Tsai08e00302018-11-26 22:32:06 -080026// EnumOf returns the protoreflect.Enum interface over e.
Joe Tsai14ac2412020-01-03 20:18:06 -080027// It returns nil if e is nil.
Joe Tsai0fc49f82019-05-01 12:29:25 -070028func (Export) EnumOf(e enum) pref.Enum {
Joe Tsai14ac2412020-01-03 20:18:06 -080029 switch e := e.(type) {
30 case nil:
31 return nil
32 case pref.Enum:
33 return e
34 default:
35 return legacyWrapEnum(reflect.ValueOf(e))
Joe Tsai08e00302018-11-26 22:32:06 -080036 }
Joe Tsai08e00302018-11-26 22:32:06 -080037}
38
Joe Tsai0fc49f82019-05-01 12:29:25 -070039// EnumDescriptorOf returns the protoreflect.EnumDescriptor for e.
Joe Tsai14ac2412020-01-03 20:18:06 -080040// It returns nil if e is nil.
Joe Tsai0fc49f82019-05-01 12:29:25 -070041func (Export) EnumDescriptorOf(e enum) pref.EnumDescriptor {
Joe Tsai14ac2412020-01-03 20:18:06 -080042 switch e := e.(type) {
43 case nil:
44 return nil
45 case pref.Enum:
46 return e.Descriptor()
47 default:
48 return LegacyLoadEnumDesc(reflect.TypeOf(e))
Joe Tsai0fc49f82019-05-01 12:29:25 -070049 }
Joe Tsai0fc49f82019-05-01 12:29:25 -070050}
51
Joe Tsai6663f3a2019-09-03 15:33:00 -070052// EnumTypeOf returns the protoreflect.EnumType for e.
Joe Tsai14ac2412020-01-03 20:18:06 -080053// It returns nil if e is nil.
Joe Tsai6663f3a2019-09-03 15:33:00 -070054func (Export) EnumTypeOf(e enum) pref.EnumType {
Joe Tsai14ac2412020-01-03 20:18:06 -080055 switch e := e.(type) {
56 case nil:
57 return nil
58 case pref.Enum:
59 return e.Type()
60 default:
61 return legacyLoadEnumType(reflect.TypeOf(e))
Joe Tsai6663f3a2019-09-03 15:33:00 -070062 }
Joe Tsai6663f3a2019-09-03 15:33:00 -070063}
64
Joe Tsai8e506a82019-03-16 00:05:34 -070065// EnumStringOf returns the enum value as a string, either as the name if
66// the number is resolvable, or the number formatted as a string.
67func (Export) EnumStringOf(ed pref.EnumDescriptor, n pref.EnumNumber) string {
68 ev := ed.Values().ByNumber(n)
69 if ev != nil {
70 return string(ev.Name())
71 }
72 return strconv.Itoa(int(n))
73}
74
Joe Tsai0fc49f82019-05-01 12:29:25 -070075// message is any message type generated by protoc-gen-go
76// and must be a pointer to a named struct type.
77type message = interface{}
78
Joe Tsai6663f3a2019-09-03 15:33:00 -070079// legacyMessageWrapper wraps a v2 message as a v1 message.
Joe Tsaiea5ada12019-09-04 22:41:40 -070080type legacyMessageWrapper struct{ m pref.ProtoMessage }
Joe Tsai6663f3a2019-09-03 15:33:00 -070081
82func (m legacyMessageWrapper) Reset() { proto.Reset(m.m) }
83func (m legacyMessageWrapper) String() string { return Export{}.MessageStringOf(m.m) }
84func (m legacyMessageWrapper) ProtoMessage() {}
85
86// ProtoMessageV1Of converts either a v1 or v2 message to a v1 message.
Joe Tsai14ac2412020-01-03 20:18:06 -080087// It returns nil if m is nil.
Joe Tsai6663f3a2019-09-03 15:33:00 -070088func (Export) ProtoMessageV1Of(m message) piface.MessageV1 {
89 switch mv := m.(type) {
Joe Tsai531a03a2019-09-06 00:30:59 -070090 case nil:
91 return nil
Joe Tsai6663f3a2019-09-03 15:33:00 -070092 case piface.MessageV1:
93 return mv
Joe Tsaifd528ff2019-09-03 16:30:39 -070094 case unwrapper:
95 return Export{}.ProtoMessageV1Of(mv.protoUnwrap())
Joe Tsaiea5ada12019-09-04 22:41:40 -070096 case pref.ProtoMessage:
Joe Tsai6663f3a2019-09-03 15:33:00 -070097 return legacyMessageWrapper{mv}
98 default:
99 panic(fmt.Sprintf("message %T is neither a v1 or v2 Message", m))
Joe Tsai08e00302018-11-26 22:32:06 -0800100 }
Joe Tsai08e00302018-11-26 22:32:06 -0800101}
102
Joe Tsaiea5ada12019-09-04 22:41:40 -0700103func (Export) protoMessageV2Of(m message) pref.ProtoMessage {
104 switch mv := m.(type) {
Joe Tsai531a03a2019-09-06 00:30:59 -0700105 case nil:
106 return nil
Joe Tsaiea5ada12019-09-04 22:41:40 -0700107 case pref.ProtoMessage:
108 return mv
109 case legacyMessageWrapper:
110 return mv.m
111 case piface.MessageV1:
112 return nil
113 default:
114 panic(fmt.Sprintf("message %T is neither a v1 or v2 Message", m))
115 }
116}
117
Joe Tsai6663f3a2019-09-03 15:33:00 -0700118// ProtoMessageV2Of converts either a v1 or v2 message to a v2 message.
Joe Tsai14ac2412020-01-03 20:18:06 -0800119// It returns nil if m is nil.
Joe Tsai6663f3a2019-09-03 15:33:00 -0700120func (Export) ProtoMessageV2Of(m message) pref.ProtoMessage {
Joe Tsai14ac2412020-01-03 20:18:06 -0800121 if m == nil {
122 return nil
123 }
124 if mv := (Export{}).protoMessageV2Of(m); mv != nil {
Joe Tsai6663f3a2019-09-03 15:33:00 -0700125 return mv
Joe Tsai08e00302018-11-26 22:32:06 -0800126 }
Joe Tsaiea5ada12019-09-04 22:41:40 -0700127 return legacyWrapMessage(reflect.ValueOf(m))
Joe Tsai6663f3a2019-09-03 15:33:00 -0700128}
129
130// MessageOf returns the protoreflect.Message interface over m.
Joe Tsai14ac2412020-01-03 20:18:06 -0800131// It returns nil if m is nil.
Joe Tsai6663f3a2019-09-03 15:33:00 -0700132func (Export) MessageOf(m message) pref.Message {
Joe Tsai14ac2412020-01-03 20:18:06 -0800133 if m == nil {
134 return nil
135 }
Joe Tsaiea5ada12019-09-04 22:41:40 -0700136 if mv := (Export{}).protoMessageV2Of(m); mv != nil {
Joe Tsai6663f3a2019-09-03 15:33:00 -0700137 return mv.ProtoReflect()
Joe Tsai6663f3a2019-09-03 15:33:00 -0700138 }
Joe Tsaiea5ada12019-09-04 22:41:40 -0700139 return legacyWrapMessage(reflect.ValueOf(m)).ProtoReflect()
Joe Tsai08e00302018-11-26 22:32:06 -0800140}
141
Joe Tsai0fc49f82019-05-01 12:29:25 -0700142// MessageDescriptorOf returns the protoreflect.MessageDescriptor for m.
Joe Tsai14ac2412020-01-03 20:18:06 -0800143// It returns nil if m is nil.
Joe Tsai0fc49f82019-05-01 12:29:25 -0700144func (Export) MessageDescriptorOf(m message) pref.MessageDescriptor {
Joe Tsai14ac2412020-01-03 20:18:06 -0800145 if m == nil {
146 return nil
147 }
Joe Tsaiea5ada12019-09-04 22:41:40 -0700148 if mv := (Export{}).protoMessageV2Of(m); mv != nil {
Joe Tsai0fc49f82019-05-01 12:29:25 -0700149 return mv.ProtoReflect().Descriptor()
Joe Tsai08e00302018-11-26 22:32:06 -0800150 }
Joe Tsaiea5ada12019-09-04 22:41:40 -0700151 return LegacyLoadMessageDesc(reflect.TypeOf(m))
Joe Tsai6663f3a2019-09-03 15:33:00 -0700152}
153
154// MessageTypeOf returns the protoreflect.MessageType for m.
Joe Tsai14ac2412020-01-03 20:18:06 -0800155// It returns nil if m is nil.
Joe Tsai6663f3a2019-09-03 15:33:00 -0700156func (Export) MessageTypeOf(m message) pref.MessageType {
Joe Tsai14ac2412020-01-03 20:18:06 -0800157 if m == nil {
158 return nil
159 }
Joe Tsaiea5ada12019-09-04 22:41:40 -0700160 if mv := (Export{}).protoMessageV2Of(m); mv != nil {
Joe Tsai6663f3a2019-09-03 15:33:00 -0700161 return mv.ProtoReflect().Type()
Joe Tsai6663f3a2019-09-03 15:33:00 -0700162 }
Joe Tsaiea5ada12019-09-04 22:41:40 -0700163 return legacyLoadMessageInfo(reflect.TypeOf(m), "")
Joe Tsai08e00302018-11-26 22:32:06 -0800164}
Joe Tsai1321a0e2019-03-20 09:46:22 -0700165
166// MessageStringOf returns the message value as a string,
167// which is the message serialized in the protobuf text format.
168func (Export) MessageStringOf(m pref.ProtoMessage) string {
Joe Tsai74b14602020-01-06 15:44:09 -0800169 return prototext.MarshalOptions{Multiline: false}.Format(m)
Joe Tsai1321a0e2019-03-20 09:46:22 -0700170}
Joe Tsai21ade492019-05-22 13:42:54 -0400171
Damien Neilf1e905b2019-08-08 15:45:59 -0700172// ExtensionDescFromType returns the legacy protoV1.ExtensionDesc for t.
173func (Export) ExtensionDescFromType(t pref.ExtensionType) *ExtensionInfo {
Joe Tsaifd4c6052019-09-16 13:30:15 -0700174 // TODO: Delete this function when v1 directly does this assertion.
175 if xt, ok := t.(*ExtensionInfo); ok {
176 return xt
177 }
178 return nil
Joe Tsai21ade492019-05-22 13:42:54 -0400179}