vgno/roc-config

View on GitHub
src/validation/validateSettings.js

Summary

Maintainability
A
1 hr
Test Coverage
import { isPlainObject, isFunction } from 'lodash';

import isValid from './helpers/isValid';
import throwValidationError from './helpers/throwValidationError';
import { REQUIRED_ERROR } from './validators/required';

/**
 * Validates the provided settings objects and might throw.
 */
export default function validateSettings(settings = {}, meta = {}, allowRequiredFailure = false, path = 'settings') {
    const validateKeys = Object.keys(meta);

    for (const validateKey of validateKeys) {
        const configValue = settings[validateKey];
        const validator = meta[validateKey].validator && isFunction(meta[validateKey].validator) ?
            meta[validateKey].validator :
            undefined;
        const newPath = path ? `${path}.${validateKey}` : validateKey;

        // process validation nodes recursively
        if (isPlainObject(configValue) && !validator) {
            validateSettings(configValue, meta[validateKey], allowRequiredFailure, newPath);
        } else {
            assertValid(configValue, newPath, validator, allowRequiredFailure);
        }
    }
}

function assertValid(value, key, validator, allowRequiredFailure = false) {
    const result = isValid(value, validator);
    if (
        !(allowRequiredFailure && result === REQUIRED_ERROR) &&
        result !== true
    ) {
        throwValidationError(...processResult(key, result, value), 'property');
    }
}

function processResult(key, result, value) {
    if (isPlainObject(result)) {
        return [
            `${key}${result.key}`,
            result.message,
            result.value,
        ];
    }
    return [
        key,
        result,
        value,
    ];
}