Commit f8e7a46b authored by Jisi Liu's avatar Jisi Liu

Merge pull request #221 from google/python_proto3

Proto3 Python changes for v3.0.0-alpha-2
parents 3feaac7d 0b70a437
...@@ -197,7 +197,6 @@ javanano_EXTRA_DIST= ...@@ -197,7 +197,6 @@ javanano_EXTRA_DIST=
python_EXTRA_DIST= \ python_EXTRA_DIST= \
python/google/protobuf/internal/api_implementation.cc \ python/google/protobuf/internal/api_implementation.cc \
python/google/protobuf/internal/api_implementation.py \ python/google/protobuf/internal/api_implementation.py \
python/google/protobuf/internal/api_implementation_default_test.py \
python/google/protobuf/internal/containers.py \ python/google/protobuf/internal/containers.py \
python/google/protobuf/internal/cpp_message.py \ python/google/protobuf/internal/cpp_message.py \
python/google/protobuf/internal/decoder.py \ python/google/protobuf/internal/decoder.py \
...@@ -221,6 +220,7 @@ python_EXTRA_DIST= \ ...@@ -221,6 +220,7 @@ python_EXTRA_DIST= \
python/google/protobuf/internal/more_extensions.proto \ python/google/protobuf/internal/more_extensions.proto \
python/google/protobuf/internal/more_extensions_dynamic.proto \ python/google/protobuf/internal/more_extensions_dynamic.proto \
python/google/protobuf/internal/more_messages.proto \ python/google/protobuf/internal/more_messages.proto \
python/google/protobuf/internal/_parameterized.py \
python/google/protobuf/internal/proto_builder_test.py \ python/google/protobuf/internal/proto_builder_test.py \
python/google/protobuf/internal/python_message.py \ python/google/protobuf/internal/python_message.py \
python/google/protobuf/internal/reflection_test.py \ python/google/protobuf/internal/reflection_test.py \
...@@ -242,16 +242,17 @@ python_EXTRA_DIST= \ ...@@ -242,16 +242,17 @@ python_EXTRA_DIST= \
python/google/protobuf/pyext/cpp_message.py \ python/google/protobuf/pyext/cpp_message.py \
python/google/protobuf/pyext/descriptor.h \ python/google/protobuf/pyext/descriptor.h \
python/google/protobuf/pyext/descriptor.cc \ python/google/protobuf/pyext/descriptor.cc \
python/google/protobuf/pyext/descriptor_cpp2_test.py \ python/google/protobuf/pyext/descriptor_pool.h \
python/google/protobuf/pyext/descriptor_pool.cc \
python/google/protobuf/pyext/descriptor_containers.h \
python/google/protobuf/pyext/descriptor_containers.cc \
python/google/protobuf/pyext/extension_dict.h \ python/google/protobuf/pyext/extension_dict.h \
python/google/protobuf/pyext/extension_dict.cc \ python/google/protobuf/pyext/extension_dict.cc \
python/google/protobuf/pyext/message.h \ python/google/protobuf/pyext/message.h \
python/google/protobuf/pyext/message.cc \ python/google/protobuf/pyext/message.cc \
python/google/protobuf/pyext/message_factory_cpp2_test.py \
python/google/protobuf/pyext/proto2_api_test.proto \ python/google/protobuf/pyext/proto2_api_test.proto \
python/google/protobuf/pyext/python.proto \ python/google/protobuf/pyext/python.proto \
python/google/protobuf/pyext/python_protobuf.h \ python/google/protobuf/pyext/python_protobuf.h \
python/google/protobuf/pyext/reflection_cpp2_generated_test.py \
python/google/protobuf/pyext/repeated_composite_container.h \ python/google/protobuf/pyext/repeated_composite_container.h \
python/google/protobuf/pyext/repeated_composite_container.cc \ python/google/protobuf/pyext/repeated_composite_container.cc \
python/google/protobuf/pyext/repeated_scalar_container.h \ python/google/protobuf/pyext/repeated_scalar_container.h \
......
...@@ -48,8 +48,9 @@ Installation ...@@ -48,8 +48,9 @@ Installation
$ python setup.py build $ python setup.py build
$ python setup.py google_test $ python setup.py google_test
If you want to test c++ implementation, run: If you want to build/test c++ implementation, run:
$ python setup.py test --cpp_implementation $ python setup.py build --cpp_implementation
$ python setup.py google_test --cpp_implementation
If some tests fail, this library may not work correctly on your If some tests fail, this library may not work correctly on your
system. Continue at your own risk. system. Continue at your own risk.
......
This diff is collapsed.
...@@ -64,6 +64,9 @@ from google.protobuf import descriptor_database ...@@ -64,6 +64,9 @@ from google.protobuf import descriptor_database
from google.protobuf import text_encoding from google.protobuf import text_encoding
_USE_C_DESCRIPTORS = descriptor._USE_C_DESCRIPTORS
def _NormalizeFullyQualifiedName(name): def _NormalizeFullyQualifiedName(name):
"""Remove leading period from fully-qualified type name. """Remove leading period from fully-qualified type name.
...@@ -271,58 +274,81 @@ class DescriptorPool(object): ...@@ -271,58 +274,81 @@ class DescriptorPool(object):
file_descriptor = descriptor.FileDescriptor( file_descriptor = descriptor.FileDescriptor(
name=file_proto.name, name=file_proto.name,
package=file_proto.package, package=file_proto.package,
syntax=file_proto.syntax,
options=file_proto.options, options=file_proto.options,
serialized_pb=file_proto.SerializeToString(), serialized_pb=file_proto.SerializeToString(),
dependencies=direct_deps) dependencies=direct_deps)
scope = {} if _USE_C_DESCRIPTORS:
# When using C++ descriptors, all objects defined in the file were added
# This loop extracts all the message and enum types from all the # to the C++ database when the FileDescriptor was built above.
# dependencoes of the file_proto. This is necessary to create the # Just add them to this descriptor pool.
# scope of available message types when defining the passed in def _AddMessageDescriptor(message_desc):
# file proto. self._descriptors[message_desc.full_name] = message_desc
for dependency in built_deps: for nested in message_desc.nested_types:
scope.update(self._ExtractSymbols( _AddMessageDescriptor(nested)
dependency.message_types_by_name.values())) for enum_type in message_desc.enum_types:
scope.update((_PrefixWithDot(enum.full_name), enum) _AddEnumDescriptor(enum_type)
for enum in dependency.enum_types_by_name.values()) def _AddEnumDescriptor(enum_desc):
self._enum_descriptors[enum_desc.full_name] = enum_desc
for message_type in file_proto.message_type: for message_type in file_descriptor.message_types_by_name.values():
message_desc = self._ConvertMessageDescriptor( _AddMessageDescriptor(message_type)
message_type, file_proto.package, file_descriptor, scope) for enum_type in file_descriptor.enum_types_by_name.values():
file_descriptor.message_types_by_name[message_desc.name] = message_desc _AddEnumDescriptor(enum_type)
for enum_type in file_proto.enum_type:
file_descriptor.enum_types_by_name[enum_type.name] = (
self._ConvertEnumDescriptor(enum_type, file_proto.package,
file_descriptor, None, scope))
for index, extension_proto in enumerate(file_proto.extension):
extension_desc = self.MakeFieldDescriptor(
extension_proto, file_proto.package, index, is_extension=True)
extension_desc.containing_type = self._GetTypeFromScope(
file_descriptor.package, extension_proto.extendee, scope)
self.SetFieldType(extension_proto, extension_desc,
file_descriptor.package, scope)
file_descriptor.extensions_by_name[extension_desc.name] = extension_desc
for desc_proto in file_proto.message_type:
self.SetAllFieldTypes(file_proto.package, desc_proto, scope)
if file_proto.package:
desc_proto_prefix = _PrefixWithDot(file_proto.package)
else: else:
desc_proto_prefix = '' scope = {}
# This loop extracts all the message and enum types from all the
# dependencies of the file_proto. This is necessary to create the
# scope of available message types when defining the passed in
# file proto.
for dependency in built_deps:
scope.update(self._ExtractSymbols(
dependency.message_types_by_name.values()))
scope.update((_PrefixWithDot(enum.full_name), enum)
for enum in dependency.enum_types_by_name.values())
for message_type in file_proto.message_type:
message_desc = self._ConvertMessageDescriptor(
message_type, file_proto.package, file_descriptor, scope,
file_proto.syntax)
file_descriptor.message_types_by_name[message_desc.name] = (
message_desc)
for enum_type in file_proto.enum_type:
file_descriptor.enum_types_by_name[enum_type.name] = (
self._ConvertEnumDescriptor(enum_type, file_proto.package,
file_descriptor, None, scope))
for index, extension_proto in enumerate(file_proto.extension):
extension_desc = self.MakeFieldDescriptor(
extension_proto, file_proto.package, index, is_extension=True)
extension_desc.containing_type = self._GetTypeFromScope(
file_descriptor.package, extension_proto.extendee, scope)
self.SetFieldType(extension_proto, extension_desc,
file_descriptor.package, scope)
file_descriptor.extensions_by_name[extension_desc.name] = (
extension_desc)
for desc_proto in file_proto.message_type:
self.SetAllFieldTypes(file_proto.package, desc_proto, scope)
if file_proto.package:
desc_proto_prefix = _PrefixWithDot(file_proto.package)
else:
desc_proto_prefix = ''
for desc_proto in file_proto.message_type:
desc = self._GetTypeFromScope(
desc_proto_prefix, desc_proto.name, scope)
file_descriptor.message_types_by_name[desc_proto.name] = desc
for desc_proto in file_proto.message_type:
desc = self._GetTypeFromScope(desc_proto_prefix, desc_proto.name, scope)
file_descriptor.message_types_by_name[desc_proto.name] = desc
self.Add(file_proto) self.Add(file_proto)
self._file_descriptors[file_proto.name] = file_descriptor self._file_descriptors[file_proto.name] = file_descriptor
return self._file_descriptors[file_proto.name] return self._file_descriptors[file_proto.name]
def _ConvertMessageDescriptor(self, desc_proto, package=None, file_desc=None, def _ConvertMessageDescriptor(self, desc_proto, package=None, file_desc=None,
scope=None): scope=None, syntax=None):
"""Adds the proto to the pool in the specified package. """Adds the proto to the pool in the specified package.
Args: Args:
...@@ -349,7 +375,8 @@ class DescriptorPool(object): ...@@ -349,7 +375,8 @@ class DescriptorPool(object):
scope = {} scope = {}
nested = [ nested = [
self._ConvertMessageDescriptor(nested, desc_name, file_desc, scope) self._ConvertMessageDescriptor(
nested, desc_name, file_desc, scope, syntax)
for nested in desc_proto.nested_type] for nested in desc_proto.nested_type]
enums = [ enums = [
self._ConvertEnumDescriptor(enum, desc_name, file_desc, None, scope) self._ConvertEnumDescriptor(enum, desc_name, file_desc, None, scope)
...@@ -383,7 +410,8 @@ class DescriptorPool(object): ...@@ -383,7 +410,8 @@ class DescriptorPool(object):
extension_ranges=extension_ranges, extension_ranges=extension_ranges,
file=file_desc, file=file_desc,
serialized_start=None, serialized_start=None,
serialized_end=None) serialized_end=None,
syntax=syntax)
for nested in desc.nested_types: for nested in desc.nested_types:
nested.containing_type = desc nested.containing_type = desc
for enum in desc.enum_types: for enum in desc.enum_types:
......
This diff is collapsed.
...@@ -50,10 +50,7 @@ namespace python { ...@@ -50,10 +50,7 @@ namespace python {
// and // and
// PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION_VERSION=2 // PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION_VERSION=2
#ifdef PYTHON_PROTO2_CPP_IMPL_V1 #ifdef PYTHON_PROTO2_CPP_IMPL_V1
#if PY_MAJOR_VERSION >= 3 #error "PYTHON_PROTO2_CPP_IMPL_V1 is no longer supported."
#error "PYTHON_PROTO2_CPP_IMPL_V1 is not supported under Python 3."
#endif
static int kImplVersion = 1;
#else #else
#ifdef PYTHON_PROTO2_CPP_IMPL_V2 #ifdef PYTHON_PROTO2_CPP_IMPL_V2
static int kImplVersion = 2; static int kImplVersion = 2;
...@@ -62,14 +59,7 @@ static int kImplVersion = 2; ...@@ -62,14 +59,7 @@ static int kImplVersion = 2;
static int kImplVersion = 0; static int kImplVersion = 0;
#else #else
// The defaults are set here. Python 3 uses the fast C++ APIv2 by default. static int kImplVersion = -1; // -1 means "Unspecified by compiler flags".
// Python 2 still uses the Python version by default until some compatibility
// issues can be worked around.
#if PY_MAJOR_VERSION >= 3
static int kImplVersion = 2;
#else
static int kImplVersion = 0;
#endif
#endif // PYTHON_PROTO2_PYTHON_IMPL #endif // PYTHON_PROTO2_PYTHON_IMPL
#endif // PYTHON_PROTO2_CPP_IMPL_V2 #endif // PYTHON_PROTO2_CPP_IMPL_V2
......
...@@ -40,14 +40,33 @@ try: ...@@ -40,14 +40,33 @@ try:
# The compile-time constants in the _api_implementation module can be used to # The compile-time constants in the _api_implementation module can be used to
# switch to a certain implementation of the Python API at build time. # switch to a certain implementation of the Python API at build time.
_api_version = _api_implementation.api_version _api_version = _api_implementation.api_version
del _api_implementation _proto_extension_modules_exist_in_build = True
except ImportError: except ImportError:
_api_version = 0 _api_version = -1 # Unspecified by compiler flags.
_proto_extension_modules_exist_in_build = False
if _api_version == 1:
raise ValueError('api_version=1 is no longer supported.')
if _api_version < 0: # Still unspecified?
try:
# The presence of this module in a build allows the proto implementation to
# be upgraded merely via build deps rather than a compiler flag or the
# runtime environment variable.
# pylint: disable=g-import-not-at-top
from google.protobuf import _use_fast_cpp_protos
# Work around a known issue in the classic bootstrap .par import hook.
if not _use_fast_cpp_protos:
raise ImportError('_use_fast_cpp_protos import succeeded but was None')
del _use_fast_cpp_protos
_api_version = 2
except ImportError:
if _proto_extension_modules_exist_in_build:
if sys.version_info[0] >= 3: # Python 3 defaults to C++ impl v2.
_api_version = 2
# TODO(b/17427486): Make Python 2 default to C++ impl v2.
_default_implementation_type = ( _default_implementation_type = (
'python' if _api_version == 0 else 'cpp') 'python' if _api_version <= 0 else 'cpp')
_default_version_str = (
'1' if _api_version <= 1 else '2')
# This environment variable can be used to switch to a certain implementation # This environment variable can be used to switch to a certain implementation
# of the Python API, overriding the compile-time constants in the # of the Python API, overriding the compile-time constants in the
...@@ -64,13 +83,12 @@ if _implementation_type != 'python': ...@@ -64,13 +83,12 @@ if _implementation_type != 'python':
# _api_implementation module. Right now only 1 and 2 are valid values. Any other # _api_implementation module. Right now only 1 and 2 are valid values. Any other
# value will be ignored. # value will be ignored.
_implementation_version_str = os.getenv( _implementation_version_str = os.getenv(
'PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION_VERSION', 'PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION_VERSION', '2')
_default_version_str)
if _implementation_version_str not in ('1', '2'): if _implementation_version_str != '2':
raise ValueError( raise ValueError(
"unsupported PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION_VERSION: '" + 'unsupported PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION_VERSION: "' +
_implementation_version_str + "' (supported versions: 1, 2)" _implementation_version_str + '" (supported versions: 2)'
) )
_implementation_version = int(_implementation_version_str) _implementation_version = int(_implementation_version_str)
......
#! /usr/bin/python
#
# Protocol Buffers - Google's data interchange format
# Copyright 2008 Google Inc. All rights reserved.
# https://developers.google.com/protocol-buffers/
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are
# met:
#
# * Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# * Redistributions in binary form must reproduce the above
# copyright notice, this list of conditions and the following disclaimer
# in the documentation and/or other materials provided with the
# distribution.
# * Neither the name of Google Inc. nor the names of its
# contributors may be used to endorse or promote products derived from
# this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
"""Test that the api_implementation defaults are what we expect."""
import os
import sys
# Clear environment implementation settings before the google3 imports.
os.environ.pop('PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION', None)
os.environ.pop('PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION_VERSION', None)
# pylint: disable=g-import-not-at-top
from google.apputils import basetest
from google.protobuf.internal import api_implementation
class ApiImplementationDefaultTest(basetest.TestCase):
if sys.version_info.major <= 2:
def testThatPythonIsTheDefault(self):
"""If -DPYTHON_PROTO_*IMPL* was given at build time, this may fail."""
self.assertEqual('python', api_implementation.Type())
else:
def testThatCppApiV2IsTheDefault(self):
"""If -DPYTHON_PROTO_*IMPL* was given at build time, this may fail."""
self.assertEqual('cpp', api_implementation.Type())
self.assertEqual(2, api_implementation.Version())
if __name__ == '__main__':
basetest.main()
...@@ -41,7 +41,6 @@ are: ...@@ -41,7 +41,6 @@ are:
__author__ = 'petar@google.com (Petar Petrov)' __author__ = 'petar@google.com (Petar Petrov)'
class BaseContainer(object): class BaseContainer(object):
"""Base container class.""" """Base container class."""
...@@ -119,15 +118,23 @@ class RepeatedScalarFieldContainer(BaseContainer): ...@@ -119,15 +118,23 @@ class RepeatedScalarFieldContainer(BaseContainer):
self._message_listener.Modified() self._message_listener.Modified()
def extend(self, elem_seq): def extend(self, elem_seq):
"""Extends by appending the given sequence. Similar to list.extend().""" """Extends by appending the given iterable. Similar to list.extend()."""
if not elem_seq:
return
new_values = [] if elem_seq is None:
for elem in elem_seq: return
new_values.append(self._type_checker.CheckValue(elem)) try:
self._values.extend(new_values) elem_seq_iter = iter(elem_seq)
self._message_listener.Modified() except TypeError:
if not elem_seq:
# silently ignore falsy inputs :-/.
# TODO(ptucker): Deprecate this behavior. b/18413862
return
raise
new_values = [self._type_checker.CheckValue(elem) for elem in elem_seq_iter]
if new_values:
self._values.extend(new_values)
self._message_listener.Modified()
def MergeFrom(self, other): def MergeFrom(self, other):
"""Appends the contents of another repeated field of the same type to this """Appends the contents of another repeated field of the same type to this
...@@ -141,6 +148,12 @@ class RepeatedScalarFieldContainer(BaseContainer): ...@@ -141,6 +148,12 @@ class RepeatedScalarFieldContainer(BaseContainer):
self._values.remove(elem) self._values.remove(elem)
self._message_listener.Modified() self._message_listener.Modified()
def pop(self, key=-1):
"""Removes and returns an item at a given index. Similar to list.pop()."""
value = self._values[key]
self.__delitem__(key)
return value
def __setitem__(self, key, value): def __setitem__(self, key, value):
"""Sets the item on the specified position.""" """Sets the item on the specified position."""
if isinstance(key, slice): # PY3 if isinstance(key, slice): # PY3
...@@ -245,6 +258,12 @@ class RepeatedCompositeFieldContainer(BaseContainer): ...@@ -245,6 +258,12 @@ class RepeatedCompositeFieldContainer(BaseContainer):
self._values.remove(elem) self._values.remove(elem)
self._message_listener.Modified() self._message_listener.Modified()
def pop(self, key=-1):
"""Removes and returns an item at a given index. Similar to list.pop()."""
value = self._values[key]
self.__delitem__(key)
return value
def __getslice__(self, start, stop): def __getslice__(self, start, stop):
"""Retrieves the subset of items from between the specified indices.""" """Retrieves the subset of items from between the specified indices."""
return self._values[start:stop] return self._values[start:stop]
......
...@@ -621,9 +621,6 @@ def MessageDecoder(field_number, is_repeated, is_packed, key, new_default): ...@@ -621,9 +621,6 @@ def MessageDecoder(field_number, is_repeated, is_packed, key, new_default):
if value is None: if value is None:
value = field_dict.setdefault(key, new_default(message)) value = field_dict.setdefault(key, new_default(message))
while 1: while 1:
value = field_dict.get(key)
if value is None:
value = field_dict.setdefault(key, new_default(message))
# Read length. # Read length.
(size, pos) = local_DecodeVarint(buffer, pos) (size, pos) = local_DecodeVarint(buffer, pos)
new_pos = pos + size new_pos = pos + size
......
...@@ -34,12 +34,16 @@ ...@@ -34,12 +34,16 @@
__author__ = 'robinson@google.com (Will Robinson)' __author__ = 'robinson@google.com (Will Robinson)'
import sys
from google.apputils import basetest from google.apputils import basetest
from google.protobuf import unittest_custom_options_pb2 from google.protobuf import unittest_custom_options_pb2
from google.protobuf import unittest_import_pb2 from google.protobuf import unittest_import_pb2
from google.protobuf import unittest_pb2 from google.protobuf import unittest_pb2
from google.protobuf import descriptor_pb2 from google.protobuf import descriptor_pb2
from google.protobuf.internal import api_implementation
from google.protobuf import descriptor from google.protobuf import descriptor
from google.protobuf import symbol_database
from google.protobuf import text_format from google.protobuf import text_format
...@@ -51,41 +55,28 @@ name: 'TestEmptyMessage' ...@@ -51,41 +55,28 @@ name: 'TestEmptyMessage'
class DescriptorTest(basetest.TestCase): class DescriptorTest(basetest.TestCase):
def setUp(self): def setUp(self):
self.my_file = descriptor.FileDescriptor( file_proto = descriptor_pb2.FileDescriptorProto(
name='some/filename/some.proto', name='some/filename/some.proto',
package='protobuf_unittest' package='protobuf_unittest')
) message_proto = file_proto.message_type.add(
self.my_enum = descriptor.EnumDescriptor( name='NestedMessage')
name='ForeignEnum', message_proto.field.add(
full_name='protobuf_unittest.ForeignEnum', name='bb',
filename=None, number=1,
file=self.my_file, type=descriptor_pb2.FieldDescriptorProto.TYPE_INT32,
values=[ label=descriptor_pb2.FieldDescriptorProto.LABEL_OPTIONAL)
descriptor.EnumValueDescriptor(name='FOREIGN_FOO', index=0, number=4), enum_proto = message_proto.enum_type.add(
descriptor.EnumValueDescriptor(name='FOREIGN_BAR', index=1, number=5), name='ForeignEnum')
descriptor.EnumValueDescriptor(name='FOREIGN_BAZ', index=2, number=6), enum_proto.value.add(name='FOREIGN_FOO', number=4)
]) enum_proto.value.add(name='FOREIGN_BAR', number=5)
self.my_message = descriptor.Descriptor( enum_proto.value.add(name='FOREIGN_BAZ', number=6)
name='NestedMessage',
full_name='protobuf_unittest.TestAllTypes.NestedMessage', descriptor_pool = symbol_database.Default().pool
filename=None, descriptor_pool.Add(file_proto)
file=self.my_file, self.my_file = descriptor_pool.FindFileByName(file_proto.name)
containing_type=None, self.my_message = self.my_file.message_types_by_name[message_proto.name]
fields=[ self.my_enum = self.my_message.enum_types_by_name[enum_proto.name]
descriptor.FieldDescriptor(
name='bb',
full_name='protobuf_unittest.TestAllTypes.NestedMessage.bb',
index=0, number=1,
type=5, cpp_type=1, label=1,
has_default_value=False, default_value=0,
message_type=None, enum_type=None, containing_type=None,
is_extension=False, extension_scope=None),
],
nested_types=[],
enum_types=[
self.my_enum,
],
extensions=[])
self.my_method = descriptor.MethodDescriptor( self.my_method = descriptor.MethodDescriptor(
name='Bar', name='Bar',
full_name='protobuf_unittest.TestService.Bar', full_name='protobuf_unittest.TestService.Bar',
...@@ -173,6 +164,11 @@ class DescriptorTest(basetest.TestCase): ...@@ -173,6 +164,11 @@ class DescriptorTest(basetest.TestCase):
self.assertEqual(unittest_custom_options_pb2.METHODOPT1_VAL2, self.assertEqual(unittest_custom_options_pb2.METHODOPT1_VAL2,
method_options.Extensions[method_opt1]) method_options.Extensions[method_opt1])
message_descriptor = (
unittest_custom_options_pb2.DummyMessageContainingEnum.DESCRIPTOR)
self.assertTrue(file_descriptor.has_options)
self.assertFalse(message_descriptor.has_options)
def testDifferentCustomOptionTypes(self): def testDifferentCustomOptionTypes(self):
kint32min = -2**31 kint32min = -2**31
kint64min = -2**63 kint64min = -2**63
...@@ -394,6 +390,108 @@ class DescriptorTest(basetest.TestCase): ...@@ -394,6 +390,108 @@ class DescriptorTest(basetest.TestCase):
self.assertEqual(self.my_file.name, 'some/filename/some.proto') self.assertEqual(self.my_file.name, 'some/filename/some.proto')
self.assertEqual(self.my_file.package, 'protobuf_unittest') self.assertEqual(self.my_file.package, 'protobuf_unittest')
@basetest.unittest.skipIf(
api_implementation.Type() != 'cpp' or api_implementation.Version() != 2,
'Immutability of descriptors is only enforced in v2 implementation')
def testImmutableCppDescriptor(self):
message_descriptor = unittest_pb2.TestAllTypes.DESCRIPTOR
with self.assertRaises(AttributeError):
message_descriptor.fields_by_name = None
with self.assertRaises(TypeError):
message_descriptor.fields_by_name['Another'] = None
with self.assertRaises(TypeError):
message_descriptor.fields.append(None)
class GeneratedDescriptorTest(basetest.TestCase):
"""Tests for the properties of descriptors in generated code."""
def CheckMessageDescriptor(self, message_descriptor):
# Basic properties
self.assertEqual(message_descriptor.name, 'TestAllTypes')
self.assertEqual(message_descriptor.full_name,
'protobuf_unittest.TestAllTypes')
# Test equality and hashability
self.assertEqual(message_descriptor, message_descriptor)
self.assertEqual(message_descriptor.fields[0].containing_type,
message_descriptor)
self.assertIn(message_descriptor, [message_descriptor])
self.assertIn(message_descriptor, {message_descriptor: None})
# Test field containers
self.CheckDescriptorSequence(message_descriptor.fields)
self.CheckDescriptorMapping(message_descriptor.fields_by_name)
self.CheckDescriptorMapping(message_descriptor.fields_by_number)
def CheckFieldDescriptor(self, field_descriptor):
# Basic properties
self.assertEqual(field_descriptor.name, 'optional_int32')
self.assertEqual(field_descriptor.full_name,
'protobuf_unittest.TestAllTypes.optional_int32')
self.assertEqual(field_descriptor.containing_type.name, 'TestAllTypes')
# Test equality and hashability
self.assertEqual(field_descriptor, field_descriptor)
self.assertEqual(
field_descriptor.containing_type.fields_by_name['optional_int32'],
field_descriptor)
self.assertIn(field_descriptor, [field_descriptor])
self.assertIn(field_descriptor, {field_descriptor: None})
def CheckDescriptorSequence(self, sequence):
# Verifies that a property like 'messageDescriptor.fields' has all the
# properties of an immutable abc.Sequence.
self.assertGreater(len(sequence), 0) # Sized
self.assertEqual(len(sequence), len(list(sequence))) # Iterable
item = sequence[0]
self.assertEqual(item, sequence[0])
self.assertIn(item, sequence) # Container
self.assertEqual(sequence.index(item), 0)
self.assertEqual(sequence.count(item), 1)
reversed_iterator = reversed(sequence)
self.assertEqual(list(reversed_iterator), list(sequence)[::-1])
self.assertRaises(StopIteration, next, reversed_iterator)
def CheckDescriptorMapping(self, mapping):
# Verifies that a property like 'messageDescriptor.fields' has all the
# properties of an immutable abc.Mapping.
self.assertGreater(len(mapping), 0) # Sized
self.assertEqual(len(mapping), len(list(mapping))) # Iterable
if sys.version_info.major >= 3:
key, item = next(iter(mapping.items()))
else:
key, item = mapping.items()[0]
self.assertIn(key, mapping) # Container
self.assertEqual(mapping.get(key), item)
# keys(), iterkeys() &co
item = (next(iter(mapping.keys())), next(iter(mapping.values())))
self.assertEqual(item, next(iter(mapping.items())))
if sys.version_info.major < 3:
def CheckItems(seq, iterator):
self.assertEqual(next(iterator), seq[0])
self.assertEqual(list(iterator), seq[1:])
CheckItems(mapping.keys(), mapping.iterkeys())
CheckItems(mapping.values(), mapping.itervalues())
CheckItems(mapping.items(), mapping.iteritems())
def testDescriptor(self):
message_descriptor = unittest_pb2.TestAllTypes.DESCRIPTOR
self.CheckMessageDescriptor(message_descriptor)
field_descriptor = message_descriptor.fields_by_name['optional_int32']
self.CheckFieldDescriptor(field_descriptor)
def testCppDescriptorContainer(self):
# Check that the collection is still valid even if the parent disappeared.
enum = unittest_pb2.TestAllTypes.DESCRIPTOR.enum_types_by_name['NestedEnum']
values = enum.values
del enum
self.assertEqual('FOO', values[0].name)
def testCppDescriptorContainer_Iterator(self):
# Same test with the iterator
enum = unittest_pb2.TestAllTypes.DESCRIPTOR.enum_types_by_name['NestedEnum']
values_iter = iter(enum.values)
del enum
self.assertEqual('FOO', next(values_iter).name)
class DescriptorCopyToProtoTest(basetest.TestCase): class DescriptorCopyToProtoTest(basetest.TestCase):
"""Tests for CopyTo functions of Descriptor.""" """Tests for CopyTo functions of Descriptor."""
...@@ -588,10 +686,12 @@ class DescriptorCopyToProtoTest(basetest.TestCase): ...@@ -588,10 +686,12 @@ class DescriptorCopyToProtoTest(basetest.TestCase):
output_type: '.protobuf_unittest.BarResponse' output_type: '.protobuf_unittest.BarResponse'
> >
""" """
self._InternalTestCopyToProto( # TODO(rocking): enable this test after the proto descriptor change is
unittest_pb2.TestService.DESCRIPTOR, # checked in.
descriptor_pb2.ServiceDescriptorProto, #self._InternalTestCopyToProto(
TEST_SERVICE_ASCII) # unittest_pb2.TestService.DESCRIPTOR,
# descriptor_pb2.ServiceDescriptorProto,
# TEST_SERVICE_ASCII)
class MakeDescriptorTest(basetest.TestCase): class MakeDescriptorTest(basetest.TestCase):
......
...@@ -219,12 +219,20 @@ def _AttachFieldHelpers(cls, field_descriptor): ...@@ -219,12 +219,20 @@ def _AttachFieldHelpers(cls, field_descriptor):
def AddDecoder(wiretype, is_packed): def AddDecoder(wiretype, is_packed):
tag_bytes = encoder.TagBytes(field_descriptor.number, wiretype) tag_bytes = encoder.TagBytes(field_descriptor.number, wiretype)
cls._decoders_by_tag[tag_bytes] = ( decode_type = field_descriptor.type
type_checkers.TYPE_TO_DECODER[field_descriptor.type]( if (decode_type == _FieldDescriptor.TYPE_ENUM and
field_descriptor.number, is_repeated, is_packed, type_checkers.SupportsOpenEnums(field_descriptor)):
field_descriptor, field_descriptor._default_constructor), decode_type = _FieldDescriptor.TYPE_INT32
field_descriptor if field_descriptor.containing_oneof is not None
else None) oneof_descriptor = None
if field_descriptor.containing_oneof is not None:
oneof_descriptor = field_descriptor
field_decoder = type_checkers.TYPE_TO_DECODER[decode_type](
field_descriptor.number, is_repeated, is_packed,
field_descriptor, field_descriptor._default_constructor)
cls._decoders_by_tag[tag_bytes] = (field_decoder, oneof_descriptor)
AddDecoder(type_checkers.FIELD_TYPE_TO_WIRE_TYPE[field_descriptor.type], AddDecoder(type_checkers.FIELD_TYPE_TO_WIRE_TYPE[field_descriptor.type],
False) False)
...@@ -296,6 +304,8 @@ def _DefaultValueConstructorForField(field): ...@@ -296,6 +304,8 @@ def _DefaultValueConstructorForField(field):
def MakeSubMessageDefault(message): def MakeSubMessageDefault(message):
result = message_type._concrete_class() result = message_type._concrete_class()
result._SetListener(message._listener_for_children) result._SetListener(message._listener_for_children)
if field.containing_oneof:
message._UpdateOneofState(field)
return result return result
return MakeSubMessageDefault return MakeSubMessageDefault
...@@ -476,6 +486,7 @@ def _AddPropertiesForNonRepeatedScalarField(field, cls): ...@@ -476,6 +486,7 @@ def _AddPropertiesForNonRepeatedScalarField(field, cls):
type_checker = type_checkers.GetTypeChecker(field) type_checker = type_checkers.GetTypeChecker(field)
default_value = field.default_value default_value = field.default_value
valid_values = set() valid_values = set()
is_proto3 = field.containing_type.syntax == "proto3"
def getter(self): def getter(self):
# TODO(protobuf-team): This may be broken since there may not be # TODO(protobuf-team): This may be broken since there may not be
...@@ -483,15 +494,24 @@ def _AddPropertiesForNonRepeatedScalarField(field, cls): ...@@ -483,15 +494,24 @@ def _AddPropertiesForNonRepeatedScalarField(field, cls):
return self._fields.get(field, default_value) return self._fields.get(field, default_value)
getter.__module__ = None getter.__module__ = None
getter.__doc__ = 'Getter for %s.' % proto_field_name getter.__doc__ = 'Getter for %s.' % proto_field_name
clear_when_set_to_default = is_proto3 and not field.containing_oneof
def field_setter(self, new_value): def field_setter(self, new_value):
# pylint: disable=protected-access # pylint: disable=protected-access
self._fields[field] = type_checker.CheckValue(new_value) # Testing the value for truthiness captures all of the proto3 defaults
# (0, 0.0, enum 0, and False).
new_value = type_checker.CheckValue(new_value)
if clear_when_set_to_default and not new_value:
self._fields.pop(field, None)
else:
self._fields[field] = new_value
# Check _cached_byte_size_dirty inline to improve performance, since scalar # Check _cached_byte_size_dirty inline to improve performance, since scalar
# setters are called frequently. # setters are called frequently.
if not self._cached_byte_size_dirty: if not self._cached_byte_size_dirty:
self._Modified() self._Modified()
if field.containing_oneof is not None: if field.containing_oneof:
def setter(self, new_value): def setter(self, new_value):
field_setter(self, new_value) field_setter(self, new_value)
self._UpdateOneofState(field) self._UpdateOneofState(field)
...@@ -624,24 +644,35 @@ def _AddListFieldsMethod(message_descriptor, cls): ...@@ -624,24 +644,35 @@ def _AddListFieldsMethod(message_descriptor, cls):
cls.ListFields = ListFields cls.ListFields = ListFields
_Proto3HasError = 'Protocol message has no non-repeated submessage field "%s"'
_Proto2HasError = 'Protocol message has no non-repeated field "%s"'
def _AddHasFieldMethod(message_descriptor, cls): def _AddHasFieldMethod(message_descriptor, cls):
"""Helper for _AddMessageMethods().""" """Helper for _AddMessageMethods()."""
singular_fields = {} is_proto3 = (message_descriptor.syntax == "proto3")
error_msg = _Proto3HasError if is_proto3 else _Proto2HasError
hassable_fields = {}
for field in message_descriptor.fields: for field in message_descriptor.fields:
if field.label != _FieldDescriptor.LABEL_REPEATED: if field.label == _FieldDescriptor.LABEL_REPEATED:
singular_fields[field.name] = field continue
# Fields inside oneofs are never repeated (enforced by the compiler). # For proto3, only submessages and fields inside a oneof have presence.
for field in message_descriptor.oneofs: if (is_proto3 and field.cpp_type != _FieldDescriptor.CPPTYPE_MESSAGE and
singular_fields[field.name] = field not field.containing_oneof):
continue
hassable_fields[field.name] = field
if not is_proto3:
# Fields inside oneofs are never repeated (enforced by the compiler).
for oneof in message_descriptor.oneofs:
hassable_fields[oneof.name] = oneof
def HasField(self, field_name): def HasField(self, field_name):
try: try:
field = singular_fields[field_name] field = hassable_fields[field_name]
except KeyError: except KeyError:
raise ValueError( raise ValueError(error_msg % field_name)
'Protocol message has no singular "%s" field.' % field_name)
if isinstance(field, descriptor_mod.OneofDescriptor): if isinstance(field, descriptor_mod.OneofDescriptor):
try: try:
...@@ -871,6 +902,7 @@ def _AddMergeFromStringMethod(message_descriptor, cls): ...@@ -871,6 +902,7 @@ def _AddMergeFromStringMethod(message_descriptor, cls):
local_ReadTag = decoder.ReadTag local_ReadTag = decoder.ReadTag
local_SkipField = decoder.SkipField local_SkipField = decoder.SkipField
decoders_by_tag = cls._decoders_by_tag decoders_by_tag = cls._decoders_by_tag
is_proto3 = message_descriptor.syntax == "proto3"
def InternalParse(self, buffer, pos, end): def InternalParse(self, buffer, pos, end):
self._Modified() self._Modified()
...@@ -884,9 +916,11 @@ def _AddMergeFromStringMethod(message_descriptor, cls): ...@@ -884,9 +916,11 @@ def _AddMergeFromStringMethod(message_descriptor, cls):
new_pos = local_SkipField(buffer, new_pos, end, tag_bytes) new_pos = local_SkipField(buffer, new_pos, end, tag_bytes)
if new_pos == -1: if new_pos == -1:
return pos return pos
if not unknown_field_list: if not is_proto3:
unknown_field_list = self._unknown_fields = [] if not unknown_field_list:
unknown_field_list.append((tag_bytes, buffer[value_start_pos:new_pos])) unknown_field_list = self._unknown_fields = []
unknown_field_list.append(
(tag_bytes, buffer[value_start_pos:new_pos]))
pos = new_pos pos = new_pos
else: else:
pos = field_decoder(buffer, new_pos, end, self, field_dict) pos = field_decoder(buffer, new_pos, end, self, field_dict)
...@@ -1008,6 +1042,8 @@ def _AddMergeFromMethod(cls): ...@@ -1008,6 +1042,8 @@ def _AddMergeFromMethod(cls):
# Construct a new object to represent this field. # Construct a new object to represent this field.
field_value = field._default_constructor(self) field_value = field._default_constructor(self)
fields[field] = field_value fields[field] = field_value
if field.containing_oneof:
self._UpdateOneofState(field)
field_value.MergeFrom(value) field_value.MergeFrom(value)
else: else:
self._fields[field] = value self._fields[field] = value
...@@ -1252,11 +1288,10 @@ class _ExtensionDict(object): ...@@ -1252,11 +1288,10 @@ class _ExtensionDict(object):
# It's slightly wasteful to lookup the type checker each time, # It's slightly wasteful to lookup the type checker each time,
# but we expect this to be a vanishingly uncommon case anyway. # but we expect this to be a vanishingly uncommon case anyway.
type_checker = type_checkers.GetTypeChecker( type_checker = type_checkers.GetTypeChecker(extension_handle)
extension_handle)
# pylint: disable=protected-access # pylint: disable=protected-access
self._extended_message._fields[extension_handle] = ( self._extended_message._fields[extension_handle] = (
type_checker.CheckValue(value)) type_checker.CheckValue(value))
self._extended_message._Modified() self._extended_message._Modified()
def _FindExtensionByName(self, name): def _FindExtensionByName(self, name):
......
...@@ -1792,6 +1792,17 @@ class ReflectionTest(basetest.TestCase): ...@@ -1792,6 +1792,17 @@ class ReflectionTest(basetest.TestCase):
# Just check the default value. # Just check the default value.
self.assertEqual(57, msg.inner.value) self.assertEqual(57, msg.inner.value)
@basetest.unittest.skipIf(
api_implementation.Type() != 'cpp' or api_implementation.Version() != 2,
'CPPv2-specific test')
def testBadArguments(self):
# Some of these assertions used to segfault.
from google.protobuf.pyext import _message
self.assertRaises(TypeError, _message.Message._GetFieldDescriptor, 3)
self.assertRaises(TypeError, _message.Message._GetExtensionDescriptor, 42)
self.assertRaises(TypeError,
unittest_pb2.TestAllTypes().__getattribute__, 42)
# Since we had so many tests for protocol buffer equality, we broke these out # Since we had so many tests for protocol buffer equality, we broke these out
# into separate TestCase classes. # into separate TestCase classes.
......
...@@ -59,6 +59,8 @@ from google.protobuf import descriptor ...@@ -59,6 +59,8 @@ from google.protobuf import descriptor
_FieldDescriptor = descriptor.FieldDescriptor _FieldDescriptor = descriptor.FieldDescriptor
def SupportsOpenEnums(field_descriptor):
return field_descriptor.containing_type.syntax == "proto3"
def GetTypeChecker(field): def GetTypeChecker(field):
"""Returns a type checker for a message field of the specified types. """Returns a type checker for a message field of the specified types.
...@@ -74,7 +76,11 @@ def GetTypeChecker(field): ...@@ -74,7 +76,11 @@ def GetTypeChecker(field):
field.type == _FieldDescriptor.TYPE_STRING): field.type == _FieldDescriptor.TYPE_STRING):
return UnicodeValueChecker() return UnicodeValueChecker()
if field.cpp_type == _FieldDescriptor.CPPTYPE_ENUM: if field.cpp_type == _FieldDescriptor.CPPTYPE_ENUM:
return EnumValueChecker(field.enum_type) if SupportsOpenEnums(field):
# When open enums are supported, any int32 can be assigned.
return _VALUE_CHECKERS[_FieldDescriptor.CPPTYPE_INT32]
else:
return EnumValueChecker(field.enum_type)
return _VALUE_CHECKERS[field.cpp_type] return _VALUE_CHECKERS[field.cpp_type]
......
...@@ -38,6 +38,7 @@ __author__ = 'bohdank@google.com (Bohdan Koval)' ...@@ -38,6 +38,7 @@ __author__ = 'bohdank@google.com (Bohdan Koval)'
from google.apputils import basetest from google.apputils import basetest
from google.protobuf import unittest_mset_pb2 from google.protobuf import unittest_mset_pb2
from google.protobuf import unittest_pb2 from google.protobuf import unittest_pb2
from google.protobuf import unittest_proto3_arena_pb2
from google.protobuf.internal import api_implementation from google.protobuf.internal import api_implementation
from google.protobuf.internal import encoder from google.protobuf.internal import encoder
from google.protobuf.internal import missing_enum_values_pb2 from google.protobuf.internal import missing_enum_values_pb2
...@@ -45,10 +46,81 @@ from google.protobuf.internal import test_util ...@@ -45,10 +46,81 @@ from google.protobuf.internal import test_util
from google.protobuf.internal import type_checkers from google.protobuf.internal import type_checkers
class UnknownFieldsTest(basetest.TestCase):
def setUp(self):
self.descriptor = unittest_pb2.TestAllTypes.DESCRIPTOR
self.all_fields = unittest_pb2.TestAllTypes()
test_util.SetAllFields(self.all_fields)
self.all_fields_data = self.all_fields.SerializeToString()
self.empty_message = unittest_pb2.TestEmptyMessage()
self.empty_message.ParseFromString(self.all_fields_data)
def testSerialize(self):
data = self.empty_message.SerializeToString()
# Don't use assertEqual because we don't want to dump raw binary data to
# stdout.
self.assertTrue(data == self.all_fields_data)
def testSerializeProto3(self):
# Verify that proto3 doesn't preserve unknown fields.
message = unittest_proto3_arena_pb2.TestEmptyMessage()
message.ParseFromString(self.all_fields_data)
self.assertEqual(0, len(message.SerializeToString()))
def testByteSize(self):
self.assertEqual(self.all_fields.ByteSize(), self.empty_message.ByteSize())
def testListFields(self):
# Make sure ListFields doesn't return unknown fields.
self.assertEqual(0, len(self.empty_message.ListFields()))
def testSerializeMessageSetWireFormatUnknownExtension(self):
# Create a message using the message set wire format with an unknown
# message.
raw = unittest_mset_pb2.RawMessageSet()
# Add an unknown extension.
item = raw.item.add()
item.type_id = 1545009
message1 = unittest_mset_pb2.TestMessageSetExtension1()
message1.i = 12345
item.message = message1.SerializeToString()
serialized = raw.SerializeToString()
# Parse message using the message set wire format.
proto = unittest_mset_pb2.TestMessageSet()
proto.MergeFromString(serialized)
# Verify that the unknown extension is serialized unchanged
reserialized = proto.SerializeToString()
new_raw = unittest_mset_pb2.RawMessageSet()
new_raw.MergeFromString(reserialized)
self.assertEqual(raw, new_raw)
# C++ implementation for proto2 does not currently take into account unknown
# fields when checking equality.
#
# TODO(haberman): fix this.
@basetest.unittest.skipIf(
api_implementation.Type() == 'cpp' and api_implementation.Version() == 2,
'C++ implementation does not expose unknown fields to Python')
def testEquals(self):
message = unittest_pb2.TestEmptyMessage()
message.ParseFromString(self.all_fields_data)
self.assertEqual(self.empty_message, message)
self.all_fields.ClearField('optional_string')
message.ParseFromString(self.all_fields.SerializeToString())
self.assertNotEqual(self.empty_message, message)
@basetest.unittest.skipIf( @basetest.unittest.skipIf(
api_implementation.Type() == 'cpp' and api_implementation.Version() == 2, api_implementation.Type() == 'cpp' and api_implementation.Version() == 2,
'C++ implementation does not expose unknown fields to Python') 'C++ implementation does not expose unknown fields to Python')
class UnknownFieldsTest(basetest.TestCase): class UnknownFieldsAccessorsTest(basetest.TestCase):
def setUp(self): def setUp(self):
self.descriptor = unittest_pb2.TestAllTypes.DESCRIPTOR self.descriptor = unittest_pb2.TestAllTypes.DESCRIPTOR
...@@ -98,13 +170,6 @@ class UnknownFieldsTest(basetest.TestCase): ...@@ -98,13 +170,6 @@ class UnknownFieldsTest(basetest.TestCase):
value = self.GetField('optionalgroup') value = self.GetField('optionalgroup')
self.assertEqual(self.all_fields.optionalgroup, value) self.assertEqual(self.all_fields.optionalgroup, value)
def testSerialize(self):
data = self.empty_message.SerializeToString()
# Don't use assertEqual because we don't want to dump raw binary data to
# stdout.
self.assertTrue(data == self.all_fields_data)
def testCopyFrom(self): def testCopyFrom(self):
message = unittest_pb2.TestEmptyMessage() message = unittest_pb2.TestEmptyMessage()
message.CopyFrom(self.empty_message) message.CopyFrom(self.empty_message)
...@@ -132,51 +197,12 @@ class UnknownFieldsTest(basetest.TestCase): ...@@ -132,51 +197,12 @@ class UnknownFieldsTest(basetest.TestCase):
self.empty_message.Clear() self.empty_message.Clear()
self.assertEqual(0, len(self.empty_message._unknown_fields)) self.assertEqual(0, len(self.empty_message._unknown_fields))
def testByteSize(self):
self.assertEqual(self.all_fields.ByteSize(), self.empty_message.ByteSize())
def testUnknownExtensions(self): def testUnknownExtensions(self):
message = unittest_pb2.TestEmptyMessageWithExtensions() message = unittest_pb2.TestEmptyMessageWithExtensions()
message.ParseFromString(self.all_fields_data) message.ParseFromString(self.all_fields_data)
self.assertEqual(self.empty_message._unknown_fields, self.assertEqual(self.empty_message._unknown_fields,
message._unknown_fields) message._unknown_fields)
def testListFields(self):
# Make sure ListFields doesn't return unknown fields.
self.assertEqual(0, len(self.empty_message.ListFields()))
def testSerializeMessageSetWireFormatUnknownExtension(self):
# Create a message using the message set wire format with an unknown
# message.
raw = unittest_mset_pb2.RawMessageSet()
# Add an unknown extension.
item = raw.item.add()
item.type_id = 1545009
message1 = unittest_mset_pb2.TestMessageSetExtension1()
message1.i = 12345
item.message = message1.SerializeToString()
serialized = raw.SerializeToString()
# Parse message using the message set wire format.
proto = unittest_mset_pb2.TestMessageSet()
proto.MergeFromString(serialized)
# Verify that the unknown extension is serialized unchanged
reserialized = proto.SerializeToString()
new_raw = unittest_mset_pb2.RawMessageSet()
new_raw.MergeFromString(reserialized)
self.assertEqual(raw, new_raw)
def testEquals(self):
message = unittest_pb2.TestEmptyMessage()
message.ParseFromString(self.all_fields_data)
self.assertEqual(self.empty_message, message)
self.all_fields.ClearField('optional_string')
message.ParseFromString(self.all_fields.SerializeToString())
self.assertNotEqual(self.empty_message, message)
@basetest.unittest.skipIf( @basetest.unittest.skipIf(
......
...@@ -36,7 +36,6 @@ ...@@ -36,7 +36,6 @@
__author__ = 'robinson@google.com (Will Robinson)' __author__ = 'robinson@google.com (Will Robinson)'
class Error(Exception): pass class Error(Exception): pass
class DecodeError(Error): pass class DecodeError(Error): pass
class EncodeError(Error): pass class EncodeError(Error): pass
......
This diff is collapsed.
...@@ -34,105 +34,55 @@ ...@@ -34,105 +34,55 @@
#define GOOGLE_PROTOBUF_PYTHON_CPP_DESCRIPTOR_H__ #define GOOGLE_PROTOBUF_PYTHON_CPP_DESCRIPTOR_H__
#include <Python.h> #include <Python.h>
#include <structmember.h>
#include <google/protobuf/stubs/hash.h>
#include <google/protobuf/descriptor.h> #include <google/protobuf/descriptor.h>
#if PY_VERSION_HEX < 0x02050000 && !defined(PY_SSIZE_T_MIN)
typedef int Py_ssize_t;
#define PY_SSIZE_T_MAX INT_MAX
#define PY_SSIZE_T_MIN INT_MIN
#endif
namespace google { namespace google {
namespace protobuf { namespace protobuf {
namespace python { namespace python {
typedef struct CMessageDescriptor { extern PyTypeObject PyMessageDescriptor_Type;
PyObject_HEAD extern PyTypeObject PyFieldDescriptor_Type;
extern PyTypeObject PyEnumDescriptor_Type;
// The proto2 descriptor that this object represents. extern PyTypeObject PyEnumValueDescriptor_Type;
const google::protobuf::Descriptor* descriptor; extern PyTypeObject PyFileDescriptor_Type;
} CMessageDescriptor; extern PyTypeObject PyOneofDescriptor_Type;
// Return a new reference to a Descriptor object.
typedef struct CFieldDescriptor { // The C++ pointer is usually borrowed from the global DescriptorPool.
PyObject_HEAD // In any case, it must stay alive as long as the Python object.
PyObject* PyMessageDescriptor_New(const Descriptor* descriptor);
// The proto2 descriptor that this object represents. PyObject* PyFieldDescriptor_New(const FieldDescriptor* descriptor);
const google::protobuf::FieldDescriptor* descriptor; PyObject* PyEnumDescriptor_New(const EnumDescriptor* descriptor);
} CFieldDescriptor; PyObject* PyEnumValueDescriptor_New(const EnumValueDescriptor* descriptor);
PyObject* PyOneofDescriptor_New(const OneofDescriptor* descriptor);
PyObject* PyFileDescriptor_New(const FileDescriptor* file_descriptor);
// Wraps operations to the global DescriptorPool which contains information
// about all messages and fields. // Alternate constructor of PyFileDescriptor, used when we already have a
// // serialized FileDescriptorProto that can be cached.
// There is normally one pool per process. We make it a Python object only // Returns a new reference.
// because it contains many Python references. PyObject* PyFileDescriptor_NewWithPb(const FileDescriptor* file_descriptor,
// TODO(amauryfa): See whether such objects can appear in reference cycles, and PyObject* serialized_pb);
// consider adding support for the cyclic GC.
//
// "Methods" that interacts with this DescriptorPool are in the cdescriptor_pool
// namespace.
typedef struct PyDescriptorPool {
PyObject_HEAD
google::protobuf::DescriptorPool* pool;
// Make our own mapping to retrieve Python classes from C++ descriptors.
//
// Descriptor pointers stored here are owned by the DescriptorPool above.
// Python references to classes are owned by this PyDescriptorPool.
typedef hash_map<const Descriptor *, PyObject *> ClassesByMessageMap;
ClassesByMessageMap *classes_by_descriptor;
} PyDescriptorPool;
extern PyTypeObject CMessageDescriptor_Type;
extern PyTypeObject CFieldDescriptor_Type;
extern PyTypeObject PyDescriptorPool_Type;
namespace cdescriptor_pool {
// Builds a new DescriptorPool. Normally called only once per process.
PyDescriptorPool* NewDescriptorPool();
// Looks up a message by name.
// Returns a message Descriptor, or NULL if not found.
const google::protobuf::Descriptor* FindMessageTypeByName(PyDescriptorPool* self,
const string& name);
// Registers a new Python class for the given message descriptor.
// Returns the message Descriptor.
// On error, returns NULL with a Python exception set.
const google::protobuf::Descriptor* RegisterMessageClass(
PyDescriptorPool* self, PyObject *message_class, PyObject *descriptor);
// Retrieves the Python class registered with the given message descriptor. // Return the C++ descriptor pointer.
// // This function checks the parameter type; on error, return NULL with a Python
// Returns a *borrowed* reference if found, otherwise returns NULL with an
// exception set. // exception set.
PyObject *GetMessageClass(PyDescriptorPool* self, const Descriptor* PyMessageDescriptor_AsDescriptor(PyObject* obj);
const Descriptor *message_descriptor); const FieldDescriptor* PyFieldDescriptor_AsDescriptor(PyObject* obj);
// Looks up a field by name. Returns a CDescriptor corresponding to // Returns the raw C++ pointer.
// the field on success, or NULL on failure. const void* PyDescriptor_AsVoidPtr(PyObject* obj);
//
// Returns a new reference.
PyObject* FindFieldByName(PyDescriptorPool* self, PyObject* name);
// Looks up an extension by name. Returns a CDescriptor corresponding // Check that the calling Python code is the global scope of a _pb2.py module.
// to the field on success, or NULL on failure. // This function is used to support the current code generated by the proto
// compiler, which insists on modifying descriptors after they have been
// created.
// //
// Returns a new reference. // stacklevel indicates which Python frame should be the _pb2.py module.
PyObject* FindExtensionByName(PyDescriptorPool* self, PyObject* arg); //
} // namespace cdescriptor_pool // Don't use this function outside descriptor classes.
bool _CalledFromGeneratedFile(int stacklevel);
PyObject* Python_BuildFile(PyObject* ignored, PyObject* args);
bool InitDescriptor(); bool InitDescriptor();
} // namespace python } // namespace python
......
This diff is collapsed.
// Protocol Buffers - Google's data interchange format
// Copyright 2008 Google Inc. All rights reserved.
// https://developers.google.com/protocol-buffers/
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
//
// * Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above
// copyright notice, this list of conditions and the following disclaimer
// in the documentation and/or other materials provided with the
// distribution.
// * Neither the name of Google Inc. nor the names of its
// contributors may be used to endorse or promote products derived from
// this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
// Mappings and Sequences of descriptors.
// They implement containers like fields_by_name, EnumDescriptor.values...
// See descriptor_containers.cc for more description.
#include <Python.h>
namespace google {
namespace protobuf {
class Descriptor;
class FileDescriptor;
class EnumDescriptor;
class OneofDescriptor;
namespace python {
// Initialize the various types and objects.
bool InitDescriptorMappingTypes();
// Each function below returns a Mapping, or a Sequence of descriptors.
// They all return a new reference.
namespace message_descriptor {
PyObject* NewMessageFieldsByName(const Descriptor* descriptor);
PyObject* NewMessageFieldsByNumber(const Descriptor* descriptor);
PyObject* NewMessageFieldsSeq(const Descriptor* descriptor);
PyObject* NewMessageNestedTypesSeq(const Descriptor* descriptor);
PyObject* NewMessageNestedTypesByName(const Descriptor* descriptor);
PyObject* NewMessageEnumsByName(const Descriptor* descriptor);
PyObject* NewMessageEnumsSeq(const Descriptor* descriptor);
PyObject* NewMessageEnumValuesByName(const Descriptor* descriptor);
PyObject* NewMessageExtensionsByName(const Descriptor* descriptor);
PyObject* NewMessageExtensionsSeq(const Descriptor* descriptor);
PyObject* NewMessageOneofsByName(const Descriptor* descriptor);
PyObject* NewMessageOneofsSeq(const Descriptor* descriptor);
} // namespace message_descriptor
namespace enum_descriptor {
PyObject* NewEnumValuesByName(const EnumDescriptor* descriptor);
PyObject* NewEnumValuesByNumber(const EnumDescriptor* descriptor);
PyObject* NewEnumValuesSeq(const EnumDescriptor* descriptor);
} // namespace enum_descriptor
namespace oneof_descriptor {
PyObject* NewOneofFieldsSeq(const OneofDescriptor* descriptor);
} // namespace oneof_descriptor
namespace file_descriptor {
PyObject* NewFileMessageTypesByName(const FileDescriptor* descriptor);
PyObject* NewFileEnumTypesByName(const FileDescriptor* descriptor);
PyObject* NewFileExtensionsByName(const FileDescriptor* descriptor);
PyObject* NewFileDependencies(const FileDescriptor* descriptor);
PyObject* NewFilePublicDependencies(const FileDescriptor* descriptor);
} // namespace file_descriptor
} // namespace python
} // namespace protobuf
} // namespace google
#! /usr/bin/python
#
# Protocol Buffers - Google's data interchange format
# Copyright 2008 Google Inc. All rights reserved.
# https://developers.google.com/protocol-buffers/
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are
# met:
#
# * Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# * Redistributions in binary form must reproduce the above
# copyright notice, this list of conditions and the following disclaimer
# in the documentation and/or other materials provided with the
# distribution.
# * Neither the name of Google Inc. nor the names of its
# contributors may be used to endorse or promote products derived from
# this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
"""Tests for google.protobuf.pyext behavior."""
__author__ = 'anuraag@google.com (Anuraag Agrawal)'
import os
os.environ['PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION'] = 'cpp'
os.environ['PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION_VERSION'] = '2'
# We must set the implementation version above before the google3 imports.
# pylint: disable=g-import-not-at-top
from google.apputils import basetest
from google.protobuf.internal import api_implementation
# Run all tests from the original module by putting them in our namespace.
# pylint: disable=wildcard-import
from google.protobuf.internal.descriptor_test import *
class ConfirmCppApi2Test(basetest.TestCase):
def testImplementationSetting(self):
self.assertEqual('cpp', api_implementation.Type())
self.assertEqual(2, api_implementation.Version())
if __name__ == '__main__':
basetest.main()
This diff is collapsed.
// Protocol Buffers - Google's data interchange format
// Copyright 2008 Google Inc. All rights reserved.
// https://developers.google.com/protocol-buffers/
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
//
// * Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above
// copyright notice, this list of conditions and the following disclaimer
// in the documentation and/or other materials provided with the
// distribution.
// * Neither the name of Google Inc. nor the names of its
// contributors may be used to endorse or promote products derived from
// this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#ifndef GOOGLE_PROTOBUF_PYTHON_CPP_DESCRIPTOR_POOL_H__
#define GOOGLE_PROTOBUF_PYTHON_CPP_DESCRIPTOR_POOL_H__
#include <Python.h>
#include <google/protobuf/stubs/hash.h>
#include <google/protobuf/descriptor.h>
namespace google {
namespace protobuf {
namespace python {
// Wraps operations to the global DescriptorPool which contains information
// about all messages and fields.
//
// There is normally one pool per process. We make it a Python object only
// because it contains many Python references.
// TODO(amauryfa): See whether such objects can appear in reference cycles, and
// consider adding support for the cyclic GC.
//
// "Methods" that interacts with this DescriptorPool are in the cdescriptor_pool
// namespace.
typedef struct PyDescriptorPool {
PyObject_HEAD
DescriptorPool* pool;
// Make our own mapping to retrieve Python classes from C++ descriptors.
//
// Descriptor pointers stored here are owned by the DescriptorPool above.
// Python references to classes are owned by this PyDescriptorPool.
typedef hash_map<const Descriptor*, PyObject*> ClassesByMessageMap;
ClassesByMessageMap* classes_by_descriptor;
// Store interned descriptors, so that the same C++ descriptor yields the same
// Python object. Objects are not immortal: this map does not own the
// references, and items are deleted when the last reference to the object is
// released.
// This is enough to support the "is" operator on live objects.
// All descriptors are stored here.
hash_map<const void*, PyObject*>* interned_descriptors;
// Cache the options for any kind of descriptor.
// Descriptor pointers are owned by the DescriptorPool above.
// Python objects are owned by the map.
hash_map<const void*, PyObject*>* descriptor_options;
} PyDescriptorPool;
extern PyTypeObject PyDescriptorPool_Type;
namespace cdescriptor_pool {
// Builds a new DescriptorPool. Normally called only once per process.
PyDescriptorPool* NewDescriptorPool();
// Looks up a message by name.
// Returns a message Descriptor, or NULL if not found.
const Descriptor* FindMessageTypeByName(PyDescriptorPool* self,
const string& name);
// Registers a new Python class for the given message descriptor.
// Returns the message Descriptor.
// On error, returns NULL with a Python exception set.
const Descriptor* RegisterMessageClass(
PyDescriptorPool* self, PyObject* message_class, PyObject* descriptor);
// Retrieves the Python class registered with the given message descriptor.
//
// Returns a *borrowed* reference if found, otherwise returns NULL with an
// exception set.
PyObject* GetMessageClass(PyDescriptorPool* self,
const Descriptor* message_descriptor);
// Looks up a message by name. Returns a PyMessageDescriptor corresponding to
// the field on success, or NULL on failure.
//
// Returns a new reference.
PyObject* FindMessageByName(PyDescriptorPool* self, PyObject* name);
// Looks up a field by name. Returns a PyFieldDescriptor corresponding to
// the field on success, or NULL on failure.
//
// Returns a new reference.
PyObject* FindFieldByName(PyDescriptorPool* self, PyObject* name);
// Looks up an extension by name. Returns a PyFieldDescriptor corresponding
// to the field on success, or NULL on failure.
//
// Returns a new reference.
PyObject* FindExtensionByName(PyDescriptorPool* self, PyObject* arg);
// Looks up an enum type by name. Returns a PyEnumDescriptor corresponding
// to the field on success, or NULL on failure.
//
// Returns a new reference.
PyObject* FindEnumTypeByName(PyDescriptorPool* self, PyObject* arg);
// Looks up a oneof by name. Returns a COneofDescriptor corresponding
// to the oneof on success, or NULL on failure.
//
// Returns a new reference.
PyObject* FindOneofByName(PyDescriptorPool* self, PyObject* arg);
} // namespace cdescriptor_pool
// Implement the Python "_BuildFile" method, it takes a serialized
// FileDescriptorProto, and adds it to the C++ DescriptorPool.
// It returns a new FileDescriptor object, or NULL when an exception is raised.
PyObject* Python_BuildFile(PyObject* ignored, PyObject* args);
// Retrieve the global descriptor pool owned by the _message module.
PyDescriptorPool* GetDescriptorPool();
// Initialize objects used by this module.
bool InitDescriptorPool();
} // namespace python
} // namespace protobuf
} // namespace google
#endif // GOOGLE_PROTOBUF_PYTHON_CPP_DESCRIPTOR_POOL_H__
...@@ -38,6 +38,7 @@ ...@@ -38,6 +38,7 @@
#include <google/protobuf/dynamic_message.h> #include <google/protobuf/dynamic_message.h>
#include <google/protobuf/message.h> #include <google/protobuf/message.h>
#include <google/protobuf/pyext/descriptor.h> #include <google/protobuf/pyext/descriptor.h>
#include <google/protobuf/pyext/descriptor_pool.h>
#include <google/protobuf/pyext/message.h> #include <google/protobuf/pyext/message.h>
#include <google/protobuf/pyext/repeated_composite_container.h> #include <google/protobuf/pyext/repeated_composite_container.h>
#include <google/protobuf/pyext/repeated_scalar_container.h> #include <google/protobuf/pyext/repeated_scalar_container.h>
...@@ -48,13 +49,11 @@ namespace google { ...@@ -48,13 +49,11 @@ namespace google {
namespace protobuf { namespace protobuf {
namespace python { namespace python {
extern google::protobuf::DynamicMessageFactory* global_message_factory;
namespace extension_dict { namespace extension_dict {
// TODO(tibell): Always use self->message for clarity, just like in // TODO(tibell): Always use self->message for clarity, just like in
// RepeatedCompositeContainer. // RepeatedCompositeContainer.
static google::protobuf::Message* GetMessage(ExtensionDict* self) { static Message* GetMessage(ExtensionDict* self) {
if (self->parent != NULL) { if (self->parent != NULL) {
return self->parent->message; return self->parent->message;
} else { } else {
...@@ -73,10 +72,9 @@ PyObject* len(ExtensionDict* self) { ...@@ -73,10 +72,9 @@ PyObject* len(ExtensionDict* self) {
// TODO(tibell): Use VisitCompositeField. // TODO(tibell): Use VisitCompositeField.
int ReleaseExtension(ExtensionDict* self, int ReleaseExtension(ExtensionDict* self,
PyObject* extension, PyObject* extension,
const google::protobuf::FieldDescriptor* descriptor) { const FieldDescriptor* descriptor) {
if (descriptor->label() == google::protobuf::FieldDescriptor::LABEL_REPEATED) { if (descriptor->label() == FieldDescriptor::LABEL_REPEATED) {
if (descriptor->cpp_type() == if (descriptor->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE) {
google::protobuf::FieldDescriptor::CPPTYPE_MESSAGE) {
if (repeated_composite_container::Release( if (repeated_composite_container::Release(
reinterpret_cast<RepeatedCompositeContainer*>( reinterpret_cast<RepeatedCompositeContainer*>(
extension)) < 0) { extension)) < 0) {
...@@ -89,8 +87,7 @@ int ReleaseExtension(ExtensionDict* self, ...@@ -89,8 +87,7 @@ int ReleaseExtension(ExtensionDict* self,
return -1; return -1;
} }
} }
} else if (descriptor->cpp_type() == } else if (descriptor->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE) {
google::protobuf::FieldDescriptor::CPPTYPE_MESSAGE) {
if (cmessage::ReleaseSubMessage( if (cmessage::ReleaseSubMessage(
GetMessage(self), descriptor, GetMessage(self), descriptor,
reinterpret_cast<CMessage*>(extension)) < 0) { reinterpret_cast<CMessage*>(extension)) < 0) {
...@@ -102,8 +99,7 @@ int ReleaseExtension(ExtensionDict* self, ...@@ -102,8 +99,7 @@ int ReleaseExtension(ExtensionDict* self,
} }
PyObject* subscript(ExtensionDict* self, PyObject* key) { PyObject* subscript(ExtensionDict* self, PyObject* key) {
const google::protobuf::FieldDescriptor* descriptor = const FieldDescriptor* descriptor = cmessage::GetExtensionDescriptor(key);
cmessage::GetExtensionDescriptor(key);
if (descriptor == NULL) { if (descriptor == NULL) {
return NULL; return NULL;
} }
...@@ -162,8 +158,7 @@ PyObject* subscript(ExtensionDict* self, PyObject* key) { ...@@ -162,8 +158,7 @@ PyObject* subscript(ExtensionDict* self, PyObject* key) {
} }
int ass_subscript(ExtensionDict* self, PyObject* key, PyObject* value) { int ass_subscript(ExtensionDict* self, PyObject* key, PyObject* value) {
const google::protobuf::FieldDescriptor* descriptor = const FieldDescriptor* descriptor = cmessage::GetExtensionDescriptor(key);
cmessage::GetExtensionDescriptor(key);
if (descriptor == NULL) { if (descriptor == NULL) {
return -1; return -1;
} }
...@@ -187,7 +182,7 @@ int ass_subscript(ExtensionDict* self, PyObject* key, PyObject* value) { ...@@ -187,7 +182,7 @@ int ass_subscript(ExtensionDict* self, PyObject* key, PyObject* value) {
} }
PyObject* ClearExtension(ExtensionDict* self, PyObject* extension) { PyObject* ClearExtension(ExtensionDict* self, PyObject* extension) {
const google::protobuf::FieldDescriptor* descriptor = const FieldDescriptor* descriptor =
cmessage::GetExtensionDescriptor(extension); cmessage::GetExtensionDescriptor(extension);
if (descriptor == NULL) { if (descriptor == NULL) {
return NULL; return NULL;
...@@ -208,7 +203,7 @@ PyObject* ClearExtension(ExtensionDict* self, PyObject* extension) { ...@@ -208,7 +203,7 @@ PyObject* ClearExtension(ExtensionDict* self, PyObject* extension) {
} }
PyObject* HasExtension(ExtensionDict* self, PyObject* extension) { PyObject* HasExtension(ExtensionDict* self, PyObject* extension) {
const google::protobuf::FieldDescriptor* descriptor = const FieldDescriptor* descriptor =
cmessage::GetExtensionDescriptor(extension); cmessage::GetExtensionDescriptor(extension);
if (descriptor == NULL) { if (descriptor == NULL) {
return NULL; return NULL;
......
...@@ -41,7 +41,6 @@ ...@@ -41,7 +41,6 @@
#include <google/protobuf/stubs/shared_ptr.h> #include <google/protobuf/stubs/shared_ptr.h>
#endif #endif
namespace google { namespace google {
namespace protobuf { namespace protobuf {
...@@ -94,7 +93,7 @@ PyObject* len(ExtensionDict* self); ...@@ -94,7 +93,7 @@ PyObject* len(ExtensionDict* self);
// Returns 0 on success, -1 on failure. // Returns 0 on success, -1 on failure.
int ReleaseExtension(ExtensionDict* self, int ReleaseExtension(ExtensionDict* self,
PyObject* extension, PyObject* extension,
const google::protobuf::FieldDescriptor* descriptor); const FieldDescriptor* descriptor);
// Gets an extension from the dict for the given extension descriptor. // Gets an extension from the dict for the given extension descriptor.
// //
......
This diff is collapsed.
...@@ -42,7 +42,6 @@ ...@@ -42,7 +42,6 @@
#endif #endif
#include <string> #include <string>
namespace google { namespace google {
namespace protobuf { namespace protobuf {
...@@ -50,12 +49,12 @@ class Message; ...@@ -50,12 +49,12 @@ class Message;
class Reflection; class Reflection;
class FieldDescriptor; class FieldDescriptor;
class Descriptor; class Descriptor;
class DynamicMessageFactory;
using internal::shared_ptr; using internal::shared_ptr;
namespace python { namespace python {
struct PyDescriptorPool;
struct ExtensionDict; struct ExtensionDict;
typedef struct CMessage { typedef struct CMessage {
...@@ -84,7 +83,7 @@ typedef struct CMessage { ...@@ -84,7 +83,7 @@ typedef struct CMessage {
// Used together with the parent's message when making a default message // Used together with the parent's message when making a default message
// instance mutable. // instance mutable.
// The pointer is owned by the global DescriptorPool. // The pointer is owned by the global DescriptorPool.
const google::protobuf::FieldDescriptor* parent_field_descriptor; const FieldDescriptor* parent_field_descriptor;
// Pointer to the C++ Message object for this CMessage. The // Pointer to the C++ Message object for this CMessage. The
// CMessage does not own this pointer. // CMessage does not own this pointer.
...@@ -115,27 +114,26 @@ namespace cmessage { ...@@ -115,27 +114,26 @@ namespace cmessage {
// Internal function to create a new empty Message Python object, but with empty // Internal function to create a new empty Message Python object, but with empty
// pointers to the C++ objects. // pointers to the C++ objects.
// The caller must fill self->message, self->owner and eventually self->parent. // The caller must fill self->message, self->owner and eventually self->parent.
CMessage* NewEmptyMessage(PyObject* type, CMessage* NewEmptyMessage(PyObject* type, const Descriptor* descriptor);
const google::protobuf::Descriptor* descriptor);
// Release a submessage from its proto tree, making it a new top-level messgae. // Release a submessage from its proto tree, making it a new top-level messgae.
// A new message will be created if this is a read-only default instance. // A new message will be created if this is a read-only default instance.
// //
// Corresponds to reflection api method ReleaseMessage. // Corresponds to reflection api method ReleaseMessage.
int ReleaseSubMessage(google::protobuf::Message* message, int ReleaseSubMessage(Message* message,
const google::protobuf::FieldDescriptor* field_descriptor, const FieldDescriptor* field_descriptor,
CMessage* child_cmessage); CMessage* child_cmessage);
// Retrieves the C++ descriptor of a Python Extension descriptor. // Retrieves the C++ descriptor of a Python Extension descriptor.
// On error, return NULL with an exception set. // On error, return NULL with an exception set.
const google::protobuf::FieldDescriptor* GetExtensionDescriptor(PyObject* extension); const FieldDescriptor* GetExtensionDescriptor(PyObject* extension);
// Initializes a new CMessage instance for a submessage. Only called once per // Initializes a new CMessage instance for a submessage. Only called once per
// submessage as the result is cached in composite_fields. // submessage as the result is cached in composite_fields.
// //
// Corresponds to reflection api method GetMessage. // Corresponds to reflection api method GetMessage.
PyObject* InternalGetSubMessage( PyObject* InternalGetSubMessage(
CMessage* self, const google::protobuf::FieldDescriptor* field_descriptor); CMessage* self, const FieldDescriptor* field_descriptor);
// Deletes a range of C++ submessages in a repeated field (following a // Deletes a range of C++ submessages in a repeated field (following a
// removal in a RepeatedCompositeContainer). // removal in a RepeatedCompositeContainer).
...@@ -146,20 +144,20 @@ PyObject* InternalGetSubMessage( ...@@ -146,20 +144,20 @@ PyObject* InternalGetSubMessage(
// by slice will be removed from cmessage_list by this function. // by slice will be removed from cmessage_list by this function.
// //
// Corresponds to reflection api method RemoveLast. // Corresponds to reflection api method RemoveLast.
int InternalDeleteRepeatedField(google::protobuf::Message* message, int InternalDeleteRepeatedField(Message* message,
const google::protobuf::FieldDescriptor* field_descriptor, const FieldDescriptor* field_descriptor,
PyObject* slice, PyObject* cmessage_list); PyObject* slice, PyObject* cmessage_list);
// Sets the specified scalar value to the message. // Sets the specified scalar value to the message.
int InternalSetScalar(CMessage* self, int InternalSetScalar(CMessage* self,
const google::protobuf::FieldDescriptor* field_descriptor, const FieldDescriptor* field_descriptor,
PyObject* value); PyObject* value);
// Retrieves the specified scalar value from the message. // Retrieves the specified scalar value from the message.
// //
// Returns a new python reference. // Returns a new python reference.
PyObject* InternalGetScalar(CMessage* self, PyObject* InternalGetScalar(CMessage* self,
const google::protobuf::FieldDescriptor* field_descriptor); const FieldDescriptor* field_descriptor);
// Clears the message, removing all contained data. Extension dictionary and // Clears the message, removing all contained data. Extension dictionary and
// submessages are released first if there are remaining external references. // submessages are released first if there are remaining external references.
...@@ -175,8 +173,7 @@ PyObject* Clear(CMessage* self); ...@@ -175,8 +173,7 @@ PyObject* Clear(CMessage* self);
// //
// Corresponds to reflection api method ClearField. // Corresponds to reflection api method ClearField.
PyObject* ClearFieldByDescriptor( PyObject* ClearFieldByDescriptor(
CMessage* self, CMessage* self, const FieldDescriptor* descriptor);
const google::protobuf::FieldDescriptor* descriptor);
// Clears the data for the given field name. The message is released if there // Clears the data for the given field name. The message is released if there
// are any external references. // are any external references.
...@@ -189,7 +186,7 @@ PyObject* ClearField(CMessage* self, PyObject* arg); ...@@ -189,7 +186,7 @@ PyObject* ClearField(CMessage* self, PyObject* arg);
// //
// Corresponds to reflection api method HasField // Corresponds to reflection api method HasField
PyObject* HasFieldByDescriptor( PyObject* HasFieldByDescriptor(
CMessage* self, const google::protobuf::FieldDescriptor* field_descriptor); CMessage* self, const FieldDescriptor* field_descriptor);
// Checks if the message has the named field. // Checks if the message has the named field.
// //
...@@ -220,18 +217,16 @@ int SetOwner(CMessage* self, const shared_ptr<Message>& new_owner); ...@@ -220,18 +217,16 @@ int SetOwner(CMessage* self, const shared_ptr<Message>& new_owner);
int AssureWritable(CMessage* self); int AssureWritable(CMessage* self);
} // namespace cmessage DynamicMessageFactory* GetMessageFactory();
// Retrieve the global descriptor pool owned by the _message module. } // namespace cmessage
PyDescriptorPool* GetDescriptorPool();
/* Is 64bit */ /* Is 64bit */
#define IS_64BIT (SIZEOF_LONG == 8) #define IS_64BIT (SIZEOF_LONG == 8)
#define FIELD_IS_REPEATED(field_descriptor) \ #define FIELD_IS_REPEATED(field_descriptor) \
((field_descriptor)->label() == google::protobuf::FieldDescriptor::LABEL_REPEATED) ((field_descriptor)->label() == FieldDescriptor::LABEL_REPEATED)
#define GOOGLE_CHECK_GET_INT32(arg, value, err) \ #define GOOGLE_CHECK_GET_INT32(arg, value, err) \
int32 value; \ int32 value; \
...@@ -294,18 +289,17 @@ bool CheckAndGetDouble(PyObject* arg, double* value); ...@@ -294,18 +289,17 @@ bool CheckAndGetDouble(PyObject* arg, double* value);
bool CheckAndGetFloat(PyObject* arg, float* value); bool CheckAndGetFloat(PyObject* arg, float* value);
bool CheckAndGetBool(PyObject* arg, bool* value); bool CheckAndGetBool(PyObject* arg, bool* value);
bool CheckAndSetString( bool CheckAndSetString(
PyObject* arg, google::protobuf::Message* message, PyObject* arg, Message* message,
const google::protobuf::FieldDescriptor* descriptor, const FieldDescriptor* descriptor,
const google::protobuf::Reflection* reflection, const Reflection* reflection,
bool append, bool append,
int index); int index);
PyObject* ToStringObject( PyObject* ToStringObject(const FieldDescriptor* descriptor, string value);
const google::protobuf::FieldDescriptor* descriptor, string value);
// Check if the passed field descriptor belongs to the given message. // Check if the passed field descriptor belongs to the given message.
// If not, return false and set a Python exception (a KeyError) // If not, return false and set a Python exception (a KeyError)
bool CheckFieldBelongsToMessage(const google::protobuf::FieldDescriptor* field_descriptor, bool CheckFieldBelongsToMessage(const FieldDescriptor* field_descriptor,
const google::protobuf::Message* message); const Message* message);
extern PyObject* PickleError_class; extern PyObject* PickleError_class;
......
#! /usr/bin/python
#
# Protocol Buffers - Google's data interchange format
# Copyright 2008 Google Inc. All rights reserved.
# https://developers.google.com/protocol-buffers/
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are
# met:
#
# * Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# * Redistributions in binary form must reproduce the above
# copyright notice, this list of conditions and the following disclaimer
# in the documentation and/or other materials provided with the
# distribution.
# * Neither the name of Google Inc. nor the names of its
# contributors may be used to endorse or promote products derived from
# this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
"""Tests for google.protobuf.message_factory."""
import os
os.environ['PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION'] = 'cpp'
os.environ['PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION_VERSION'] = '2'
# We must set the implementation version above before the google3 imports.
# pylint: disable=g-import-not-at-top
from google.apputils import basetest
from google.protobuf.internal import api_implementation
# Run all tests from the original module by putting them in our namespace.
# pylint: disable=wildcard-import
from google.protobuf.internal.message_factory_test import *
class ConfirmCppApi2Test(basetest.TestCase):
def testImplementationSetting(self):
self.assertEqual('cpp', api_implementation.Type())
self.assertEqual(2, api_implementation.Version())
if __name__ == '__main__':
basetest.main()
#! /usr/bin/python
# -*- coding: utf-8 -*-
#
# Protocol Buffers - Google's data interchange format
# Copyright 2008 Google Inc. All rights reserved.
# https://developers.google.com/protocol-buffers/
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are
# met:
#
# * Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# * Redistributions in binary form must reproduce the above
# copyright notice, this list of conditions and the following disclaimer
# in the documentation and/or other materials provided with the
# distribution.
# * Neither the name of Google Inc. nor the names of its
# contributors may be used to endorse or promote products derived from
# this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
"""Unittest for reflection.py, which tests the generated C++ implementation."""
__author__ = 'jasonh@google.com (Jason Hsueh)'
import os
os.environ['PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION'] = 'cpp'
os.environ['PROTOCOL_BUFFERS_PYTHON_IMPLEMENTATION_VERSION'] = '2'
from google.apputils import basetest
from google.protobuf.internal import api_implementation
from google.protobuf.internal import more_extensions_dynamic_pb2
from google.protobuf.internal import more_extensions_pb2
from google.protobuf.internal.reflection_test import *
class ReflectionCppTest(basetest.TestCase):
def testImplementationSetting(self):
self.assertEqual('cpp', api_implementation.Type())
self.assertEqual(2, api_implementation.Version())
def testExtensionOfGeneratedTypeInDynamicFile(self):
"""Tests that a file built dynamically can extend a generated C++ type.
The C++ implementation uses a DescriptorPool that has the generated
DescriptorPool as an underlay. Typically, a type can only find
extensions in its own pool. With the python C-extension, the generated C++
extendee may be available, but not the extension. This tests that the
C-extension implements the correct special handling to make such extensions
available.
"""
pb1 = more_extensions_pb2.ExtendedMessage()
# Test that basic accessors work.
self.assertFalse(
pb1.HasExtension(more_extensions_dynamic_pb2.dynamic_int32_extension))
self.assertFalse(
pb1.HasExtension(more_extensions_dynamic_pb2.dynamic_message_extension))
pb1.Extensions[more_extensions_dynamic_pb2.dynamic_int32_extension] = 17
pb1.Extensions[more_extensions_dynamic_pb2.dynamic_message_extension].a = 24
self.assertTrue(
pb1.HasExtension(more_extensions_dynamic_pb2.dynamic_int32_extension))
self.assertTrue(
pb1.HasExtension(more_extensions_dynamic_pb2.dynamic_message_extension))
# Now serialize the data and parse to a new message.
pb2 = more_extensions_pb2.ExtendedMessage()
pb2.MergeFromString(pb1.SerializeToString())
self.assertTrue(
pb2.HasExtension(more_extensions_dynamic_pb2.dynamic_int32_extension))
self.assertTrue(
pb2.HasExtension(more_extensions_dynamic_pb2.dynamic_message_extension))
self.assertEqual(
17, pb2.Extensions[more_extensions_dynamic_pb2.dynamic_int32_extension])
self.assertEqual(
24,
pb2.Extensions[more_extensions_dynamic_pb2.dynamic_message_extension].a)
if __name__ == '__main__':
basetest.main()
...@@ -43,7 +43,6 @@ ...@@ -43,7 +43,6 @@
#include <string> #include <string>
#include <vector> #include <vector>
namespace google { namespace google {
namespace protobuf { namespace protobuf {
...@@ -82,7 +81,7 @@ typedef struct RepeatedCompositeContainer { ...@@ -82,7 +81,7 @@ typedef struct RepeatedCompositeContainer {
// A descriptor used to modify the underlying 'message'. // A descriptor used to modify the underlying 'message'.
// The pointer is owned by the global DescriptorPool. // The pointer is owned by the global DescriptorPool.
const google::protobuf::FieldDescriptor* parent_field_descriptor; const FieldDescriptor* parent_field_descriptor;
// Pointer to the C++ Message that contains this container. The // Pointer to the C++ Message that contains this container. The
// RepeatedCompositeContainer does not own this pointer. // RepeatedCompositeContainer does not own this pointer.
...@@ -106,7 +105,7 @@ namespace repeated_composite_container { ...@@ -106,7 +105,7 @@ namespace repeated_composite_container {
// field descriptor. // field descriptor.
PyObject *NewContainer( PyObject *NewContainer(
CMessage* parent, CMessage* parent,
const google::protobuf::FieldDescriptor* parent_field_descriptor, const FieldDescriptor* parent_field_descriptor,
PyObject *concrete_class); PyObject *concrete_class);
// Returns the number of items in this repeated composite container. // Returns the number of items in this repeated composite container.
...@@ -150,8 +149,7 @@ int AssignSubscript(RepeatedCompositeContainer* self, ...@@ -150,8 +149,7 @@ int AssignSubscript(RepeatedCompositeContainer* self,
// Releases the messages in the container to the given message. // Releases the messages in the container to the given message.
// //
// Returns 0 on success, -1 on failure. // Returns 0 on success, -1 on failure.
int ReleaseToMessage(RepeatedCompositeContainer* self, int ReleaseToMessage(RepeatedCompositeContainer* self, Message* new_message);
google::protobuf::Message* new_message);
// Releases the messages in the container to a new message. // Releases the messages in the container to a new message.
// //
......
...@@ -77,7 +77,7 @@ typedef struct RepeatedScalarContainer { ...@@ -77,7 +77,7 @@ typedef struct RepeatedScalarContainer {
// field. Used together with the parent's message when making a // field. Used together with the parent's message when making a
// default message instance mutable. // default message instance mutable.
// The pointer is owned by the global DescriptorPool. // The pointer is owned by the global DescriptorPool.
const google::protobuf::FieldDescriptor* parent_field_descriptor; const FieldDescriptor* parent_field_descriptor;
} RepeatedScalarContainer; } RepeatedScalarContainer;
extern PyTypeObject RepeatedScalarContainer_Type; extern PyTypeObject RepeatedScalarContainer_Type;
...@@ -87,7 +87,7 @@ namespace repeated_scalar_container { ...@@ -87,7 +87,7 @@ namespace repeated_scalar_container {
// Builds a RepeatedScalarContainer object, from a parent message and a // Builds a RepeatedScalarContainer object, from a parent message and a
// field descriptor. // field descriptor.
extern PyObject *NewContainer( extern PyObject *NewContainer(
CMessage* parent, const google::protobuf::FieldDescriptor* parent_field_descriptor); CMessage* parent, const FieldDescriptor* parent_field_descriptor);
// Appends the scalar 'item' to the end of the container 'self'. // Appends the scalar 'item' to the end of the container 'self'.
// //
......
...@@ -56,18 +56,12 @@ _FieldDescriptor = descriptor_mod.FieldDescriptor ...@@ -56,18 +56,12 @@ _FieldDescriptor = descriptor_mod.FieldDescriptor
if api_implementation.Type() == 'cpp': if api_implementation.Type() == 'cpp':
if api_implementation.Version() == 2: from google.protobuf.pyext import cpp_message as message_impl
from google.protobuf.pyext import cpp_message
_NewMessage = cpp_message.NewMessage
_InitMessage = cpp_message.InitMessage
else:
from google.protobuf.internal import cpp_message
_NewMessage = cpp_message.NewMessage
_InitMessage = cpp_message.InitMessage
else: else:
from google.protobuf.internal import python_message from google.protobuf.internal import python_message as message_impl
_NewMessage = python_message.NewMessage
_InitMessage = python_message.InitMessage _NewMessage = message_impl.NewMessage
_InitMessage = message_impl.InitMessage
class GeneratedProtocolMessageType(type): class GeneratedProtocolMessageType(type):
...@@ -127,7 +121,6 @@ class GeneratedProtocolMessageType(type): ...@@ -127,7 +121,6 @@ class GeneratedProtocolMessageType(type):
superclass = super(GeneratedProtocolMessageType, cls) superclass = super(GeneratedProtocolMessageType, cls)
new_class = superclass.__new__(cls, name, bases, dictionary) new_class = superclass.__new__(cls, name, bases, dictionary)
setattr(descriptor, '_concrete_class', new_class)
return new_class return new_class
def __init__(cls, name, bases, dictionary): def __init__(cls, name, bases, dictionary):
...@@ -151,6 +144,7 @@ class GeneratedProtocolMessageType(type): ...@@ -151,6 +144,7 @@ class GeneratedProtocolMessageType(type):
_InitMessage(descriptor, cls) _InitMessage(descriptor, cls)
superclass = super(GeneratedProtocolMessageType, cls) superclass = super(GeneratedProtocolMessageType, cls)
superclass.__init__(name, bases, dictionary) superclass.__init__(name, bases, dictionary)
setattr(descriptor, '_concrete_class', cls)
def ParseMessage(descriptor, byte_str): def ParseMessage(descriptor, byte_str):
......
This diff is collapsed.
This diff is collapsed.
...@@ -325,6 +325,7 @@ EXTRA_DIST = \ ...@@ -325,6 +325,7 @@ EXTRA_DIST = \
google/protobuf/io/gzip_stream_unittest.sh \ google/protobuf/io/gzip_stream_unittest.sh \
google/protobuf/testdata/golden_message \ google/protobuf/testdata/golden_message \
google/protobuf/testdata/golden_message_oneof_implemented \ google/protobuf/testdata/golden_message_oneof_implemented \
google/protobuf/testdata/golden_message_proto3 \
google/protobuf/testdata/golden_packed_fields_message \ google/protobuf/testdata/golden_packed_fields_message \
google/protobuf/testdata/bad_utf8_string \ google/protobuf/testdata/bad_utf8_string \
google/protobuf/testdata/text_format_unittest_data.txt \ google/protobuf/testdata/text_format_unittest_data.txt \
......
This diff is collapsed.
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment