You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

352 lines
12 KiB

4 years ago
  1. /**
  2. * @fileoverview Rule to require or disallow yoda comparisons
  3. * @author Nicholas C. Zakas
  4. */
  5. "use strict";
  6. //--------------------------------------------------------------------------
  7. // Requirements
  8. //--------------------------------------------------------------------------
  9. const astUtils = require("./utils/ast-utils");
  10. //--------------------------------------------------------------------------
  11. // Helpers
  12. //--------------------------------------------------------------------------
  13. /**
  14. * Determines whether an operator is a comparison operator.
  15. * @param {string} operator The operator to check.
  16. * @returns {boolean} Whether or not it is a comparison operator.
  17. */
  18. function isComparisonOperator(operator) {
  19. return /^(==|===|!=|!==|<|>|<=|>=)$/u.test(operator);
  20. }
  21. /**
  22. * Determines whether an operator is an equality operator.
  23. * @param {string} operator The operator to check.
  24. * @returns {boolean} Whether or not it is an equality operator.
  25. */
  26. function isEqualityOperator(operator) {
  27. return /^(==|===)$/u.test(operator);
  28. }
  29. /**
  30. * Determines whether an operator is one used in a range test.
  31. * Allowed operators are `<` and `<=`.
  32. * @param {string} operator The operator to check.
  33. * @returns {boolean} Whether the operator is used in range tests.
  34. */
  35. function isRangeTestOperator(operator) {
  36. return ["<", "<="].indexOf(operator) >= 0;
  37. }
  38. /**
  39. * Determines whether a non-Literal node is a negative number that should be
  40. * treated as if it were a single Literal node.
  41. * @param {ASTNode} node Node to test.
  42. * @returns {boolean} True if the node is a negative number that looks like a
  43. * real literal and should be treated as such.
  44. */
  45. function isNegativeNumericLiteral(node) {
  46. return (
  47. node.type === "UnaryExpression" &&
  48. node.operator === "-" &&
  49. node.prefix &&
  50. astUtils.isNumericLiteral(node.argument)
  51. );
  52. }
  53. /**
  54. * Determines whether a node is a Template Literal which can be determined statically.
  55. * @param {ASTNode} node Node to test
  56. * @returns {boolean} True if the node is a Template Literal without expression.
  57. */
  58. function isStaticTemplateLiteral(node) {
  59. return node.type === "TemplateLiteral" && node.expressions.length === 0;
  60. }
  61. /**
  62. * Determines whether a non-Literal node should be treated as a single Literal node.
  63. * @param {ASTNode} node Node to test
  64. * @returns {boolean} True if the node should be treated as a single Literal node.
  65. */
  66. function looksLikeLiteral(node) {
  67. return isNegativeNumericLiteral(node) || isStaticTemplateLiteral(node);
  68. }
  69. /**
  70. * Attempts to derive a Literal node from nodes that are treated like literals.
  71. * @param {ASTNode} node Node to normalize.
  72. * @returns {ASTNode} One of the following options.
  73. * 1. The original node if the node is already a Literal
  74. * 2. A normalized Literal node with the negative number as the value if the
  75. * node represents a negative number literal.
  76. * 3. A normalized Literal node with the string as the value if the node is
  77. * a Template Literal without expression.
  78. * 4. Otherwise `null`.
  79. */
  80. function getNormalizedLiteral(node) {
  81. if (node.type === "Literal") {
  82. return node;
  83. }
  84. if (isNegativeNumericLiteral(node)) {
  85. return {
  86. type: "Literal",
  87. value: -node.argument.value,
  88. raw: `-${node.argument.value}`
  89. };
  90. }
  91. if (isStaticTemplateLiteral(node)) {
  92. return {
  93. type: "Literal",
  94. value: node.quasis[0].value.cooked,
  95. raw: node.quasis[0].value.raw
  96. };
  97. }
  98. return null;
  99. }
  100. //------------------------------------------------------------------------------
  101. // Rule Definition
  102. //------------------------------------------------------------------------------
  103. module.exports = {
  104. meta: {
  105. type: "suggestion",
  106. docs: {
  107. description: 'require or disallow "Yoda" conditions',
  108. category: "Best Practices",
  109. recommended: false,
  110. url: "https://eslint.org/docs/rules/yoda"
  111. },
  112. schema: [
  113. {
  114. enum: ["always", "never"]
  115. },
  116. {
  117. type: "object",
  118. properties: {
  119. exceptRange: {
  120. type: "boolean",
  121. default: false
  122. },
  123. onlyEquality: {
  124. type: "boolean",
  125. default: false
  126. }
  127. },
  128. additionalProperties: false
  129. }
  130. ],
  131. fixable: "code",
  132. messages: {
  133. expected:
  134. "Expected literal to be on the {{expectedSide}} side of {{operator}}."
  135. }
  136. },
  137. create(context) {
  138. // Default to "never" (!always) if no option
  139. const always = context.options[0] === "always";
  140. const exceptRange =
  141. context.options[1] && context.options[1].exceptRange;
  142. const onlyEquality =
  143. context.options[1] && context.options[1].onlyEquality;
  144. const sourceCode = context.getSourceCode();
  145. /**
  146. * Determines whether node represents a range test.
  147. * A range test is a "between" test like `(0 <= x && x < 1)` or an "outside"
  148. * test like `(x < 0 || 1 <= x)`. It must be wrapped in parentheses, and
  149. * both operators must be `<` or `<=`. Finally, the literal on the left side
  150. * must be less than or equal to the literal on the right side so that the
  151. * test makes any sense.
  152. * @param {ASTNode} node LogicalExpression node to test.
  153. * @returns {boolean} Whether node is a range test.
  154. */
  155. function isRangeTest(node) {
  156. const left = node.left,
  157. right = node.right;
  158. /**
  159. * Determines whether node is of the form `0 <= x && x < 1`.
  160. * @returns {boolean} Whether node is a "between" range test.
  161. */
  162. function isBetweenTest() {
  163. if (node.operator === "&&" && astUtils.isSameReference(left.right, right.left)) {
  164. const leftLiteral = getNormalizedLiteral(left.left);
  165. const rightLiteral = getNormalizedLiteral(right.right);
  166. if (leftLiteral === null && rightLiteral === null) {
  167. return false;
  168. }
  169. if (rightLiteral === null || leftLiteral === null) {
  170. return true;
  171. }
  172. if (leftLiteral.value <= rightLiteral.value) {
  173. return true;
  174. }
  175. }
  176. return false;
  177. }
  178. /**
  179. * Determines whether node is of the form `x < 0 || 1 <= x`.
  180. * @returns {boolean} Whether node is an "outside" range test.
  181. */
  182. function isOutsideTest() {
  183. if (node.operator === "||" && astUtils.isSameReference(left.left, right.right)) {
  184. const leftLiteral = getNormalizedLiteral(left.right);
  185. const rightLiteral = getNormalizedLiteral(right.left);
  186. if (leftLiteral === null && rightLiteral === null) {
  187. return false;
  188. }
  189. if (rightLiteral === null || leftLiteral === null) {
  190. return true;
  191. }
  192. if (leftLiteral.value <= rightLiteral.value) {
  193. return true;
  194. }
  195. }
  196. return false;
  197. }
  198. /**
  199. * Determines whether node is wrapped in parentheses.
  200. * @returns {boolean} Whether node is preceded immediately by an open
  201. * paren token and followed immediately by a close
  202. * paren token.
  203. */
  204. function isParenWrapped() {
  205. return astUtils.isParenthesised(sourceCode, node);
  206. }
  207. return (
  208. node.type === "LogicalExpression" &&
  209. left.type === "BinaryExpression" &&
  210. right.type === "BinaryExpression" &&
  211. isRangeTestOperator(left.operator) &&
  212. isRangeTestOperator(right.operator) &&
  213. (isBetweenTest() || isOutsideTest()) &&
  214. isParenWrapped()
  215. );
  216. }
  217. const OPERATOR_FLIP_MAP = {
  218. "===": "===",
  219. "!==": "!==",
  220. "==": "==",
  221. "!=": "!=",
  222. "<": ">",
  223. ">": "<",
  224. "<=": ">=",
  225. ">=": "<="
  226. };
  227. /**
  228. * Returns a string representation of a BinaryExpression node with its sides/operator flipped around.
  229. * @param {ASTNode} node The BinaryExpression node
  230. * @returns {string} A string representation of the node with the sides and operator flipped
  231. */
  232. function getFlippedString(node) {
  233. const tokenBefore = sourceCode.getTokenBefore(node);
  234. const operatorToken = sourceCode.getFirstTokenBetween(
  235. node.left,
  236. node.right,
  237. token => token.value === node.operator
  238. );
  239. const textBeforeOperator = sourceCode
  240. .getText()
  241. .slice(
  242. sourceCode.getTokenBefore(operatorToken).range[1],
  243. operatorToken.range[0]
  244. );
  245. const textAfterOperator = sourceCode
  246. .getText()
  247. .slice(
  248. operatorToken.range[1],
  249. sourceCode.getTokenAfter(operatorToken).range[0]
  250. );
  251. const leftText = sourceCode
  252. .getText()
  253. .slice(
  254. node.range[0],
  255. sourceCode.getTokenBefore(operatorToken).range[1]
  256. );
  257. const firstRightToken = sourceCode.getTokenAfter(operatorToken);
  258. const rightText = sourceCode
  259. .getText()
  260. .slice(firstRightToken.range[0], node.range[1]);
  261. let prefix = "";
  262. if (
  263. tokenBefore &&
  264. tokenBefore.range[1] === node.range[0] &&
  265. !astUtils.canTokensBeAdjacent(tokenBefore, firstRightToken)
  266. ) {
  267. prefix = " ";
  268. }
  269. return (
  270. prefix +
  271. rightText +
  272. textBeforeOperator +
  273. OPERATOR_FLIP_MAP[operatorToken.value] +
  274. textAfterOperator +
  275. leftText
  276. );
  277. }
  278. //--------------------------------------------------------------------------
  279. // Public
  280. //--------------------------------------------------------------------------
  281. return {
  282. BinaryExpression(node) {
  283. const expectedLiteral = always ? node.left : node.right;
  284. const expectedNonLiteral = always ? node.right : node.left;
  285. // If `expectedLiteral` is not a literal, and `expectedNonLiteral` is a literal, raise an error.
  286. if (
  287. (expectedNonLiteral.type === "Literal" ||
  288. looksLikeLiteral(expectedNonLiteral)) &&
  289. !(
  290. expectedLiteral.type === "Literal" ||
  291. looksLikeLiteral(expectedLiteral)
  292. ) &&
  293. !(!isEqualityOperator(node.operator) && onlyEquality) &&
  294. isComparisonOperator(node.operator) &&
  295. !(exceptRange && isRangeTest(context.getAncestors().pop()))
  296. ) {
  297. context.report({
  298. node,
  299. messageId: "expected",
  300. data: {
  301. operator: node.operator,
  302. expectedSide: always ? "left" : "right"
  303. },
  304. fix: fixer =>
  305. fixer.replaceText(node, getFlippedString(node))
  306. });
  307. }
  308. }
  309. };
  310. }
  311. };