|
|
/*!
|
|
* Module dependencies.
|
|
*/
|
|
|
|
var SchemaType = require('../schematype')
|
|
, CastError = SchemaType.CastError
|
|
, utils = require('../utils')
|
|
, Document
|
|
|
|
/**
|
|
* String SchemaType constructor.
|
|
*
|
|
* @param {String} key
|
|
* @param {Object} options
|
|
* @inherits SchemaType
|
|
* @api private
|
|
*/
|
|
|
|
function SchemaString (key, options) {
|
|
this.enumValues = [];
|
|
this.regExp = null;
|
|
SchemaType.call(this, key, options, 'String');
|
|
};
|
|
|
|
/*!
|
|
* Inherits from SchemaType.
|
|
*/
|
|
|
|
SchemaString.prototype.__proto__ = SchemaType.prototype;
|
|
|
|
/**
|
|
* Adds enumeration values and a coinciding validator.
|
|
*
|
|
* ####Example:
|
|
*
|
|
* var states = 'opening open closing closed'.split(' ')
|
|
* var s = new Schema({ state: { type: String, enum: states })
|
|
* var M = db.model('M', s)
|
|
* var m = new M({ state: 'invalid' })
|
|
* m.save(function (err) {
|
|
* console.error(err) // validator error
|
|
* m.state = 'open'
|
|
* m.save() // success
|
|
* })
|
|
*
|
|
* @param {String} [args...] enumeration values
|
|
* @return {SchemaType} this
|
|
* @api public
|
|
*/
|
|
|
|
SchemaString.prototype.enum = function () {
|
|
var len = arguments.length;
|
|
|
|
if (!len || undefined === arguments[0] || false === arguments[0]) {
|
|
if (this.enumValidator){
|
|
this.enumValidator = false;
|
|
this.validators = this.validators.filter(function(v){
|
|
return v[1] != 'enum';
|
|
});
|
|
}
|
|
return this;
|
|
}
|
|
|
|
for (var i = 0; i < len; i++) {
|
|
if (undefined !== arguments[i]) {
|
|
this.enumValues.push(this.cast(arguments[i]));
|
|
}
|
|
}
|
|
|
|
if (!this.enumValidator) {
|
|
var values = this.enumValues;
|
|
this.enumValidator = function(v){
|
|
return undefined === v || ~values.indexOf(v);
|
|
};
|
|
this.validators.push([this.enumValidator, 'enum']);
|
|
}
|
|
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Adds a lowercase setter.
|
|
*
|
|
* ####Example:
|
|
*
|
|
* var s = new Schema({ email: { type: String, lowercase: true }})
|
|
* var M = db.model('M', s);
|
|
* var m = new M({ email: 'SomeEmail@example.COM' });
|
|
* console.log(m.email) // someemail@example.com
|
|
*
|
|
* @api public
|
|
* @return {SchemaType} this
|
|
*/
|
|
|
|
SchemaString.prototype.lowercase = function () {
|
|
return this.set(function (v, self) {
|
|
if ('string' != typeof v) v = self.cast(v)
|
|
if (v) return v.toLowerCase();
|
|
return v;
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Adds an uppercase setter.
|
|
*
|
|
* ####Example:
|
|
*
|
|
* var s = new Schema({ caps: { type: String, uppercase: true }})
|
|
* var M = db.model('M', s);
|
|
* var m = new M({ caps: 'an example' });
|
|
* console.log(m.caps) // AN EXAMPLE
|
|
*
|
|
* @api public
|
|
* @return {SchemaType} this
|
|
*/
|
|
|
|
SchemaString.prototype.uppercase = function () {
|
|
return this.set(function (v, self) {
|
|
if ('string' != typeof v) v = self.cast(v)
|
|
if (v) return v.toUpperCase();
|
|
return v;
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Adds a trim setter.
|
|
*
|
|
* The string value will be trimmed when set.
|
|
*
|
|
* ####Example:
|
|
*
|
|
* var s = new Schema({ name: { type: String, trim: true }})
|
|
* var M = db.model('M', s)
|
|
* var string = ' some name '
|
|
* console.log(string.length) // 11
|
|
* var m = new M({ name: string })
|
|
* console.log(m.name.length) // 9
|
|
*
|
|
* @api public
|
|
* @return {SchemaType} this
|
|
*/
|
|
|
|
SchemaString.prototype.trim = function () {
|
|
return this.set(function (v, self) {
|
|
if ('string' != typeof v) v = self.cast(v)
|
|
if (v) return v.trim();
|
|
return v;
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Sets a regexp validator.
|
|
*
|
|
* Any value that does not pass `regExp`.test(val) will fail validation.
|
|
*
|
|
* ####Example:
|
|
*
|
|
* var s = new Schema({ name: { type: String, match: /^a/ }})
|
|
* var M = db.model('M', s)
|
|
* var m = new M({ name: 'invalid' })
|
|
* m.validate(function (err) {
|
|
* console.error(err) // validation error
|
|
* m.name = 'apples'
|
|
* m.validate(function (err) {
|
|
* assert.ok(err) // success
|
|
* })
|
|
* })
|
|
*
|
|
* @param {RegExp} regExp regular expression to test against
|
|
* @return {SchemaType} this
|
|
* @api public
|
|
*/
|
|
|
|
SchemaString.prototype.match = function match (regExp) {
|
|
this.validators.push([function(v){
|
|
return null != v && '' !== v
|
|
? regExp.test(v)
|
|
: true
|
|
}, 'regexp']);
|
|
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Check required
|
|
*
|
|
* @param {String|null|undefined} value
|
|
* @api private
|
|
*/
|
|
|
|
SchemaString.prototype.checkRequired = function checkRequired (value, doc) {
|
|
if (SchemaType._isRef(this, value, doc, true)) {
|
|
return null != value;
|
|
} else {
|
|
return (value instanceof String || typeof value == 'string') && value.length;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Casts to String
|
|
*
|
|
* @api private
|
|
*/
|
|
|
|
SchemaString.prototype.cast = function (value, doc, init) {
|
|
if (SchemaType._isRef(this, value, doc, init)) {
|
|
// wait! we may need to cast this to a document
|
|
|
|
if (null == value) {
|
|
return value;
|
|
}
|
|
|
|
// lazy load
|
|
Document || (Document = require('./../document'));
|
|
|
|
if (value instanceof Document) {
|
|
value.$__.wasPopulated = true;
|
|
return value;
|
|
}
|
|
|
|
// setting a populated path
|
|
if ('string' == typeof value) {
|
|
return value;
|
|
} else if (Buffer.isBuffer(value) || !utils.isObject(value)) {
|
|
throw new CastError('string', value, this.path);
|
|
}
|
|
|
|
// Handle the case where user directly sets a populated
|
|
// path to a plain object; cast to the Model used in
|
|
// the population query.
|
|
var path = doc.$__fullPath(this.path);
|
|
var owner = doc.ownerDocument ? doc.ownerDocument() : doc;
|
|
var pop = owner.populated(path, true);
|
|
var ret = new pop.options.model(value);
|
|
ret.$__.wasPopulated = true;
|
|
return ret;
|
|
}
|
|
|
|
if (value === null) {
|
|
return value;
|
|
}
|
|
|
|
if ('undefined' !== typeof value) {
|
|
// handle documents being passed
|
|
if (value._id && 'string' == typeof value._id) {
|
|
return value._id;
|
|
}
|
|
if (value.toString) {
|
|
return value.toString();
|
|
}
|
|
}
|
|
|
|
|
|
throw new CastError('string', value, this.path);
|
|
};
|
|
|
|
/*!
|
|
* ignore
|
|
*/
|
|
|
|
function handleSingle (val) {
|
|
return this.castForQuery(val);
|
|
}
|
|
|
|
function handleArray (val) {
|
|
var self = this;
|
|
return val.map(function (m) {
|
|
return self.castForQuery(m);
|
|
});
|
|
}
|
|
|
|
SchemaString.prototype.$conditionalHandlers = {
|
|
'$ne' : handleSingle
|
|
, '$in' : handleArray
|
|
, '$nin': handleArray
|
|
, '$gt' : handleSingle
|
|
, '$lt' : handleSingle
|
|
, '$gte': handleSingle
|
|
, '$lte': handleSingle
|
|
, '$all': handleArray
|
|
, '$regex': handleSingle
|
|
, '$options': handleSingle
|
|
};
|
|
|
|
/**
|
|
* Casts contents for queries.
|
|
*
|
|
* @param {String} $conditional
|
|
* @param {any} [val]
|
|
* @api private
|
|
*/
|
|
|
|
SchemaString.prototype.castForQuery = function ($conditional, val) {
|
|
var handler;
|
|
if (arguments.length === 2) {
|
|
handler = this.$conditionalHandlers[$conditional];
|
|
if (!handler)
|
|
throw new Error("Can't use " + $conditional + " with String.");
|
|
return handler.call(this, val);
|
|
} else {
|
|
val = $conditional;
|
|
if (val instanceof RegExp) return val;
|
|
return this.cast(val);
|
|
}
|
|
};
|
|
|
|
/*!
|
|
* Module exports.
|
|
*/
|
|
|
|
module.exports = SchemaString;
|