1 // Protocol Buffers - Google's data interchange format
2 // Copyright 2008 Google Inc. All rights reserved.
3 // https://developers.google.com/protocol-buffers/
4 //
5 // Redistribution and use in source and binary forms, with or without
6 // modification, are permitted provided that the following conditions are
7 // met:
8 //
9 // * Redistributions of source code must retain the above copyright
10 // notice, this list of conditions and the following disclaimer.
11 // * Redistributions in binary form must reproduce the above
12 // copyright notice, this list of conditions and the following disclaimer
13 // in the documentation and/or other materials provided with the
14 // distribution.
15 // * Neither the name of Google Inc. nor the names of its
16 // contributors may be used to endorse or promote products derived from
17 // this software without specific prior written permission.
18 //
19 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
31 #include <google/protobuf/compiler/js/js_generator.h>
32
33 #include <assert.h>
34 #include <algorithm>
35 #include <limits>
36 #include <map>
37 #include <memory>
38 #include <string>
39 #include <utility>
40 #include <vector>
41
42 #include <google/protobuf/stubs/logging.h>
43 #include <google/protobuf/stubs/common.h>
44 #include <google/protobuf/stubs/stringprintf.h>
45 #include <google/protobuf/stubs/strutil.h>
46
47 #include <google/protobuf/compiler/scc.h>
48 #include <google/protobuf/compiler/js/well_known_types_embed.h>
49 #include <google/protobuf/io/printer.h>
50 #include <google/protobuf/io/zero_copy_stream.h>
51 #include <google/protobuf/descriptor.pb.h>
52 #include <google/protobuf/descriptor.h>
53
54
55 namespace google {
56 namespace protobuf {
57 namespace compiler {
58 namespace js {
59
60 // Sorted list of JavaScript keywords. These cannot be used as names. If they
61 // appear, we prefix them with "pb_".
62 const char* kKeyword[] = {
63 "abstract", "boolean", "break", "byte", "case",
64 "catch", "char", "class", "const", "continue",
65 "debugger", "default", "delete", "do", "double",
66 "else", "enum", "export", "extends", "false",
67 "final", "finally", "float", "for", "function",
68 "goto", "if", "implements", "import", "in",
69 "instanceof", "int", "interface", "long", "native",
70 "new", "null", "package", "private", "protected",
71 "public", "return", "short", "static", "super",
72 "switch", "synchronized", "this", "throw", "throws",
73 "transient", "try", "typeof", "var", "void",
74 "volatile", "while", "with",
75 };
76
77 static const int kNumKeyword = sizeof(kKeyword) / sizeof(char*);
78
79 namespace {
80
81 // The mode of operation for bytes fields. Historically JSPB always carried
82 // bytes as JS {string}, containing base64 content by convention. With binary
83 // and proto3 serialization the new convention is to represent it as binary
84 // data in Uint8Array. See b/26173701 for background on the migration.
85 enum BytesMode {
86 BYTES_DEFAULT, // Default type for getBytesField to return.
87 BYTES_B64, // Explicitly coerce to base64 string where needed.
88 BYTES_U8, // Explicitly coerce to Uint8Array where needed.
89 };
90
IsReserved(const std::string & ident)91 bool IsReserved(const std::string& ident) {
92 for (int i = 0; i < kNumKeyword; i++) {
93 if (ident == kKeyword[i]) {
94 return true;
95 }
96 }
97 return false;
98 }
99
StrEndsWith(StringPiece sp,StringPiece x)100 bool StrEndsWith(StringPiece sp, StringPiece x) {
101 return sp.size() >= x.size() && sp.substr(sp.size() - x.size()) == x;
102 }
103
104 // Returns a copy of |filename| with any trailing ".protodevel" or ".proto
105 // suffix stripped.
106 // TODO(haberman): Unify with copy in compiler/cpp/internal/helpers.cc.
StripProto(const std::string & filename)107 std::string StripProto(const std::string& filename) {
108 const char* suffix =
109 StrEndsWith(filename, ".protodevel") ? ".protodevel" : ".proto";
110 return StripSuffixString(filename, suffix);
111 }
112
GetSnakeFilename(const std::string & filename)113 std::string GetSnakeFilename(const std::string& filename) {
114 std::string snake_name = filename;
115 ReplaceCharacters(&snake_name, "/", '_');
116 return snake_name;
117 }
118
119 // Given a filename like foo/bar/baz.proto, returns the corresponding JavaScript
120 // file foo/bar/baz.js.
GetJSFilename(const GeneratorOptions & options,const std::string & filename)121 std::string GetJSFilename(const GeneratorOptions& options,
122 const std::string& filename) {
123 return StripProto(filename) + options.GetFileNameExtension();
124 }
125
126 // Given a filename like foo/bar/baz.proto, returns the root directory
127 // path ../../
GetRootPath(const std::string & from_filename,const std::string & to_filename)128 string GetRootPath(const std::string& from_filename,
129 const std::string& to_filename) {
130 if (to_filename.find("google/protobuf") == 0) {
131 // Well-known types (.proto files in the google/protobuf directory) are
132 // assumed to come from the 'google-protobuf' npm package. We may want to
133 // generalize this exception later by letting others put generated code in
134 // their own npm packages.
135 return "google-protobuf/";
136 }
137
138 size_t slashes = std::count(from_filename.begin(), from_filename.end(), '/');
139 if (slashes == 0) {
140 return "./";
141 }
142 std::string result = "";
143 for (size_t i = 0; i < slashes; i++) {
144 result += "../";
145 }
146 return result;
147 }
148
149 // Returns the alias we assign to the module of the given .proto filename
150 // when importing.
ModuleAlias(const std::string & filename)151 std::string ModuleAlias(const std::string& filename) {
152 // This scheme could technically cause problems if a file includes any 2 of:
153 // foo/bar_baz.proto
154 // foo_bar_baz.proto
155 // foo_bar/baz.proto
156 //
157 // We'll worry about this problem if/when we actually see it. This name isn't
158 // exposed to users so we can change it later if we need to.
159 std::string basename = StripProto(filename);
160 ReplaceCharacters(&basename, "-", '$');
161 ReplaceCharacters(&basename, "/", '_');
162 ReplaceCharacters(&basename, ".", '_');
163 return basename + "_pb";
164 }
165
166 // Returns the fully normalized JavaScript namespace for the given
167 // file descriptor's package.
GetNamespace(const GeneratorOptions & options,const FileDescriptor * file)168 std::string GetNamespace(const GeneratorOptions& options,
169 const FileDescriptor* file) {
170 if (!options.namespace_prefix.empty()) {
171 return options.namespace_prefix;
172 } else if (!file->package().empty()) {
173 return "proto." + file->package();
174 } else {
175 return "proto";
176 }
177 }
178
179 // Returns the name of the message with a leading dot and taking into account
180 // nesting, for example ".OuterMessage.InnerMessage", or returns empty if
181 // descriptor is null. This function does not handle namespacing, only message
182 // nesting.
GetNestedMessageName(const Descriptor * descriptor)183 std::string GetNestedMessageName(const Descriptor* descriptor) {
184 if (descriptor == NULL) {
185 return "";
186 }
187 std::string result = StripPrefixString(
188 descriptor->full_name(), descriptor->file()->package());
189 // Add a leading dot if one is not already present.
190 if (!result.empty() && result[0] != '.') {
191 result = "." + result;
192 }
193 return result;
194 }
195
196 // Returns the path prefix for a message or enumeration that
197 // lives under the given file and containing type.
GetPrefix(const GeneratorOptions & options,const FileDescriptor * file_descriptor,const Descriptor * containing_type)198 std::string GetPrefix(const GeneratorOptions& options,
199 const FileDescriptor* file_descriptor,
200 const Descriptor* containing_type) {
201 std::string prefix = GetNamespace(options, file_descriptor) +
202 GetNestedMessageName(containing_type);
203 if (!prefix.empty()) {
204 prefix += ".";
205 }
206 return prefix;
207 }
208
209 // Returns the fully normalized JavaScript path prefix for the given
210 // message descriptor.
GetMessagePathPrefix(const GeneratorOptions & options,const Descriptor * descriptor)211 std::string GetMessagePathPrefix(const GeneratorOptions& options,
212 const Descriptor* descriptor) {
213 return GetPrefix(options, descriptor->file(), descriptor->containing_type());
214 }
215
216 // Returns the fully normalized JavaScript path for the given
217 // message descriptor.
GetMessagePath(const GeneratorOptions & options,const Descriptor * descriptor)218 std::string GetMessagePath(const GeneratorOptions& options,
219 const Descriptor* descriptor) {
220 return GetMessagePathPrefix(options, descriptor) + descriptor->name();
221 }
222
223 // Returns the fully normalized JavaScript path prefix for the given
224 // enumeration descriptor.
GetEnumPathPrefix(const GeneratorOptions & options,const EnumDescriptor * enum_descriptor)225 std::string GetEnumPathPrefix(const GeneratorOptions& options,
226 const EnumDescriptor* enum_descriptor) {
227 return GetPrefix(options, enum_descriptor->file(),
228 enum_descriptor->containing_type());
229 }
230
231 // Returns the fully normalized JavaScript path for the given
232 // enumeration descriptor.
GetEnumPath(const GeneratorOptions & options,const EnumDescriptor * enum_descriptor)233 std::string GetEnumPath(const GeneratorOptions& options,
234 const EnumDescriptor* enum_descriptor) {
235 return GetEnumPathPrefix(options, enum_descriptor) + enum_descriptor->name();
236 }
237
MaybeCrossFileRef(const GeneratorOptions & options,const FileDescriptor * from_file,const Descriptor * to_message)238 std::string MaybeCrossFileRef(const GeneratorOptions& options,
239 const FileDescriptor* from_file,
240 const Descriptor* to_message) {
241 if ((options.import_style == GeneratorOptions::kImportCommonJs ||
242 options.import_style == GeneratorOptions::kImportCommonJsStrict) &&
243 from_file != to_message->file()) {
244 // Cross-file ref in CommonJS needs to use the module alias instead of
245 // the global name.
246 return ModuleAlias(to_message->file()->name()) +
247 GetNestedMessageName(to_message->containing_type()) + "." +
248 to_message->name();
249 } else {
250 // Within a single file we use a full name.
251 return GetMessagePath(options, to_message);
252 }
253 }
254
SubmessageTypeRef(const GeneratorOptions & options,const FieldDescriptor * field)255 std::string SubmessageTypeRef(const GeneratorOptions& options,
256 const FieldDescriptor* field) {
257 GOOGLE_CHECK(field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE);
258 return MaybeCrossFileRef(options, field->file(), field->message_type());
259 }
260
261 // - Object field name: LOWER_UNDERSCORE -> LOWER_CAMEL, except for group fields
262 // (UPPER_CAMEL -> LOWER_CAMEL), with "List" (or "Map") appended if appropriate,
263 // and with reserved words triggering a "pb_" prefix.
264 // - Getters/setters: LOWER_UNDERSCORE -> UPPER_CAMEL, except for group fields
265 // (use the name directly), then append "List" if appropriate, then append "$"
266 // if resulting name is equal to a reserved word.
267 // - Enums: just uppercase.
268
269 // Locale-independent version of ToLower that deals only with ASCII A-Z.
ToLowerASCII(char c)270 char ToLowerASCII(char c) {
271 if (c >= 'A' && c <= 'Z') {
272 return (c - 'A') + 'a';
273 } else {
274 return c;
275 }
276 }
277
ParseLowerUnderscore(const std::string & input)278 std::vector<std::string> ParseLowerUnderscore(const std::string& input) {
279 std::vector<std::string> words;
280 std::string running = "";
281 for (int i = 0; i < input.size(); i++) {
282 if (input[i] == '_') {
283 if (!running.empty()) {
284 words.push_back(running);
285 running.clear();
286 }
287 } else {
288 running += ToLowerASCII(input[i]);
289 }
290 }
291 if (!running.empty()) {
292 words.push_back(running);
293 }
294 return words;
295 }
296
ParseUpperCamel(const std::string & input)297 std::vector<std::string> ParseUpperCamel(const std::string& input) {
298 std::vector<std::string> words;
299 std::string running = "";
300 for (int i = 0; i < input.size(); i++) {
301 if (input[i] >= 'A' && input[i] <= 'Z' && !running.empty()) {
302 words.push_back(running);
303 running.clear();
304 }
305 running += ToLowerASCII(input[i]);
306 }
307 if (!running.empty()) {
308 words.push_back(running);
309 }
310 return words;
311 }
312
ToLowerCamel(const std::vector<std::string> & words)313 std::string ToLowerCamel(const std::vector<std::string>& words) {
314 std::string result;
315 for (int i = 0; i < words.size(); i++) {
316 std::string word = words[i];
317 if (i == 0 && (word[0] >= 'A' && word[0] <= 'Z')) {
318 word[0] = (word[0] - 'A') + 'a';
319 } else if (i != 0 && (word[0] >= 'a' && word[0] <= 'z')) {
320 word[0] = (word[0] - 'a') + 'A';
321 }
322 result += word;
323 }
324 return result;
325 }
326
ToUpperCamel(const std::vector<std::string> & words)327 std::string ToUpperCamel(const std::vector<std::string>& words) {
328 std::string result;
329 for (int i = 0; i < words.size(); i++) {
330 std::string word = words[i];
331 if (word[0] >= 'a' && word[0] <= 'z') {
332 word[0] = (word[0] - 'a') + 'A';
333 }
334 result += word;
335 }
336 return result;
337 }
338
339 // Based on code from descriptor.cc (Thanks Kenton!)
340 // Uppercases the entire string, turning ValueName into
341 // VALUENAME.
ToEnumCase(const std::string & input)342 std::string ToEnumCase(const std::string& input) {
343 std::string result;
344 result.reserve(input.size());
345
346 for (int i = 0; i < input.size(); i++) {
347 if ('a' <= input[i] && input[i] <= 'z') {
348 result.push_back(input[i] - 'a' + 'A');
349 } else {
350 result.push_back(input[i]);
351 }
352 }
353
354 return result;
355 }
356
ToLower(const std::string & input)357 std::string ToLower(const std::string& input) {
358 std::string result;
359 result.reserve(input.size());
360
361 for (int i = 0; i < input.size(); i++) {
362 if ('A' <= input[i] && input[i] <= 'Z') {
363 result.push_back(input[i] - 'A' + 'a');
364 } else {
365 result.push_back(input[i]);
366 }
367 }
368
369 return result;
370 }
371
372 // When we're generating one output file per SCC, this is the filename
373 // that top-level extensions should go in.
374 // e.g. one proto file (test.proto):
375 // package a;
376 // extends Foo {
377 // ...
378 // }
379 // If "with_filename" equals true, the extension filename will be
380 // "proto.a_test_extensions.js", otherwise will be "proto.a.js"
GetExtensionFileName(const GeneratorOptions & options,const FileDescriptor * file,bool with_filename)381 std::string GetExtensionFileName(const GeneratorOptions& options,
382 const FileDescriptor* file,
383 bool with_filename) {
384 std::string snake_name = StripProto(GetSnakeFilename(file->name()));
385 return options.output_dir + "/" + ToLower(GetNamespace(options, file)) +
386 (with_filename ? ("_" + snake_name + "_extensions") : "") +
387 options.GetFileNameExtension();
388 }
389 // When we're generating one output file per SCC, this is the filename
390 // that all messages in the SCC should go in.
391 // If with_package equals true, filename will have package prefix,
392 // If the filename length is longer than 200, the filename will be the
393 // SCC's proto filename with suffix "_long_sccs_(index)" (if with_package equals
394 // true it still has package prefix)
GetMessagesFileName(const GeneratorOptions & options,const SCC * scc,bool with_package)395 std::string GetMessagesFileName(const GeneratorOptions& options, const SCC* scc,
396 bool with_package) {
397 static std::map<const Descriptor*, std::string>* long_name_dict =
398 new std::map<const Descriptor*, std::string>();
399 std::string package_base =
400 with_package
401 ? ToLower(GetNamespace(options, scc->GetRepresentative()->file()) +
402 "_")
403 : "";
404 std::string filename_base = "";
405 std::vector<std::string> all_message_names;
406 for (auto one_desc : scc->descriptors) {
407 if (one_desc->containing_type() == nullptr) {
408 all_message_names.push_back(ToLower(one_desc->name()));
409 }
410 }
411 sort(all_message_names.begin(), all_message_names.end());
412 for (auto one_message : all_message_names) {
413 if (!filename_base.empty()) {
414 filename_base += "_";
415 }
416 filename_base += one_message;
417 }
418 if (filename_base.size() + package_base.size() > 200) {
419 if ((*long_name_dict).find(scc->GetRepresentative()) ==
420 (*long_name_dict).end()) {
421 std::string snake_name = StripProto(
422 GetSnakeFilename(scc->GetRepresentative()->file()->name()));
423 (*long_name_dict)[scc->GetRepresentative()] =
424 StrCat(snake_name, "_long_sccs_",
425 static_cast<uint64>((*long_name_dict).size()));
426 }
427 filename_base = (*long_name_dict)[scc->GetRepresentative()];
428 }
429 return options.output_dir + "/" + package_base + filename_base +
430 options.GetFileNameExtension();
431 }
432
433 // When we're generating one output file per type name, this is the filename
434 // that a top-level enum should go in.
435 // If with_package equals true, filename will have package prefix.
GetEnumFileName(const GeneratorOptions & options,const EnumDescriptor * desc,bool with_package)436 std::string GetEnumFileName(const GeneratorOptions& options,
437 const EnumDescriptor* desc, bool with_package) {
438 return options.output_dir + "/" +
439 (with_package ? ToLower(GetNamespace(options, desc->file()) + "_")
440 : "") +
441 ToLower(desc->name()) + options.GetFileNameExtension();
442 }
443
444 // Returns the message/response ID, if set.
GetMessageId(const Descriptor * desc)445 std::string GetMessageId(const Descriptor* desc) {
446 return std::string();
447 }
448
IgnoreExtensionField(const FieldDescriptor * field)449 bool IgnoreExtensionField(const FieldDescriptor* field) {
450 // Exclude descriptor extensions from output "to avoid clutter" (from original
451 // codegen).
452 if (!field->is_extension()) return false;
453 const FileDescriptor* file = field->containing_type()->file();
454 return file->name() == "net/proto2/proto/descriptor.proto" ||
455 file->name() == "google/protobuf/descriptor.proto";
456 }
457
458
459 // Used inside Google only -- do not remove.
IsResponse(const Descriptor * desc)460 bool IsResponse(const Descriptor* desc) { return false; }
461
IgnoreField(const FieldDescriptor * field)462 bool IgnoreField(const FieldDescriptor* field) {
463 return IgnoreExtensionField(field);
464 }
465
466
467 // Do we ignore this message type?
IgnoreMessage(const Descriptor * d)468 bool IgnoreMessage(const Descriptor* d) { return d->options().map_entry(); }
469
470 // Does JSPB ignore this entire oneof? True only if all fields are ignored.
IgnoreOneof(const OneofDescriptor * oneof)471 bool IgnoreOneof(const OneofDescriptor* oneof) {
472 for (int i = 0; i < oneof->field_count(); i++) {
473 if (!IgnoreField(oneof->field(i))) {
474 return false;
475 }
476 }
477 return true;
478 }
479
JSIdent(const GeneratorOptions & options,const FieldDescriptor * field,bool is_upper_camel,bool is_map,bool drop_list)480 std::string JSIdent(const GeneratorOptions& options,
481 const FieldDescriptor* field, bool is_upper_camel,
482 bool is_map, bool drop_list) {
483 std::string result;
484 if (field->type() == FieldDescriptor::TYPE_GROUP) {
485 result = is_upper_camel
486 ? ToUpperCamel(ParseUpperCamel(field->message_type()->name()))
487 : ToLowerCamel(ParseUpperCamel(field->message_type()->name()));
488 } else {
489 result = is_upper_camel ? ToUpperCamel(ParseLowerUnderscore(field->name()))
490 : ToLowerCamel(ParseLowerUnderscore(field->name()));
491 }
492 if (is_map || field->is_map()) {
493 // JSPB-style or proto3-style map.
494 result += "Map";
495 } else if (!drop_list && field->is_repeated()) {
496 // Repeated field.
497 result += "List";
498 }
499 return result;
500 }
501
JSObjectFieldName(const GeneratorOptions & options,const FieldDescriptor * field)502 std::string JSObjectFieldName(const GeneratorOptions& options,
503 const FieldDescriptor* field) {
504 std::string name = JSIdent(options, field,
505 /* is_upper_camel = */ false,
506 /* is_map = */ false,
507 /* drop_list = */ false);
508 if (IsReserved(name)) {
509 name = "pb_" + name;
510 }
511 return name;
512 }
513
JSByteGetterSuffix(BytesMode bytes_mode)514 std::string JSByteGetterSuffix(BytesMode bytes_mode) {
515 switch (bytes_mode) {
516 case BYTES_DEFAULT:
517 return "";
518 case BYTES_B64:
519 return "B64";
520 case BYTES_U8:
521 return "U8";
522 default:
523 assert(false);
524 }
525 return "";
526 }
527
528 // Returns the field name as a capitalized portion of a getter/setter method
529 // name, e.g. MyField for .getMyField().
JSGetterName(const GeneratorOptions & options,const FieldDescriptor * field,BytesMode bytes_mode=BYTES_DEFAULT,bool drop_list=false)530 std::string JSGetterName(const GeneratorOptions& options,
531 const FieldDescriptor* field,
532 BytesMode bytes_mode = BYTES_DEFAULT,
533 bool drop_list = false) {
534 std::string name = JSIdent(options, field,
535 /* is_upper_camel = */ true,
536 /* is_map = */ false, drop_list);
537 if (field->type() == FieldDescriptor::TYPE_BYTES) {
538 std::string suffix = JSByteGetterSuffix(bytes_mode);
539 if (!suffix.empty()) {
540 name += "_as" + suffix;
541 }
542 }
543 if (name == "Extension" || name == "JsPbMessageId") {
544 // Avoid conflicts with base-class names.
545 name += "$";
546 }
547 return name;
548 }
549
550
JSOneofName(const OneofDescriptor * oneof)551 std::string JSOneofName(const OneofDescriptor* oneof) {
552 return ToUpperCamel(ParseLowerUnderscore(oneof->name()));
553 }
554
555 // Returns the index corresponding to this field in the JSPB array (underlying
556 // data storage array).
JSFieldIndex(const FieldDescriptor * field)557 std::string JSFieldIndex(const FieldDescriptor* field) {
558 // Determine whether this field is a member of a group. Group fields are a bit
559 // wonky: their "containing type" is a message type created just for the
560 // group, and that type's parent type has a field with the group-message type
561 // as its message type and TYPE_GROUP as its field type. For such fields, the
562 // index we use is relative to the field number of the group submessage field.
563 // For all other fields, we just use the field number.
564 const Descriptor* containing_type = field->containing_type();
565 const Descriptor* parent_type = containing_type->containing_type();
566 if (parent_type != NULL) {
567 for (int i = 0; i < parent_type->field_count(); i++) {
568 if (parent_type->field(i)->type() == FieldDescriptor::TYPE_GROUP &&
569 parent_type->field(i)->message_type() == containing_type) {
570 return StrCat(field->number() - parent_type->field(i)->number());
571 }
572 }
573 }
574 return StrCat(field->number());
575 }
576
JSOneofIndex(const OneofDescriptor * oneof)577 std::string JSOneofIndex(const OneofDescriptor* oneof) {
578 int index = -1;
579 for (int i = 0; i < oneof->containing_type()->oneof_decl_count(); i++) {
580 const OneofDescriptor* o = oneof->containing_type()->oneof_decl(i);
581 // If at least one field in this oneof is not JSPB-ignored, count the oneof.
582 for (int j = 0; j < o->field_count(); j++) {
583 const FieldDescriptor* f = o->field(j);
584 if (!IgnoreField(f)) {
585 index++;
586 break; // inner loop
587 }
588 }
589 if (o == oneof) {
590 break;
591 }
592 }
593 return StrCat(index);
594 }
595
596 // Decodes a codepoint in \x0000 -- \xFFFF.
DecodeUTF8Codepoint(uint8 * bytes,size_t * length)597 uint16 DecodeUTF8Codepoint(uint8* bytes, size_t* length) {
598 if (*length == 0) {
599 return 0;
600 }
601 size_t expected = 0;
602 if ((*bytes & 0x80) == 0) {
603 expected = 1;
604 } else if ((*bytes & 0xe0) == 0xc0) {
605 expected = 2;
606 } else if ((*bytes & 0xf0) == 0xe0) {
607 expected = 3;
608 } else {
609 // Too long -- don't accept.
610 *length = 0;
611 return 0;
612 }
613
614 if (*length < expected) {
615 // Not enough bytes -- don't accept.
616 *length = 0;
617 return 0;
618 }
619
620 *length = expected;
621 switch (expected) {
622 case 1:
623 return bytes[0];
624 case 2:
625 return ((bytes[0] & 0x1F) << 6) | ((bytes[1] & 0x3F) << 0);
626 case 3:
627 return ((bytes[0] & 0x0F) << 12) | ((bytes[1] & 0x3F) << 6) |
628 ((bytes[2] & 0x3F) << 0);
629 default:
630 return 0;
631 }
632 }
633
634 // Escapes the contents of a string to be included within double-quotes ("") in
635 // JavaScript. The input data should be a UTF-8 encoded C++ string of chars.
636 // Returns false if |out| was truncated because |in| contained invalid UTF-8 or
637 // codepoints outside the BMP.
638 // TODO(b/115551870): Support codepoints outside the BMP.
EscapeJSString(const std::string & in,std::string * out)639 bool EscapeJSString(const std::string& in, std::string* out) {
640 size_t decoded = 0;
641 for (size_t i = 0; i < in.size(); i += decoded) {
642 uint16 codepoint = 0;
643 // Decode the next UTF-8 codepoint.
644 size_t have_bytes = in.size() - i;
645 uint8 bytes[3] = {
646 static_cast<uint8>(in[i]),
647 static_cast<uint8>(((i + 1) < in.size()) ? in[i + 1] : 0),
648 static_cast<uint8>(((i + 2) < in.size()) ? in[i + 2] : 0),
649 };
650 codepoint = DecodeUTF8Codepoint(bytes, &have_bytes);
651 if (have_bytes == 0) {
652 return false;
653 }
654 decoded = have_bytes;
655
656 switch (codepoint) {
657 case '\'':
658 *out += "\\x27";
659 break;
660 case '"':
661 *out += "\\x22";
662 break;
663 case '<':
664 *out += "\\x3c";
665 break;
666 case '=':
667 *out += "\\x3d";
668 break;
669 case '>':
670 *out += "\\x3e";
671 break;
672 case '&':
673 *out += "\\x26";
674 break;
675 case '\b':
676 *out += "\\b";
677 break;
678 case '\t':
679 *out += "\\t";
680 break;
681 case '\n':
682 *out += "\\n";
683 break;
684 case '\f':
685 *out += "\\f";
686 break;
687 case '\r':
688 *out += "\\r";
689 break;
690 case '\\':
691 *out += "\\\\";
692 break;
693 default:
694 // TODO(b/115551870): Once we're supporting codepoints outside the BMP,
695 // use a single Unicode codepoint escape if the output language is
696 // ECMAScript 2015 or above. Otherwise, use a surrogate pair.
697 // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Lexical_grammar#String_literals
698 if (codepoint >= 0x20 && codepoint <= 0x7e) {
699 *out += static_cast<char>(codepoint);
700 } else if (codepoint >= 0x100) {
701 *out += StringPrintf("\\u%04x", codepoint);
702 } else {
703 *out += StringPrintf("\\x%02x", codepoint);
704 }
705 break;
706 }
707 }
708 return true;
709 }
710
EscapeBase64(const std::string & in)711 std::string EscapeBase64(const std::string& in) {
712 static const char* kAlphabet =
713 "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
714 std::string result;
715
716 for (size_t i = 0; i < in.size(); i += 3) {
717 int value = (in[i] << 16) | (((i + 1) < in.size()) ? (in[i + 1] << 8) : 0) |
718 (((i + 2) < in.size()) ? (in[i + 2] << 0) : 0);
719 result += kAlphabet[(value >> 18) & 0x3f];
720 result += kAlphabet[(value >> 12) & 0x3f];
721 if ((i + 1) < in.size()) {
722 result += kAlphabet[(value >> 6) & 0x3f];
723 } else {
724 result += '=';
725 }
726 if ((i + 2) < in.size()) {
727 result += kAlphabet[(value >> 0) & 0x3f];
728 } else {
729 result += '=';
730 }
731 }
732
733 return result;
734 }
735
736 // Post-process the result of SimpleFtoa/SimpleDtoa to *exactly* match the
737 // original codegen's formatting (which is just .toString() on java.lang.Double
738 // or java.lang.Float).
PostProcessFloat(std::string result)739 std::string PostProcessFloat(std::string result) {
740 // If inf, -inf or nan, replace with +Infinity, -Infinity or NaN.
741 if (result == "inf") {
742 return "Infinity";
743 } else if (result == "-inf") {
744 return "-Infinity";
745 } else if (result == "nan") {
746 return "NaN";
747 }
748
749 // If scientific notation (e.g., "1e10"), (i) capitalize the "e", (ii)
750 // ensure that the mantissa (portion prior to the "e") has at least one
751 // fractional digit (after the decimal point), and (iii) strip any unnecessary
752 // leading zeroes and/or '+' signs from the exponent.
753 std::string::size_type exp_pos = result.find('e');
754 if (exp_pos != std::string::npos) {
755 std::string mantissa = result.substr(0, exp_pos);
756 std::string exponent = result.substr(exp_pos + 1);
757
758 // Add ".0" to mantissa if no fractional part exists.
759 if (mantissa.find('.') == std::string::npos) {
760 mantissa += ".0";
761 }
762
763 // Strip the sign off the exponent and store as |exp_neg|.
764 bool exp_neg = false;
765 if (!exponent.empty() && exponent[0] == '+') {
766 exponent = exponent.substr(1);
767 } else if (!exponent.empty() && exponent[0] == '-') {
768 exp_neg = true;
769 exponent = exponent.substr(1);
770 }
771
772 // Strip any leading zeroes off the exponent.
773 while (exponent.size() > 1 && exponent[0] == '0') {
774 exponent = exponent.substr(1);
775 }
776
777 return mantissa + "E" + string(exp_neg ? "-" : "") + exponent;
778 }
779
780 // Otherwise, this is an ordinary decimal number. Append ".0" if result has no
781 // decimal/fractional part in order to match output of original codegen.
782 if (result.find('.') == std::string::npos) {
783 result += ".0";
784 }
785
786 return result;
787 }
788
FloatToString(float value)789 std::string FloatToString(float value) {
790 std::string result = SimpleFtoa(value);
791 return PostProcessFloat(result);
792 }
793
DoubleToString(double value)794 std::string DoubleToString(double value) {
795 std::string result = SimpleDtoa(value);
796 return PostProcessFloat(result);
797 }
798
799 // Return true if this is an integral field that should be represented as string
800 // in JS.
IsIntegralFieldWithStringJSType(const FieldDescriptor * field)801 bool IsIntegralFieldWithStringJSType(const FieldDescriptor* field) {
802 switch (field->cpp_type()) {
803 case FieldDescriptor::CPPTYPE_INT64:
804 case FieldDescriptor::CPPTYPE_UINT64:
805 // The default value of JSType is JS_NORMAL, which behaves the same as
806 // JS_NUMBER.
807 return field->options().jstype() == FieldOptions::JS_STRING;
808 default:
809 return false;
810 }
811 }
812
MaybeNumberString(const FieldDescriptor * field,const std::string & orig)813 std::string MaybeNumberString(const FieldDescriptor* field,
814 const std::string& orig) {
815 return IsIntegralFieldWithStringJSType(field) ? ("\"" + orig + "\"") : orig;
816 }
817
JSFieldDefault(const FieldDescriptor * field)818 std::string JSFieldDefault(const FieldDescriptor* field) {
819 if (field->is_repeated()) {
820 return "[]";
821 }
822
823 switch (field->cpp_type()) {
824 case FieldDescriptor::CPPTYPE_INT32:
825 return MaybeNumberString(field,
826 StrCat(field->default_value_int32()));
827 case FieldDescriptor::CPPTYPE_UINT32:
828 // The original codegen is in Java, and Java protobufs store unsigned
829 // integer values as signed integer values. In order to exactly match the
830 // output, we need to reinterpret as base-2 signed. Ugh.
831 return MaybeNumberString(
832 field,
833 StrCat(static_cast<int32>(field->default_value_uint32())));
834 case FieldDescriptor::CPPTYPE_INT64:
835 return MaybeNumberString(field,
836 StrCat(field->default_value_int64()));
837 case FieldDescriptor::CPPTYPE_UINT64:
838 // See above note for uint32 -- reinterpreting as signed.
839 return MaybeNumberString(
840 field,
841 StrCat(static_cast<int64>(field->default_value_uint64())));
842 case FieldDescriptor::CPPTYPE_ENUM:
843 return StrCat(field->default_value_enum()->number());
844 case FieldDescriptor::CPPTYPE_BOOL:
845 return field->default_value_bool() ? "true" : "false";
846 case FieldDescriptor::CPPTYPE_FLOAT:
847 return FloatToString(field->default_value_float());
848 case FieldDescriptor::CPPTYPE_DOUBLE:
849 return DoubleToString(field->default_value_double());
850 case FieldDescriptor::CPPTYPE_STRING:
851 if (field->type() == FieldDescriptor::TYPE_STRING) {
852 std::string out;
853 bool is_valid = EscapeJSString(field->default_value_string(), &out);
854 if (!is_valid) {
855 // TODO(b/115551870): Decide whether this should be a hard error.
856 GOOGLE_LOG(WARNING) << "The default value for field " << field->full_name()
857 << " was truncated since it contained invalid UTF-8 or"
858 " codepoints outside the basic multilingual plane.";
859 }
860 return "\"" + out + "\"";
861 } else { // Bytes
862 return "\"" + EscapeBase64(field->default_value_string()) + "\"";
863 }
864 case FieldDescriptor::CPPTYPE_MESSAGE:
865 return "null";
866 }
867 GOOGLE_LOG(FATAL) << "Shouldn't reach here.";
868 return "";
869 }
870
ProtoTypeName(const GeneratorOptions & options,const FieldDescriptor * field)871 std::string ProtoTypeName(const GeneratorOptions& options,
872 const FieldDescriptor* field) {
873 switch (field->type()) {
874 case FieldDescriptor::TYPE_BOOL:
875 return "bool";
876 case FieldDescriptor::TYPE_INT32:
877 return "int32";
878 case FieldDescriptor::TYPE_UINT32:
879 return "uint32";
880 case FieldDescriptor::TYPE_SINT32:
881 return "sint32";
882 case FieldDescriptor::TYPE_FIXED32:
883 return "fixed32";
884 case FieldDescriptor::TYPE_SFIXED32:
885 return "sfixed32";
886 case FieldDescriptor::TYPE_INT64:
887 return "int64";
888 case FieldDescriptor::TYPE_UINT64:
889 return "uint64";
890 case FieldDescriptor::TYPE_SINT64:
891 return "sint64";
892 case FieldDescriptor::TYPE_FIXED64:
893 return "fixed64";
894 case FieldDescriptor::TYPE_SFIXED64:
895 return "sfixed64";
896 case FieldDescriptor::TYPE_FLOAT:
897 return "float";
898 case FieldDescriptor::TYPE_DOUBLE:
899 return "double";
900 case FieldDescriptor::TYPE_STRING:
901 return "string";
902 case FieldDescriptor::TYPE_BYTES:
903 return "bytes";
904 case FieldDescriptor::TYPE_GROUP:
905 return GetMessagePath(options, field->message_type());
906 case FieldDescriptor::TYPE_ENUM:
907 return GetEnumPath(options, field->enum_type());
908 case FieldDescriptor::TYPE_MESSAGE:
909 return GetMessagePath(options, field->message_type());
910 default:
911 return "";
912 }
913 }
914
JSIntegerTypeName(const FieldDescriptor * field)915 std::string JSIntegerTypeName(const FieldDescriptor* field) {
916 return IsIntegralFieldWithStringJSType(field) ? "string" : "number";
917 }
918
JSStringTypeName(const GeneratorOptions & options,const FieldDescriptor * field,BytesMode bytes_mode)919 std::string JSStringTypeName(const GeneratorOptions& options,
920 const FieldDescriptor* field,
921 BytesMode bytes_mode) {
922 if (field->type() == FieldDescriptor::TYPE_BYTES) {
923 switch (bytes_mode) {
924 case BYTES_DEFAULT:
925 return "(string|Uint8Array)";
926 case BYTES_B64:
927 return "string";
928 case BYTES_U8:
929 return "Uint8Array";
930 default:
931 assert(false);
932 }
933 }
934 return "string";
935 }
936
JSTypeName(const GeneratorOptions & options,const FieldDescriptor * field,BytesMode bytes_mode)937 std::string JSTypeName(const GeneratorOptions& options,
938 const FieldDescriptor* field, BytesMode bytes_mode) {
939 switch (field->cpp_type()) {
940 case FieldDescriptor::CPPTYPE_BOOL:
941 return "boolean";
942 case FieldDescriptor::CPPTYPE_INT32:
943 return JSIntegerTypeName(field);
944 case FieldDescriptor::CPPTYPE_INT64:
945 return JSIntegerTypeName(field);
946 case FieldDescriptor::CPPTYPE_UINT32:
947 return JSIntegerTypeName(field);
948 case FieldDescriptor::CPPTYPE_UINT64:
949 return JSIntegerTypeName(field);
950 case FieldDescriptor::CPPTYPE_FLOAT:
951 return "number";
952 case FieldDescriptor::CPPTYPE_DOUBLE:
953 return "number";
954 case FieldDescriptor::CPPTYPE_STRING:
955 return JSStringTypeName(options, field, bytes_mode);
956 case FieldDescriptor::CPPTYPE_ENUM:
957 return GetEnumPath(options, field->enum_type());
958 case FieldDescriptor::CPPTYPE_MESSAGE:
959 return GetMessagePath(options, field->message_type());
960 default:
961 return "";
962 }
963 }
964
965 // Used inside Google only -- do not remove.
UseBrokenPresenceSemantics(const GeneratorOptions & options,const FieldDescriptor * field)966 bool UseBrokenPresenceSemantics(const GeneratorOptions& options,
967 const FieldDescriptor* field) {
968 return false;
969 }
970
971 // Returns true for fields that return "null" from accessors when they are
972 // unset. This should normally only be true for non-repeated submessages, but we
973 // have legacy users who relied on old behavior where accessors behaved this
974 // way.
ReturnsNullWhenUnset(const GeneratorOptions & options,const FieldDescriptor * field)975 bool ReturnsNullWhenUnset(const GeneratorOptions& options,
976 const FieldDescriptor* field) {
977 if (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE &&
978 field->is_optional()) {
979 return true;
980 }
981
982 // TODO(haberman): remove this case and unconditionally return false.
983 return UseBrokenPresenceSemantics(options, field) && !field->is_repeated() &&
984 !field->has_default_value();
985 }
986
987 // In a sane world, this would be the same as ReturnsNullWhenUnset(). But in
988 // the status quo, some fields declare that they never return null/undefined
989 // even though they actually do:
990 // * required fields
991 // * optional enum fields
992 // * proto3 primitive fields.
DeclaredReturnTypeIsNullable(const GeneratorOptions & options,const FieldDescriptor * field)993 bool DeclaredReturnTypeIsNullable(const GeneratorOptions& options,
994 const FieldDescriptor* field) {
995 if (field->is_required() || field->type() == FieldDescriptor::TYPE_ENUM) {
996 return false;
997 }
998
999 if (field->file()->syntax() == FileDescriptor::SYNTAX_PROTO3 &&
1000 field->cpp_type() != FieldDescriptor::CPPTYPE_MESSAGE) {
1001 return false;
1002 }
1003
1004 return ReturnsNullWhenUnset(options, field);
1005 }
1006
SetterAcceptsUndefined(const GeneratorOptions & options,const FieldDescriptor * field)1007 bool SetterAcceptsUndefined(const GeneratorOptions& options,
1008 const FieldDescriptor* field) {
1009 if (ReturnsNullWhenUnset(options, field)) {
1010 return true;
1011 }
1012
1013 // Broken presence semantics always accepts undefined for setters.
1014 return UseBrokenPresenceSemantics(options, field);
1015 }
1016
SetterAcceptsNull(const GeneratorOptions & options,const FieldDescriptor * field)1017 bool SetterAcceptsNull(const GeneratorOptions& options,
1018 const FieldDescriptor* field) {
1019 if (ReturnsNullWhenUnset(options, field)) {
1020 return true;
1021 }
1022
1023 // With broken presence semantics, fields with defaults accept "null" for
1024 // setters, but other fields do not. This is a strange quirk of the old
1025 // codegen.
1026 return UseBrokenPresenceSemantics(options, field) &&
1027 field->has_default_value();
1028 }
1029
1030 // Returns types which are known to by non-nullable by default.
1031 // The style guide requires that we omit "!" in this case.
IsPrimitive(const std::string & type)1032 bool IsPrimitive(const std::string& type) {
1033 return type == "undefined" || type == "string" || type == "number" ||
1034 type == "boolean";
1035 }
1036
JSFieldTypeAnnotation(const GeneratorOptions & options,const FieldDescriptor * field,bool is_setter_argument,bool force_present,bool singular_if_not_packed,BytesMode bytes_mode=BYTES_DEFAULT,bool force_singular=false)1037 std::string JSFieldTypeAnnotation(const GeneratorOptions& options,
1038 const FieldDescriptor* field,
1039 bool is_setter_argument, bool force_present,
1040 bool singular_if_not_packed,
1041 BytesMode bytes_mode = BYTES_DEFAULT,
1042 bool force_singular = false) {
1043 std::string jstype = JSTypeName(options, field, bytes_mode);
1044
1045 if (!force_singular && field->is_repeated() &&
1046 (field->is_packed() || !singular_if_not_packed)) {
1047 if (field->type() == FieldDescriptor::TYPE_BYTES &&
1048 bytes_mode == BYTES_DEFAULT) {
1049 jstype = "(Array<!Uint8Array>|Array<string>)";
1050 } else {
1051 if (!IsPrimitive(jstype)) {
1052 jstype = "!" + jstype;
1053 }
1054 jstype = "Array<" + jstype + ">";
1055 }
1056 }
1057
1058 bool is_null_or_undefined = false;
1059
1060 if (is_setter_argument) {
1061 if (SetterAcceptsNull(options, field)) {
1062 jstype = "?" + jstype;
1063 is_null_or_undefined = true;
1064 }
1065
1066 if (SetterAcceptsUndefined(options, field)) {
1067 jstype += "|undefined";
1068 is_null_or_undefined = true;
1069 }
1070 } else if (force_present) {
1071 // Don't add null or undefined.
1072 } else {
1073 if (DeclaredReturnTypeIsNullable(options, field)) {
1074 jstype = "?" + jstype;
1075 is_null_or_undefined = true;
1076 }
1077 }
1078
1079 if (!is_null_or_undefined && !IsPrimitive(jstype)) {
1080 jstype = "!" + jstype;
1081 }
1082
1083 return jstype;
1084 }
1085
JSBinaryReaderMethodType(const FieldDescriptor * field)1086 std::string JSBinaryReaderMethodType(const FieldDescriptor* field) {
1087 std::string name = field->type_name();
1088 if (name[0] >= 'a' && name[0] <= 'z') {
1089 name[0] = (name[0] - 'a') + 'A';
1090 }
1091 return IsIntegralFieldWithStringJSType(field) ? (name + "String") : name;
1092 }
1093
JSBinaryReadWriteMethodName(const FieldDescriptor * field,bool is_writer)1094 std::string JSBinaryReadWriteMethodName(const FieldDescriptor* field,
1095 bool is_writer) {
1096 std::string name = JSBinaryReaderMethodType(field);
1097 if (field->is_packed()) {
1098 name = "Packed" + name;
1099 } else if (is_writer && field->is_repeated()) {
1100 name = "Repeated" + name;
1101 }
1102 return name;
1103 }
1104
JSBinaryReaderMethodName(const GeneratorOptions & options,const FieldDescriptor * field)1105 std::string JSBinaryReaderMethodName(const GeneratorOptions& options,
1106 const FieldDescriptor* field) {
1107 return "jspb.BinaryReader.prototype.read" +
1108 JSBinaryReadWriteMethodName(field, /* is_writer = */ false);
1109 }
1110
JSBinaryWriterMethodName(const GeneratorOptions & options,const FieldDescriptor * field)1111 std::string JSBinaryWriterMethodName(const GeneratorOptions& options,
1112 const FieldDescriptor* field) {
1113 if (field->containing_type() &&
1114 field->containing_type()->options().message_set_wire_format()) {
1115 return "jspb.BinaryWriter.prototype.writeMessageSet";
1116 }
1117 return "jspb.BinaryWriter.prototype.write" +
1118 JSBinaryReadWriteMethodName(field, /* is_writer = */ true);
1119 }
1120
JSReturnClause(const FieldDescriptor * desc)1121 std::string JSReturnClause(const FieldDescriptor* desc) {
1122 return "";
1123 }
1124
JSTypeTag(const FieldDescriptor * desc)1125 std::string JSTypeTag(const FieldDescriptor* desc) {
1126 switch (desc->type()) {
1127 case FieldDescriptor::TYPE_DOUBLE:
1128 case FieldDescriptor::TYPE_FLOAT:
1129 return "Float";
1130 case FieldDescriptor::TYPE_INT32:
1131 case FieldDescriptor::TYPE_UINT32:
1132 case FieldDescriptor::TYPE_INT64:
1133 case FieldDescriptor::TYPE_UINT64:
1134 case FieldDescriptor::TYPE_FIXED32:
1135 case FieldDescriptor::TYPE_FIXED64:
1136 case FieldDescriptor::TYPE_SINT32:
1137 case FieldDescriptor::TYPE_SINT64:
1138 case FieldDescriptor::TYPE_SFIXED32:
1139 case FieldDescriptor::TYPE_SFIXED64:
1140 if (IsIntegralFieldWithStringJSType(desc)) {
1141 return "StringInt";
1142 } else {
1143 return "Int";
1144 }
1145 case FieldDescriptor::TYPE_BOOL:
1146 return "Boolean";
1147 case FieldDescriptor::TYPE_STRING:
1148 return "String";
1149 case FieldDescriptor::TYPE_BYTES:
1150 return "Bytes";
1151 case FieldDescriptor::TYPE_ENUM:
1152 return "Enum";
1153 default:
1154 assert(false);
1155 }
1156 return "";
1157 }
1158
JSReturnDoc(const GeneratorOptions & options,const FieldDescriptor * desc)1159 std::string JSReturnDoc(const GeneratorOptions& options,
1160 const FieldDescriptor* desc) {
1161 return "";
1162 }
1163
HasRepeatedFields(const GeneratorOptions & options,const Descriptor * desc)1164 bool HasRepeatedFields(const GeneratorOptions& options,
1165 const Descriptor* desc) {
1166 for (int i = 0; i < desc->field_count(); i++) {
1167 if (desc->field(i)->is_repeated() && !desc->field(i)->is_map()) {
1168 return true;
1169 }
1170 }
1171 return false;
1172 }
1173
1174 static const char* kRepeatedFieldArrayName = ".repeatedFields_";
1175
RepeatedFieldsArrayName(const GeneratorOptions & options,const Descriptor * desc)1176 std::string RepeatedFieldsArrayName(const GeneratorOptions& options,
1177 const Descriptor* desc) {
1178 return HasRepeatedFields(options, desc)
1179 ? (GetMessagePath(options, desc) + kRepeatedFieldArrayName)
1180 : "null";
1181 }
1182
HasOneofFields(const Descriptor * desc)1183 bool HasOneofFields(const Descriptor* desc) {
1184 for (int i = 0; i < desc->field_count(); i++) {
1185 if (desc->field(i)->containing_oneof()) {
1186 return true;
1187 }
1188 }
1189 return false;
1190 }
1191
1192 static const char* kOneofGroupArrayName = ".oneofGroups_";
1193
OneofFieldsArrayName(const GeneratorOptions & options,const Descriptor * desc)1194 std::string OneofFieldsArrayName(const GeneratorOptions& options,
1195 const Descriptor* desc) {
1196 return HasOneofFields(desc)
1197 ? (GetMessagePath(options, desc) + kOneofGroupArrayName)
1198 : "null";
1199 }
1200
RepeatedFieldNumberList(const GeneratorOptions & options,const Descriptor * desc)1201 std::string RepeatedFieldNumberList(const GeneratorOptions& options,
1202 const Descriptor* desc) {
1203 std::vector<std::string> numbers;
1204 for (int i = 0; i < desc->field_count(); i++) {
1205 if (desc->field(i)->is_repeated() && !desc->field(i)->is_map()) {
1206 numbers.push_back(JSFieldIndex(desc->field(i)));
1207 }
1208 }
1209 return "[" + Join(numbers, ",") + "]";
1210 }
1211
OneofGroupList(const Descriptor * desc)1212 std::string OneofGroupList(const Descriptor* desc) {
1213 // List of arrays (one per oneof), each of which is a list of field indices
1214 std::vector<std::string> oneof_entries;
1215 for (int i = 0; i < desc->oneof_decl_count(); i++) {
1216 const OneofDescriptor* oneof = desc->oneof_decl(i);
1217 if (IgnoreOneof(oneof)) {
1218 continue;
1219 }
1220
1221 std::vector<std::string> oneof_fields;
1222 for (int j = 0; j < oneof->field_count(); j++) {
1223 if (IgnoreField(oneof->field(j))) {
1224 continue;
1225 }
1226 oneof_fields.push_back(JSFieldIndex(oneof->field(j)));
1227 }
1228 oneof_entries.push_back("[" + Join(oneof_fields, ",") + "]");
1229 }
1230 return "[" + Join(oneof_entries, ",") + "]";
1231 }
1232
JSOneofArray(const GeneratorOptions & options,const FieldDescriptor * field)1233 std::string JSOneofArray(const GeneratorOptions& options,
1234 const FieldDescriptor* field) {
1235 return OneofFieldsArrayName(options, field->containing_type()) + "[" +
1236 JSOneofIndex(field->containing_oneof()) + "]";
1237 }
1238
RelativeTypeName(const FieldDescriptor * field)1239 std::string RelativeTypeName(const FieldDescriptor* field) {
1240 assert(field->cpp_type() == FieldDescriptor::CPPTYPE_ENUM ||
1241 field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE);
1242 // For a field with an enum or message type, compute a name relative to the
1243 // path name of the message type containing this field.
1244 std::string package = field->file()->package();
1245 std::string containing_type = field->containing_type()->full_name() + ".";
1246 std::string type = (field->cpp_type() == FieldDescriptor::CPPTYPE_ENUM)
1247 ? field->enum_type()->full_name()
1248 : field->message_type()->full_name();
1249
1250 // |prefix| is advanced as we find separators '.' past the common package
1251 // prefix that yield common prefixes in the containing type's name and this
1252 // type's name.
1253 int prefix = 0;
1254 for (int i = 0; i < type.size() && i < containing_type.size(); i++) {
1255 if (type[i] != containing_type[i]) {
1256 break;
1257 }
1258 if (type[i] == '.' && i >= package.size()) {
1259 prefix = i + 1;
1260 }
1261 }
1262
1263 return type.substr(prefix);
1264 }
1265
JSExtensionsObjectName(const GeneratorOptions & options,const FileDescriptor * from_file,const Descriptor * desc)1266 std::string JSExtensionsObjectName(const GeneratorOptions& options,
1267 const FileDescriptor* from_file,
1268 const Descriptor* desc) {
1269 if (desc->full_name() == "google.protobuf.bridge.MessageSet") {
1270 // TODO(haberman): fix this for the kImportCommonJs case.
1271 return "jspb.Message.messageSetExtensions";
1272 } else {
1273 return MaybeCrossFileRef(options, from_file, desc) + ".extensions";
1274 }
1275 }
1276
1277 static const int kMapKeyField = 1;
1278 static const int kMapValueField = 2;
1279
MapFieldKey(const FieldDescriptor * field)1280 const FieldDescriptor* MapFieldKey(const FieldDescriptor* field) {
1281 assert(field->is_map());
1282 return field->message_type()->FindFieldByNumber(kMapKeyField);
1283 }
1284
MapFieldValue(const FieldDescriptor * field)1285 const FieldDescriptor* MapFieldValue(const FieldDescriptor* field) {
1286 assert(field->is_map());
1287 return field->message_type()->FindFieldByNumber(kMapValueField);
1288 }
1289
FieldDefinition(const GeneratorOptions & options,const FieldDescriptor * field)1290 std::string FieldDefinition(const GeneratorOptions& options,
1291 const FieldDescriptor* field) {
1292 if (field->is_map()) {
1293 const FieldDescriptor* key_field = MapFieldKey(field);
1294 const FieldDescriptor* value_field = MapFieldValue(field);
1295 std::string key_type = ProtoTypeName(options, key_field);
1296 std::string value_type;
1297 if (value_field->type() == FieldDescriptor::TYPE_ENUM ||
1298 value_field->type() == FieldDescriptor::TYPE_MESSAGE) {
1299 value_type = RelativeTypeName(value_field);
1300 } else {
1301 value_type = ProtoTypeName(options, value_field);
1302 }
1303 return StringPrintf("map<%s, %s> %s = %d;", key_type.c_str(),
1304 value_type.c_str(), field->name().c_str(),
1305 field->number());
1306 } else {
1307 std::string qualifier =
1308 field->is_repeated() ? "repeated"
1309 : (field->is_optional() ? "optional" : "required");
1310 std::string type, name;
1311 if (field->type() == FieldDescriptor::TYPE_ENUM ||
1312 field->type() == FieldDescriptor::TYPE_MESSAGE) {
1313 type = RelativeTypeName(field);
1314 name = field->name();
1315 } else if (field->type() == FieldDescriptor::TYPE_GROUP) {
1316 type = "group";
1317 name = field->message_type()->name();
1318 } else {
1319 type = ProtoTypeName(options, field);
1320 name = field->name();
1321 }
1322 return StringPrintf("%s %s %s = %d;", qualifier.c_str(), type.c_str(),
1323 name.c_str(), field->number());
1324 }
1325 }
1326
FieldComments(const FieldDescriptor * field,BytesMode bytes_mode)1327 std::string FieldComments(const FieldDescriptor* field, BytesMode bytes_mode) {
1328 std::string comments;
1329 if (field->type() == FieldDescriptor::TYPE_BYTES && bytes_mode == BYTES_U8) {
1330 comments +=
1331 " * Note that Uint8Array is not supported on all browsers.\n"
1332 " * @see http://caniuse.com/Uint8Array\n";
1333 }
1334 return comments;
1335 }
1336
ShouldGenerateExtension(const FieldDescriptor * field)1337 bool ShouldGenerateExtension(const FieldDescriptor* field) {
1338 return field->is_extension() && !IgnoreField(field);
1339 }
1340
HasExtensions(const Descriptor * desc)1341 bool HasExtensions(const Descriptor* desc) {
1342 for (int i = 0; i < desc->extension_count(); i++) {
1343 if (ShouldGenerateExtension(desc->extension(i))) {
1344 return true;
1345 }
1346 }
1347 for (int i = 0; i < desc->nested_type_count(); i++) {
1348 if (HasExtensions(desc->nested_type(i))) {
1349 return true;
1350 }
1351 }
1352 return false;
1353 }
1354
HasExtensions(const FileDescriptor * file)1355 bool HasExtensions(const FileDescriptor* file) {
1356 for (int i = 0; i < file->extension_count(); i++) {
1357 if (ShouldGenerateExtension(file->extension(i))) {
1358 return true;
1359 }
1360 }
1361 for (int i = 0; i < file->message_type_count(); i++) {
1362 if (HasExtensions(file->message_type(i))) {
1363 return true;
1364 }
1365 }
1366 return false;
1367 }
1368
HasMap(const GeneratorOptions & options,const Descriptor * desc)1369 bool HasMap(const GeneratorOptions& options, const Descriptor* desc) {
1370 for (int i = 0; i < desc->field_count(); i++) {
1371 if (desc->field(i)->is_map()) {
1372 return true;
1373 }
1374 }
1375 for (int i = 0; i < desc->nested_type_count(); i++) {
1376 if (HasMap(options, desc->nested_type(i))) {
1377 return true;
1378 }
1379 }
1380 return false;
1381 }
1382
FileHasMap(const GeneratorOptions & options,const FileDescriptor * desc)1383 bool FileHasMap(const GeneratorOptions& options, const FileDescriptor* desc) {
1384 for (int i = 0; i < desc->message_type_count(); i++) {
1385 if (HasMap(options, desc->message_type(i))) {
1386 return true;
1387 }
1388 }
1389 return false;
1390 }
1391
IsExtendable(const Descriptor * desc)1392 bool IsExtendable(const Descriptor* desc) {
1393 return desc->extension_range_count() > 0;
1394 }
1395
1396 // Returns the max index in the underlying data storage array beyond which the
1397 // extension object is used.
GetPivot(const Descriptor * desc)1398 std::string GetPivot(const Descriptor* desc) {
1399 static const int kDefaultPivot = 500;
1400
1401 // Find the max field number
1402 int max_field_number = 0;
1403 for (int i = 0; i < desc->field_count(); i++) {
1404 if (!IgnoreField(desc->field(i)) &&
1405 desc->field(i)->number() > max_field_number) {
1406 max_field_number = desc->field(i)->number();
1407 }
1408 }
1409
1410 int pivot = -1;
1411 if (IsExtendable(desc) || (max_field_number >= kDefaultPivot)) {
1412 pivot = ((max_field_number + 1) < kDefaultPivot) ? (max_field_number + 1)
1413 : kDefaultPivot;
1414 }
1415
1416 return StrCat(pivot);
1417 }
1418
1419 // Whether this field represents presence. For fields with presence, we
1420 // generate extra methods (clearFoo() and hasFoo()) for this field.
HasFieldPresence(const GeneratorOptions & options,const FieldDescriptor * field)1421 bool HasFieldPresence(const GeneratorOptions& options,
1422 const FieldDescriptor* field) {
1423 if (field->is_repeated() || field->is_map()) {
1424 // We say repeated fields and maps don't have presence, but we still do
1425 // generate clearFoo() methods for them through a special case elsewhere.
1426 return false;
1427 }
1428
1429 if (UseBrokenPresenceSemantics(options, field)) {
1430 // Proto3 files with broken presence semantics have field presence.
1431 return true;
1432 }
1433
1434 return field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE ||
1435 field->containing_oneof() != NULL ||
1436 field->file()->syntax() == FileDescriptor::SYNTAX_PROTO2;
1437 }
1438
1439 // We use this to implement the semantics that same file can be generated
1440 // multiple times, but only the last one keep the short name. Others all use
1441 // long name with extra information to distinguish (For message and enum, the
1442 // extra information is package name, for file level extension, the extra
1443 // information is proto's filename).
1444 // We never actually write the files, but we keep a set of which descriptors
1445 // were the final one for a given filename.
1446 class FileDeduplicator {
1447 public:
FileDeduplicator(const GeneratorOptions & options)1448 explicit FileDeduplicator(const GeneratorOptions& options)
1449 : error_on_conflict_(options.error_on_name_conflict) {}
1450
1451 // params:
1452 // filenames: a pair of {short filename, full filename}
1453 // (short filename don't have extra information, full filename
1454 // contains extra information)
1455 // desc: The Descriptor or SCC pointer or EnumDescriptor.
1456 // error: The returned error information.
AddFile(const std::pair<std::string,std::string> filenames,const void * desc,std::string * error)1457 bool AddFile(const std::pair<std::string, std::string> filenames,
1458 const void* desc, std::string* error) {
1459 if (descs_by_shortname_.find(filenames.first) !=
1460 descs_by_shortname_.end()) {
1461 if (error_on_conflict_) {
1462 *error = "Name conflict: file name " + filenames.first +
1463 " would be generated by two descriptors";
1464 return false;
1465 }
1466 // Change old pointer's actual name to full name.
1467 auto short_name_desc = descs_by_shortname_[filenames.first];
1468 allowed_descs_actual_name_[short_name_desc] =
1469 allowed_descs_full_name_[short_name_desc];
1470 }
1471 descs_by_shortname_[filenames.first] = desc;
1472 allowed_descs_actual_name_[desc] = filenames.first;
1473 allowed_descs_full_name_[desc] = filenames.second;
1474
1475 return true;
1476 }
1477
GetAllowedMap(std::map<const void *,std::string> * allowed_set)1478 void GetAllowedMap(std::map<const void*, std::string>* allowed_set) {
1479 *allowed_set = allowed_descs_actual_name_;
1480 }
1481
1482 private:
1483 bool error_on_conflict_;
1484 // The map that restores all the descs that are using short name as filename.
1485 std::map<std::string, const void*> descs_by_shortname_;
1486 // The final actual filename map.
1487 std::map<const void*, std::string> allowed_descs_actual_name_;
1488 // The full name map.
1489 std::map<const void*, std::string> allowed_descs_full_name_;
1490 };
1491
DepthFirstSearch(const FileDescriptor * file,std::vector<const FileDescriptor * > * list,std::set<const FileDescriptor * > * seen)1492 void DepthFirstSearch(const FileDescriptor* file,
1493 std::vector<const FileDescriptor*>* list,
1494 std::set<const FileDescriptor*>* seen) {
1495 if (!seen->insert(file).second) {
1496 return;
1497 }
1498
1499 // Add all dependencies.
1500 for (int i = 0; i < file->dependency_count(); i++) {
1501 DepthFirstSearch(file->dependency(i), list, seen);
1502 }
1503
1504 // Add this file.
1505 list->push_back(file);
1506 }
1507
1508 // A functor for the predicate to remove_if() below. Returns true if a given
1509 // FileDescriptor is not in the given set.
1510 class NotInSet {
1511 public:
NotInSet(const std::set<const FileDescriptor * > & file_set)1512 explicit NotInSet(const std::set<const FileDescriptor*>& file_set)
1513 : file_set_(file_set) {}
1514
operator ()(const FileDescriptor * file)1515 bool operator()(const FileDescriptor* file) {
1516 return file_set_.count(file) == 0;
1517 }
1518
1519 private:
1520 const std::set<const FileDescriptor*>& file_set_;
1521 };
1522
1523 // This function generates an ordering of the input FileDescriptors that matches
1524 // the logic of the old code generator. The order is significant because two
1525 // different input files can generate the same output file, and the last one
1526 // needs to win.
GenerateJspbFileOrder(const std::vector<const FileDescriptor * > & input,std::vector<const FileDescriptor * > * ordered)1527 void GenerateJspbFileOrder(const std::vector<const FileDescriptor*>& input,
1528 std::vector<const FileDescriptor*>* ordered) {
1529 // First generate an ordering of all reachable files (including dependencies)
1530 // with depth-first search. This mimics the behavior of --include_imports,
1531 // which is what the old codegen used.
1532 ordered->clear();
1533 std::set<const FileDescriptor*> seen;
1534 std::set<const FileDescriptor*> input_set;
1535 for (int i = 0; i < input.size(); i++) {
1536 DepthFirstSearch(input[i], ordered, &seen);
1537 input_set.insert(input[i]);
1538 }
1539
1540 // Now remove the entries that are not actually in our input list.
1541 ordered->erase(
1542 std::remove_if(ordered->begin(), ordered->end(), NotInSet(input_set)),
1543 ordered->end());
1544 }
1545
1546 // If we're generating code in file-per-type mode, avoid overwriting files
1547 // by choosing the last descriptor that writes each filename and permitting
1548 // only those to generate code.
1549
1550 struct DepsGenerator {
operator ()google::protobuf::compiler::js::__anonb14e560c0111::DepsGenerator1551 std::vector<const Descriptor*> operator()(const Descriptor* desc) const {
1552 std::vector<const Descriptor*> deps;
1553 auto maybe_add = [&](const Descriptor* d) {
1554 if (d) deps.push_back(d);
1555 };
1556 for (int i = 0; i < desc->field_count(); i++) {
1557 if (!IgnoreField(desc->field(i))) {
1558 maybe_add(desc->field(i)->message_type());
1559 }
1560 }
1561 for (int i = 0; i < desc->extension_count(); i++) {
1562 maybe_add(desc->extension(i)->message_type());
1563 maybe_add(desc->extension(i)->containing_type());
1564 }
1565 for (int i = 0; i < desc->nested_type_count(); i++) {
1566 maybe_add(desc->nested_type(i));
1567 }
1568 maybe_add(desc->containing_type());
1569
1570 return deps;
1571 }
1572 };
1573
GenerateJspbAllowedMap(const GeneratorOptions & options,const std::vector<const FileDescriptor * > & files,std::map<const void *,std::string> * allowed_set,SCCAnalyzer<DepsGenerator> * analyzer,std::string * error)1574 bool GenerateJspbAllowedMap(const GeneratorOptions& options,
1575 const std::vector<const FileDescriptor*>& files,
1576 std::map<const void*, std::string>* allowed_set,
1577 SCCAnalyzer<DepsGenerator>* analyzer,
1578 std::string* error) {
1579 std::vector<const FileDescriptor*> files_ordered;
1580 GenerateJspbFileOrder(files, &files_ordered);
1581
1582 // Choose the last descriptor for each filename.
1583 FileDeduplicator dedup(options);
1584 std::set<const SCC*> added;
1585 for (int i = 0; i < files_ordered.size(); i++) {
1586 for (int j = 0; j < files_ordered[i]->message_type_count(); j++) {
1587 const Descriptor* desc = files_ordered[i]->message_type(j);
1588 if (added.insert(analyzer->GetSCC(desc)).second &&
1589 !dedup.AddFile(
1590 std::make_pair(
1591 GetMessagesFileName(options, analyzer->GetSCC(desc), false),
1592 GetMessagesFileName(options, analyzer->GetSCC(desc), true)),
1593 analyzer->GetSCC(desc), error)) {
1594 return false;
1595 }
1596 }
1597 for (int j = 0; j < files_ordered[i]->enum_type_count(); j++) {
1598 const EnumDescriptor* desc = files_ordered[i]->enum_type(j);
1599 if (!dedup.AddFile(std::make_pair(GetEnumFileName(options, desc, false),
1600 GetEnumFileName(options, desc, true)),
1601 desc, error)) {
1602 return false;
1603 }
1604 }
1605
1606 // Pull out all free-floating extensions and generate files for those too.
1607 bool has_extension = false;
1608
1609 for (int j = 0; j < files_ordered[i]->extension_count(); j++) {
1610 if (ShouldGenerateExtension(files_ordered[i]->extension(j))) {
1611 has_extension = true;
1612 }
1613 }
1614
1615 if (has_extension) {
1616 if (!dedup.AddFile(
1617 std::make_pair(
1618 GetExtensionFileName(options, files_ordered[i], false),
1619 GetExtensionFileName(options, files_ordered[i], true)),
1620 files_ordered[i], error)) {
1621 return false;
1622 }
1623 }
1624 }
1625
1626 dedup.GetAllowedMap(allowed_set);
1627
1628 return true;
1629 }
1630
1631 // Embeds base64 encoded GeneratedCodeInfo proto in a comment at the end of
1632 // file.
EmbedCodeAnnotations(const GeneratedCodeInfo & annotations,io::Printer * printer)1633 void EmbedCodeAnnotations(const GeneratedCodeInfo& annotations,
1634 io::Printer* printer) {
1635 // Serialize annotations proto into base64 string.
1636 std::string meta_content;
1637 annotations.SerializeToString(&meta_content);
1638 std::string meta_64;
1639 Base64Escape(meta_content, &meta_64);
1640
1641 // Print base64 encoded annotations at the end of output file in
1642 // a comment.
1643 printer->Print("\n// Below is base64 encoded GeneratedCodeInfo proto");
1644 printer->Print("\n// $encoded_proto$\n", "encoded_proto", meta_64);
1645 }
1646
IsWellKnownTypeFile(const FileDescriptor * file)1647 bool IsWellKnownTypeFile(const FileDescriptor* file) {
1648 return HasPrefixString(file->name(), "google/protobuf/");
1649 }
1650
1651 } // anonymous namespace
1652
GenerateHeader(const GeneratorOptions & options,const FileDescriptor * file,io::Printer * printer) const1653 void Generator::GenerateHeader(const GeneratorOptions& options,
1654 const FileDescriptor* file,
1655 io::Printer* printer) const {
1656 if (file != nullptr) {
1657 printer->Print("// source: $filename$\n", "filename", file->name());
1658 }
1659 printer->Print(
1660 "/**\n"
1661 " * @fileoverview\n"
1662 " * @enhanceable\n"
1663 " * @suppress {messageConventions} JS Compiler reports an "
1664 "error if a variable or\n"
1665 " * field starts with 'MSG_' and isn't a translatable "
1666 "message.\n"
1667 " * @public\n"
1668 " */\n"
1669 "// GENERATED CODE -- DO NOT EDIT!\n"
1670 "\n");
1671 }
1672
FindProvidesForFile(const GeneratorOptions & options,io::Printer * printer,const FileDescriptor * file,std::set<std::string> * provided) const1673 void Generator::FindProvidesForFile(const GeneratorOptions& options,
1674 io::Printer* printer,
1675 const FileDescriptor* file,
1676 std::set<std::string>* provided) const {
1677 for (int i = 0; i < file->message_type_count(); i++) {
1678 FindProvidesForMessage(options, printer, file->message_type(i), provided);
1679 }
1680 for (int i = 0; i < file->enum_type_count(); i++) {
1681 FindProvidesForEnum(options, printer, file->enum_type(i), provided);
1682 }
1683 }
1684
FindProvides(const GeneratorOptions & options,io::Printer * printer,const std::vector<const FileDescriptor * > & files,std::set<std::string> * provided) const1685 void Generator::FindProvides(const GeneratorOptions& options,
1686 io::Printer* printer,
1687 const std::vector<const FileDescriptor*>& files,
1688 std::set<std::string>* provided) const {
1689 for (int i = 0; i < files.size(); i++) {
1690 FindProvidesForFile(options, printer, files[i], provided);
1691 }
1692
1693 printer->Print("\n");
1694 }
1695
FindProvidesForOneOfEnum(const GeneratorOptions & options,const OneofDescriptor * oneof,std::set<std::string> * provided)1696 void FindProvidesForOneOfEnum(const GeneratorOptions& options,
1697 const OneofDescriptor* oneof,
1698 std::set<std::string>* provided) {
1699 std::string name = GetMessagePath(options, oneof->containing_type()) + "." +
1700 JSOneofName(oneof) + "Case";
1701 provided->insert(name);
1702 }
1703
FindProvidesForOneOfEnums(const GeneratorOptions & options,io::Printer * printer,const Descriptor * desc,std::set<std::string> * provided)1704 void FindProvidesForOneOfEnums(const GeneratorOptions& options,
1705 io::Printer* printer, const Descriptor* desc,
1706 std::set<std::string>* provided) {
1707 if (HasOneofFields(desc)) {
1708 for (int i = 0; i < desc->oneof_decl_count(); i++) {
1709 if (IgnoreOneof(desc->oneof_decl(i))) {
1710 continue;
1711 }
1712 FindProvidesForOneOfEnum(options, desc->oneof_decl(i), provided);
1713 }
1714 }
1715 }
1716
FindProvidesForMessage(const GeneratorOptions & options,io::Printer * printer,const Descriptor * desc,std::set<std::string> * provided) const1717 void Generator::FindProvidesForMessage(const GeneratorOptions& options,
1718 io::Printer* printer,
1719 const Descriptor* desc,
1720 std::set<std::string>* provided) const {
1721 if (IgnoreMessage(desc)) {
1722 return;
1723 }
1724
1725 std::string name = GetMessagePath(options, desc);
1726 provided->insert(name);
1727
1728 for (int i = 0; i < desc->enum_type_count(); i++) {
1729 FindProvidesForEnum(options, printer, desc->enum_type(i), provided);
1730 }
1731
1732 FindProvidesForOneOfEnums(options, printer, desc, provided);
1733
1734 for (int i = 0; i < desc->nested_type_count(); i++) {
1735 FindProvidesForMessage(options, printer, desc->nested_type(i), provided);
1736 }
1737 }
FindProvidesForEnum(const GeneratorOptions & options,io::Printer * printer,const EnumDescriptor * enumdesc,std::set<std::string> * provided) const1738 void Generator::FindProvidesForEnum(const GeneratorOptions& options,
1739 io::Printer* printer,
1740 const EnumDescriptor* enumdesc,
1741 std::set<std::string>* provided) const {
1742 std::string name = GetEnumPath(options, enumdesc);
1743 provided->insert(name);
1744 }
1745
FindProvidesForFields(const GeneratorOptions & options,io::Printer * printer,const std::vector<const FieldDescriptor * > & fields,std::set<std::string> * provided) const1746 void Generator::FindProvidesForFields(
1747 const GeneratorOptions& options, io::Printer* printer,
1748 const std::vector<const FieldDescriptor*>& fields,
1749 std::set<std::string>* provided) const {
1750 for (int i = 0; i < fields.size(); i++) {
1751 const FieldDescriptor* field = fields[i];
1752
1753 if (IgnoreField(field)) {
1754 continue;
1755 }
1756
1757 std::string name = GetNamespace(options, field->file()) + "." +
1758 JSObjectFieldName(options, field);
1759 provided->insert(name);
1760 }
1761 }
1762
GenerateProvides(const GeneratorOptions & options,io::Printer * printer,std::set<std::string> * provided) const1763 void Generator::GenerateProvides(const GeneratorOptions& options,
1764 io::Printer* printer,
1765 std::set<std::string>* provided) const {
1766 for (std::set<std::string>::iterator it = provided->begin();
1767 it != provided->end(); ++it) {
1768 if (options.import_style == GeneratorOptions::kImportClosure) {
1769 printer->Print("goog.provide('$name$');\n", "name", *it);
1770 } else {
1771 // We aren't using Closure's import system, but we use goog.exportSymbol()
1772 // to construct the expected tree of objects, eg.
1773 //
1774 // goog.exportSymbol('foo.bar.Baz', null, this);
1775 //
1776 // // Later generated code expects foo.bar = {} to exist:
1777 // foo.bar.Baz = function() { /* ... */ }
1778
1779 // Do not use global scope in strict mode
1780 if (options.import_style == GeneratorOptions::kImportCommonJsStrict) {
1781 std::string namespaceObject = *it;
1782 // Remove "proto." from the namespace object
1783 GOOGLE_CHECK_EQ(0, namespaceObject.compare(0, 6, "proto."));
1784 namespaceObject.erase(0, 6);
1785 printer->Print("goog.exportSymbol('$name$', null, proto);\n", "name",
1786 namespaceObject);
1787 } else {
1788 printer->Print("goog.exportSymbol('$name$', null, global);\n", "name",
1789 *it);
1790 }
1791 }
1792 }
1793 }
1794
GenerateRequiresForSCC(const GeneratorOptions & options,io::Printer * printer,const SCC * scc,std::set<std::string> * provided) const1795 void Generator::GenerateRequiresForSCC(const GeneratorOptions& options,
1796 io::Printer* printer, const SCC* scc,
1797 std::set<std::string>* provided) const {
1798 std::set<std::string> required;
1799 std::set<std::string> forwards;
1800 bool have_message = false;
1801 bool has_extension = false;
1802 bool has_map = false;
1803 for (auto desc : scc->descriptors) {
1804 if (desc->containing_type() == nullptr) {
1805 FindRequiresForMessage(options, desc, &required, &forwards,
1806 &have_message);
1807 has_extension = (has_extension || HasExtensions(desc));
1808 has_map = (has_map || HasMap(options, desc));
1809 }
1810 }
1811
1812 GenerateRequiresImpl(options, printer, &required, &forwards, provided,
1813 /* require_jspb = */ have_message,
1814 /* require_extension = */ has_extension,
1815 /* require_map = */ has_map);
1816 }
1817
GenerateRequiresForLibrary(const GeneratorOptions & options,io::Printer * printer,const std::vector<const FileDescriptor * > & files,std::set<std::string> * provided) const1818 void Generator::GenerateRequiresForLibrary(
1819 const GeneratorOptions& options, io::Printer* printer,
1820 const std::vector<const FileDescriptor*>& files,
1821 std::set<std::string>* provided) const {
1822 GOOGLE_CHECK_EQ(options.import_style, GeneratorOptions::kImportClosure);
1823 // For Closure imports we need to import every message type individually.
1824 std::set<std::string> required;
1825 std::set<std::string> forwards;
1826 bool have_extensions = false;
1827 bool have_map = false;
1828 bool have_message = false;
1829
1830 for (int i = 0; i < files.size(); i++) {
1831 for (int j = 0; j < files[i]->message_type_count(); j++) {
1832 const Descriptor* desc = files[i]->message_type(j);
1833 if (!IgnoreMessage(desc)) {
1834 FindRequiresForMessage(options, desc, &required, &forwards,
1835 &have_message);
1836 }
1837 }
1838
1839 if (!have_extensions && HasExtensions(files[i])) {
1840 have_extensions = true;
1841 }
1842
1843 if (!have_map && FileHasMap(options, files[i])) {
1844 have_map = true;
1845 }
1846
1847 for (int j = 0; j < files[i]->extension_count(); j++) {
1848 const FieldDescriptor* extension = files[i]->extension(j);
1849 if (IgnoreField(extension)) {
1850 continue;
1851 }
1852 if (extension->containing_type()->full_name() !=
1853 "google.protobuf.bridge.MessageSet") {
1854 required.insert(GetMessagePath(options, extension->containing_type()));
1855 }
1856 FindRequiresForField(options, extension, &required, &forwards);
1857 have_extensions = true;
1858 }
1859 }
1860
1861 GenerateRequiresImpl(options, printer, &required, &forwards, provided,
1862 /* require_jspb = */ have_message,
1863 /* require_extension = */ have_extensions,
1864 /* require_map = */ have_map);
1865 }
1866
GenerateRequiresForExtensions(const GeneratorOptions & options,io::Printer * printer,const std::vector<const FieldDescriptor * > & fields,std::set<std::string> * provided) const1867 void Generator::GenerateRequiresForExtensions(
1868 const GeneratorOptions& options, io::Printer* printer,
1869 const std::vector<const FieldDescriptor*>& fields,
1870 std::set<std::string>* provided) const {
1871 std::set<std::string> required;
1872 std::set<std::string> forwards;
1873 for (int i = 0; i < fields.size(); i++) {
1874 const FieldDescriptor* field = fields[i];
1875 if (IgnoreField(field)) {
1876 continue;
1877 }
1878 FindRequiresForExtension(options, field, &required, &forwards);
1879 }
1880
1881 GenerateRequiresImpl(options, printer, &required, &forwards, provided,
1882 /* require_jspb = */ false,
1883 /* require_extension = */ fields.size() > 0,
1884 /* require_map = */ false);
1885 }
1886
GenerateRequiresImpl(const GeneratorOptions & options,io::Printer * printer,std::set<std::string> * required,std::set<std::string> * forwards,std::set<std::string> * provided,bool require_jspb,bool require_extension,bool require_map) const1887 void Generator::GenerateRequiresImpl(const GeneratorOptions& options,
1888 io::Printer* printer,
1889 std::set<std::string>* required,
1890 std::set<std::string>* forwards,
1891 std::set<std::string>* provided,
1892 bool require_jspb, bool require_extension,
1893 bool require_map) const {
1894 if (require_jspb) {
1895 required->insert("jspb.Message");
1896 required->insert("jspb.BinaryReader");
1897 required->insert("jspb.BinaryWriter");
1898 }
1899 if (require_extension) {
1900 required->insert("jspb.ExtensionFieldBinaryInfo");
1901 required->insert("jspb.ExtensionFieldInfo");
1902 }
1903 if (require_map) {
1904 required->insert("jspb.Map");
1905 }
1906
1907 std::set<std::string>::iterator it;
1908 for (it = required->begin(); it != required->end(); ++it) {
1909 if (provided->find(*it) != provided->end()) {
1910 continue;
1911 }
1912 printer->Print("goog.require('$name$');\n", "name", *it);
1913 }
1914
1915 printer->Print("\n");
1916
1917 for (it = forwards->begin(); it != forwards->end(); ++it) {
1918 if (provided->find(*it) != provided->end()) {
1919 continue;
1920 }
1921 printer->Print("goog.forwardDeclare('$name$');\n", "name", *it);
1922 }
1923 }
1924
NamespaceOnly(const Descriptor * desc)1925 bool NamespaceOnly(const Descriptor* desc) {
1926 return false;
1927 }
1928
FindRequiresForMessage(const GeneratorOptions & options,const Descriptor * desc,std::set<std::string> * required,std::set<std::string> * forwards,bool * have_message) const1929 void Generator::FindRequiresForMessage(const GeneratorOptions& options,
1930 const Descriptor* desc,
1931 std::set<std::string>* required,
1932 std::set<std::string>* forwards,
1933 bool* have_message) const {
1934
1935 if (!NamespaceOnly(desc)) {
1936 *have_message = true;
1937 for (int i = 0; i < desc->field_count(); i++) {
1938 const FieldDescriptor* field = desc->field(i);
1939 if (IgnoreField(field)) {
1940 continue;
1941 }
1942 FindRequiresForField(options, field, required, forwards);
1943 }
1944 }
1945
1946 for (int i = 0; i < desc->extension_count(); i++) {
1947 const FieldDescriptor* field = desc->extension(i);
1948 if (IgnoreField(field)) {
1949 continue;
1950 }
1951 FindRequiresForExtension(options, field, required, forwards);
1952 }
1953
1954 for (int i = 0; i < desc->nested_type_count(); i++) {
1955 FindRequiresForMessage(options, desc->nested_type(i), required, forwards,
1956 have_message);
1957 }
1958 }
1959
FindRequiresForField(const GeneratorOptions & options,const FieldDescriptor * field,std::set<std::string> * required,std::set<std::string> * forwards) const1960 void Generator::FindRequiresForField(const GeneratorOptions& options,
1961 const FieldDescriptor* field,
1962 std::set<std::string>* required,
1963 std::set<std::string>* forwards) const {
1964 if (field->cpp_type() == FieldDescriptor::CPPTYPE_ENUM &&
1965 // N.B.: file-level extensions with enum type do *not* create
1966 // dependencies, as per original codegen.
1967 !(field->is_extension() && field->extension_scope() == nullptr)) {
1968 if (options.add_require_for_enums) {
1969 required->insert(GetEnumPath(options, field->enum_type()));
1970 } else {
1971 forwards->insert(GetEnumPath(options, field->enum_type()));
1972 }
1973 } else if (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE) {
1974 if (!IgnoreMessage(field->message_type())) {
1975 required->insert(GetMessagePath(options, field->message_type()));
1976 }
1977 }
1978 }
1979
FindRequiresForExtension(const GeneratorOptions & options,const FieldDescriptor * field,std::set<std::string> * required,std::set<std::string> * forwards) const1980 void Generator::FindRequiresForExtension(
1981 const GeneratorOptions& options, const FieldDescriptor* field,
1982 std::set<std::string>* required, std::set<std::string>* forwards) const {
1983 if (field->containing_type()->full_name() != "google.protobuf.bridge.MessageSet") {
1984 required->insert(GetMessagePath(options, field->containing_type()));
1985 }
1986 FindRequiresForField(options, field, required, forwards);
1987 }
1988
GenerateTestOnly(const GeneratorOptions & options,io::Printer * printer) const1989 void Generator::GenerateTestOnly(const GeneratorOptions& options,
1990 io::Printer* printer) const {
1991 if (options.testonly) {
1992 printer->Print("goog.setTestOnly();\n\n");
1993 }
1994 printer->Print("\n");
1995 }
1996
GenerateClassesAndEnums(const GeneratorOptions & options,io::Printer * printer,const FileDescriptor * file) const1997 void Generator::GenerateClassesAndEnums(const GeneratorOptions& options,
1998 io::Printer* printer,
1999 const FileDescriptor* file) const {
2000 for (int i = 0; i < file->message_type_count(); i++) {
2001 GenerateClassConstructorAndDeclareExtensionFieldInfo(options, printer,
2002 file->message_type(i));
2003 }
2004 for (int i = 0; i < file->message_type_count(); i++) {
2005 GenerateClass(options, printer, file->message_type(i));
2006 }
2007 for (int i = 0; i < file->enum_type_count(); i++) {
2008 GenerateEnum(options, printer, file->enum_type(i));
2009 }
2010 }
2011
GenerateClass(const GeneratorOptions & options,io::Printer * printer,const Descriptor * desc) const2012 void Generator::GenerateClass(const GeneratorOptions& options,
2013 io::Printer* printer,
2014 const Descriptor* desc) const {
2015 if (IgnoreMessage(desc)) {
2016 return;
2017 }
2018
2019 if (!NamespaceOnly(desc)) {
2020 printer->Print("\n");
2021 GenerateClassFieldInfo(options, printer, desc);
2022
2023
2024 GenerateClassToObject(options, printer, desc);
2025 // These must come *before* the extension-field info generation in
2026 // GenerateClassRegistration so that references to the binary
2027 // serialization/deserialization functions may be placed in the extension
2028 // objects.
2029 GenerateClassDeserializeBinary(options, printer, desc);
2030 GenerateClassSerializeBinary(options, printer, desc);
2031 }
2032
2033 // Recurse on nested types. These must come *before* the extension-field
2034 // info generation in GenerateClassRegistration so that extensions that
2035 // reference nested types proceed the definitions of the nested types.
2036 for (int i = 0; i < desc->enum_type_count(); i++) {
2037 GenerateEnum(options, printer, desc->enum_type(i));
2038 }
2039 for (int i = 0; i < desc->nested_type_count(); i++) {
2040 GenerateClass(options, printer, desc->nested_type(i));
2041 }
2042
2043 if (!NamespaceOnly(desc)) {
2044 GenerateClassRegistration(options, printer, desc);
2045 GenerateClassFields(options, printer, desc);
2046
2047 if (options.import_style != GeneratorOptions::kImportClosure) {
2048 for (int i = 0; i < desc->extension_count(); i++) {
2049 GenerateExtension(options, printer, desc->extension(i));
2050 }
2051 }
2052 }
2053 }
2054
GenerateClassConstructor(const GeneratorOptions & options,io::Printer * printer,const Descriptor * desc) const2055 void Generator::GenerateClassConstructor(const GeneratorOptions& options,
2056 io::Printer* printer,
2057 const Descriptor* desc) const {
2058 printer->Print(
2059 "/**\n"
2060 " * Generated by JsPbCodeGenerator.\n"
2061 " * @param {Array=} opt_data Optional initial data array, typically "
2062 "from a\n"
2063 " * server response, or constructed directly in Javascript. The array "
2064 "is used\n"
2065 " * in place and becomes part of the constructed object. It is not "
2066 "cloned.\n"
2067 " * If no data is provided, the constructed object will be empty, but "
2068 "still\n"
2069 " * valid.\n"
2070 " * @extends {jspb.Message}\n"
2071 " * @constructor\n"
2072 " */\n"
2073 "$classprefix$$classname$ = function(opt_data) {\n",
2074 "classprefix", GetMessagePathPrefix(options, desc), "classname",
2075 desc->name());
2076 printer->Annotate("classname", desc);
2077 std::string message_id = GetMessageId(desc);
2078 printer->Print(
2079 " jspb.Message.initialize(this, opt_data, $messageId$, $pivot$, "
2080 "$rptfields$, $oneoffields$);\n",
2081 "messageId",
2082 !message_id.empty() ? ("'" + message_id + "'")
2083 : (IsResponse(desc) ? "''" : "0"),
2084 "pivot", GetPivot(desc), "rptfields",
2085 RepeatedFieldsArrayName(options, desc), "oneoffields",
2086 OneofFieldsArrayName(options, desc));
2087 printer->Print(
2088 "};\n"
2089 "goog.inherits($classname$, jspb.Message);\n"
2090 "if (goog.DEBUG && !COMPILED) {\n"
2091 // displayName overrides Function.prototype.displayName
2092 // http://google3/javascript/externs/es3.js?l=511
2093 " /**\n"
2094 " * @public\n"
2095 " * @override\n"
2096 " */\n"
2097 " $classname$.displayName = '$classname$';\n"
2098 "}\n",
2099 "classname", GetMessagePath(options, desc));
2100 }
2101
GenerateClassConstructorAndDeclareExtensionFieldInfo(const GeneratorOptions & options,io::Printer * printer,const Descriptor * desc) const2102 void Generator::GenerateClassConstructorAndDeclareExtensionFieldInfo(
2103 const GeneratorOptions& options, io::Printer* printer,
2104 const Descriptor* desc) const {
2105 if (!NamespaceOnly(desc)) {
2106 GenerateClassConstructor(options, printer, desc);
2107 if (IsExtendable(desc) && desc->full_name() != "google.protobuf.bridge.MessageSet") {
2108 GenerateClassExtensionFieldInfo(options, printer, desc);
2109 }
2110 }
2111 for (int i = 0; i < desc->nested_type_count(); i++) {
2112 if (!IgnoreMessage(desc->nested_type(i))) {
2113 GenerateClassConstructorAndDeclareExtensionFieldInfo(
2114 options, printer, desc->nested_type(i));
2115 }
2116 }
2117 }
2118
GenerateClassFieldInfo(const GeneratorOptions & options,io::Printer * printer,const Descriptor * desc) const2119 void Generator::GenerateClassFieldInfo(const GeneratorOptions& options,
2120 io::Printer* printer,
2121 const Descriptor* desc) const {
2122 if (HasRepeatedFields(options, desc)) {
2123 printer->Print(
2124 "/**\n"
2125 " * List of repeated fields within this message type.\n"
2126 " * @private {!Array<number>}\n"
2127 " * @const\n"
2128 " */\n"
2129 "$classname$$rptfieldarray$ = $rptfields$;\n"
2130 "\n",
2131 "classname", GetMessagePath(options, desc), "rptfieldarray",
2132 kRepeatedFieldArrayName, "rptfields",
2133 RepeatedFieldNumberList(options, desc));
2134 }
2135
2136 if (HasOneofFields(desc)) {
2137 printer->Print(
2138 "/**\n"
2139 " * Oneof group definitions for this message. Each group defines the "
2140 "field\n"
2141 " * numbers belonging to that group. When of these fields' value is "
2142 "set, all\n"
2143 " * other fields in the group are cleared. During deserialization, if "
2144 "multiple\n"
2145 " * fields are encountered for a group, only the last value seen will "
2146 "be kept.\n"
2147 " * @private {!Array<!Array<number>>}\n"
2148 " * @const\n"
2149 " */\n"
2150 "$classname$$oneofgrouparray$ = $oneofgroups$;\n"
2151 "\n",
2152 "classname", GetMessagePath(options, desc), "oneofgrouparray",
2153 kOneofGroupArrayName, "oneofgroups", OneofGroupList(desc));
2154
2155 for (int i = 0; i < desc->oneof_decl_count(); i++) {
2156 if (IgnoreOneof(desc->oneof_decl(i))) {
2157 continue;
2158 }
2159 GenerateOneofCaseDefinition(options, printer, desc->oneof_decl(i));
2160 }
2161 }
2162 }
2163
GenerateClassXid(const GeneratorOptions & options,io::Printer * printer,const Descriptor * desc) const2164 void Generator::GenerateClassXid(const GeneratorOptions& options,
2165 io::Printer* printer,
2166 const Descriptor* desc) const {
2167 printer->Print(
2168 "\n"
2169 "\n"
2170 "$class$.prototype.messageXid = xid('$class$');\n",
2171 "class", GetMessagePath(options, desc));
2172 }
2173
GenerateOneofCaseDefinition(const GeneratorOptions & options,io::Printer * printer,const OneofDescriptor * oneof) const2174 void Generator::GenerateOneofCaseDefinition(
2175 const GeneratorOptions& options, io::Printer* printer,
2176 const OneofDescriptor* oneof) const {
2177 printer->Print(
2178 "/**\n"
2179 " * @enum {number}\n"
2180 " */\n"
2181 "$classname$.$oneof$Case = {\n"
2182 " $upcase$_NOT_SET: 0",
2183 "classname", GetMessagePath(options, oneof->containing_type()), "oneof",
2184 JSOneofName(oneof), "upcase", ToEnumCase(oneof->name()));
2185
2186 for (int i = 0; i < oneof->field_count(); i++) {
2187 if (IgnoreField(oneof->field(i))) {
2188 continue;
2189 }
2190
2191 printer->Print(
2192 ",\n"
2193 " $upcase$: $number$",
2194 "upcase", ToEnumCase(oneof->field(i)->name()), "number",
2195 JSFieldIndex(oneof->field(i)));
2196 printer->Annotate("upcase", oneof->field(i));
2197 }
2198
2199 printer->Print(
2200 "\n"
2201 "};\n"
2202 "\n"
2203 "/**\n"
2204 " * @return {$class$.$oneof$Case}\n"
2205 " */\n"
2206 "$class$.prototype.get$oneof$Case = function() {\n"
2207 " return /** @type {$class$.$oneof$Case} */(jspb.Message."
2208 "computeOneofCase(this, $class$.oneofGroups_[$oneofindex$]));\n"
2209 "};\n"
2210 "\n",
2211 "class", GetMessagePath(options, oneof->containing_type()), "oneof",
2212 JSOneofName(oneof), "oneofindex", JSOneofIndex(oneof));
2213 }
2214
GenerateClassToObject(const GeneratorOptions & options,io::Printer * printer,const Descriptor * desc) const2215 void Generator::GenerateClassToObject(const GeneratorOptions& options,
2216 io::Printer* printer,
2217 const Descriptor* desc) const {
2218 printer->Print(
2219 "\n"
2220 "\n"
2221 "if (jspb.Message.GENERATE_TO_OBJECT) {\n"
2222 "/**\n"
2223 " * Creates an object representation of this proto.\n"
2224 " * Field names that are reserved in JavaScript and will be renamed to "
2225 "pb_name.\n"
2226 " * Optional fields that are not set will be set to undefined.\n"
2227 " * To access a reserved field use, foo.pb_<name>, eg, foo.pb_default.\n"
2228 " * For the list of reserved names please see:\n"
2229 " * net/proto2/compiler/js/internal/generator.cc#kKeyword.\n"
2230 " * @param {boolean=} opt_includeInstance Deprecated. whether to include "
2231 "the\n"
2232 " * JSPB instance for transitional soy proto support:\n"
2233 " * http://goto/soy-param-migration\n"
2234 " * @return {!Object}\n"
2235 " */\n"
2236 "$classname$.prototype.toObject = function(opt_includeInstance) {\n"
2237 " return $classname$.toObject(opt_includeInstance, this);\n"
2238 "};\n"
2239 "\n"
2240 "\n"
2241 "/**\n"
2242 " * Static version of the {@see toObject} method.\n"
2243 " * @param {boolean|undefined} includeInstance Deprecated. Whether to "
2244 "include\n"
2245 " * the JSPB instance for transitional soy proto support:\n"
2246 " * http://goto/soy-param-migration\n"
2247 " * @param {!$classname$} msg The msg instance to transform.\n"
2248 " * @return {!Object}\n"
2249 " * @suppress {unusedLocalVariables} f is only used for nested messages\n"
2250 " */\n"
2251 "$classname$.toObject = function(includeInstance, msg) {\n"
2252 " var f, obj = {",
2253 "classname", GetMessagePath(options, desc));
2254
2255 bool first = true;
2256 for (int i = 0; i < desc->field_count(); i++) {
2257 const FieldDescriptor* field = desc->field(i);
2258 if (IgnoreField(field)) {
2259 continue;
2260 }
2261
2262 if (!first) {
2263 printer->Print(",\n ");
2264 } else {
2265 printer->Print("\n ");
2266 first = false;
2267 }
2268
2269 GenerateClassFieldToObject(options, printer, field);
2270 }
2271
2272 if (!first) {
2273 printer->Print("\n };\n\n");
2274 } else {
2275 printer->Print("\n\n };\n\n");
2276 }
2277
2278 if (IsExtendable(desc)) {
2279 printer->Print(
2280 " jspb.Message.toObjectExtension(/** @type {!jspb.Message} */ (msg), "
2281 "obj,\n"
2282 " $extObject$, $class$.prototype.getExtension,\n"
2283 " includeInstance);\n",
2284 "extObject", JSExtensionsObjectName(options, desc->file(), desc),
2285 "class", GetMessagePath(options, desc));
2286 }
2287
2288 printer->Print(
2289 " if (includeInstance) {\n"
2290 " obj.$$jspbMessageInstance = msg;\n"
2291 " }\n"
2292 " return obj;\n"
2293 "};\n"
2294 "}\n"
2295 "\n"
2296 "\n",
2297 "classname", GetMessagePath(options, desc));
2298 }
2299
GenerateFieldValueExpression(io::Printer * printer,const char * obj_reference,const FieldDescriptor * field,bool use_default) const2300 void Generator::GenerateFieldValueExpression(io::Printer* printer,
2301 const char* obj_reference,
2302 const FieldDescriptor* field,
2303 bool use_default) const {
2304 const bool is_float_or_double =
2305 field->cpp_type() == FieldDescriptor::CPPTYPE_FLOAT ||
2306 field->cpp_type() == FieldDescriptor::CPPTYPE_DOUBLE;
2307 const bool is_boolean = field->cpp_type() == FieldDescriptor::CPPTYPE_BOOL;
2308
2309 const string with_default = use_default ? "WithDefault" : "";
2310 const string default_arg =
2311 use_default ? StrCat(", ", JSFieldDefault(field)) : "";
2312 const string cardinality = field->is_repeated() ? "Repeated" : "";
2313 string type = "";
2314 if (is_float_or_double) {
2315 type = "FloatingPoint";
2316 }
2317 if (is_boolean) {
2318 type = "Boolean";
2319 }
2320
2321 // Prints the appropriate function, among:
2322 // - getField
2323 // - getBooleanField
2324 // - getFloatingPointField => Replaced by getOptionalFloatingPointField to
2325 // preserve backward compatibility.
2326 // - getFieldWithDefault
2327 // - getBooleanFieldWithDefault
2328 // - getFloatingPointFieldWithDefault
2329 // - getRepeatedField
2330 // - getRepeatedBooleanField
2331 // - getRepeatedFloatingPointField
2332 if (is_float_or_double && !field->is_repeated() && !use_default) {
2333 printer->Print(
2334 "jspb.Message.getOptionalFloatingPointField($obj$, "
2335 "$index$$default$)",
2336 "obj", obj_reference, "index", JSFieldIndex(field), "default",
2337 default_arg);
2338 } else {
2339 printer->Print(
2340 "jspb.Message.get$cardinality$$type$Field$with_default$($obj$, "
2341 "$index$$default$)",
2342 "cardinality", cardinality, "type", type, "with_default", with_default,
2343 "obj", obj_reference, "index", JSFieldIndex(field), "default",
2344 default_arg);
2345 }
2346 }
2347
GenerateClassFieldToObject(const GeneratorOptions & options,io::Printer * printer,const FieldDescriptor * field) const2348 void Generator::GenerateClassFieldToObject(const GeneratorOptions& options,
2349 io::Printer* printer,
2350 const FieldDescriptor* field) const {
2351 printer->Print("$fieldname$: ", "fieldname",
2352 JSObjectFieldName(options, field));
2353
2354 if (field->is_map()) {
2355 const FieldDescriptor* value_field = MapFieldValue(field);
2356 // If the map values are of a message type, we must provide their static
2357 // toObject() method; otherwise we pass undefined for that argument.
2358 std::string value_to_object;
2359 if (value_field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE) {
2360 value_to_object =
2361 GetMessagePath(options, value_field->message_type()) + ".toObject";
2362 } else {
2363 value_to_object = "undefined";
2364 }
2365 printer->Print(
2366 "(f = msg.get$name$()) ? f.toObject(includeInstance, $valuetoobject$) "
2367 ": []",
2368 "name", JSGetterName(options, field), "valuetoobject", value_to_object);
2369 } else if (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE) {
2370 // Message field.
2371 if (field->is_repeated()) {
2372 {
2373 printer->Print(
2374 "jspb.Message.toObjectList(msg.get$getter$(),\n"
2375 " $type$.toObject, includeInstance)",
2376 "getter", JSGetterName(options, field), "type",
2377 SubmessageTypeRef(options, field));
2378 }
2379 } else {
2380 printer->Print(
2381 "(f = msg.get$getter$()) && "
2382 "$type$.toObject(includeInstance, f)",
2383 "getter", JSGetterName(options, field), "type",
2384 SubmessageTypeRef(options, field));
2385 }
2386 } else if (field->type() == FieldDescriptor::TYPE_BYTES) {
2387 // For bytes fields we want to always return the B64 data.
2388 printer->Print("msg.get$getter$()", "getter",
2389 JSGetterName(options, field, BYTES_B64));
2390 } else {
2391 bool use_default = field->has_default_value();
2392
2393 if (field->file()->syntax() == FileDescriptor::SYNTAX_PROTO3 &&
2394 // Repeated fields get initialized to their default in the constructor
2395 // (why?), so we emit a plain getField() call for them.
2396 !field->is_repeated() && !UseBrokenPresenceSemantics(options, field)) {
2397 // Proto3 puts all defaults (including implicit defaults) in toObject().
2398 // But for proto2 we leave the existing semantics unchanged: unset fields
2399 // without default are unset.
2400 use_default = true;
2401 }
2402
2403 // We don't implement this by calling the accessors, because the semantics
2404 // of the accessors are changing independently of the toObject() semantics.
2405 // We are migrating the accessors to return defaults instead of null, but
2406 // it may take longer to migrate toObject (or we might not want to do it at
2407 // all). So we want to generate independent code.
2408 // The accessor for unset optional values without default should return
2409 // null. Those are converted to undefined in the generated object.
2410 if (!use_default) {
2411 printer->Print("(f = ");
2412 }
2413 GenerateFieldValueExpression(printer, "msg", field, use_default);
2414 if (!use_default) {
2415 printer->Print(") == null ? undefined : f");
2416 }
2417 }
2418 }
2419
GenerateObjectTypedef(const GeneratorOptions & options,io::Printer * printer,const Descriptor * desc) const2420 void Generator::GenerateObjectTypedef(const GeneratorOptions& options,
2421 io::Printer* printer,
2422 const Descriptor* desc) const {
2423 // TODO(b/122687752): Consider renaming nested messages called ObjectFormat
2424 // to prevent collisions.
2425 const std::string type_name = GetMessagePath(options, desc) + ".ObjectFormat";
2426
2427 printer->Print(
2428 "/**\n"
2429 " * The raw object form of $messageName$ as accepted by the `fromObject` "
2430 "method.\n"
2431 " * @record\n"
2432 " */\n"
2433 "$typeName$ = function() {\n",
2434 "messageName", desc->name(), "typeName", type_name);
2435
2436 for (int i = 0; i < desc->field_count(); i++) {
2437 if (i > 0) {
2438 printer->Print("\n");
2439 }
2440 printer->Print(
2441 " /** @type {$fieldType$|undefined} */\n"
2442 " this.$fieldName$;\n",
2443 "fieldName", JSObjectFieldName(options, desc->field(i)),
2444 // TODO(b/121097361): Add type checking for field values.
2445 "fieldType", "?");
2446 }
2447
2448 printer->Print("};\n\n");
2449 }
2450
GenerateClassFromObject(const GeneratorOptions & options,io::Printer * printer,const Descriptor * desc) const2451 void Generator::GenerateClassFromObject(const GeneratorOptions& options,
2452 io::Printer* printer,
2453 const Descriptor* desc) const {
2454 printer->Print("if (jspb.Message.GENERATE_FROM_OBJECT) {\n\n");
2455
2456 GenerateObjectTypedef(options, printer, desc);
2457
2458 printer->Print(
2459 "/**\n"
2460 " * Loads data from an object into a new instance of this proto.\n"
2461 " * @param {!$classname$.ObjectFormat} obj\n"
2462 " * The object representation of this proto to load the data from.\n"
2463 " * @return {!$classname$}\n"
2464 " */\n"
2465 "$classname$.fromObject = function(obj) {\n"
2466 " var msg = new $classname$();\n",
2467 "classname", GetMessagePath(options, desc));
2468
2469 for (int i = 0; i < desc->field_count(); i++) {
2470 const FieldDescriptor* field = desc->field(i);
2471 if (!IgnoreField(field)) {
2472 GenerateClassFieldFromObject(options, printer, field);
2473 }
2474 }
2475
2476 printer->Print(
2477 " return msg;\n"
2478 "};\n"
2479 "}\n\n");
2480 }
2481
GenerateClassFieldFromObject(const GeneratorOptions & options,io::Printer * printer,const FieldDescriptor * field) const2482 void Generator::GenerateClassFieldFromObject(
2483 const GeneratorOptions& options, io::Printer* printer,
2484 const FieldDescriptor* field) const {
2485 if (field->is_map()) {
2486 const FieldDescriptor* value_field = MapFieldValue(field);
2487 if (value_field->type() == FieldDescriptor::TYPE_MESSAGE) {
2488 // Since the map values are of message type, we have to do some extra work
2489 // to recursively call fromObject() on them before setting the map field.
2490 printer->Print(
2491 " obj.$name$ && jspb.Message.setWrapperField(\n"
2492 " msg, $index$, jspb.Map.fromObject(obj.$name$, $fieldclass$, "
2493 "$fieldclass$.fromObject));\n",
2494 "name", JSObjectFieldName(options, field), "index",
2495 JSFieldIndex(field), "fieldclass",
2496 GetMessagePath(options, value_field->message_type()));
2497 } else {
2498 // `msg` is a newly-constructed message object that has not yet built any
2499 // map containers wrapping underlying arrays, so we can simply directly
2500 // set the array here without fear of a stale wrapper.
2501 printer->Print(
2502 " obj.$name$ && "
2503 "jspb.Message.setField(msg, $index$, obj.$name$);\n",
2504 "name", JSObjectFieldName(options, field), "index",
2505 JSFieldIndex(field));
2506 }
2507 } else if (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE) {
2508 // Message field (singular or repeated)
2509 if (field->is_repeated()) {
2510 {
2511 printer->Print(
2512 " obj.$name$ && "
2513 "jspb.Message.setRepeatedWrapperField(\n"
2514 " msg, $index$, obj.$name$.map(\n"
2515 " $fieldclass$.fromObject));\n",
2516 "name", JSObjectFieldName(options, field), "index",
2517 JSFieldIndex(field), "fieldclass",
2518 SubmessageTypeRef(options, field));
2519 }
2520 } else {
2521 printer->Print(
2522 " obj.$name$ && jspb.Message.setWrapperField(\n"
2523 " msg, $index$, $fieldclass$.fromObject(obj.$name$));\n",
2524 "name", JSObjectFieldName(options, field), "index",
2525 JSFieldIndex(field), "fieldclass", SubmessageTypeRef(options, field));
2526 }
2527 } else {
2528 // Simple (primitive) field.
2529 printer->Print(
2530 " obj.$name$ != null && jspb.Message.setField(msg, $index$, "
2531 "obj.$name$);\n",
2532 "name", JSObjectFieldName(options, field), "index",
2533 JSFieldIndex(field));
2534 }
2535 }
2536
GenerateClassRegistration(const GeneratorOptions & options,io::Printer * printer,const Descriptor * desc) const2537 void Generator::GenerateClassRegistration(const GeneratorOptions& options,
2538 io::Printer* printer,
2539 const Descriptor* desc) const {
2540 // Register any extensions defined inside this message type.
2541 for (int i = 0; i < desc->extension_count(); i++) {
2542 const FieldDescriptor* extension = desc->extension(i);
2543 if (ShouldGenerateExtension(extension)) {
2544 GenerateExtension(options, printer, extension);
2545 }
2546 }
2547
2548 }
2549
GenerateClassFields(const GeneratorOptions & options,io::Printer * printer,const Descriptor * desc) const2550 void Generator::GenerateClassFields(const GeneratorOptions& options,
2551 io::Printer* printer,
2552 const Descriptor* desc) const {
2553 for (int i = 0; i < desc->field_count(); i++) {
2554 if (!IgnoreField(desc->field(i))) {
2555 GenerateClassField(options, printer, desc->field(i));
2556 }
2557 }
2558 }
2559
GenerateBytesWrapper(const GeneratorOptions & options,io::Printer * printer,const FieldDescriptor * field,BytesMode bytes_mode)2560 void GenerateBytesWrapper(const GeneratorOptions& options, io::Printer* printer,
2561 const FieldDescriptor* field, BytesMode bytes_mode) {
2562 std::string type =
2563 JSFieldTypeAnnotation(options, field,
2564 /* is_setter_argument = */ false,
2565 /* force_present = */ false,
2566 /* singular_if_not_packed = */ false, bytes_mode);
2567 printer->Print(
2568 "/**\n"
2569 " * $fielddef$\n"
2570 "$comment$"
2571 " * This is a type-conversion wrapper around `get$defname$()`\n"
2572 " * @return {$type$}\n"
2573 " */\n"
2574 "$class$.prototype.get$name$ = function() {\n"
2575 " return /** @type {$type$} */ (jspb.Message.bytes$list$As$suffix$(\n"
2576 " this.get$defname$()));\n"
2577 "};\n"
2578 "\n"
2579 "\n",
2580 "fielddef", FieldDefinition(options, field), "comment",
2581 FieldComments(field, bytes_mode), "type", type, "class",
2582 GetMessagePath(options, field->containing_type()), "name",
2583 JSGetterName(options, field, bytes_mode), "list",
2584 field->is_repeated() ? "List" : "", "suffix",
2585 JSByteGetterSuffix(bytes_mode), "defname",
2586 JSGetterName(options, field, BYTES_DEFAULT));
2587 }
2588
GenerateClassField(const GeneratorOptions & options,io::Printer * printer,const FieldDescriptor * field) const2589 void Generator::GenerateClassField(const GeneratorOptions& options,
2590 io::Printer* printer,
2591 const FieldDescriptor* field) const {
2592 if (field->is_map()) {
2593 const FieldDescriptor* key_field = MapFieldKey(field);
2594 const FieldDescriptor* value_field = MapFieldValue(field);
2595 // Map field: special handling to instantiate the map object on demand.
2596 std::string key_type =
2597 JSFieldTypeAnnotation(options, key_field,
2598 /* is_setter_argument = */ false,
2599 /* force_present = */ true,
2600 /* singular_if_not_packed = */ false);
2601 std::string value_type =
2602 JSFieldTypeAnnotation(options, value_field,
2603 /* is_setter_argument = */ false,
2604 /* force_present = */ true,
2605 /* singular_if_not_packed = */ false);
2606
2607 printer->Print(
2608 "/**\n"
2609 " * $fielddef$\n"
2610 " * @param {boolean=} opt_noLazyCreate Do not create the map if\n"
2611 " * empty, instead returning `undefined`\n"
2612 " * @return {!jspb.Map<$keytype$,$valuetype$>}\n"
2613 " */\n",
2614 "fielddef", FieldDefinition(options, field), "keytype", key_type,
2615 "valuetype", value_type);
2616 printer->Print(
2617 "$class$.prototype.$gettername$ = function(opt_noLazyCreate) {\n"
2618 " return /** @type {!jspb.Map<$keytype$,$valuetype$>} */ (\n",
2619 "class", GetMessagePath(options, field->containing_type()),
2620 "gettername", "get" + JSGetterName(options, field), "keytype", key_type,
2621 "valuetype", value_type);
2622 printer->Annotate("gettername", field);
2623 printer->Print(
2624 " jspb.Message.getMapField(this, $index$, opt_noLazyCreate",
2625 "index", JSFieldIndex(field));
2626
2627 if (value_field->type() == FieldDescriptor::TYPE_MESSAGE) {
2628 printer->Print(
2629 ",\n"
2630 " $messageType$",
2631 "messageType", GetMessagePath(options, value_field->message_type()));
2632 } else {
2633 printer->Print(
2634 ",\n"
2635 " null");
2636 }
2637
2638 printer->Print("));\n");
2639
2640 printer->Print(
2641 "};\n"
2642 "\n"
2643 "\n");
2644 } else if (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE) {
2645 // Message field: special handling in order to wrap the underlying data
2646 // array with a message object.
2647
2648 printer->Print(
2649 "/**\n"
2650 " * $fielddef$\n"
2651 "$comment$"
2652 " * @return {$type$}\n"
2653 " */\n",
2654 "fielddef", FieldDefinition(options, field), "comment",
2655 FieldComments(field, BYTES_DEFAULT), "type",
2656 JSFieldTypeAnnotation(options, field,
2657 /* is_setter_argument = */ false,
2658 /* force_present = */ false,
2659 /* singular_if_not_packed = */ false));
2660 printer->Print(
2661 "$class$.prototype.$gettername$ = function() {\n"
2662 " return /** @type{$type$} */ (\n"
2663 " jspb.Message.get$rpt$WrapperField(this, $wrapperclass$, "
2664 "$index$$required$));\n"
2665 "};\n"
2666 "\n"
2667 "\n",
2668 "class", GetMessagePath(options, field->containing_type()),
2669 "gettername", "get" + JSGetterName(options, field), "type",
2670 JSFieldTypeAnnotation(options, field,
2671 /* is_setter_argument = */ false,
2672 /* force_present = */ false,
2673 /* singular_if_not_packed = */ false),
2674 "rpt", (field->is_repeated() ? "Repeated" : ""), "index",
2675 JSFieldIndex(field), "wrapperclass", SubmessageTypeRef(options, field),
2676 "required",
2677 (field->label() == FieldDescriptor::LABEL_REQUIRED ? ", 1" : ""));
2678 printer->Annotate("gettername", field);
2679 printer->Print(
2680 "/** @param {$optionaltype$} value$returndoc$ */\n"
2681 "$class$.prototype.$settername$ = function(value) {\n"
2682 " jspb.Message.set$oneoftag$$repeatedtag$WrapperField(",
2683 "optionaltype",
2684 JSFieldTypeAnnotation(options, field,
2685 /* is_setter_argument = */ true,
2686 /* force_present = */ false,
2687 /* singular_if_not_packed = */ false),
2688 "returndoc", JSReturnDoc(options, field), "class",
2689 GetMessagePath(options, field->containing_type()), "settername",
2690 "set" + JSGetterName(options, field), "oneoftag",
2691 (field->containing_oneof() ? "Oneof" : ""), "repeatedtag",
2692 (field->is_repeated() ? "Repeated" : ""));
2693 printer->Annotate("settername", field);
2694
2695 printer->Print(
2696 "this, $index$$oneofgroup$, value);$returnvalue$\n"
2697 "};\n"
2698 "\n"
2699 "\n",
2700 "index", JSFieldIndex(field), "oneofgroup",
2701 (field->containing_oneof() ? (", " + JSOneofArray(options, field))
2702 : ""),
2703 "returnvalue", JSReturnClause(field));
2704
2705 if (field->is_repeated()) {
2706 GenerateRepeatedMessageHelperMethods(options, printer, field);
2707 }
2708
2709 } else {
2710 bool untyped =
2711 false;
2712
2713 // Simple (primitive) field, either singular or repeated.
2714
2715 // TODO(b/26173701): Always use BYTES_DEFAULT for the getter return type;
2716 // at this point we "lie" to non-binary users and tell the return
2717 // type is always base64 string, pending a LSC to migrate to typed getters.
2718 BytesMode bytes_mode =
2719 field->type() == FieldDescriptor::TYPE_BYTES && !options.binary
2720 ? BYTES_B64
2721 : BYTES_DEFAULT;
2722 std::string typed_annotation =
2723 JSFieldTypeAnnotation(options, field,
2724 /* is_setter_argument = */ false,
2725 /* force_present = */ false,
2726 /* singular_if_not_packed = */ false,
2727 /* bytes_mode = */ bytes_mode);
2728 if (untyped) {
2729 printer->Print(
2730 "/**\n"
2731 " * @return {?} Raw field, untyped.\n"
2732 " */\n");
2733 } else {
2734 printer->Print(
2735 "/**\n"
2736 " * $fielddef$\n"
2737 "$comment$"
2738 " * @return {$type$}\n"
2739 " */\n",
2740 "fielddef", FieldDefinition(options, field), "comment",
2741 FieldComments(field, bytes_mode), "type", typed_annotation);
2742 }
2743
2744 printer->Print("$class$.prototype.$gettername$ = function() {\n", "class",
2745 GetMessagePath(options, field->containing_type()),
2746 "gettername", "get" + JSGetterName(options, field));
2747 printer->Annotate("gettername", field);
2748
2749 if (untyped) {
2750 printer->Print(" return ");
2751 } else {
2752 printer->Print(" return /** @type {$type$} */ (", "type",
2753 typed_annotation);
2754 }
2755
2756 bool use_default = !ReturnsNullWhenUnset(options, field);
2757
2758 // Raw fields with no default set should just return undefined.
2759 if (untyped && !field->has_default_value()) {
2760 use_default = false;
2761 }
2762
2763 // Repeated fields get initialized to their default in the constructor
2764 // (why?), so we emit a plain getField() call for them.
2765 if (field->is_repeated()) {
2766 use_default = false;
2767 }
2768
2769 GenerateFieldValueExpression(printer, "this", field, use_default);
2770
2771 if (untyped) {
2772 printer->Print(
2773 ";\n"
2774 "};\n"
2775 "\n"
2776 "\n");
2777 } else {
2778 printer->Print(
2779 ");\n"
2780 "};\n"
2781 "\n"
2782 "\n");
2783 }
2784
2785 if (field->type() == FieldDescriptor::TYPE_BYTES && !untyped) {
2786 GenerateBytesWrapper(options, printer, field, BYTES_B64);
2787 GenerateBytesWrapper(options, printer, field, BYTES_U8);
2788 }
2789
2790 if (untyped) {
2791 printer->Print(
2792 "/**\n"
2793 " * @param {*} value$returndoc$\n"
2794 " */\n",
2795 "returndoc", JSReturnDoc(options, field));
2796 } else {
2797 printer->Print(
2798 "/** @param {$optionaltype$} value$returndoc$ */\n", "optionaltype",
2799 JSFieldTypeAnnotation(options, field,
2800 /* is_setter_argument = */ true,
2801 /* force_present = */ false,
2802 /* singular_if_not_packed = */ false),
2803 "returndoc", JSReturnDoc(options, field));
2804 }
2805
2806 if (field->file()->syntax() == FileDescriptor::SYNTAX_PROTO3 &&
2807 !field->is_repeated() && !field->is_map() &&
2808 !HasFieldPresence(options, field)) {
2809 // Proto3 non-repeated and non-map fields without presence use the
2810 // setProto3*Field function.
2811 printer->Print(
2812 "$class$.prototype.$settername$ = function(value) {\n"
2813 " jspb.Message.setProto3$typetag$Field(this, $index$, "
2814 "value);$returnvalue$\n"
2815 "};\n"
2816 "\n"
2817 "\n",
2818 "class", GetMessagePath(options, field->containing_type()),
2819 "settername", "set" + JSGetterName(options, field), "typetag",
2820 JSTypeTag(field), "index", JSFieldIndex(field), "returnvalue",
2821 JSReturnClause(field));
2822 printer->Annotate("settername", field);
2823 } else {
2824 // Otherwise, use the regular setField function.
2825 printer->Print(
2826 "$class$.prototype.$settername$ = function(value) {\n"
2827 " jspb.Message.set$oneoftag$Field(this, $index$",
2828 "class", GetMessagePath(options, field->containing_type()),
2829 "settername", "set" + JSGetterName(options, field), "oneoftag",
2830 (field->containing_oneof() ? "Oneof" : ""), "index",
2831 JSFieldIndex(field));
2832 printer->Annotate("settername", field);
2833 printer->Print(
2834 "$oneofgroup$, $type$value$rptvalueinit$$typeclose$);$returnvalue$\n"
2835 "};\n"
2836 "\n"
2837 "\n",
2838 "type",
2839 untyped ? "/** @type{string|number|boolean|Array|undefined} */(" : "",
2840 "typeclose", untyped ? ")" : "", "oneofgroup",
2841 (field->containing_oneof() ? (", " + JSOneofArray(options, field))
2842 : ""),
2843 "returnvalue", JSReturnClause(field), "rptvalueinit",
2844 (field->is_repeated() ? " || []" : ""));
2845 }
2846
2847 if (untyped) {
2848 printer->Print(
2849 "/**\n"
2850 " * Clears the value.$returndoc$\n"
2851 " */\n",
2852 "returndoc", JSReturnDoc(options, field));
2853 }
2854
2855 if (field->is_repeated()) {
2856 GenerateRepeatedPrimitiveHelperMethods(options, printer, field, untyped);
2857 }
2858 }
2859
2860 // Generate clearFoo() method for map fields, repeated fields, and other
2861 // fields with presence.
2862 if (field->is_map()) {
2863 // clang-format off
2864 printer->Print(
2865 "/**\n"
2866 " * Clears values from the map. The map will be non-null."
2867 "$returndoc$\n"
2868 " */\n"
2869 "$class$.prototype.$clearername$ = function() {\n"
2870 " this.$gettername$().clear();$returnvalue$\n"
2871 "};\n"
2872 "\n"
2873 "\n",
2874 "returndoc", JSReturnDoc(options, field),
2875 "class", GetMessagePath(options, field->containing_type()),
2876 "clearername", "clear" + JSGetterName(options, field),
2877 "gettername", "get" + JSGetterName(options, field),
2878 "returnvalue", JSReturnClause(field));
2879 // clang-format on
2880 printer->Annotate("clearername", field);
2881 } else if (field->is_repeated() ||
2882 (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE &&
2883 !field->is_required())) {
2884 // Fields where we can delegate to the regular setter.
2885 // clang-format off
2886 printer->Print(
2887 "/**\n"
2888 " * $jsdoc$$returndoc$\n"
2889 " */\n"
2890 "$class$.prototype.$clearername$ = function() {\n"
2891 " this.$settername$($clearedvalue$);$returnvalue$\n"
2892 "};\n"
2893 "\n"
2894 "\n",
2895 "jsdoc", field->is_repeated()
2896 ? "Clears the list making it empty but non-null."
2897 : "Clears the message field making it undefined.",
2898 "returndoc", JSReturnDoc(options, field),
2899 "class", GetMessagePath(options, field->containing_type()),
2900 "clearername", "clear" + JSGetterName(options, field),
2901 "settername", "set" + JSGetterName(options, field),
2902 "clearedvalue", (field->is_repeated() ? "[]" : "undefined"),
2903 "returnvalue", JSReturnClause(field));
2904 // clang-format on
2905 printer->Annotate("clearername", field);
2906 } else if (HasFieldPresence(options, field)) {
2907 // Fields where we can't delegate to the regular setter because it doesn't
2908 // accept "undefined" as an argument.
2909 // clang-format off
2910 printer->Print(
2911 "/**\n"
2912 " * Clears the field making it undefined.$returndoc$\n"
2913 " */\n"
2914 "$class$.prototype.$clearername$ = function() {\n"
2915 " jspb.Message.set$maybeoneof$Field(this, "
2916 "$index$$maybeoneofgroup$, ",
2917 "returndoc", JSReturnDoc(options, field),
2918 "class", GetMessagePath(options, field->containing_type()),
2919 "clearername", "clear" + JSGetterName(options, field),
2920 "maybeoneof", (field->containing_oneof() ? "Oneof" : ""),
2921 "maybeoneofgroup", (field->containing_oneof()
2922 ? (", " + JSOneofArray(options, field))
2923 : ""),
2924 "index", JSFieldIndex(field));
2925 // clang-format on
2926 printer->Annotate("clearername", field);
2927 printer->Print(
2928 "$clearedvalue$);$returnvalue$\n"
2929 "};\n"
2930 "\n"
2931 "\n",
2932 "clearedvalue", (field->is_repeated() ? "[]" : "undefined"),
2933 "returnvalue", JSReturnClause(field));
2934 }
2935
2936 if (HasFieldPresence(options, field)) {
2937 printer->Print(
2938 "/**\n"
2939 " * Returns whether this field is set.\n"
2940 " * @return {boolean}\n"
2941 " */\n"
2942 "$class$.prototype.$hasername$ = function() {\n"
2943 " return jspb.Message.getField(this, $index$) != null;\n"
2944 "};\n"
2945 "\n"
2946 "\n",
2947 "class", GetMessagePath(options, field->containing_type()), "hasername",
2948 "has" + JSGetterName(options, field), "index", JSFieldIndex(field));
2949 printer->Annotate("hasername", field);
2950 }
2951 }
2952
GenerateRepeatedPrimitiveHelperMethods(const GeneratorOptions & options,io::Printer * printer,const FieldDescriptor * field,bool untyped) const2953 void Generator::GenerateRepeatedPrimitiveHelperMethods(
2954 const GeneratorOptions& options, io::Printer* printer,
2955 const FieldDescriptor* field, bool untyped) const {
2956 // clang-format off
2957 printer->Print(
2958 "/**\n"
2959 " * @param {$optionaltype$} value\n"
2960 " * @param {number=} opt_index$returndoc$\n"
2961 " */\n"
2962 "$class$.prototype.$addername$ = function(value, opt_index) {\n"
2963 " jspb.Message.addToRepeatedField(this, $index$",
2964 "class", GetMessagePath(options, field->containing_type()), "addername",
2965 "add" + JSGetterName(options, field, BYTES_DEFAULT,
2966 /* drop_list = */ true),
2967 "optionaltype",
2968 JSFieldTypeAnnotation(
2969 options, field,
2970 /* is_setter_argument = */ false,
2971 /* force_present = */ true,
2972 /* singular_if_not_packed = */ false,
2973 BYTES_DEFAULT,
2974 /* force_singular = */ true),
2975 "index", JSFieldIndex(field),
2976 "returndoc", JSReturnDoc(options, field));
2977 printer->Annotate("addername", field);
2978 printer->Print(
2979 "$oneofgroup$, $type$value$rptvalueinit$$typeclose$, "
2980 "opt_index);$returnvalue$\n"
2981 "};\n"
2982 "\n"
2983 "\n",
2984 "type", untyped ? "/** @type{string|number|boolean|!Uint8Array} */(" : "",
2985 "typeclose", untyped ? ")" : "", "oneofgroup",
2986 (field->containing_oneof() ? (", " + JSOneofArray(options, field)) : ""),
2987 "rptvalueinit", "",
2988 "returnvalue", JSReturnClause(field));
2989 // clang-format on
2990 }
2991
GenerateRepeatedMessageHelperMethods(const GeneratorOptions & options,io::Printer * printer,const FieldDescriptor * field) const2992 void Generator::GenerateRepeatedMessageHelperMethods(
2993 const GeneratorOptions& options, io::Printer* printer,
2994 const FieldDescriptor* field) const {
2995 printer->Print(
2996 "/**\n"
2997 " * @param {!$optionaltype$=} opt_value\n"
2998 " * @param {number=} opt_index\n"
2999 " * @return {!$optionaltype$}\n"
3000 " */\n"
3001 "$class$.prototype.$addername$ = function(opt_value, opt_index) {\n"
3002 " return jspb.Message.addTo$repeatedtag$WrapperField(",
3003 "optionaltype", JSTypeName(options, field, BYTES_DEFAULT), "class",
3004 GetMessagePath(options, field->containing_type()), "addername",
3005 "add" + JSGetterName(options, field, BYTES_DEFAULT,
3006 /* drop_list = */ true),
3007 "repeatedtag", (field->is_repeated() ? "Repeated" : ""));
3008
3009 printer->Annotate("addername", field);
3010 printer->Print(
3011 "this, $index$$oneofgroup$, opt_value, $ctor$, opt_index);\n"
3012 "};\n"
3013 "\n"
3014 "\n",
3015 "index", JSFieldIndex(field), "oneofgroup",
3016 (field->containing_oneof() ? (", " + JSOneofArray(options, field)) : ""),
3017 "ctor", GetMessagePath(options, field->message_type()));
3018 }
3019
GenerateClassExtensionFieldInfo(const GeneratorOptions & options,io::Printer * printer,const Descriptor * desc) const3020 void Generator::GenerateClassExtensionFieldInfo(const GeneratorOptions& options,
3021 io::Printer* printer,
3022 const Descriptor* desc) const {
3023 if (IsExtendable(desc)) {
3024 printer->Print(
3025 "\n"
3026 "/**\n"
3027 " * The extensions registered with this message class. This is a "
3028 "map of\n"
3029 " * extension field number to fieldInfo object.\n"
3030 " *\n"
3031 " * For example:\n"
3032 " * { 123: {fieldIndex: 123, fieldName: {my_field_name: 0}, "
3033 "ctor: proto.example.MyMessage} }\n"
3034 " *\n"
3035 " * fieldName contains the JsCompiler renamed field name property "
3036 "so that it\n"
3037 " * works in OPTIMIZED mode.\n"
3038 " *\n"
3039 " * @type {!Object<number, jspb.ExtensionFieldInfo>}\n"
3040 " */\n"
3041 "$class$.extensions = {};\n"
3042 "\n",
3043 "class", GetMessagePath(options, desc));
3044
3045 printer->Print(
3046 "\n"
3047 "/**\n"
3048 " * The extensions registered with this message class. This is a "
3049 "map of\n"
3050 " * extension field number to fieldInfo object.\n"
3051 " *\n"
3052 " * For example:\n"
3053 " * { 123: {fieldIndex: 123, fieldName: {my_field_name: 0}, "
3054 "ctor: proto.example.MyMessage} }\n"
3055 " *\n"
3056 " * fieldName contains the JsCompiler renamed field name property "
3057 "so that it\n"
3058 " * works in OPTIMIZED mode.\n"
3059 " *\n"
3060 " * @type {!Object<number, jspb.ExtensionFieldBinaryInfo>}\n"
3061 " */\n"
3062 "$class$.extensionsBinary = {};\n"
3063 "\n",
3064 "class", GetMessagePath(options, desc));
3065 }
3066 }
3067
3068
GenerateClassDeserializeBinary(const GeneratorOptions & options,io::Printer * printer,const Descriptor * desc) const3069 void Generator::GenerateClassDeserializeBinary(const GeneratorOptions& options,
3070 io::Printer* printer,
3071 const Descriptor* desc) const {
3072 // TODO(cfallin): Handle lazy decoding when requested by field option and/or
3073 // by default for 'bytes' fields and packed repeated fields.
3074
3075 printer->Print(
3076 "/**\n"
3077 " * Deserializes binary data (in protobuf wire format).\n"
3078 " * @param {jspb.ByteSource} bytes The bytes to deserialize.\n"
3079 " * @return {!$class$}\n"
3080 " */\n"
3081 "$class$.deserializeBinary = function(bytes) {\n"
3082 " var reader = new jspb.BinaryReader(bytes);\n"
3083 " var msg = new $class$;\n"
3084 " return $class$.deserializeBinaryFromReader(msg, reader);\n"
3085 "};\n"
3086 "\n"
3087 "\n"
3088 "/**\n"
3089 " * Deserializes binary data (in protobuf wire format) from the\n"
3090 " * given reader into the given message object.\n"
3091 " * @param {!$class$} msg The message object to deserialize into.\n"
3092 " * @param {!jspb.BinaryReader} reader The BinaryReader to use.\n"
3093 " * @return {!$class$}\n"
3094 " */\n"
3095 "$class$.deserializeBinaryFromReader = function(msg, reader) {\n"
3096 " while (reader.nextField()) {\n",
3097 "class", GetMessagePath(options, desc));
3098 printer->Print(
3099 " if (reader.isEndGroup()) {\n"
3100 " break;\n"
3101 " }\n"
3102 " var field = reader.getFieldNumber();\n"
3103 " switch (field) {\n");
3104
3105 for (int i = 0; i < desc->field_count(); i++) {
3106 if (!IgnoreField(desc->field(i))) {
3107 GenerateClassDeserializeBinaryField(options, printer, desc->field(i));
3108 }
3109 }
3110
3111 printer->Print(" default:\n");
3112 if (IsExtendable(desc)) {
3113 printer->Print(
3114 " jspb.Message.readBinaryExtension(msg, reader,\n"
3115 " $extobj$Binary,\n"
3116 " $class$.prototype.getExtension,\n"
3117 " $class$.prototype.setExtension);\n"
3118 " break;\n"
3119 " }\n",
3120 "extobj", JSExtensionsObjectName(options, desc->file(), desc),
3121 "class", GetMessagePath(options, desc));
3122 } else {
3123 printer->Print(
3124 " reader.skipField();\n"
3125 " break;\n"
3126 " }\n");
3127 }
3128
3129 printer->Print(
3130 " }\n"
3131 " return msg;\n"
3132 "};\n"
3133 "\n"
3134 "\n");
3135 }
3136
GenerateClassDeserializeBinaryField(const GeneratorOptions & options,io::Printer * printer,const FieldDescriptor * field) const3137 void Generator::GenerateClassDeserializeBinaryField(
3138 const GeneratorOptions& options, io::Printer* printer,
3139 const FieldDescriptor* field) const {
3140 printer->Print(" case $num$:\n", "num", StrCat(field->number()));
3141
3142 if (field->is_map()) {
3143 const FieldDescriptor* key_field = MapFieldKey(field);
3144 const FieldDescriptor* value_field = MapFieldValue(field);
3145 printer->Print(
3146 " var value = msg.get$name$();\n"
3147 " reader.readMessage(value, function(message, reader) {\n",
3148 "name", JSGetterName(options, field));
3149
3150 printer->Print(
3151 " jspb.Map.deserializeBinary(message, reader, "
3152 "$keyReaderFn$, $valueReaderFn$",
3153 "keyReaderFn", JSBinaryReaderMethodName(options, key_field),
3154 "valueReaderFn", JSBinaryReaderMethodName(options, value_field));
3155
3156 if (value_field->type() == FieldDescriptor::TYPE_MESSAGE) {
3157 printer->Print(", $messageType$.deserializeBinaryFromReader",
3158 "messageType",
3159 GetMessagePath(options, value_field->message_type()));
3160 } else {
3161 printer->Print(", null");
3162 }
3163 printer->Print(", $defaultKey$", "defaultKey", JSFieldDefault(key_field));
3164 printer->Print(");\n");
3165 printer->Print(" });\n");
3166 } else {
3167 if (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE) {
3168 printer->Print(
3169 " var value = new $fieldclass$;\n"
3170 " reader.read$msgOrGroup$($grpfield$value,"
3171 "$fieldclass$.deserializeBinaryFromReader);\n",
3172 "fieldclass", SubmessageTypeRef(options, field), "msgOrGroup",
3173 (field->type() == FieldDescriptor::TYPE_GROUP) ? "Group" : "Message",
3174 "grpfield",
3175 (field->type() == FieldDescriptor::TYPE_GROUP)
3176 ? (StrCat(field->number()) + ", ")
3177 : "");
3178 } else {
3179 printer->Print(
3180 " var value = /** @type {$fieldtype$} */ "
3181 "(reader.read$reader$());\n",
3182 "fieldtype",
3183 JSFieldTypeAnnotation(options, field, false, true,
3184 /* singular_if_not_packed */ true, BYTES_U8),
3185 "reader",
3186 JSBinaryReadWriteMethodName(field, /* is_writer = */ false));
3187 }
3188
3189 if (field->is_repeated() && !field->is_packed()) {
3190 printer->Print(
3191 " msg.add$name$(value);\n", "name",
3192 JSGetterName(options, field, BYTES_DEFAULT, /* drop_list = */ true));
3193 } else {
3194 // Singular fields, and packed repeated fields, receive a |value| either
3195 // as the field's value or as the array of all the field's values; set
3196 // this as the field's value directly.
3197 printer->Print(" msg.set$name$(value);\n", "name",
3198 JSGetterName(options, field));
3199 }
3200 }
3201
3202 printer->Print(" break;\n");
3203 }
3204
GenerateClassSerializeBinary(const GeneratorOptions & options,io::Printer * printer,const Descriptor * desc) const3205 void Generator::GenerateClassSerializeBinary(const GeneratorOptions& options,
3206 io::Printer* printer,
3207 const Descriptor* desc) const {
3208 printer->Print(
3209 "/**\n"
3210 " * Serializes the message to binary data (in protobuf wire format).\n"
3211 " * @return {!Uint8Array}\n"
3212 " */\n"
3213 "$class$.prototype.serializeBinary = function() {\n"
3214 " var writer = new jspb.BinaryWriter();\n"
3215 " $class$.serializeBinaryToWriter(this, writer);\n"
3216 " return writer.getResultBuffer();\n"
3217 "};\n"
3218 "\n"
3219 "\n"
3220 "/**\n"
3221 " * Serializes the given message to binary data (in protobuf wire\n"
3222 " * format), writing to the given BinaryWriter.\n"
3223 " * @param {!$class$} message\n"
3224 " * @param {!jspb.BinaryWriter} writer\n"
3225 " * @suppress {unusedLocalVariables} f is only used for nested messages\n"
3226 " */\n"
3227 "$class$.serializeBinaryToWriter = function(message, "
3228 "writer) {\n"
3229 " var f = undefined;\n",
3230 "class", GetMessagePath(options, desc));
3231
3232 for (int i = 0; i < desc->field_count(); i++) {
3233 if (!IgnoreField(desc->field(i))) {
3234 GenerateClassSerializeBinaryField(options, printer, desc->field(i));
3235 }
3236 }
3237
3238 if (IsExtendable(desc)) {
3239 printer->Print(
3240 " jspb.Message.serializeBinaryExtensions(message, writer,\n"
3241 " $extobj$Binary, $class$.prototype.getExtension);\n",
3242 "extobj", JSExtensionsObjectName(options, desc->file(), desc), "class",
3243 GetMessagePath(options, desc));
3244 }
3245
3246 printer->Print(
3247 "};\n"
3248 "\n"
3249 "\n");
3250 }
3251
GenerateClassSerializeBinaryField(const GeneratorOptions & options,io::Printer * printer,const FieldDescriptor * field) const3252 void Generator::GenerateClassSerializeBinaryField(
3253 const GeneratorOptions& options, io::Printer* printer,
3254 const FieldDescriptor* field) const {
3255 if (HasFieldPresence(options, field) &&
3256 field->cpp_type() != FieldDescriptor::CPPTYPE_MESSAGE) {
3257 std::string typed_annotation =
3258 JSFieldTypeAnnotation(options, field,
3259 /* is_setter_argument = */ false,
3260 /* force_present = */ false,
3261 /* singular_if_not_packed = */ false,
3262 /* bytes_mode = */ BYTES_DEFAULT);
3263 printer->Print(
3264 " f = /** @type {$type$} */ "
3265 "(jspb.Message.getField(message, $index$));\n",
3266 "index", JSFieldIndex(field), "type", typed_annotation);
3267 } else {
3268 printer->Print(
3269 " f = message.get$name$($nolazy$);\n", "name",
3270 JSGetterName(options, field, BYTES_U8),
3271 // No lazy creation for maps containers -- fastpath the empty case.
3272 "nolazy", field->is_map() ? "true" : "");
3273 }
3274
3275 // Print an `if (condition)` statement that evaluates to true if the field
3276 // goes on the wire.
3277 if (field->is_map()) {
3278 printer->Print(" if (f && f.getLength() > 0) {\n");
3279 } else if (field->is_repeated()) {
3280 printer->Print(" if (f.length > 0) {\n");
3281 } else {
3282 if (HasFieldPresence(options, field)) {
3283 printer->Print(" if (f != null) {\n");
3284 } else {
3285 // No field presence: serialize onto the wire only if value is
3286 // non-default. Defaults are documented here:
3287 // https://goto.google.com/lhdfm
3288 switch (field->cpp_type()) {
3289 case FieldDescriptor::CPPTYPE_INT32:
3290 case FieldDescriptor::CPPTYPE_INT64:
3291 case FieldDescriptor::CPPTYPE_UINT32:
3292 case FieldDescriptor::CPPTYPE_UINT64: {
3293 if (IsIntegralFieldWithStringJSType(field)) {
3294 // We can use `parseInt` here even though it will not be precise for
3295 // 64-bit quantities because we are only testing for zero/nonzero,
3296 // and JS numbers (64-bit floating point values, i.e., doubles) are
3297 // integer-precise in the range that includes zero.
3298 printer->Print(" if (parseInt(f, 10) !== 0) {\n");
3299 } else {
3300 printer->Print(" if (f !== 0) {\n");
3301 }
3302 break;
3303 }
3304
3305 case FieldDescriptor::CPPTYPE_ENUM:
3306 case FieldDescriptor::CPPTYPE_FLOAT:
3307 case FieldDescriptor::CPPTYPE_DOUBLE:
3308 printer->Print(" if (f !== 0.0) {\n");
3309 break;
3310 case FieldDescriptor::CPPTYPE_BOOL:
3311 printer->Print(" if (f) {\n");
3312 break;
3313 case FieldDescriptor::CPPTYPE_STRING:
3314 printer->Print(" if (f.length > 0) {\n");
3315 break;
3316 default:
3317 assert(false);
3318 break;
3319 }
3320 }
3321 }
3322
3323 // Write the field on the wire.
3324 if (field->is_map()) {
3325 const FieldDescriptor* key_field = MapFieldKey(field);
3326 const FieldDescriptor* value_field = MapFieldValue(field);
3327 printer->Print(
3328 " f.serializeBinary($index$, writer, "
3329 "$keyWriterFn$, $valueWriterFn$",
3330 "index", StrCat(field->number()), "keyWriterFn",
3331 JSBinaryWriterMethodName(options, key_field), "valueWriterFn",
3332 JSBinaryWriterMethodName(options, value_field));
3333
3334 if (value_field->type() == FieldDescriptor::TYPE_MESSAGE) {
3335 printer->Print(", $messageType$.serializeBinaryToWriter", "messageType",
3336 GetMessagePath(options, value_field->message_type()));
3337 }
3338
3339 printer->Print(");\n");
3340 } else {
3341 printer->Print(
3342 " writer.write$method$(\n"
3343 " $index$,\n"
3344 " f",
3345 "method", JSBinaryReadWriteMethodName(field, /* is_writer = */ true),
3346 "index", StrCat(field->number()));
3347
3348 if (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE &&
3349 !field->is_map()) {
3350 printer->Print(
3351 ",\n"
3352 " $submsg$.serializeBinaryToWriter\n",
3353 "submsg", SubmessageTypeRef(options, field));
3354 } else {
3355 printer->Print("\n");
3356 }
3357
3358 printer->Print(" );\n");
3359 }
3360
3361 // Close the `if`.
3362 printer->Print(" }\n");
3363 }
3364
GenerateEnum(const GeneratorOptions & options,io::Printer * printer,const EnumDescriptor * enumdesc) const3365 void Generator::GenerateEnum(const GeneratorOptions& options,
3366 io::Printer* printer,
3367 const EnumDescriptor* enumdesc) const {
3368 printer->Print(
3369 "/**\n"
3370 " * @enum {number}\n"
3371 " */\n"
3372 "$enumprefix$$name$ = {\n",
3373 "enumprefix", GetEnumPathPrefix(options, enumdesc), "name",
3374 enumdesc->name());
3375 printer->Annotate("name", enumdesc);
3376
3377 for (int i = 0; i < enumdesc->value_count(); i++) {
3378 const EnumValueDescriptor* value = enumdesc->value(i);
3379 printer->Print(" $name$: $value$$comma$\n", "name",
3380 ToEnumCase(value->name()), "value",
3381 StrCat(value->number()), "comma",
3382 (i == enumdesc->value_count() - 1) ? "" : ",");
3383 printer->Annotate("name", value);
3384 }
3385
3386 printer->Print(
3387 "};\n"
3388 "\n");
3389 }
3390
GenerateExtension(const GeneratorOptions & options,io::Printer * printer,const FieldDescriptor * field) const3391 void Generator::GenerateExtension(const GeneratorOptions& options,
3392 io::Printer* printer,
3393 const FieldDescriptor* field) const {
3394 std::string extension_scope =
3395 (field->extension_scope()
3396 ? GetMessagePath(options, field->extension_scope())
3397 : GetNamespace(options, field->file()));
3398
3399 const std::string extension_object_name = JSObjectFieldName(options, field);
3400 printer->Print(
3401 "\n"
3402 "/**\n"
3403 " * A tuple of {field number, class constructor} for the extension\n"
3404 " * field named `$nameInComment$`.\n"
3405 " * @type {!jspb.ExtensionFieldInfo<$extensionType$>}\n"
3406 " */\n"
3407 "$class$.$name$ = new jspb.ExtensionFieldInfo(\n",
3408 "nameInComment", extension_object_name, "name", extension_object_name,
3409 "class", extension_scope, "extensionType",
3410 JSFieldTypeAnnotation(options, field,
3411 /* is_setter_argument = */ false,
3412 /* force_present = */ true,
3413 /* singular_if_not_packed = */ false));
3414 printer->Annotate("name", field);
3415 printer->Print(
3416 " $index$,\n"
3417 " {$name$: 0},\n"
3418 " $ctor$,\n"
3419 " /** @type {?function((boolean|undefined),!jspb.Message=): "
3420 "!Object} */ (\n"
3421 " $toObject$),\n"
3422 " $repeated$);\n",
3423 "index", StrCat(field->number()), "name", extension_object_name,
3424 "ctor",
3425 (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE
3426 ? SubmessageTypeRef(options, field)
3427 : std::string("null")),
3428 "toObject",
3429 (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE
3430 ? (SubmessageTypeRef(options, field) + ".toObject")
3431 : std::string("null")),
3432 "repeated", (field->is_repeated() ? "1" : "0"));
3433
3434 printer->Print(
3435 "\n"
3436 "$extendName$Binary[$index$] = new jspb.ExtensionFieldBinaryInfo(\n"
3437 " $class$.$name$,\n"
3438 " $binaryReaderFn$,\n"
3439 " $binaryWriterFn$,\n"
3440 " $binaryMessageSerializeFn$,\n"
3441 " $binaryMessageDeserializeFn$,\n",
3442 "extendName",
3443 JSExtensionsObjectName(options, field->file(), field->containing_type()),
3444 "index", StrCat(field->number()), "class", extension_scope, "name",
3445 extension_object_name, "binaryReaderFn",
3446 JSBinaryReaderMethodName(options, field), "binaryWriterFn",
3447 JSBinaryWriterMethodName(options, field), "binaryMessageSerializeFn",
3448 (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE)
3449 ? (SubmessageTypeRef(options, field) + ".serializeBinaryToWriter")
3450 : "undefined",
3451 "binaryMessageDeserializeFn",
3452 (field->cpp_type() == FieldDescriptor::CPPTYPE_MESSAGE)
3453 ? (SubmessageTypeRef(options, field) + ".deserializeBinaryFromReader")
3454 : "undefined");
3455
3456 printer->Print(" $isPacked$);\n", "isPacked",
3457 (field->is_packed() ? "true" : "false"));
3458
3459 printer->Print(
3460 "// This registers the extension field with the extended class, so that\n"
3461 "// toObject() will function correctly.\n"
3462 "$extendName$[$index$] = $class$.$name$;\n"
3463 "\n",
3464 "extendName",
3465 JSExtensionsObjectName(options, field->file(), field->containing_type()),
3466 "index", StrCat(field->number()), "class", extension_scope, "name",
3467 extension_object_name);
3468 }
3469
ParseFromOptions(const std::vector<std::pair<std::string,std::string>> & options,std::string * error)3470 bool GeneratorOptions::ParseFromOptions(
3471 const std::vector<std::pair<std::string, std::string> >& options,
3472 std::string* error) {
3473 for (int i = 0; i < options.size(); i++) {
3474 if (options[i].first == "add_require_for_enums") {
3475 if (options[i].second != "") {
3476 *error = "Unexpected option value for add_require_for_enums";
3477 return false;
3478 }
3479 add_require_for_enums = true;
3480 } else if (options[i].first == "binary") {
3481 if (options[i].second != "") {
3482 *error = "Unexpected option value for binary";
3483 return false;
3484 }
3485 binary = true;
3486 } else if (options[i].first == "testonly") {
3487 if (options[i].second != "") {
3488 *error = "Unexpected option value for testonly";
3489 return false;
3490 }
3491 testonly = true;
3492 } else if (options[i].first == "error_on_name_conflict") {
3493 if (options[i].second != "") {
3494 *error = "Unexpected option value for error_on_name_conflict";
3495 return false;
3496 }
3497 error_on_name_conflict = true;
3498 } else if (options[i].first == "output_dir") {
3499 output_dir = options[i].second;
3500 } else if (options[i].first == "namespace_prefix") {
3501 namespace_prefix = options[i].second;
3502 } else if (options[i].first == "library") {
3503 library = options[i].second;
3504 } else if (options[i].first == "import_style") {
3505 if (options[i].second == "closure") {
3506 import_style = kImportClosure;
3507 } else if (options[i].second == "commonjs") {
3508 import_style = kImportCommonJs;
3509 } else if (options[i].second == "commonjs_strict") {
3510 import_style = kImportCommonJsStrict;
3511 } else if (options[i].second == "browser") {
3512 import_style = kImportBrowser;
3513 } else if (options[i].second == "es6") {
3514 import_style = kImportEs6;
3515 } else {
3516 *error = "Unknown import style " + options[i].second + ", expected " +
3517 "one of: closure, commonjs, browser, es6.";
3518 }
3519 } else if (options[i].first == "extension") {
3520 extension = options[i].second;
3521 } else if (options[i].first == "one_output_file_per_input_file") {
3522 if (!options[i].second.empty()) {
3523 *error = "Unexpected option value for one_output_file_per_input_file";
3524 return false;
3525 }
3526 one_output_file_per_input_file = true;
3527 } else if (options[i].first == "annotate_code") {
3528 if (!options[i].second.empty()) {
3529 *error = "Unexpected option value for annotate_code";
3530 return false;
3531 }
3532 annotate_code = true;
3533 } else {
3534 // Assume any other option is an output directory, as long as it is a bare
3535 // `key` rather than a `key=value` option.
3536 if (options[i].second != "") {
3537 *error = "Unknown option: " + options[i].first;
3538 return false;
3539 }
3540 output_dir = options[i].first;
3541 }
3542 }
3543
3544 if (import_style != kImportClosure &&
3545 (add_require_for_enums || testonly || !library.empty() ||
3546 error_on_name_conflict || extension != ".js" ||
3547 one_output_file_per_input_file)) {
3548 *error =
3549 "The add_require_for_enums, testonly, library, error_on_name_conflict, "
3550 "extension, and one_output_file_per_input_file options should only be "
3551 "used for import_style=closure";
3552 return false;
3553 }
3554
3555 return true;
3556 }
3557
output_mode() const3558 GeneratorOptions::OutputMode GeneratorOptions::output_mode() const {
3559 // We use one output file per input file if we are not using Closure or if
3560 // this is explicitly requested.
3561 if (import_style != kImportClosure || one_output_file_per_input_file) {
3562 return kOneOutputFilePerInputFile;
3563 }
3564
3565 // If a library name is provided, we put everything in that one file.
3566 if (!library.empty()) {
3567 return kEverythingInOneFile;
3568 }
3569
3570 // Otherwise, we create one output file per SCC.
3571 return kOneOutputFilePerSCC;
3572 }
3573
GenerateFilesInDepOrder(const GeneratorOptions & options,io::Printer * printer,const std::vector<const FileDescriptor * > & files) const3574 void Generator::GenerateFilesInDepOrder(
3575 const GeneratorOptions& options, io::Printer* printer,
3576 const std::vector<const FileDescriptor*>& files) const {
3577 // Build a std::set over all files so that the DFS can detect when it recurses
3578 // into a dep not specified in the user's command line.
3579 std::set<const FileDescriptor*> all_files(files.begin(), files.end());
3580 // Track the in-progress set of files that have been generated already.
3581 std::set<const FileDescriptor*> generated;
3582 for (int i = 0; i < files.size(); i++) {
3583 GenerateFileAndDeps(options, printer, files[i], &all_files, &generated);
3584 }
3585 }
3586
GenerateFileAndDeps(const GeneratorOptions & options,io::Printer * printer,const FileDescriptor * root,std::set<const FileDescriptor * > * all_files,std::set<const FileDescriptor * > * generated) const3587 void Generator::GenerateFileAndDeps(
3588 const GeneratorOptions& options, io::Printer* printer,
3589 const FileDescriptor* root, std::set<const FileDescriptor*>* all_files,
3590 std::set<const FileDescriptor*>* generated) const {
3591 // Skip if already generated.
3592 if (generated->find(root) != generated->end()) {
3593 return;
3594 }
3595 generated->insert(root);
3596
3597 // Generate all dependencies before this file's content.
3598 for (int i = 0; i < root->dependency_count(); i++) {
3599 const FileDescriptor* dep = root->dependency(i);
3600 GenerateFileAndDeps(options, printer, dep, all_files, generated);
3601 }
3602
3603 // Generate this file's content. Only generate if the file is part of the
3604 // original set requested to be generated; i.e., don't take all transitive
3605 // deps down to the roots.
3606 if (all_files->find(root) != all_files->end()) {
3607 GenerateClassesAndEnums(options, printer, root);
3608 }
3609 }
3610
GenerateFile(const FileDescriptor * file,const GeneratorOptions & options,GeneratorContext * context,bool use_short_name) const3611 bool Generator::GenerateFile(const FileDescriptor* file,
3612 const GeneratorOptions& options,
3613 GeneratorContext* context,
3614 bool use_short_name) const {
3615 std::string filename =
3616 options.output_dir + "/" +
3617 GetJSFilename(options, use_short_name
3618 ? file->name().substr(file->name().rfind('/'))
3619 : file->name());
3620 std::unique_ptr<io::ZeroCopyOutputStream> output(context->Open(filename));
3621 GOOGLE_CHECK(output);
3622 GeneratedCodeInfo annotations;
3623 io::AnnotationProtoCollector<GeneratedCodeInfo> annotation_collector(
3624 &annotations);
3625 io::Printer printer(output.get(), '$',
3626 options.annotate_code ? &annotation_collector : nullptr);
3627
3628 GenerateFile(options, &printer, file);
3629
3630 if (printer.failed()) {
3631 return false;
3632 }
3633
3634 if (options.annotate_code) {
3635 EmbedCodeAnnotations(annotations, &printer);
3636 }
3637
3638 return true;
3639 }
3640
GenerateFile(const GeneratorOptions & options,io::Printer * printer,const FileDescriptor * file) const3641 void Generator::GenerateFile(const GeneratorOptions& options,
3642 io::Printer* printer,
3643 const FileDescriptor* file) const {
3644 GenerateHeader(options, file, printer);
3645
3646 // Generate "require" statements.
3647 if ((options.import_style == GeneratorOptions::kImportCommonJs ||
3648 options.import_style == GeneratorOptions::kImportCommonJsStrict)) {
3649 printer->Print("var jspb = require('google-protobuf');\n");
3650 printer->Print("var goog = jspb;\n");
3651
3652 // Do not use global scope in strict mode
3653 if (options.import_style == GeneratorOptions::kImportCommonJsStrict) {
3654 printer->Print("var proto = {};\n\n");
3655 } else {
3656 printer->Print("var global = Function('return this')();\n\n");
3657 }
3658
3659 for (int i = 0; i < file->dependency_count(); i++) {
3660 const std::string& name = file->dependency(i)->name();
3661 printer->Print(
3662 "var $alias$ = require('$file$');\n"
3663 "goog.object.extend(proto, $alias$);\n",
3664 "alias", ModuleAlias(name), "file",
3665 GetRootPath(file->name(), name) + GetJSFilename(options, name));
3666 }
3667 }
3668
3669 std::set<std::string> provided;
3670 std::set<const FieldDescriptor*> extensions;
3671 for (int i = 0; i < file->extension_count(); i++) {
3672 // We honor the jspb::ignore option here only when working with
3673 // Closure-style imports. Use of this option is discouraged and so we want
3674 // to avoid adding new support for it.
3675 if (options.import_style == GeneratorOptions::kImportClosure &&
3676 IgnoreField(file->extension(i))) {
3677 continue;
3678 }
3679 provided.insert(GetNamespace(options, file) + "." +
3680 JSObjectFieldName(options, file->extension(i)));
3681 extensions.insert(file->extension(i));
3682 }
3683
3684 FindProvidesForFile(options, printer, file, &provided);
3685 GenerateProvides(options, printer, &provided);
3686 std::vector<const FileDescriptor*> files;
3687 files.push_back(file);
3688 if (options.import_style == GeneratorOptions::kImportClosure) {
3689 GenerateRequiresForLibrary(options, printer, files, &provided);
3690 }
3691
3692 GenerateClassesAndEnums(options, printer, file);
3693
3694 // Generate code for top-level extensions. Extensions nested inside messages
3695 // are emitted inside GenerateClassesAndEnums().
3696 for (std::set<const FieldDescriptor*>::const_iterator it = extensions.begin();
3697 it != extensions.end(); ++it) {
3698 GenerateExtension(options, printer, *it);
3699 }
3700
3701 // if provided is empty, do not export anything
3702 if (options.import_style == GeneratorOptions::kImportCommonJs &&
3703 !provided.empty()) {
3704 printer->Print("goog.object.extend(exports, $package$);\n", "package",
3705 GetNamespace(options, file));
3706 } else if (options.import_style == GeneratorOptions::kImportCommonJsStrict) {
3707 printer->Print("goog.object.extend(exports, proto);\n", "package",
3708 GetNamespace(options, file));
3709 }
3710
3711 // Emit well-known type methods.
3712 for (FileToc* toc = well_known_types_js; toc->name != NULL; toc++) {
3713 std::string name = std::string("google/protobuf/") + toc->name;
3714 if (name == StripProto(file->name()) + ".js") {
3715 printer->Print(toc->data);
3716 }
3717 }
3718 }
3719
GenerateAll(const std::vector<const FileDescriptor * > & files,const std::string & parameter,GeneratorContext * context,std::string * error) const3720 bool Generator::GenerateAll(const std::vector<const FileDescriptor*>& files,
3721 const std::string& parameter,
3722 GeneratorContext* context,
3723 std::string* error) const {
3724 std::vector<std::pair<std::string, std::string> > option_pairs;
3725 ParseGeneratorParameter(parameter, &option_pairs);
3726 GeneratorOptions options;
3727 if (!options.ParseFromOptions(option_pairs, error)) {
3728 return false;
3729 }
3730
3731
3732 if (options.output_mode() == GeneratorOptions::kEverythingInOneFile) {
3733 // All output should go in a single file.
3734 std::string filename = options.output_dir + "/" + options.library +
3735 options.GetFileNameExtension();
3736 std::unique_ptr<io::ZeroCopyOutputStream> output(context->Open(filename));
3737 GOOGLE_CHECK(output.get());
3738 GeneratedCodeInfo annotations;
3739 io::AnnotationProtoCollector<GeneratedCodeInfo> annotation_collector(
3740 &annotations);
3741 io::Printer printer(
3742 output.get(), '$',
3743 options.annotate_code ? &annotation_collector : nullptr);
3744
3745 // Pull out all extensions -- we need these to generate all
3746 // provides/requires.
3747 std::vector<const FieldDescriptor*> extensions;
3748 for (int i = 0; i < files.size(); i++) {
3749 for (int j = 0; j < files[i]->extension_count(); j++) {
3750 const FieldDescriptor* extension = files[i]->extension(j);
3751 extensions.push_back(extension);
3752 }
3753 }
3754
3755 if (files.size() == 1) {
3756 GenerateHeader(options, files[0], &printer);
3757 } else {
3758 GenerateHeader(options, nullptr, &printer);
3759 }
3760
3761 std::set<std::string> provided;
3762 FindProvides(options, &printer, files, &provided);
3763 FindProvidesForFields(options, &printer, extensions, &provided);
3764 GenerateProvides(options, &printer, &provided);
3765 GenerateTestOnly(options, &printer);
3766 GenerateRequiresForLibrary(options, &printer, files, &provided);
3767
3768 GenerateFilesInDepOrder(options, &printer, files);
3769
3770 for (int i = 0; i < extensions.size(); i++) {
3771 if (ShouldGenerateExtension(extensions[i])) {
3772 GenerateExtension(options, &printer, extensions[i]);
3773 }
3774 }
3775
3776 if (printer.failed()) {
3777 return false;
3778 }
3779 if (options.annotate_code) {
3780 EmbedCodeAnnotations(annotations, &printer);
3781 }
3782 } else if (options.output_mode() == GeneratorOptions::kOneOutputFilePerSCC) {
3783 std::set<const Descriptor*> have_printed;
3784 SCCAnalyzer<DepsGenerator> analyzer;
3785 std::map<const void*, std::string> allowed_map;
3786 if (!GenerateJspbAllowedMap(options, files, &allowed_map, &analyzer,
3787 error)) {
3788 return false;
3789 }
3790
3791 bool generated = false;
3792 for (int i = 0; i < files.size(); i++) {
3793 const FileDescriptor* file = files[i];
3794 // Force well known type to generate in a whole file.
3795 if (IsWellKnownTypeFile(file)) {
3796 if (!GenerateFile(file, options, context, true)) {
3797 return false;
3798 }
3799 generated = true;
3800 continue;
3801 }
3802 for (int j = 0; j < file->message_type_count(); j++) {
3803 const Descriptor* desc = file->message_type(j);
3804 if (have_printed.count(desc) ||
3805 allowed_map.count(analyzer.GetSCC(desc)) == 0) {
3806 continue;
3807 }
3808
3809 generated = true;
3810 const SCC* scc = analyzer.GetSCC(desc);
3811 const std::string& filename = allowed_map[scc];
3812 std::unique_ptr<io::ZeroCopyOutputStream> output(
3813 context->Open(filename));
3814 GOOGLE_CHECK(output.get());
3815 GeneratedCodeInfo annotations;
3816 io::AnnotationProtoCollector<GeneratedCodeInfo> annotation_collector(
3817 &annotations);
3818 io::Printer printer(
3819 output.get(), '$',
3820 options.annotate_code ? &annotation_collector : nullptr);
3821
3822 GenerateHeader(options, file, &printer);
3823
3824 std::set<std::string> provided;
3825 for (auto one_desc : scc->descriptors) {
3826 if (one_desc->containing_type() == nullptr) {
3827 FindProvidesForMessage(options, &printer, one_desc, &provided);
3828 }
3829 }
3830 GenerateProvides(options, &printer, &provided);
3831 GenerateTestOnly(options, &printer);
3832 GenerateRequiresForSCC(options, &printer, scc, &provided);
3833
3834 for (auto one_desc : scc->descriptors) {
3835 if (one_desc->containing_type() == nullptr) {
3836 GenerateClassConstructorAndDeclareExtensionFieldInfo(
3837 options, &printer, one_desc);
3838 }
3839 }
3840 for (auto one_desc : scc->descriptors) {
3841 if (one_desc->containing_type() == nullptr) {
3842 GenerateClass(options, &printer, one_desc);
3843 }
3844 }
3845
3846 for (auto one_desc : scc->descriptors) {
3847 have_printed.insert(one_desc);
3848 }
3849
3850 if (printer.failed()) {
3851 return false;
3852 }
3853 if (options.annotate_code) {
3854 EmbedCodeAnnotations(annotations, &printer);
3855 }
3856 }
3857 for (int j = 0; j < file->enum_type_count(); j++) {
3858 const EnumDescriptor* enumdesc = file->enum_type(j);
3859 if (allowed_map.count(enumdesc) == 0) {
3860 continue;
3861 }
3862
3863 generated = true;
3864 const std::string& filename = allowed_map[enumdesc];
3865 std::unique_ptr<io::ZeroCopyOutputStream> output(
3866 context->Open(filename));
3867 GOOGLE_CHECK(output.get());
3868 GeneratedCodeInfo annotations;
3869 io::AnnotationProtoCollector<GeneratedCodeInfo> annotation_collector(
3870 &annotations);
3871 io::Printer printer(
3872 output.get(), '$',
3873 options.annotate_code ? &annotation_collector : nullptr);
3874
3875 GenerateHeader(options, file, &printer);
3876
3877 std::set<std::string> provided;
3878 FindProvidesForEnum(options, &printer, enumdesc, &provided);
3879 GenerateProvides(options, &printer, &provided);
3880 GenerateTestOnly(options, &printer);
3881
3882 GenerateEnum(options, &printer, enumdesc);
3883
3884 if (printer.failed()) {
3885 return false;
3886 }
3887 if (options.annotate_code) {
3888 EmbedCodeAnnotations(annotations, &printer);
3889 }
3890 }
3891 // File-level extensions (message-level extensions are generated under
3892 // the enclosing message).
3893 if (allowed_map.count(file) == 1) {
3894 generated = true;
3895 const std::string& filename = allowed_map[file];
3896
3897 std::unique_ptr<io::ZeroCopyOutputStream> output(
3898 context->Open(filename));
3899 GOOGLE_CHECK(output.get());
3900 GeneratedCodeInfo annotations;
3901 io::AnnotationProtoCollector<GeneratedCodeInfo> annotation_collector(
3902 &annotations);
3903 io::Printer printer(
3904 output.get(), '$',
3905 options.annotate_code ? &annotation_collector : nullptr);
3906
3907 GenerateHeader(options, file, &printer);
3908
3909 std::set<std::string> provided;
3910 std::vector<const FieldDescriptor*> fields;
3911
3912 for (int j = 0; j < files[i]->extension_count(); j++) {
3913 if (ShouldGenerateExtension(files[i]->extension(j))) {
3914 fields.push_back(files[i]->extension(j));
3915 }
3916 }
3917
3918 FindProvidesForFields(options, &printer, fields, &provided);
3919 GenerateProvides(options, &printer, &provided);
3920 GenerateTestOnly(options, &printer);
3921 GenerateRequiresForExtensions(options, &printer, fields, &provided);
3922
3923 for (int j = 0; j < files[i]->extension_count(); j++) {
3924 if (ShouldGenerateExtension(files[i]->extension(j))) {
3925 GenerateExtension(options, &printer, files[i]->extension(j));
3926 }
3927 }
3928 if (options.annotate_code) {
3929 EmbedCodeAnnotations(annotations, &printer);
3930 }
3931 }
3932 }
3933 if (!generated) {
3934 std::string filename = options.output_dir + "/" +
3935 "empty_no_content_void_file" +
3936 options.GetFileNameExtension();
3937 std::unique_ptr<io::ZeroCopyOutputStream> output(context->Open(filename));
3938 }
3939 } else /* options.output_mode() == kOneOutputFilePerInputFile */ {
3940 // Generate one output file per input (.proto) file.
3941
3942 for (int i = 0; i < files.size(); i++) {
3943 const FileDescriptor* file = files[i];
3944 if (!GenerateFile(file, options, context, false)) {
3945 return false;
3946 }
3947 }
3948 }
3949 return true;
3950 }
3951
3952 } // namespace js
3953 } // namespace compiler
3954 } // namespace protobuf
3955 } // namespace google
3956