stylelint
Version:
A mighty CSS linter that helps you avoid errors and enforce conventions.
82 lines (64 loc) • 2.46 kB
JavaScript
import { isTokenString, tokenize } from '@csstools/css-tokenizer';
import { fork } from 'css-tree';
import { atRuleRegexes, descriptorRegexes } from '../../utils/regexes.mjs';
import { declarationValueIndex } from '../../utils/nodeFieldIndices.mjs';
import report from '../../utils/report.mjs';
import ruleMessages from '../../utils/ruleMessages.mjs';
import validateOptions from '../../utils/validateOptions.mjs';
const ruleName = 'syntax-string-no-invalid';
const messages = ruleMessages(ruleName, {
rejected: (syntaxString) => `Unexpected invalid syntax string "${syntaxString}"`,
});
const meta = {
url: 'https://stylelint.io/user-guide/rules/syntax-string-no-invalid',
};
/** @type {import('stylelint').CoreRules[ruleName]} */
const rule = (primary) => {
return (root, result) => {
const validOptions = validateOptions(result, ruleName, { actual: primary });
if (!validOptions) {
return;
}
const forkedLexer = fork({
properties: {
'-stylelint-syntax': "'*' | <syntax-component> [ <syntax-combinator> <syntax-component> ]*",
},
types: {
'syntax-component':
"<syntax-single-component> <syntax-multiplier>? | '<' transform-list '>'",
'syntax-single-component': "'<' <syntax-type-name> '>' | <ident>",
'syntax-type-name':
'angle | color | custom-ident | image | integer | length | length-percentage | number | percentage | resolution | string | time | url | transform-function',
'syntax-combinator': "'|'",
'syntax-multiplier': "[ '#' | '+' ]",
},
}).lexer;
root.walkAtRules(atRuleRegexes.propertyName, (atRule) => {
atRule.walkDecls(descriptorRegexes.syntaxName, (decl) => {
const tokens = tokenize({ css: decl.value }).filter((token) => isTokenString(token));
tokens.forEach((token) => {
const { error } = forkedLexer.matchProperty('-stylelint-syntax', token[4].value);
if (!error) return;
if (!('mismatchLength' in error)) return;
const { name, rawMessage } = error;
if (name !== 'SyntaxMatchError') return;
if (rawMessage !== 'Mismatch') return;
const valueIndex = declarationValueIndex(decl);
report({
message: messages.rejected,
messageArgs: [token[1]],
node: decl,
index: valueIndex + token[2],
endIndex: valueIndex + token[3] + 1,
result,
ruleName,
});
});
});
});
};
};
rule.ruleName = ruleName;
rule.messages = messages;
rule.meta = meta;
export default rule;