Commit ba52f2b6 authored by Joshua Haberman's avatar Joshua Haberman Committed by GitHub

Merge pull request #1788 from google/rubypackagecap

Ruby: translate package names from snake_case -> PascalCase.
parents c43f7188 97e20261
...@@ -705,6 +705,7 @@ ruby_EXTRA_DIST= \ ...@@ -705,6 +705,7 @@ ruby_EXTRA_DIST= \
ruby/tests/repeated_field_test.rb \ ruby/tests/repeated_field_test.rb \
ruby/tests/stress.rb \ ruby/tests/stress.rb \
ruby/tests/generated_code.proto \ ruby/tests/generated_code.proto \
ruby/tests/test_import.proto \
ruby/tests/generated_code_test.rb \ ruby/tests/generated_code_test.rb \
ruby/travis-test.sh ruby/travis-test.sh
......
...@@ -80,10 +80,15 @@ end ...@@ -80,10 +80,15 @@ end
# Proto for tests. # Proto for tests.
genproto_output << "tests/generated_code.rb" genproto_output << "tests/generated_code.rb"
genproto_output << "tests/test_import.rb"
file "tests/generated_code.rb" => "tests/generated_code.proto" do |file_task| file "tests/generated_code.rb" => "tests/generated_code.proto" do |file_task|
sh "../src/protoc --ruby_out=. tests/generated_code.proto" sh "../src/protoc --ruby_out=. tests/generated_code.proto"
end end
file "tests/test_import.rb" => "tests/test_import.proto" do |file_task|
sh "../src/protoc --ruby_out=. tests/test_import.proto"
end
task :genproto => genproto_output task :genproto => genproto_output
task :clean do task :clean do
......
syntax = "proto3"; syntax = "proto3";
package A.B.C; package a.b.c;
message TestMessage { message TestMessage {
int32 optional_int32 = 1; int32 optional_int32 = 1;
......
...@@ -4,6 +4,7 @@ ...@@ -4,6 +4,7 @@
$LOAD_PATH.unshift(File.expand_path(File.dirname(__FILE__))) $LOAD_PATH.unshift(File.expand_path(File.dirname(__FILE__)))
require 'generated_code_pb' require 'generated_code_pb'
require 'test_import_pb'
require 'test/unit' require 'test/unit'
class GeneratedCodeTest < Test::Unit::TestCase class GeneratedCodeTest < Test::Unit::TestCase
...@@ -13,5 +14,6 @@ class GeneratedCodeTest < Test::Unit::TestCase ...@@ -13,5 +14,6 @@ class GeneratedCodeTest < Test::Unit::TestCase
# successfully creates message definitions and classes, not to test every # successfully creates message definitions and classes, not to test every
# aspect of the extension (basic.rb is for that). # aspect of the extension (basic.rb is for that).
m = A::B::C::TestMessage.new() m = A::B::C::TestMessage.new()
m2 = FooBar::TestImportedMessage.new()
end end
end end
syntax = "proto3";
package foo_bar;
message TestImportedMessage {}
...@@ -237,15 +237,52 @@ void GenerateEnum(const google::protobuf::EnumDescriptor* en, ...@@ -237,15 +237,52 @@ void GenerateEnum(const google::protobuf::EnumDescriptor* en,
"end\n"); "end\n");
} }
// Module names, class names, and enum value names need to be Ruby constants, // Locale-agnostic utility functions.
// which must start with a capital letter. 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 RubifyConstant(const std::string& name) {
std::string ret = name; std::string ret = name;
if (!ret.empty()) { if (!ret.empty()) {
if (ret[0] >= 'a' && ret[0] <= 'z') { if (IsLower(ret[0])) {
// If it starts with a lowercase letter, capitalize it. // If it starts with a lowercase letter, capitalize it.
ret[0] = ret[0] - 'a' + 'A'; ret[0] = ToUpper(ret[0]);
} else if (ret[0] < 'A' || ret[0] > 'Z') { } else if (!IsAlpha(ret[0])) {
// Otherwise (e.g. if it begins with an underscore), we need to come up // 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 // 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 // here, e.g. try to strip leading underscores, but this may cause other
...@@ -254,6 +291,7 @@ std::string RubifyConstant(const std::string& name) { ...@@ -254,6 +291,7 @@ std::string RubifyConstant(const std::string& name) {
ret = "PB_" + ret; ret = "PB_" + ret;
} }
} }
return ret; return ret;
} }
...@@ -314,7 +352,7 @@ int GeneratePackageModules( ...@@ -314,7 +352,7 @@ int GeneratePackageModules(
component = package_name.substr(0, dot_index); component = package_name.substr(0, dot_index);
package_name = package_name.substr(dot_index + 1); package_name = package_name.substr(dot_index + 1);
} }
component = RubifyConstant(component); component = PackageToModule(component);
printer->Print( printer->Print(
"module $name$\n", "module $name$\n",
"name", component); "name", component);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment