| /** | |
|  * @fileoverview A rule to disallow `this` keywords in contexts where the value of `this` is `undefined`. | |
|  * @author Toru Nagashima | |
|  */ | |
| 
 | |
| "use strict"; | |
| 
 | |
| //------------------------------------------------------------------------------ | |
| // Requirements | |
| //------------------------------------------------------------------------------ | |
|  | |
| const astUtils = require("./utils/ast-utils"); | |
| 
 | |
| //------------------------------------------------------------------------------ | |
| // Helpers | |
| //------------------------------------------------------------------------------ | |
|  | |
| /** | |
|  * Determines if the given code path is a code path with lexical `this` binding. | |
|  * That is, if `this` within the code path refers to `this` of surrounding code path. | |
|  * @param {CodePath} codePath Code path. | |
|  * @param {ASTNode} node Node that started the code path. | |
|  * @returns {boolean} `true` if it is a code path with lexical `this` binding. | |
|  */ | |
| function isCodePathWithLexicalThis(codePath, node) { | |
|     return codePath.origin === "function" && node.type === "ArrowFunctionExpression"; | |
| } | |
| 
 | |
| //------------------------------------------------------------------------------ | |
| // Rule Definition | |
| //------------------------------------------------------------------------------ | |
|  | |
| /** @type {import('../shared/types').Rule} */ | |
| module.exports = { | |
|     meta: { | |
|         type: "suggestion", | |
| 
 | |
|         docs: { | |
|             description: "Disallow use of `this` in contexts where the value of `this` is `undefined`", | |
|             recommended: false, | |
|             url: "https://eslint.org/docs/latest/rules/no-invalid-this" | |
|         }, | |
| 
 | |
|         schema: [ | |
|             { | |
|                 type: "object", | |
|                 properties: { | |
|                     capIsConstructor: { | |
|                         type: "boolean", | |
|                         default: true | |
|                     } | |
|                 }, | |
|                 additionalProperties: false | |
|             } | |
|         ], | |
| 
 | |
|         messages: { | |
|             unexpectedThis: "Unexpected 'this'." | |
|         } | |
|     }, | |
| 
 | |
|     create(context) { | |
|         const options = context.options[0] || {}; | |
|         const capIsConstructor = options.capIsConstructor !== false; | |
|         const stack = [], | |
|             sourceCode = context.sourceCode; | |
| 
 | |
|         /** | |
|          * Gets the current checking context. | |
|          * | |
|          * The return value has a flag that whether or not `this` keyword is valid. | |
|          * The flag is initialized when got at the first time. | |
|          * @returns {{valid: boolean}} | |
|          *   an object which has a flag that whether or not `this` keyword is valid. | |
|          */ | |
|         stack.getCurrent = function() { | |
|             const current = this[this.length - 1]; | |
| 
 | |
|             if (!current.init) { | |
|                 current.init = true; | |
|                 current.valid = !astUtils.isDefaultThisBinding( | |
|                     current.node, | |
|                     sourceCode, | |
|                     { capIsConstructor } | |
|                 ); | |
|             } | |
|             return current; | |
|         }; | |
| 
 | |
|         return { | |
| 
 | |
|             onCodePathStart(codePath, node) { | |
|                 if (isCodePathWithLexicalThis(codePath, node)) { | |
|                     return; | |
|                 } | |
| 
 | |
|                 if (codePath.origin === "program") { | |
|                     const scope = sourceCode.getScope(node); | |
|                     const features = context.languageOptions.parserOptions.ecmaFeatures || {}; | |
| 
 | |
|                     // `this` at the top level of scripts always refers to the global object | |
|                     stack.push({ | |
|                         init: true, | |
|                         node, | |
|                         valid: !( | |
|                             node.sourceType === "module" || | |
|                             (features.globalReturn && scope.childScopes[0].isStrict) | |
|                         ) | |
|                     }); | |
| 
 | |
|                     return; | |
|                 } | |
| 
 | |
|                 /* | |
|                  * `init: false` means that `valid` isn't determined yet. | |
|                  * Most functions don't use `this`, and the calculation for `valid` | |
|                  * is relatively costly, so we'll calculate it lazily when the first | |
|                  * `this` within the function is traversed. A special case are non-strict | |
|                  * functions, because `this` refers to the global object and therefore is | |
|                  * always valid, so we can set `init: true` right away. | |
|                  */ | |
|                 stack.push({ | |
|                     init: !sourceCode.getScope(node).isStrict, | |
|                     node, | |
|                     valid: true | |
|                 }); | |
|             }, | |
| 
 | |
|             onCodePathEnd(codePath, node) { | |
|                 if (isCodePathWithLexicalThis(codePath, node)) { | |
|                     return; | |
|                 } | |
| 
 | |
|                 stack.pop(); | |
|             }, | |
| 
 | |
|             // Reports if `this` of the current context is invalid. | |
|             ThisExpression(node) { | |
|                 const current = stack.getCurrent(); | |
| 
 | |
|                 if (current && !current.valid) { | |
|                     context.report({ | |
|                         node, | |
|                         messageId: "unexpectedThis" | |
|                     }); | |
|                 } | |
|             } | |
|         }; | |
|     } | |
| };
 |