'use strict';
|
|
const Hoek = require('@hapi/hoek');
|
|
const Any = require('../any');
|
const Ref = require('../../ref');
|
|
|
const internals = {
|
precisionRx: /(?:\.(\d+))?(?:[eE]([+-]?\d+))?$/,
|
normalizeExponent(str) {
|
|
return str
|
.replace(/\.?0+e/, 'e')
|
.replace(/e\+/, 'e')
|
.replace(/^\+/, '')
|
.replace(/^(-?)0+([1-9])/, '$1$2');
|
},
|
normalizeDecimal(str) {
|
|
str = str
|
.replace(/^\+/, '')
|
.replace(/\.0+$/, '')
|
.replace(/^(-?)0+([1-9])/, '$1$2');
|
|
if (str.includes('.') && str.endsWith('0')) {
|
str = str.replace(/0+$/, '');
|
}
|
|
return str;
|
}
|
};
|
|
|
internals.Number = class extends Any {
|
|
constructor() {
|
|
super();
|
this._type = 'number';
|
this._flags.unsafe = false;
|
this._invalids.add(Infinity);
|
this._invalids.add(-Infinity);
|
}
|
|
_base(value, state, options) {
|
|
const result = {
|
errors: null,
|
value
|
};
|
|
if (typeof value === 'string' &&
|
options.convert) {
|
|
const matches = value.match(/^\s*[+-]?\d+(?:\.\d+)?(?:e([+-]?\d+))?\s*$/i);
|
if (matches) {
|
|
value = value.trim();
|
result.value = parseFloat(value);
|
|
if (!this._flags.unsafe) {
|
if (value.includes('e')) {
|
if (internals.normalizeExponent(`${result.value / Math.pow(10, matches[1])}e${matches[1]}`) !== internals.normalizeExponent(value)) {
|
result.errors = this.createError('number.unsafe', { value }, state, options);
|
return result;
|
}
|
}
|
else {
|
if (result.value.toString() !== internals.normalizeDecimal(value)) {
|
result.errors = this.createError('number.unsafe', { value }, state, options);
|
return result;
|
}
|
}
|
}
|
}
|
}
|
|
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;
|
}
|
|
if (isNumber) {
|
if (!this._flags.unsafe &&
|
(value > Number.MAX_SAFE_INTEGER || value < Number.MIN_SAFE_INTEGER)) {
|
result.errors = this.createError('number.unsafe', { value }, state, options);
|
}
|
}
|
else {
|
result.errors = this.createError('number.base', { value }, 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.reference || 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 Math.trunc(value) - value === 0 ? value : this.createError('number.integer', { value }, state, options);
|
});
|
}
|
|
unsafe(enabled = true) {
|
|
Hoek.assert(typeof enabled === 'boolean', 'enabled must be a boolean');
|
|
if (this._flags.unsafe === enabled) {
|
return this;
|
}
|
|
const obj = this.clone();
|
obj._flags.unsafe = enabled;
|
return obj;
|
}
|
|
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(Number.isSafeInteger(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;
|
}
|
|
port() {
|
|
return this._test('port', undefined, function (value, state, options) {
|
|
if (!Number.isSafeInteger(value) || value < 0 || value > 65535) {
|
return this.createError('number.port', { value }, state, options);
|
}
|
|
return value;
|
});
|
}
|
|
};
|
|
|
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.reference || 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();
|