@html-eslint/eslint-plugin
Version:
ESLint plugin for HTML
94 lines (86 loc) • 2.22 kB
JavaScript
/**
* @import {
* Attribute,
* ScriptTag,
* StyleTag,
* Tag
* } from "@html-eslint/types"
* @import {
* RuleModule,
* SuggestionReportDescriptor
* } from "../types"
*/
const { RULE_CATEGORY } = require("../constants");
const { createVisitors } = require("./utils/visitors");
const { getRuleUrl } = require("./utils/rule");
const { hasTemplate } = require("./utils/node");
const MESSAGE_IDS = {
DUPLICATE_ATTRS: "duplicateAttrs",
REMOVE_ATTR: "removeAttr",
};
/** @type {RuleModule<[]>} */
module.exports = {
meta: {
type: "code",
docs: {
description: "Disallow duplicate attributes",
category: RULE_CATEGORY.BEST_PRACTICE,
recommended: true,
url: getRuleUrl("no-duplicate-attrs"),
},
fixable: null,
hasSuggestions: true,
schema: [],
messages: {
[MESSAGE_IDS.DUPLICATE_ATTRS]:
"The attribute '{{attrName}}' is duplicated.",
[MESSAGE_IDS.REMOVE_ATTR]:
"Remove this duplicate '{{attrName}}' attribute.",
},
},
create(context) {
/**
* @param {Attribute} node
* @returns {SuggestionReportDescriptor[]}
*/
function getSuggestions(node) {
return [
{
messageId: MESSAGE_IDS.REMOVE_ATTR,
fix: (fixer) => fixer.removeRange(node.range),
data: {
attrName: node.key.value,
},
},
];
}
/** @param {Tag | StyleTag | ScriptTag} node */
function check(node) {
if (Array.isArray(node.attributes)) {
const attrsSet = new Set();
node.attributes.forEach((attr) => {
if (hasTemplate(attr.key)) {
return;
}
if (attrsSet.has(attr.key.value.toLowerCase())) {
context.report({
node: attr,
data: {
attrName: attr.key.value,
},
messageId: MESSAGE_IDS.DUPLICATE_ATTRS,
suggest: getSuggestions(attr),
});
} else {
attrsSet.add(attr.key.value.toLowerCase());
}
});
}
}
return createVisitors(context, {
Tag: check,
StyleTag: check,
ScriptTag: check,
});
},
};