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.
		
		
		
		
		
			
		
			
				
					
					
						
							247 lines
						
					
					
						
							7.3 KiB
						
					
					
				
			
		
		
	
	
							247 lines
						
					
					
						
							7.3 KiB
						
					
					
				/* | 
						|
  Copyright (C) 2015 Yusuke Suzuki <utatane.tea@gmail.com> | 
						|
 | 
						|
  Redistribution and use in source and binary forms, with or without | 
						|
  modification, are permitted provided that the following conditions are met: | 
						|
 | 
						|
    * Redistributions of source code must retain the above copyright | 
						|
      notice, this list of conditions and the following disclaimer. | 
						|
    * Redistributions in binary form must reproduce the above copyright | 
						|
      notice, this list of conditions and the following disclaimer in the | 
						|
      documentation and/or other materials provided with the distribution. | 
						|
 | 
						|
  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | 
						|
  AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
						|
  IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | 
						|
  ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY | 
						|
  DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | 
						|
  (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | 
						|
  LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | 
						|
  ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
						|
  (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF | 
						|
  THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
						|
*/ | 
						|
"use strict"; | 
						|
 | 
						|
/* eslint-disable no-underscore-dangle */ | 
						|
 | 
						|
const Scope = require("./scope"); | 
						|
const assert = require("assert"); | 
						|
 | 
						|
const GlobalScope = Scope.GlobalScope; | 
						|
const CatchScope = Scope.CatchScope; | 
						|
const WithScope = Scope.WithScope; | 
						|
const ModuleScope = Scope.ModuleScope; | 
						|
const ClassScope = Scope.ClassScope; | 
						|
const SwitchScope = Scope.SwitchScope; | 
						|
const FunctionScope = Scope.FunctionScope; | 
						|
const ForScope = Scope.ForScope; | 
						|
const FunctionExpressionNameScope = Scope.FunctionExpressionNameScope; | 
						|
const BlockScope = Scope.BlockScope; | 
						|
 | 
						|
/** | 
						|
 * @class ScopeManager | 
						|
 */ | 
						|
class ScopeManager { | 
						|
    constructor(options) { | 
						|
        this.scopes = []; | 
						|
        this.globalScope = null; | 
						|
        this.__nodeToScope = new WeakMap(); | 
						|
        this.__currentScope = null; | 
						|
        this.__options = options; | 
						|
        this.__declaredVariables = new WeakMap(); | 
						|
    } | 
						|
 | 
						|
    __useDirective() { | 
						|
        return this.__options.directive; | 
						|
    } | 
						|
 | 
						|
    __isOptimistic() { | 
						|
        return this.__options.optimistic; | 
						|
    } | 
						|
 | 
						|
    __ignoreEval() { | 
						|
        return this.__options.ignoreEval; | 
						|
    } | 
						|
 | 
						|
    __isNodejsScope() { | 
						|
        return this.__options.nodejsScope; | 
						|
    } | 
						|
 | 
						|
    isModule() { | 
						|
        return this.__options.sourceType === "module"; | 
						|
    } | 
						|
 | 
						|
    isImpliedStrict() { | 
						|
        return this.__options.impliedStrict; | 
						|
    } | 
						|
 | 
						|
    isStrictModeSupported() { | 
						|
        return this.__options.ecmaVersion >= 5; | 
						|
    } | 
						|
 | 
						|
    // Returns appropriate scope for this node. | 
						|
    __get(node) { | 
						|
        return this.__nodeToScope.get(node); | 
						|
    } | 
						|
 | 
						|
    /** | 
						|
     * Get variables that are declared by the node. | 
						|
     * | 
						|
     * "are declared by the node" means the node is same as `Variable.defs[].node` or `Variable.defs[].parent`. | 
						|
     * If the node declares nothing, this method returns an empty array. | 
						|
     * CAUTION: This API is experimental. See https://github.com/estools/escope/pull/69 for more details. | 
						|
     * | 
						|
     * @param {Espree.Node} node - a node to get. | 
						|
     * @returns {Variable[]} variables that declared by the node. | 
						|
     */ | 
						|
    getDeclaredVariables(node) { | 
						|
        return this.__declaredVariables.get(node) || []; | 
						|
    } | 
						|
 | 
						|
    /** | 
						|
     * acquire scope from node. | 
						|
     * @method ScopeManager#acquire | 
						|
     * @param {Espree.Node} node - node for the acquired scope. | 
						|
     * @param {boolean=} inner - look up the most inner scope, default value is false. | 
						|
     * @returns {Scope?} Scope from node | 
						|
     */ | 
						|
    acquire(node, inner) { | 
						|
 | 
						|
        /** | 
						|
         * predicate | 
						|
         * @param {Scope} testScope - scope to test | 
						|
         * @returns {boolean} predicate | 
						|
         */ | 
						|
        function predicate(testScope) { | 
						|
            if (testScope.type === "function" && testScope.functionExpressionScope) { | 
						|
                return false; | 
						|
            } | 
						|
            return true; | 
						|
        } | 
						|
 | 
						|
        const scopes = this.__get(node); | 
						|
 | 
						|
        if (!scopes || scopes.length === 0) { | 
						|
            return null; | 
						|
        } | 
						|
 | 
						|
        // Heuristic selection from all scopes. | 
						|
        // If you would like to get all scopes, please use ScopeManager#acquireAll. | 
						|
        if (scopes.length === 1) { | 
						|
            return scopes[0]; | 
						|
        } | 
						|
 | 
						|
        if (inner) { | 
						|
            for (let i = scopes.length - 1; i >= 0; --i) { | 
						|
                const scope = scopes[i]; | 
						|
 | 
						|
                if (predicate(scope)) { | 
						|
                    return scope; | 
						|
                } | 
						|
            } | 
						|
        } else { | 
						|
            for (let i = 0, iz = scopes.length; i < iz; ++i) { | 
						|
                const scope = scopes[i]; | 
						|
 | 
						|
                if (predicate(scope)) { | 
						|
                    return scope; | 
						|
                } | 
						|
            } | 
						|
        } | 
						|
 | 
						|
        return null; | 
						|
    } | 
						|
 | 
						|
    /** | 
						|
     * acquire all scopes from node. | 
						|
     * @method ScopeManager#acquireAll | 
						|
     * @param {Espree.Node} node - node for the acquired scope. | 
						|
     * @returns {Scopes?} Scope array | 
						|
     */ | 
						|
    acquireAll(node) { | 
						|
        return this.__get(node); | 
						|
    } | 
						|
 | 
						|
    /** | 
						|
     * release the node. | 
						|
     * @method ScopeManager#release | 
						|
     * @param {Espree.Node} node - releasing node. | 
						|
     * @param {boolean=} inner - look up the most inner scope, default value is false. | 
						|
     * @returns {Scope?} upper scope for the node. | 
						|
     */ | 
						|
    release(node, inner) { | 
						|
        const scopes = this.__get(node); | 
						|
 | 
						|
        if (scopes && scopes.length) { | 
						|
            const scope = scopes[0].upper; | 
						|
 | 
						|
            if (!scope) { | 
						|
                return null; | 
						|
            } | 
						|
            return this.acquire(scope.block, inner); | 
						|
        } | 
						|
        return null; | 
						|
    } | 
						|
 | 
						|
    attach() { } // eslint-disable-line class-methods-use-this | 
						|
 | 
						|
    detach() { } // eslint-disable-line class-methods-use-this | 
						|
 | 
						|
    __nestScope(scope) { | 
						|
        if (scope instanceof GlobalScope) { | 
						|
            assert(this.__currentScope === null); | 
						|
            this.globalScope = scope; | 
						|
        } | 
						|
        this.__currentScope = scope; | 
						|
        return scope; | 
						|
    } | 
						|
 | 
						|
    __nestGlobalScope(node) { | 
						|
        return this.__nestScope(new GlobalScope(this, node)); | 
						|
    } | 
						|
 | 
						|
    __nestBlockScope(node) { | 
						|
        return this.__nestScope(new BlockScope(this, this.__currentScope, node)); | 
						|
    } | 
						|
 | 
						|
    __nestFunctionScope(node, isMethodDefinition) { | 
						|
        return this.__nestScope(new FunctionScope(this, this.__currentScope, node, isMethodDefinition)); | 
						|
    } | 
						|
 | 
						|
    __nestForScope(node) { | 
						|
        return this.__nestScope(new ForScope(this, this.__currentScope, node)); | 
						|
    } | 
						|
 | 
						|
    __nestCatchScope(node) { | 
						|
        return this.__nestScope(new CatchScope(this, this.__currentScope, node)); | 
						|
    } | 
						|
 | 
						|
    __nestWithScope(node) { | 
						|
        return this.__nestScope(new WithScope(this, this.__currentScope, node)); | 
						|
    } | 
						|
 | 
						|
    __nestClassScope(node) { | 
						|
        return this.__nestScope(new ClassScope(this, this.__currentScope, node)); | 
						|
    } | 
						|
 | 
						|
    __nestSwitchScope(node) { | 
						|
        return this.__nestScope(new SwitchScope(this, this.__currentScope, node)); | 
						|
    } | 
						|
 | 
						|
    __nestModuleScope(node) { | 
						|
        return this.__nestScope(new ModuleScope(this, this.__currentScope, node)); | 
						|
    } | 
						|
 | 
						|
    __nestFunctionExpressionNameScope(node) { | 
						|
        return this.__nestScope(new FunctionExpressionNameScope(this, this.__currentScope, node)); | 
						|
    } | 
						|
 | 
						|
    __isES6() { | 
						|
        return this.__options.ecmaVersion >= 6; | 
						|
    } | 
						|
} | 
						|
 | 
						|
module.exports = ScopeManager; | 
						|
 | 
						|
/* vim: set sw=4 ts=4 et tw=80 : */
 | 
						|
 |