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.
 
 
 
 

201 lines
6.5 KiB

/**
* @author Toru Nagashima <https://github.com/mysticatea>
* See LICENSE file in root directory for full license.
*/
"use strict"
const getLinters = require("../internal/get-linters")
const { toRuleIdLocation } = require("../internal/utils")
const quotedName = /'(.+?)'/u
/**
* Get the severity of a given rule.
* @param {object} config The config object to check.
* @param {string} ruleId The rule ID to check.
* @returns {number} The severity of the rule.
*/
function getSeverity(config, ruleId) {
const rules = config && config.rules
const ruleOptions = rules && rules[ruleId]
const severity = Array.isArray(ruleOptions) ? ruleOptions[0] : ruleOptions
switch (severity) {
case 2:
case "error":
return 2
case 1:
case "warn":
return 1
default:
return 0
}
}
/**
* Get the comment which is at a given message location.
* @param {Message} message The message to get.
* @param {SourceCode|undefined} sourceCode The source code object to get.
* @returns {Comment|undefined} The gotten comment.
*/
function getCommentAt(message, sourceCode) {
if (sourceCode != null) {
const loc = { line: message.line, column: message.column - 1 }
const index = sourceCode.getIndexFromLoc(loc)
const options = { includeComments: true }
const comment = sourceCode.getTokenByRangeStart(index, options)
if (
comment != null &&
(comment.type === "Line" || comment.type === "Block")
) {
return comment
}
}
return undefined
}
/**
* Check whether a given message is a `reportUnusedDisableDirectives` error.
* @param {Message} message The message.
* @returns {boolean} `true` if the message is a `reportUnusedDisableDirectives` error.
*/
function isUnusedDisableDirectiveError(message) {
return (
!message.fatal &&
!message.ruleId &&
message.message.includes("eslint-disable")
)
}
/**
* Create `eslint-comments/no-unused-disable` error.
* @param {string} ruleId The ruleId.
* @param {number} severity The severity of the rule.
* @param {Message} message The original message.
* @param {Comment|undefined} comment The directive comment.
* @returns {Message} The created error.
*/
function createNoUnusedDisableError(ruleId, severity, message, comment) {
const clone = Object.assign({}, message)
const match = quotedName.exec(message.message)
const targetRuleId = match && match[1]
clone.ruleId = ruleId
clone.severity = severity
clone.message = targetRuleId
? `'${targetRuleId}' rule is disabled but never reported.`
: "ESLint rules are disabled but never reported."
clone.suggestions = []
if (comment != null) {
if (targetRuleId) {
const loc = toRuleIdLocation(comment, targetRuleId)
clone.line = loc.start.line
clone.column = loc.start.column + 1
clone.endLine = loc.end.line
clone.endColumn = loc.end.column + 1
} else {
clone.endLine = comment.loc.end.line
clone.endColumn = comment.loc.end.column + 1
}
// Remove the whole node if it is the only rule, otherwise
// don't try to fix because it is quite complicated.
if (!comment.value.includes(",") && !comment.value.includes("--")) {
// We can't use the typical `fixer` helper because we are injecting
// this message after the fixes are resolved.
clone.suggestions = [
{
desc: "Remove `eslint-disable` comment.",
fix: {
range: comment.range,
text: comment.value.includes("\n") ? "\n" : "",
},
},
]
}
}
return clone
}
/**
* Convert `reportUnusedDisableDirectives` errors to `eslint-comments/no-unused-disable` errors.
* @param {Message[]} messages The original messages.
* @param {SourceCode|undefined} sourceCode The source code object.
* @param {string} ruleId The rule ID to convert.
* @param {number} severity The severity of the rule.
* @param {boolean} keepAsIs The flag to keep original errors as is.
* @returns {Message[]} The converted messages.
*/
function convert(messages, sourceCode, ruleId, severity, keepAsIs) {
for (let i = messages.length - 1; i >= 0; --i) {
const message = messages[i]
if (!isUnusedDisableDirectiveError(message)) {
continue
}
const newMessage = createNoUnusedDisableError(
ruleId,
severity,
message,
getCommentAt(message, sourceCode)
)
if (keepAsIs) {
messages.splice(i + 1, 0, newMessage)
} else {
messages.splice(i, 1, newMessage)
}
}
return messages
}
module.exports = (ruleId = "eslint-comments/no-unused-disable") => {
for (const Linter of getLinters()) {
const verify0 = Linter.prototype._verifyWithoutProcessors
Object.defineProperty(Linter.prototype, "_verifyWithoutProcessors", {
value: function _verifyWithoutProcessors(
textOrSourceCode,
config,
filenameOrOptions
) {
const severity = getSeverity(config, ruleId)
if (severity === 0) {
return verify0.call(
this,
textOrSourceCode,
config,
filenameOrOptions
)
}
const options =
typeof filenameOrOptions === "string"
? { filename: filenameOrOptions }
: filenameOrOptions || {}
const reportUnusedDisableDirectives = Boolean(
options.reportUnusedDisableDirectives
)
const messages = verify0.call(
this,
textOrSourceCode,
config,
Object.assign({}, options, {
reportUnusedDisableDirectives: true,
})
)
return convert(
messages,
this.getSourceCode(),
ruleId,
severity,
reportUnusedDisableDirectives
)
},
configurable: true,
writable: true,
})
}
}