176 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			176 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
'use strict';
 | 
						|
 | 
						|
const Clone = require('@hapi/hoek/lib/clone');
 | 
						|
 | 
						|
const Common = require('./common');
 | 
						|
 | 
						|
 | 
						|
const internals = {
 | 
						|
    annotations: Symbol('annotations')
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
exports.error = function (stripColorCodes) {
 | 
						|
 | 
						|
    if (!this._original ||
 | 
						|
        typeof this._original !== 'object') {
 | 
						|
 | 
						|
        return this.details[0].message;
 | 
						|
    }
 | 
						|
 | 
						|
    const redFgEscape = stripColorCodes ? '' : '\u001b[31m';
 | 
						|
    const redBgEscape = stripColorCodes ? '' : '\u001b[41m';
 | 
						|
    const endColor = stripColorCodes ? '' : '\u001b[0m';
 | 
						|
 | 
						|
    const obj = Clone(this._original);
 | 
						|
 | 
						|
    for (let i = this.details.length - 1; i >= 0; --i) {        // Reverse order to process deepest child first
 | 
						|
        const pos = i + 1;
 | 
						|
        const error = this.details[i];
 | 
						|
        const path = error.path;
 | 
						|
        let node = obj;
 | 
						|
        for (let j = 0; ; ++j) {
 | 
						|
            const seg = path[j];
 | 
						|
 | 
						|
            if (Common.isSchema(node)) {
 | 
						|
                node = node.clone();                              // joi schemas are not cloned by hoek, we have to take this extra step
 | 
						|
            }
 | 
						|
 | 
						|
            if (j + 1 < path.length &&
 | 
						|
                typeof node[seg] !== 'string') {
 | 
						|
 | 
						|
                node = node[seg];
 | 
						|
            }
 | 
						|
            else {
 | 
						|
                const refAnnotations = node[internals.annotations] || { errors: {}, missing: {} };
 | 
						|
                node[internals.annotations] = refAnnotations;
 | 
						|
 | 
						|
                const cacheKey = seg || error.context.key;
 | 
						|
 | 
						|
                if (node[seg] !== undefined) {
 | 
						|
                    refAnnotations.errors[cacheKey] = refAnnotations.errors[cacheKey] || [];
 | 
						|
                    refAnnotations.errors[cacheKey].push(pos);
 | 
						|
                }
 | 
						|
                else {
 | 
						|
                    refAnnotations.missing[cacheKey] = pos;
 | 
						|
                }
 | 
						|
 | 
						|
                break;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    const replacers = {
 | 
						|
        key: /_\$key\$_([, \d]+)_\$end\$_"/g,
 | 
						|
        missing: /"_\$miss\$_([^|]+)\|(\d+)_\$end\$_": "__missing__"/g,
 | 
						|
        arrayIndex: /\s*"_\$idx\$_([, \d]+)_\$end\$_",?\n(.*)/g,
 | 
						|
        specials: /"\[(NaN|Symbol.*|-?Infinity|function.*|\(.*)]"/g
 | 
						|
    };
 | 
						|
 | 
						|
    let message = internals.safeStringify(obj, 2)
 | 
						|
        .replace(replacers.key, ($0, $1) => `" ${redFgEscape}[${$1}]${endColor}`)
 | 
						|
        .replace(replacers.missing, ($0, $1, $2) => `${redBgEscape}"${$1}"${endColor}${redFgEscape} [${$2}]: -- missing --${endColor}`)
 | 
						|
        .replace(replacers.arrayIndex, ($0, $1, $2) => `\n${$2} ${redFgEscape}[${$1}]${endColor}`)
 | 
						|
        .replace(replacers.specials, ($0, $1) => $1);
 | 
						|
 | 
						|
    message = `${message}\n${redFgEscape}`;
 | 
						|
 | 
						|
    for (let i = 0; i < this.details.length; ++i) {
 | 
						|
        const pos = i + 1;
 | 
						|
        message = `${message}\n[${pos}] ${this.details[i].message}`;
 | 
						|
    }
 | 
						|
 | 
						|
    message = message + endColor;
 | 
						|
 | 
						|
    return message;
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
// Inspired by json-stringify-safe
 | 
						|
 | 
						|
internals.safeStringify = function (obj, spaces) {
 | 
						|
 | 
						|
    return JSON.stringify(obj, internals.serializer(), spaces);
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
internals.serializer = function () {
 | 
						|
 | 
						|
    const keys = [];
 | 
						|
    const stack = [];
 | 
						|
 | 
						|
    const cycleReplacer = (key, value) => {
 | 
						|
 | 
						|
        if (stack[0] === value) {
 | 
						|
            return '[Circular ~]';
 | 
						|
        }
 | 
						|
 | 
						|
        return '[Circular ~.' + keys.slice(0, stack.indexOf(value)).join('.') + ']';
 | 
						|
    };
 | 
						|
 | 
						|
    return function (key, value) {
 | 
						|
 | 
						|
        if (stack.length > 0) {
 | 
						|
            const thisPos = stack.indexOf(this);
 | 
						|
            if (~thisPos) {
 | 
						|
                stack.length = thisPos + 1;
 | 
						|
                keys.length = thisPos + 1;
 | 
						|
                keys[thisPos] = key;
 | 
						|
            }
 | 
						|
            else {
 | 
						|
                stack.push(this);
 | 
						|
                keys.push(key);
 | 
						|
            }
 | 
						|
 | 
						|
            if (~stack.indexOf(value)) {
 | 
						|
                value = cycleReplacer.call(this, key, value);
 | 
						|
            }
 | 
						|
        }
 | 
						|
        else {
 | 
						|
            stack.push(value);
 | 
						|
        }
 | 
						|
 | 
						|
        if (value) {
 | 
						|
            const annotations = value[internals.annotations];
 | 
						|
            if (annotations) {
 | 
						|
                if (Array.isArray(value)) {
 | 
						|
                    const annotated = [];
 | 
						|
 | 
						|
                    for (let i = 0; i < value.length; ++i) {
 | 
						|
                        if (annotations.errors[i]) {
 | 
						|
                            annotated.push(`_$idx$_${annotations.errors[i].sort().join(', ')}_$end$_`);
 | 
						|
                        }
 | 
						|
 | 
						|
                        annotated.push(value[i]);
 | 
						|
                    }
 | 
						|
 | 
						|
                    value = annotated;
 | 
						|
                }
 | 
						|
                else {
 | 
						|
                    for (const errorKey in annotations.errors) {
 | 
						|
                        value[`${errorKey}_$key$_${annotations.errors[errorKey].sort().join(', ')}_$end$_`] = value[errorKey];
 | 
						|
                        value[errorKey] = undefined;
 | 
						|
                    }
 | 
						|
 | 
						|
                    for (const missingKey in annotations.missing) {
 | 
						|
                        value[`_$miss$_${missingKey}|${annotations.missing[missingKey]}_$end$_`] = '__missing__';
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                return value;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        if (value === Infinity ||
 | 
						|
            value === -Infinity ||
 | 
						|
            Number.isNaN(value) ||
 | 
						|
            typeof value === 'function' ||
 | 
						|
            typeof value === 'symbol') {
 | 
						|
 | 
						|
            return '[' + value.toString() + ']';
 | 
						|
        }
 | 
						|
 | 
						|
        return value;
 | 
						|
    };
 | 
						|
};
 |