Blame view

node_modules/eslint/lib/rules/no-func-assign.js 2.75 KB
c39994410   Ryan Glover   wip converting to...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
  /**
   * @fileoverview Rule to flag use of function declaration identifiers as variables.
   * @author Ian Christian Myers
   * @copyright 2013 Ian Christian Myers. All rights reserved.
   */
  
  "use strict";
  
  var astUtils = require("../ast-utils");
  
  //------------------------------------------------------------------------------
  // Rule Definition
  //------------------------------------------------------------------------------
  
  module.exports = function(context) {
  
      var unresolved = Object.create(null);
  
      /**
       * Collects unresolved references from the global scope, then creates a map to references from its name.
       * Usage of the map is explained at `checkVariable(variable)`.
       * @returns {void}
       */
      function collectUnresolvedReferences() {
          unresolved = Object.create(null);
  
          var references = context.getScope().through;
          for (var i = 0; i < references.length; ++i) {
              var reference = references[i];
              var name = reference.identifier.name;
  
              if (name in unresolved === false) {
                  unresolved[name] = [];
              }
              unresolved[name].push(reference);
          }
      }
  
      /**
       * Reports a reference if is non initializer and writable.
       * @param {References} references - Collection of reference to check.
       * @returns {void}
       */
      function checkReference(references) {
          astUtils.getModifyingReferences(references).forEach(function(reference) {
              context.report(
                  reference.identifier,
                  "'{{name}}' is a function.",
                  {name: reference.identifier.name});
          });
      }
  
      /**
       * Finds and reports references that are non initializer and writable.
       * @param {Variable} variable - A variable to check.
       * @returns {void}
       */
      function checkVariable(variable) {
          if (variable.defs[0].type === "FunctionName") {
              // If the function is in global scope, its references are not resolved (by escope's design).
              // So when references of the function are nothing, this checks in unresolved.
              if (variable.references.length > 0) {
                  checkReference(variable.references);
              } else if (unresolved[variable.name]) {
                  checkReference(unresolved[variable.name]);
              }
          }
      }
  
      /**
       * Checks parameters of a given function node.
       * @param {ASTNode} node - A function node to check.
       * @returns {void}
       */
      function checkForFunction(node) {
          context.getDeclaredVariables(node).forEach(checkVariable);
      }
  
      return {
          "Program": collectUnresolvedReferences,
          "FunctionDeclaration": checkForFunction,
          "FunctionExpression": checkForFunction
      };
  
  };
  
  module.exports.schema = [];