blob: c438c1d23e5a1ce2eb6af68b031ca72673de75cf [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 Tsai08e00302018-11-26 22:32:06 -08005package legacy
Joe Tsai90fe9962018-10-18 11:06:29 -07006
7import (
8 "bytes"
9 "compress/gzip"
10 "io/ioutil"
11 "sync"
12
Damien Neil4be2fb42018-12-17 11:16:16 -080013 "github.com/golang/protobuf/v2/proto"
Joe Tsaieeca8bb2018-12-04 16:24:22 -080014 pref "github.com/golang/protobuf/v2/reflect/protoreflect"
Joe Tsaie1f8d502018-11-26 18:55:29 -080015 descriptorpb "github.com/golang/protobuf/v2/types/descriptor"
Joe Tsai90fe9962018-10-18 11:06:29 -070016)
17
18// Every enum and message type generated by protoc-gen-go since commit 2fc053c5
19// on February 25th, 2016 has had a method to get the raw descriptor.
20// Types that were not generated by protoc-gen-go or were generated prior
21// to that version are not supported.
22//
23// The []byte returned is the encoded form of a FileDescriptorProto message
24// compressed using GZIP. The []int is the path from the top-level file
25// to the specific message or enum declaration.
26type (
Joe Tsai6dbffb72018-12-04 14:06:19 -080027 enumV1 interface {
Joe Tsai90fe9962018-10-18 11:06:29 -070028 EnumDescriptor() ([]byte, []int)
29 }
Joe Tsai6dbffb72018-12-04 14:06:19 -080030 messageV1 interface {
Joe Tsai90fe9962018-10-18 11:06:29 -070031 Descriptor() ([]byte, []int)
32 }
33)
34
Joe Tsaie1f8d502018-11-26 18:55:29 -080035var fileDescCache sync.Map // map[*byte]*descriptorpb.FileDescriptorProto
Joe Tsai90fe9962018-10-18 11:06:29 -070036
Joe Tsai6dbffb72018-12-04 14:06:19 -080037// loadFileDesc unmarshals b as a compressed FileDescriptorProto message.
Joe Tsai90fe9962018-10-18 11:06:29 -070038//
39// This assumes that b is immutable and that b does not refer to part of a
40// concatenated series of GZIP files (which would require shenanigans that
41// rely on the concatenation properties of both protobufs and GZIP).
42// File descriptors generated by protoc-gen-go do not rely on that property.
Joe Tsaie1f8d502018-11-26 18:55:29 -080043func loadFileDesc(b []byte) *descriptorpb.FileDescriptorProto {
Joe Tsai90fe9962018-10-18 11:06:29 -070044 // Fast-path: check whether we already have a cached file descriptor.
Joe Tsaib9365042019-03-19 14:14:29 -070045 if fd, ok := fileDescCache.Load(&b[0]); ok {
46 return fd.(*descriptorpb.FileDescriptorProto)
Joe Tsai90fe9962018-10-18 11:06:29 -070047 }
48
49 // Slow-path: decompress and unmarshal the file descriptor proto.
Joe Tsaib9365042019-03-19 14:14:29 -070050 fd := new(descriptorpb.FileDescriptorProto)
Joe Tsai90fe9962018-10-18 11:06:29 -070051 zr, err := gzip.NewReader(bytes.NewReader(b))
52 if err != nil {
53 panic(err)
54 }
55 b, err = ioutil.ReadAll(zr)
56 if err != nil {
57 panic(err)
58 }
Joe Tsaib9365042019-03-19 14:14:29 -070059 err = proto.UnmarshalOptions{DiscardUnknown: true}.Unmarshal(b, fd)
Joe Tsai69996ba2019-03-14 14:06:05 -070060 if err != nil {
Joe Tsai90fe9962018-10-18 11:06:29 -070061 panic(err)
62 }
Joe Tsaib9365042019-03-19 14:14:29 -070063 if fd, ok := fileDescCache.LoadOrStore(&b[0], fd); ok {
64 return fd.(*descriptorpb.FileDescriptorProto)
65 }
66 return fd
Joe Tsai90fe9962018-10-18 11:06:29 -070067}
Joe Tsai08e00302018-11-26 22:32:06 -080068
69// parentFileDescriptor returns the parent protoreflect.FileDescriptor for the
70// provide descriptor. It returns nil if there is no parent.
71func parentFileDescriptor(d pref.Descriptor) pref.FileDescriptor {
72 for ok := true; ok; d, ok = d.Parent() {
73 if fd, _ := d.(pref.FileDescriptor); fd != nil {
74 return fd
75 }
76 }
77 return nil
78}