Protocol Buffers - Google's data interchange format (grpc依赖)
https://developers.google.com/protocol-buffers/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
144 lines
5.2 KiB
144 lines
5.2 KiB
// Copyright (c) 2009-2022, Google LLC |
|
// All rights reserved. |
|
// |
|
// 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 LLC 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 Google LLC 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 <assert.h> |
|
|
|
#include <iostream> |
|
#include <string> |
|
|
|
#include "google/protobuf/compiler/plugin.upb.h" |
|
#include "google/protobuf/compiler/plugin.upbdefs.h" |
|
#include "upb/json_decode.h" |
|
#include "upb/json_encode.h" |
|
#include "upb/reflection/def.h" |
|
#include "upb/upb.h" |
|
#include "upbc/code_generator_request.h" |
|
#include "upbc/code_generator_request.upb.h" |
|
#include "upbc/code_generator_request.upbdefs.h" |
|
#include "upbc/subprocess.h" |
|
|
|
static constexpr char kDefaultPlugin[] = "protoc_dart_plugin"; |
|
|
|
static std::string JsonEncode(const upbc_CodeGeneratorRequest* request, |
|
upb_Arena* a) { |
|
upb_DefPool* s = upb_DefPool_New(); |
|
const upb_MessageDef* m = upbc_CodeGeneratorRequest_getmsgdef(s); |
|
|
|
upb_Status status; |
|
upb_Status_Clear(&status); |
|
|
|
const size_t json_size = upb_JsonEncode(request, m, s, 0, NULL, 0, &status); |
|
assert(upb_Status_IsOk(&status)); |
|
|
|
char* json_buf = (char*)upb_Arena_Malloc(a, json_size + 1); |
|
|
|
(void)upb_JsonEncode(request, m, s, 0, json_buf, json_size + 1, &status); |
|
assert(upb_Status_IsOk(&status)); |
|
|
|
upb_DefPool_Free(s); |
|
|
|
return std::string(json_buf, json_size); |
|
} |
|
|
|
static google_protobuf_compiler_CodeGeneratorResponse* JsonDecode( |
|
const std::string& json, upb_Arena* a) { |
|
google_protobuf_compiler_CodeGeneratorResponse* response = |
|
google_protobuf_compiler_CodeGeneratorResponse_new(a); |
|
|
|
upb_DefPool* s = upb_DefPool_New(); |
|
const upb_MessageDef* m = google_protobuf_compiler_CodeGeneratorResponse_getmsgdef(s); |
|
|
|
upb_Status status; |
|
upb_Status_Clear(&status); |
|
|
|
(void)upb_JsonDecode(json.c_str(), json.size(), response, m, s, 0, a, |
|
&status); |
|
assert(upb_Status_IsOk(&status)); |
|
|
|
upb_DefPool_Free(s); |
|
|
|
return response; |
|
} |
|
|
|
static std::string Serialize( |
|
const google_protobuf_compiler_CodeGeneratorResponse* response, upb_Arena* a) { |
|
size_t len = 0; |
|
const char* buf = |
|
google_protobuf_compiler_CodeGeneratorResponse_serialize(response, a, &len); |
|
return std::string(buf, len); |
|
} |
|
|
|
int main() { |
|
upb_Arena* a = upb_Arena_New(); |
|
|
|
// Read (binary) stdin into a string. |
|
const std::string input = {std::istreambuf_iterator<char>(std::cin), |
|
std::istreambuf_iterator<char>()}; |
|
|
|
// Parse the request. |
|
auto inner_request = google_protobuf_compiler_CodeGeneratorRequest_parse( |
|
input.c_str(), input.size(), a); |
|
|
|
// Check the request for a plugin name. |
|
std::string plugin = kDefaultPlugin; |
|
if (google_protobuf_compiler_CodeGeneratorRequest_has_parameter(inner_request)) { |
|
auto param = google_protobuf_compiler_CodeGeneratorRequest_parameter(inner_request); |
|
plugin = std::string(param.data, param.size); |
|
} |
|
|
|
// Wrap the request inside a upbc_CodeGeneratorRequest. |
|
upb_Status status; |
|
upb_Status_Clear(&status); |
|
auto outer_request = upbc_MakeCodeGeneratorRequest(inner_request, a, &status); |
|
if (!upb_Status_IsOk(&status)) { |
|
std::cerr << status.msg << std::endl; |
|
return -1; |
|
} |
|
|
|
const std::string json_request = JsonEncode(outer_request, a); |
|
|
|
// Launch the subprocess. |
|
upbc::Subprocess subprocess; |
|
subprocess.Start(plugin, upbc::Subprocess::SEARCH_PATH); |
|
|
|
// Exchange JSON strings with the subprocess. |
|
std::string json_response, error; |
|
const bool ok = subprocess.Communicate(json_request, &json_response, &error); |
|
if (!ok) { |
|
// Dump the JSON request to stderr if we can't launch the next plugin. |
|
std::cerr << json_request << std::endl; |
|
return -1; |
|
} |
|
|
|
// Decode and serialize the JSON response. |
|
const auto response = JsonDecode(json_response, a); |
|
const std::string output = Serialize(response, a); |
|
|
|
// Question: Is this sufficient for sending reliably to stdout? |
|
std::cout << output; |
|
|
|
upb_Arena_Free(a); |
|
return 0; |
|
}
|
|
|