1
0
mirror of https://github.com/thangisme/notes.git synced 2024-09-27 08:05:54 -04:00
notes/node_modules/stylelint/lib/utils/validateOptions.js
Patrick Marsceill b7b0d0d7bf
Initial commit
2017-03-09 13:16:08 -05:00

133 lines
3.7 KiB
JavaScript

/* @flow */
"use strict"
const _ = require("lodash")
const ignoredOptions = [ "severity", "message" ]
/**
* Validate a rule's options.
*
* See existing rules for examples.
*
* @param {Result} result - postcss result
* @param {string} ruleName
* @param {...object} ...optionDescriptions - Each optionDescription can
* have the following properties:
* - `actual` (required): the actual passed option value or object.
* - `possible` (required): a schema representation of what values are
* valid for those options. `possible` should be an object if the
* options are an object, with corresponding keys; if the options are not an
* object, `possible` isn't, either. All `possible` value representations
* should be **arrays of either values or functions**. Values are === checked
* against `actual`. Functions are fed `actual` as an argument and their
* return value is interpreted: truthy = valid, falsy = invalid.
* - `optional` (optional): If this is `true`, `actual` can be undefined.
* @return {boolean} Whether or not the options are valid (true = valid)
*/
module.exports = function (
result/*: Object*/,
ruleName/*: string*/
)/*: boolean*/ {
let noErrors = true
const optionDescriptions = Array.from(arguments).slice(2)
optionDescriptions.forEach(optionDescription => {
validate(optionDescription, ruleName, complain)
})
function complain(message) {
noErrors = false
result.warn(message, {
stylelintType: "invalidOption",
})
_.set(result, "stylelint.stylelintError", true)
}
return noErrors
}
function validate(opts, ruleName, complain) {
const possible = opts.possible
const actual = opts.actual
const optional = opts.optional
if (actual === null || _.isEqual(actual, [null])) {
return
}
const nothingPossible = possible === undefined || Array.isArray(possible) && possible.length === 0
if (nothingPossible && actual === true) {
return
}
if (actual === undefined) {
if (nothingPossible || optional) {
return
}
complain(`Expected option value for rule "${ruleName}"`)
return
} else if (nothingPossible) {
complain(`Unexpected option value "${actual}" for rule "${ruleName}"`)
return
}
// If `possible` is a function ...
if (_.isFunction(possible)) {
if (!possible(actual)) {
complain(`Invalid option "${JSON.stringify(actual)}" for rule ${ruleName}`)
}
return
}
// If `possible` is an array instead of an object ...
if (!_.isPlainObject(possible)) {
[].concat(actual).forEach(a => {
if (isValid(possible, a)) {
return
}
complain(`Invalid option value "${a}" for rule "${ruleName}"`)
})
return
}
// If possible is an object ...
if (!_.isPlainObject(actual)) {
complain(`Invalid option value ${JSON.stringify(actual)} for rule "${ruleName}": ` + "should be an object")
return
}
Object.keys(actual).forEach(optionName => {
if (ignoredOptions.indexOf(optionName) !== -1) {
return
}
if (!possible[optionName]) {
complain(`Invalid option name "${optionName}" for rule "${ruleName}"`)
return
}
const actualOptionValue = actual[optionName];[].concat(actualOptionValue).forEach(a => {
if (isValid(possible[optionName], a)) {
return
}
complain(`Invalid value "${a}" for option "${optionName}" of rule "${ruleName}"`)
})
})
}
function isValid(possible, actual) {
const possibleList = [].concat(possible)
for (let i = 0, l = possibleList.length; i < l; i++) {
const possibility = possibleList[i]
if (typeof possibility === "function" && possibility(actual)) {
return true
}
if (actual === possibility) {
return true
}
}
}