eslint-plugin-sonarjs
Version:
SonarJS rules for ESLint
97 lines (96 loc) • 3.96 kB
JavaScript
/*
* SonarQube JavaScript Plugin
* Copyright (C) 2011-2025 SonarSource SA
* mailto:info AT sonarsource DOT com
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the Sonar Source-Available License Version 1, as published by SonarSource SA.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
* See the Sonar Source-Available License for more details.
*
* You should have received a copy of the Sonar Source-Available License
* along with this program; if not, see https://sonarsource.com/license/ssal/
*/
// https://sonarsource.github.io/rspec/#/rspec/S1192
Object.defineProperty(exports, "__esModule", { value: true });
exports.rule = void 0;
const index_js_1 = require("../helpers/index.js");
const meta_js_1 = require("./meta.js");
// Number of times a literal must be duplicated to trigger an issue
const MIN_LENGTH = 10;
const NO_SEPARATOR_REGEXP = /^\w*$/;
const EXCLUDED_CONTEXTS = [
'ImportDeclaration',
'ImportExpression',
'JSXAttribute',
'ExportAllDeclaration',
'ExportNamedDeclaration',
];
const message = 'Define a constant instead of duplicating this literal {{times}} times.';
const DEFAULT_OPTIONS = {
threshold: 3,
ignoreStrings: 'application/json',
};
const messages = {
defineConstant: message,
};
exports.rule = {
meta: (0, index_js_1.generateMeta)(meta_js_1.meta, { messages, schema: meta_js_1.schema }, true),
create(context) {
const literalsByValue = new Map();
const { threshold, ignoreStrings } = {
...DEFAULT_OPTIONS,
...context.options[0],
};
const whitelist = ignoreStrings.split(',');
return {
Literal: (node) => {
const literal = node;
const { parent } = literal;
if (typeof literal.value === 'string' &&
parent &&
!['ExpressionStatement', 'TSLiteralType'].includes(parent.type)) {
const stringContent = literal.value.trim();
if (!whitelist.includes(literal.value) &&
!isExcludedByUsageContext(context, literal) &&
stringContent.length >= MIN_LENGTH &&
!NO_SEPARATOR_REGEXP.exec(stringContent)) {
const sameStringLiterals = literalsByValue.get(stringContent) || [];
sameStringLiterals.push(literal);
literalsByValue.set(stringContent, sameStringLiterals);
}
}
},
'Program:exit'() {
literalsByValue.forEach(literals => {
if (literals.length >= threshold) {
const [primaryNode, ...secondaryNodes] = literals;
const secondaryIssues = secondaryNodes.map(node => (0, index_js_1.toSecondaryLocation)(node, 'Duplication'));
(0, index_js_1.report)(context, {
message,
node: primaryNode,
data: { times: literals.length.toString() },
}, secondaryIssues);
}
});
},
};
},
};
function isExcludedByUsageContext(context, literal) {
const { parent } = literal;
const parentType = parent.type;
return (EXCLUDED_CONTEXTS.includes(parentType) ||
isRequireContext(parent, context) ||
isObjectPropertyKey(parent, literal));
}
function isRequireContext(parent, context) {
return (parent.type === 'CallExpression' && context.sourceCode.getText(parent.callee) === 'require');
}
function isObjectPropertyKey(parent, literal) {
return parent.type === 'Property' && parent.key === literal;
}
;