Blame view

node_modules/eslint/lib/rules/no-native-reassign.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
  /**
   * @fileoverview Rule to flag when re-assigning native objects
   * @author Ilya Volodin
   */
  
  "use strict";
  
  //------------------------------------------------------------------------------
  // Rule Definition
  //------------------------------------------------------------------------------
  
  module.exports = function(context) {
  
      var config = context.options[0];
      var exceptions = (config && config.exceptions) || [];
  
      /**
       * Gets the names of writeable built-in variables.
       * @param {escope.Scope} scope - A scope to get.
       * @returns {object} A map that its key is variable names.
       */
      function getBuiltinGlobals(scope) {
          return scope.variables.reduce(function(retv, variable) {
              if (variable.writeable === false && variable.name !== "__proto__") {
                  retv[variable.name] = true;
              }
              return retv;
          }, Object.create(null));
      }
  
      /**
       * Reports if a given reference's name is same as native object's.
       * @param {object} builtins - A map that its key is a variable name.
       * @param {Reference} reference - A reference to check.
       * @param {int} index - The index of the reference in the references.
       * @param {Reference[]} references - The array that the reference belongs to.
       * @returns {void}
       */
      function checkThroughReference(builtins, reference, index, references) {
          var identifier = reference.identifier;
  
          if (identifier &&
              builtins[identifier.name] &&
              exceptions.indexOf(identifier.name) === -1 &&
              reference.init === false &&
              reference.isWrite() &&
              // Destructuring assignments can have multiple default value,
              // so possibly there are multiple writeable references for the same identifier.
              (index === 0 || references[index - 1].identifier !== identifier)
          ) {
              context.report(
                  identifier,
                  "{{name}} is a read-only native object.",
                  {name: identifier.name});
          }
      }
  
      return {
          // Checks assignments of global variables.
          // References to implicit global variables are not resolved,
          // so those are in the `through` of the global scope.
          "Program": function() {
              var globalScope = context.getScope();
              var builtins = getBuiltinGlobals(globalScope);
              globalScope.through.forEach(checkThroughReference.bind(null, builtins));
          }
      };
  
  };
  
  module.exports.schema = [
      {
          "type": "object",
          "properties": {
              "exceptions": {
                  "type": "array",
                  "items": {"type": "string"},
                  "uniqueItems": true
              }
          },
          "additionalProperties": false
      }
  ];