eslint-plugin-sonarjs
Version:
SonarJS rules for ESLint
101 lines (100 loc) • 3.92 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/S5856/javascript
Object.defineProperty(exports, "__esModule", { value: true });
exports.rule = void 0;
const index_js_1 = require("../helpers/index.js");
const regexpp_1 = require("@eslint-community/regexpp");
const meta_js_1 = require("./meta.js");
const validator = new regexpp_1.RegExpValidator();
exports.rule = {
meta: (0, index_js_1.generateMeta)(meta_js_1.meta),
create(context) {
function getFlags(node) {
if (node.arguments.length < 2) {
return '';
}
if ((0, index_js_1.isStringLiteral)(node.arguments[1])) {
return node.arguments[1].value;
}
return null;
}
function validateRegExpPattern(pattern, uFlag) {
try {
validator.validatePattern(pattern, undefined, undefined, uFlag);
return null;
}
catch (err) {
return err.message;
}
}
function validateRegExpFlags(flags) {
try {
validator.validateFlags(flags);
return null;
}
catch {
return `Invalid flags supplied to RegExp constructor '${flags}'`;
}
}
function isRegExpConstructor(call) {
const { callee } = call;
return callee.type === 'Identifier' && callee.name === 'RegExp';
}
function isStringMatch(call) {
const services = context.sourceCode.parserServices;
if (!(0, index_js_1.isRequiredParserServices)(services)) {
return false;
}
const { callee } = call;
return (callee.type === 'MemberExpression' &&
(0, index_js_1.isStringType)((0, index_js_1.getTypeFromTreeNode)(callee.object, services)) &&
(0, index_js_1.isIdentifier)(callee.property, 'match'));
}
function getPattern(call) {
if ((0, index_js_1.isStringLiteral)(call.arguments[0])) {
return call.arguments[0].value;
}
return null;
}
return {
'CallExpression, NewExpression'(node) {
const call = node;
if (!isRegExpConstructor(call) && !isStringMatch(call)) {
return;
}
const pattern = getPattern(call);
if (!pattern) {
return;
}
const flags = getFlags(call);
const message = (flags && validateRegExpFlags(flags)) ||
// If flags are unknown, report the regex only if its pattern is invalid both with and without the "u" flag
(flags === null
? validateRegExpPattern(pattern, true) && validateRegExpPattern(pattern, false)
: validateRegExpPattern(pattern, flags.includes('u')));
if (message) {
context.report({
node,
message,
});
}
},
};
},
};
;