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.

145 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;
}