You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
183 lines
4.5 KiB
183 lines
4.5 KiB
/** |
|
* @author Yosuke Ota |
|
* See LICENSE file in root directory for full license. |
|
*/ |
|
'use strict' |
|
|
|
const utils = require('../utils') |
|
const regexp = require('../utils/regexp') |
|
|
|
/** |
|
* @typedef {import('../utils').ComponentProp} ComponentProp |
|
*/ |
|
|
|
/** |
|
* @typedef {object} ParsedOption |
|
* @property { (name: string) => boolean } test |
|
* @property {string|undefined} [message] |
|
* @property {string|undefined} [suggest] |
|
*/ |
|
|
|
/** |
|
* @param {string} str |
|
* @returns {(str: string) => boolean} |
|
*/ |
|
function buildMatcher(str) { |
|
if (regexp.isRegExp(str)) { |
|
const re = regexp.toRegExp(str) |
|
return (s) => { |
|
re.lastIndex = 0 |
|
return re.test(s) |
|
} |
|
} |
|
return (s) => s === str |
|
} |
|
/** |
|
* @param {string|{name:string, message?: string, suggest?:string}} option |
|
* @returns {ParsedOption} |
|
*/ |
|
function parseOption(option) { |
|
if (typeof option === 'string') { |
|
const matcher = buildMatcher(option) |
|
return { |
|
test(name) { |
|
return matcher(name) |
|
} |
|
} |
|
} |
|
const parsed = parseOption(option.name) |
|
parsed.message = option.message |
|
parsed.suggest = option.suggest |
|
return parsed |
|
} |
|
|
|
module.exports = { |
|
meta: { |
|
hasSuggestions: true, |
|
type: 'suggestion', |
|
docs: { |
|
description: 'disallow specific props', |
|
categories: undefined, |
|
url: 'https://eslint.vuejs.org/rules/no-restricted-props.html' |
|
}, |
|
fixable: null, |
|
schema: { |
|
type: 'array', |
|
items: { |
|
oneOf: [ |
|
{ type: ['string'] }, |
|
{ |
|
type: 'object', |
|
properties: { |
|
name: { type: 'string' }, |
|
message: { type: 'string', minLength: 1 }, |
|
suggest: { type: 'string' } |
|
}, |
|
required: ['name'], |
|
additionalProperties: false |
|
} |
|
] |
|
}, |
|
uniqueItems: true, |
|
minItems: 0 |
|
}, |
|
|
|
messages: { |
|
// eslint-disable-next-line eslint-plugin/report-message-format |
|
restrictedProp: '{{message}}', |
|
instead: 'Instead, change to `{{suggest}}`.' |
|
} |
|
}, |
|
/** @param {RuleContext} context */ |
|
create(context) { |
|
/** @type {ParsedOption[]} */ |
|
const options = context.options.map(parseOption) |
|
|
|
/** |
|
* @param {ComponentProp[]} props |
|
* @param { { [key: string]: Property | undefined } } [withDefaultsProps] |
|
*/ |
|
function processProps(props, withDefaultsProps) { |
|
for (const prop of props) { |
|
if (!prop.propName) { |
|
continue |
|
} |
|
|
|
for (const option of options) { |
|
if (option.test(prop.propName)) { |
|
const message = |
|
option.message || |
|
`Using \`${prop.propName}\` props is not allowed.` |
|
context.report({ |
|
node: prop.key, |
|
messageId: 'restrictedProp', |
|
data: { message }, |
|
suggest: createSuggest( |
|
prop.key, |
|
option, |
|
withDefaultsProps && withDefaultsProps[prop.propName] |
|
) |
|
}) |
|
break |
|
} |
|
} |
|
} |
|
} |
|
return utils.compositingVisitors( |
|
utils.defineScriptSetupVisitor(context, { |
|
onDefinePropsEnter(node, props) { |
|
processProps(props, utils.getWithDefaultsProps(node)) |
|
} |
|
}), |
|
utils.defineVueVisitor(context, { |
|
onVueObjectEnter(node) { |
|
processProps(utils.getComponentPropsFromOptions(node)) |
|
} |
|
}) |
|
) |
|
} |
|
} |
|
|
|
/** |
|
* @param {Expression} node |
|
* @param {ParsedOption} option |
|
* @param {Property} [withDefault] |
|
* @returns {Rule.SuggestionReportDescriptor[]} |
|
*/ |
|
function createSuggest(node, option, withDefault) { |
|
if (!option.suggest) { |
|
return [] |
|
} |
|
|
|
/** @type {string} */ |
|
let replaceText |
|
if (node.type === 'Literal' || node.type === 'TemplateLiteral') { |
|
replaceText = JSON.stringify(option.suggest) |
|
} else if (node.type === 'Identifier') { |
|
replaceText = /^[a-z]\w*$/iu.exec(option.suggest) |
|
? option.suggest |
|
: JSON.stringify(option.suggest) |
|
} else { |
|
return [] |
|
} |
|
|
|
return [ |
|
{ |
|
fix(fixer) { |
|
const fixes = [fixer.replaceText(node, replaceText)] |
|
if (withDefault) { |
|
if (withDefault.shorthand) { |
|
fixes.push( |
|
fixer.insertTextBefore(withDefault.value, `${replaceText}:`) |
|
) |
|
} else { |
|
fixes.push(fixer.replaceText(withDefault.key, replaceText)) |
|
} |
|
} |
|
return fixes.sort((a, b) => a.range[0] - b.range[0]) |
|
}, |
|
messageId: 'instead', |
|
data: { suggest: option.suggest } |
|
} |
|
] |
|
}
|
|
|