xref: /aosp_15_r20/external/grpc-grpc/src/compiler/ruby_generator.cc (revision cc02d7e222339f7a4f6ba5f422e6413f4bd931f2)
1*cc02d7e2SAndroid Build Coastguard Worker /*
2*cc02d7e2SAndroid Build Coastguard Worker  *
3*cc02d7e2SAndroid Build Coastguard Worker  * Copyright 2015 gRPC authors.
4*cc02d7e2SAndroid Build Coastguard Worker  *
5*cc02d7e2SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
6*cc02d7e2SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
7*cc02d7e2SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
8*cc02d7e2SAndroid Build Coastguard Worker  *
9*cc02d7e2SAndroid Build Coastguard Worker  *     http://www.apache.org/licenses/LICENSE-2.0
10*cc02d7e2SAndroid Build Coastguard Worker  *
11*cc02d7e2SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
12*cc02d7e2SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
13*cc02d7e2SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14*cc02d7e2SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
15*cc02d7e2SAndroid Build Coastguard Worker  * limitations under the License.
16*cc02d7e2SAndroid Build Coastguard Worker  *
17*cc02d7e2SAndroid Build Coastguard Worker  */
18*cc02d7e2SAndroid Build Coastguard Worker 
19*cc02d7e2SAndroid Build Coastguard Worker #include "src/compiler/ruby_generator.h"
20*cc02d7e2SAndroid Build Coastguard Worker 
21*cc02d7e2SAndroid Build Coastguard Worker #include <cctype>
22*cc02d7e2SAndroid Build Coastguard Worker #include <map>
23*cc02d7e2SAndroid Build Coastguard Worker #include <vector>
24*cc02d7e2SAndroid Build Coastguard Worker 
25*cc02d7e2SAndroid Build Coastguard Worker #include "src/compiler/config.h"
26*cc02d7e2SAndroid Build Coastguard Worker #include "src/compiler/ruby_generator_helpers-inl.h"
27*cc02d7e2SAndroid Build Coastguard Worker #include "src/compiler/ruby_generator_map-inl.h"
28*cc02d7e2SAndroid Build Coastguard Worker #include "src/compiler/ruby_generator_string-inl.h"
29*cc02d7e2SAndroid Build Coastguard Worker 
30*cc02d7e2SAndroid Build Coastguard Worker using grpc::protobuf::FileDescriptor;
31*cc02d7e2SAndroid Build Coastguard Worker using grpc::protobuf::MethodDescriptor;
32*cc02d7e2SAndroid Build Coastguard Worker using grpc::protobuf::ServiceDescriptor;
33*cc02d7e2SAndroid Build Coastguard Worker using grpc::protobuf::io::Printer;
34*cc02d7e2SAndroid Build Coastguard Worker using grpc::protobuf::io::StringOutputStream;
35*cc02d7e2SAndroid Build Coastguard Worker using std::map;
36*cc02d7e2SAndroid Build Coastguard Worker using std::vector;
37*cc02d7e2SAndroid Build Coastguard Worker 
38*cc02d7e2SAndroid Build Coastguard Worker namespace grpc_ruby_generator {
39*cc02d7e2SAndroid Build Coastguard Worker namespace {
40*cc02d7e2SAndroid Build Coastguard Worker 
41*cc02d7e2SAndroid Build Coastguard Worker // Prints out the method using the ruby gRPC DSL.
PrintMethod(const MethodDescriptor * method,Printer * out)42*cc02d7e2SAndroid Build Coastguard Worker void PrintMethod(const MethodDescriptor* method, Printer* out) {
43*cc02d7e2SAndroid Build Coastguard Worker   std::string input_type = RubyTypeOf(method->input_type());
44*cc02d7e2SAndroid Build Coastguard Worker   if (method->client_streaming()) {
45*cc02d7e2SAndroid Build Coastguard Worker     input_type = "stream(" + input_type + ")";
46*cc02d7e2SAndroid Build Coastguard Worker   }
47*cc02d7e2SAndroid Build Coastguard Worker   std::string output_type = RubyTypeOf(method->output_type());
48*cc02d7e2SAndroid Build Coastguard Worker   if (method->server_streaming()) {
49*cc02d7e2SAndroid Build Coastguard Worker     output_type = "stream(" + output_type + ")";
50*cc02d7e2SAndroid Build Coastguard Worker   }
51*cc02d7e2SAndroid Build Coastguard Worker   std::map<std::string, std::string> method_vars = ListToDict({
52*cc02d7e2SAndroid Build Coastguard Worker       "mth.name",
53*cc02d7e2SAndroid Build Coastguard Worker       method->name(),
54*cc02d7e2SAndroid Build Coastguard Worker       "input.type",
55*cc02d7e2SAndroid Build Coastguard Worker       input_type,
56*cc02d7e2SAndroid Build Coastguard Worker       "output.type",
57*cc02d7e2SAndroid Build Coastguard Worker       output_type,
58*cc02d7e2SAndroid Build Coastguard Worker   });
59*cc02d7e2SAndroid Build Coastguard Worker   out->Print(GetRubyComments(method, true).c_str());
60*cc02d7e2SAndroid Build Coastguard Worker   out->Print(method_vars, "rpc :$mth.name$, $input.type$, $output.type$\n");
61*cc02d7e2SAndroid Build Coastguard Worker   out->Print(GetRubyComments(method, false).c_str());
62*cc02d7e2SAndroid Build Coastguard Worker }
63*cc02d7e2SAndroid Build Coastguard Worker 
64*cc02d7e2SAndroid Build Coastguard Worker // Prints out the service using the ruby gRPC DSL.
PrintService(const ServiceDescriptor * service,Printer * out)65*cc02d7e2SAndroid Build Coastguard Worker void PrintService(const ServiceDescriptor* service, Printer* out) {
66*cc02d7e2SAndroid Build Coastguard Worker   if (service->method_count() == 0) {
67*cc02d7e2SAndroid Build Coastguard Worker     return;
68*cc02d7e2SAndroid Build Coastguard Worker   }
69*cc02d7e2SAndroid Build Coastguard Worker 
70*cc02d7e2SAndroid Build Coastguard Worker   // Begin the service module
71*cc02d7e2SAndroid Build Coastguard Worker   std::map<std::string, std::string> module_vars = ListToDict({
72*cc02d7e2SAndroid Build Coastguard Worker       "module.name",
73*cc02d7e2SAndroid Build Coastguard Worker       Modularize(service->name()),
74*cc02d7e2SAndroid Build Coastguard Worker   });
75*cc02d7e2SAndroid Build Coastguard Worker   out->Print(module_vars, "module $module.name$\n");
76*cc02d7e2SAndroid Build Coastguard Worker   out->Indent();
77*cc02d7e2SAndroid Build Coastguard Worker 
78*cc02d7e2SAndroid Build Coastguard Worker   out->Print(GetRubyComments(service, true).c_str());
79*cc02d7e2SAndroid Build Coastguard Worker   out->Print("class Service\n");
80*cc02d7e2SAndroid Build Coastguard Worker 
81*cc02d7e2SAndroid Build Coastguard Worker   // Write the indented class body.
82*cc02d7e2SAndroid Build Coastguard Worker   out->Indent();
83*cc02d7e2SAndroid Build Coastguard Worker   out->Print("\n");
84*cc02d7e2SAndroid Build Coastguard Worker   out->Print("include ::GRPC::GenericService\n");
85*cc02d7e2SAndroid Build Coastguard Worker   out->Print("\n");
86*cc02d7e2SAndroid Build Coastguard Worker   out->Print("self.marshal_class_method = :encode\n");
87*cc02d7e2SAndroid Build Coastguard Worker   out->Print("self.unmarshal_class_method = :decode\n");
88*cc02d7e2SAndroid Build Coastguard Worker   std::map<std::string, std::string> pkg_vars =
89*cc02d7e2SAndroid Build Coastguard Worker       ListToDict({"service_full_name", service->full_name()});
90*cc02d7e2SAndroid Build Coastguard Worker   out->Print(pkg_vars, "self.service_name = '$service_full_name$'\n");
91*cc02d7e2SAndroid Build Coastguard Worker   out->Print("\n");
92*cc02d7e2SAndroid Build Coastguard Worker   for (int i = 0; i < service->method_count(); ++i) {
93*cc02d7e2SAndroid Build Coastguard Worker     PrintMethod(service->method(i), out);
94*cc02d7e2SAndroid Build Coastguard Worker   }
95*cc02d7e2SAndroid Build Coastguard Worker   out->Outdent();
96*cc02d7e2SAndroid Build Coastguard Worker 
97*cc02d7e2SAndroid Build Coastguard Worker   out->Print("end\n");
98*cc02d7e2SAndroid Build Coastguard Worker   out->Print("\n");
99*cc02d7e2SAndroid Build Coastguard Worker   out->Print("Stub = Service.rpc_stub_class\n");
100*cc02d7e2SAndroid Build Coastguard Worker 
101*cc02d7e2SAndroid Build Coastguard Worker   // End the service module
102*cc02d7e2SAndroid Build Coastguard Worker   out->Outdent();
103*cc02d7e2SAndroid Build Coastguard Worker   out->Print("end\n");
104*cc02d7e2SAndroid Build Coastguard Worker   out->Print(GetRubyComments(service, false).c_str());
105*cc02d7e2SAndroid Build Coastguard Worker }
106*cc02d7e2SAndroid Build Coastguard Worker 
107*cc02d7e2SAndroid Build Coastguard Worker }  // namespace
108*cc02d7e2SAndroid Build Coastguard Worker 
109*cc02d7e2SAndroid Build Coastguard Worker // The following functions are copied directly from the source for the protoc
110*cc02d7e2SAndroid Build Coastguard Worker // ruby generator
111*cc02d7e2SAndroid Build Coastguard Worker // to ensure compatibility (with the exception of int and string type changes).
112*cc02d7e2SAndroid Build Coastguard Worker // See
113*cc02d7e2SAndroid Build Coastguard Worker // https://github.com/protocolbuffers/protobuf/blob/63895855d7b1298bee97591cbafced49f23902da/src/google/protobuf/compiler/ruby/ruby_generator.cc#L312
114*cc02d7e2SAndroid Build Coastguard Worker // TODO: keep up to date with protoc code generation, though this behavior isn't
115*cc02d7e2SAndroid Build Coastguard Worker // expected to change
116*cc02d7e2SAndroid Build Coastguard Worker 
117*cc02d7e2SAndroid Build Coastguard Worker // Locale-agnostic utility functions.
IsLower(char ch)118*cc02d7e2SAndroid Build Coastguard Worker bool IsLower(char ch) { return ch >= 'a' && ch <= 'z'; }
119*cc02d7e2SAndroid Build Coastguard Worker 
IsUpper(char ch)120*cc02d7e2SAndroid Build Coastguard Worker bool IsUpper(char ch) { return ch >= 'A' && ch <= 'Z'; }
121*cc02d7e2SAndroid Build Coastguard Worker 
IsAlpha(char ch)122*cc02d7e2SAndroid Build Coastguard Worker bool IsAlpha(char ch) { return IsLower(ch) || IsUpper(ch); }
123*cc02d7e2SAndroid Build Coastguard Worker 
UpperChar(char ch)124*cc02d7e2SAndroid Build Coastguard Worker char UpperChar(char ch) { return IsLower(ch) ? (ch - 'a' + 'A') : ch; }
125*cc02d7e2SAndroid Build Coastguard Worker 
126*cc02d7e2SAndroid Build Coastguard Worker // Package names in protobuf are snake_case by convention, but Ruby module
127*cc02d7e2SAndroid Build Coastguard Worker // names must be PascalCased.
128*cc02d7e2SAndroid Build Coastguard Worker //
129*cc02d7e2SAndroid Build Coastguard Worker //   foo_bar_baz -> FooBarBaz
PackageToModule(const std::string & name)130*cc02d7e2SAndroid Build Coastguard Worker std::string PackageToModule(const std::string& name) {
131*cc02d7e2SAndroid Build Coastguard Worker   bool next_upper = true;
132*cc02d7e2SAndroid Build Coastguard Worker   std::string result;
133*cc02d7e2SAndroid Build Coastguard Worker   result.reserve(name.size());
134*cc02d7e2SAndroid Build Coastguard Worker 
135*cc02d7e2SAndroid Build Coastguard Worker   for (std::string::size_type i = 0; i < name.size(); i++) {
136*cc02d7e2SAndroid Build Coastguard Worker     if (name[i] == '_') {
137*cc02d7e2SAndroid Build Coastguard Worker       next_upper = true;
138*cc02d7e2SAndroid Build Coastguard Worker     } else {
139*cc02d7e2SAndroid Build Coastguard Worker       if (next_upper) {
140*cc02d7e2SAndroid Build Coastguard Worker         result.push_back(UpperChar(name[i]));
141*cc02d7e2SAndroid Build Coastguard Worker       } else {
142*cc02d7e2SAndroid Build Coastguard Worker         result.push_back(name[i]);
143*cc02d7e2SAndroid Build Coastguard Worker       }
144*cc02d7e2SAndroid Build Coastguard Worker       next_upper = false;
145*cc02d7e2SAndroid Build Coastguard Worker     }
146*cc02d7e2SAndroid Build Coastguard Worker   }
147*cc02d7e2SAndroid Build Coastguard Worker 
148*cc02d7e2SAndroid Build Coastguard Worker   return result;
149*cc02d7e2SAndroid Build Coastguard Worker }
150*cc02d7e2SAndroid Build Coastguard Worker 
151*cc02d7e2SAndroid Build Coastguard Worker // Class and enum names in protobuf should be PascalCased by convention, but
152*cc02d7e2SAndroid Build Coastguard Worker // since there is nothing enforcing this we need to ensure that they are valid
153*cc02d7e2SAndroid Build Coastguard Worker // Ruby constants.  That mainly means making sure that the first character is
154*cc02d7e2SAndroid Build Coastguard Worker // an upper-case letter.
RubifyConstant(const std::string & name)155*cc02d7e2SAndroid Build Coastguard Worker std::string RubifyConstant(const std::string& name) {
156*cc02d7e2SAndroid Build Coastguard Worker   std::string ret = name;
157*cc02d7e2SAndroid Build Coastguard Worker   if (!ret.empty()) {
158*cc02d7e2SAndroid Build Coastguard Worker     if (IsLower(ret[0])) {
159*cc02d7e2SAndroid Build Coastguard Worker       // If it starts with a lowercase letter, capitalize it.
160*cc02d7e2SAndroid Build Coastguard Worker       ret[0] = UpperChar(ret[0]);
161*cc02d7e2SAndroid Build Coastguard Worker     } else if (!IsAlpha(ret[0])) {
162*cc02d7e2SAndroid Build Coastguard Worker       // Otherwise (e.g. if it begins with an underscore), we need to come up
163*cc02d7e2SAndroid Build Coastguard Worker       // with some prefix that starts with a capital letter. We could be smarter
164*cc02d7e2SAndroid Build Coastguard Worker       // here, e.g. try to strip leading underscores, but this may cause other
165*cc02d7e2SAndroid Build Coastguard Worker       // problems if the user really intended the name. So let's just prepend a
166*cc02d7e2SAndroid Build Coastguard Worker       // well-known suffix.
167*cc02d7e2SAndroid Build Coastguard Worker       ret = "PB_" + ret;
168*cc02d7e2SAndroid Build Coastguard Worker     }
169*cc02d7e2SAndroid Build Coastguard Worker   }
170*cc02d7e2SAndroid Build Coastguard Worker 
171*cc02d7e2SAndroid Build Coastguard Worker   return ret;
172*cc02d7e2SAndroid Build Coastguard Worker }
173*cc02d7e2SAndroid Build Coastguard Worker // end copying of protoc generator for ruby code
174*cc02d7e2SAndroid Build Coastguard Worker 
GetServices(const FileDescriptor * file)175*cc02d7e2SAndroid Build Coastguard Worker std::string GetServices(const FileDescriptor* file) {
176*cc02d7e2SAndroid Build Coastguard Worker   std::string output;
177*cc02d7e2SAndroid Build Coastguard Worker   {
178*cc02d7e2SAndroid Build Coastguard Worker     // Scope the output stream so it closes and finalizes output to the string.
179*cc02d7e2SAndroid Build Coastguard Worker 
180*cc02d7e2SAndroid Build Coastguard Worker     StringOutputStream output_stream(&output);
181*cc02d7e2SAndroid Build Coastguard Worker     Printer out(&output_stream, '$');
182*cc02d7e2SAndroid Build Coastguard Worker 
183*cc02d7e2SAndroid Build Coastguard Worker     // Don't write out any output if there no services, to avoid empty service
184*cc02d7e2SAndroid Build Coastguard Worker     // files being generated for proto files that don't declare any.
185*cc02d7e2SAndroid Build Coastguard Worker     if (file->service_count() == 0) {
186*cc02d7e2SAndroid Build Coastguard Worker       return output;
187*cc02d7e2SAndroid Build Coastguard Worker     }
188*cc02d7e2SAndroid Build Coastguard Worker 
189*cc02d7e2SAndroid Build Coastguard Worker     std::string package_name = RubyPackage(file);
190*cc02d7e2SAndroid Build Coastguard Worker 
191*cc02d7e2SAndroid Build Coastguard Worker     // Write out a file header.
192*cc02d7e2SAndroid Build Coastguard Worker     std::map<std::string, std::string> header_comment_vars = ListToDict({
193*cc02d7e2SAndroid Build Coastguard Worker         "file.name",
194*cc02d7e2SAndroid Build Coastguard Worker         file->name(),
195*cc02d7e2SAndroid Build Coastguard Worker         "file.package",
196*cc02d7e2SAndroid Build Coastguard Worker         package_name,
197*cc02d7e2SAndroid Build Coastguard Worker     });
198*cc02d7e2SAndroid Build Coastguard Worker     out.Print("# Generated by the protocol buffer compiler.  DO NOT EDIT!\n");
199*cc02d7e2SAndroid Build Coastguard Worker     out.Print(header_comment_vars,
200*cc02d7e2SAndroid Build Coastguard Worker               "# Source: $file.name$ for package '$file.package$'\n");
201*cc02d7e2SAndroid Build Coastguard Worker 
202*cc02d7e2SAndroid Build Coastguard Worker     std::string leading_comments = GetRubyComments(file, true);
203*cc02d7e2SAndroid Build Coastguard Worker     if (!leading_comments.empty()) {
204*cc02d7e2SAndroid Build Coastguard Worker       out.Print("# Original file comments:\n");
205*cc02d7e2SAndroid Build Coastguard Worker       out.PrintRaw(leading_comments.c_str());
206*cc02d7e2SAndroid Build Coastguard Worker     }
207*cc02d7e2SAndroid Build Coastguard Worker 
208*cc02d7e2SAndroid Build Coastguard Worker     out.Print("\n");
209*cc02d7e2SAndroid Build Coastguard Worker     out.Print("require 'grpc'\n");
210*cc02d7e2SAndroid Build Coastguard Worker     // Write out require statemment to import the separately generated file
211*cc02d7e2SAndroid Build Coastguard Worker     // that defines the messages used by the service. This is generated by the
212*cc02d7e2SAndroid Build Coastguard Worker     // main ruby plugin.
213*cc02d7e2SAndroid Build Coastguard Worker     std::map<std::string, std::string> dep_vars = ListToDict({
214*cc02d7e2SAndroid Build Coastguard Worker         "dep.name",
215*cc02d7e2SAndroid Build Coastguard Worker         MessagesRequireName(file),
216*cc02d7e2SAndroid Build Coastguard Worker     });
217*cc02d7e2SAndroid Build Coastguard Worker     out.Print(dep_vars, "require '$dep.name$'\n");
218*cc02d7e2SAndroid Build Coastguard Worker 
219*cc02d7e2SAndroid Build Coastguard Worker     // Write out services within the modules
220*cc02d7e2SAndroid Build Coastguard Worker     out.Print("\n");
221*cc02d7e2SAndroid Build Coastguard Worker     std::vector<std::string> modules = Split(package_name, '.');
222*cc02d7e2SAndroid Build Coastguard Worker     for (size_t i = 0; i < modules.size(); ++i) {
223*cc02d7e2SAndroid Build Coastguard Worker       std::map<std::string, std::string> module_vars = ListToDict({
224*cc02d7e2SAndroid Build Coastguard Worker           "module.name",
225*cc02d7e2SAndroid Build Coastguard Worker           PackageToModule(modules[i]),
226*cc02d7e2SAndroid Build Coastguard Worker       });
227*cc02d7e2SAndroid Build Coastguard Worker       out.Print(module_vars, "module $module.name$\n");
228*cc02d7e2SAndroid Build Coastguard Worker       out.Indent();
229*cc02d7e2SAndroid Build Coastguard Worker     }
230*cc02d7e2SAndroid Build Coastguard Worker     for (int i = 0; i < file->service_count(); ++i) {
231*cc02d7e2SAndroid Build Coastguard Worker       auto service = file->service(i);
232*cc02d7e2SAndroid Build Coastguard Worker       PrintService(service, &out);
233*cc02d7e2SAndroid Build Coastguard Worker     }
234*cc02d7e2SAndroid Build Coastguard Worker     for (size_t i = 0; i < modules.size(); ++i) {
235*cc02d7e2SAndroid Build Coastguard Worker       out.Outdent();
236*cc02d7e2SAndroid Build Coastguard Worker       out.Print("end\n");
237*cc02d7e2SAndroid Build Coastguard Worker     }
238*cc02d7e2SAndroid Build Coastguard Worker 
239*cc02d7e2SAndroid Build Coastguard Worker     out.Print(GetRubyComments(file, false).c_str());
240*cc02d7e2SAndroid Build Coastguard Worker   }
241*cc02d7e2SAndroid Build Coastguard Worker   return output;
242*cc02d7e2SAndroid Build Coastguard Worker }
243*cc02d7e2SAndroid Build Coastguard Worker 
244*cc02d7e2SAndroid Build Coastguard Worker }  // namespace grpc_ruby_generator
245