| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512 |
- // Protocol Buffers - Google's data interchange format
- // Copyright 2008 Google Inc. All rights reserved.
- // https://developers.google.com/protocol-buffers/
- //
- // 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 Inc. 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 THE COPYRIGHT
- // OWNER OR CONTRIBUTORS 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 <sstream>
- #include <google/protobuf/compiler/code_generator.h>
- #include <google/protobuf/compiler/plugin.h>
- #include <google/protobuf/descriptor.h>
- #include <google/protobuf/descriptor.pb.h>
- #include <google/protobuf/io/printer.h>
- #include <google/protobuf/io/zero_copy_stream.h>
- #include <google/protobuf/compiler/ruby/ruby_generator.h>
- namespace google {
- namespace protobuf {
- namespace compiler {
- namespace ruby {
- // Forward decls.
- std::string IntToString(int32 value);
- std::string GetRequireName(const std::string& proto_file);
- std::string LabelForField(google::protobuf::FieldDescriptor* field);
- std::string TypeName(google::protobuf::FieldDescriptor* field);
- void GenerateMessage(const google::protobuf::Descriptor* message,
- google::protobuf::io::Printer* printer);
- void GenerateEnum(const google::protobuf::EnumDescriptor* en,
- google::protobuf::io::Printer* printer);
- void GenerateMessageAssignment(
- const std::string& prefix,
- const google::protobuf::Descriptor* message,
- google::protobuf::io::Printer* printer);
- void GenerateEnumAssignment(
- const std::string& prefix,
- const google::protobuf::EnumDescriptor* en,
- google::protobuf::io::Printer* printer);
- std::string IntToString(int32 value) {
- std::ostringstream os;
- os << value;
- return os.str();
- }
- std::string GetRequireName(const std::string& proto_file) {
- int lastindex = proto_file.find_last_of(".");
- return proto_file.substr(0, lastindex) + "_pb";
- }
- std::string GetOutputFilename(const std::string& proto_file) {
- return GetRequireName(proto_file) + ".rb";
- }
- std::string LabelForField(const google::protobuf::FieldDescriptor* field) {
- switch (field->label()) {
- case FieldDescriptor::LABEL_OPTIONAL: return "optional";
- case FieldDescriptor::LABEL_REQUIRED: return "required";
- case FieldDescriptor::LABEL_REPEATED: return "repeated";
- default: assert(false); return "";
- }
- }
- std::string TypeName(const google::protobuf::FieldDescriptor* field) {
- switch (field->type()) {
- case FieldDescriptor::TYPE_INT32: return "int32";
- case FieldDescriptor::TYPE_INT64: return "int64";
- case FieldDescriptor::TYPE_UINT32: return "uint32";
- case FieldDescriptor::TYPE_UINT64: return "uint64";
- case FieldDescriptor::TYPE_SINT32: return "sint32";
- case FieldDescriptor::TYPE_SINT64: return "sint64";
- case FieldDescriptor::TYPE_FIXED32: return "fixed32";
- case FieldDescriptor::TYPE_FIXED64: return "fixed64";
- case FieldDescriptor::TYPE_SFIXED32: return "sfixed32";
- case FieldDescriptor::TYPE_SFIXED64: return "sfixed64";
- case FieldDescriptor::TYPE_DOUBLE: return "double";
- case FieldDescriptor::TYPE_FLOAT: return "float";
- case FieldDescriptor::TYPE_BOOL: return "bool";
- case FieldDescriptor::TYPE_ENUM: return "enum";
- case FieldDescriptor::TYPE_STRING: return "string";
- case FieldDescriptor::TYPE_BYTES: return "bytes";
- case FieldDescriptor::TYPE_MESSAGE: return "message";
- case FieldDescriptor::TYPE_GROUP: return "group";
- default: assert(false); return "";
- }
- }
- void GenerateField(const google::protobuf::FieldDescriptor* field,
- google::protobuf::io::Printer* printer) {
- if (field->is_map()) {
- const FieldDescriptor* key_field =
- field->message_type()->FindFieldByNumber(1);
- const FieldDescriptor* value_field =
- field->message_type()->FindFieldByNumber(2);
- printer->Print(
- "map :$name$, :$key_type$, :$value_type$, $number$",
- "name", field->name(),
- "key_type", TypeName(key_field),
- "value_type", TypeName(value_field),
- "number", IntToString(field->number()));
- if (value_field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE) {
- printer->Print(
- ", \"$subtype$\"\n",
- "subtype", value_field->message_type()->full_name());
- } else if (value_field->cpp_type() == FieldDescriptor::CPPTYPE_ENUM) {
- printer->Print(
- ", \"$subtype$\"\n",
- "subtype", value_field->enum_type()->full_name());
- } else {
- printer->Print("\n");
- }
- } else {
- printer->Print(
- "$label$ :$name$, ",
- "label", LabelForField(field),
- "name", field->name());
- printer->Print(
- ":$type$, $number$",
- "type", TypeName(field),
- "number", IntToString(field->number()));
- if (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE) {
- printer->Print(
- ", \"$subtype$\"\n",
- "subtype", field->message_type()->full_name());
- } else if (field->cpp_type() == FieldDescriptor::CPPTYPE_ENUM) {
- printer->Print(
- ", \"$subtype$\"\n",
- "subtype", field->enum_type()->full_name());
- } else {
- printer->Print("\n");
- }
- }
- }
- void GenerateOneof(const google::protobuf::OneofDescriptor* oneof,
- google::protobuf::io::Printer* printer) {
- printer->Print(
- "oneof :$name$ do\n",
- "name", oneof->name());
- printer->Indent();
- for (int i = 0; i < oneof->field_count(); i++) {
- const FieldDescriptor* field = oneof->field(i);
- GenerateField(field, printer);
- }
- printer->Outdent();
- printer->Print("end\n");
- }
- void GenerateMessage(const google::protobuf::Descriptor* message,
- google::protobuf::io::Printer* printer) {
- // Don't generate MapEntry messages -- we use the Ruby extension's native
- // support for map fields instead.
- if (message->options().map_entry()) {
- return;
- }
- printer->Print(
- "add_message \"$name$\" do\n",
- "name", message->full_name());
- printer->Indent();
- for (int i = 0; i < message->field_count(); i++) {
- const FieldDescriptor* field = message->field(i);
- if (!field->containing_oneof()) {
- GenerateField(field, printer);
- }
- }
- for (int i = 0; i < message->oneof_decl_count(); i++) {
- const OneofDescriptor* oneof = message->oneof_decl(i);
- GenerateOneof(oneof, printer);
- }
- printer->Outdent();
- printer->Print("end\n");
- for (int i = 0; i < message->nested_type_count(); i++) {
- GenerateMessage(message->nested_type(i), printer);
- }
- for (int i = 0; i < message->enum_type_count(); i++) {
- GenerateEnum(message->enum_type(i), printer);
- }
- }
- void GenerateEnum(const google::protobuf::EnumDescriptor* en,
- google::protobuf::io::Printer* printer) {
- printer->Print(
- "add_enum \"$name$\" do\n",
- "name", en->full_name());
- printer->Indent();
- for (int i = 0; i < en->value_count(); i++) {
- const EnumValueDescriptor* value = en->value(i);
- printer->Print(
- "value :$name$, $number$\n",
- "name", value->name(),
- "number", IntToString(value->number()));
- }
- printer->Outdent();
- printer->Print(
- "end\n");
- }
- // Locale-agnostic utility functions.
- bool IsLower(char ch) { return ch >= 'a' && ch <= 'z'; }
- bool IsUpper(char ch) { return ch >= 'A' && ch <= 'Z'; }
- bool IsAlpha(char ch) { return IsLower(ch) || IsUpper(ch); }
- char ToUpper(char ch) { return IsLower(ch) ? (ch - 'a' + 'A') : ch; }
- // Package names in protobuf are snake_case by convention, but Ruby module
- // names must be PascalCased.
- //
- // foo_bar_baz -> FooBarBaz
- std::string PackageToModule(const std::string& name) {
- bool next_upper = true;
- std::string result;
- result.reserve(name.size());
- for (int i = 0; i < name.size(); i++) {
- if (name[i] == '_') {
- next_upper = true;
- } else {
- if (next_upper) {
- result.push_back(ToUpper(name[i]));
- } else {
- result.push_back(name[i]);
- }
- next_upper = false;
- }
- }
- return result;
- }
- // Class and enum names in protobuf should be PascalCased by convention, but
- // since there is nothing enforcing this we need to ensure that they are valid
- // Ruby constants. That mainly means making sure that the first character is
- // an upper-case letter.
- std::string RubifyConstant(const std::string& name) {
- std::string ret = name;
- if (!ret.empty()) {
- if (IsLower(ret[0])) {
- // If it starts with a lowercase letter, capitalize it.
- ret[0] = ToUpper(ret[0]);
- } else if (!IsAlpha(ret[0])) {
- // Otherwise (e.g. if it begins with an underscore), we need to come up
- // with some prefix that starts with a capital letter. We could be smarter
- // here, e.g. try to strip leading underscores, but this may cause other
- // problems if the user really intended the name. So let's just prepend a
- // well-known suffix.
- ret = "PB_" + ret;
- }
- }
- return ret;
- }
- void GenerateMessageAssignment(
- const std::string& prefix,
- const google::protobuf::Descriptor* message,
- google::protobuf::io::Printer* printer) {
- // Don't generate MapEntry messages -- we use the Ruby extension's native
- // support for map fields instead.
- if (message->options().map_entry()) {
- return;
- }
- printer->Print(
- "$prefix$$name$ = ",
- "prefix", prefix,
- "name", RubifyConstant(message->name()));
- printer->Print(
- "Google::Protobuf::DescriptorPool.generated_pool."
- "lookup(\"$full_name$\").msgclass\n",
- "full_name", message->full_name());
- std::string nested_prefix = prefix + message->name() + "::";
- for (int i = 0; i < message->nested_type_count(); i++) {
- GenerateMessageAssignment(nested_prefix, message->nested_type(i), printer);
- }
- for (int i = 0; i < message->enum_type_count(); i++) {
- GenerateEnumAssignment(nested_prefix, message->enum_type(i), printer);
- }
- }
- void GenerateEnumAssignment(
- const std::string& prefix,
- const google::protobuf::EnumDescriptor* en,
- google::protobuf::io::Printer* printer) {
- printer->Print(
- "$prefix$$name$ = ",
- "prefix", prefix,
- "name", RubifyConstant(en->name()));
- printer->Print(
- "Google::Protobuf::DescriptorPool.generated_pool."
- "lookup(\"$full_name$\").enummodule\n",
- "full_name", en->full_name());
- }
- int GeneratePackageModules(
- const FileDescriptor* file,
- google::protobuf::io::Printer* printer) {
- int levels = 0;
- bool need_change_to_module;
- std::string package_name;
- if (file->options().has_ruby_package()) {
- package_name = file->options().ruby_package();
- need_change_to_module = false;
- } else {
- package_name = file->package();
- need_change_to_module = true;
- }
- while (!package_name.empty()) {
- size_t dot_index = package_name.find(".");
- string component;
- if (dot_index == string::npos) {
- component = package_name;
- package_name = "";
- } else {
- component = package_name.substr(0, dot_index);
- package_name = package_name.substr(dot_index + 1);
- }
- if (need_change_to_module) {
- component = PackageToModule(component);
- }
- printer->Print(
- "module $name$\n",
- "name", component);
- printer->Indent();
- levels++;
- }
- return levels;
- }
- void EndPackageModules(
- int levels,
- google::protobuf::io::Printer* printer) {
- while (levels > 0) {
- levels--;
- printer->Outdent();
- printer->Print(
- "end\n");
- }
- }
- bool UsesTypeFromFile(const Descriptor* message, const FileDescriptor* file,
- string* error) {
- for (int i = 0; i < message->field_count(); i++) {
- const FieldDescriptor* field = message->field(i);
- if ((field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE &&
- field->message_type()->file() == file) ||
- (field->type() == FieldDescriptor::TYPE_ENUM &&
- field->enum_type()->file() == file)) {
- *error = "proto3 message field " + field->full_name() + " in file " +
- file->name() + " has a dependency on a type from proto2 file " +
- file->name() +
- ". Ruby doesn't support proto2 yet, so we must fail.";
- return true;
- }
- }
- for (int i = 0; i < message->nested_type_count(); i++) {
- if (UsesTypeFromFile(message->nested_type(i), file, error)) {
- return true;
- }
- }
- return false;
- }
- // Ruby doesn't currently support proto2. This causes a failure even for proto3
- // files that import proto2. But in some cases, the proto2 file is only being
- // imported to extend another proto2 message. The prime example is declaring
- // custom options by extending FileOptions/FieldOptions/etc.
- //
- // If the proto3 messages don't have any proto2 submessages, it is safe to omit
- // the dependency completely. Users won't be able to use any proto2 extensions,
- // but they already couldn't because proto2 messages aren't supported.
- //
- // If/when we add proto2 support, we should remove this.
- bool MaybeEmitDependency(const FileDescriptor* import,
- const FileDescriptor* from,
- io::Printer* printer,
- string* error) {
- if (import->syntax() == FileDescriptor::SYNTAX_PROTO2) {
- for (int i = 0; i < from->message_type_count(); i++) {
- if (UsesTypeFromFile(from->message_type(i), import, error)) {
- // Error text was already set by UsesTypeFromFile().
- return false;
- }
- }
- // Ok to omit this proto2 dependency -- so we won't print anything.
- GOOGLE_LOG(WARNING) << "Omitting proto2 dependency '" << import->name()
- << "' from proto3 output file '"
- << GetOutputFilename(from->name())
- << "' because we don't support proto2 and no proto2 "
- "types from that file are being used.";
- return true;
- } else {
- printer->Print(
- "require '$name$'\n", "name", GetRequireName(import->name()));
- return true;
- }
- }
- bool GenerateFile(const FileDescriptor* file, io::Printer* printer,
- string* error) {
- printer->Print(
- "# Generated by the protocol buffer compiler. DO NOT EDIT!\n"
- "# source: $filename$\n"
- "\n",
- "filename", file->name());
- printer->Print(
- "require 'google/protobuf'\n\n");
- for (int i = 0; i < file->dependency_count(); i++) {
- if (!MaybeEmitDependency(file->dependency(i), file, printer, error)) {
- return false;
- }
- }
- printer->Print(
- "Google::Protobuf::DescriptorPool.generated_pool.build do\n");
- printer->Indent();
- for (int i = 0; i < file->message_type_count(); i++) {
- GenerateMessage(file->message_type(i), printer);
- }
- for (int i = 0; i < file->enum_type_count(); i++) {
- GenerateEnum(file->enum_type(i), printer);
- }
- printer->Outdent();
- printer->Print(
- "end\n\n");
- int levels = GeneratePackageModules(file, printer);
- for (int i = 0; i < file->message_type_count(); i++) {
- GenerateMessageAssignment("", file->message_type(i), printer);
- }
- for (int i = 0; i < file->enum_type_count(); i++) {
- GenerateEnumAssignment("", file->enum_type(i), printer);
- }
- EndPackageModules(levels, printer);
- return true;
- }
- bool Generator::Generate(
- const FileDescriptor* file,
- const string& parameter,
- GeneratorContext* generator_context,
- string* error) const {
- if (file->syntax() != FileDescriptor::SYNTAX_PROTO3) {
- *error =
- "Can only generate Ruby code for proto3 .proto files.\n"
- "Please add 'syntax = \"proto3\";' to the top of your .proto file.\n";
- return false;
- }
- std::unique_ptr<io::ZeroCopyOutputStream> output(
- generator_context->Open(GetOutputFilename(file->name())));
- io::Printer printer(output.get(), '$');
- return GenerateFile(file, &printer, error);
- }
- } // namespace ruby
- } // namespace compiler
- } // namespace protobuf
- } // namespace google
|