Commit c4d70123 authored by Adam Cozzette's avatar Adam Cozzette

Fixed references to foreign nested messages with CommonJS-style imports

A bug was causing generated JSPB code with CommonJS-style imports to
refer incorrectly to nested messages from other .proto files. The
generated code would have things like "test_pb.InnerMessage" instead of
"test_pb.OuterMessage.InnerMessage". This commit fixes the problem by
correctly taking into account any message nesting.
parent 787f3fb1
...@@ -1040,4 +1040,18 @@ describe('Message test suite', function() { ...@@ -1040,4 +1040,18 @@ describe('Message test suite', function() {
assertNan(message.getDefaultDoubleField()); assertNan(message.getDefaultDoubleField());
}); });
// Verify that we can successfully use a field referring to a nested message
// from a different .proto file.
it('testForeignNestedMessage', function() {
var msg = new proto.jspb.test.ForeignNestedFieldMessage();
var nested = new proto.jspb.test.Deeply.Nested.Message();
nested.setCount(5);
msg.setDeeplyNestedMessage(nested);
// After a serialization-deserialization round trip we should get back the
// same data we started with.
var serialized = msg.serializeBinary();
var deserialized = proto.jspb.test.ForeignNestedFieldMessage.deserializeBinary(serialized);
assertEquals(5, deserialized.getDeeplyNestedMessage().getCount());
});
}); });
...@@ -260,3 +260,11 @@ enum MapValueEnumNoBinary { ...@@ -260,3 +260,11 @@ enum MapValueEnumNoBinary {
message MapValueMessageNoBinary { message MapValueMessageNoBinary {
optional int32 foo = 1; optional int32 foo = 1;
} }
message Deeply {
message Nested {
message Message {
optional int32 count = 1;
}
}
}
...@@ -35,6 +35,8 @@ option java_multiple_files = true; ...@@ -35,6 +35,8 @@ option java_multiple_files = true;
package jspb.test; package jspb.test;
import "test.proto";
message TestExtensionsMessage { message TestExtensionsMessage {
optional int32 intfield = 1; optional int32 intfield = 1;
extensions 100 to max; extensions 100 to max;
...@@ -52,3 +54,7 @@ extend TestExtensionsMessage { ...@@ -52,3 +54,7 @@ extend TestExtensionsMessage {
optional ExtensionMessage floating_msg_field = 101; optional ExtensionMessage floating_msg_field = 101;
optional string floating_str_field = 102; optional string floating_str_field = 102;
} }
message ForeignNestedFieldMessage {
optional Deeply.Nested.Message deeply_nested_message = 1;
}
...@@ -208,28 +208,28 @@ string GetPath(const GeneratorOptions& options, ...@@ -208,28 +208,28 @@ string GetPath(const GeneratorOptions& options,
} }
} }
// Forward declare, so that GetPrefix can call this method, // Returns the name of the message with a leading dot and taking into account
// which in turn, calls GetPrefix. // nesting, for example ".OuterMessage.InnerMessage", or returns empty if
string GetPath(const GeneratorOptions& options, // descriptor is null. This function does not handle namespacing, only message
const Descriptor* descriptor); // nesting.
string GetNestedMessageName(const Descriptor* descriptor) {
if (descriptor == NULL) {
return "";
}
return StripPrefixString(descriptor->full_name(),
descriptor->file()->package());
}
// Returns the path prefix for a message or enumeration that // Returns the path prefix for a message or enumeration that
// lives under the given file and containing type. // lives under the given file and containing type.
string GetPrefix(const GeneratorOptions& options, string GetPrefix(const GeneratorOptions& options,
const FileDescriptor* file_descriptor, const FileDescriptor* file_descriptor,
const Descriptor* containing_type) { const Descriptor* containing_type) {
string prefix = ""; string prefix = GetPath(options, file_descriptor) +
GetNestedMessageName(containing_type);
if (containing_type == NULL) {
prefix = GetPath(options, file_descriptor);
} else {
prefix = GetPath(options, containing_type);
}
if (!prefix.empty()) { if (!prefix.empty()) {
prefix += "."; prefix += ".";
} }
return prefix; return prefix;
} }
...@@ -277,7 +277,9 @@ string MaybeCrossFileRef(const GeneratorOptions& options, ...@@ -277,7 +277,9 @@ string MaybeCrossFileRef(const GeneratorOptions& options,
from_file != to_message->file()) { from_file != to_message->file()) {
// Cross-file ref in CommonJS needs to use the module alias instead of // Cross-file ref in CommonJS needs to use the module alias instead of
// the global name. // the global name.
return ModuleAlias(to_message->file()->name()) + "." + to_message->name(); return ModuleAlias(to_message->file()->name()) +
GetNestedMessageName(to_message->containing_type()) +
"." + to_message->name();
} else { } else {
// Within a single file we use a full name. // Within a single file we use a full name.
return GetPath(options, to_message); return GetPath(options, to_message);
......
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