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.

181 lines
5.5 KiB

4 years ago
  1. /**
  2. * @fileoverview Rule to flag the use of redundant constructors in classes.
  3. * @author Alberto Rodríguez
  4. */
  5. "use strict";
  6. //------------------------------------------------------------------------------
  7. // Helpers
  8. //------------------------------------------------------------------------------
  9. /**
  10. * Checks whether a given array of statements is a single call of `super`.
  11. * @param {ASTNode[]} body An array of statements to check.
  12. * @returns {boolean} `true` if the body is a single call of `super`.
  13. */
  14. function isSingleSuperCall(body) {
  15. return (
  16. body.length === 1 &&
  17. body[0].type === "ExpressionStatement" &&
  18. body[0].expression.type === "CallExpression" &&
  19. body[0].expression.callee.type === "Super"
  20. );
  21. }
  22. /**
  23. * Checks whether a given node is a pattern which doesn't have any side effects.
  24. * Default parameters and Destructuring parameters can have side effects.
  25. * @param {ASTNode} node A pattern node.
  26. * @returns {boolean} `true` if the node doesn't have any side effects.
  27. */
  28. function isSimple(node) {
  29. return node.type === "Identifier" || node.type === "RestElement";
  30. }
  31. /**
  32. * Checks whether a given array of expressions is `...arguments` or not.
  33. * `super(...arguments)` passes all arguments through.
  34. * @param {ASTNode[]} superArgs An array of expressions to check.
  35. * @returns {boolean} `true` if the superArgs is `...arguments`.
  36. */
  37. function isSpreadArguments(superArgs) {
  38. return (
  39. superArgs.length === 1 &&
  40. superArgs[0].type === "SpreadElement" &&
  41. superArgs[0].argument.type === "Identifier" &&
  42. superArgs[0].argument.name === "arguments"
  43. );
  44. }
  45. /**
  46. * Checks whether given 2 nodes are identifiers which have the same name or not.
  47. * @param {ASTNode} ctorParam A node to check.
  48. * @param {ASTNode} superArg A node to check.
  49. * @returns {boolean} `true` if the nodes are identifiers which have the same
  50. * name.
  51. */
  52. function isValidIdentifierPair(ctorParam, superArg) {
  53. return (
  54. ctorParam.type === "Identifier" &&
  55. superArg.type === "Identifier" &&
  56. ctorParam.name === superArg.name
  57. );
  58. }
  59. /**
  60. * Checks whether given 2 nodes are a rest/spread pair which has the same values.
  61. * @param {ASTNode} ctorParam A node to check.
  62. * @param {ASTNode} superArg A node to check.
  63. * @returns {boolean} `true` if the nodes are a rest/spread pair which has the
  64. * same values.
  65. */
  66. function isValidRestSpreadPair(ctorParam, superArg) {
  67. return (
  68. ctorParam.type === "RestElement" &&
  69. superArg.type === "SpreadElement" &&
  70. isValidIdentifierPair(ctorParam.argument, superArg.argument)
  71. );
  72. }
  73. /**
  74. * Checks whether given 2 nodes have the same value or not.
  75. * @param {ASTNode} ctorParam A node to check.
  76. * @param {ASTNode} superArg A node to check.
  77. * @returns {boolean} `true` if the nodes have the same value or not.
  78. */
  79. function isValidPair(ctorParam, superArg) {
  80. return (
  81. isValidIdentifierPair(ctorParam, superArg) ||
  82. isValidRestSpreadPair(ctorParam, superArg)
  83. );
  84. }
  85. /**
  86. * Checks whether the parameters of a constructor and the arguments of `super()`
  87. * have the same values or not.
  88. * @param {ASTNode} ctorParams The parameters of a constructor to check.
  89. * @param {ASTNode} superArgs The arguments of `super()` to check.
  90. * @returns {boolean} `true` if those have the same values.
  91. */
  92. function isPassingThrough(ctorParams, superArgs) {
  93. if (ctorParams.length !== superArgs.length) {
  94. return false;
  95. }
  96. for (let i = 0; i < ctorParams.length; ++i) {
  97. if (!isValidPair(ctorParams[i], superArgs[i])) {
  98. return false;
  99. }
  100. }
  101. return true;
  102. }
  103. /**
  104. * Checks whether the constructor body is a redundant super call.
  105. * @param {Array} body constructor body content.
  106. * @param {Array} ctorParams The params to check against super call.
  107. * @returns {boolean} true if the constructor body is redundant
  108. */
  109. function isRedundantSuperCall(body, ctorParams) {
  110. return (
  111. isSingleSuperCall(body) &&
  112. ctorParams.every(isSimple) &&
  113. (
  114. isSpreadArguments(body[0].expression.arguments) ||
  115. isPassingThrough(ctorParams, body[0].expression.arguments)
  116. )
  117. );
  118. }
  119. //------------------------------------------------------------------------------
  120. // Rule Definition
  121. //------------------------------------------------------------------------------
  122. module.exports = {
  123. meta: {
  124. type: "suggestion",
  125. docs: {
  126. description: "disallow unnecessary constructors",
  127. category: "ECMAScript 6",
  128. recommended: false,
  129. url: "https://eslint.org/docs/rules/no-useless-constructor"
  130. },
  131. schema: [],
  132. messages: {
  133. noUselessConstructor: "Useless constructor."
  134. }
  135. },
  136. create(context) {
  137. /**
  138. * Checks whether a node is a redundant constructor
  139. * @param {ASTNode} node node to check
  140. * @returns {void}
  141. */
  142. function checkForConstructor(node) {
  143. if (node.kind !== "constructor") {
  144. return;
  145. }
  146. const body = node.value.body.body;
  147. const ctorParams = node.value.params;
  148. const superClass = node.parent.parent.superClass;
  149. if (superClass ? isRedundantSuperCall(body, ctorParams) : (body.length === 0)) {
  150. context.report({
  151. node,
  152. messageId: "noUselessConstructor"
  153. });
  154. }
  155. }
  156. return {
  157. MethodDefinition: checkForConstructor
  158. };
  159. }
  160. };