| /** |
| * @fileoverview A rule to choose between single and double quote marks |
| * @author Matt DuVall <http://www.mattduvall.com/>, Brandon Payton |
| */ |
| |
| "use strict"; |
| |
| //------------------------------------------------------------------------------ |
| // Requirements |
| //------------------------------------------------------------------------------ |
| |
| const astUtils = require("./utils/ast-utils"); |
| |
| //------------------------------------------------------------------------------ |
| // Constants |
| //------------------------------------------------------------------------------ |
| |
| const QUOTE_SETTINGS = { |
| double: { |
| quote: "\"", |
| alternateQuote: "'", |
| description: "doublequote" |
| }, |
| single: { |
| quote: "'", |
| alternateQuote: "\"", |
| description: "singlequote" |
| }, |
| backtick: { |
| quote: "`", |
| alternateQuote: "\"", |
| description: "backtick" |
| } |
| }; |
| |
| // An unescaped newline is a newline preceded by an even number of backslashes. |
| const UNESCAPED_LINEBREAK_PATTERN = new RegExp(String.raw`(^|[^\\])(\\\\)*[${Array.from(astUtils.LINEBREAKS).join("")}]`, "u"); |
| |
| /** |
| * Switches quoting of javascript string between ' " and ` |
| * escaping and unescaping as necessary. |
| * Only escaping of the minimal set of characters is changed. |
| * Note: escaping of newlines when switching from backtick to other quotes is not handled. |
| * @param {string} str - A string to convert. |
| * @returns {string} The string with changed quotes. |
| * @private |
| */ |
| QUOTE_SETTINGS.double.convert = |
| QUOTE_SETTINGS.single.convert = |
| QUOTE_SETTINGS.backtick.convert = function(str) { |
| const newQuote = this.quote; |
| const oldQuote = str[0]; |
| |
| if (newQuote === oldQuote) { |
| return str; |
| } |
| return newQuote + str.slice(1, -1).replace(/\\(\$\{|\r\n?|\n|.)|["'`]|\$\{|(\r\n?|\n)/gu, (match, escaped, newline) => { |
| if (escaped === oldQuote || oldQuote === "`" && escaped === "${") { |
| return escaped; // unescape |
| } |
| if (match === newQuote || newQuote === "`" && match === "${") { |
| return `\\${match}`; // escape |
| } |
| if (newline && oldQuote === "`") { |
| return "\\n"; // escape newlines |
| } |
| return match; |
| }) + newQuote; |
| }; |
| |
| const AVOID_ESCAPE = "avoid-escape"; |
| |
| //------------------------------------------------------------------------------ |
| // Rule Definition |
| //------------------------------------------------------------------------------ |
| |
| module.exports = { |
| meta: { |
| type: "layout", |
| |
| docs: { |
| description: "enforce the consistent use of either backticks, double, or single quotes", |
| category: "Stylistic Issues", |
| recommended: false, |
| url: "https://eslint.org/docs/rules/quotes" |
| }, |
| |
| fixable: "code", |
| |
| schema: [ |
| { |
| enum: ["single", "double", "backtick"] |
| }, |
| { |
| anyOf: [ |
| { |
| enum: ["avoid-escape"] |
| }, |
| { |
| type: "object", |
| properties: { |
| avoidEscape: { |
| type: "boolean" |
| }, |
| allowTemplateLiterals: { |
| type: "boolean" |
| } |
| }, |
| additionalProperties: false |
| } |
| ] |
| } |
| ] |
| }, |
| |
| create(context) { |
| |
| const quoteOption = context.options[0], |
| settings = QUOTE_SETTINGS[quoteOption || "double"], |
| options = context.options[1], |
| allowTemplateLiterals = options && options.allowTemplateLiterals === true, |
| sourceCode = context.getSourceCode(); |
| let avoidEscape = options && options.avoidEscape === true; |
| |
| // deprecated |
| if (options === AVOID_ESCAPE) { |
| avoidEscape = true; |
| } |
| |
| /** |
| * Determines if a given node is part of JSX syntax. |
| * |
| * This function returns `true` in the following cases: |
| * |
| * - `<div className="foo"></div>` ... If the literal is an attribute value, the parent of the literal is `JSXAttribute`. |
| * - `<div>foo</div>` ... If the literal is a text content, the parent of the literal is `JSXElement`. |
| * - `<>foo</>` ... If the literal is a text content, the parent of the literal is `JSXFragment`. |
| * |
| * In particular, this function returns `false` in the following cases: |
| * |
| * - `<div className={"foo"}></div>` |
| * - `<div>{"foo"}</div>` |
| * |
| * In both cases, inside of the braces is handled as normal JavaScript. |
| * The braces are `JSXExpressionContainer` nodes. |
| * |
| * @param {ASTNode} node The Literal node to check. |
| * @returns {boolean} True if the node is a part of JSX, false if not. |
| * @private |
| */ |
| function isJSXLiteral(node) { |
| return node.parent.type === "JSXAttribute" || node.parent.type === "JSXElement" || node.parent.type === "JSXFragment"; |
| } |
| |
| /** |
| * Checks whether or not a given node is a directive. |
| * The directive is a `ExpressionStatement` which has only a string literal. |
| * @param {ASTNode} node - A node to check. |
| * @returns {boolean} Whether or not the node is a directive. |
| * @private |
| */ |
| function isDirective(node) { |
| return ( |
| node.type === "ExpressionStatement" && |
| node.expression.type === "Literal" && |
| typeof node.expression.value === "string" |
| ); |
| } |
| |
| /** |
| * Checks whether or not a given node is a part of directive prologues. |
| * See also: http://www.ecma-international.org/ecma-262/6.0/#sec-directive-prologues-and-the-use-strict-directive |
| * @param {ASTNode} node - A node to check. |
| * @returns {boolean} Whether or not the node is a part of directive prologues. |
| * @private |
| */ |
| function isPartOfDirectivePrologue(node) { |
| const block = node.parent.parent; |
| |
| if (block.type !== "Program" && (block.type !== "BlockStatement" || !astUtils.isFunction(block.parent))) { |
| return false; |
| } |
| |
| // Check the node is at a prologue. |
| for (let i = 0; i < block.body.length; ++i) { |
| const statement = block.body[i]; |
| |
| if (statement === node.parent) { |
| return true; |
| } |
| if (!isDirective(statement)) { |
| break; |
| } |
| } |
| |
| return false; |
| } |
| |
| /** |
| * Checks whether or not a given node is allowed as non backtick. |
| * @param {ASTNode} node - A node to check. |
| * @returns {boolean} Whether or not the node is allowed as non backtick. |
| * @private |
| */ |
| function isAllowedAsNonBacktick(node) { |
| const parent = node.parent; |
| |
| switch (parent.type) { |
| |
| // Directive Prologues. |
| case "ExpressionStatement": |
| return isPartOfDirectivePrologue(node); |
| |
| // LiteralPropertyName. |
| case "Property": |
| case "MethodDefinition": |
| return parent.key === node && !parent.computed; |
| |
| // ModuleSpecifier. |
| case "ImportDeclaration": |
| case "ExportNamedDeclaration": |
| case "ExportAllDeclaration": |
| return parent.source === node; |
| |
| // Others don't allow. |
| default: |
| return false; |
| } |
| } |
| |
| /** |
| * Checks whether or not a given TemplateLiteral node is actually using any of the special features provided by template literal strings. |
| * @param {ASTNode} node - A TemplateLiteral node to check. |
| * @returns {boolean} Whether or not the TemplateLiteral node is using any of the special features provided by template literal strings. |
| * @private |
| */ |
| function isUsingFeatureOfTemplateLiteral(node) { |
| const hasTag = node.parent.type === "TaggedTemplateExpression" && node === node.parent.quasi; |
| |
| if (hasTag) { |
| return true; |
| } |
| |
| const hasStringInterpolation = node.expressions.length > 0; |
| |
| if (hasStringInterpolation) { |
| return true; |
| } |
| |
| const isMultilineString = node.quasis.length >= 1 && UNESCAPED_LINEBREAK_PATTERN.test(node.quasis[0].value.raw); |
| |
| if (isMultilineString) { |
| return true; |
| } |
| |
| return false; |
| } |
| |
| return { |
| |
| Literal(node) { |
| const val = node.value, |
| rawVal = node.raw; |
| |
| if (settings && typeof val === "string") { |
| let isValid = (quoteOption === "backtick" && isAllowedAsNonBacktick(node)) || |
| isJSXLiteral(node) || |
| astUtils.isSurroundedBy(rawVal, settings.quote); |
| |
| if (!isValid && avoidEscape) { |
| isValid = astUtils.isSurroundedBy(rawVal, settings.alternateQuote) && rawVal.indexOf(settings.quote) >= 0; |
| } |
| |
| if (!isValid) { |
| context.report({ |
| node, |
| message: "Strings must use {{description}}.", |
| data: { |
| description: settings.description |
| }, |
| fix(fixer) { |
| return fixer.replaceText(node, settings.convert(node.raw)); |
| } |
| }); |
| } |
| } |
| }, |
| |
| TemplateLiteral(node) { |
| |
| // Don't throw an error if backticks are expected or a template literal feature is in use. |
| if ( |
| allowTemplateLiterals || |
| quoteOption === "backtick" || |
| isUsingFeatureOfTemplateLiteral(node) |
| ) { |
| return; |
| } |
| |
| context.report({ |
| node, |
| message: "Strings must use {{description}}.", |
| data: { |
| description: settings.description |
| }, |
| fix(fixer) { |
| if (isPartOfDirectivePrologue(node)) { |
| |
| /* |
| * TemplateLiterals in a directive prologue aren't actually directives, but if they're |
| * in the directive prologue, then fixing them might turn them into directives and change |
| * the behavior of the code. |
| */ |
| return null; |
| } |
| return fixer.replaceText(node, settings.convert(sourceCode.getText(node))); |
| } |
| }); |
| } |
| }; |
| |
| } |
| }; |