1# Protocol Buffers - Google's data interchange format 2# Copyright 2008 Google Inc. All rights reserved. 3# https://developers.google.com/protocol-buffers/ 4# 5# Redistribution and use in source and binary forms, with or without 6# modification, are permitted provided that the following conditions are 7# met: 8# 9# * Redistributions of source code must retain the above copyright 10# notice, this list of conditions and the following disclaimer. 11# * Redistributions in binary form must reproduce the above 12# copyright notice, this list of conditions and the following disclaimer 13# in the documentation and/or other materials provided with the 14# distribution. 15# * Neither the name of Google Inc. nor the names of its 16# contributors may be used to endorse or promote products derived from 17# this software without specific prior written permission. 18# 19# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 20# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 21# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 22# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 23# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 24# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 25# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 26# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 27# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 28# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 29# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 30 31"""Descriptors essentially contain exactly the information found in a .proto 32file, in types that make this information accessible in Python. 33""" 34 35__author__ = 'robinson@google.com (Will Robinson)' 36 37import six 38 39from google.protobuf.internal import api_implementation 40 41_USE_C_DESCRIPTORS = False 42if api_implementation.Type() == 'cpp': 43 # Used by MakeDescriptor in cpp mode 44 import os 45 import uuid 46 from google.protobuf.pyext import _message 47 _USE_C_DESCRIPTORS = getattr(_message, '_USE_C_DESCRIPTORS', False) 48 49 50class Error(Exception): 51 """Base error for this module.""" 52 53 54class TypeTransformationError(Error): 55 """Error transforming between python proto type and corresponding C++ type.""" 56 57 58if _USE_C_DESCRIPTORS: 59 # This metaclass allows to override the behavior of code like 60 # isinstance(my_descriptor, FieldDescriptor) 61 # and make it return True when the descriptor is an instance of the extension 62 # type written in C++. 63 class DescriptorMetaclass(type): 64 def __instancecheck__(cls, obj): 65 if super(DescriptorMetaclass, cls).__instancecheck__(obj): 66 return True 67 if isinstance(obj, cls._C_DESCRIPTOR_CLASS): 68 return True 69 return False 70else: 71 # The standard metaclass; nothing changes. 72 DescriptorMetaclass = type 73 74 75class DescriptorBase(six.with_metaclass(DescriptorMetaclass)): 76 77 """Descriptors base class. 78 79 This class is the base of all descriptor classes. It provides common options 80 related functionality. 81 82 Attributes: 83 has_options: True if the descriptor has non-default options. Usually it 84 is not necessary to read this -- just call GetOptions() which will 85 happily return the default instance. However, it's sometimes useful 86 for efficiency, and also useful inside the protobuf implementation to 87 avoid some bootstrapping issues. 88 """ 89 90 if _USE_C_DESCRIPTORS: 91 # The class, or tuple of classes, that are considered as "virtual 92 # subclasses" of this descriptor class. 93 _C_DESCRIPTOR_CLASS = () 94 95 def __init__(self, options, options_class_name): 96 """Initialize the descriptor given its options message and the name of the 97 class of the options message. The name of the class is required in case 98 the options message is None and has to be created. 99 """ 100 self._options = options 101 self._options_class_name = options_class_name 102 103 # Does this descriptor have non-default options? 104 self.has_options = options is not None 105 106 def _SetOptions(self, options, options_class_name): 107 """Sets the descriptor's options 108 109 This function is used in generated proto2 files to update descriptor 110 options. It must not be used outside proto2. 111 """ 112 self._options = options 113 self._options_class_name = options_class_name 114 115 # Does this descriptor have non-default options? 116 self.has_options = options is not None 117 118 def GetOptions(self): 119 """Retrieves descriptor options. 120 121 This method returns the options set or creates the default options for the 122 descriptor. 123 """ 124 if self._options: 125 return self._options 126 from google.protobuf import descriptor_pb2 127 try: 128 options_class = getattr(descriptor_pb2, self._options_class_name) 129 except AttributeError: 130 raise RuntimeError('Unknown options class name %s!' % 131 (self._options_class_name)) 132 self._options = options_class() 133 return self._options 134 135 136class _NestedDescriptorBase(DescriptorBase): 137 """Common class for descriptors that can be nested.""" 138 139 def __init__(self, options, options_class_name, name, full_name, 140 file, containing_type, serialized_start=None, 141 serialized_end=None): 142 """Constructor. 143 144 Args: 145 options: Protocol message options or None 146 to use default message options. 147 options_class_name: (str) The class name of the above options. 148 149 name: (str) Name of this protocol message type. 150 full_name: (str) Fully-qualified name of this protocol message type, 151 which will include protocol "package" name and the name of any 152 enclosing types. 153 file: (FileDescriptor) Reference to file info. 154 containing_type: if provided, this is a nested descriptor, with this 155 descriptor as parent, otherwise None. 156 serialized_start: The start index (inclusive) in block in the 157 file.serialized_pb that describes this descriptor. 158 serialized_end: The end index (exclusive) in block in the 159 file.serialized_pb that describes this descriptor. 160 """ 161 super(_NestedDescriptorBase, self).__init__( 162 options, options_class_name) 163 164 self.name = name 165 # TODO(falk): Add function to calculate full_name instead of having it in 166 # memory? 167 self.full_name = full_name 168 self.file = file 169 self.containing_type = containing_type 170 171 self._serialized_start = serialized_start 172 self._serialized_end = serialized_end 173 174 def GetTopLevelContainingType(self): 175 """Returns the root if this is a nested type, or itself if its the root.""" 176 desc = self 177 while desc.containing_type is not None: 178 desc = desc.containing_type 179 return desc 180 181 def CopyToProto(self, proto): 182 """Copies this to the matching proto in descriptor_pb2. 183 184 Args: 185 proto: An empty proto instance from descriptor_pb2. 186 187 Raises: 188 Error: If self couldnt be serialized, due to to few constructor arguments. 189 """ 190 if (self.file is not None and 191 self._serialized_start is not None and 192 self._serialized_end is not None): 193 proto.ParseFromString(self.file.serialized_pb[ 194 self._serialized_start:self._serialized_end]) 195 else: 196 raise Error('Descriptor does not contain serialization.') 197 198 199class Descriptor(_NestedDescriptorBase): 200 201 """Descriptor for a protocol message type. 202 203 A Descriptor instance has the following attributes: 204 205 name: (str) Name of this protocol message type. 206 full_name: (str) Fully-qualified name of this protocol message type, 207 which will include protocol "package" name and the name of any 208 enclosing types. 209 210 containing_type: (Descriptor) Reference to the descriptor of the 211 type containing us, or None if this is top-level. 212 213 fields: (list of FieldDescriptors) Field descriptors for all 214 fields in this type. 215 fields_by_number: (dict int -> FieldDescriptor) Same FieldDescriptor 216 objects as in |fields|, but indexed by "number" attribute in each 217 FieldDescriptor. 218 fields_by_name: (dict str -> FieldDescriptor) Same FieldDescriptor 219 objects as in |fields|, but indexed by "name" attribute in each 220 FieldDescriptor. 221 fields_by_camelcase_name: (dict str -> FieldDescriptor) Same 222 FieldDescriptor objects as in |fields|, but indexed by 223 "camelcase_name" attribute in each FieldDescriptor. 224 225 nested_types: (list of Descriptors) Descriptor references 226 for all protocol message types nested within this one. 227 nested_types_by_name: (dict str -> Descriptor) Same Descriptor 228 objects as in |nested_types|, but indexed by "name" attribute 229 in each Descriptor. 230 231 enum_types: (list of EnumDescriptors) EnumDescriptor references 232 for all enums contained within this type. 233 enum_types_by_name: (dict str ->EnumDescriptor) Same EnumDescriptor 234 objects as in |enum_types|, but indexed by "name" attribute 235 in each EnumDescriptor. 236 enum_values_by_name: (dict str -> EnumValueDescriptor) Dict mapping 237 from enum value name to EnumValueDescriptor for that value. 238 239 extensions: (list of FieldDescriptor) All extensions defined directly 240 within this message type (NOT within a nested type). 241 extensions_by_name: (dict, string -> FieldDescriptor) Same FieldDescriptor 242 objects as |extensions|, but indexed by "name" attribute of each 243 FieldDescriptor. 244 245 is_extendable: Does this type define any extension ranges? 246 247 oneofs: (list of OneofDescriptor) The list of descriptors for oneof fields 248 in this message. 249 oneofs_by_name: (dict str -> OneofDescriptor) Same objects as in |oneofs|, 250 but indexed by "name" attribute. 251 252 file: (FileDescriptor) Reference to file descriptor. 253 """ 254 255 if _USE_C_DESCRIPTORS: 256 _C_DESCRIPTOR_CLASS = _message.Descriptor 257 258 def __new__(cls, name, full_name, filename, containing_type, fields, 259 nested_types, enum_types, extensions, options=None, 260 is_extendable=True, extension_ranges=None, oneofs=None, 261 file=None, serialized_start=None, serialized_end=None, # pylint: disable=redefined-builtin 262 syntax=None): 263 _message.Message._CheckCalledFromGeneratedFile() 264 return _message.default_pool.FindMessageTypeByName(full_name) 265 266 # NOTE(tmarek): The file argument redefining a builtin is nothing we can 267 # fix right now since we don't know how many clients already rely on the 268 # name of the argument. 269 def __init__(self, name, full_name, filename, containing_type, fields, 270 nested_types, enum_types, extensions, options=None, 271 is_extendable=True, extension_ranges=None, oneofs=None, 272 file=None, serialized_start=None, serialized_end=None, # pylint: disable=redefined-builtin 273 syntax=None): 274 """Arguments to __init__() are as described in the description 275 of Descriptor fields above. 276 277 Note that filename is an obsolete argument, that is not used anymore. 278 Please use file.name to access this as an attribute. 279 """ 280 super(Descriptor, self).__init__( 281 options, 'MessageOptions', name, full_name, file, 282 containing_type, serialized_start=serialized_start, 283 serialized_end=serialized_end) 284 285 # We have fields in addition to fields_by_name and fields_by_number, 286 # so that: 287 # 1. Clients can index fields by "order in which they're listed." 288 # 2. Clients can easily iterate over all fields with the terse 289 # syntax: for f in descriptor.fields: ... 290 self.fields = fields 291 for field in self.fields: 292 field.containing_type = self 293 self.fields_by_number = dict((f.number, f) for f in fields) 294 self.fields_by_name = dict((f.name, f) for f in fields) 295 self._fields_by_camelcase_name = None 296 297 self.nested_types = nested_types 298 for nested_type in nested_types: 299 nested_type.containing_type = self 300 self.nested_types_by_name = dict((t.name, t) for t in nested_types) 301 302 self.enum_types = enum_types 303 for enum_type in self.enum_types: 304 enum_type.containing_type = self 305 self.enum_types_by_name = dict((t.name, t) for t in enum_types) 306 self.enum_values_by_name = dict( 307 (v.name, v) for t in enum_types for v in t.values) 308 309 self.extensions = extensions 310 for extension in self.extensions: 311 extension.extension_scope = self 312 self.extensions_by_name = dict((f.name, f) for f in extensions) 313 self.is_extendable = is_extendable 314 self.extension_ranges = extension_ranges 315 self.oneofs = oneofs if oneofs is not None else [] 316 self.oneofs_by_name = dict((o.name, o) for o in self.oneofs) 317 for oneof in self.oneofs: 318 oneof.containing_type = self 319 self.syntax = syntax or "proto2" 320 321 @property 322 def fields_by_camelcase_name(self): 323 if self._fields_by_camelcase_name is None: 324 self._fields_by_camelcase_name = dict( 325 (f.camelcase_name, f) for f in self.fields) 326 return self._fields_by_camelcase_name 327 328 def EnumValueName(self, enum, value): 329 """Returns the string name of an enum value. 330 331 This is just a small helper method to simplify a common operation. 332 333 Args: 334 enum: string name of the Enum. 335 value: int, value of the enum. 336 337 Returns: 338 string name of the enum value. 339 340 Raises: 341 KeyError if either the Enum doesn't exist or the value is not a valid 342 value for the enum. 343 """ 344 return self.enum_types_by_name[enum].values_by_number[value].name 345 346 def CopyToProto(self, proto): 347 """Copies this to a descriptor_pb2.DescriptorProto. 348 349 Args: 350 proto: An empty descriptor_pb2.DescriptorProto. 351 """ 352 # This function is overridden to give a better doc comment. 353 super(Descriptor, self).CopyToProto(proto) 354 355 356# TODO(robinson): We should have aggressive checking here, 357# for example: 358# * If you specify a repeated field, you should not be allowed 359# to specify a default value. 360# * [Other examples here as needed]. 361# 362# TODO(robinson): for this and other *Descriptor classes, we 363# might also want to lock things down aggressively (e.g., 364# prevent clients from setting the attributes). Having 365# stronger invariants here in general will reduce the number 366# of runtime checks we must do in reflection.py... 367class FieldDescriptor(DescriptorBase): 368 369 """Descriptor for a single field in a .proto file. 370 371 A FieldDescriptor instance has the following attributes: 372 373 name: (str) Name of this field, exactly as it appears in .proto. 374 full_name: (str) Name of this field, including containing scope. This is 375 particularly relevant for extensions. 376 camelcase_name: (str) Camelcase name of this field. 377 index: (int) Dense, 0-indexed index giving the order that this 378 field textually appears within its message in the .proto file. 379 number: (int) Tag number declared for this field in the .proto file. 380 381 type: (One of the TYPE_* constants below) Declared type. 382 cpp_type: (One of the CPPTYPE_* constants below) C++ type used to 383 represent this field. 384 385 label: (One of the LABEL_* constants below) Tells whether this 386 field is optional, required, or repeated. 387 has_default_value: (bool) True if this field has a default value defined, 388 otherwise false. 389 default_value: (Varies) Default value of this field. Only 390 meaningful for non-repeated scalar fields. Repeated fields 391 should always set this to [], and non-repeated composite 392 fields should always set this to None. 393 394 containing_type: (Descriptor) Descriptor of the protocol message 395 type that contains this field. Set by the Descriptor constructor 396 if we're passed into one. 397 Somewhat confusingly, for extension fields, this is the 398 descriptor of the EXTENDED message, not the descriptor 399 of the message containing this field. (See is_extension and 400 extension_scope below). 401 message_type: (Descriptor) If a composite field, a descriptor 402 of the message type contained in this field. Otherwise, this is None. 403 enum_type: (EnumDescriptor) If this field contains an enum, a 404 descriptor of that enum. Otherwise, this is None. 405 406 is_extension: True iff this describes an extension field. 407 extension_scope: (Descriptor) Only meaningful if is_extension is True. 408 Gives the message that immediately contains this extension field. 409 Will be None iff we're a top-level (file-level) extension field. 410 411 options: (descriptor_pb2.FieldOptions) Protocol message field options or 412 None to use default field options. 413 414 containing_oneof: (OneofDescriptor) If the field is a member of a oneof 415 union, contains its descriptor. Otherwise, None. 416 """ 417 418 # Must be consistent with C++ FieldDescriptor::Type enum in 419 # descriptor.h. 420 # 421 # TODO(robinson): Find a way to eliminate this repetition. 422 TYPE_DOUBLE = 1 423 TYPE_FLOAT = 2 424 TYPE_INT64 = 3 425 TYPE_UINT64 = 4 426 TYPE_INT32 = 5 427 TYPE_FIXED64 = 6 428 TYPE_FIXED32 = 7 429 TYPE_BOOL = 8 430 TYPE_STRING = 9 431 TYPE_GROUP = 10 432 TYPE_MESSAGE = 11 433 TYPE_BYTES = 12 434 TYPE_UINT32 = 13 435 TYPE_ENUM = 14 436 TYPE_SFIXED32 = 15 437 TYPE_SFIXED64 = 16 438 TYPE_SINT32 = 17 439 TYPE_SINT64 = 18 440 MAX_TYPE = 18 441 442 # Must be consistent with C++ FieldDescriptor::CppType enum in 443 # descriptor.h. 444 # 445 # TODO(robinson): Find a way to eliminate this repetition. 446 CPPTYPE_INT32 = 1 447 CPPTYPE_INT64 = 2 448 CPPTYPE_UINT32 = 3 449 CPPTYPE_UINT64 = 4 450 CPPTYPE_DOUBLE = 5 451 CPPTYPE_FLOAT = 6 452 CPPTYPE_BOOL = 7 453 CPPTYPE_ENUM = 8 454 CPPTYPE_STRING = 9 455 CPPTYPE_MESSAGE = 10 456 MAX_CPPTYPE = 10 457 458 _PYTHON_TO_CPP_PROTO_TYPE_MAP = { 459 TYPE_DOUBLE: CPPTYPE_DOUBLE, 460 TYPE_FLOAT: CPPTYPE_FLOAT, 461 TYPE_ENUM: CPPTYPE_ENUM, 462 TYPE_INT64: CPPTYPE_INT64, 463 TYPE_SINT64: CPPTYPE_INT64, 464 TYPE_SFIXED64: CPPTYPE_INT64, 465 TYPE_UINT64: CPPTYPE_UINT64, 466 TYPE_FIXED64: CPPTYPE_UINT64, 467 TYPE_INT32: CPPTYPE_INT32, 468 TYPE_SFIXED32: CPPTYPE_INT32, 469 TYPE_SINT32: CPPTYPE_INT32, 470 TYPE_UINT32: CPPTYPE_UINT32, 471 TYPE_FIXED32: CPPTYPE_UINT32, 472 TYPE_BYTES: CPPTYPE_STRING, 473 TYPE_STRING: CPPTYPE_STRING, 474 TYPE_BOOL: CPPTYPE_BOOL, 475 TYPE_MESSAGE: CPPTYPE_MESSAGE, 476 TYPE_GROUP: CPPTYPE_MESSAGE 477 } 478 479 # Must be consistent with C++ FieldDescriptor::Label enum in 480 # descriptor.h. 481 # 482 # TODO(robinson): Find a way to eliminate this repetition. 483 LABEL_OPTIONAL = 1 484 LABEL_REQUIRED = 2 485 LABEL_REPEATED = 3 486 MAX_LABEL = 3 487 488 # Must be consistent with C++ constants kMaxNumber, kFirstReservedNumber, 489 # and kLastReservedNumber in descriptor.h 490 MAX_FIELD_NUMBER = (1 << 29) - 1 491 FIRST_RESERVED_FIELD_NUMBER = 19000 492 LAST_RESERVED_FIELD_NUMBER = 19999 493 494 if _USE_C_DESCRIPTORS: 495 _C_DESCRIPTOR_CLASS = _message.FieldDescriptor 496 497 def __new__(cls, name, full_name, index, number, type, cpp_type, label, 498 default_value, message_type, enum_type, containing_type, 499 is_extension, extension_scope, options=None, 500 has_default_value=True, containing_oneof=None): 501 _message.Message._CheckCalledFromGeneratedFile() 502 if is_extension: 503 return _message.default_pool.FindExtensionByName(full_name) 504 else: 505 return _message.default_pool.FindFieldByName(full_name) 506 507 def __init__(self, name, full_name, index, number, type, cpp_type, label, 508 default_value, message_type, enum_type, containing_type, 509 is_extension, extension_scope, options=None, 510 has_default_value=True, containing_oneof=None): 511 """The arguments are as described in the description of FieldDescriptor 512 attributes above. 513 514 Note that containing_type may be None, and may be set later if necessary 515 (to deal with circular references between message types, for example). 516 Likewise for extension_scope. 517 """ 518 super(FieldDescriptor, self).__init__(options, 'FieldOptions') 519 self.name = name 520 self.full_name = full_name 521 self._camelcase_name = None 522 self.index = index 523 self.number = number 524 self.type = type 525 self.cpp_type = cpp_type 526 self.label = label 527 self.has_default_value = has_default_value 528 self.default_value = default_value 529 self.containing_type = containing_type 530 self.message_type = message_type 531 self.enum_type = enum_type 532 self.is_extension = is_extension 533 self.extension_scope = extension_scope 534 self.containing_oneof = containing_oneof 535 if api_implementation.Type() == 'cpp': 536 if is_extension: 537 self._cdescriptor = _message.default_pool.FindExtensionByName(full_name) 538 else: 539 self._cdescriptor = _message.default_pool.FindFieldByName(full_name) 540 else: 541 self._cdescriptor = None 542 543 @property 544 def camelcase_name(self): 545 if self._camelcase_name is None: 546 self._camelcase_name = _ToCamelCase(self.name) 547 return self._camelcase_name 548 549 @staticmethod 550 def ProtoTypeToCppProtoType(proto_type): 551 """Converts from a Python proto type to a C++ Proto Type. 552 553 The Python ProtocolBuffer classes specify both the 'Python' datatype and the 554 'C++' datatype - and they're not the same. This helper method should 555 translate from one to another. 556 557 Args: 558 proto_type: the Python proto type (descriptor.FieldDescriptor.TYPE_*) 559 Returns: 560 descriptor.FieldDescriptor.CPPTYPE_*, the C++ type. 561 Raises: 562 TypeTransformationError: when the Python proto type isn't known. 563 """ 564 try: 565 return FieldDescriptor._PYTHON_TO_CPP_PROTO_TYPE_MAP[proto_type] 566 except KeyError: 567 raise TypeTransformationError('Unknown proto_type: %s' % proto_type) 568 569 570class EnumDescriptor(_NestedDescriptorBase): 571 572 """Descriptor for an enum defined in a .proto file. 573 574 An EnumDescriptor instance has the following attributes: 575 576 name: (str) Name of the enum type. 577 full_name: (str) Full name of the type, including package name 578 and any enclosing type(s). 579 580 values: (list of EnumValueDescriptors) List of the values 581 in this enum. 582 values_by_name: (dict str -> EnumValueDescriptor) Same as |values|, 583 but indexed by the "name" field of each EnumValueDescriptor. 584 values_by_number: (dict int -> EnumValueDescriptor) Same as |values|, 585 but indexed by the "number" field of each EnumValueDescriptor. 586 containing_type: (Descriptor) Descriptor of the immediate containing 587 type of this enum, or None if this is an enum defined at the 588 top level in a .proto file. Set by Descriptor's constructor 589 if we're passed into one. 590 file: (FileDescriptor) Reference to file descriptor. 591 options: (descriptor_pb2.EnumOptions) Enum options message or 592 None to use default enum options. 593 """ 594 595 if _USE_C_DESCRIPTORS: 596 _C_DESCRIPTOR_CLASS = _message.EnumDescriptor 597 598 def __new__(cls, name, full_name, filename, values, 599 containing_type=None, options=None, file=None, 600 serialized_start=None, serialized_end=None): 601 _message.Message._CheckCalledFromGeneratedFile() 602 return _message.default_pool.FindEnumTypeByName(full_name) 603 604 def __init__(self, name, full_name, filename, values, 605 containing_type=None, options=None, file=None, 606 serialized_start=None, serialized_end=None): 607 """Arguments are as described in the attribute description above. 608 609 Note that filename is an obsolete argument, that is not used anymore. 610 Please use file.name to access this as an attribute. 611 """ 612 super(EnumDescriptor, self).__init__( 613 options, 'EnumOptions', name, full_name, file, 614 containing_type, serialized_start=serialized_start, 615 serialized_end=serialized_end) 616 617 self.values = values 618 for value in self.values: 619 value.type = self 620 self.values_by_name = dict((v.name, v) for v in values) 621 self.values_by_number = dict((v.number, v) for v in values) 622 623 def CopyToProto(self, proto): 624 """Copies this to a descriptor_pb2.EnumDescriptorProto. 625 626 Args: 627 proto: An empty descriptor_pb2.EnumDescriptorProto. 628 """ 629 # This function is overridden to give a better doc comment. 630 super(EnumDescriptor, self).CopyToProto(proto) 631 632 633class EnumValueDescriptor(DescriptorBase): 634 635 """Descriptor for a single value within an enum. 636 637 name: (str) Name of this value. 638 index: (int) Dense, 0-indexed index giving the order that this 639 value appears textually within its enum in the .proto file. 640 number: (int) Actual number assigned to this enum value. 641 type: (EnumDescriptor) EnumDescriptor to which this value 642 belongs. Set by EnumDescriptor's constructor if we're 643 passed into one. 644 options: (descriptor_pb2.EnumValueOptions) Enum value options message or 645 None to use default enum value options options. 646 """ 647 648 if _USE_C_DESCRIPTORS: 649 _C_DESCRIPTOR_CLASS = _message.EnumValueDescriptor 650 651 def __new__(cls, name, index, number, type=None, options=None): 652 _message.Message._CheckCalledFromGeneratedFile() 653 # There is no way we can build a complete EnumValueDescriptor with the 654 # given parameters (the name of the Enum is not known, for example). 655 # Fortunately generated files just pass it to the EnumDescriptor() 656 # constructor, which will ignore it, so returning None is good enough. 657 return None 658 659 def __init__(self, name, index, number, type=None, options=None): 660 """Arguments are as described in the attribute description above.""" 661 super(EnumValueDescriptor, self).__init__(options, 'EnumValueOptions') 662 self.name = name 663 self.index = index 664 self.number = number 665 self.type = type 666 667 668class OneofDescriptor(DescriptorBase): 669 """Descriptor for a oneof field. 670 671 name: (str) Name of the oneof field. 672 full_name: (str) Full name of the oneof field, including package name. 673 index: (int) 0-based index giving the order of the oneof field inside 674 its containing type. 675 containing_type: (Descriptor) Descriptor of the protocol message 676 type that contains this field. Set by the Descriptor constructor 677 if we're passed into one. 678 fields: (list of FieldDescriptor) The list of field descriptors this 679 oneof can contain. 680 """ 681 682 if _USE_C_DESCRIPTORS: 683 _C_DESCRIPTOR_CLASS = _message.OneofDescriptor 684 685 def __new__( 686 cls, name, full_name, index, containing_type, fields, options=None): 687 _message.Message._CheckCalledFromGeneratedFile() 688 return _message.default_pool.FindOneofByName(full_name) 689 690 def __init__( 691 self, name, full_name, index, containing_type, fields, options=None): 692 """Arguments are as described in the attribute description above.""" 693 super(OneofDescriptor, self).__init__(options, 'OneofOptions') 694 self.name = name 695 self.full_name = full_name 696 self.index = index 697 self.containing_type = containing_type 698 self.fields = fields 699 700 701class ServiceDescriptor(_NestedDescriptorBase): 702 703 """Descriptor for a service. 704 705 name: (str) Name of the service. 706 full_name: (str) Full name of the service, including package name. 707 index: (int) 0-indexed index giving the order that this services 708 definition appears withing the .proto file. 709 methods: (list of MethodDescriptor) List of methods provided by this 710 service. 711 methods_by_name: (dict str -> MethodDescriptor) Same MethodDescriptor 712 objects as in |methods_by_name|, but indexed by "name" attribute in each 713 MethodDescriptor. 714 options: (descriptor_pb2.ServiceOptions) Service options message or 715 None to use default service options. 716 file: (FileDescriptor) Reference to file info. 717 """ 718 719 if _USE_C_DESCRIPTORS: 720 _C_DESCRIPTOR_CLASS = _message.ServiceDescriptor 721 722 def __new__(cls, name, full_name, index, methods, options=None, file=None, # pylint: disable=redefined-builtin 723 serialized_start=None, serialized_end=None): 724 _message.Message._CheckCalledFromGeneratedFile() # pylint: disable=protected-access 725 return _message.default_pool.FindServiceByName(full_name) 726 727 def __init__(self, name, full_name, index, methods, options=None, file=None, 728 serialized_start=None, serialized_end=None): 729 super(ServiceDescriptor, self).__init__( 730 options, 'ServiceOptions', name, full_name, file, 731 None, serialized_start=serialized_start, 732 serialized_end=serialized_end) 733 self.index = index 734 self.methods = methods 735 self.methods_by_name = dict((m.name, m) for m in methods) 736 # Set the containing service for each method in this service. 737 for method in self.methods: 738 method.containing_service = self 739 740 def FindMethodByName(self, name): 741 """Searches for the specified method, and returns its descriptor.""" 742 return self.methods_by_name.get(name, None) 743 744 def CopyToProto(self, proto): 745 """Copies this to a descriptor_pb2.ServiceDescriptorProto. 746 747 Args: 748 proto: An empty descriptor_pb2.ServiceDescriptorProto. 749 """ 750 # This function is overridden to give a better doc comment. 751 super(ServiceDescriptor, self).CopyToProto(proto) 752 753 754class MethodDescriptor(DescriptorBase): 755 756 """Descriptor for a method in a service. 757 758 name: (str) Name of the method within the service. 759 full_name: (str) Full name of method. 760 index: (int) 0-indexed index of the method inside the service. 761 containing_service: (ServiceDescriptor) The service that contains this 762 method. 763 input_type: The descriptor of the message that this method accepts. 764 output_type: The descriptor of the message that this method returns. 765 options: (descriptor_pb2.MethodOptions) Method options message or 766 None to use default method options. 767 """ 768 769 if _USE_C_DESCRIPTORS: 770 _C_DESCRIPTOR_CLASS = _message.MethodDescriptor 771 772 def __new__(cls, name, full_name, index, containing_service, 773 input_type, output_type, options=None): 774 _message.Message._CheckCalledFromGeneratedFile() # pylint: disable=protected-access 775 return _message.default_pool.FindMethodByName(full_name) 776 777 def __init__(self, name, full_name, index, containing_service, 778 input_type, output_type, options=None): 779 """The arguments are as described in the description of MethodDescriptor 780 attributes above. 781 782 Note that containing_service may be None, and may be set later if necessary. 783 """ 784 super(MethodDescriptor, self).__init__(options, 'MethodOptions') 785 self.name = name 786 self.full_name = full_name 787 self.index = index 788 self.containing_service = containing_service 789 self.input_type = input_type 790 self.output_type = output_type 791 792 793class FileDescriptor(DescriptorBase): 794 """Descriptor for a file. Mimics the descriptor_pb2.FileDescriptorProto. 795 796 Note that enum_types_by_name, extensions_by_name, and dependencies 797 fields are only set by the message_factory module, and not by the 798 generated proto code. 799 800 name: name of file, relative to root of source tree. 801 package: name of the package 802 syntax: string indicating syntax of the file (can be "proto2" or "proto3") 803 serialized_pb: (str) Byte string of serialized 804 descriptor_pb2.FileDescriptorProto. 805 dependencies: List of other FileDescriptors this FileDescriptor depends on. 806 public_dependencies: A list of FileDescriptors, subset of the dependencies 807 above, which were declared as "public". 808 message_types_by_name: Dict of message names of their descriptors. 809 enum_types_by_name: Dict of enum names and their descriptors. 810 extensions_by_name: Dict of extension names and their descriptors. 811 services_by_name: Dict of services names and their descriptors. 812 pool: the DescriptorPool this descriptor belongs to. When not passed to the 813 constructor, the global default pool is used. 814 """ 815 816 if _USE_C_DESCRIPTORS: 817 _C_DESCRIPTOR_CLASS = _message.FileDescriptor 818 819 def __new__(cls, name, package, options=None, serialized_pb=None, 820 dependencies=None, public_dependencies=None, 821 syntax=None, pool=None): 822 # FileDescriptor() is called from various places, not only from generated 823 # files, to register dynamic proto files and messages. 824 if serialized_pb: 825 # TODO(amauryfa): use the pool passed as argument. This will work only 826 # for C++-implemented DescriptorPools. 827 return _message.default_pool.AddSerializedFile(serialized_pb) 828 else: 829 return super(FileDescriptor, cls).__new__(cls) 830 831 def __init__(self, name, package, options=None, serialized_pb=None, 832 dependencies=None, public_dependencies=None, 833 syntax=None, pool=None): 834 """Constructor.""" 835 super(FileDescriptor, self).__init__(options, 'FileOptions') 836 837 if pool is None: 838 from google.protobuf import descriptor_pool 839 pool = descriptor_pool.Default() 840 self.pool = pool 841 self.message_types_by_name = {} 842 self.name = name 843 self.package = package 844 self.syntax = syntax or "proto2" 845 self.serialized_pb = serialized_pb 846 847 self.enum_types_by_name = {} 848 self.extensions_by_name = {} 849 self.services_by_name = {} 850 self.dependencies = (dependencies or []) 851 self.public_dependencies = (public_dependencies or []) 852 853 if (api_implementation.Type() == 'cpp' and 854 self.serialized_pb is not None): 855 _message.default_pool.AddSerializedFile(self.serialized_pb) 856 857 def CopyToProto(self, proto): 858 """Copies this to a descriptor_pb2.FileDescriptorProto. 859 860 Args: 861 proto: An empty descriptor_pb2.FileDescriptorProto. 862 """ 863 proto.ParseFromString(self.serialized_pb) 864 865 866def _ParseOptions(message, string): 867 """Parses serialized options. 868 869 This helper function is used to parse serialized options in generated 870 proto2 files. It must not be used outside proto2. 871 """ 872 message.ParseFromString(string) 873 return message 874 875 876def _ToCamelCase(name): 877 """Converts name to camel-case and returns it.""" 878 capitalize_next = False 879 result = [] 880 881 for c in name: 882 if c == '_': 883 if result: 884 capitalize_next = True 885 elif capitalize_next: 886 result.append(c.upper()) 887 capitalize_next = False 888 else: 889 result += c 890 891 # Lower-case the first letter. 892 if result and result[0].isupper(): 893 result[0] = result[0].lower() 894 return ''.join(result) 895 896 897def MakeDescriptor(desc_proto, package='', build_file_if_cpp=True, 898 syntax=None): 899 """Make a protobuf Descriptor given a DescriptorProto protobuf. 900 901 Handles nested descriptors. Note that this is limited to the scope of defining 902 a message inside of another message. Composite fields can currently only be 903 resolved if the message is defined in the same scope as the field. 904 905 Args: 906 desc_proto: The descriptor_pb2.DescriptorProto protobuf message. 907 package: Optional package name for the new message Descriptor (string). 908 build_file_if_cpp: Update the C++ descriptor pool if api matches. 909 Set to False on recursion, so no duplicates are created. 910 syntax: The syntax/semantics that should be used. Set to "proto3" to get 911 proto3 field presence semantics. 912 Returns: 913 A Descriptor for protobuf messages. 914 """ 915 if api_implementation.Type() == 'cpp' and build_file_if_cpp: 916 # The C++ implementation requires all descriptors to be backed by the same 917 # definition in the C++ descriptor pool. To do this, we build a 918 # FileDescriptorProto with the same definition as this descriptor and build 919 # it into the pool. 920 from google.protobuf import descriptor_pb2 921 file_descriptor_proto = descriptor_pb2.FileDescriptorProto() 922 file_descriptor_proto.message_type.add().MergeFrom(desc_proto) 923 924 # Generate a random name for this proto file to prevent conflicts with any 925 # imported ones. We need to specify a file name so the descriptor pool 926 # accepts our FileDescriptorProto, but it is not important what that file 927 # name is actually set to. 928 proto_name = str(uuid.uuid4()) 929 930 if package: 931 file_descriptor_proto.name = os.path.join(package.replace('.', '/'), 932 proto_name + '.proto') 933 file_descriptor_proto.package = package 934 else: 935 file_descriptor_proto.name = proto_name + '.proto' 936 937 _message.default_pool.Add(file_descriptor_proto) 938 result = _message.default_pool.FindFileByName(file_descriptor_proto.name) 939 940 if _USE_C_DESCRIPTORS: 941 return result.message_types_by_name[desc_proto.name] 942 943 full_message_name = [desc_proto.name] 944 if package: full_message_name.insert(0, package) 945 946 # Create Descriptors for enum types 947 enum_types = {} 948 for enum_proto in desc_proto.enum_type: 949 full_name = '.'.join(full_message_name + [enum_proto.name]) 950 enum_desc = EnumDescriptor( 951 enum_proto.name, full_name, None, [ 952 EnumValueDescriptor(enum_val.name, ii, enum_val.number) 953 for ii, enum_val in enumerate(enum_proto.value)]) 954 enum_types[full_name] = enum_desc 955 956 # Create Descriptors for nested types 957 nested_types = {} 958 for nested_proto in desc_proto.nested_type: 959 full_name = '.'.join(full_message_name + [nested_proto.name]) 960 # Nested types are just those defined inside of the message, not all types 961 # used by fields in the message, so no loops are possible here. 962 nested_desc = MakeDescriptor(nested_proto, 963 package='.'.join(full_message_name), 964 build_file_if_cpp=False, 965 syntax=syntax) 966 nested_types[full_name] = nested_desc 967 968 fields = [] 969 for field_proto in desc_proto.field: 970 full_name = '.'.join(full_message_name + [field_proto.name]) 971 enum_desc = None 972 nested_desc = None 973 if field_proto.HasField('type_name'): 974 type_name = field_proto.type_name 975 full_type_name = '.'.join(full_message_name + 976 [type_name[type_name.rfind('.')+1:]]) 977 if full_type_name in nested_types: 978 nested_desc = nested_types[full_type_name] 979 elif full_type_name in enum_types: 980 enum_desc = enum_types[full_type_name] 981 # Else type_name references a non-local type, which isn't implemented 982 field = FieldDescriptor( 983 field_proto.name, full_name, field_proto.number - 1, 984 field_proto.number, field_proto.type, 985 FieldDescriptor.ProtoTypeToCppProtoType(field_proto.type), 986 field_proto.label, None, nested_desc, enum_desc, None, False, None, 987 options=field_proto.options, has_default_value=False) 988 fields.append(field) 989 990 desc_name = '.'.join(full_message_name) 991 return Descriptor(desc_proto.name, desc_name, None, None, fields, 992 list(nested_types.values()), list(enum_types.values()), [], 993 options=desc_proto.options) 994