mirror of
https://github.com/protocolbuffers/protobuf-go.git
synced 2025-01-04 02:38:50 +00:00
e407ee162b
This is a breaking change. The replacement is the Files.FindDescriptorByName method, which is more flexible as it handles all descriptor types. Change-Id: I2ccd544a7630396a2428b1d41f836c5246070912 Reviewed-on: https://go-review.googlesource.com/c/protobuf/+/183700 Reviewed-by: Damien Neil <dneil@google.com>
150 lines
4.7 KiB
Go
150 lines
4.7 KiB
Go
// Copyright 2019 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 filedesc provides functionality for constructing descriptors.
|
|
package filedesc
|
|
|
|
import (
|
|
"google.golang.org/protobuf/internal/encoding/wire"
|
|
"google.golang.org/protobuf/internal/fieldnum"
|
|
"google.golang.org/protobuf/reflect/protoreflect"
|
|
pref "google.golang.org/protobuf/reflect/protoreflect"
|
|
preg "google.golang.org/protobuf/reflect/protoregistry"
|
|
)
|
|
|
|
// DescBuilder construct a protoreflect.FileDescriptor from the raw descriptor.
|
|
type DescBuilder struct {
|
|
// RawDescriptor is the wire-encoded bytes of FileDescriptorProto
|
|
// and must be populated.
|
|
RawDescriptor []byte
|
|
|
|
// NumEnums is the total number of enums declared in the file.
|
|
NumEnums int32
|
|
// NumMessages is the total number of messages declared in the file.
|
|
// It includes the implicit message declarations for map entries.
|
|
NumMessages int32
|
|
// NumExtensions is the total number of extensions declared in the file.
|
|
NumExtensions int32
|
|
// NumServices is the total number of services declared in the file.
|
|
NumServices int32
|
|
|
|
// TypeResolver resolves extension field types for descriptor options.
|
|
// If nil, it uses protoregistry.GlobalTypes.
|
|
TypeResolver interface {
|
|
preg.ExtensionTypeResolver
|
|
}
|
|
|
|
// FileRegistry is use to lookup file, enum, and message dependencies.
|
|
// Once constructed, the file descriptor is registered here.
|
|
// If nil, it uses protoregistry.GlobalFiles.
|
|
FileRegistry interface {
|
|
FindFileByPath(string) (protoreflect.FileDescriptor, error)
|
|
FindDescriptorByName(pref.FullName) (pref.Descriptor, error)
|
|
Register(...pref.FileDescriptor) error
|
|
}
|
|
}
|
|
|
|
// resolverByIndex is an interface DescBuilder.FileRegistry may implement.
|
|
// If so, it permits looking up an enum or message dependency based on the
|
|
// sub-list and element index into filetype.TypeBuilder.DependencyIndexes.
|
|
type resolverByIndex interface {
|
|
FindEnumByIndex(int32, int32, []Enum, []Message) pref.EnumDescriptor
|
|
FindMessageByIndex(int32, int32, []Enum, []Message) pref.MessageDescriptor
|
|
}
|
|
|
|
// Indexes of each sub-list in filetype.TypeBuilder.DependencyIndexes.
|
|
const (
|
|
listFieldDeps int32 = iota
|
|
listExtTargets
|
|
listExtDeps
|
|
listMethInDeps
|
|
listMethOutDeps
|
|
)
|
|
|
|
// Build constructs a FileDescriptor given the parameters set in DescBuilder.
|
|
// It assumes that the inputs are well-formed and panics if any inconsistencies
|
|
// are encountered.
|
|
//
|
|
// If NumEnums+NumMessages+NumExtensions+NumServices is zero,
|
|
// then Build automatically derives them from the raw descriptor.
|
|
func (db DescBuilder) Build() (out struct {
|
|
File pref.FileDescriptor
|
|
|
|
// Enums is all enum descriptors in "flattened ordering".
|
|
Enums []Enum
|
|
// Messages is all message descriptors in "flattened ordering".
|
|
// It includes the implicit message declarations for map entries.
|
|
Messages []Message
|
|
// Extensions is all extension descriptors in "flattened ordering".
|
|
Extensions []Extension
|
|
// Service is all service descriptors in "flattened ordering".
|
|
Services []Service
|
|
}) {
|
|
// Populate the counts if uninitialized.
|
|
if db.NumEnums+db.NumMessages+db.NumExtensions+db.NumServices == 0 {
|
|
db.unmarshalCounts(db.RawDescriptor, true)
|
|
}
|
|
|
|
// Initialize resolvers and registries if unpopulated.
|
|
if db.TypeResolver == nil {
|
|
db.TypeResolver = preg.GlobalTypes
|
|
}
|
|
if db.FileRegistry == nil {
|
|
db.FileRegistry = preg.GlobalFiles
|
|
}
|
|
|
|
fd := newRawFile(db)
|
|
out.File = fd
|
|
out.Enums = fd.allEnums
|
|
out.Messages = fd.allMessages
|
|
out.Extensions = fd.allExtensions
|
|
out.Services = fd.allServices
|
|
|
|
if err := db.FileRegistry.Register(fd); err != nil {
|
|
panic(err)
|
|
}
|
|
return out
|
|
}
|
|
|
|
// unmarshalCounts counts the number of enum, message, extension, and service
|
|
// declarations in the raw message, which is either a FileDescriptorProto
|
|
// or a MessageDescriptorProto depending on whether isFile is set.
|
|
func (db *DescBuilder) unmarshalCounts(b []byte, isFile bool) {
|
|
for len(b) > 0 {
|
|
num, typ, n := wire.ConsumeTag(b)
|
|
b = b[n:]
|
|
switch typ {
|
|
case wire.BytesType:
|
|
v, m := wire.ConsumeBytes(b)
|
|
b = b[m:]
|
|
if isFile {
|
|
switch num {
|
|
case fieldnum.FileDescriptorProto_EnumType:
|
|
db.NumEnums++
|
|
case fieldnum.FileDescriptorProto_MessageType:
|
|
db.unmarshalCounts(v, false)
|
|
db.NumMessages++
|
|
case fieldnum.FileDescriptorProto_Extension:
|
|
db.NumExtensions++
|
|
case fieldnum.FileDescriptorProto_Service:
|
|
db.NumServices++
|
|
}
|
|
} else {
|
|
switch num {
|
|
case fieldnum.DescriptorProto_EnumType:
|
|
db.NumEnums++
|
|
case fieldnum.DescriptorProto_NestedType:
|
|
db.unmarshalCounts(v, false)
|
|
db.NumMessages++
|
|
case fieldnum.DescriptorProto_Extension:
|
|
db.NumExtensions++
|
|
}
|
|
}
|
|
default:
|
|
m := wire.ConsumeFieldValue(num, typ, b)
|
|
b = b[m:]
|
|
}
|
|
}
|
|
}
|