blob: d7b844d38a77aaa54a9b4b858ca200d309c36220 [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 Tsai0fc49f82019-05-01 12:29:25 -070027func (Export) EnumOf(e enum) pref.Enum {
Damien Neila8593ba2019-01-08 16:18:07 -080028 if ev, ok := e.(pref.Enum); ok {
29 return ev
Joe Tsai08e00302018-11-26 22:32:06 -080030 }
Joe Tsai21ade492019-05-22 13:42:54 -040031 return legacyWrapEnum(reflect.ValueOf(e))
Joe Tsai08e00302018-11-26 22:32:06 -080032}
33
Joe Tsai0fc49f82019-05-01 12:29:25 -070034// EnumDescriptorOf returns the protoreflect.EnumDescriptor for e.
35func (Export) EnumDescriptorOf(e enum) pref.EnumDescriptor {
36 if ev, ok := e.(pref.Enum); ok {
37 return ev.Descriptor()
38 }
Joe Tsai21ade492019-05-22 13:42:54 -040039 return LegacyLoadEnumDesc(reflect.TypeOf(e))
Joe Tsai0fc49f82019-05-01 12:29:25 -070040}
41
Joe Tsai6663f3a2019-09-03 15:33:00 -070042// EnumTypeOf returns the protoreflect.EnumType for e.
43func (Export) EnumTypeOf(e enum) pref.EnumType {
44 if ev, ok := e.(pref.Enum); ok {
45 return ev.Type()
46 }
47 return legacyLoadEnumType(reflect.TypeOf(e))
48}
49
Joe Tsai8e506a82019-03-16 00:05:34 -070050// EnumStringOf returns the enum value as a string, either as the name if
51// the number is resolvable, or the number formatted as a string.
52func (Export) EnumStringOf(ed pref.EnumDescriptor, n pref.EnumNumber) string {
53 ev := ed.Values().ByNumber(n)
54 if ev != nil {
55 return string(ev.Name())
56 }
57 return strconv.Itoa(int(n))
58}
59
Joe Tsai0fc49f82019-05-01 12:29:25 -070060// message is any message type generated by protoc-gen-go
61// and must be a pointer to a named struct type.
62type message = interface{}
63
Joe Tsai6663f3a2019-09-03 15:33:00 -070064// legacyMessageWrapper wraps a v2 message as a v1 message.
Joe Tsaiea5ada12019-09-04 22:41:40 -070065type legacyMessageWrapper struct{ m pref.ProtoMessage }
Joe Tsai6663f3a2019-09-03 15:33:00 -070066
67func (m legacyMessageWrapper) Reset() { proto.Reset(m.m) }
68func (m legacyMessageWrapper) String() string { return Export{}.MessageStringOf(m.m) }
69func (m legacyMessageWrapper) ProtoMessage() {}
70
71// ProtoMessageV1Of converts either a v1 or v2 message to a v1 message.
72func (Export) ProtoMessageV1Of(m message) piface.MessageV1 {
73 switch mv := m.(type) {
Joe Tsai531a03a2019-09-06 00:30:59 -070074 case nil:
75 return nil
Joe Tsai6663f3a2019-09-03 15:33:00 -070076 case piface.MessageV1:
77 return mv
Joe Tsaifd528ff2019-09-03 16:30:39 -070078 case unwrapper:
79 return Export{}.ProtoMessageV1Of(mv.protoUnwrap())
Joe Tsaiea5ada12019-09-04 22:41:40 -070080 case pref.ProtoMessage:
Joe Tsai6663f3a2019-09-03 15:33:00 -070081 return legacyMessageWrapper{mv}
82 default:
83 panic(fmt.Sprintf("message %T is neither a v1 or v2 Message", m))
Joe Tsai08e00302018-11-26 22:32:06 -080084 }
Joe Tsai08e00302018-11-26 22:32:06 -080085}
86
Joe Tsaiea5ada12019-09-04 22:41:40 -070087func (Export) protoMessageV2Of(m message) pref.ProtoMessage {
88 switch mv := m.(type) {
Joe Tsai531a03a2019-09-06 00:30:59 -070089 case nil:
90 return nil
Joe Tsaiea5ada12019-09-04 22:41:40 -070091 case pref.ProtoMessage:
92 return mv
93 case legacyMessageWrapper:
94 return mv.m
95 case piface.MessageV1:
96 return nil
97 default:
98 panic(fmt.Sprintf("message %T is neither a v1 or v2 Message", m))
99 }
100}
101
Joe Tsai6663f3a2019-09-03 15:33:00 -0700102// ProtoMessageV2Of converts either a v1 or v2 message to a v2 message.
103func (Export) ProtoMessageV2Of(m message) pref.ProtoMessage {
Joe Tsai531a03a2019-09-06 00:30:59 -0700104 if mv := (Export{}).protoMessageV2Of(m); mv != nil || m == nil {
Joe Tsai6663f3a2019-09-03 15:33:00 -0700105 return mv
Joe Tsai08e00302018-11-26 22:32:06 -0800106 }
Joe Tsaiea5ada12019-09-04 22:41:40 -0700107 return legacyWrapMessage(reflect.ValueOf(m))
Joe Tsai6663f3a2019-09-03 15:33:00 -0700108}
109
110// MessageOf returns the protoreflect.Message interface over m.
111func (Export) MessageOf(m message) pref.Message {
Joe Tsaiea5ada12019-09-04 22:41:40 -0700112 if mv := (Export{}).protoMessageV2Of(m); mv != nil {
Joe Tsai6663f3a2019-09-03 15:33:00 -0700113 return mv.ProtoReflect()
Joe Tsai6663f3a2019-09-03 15:33:00 -0700114 }
Joe Tsaiea5ada12019-09-04 22:41:40 -0700115 return legacyWrapMessage(reflect.ValueOf(m)).ProtoReflect()
Joe Tsai08e00302018-11-26 22:32:06 -0800116}
117
Joe Tsai0fc49f82019-05-01 12:29:25 -0700118// MessageDescriptorOf returns the protoreflect.MessageDescriptor for m.
119func (Export) MessageDescriptorOf(m message) pref.MessageDescriptor {
Joe Tsaiea5ada12019-09-04 22:41:40 -0700120 if mv := (Export{}).protoMessageV2Of(m); mv != nil {
Joe Tsai0fc49f82019-05-01 12:29:25 -0700121 return mv.ProtoReflect().Descriptor()
Joe Tsai08e00302018-11-26 22:32:06 -0800122 }
Joe Tsaiea5ada12019-09-04 22:41:40 -0700123 return LegacyLoadMessageDesc(reflect.TypeOf(m))
Joe Tsai6663f3a2019-09-03 15:33:00 -0700124}
125
126// MessageTypeOf returns the protoreflect.MessageType for m.
127func (Export) MessageTypeOf(m message) pref.MessageType {
Joe Tsaiea5ada12019-09-04 22:41:40 -0700128 if mv := (Export{}).protoMessageV2Of(m); mv != nil {
Joe Tsai6663f3a2019-09-03 15:33:00 -0700129 return mv.ProtoReflect().Type()
Joe Tsai6663f3a2019-09-03 15:33:00 -0700130 }
Joe Tsaiea5ada12019-09-04 22:41:40 -0700131 return legacyLoadMessageInfo(reflect.TypeOf(m), "")
Joe Tsai08e00302018-11-26 22:32:06 -0800132}
Joe Tsai1321a0e2019-03-20 09:46:22 -0700133
134// MessageStringOf returns the message value as a string,
135// which is the message serialized in the protobuf text format.
136func (Export) MessageStringOf(m pref.ProtoMessage) string {
Joe Tsaid01061a2019-10-05 20:56:32 -0700137 v := reflect.ValueOf(m)
138 if m == nil || (v.Kind() == reflect.Ptr && v.IsNil()) {
139 return "<nil>"
140 }
Joe Tsaicd4a31e2019-09-14 19:14:24 -0700141 b, _ := prototext.MarshalOptions{
142 AllowPartial: true,
143 EmitUnknown: true,
144 }.Marshal(m)
Joe Tsai1321a0e2019-03-20 09:46:22 -0700145 return string(b)
146}
Joe Tsai21ade492019-05-22 13:42:54 -0400147
Damien Neilf1e905b2019-08-08 15:45:59 -0700148// ExtensionDescFromType returns the legacy protoV1.ExtensionDesc for t.
149func (Export) ExtensionDescFromType(t pref.ExtensionType) *ExtensionInfo {
Joe Tsaifd4c6052019-09-16 13:30:15 -0700150 // TODO: Delete this function when v1 directly does this assertion.
151 if xt, ok := t.(*ExtensionInfo); ok {
152 return xt
153 }
154 return nil
Joe Tsai21ade492019-05-22 13:42:54 -0400155}