'use strict'; // Load modules const Any = require('./any'); const Ref = require('./ref'); const Hoek = require('hoek'); // Declare internals const internals = { precisionRx: /(?:\.(\d+))?(?:[eE]([+-]?\d+))?$/ }; internals.Number = class extends Any { constructor() { super(); this._type = 'number'; this._invalids.add(Infinity); this._invalids.add(-Infinity); } _base(value, state, options) { const result = { errors: null, value }; if (typeof value === 'string' && options.convert) { const number = parseFloat(value); result.value = (isNaN(number) || !isFinite(value)) ? NaN : number; } const isNumber = typeof result.value === 'number' && !isNaN(result.value); if (options.convert && 'precision' in this._flags && isNumber) { // This is conceptually equivalent to using toFixed but it should be much faster const precision = Math.pow(10, this._flags.precision); result.value = Math.round(result.value * precision) / precision; } result.errors = isNumber ? null : this.createError('number.base', null, state, options); return result; } multiple(base) { const isRef = Ref.isRef(base); if (!isRef) { Hoek.assert(typeof base === 'number' && isFinite(base), 'multiple must be a number'); Hoek.assert(base > 0, 'multiple must be greater than 0'); } return this._test('multiple', base, function (value, state, options) { const divisor = isRef ? base(state.parent, options) : base; if (isRef && (typeof divisor !== 'number' || !isFinite(divisor))) { return this.createError('number.ref', { ref: base.key }, state, options); } if (value % divisor === 0) { return value; } return this.createError('number.multiple', { multiple: base, value }, state, options); }); } integer() { return this._test('integer', undefined, function (value, state, options) { return Hoek.isInteger(value) ? value : this.createError('number.integer', { value }, state, options); }); } negative() { return this._test('negative', undefined, function (value, state, options) { if (value < 0) { return value; } return this.createError('number.negative', { value }, state, options); }); } positive() { return this._test('positive', undefined, function (value, state, options) { if (value > 0) { return value; } return this.createError('number.positive', { value }, state, options); }); } precision(limit) { Hoek.assert(Hoek.isInteger(limit), 'limit must be an integer'); Hoek.assert(!('precision' in this._flags), 'precision already set'); const obj = this._test('precision', limit, function (value, state, options) { const places = value.toString().match(internals.precisionRx); const decimals = Math.max((places[1] ? places[1].length : 0) - (places[2] ? parseInt(places[2], 10) : 0), 0); if (decimals <= limit) { return value; } return this.createError('number.precision', { limit, value }, state, options); }); obj._flags.precision = limit; return obj; } }; internals.compare = function (type, compare) { return function (limit) { const isRef = Ref.isRef(limit); const isNumber = typeof limit === 'number' && !isNaN(limit); Hoek.assert(isNumber || isRef, 'limit must be a number or reference'); return this._test(type, limit, function (value, state, options) { let compareTo; if (isRef) { compareTo = limit(state.parent, options); if (!(typeof compareTo === 'number' && !isNaN(compareTo))) { return this.createError('number.ref', { ref: limit.key }, state, options); } } else { compareTo = limit; } if (compare(value, compareTo)) { return value; } return this.createError('number.' + type, { limit: compareTo, value }, state, options); }); }; }; internals.Number.prototype.min = internals.compare('min', (value, limit) => value >= limit); internals.Number.prototype.max = internals.compare('max', (value, limit) => value <= limit); internals.Number.prototype.greater = internals.compare('greater', (value, limit) => value > limit); internals.Number.prototype.less = internals.compare('less', (value, limit) => value < limit); module.exports = new internals.Number();