• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2014 Google Inc. All rights reserved.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *     http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 // independent from idl_parser, since this code is not needed for most clients
18 #include <unordered_set>
19 #include <unordered_map>
20 #include <cassert>
21 
22 #include "flatbuffers/flatbuffers.h"
23 #include "flatbuffers/idl.h"
24 #include "flatbuffers/util.h"
25 #include "flatbuffers/code_generators.h"
26 
27 namespace flatbuffers {
28 
29 const std::string kGeneratedFileNamePostfix = "_generated";
30 
31 struct JsLanguageParameters {
32   IDLOptions::Language language;
33   std::string file_extension;
34 };
35 
36 struct ReexportDescription {
37   std::string symbol;
38   std::string source_namespace;
39   std::string target_namespace;
40 };
41 
GetJsLangParams(IDLOptions::Language lang)42 const JsLanguageParameters& GetJsLangParams(IDLOptions::Language lang) {
43   static JsLanguageParameters js_language_parameters[] = {
44     {
45       IDLOptions::kJs,
46       ".js",
47     },
48     {
49       IDLOptions::kTs,
50       ".ts",
51     },
52   };
53 
54   if (lang == IDLOptions::kJs) {
55     return js_language_parameters[0];
56   } else {
57     assert(lang == IDLOptions::kTs);
58     return js_language_parameters[1];
59   }
60 }
61 
GeneratedFileName(const std::string & path,const std::string & file_name,const JsLanguageParameters & lang)62 static std::string GeneratedFileName(const std::string &path,
63                                      const std::string &file_name,
64                                      const JsLanguageParameters &lang) {
65   return path + file_name + kGeneratedFileNamePostfix + lang.file_extension;
66 }
67 
68 namespace js {
69 // Iterate through all definitions we haven't generate code for (enums, structs,
70 // and tables) and output them to a single file.
71 class JsGenerator : public BaseGenerator {
72  public:
73   typedef std::unordered_set<std::string> imported_fileset;
74   typedef std::unordered_multimap<std::string, ReexportDescription>
75           reexport_map;
76 
JsGenerator(const Parser & parser,const std::string & path,const std::string & file_name)77   JsGenerator(const Parser &parser, const std::string &path,
78               const std::string &file_name)
79       : BaseGenerator(parser, path, file_name, "", "."),
80         lang_(GetJsLangParams(parser_.opts.lang))
81   {
82   };
83   // Iterate through all definitions we haven't generate code for (enums,
84   // structs, and tables) and output them to a single file.
generate()85   bool generate() {
86     imported_fileset imported_files;
87     reexport_map reexports;
88 
89     std::string enum_code, struct_code, import_code, exports_code, code;
90     generateEnums(&enum_code, &exports_code, reexports);
91     generateStructs(&struct_code, &exports_code, imported_files);
92     generateImportDependencies(&import_code, imported_files);
93     generateReexports(&import_code, reexports, imported_files);
94 
95     code = code + "// " + FlatBuffersGeneratedWarning() + "\n\n";
96 
97     // Generate code for all the namespace declarations.
98     GenNamespaces(&code, &exports_code);
99 
100     // Output the main declaration code from above.
101     code += import_code;
102 
103     code += enum_code;
104     code += struct_code;
105 
106     if (lang_.language == IDLOptions::kJs && !exports_code.empty() &&
107         !parser_.opts.skip_js_exports) {
108       code += "// Exports for Node.js and RequireJS\n";
109       code += exports_code;
110     }
111 
112     return SaveFile(GeneratedFileName(path_, file_name_, lang_).c_str(), code,
113                     false);
114   }
115 
116  private:
117   JsLanguageParameters lang_;
118 
119   // Generate code for imports
generateImportDependencies(std::string * code_ptr,const imported_fileset & imported_files)120   void generateImportDependencies(std::string *code_ptr,
121                                   const imported_fileset &imported_files) {
122     std::string &code = *code_ptr;
123     for (auto it = imported_files.begin(); it != imported_files.end(); ++it) {
124       const auto &file = *it;
125       const auto basename =
126           flatbuffers::StripPath(flatbuffers::StripExtension(file));
127       if (basename != file_name_) {
128         const auto file_name = basename + kGeneratedFileNamePostfix;
129         code += GenPrefixedImport(file, file_name);
130       }
131     }
132   }
133 
134   // Generate reexports, which might not have been explicitly imported using the
135   // "export import" trick
generateReexports(std::string * code_ptr,const reexport_map & reexports,imported_fileset imported_files)136   void generateReexports(std::string *code_ptr,
137                          const reexport_map &reexports,
138                          imported_fileset imported_files) {
139       if (!parser_.opts.reexport_ts_modules ||
140           lang_.language != IDLOptions::kTs) {
141           return;
142       }
143 
144       std::string &code = *code_ptr;
145       for (auto it = reexports.begin(); it != reexports.end(); ++it) {
146         const auto &file = *it;
147         const auto basename =
148             flatbuffers::StripPath(flatbuffers::StripExtension(file.first));
149         if (basename != file_name_) {
150           const auto file_name = basename + kGeneratedFileNamePostfix;
151 
152           if (imported_files.find(file.first) == imported_files.end()) {
153             code += GenPrefixedImport(file.first, file_name);
154             imported_files.emplace(file.first);
155           }
156 
157           code += "export namespace " + file.second.target_namespace + " { \n";
158           code += "export import " + file.second.symbol + " = ";
159           code += GenFileNamespacePrefix(file.first) + "." +
160                   file.second.source_namespace + "." + file.second.symbol +
161                   "; }\n";
162         }
163       }
164   }
165 
166   // Generate code for all enums.
generateEnums(std::string * enum_code_ptr,std::string * exports_code_ptr,reexport_map & reexports)167   void generateEnums(std::string *enum_code_ptr,
168                      std::string *exports_code_ptr,
169                      reexport_map &reexports) {
170     for (auto it = parser_.enums_.vec.begin(); it != parser_.enums_.vec.end();
171          ++it) {
172       auto &enum_def = **it;
173       GenEnum(enum_def, enum_code_ptr, exports_code_ptr, reexports);
174     }
175   }
176 
177   // Generate code for all structs.
generateStructs(std::string * decl_code_ptr,std::string * exports_code_ptr,imported_fileset & imported_files)178   void generateStructs(std::string *decl_code_ptr,
179                        std::string *exports_code_ptr,
180                        imported_fileset &imported_files) {
181     for (auto it = parser_.structs_.vec.begin();
182          it != parser_.structs_.vec.end(); ++it) {
183       auto &struct_def = **it;
184       GenStruct(parser_, struct_def, decl_code_ptr, exports_code_ptr,
185                 imported_files);
186     }
187   }
GenNamespaces(std::string * code_ptr,std::string * exports_ptr)188   void GenNamespaces(std::string *code_ptr, std::string *exports_ptr) {
189   if (lang_.language == IDLOptions::kTs &&
190       parser_.opts.skip_flatbuffers_import) {
191     return;
192   }
193 
194   std::set<std::string> namespaces;
195 
196   for (auto it = parser_.namespaces_.begin();
197        it != parser_.namespaces_.end(); ++it) {
198     std::string namespace_so_far;
199 
200     // Gather all parent namespaces for this namespace
201     for (auto component = (*it)->components.begin();
202          component != (*it)->components.end(); ++component) {
203       if (!namespace_so_far.empty()) {
204         namespace_so_far += '.';
205       }
206       namespace_so_far += *component;
207       namespaces.insert(namespace_so_far);
208     }
209   }
210 
211   // Make sure parent namespaces come before child namespaces
212   std::vector<std::string> sorted_namespaces(
213     namespaces.begin(), namespaces.end());
214   std::sort(sorted_namespaces.begin(), sorted_namespaces.end());
215 
216   // Emit namespaces in a form that Closure Compiler can optimize
217   std::string &code = *code_ptr;
218   std::string &exports = *exports_ptr;
219   for (auto it = sorted_namespaces.begin();
220        it != sorted_namespaces.end(); it++) {
221     if (lang_.language == IDLOptions::kTs) {
222       if (it->find('.') == std::string::npos) {
223         code += "import { flatbuffers } from \"./flatbuffers\"\n";
224         break;
225       }
226     } else {
227       code += "/**\n * @const\n * @namespace\n */\n";
228       if (it->find('.') == std::string::npos) {
229         code += "var ";
230         if(parser_.opts.use_goog_js_export_format) {
231           exports += "goog.exportSymbol('" + *it + "', " + *it + ");\n";
232         } else {
233           exports += "this." + *it + " = " + *it + ";\n";
234         }
235       }
236       code += *it + " = " + *it + " || {};\n\n";
237     }
238   }
239 }
240 
241 // Generate a documentation comment, if available.
GenDocComment(const std::vector<std::string> & dc,std::string * code_ptr,const std::string & extra_lines,const char * indent=nullptr)242 static void GenDocComment(const std::vector<std::string> &dc,
243                           std::string *code_ptr,
244                           const std::string &extra_lines,
245                           const char *indent = nullptr) {
246   if (dc.empty() && extra_lines.empty()) {
247     // Don't output empty comment blocks with 0 lines of comment content.
248     return;
249   }
250 
251   std::string &code = *code_ptr;
252   if (indent) code += indent;
253   code += "/**\n";
254   for (auto it = dc.begin(); it != dc.end(); ++it) {
255     if (indent) code += indent;
256     code += " *" + *it + "\n";
257   }
258   if (!extra_lines.empty()) {
259     if (!dc.empty()) {
260       if (indent) code += indent;
261       code += " *\n";
262     }
263     if (indent) code += indent;
264     std::string::size_type start = 0;
265     for (;;) {
266       auto end = extra_lines.find('\n', start);
267       if (end != std::string::npos) {
268         code += " * " + extra_lines.substr(start, end - start) + "\n";
269         start = end + 1;
270       } else {
271         code += " * " + extra_lines.substr(start) + "\n";
272         break;
273       }
274     }
275   }
276   if (indent) code += indent;
277   code += " */\n";
278 }
279 
GenDocComment(std::string * code_ptr,const std::string & extra_lines)280 static void GenDocComment(std::string *code_ptr,
281                           const std::string &extra_lines) {
282   GenDocComment(std::vector<std::string>(), code_ptr, extra_lines);
283 }
284 
285 // Generate an enum declaration and an enum string lookup table.
GenEnum(EnumDef & enum_def,std::string * code_ptr,std::string * exports_ptr,reexport_map & reexports)286 void GenEnum(EnumDef &enum_def, std::string *code_ptr,
287              std::string *exports_ptr, reexport_map &reexports) {
288   if (enum_def.generated) return;
289   std::string &code = *code_ptr;
290   std::string &exports = *exports_ptr;
291   GenDocComment(enum_def.doc_comment, code_ptr, "@enum");
292   std::string ns = GetNameSpace(enum_def);
293   if (lang_.language == IDLOptions::kTs) {
294     if (!ns.empty()) {
295       code += "export namespace " + ns + "{\n";
296     }
297     code += "export enum " + enum_def.name + "{\n";
298   } else {
299     if (enum_def.defined_namespace->components.empty()) {
300       code += "var ";
301       if(parser_.opts.use_goog_js_export_format) {
302         exports += "goog.exportSymbol('" + enum_def.name + "', " +
303                    enum_def.name + ");\n";
304       } else {
305         exports += "this." + enum_def.name + " = " + enum_def.name + ";\n";
306       }
307     }
308     code += WrapInNameSpace(enum_def) + " = {\n";
309   }
310   for (auto it = enum_def.vals.vec.begin();
311        it != enum_def.vals.vec.end(); ++it) {
312     auto &ev = **it;
313     if (!ev.doc_comment.empty()) {
314       if (it != enum_def.vals.vec.begin()) {
315         code += '\n';
316       }
317       GenDocComment(ev.doc_comment, code_ptr, "", "  ");
318     }
319     code += "  " + ev.name;
320     code += lang_.language == IDLOptions::kTs ? "= " : ": ";
321     code += NumToString(ev.value);
322     code += (it + 1) != enum_def.vals.vec.end() ? ",\n" : "\n";
323 
324     if (ev.union_type.struct_def) {
325         ReexportDescription desc = {
326           ev.name,
327           GetNameSpace(*ev.union_type.struct_def),
328           GetNameSpace(enum_def)
329         };
330         reexports.insert(std::make_pair(ev.union_type.struct_def->file,
331                                         std::move(desc)));
332     }
333   }
334 
335   if (lang_.language == IDLOptions::kTs && !ns.empty()) {
336     code += "}";
337   }
338   code += "};\n\n";
339 }
340 
GenType(const Type & type)341 static std::string GenType(const Type &type) {
342   switch (type.base_type) {
343     case BASE_TYPE_BOOL:
344     case BASE_TYPE_CHAR: return "Int8";
345     case BASE_TYPE_UTYPE:
346     case BASE_TYPE_UCHAR: return "Uint8";
347     case BASE_TYPE_SHORT: return "Int16";
348     case BASE_TYPE_USHORT: return "Uint16";
349     case BASE_TYPE_INT: return "Int32";
350     case BASE_TYPE_UINT: return "Uint32";
351     case BASE_TYPE_LONG: return "Int64";
352     case BASE_TYPE_ULONG: return "Uint64";
353     case BASE_TYPE_FLOAT: return "Float32";
354     case BASE_TYPE_DOUBLE: return "Float64";
355     case BASE_TYPE_STRING: return "String";
356     case BASE_TYPE_VECTOR: return GenType(type.VectorType());
357     case BASE_TYPE_STRUCT: return type.struct_def->name;
358     default: return "Table";
359   }
360 }
361 
GenGetter(const Type & type,const std::string & arguments)362 std::string GenGetter(const Type &type, const std::string &arguments) {
363   switch (type.base_type) {
364     case BASE_TYPE_STRING: return "this.bb.__string" + arguments;
365     case BASE_TYPE_STRUCT: return "this.bb.__struct" + arguments;
366     case BASE_TYPE_UNION:  return "this.bb.__union" + arguments;
367     case BASE_TYPE_VECTOR: return GenGetter(type.VectorType(), arguments);
368     default: {
369       auto getter = "this.bb.read" + MakeCamel(GenType(type)) + arguments;
370       if (type.base_type == BASE_TYPE_BOOL) {
371         getter = "!!" + getter;
372       }
373       if (type.enum_def) {
374         getter = "/** @type {" + WrapInNameSpace(*type.enum_def) + "} */ (" +
375           getter + ")";
376       }
377       return getter;
378     }
379   }
380 }
381 
GenDefaultValue(const Value & value,const std::string & context)382 std::string GenDefaultValue(const Value &value, const std::string &context) {
383   if (value.type.enum_def) {
384     if (auto val = value.type.enum_def->ReverseLookup(
385         atoi(value.constant.c_str()), false)) {
386       if (lang_.language == IDLOptions::kTs) {
387         return GenPrefixedTypeName(WrapInNameSpace(*value.type.enum_def),
388                                    value.type.enum_def->file) + "." + val->name;
389       } else {
390         return WrapInNameSpace(*value.type.enum_def) + "." + val->name;
391       }
392     } else {
393       return "/** @type {" + WrapInNameSpace(*value.type.enum_def) + "} */ ("
394         + value.constant + ")";
395     }
396   }
397 
398   switch (value.type.base_type) {
399     case BASE_TYPE_BOOL:
400       return value.constant == "0" ? "false" : "true";
401 
402     case BASE_TYPE_STRING:
403       return "null";
404 
405     case BASE_TYPE_LONG:
406     case BASE_TYPE_ULONG: {
407       int64_t constant = StringToInt(value.constant.c_str());
408       return context + ".createLong(" + NumToString((int32_t)constant) +
409         ", " + NumToString((int32_t)(constant >> 32)) + ")";
410     }
411 
412     default:
413       return value.constant;
414   }
415 }
416 
GenTypeName(const Type & type,bool input,bool allowNull=false)417 std::string GenTypeName(const Type &type, bool input, bool allowNull = false) {
418   if (!input) {
419     if (type.base_type == BASE_TYPE_STRING || type.base_type == BASE_TYPE_STRUCT) {
420       std::string name;
421       if (type.base_type == BASE_TYPE_STRING) {
422         name = "string|Uint8Array";
423       } else {
424         name = WrapInNameSpace(*type.struct_def);
425       }
426       return (allowNull) ? (name + "|null") : (name);
427     }
428   }
429 
430   switch (type.base_type) {
431     case BASE_TYPE_BOOL: return "boolean";
432     case BASE_TYPE_LONG:
433     case BASE_TYPE_ULONG: return "flatbuffers.Long";
434     default:
435       if (IsScalar(type.base_type)) {
436         if (type.enum_def) {
437           return WrapInNameSpace(*type.enum_def);
438         }
439         return "number";
440       }
441       return "flatbuffers.Offset";
442   }
443 }
444 
445 // Returns the method name for use with add/put calls.
GenWriteMethod(const Type & type)446 static std::string GenWriteMethod(const Type &type) {
447   // Forward to signed versions since unsigned versions don't exist
448   switch (type.base_type) {
449     case BASE_TYPE_UTYPE:
450     case BASE_TYPE_UCHAR: return GenWriteMethod(Type(BASE_TYPE_CHAR));
451     case BASE_TYPE_USHORT: return GenWriteMethod(Type(BASE_TYPE_SHORT));
452     case BASE_TYPE_UINT: return GenWriteMethod(Type(BASE_TYPE_INT));
453     case BASE_TYPE_ULONG: return GenWriteMethod(Type(BASE_TYPE_LONG));
454     default: break;
455   }
456 
457   return IsScalar(type.base_type)
458     ? MakeCamel(GenType(type))
459     : (IsStruct(type) ? "Struct" : "Offset");
460 }
461 
462 template <typename T>
MaybeAdd(T value)463 static std::string MaybeAdd(T value) {
464   return value != 0 ? " + " + NumToString(value) : "";
465 }
466 
467 template <typename T>
MaybeScale(T value)468 static std::string MaybeScale(T value) {
469   return value != 1 ? " * " + NumToString(value) : "";
470 }
471 
GenFileNamespacePrefix(const std::string & file)472 static std::string GenFileNamespacePrefix(const std::string &file) {
473   return "NS" + std::to_string(
474         static_cast<unsigned long long>(std::hash<std::string>()(file)));
475 }
476 
GenPrefixedImport(const std::string & full_file_name,const std::string & base_file_name)477 static std::string GenPrefixedImport(const std::string &full_file_name,
478                                      const std::string &base_file_name) {
479   return "import * as "+ GenFileNamespacePrefix(full_file_name) +
480          " from \"./" + base_file_name + "\";\n";
481 }
482 
483 // Adds a source-dependent prefix, for of import * statements.
GenPrefixedTypeName(const std::string & typeName,const std::string & file)484 std::string GenPrefixedTypeName(const std::string &typeName,
485                                 const std::string &file) {
486   const auto basename =
487       flatbuffers::StripPath(flatbuffers::StripExtension(file));
488   if (basename == file_name_) {
489     return typeName;
490   }
491   return GenFileNamespacePrefix(file) + "." + typeName;
492 }
493 
GenStructArgs(const StructDef & struct_def,std::string * annotations,std::string * arguments,const std::string & nameprefix)494 void GenStructArgs(const StructDef &struct_def,
495                           std::string *annotations,
496                           std::string *arguments,
497                           const std::string &nameprefix) {
498   for (auto it = struct_def.fields.vec.begin();
499        it != struct_def.fields.vec.end(); ++it) {
500     auto &field = **it;
501     if (IsStruct(field.value.type)) {
502       // Generate arguments for a struct inside a struct. To ensure names
503       // don't clash, and to make it obvious these arguments are constructing
504       // a nested struct, prefix the name with the field name.
505       GenStructArgs(*field.value.type.struct_def, annotations, arguments,
506                     nameprefix + field.name + "_");
507     } else {
508       *annotations += "@param {" + GenTypeName(field.value.type, true);
509       *annotations += "} " + nameprefix + field.name + "\n";
510 
511       if (lang_.language == IDLOptions::kTs) {
512           *arguments += ", " + nameprefix + field.name + ": " +
513                         GenTypeName(field.value.type, true);
514       } else {
515           *arguments += ", " + nameprefix + field.name;
516       }
517     }
518   }
519 }
520 
GenStructBody(const StructDef & struct_def,std::string * body,const std::string & nameprefix)521 static void GenStructBody(const StructDef &struct_def,
522                           std::string *body,
523                           const std::string &nameprefix) {
524   *body += "  builder.prep(";
525   *body += NumToString(struct_def.minalign) + ", ";
526   *body += NumToString(struct_def.bytesize) + ");\n";
527 
528   for (auto it = struct_def.fields.vec.rbegin();
529        it != struct_def.fields.vec.rend(); ++it) {
530     auto &field = **it;
531     if (field.padding) {
532       *body += "  builder.pad(" + NumToString(field.padding) + ");\n";
533     }
534     if (IsStruct(field.value.type)) {
535       // Generate arguments for a struct inside a struct. To ensure names
536       // don't clash, and to make it obvious these arguments are constructing
537       // a nested struct, prefix the name with the field name.
538       GenStructBody(*field.value.type.struct_def, body,
539                     nameprefix + field.name + "_");
540     } else {
541       *body += "  builder.write" + GenWriteMethod(field.value.type) + "(";
542       if (field.value.type.base_type == BASE_TYPE_BOOL) {
543         *body += "+";
544       }
545       *body += nameprefix + field.name + ");\n";
546     }
547   }
548 }
549 
550 // Generate an accessor struct with constructor for a flatbuffers struct.
GenStruct(const Parser & parser,StructDef & struct_def,std::string * code_ptr,std::string * exports_ptr,imported_fileset & imported_files)551 void GenStruct(const Parser &parser, StructDef &struct_def,
552                std::string *code_ptr, std::string *exports_ptr,
553                imported_fileset &imported_files) {
554   if (struct_def.generated) return;
555   std::string &code = *code_ptr;
556   std::string &exports = *exports_ptr;
557 
558   std::string object_name;
559   std::string object_namespace = GetNameSpace(struct_def);
560 
561   // Emit constructor
562   if (lang_.language == IDLOptions::kTs) {
563     object_name = struct_def.name;
564     GenDocComment(struct_def.doc_comment, code_ptr, "@constructor");
565     if (!object_namespace.empty()) {
566       code += "export namespace " + object_namespace + "{\n";
567     }
568     code += "export class " + struct_def.name;
569     code += " {\n";
570     code += "  /**\n";
571     code += "   * @type {flatbuffers.ByteBuffer}\n";
572     code += "   */\n";
573     code += "  bb: flatbuffers.ByteBuffer;\n";
574     code += "\n";
575     code += "  /**\n";
576     code += "   * @type {number}\n";
577     code += "   */\n";
578     code += "  bb_pos:number = 0;\n";
579   } else {
580     bool isStatement = struct_def.defined_namespace->components.empty();
581     object_name = WrapInNameSpace(struct_def);
582     GenDocComment(struct_def.doc_comment, code_ptr, "@constructor");
583     if (isStatement) {
584       if(parser_.opts.use_goog_js_export_format) {
585         exports += "goog.exportSymbol('" + struct_def.name + "', " +
586           struct_def.name + ");\n";
587       } else {
588         exports += "this." + struct_def.name + " = " + struct_def.name + ";\n";
589       }
590       code += "function " + object_name;
591     } else {
592       code += object_name + " = function";
593     }
594     code += "() {\n";
595     code += "  /**\n";
596     code += "   * @type {flatbuffers.ByteBuffer}\n";
597     code += "   */\n";
598     code += "  this.bb = null;\n";
599     code += "\n";
600     code += "  /**\n";
601     code += "   * @type {number}\n";
602     code += "   */\n";
603     code += "  this.bb_pos = 0;\n";
604     code += isStatement ? "}\n\n" : "};\n\n";
605   }
606 
607   // Generate the __init method that sets the field in a pre-existing
608   // accessor object. This is to allow object reuse.
609   code += "/**\n";
610   code += " * @param {number} i\n";
611   code += " * @param {flatbuffers.ByteBuffer} bb\n";
612   code += " * @returns {" + object_name + "}\n";
613   code += " */\n";
614 
615   if (lang_.language == IDLOptions::kTs) {
616     code += "__init(i:number, bb:flatbuffers.ByteBuffer):" + object_name +
617             " {\n";
618   } else {
619     code += object_name + ".prototype.__init = function(i, bb) {\n";
620   }
621 
622   code += "  this.bb_pos = i;\n";
623   code += "  this.bb = bb;\n";
624   code += "  return this;\n";
625   code += "};\n\n";
626 
627   // Generate a special accessor for the table that when used as the root of a
628   // FlatBuffer
629   if (!struct_def.fixed) {
630     GenDocComment(code_ptr,
631       "@param {flatbuffers.ByteBuffer} bb\n"
632       "@param {" + object_name + "=} obj\n"
633       "@returns {" + object_name + "}");
634     if (lang_.language == IDLOptions::kTs) {
635       code += "static getRootAs" + struct_def.name;
636       code += "(bb:flatbuffers.ByteBuffer, obj?:" + object_name + "):" +
637               object_name + " {\n";
638     } else {
639       code += object_name + ".getRootAs" + struct_def.name;
640       code += " = function(bb, obj) {\n";
641     }
642     code += "  return (obj || new " + object_name;
643     code += ").__init(bb.readInt32(bb.position()) + bb.position(), bb);\n";
644     code += "};\n\n";
645 
646     // Generate the identifier check method
647     if (parser_.root_struct_def_ == &struct_def &&
648         !parser_.file_identifier_.empty()) {
649       GenDocComment(code_ptr,
650         "@param {flatbuffers.ByteBuffer} bb\n"
651         "@returns {boolean}");
652       if (lang_.language == IDLOptions::kTs) {
653         code +=
654             "static bufferHasIdentifier(bb:flatbuffers.ByteBuffer):boolean {\n";
655       } else {
656         code += object_name + ".bufferHasIdentifier = function(bb) {\n";
657       }
658 
659       code += "  return bb.__has_identifier('" + parser_.file_identifier_;
660       code += "');\n};\n\n";
661     }
662   }
663 
664   // Emit field accessors
665   for (auto it = struct_def.fields.vec.begin();
666        it != struct_def.fields.vec.end(); ++it) {
667     auto &field = **it;
668     if (field.deprecated) continue;
669     auto offset_prefix = "  var offset = this.bb.__offset(this.bb_pos, " +
670       NumToString(field.value.offset) + ");\n  return offset ? ";
671 
672     // Emit a scalar field
673     if (IsScalar(field.value.type.base_type) ||
674         field.value.type.base_type == BASE_TYPE_STRING) {
675       GenDocComment(field.doc_comment, code_ptr,
676         std::string(field.value.type.base_type == BASE_TYPE_STRING ?
677           "@param {flatbuffers.Encoding=} optionalEncoding\n" : "") +
678         "@returns {" + GenTypeName(field.value.type, false, true) + "}");
679       if (lang_.language == IDLOptions::kTs) {
680         std::string prefix = MakeCamel(field.name, false) + "(";
681         if (field.value.type.base_type == BASE_TYPE_STRING) {
682           code += prefix + "):string|null\n";
683           code += prefix + "optionalEncoding:flatbuffers.Encoding"+"):" +
684                   GenTypeName(field.value.type, false, true)+"\n";
685           code += prefix + "optionalEncoding?:any";
686         } else {
687           code += prefix;
688         }
689         if (field.value.type.enum_def) {
690           code += "):" +
691                   GenPrefixedTypeName(GenTypeName(field.value.type, false, true),
692                                       field.value.type.enum_def->file) + " {\n";
693         } else {
694           code += "):" + GenTypeName(field.value.type, false, true) + " {\n";
695         }
696       } else {
697         code += object_name + ".prototype." + MakeCamel(field.name, false);
698         code += " = function(";
699         if (field.value.type.base_type == BASE_TYPE_STRING) {
700           code += "optionalEncoding";
701         }
702         code += ") {\n";
703       }
704 
705       if (struct_def.fixed) {
706         code += "  return " + GenGetter(field.value.type, "(this.bb_pos" +
707           MaybeAdd(field.value.offset) + ")") + ";\n";
708       } else {
709         std::string index = "this.bb_pos + offset";
710         if (field.value.type.base_type == BASE_TYPE_STRING) {
711           index += ", optionalEncoding";
712         }
713         code += offset_prefix + GenGetter(field.value.type,
714           "(" + index + ")") + " : " + GenDefaultValue(field.value, "this.bb");
715         code += ";\n";
716       }
717     }
718 
719     // Emit an object field
720     else {
721       switch (field.value.type.base_type) {
722         case BASE_TYPE_STRUCT: {
723           auto type = WrapInNameSpace(*field.value.type.struct_def);
724           GenDocComment(field.doc_comment, code_ptr,
725             "@param {" + type + "=} obj\n@returns {" + type + "|null}");
726           if (lang_.language == IDLOptions::kTs) {
727             type = GenPrefixedTypeName(type, field.value.type.struct_def->file);
728             code += MakeCamel(field.name, false);
729             code += "(obj?:" + type + "):" + type + "|null {\n";
730           } else {
731             code += object_name + ".prototype." + MakeCamel(field.name, false);
732             code += " = function(obj) {\n";
733           }
734 
735           if (struct_def.fixed) {
736             code += "  return (obj || new " + type;
737             code += ").__init(this.bb_pos";
738             code += MaybeAdd(field.value.offset) + ", this.bb);\n";
739           } else {
740             code += offset_prefix + "(obj || new " + type + ").__init(";
741             code += field.value.type.struct_def->fixed
742               ? "this.bb_pos + offset"
743               : "this.bb.__indirect(this.bb_pos + offset)";
744             code += ", this.bb) : null;\n";
745           }
746 
747           if (lang_.language == IDLOptions::kTs) {
748             imported_files.insert(field.value.type.struct_def->file);
749           }
750 
751           break;
752         }
753 
754         case BASE_TYPE_VECTOR: {
755           auto vectortype = field.value.type.VectorType();
756           auto vectortypename = GenTypeName(vectortype, false);
757           auto inline_size = InlineSize(vectortype);
758           auto index = "this.bb.__vector(this.bb_pos + offset) + index" +
759                        MaybeScale(inline_size);
760           std::string args = "@param {number} index\n";
761           std::string ret_type;
762           bool is_union = false;
763           switch (vectortype.base_type) {
764             case BASE_TYPE_STRUCT:
765               args += "@param {" + vectortypename + "=} obj\n";
766               ret_type = vectortypename;
767               break;
768             case BASE_TYPE_STRING:
769               args += "@param {flatbuffers.Encoding=} optionalEncoding\n";
770               ret_type = vectortypename;
771               break;
772             case BASE_TYPE_UNION:
773               args += "@param {flatbuffers.Table=} obj\n";
774               ret_type = "?flatbuffers.Table";
775               is_union = true;
776               break;
777             default:
778               ret_type = vectortypename;
779           }
780           GenDocComment(field.doc_comment, code_ptr, args +
781             "@returns {" + ret_type + "}");
782           if (lang_.language == IDLOptions::kTs) {
783             std::string prefix = MakeCamel(field.name, false);
784             if (is_union) {
785               prefix += "<T extends flatbuffers.Table>";
786             }
787             prefix += "(index: number";
788             if (is_union) {
789               vectortypename = "T";
790               code += prefix + ", obj:T";
791             } else if (vectortype.base_type == BASE_TYPE_STRUCT) {
792               vectortypename = GenPrefixedTypeName(vectortypename,
793                                                    vectortype.struct_def->file);
794               code += prefix + ", obj?:" + vectortypename;
795               imported_files.insert(vectortype.struct_def->file);
796             } else if (vectortype.base_type == BASE_TYPE_STRING) {
797               code += prefix + "):string\n";
798               code += prefix + ",optionalEncoding:flatbuffers.Encoding" + "):" +
799                       vectortypename + "\n";
800               code += prefix + ",optionalEncoding?:any";
801             } else {
802               code += prefix;
803             }
804             code += "):" + vectortypename + "|null {\n";
805           } else {
806             code += object_name + ".prototype." + MakeCamel(field.name, false);
807             code += " = function(index";
808             if (vectortype.base_type == BASE_TYPE_STRUCT || is_union) {
809               code += ", obj";
810             } else if (vectortype.base_type == BASE_TYPE_STRING) {
811               code += ", optionalEncoding";
812             }
813             code += ") {\n";
814           }
815 
816           if (vectortype.base_type == BASE_TYPE_STRUCT) {
817             code += offset_prefix + "(obj || new " + vectortypename;
818             code += ").__init(";
819             code += vectortype.struct_def->fixed
820               ? index
821               : "this.bb.__indirect(" + index + ")";
822             code += ", this.bb)";
823           } else {
824             if (is_union) {
825               index = "obj, " + index;
826             } else if (vectortype.base_type == BASE_TYPE_STRING) {
827               index += ", optionalEncoding";
828             }
829             code += offset_prefix + GenGetter(vectortype, "(" + index + ")");
830           }
831           code += " : ";
832           if (field.value.type.element == BASE_TYPE_BOOL) {
833             code += "false";
834           } else if (field.value.type.element == BASE_TYPE_LONG ||
835               field.value.type.element == BASE_TYPE_ULONG) {
836             code += "this.bb.createLong(0, 0)";
837           } else if (IsScalar(field.value.type.element)) {
838             if (field.value.type.enum_def) {
839               code += "/** @type {" +
840                 WrapInNameSpace(*field.value.type.enum_def) + "} */ (" +
841                 field.value.constant + ")";
842             } else {
843               code += "0";
844             }
845           } else {
846             code += "null";
847           }
848           code += ";\n";
849           break;
850         }
851 
852         case BASE_TYPE_UNION:
853           GenDocComment(field.doc_comment, code_ptr,
854             "@param {flatbuffers.Table} obj\n"
855             "@returns {?flatbuffers.Table}");
856           if (lang_.language == IDLOptions::kTs) {
857             code += MakeCamel(field.name, false);
858             code += "<T extends flatbuffers.Table>(obj:T):T|null {\n";
859           } else {
860             code += object_name + ".prototype." + MakeCamel(field.name, false);
861             code += " = function(obj) {\n";
862           }
863 
864           code += offset_prefix + GenGetter(field.value.type,
865             "(obj, this.bb_pos + offset)") + " : null;\n";
866           break;
867 
868         default:
869           assert(0);
870       }
871     }
872     code += "};\n\n";
873 
874     if(parser_.opts.use_goog_js_export_format) {
875       exports += "goog.exportProperty(" + object_name + ".prototype, '" +
876         MakeCamel(field.name, false) + "', " + object_name + ".prototype." +
877         MakeCamel(field.name, false) + ");\n";
878     }
879 
880     // Adds the mutable scalar value to the output
881     if (IsScalar(field.value.type.base_type) && parser.opts.mutable_buffer) {
882       std::string annotations =
883           "@param {" + GenTypeName(field.value.type, true) + "} value\n";
884       GenDocComment(code_ptr, annotations +
885         "@returns {boolean}");
886 
887       if (lang_.language == IDLOptions::kTs) {
888         std::string type;
889         if (field.value.type.enum_def) {
890           type = GenPrefixedTypeName(GenTypeName(field.value.type, true),
891                                      field.value.type.enum_def->file);
892         } else {
893           type = GenTypeName(field.value.type, true);
894         }
895 
896         code += "mutate_" + field.name + "(value:" + type + "):boolean {\n";
897       } else {
898         code += object_name + ".prototype.mutate_" + field.name +
899                 " = function(value) {\n";
900       }
901 
902       code += "  var offset = this.bb.__offset(this.bb_pos, " +
903               NumToString(field.value.offset) + ");\n\n";
904       code += "  if (offset === 0) {\n";
905       code += "    return false;\n";
906       code += "  }\n\n";
907 
908       // special case for bools, which are treated as uint8
909       code += "  this.bb.write" + MakeCamel(GenType(field.value.type)) +
910               "(this.bb_pos + offset, ";
911       if (field.value.type.base_type == BASE_TYPE_BOOL &&
912           lang_.language == IDLOptions::kTs) {
913           code += "+";
914       }
915 
916       code += "value);\n";
917       code += "  return true;\n";
918       code += "};\n\n";
919 
920       if(parser_.opts.use_goog_js_export_format) {
921         exports += "goog.exportProperty(" + object_name +
922           ".prototype, 'mutate_" + field.name + "', " + object_name +
923           ".prototype.mutate_" + field.name + ");\n";
924       }
925     }
926 
927     // Emit vector helpers
928     if (field.value.type.base_type == BASE_TYPE_VECTOR) {
929       // Emit a length helper
930       GenDocComment(code_ptr, "@returns {number}");
931       if (lang_.language == IDLOptions::kTs) {
932         code += MakeCamel(field.name, false);
933         code += "Length():number {\n" + offset_prefix;
934       } else {
935         code += object_name + ".prototype." + MakeCamel(field.name, false);
936         code += "Length = function() {\n" + offset_prefix;
937       }
938 
939       code += "this.bb.__vector_len(this.bb_pos + offset) : 0;\n};\n\n";
940 
941       if(parser_.opts.use_goog_js_export_format) {
942         exports += "goog.exportProperty(" + object_name + ".prototype, '" +
943           MakeCamel(field.name, false) + "Length', " + object_name +
944           ".prototype." + MakeCamel(field.name, false) + "Length);\n";
945       }
946 
947       // For scalar types, emit a typed array helper
948       auto vectorType = field.value.type.VectorType();
949       if (IsScalar(vectorType.base_type) && !IsLong(vectorType.base_type)) {
950         GenDocComment(code_ptr, "@returns {" + GenType(vectorType) + "Array}");
951 
952         if (lang_.language == IDLOptions::kTs) {
953           code += MakeCamel(field.name, false);
954           code += "Array():" + GenType(vectorType) + "Array|null {\n" +
955                   offset_prefix;
956         } else {
957           code += object_name + ".prototype." + MakeCamel(field.name, false);
958           code += "Array = function() {\n" + offset_prefix;
959         }
960 
961         code += "new " + GenType(vectorType) + "Array(this.bb.bytes().buffer, "
962           "this.bb.bytes().byteOffset + this.bb.__vector(this.bb_pos + offset), "
963           "this.bb.__vector_len(this.bb_pos + offset)) : null;\n};\n\n";
964 
965         if(parser_.opts.use_goog_js_export_format) {
966           exports += "goog.exportProperty(" + object_name + ".prototype, '" +
967             MakeCamel(field.name, false) + "Array', " + object_name +
968             ".prototype." + MakeCamel(field.name, false) + "Array);\n";
969         }
970       }
971     }
972   }
973 
974   // Emit a factory constructor
975   if (struct_def.fixed) {
976     std::string annotations = "@param {flatbuffers.Builder} builder\n";
977     std::string arguments;
978     GenStructArgs(struct_def, &annotations, &arguments, "");
979     GenDocComment(code_ptr, annotations +
980       "@returns {flatbuffers.Offset}");
981 
982     if (lang_.language == IDLOptions::kTs) {
983       code += "static create" + struct_def.name + "(builder:flatbuffers.Builder";
984       code += arguments + "):flatbuffers.Offset {\n";
985     } else {
986       code += object_name + ".create" + struct_def.name + " = function(builder";
987       code += arguments + ") {\n";
988     }
989 
990     GenStructBody(struct_def, &code, "");
991     code += "  return builder.offset();\n};\n\n";
992   } else {
993     // Generate a method to start building a new object
994     GenDocComment(code_ptr,
995       "@param {flatbuffers.Builder} builder");
996 
997     if (lang_.language == IDLOptions::kTs) {
998       code += "static start" + struct_def.name;
999       code += "(builder:flatbuffers.Builder) {\n";
1000     } else {
1001       code += object_name + ".start" + struct_def.name;
1002       code += " = function(builder) {\n";
1003     }
1004 
1005     code += "  builder.startObject(" + NumToString(
1006       struct_def.fields.vec.size()) + ");\n";
1007     code += "};\n\n";
1008 
1009     // Generate a set of static methods that allow table construction
1010     for (auto it = struct_def.fields.vec.begin();
1011          it != struct_def.fields.vec.end(); ++it) {
1012       auto &field = **it;
1013       if (field.deprecated) continue;
1014       auto argname = MakeCamel(field.name, false);
1015       if (!IsScalar(field.value.type.base_type)) {
1016         argname += "Offset";
1017       }
1018 
1019       // Generate the field insertion method
1020       GenDocComment(code_ptr,
1021         "@param {flatbuffers.Builder} builder\n"
1022         "@param {" + GenTypeName(field.value.type, true) + "} " +
1023         argname);
1024 
1025       if (lang_.language == IDLOptions::kTs) {
1026         std::string argType;
1027         if (field.value.type.enum_def) {
1028           argType = GenPrefixedTypeName(GenTypeName(field.value.type, true),
1029                                         field.value.type.enum_def->file);
1030         } else {
1031           argType = GenTypeName(field.value.type, true);
1032         }
1033 
1034         code += "static add" + MakeCamel(field.name);
1035         code += "(builder:flatbuffers.Builder, " + argname + ":" + argType +
1036                 ") {\n";
1037       } else {
1038         code += object_name + ".add" + MakeCamel(field.name);
1039         code += " = function(builder, " + argname + ") {\n";
1040       }
1041 
1042       code += "  builder.addField" + GenWriteMethod(field.value.type) + "(";
1043       code += NumToString(it - struct_def.fields.vec.begin()) + ", ";
1044       if (field.value.type.base_type == BASE_TYPE_BOOL) {
1045         code += "+";
1046       }
1047       code += argname + ", ";
1048       if (!IsScalar(field.value.type.base_type)) {
1049         code += "0";
1050       } else {
1051         if (field.value.type.base_type == BASE_TYPE_BOOL) {
1052           code += "+";
1053         }
1054         code += GenDefaultValue(field.value, "builder");
1055       }
1056       code += ");\n};\n\n";
1057 
1058       if (field.value.type.base_type == BASE_TYPE_VECTOR) {
1059         auto vector_type = field.value.type.VectorType();
1060         auto alignment = InlineAlignment(vector_type);
1061         auto elem_size = InlineSize(vector_type);
1062 
1063         // Generate a method to create a vector from a JavaScript array
1064         if (!IsStruct(vector_type)) {
1065           GenDocComment(code_ptr,
1066             "@param {flatbuffers.Builder} builder\n"
1067             "@param {Array.<" + GenTypeName(vector_type, true) +
1068             ">} data\n"
1069             "@returns {flatbuffers.Offset}");
1070 
1071           if (lang_.language == IDLOptions::kTs) {
1072             code += "static create" + MakeCamel(field.name);
1073             std::string type = GenTypeName(vector_type, true) + "[]";
1074             if (type == "number[]") {
1075               type += " | Uint8Array";
1076             }
1077             code += "Vector(builder:flatbuffers.Builder, data:" + type +
1078                     "):flatbuffers.Offset {\n";
1079           } else {
1080             code += object_name + ".create" + MakeCamel(field.name);
1081             code += "Vector = function(builder, data) {\n";
1082           }
1083 
1084           code += "  builder.startVector(" + NumToString(elem_size);
1085           code += ", data.length, " + NumToString(alignment) + ");\n";
1086           code += "  for (var i = data.length - 1; i >= 0; i--) {\n";
1087           code += "    builder.add" + GenWriteMethod(vector_type) + "(";
1088           if (vector_type.base_type == BASE_TYPE_BOOL) {
1089             code += "+";
1090           }
1091           code += "data[i]);\n";
1092           code += "  }\n";
1093           code += "  return builder.endVector();\n";
1094           code += "};\n\n";
1095         }
1096 
1097         // Generate a method to start a vector, data to be added manually after
1098         GenDocComment(code_ptr,
1099           "@param {flatbuffers.Builder} builder\n"
1100           "@param {number} numElems");
1101 
1102         if (lang_.language == IDLOptions::kTs) {
1103           code += "static start" + MakeCamel(field.name);
1104           code += "Vector(builder:flatbuffers.Builder, numElems:number) {\n";
1105         } else {
1106           code += object_name + ".start" + MakeCamel(field.name);
1107           code += "Vector = function(builder, numElems) {\n";
1108         }
1109 
1110         code += "  builder.startVector(" + NumToString(elem_size);
1111         code += ", numElems, " + NumToString(alignment) + ");\n";
1112         code += "};\n\n";
1113       }
1114     }
1115 
1116     // Generate a method to stop building a new object
1117     GenDocComment(code_ptr,
1118       "@param {flatbuffers.Builder} builder\n"
1119       "@returns {flatbuffers.Offset}");
1120 
1121     if (lang_.language == IDLOptions::kTs) {
1122       code += "static end" + struct_def.name;
1123       code += "(builder:flatbuffers.Builder):flatbuffers.Offset {\n";
1124     } else {
1125       code += object_name + ".end" + struct_def.name;
1126       code += " = function(builder) {\n";
1127     }
1128 
1129     code += "  var offset = builder.endObject();\n";
1130     for (auto it = struct_def.fields.vec.begin();
1131          it != struct_def.fields.vec.end(); ++it) {
1132       auto &field = **it;
1133       if (!field.deprecated && field.required) {
1134         code += "  builder.requiredField(offset, ";
1135         code += NumToString(field.value.offset);
1136         code += "); // " + field.name + "\n";
1137       }
1138     }
1139     code += "  return offset;\n";
1140     code += "};\n\n";
1141 
1142     // Generate the method to complete buffer construction
1143     if (parser_.root_struct_def_ == &struct_def) {
1144       GenDocComment(code_ptr,
1145         "@param {flatbuffers.Builder} builder\n"
1146         "@param {flatbuffers.Offset} offset");
1147 
1148       if (lang_.language == IDLOptions::kTs) {
1149         code += "static finish" + struct_def.name + "Buffer";
1150         code += "(builder:flatbuffers.Builder, offset:flatbuffers.Offset) {\n";
1151       } else {
1152         code += object_name + ".finish" + struct_def.name + "Buffer";
1153         code += " = function(builder, offset) {\n";
1154       }
1155 
1156       code += "  builder.finish(offset";
1157       if (!parser_.file_identifier_.empty()) {
1158         code += ", '" + parser_.file_identifier_ + "'";
1159       }
1160       code += ");\n";
1161       code += "};\n\n";
1162     }
1163   }
1164 
1165   if (lang_.language == IDLOptions::kTs) {
1166     if (!object_namespace.empty()) {
1167       code += "}\n";
1168     }
1169     code += "}\n";
1170   }
1171 }
1172 };
1173 }  // namespace js
1174 
GenerateJS(const Parser & parser,const std::string & path,const std::string & file_name)1175 bool GenerateJS(const Parser &parser, const std::string &path,
1176                 const std::string &file_name) {
1177   js::JsGenerator generator(parser, path, file_name);
1178   return generator.generate();
1179 }
1180 
JSMakeRule(const Parser & parser,const std::string & path,const std::string & file_name)1181 std::string JSMakeRule(const Parser &parser,
1182                        const std::string &path,
1183                        const std::string &file_name) {
1184   assert(parser.opts.lang <= IDLOptions::kMAX);
1185   const auto &lang = GetJsLangParams(parser.opts.lang);
1186 
1187   std::string filebase = flatbuffers::StripPath(
1188       flatbuffers::StripExtension(file_name));
1189   std::string make_rule = GeneratedFileName(path, filebase, lang) + ": ";
1190 
1191   auto included_files = parser.GetIncludedFilesRecursive(file_name);
1192   for (auto it = included_files.begin();
1193        it != included_files.end(); ++it) {
1194     make_rule += " " + *it;
1195   }
1196 return make_rule;
1197 }
1198 
1199 }  // namespace flatbuffers
1200