/** * @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, }) } }