Commit a6d624ba authored by Hao Nguyen's avatar Hao Nguyen

Merge branch 'master' of https://github.com/haon4/protobuf

parents 3d46d8df cecba296
...@@ -21,7 +21,9 @@ add_custom_command( ...@@ -21,7 +21,9 @@ add_custom_command(
add_executable(conformance_test_runner add_executable(conformance_test_runner
${protobuf_source_dir}/conformance/conformance.pb.cc ${protobuf_source_dir}/conformance/conformance.pb.cc
${protobuf_source_dir}/conformance/conformance_test.cc ${protobuf_source_dir}/conformance/conformance_test.cc
${protobuf_source_dir}/conformance/conformance_test_impl.cc ${protobuf_source_dir}/conformance/binary_json_conformance_main.cc
${protobuf_source_dir}/conformance/binary_json_conformance_suite.cc
${protobuf_source_dir}/conformance/binary_json_conformance_suite.h
${protobuf_source_dir}/conformance/conformance_test_runner.cc ${protobuf_source_dir}/conformance/conformance_test_runner.cc
${protobuf_source_dir}/conformance/third_party/jsoncpp/json.h ${protobuf_source_dir}/conformance/third_party/jsoncpp/json.h
${protobuf_source_dir}/conformance/third_party/jsoncpp/jsoncpp.cpp ${protobuf_source_dir}/conformance/third_party/jsoncpp/jsoncpp.cpp
......
...@@ -5,6 +5,7 @@ set(libprotobuf_lite_files ...@@ -5,6 +5,7 @@ set(libprotobuf_lite_files
${protobuf_source_dir}/src/google/protobuf/generated_message_table_driven_lite.cc ${protobuf_source_dir}/src/google/protobuf/generated_message_table_driven_lite.cc
${protobuf_source_dir}/src/google/protobuf/generated_message_util.cc ${protobuf_source_dir}/src/google/protobuf/generated_message_util.cc
${protobuf_source_dir}/src/google/protobuf/implicit_weak_message.cc ${protobuf_source_dir}/src/google/protobuf/implicit_weak_message.cc
${protobuf_source_dir}/src/google/protobuf/parse_context.cc
${protobuf_source_dir}/src/google/protobuf/io/coded_stream.cc ${protobuf_source_dir}/src/google/protobuf/io/coded_stream.cc
${protobuf_source_dir}/src/google/protobuf/io/zero_copy_stream.cc ${protobuf_source_dir}/src/google/protobuf/io/zero_copy_stream.cc
${protobuf_source_dir}/src/google/protobuf/io/zero_copy_stream_impl_lite.cc ${protobuf_source_dir}/src/google/protobuf/io/zero_copy_stream_impl_lite.cc
...@@ -30,6 +31,7 @@ set(libprotobuf_lite_includes ...@@ -30,6 +31,7 @@ set(libprotobuf_lite_includes
${protobuf_source_dir}/src/google/protobuf/extension_set.h ${protobuf_source_dir}/src/google/protobuf/extension_set.h
${protobuf_source_dir}/src/google/protobuf/generated_message_util.h ${protobuf_source_dir}/src/google/protobuf/generated_message_util.h
${protobuf_source_dir}/src/google/protobuf/implicit_weak_message.h ${protobuf_source_dir}/src/google/protobuf/implicit_weak_message.h
${protobuf_source_dir}/src/google/protobuf/parse_context.h
${protobuf_source_dir}/src/google/protobuf/io/coded_stream.h ${protobuf_source_dir}/src/google/protobuf/io/coded_stream.h
${protobuf_source_dir}/src/google/protobuf/io/zero_copy_stream.h ${protobuf_source_dir}/src/google/protobuf/io/zero_copy_stream.h
${protobuf_source_dir}/src/google/protobuf/io/zero_copy_stream_impl_lite.h ${protobuf_source_dir}/src/google/protobuf/io/zero_copy_stream_impl_lite.h
......
...@@ -206,7 +206,9 @@ EXTRA_DIST = \ ...@@ -206,7 +206,9 @@ EXTRA_DIST = \
conformance_test_runner_LDADD = $(top_srcdir)/src/libprotobuf.la conformance_test_runner_LDADD = $(top_srcdir)/src/libprotobuf.la
conformance_test_runner_SOURCES = conformance_test.h conformance_test.cc \ conformance_test_runner_SOURCES = conformance_test.h conformance_test.cc \
conformance_test_impl.cc \ binary_json_conformance_main.cc \
binary_json_conformance_suite.h \
binary_json_conformance_suite.cc \
conformance_test_runner.cc \ conformance_test_runner.cc \
third_party/jsoncpp/json.h \ third_party/jsoncpp/json.h \
third_party/jsoncpp/jsoncpp.cpp third_party/jsoncpp/jsoncpp.cpp
......
// 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.
#include "binary_json_conformance_suite.h"
#include "conformance_test.h"
int main(int argc, char *argv[]) {
google::protobuf::BinaryAndJsonConformanceSuite suite;
return google::protobuf::ForkPipeRunner::Run(argc, argv, &suite);
}
...@@ -28,6 +28,7 @@ ...@@ -28,6 +28,7 @@
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#include "binary_json_conformance_suite.h"
#include "conformance_test.h" #include "conformance_test.h"
#include "third_party/jsoncpp/json.h" #include "third_party/jsoncpp/json.h"
...@@ -37,11 +38,13 @@ ...@@ -37,11 +38,13 @@
#include <google/protobuf/stubs/common.h> #include <google/protobuf/stubs/common.h>
#include <google/protobuf/stubs/strutil.h> #include <google/protobuf/stubs/strutil.h>
#include <google/protobuf/text_format.h> #include <google/protobuf/text_format.h>
#include <google/protobuf/util/json_util.h>
#include <google/protobuf/util/type_resolver_util.h> #include <google/protobuf/util/type_resolver_util.h>
#include <google/protobuf/wire_format_lite.h> #include <google/protobuf/wire_format_lite.h>
using conformance::ConformanceRequest; using conformance::ConformanceRequest;
using conformance::ConformanceResponse; using conformance::ConformanceResponse;
using conformance::WireFormat;
using google::protobuf::Descriptor; using google::protobuf::Descriptor;
using google::protobuf::FieldDescriptor; using google::protobuf::FieldDescriptor;
using google::protobuf::Message; using google::protobuf::Message;
...@@ -189,73 +192,83 @@ std::unique_ptr<Message> NewTestMessage(bool is_proto3) { ...@@ -189,73 +192,83 @@ std::unique_ptr<Message> NewTestMessage(bool is_proto3) {
namespace google { namespace google {
namespace protobuf { namespace protobuf {
class ConformanceTestSuiteImpl : public ConformanceTestSuite { bool BinaryAndJsonConformanceSuite::ParseJsonResponse(
public: const ConformanceResponse& response,
ConformanceTestSuiteImpl() {} Message* test_message) {
string binary_protobuf;
private: util::Status status =
void RunSuiteImpl(); JsonToBinaryString(type_resolver_.get(), type_url_,
void RunValidJsonTest(const string& test_name, response.json_payload(), &binary_protobuf);
ConformanceLevel level,
const string& input_json, if (!status.ok()) {
const string& equivalent_text_format); return false;
void RunValidJsonTestWithProtobufInput( }
const string& test_name,
ConformanceLevel level, if (!test_message->ParseFromString(binary_protobuf)) {
const protobuf_test_messages::proto3::TestAllTypesProto3& input, GOOGLE_LOG(FATAL)
const string& equivalent_text_format); << "INTERNAL ERROR: internal JSON->protobuf transcode "
void RunValidJsonIgnoreUnknownTest( << "yielded unparseable proto.";
const string& test_name, ConformanceLevel level, const string& input_json, return false;
const string& equivalent_text_format); }
void RunValidProtobufTest(const string& test_name, ConformanceLevel level,
const string& input_protobuf, return true;
const string& equivalent_text_format, }
bool is_proto3);
void RunValidBinaryProtobufTest(const string& test_name, bool BinaryAndJsonConformanceSuite::ParseResponse(
ConformanceLevel level, const ConformanceResponse& response,
const string& input_protobuf, const ConformanceRequestSetting& setting,
bool is_proto3); Message* test_message) {
void RunValidProtobufTestWithMessage( const ConformanceRequest& request = setting.GetRequest();
const string& test_name, ConformanceLevel level, WireFormat requested_output = request.requested_output_format();
const Message *input, const string& test_name = setting.GetTestName();
const string& equivalent_text_format, ConformanceLevel level = setting.GetLevel();
bool is_proto3);
switch (response.result_case()) {
typedef std::function<bool(const Json::Value&)> Validator; case ConformanceResponse::kProtobufPayload: {
void RunValidJsonTestWithValidator(const string& test_name, if (requested_output != conformance::PROTOBUF) {
ConformanceLevel level, ReportFailure(
const string& input_json, test_name, level, request, response,
const Validator& validator); StrCat("Test was asked for ", WireFormatToString(requested_output),
void ExpectParseFailureForJson(const string& test_name, " output but provided PROTOBUF instead.").c_str());
ConformanceLevel level, return false;
const string& input_json); }
void ExpectSerializeFailureForJson(const string& test_name,
ConformanceLevel level, if (!test_message->ParseFromString(response.protobuf_payload())) {
const string& text_format); ReportFailure(test_name, level, request, response,
void ExpectParseFailureForProtoWithProtoVersion (const string& proto, "Protobuf output we received from test was unparseable.");
const string& test_name, return false;
ConformanceLevel level, }
bool is_proto3);
void ExpectParseFailureForProto(const std::string& proto, break;
const std::string& test_name, }
ConformanceLevel level);
void ExpectHardParseFailureForProto(const std::string& proto, case ConformanceResponse::kJsonPayload: {
const std::string& test_name, if (requested_output != conformance::JSON) {
ConformanceLevel level); ReportFailure(
void TestPrematureEOFForType(google::protobuf::FieldDescriptor::Type type); test_name, level, request, response,
void TestIllegalTags(); StrCat("Test was asked for ", WireFormatToString(requested_output),
template <class MessageType> " output but provided JSON instead.").c_str());
void TestOneofMessage (MessageType &message, return false;
bool is_proto3); }
template <class MessageType>
void TestUnknownMessage (MessageType &message, if (!ParseJsonResponse(response, test_message)) {
bool is_proto3); ReportFailure(test_name, level, request, response,
void TestValidDataForType( "JSON output we received from test was unparseable.");
google::protobuf::FieldDescriptor::Type, return false;
std::vector<std::pair<std::string, std::string>> values); }
};
break;
void ConformanceTestSuiteImpl::ExpectParseFailureForProtoWithProtoVersion ( }
default:
GOOGLE_LOG(FATAL) << test_name << ": unknown payload type: "
<< response.result_case();
}
return true;
}
void BinaryAndJsonConformanceSuite::ExpectParseFailureForProtoWithProtoVersion (
const string& proto, const string& test_name, ConformanceLevel level, const string& proto, const string& test_name, ConformanceLevel level,
bool is_proto3) { bool is_proto3) {
std::unique_ptr<Message> prototype = NewTestMessage(is_proto3); std::unique_ptr<Message> prototype = NewTestMessage(is_proto3);
...@@ -285,7 +298,7 @@ void ConformanceTestSuiteImpl::ExpectParseFailureForProtoWithProtoVersion ( ...@@ -285,7 +298,7 @@ void ConformanceTestSuiteImpl::ExpectParseFailureForProtoWithProtoVersion (
} }
// Expect that this precise protobuf will cause a parse error. // Expect that this precise protobuf will cause a parse error.
void ConformanceTestSuiteImpl::ExpectParseFailureForProto( void BinaryAndJsonConformanceSuite::ExpectParseFailureForProto(
const string& proto, const string& test_name, ConformanceLevel level) { const string& proto, const string& test_name, ConformanceLevel level) {
ExpectParseFailureForProtoWithProtoVersion(proto, test_name, level, true); ExpectParseFailureForProtoWithProtoVersion(proto, test_name, level, true);
ExpectParseFailureForProtoWithProtoVersion(proto, test_name, level, false); ExpectParseFailureForProtoWithProtoVersion(proto, test_name, level, false);
...@@ -296,12 +309,12 @@ void ConformanceTestSuiteImpl::ExpectParseFailureForProto( ...@@ -296,12 +309,12 @@ void ConformanceTestSuiteImpl::ExpectParseFailureForProto(
// data verbatim and once with this data followed by some valid data. // data verbatim and once with this data followed by some valid data.
// //
// TODO(haberman): implement the second of these. // TODO(haberman): implement the second of these.
void ConformanceTestSuiteImpl::ExpectHardParseFailureForProto( void BinaryAndJsonConformanceSuite::ExpectHardParseFailureForProto(
const string& proto, const string& test_name, ConformanceLevel level) { const string& proto, const string& test_name, ConformanceLevel level) {
return ExpectParseFailureForProto(proto, test_name, level); return ExpectParseFailureForProto(proto, test_name, level);
} }
void ConformanceTestSuiteImpl::RunValidJsonTest( void BinaryAndJsonConformanceSuite::RunValidJsonTest(
const string& test_name, ConformanceLevel level, const string& input_json, const string& test_name, ConformanceLevel level, const string& input_json,
const string& equivalent_text_format) { const string& equivalent_text_format) {
TestAllTypesProto3 prototype; TestAllTypesProto3 prototype;
...@@ -317,7 +330,7 @@ void ConformanceTestSuiteImpl::RunValidJsonTest( ...@@ -317,7 +330,7 @@ void ConformanceTestSuiteImpl::RunValidJsonTest(
RunValidInputTest(setting2, equivalent_text_format); RunValidInputTest(setting2, equivalent_text_format);
} }
void ConformanceTestSuiteImpl::RunValidJsonTestWithProtobufInput( void BinaryAndJsonConformanceSuite::RunValidJsonTestWithProtobufInput(
const string& test_name, ConformanceLevel level, const TestAllTypesProto3& input, const string& test_name, ConformanceLevel level, const TestAllTypesProto3& input,
const string& equivalent_text_format) { const string& equivalent_text_format) {
ConformanceRequestSetting setting( ConformanceRequestSetting setting(
...@@ -327,7 +340,7 @@ void ConformanceTestSuiteImpl::RunValidJsonTestWithProtobufInput( ...@@ -327,7 +340,7 @@ void ConformanceTestSuiteImpl::RunValidJsonTestWithProtobufInput(
RunValidInputTest(setting, equivalent_text_format); RunValidInputTest(setting, equivalent_text_format);
} }
void ConformanceTestSuiteImpl::RunValidJsonIgnoreUnknownTest( void BinaryAndJsonConformanceSuite::RunValidJsonIgnoreUnknownTest(
const string& test_name, ConformanceLevel level, const string& input_json, const string& test_name, ConformanceLevel level, const string& input_json,
const string& equivalent_text_format) { const string& equivalent_text_format) {
TestAllTypesProto3 prototype; TestAllTypesProto3 prototype;
...@@ -338,7 +351,7 @@ void ConformanceTestSuiteImpl::RunValidJsonIgnoreUnknownTest( ...@@ -338,7 +351,7 @@ void ConformanceTestSuiteImpl::RunValidJsonIgnoreUnknownTest(
RunValidInputTest(setting, equivalent_text_format); RunValidInputTest(setting, equivalent_text_format);
} }
void ConformanceTestSuiteImpl::RunValidProtobufTest( void BinaryAndJsonConformanceSuite::RunValidProtobufTest(
const string& test_name, ConformanceLevel level, const string& test_name, ConformanceLevel level,
const string& input_protobuf, const string& equivalent_text_format, const string& input_protobuf, const string& equivalent_text_format,
bool is_proto3) { bool is_proto3) {
...@@ -359,7 +372,7 @@ void ConformanceTestSuiteImpl::RunValidProtobufTest( ...@@ -359,7 +372,7 @@ void ConformanceTestSuiteImpl::RunValidProtobufTest(
} }
} }
void ConformanceTestSuiteImpl::RunValidBinaryProtobufTest( void BinaryAndJsonConformanceSuite::RunValidBinaryProtobufTest(
const string& test_name, ConformanceLevel level, const string& test_name, ConformanceLevel level,
const string& input_protobuf, bool is_proto3) { const string& input_protobuf, bool is_proto3) {
std::unique_ptr<Message> prototype = NewTestMessage(is_proto3); std::unique_ptr<Message> prototype = NewTestMessage(is_proto3);
...@@ -370,7 +383,7 @@ void ConformanceTestSuiteImpl::RunValidBinaryProtobufTest( ...@@ -370,7 +383,7 @@ void ConformanceTestSuiteImpl::RunValidBinaryProtobufTest(
RunValidBinaryInputTest(setting, input_protobuf); RunValidBinaryInputTest(setting, input_protobuf);
} }
void ConformanceTestSuiteImpl::RunValidProtobufTestWithMessage( void BinaryAndJsonConformanceSuite::RunValidProtobufTestWithMessage(
const string& test_name, ConformanceLevel level, const Message *input, const string& test_name, ConformanceLevel level, const Message *input,
const string& equivalent_text_format, bool is_proto3) { const string& equivalent_text_format, bool is_proto3) {
RunValidProtobufTest(test_name, level, input->SerializeAsString(), RunValidProtobufTest(test_name, level, input->SerializeAsString(),
...@@ -382,7 +395,7 @@ void ConformanceTestSuiteImpl::RunValidProtobufTestWithMessage( ...@@ -382,7 +395,7 @@ void ConformanceTestSuiteImpl::RunValidProtobufTestWithMessage(
// numbers while the parser is allowed to accept them as JSON strings). This // numbers while the parser is allowed to accept them as JSON strings). This
// method allows strict checking on a proto3 JSON serializer by inspecting // method allows strict checking on a proto3 JSON serializer by inspecting
// the JSON output directly. // the JSON output directly.
void ConformanceTestSuiteImpl::RunValidJsonTestWithValidator( void BinaryAndJsonConformanceSuite::RunValidJsonTestWithValidator(
const string& test_name, ConformanceLevel level, const string& input_json, const string& test_name, ConformanceLevel level, const string& input_json,
const Validator& validator) { const Validator& validator) {
TestAllTypesProto3 prototype; TestAllTypesProto3 prototype;
...@@ -426,7 +439,7 @@ void ConformanceTestSuiteImpl::RunValidJsonTestWithValidator( ...@@ -426,7 +439,7 @@ void ConformanceTestSuiteImpl::RunValidJsonTestWithValidator(
ReportSuccess(effective_test_name); ReportSuccess(effective_test_name);
} }
void ConformanceTestSuiteImpl::ExpectParseFailureForJson( void BinaryAndJsonConformanceSuite::ExpectParseFailureForJson(
const string& test_name, ConformanceLevel level, const string& input_json) { const string& test_name, ConformanceLevel level, const string& input_json) {
TestAllTypesProto3 prototype; TestAllTypesProto3 prototype;
// We don't expect output, but if the program erroneously accepts the protobuf // We don't expect output, but if the program erroneously accepts the protobuf
...@@ -452,7 +465,7 @@ void ConformanceTestSuiteImpl::ExpectParseFailureForJson( ...@@ -452,7 +465,7 @@ void ConformanceTestSuiteImpl::ExpectParseFailureForJson(
} }
} }
void ConformanceTestSuiteImpl::ExpectSerializeFailureForJson( void BinaryAndJsonConformanceSuite::ExpectSerializeFailureForJson(
const string& test_name, ConformanceLevel level, const string& text_format) { const string& test_name, ConformanceLevel level, const string& text_format) {
TestAllTypesProto3 payload_message; TestAllTypesProto3 payload_message;
GOOGLE_CHECK( GOOGLE_CHECK(
...@@ -482,7 +495,7 @@ void ConformanceTestSuiteImpl::ExpectSerializeFailureForJson( ...@@ -482,7 +495,7 @@ void ConformanceTestSuiteImpl::ExpectSerializeFailureForJson(
} }
//TODO: proto2? //TODO: proto2?
void ConformanceTestSuiteImpl::TestPrematureEOFForType( void BinaryAndJsonConformanceSuite::TestPrematureEOFForType(
FieldDescriptor::Type type) { FieldDescriptor::Type type) {
// Incomplete values for each wire type. // Incomplete values for each wire type.
static const string incompletes[6] = { static const string incompletes[6] = {
...@@ -570,7 +583,7 @@ void ConformanceTestSuiteImpl::TestPrematureEOFForType( ...@@ -570,7 +583,7 @@ void ConformanceTestSuiteImpl::TestPrematureEOFForType(
} }
} }
void ConformanceTestSuiteImpl::TestValidDataForType( void BinaryAndJsonConformanceSuite::TestValidDataForType(
FieldDescriptor::Type type, FieldDescriptor::Type type,
std::vector<std::pair<std::string, std::string>> values) { std::vector<std::pair<std::string, std::string>> values) {
for (int is_proto3 = 0; is_proto3 < 2; is_proto3++) { for (int is_proto3 = 0; is_proto3 < 2; is_proto3++) {
...@@ -606,7 +619,7 @@ void ConformanceTestSuiteImpl::TestValidDataForType( ...@@ -606,7 +619,7 @@ void ConformanceTestSuiteImpl::TestValidDataForType(
} }
// TODO: proto2? // TODO: proto2?
void ConformanceTestSuiteImpl::TestIllegalTags() { void BinaryAndJsonConformanceSuite::TestIllegalTags() {
// field num 0 is illegal // field num 0 is illegal
string nullfield[] = { string nullfield[] = {
"\1DEADBEEF", "\1DEADBEEF",
...@@ -621,7 +634,7 @@ void ConformanceTestSuiteImpl::TestIllegalTags() { ...@@ -621,7 +634,7 @@ void ConformanceTestSuiteImpl::TestIllegalTags() {
} }
} }
template <class MessageType> template <class MessageType>
void ConformanceTestSuiteImpl::TestOneofMessage ( void BinaryAndJsonConformanceSuite::TestOneofMessage (
MessageType &message, bool is_proto3) { MessageType &message, bool is_proto3) {
message.set_oneof_uint32(0); message.set_oneof_uint32(0);
RunValidProtobufTestWithMessage( RunValidProtobufTestWithMessage(
...@@ -660,14 +673,14 @@ void ConformanceTestSuiteImpl::TestOneofMessage ( ...@@ -660,14 +673,14 @@ void ConformanceTestSuiteImpl::TestOneofMessage (
} }
template <class MessageType> template <class MessageType>
void ConformanceTestSuiteImpl::TestUnknownMessage( void BinaryAndJsonConformanceSuite::TestUnknownMessage(
MessageType& message, bool is_proto3) { MessageType& message, bool is_proto3) {
message.ParseFromString("\xA8\x1F\x01"); message.ParseFromString("\xA8\x1F\x01");
RunValidBinaryProtobufTest("UnknownVarint", REQUIRED, RunValidBinaryProtobufTest("UnknownVarint", REQUIRED,
message.SerializeAsString(), is_proto3); message.SerializeAsString(), is_proto3);
} }
void ConformanceTestSuiteImpl::RunSuiteImpl() { void BinaryAndJsonConformanceSuite::RunSuiteImpl() {
type_resolver_.reset(NewTypeResolverForDescriptorPool( type_resolver_.reset(NewTypeResolverForDescriptorPool(
kTypeUrlPrefix, DescriptorPool::generated_pool())); kTypeUrlPrefix, DescriptorPool::generated_pool()));
type_url_ = GetTypeUrl(TestAllTypesProto3::descriptor()); type_url_ = GetTypeUrl(TestAllTypesProto3::descriptor());
...@@ -2377,8 +2390,3 @@ void ConformanceTestSuiteImpl::RunSuiteImpl() { ...@@ -2377,8 +2390,3 @@ void ConformanceTestSuiteImpl::RunSuiteImpl() {
} // namespace protobuf } // namespace protobuf
} // namespace google } // namespace google
int main(int argc, char *argv[]) {
google::protobuf::ConformanceTestSuiteImpl suite;
return google::protobuf::ForkPipeRunner::Run(argc, argv, &suite);
}
// 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 CONFORMANCE_BINARY_JSON_CONFORMANCE_SUITE_H
#define CONFORMANCE_BINARY_JSON_CONFORMANCE_SUITE_H
#include "conformance_test.h"
#include "third_party/jsoncpp/json.h"
namespace google {
namespace protobuf {
class BinaryAndJsonConformanceSuite : public ConformanceTestSuite {
public:
BinaryAndJsonConformanceSuite() {}
private:
void RunSuiteImpl();
void RunValidJsonTest(const string& test_name,
ConformanceLevel level,
const string& input_json,
const string& equivalent_text_format);
void RunValidJsonTestWithProtobufInput(
const string& test_name,
ConformanceLevel level,
const protobuf_test_messages::proto3::TestAllTypesProto3& input,
const string& equivalent_text_format);
void RunValidJsonIgnoreUnknownTest(
const string& test_name, ConformanceLevel level, const string& input_json,
const string& equivalent_text_format);
void RunValidProtobufTest(const string& test_name, ConformanceLevel level,
const string& input_protobuf,
const string& equivalent_text_format,
bool is_proto3);
void RunValidBinaryProtobufTest(const string& test_name,
ConformanceLevel level,
const string& input_protobuf,
bool is_proto3);
void RunValidProtobufTestWithMessage(
const string& test_name, ConformanceLevel level,
const Message *input,
const string& equivalent_text_format,
bool is_proto3);
bool ParseJsonResponse(
const conformance::ConformanceResponse& response,
Message* test_message);
bool ParseResponse(
const conformance::ConformanceResponse& response,
const ConformanceRequestSetting& setting,
Message* test_message) override;
typedef std::function<bool(const Json::Value&)> Validator;
void RunValidJsonTestWithValidator(const string& test_name,
ConformanceLevel level,
const string& input_json,
const Validator& validator);
void ExpectParseFailureForJson(const string& test_name,
ConformanceLevel level,
const string& input_json);
void ExpectSerializeFailureForJson(const string& test_name,
ConformanceLevel level,
const string& text_format);
void ExpectParseFailureForProtoWithProtoVersion (const string& proto,
const string& test_name,
ConformanceLevel level,
bool is_proto3);
void ExpectParseFailureForProto(const std::string& proto,
const std::string& test_name,
ConformanceLevel level);
void ExpectHardParseFailureForProto(const std::string& proto,
const std::string& test_name,
ConformanceLevel level);
void TestPrematureEOFForType(google::protobuf::FieldDescriptor::Type type);
void TestIllegalTags();
template <class MessageType>
void TestOneofMessage (MessageType &message,
bool is_proto3);
template <class MessageType>
void TestUnknownMessage (MessageType &message,
bool is_proto3);
void TestValidDataForType(
google::protobuf::FieldDescriptor::Type,
std::vector<std::pair<std::string, std::string>> values);
std::unique_ptr<google::protobuf::util::TypeResolver>
type_resolver_;
std::string type_url_;
};
} // namespace protobuf
} // namespace google
#endif // CONFORMANCE_BINARY_JSON_CONFORMANCE_SUITE_H
...@@ -55,6 +55,7 @@ enum WireFormat { ...@@ -55,6 +55,7 @@ enum WireFormat {
UNSPECIFIED = 0; UNSPECIFIED = 0;
PROTOBUF = 1; PROTOBUF = 1;
JSON = 2; JSON = 2;
JSPB = 3; // Google internal only. Opensource testees just skip it.
} }
enum TestCategory { enum TestCategory {
...@@ -67,6 +68,8 @@ enum TestCategory { ...@@ -67,6 +68,8 @@ enum TestCategory {
// https://developers.google.com/protocol-buffers/docs/proto3#json_options // https://developers.google.com/protocol-buffers/docs/proto3#json_options
// for more detail. // for more detail.
JSON_IGNORE_UNKNOWN_PARSING_TEST = 3; JSON_IGNORE_UNKNOWN_PARSING_TEST = 3;
JSPB_TEST = 4; // Test jspb wire format. Google internal only.
// Opensource testees just skip it.
} }
// Represents a single test case's input. The testee should: // Represents a single test case's input. The testee should:
...@@ -85,6 +88,8 @@ message ConformanceRequest { ...@@ -85,6 +88,8 @@ message ConformanceRequest {
oneof payload { oneof payload {
bytes protobuf_payload = 1; bytes protobuf_payload = 1;
string json_payload = 2; string json_payload = 2;
string jspb_payload = 7; // Google internal only.
// Opensource testees just skip it.
} }
// Which format should the testee serialize its message to? // Which format should the testee serialize its message to?
...@@ -99,6 +104,9 @@ message ConformanceRequest { ...@@ -99,6 +104,9 @@ message ConformanceRequest {
// spedific support in testee programs. Refer to the defintion of TestCategory // spedific support in testee programs. Refer to the defintion of TestCategory
// for more information. // for more information.
TestCategory test_category = 5; TestCategory test_category = 5;
// Specify details for how to encode jspb.
JspbEncodingConfig jspb_encoding_options = 6;
} }
// Represents a single test case's output. // Represents a single test case's output.
...@@ -132,5 +140,16 @@ message ConformanceResponse { ...@@ -132,5 +140,16 @@ message ConformanceResponse {
// For when the testee skipped the test, likely because a certain feature // For when the testee skipped the test, likely because a certain feature
// wasn't supported, like JSON input/output. // wasn't supported, like JSON input/output.
string skipped = 5; string skipped = 5;
// If the input was successfully parsed and the requested output was JSPB,
// serialize to JSPB and set it in this field. JSPB is google internal only
// format. Opensource testees can just skip it.
string jspb_payload = 7;
} }
} }
// Encoding options for jspb format.
message JspbEncodingConfig {
// Encode the value field of Any as jspb array if ture, otherwise binary.
bool use_jspb_array_any_format = 1;
}
...@@ -137,6 +137,11 @@ void DoTest(const ConformanceRequest& request, ConformanceResponse* response) { ...@@ -137,6 +137,11 @@ void DoTest(const ConformanceRequest& request, ConformanceResponse* response) {
case ConformanceRequest::PAYLOAD_NOT_SET: case ConformanceRequest::PAYLOAD_NOT_SET:
GOOGLE_LOG(FATAL) << "Request didn't have payload."; GOOGLE_LOG(FATAL) << "Request didn't have payload.";
break; break;
default:
GOOGLE_LOG(FATAL) << "unknown payload type: "
<< request.payload_case();
break;
} }
switch (request.requested_output_format()) { switch (request.requested_output_format()) {
......
...@@ -3,6 +3,7 @@ ...@@ -3,6 +3,7 @@
require_once("Conformance/WireFormat.php"); require_once("Conformance/WireFormat.php");
require_once("Conformance/ConformanceResponse.php"); require_once("Conformance/ConformanceResponse.php");
require_once("Conformance/ConformanceRequest.php"); require_once("Conformance/ConformanceRequest.php");
require_once("Conformance/JspbEncodingConfig.php");
require_once("Conformance/TestCategory.php"); require_once("Conformance/TestCategory.php");
require_once("Protobuf_test_messages/Proto3/ForeignMessage.php"); require_once("Protobuf_test_messages/Proto3/ForeignMessage.php");
require_once("Protobuf_test_messages/Proto3/ForeignEnum.php"); require_once("Protobuf_test_messages/Proto3/ForeignEnum.php");
......
...@@ -80,6 +80,11 @@ ConformanceTestSuite::ConformanceRequestSetting::ConformanceRequestSetting( ...@@ -80,6 +80,11 @@ ConformanceTestSuite::ConformanceRequestSetting::ConformanceRequestSetting(
break; break;
} }
case conformance::JSPB: {
request_.set_jspb_payload(input);
break;
}
default: default:
GOOGLE_LOG(FATAL) << "Unspecified input format"; GOOGLE_LOG(FATAL) << "Unspecified input format";
} }
...@@ -215,23 +220,27 @@ void ConformanceTestSuite::RunValidInputTest( ...@@ -215,23 +220,27 @@ void ConformanceTestSuite::RunValidInputTest(
void ConformanceTestSuite::RunValidBinaryInputTest( void ConformanceTestSuite::RunValidBinaryInputTest(
const ConformanceRequestSetting& setting, const ConformanceRequestSetting& setting,
const string& equivalent_wire_format) { const string& equivalent_wire_format) {
const ConformanceRequest& request = setting.GetRequest();
ConformanceResponse response;
RunTest(setting.GetTestName(), request, &response);
VerifyResponse(setting, equivalent_wire_format, response, true);
}
void ConformanceTestSuite::VerifyResponse(
const ConformanceRequestSetting& setting,
const string& equivalent_wire_format,
const ConformanceResponse& response,
bool need_report_success) {
Message* test_message = setting.GetTestMessage();
const ConformanceRequest& request = setting.GetRequest();
const string& test_name = setting.GetTestName(); const string& test_name = setting.GetTestName();
ConformanceLevel level = setting.GetLevel(); ConformanceLevel level = setting.GetLevel();
Message* reference_message = setting.GetTestMessage(); Message* reference_message = setting.GetTestMessage();
GOOGLE_CHECK( GOOGLE_CHECK(
reference_message->ParseFromString(equivalent_wire_format)) reference_message->ParseFromString(equivalent_wire_format))
<< "Failed to parse wire data for test case: " << test_name; << "Failed to parse wire data for test case: " << test_name;
const ConformanceRequest& request = setting.GetRequest();
ConformanceResponse response;
RunTest(test_name, request, &response);
Message* test_message = setting.GetTestMessage();
WireFormat requested_output = request.requested_output_format();
switch (response.result_case()) { switch (response.result_case()) {
case ConformanceResponse::RESULT_NOT_SET: case ConformanceResponse::RESULT_NOT_SET:
ReportFailure(test_name, level, request, response, ReportFailure(test_name, level, request, response,
...@@ -249,53 +258,8 @@ void ConformanceTestSuite::RunValidBinaryInputTest( ...@@ -249,53 +258,8 @@ void ConformanceTestSuite::RunValidBinaryInputTest(
ReportSkip(test_name, request, response); ReportSkip(test_name, request, response);
return; return;
case ConformanceResponse::kJsonPayload: {
if (requested_output != conformance::JSON) {
ReportFailure(
test_name, level, request, response,
"Test was asked for protobuf output but provided JSON instead.");
return;
}
string binary_protobuf;
Status status =
JsonToBinaryString(type_resolver_.get(), type_url_,
response.json_payload(), &binary_protobuf);
if (!status.ok()) {
ReportFailure(test_name, level, request, response,
"JSON output we received from test was unparseable.");
return;
}
if (!test_message->ParseFromString(binary_protobuf)) {
ReportFailure(test_name, level, request, response,
"INTERNAL ERROR: internal JSON->protobuf transcode "
"yielded unparseable proto.");
return;
}
break;
}
case ConformanceResponse::kProtobufPayload: {
if (requested_output != conformance::PROTOBUF) {
ReportFailure(
test_name, level, request, response,
"Test was asked for JSON output but provided protobuf instead.");
return;
}
if (!test_message->ParseFromString(response.protobuf_payload())) {
ReportFailure(test_name, level, request, response,
"Protobuf output we received from test was unparseable.");
return;
}
break;
}
default: default:
GOOGLE_LOG(FATAL) << test_name << ": unknown payload type: " if (!ParseResponse(response, setting, test_message)) return;
<< response.result_case();
} }
MessageDifferencer differencer; MessageDifferencer differencer;
...@@ -308,7 +272,9 @@ void ConformanceTestSuite::RunValidBinaryInputTest( ...@@ -308,7 +272,9 @@ void ConformanceTestSuite::RunValidBinaryInputTest(
bool check; bool check;
check = differencer.Compare(*reference_message, *test_message); check = differencer.Compare(*reference_message, *test_message);
if (check) { if (check) {
ReportSuccess(test_name); if (need_report_success) {
ReportSuccess(test_name);
}
} else { } else {
ReportFailure(test_name, level, request, response, ReportFailure(test_name, level, request, response,
"Output was not equivalent to reference message: %s.", "Output was not equivalent to reference message: %s.",
...@@ -375,6 +341,24 @@ bool ConformanceTestSuite::CheckSetEmpty( ...@@ -375,6 +341,24 @@ bool ConformanceTestSuite::CheckSetEmpty(
} }
} }
string ConformanceTestSuite::WireFormatToString(
WireFormat wire_format) {
switch (wire_format) {
case conformance::PROTOBUF:
return "PROTOBUF";
case conformance::JSON:
return "JSON";
case conformance::JSPB:
return "JSPB";
case conformance::UNSPECIFIED:
return "UNSPECIFIED";
default:
GOOGLE_LOG(FATAL) << "unknown wire type: "
<< wire_format;
}
return "";
}
bool ConformanceTestSuite::RunSuite( bool ConformanceTestSuite::RunSuite(
ConformanceTestRunner* runner, std::string* output) { ConformanceTestRunner* runner, std::string* output) {
runner_ = runner; runner_ = runner;
......
...@@ -87,7 +87,6 @@ class ForkPipeRunner : public ConformanceTestRunner { ...@@ -87,7 +87,6 @@ class ForkPipeRunner : public ConformanceTestRunner {
static int Run(int argc, char *argv[], static int Run(int argc, char *argv[],
ConformanceTestSuite* suite); ConformanceTestSuite* suite);
private:
ForkPipeRunner(const std::string &executable) ForkPipeRunner(const std::string &executable)
: child_pid_(-1), executable_(executable) {} : child_pid_(-1), executable_(executable) {}
...@@ -97,24 +96,7 @@ class ForkPipeRunner : public ConformanceTestRunner { ...@@ -97,24 +96,7 @@ class ForkPipeRunner : public ConformanceTestRunner {
const std::string& request, const std::string& request,
std::string* response); std::string* response);
// TODO(haberman): make this work on Windows, instead of using these private:
// UNIX-specific APIs.
//
// There is a platform-agnostic API in
// src/google/protobuf/compiler/subprocess.h
//
// However that API only supports sending a single message to the subprocess.
// We really want to be able to send messages and receive responses one at a
// time:
//
// 1. Spawning a new process for each test would take way too long for thousands
// of tests and subprocesses like java that can take 100ms or more to start
// up.
//
// 2. Sending all the tests in one big message and receiving all results in one
// big message would take away our visibility about which test(s) caused a
// crash or other fatal error. It would also give us only a single failure
// instead of all of them.
void SpawnTestProgram(); void SpawnTestProgram();
void CheckedWrite(int fd, const void *buf, size_t len); void CheckedWrite(int fd, const void *buf, size_t len);
...@@ -237,6 +219,7 @@ class ConformanceTestSuite { ...@@ -237,6 +219,7 @@ class ConformanceTestSuite {
protected: protected:
virtual string InputFormatString(conformance::WireFormat format) const; virtual string InputFormatString(conformance::WireFormat format) const;
virtual string OutputFormatString(conformance::WireFormat format) const; virtual string OutputFormatString(conformance::WireFormat format) const;
conformance::ConformanceRequest request_;
private: private:
ConformanceLevel level_; ConformanceLevel level_;
...@@ -244,11 +227,24 @@ class ConformanceTestSuite { ...@@ -244,11 +227,24 @@ class ConformanceTestSuite {
::conformance::WireFormat output_format_; ::conformance::WireFormat output_format_;
const Message& prototype_message_; const Message& prototype_message_;
string test_name_; string test_name_;
conformance::ConformanceRequest request_;
}; };
bool CheckSetEmpty(const std::set<string>& set_to_check, bool CheckSetEmpty(const std::set<string>& set_to_check,
const std::string& write_to_file, const std::string& msg); const std::string& write_to_file, const std::string& msg);
string WireFormatToString(conformance::WireFormat wire_format);
// Parse payload in the response to the given message. Returns true on
// success.
virtual bool ParseResponse(
const conformance::ConformanceResponse& response,
const ConformanceRequestSetting& setting,
Message* test_message) = 0;
void VerifyResponse(
const ConformanceRequestSetting& setting,
const string& equivalent_wire_format,
const conformance::ConformanceResponse& response,
bool need_report_success);
void ReportSuccess(const std::string& test_name); void ReportSuccess(const std::string& test_name);
void ReportFailure(const string& test_name, void ReportFailure(const string& test_name,
...@@ -295,10 +291,6 @@ class ConformanceTestSuite { ...@@ -295,10 +291,6 @@ class ConformanceTestSuite {
// The set of tests that the testee opted out of; // The set of tests that the testee opted out of;
std::set<std::string> skipped_; std::set<std::string> skipped_;
std::unique_ptr<google::protobuf::util::TypeResolver>
type_resolver_;
std::string type_url_;
}; };
} // namespace protobuf } // namespace protobuf
......
...@@ -212,6 +212,24 @@ int ForkPipeRunner::Run( ...@@ -212,6 +212,24 @@ int ForkPipeRunner::Run(
return ok ? EXIT_SUCCESS : EXIT_FAILURE; return ok ? EXIT_SUCCESS : EXIT_FAILURE;
} }
// TODO(haberman): make this work on Windows, instead of using these
// UNIX-specific APIs.
//
// There is a platform-agnostic API in
// src/google/protobuf/compiler/subprocess.h
//
// However that API only supports sending a single message to the subprocess.
// We really want to be able to send messages and receive responses one at a
// time:
//
// 1. Spawning a new process for each test would take way too long for thousands
// of tests and subprocesses like java that can take 100ms or more to start
// up.
//
// 2. Sending all the tests in one big message and receiving all results in one
// big message would take away our visibility about which test(s) caused a
// crash or other fatal error. It would also give us only a single failure
// instead of all of them.
void ForkPipeRunner::SpawnTestProgram() { void ForkPipeRunner::SpawnTestProgram() {
int toproc_pipe_fd[2]; int toproc_pipe_fd[2];
int fromproc_pipe_fd[2]; int fromproc_pipe_fd[2];
......
...@@ -24,26 +24,31 @@ namespace Conformance { ...@@ -24,26 +24,31 @@ namespace Conformance {
static ConformanceReflection() { static ConformanceReflection() {
byte[] descriptorData = global::System.Convert.FromBase64String( byte[] descriptorData = global::System.Convert.FromBase64String(
string.Concat( string.Concat(
"ChFjb25mb3JtYW5jZS5wcm90bxILY29uZm9ybWFuY2Ui1QEKEkNvbmZvcm1h", "ChFjb25mb3JtYW5jZS5wcm90bxILY29uZm9ybWFuY2UirQIKEkNvbmZvcm1h",
"bmNlUmVxdWVzdBIaChBwcm90b2J1Zl9wYXlsb2FkGAEgASgMSAASFgoManNv", "bmNlUmVxdWVzdBIaChBwcm90b2J1Zl9wYXlsb2FkGAEgASgMSAASFgoManNv",
"bl9wYXlsb2FkGAIgASgJSAASOAoXcmVxdWVzdGVkX291dHB1dF9mb3JtYXQY", "bl9wYXlsb2FkGAIgASgJSAASFgoManNwYl9wYXlsb2FkGAcgASgJSAASOAoX",
"AyABKA4yFy5jb25mb3JtYW5jZS5XaXJlRm9ybWF0EhQKDG1lc3NhZ2VfdHlw", "cmVxdWVzdGVkX291dHB1dF9mb3JtYXQYAyABKA4yFy5jb25mb3JtYW5jZS5X",
"ZRgEIAEoCRIwCg10ZXN0X2NhdGVnb3J5GAUgASgOMhkuY29uZm9ybWFuY2Uu", "aXJlRm9ybWF0EhQKDG1lc3NhZ2VfdHlwZRgEIAEoCRIwCg10ZXN0X2NhdGVn",
"VGVzdENhdGVnb3J5QgkKB3BheWxvYWQisQEKE0NvbmZvcm1hbmNlUmVzcG9u", "b3J5GAUgASgOMhkuY29uZm9ybWFuY2UuVGVzdENhdGVnb3J5Ej4KFWpzcGJf",
"c2USFQoLcGFyc2VfZXJyb3IYASABKAlIABIZCg9zZXJpYWxpemVfZXJyb3IY", "ZW5jb2Rpbmdfb3B0aW9ucxgGIAEoCzIfLmNvbmZvcm1hbmNlLkpzcGJFbmNv",
"BiABKAlIABIXCg1ydW50aW1lX2Vycm9yGAIgASgJSAASGgoQcHJvdG9idWZf", "ZGluZ0NvbmZpZ0IJCgdwYXlsb2FkIskBChNDb25mb3JtYW5jZVJlc3BvbnNl",
"cGF5bG9hZBgDIAEoDEgAEhYKDGpzb25fcGF5bG9hZBgEIAEoCUgAEhEKB3Nr", "EhUKC3BhcnNlX2Vycm9yGAEgASgJSAASGQoPc2VyaWFsaXplX2Vycm9yGAYg",
"aXBwZWQYBSABKAlIAEIICgZyZXN1bHQqNQoKV2lyZUZvcm1hdBIPCgtVTlNQ", "ASgJSAASFwoNcnVudGltZV9lcnJvchgCIAEoCUgAEhoKEHByb3RvYnVmX3Bh",
"RUNJRklFRBAAEgwKCFBST1RPQlVGEAESCAoESlNPThACKmoKDFRlc3RDYXRl", "eWxvYWQYAyABKAxIABIWCgxqc29uX3BheWxvYWQYBCABKAlIABIRCgdza2lw",
"Z29yeRIUChBVTlNQRUNJRklFRF9URVNUEAASDwoLQklOQVJZX1RFU1QQARIN", "cGVkGAUgASgJSAASFgoManNwYl9wYXlsb2FkGAcgASgJSABCCAoGcmVzdWx0",
"CglKU09OX1RFU1QQAhIkCiBKU09OX0lHTk9SRV9VTktOT1dOX1BBUlNJTkdf", "IjcKEkpzcGJFbmNvZGluZ0NvbmZpZxIhChl1c2VfanNwYl9hcnJheV9hbnlf",
"VEVTVBADQiEKH2NvbS5nb29nbGUucHJvdG9idWYuY29uZm9ybWFuY2ViBnBy", "Zm9ybWF0GAEgASgIKj8KCldpcmVGb3JtYXQSDwoLVU5TUEVDSUZJRUQQABIM",
"b3RvMw==")); "CghQUk9UT0JVRhABEggKBEpTT04QAhIICgRKU1BCEAMqeQoMVGVzdENhdGVn",
"b3J5EhQKEFVOU1BFQ0lGSUVEX1RFU1QQABIPCgtCSU5BUllfVEVTVBABEg0K",
"CUpTT05fVEVTVBACEiQKIEpTT05fSUdOT1JFX1VOS05PV05fUEFSU0lOR19U",
"RVNUEAMSDQoJSlNQQl9URVNUEARCIQofY29tLmdvb2dsZS5wcm90b2J1Zi5j",
"b25mb3JtYW5jZWIGcHJvdG8z"));
descriptor = pbr::FileDescriptor.FromGeneratedCode(descriptorData, descriptor = pbr::FileDescriptor.FromGeneratedCode(descriptorData,
new pbr::FileDescriptor[] { }, new pbr::FileDescriptor[] { },
new pbr::GeneratedClrTypeInfo(new[] {typeof(global::Conformance.WireFormat), typeof(global::Conformance.TestCategory), }, new pbr::GeneratedClrTypeInfo[] { new pbr::GeneratedClrTypeInfo(new[] {typeof(global::Conformance.WireFormat), typeof(global::Conformance.TestCategory), }, new pbr::GeneratedClrTypeInfo[] {
new pbr::GeneratedClrTypeInfo(typeof(global::Conformance.ConformanceRequest), global::Conformance.ConformanceRequest.Parser, new[]{ "ProtobufPayload", "JsonPayload", "RequestedOutputFormat", "MessageType", "TestCategory" }, new[]{ "Payload" }, null, null), new pbr::GeneratedClrTypeInfo(typeof(global::Conformance.ConformanceRequest), global::Conformance.ConformanceRequest.Parser, new[]{ "ProtobufPayload", "JsonPayload", "JspbPayload", "RequestedOutputFormat", "MessageType", "TestCategory", "JspbEncodingOptions" }, new[]{ "Payload" }, null, null),
new pbr::GeneratedClrTypeInfo(typeof(global::Conformance.ConformanceResponse), global::Conformance.ConformanceResponse.Parser, new[]{ "ParseError", "SerializeError", "RuntimeError", "ProtobufPayload", "JsonPayload", "Skipped" }, new[]{ "Result" }, null, null) new pbr::GeneratedClrTypeInfo(typeof(global::Conformance.ConformanceResponse), global::Conformance.ConformanceResponse.Parser, new[]{ "ParseError", "SerializeError", "RuntimeError", "ProtobufPayload", "JsonPayload", "Skipped", "JspbPayload" }, new[]{ "Result" }, null, null),
new pbr::GeneratedClrTypeInfo(typeof(global::Conformance.JspbEncodingConfig), global::Conformance.JspbEncodingConfig.Parser, new[]{ "UseJspbArrayAnyFormat" }, null, null, null)
})); }));
} }
#endregion #endregion
...@@ -54,6 +59,10 @@ namespace Conformance { ...@@ -54,6 +59,10 @@ namespace Conformance {
[pbr::OriginalName("UNSPECIFIED")] Unspecified = 0, [pbr::OriginalName("UNSPECIFIED")] Unspecified = 0,
[pbr::OriginalName("PROTOBUF")] Protobuf = 1, [pbr::OriginalName("PROTOBUF")] Protobuf = 1,
[pbr::OriginalName("JSON")] Json = 2, [pbr::OriginalName("JSON")] Json = 2,
/// <summary>
/// Google internal only. Opensource testees just skip it.
/// </summary>
[pbr::OriginalName("JSPB")] Jspb = 3,
} }
public enum TestCategory { public enum TestCategory {
...@@ -74,6 +83,10 @@ namespace Conformance { ...@@ -74,6 +83,10 @@ namespace Conformance {
/// for more detail. /// for more detail.
/// </summary> /// </summary>
[pbr::OriginalName("JSON_IGNORE_UNKNOWN_PARSING_TEST")] JsonIgnoreUnknownParsingTest = 3, [pbr::OriginalName("JSON_IGNORE_UNKNOWN_PARSING_TEST")] JsonIgnoreUnknownParsingTest = 3,
/// <summary>
/// Test jspb wire format. Google internal only.
/// </summary>
[pbr::OriginalName("JSPB_TEST")] JspbTest = 4,
} }
#endregion #endregion
...@@ -114,6 +127,7 @@ namespace Conformance { ...@@ -114,6 +127,7 @@ namespace Conformance {
requestedOutputFormat_ = other.requestedOutputFormat_; requestedOutputFormat_ = other.requestedOutputFormat_;
messageType_ = other.messageType_; messageType_ = other.messageType_;
testCategory_ = other.testCategory_; testCategory_ = other.testCategory_;
jspbEncodingOptions_ = other.jspbEncodingOptions_ != null ? other.jspbEncodingOptions_.Clone() : null;
switch (other.PayloadCase) { switch (other.PayloadCase) {
case PayloadOneofCase.ProtobufPayload: case PayloadOneofCase.ProtobufPayload:
ProtobufPayload = other.ProtobufPayload; ProtobufPayload = other.ProtobufPayload;
...@@ -121,6 +135,9 @@ namespace Conformance { ...@@ -121,6 +135,9 @@ namespace Conformance {
case PayloadOneofCase.JsonPayload: case PayloadOneofCase.JsonPayload:
JsonPayload = other.JsonPayload; JsonPayload = other.JsonPayload;
break; break;
case PayloadOneofCase.JspbPayload:
JspbPayload = other.JspbPayload;
break;
} }
_unknownFields = pb::UnknownFieldSet.Clone(other._unknownFields); _unknownFields = pb::UnknownFieldSet.Clone(other._unknownFields);
...@@ -153,6 +170,20 @@ namespace Conformance { ...@@ -153,6 +170,20 @@ namespace Conformance {
} }
} }
/// <summary>Field number for the "jspb_payload" field.</summary>
public const int JspbPayloadFieldNumber = 7;
/// <summary>
/// Google internal only.
/// </summary>
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public string JspbPayload {
get { return payloadCase_ == PayloadOneofCase.JspbPayload ? (string) payload_ : ""; }
set {
payload_ = pb::ProtoPreconditions.CheckNotNull(value, "value");
payloadCase_ = PayloadOneofCase.JspbPayload;
}
}
/// <summary>Field number for the "requested_output_format" field.</summary> /// <summary>Field number for the "requested_output_format" field.</summary>
public const int RequestedOutputFormatFieldNumber = 3; public const int RequestedOutputFormatFieldNumber = 3;
private global::Conformance.WireFormat requestedOutputFormat_ = 0; private global::Conformance.WireFormat requestedOutputFormat_ = 0;
...@@ -199,12 +230,27 @@ namespace Conformance { ...@@ -199,12 +230,27 @@ namespace Conformance {
} }
} }
/// <summary>Field number for the "jspb_encoding_options" field.</summary>
public const int JspbEncodingOptionsFieldNumber = 6;
private global::Conformance.JspbEncodingConfig jspbEncodingOptions_;
/// <summary>
/// Specify details for how to encode jspb.
/// </summary>
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public global::Conformance.JspbEncodingConfig JspbEncodingOptions {
get { return jspbEncodingOptions_; }
set {
jspbEncodingOptions_ = value;
}
}
private object payload_; private object payload_;
/// <summary>Enum of possible cases for the "payload" oneof.</summary> /// <summary>Enum of possible cases for the "payload" oneof.</summary>
public enum PayloadOneofCase { public enum PayloadOneofCase {
None = 0, None = 0,
ProtobufPayload = 1, ProtobufPayload = 1,
JsonPayload = 2, JsonPayload = 2,
JspbPayload = 7,
} }
private PayloadOneofCase payloadCase_ = PayloadOneofCase.None; private PayloadOneofCase payloadCase_ = PayloadOneofCase.None;
[global::System.Diagnostics.DebuggerNonUserCodeAttribute] [global::System.Diagnostics.DebuggerNonUserCodeAttribute]
...@@ -233,9 +279,11 @@ namespace Conformance { ...@@ -233,9 +279,11 @@ namespace Conformance {
} }
if (ProtobufPayload != other.ProtobufPayload) return false; if (ProtobufPayload != other.ProtobufPayload) return false;
if (JsonPayload != other.JsonPayload) return false; if (JsonPayload != other.JsonPayload) return false;
if (JspbPayload != other.JspbPayload) return false;
if (RequestedOutputFormat != other.RequestedOutputFormat) return false; if (RequestedOutputFormat != other.RequestedOutputFormat) return false;
if (MessageType != other.MessageType) return false; if (MessageType != other.MessageType) return false;
if (TestCategory != other.TestCategory) return false; if (TestCategory != other.TestCategory) return false;
if (!object.Equals(JspbEncodingOptions, other.JspbEncodingOptions)) return false;
if (PayloadCase != other.PayloadCase) return false; if (PayloadCase != other.PayloadCase) return false;
return Equals(_unknownFields, other._unknownFields); return Equals(_unknownFields, other._unknownFields);
} }
...@@ -245,9 +293,11 @@ namespace Conformance { ...@@ -245,9 +293,11 @@ namespace Conformance {
int hash = 1; int hash = 1;
if (payloadCase_ == PayloadOneofCase.ProtobufPayload) hash ^= ProtobufPayload.GetHashCode(); if (payloadCase_ == PayloadOneofCase.ProtobufPayload) hash ^= ProtobufPayload.GetHashCode();
if (payloadCase_ == PayloadOneofCase.JsonPayload) hash ^= JsonPayload.GetHashCode(); if (payloadCase_ == PayloadOneofCase.JsonPayload) hash ^= JsonPayload.GetHashCode();
if (payloadCase_ == PayloadOneofCase.JspbPayload) hash ^= JspbPayload.GetHashCode();
if (RequestedOutputFormat != 0) hash ^= RequestedOutputFormat.GetHashCode(); if (RequestedOutputFormat != 0) hash ^= RequestedOutputFormat.GetHashCode();
if (MessageType.Length != 0) hash ^= MessageType.GetHashCode(); if (MessageType.Length != 0) hash ^= MessageType.GetHashCode();
if (TestCategory != 0) hash ^= TestCategory.GetHashCode(); if (TestCategory != 0) hash ^= TestCategory.GetHashCode();
if (jspbEncodingOptions_ != null) hash ^= JspbEncodingOptions.GetHashCode();
hash ^= (int) payloadCase_; hash ^= (int) payloadCase_;
if (_unknownFields != null) { if (_unknownFields != null) {
hash ^= _unknownFields.GetHashCode(); hash ^= _unknownFields.GetHashCode();
...@@ -282,6 +332,14 @@ namespace Conformance { ...@@ -282,6 +332,14 @@ namespace Conformance {
output.WriteRawTag(40); output.WriteRawTag(40);
output.WriteEnum((int) TestCategory); output.WriteEnum((int) TestCategory);
} }
if (jspbEncodingOptions_ != null) {
output.WriteRawTag(50);
output.WriteMessage(JspbEncodingOptions);
}
if (payloadCase_ == PayloadOneofCase.JspbPayload) {
output.WriteRawTag(58);
output.WriteString(JspbPayload);
}
if (_unknownFields != null) { if (_unknownFields != null) {
_unknownFields.WriteTo(output); _unknownFields.WriteTo(output);
} }
...@@ -296,6 +354,9 @@ namespace Conformance { ...@@ -296,6 +354,9 @@ namespace Conformance {
if (payloadCase_ == PayloadOneofCase.JsonPayload) { if (payloadCase_ == PayloadOneofCase.JsonPayload) {
size += 1 + pb::CodedOutputStream.ComputeStringSize(JsonPayload); size += 1 + pb::CodedOutputStream.ComputeStringSize(JsonPayload);
} }
if (payloadCase_ == PayloadOneofCase.JspbPayload) {
size += 1 + pb::CodedOutputStream.ComputeStringSize(JspbPayload);
}
if (RequestedOutputFormat != 0) { if (RequestedOutputFormat != 0) {
size += 1 + pb::CodedOutputStream.ComputeEnumSize((int) RequestedOutputFormat); size += 1 + pb::CodedOutputStream.ComputeEnumSize((int) RequestedOutputFormat);
} }
...@@ -305,6 +366,9 @@ namespace Conformance { ...@@ -305,6 +366,9 @@ namespace Conformance {
if (TestCategory != 0) { if (TestCategory != 0) {
size += 1 + pb::CodedOutputStream.ComputeEnumSize((int) TestCategory); size += 1 + pb::CodedOutputStream.ComputeEnumSize((int) TestCategory);
} }
if (jspbEncodingOptions_ != null) {
size += 1 + pb::CodedOutputStream.ComputeMessageSize(JspbEncodingOptions);
}
if (_unknownFields != null) { if (_unknownFields != null) {
size += _unknownFields.CalculateSize(); size += _unknownFields.CalculateSize();
} }
...@@ -325,6 +389,12 @@ namespace Conformance { ...@@ -325,6 +389,12 @@ namespace Conformance {
if (other.TestCategory != 0) { if (other.TestCategory != 0) {
TestCategory = other.TestCategory; TestCategory = other.TestCategory;
} }
if (other.jspbEncodingOptions_ != null) {
if (jspbEncodingOptions_ == null) {
JspbEncodingOptions = new global::Conformance.JspbEncodingConfig();
}
JspbEncodingOptions.MergeFrom(other.JspbEncodingOptions);
}
switch (other.PayloadCase) { switch (other.PayloadCase) {
case PayloadOneofCase.ProtobufPayload: case PayloadOneofCase.ProtobufPayload:
ProtobufPayload = other.ProtobufPayload; ProtobufPayload = other.ProtobufPayload;
...@@ -332,6 +402,9 @@ namespace Conformance { ...@@ -332,6 +402,9 @@ namespace Conformance {
case PayloadOneofCase.JsonPayload: case PayloadOneofCase.JsonPayload:
JsonPayload = other.JsonPayload; JsonPayload = other.JsonPayload;
break; break;
case PayloadOneofCase.JspbPayload:
JspbPayload = other.JspbPayload;
break;
} }
_unknownFields = pb::UnknownFieldSet.MergeFrom(_unknownFields, other._unknownFields); _unknownFields = pb::UnknownFieldSet.MergeFrom(_unknownFields, other._unknownFields);
...@@ -367,6 +440,17 @@ namespace Conformance { ...@@ -367,6 +440,17 @@ namespace Conformance {
TestCategory = (global::Conformance.TestCategory) input.ReadEnum(); TestCategory = (global::Conformance.TestCategory) input.ReadEnum();
break; break;
} }
case 50: {
if (jspbEncodingOptions_ == null) {
JspbEncodingOptions = new global::Conformance.JspbEncodingConfig();
}
input.ReadMessage(JspbEncodingOptions);
break;
}
case 58: {
JspbPayload = input.ReadString();
break;
}
} }
} }
} }
...@@ -420,6 +504,9 @@ namespace Conformance { ...@@ -420,6 +504,9 @@ namespace Conformance {
case ResultOneofCase.Skipped: case ResultOneofCase.Skipped:
Skipped = other.Skipped; Skipped = other.Skipped;
break; break;
case ResultOneofCase.JspbPayload:
JspbPayload = other.JspbPayload;
break;
} }
_unknownFields = pb::UnknownFieldSet.Clone(other._unknownFields); _unknownFields = pb::UnknownFieldSet.Clone(other._unknownFields);
...@@ -525,6 +612,22 @@ namespace Conformance { ...@@ -525,6 +612,22 @@ namespace Conformance {
} }
} }
/// <summary>Field number for the "jspb_payload" field.</summary>
public const int JspbPayloadFieldNumber = 7;
/// <summary>
/// If the input was successfully parsed and the requested output was JSPB,
/// serialize to JSPB and set it in this field. JSPB is google internal only
/// format. Opensource testees can just skip it.
/// </summary>
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public string JspbPayload {
get { return resultCase_ == ResultOneofCase.JspbPayload ? (string) result_ : ""; }
set {
result_ = pb::ProtoPreconditions.CheckNotNull(value, "value");
resultCase_ = ResultOneofCase.JspbPayload;
}
}
private object result_; private object result_;
/// <summary>Enum of possible cases for the "result" oneof.</summary> /// <summary>Enum of possible cases for the "result" oneof.</summary>
public enum ResultOneofCase { public enum ResultOneofCase {
...@@ -535,6 +638,7 @@ namespace Conformance { ...@@ -535,6 +638,7 @@ namespace Conformance {
ProtobufPayload = 3, ProtobufPayload = 3,
JsonPayload = 4, JsonPayload = 4,
Skipped = 5, Skipped = 5,
JspbPayload = 7,
} }
private ResultOneofCase resultCase_ = ResultOneofCase.None; private ResultOneofCase resultCase_ = ResultOneofCase.None;
[global::System.Diagnostics.DebuggerNonUserCodeAttribute] [global::System.Diagnostics.DebuggerNonUserCodeAttribute]
...@@ -567,6 +671,7 @@ namespace Conformance { ...@@ -567,6 +671,7 @@ namespace Conformance {
if (ProtobufPayload != other.ProtobufPayload) return false; if (ProtobufPayload != other.ProtobufPayload) return false;
if (JsonPayload != other.JsonPayload) return false; if (JsonPayload != other.JsonPayload) return false;
if (Skipped != other.Skipped) return false; if (Skipped != other.Skipped) return false;
if (JspbPayload != other.JspbPayload) return false;
if (ResultCase != other.ResultCase) return false; if (ResultCase != other.ResultCase) return false;
return Equals(_unknownFields, other._unknownFields); return Equals(_unknownFields, other._unknownFields);
} }
...@@ -580,6 +685,7 @@ namespace Conformance { ...@@ -580,6 +685,7 @@ namespace Conformance {
if (resultCase_ == ResultOneofCase.ProtobufPayload) hash ^= ProtobufPayload.GetHashCode(); if (resultCase_ == ResultOneofCase.ProtobufPayload) hash ^= ProtobufPayload.GetHashCode();
if (resultCase_ == ResultOneofCase.JsonPayload) hash ^= JsonPayload.GetHashCode(); if (resultCase_ == ResultOneofCase.JsonPayload) hash ^= JsonPayload.GetHashCode();
if (resultCase_ == ResultOneofCase.Skipped) hash ^= Skipped.GetHashCode(); if (resultCase_ == ResultOneofCase.Skipped) hash ^= Skipped.GetHashCode();
if (resultCase_ == ResultOneofCase.JspbPayload) hash ^= JspbPayload.GetHashCode();
hash ^= (int) resultCase_; hash ^= (int) resultCase_;
if (_unknownFields != null) { if (_unknownFields != null) {
hash ^= _unknownFields.GetHashCode(); hash ^= _unknownFields.GetHashCode();
...@@ -618,6 +724,10 @@ namespace Conformance { ...@@ -618,6 +724,10 @@ namespace Conformance {
output.WriteRawTag(50); output.WriteRawTag(50);
output.WriteString(SerializeError); output.WriteString(SerializeError);
} }
if (resultCase_ == ResultOneofCase.JspbPayload) {
output.WriteRawTag(58);
output.WriteString(JspbPayload);
}
if (_unknownFields != null) { if (_unknownFields != null) {
_unknownFields.WriteTo(output); _unknownFields.WriteTo(output);
} }
...@@ -644,6 +754,9 @@ namespace Conformance { ...@@ -644,6 +754,9 @@ namespace Conformance {
if (resultCase_ == ResultOneofCase.Skipped) { if (resultCase_ == ResultOneofCase.Skipped) {
size += 1 + pb::CodedOutputStream.ComputeStringSize(Skipped); size += 1 + pb::CodedOutputStream.ComputeStringSize(Skipped);
} }
if (resultCase_ == ResultOneofCase.JspbPayload) {
size += 1 + pb::CodedOutputStream.ComputeStringSize(JspbPayload);
}
if (_unknownFields != null) { if (_unknownFields != null) {
size += _unknownFields.CalculateSize(); size += _unknownFields.CalculateSize();
} }
...@@ -674,6 +787,9 @@ namespace Conformance { ...@@ -674,6 +787,9 @@ namespace Conformance {
case ResultOneofCase.Skipped: case ResultOneofCase.Skipped:
Skipped = other.Skipped; Skipped = other.Skipped;
break; break;
case ResultOneofCase.JspbPayload:
JspbPayload = other.JspbPayload;
break;
} }
_unknownFields = pb::UnknownFieldSet.MergeFrom(_unknownFields, other._unknownFields); _unknownFields = pb::UnknownFieldSet.MergeFrom(_unknownFields, other._unknownFields);
...@@ -713,6 +829,147 @@ namespace Conformance { ...@@ -713,6 +829,147 @@ namespace Conformance {
SerializeError = input.ReadString(); SerializeError = input.ReadString();
break; break;
} }
case 58: {
JspbPayload = input.ReadString();
break;
}
}
}
}
}
/// <summary>
/// Encoding options for jspb format.
/// </summary>
public sealed partial class JspbEncodingConfig : pb::IMessage<JspbEncodingConfig> {
private static readonly pb::MessageParser<JspbEncodingConfig> _parser = new pb::MessageParser<JspbEncodingConfig>(() => new JspbEncodingConfig());
private pb::UnknownFieldSet _unknownFields;
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public static pb::MessageParser<JspbEncodingConfig> Parser { get { return _parser; } }
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public static pbr::MessageDescriptor Descriptor {
get { return global::Conformance.ConformanceReflection.Descriptor.MessageTypes[2]; }
}
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
pbr::MessageDescriptor pb::IMessage.Descriptor {
get { return Descriptor; }
}
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public JspbEncodingConfig() {
OnConstruction();
}
partial void OnConstruction();
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public JspbEncodingConfig(JspbEncodingConfig other) : this() {
useJspbArrayAnyFormat_ = other.useJspbArrayAnyFormat_;
_unknownFields = pb::UnknownFieldSet.Clone(other._unknownFields);
}
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public JspbEncodingConfig Clone() {
return new JspbEncodingConfig(this);
}
/// <summary>Field number for the "use_jspb_array_any_format" field.</summary>
public const int UseJspbArrayAnyFormatFieldNumber = 1;
private bool useJspbArrayAnyFormat_;
/// <summary>
/// Encode the value field of Any as jspb array if ture, otherwise binary.
/// </summary>
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public bool UseJspbArrayAnyFormat {
get { return useJspbArrayAnyFormat_; }
set {
useJspbArrayAnyFormat_ = value;
}
}
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public override bool Equals(object other) {
return Equals(other as JspbEncodingConfig);
}
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public bool Equals(JspbEncodingConfig other) {
if (ReferenceEquals(other, null)) {
return false;
}
if (ReferenceEquals(other, this)) {
return true;
}
if (UseJspbArrayAnyFormat != other.UseJspbArrayAnyFormat) return false;
return Equals(_unknownFields, other._unknownFields);
}
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public override int GetHashCode() {
int hash = 1;
if (UseJspbArrayAnyFormat != false) hash ^= UseJspbArrayAnyFormat.GetHashCode();
if (_unknownFields != null) {
hash ^= _unknownFields.GetHashCode();
}
return hash;
}
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public override string ToString() {
return pb::JsonFormatter.ToDiagnosticString(this);
}
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public void WriteTo(pb::CodedOutputStream output) {
if (UseJspbArrayAnyFormat != false) {
output.WriteRawTag(8);
output.WriteBool(UseJspbArrayAnyFormat);
}
if (_unknownFields != null) {
_unknownFields.WriteTo(output);
}
}
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public int CalculateSize() {
int size = 0;
if (UseJspbArrayAnyFormat != false) {
size += 1 + 1;
}
if (_unknownFields != null) {
size += _unknownFields.CalculateSize();
}
return size;
}
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public void MergeFrom(JspbEncodingConfig other) {
if (other == null) {
return;
}
if (other.UseJspbArrayAnyFormat != false) {
UseJspbArrayAnyFormat = other.UseJspbArrayAnyFormat;
}
_unknownFields = pb::UnknownFieldSet.MergeFrom(_unknownFields, other._unknownFields);
}
[global::System.Diagnostics.DebuggerNonUserCodeAttribute]
public void MergeFrom(pb::CodedInputStream input) {
uint tag;
while ((tag = input.ReadTag()) != 0) {
switch(tag) {
default:
if (!pb::UnknownFieldSet.MergeFieldFrom(ref _unknownFields, input)) {
return;
}
break;
case 8: {
UseJspbArrayAnyFormat = input.ReadBool();
break;
}
} }
} }
} }
......
...@@ -22,6 +22,11 @@ ...@@ -22,6 +22,11 @@
<groupId>com.google.guava</groupId> <groupId>com.google.guava</groupId>
<artifactId>guava</artifactId> <artifactId>guava</artifactId>
</dependency> </dependency>
<dependency>
<groupId>com.google.errorprone</groupId>
<artifactId>error_prone_annotations</artifactId>
<version>2.3.2</version>
</dependency>
<dependency> <dependency>
<groupId>com.google.guava</groupId> <groupId>com.google.guava</groupId>
<artifactId>guava-testlib</artifactId> <artifactId>guava-testlib</artifactId>
......
...@@ -41,6 +41,7 @@ class Label ...@@ -41,6 +41,7 @@ class Label
return self::$valueToName[$value]; return self::$valueToName[$value];
} }
public static function value($name) public static function value($name)
{ {
$const = __CLASS__ . '::' . strtoupper($name); $const = __CLASS__ . '::' . strtoupper($name);
......
...@@ -136,6 +136,7 @@ class Type ...@@ -136,6 +136,7 @@ class Type
return self::$valueToName[$value]; return self::$valueToName[$value];
} }
public static function value($name) public static function value($name)
{ {
$const = __CLASS__ . '::' . strtoupper($name); $const = __CLASS__ . '::' . strtoupper($name);
......
...@@ -41,6 +41,7 @@ class CType ...@@ -41,6 +41,7 @@ class CType
return self::$valueToName[$value]; return self::$valueToName[$value];
} }
public static function value($name) public static function value($name)
{ {
$const = __CLASS__ . '::' . strtoupper($name); $const = __CLASS__ . '::' . strtoupper($name);
......
...@@ -45,6 +45,7 @@ class JSType ...@@ -45,6 +45,7 @@ class JSType
return self::$valueToName[$value]; return self::$valueToName[$value];
} }
public static function value($name) public static function value($name)
{ {
$const = __CLASS__ . '::' . strtoupper($name); $const = __CLASS__ . '::' . strtoupper($name);
......
...@@ -47,6 +47,7 @@ class OptimizeMode ...@@ -47,6 +47,7 @@ class OptimizeMode
return self::$valueToName[$value]; return self::$valueToName[$value];
} }
public static function value($name) public static function value($name)
{ {
$const = __CLASS__ . '::' . strtoupper($name); $const = __CLASS__ . '::' . strtoupper($name);
......
...@@ -47,6 +47,7 @@ class IdempotencyLevel ...@@ -47,6 +47,7 @@ class IdempotencyLevel
return self::$valueToName[$value]; return self::$valueToName[$value];
} }
public static function value($name) public static function value($name)
{ {
$const = __CLASS__ . '::' . strtoupper($name); $const = __CLASS__ . '::' . strtoupper($name);
......
load("@bazel_skylib//lib:versions.bzl", "versions") load("@bazel_skylib//lib:versions.bzl", "versions")
def _GetPath(ctx, path): def _GetPath(ctx, path):
if ctx.label.workspace_root: if ctx.label.workspace_root:
return ctx.label.workspace_root + '/' + path return ctx.label.workspace_root + "/" + path
else: else:
return path return path
def _IsNewExternal(ctx): def _IsNewExternal(ctx):
# Bazel 0.4.4 and older have genfiles paths that look like: # Bazel 0.4.4 and older have genfiles paths that look like:
# bazel-out/local-fastbuild/genfiles/external/repo/foo # bazel-out/local-fastbuild/genfiles/external/repo/foo
# After the exec root rearrangement, they look like: # After the exec root rearrangement, they look like:
# ../repo/bazel-out/local-fastbuild/genfiles/foo # ../repo/bazel-out/local-fastbuild/genfiles/foo
return ctx.label.workspace_root.startswith("../") return ctx.label.workspace_root.startswith("../")
def _GenDir(ctx): def _GenDir(ctx):
if _IsNewExternal(ctx): if _IsNewExternal(ctx):
# We are using the fact that Bazel 0.4.4+ provides repository-relative paths # We are using the fact that Bazel 0.4.4+ provides repository-relative paths
# for ctx.genfiles_dir. # for ctx.genfiles_dir.
return ctx.genfiles_dir.path + ( return ctx.genfiles_dir.path + (
"/" + ctx.attr.includes[0] if ctx.attr.includes and ctx.attr.includes[0] else "") "/" + ctx.attr.includes[0] if ctx.attr.includes and ctx.attr.includes[0] else ""
# This means that we're either in the old version OR the new version in the local repo. )
# Either way, appending the source path to the genfiles dir works.
return ctx.var["GENDIR"] + "/" + _SourceDir(ctx) # This means that we're either in the old version OR the new version in the local repo.
# Either way, appending the source path to the genfiles dir works.
return ctx.var["GENDIR"] + "/" + _SourceDir(ctx)
def _SourceDir(ctx): def _SourceDir(ctx):
if not ctx.attr.includes: if not ctx.attr.includes:
return ctx.label.workspace_root return ctx.label.workspace_root
if not ctx.attr.includes[0]: if not ctx.attr.includes[0]:
return _GetPath(ctx, ctx.label.package) return _GetPath(ctx, ctx.label.package)
if not ctx.label.package: if not ctx.label.package:
return _GetPath(ctx, ctx.attr.includes[0]) return _GetPath(ctx, ctx.attr.includes[0])
return _GetPath(ctx, ctx.label.package + '/' + ctx.attr.includes[0]) return _GetPath(ctx, ctx.label.package + "/" + ctx.attr.includes[0])
def _CcHdrs(srcs, use_grpc_plugin=False): def _CcHdrs(srcs, use_grpc_plugin = False):
ret = [s[:-len(".proto")] + ".pb.h" for s in srcs] ret = [s[:-len(".proto")] + ".pb.h" for s in srcs]
if use_grpc_plugin: if use_grpc_plugin:
ret += [s[:-len(".proto")] + ".grpc.pb.h" for s in srcs] ret += [s[:-len(".proto")] + ".grpc.pb.h" for s in srcs]
return ret return ret
def _CcSrcs(srcs, use_grpc_plugin=False): def _CcSrcs(srcs, use_grpc_plugin = False):
ret = [s[:-len(".proto")] + ".pb.cc" for s in srcs] ret = [s[:-len(".proto")] + ".pb.cc" for s in srcs]
if use_grpc_plugin: if use_grpc_plugin:
ret += [s[:-len(".proto")] + ".grpc.pb.cc" for s in srcs] ret += [s[:-len(".proto")] + ".grpc.pb.cc" for s in srcs]
return ret return ret
def _CcOuts(srcs, use_grpc_plugin=False): def _CcOuts(srcs, use_grpc_plugin = False):
return _CcHdrs(srcs, use_grpc_plugin) + _CcSrcs(srcs, use_grpc_plugin) return _CcHdrs(srcs, use_grpc_plugin) + _CcSrcs(srcs, use_grpc_plugin)
def _PyOuts(srcs, use_grpc_plugin=False): def _PyOuts(srcs, use_grpc_plugin = False):
ret = [s[:-len(".proto")] + "_pb2.py" for s in srcs] ret = [s[:-len(".proto")] + "_pb2.py" for s in srcs]
if use_grpc_plugin: if use_grpc_plugin:
ret += [s[:-len(".proto")] + "_pb2_grpc.py" for s in srcs] ret += [s[:-len(".proto")] + "_pb2_grpc.py" for s in srcs]
return ret return ret
def _RelativeOutputPath(path, include, dest=""): def _RelativeOutputPath(path, include, dest = ""):
if include == None: if include == None:
return path return path
if not path.startswith(include): if not path.startswith(include):
fail("Include path %s isn't part of the path %s." % (include, path)) fail("Include path %s isn't part of the path %s." % (include, path))
if include and include[-1] != '/': if include and include[-1] != "/":
include = include + '/' include = include + "/"
if dest and dest[-1] != '/': if dest and dest[-1] != "/":
dest = dest + '/' dest = dest + "/"
path = path[len(include):] path = path[len(include):]
return dest + path return dest + path
def _proto_gen_impl(ctx): def _proto_gen_impl(ctx):
"""General implementation for generating protos""" """General implementation for generating protos"""
srcs = ctx.files.srcs srcs = ctx.files.srcs
deps = [] deps = []
deps += ctx.files.srcs deps += ctx.files.srcs
source_dir = _SourceDir(ctx) source_dir = _SourceDir(ctx)
gen_dir = _GenDir(ctx).rstrip('/') gen_dir = _GenDir(ctx).rstrip("/")
if source_dir: if source_dir:
import_flags = ["-I" + source_dir, "-I" + gen_dir] import_flags = ["-I" + source_dir, "-I" + gen_dir]
else:
import_flags = ["-I."]
for dep in ctx.attr.deps:
import_flags += dep.proto.import_flags
deps += dep.proto.deps
if not ctx.attr.gen_cc and not ctx.attr.gen_py and not ctx.executable.plugin:
return struct(
proto=struct(
srcs=srcs,
import_flags=import_flags,
deps=deps,
),
)
for src in srcs:
args = []
in_gen_dir = src.root.path == gen_dir
if in_gen_dir:
import_flags_real = []
for f in depset(import_flags):
path = f.replace('-I', '')
import_flags_real.append('-I$(realpath -s %s)' % path)
outs = []
use_grpc_plugin = (ctx.attr.plugin_language == "grpc" and ctx.attr.plugin)
path_tpl = "$(realpath %s)" if in_gen_dir else "%s"
if ctx.attr.gen_cc:
args += [("--cpp_out=" + path_tpl) % gen_dir]
outs.extend(_CcOuts([src.basename], use_grpc_plugin=use_grpc_plugin))
if ctx.attr.gen_py:
args += [("--python_out=" + path_tpl) % gen_dir]
outs.extend(_PyOuts([src.basename], use_grpc_plugin=use_grpc_plugin))
outs = [ctx.actions.declare_file(out, sibling=src) for out in outs]
inputs = [src] + deps
if ctx.executable.plugin:
plugin = ctx.executable.plugin
lang = ctx.attr.plugin_language
if not lang and plugin.basename.startswith('protoc-gen-'):
lang = plugin.basename[len('protoc-gen-'):]
if not lang:
fail("cannot infer the target language of plugin", "plugin_language")
outdir = "." if in_gen_dir else gen_dir
if ctx.attr.plugin_options:
outdir = ",".join(ctx.attr.plugin_options) + ":" + outdir
args += [("--plugin=protoc-gen-%s=" + path_tpl) % (lang, plugin.path)]
args += ["--%s_out=%s" % (lang, outdir)]
inputs += [plugin]
if not in_gen_dir:
ctx.actions.run(
inputs=inputs,
outputs=outs,
arguments=args + import_flags + [src.path],
executable=ctx.executable.protoc,
mnemonic="ProtoCompile",
use_default_shell_env=True,
)
else: else:
for out in outs: import_flags = ["-I."]
orig_command = " ".join(
["$(realpath %s)" % ctx.executable.protoc.path] + args + for dep in ctx.attr.deps:
import_flags_real + ["-I.", src.basename]) import_flags += dep.proto.import_flags
command = ";".join([ deps += dep.proto.deps
'CMD="%s"' % orig_command,
"cd %s" % src.dirname, if not ctx.attr.gen_cc and not ctx.attr.gen_py and not ctx.executable.plugin:
"${CMD}", return struct(
"cd -", proto = struct(
]) srcs = srcs,
generated_out = '/'.join([gen_dir, out.basename]) import_flags = import_flags,
if generated_out != out.path: deps = deps,
command += ";mv %s %s" % (generated_out, out.path) ),
ctx.actions.run_shell(
inputs=inputs + [ctx.executable.protoc],
outputs=[out],
command=command,
mnemonic="ProtoCompile",
use_default_shell_env=True,
) )
return struct( for src in srcs:
proto=struct( args = []
srcs=srcs,
import_flags=import_flags, in_gen_dir = src.root.path == gen_dir
deps=deps, if in_gen_dir:
), import_flags_real = []
) for f in depset(import_flags).to_list():
path = f.replace("-I", "")
import_flags_real.append("-I$(realpath -s %s)" % path)
outs = []
use_grpc_plugin = (ctx.attr.plugin_language == "grpc" and ctx.attr.plugin)
path_tpl = "$(realpath %s)" if in_gen_dir else "%s"
if ctx.attr.gen_cc:
args += [("--cpp_out=" + path_tpl) % gen_dir]
outs.extend(_CcOuts([src.basename], use_grpc_plugin = use_grpc_plugin))
if ctx.attr.gen_py:
args += [("--python_out=" + path_tpl) % gen_dir]
outs.extend(_PyOuts([src.basename], use_grpc_plugin = use_grpc_plugin))
outs = [ctx.actions.declare_file(out, sibling = src) for out in outs]
inputs = [src] + deps
if ctx.executable.plugin:
plugin = ctx.executable.plugin
lang = ctx.attr.plugin_language
if not lang and plugin.basename.startswith("protoc-gen-"):
lang = plugin.basename[len("protoc-gen-"):]
if not lang:
fail("cannot infer the target language of plugin", "plugin_language")
outdir = "." if in_gen_dir else gen_dir
if ctx.attr.plugin_options:
outdir = ",".join(ctx.attr.plugin_options) + ":" + outdir
args += [("--plugin=protoc-gen-%s=" + path_tpl) % (lang, plugin.path)]
args += ["--%s_out=%s" % (lang, outdir)]
inputs += [plugin]
if not in_gen_dir:
ctx.actions.run(
inputs = inputs,
outputs = outs,
arguments = args + import_flags + [src.path],
executable = ctx.executable.protoc,
mnemonic = "ProtoCompile",
use_default_shell_env = True,
)
else:
for out in outs:
orig_command = " ".join(
["$(realpath %s)" % ctx.executable.protoc.path] + args +
import_flags_real + ["-I.", src.basename],
)
command = ";".join([
'CMD="%s"' % orig_command,
"cd %s" % src.dirname,
"${CMD}",
"cd -",
])
generated_out = "/".join([gen_dir, out.basename])
if generated_out != out.path:
command += ";mv %s %s" % (generated_out, out.path)
ctx.actions.run_shell(
inputs = inputs + [ctx.executable.protoc],
outputs = [out],
command = command,
mnemonic = "ProtoCompile",
use_default_shell_env = True,
)
return struct(
proto = struct(
srcs = srcs,
import_flags = import_flags,
deps = deps,
),
)
proto_gen = rule( proto_gen = rule(
attrs = { attrs = {
...@@ -218,245 +221,251 @@ Args: ...@@ -218,245 +221,251 @@ Args:
def cc_proto_library( def cc_proto_library(
name, name,
srcs=[], srcs = [],
deps=[], deps = [],
cc_libs=[], cc_libs = [],
include=None, include = None,
protoc="@com_google_protobuf//:protoc", protoc = "@com_google_protobuf//:protoc",
internal_bootstrap_hack=False, internal_bootstrap_hack = False,
use_grpc_plugin=False, use_grpc_plugin = False,
default_runtime="@com_google_protobuf//:protobuf", default_runtime = "@com_google_protobuf//:protobuf",
**kargs): **kargs):
"""Bazel rule to create a C++ protobuf library from proto source files """Bazel rule to create a C++ protobuf library from proto source files
NOTE: the rule is only an internal workaround to generate protos. The NOTE: the rule is only an internal workaround to generate protos. The
interface may change and the rule may be removed when bazel has introduced interface may change and the rule may be removed when bazel has introduced
the native rule. the native rule.
Args: Args:
name: the name of the cc_proto_library. name: the name of the cc_proto_library.
srcs: the .proto files of the cc_proto_library. srcs: the .proto files of the cc_proto_library.
deps: a list of dependency labels; must be cc_proto_library. deps: a list of dependency labels; must be cc_proto_library.
cc_libs: a list of other cc_library targets depended by the generated cc_libs: a list of other cc_library targets depended by the generated
cc_library. cc_library.
include: a string indicating the include path of the .proto files. include: a string indicating the include path of the .proto files.
protoc: the label of the protocol compiler to generate the sources. protoc: the label of the protocol compiler to generate the sources.
internal_bootstrap_hack: a flag indicate the cc_proto_library is used only internal_bootstrap_hack: a flag indicate the cc_proto_library is used only
for bootstraping. When it is set to True, no files will be generated. for bootstraping. When it is set to True, no files will be generated.
The rule will simply be a provider for .proto files, so that other The rule will simply be a provider for .proto files, so that other
cc_proto_library can depend on it. cc_proto_library can depend on it.
use_grpc_plugin: a flag to indicate whether to call the grpc C++ plugin use_grpc_plugin: a flag to indicate whether to call the grpc C++ plugin
when processing the proto files. when processing the proto files.
default_runtime: the implicitly default runtime which will be depended on by default_runtime: the implicitly default runtime which will be depended on by
the generated cc_library target. the generated cc_library target.
**kargs: other keyword arguments that are passed to cc_library. **kargs: other keyword arguments that are passed to cc_library.
""" """
includes = [] includes = []
if include != None: if include != None:
includes = [include] includes = [include]
if internal_bootstrap_hack: if internal_bootstrap_hack:
# For pre-checked-in generated files, we add the internal_bootstrap_hack # For pre-checked-in generated files, we add the internal_bootstrap_hack
# which will skip the codegen action. # which will skip the codegen action.
proto_gen(
name = name + "_genproto",
srcs = srcs,
deps = [s + "_genproto" for s in deps],
includes = includes,
protoc = protoc,
visibility = ["//visibility:public"],
)
# An empty cc_library to make rule dependency consistent.
native.cc_library(
name = name,
**kargs
)
return
grpc_cpp_plugin = None
if use_grpc_plugin:
grpc_cpp_plugin = "//external:grpc_cpp_plugin"
gen_srcs = _CcSrcs(srcs, use_grpc_plugin)
gen_hdrs = _CcHdrs(srcs, use_grpc_plugin)
outs = gen_srcs + gen_hdrs
proto_gen( proto_gen(
name=name + "_genproto", name = name + "_genproto",
srcs=srcs, srcs = srcs,
deps=[s + "_genproto" for s in deps], deps = [s + "_genproto" for s in deps],
includes=includes, includes = includes,
protoc=protoc, protoc = protoc,
visibility=["//visibility:public"], plugin = grpc_cpp_plugin,
plugin_language = "grpc",
gen_cc = 1,
outs = outs,
visibility = ["//visibility:public"],
) )
# An empty cc_library to make rule dependency consistent.
if default_runtime and not default_runtime in cc_libs:
cc_libs = cc_libs + [default_runtime]
if use_grpc_plugin:
cc_libs = cc_libs + ["//external:grpc_lib"]
native.cc_library( native.cc_library(
name=name, name = name,
**kargs) srcs = gen_srcs,
return hdrs = gen_hdrs,
deps = cc_libs + deps,
grpc_cpp_plugin = None includes = includes,
if use_grpc_plugin: **kargs
grpc_cpp_plugin = "//external:grpc_cpp_plugin" )
gen_srcs = _CcSrcs(srcs, use_grpc_plugin)
gen_hdrs = _CcHdrs(srcs, use_grpc_plugin)
outs = gen_srcs + gen_hdrs
proto_gen(
name=name + "_genproto",
srcs=srcs,
deps=[s + "_genproto" for s in deps],
includes=includes,
protoc=protoc,
plugin=grpc_cpp_plugin,
plugin_language="grpc",
gen_cc=1,
outs=outs,
visibility=["//visibility:public"],
)
if default_runtime and not default_runtime in cc_libs:
cc_libs = cc_libs + [default_runtime]
if use_grpc_plugin:
cc_libs = cc_libs + ["//external:grpc_lib"]
native.cc_library(
name=name,
srcs=gen_srcs,
hdrs=gen_hdrs,
deps=cc_libs + deps,
includes=includes,
**kargs)
def internal_gen_well_known_protos_java(srcs): def internal_gen_well_known_protos_java(srcs):
"""Bazel rule to generate the gen_well_known_protos_java genrule """Bazel rule to generate the gen_well_known_protos_java genrule
Args: Args:
srcs: the well known protos srcs: the well known protos
""" """
root = Label("%s//protobuf_java" % (native.repository_name())).workspace_root root = Label("%s//protobuf_java" % (native.repository_name())).workspace_root
pkg = native.package_name() + "/" if native.package_name() else "" pkg = native.package_name() + "/" if native.package_name() else ""
if root == "": if root == "":
include = " -I%ssrc " % pkg include = " -I%ssrc " % pkg
else: else:
include = " -I%s/%ssrc " % (root, pkg) include = " -I%s/%ssrc " % (root, pkg)
native.genrule( native.genrule(
name = "gen_well_known_protos_java", name = "gen_well_known_protos_java",
srcs = srcs, srcs = srcs,
outs = [ outs = [
"wellknown.srcjar", "wellknown.srcjar",
], ],
cmd = "$(location :protoc) --java_out=$(@D)/wellknown.jar" + cmd = "$(location :protoc) --java_out=$(@D)/wellknown.jar" +
" %s $(SRCS) " % include + " %s $(SRCS) " % include +
" && mv $(@D)/wellknown.jar $(@D)/wellknown.srcjar", " && mv $(@D)/wellknown.jar $(@D)/wellknown.srcjar",
tools = [":protoc"], tools = [":protoc"],
) )
def internal_copied_filegroup(name, srcs, strip_prefix, dest, **kwargs): def internal_copied_filegroup(name, srcs, strip_prefix, dest, **kwargs):
"""Macro to copy files to a different directory and then create a filegroup. """Macro to copy files to a different directory and then create a filegroup.
This is used by the //:protobuf_python py_proto_library target to work around This is used by the //:protobuf_python py_proto_library target to work around
an issue caused by Python source files that are part of the same Python an issue caused by Python source files that are part of the same Python
package being in separate directories. package being in separate directories.
Args: Args:
srcs: The source files to copy and add to the filegroup. srcs: The source files to copy and add to the filegroup.
strip_prefix: Path to the root of the files to copy. strip_prefix: Path to the root of the files to copy.
dest: The directory to copy the source files into. dest: The directory to copy the source files into.
**kwargs: extra arguments that will be passesd to the filegroup. **kwargs: extra arguments that will be passesd to the filegroup.
""" """
outs = [_RelativeOutputPath(s, strip_prefix, dest) for s in srcs] outs = [_RelativeOutputPath(s, strip_prefix, dest) for s in srcs]
native.genrule( native.genrule(
name = name + "_genrule", name = name + "_genrule",
srcs = srcs, srcs = srcs,
outs = outs, outs = outs,
cmd = " && ".join( cmd = " && ".join(
["cp $(location %s) $(location %s)" % ["cp $(location %s) $(location %s)" %
(s, _RelativeOutputPath(s, strip_prefix, dest)) for s in srcs]), (s, _RelativeOutputPath(s, strip_prefix, dest)) for s in srcs],
) ),
)
native.filegroup(
name = name, native.filegroup(
srcs = outs, name = name,
**kwargs) srcs = outs,
**kwargs
)
def py_proto_library( def py_proto_library(
name, name,
srcs=[], srcs = [],
deps=[], deps = [],
py_libs=[], py_libs = [],
py_extra_srcs=[], py_extra_srcs = [],
include=None, include = None,
default_runtime="@com_google_protobuf//:protobuf_python", default_runtime = "@com_google_protobuf//:protobuf_python",
protoc="@com_google_protobuf//:protoc", protoc = "@com_google_protobuf//:protoc",
use_grpc_plugin=False, use_grpc_plugin = False,
**kargs): **kargs):
"""Bazel rule to create a Python protobuf library from proto source files """Bazel rule to create a Python protobuf library from proto source files
NOTE: the rule is only an internal workaround to generate protos. The NOTE: the rule is only an internal workaround to generate protos. The
interface may change and the rule may be removed when bazel has introduced interface may change and the rule may be removed when bazel has introduced
the native rule. the native rule.
Args: Args:
name: the name of the py_proto_library. name: the name of the py_proto_library.
srcs: the .proto files of the py_proto_library. srcs: the .proto files of the py_proto_library.
deps: a list of dependency labels; must be py_proto_library. deps: a list of dependency labels; must be py_proto_library.
py_libs: a list of other py_library targets depended by the generated py_libs: a list of other py_library targets depended by the generated
py_library. py_library.
py_extra_srcs: extra source files that will be added to the output py_extra_srcs: extra source files that will be added to the output
py_library. This attribute is used for internal bootstrapping. py_library. This attribute is used for internal bootstrapping.
include: a string indicating the include path of the .proto files. include: a string indicating the include path of the .proto files.
default_runtime: the implicitly default runtime which will be depended on by default_runtime: the implicitly default runtime which will be depended on by
the generated py_library target. the generated py_library target.
protoc: the label of the protocol compiler to generate the sources. protoc: the label of the protocol compiler to generate the sources.
use_grpc_plugin: a flag to indicate whether to call the Python C++ plugin use_grpc_plugin: a flag to indicate whether to call the Python C++ plugin
when processing the proto files. when processing the proto files.
**kargs: other keyword arguments that are passed to cc_library. **kargs: other keyword arguments that are passed to cc_library.
""" """
outs = _PyOuts(srcs, use_grpc_plugin) outs = _PyOuts(srcs, use_grpc_plugin)
includes = [] includes = []
if include != None: if include != None:
includes = [include] includes = [include]
grpc_python_plugin = None grpc_python_plugin = None
if use_grpc_plugin: if use_grpc_plugin:
grpc_python_plugin = "//external:grpc_python_plugin" grpc_python_plugin = "//external:grpc_python_plugin"
# Note: Generated grpc code depends on Python grpc module. This dependency # Note: Generated grpc code depends on Python grpc module. This dependency
# is not explicitly listed in py_libs. Instead, host system is assumed to # is not explicitly listed in py_libs. Instead, host system is assumed to
# have grpc installed. # have grpc installed.
proto_gen(
name=name + "_genproto",
srcs=srcs,
deps=[s + "_genproto" for s in deps],
includes=includes,
protoc=protoc,
gen_py=1,
outs=outs,
visibility=["//visibility:public"],
plugin=grpc_python_plugin,
plugin_language="grpc"
)
if default_runtime and not default_runtime in py_libs + deps:
py_libs = py_libs + [default_runtime]
native.py_library(
name=name,
srcs=outs+py_extra_srcs,
deps=py_libs+deps,
imports=includes,
**kargs)
def internal_protobuf_py_tests( proto_gen(
name, name = name + "_genproto",
modules=[], srcs = srcs,
**kargs): deps = [s + "_genproto" for s in deps],
"""Bazel rules to create batch tests for protobuf internal. includes = includes,
protoc = protoc,
Args: gen_py = 1,
name: the name of the rule. outs = outs,
modules: a list of modules for tests. The macro will create a py_test for visibility = ["//visibility:public"],
each of the parameter with the source "google/protobuf/%s.py" plugin = grpc_python_plugin,
kargs: extra parameters that will be passed into the py_test. plugin_language = "grpc",
)
"""
for m in modules: if default_runtime and not default_runtime in py_libs + deps:
s = "python/google/protobuf/internal/%s.py" % m py_libs = py_libs + [default_runtime]
native.py_test(
name="py_%s" % m, native.py_library(
srcs=[s], name = name,
main=s, srcs = outs + py_extra_srcs,
**kargs) deps = py_libs + deps,
imports = includes,
**kargs
)
def internal_protobuf_py_tests(
name,
modules = [],
**kargs):
"""Bazel rules to create batch tests for protobuf internal.
Args:
name: the name of the rule.
modules: a list of modules for tests. The macro will create a py_test for
each of the parameter with the source "google/protobuf/%s.py"
kargs: extra parameters that will be passed into the py_test.
"""
for m in modules:
s = "python/google/protobuf/internal/%s.py" % m
native.py_test(
name = "py_%s" % m,
srcs = [s],
main = s,
**kargs
)
def check_protobuf_required_bazel_version(): def check_protobuf_required_bazel_version():
"""For WORKSPACE files, to check the installed version of bazel. """For WORKSPACE files, to check the installed version of bazel.
This ensures bazel supports our approach to proto_library() depending on a This ensures bazel supports our approach to proto_library() depending on a
copied filegroup. (Fixed in bazel 0.5.4) copied filegroup. (Fixed in bazel 0.5.4)
""" """
versions.check(minimum_bazel_version = "0.5.4") versions.check(minimum_bazel_version = "0.5.4")
...@@ -208,6 +208,7 @@ libprotobuf_lite_la_SOURCES = \ ...@@ -208,6 +208,7 @@ libprotobuf_lite_la_SOURCES = \
google/protobuf/generated_message_table_driven_lite.cc \ google/protobuf/generated_message_table_driven_lite.cc \
google/protobuf/implicit_weak_message.cc \ google/protobuf/implicit_weak_message.cc \
google/protobuf/message_lite.cc \ google/protobuf/message_lite.cc \
google/protobuf/parse_context.cc \
google/protobuf/repeated_field.cc \ google/protobuf/repeated_field.cc \
google/protobuf/wire_format_lite.cc \ google/protobuf/wire_format_lite.cc \
google/protobuf/io/coded_stream.cc \ google/protobuf/io/coded_stream.cc \
......
...@@ -48,9 +48,9 @@ ...@@ -48,9 +48,9 @@
#include <google/protobuf/io/printer.h> #include <google/protobuf/io/printer.h>
#include <google/protobuf/io/zero_copy_stream_impl.h> #include <google/protobuf/io/zero_copy_stream_impl.h>
#include <google/protobuf/stubs/io_win32.h> #include <google/protobuf/stubs/io_win32.h>
#include <google/protobuf/stubs/port.h>
#include <google/protobuf/stubs/strutil.h> #include <google/protobuf/stubs/strutil.h>
// NOTE: src/google/protobuf/compiler/plugin.cc makes use of cerr for some // NOTE: src/google/protobuf/compiler/plugin.cc makes use of cerr for some
// error cases, so it seems to be ok to use as a back door for errors. // error cases, so it seems to be ok to use as a back door for errors.
...@@ -364,7 +364,7 @@ string StripProto(const string& filename) { ...@@ -364,7 +364,7 @@ string StripProto(const string& filename) {
} }
} }
void StringPieceTrimWhitespace(StringPiece* input) { void TrimWhitespace(StringPiece* input) {
while (!input->empty() && ascii_isspace(*input->data())) { while (!input->empty() && ascii_isspace(*input->data())) {
input->remove_prefix(1); input->remove_prefix(1);
} }
...@@ -659,7 +659,7 @@ string GetCapitalizedType(const FieldDescriptor* field) { ...@@ -659,7 +659,7 @@ string GetCapitalizedType(const FieldDescriptor* field) {
// Some compilers report reaching end of function even though all cases of // Some compilers report reaching end of function even though all cases of
// the enum are handed in the switch. // the enum are handed in the switch.
GOOGLE_LOG(FATAL) << "Can't get here."; GOOGLE_LOG(FATAL) << "Can't get here.";
return NULL; return string();
} }
ObjectiveCType GetObjectiveCType(FieldDescriptor::Type field_type) { ObjectiveCType GetObjectiveCType(FieldDescriptor::Type field_type) {
...@@ -787,7 +787,7 @@ string GPBGenericValueFieldName(const FieldDescriptor* field) { ...@@ -787,7 +787,7 @@ string GPBGenericValueFieldName(const FieldDescriptor* field) {
// Some compilers report reaching end of function even though all cases of // Some compilers report reaching end of function even though all cases of
// the enum are handed in the switch. // the enum are handed in the switch.
GOOGLE_LOG(FATAL) << "Can't get here."; GOOGLE_LOG(FATAL) << "Can't get here.";
return NULL; return string();
} }
...@@ -859,7 +859,7 @@ string DefaultValue(const FieldDescriptor* field) { ...@@ -859,7 +859,7 @@ string DefaultValue(const FieldDescriptor* field) {
// Some compilers report reaching end of function even though all cases of // Some compilers report reaching end of function even though all cases of
// the enum are handed in the switch. // the enum are handed in the switch.
GOOGLE_LOG(FATAL) << "Can't get here."; GOOGLE_LOG(FATAL) << "Can't get here.";
return NULL; return string();
} }
bool HasNonZeroDefaultValue(const FieldDescriptor* field) { bool HasNonZeroDefaultValue(const FieldDescriptor* field) {
...@@ -1047,18 +1047,17 @@ bool ExpectedPrefixesCollector::ConsumeLine( ...@@ -1047,18 +1047,17 @@ bool ExpectedPrefixesCollector::ConsumeLine(
const StringPiece& line, string* out_error) { const StringPiece& line, string* out_error) {
int offset = line.find('='); int offset = line.find('=');
if (offset == StringPiece::npos) { if (offset == StringPiece::npos) {
*out_error = *out_error = string("Expected prefixes file line without equal sign: '") +
string("Expected prefixes file line without equal sign: '") + string(line) + "'.";
line.ToString() + "'.";
return false; return false;
} }
StringPiece package(line, 0, offset); StringPiece package = line.substr(0, offset);
StringPiece prefix(line, offset + 1, line.length() - offset - 1); StringPiece prefix = line.substr(offset + 1);
StringPieceTrimWhitespace(&package); TrimWhitespace(&package);
StringPieceTrimWhitespace(&prefix); TrimWhitespace(&prefix);
// Don't really worry about error checking the package/prefix for // Don't really worry about error checking the package/prefix for
// being valid. Assume the file is validated when it is created/edited. // being valid. Assume the file is validated when it is created/edited.
(*prefix_map_)[package.ToString()] = prefix.ToString(); (*prefix_map_)[string(package)] = string(prefix);
return true; return true;
} }
...@@ -1474,7 +1473,7 @@ class Parser { ...@@ -1474,7 +1473,7 @@ class Parser {
bool Parser::ParseChunk(StringPiece chunk) { bool Parser::ParseChunk(StringPiece chunk) {
if (!leftover_.empty()) { if (!leftover_.empty()) {
chunk.AppendToString(&leftover_); leftover_ += string(chunk);
p_ = StringPiece(leftover_); p_ = StringPiece(leftover_);
} else { } else {
p_ = chunk; p_ = chunk;
...@@ -1483,7 +1482,7 @@ bool Parser::ParseChunk(StringPiece chunk) { ...@@ -1483,7 +1482,7 @@ bool Parser::ParseChunk(StringPiece chunk) {
if (p_.empty()) { if (p_.empty()) {
leftover_.clear(); leftover_.clear();
} else { } else {
leftover_ = p_.ToString(); leftover_ = string(p_);
} }
return result; return result;
} }
...@@ -1506,7 +1505,7 @@ bool Parser::ParseLoop() { ...@@ -1506,7 +1505,7 @@ bool Parser::ParseLoop() {
while (ReadLine(&p_, &line)) { while (ReadLine(&p_, &line)) {
++line_; ++line_;
RemoveComment(&line); RemoveComment(&line);
StringPieceTrimWhitespace(&line); TrimWhitespace(&line);
if (line.size() == 0) { if (line.size() == 0) {
continue; // Blank line. continue; // Blank line.
} }
...@@ -1693,12 +1692,12 @@ bool ImportWriter::ProtoFrameworkCollector::ConsumeLine( ...@@ -1693,12 +1692,12 @@ bool ImportWriter::ProtoFrameworkCollector::ConsumeLine(
if (offset == StringPiece::npos) { if (offset == StringPiece::npos) {
*out_error = *out_error =
string("Framework/proto file mapping line without colon sign: '") + string("Framework/proto file mapping line without colon sign: '") +
line.ToString() + "'."; string(line) + "'.";
return false; return false;
} }
StringPiece framework_name(line, 0, offset); StringPiece framework_name = line.substr(0, offset);
StringPiece proto_file_list(line, offset + 1, line.length() - offset - 1); StringPiece proto_file_list = line.substr(offset + 1);
StringPieceTrimWhitespace(&framework_name); TrimWhitespace(&framework_name);
int start = 0; int start = 0;
while (start < proto_file_list.length()) { while (start < proto_file_list.length()) {
...@@ -1707,25 +1706,27 @@ bool ImportWriter::ProtoFrameworkCollector::ConsumeLine( ...@@ -1707,25 +1706,27 @@ bool ImportWriter::ProtoFrameworkCollector::ConsumeLine(
offset = proto_file_list.length(); offset = proto_file_list.length();
} }
StringPiece proto_file(proto_file_list, start, offset - start); StringPiece proto_file = proto_file_list.substr(start, offset - start);
StringPieceTrimWhitespace(&proto_file); TrimWhitespace(&proto_file);
if (proto_file.size() != 0) { if (proto_file.size() != 0) {
std::map<string, string>::iterator existing_entry = std::map<string, string>::iterator existing_entry =
map_->find(proto_file.ToString()); map_->find(string(proto_file));
if (existing_entry != map_->end()) { if (existing_entry != map_->end()) {
std::cerr << "warning: duplicate proto file reference, replacing framework entry for '" std::cerr << "warning: duplicate proto file reference, replacing "
<< proto_file.ToString() << "' with '" << framework_name.ToString() "framework entry for '"
<< "' (was '" << existing_entry->second << "')." << std::endl; << string(proto_file) << "' with '" << string(framework_name)
<< "' (was '" << existing_entry->second << "')." << std::endl;
std::cerr.flush(); std::cerr.flush();
} }
if (proto_file.find(' ') != StringPiece::npos) { if (proto_file.find(' ') != StringPiece::npos) {
std::cerr << "note: framework mapping file had a proto file with a space in, hopefully that isn't a missing comma: '" std::cerr << "note: framework mapping file had a proto file with a "
<< proto_file.ToString() << "'" << std::endl; "space in, hopefully that isn't a missing comma: '"
<< string(proto_file) << "'" << std::endl;
std::cerr.flush(); std::cerr.flush();
} }
(*map_)[proto_file.ToString()] = framework_name.ToString(); (*map_)[string(proto_file)] = string(framework_name);
} }
start = offset + 1; start = offset + 1;
......
...@@ -62,7 +62,7 @@ string PROTOC_EXPORT EscapeTrigraphs(const string& to_escape); ...@@ -62,7 +62,7 @@ string PROTOC_EXPORT EscapeTrigraphs(const string& to_escape);
string PROTOC_EXPORT StripProto(const string& filename); string PROTOC_EXPORT StripProto(const string& filename);
// Remove white space from either end of a StringPiece. // Remove white space from either end of a StringPiece.
void PROTOC_EXPORT StringPieceTrimWhitespace(StringPiece* input); void PROTOC_EXPORT TrimWhitespace(StringPiece* input);
// Returns true if the name requires a ns_returns_not_retained attribute applied // Returns true if the name requires a ns_returns_not_retained attribute applied
// to it. // to it.
......
...@@ -431,7 +431,11 @@ class BigEndian { ...@@ -431,7 +431,11 @@ class BigEndian {
} // namespace protobuf } // namespace protobuf
} // namespace google } // namespace google
#ifdef PROTOBUF_ENABLE_EXPERIMENTAL_PARSER
#define GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER 1
#else
#define GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER 0 #define GOOGLE_PROTOBUF_ENABLE_EXPERIMENTAL_PARSER 0
#endif
#include <google/protobuf/port_undef.inc> #include <google/protobuf/port_undef.inc>
......
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