| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291 | /* * * Copyright 2015 gRPC authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * *     http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * */#include <map>#include <set>#include <sstream>#include "src/compiler/config.h"#include "src/compiler/objective_c_generator.h"#include "src/compiler/objective_c_generator_helpers.h"#include <google/protobuf/compiler/objectivec/objectivec_helpers.h>using ::google::protobuf::compiler::objectivec::ClassName;using ::grpc::protobuf::FileDescriptor;using ::grpc::protobuf::MethodDescriptor;using ::grpc::protobuf::ServiceDescriptor;using ::grpc::protobuf::io::Printer;using ::grpc::protobuf::FileDescriptor;using ::std::map;using ::std::set;namespace grpc_objective_c_generator {namespace {void PrintProtoRpcDeclarationAsPragma(    Printer* printer, const MethodDescriptor* method,    map< ::grpc::string, ::grpc::string> vars) {  vars["client_stream"] = method->client_streaming() ? "stream " : "";  vars["server_stream"] = method->server_streaming() ? "stream " : "";  printer->Print(vars,                 "#pragma mark $method_name$($client_stream$$request_type$)"                 " returns ($server_stream$$response_type$)\n\n");}template <typename DescriptorType>static void PrintAllComments(const DescriptorType* desc, Printer* printer) {  std::vector<grpc::string> comments;  grpc_generator::GetComment(desc, grpc_generator::COMMENTTYPE_LEADING_DETACHED,                             &comments);  grpc_generator::GetComment(desc, grpc_generator::COMMENTTYPE_LEADING,                             &comments);  grpc_generator::GetComment(desc, grpc_generator::COMMENTTYPE_TRAILING,                             &comments);  if (comments.empty()) {    return;  }  printer->Print("/**\n");  for (auto it = comments.begin(); it != comments.end(); ++it) {    printer->Print(" * ");    size_t start_pos = it->find_first_not_of(' ');    if (start_pos != grpc::string::npos) {      printer->Print(it->c_str() + start_pos);    }    printer->Print("\n");  }  printer->Print(" */\n");}void PrintMethodSignature(Printer* printer, const MethodDescriptor* method,                          const map< ::grpc::string, ::grpc::string>& vars) {  // Print comment  PrintAllComments(method, printer);  printer->Print(vars, "- ($return_type$)$method_name$With");  if (method->client_streaming()) {    printer->Print("RequestsWriter:(GRXWriter *)requestWriter");  } else {    printer->Print(vars, "Request:($request_class$ *)request");  }  // TODO(jcanizales): Put this on a new line and align colons.  if (method->server_streaming()) {    printer->Print(vars,                   " eventHandler:(void(^)(BOOL done, "                   "$response_class$ *_Nullable response, NSError *_Nullable "                   "error))eventHandler");  } else {    printer->Print(vars,                   " handler:(void(^)($response_class$ *_Nullable response, "                   "NSError *_Nullable error))handler");  }}void PrintSimpleSignature(Printer* printer, const MethodDescriptor* method,                          map< ::grpc::string, ::grpc::string> vars) {  vars["method_name"] =      grpc_generator::LowercaseFirstLetter(vars["method_name"]);  vars["return_type"] = "void";  PrintMethodSignature(printer, method, vars);}void PrintAdvancedSignature(Printer* printer, const MethodDescriptor* method,                            map< ::grpc::string, ::grpc::string> vars) {  vars["method_name"] = "RPCTo" + vars["method_name"];  vars["return_type"] = "GRPCProtoCall *";  PrintMethodSignature(printer, method, vars);}inline map< ::grpc::string, ::grpc::string> GetMethodVars(    const MethodDescriptor* method) {  map< ::grpc::string, ::grpc::string> res;  res["method_name"] = method->name();  res["request_type"] = method->input_type()->name();  res["response_type"] = method->output_type()->name();  res["request_class"] = ClassName(method->input_type());  res["response_class"] = ClassName(method->output_type());  return res;}void PrintMethodDeclarations(Printer* printer, const MethodDescriptor* method) {  map< ::grpc::string, ::grpc::string> vars = GetMethodVars(method);  PrintProtoRpcDeclarationAsPragma(printer, method, vars);  PrintSimpleSignature(printer, method, vars);  printer->Print(";\n\n");  PrintAdvancedSignature(printer, method, vars);  printer->Print(";\n\n\n");}void PrintSimpleImplementation(Printer* printer, const MethodDescriptor* method,                               map< ::grpc::string, ::grpc::string> vars) {  printer->Print("{\n");  printer->Print(vars, "  [[self RPCTo$method_name$With");  if (method->client_streaming()) {    printer->Print("RequestsWriter:requestWriter");  } else {    printer->Print("Request:request");  }  if (method->server_streaming()) {    printer->Print(" eventHandler:eventHandler] start];\n");  } else {    printer->Print(" handler:handler] start];\n");  }  printer->Print("}\n");}void PrintAdvancedImplementation(Printer* printer,                                 const MethodDescriptor* method,                                 map< ::grpc::string, ::grpc::string> vars) {  printer->Print("{\n");  printer->Print(vars, "  return [self RPCToMethod:@\"$method_name$\"\n");  printer->Print("            requestsWriter:");  if (method->client_streaming()) {    printer->Print("requestWriter\n");  } else {    printer->Print("[GRXWriter writerWithValue:request]\n");  }  printer->Print(vars, "             responseClass:[$response_class$ class]\n");  printer->Print("        responsesWriteable:[GRXWriteable ");  if (method->server_streaming()) {    printer->Print("writeableWithEventHandler:eventHandler]];\n");  } else {    printer->Print("writeableWithSingleHandler:handler]];\n");  }  printer->Print("}\n");}void PrintMethodImplementations(Printer* printer,                                const MethodDescriptor* method) {  map< ::grpc::string, ::grpc::string> vars = GetMethodVars(method);  PrintProtoRpcDeclarationAsPragma(printer, method, vars);  // TODO(jcanizales): Print documentation from the method.  PrintSimpleSignature(printer, method, vars);  PrintSimpleImplementation(printer, method, vars);  printer->Print("// Returns a not-yet-started RPC object.\n");  PrintAdvancedSignature(printer, method, vars);  PrintAdvancedImplementation(printer, method, vars);}}  // namespace::grpc::string GetAllMessageClasses(const FileDescriptor* file) {  ::grpc::string output;  set< ::grpc::string> classes;  for (int i = 0; i < file->service_count(); i++) {    const auto service = file->service(i);    for (int i = 0; i < service->method_count(); i++) {      const auto method = service->method(i);      classes.insert(ClassName(method->input_type()));      classes.insert(ClassName(method->output_type()));    }  }  for (auto one_class : classes) {    output += "  @class " + one_class + ";\n";  }  return output;}::grpc::string GetHeader(const ServiceDescriptor* service) {  ::grpc::string output;  {    // Scope the output stream so it closes and finalizes output to the string.    grpc::protobuf::io::StringOutputStream output_stream(&output);    Printer printer(&output_stream, '$');    map< ::grpc::string, ::grpc::string> vars = {        {"service_class", ServiceClassName(service)}};    printer.Print(vars, "@protocol $service_class$ <NSObject>\n\n");    for (int i = 0; i < service->method_count(); i++) {      PrintMethodDeclarations(&printer, service->method(i));    }    printer.Print("@end\n\n");    printer.Print(        "/**\n"        " * Basic service implementation, over gRPC, that only does\n"        " * marshalling and parsing.\n"        " */\n");    printer.Print(vars,                  "@interface $service_class$ :"                  " GRPCProtoService<$service_class$>\n");    printer.Print(        "- (instancetype)initWithHost:(NSString *)host"        " NS_DESIGNATED_INITIALIZER;\n");    printer.Print("+ (instancetype)serviceWithHost:(NSString *)host;\n");    printer.Print("@end\n");  }  return output;}::grpc::string GetSource(const ServiceDescriptor* service) {  ::grpc::string output;  {    // Scope the output stream so it closes and finalizes output to the string.    grpc::protobuf::io::StringOutputStream output_stream(&output);    Printer printer(&output_stream, '$');    map< ::grpc::string, ::grpc::string> vars = {        {"service_name", service->name()},        {"service_class", ServiceClassName(service)},        {"package", service->file()->package()}};    printer.Print(vars, "@implementation $service_class$\n\n");    printer.Print("// Designated initializer\n");    printer.Print("- (instancetype)initWithHost:(NSString *)host {\n");    printer.Print(        vars,        "  return (self = [super initWithHost:host"        " packageName:@\"$package$\" serviceName:@\"$service_name$\"]);\n");    printer.Print("}\n\n");    printer.Print(        "// Override superclass initializer to disallow different"        " package and service names.\n");    printer.Print("- (instancetype)initWithHost:(NSString *)host\n");    printer.Print("                 packageName:(NSString *)packageName\n");    printer.Print("                 serviceName:(NSString *)serviceName {\n");    printer.Print("  return [self initWithHost:host];\n");    printer.Print("}\n\n");    printer.Print("+ (instancetype)serviceWithHost:(NSString *)host {\n");    printer.Print("  return [[self alloc] initWithHost:host];\n");    printer.Print("}\n\n\n");    for (int i = 0; i < service->method_count(); i++) {      PrintMethodImplementations(&printer, service->method(i));    }    printer.Print("@end\n");  }  return output;}}  // namespace grpc_objective_c_generator
 |