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