// 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 #include #include #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(std::cin), std::istreambuf_iterator()}; // 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; }