No Description
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

root.js 12KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363
  1. "use strict";
  2. module.exports = Root;
  3. // extends Namespace
  4. var Namespace = require("./namespace");
  5. ((Root.prototype = Object.create(Namespace.prototype)).constructor = Root).className = "Root";
  6. var Field = require("./field"),
  7. Enum = require("./enum"),
  8. OneOf = require("./oneof"),
  9. util = require("./util");
  10. var Type, // cyclic
  11. parse, // might be excluded
  12. common; // "
  13. /**
  14. * Constructs a new root namespace instance.
  15. * @classdesc Root namespace wrapping all types, enums, services, sub-namespaces etc. that belong together.
  16. * @extends NamespaceBase
  17. * @constructor
  18. * @param {Object.<string,*>} [options] Top level options
  19. */
  20. function Root(options) {
  21. Namespace.call(this, "", options);
  22. /**
  23. * Deferred extension fields.
  24. * @type {Field[]}
  25. */
  26. this.deferred = [];
  27. /**
  28. * Resolved file names of loaded files.
  29. * @type {string[]}
  30. */
  31. this.files = [];
  32. }
  33. /**
  34. * Loads a namespace descriptor into a root namespace.
  35. * @param {INamespace} json Nameespace descriptor
  36. * @param {Root} [root] Root namespace, defaults to create a new one if omitted
  37. * @returns {Root} Root namespace
  38. */
  39. Root.fromJSON = function fromJSON(json, root) {
  40. if (!root)
  41. root = new Root();
  42. if (json.options)
  43. root.setOptions(json.options);
  44. return root.addJSON(json.nested);
  45. };
  46. /**
  47. * Resolves the path of an imported file, relative to the importing origin.
  48. * This method exists so you can override it with your own logic in case your imports are scattered over multiple directories.
  49. * @function
  50. * @param {string} origin The file name of the importing file
  51. * @param {string} target The file name being imported
  52. * @returns {string|null} Resolved path to `target` or `null` to skip the file
  53. */
  54. Root.prototype.resolvePath = util.path.resolve;
  55. /**
  56. * Fetch content from file path or url
  57. * This method exists so you can override it with your own logic.
  58. * @function
  59. * @param {string} path File path or url
  60. * @param {FetchCallback} callback Callback function
  61. * @returns {undefined}
  62. */
  63. Root.prototype.fetch = util.fetch;
  64. // A symbol-like function to safely signal synchronous loading
  65. /* istanbul ignore next */
  66. function SYNC() {} // eslint-disable-line no-empty-function
  67. /**
  68. * Loads one or multiple .proto or preprocessed .json files into this root namespace and calls the callback.
  69. * @param {string|string[]} filename Names of one or multiple files to load
  70. * @param {IParseOptions} options Parse options
  71. * @param {LoadCallback} callback Callback function
  72. * @returns {undefined}
  73. */
  74. Root.prototype.load = function load(filename, options, callback) {
  75. if (typeof options === "function") {
  76. callback = options;
  77. options = undefined;
  78. }
  79. var self = this;
  80. if (!callback)
  81. return util.asPromise(load, self, filename, options);
  82. var sync = callback === SYNC; // undocumented
  83. // Finishes loading by calling the callback (exactly once)
  84. function finish(err, root) {
  85. /* istanbul ignore if */
  86. if (!callback)
  87. return;
  88. var cb = callback;
  89. callback = null;
  90. if (sync)
  91. throw err;
  92. cb(err, root);
  93. }
  94. // Bundled definition existence checking
  95. function getBundledFileName(filename) {
  96. var idx = filename.lastIndexOf("google/protobuf/");
  97. if (idx > -1) {
  98. var altname = filename.substring(idx);
  99. if (altname in common) return altname;
  100. }
  101. return null;
  102. }
  103. // Processes a single file
  104. function process(filename, source) {
  105. try {
  106. if (util.isString(source) && source.charAt(0) === "{")
  107. source = JSON.parse(source);
  108. if (!util.isString(source))
  109. self.setOptions(source.options).addJSON(source.nested);
  110. else {
  111. parse.filename = filename;
  112. var parsed = parse(source, self, options),
  113. resolved,
  114. i = 0;
  115. if (parsed.imports)
  116. for (; i < parsed.imports.length; ++i)
  117. if (resolved = getBundledFileName(parsed.imports[i]) || self.resolvePath(filename, parsed.imports[i]))
  118. fetch(resolved);
  119. if (parsed.weakImports)
  120. for (i = 0; i < parsed.weakImports.length; ++i)
  121. if (resolved = getBundledFileName(parsed.weakImports[i]) || self.resolvePath(filename, parsed.weakImports[i]))
  122. fetch(resolved, true);
  123. }
  124. } catch (err) {
  125. finish(err);
  126. }
  127. if (!sync && !queued)
  128. finish(null, self); // only once anyway
  129. }
  130. // Fetches a single file
  131. function fetch(filename, weak) {
  132. // Skip if already loaded / attempted
  133. if (self.files.indexOf(filename) > -1)
  134. return;
  135. self.files.push(filename);
  136. // Shortcut bundled definitions
  137. if (filename in common) {
  138. if (sync)
  139. process(filename, common[filename]);
  140. else {
  141. ++queued;
  142. setTimeout(function() {
  143. --queued;
  144. process(filename, common[filename]);
  145. });
  146. }
  147. return;
  148. }
  149. // Otherwise fetch from disk or network
  150. if (sync) {
  151. var source;
  152. try {
  153. source = util.fs.readFileSync(filename).toString("utf8");
  154. } catch (err) {
  155. if (!weak)
  156. finish(err);
  157. return;
  158. }
  159. process(filename, source);
  160. } else {
  161. ++queued;
  162. self.fetch(filename, function(err, source) {
  163. --queued;
  164. /* istanbul ignore if */
  165. if (!callback)
  166. return; // terminated meanwhile
  167. if (err) {
  168. /* istanbul ignore else */
  169. if (!weak)
  170. finish(err);
  171. else if (!queued) // can't be covered reliably
  172. finish(null, self);
  173. return;
  174. }
  175. process(filename, source);
  176. });
  177. }
  178. }
  179. var queued = 0;
  180. // Assembling the root namespace doesn't require working type
  181. // references anymore, so we can load everything in parallel
  182. if (util.isString(filename))
  183. filename = [ filename ];
  184. for (var i = 0, resolved; i < filename.length; ++i)
  185. if (resolved = self.resolvePath("", filename[i]))
  186. fetch(resolved);
  187. if (sync)
  188. return self;
  189. if (!queued)
  190. finish(null, self);
  191. return undefined;
  192. };
  193. // function load(filename:string, options:IParseOptions, callback:LoadCallback):undefined
  194. /**
  195. * Loads one or multiple .proto or preprocessed .json files into this root namespace and calls the callback.
  196. * @function Root#load
  197. * @param {string|string[]} filename Names of one or multiple files to load
  198. * @param {LoadCallback} callback Callback function
  199. * @returns {undefined}
  200. * @variation 2
  201. */
  202. // function load(filename:string, callback:LoadCallback):undefined
  203. /**
  204. * Loads one or multiple .proto or preprocessed .json files into this root namespace and returns a promise.
  205. * @function Root#load
  206. * @param {string|string[]} filename Names of one or multiple files to load
  207. * @param {IParseOptions} [options] Parse options. Defaults to {@link parse.defaults} when omitted.
  208. * @returns {Promise<Root>} Promise
  209. * @variation 3
  210. */
  211. // function load(filename:string, [options:IParseOptions]):Promise<Root>
  212. /**
  213. * Synchronously loads one or multiple .proto or preprocessed .json files into this root namespace (node only).
  214. * @function Root#loadSync
  215. * @param {string|string[]} filename Names of one or multiple files to load
  216. * @param {IParseOptions} [options] Parse options. Defaults to {@link parse.defaults} when omitted.
  217. * @returns {Root} Root namespace
  218. * @throws {Error} If synchronous fetching is not supported (i.e. in browsers) or if a file's syntax is invalid
  219. */
  220. Root.prototype.loadSync = function loadSync(filename, options) {
  221. if (!util.isNode)
  222. throw Error("not supported");
  223. return this.load(filename, options, SYNC);
  224. };
  225. /**
  226. * @override
  227. */
  228. Root.prototype.resolveAll = function resolveAll() {
  229. if (this.deferred.length)
  230. throw Error("unresolvable extensions: " + this.deferred.map(function(field) {
  231. return "'extend " + field.extend + "' in " + field.parent.fullName;
  232. }).join(", "));
  233. return Namespace.prototype.resolveAll.call(this);
  234. };
  235. // only uppercased (and thus conflict-free) children are exposed, see below
  236. var exposeRe = /^[A-Z]/;
  237. /**
  238. * Handles a deferred declaring extension field by creating a sister field to represent it within its extended type.
  239. * @param {Root} root Root instance
  240. * @param {Field} field Declaring extension field witin the declaring type
  241. * @returns {boolean} `true` if successfully added to the extended type, `false` otherwise
  242. * @inner
  243. * @ignore
  244. */
  245. function tryHandleExtension(root, field) {
  246. var extendedType = field.parent.lookup(field.extend);
  247. if (extendedType) {
  248. var sisterField = new Field(field.fullName, field.id, field.type, field.rule, undefined, field.options);
  249. sisterField.declaringField = field;
  250. field.extensionField = sisterField;
  251. extendedType.add(sisterField);
  252. return true;
  253. }
  254. return false;
  255. }
  256. /**
  257. * Called when any object is added to this root or its sub-namespaces.
  258. * @param {ReflectionObject} object Object added
  259. * @returns {undefined}
  260. * @private
  261. */
  262. Root.prototype._handleAdd = function _handleAdd(object) {
  263. if (object instanceof Field) {
  264. if (/* an extension field (implies not part of a oneof) */ object.extend !== undefined && /* not already handled */ !object.extensionField)
  265. if (!tryHandleExtension(this, object))
  266. this.deferred.push(object);
  267. } else if (object instanceof Enum) {
  268. if (exposeRe.test(object.name))
  269. object.parent[object.name] = object.values; // expose enum values as property of its parent
  270. } else if (!(object instanceof OneOf)) /* everything else is a namespace */ {
  271. if (object instanceof Type) // Try to handle any deferred extensions
  272. for (var i = 0; i < this.deferred.length;)
  273. if (tryHandleExtension(this, this.deferred[i]))
  274. this.deferred.splice(i, 1);
  275. else
  276. ++i;
  277. for (var j = 0; j < /* initializes */ object.nestedArray.length; ++j) // recurse into the namespace
  278. this._handleAdd(object._nestedArray[j]);
  279. if (exposeRe.test(object.name))
  280. object.parent[object.name] = object; // expose namespace as property of its parent
  281. }
  282. // The above also adds uppercased (and thus conflict-free) nested types, services and enums as
  283. // properties of namespaces just like static code does. This allows using a .d.ts generated for
  284. // a static module with reflection-based solutions where the condition is met.
  285. };
  286. /**
  287. * Called when any object is removed from this root or its sub-namespaces.
  288. * @param {ReflectionObject} object Object removed
  289. * @returns {undefined}
  290. * @private
  291. */
  292. Root.prototype._handleRemove = function _handleRemove(object) {
  293. if (object instanceof Field) {
  294. if (/* an extension field */ object.extend !== undefined) {
  295. if (/* already handled */ object.extensionField) { // remove its sister field
  296. object.extensionField.parent.remove(object.extensionField);
  297. object.extensionField = null;
  298. } else { // cancel the extension
  299. var index = this.deferred.indexOf(object);
  300. /* istanbul ignore else */
  301. if (index > -1)
  302. this.deferred.splice(index, 1);
  303. }
  304. }
  305. } else if (object instanceof Enum) {
  306. if (exposeRe.test(object.name))
  307. delete object.parent[object.name]; // unexpose enum values
  308. } else if (object instanceof Namespace) {
  309. for (var i = 0; i < /* initializes */ object.nestedArray.length; ++i) // recurse into the namespace
  310. this._handleRemove(object._nestedArray[i]);
  311. if (exposeRe.test(object.name))
  312. delete object.parent[object.name]; // unexpose namespaces
  313. }
  314. };
  315. // Sets up cyclic dependencies (called in index-light)
  316. Root._configure = function(Type_, parse_, common_) {
  317. Type = Type_;
  318. parse = parse_;
  319. common = common_;
  320. };