| #!/usr/bin/env python3 |
| # Protocol Buffers - Google's data interchange format |
| # Copyright 2008 Google Inc. All rights reserved. |
| # |
| # Use of this source code is governed by a BSD-style |
| # license that can be found in the LICENSE file or at |
| # https://developers.google.com/open-source/licenses/bsd |
| |
| """A conformance test implementation for the Python protobuf library. |
| |
| See conformance.proto for more information. |
| """ |
| |
| import struct |
| import sys |
| from google.protobuf import json_format |
| from google.protobuf import message |
| from google.protobuf import text_format |
| from google.protobuf import test_messages_proto2_pb2 |
| from google.protobuf import test_messages_proto3_pb2 |
| from conformance import conformance_pb2 |
| from conformance.test_protos import test_messages_edition2023_pb2 |
| from editions.golden import test_messages_proto2_editions_pb2 |
| from editions.golden import test_messages_proto3_editions_pb2 |
| |
| test_count = 0 |
| verbose = False |
| |
| |
| class ProtocolError(Exception): |
| pass |
| |
| |
| def _create_test_message(type): |
| if type == "protobuf_test_messages.proto2.TestAllTypesProto2": |
| return test_messages_proto2_pb2.TestAllTypesProto2() |
| if type == "protobuf_test_messages.proto3.TestAllTypesProto3": |
| return test_messages_proto3_pb2.TestAllTypesProto3() |
| if type == "protobuf_test_messages.editions.TestAllTypesEdition2023": |
| return test_messages_edition2023_pb2.TestAllTypesEdition2023() |
| if type == "protobuf_test_messages.editions.proto2.TestAllTypesProto2": |
| return test_messages_proto2_editions_pb2.TestAllTypesProto2() |
| if type == "protobuf_test_messages.editions.proto3.TestAllTypesProto3": |
| return test_messages_proto3_editions_pb2.TestAllTypesProto3() |
| return None |
| |
| |
| def do_test(request): |
| response = conformance_pb2.ConformanceResponse() |
| |
| if request.message_type == "conformance.FailureSet": |
| failure_set = conformance_pb2.FailureSet() |
| failures = [] |
| # TODO: Remove, this is a hack to detect if the old vs new |
| # parser is used by the cpp code. Relying on a bug in the old parser. |
| hack_proto = test_messages_proto2_pb2.TestAllTypesProto2() |
| old_parser = True |
| try: |
| hack_proto.ParseFromString(b"\322\002\001") |
| except message.DecodeError as e: |
| old_parser = False |
| if old_parser: |
| # the string above is one of the failing conformance test strings of the |
| # old parser. If we succeed the c++ implementation is using the old |
| # parser so we add the list of failing conformance tests. |
| failures = [ |
| "Required.Proto3.ProtobufInput.PrematureEofInDelimitedDataForKnownNonRepeatedValue.MESSAGE", |
| "Required.Proto3.ProtobufInput.PrematureEofInDelimitedDataForKnownRepeatedValue.MESSAGE", |
| "Required.Proto3.ProtobufInput.PrematureEofInPackedField.BOOL", |
| "Required.Proto3.ProtobufInput.PrematureEofInPackedField.DOUBLE", |
| "Required.Proto3.ProtobufInput.PrematureEofInPackedField.ENUM", |
| "Required.Proto3.ProtobufInput.PrematureEofInPackedField.FIXED32", |
| "Required.Proto3.ProtobufInput.PrematureEofInPackedField.FIXED64", |
| "Required.Proto3.ProtobufInput.PrematureEofInPackedField.FLOAT", |
| "Required.Proto3.ProtobufInput.PrematureEofInPackedField.INT32", |
| "Required.Proto3.ProtobufInput.PrematureEofInPackedField.INT64", |
| "Required.Proto3.ProtobufInput.PrematureEofInPackedField.SFIXED32", |
| "Required.Proto3.ProtobufInput.PrematureEofInPackedField.SFIXED64", |
| "Required.Proto3.ProtobufInput.PrematureEofInPackedField.SINT32", |
| "Required.Proto3.ProtobufInput.PrematureEofInPackedField.SINT64", |
| "Required.Proto3.ProtobufInput.PrematureEofInPackedField.UINT32", |
| "Required.Proto3.ProtobufInput.PrematureEofInPackedField.UINT64", |
| "Required.Proto2.ProtobufInput.PrematureEofInDelimitedDataForKnownNonRepeatedValue.MESSAGE", |
| "Required.Proto2.ProtobufInput.PrematureEofInDelimitedDataForKnownRepeatedValue.MESSAGE", |
| "Required.Proto2.ProtobufInput.PrematureEofInPackedField.BOOL", |
| "Required.Proto2.ProtobufInput.PrematureEofInPackedField.DOUBLE", |
| "Required.Proto2.ProtobufInput.PrematureEofInPackedField.ENUM", |
| "Required.Proto2.ProtobufInput.PrematureEofInPackedField.FIXED32", |
| "Required.Proto2.ProtobufInput.PrematureEofInPackedField.FIXED64", |
| "Required.Proto2.ProtobufInput.PrematureEofInPackedField.FLOAT", |
| "Required.Proto2.ProtobufInput.PrematureEofInPackedField.INT32", |
| "Required.Proto2.ProtobufInput.PrematureEofInPackedField.INT64", |
| "Required.Proto2.ProtobufInput.PrematureEofInPackedField.SFIXED32", |
| "Required.Proto2.ProtobufInput.PrematureEofInPackedField.SFIXED64", |
| "Required.Proto2.ProtobufInput.PrematureEofInPackedField.SINT32", |
| "Required.Proto2.ProtobufInput.PrematureEofInPackedField.SINT64", |
| "Required.Proto2.ProtobufInput.PrematureEofInPackedField.UINT32", |
| "Required.Proto2.ProtobufInput.PrematureEofInPackedField.UINT64", |
| ] |
| for x in failures: |
| failure_set.test.append(conformance_pb2.TestStatus(name=x)) |
| response.protobuf_payload = failure_set.SerializeToString() |
| return response |
| |
| isJson = request.WhichOneof("payload") == "json_payload" |
| test_message = _create_test_message(request.message_type) |
| |
| if (not isJson) and (test_message is None): |
| raise ProtocolError("Protobuf request doesn't have specific payload type") |
| |
| try: |
| if request.WhichOneof("payload") == "protobuf_payload": |
| try: |
| test_message.ParseFromString(request.protobuf_payload) |
| except message.DecodeError as e: |
| response.parse_error = str(e) |
| return response |
| |
| elif request.WhichOneof("payload") == "json_payload": |
| try: |
| ignore_unknown_fields = ( |
| request.test_category |
| == conformance_pb2.JSON_IGNORE_UNKNOWN_PARSING_TEST |
| ) |
| json_format.Parse( |
| request.json_payload, test_message, ignore_unknown_fields |
| ) |
| except Exception as e: |
| response.parse_error = str(e) |
| return response |
| |
| elif request.WhichOneof("payload") == "text_payload": |
| try: |
| text_format.Parse(request.text_payload, test_message) |
| except Exception as e: |
| response.parse_error = str(e) |
| return response |
| |
| else: |
| raise ProtocolError("Request didn't have payload.") |
| |
| if request.requested_output_format == conformance_pb2.UNSPECIFIED: |
| raise ProtocolError("Unspecified output format") |
| |
| elif request.requested_output_format == conformance_pb2.PROTOBUF: |
| response.protobuf_payload = test_message.SerializeToString() |
| |
| elif request.requested_output_format == conformance_pb2.JSON: |
| try: |
| response.json_payload = json_format.MessageToJson(test_message) |
| except Exception as e: |
| response.serialize_error = str(e) |
| return response |
| |
| elif request.requested_output_format == conformance_pb2.TEXT_FORMAT: |
| response.text_payload = text_format.MessageToString( |
| test_message, print_unknown_fields=request.print_unknown_fields |
| ) |
| |
| except Exception as e: |
| response.runtime_error = str(e) |
| |
| return response |
| |
| |
| def do_test_io(): |
| length_bytes = sys.stdin.buffer.read(4) |
| if len(length_bytes) == 0: |
| return False # EOF |
| elif len(length_bytes) != 4: |
| raise IOError("I/O error") |
| |
| length = struct.unpack("<I", length_bytes)[0] |
| serialized_request = sys.stdin.buffer.read(length) |
| if len(serialized_request) != length: |
| raise IOError("I/O error") |
| |
| request = conformance_pb2.ConformanceRequest() |
| request.ParseFromString(serialized_request) |
| |
| response = do_test(request) |
| |
| serialized_response = response.SerializeToString() |
| sys.stdout.buffer.write(struct.pack("<I", len(serialized_response))) |
| sys.stdout.buffer.write(serialized_response) |
| sys.stdout.buffer.flush() |
| |
| if verbose: |
| sys.stderr.write( |
| "conformance_python: request=%s, response=%s\n" |
| % ( |
| request.ShortDebugString().c_str(), |
| response.ShortDebugString().c_str(), |
| ) |
| ) |
| |
| global test_count |
| test_count += 1 |
| |
| return True |
| |
| |
| while True: |
| if not do_test_io(): |
| sys.stderr.write( |
| "conformance_python: received EOF from test runner " |
| + "after %s tests, exiting\n" % (test_count,) |
| ) |
| sys.exit(0) |