| "use strict"; |
| module.exports = Field; |
| |
| // extends ReflectionObject |
| var ReflectionObject = require("./object"); |
| ((Field.prototype = Object.create(ReflectionObject.prototype)).constructor = Field).className = "Field"; |
| |
| var Enum = require("./enum"), |
| types = require("./types"), |
| util = require("./util"); |
| |
| var Type; // cyclic |
| |
| var ruleRe = /^required|optional|repeated$/; |
| |
| /** |
| * Constructs a new message field instance. Note that {@link MapField|map fields} have their own class. |
| * @name Field |
| * @classdesc Reflected message field. |
| * @extends FieldBase |
| * @constructor |
| * @param {string} name Unique name within its namespace |
| * @param {number} id Unique id within its namespace |
| * @param {string} type Value type |
| * @param {string|Object.<string,*>} [rule="optional"] Field rule |
| * @param {string|Object.<string,*>} [extend] Extended type if different from parent |
| * @param {Object.<string,*>} [options] Declared options |
| */ |
| |
| /** |
| * Constructs a field from a field descriptor. |
| * @param {string} name Field name |
| * @param {IField} json Field descriptor |
| * @returns {Field} Created field |
| * @throws {TypeError} If arguments are invalid |
| */ |
| Field.fromJSON = function fromJSON(name, json) { |
| return new Field(name, json.id, json.type, json.rule, json.extend, json.options, json.comment); |
| }; |
| |
| /** |
| * Not an actual constructor. Use {@link Field} instead. |
| * @classdesc Base class of all reflected message fields. This is not an actual class but here for the sake of having consistent type definitions. |
| * @exports FieldBase |
| * @extends ReflectionObject |
| * @constructor |
| * @param {string} name Unique name within its namespace |
| * @param {number} id Unique id within its namespace |
| * @param {string} type Value type |
| * @param {string|Object.<string,*>} [rule="optional"] Field rule |
| * @param {string|Object.<string,*>} [extend] Extended type if different from parent |
| * @param {Object.<string,*>} [options] Declared options |
| * @param {string} [comment] Comment associated with this field |
| */ |
| function Field(name, id, type, rule, extend, options, comment) { |
| |
| if (util.isObject(rule)) { |
| comment = extend; |
| options = rule; |
| rule = extend = undefined; |
| } else if (util.isObject(extend)) { |
| comment = options; |
| options = extend; |
| extend = undefined; |
| } |
| |
| ReflectionObject.call(this, name, options); |
| |
| if (!util.isInteger(id) || id < 0) |
| throw TypeError("id must be a non-negative integer"); |
| |
| if (!util.isString(type)) |
| throw TypeError("type must be a string"); |
| |
| if (rule !== undefined && !ruleRe.test(rule = rule.toString().toLowerCase())) |
| throw TypeError("rule must be a string rule"); |
| |
| if (extend !== undefined && !util.isString(extend)) |
| throw TypeError("extend must be a string"); |
| |
| /** |
| * Field rule, if any. |
| * @type {string|undefined} |
| */ |
| if (rule === "proto3_optional") { |
| rule = "optional"; |
| } |
| this.rule = rule && rule !== "optional" ? rule : undefined; // toJSON |
| |
| /** |
| * Field type. |
| * @type {string} |
| */ |
| this.type = type; // toJSON |
| |
| /** |
| * Unique field id. |
| * @type {number} |
| */ |
| this.id = id; // toJSON, marker |
| |
| /** |
| * Extended type if different from parent. |
| * @type {string|undefined} |
| */ |
| this.extend = extend || undefined; // toJSON |
| |
| /** |
| * Whether this field is required. |
| * @type {boolean} |
| */ |
| this.required = rule === "required"; |
| |
| /** |
| * Whether this field is optional. |
| * @type {boolean} |
| */ |
| this.optional = !this.required; |
| |
| /** |
| * Whether this field is repeated. |
| * @type {boolean} |
| */ |
| this.repeated = rule === "repeated"; |
| |
| /** |
| * Whether this field is a map or not. |
| * @type {boolean} |
| */ |
| this.map = false; |
| |
| /** |
| * Message this field belongs to. |
| * @type {Type|null} |
| */ |
| this.message = null; |
| |
| /** |
| * OneOf this field belongs to, if any, |
| * @type {OneOf|null} |
| */ |
| this.partOf = null; |
| |
| /** |
| * The field type's default value. |
| * @type {*} |
| */ |
| this.typeDefault = null; |
| |
| /** |
| * The field's default value on prototypes. |
| * @type {*} |
| */ |
| this.defaultValue = null; |
| |
| /** |
| * Whether this field's value should be treated as a long. |
| * @type {boolean} |
| */ |
| this.long = util.Long ? types.long[type] !== undefined : /* istanbul ignore next */ false; |
| |
| /** |
| * Whether this field's value is a buffer. |
| * @type {boolean} |
| */ |
| this.bytes = type === "bytes"; |
| |
| /** |
| * Resolved type if not a basic type. |
| * @type {Type|Enum|null} |
| */ |
| this.resolvedType = null; |
| |
| /** |
| * Sister-field within the extended type if a declaring extension field. |
| * @type {Field|null} |
| */ |
| this.extensionField = null; |
| |
| /** |
| * Sister-field within the declaring namespace if an extended field. |
| * @type {Field|null} |
| */ |
| this.declaringField = null; |
| |
| /** |
| * Internally remembers whether this field is packed. |
| * @type {boolean|null} |
| * @private |
| */ |
| this._packed = null; |
| |
| /** |
| * Comment for this field. |
| * @type {string|null} |
| */ |
| this.comment = comment; |
| } |
| |
| /** |
| * Determines whether this field is packed. Only relevant when repeated and working with proto2. |
| * @name Field#packed |
| * @type {boolean} |
| * @readonly |
| */ |
| Object.defineProperty(Field.prototype, "packed", { |
| get: function() { |
| // defaults to packed=true if not explicity set to false |
| if (this._packed === null) |
| this._packed = this.getOption("packed") !== false; |
| return this._packed; |
| } |
| }); |
| |
| /** |
| * @override |
| */ |
| Field.prototype.setOption = function setOption(name, value, ifNotSet) { |
| if (name === "packed") // clear cached before setting |
| this._packed = null; |
| return ReflectionObject.prototype.setOption.call(this, name, value, ifNotSet); |
| }; |
| |
| /** |
| * Field descriptor. |
| * @interface IField |
| * @property {string} [rule="optional"] Field rule |
| * @property {string} type Field type |
| * @property {number} id Field id |
| * @property {Object.<string,*>} [options] Field options |
| */ |
| |
| /** |
| * Extension field descriptor. |
| * @interface IExtensionField |
| * @extends IField |
| * @property {string} extend Extended type |
| */ |
| |
| /** |
| * Converts this field to a field descriptor. |
| * @param {IToJSONOptions} [toJSONOptions] JSON conversion options |
| * @returns {IField} Field descriptor |
| */ |
| Field.prototype.toJSON = function toJSON(toJSONOptions) { |
| var keepComments = toJSONOptions ? Boolean(toJSONOptions.keepComments) : false; |
| return util.toObject([ |
| "rule" , this.rule !== "optional" && this.rule || undefined, |
| "type" , this.type, |
| "id" , this.id, |
| "extend" , this.extend, |
| "options" , this.options, |
| "comment" , keepComments ? this.comment : undefined |
| ]); |
| }; |
| |
| /** |
| * Resolves this field's type references. |
| * @returns {Field} `this` |
| * @throws {Error} If any reference cannot be resolved |
| */ |
| Field.prototype.resolve = function resolve() { |
| |
| if (this.resolved) |
| return this; |
| |
| if ((this.typeDefault = types.defaults[this.type]) === undefined) { // if not a basic type, resolve it |
| this.resolvedType = (this.declaringField ? this.declaringField.parent : this.parent).lookupTypeOrEnum(this.type); |
| if (this.resolvedType instanceof Type) |
| this.typeDefault = null; |
| else // instanceof Enum |
| this.typeDefault = this.resolvedType.values[Object.keys(this.resolvedType.values)[0]]; // first defined |
| } else if (this.options && this.options.proto3_optional) { |
| // proto3 scalar value marked optional; should default to null |
| this.typeDefault = null; |
| } |
| |
| // use explicitly set default value if present |
| if (this.options && this.options["default"] != null) { |
| this.typeDefault = this.options["default"]; |
| if (this.resolvedType instanceof Enum && typeof this.typeDefault === "string") |
| this.typeDefault = this.resolvedType.values[this.typeDefault]; |
| } |
| |
| // remove unnecessary options |
| if (this.options) { |
| if (this.options.packed === true || this.options.packed !== undefined && this.resolvedType && !(this.resolvedType instanceof Enum)) |
| delete this.options.packed; |
| if (!Object.keys(this.options).length) |
| this.options = undefined; |
| } |
| |
| // convert to internal data type if necesssary |
| if (this.long) { |
| this.typeDefault = util.Long.fromNumber(this.typeDefault, this.type.charAt(0) === "u"); |
| |
| /* istanbul ignore else */ |
| if (Object.freeze) |
| Object.freeze(this.typeDefault); // long instances are meant to be immutable anyway (i.e. use small int cache that even requires it) |
| |
| } else if (this.bytes && typeof this.typeDefault === "string") { |
| var buf; |
| if (util.base64.test(this.typeDefault)) |
| util.base64.decode(this.typeDefault, buf = util.newBuffer(util.base64.length(this.typeDefault)), 0); |
| else |
| util.utf8.write(this.typeDefault, buf = util.newBuffer(util.utf8.length(this.typeDefault)), 0); |
| this.typeDefault = buf; |
| } |
| |
| // take special care of maps and repeated fields |
| if (this.map) |
| this.defaultValue = util.emptyObject; |
| else if (this.repeated) |
| this.defaultValue = util.emptyArray; |
| else |
| this.defaultValue = this.typeDefault; |
| |
| // ensure proper value on prototype |
| if (this.parent instanceof Type) |
| this.parent.ctor.prototype[this.name] = this.defaultValue; |
| |
| return ReflectionObject.prototype.resolve.call(this); |
| }; |
| |
| /** |
| * Decorator function as returned by {@link Field.d} and {@link MapField.d} (TypeScript). |
| * @typedef FieldDecorator |
| * @type {function} |
| * @param {Object} prototype Target prototype |
| * @param {string} fieldName Field name |
| * @returns {undefined} |
| */ |
| |
| /** |
| * Field decorator (TypeScript). |
| * @name Field.d |
| * @function |
| * @param {number} fieldId Field id |
| * @param {"double"|"float"|"int32"|"uint32"|"sint32"|"fixed32"|"sfixed32"|"int64"|"uint64"|"sint64"|"fixed64"|"sfixed64"|"string"|"bool"|"bytes"|Object} fieldType Field type |
| * @param {"optional"|"required"|"repeated"} [fieldRule="optional"] Field rule |
| * @param {T} [defaultValue] Default value |
| * @returns {FieldDecorator} Decorator function |
| * @template T extends number | number[] | Long | Long[] | string | string[] | boolean | boolean[] | Uint8Array | Uint8Array[] | Buffer | Buffer[] |
| */ |
| Field.d = function decorateField(fieldId, fieldType, fieldRule, defaultValue) { |
| |
| // submessage: decorate the submessage and use its name as the type |
| if (typeof fieldType === "function") |
| fieldType = util.decorateType(fieldType).name; |
| |
| // enum reference: create a reflected copy of the enum and keep reuseing it |
| else if (fieldType && typeof fieldType === "object") |
| fieldType = util.decorateEnum(fieldType).name; |
| |
| return function fieldDecorator(prototype, fieldName) { |
| util.decorateType(prototype.constructor) |
| .add(new Field(fieldName, fieldId, fieldType, fieldRule, { "default": defaultValue })); |
| }; |
| }; |
| |
| /** |
| * Field decorator (TypeScript). |
| * @name Field.d |
| * @function |
| * @param {number} fieldId Field id |
| * @param {Constructor<T>|string} fieldType Field type |
| * @param {"optional"|"required"|"repeated"} [fieldRule="optional"] Field rule |
| * @returns {FieldDecorator} Decorator function |
| * @template T extends Message<T> |
| * @variation 2 |
| */ |
| // like Field.d but without a default value |
| |
| // Sets up cyclic dependencies (called in index-light) |
| Field._configure = function configure(Type_) { |
| Type = Type_; |
| }; |