'use strict';
|
|
const Net = require('net');
|
|
const Address = require('@hapi/address');
|
const Hoek = require('@hapi/hoek');
|
|
const Any = require('../any');
|
const Ref = require('../../ref');
|
const JoiDate = require('../date');
|
|
const Uri = require('./uri');
|
const Ip = require('./ip');
|
|
|
const internals = {
|
uriRegex: Uri.createUriRegex(),
|
ipRegex: Ip.createIpRegex(['ipv4', 'ipv6', 'ipvfuture'], 'optional'),
|
guidBrackets: {
|
'{': '}', '[': ']', '(': ')', '': ''
|
},
|
guidVersions: {
|
uuidv1: '1',
|
uuidv2: '2',
|
uuidv3: '3',
|
uuidv4: '4',
|
uuidv5: '5'
|
},
|
cidrPresences: ['required', 'optional', 'forbidden'],
|
normalizationForms: ['NFC', 'NFD', 'NFKC', 'NFKD']
|
};
|
|
|
internals.String = class extends Any {
|
|
constructor() {
|
|
super();
|
this._type = 'string';
|
this._invalids.add('');
|
}
|
|
_base(value, state, options) {
|
|
if (typeof value === 'string' &&
|
options.convert) {
|
|
if (this._flags.normalize) {
|
value = value.normalize(this._flags.normalize);
|
}
|
|
if (this._flags.case) {
|
value = (this._flags.case === 'upper' ? value.toLocaleUpperCase() : value.toLocaleLowerCase());
|
}
|
|
if (this._flags.trim) {
|
value = value.trim();
|
}
|
|
if (this._inner.replacements) {
|
|
for (let i = 0; i < this._inner.replacements.length; ++i) {
|
const replacement = this._inner.replacements[i];
|
value = value.replace(replacement.pattern, replacement.replacement);
|
}
|
}
|
|
if (this._flags.truncate) {
|
for (let i = 0; i < this._tests.length; ++i) {
|
const test = this._tests[i];
|
if (test.name === 'max') {
|
value = value.slice(0, test.arg);
|
break;
|
}
|
}
|
}
|
|
if (this._flags.byteAligned && value.length % 2 !== 0) {
|
value = `0${value}`;
|
}
|
}
|
|
return {
|
value,
|
errors: (typeof value === 'string') ? null : this.createError('string.base', { value }, state, options)
|
};
|
}
|
|
insensitive() {
|
|
if (this._flags.insensitive) {
|
return this;
|
}
|
|
const obj = this.clone();
|
obj._flags.insensitive = true;
|
return obj;
|
}
|
|
creditCard() {
|
|
return this._test('creditCard', undefined, function (value, state, options) {
|
|
let i = value.length;
|
let sum = 0;
|
let mul = 1;
|
|
while (i--) {
|
const char = value.charAt(i) * mul;
|
sum = sum + (char - (char > 9) * 9);
|
mul = mul ^ 3;
|
}
|
|
const check = (sum % 10 === 0) && (sum > 0);
|
return check ? value : this.createError('string.creditCard', { value }, state, options);
|
});
|
}
|
|
regex(pattern, patternOptions) {
|
|
Hoek.assert(pattern instanceof RegExp, 'pattern must be a RegExp');
|
Hoek.assert(!pattern.flags.includes('g') && !pattern.flags.includes('y'), 'pattern should not use global or sticky mode');
|
|
const patternObject = { pattern };
|
|
if (typeof patternOptions === 'string') {
|
patternObject.name = patternOptions;
|
}
|
else if (typeof patternOptions === 'object') {
|
patternObject.invert = !!patternOptions.invert;
|
|
if (patternOptions.name) {
|
patternObject.name = patternOptions.name;
|
}
|
}
|
|
const errorCode = ['string.regex', patternObject.invert ? '.invert' : '', patternObject.name ? '.name' : '.base'].join('');
|
|
return this._test('regex', patternObject, function (value, state, options) {
|
|
const patternMatch = patternObject.pattern.test(value);
|
|
if (patternMatch ^ patternObject.invert) {
|
return value;
|
}
|
|
return this.createError(errorCode, { name: patternObject.name, pattern: patternObject.pattern, value }, state, options);
|
});
|
}
|
|
alphanum() {
|
|
return this._test('alphanum', undefined, function (value, state, options) {
|
|
if (/^[a-zA-Z0-9]+$/.test(value)) {
|
return value;
|
}
|
|
return this.createError('string.alphanum', { value }, state, options);
|
});
|
}
|
|
token() {
|
|
return this._test('token', undefined, function (value, state, options) {
|
|
if (/^\w+$/.test(value)) {
|
return value;
|
}
|
|
return this.createError('string.token', { value }, state, options);
|
});
|
}
|
|
email(validationOptions) {
|
|
if (validationOptions) {
|
Hoek.assert(typeof validationOptions === 'object', 'email options must be an object');
|
|
// Migration validation for unsupported options
|
|
Hoek.assert(validationOptions.checkDNS === undefined, 'checkDNS option is not supported');
|
Hoek.assert(validationOptions.errorLevel === undefined, 'errorLevel option is not supported');
|
Hoek.assert(validationOptions.minDomainAtoms === undefined, 'minDomainAtoms option is not supported, use minDomainSegments instead');
|
Hoek.assert(validationOptions.tldBlacklist === undefined, 'tldBlacklist option is not supported, use tlds.deny instead');
|
Hoek.assert(validationOptions.tldWhitelist === undefined, 'tldWhitelist option is not supported, use tlds.allow instead');
|
|
// Validate options
|
|
if (validationOptions.tlds &&
|
typeof validationOptions.tlds === 'object') {
|
|
Hoek.assert(validationOptions.tlds.allow === undefined ||
|
validationOptions.tlds.allow === false ||
|
validationOptions.tlds.allow === true ||
|
Array.isArray(validationOptions.tlds.allow) ||
|
validationOptions.tlds.allow instanceof Set, 'tlds.allow must be an array, Set, or boolean');
|
|
Hoek.assert(validationOptions.tlds.deny === undefined ||
|
Array.isArray(validationOptions.tlds.deny) ||
|
validationOptions.tlds.deny instanceof Set, 'tlds.deny must be an array or Set');
|
|
const normalizeTable = (table) => {
|
|
if (table === undefined ||
|
typeof table === 'boolean' ||
|
table instanceof Set) {
|
|
return table;
|
}
|
|
return new Set(table);
|
};
|
|
validationOptions = Object.assign({}, validationOptions); // Shallow cloned
|
validationOptions.tlds = {
|
allow: normalizeTable(validationOptions.tlds.allow),
|
deny: normalizeTable(validationOptions.tlds.deny)
|
};
|
}
|
|
Hoek.assert(validationOptions.minDomainSegments === undefined ||
|
Number.isSafeInteger(validationOptions.minDomainSegments) && validationOptions.minDomainSegments > 0, 'minDomainSegments must be a positive integer');
|
}
|
|
return this._test('email', validationOptions, function (value, state, options) {
|
|
if (Address.email.isValid(value, validationOptions)) {
|
return value;
|
}
|
|
return this.createError('string.email', { value }, state, options);
|
});
|
}
|
|
ip(ipOptions = {}) {
|
|
let regex = internals.ipRegex;
|
Hoek.assert(typeof ipOptions === 'object', 'options must be an object');
|
|
if (ipOptions.cidr) {
|
Hoek.assert(typeof ipOptions.cidr === 'string', 'cidr must be a string');
|
ipOptions.cidr = ipOptions.cidr.toLowerCase();
|
|
Hoek.assert(Hoek.contain(internals.cidrPresences, ipOptions.cidr), 'cidr must be one of ' + internals.cidrPresences.join(', '));
|
|
// If we only received a `cidr` setting, create a regex for it. But we don't need to create one if `cidr` is "optional" since that is the default
|
if (!ipOptions.version && ipOptions.cidr !== 'optional') {
|
regex = Ip.createIpRegex(['ipv4', 'ipv6', 'ipvfuture'], ipOptions.cidr);
|
}
|
}
|
else {
|
|
// Set our default cidr strategy
|
ipOptions.cidr = 'optional';
|
}
|
|
let versions;
|
if (ipOptions.version) {
|
if (!Array.isArray(ipOptions.version)) {
|
ipOptions.version = [ipOptions.version];
|
}
|
|
Hoek.assert(ipOptions.version.length >= 1, 'version must have at least 1 version specified');
|
|
versions = [];
|
for (let i = 0; i < ipOptions.version.length; ++i) {
|
let version = ipOptions.version[i];
|
Hoek.assert(typeof version === 'string', 'version at position ' + i + ' must be a string');
|
version = version.toLowerCase();
|
Hoek.assert(Ip.versions[version], 'version at position ' + i + ' must be one of ' + Object.keys(Ip.versions).join(', '));
|
versions.push(version);
|
}
|
|
// Make sure we have a set of versions
|
versions = Array.from(new Set(versions));
|
|
regex = Ip.createIpRegex(versions, ipOptions.cidr);
|
}
|
|
return this._test('ip', ipOptions, function (value, state, options) {
|
|
if (regex.test(value)) {
|
return value;
|
}
|
|
if (versions) {
|
return this.createError('string.ipVersion', { value, cidr: ipOptions.cidr, version: versions }, state, options);
|
}
|
|
return this.createError('string.ip', { value, cidr: ipOptions.cidr }, state, options);
|
});
|
}
|
|
uri(uriOptions) {
|
|
let customScheme = '';
|
let allowRelative = false;
|
let relativeOnly = false;
|
let allowQuerySquareBrackets = false;
|
let regex = internals.uriRegex;
|
|
if (uriOptions) {
|
Hoek.assert(typeof uriOptions === 'object', 'options must be an object');
|
|
const unknownOptions = Object.keys(uriOptions).filter((key) => !['scheme', 'allowRelative', 'relativeOnly', 'allowQuerySquareBrackets'].includes(key));
|
Hoek.assert(unknownOptions.length === 0, `options contain unknown keys: ${unknownOptions}`);
|
|
if (uriOptions.scheme) {
|
Hoek.assert(uriOptions.scheme instanceof RegExp || typeof uriOptions.scheme === 'string' || Array.isArray(uriOptions.scheme), 'scheme must be a RegExp, String, or Array');
|
|
if (!Array.isArray(uriOptions.scheme)) {
|
uriOptions.scheme = [uriOptions.scheme];
|
}
|
|
Hoek.assert(uriOptions.scheme.length >= 1, 'scheme must have at least 1 scheme specified');
|
|
// Flatten the array into a string to be used to match the schemes.
|
for (let i = 0; i < uriOptions.scheme.length; ++i) {
|
const scheme = uriOptions.scheme[i];
|
Hoek.assert(scheme instanceof RegExp || typeof scheme === 'string', 'scheme at position ' + i + ' must be a RegExp or String');
|
|
// Add OR separators if a value already exists
|
customScheme = customScheme + (customScheme ? '|' : '');
|
|
// If someone wants to match HTTP or HTTPS for example then we need to support both RegExp and String so we don't escape their pattern unknowingly.
|
if (scheme instanceof RegExp) {
|
customScheme = customScheme + scheme.source;
|
}
|
else {
|
Hoek.assert(/[a-zA-Z][a-zA-Z0-9+-\.]*/.test(scheme), 'scheme at position ' + i + ' must be a valid scheme');
|
customScheme = customScheme + Hoek.escapeRegex(scheme);
|
}
|
}
|
}
|
|
if (uriOptions.allowRelative) {
|
allowRelative = true;
|
}
|
|
if (uriOptions.relativeOnly) {
|
relativeOnly = true;
|
}
|
|
if (uriOptions.allowQuerySquareBrackets) {
|
allowQuerySquareBrackets = true;
|
}
|
}
|
|
if (customScheme || allowRelative || relativeOnly || allowQuerySquareBrackets) {
|
regex = Uri.createUriRegex(customScheme, allowRelative, relativeOnly, allowQuerySquareBrackets);
|
}
|
|
return this._test('uri', uriOptions, function (value, state, options) {
|
|
if (regex.test(value)) {
|
return value;
|
}
|
|
if (relativeOnly) {
|
return this.createError('string.uriRelativeOnly', { value }, state, options);
|
}
|
|
if (customScheme) {
|
return this.createError('string.uriCustomScheme', { scheme: customScheme, value }, state, options);
|
}
|
|
return this.createError('string.uri', { value }, state, options);
|
});
|
}
|
|
isoDate() {
|
|
return this._test('isoDate', undefined, function (value, state, options) {
|
|
if (JoiDate._isIsoDate(value)) {
|
if (!options.convert) {
|
return value;
|
}
|
|
const d = new Date(value);
|
if (!isNaN(d.getTime())) {
|
return d.toISOString();
|
}
|
}
|
|
return this.createError('string.isoDate', { value }, state, options);
|
});
|
}
|
|
guid(guidOptions) {
|
|
let versionNumbers = '';
|
|
if (guidOptions && guidOptions.version) {
|
if (!Array.isArray(guidOptions.version)) {
|
guidOptions.version = [guidOptions.version];
|
}
|
|
Hoek.assert(guidOptions.version.length >= 1, 'version must have at least 1 valid version specified');
|
const versions = new Set();
|
|
for (let i = 0; i < guidOptions.version.length; ++i) {
|
let version = guidOptions.version[i];
|
Hoek.assert(typeof version === 'string', 'version at position ' + i + ' must be a string');
|
version = version.toLowerCase();
|
const versionNumber = internals.guidVersions[version];
|
Hoek.assert(versionNumber, 'version at position ' + i + ' must be one of ' + Object.keys(internals.guidVersions).join(', '));
|
Hoek.assert(!(versions.has(versionNumber)), 'version at position ' + i + ' must not be a duplicate.');
|
|
versionNumbers += versionNumber;
|
versions.add(versionNumber);
|
}
|
}
|
|
const guidRegex = new RegExp(`^([\\[{\\(]?)[0-9A-F]{8}([:-]?)[0-9A-F]{4}\\2?[${versionNumbers || '0-9A-F'}][0-9A-F]{3}\\2?[${versionNumbers ? '89AB' : '0-9A-F'}][0-9A-F]{3}\\2?[0-9A-F]{12}([\\]}\\)]?)$`, 'i');
|
|
return this._test('guid', guidOptions, function (value, state, options) {
|
|
const results = guidRegex.exec(value);
|
|
if (!results) {
|
return this.createError('string.guid', { value }, state, options);
|
}
|
|
// Matching braces
|
if (internals.guidBrackets[results[1]] !== results[results.length - 1]) {
|
return this.createError('string.guid', { value }, state, options);
|
}
|
|
return value;
|
});
|
}
|
|
hex(hexOptions = {}) {
|
|
Hoek.assert(typeof hexOptions === 'object', 'hex options must be an object');
|
Hoek.assert(typeof hexOptions.byteAligned === 'undefined' || typeof hexOptions.byteAligned === 'boolean',
|
'byteAligned must be boolean');
|
|
const byteAligned = hexOptions.byteAligned === true;
|
const regex = /^[a-f0-9]+$/i;
|
|
const obj = this._test('hex', regex, function (value, state, options) {
|
|
if (regex.test(value)) {
|
if (byteAligned && value.length % 2 !== 0) {
|
return this.createError('string.hexAlign', { value }, state, options);
|
}
|
|
return value;
|
}
|
|
return this.createError('string.hex', { value }, state, options);
|
});
|
|
if (byteAligned) {
|
obj._flags.byteAligned = true;
|
}
|
|
return obj;
|
}
|
|
base64(base64Options = {}) {
|
|
// Validation.
|
Hoek.assert(typeof base64Options === 'object', 'base64 options must be an object');
|
Hoek.assert(typeof base64Options.paddingRequired === 'undefined' || typeof base64Options.paddingRequired === 'boolean',
|
'paddingRequired must be boolean');
|
|
// Determine if padding is required.
|
const paddingRequired = base64Options.paddingRequired === false ?
|
base64Options.paddingRequired
|
: base64Options.paddingRequired || true;
|
|
// Set validation based on preference.
|
const regex = paddingRequired ?
|
// Padding is required.
|
/^(?:[A-Za-z0-9+\/]{2}[A-Za-z0-9+\/]{2})*(?:[A-Za-z0-9+\/]{2}==|[A-Za-z0-9+\/]{3}=)?$/
|
// Padding is optional.
|
: /^(?:[A-Za-z0-9+\/]{2}[A-Za-z0-9+\/]{2})*(?:[A-Za-z0-9+\/]{2}(==)?|[A-Za-z0-9+\/]{3}=?)?$/;
|
|
return this._test('base64', regex, function (value, state, options) {
|
|
if (regex.test(value)) {
|
return value;
|
}
|
|
return this.createError('string.base64', { value }, state, options);
|
});
|
}
|
|
dataUri(dataUriOptions = {}) {
|
|
const regex = /^data:[\w+.-]+\/[\w+.-]+;((charset=[\w-]+|base64),)?(.*)$/;
|
|
// Determine if padding is required.
|
const paddingRequired = dataUriOptions.paddingRequired === false ?
|
dataUriOptions.paddingRequired
|
: dataUriOptions.paddingRequired || true;
|
|
const base64regex = paddingRequired ?
|
/^(?:[A-Za-z0-9+\/]{4})*(?:[A-Za-z0-9+\/]{2}==|[A-Za-z0-9+\/]{3}=)?$/
|
: /^(?:[A-Za-z0-9+\/]{4})*(?:[A-Za-z0-9+\/]{2}(==)?|[A-Za-z0-9+\/]{3}=?)?$/;
|
|
return this._test('dataUri', regex, function (value, state, options) {
|
|
const matches = value.match(regex);
|
|
if (matches) {
|
if (!matches[2]) {
|
return value;
|
}
|
|
if (matches[2] !== 'base64') {
|
return value;
|
}
|
|
if (base64regex.test(matches[3])) {
|
return value;
|
}
|
}
|
|
return this.createError('string.dataUri', { value }, state, options);
|
});
|
}
|
|
hostname() {
|
|
const regex = /^(([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9])\.)*([A-Za-z0-9]|[A-Za-z0-9][A-Za-z0-9\-]*[A-Za-z0-9])$/;
|
|
return this._test('hostname', undefined, function (value, state, options) {
|
|
if ((value.length <= 255 && regex.test(value)) ||
|
Net.isIPv6(value)) {
|
|
return value;
|
}
|
|
return this.createError('string.hostname', { value }, state, options);
|
});
|
}
|
|
normalize(form = 'NFC') {
|
|
Hoek.assert(Hoek.contain(internals.normalizationForms, form), 'normalization form must be one of ' + internals.normalizationForms.join(', '));
|
|
const obj = this._test('normalize', form, function (value, state, options) {
|
|
if (options.convert ||
|
value === value.normalize(form)) {
|
|
return value;
|
}
|
|
return this.createError('string.normalize', { value, form }, state, options);
|
});
|
|
obj._flags.normalize = form;
|
return obj;
|
}
|
|
lowercase() {
|
|
const obj = this._test('lowercase', undefined, function (value, state, options) {
|
|
if (options.convert ||
|
value === value.toLocaleLowerCase()) {
|
|
return value;
|
}
|
|
return this.createError('string.lowercase', { value }, state, options);
|
});
|
|
obj._flags.case = 'lower';
|
return obj;
|
}
|
|
uppercase() {
|
|
const obj = this._test('uppercase', undefined, function (value, state, options) {
|
|
if (options.convert ||
|
value === value.toLocaleUpperCase()) {
|
|
return value;
|
}
|
|
return this.createError('string.uppercase', { value }, state, options);
|
});
|
|
obj._flags.case = 'upper';
|
return obj;
|
}
|
|
trim(enabled = true) {
|
|
Hoek.assert(typeof enabled === 'boolean', 'option must be a boolean');
|
|
if ((this._flags.trim && enabled) || (!this._flags.trim && !enabled)) {
|
return this;
|
}
|
|
let obj;
|
if (enabled) {
|
obj = this._test('trim', undefined, function (value, state, options) {
|
|
if (options.convert ||
|
value === value.trim()) {
|
|
return value;
|
}
|
|
return this.createError('string.trim', { value }, state, options);
|
});
|
}
|
else {
|
obj = this.clone();
|
obj._tests = obj._tests.filter((test) => test.name !== 'trim');
|
}
|
|
obj._flags.trim = enabled;
|
return obj;
|
}
|
|
replace(pattern, replacement) {
|
|
if (typeof pattern === 'string') {
|
pattern = new RegExp(Hoek.escapeRegex(pattern), 'g');
|
}
|
|
Hoek.assert(pattern instanceof RegExp, 'pattern must be a RegExp');
|
Hoek.assert(typeof replacement === 'string', 'replacement must be a String');
|
|
// This can not be considere a test like trim, we can't "reject"
|
// anything from this rule, so just clone the current object
|
const obj = this.clone();
|
|
if (!obj._inner.replacements) {
|
obj._inner.replacements = [];
|
}
|
|
obj._inner.replacements.push({
|
pattern,
|
replacement
|
});
|
|
return obj;
|
}
|
|
truncate(enabled) {
|
|
const value = enabled === undefined ? true : !!enabled;
|
|
if (this._flags.truncate === value) {
|
return this;
|
}
|
|
const obj = this.clone();
|
obj._flags.truncate = value;
|
return obj;
|
}
|
|
};
|
|
internals.compare = function (type, compare) {
|
|
return function (limit, encoding) {
|
|
const isRef = Ref.isRef(limit);
|
|
Hoek.assert((Number.isSafeInteger(limit) && limit >= 0) || isRef, 'limit must be a positive integer or reference');
|
Hoek.assert(!encoding || Buffer.isEncoding(encoding), 'Invalid encoding:', encoding);
|
|
return this._test(type, limit, function (value, state, options) {
|
|
let compareTo;
|
if (isRef) {
|
compareTo = limit(state.reference || state.parent, options);
|
|
if (!Number.isSafeInteger(compareTo)) {
|
return this.createError('string.ref', { ref: limit, value: compareTo }, state, options);
|
}
|
}
|
else {
|
compareTo = limit;
|
}
|
|
if (compare(value, compareTo, encoding)) {
|
return value;
|
}
|
|
return this.createError('string.' + type, { limit: compareTo, value, encoding }, state, options);
|
});
|
};
|
};
|
|
|
internals.String.prototype.min = internals.compare('min', (value, limit, encoding) => {
|
|
const length = encoding ? Buffer.byteLength(value, encoding) : value.length;
|
return length >= limit;
|
});
|
|
|
internals.String.prototype.max = internals.compare('max', (value, limit, encoding) => {
|
|
const length = encoding ? Buffer.byteLength(value, encoding) : value.length;
|
return length <= limit;
|
});
|
|
|
internals.String.prototype.length = internals.compare('length', (value, limit, encoding) => {
|
|
const length = encoding ? Buffer.byteLength(value, encoding) : value.length;
|
return length === limit;
|
});
|
|
// Aliases
|
|
internals.String.prototype.uuid = internals.String.prototype.guid;
|
|
module.exports = new internals.String();
|