no-use-before-define.js 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. /**
  2. * @fileoverview Rule to flag use of variables before they are defined
  3. * @author Ilya Volodin
  4. */
  5. "use strict";
  6. //------------------------------------------------------------------------------
  7. // Helpers
  8. //------------------------------------------------------------------------------
  9. const SENTINEL_TYPE = /^(?:(?:Function|Class)(?:Declaration|Expression)|ArrowFunctionExpression|CatchClause|ImportDeclaration|ExportNamedDeclaration)$/;
  10. const FOR_IN_OF_TYPE = /^For(?:In|Of)Statement$/;
  11. /**
  12. * Parses a given value as options.
  13. *
  14. * @param {any} options - A value to parse.
  15. * @returns {Object} The parsed options.
  16. */
  17. function parseOptions(options) {
  18. let functions = true;
  19. let classes = true;
  20. let variables = true;
  21. if (typeof options === "string") {
  22. functions = (options !== "nofunc");
  23. } else if (typeof options === "object" && options !== null) {
  24. functions = options.functions !== false;
  25. classes = options.classes !== false;
  26. variables = options.variables !== false;
  27. }
  28. return { functions, classes, variables };
  29. }
  30. /**
  31. * Checks whether or not a given variable is a function declaration.
  32. *
  33. * @param {eslint-scope.Variable} variable - A variable to check.
  34. * @returns {boolean} `true` if the variable is a function declaration.
  35. */
  36. function isFunction(variable) {
  37. return variable.defs[0].type === "FunctionName";
  38. }
  39. /**
  40. * Checks whether or not a given variable is a class declaration in an upper function scope.
  41. *
  42. * @param {eslint-scope.Variable} variable - A variable to check.
  43. * @param {eslint-scope.Reference} reference - A reference to check.
  44. * @returns {boolean} `true` if the variable is a class declaration.
  45. */
  46. function isOuterClass(variable, reference) {
  47. return (
  48. variable.defs[0].type === "ClassName" &&
  49. variable.scope.variableScope !== reference.from.variableScope
  50. );
  51. }
  52. /**
  53. * Checks whether or not a given variable is a variable declaration in an upper function scope.
  54. * @param {eslint-scope.Variable} variable - A variable to check.
  55. * @param {eslint-scope.Reference} reference - A reference to check.
  56. * @returns {boolean} `true` if the variable is a variable declaration.
  57. */
  58. function isOuterVariable(variable, reference) {
  59. return (
  60. variable.defs[0].type === "Variable" &&
  61. variable.scope.variableScope !== reference.from.variableScope
  62. );
  63. }
  64. /**
  65. * Checks whether or not a given location is inside of the range of a given node.
  66. *
  67. * @param {ASTNode} node - An node to check.
  68. * @param {number} location - A location to check.
  69. * @returns {boolean} `true` if the location is inside of the range of the node.
  70. */
  71. function isInRange(node, location) {
  72. return node && node.range[0] <= location && location <= node.range[1];
  73. }
  74. /**
  75. * Checks whether or not a given reference is inside of the initializers of a given variable.
  76. *
  77. * This returns `true` in the following cases:
  78. *
  79. * var a = a
  80. * var [a = a] = list
  81. * var {a = a} = obj
  82. * for (var a in a) {}
  83. * for (var a of a) {}
  84. *
  85. * @param {Variable} variable - A variable to check.
  86. * @param {Reference} reference - A reference to check.
  87. * @returns {boolean} `true` if the reference is inside of the initializers.
  88. */
  89. function isInInitializer(variable, reference) {
  90. if (variable.scope !== reference.from) {
  91. return false;
  92. }
  93. let node = variable.identifiers[0].parent;
  94. const location = reference.identifier.range[1];
  95. while (node) {
  96. if (node.type === "VariableDeclarator") {
  97. if (isInRange(node.init, location)) {
  98. return true;
  99. }
  100. if (FOR_IN_OF_TYPE.test(node.parent.parent.type) &&
  101. isInRange(node.parent.parent.right, location)
  102. ) {
  103. return true;
  104. }
  105. break;
  106. } else if (node.type === "AssignmentPattern") {
  107. if (isInRange(node.right, location)) {
  108. return true;
  109. }
  110. } else if (SENTINEL_TYPE.test(node.type)) {
  111. break;
  112. }
  113. node = node.parent;
  114. }
  115. return false;
  116. }
  117. //------------------------------------------------------------------------------
  118. // Rule Definition
  119. //------------------------------------------------------------------------------
  120. module.exports = {
  121. meta: {
  122. docs: {
  123. description: "disallow the use of variables before they are defined",
  124. category: "Variables",
  125. recommended: false,
  126. url: "https://eslint.org/docs/rules/no-use-before-define"
  127. },
  128. schema: [
  129. {
  130. oneOf: [
  131. {
  132. enum: ["nofunc"]
  133. },
  134. {
  135. type: "object",
  136. properties: {
  137. functions: { type: "boolean" },
  138. classes: { type: "boolean" },
  139. variables: { type: "boolean" }
  140. },
  141. additionalProperties: false
  142. }
  143. ]
  144. }
  145. ]
  146. },
  147. create(context) {
  148. const options = parseOptions(context.options[0]);
  149. /**
  150. * Determines whether a given use-before-define case should be reported according to the options.
  151. * @param {eslint-scope.Variable} variable The variable that gets used before being defined
  152. * @param {eslint-scope.Reference} reference The reference to the variable
  153. * @returns {boolean} `true` if the usage should be reported
  154. */
  155. function isForbidden(variable, reference) {
  156. if (isFunction(variable)) {
  157. return options.functions;
  158. }
  159. if (isOuterClass(variable, reference)) {
  160. return options.classes;
  161. }
  162. if (isOuterVariable(variable, reference)) {
  163. return options.variables;
  164. }
  165. return true;
  166. }
  167. /**
  168. * Finds and validates all variables in a given scope.
  169. * @param {Scope} scope The scope object.
  170. * @returns {void}
  171. * @private
  172. */
  173. function findVariablesInScope(scope) {
  174. scope.references.forEach(reference => {
  175. const variable = reference.resolved;
  176. /*
  177. * Skips when the reference is:
  178. * - initialization's.
  179. * - referring to an undefined variable.
  180. * - referring to a global environment variable (there're no identifiers).
  181. * - located preceded by the variable (except in initializers).
  182. * - allowed by options.
  183. */
  184. if (reference.init ||
  185. !variable ||
  186. variable.identifiers.length === 0 ||
  187. (variable.identifiers[0].range[1] < reference.identifier.range[1] && !isInInitializer(variable, reference)) ||
  188. !isForbidden(variable, reference)
  189. ) {
  190. return;
  191. }
  192. // Reports.
  193. context.report({
  194. node: reference.identifier,
  195. message: "'{{name}}' was used before it was defined.",
  196. data: reference.identifier
  197. });
  198. });
  199. }
  200. /**
  201. * Validates variables inside of a node's scope.
  202. * @param {ASTNode} node The node to check.
  203. * @returns {void}
  204. * @private
  205. */
  206. function findVariables() {
  207. const scope = context.getScope();
  208. findVariablesInScope(scope);
  209. }
  210. const ruleDefinition = {
  211. "Program:exit"(node) {
  212. const scope = context.getScope(),
  213. ecmaFeatures = context.parserOptions.ecmaFeatures || {};
  214. findVariablesInScope(scope);
  215. // both Node.js and Modules have an extra scope
  216. if (ecmaFeatures.globalReturn || node.sourceType === "module") {
  217. findVariablesInScope(scope.childScopes[0]);
  218. }
  219. }
  220. };
  221. if (context.parserOptions.ecmaVersion >= 6) {
  222. ruleDefinition["BlockStatement:exit"] =
  223. ruleDefinition["SwitchStatement:exit"] = findVariables;
  224. ruleDefinition["ArrowFunctionExpression:exit"] = function(node) {
  225. if (node.body.type !== "BlockStatement") {
  226. findVariables();
  227. }
  228. };
  229. } else {
  230. ruleDefinition["FunctionExpression:exit"] =
  231. ruleDefinition["FunctionDeclaration:exit"] =
  232. ruleDefinition["ArrowFunctionExpression:exit"] = findVariables;
  233. }
  234. return ruleDefinition;
  235. }
  236. };