| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760 | /** * @fileoverview A class of the code path analyzer. * @author Toru Nagashima */"use strict";//------------------------------------------------------------------------------// Requirements//------------------------------------------------------------------------------const assert = require("assert"),    { breakableTypePattern } = require("../../shared/ast-utils"),    CodePath = require("./code-path"),    CodePathSegment = require("./code-path-segment"),    IdGenerator = require("./id-generator"),    debug = require("./debug-helpers");//------------------------------------------------------------------------------// Helpers//------------------------------------------------------------------------------/** * Checks whether or not a given node is a `case` node (not `default` node). * @param {ASTNode} node A `SwitchCase` node to check. * @returns {boolean} `true` if the node is a `case` node (not `default` node). */function isCaseNode(node) {    return Boolean(node.test);}/** * Checks whether the given logical operator is taken into account for the code * path analysis. * @param {string} operator The operator found in the LogicalExpression node * @returns {boolean} `true` if the operator is "&&" or "||" or "??" */function isHandledLogicalOperator(operator) {    return operator === "&&" || operator === "||" || operator === "??";}/** * Checks whether the given assignment operator is a logical assignment operator. * Logical assignments are taken into account for the code path analysis * because of their short-circuiting semantics. * @param {string} operator The operator found in the AssignmentExpression node * @returns {boolean} `true` if the operator is "&&=" or "||=" or "??=" */function isLogicalAssignmentOperator(operator) {    return operator === "&&=" || operator === "||=" || operator === "??=";}/** * Gets the label if the parent node of a given node is a LabeledStatement. * @param {ASTNode} node A node to get. * @returns {string|null} The label or `null`. */function getLabel(node) {    if (node.parent.type === "LabeledStatement") {        return node.parent.label.name;    }    return null;}/** * Checks whether or not a given logical expression node goes different path * between the `true` case and the `false` case. * @param {ASTNode} node A node to check. * @returns {boolean} `true` if the node is a test of a choice statement. */function isForkingByTrueOrFalse(node) {    const parent = node.parent;    switch (parent.type) {        case "ConditionalExpression":        case "IfStatement":        case "WhileStatement":        case "DoWhileStatement":        case "ForStatement":            return parent.test === node;        case "LogicalExpression":            return isHandledLogicalOperator(parent.operator);        case "AssignmentExpression":            return isLogicalAssignmentOperator(parent.operator);        default:            return false;    }}/** * Gets the boolean value of a given literal node. * * This is used to detect infinity loops (e.g. `while (true) {}`). * Statements preceded by an infinity loop are unreachable if the loop didn't * have any `break` statement. * @param {ASTNode} node A node to get. * @returns {boolean|undefined} a boolean value if the node is a Literal node, *   otherwise `undefined`. */function getBooleanValueIfSimpleConstant(node) {    if (node.type === "Literal") {        return Boolean(node.value);    }    return void 0;}/** * Checks that a given identifier node is a reference or not. * * This is used to detect the first throwable node in a `try` block. * @param {ASTNode} node An Identifier node to check. * @returns {boolean} `true` if the node is a reference. */function isIdentifierReference(node) {    const parent = node.parent;    switch (parent.type) {        case "LabeledStatement":        case "BreakStatement":        case "ContinueStatement":        case "ArrayPattern":        case "RestElement":        case "ImportSpecifier":        case "ImportDefaultSpecifier":        case "ImportNamespaceSpecifier":        case "CatchClause":            return false;        case "FunctionDeclaration":        case "FunctionExpression":        case "ArrowFunctionExpression":        case "ClassDeclaration":        case "ClassExpression":        case "VariableDeclarator":            return parent.id !== node;        case "Property":        case "MethodDefinition":            return (                parent.key !== node ||                parent.computed ||                parent.shorthand            );        case "AssignmentPattern":            return parent.key !== node;        default:            return true;    }}/** * Updates the current segment with the head segment. * This is similar to local branches and tracking branches of git. * * To separate the current and the head is in order to not make useless segments. * * In this process, both "onCodePathSegmentStart" and "onCodePathSegmentEnd" * events are fired. * @param {CodePathAnalyzer} analyzer The instance. * @param {ASTNode} node The current AST node. * @returns {void} */function forwardCurrentToHead(analyzer, node) {    const codePath = analyzer.codePath;    const state = CodePath.getState(codePath);    const currentSegments = state.currentSegments;    const headSegments = state.headSegments;    const end = Math.max(currentSegments.length, headSegments.length);    let i, currentSegment, headSegment;    // Fires leaving events.    for (i = 0; i < end; ++i) {        currentSegment = currentSegments[i];        headSegment = headSegments[i];        if (currentSegment !== headSegment && currentSegment) {            debug.dump(`onCodePathSegmentEnd ${currentSegment.id}`);            if (currentSegment.reachable) {                analyzer.emitter.emit(                    "onCodePathSegmentEnd",                    currentSegment,                    node                );            }        }    }    // Update state.    state.currentSegments = headSegments;    // Fires entering events.    for (i = 0; i < end; ++i) {        currentSegment = currentSegments[i];        headSegment = headSegments[i];        if (currentSegment !== headSegment && headSegment) {            debug.dump(`onCodePathSegmentStart ${headSegment.id}`);            CodePathSegment.markUsed(headSegment);            if (headSegment.reachable) {                analyzer.emitter.emit(                    "onCodePathSegmentStart",                    headSegment,                    node                );            }        }    }}/** * Updates the current segment with empty. * This is called at the last of functions or the program. * @param {CodePathAnalyzer} analyzer The instance. * @param {ASTNode} node The current AST node. * @returns {void} */function leaveFromCurrentSegment(analyzer, node) {    const state = CodePath.getState(analyzer.codePath);    const currentSegments = state.currentSegments;    for (let i = 0; i < currentSegments.length; ++i) {        const currentSegment = currentSegments[i];        debug.dump(`onCodePathSegmentEnd ${currentSegment.id}`);        if (currentSegment.reachable) {            analyzer.emitter.emit(                "onCodePathSegmentEnd",                currentSegment,                node            );        }    }    state.currentSegments = [];}/** * Updates the code path due to the position of a given node in the parent node * thereof. * * For example, if the node is `parent.consequent`, this creates a fork from the * current path. * @param {CodePathAnalyzer} analyzer The instance. * @param {ASTNode} node The current AST node. * @returns {void} */function preprocess(analyzer, node) {    const codePath = analyzer.codePath;    const state = CodePath.getState(codePath);    const parent = node.parent;    switch (parent.type) {        // The `arguments.length == 0` case is in `postprocess` function.        case "CallExpression":            if (parent.optional === true && parent.arguments.length >= 1 && parent.arguments[0] === node) {                state.makeOptionalRight();            }            break;        case "MemberExpression":            if (parent.optional === true && parent.property === node) {                state.makeOptionalRight();            }            break;        case "LogicalExpression":            if (                parent.right === node &&                isHandledLogicalOperator(parent.operator)            ) {                state.makeLogicalRight();            }            break;        case "AssignmentExpression":            if (                parent.right === node &&                isLogicalAssignmentOperator(parent.operator)            ) {                state.makeLogicalRight();            }            break;        case "ConditionalExpression":        case "IfStatement":            /*             * Fork if this node is at `consequent`/`alternate`.             * `popForkContext()` exists at `IfStatement:exit` and             * `ConditionalExpression:exit`.             */            if (parent.consequent === node) {                state.makeIfConsequent();            } else if (parent.alternate === node) {                state.makeIfAlternate();            }            break;        case "SwitchCase":            if (parent.consequent[0] === node) {                state.makeSwitchCaseBody(false, !parent.test);            }            break;        case "TryStatement":            if (parent.handler === node) {                state.makeCatchBlock();            } else if (parent.finalizer === node) {                state.makeFinallyBlock();            }            break;        case "WhileStatement":            if (parent.test === node) {                state.makeWhileTest(getBooleanValueIfSimpleConstant(node));            } else {                assert(parent.body === node);                state.makeWhileBody();            }            break;        case "DoWhileStatement":            if (parent.body === node) {                state.makeDoWhileBody();            } else {                assert(parent.test === node);                state.makeDoWhileTest(getBooleanValueIfSimpleConstant(node));            }            break;        case "ForStatement":            if (parent.test === node) {                state.makeForTest(getBooleanValueIfSimpleConstant(node));            } else if (parent.update === node) {                state.makeForUpdate();            } else if (parent.body === node) {                state.makeForBody();            }            break;        case "ForInStatement":        case "ForOfStatement":            if (parent.left === node) {                state.makeForInOfLeft();            } else if (parent.right === node) {                state.makeForInOfRight();            } else {                assert(parent.body === node);                state.makeForInOfBody();            }            break;        case "AssignmentPattern":            /*             * Fork if this node is at `right`.             * `left` is executed always, so it uses the current path.             * `popForkContext()` exists at `AssignmentPattern:exit`.             */            if (parent.right === node) {                state.pushForkContext();                state.forkBypassPath();                state.forkPath();            }            break;        default:            break;    }}/** * Updates the code path due to the type of a given node in entering. * @param {CodePathAnalyzer} analyzer The instance. * @param {ASTNode} node The current AST node. * @returns {void} */function processCodePathToEnter(analyzer, node) {    let codePath = analyzer.codePath;    let state = codePath && CodePath.getState(codePath);    const parent = node.parent;    switch (node.type) {        case "Program":        case "FunctionDeclaration":        case "FunctionExpression":        case "ArrowFunctionExpression":            if (codePath) {                // Emits onCodePathSegmentStart events if updated.                forwardCurrentToHead(analyzer, node);                debug.dumpState(node, state, false);            }            // Create the code path of this scope.            codePath = analyzer.codePath = new CodePath(                analyzer.idGenerator.next(),                codePath,                analyzer.onLooped            );            state = CodePath.getState(codePath);            // Emits onCodePathStart events.            debug.dump(`onCodePathStart ${codePath.id}`);            analyzer.emitter.emit("onCodePathStart", codePath, node);            break;        case "ChainExpression":            state.pushChainContext();            break;        case "CallExpression":            if (node.optional === true) {                state.makeOptionalNode();            }            break;        case "MemberExpression":            if (node.optional === true) {                state.makeOptionalNode();            }            break;        case "LogicalExpression":            if (isHandledLogicalOperator(node.operator)) {                state.pushChoiceContext(                    node.operator,                    isForkingByTrueOrFalse(node)                );            }            break;        case "AssignmentExpression":            if (isLogicalAssignmentOperator(node.operator)) {                state.pushChoiceContext(                    node.operator.slice(0, -1), // removes `=` from the end                    isForkingByTrueOrFalse(node)                );            }            break;        case "ConditionalExpression":        case "IfStatement":            state.pushChoiceContext("test", false);            break;        case "SwitchStatement":            state.pushSwitchContext(                node.cases.some(isCaseNode),                getLabel(node)            );            break;        case "TryStatement":            state.pushTryContext(Boolean(node.finalizer));            break;        case "SwitchCase":            /*             * Fork if this node is after the 2st node in `cases`.             * It's similar to `else` blocks.             * The next `test` node is processed in this path.             */            if (parent.discriminant !== node && parent.cases[0] !== node) {                state.forkPath();            }            break;        case "WhileStatement":        case "DoWhileStatement":        case "ForStatement":        case "ForInStatement":        case "ForOfStatement":            state.pushLoopContext(node.type, getLabel(node));            break;        case "LabeledStatement":            if (!breakableTypePattern.test(node.body.type)) {                state.pushBreakContext(false, node.label.name);            }            break;        default:            break;    }    // Emits onCodePathSegmentStart events if updated.    forwardCurrentToHead(analyzer, node);    debug.dumpState(node, state, false);}/** * Updates the code path due to the type of a given node in leaving. * @param {CodePathAnalyzer} analyzer The instance. * @param {ASTNode} node The current AST node. * @returns {void} */function processCodePathToExit(analyzer, node) {    const codePath = analyzer.codePath;    const state = CodePath.getState(codePath);    let dontForward = false;    switch (node.type) {        case "ChainExpression":            state.popChainContext();            break;        case "IfStatement":        case "ConditionalExpression":            state.popChoiceContext();            break;        case "LogicalExpression":            if (isHandledLogicalOperator(node.operator)) {                state.popChoiceContext();            }            break;        case "AssignmentExpression":            if (isLogicalAssignmentOperator(node.operator)) {                state.popChoiceContext();            }            break;        case "SwitchStatement":            state.popSwitchContext();            break;        case "SwitchCase":            /*             * This is the same as the process at the 1st `consequent` node in             * `preprocess` function.             * Must do if this `consequent` is empty.             */            if (node.consequent.length === 0) {                state.makeSwitchCaseBody(true, !node.test);            }            if (state.forkContext.reachable) {                dontForward = true;            }            break;        case "TryStatement":            state.popTryContext();            break;        case "BreakStatement":            forwardCurrentToHead(analyzer, node);            state.makeBreak(node.label && node.label.name);            dontForward = true;            break;        case "ContinueStatement":            forwardCurrentToHead(analyzer, node);            state.makeContinue(node.label && node.label.name);            dontForward = true;            break;        case "ReturnStatement":            forwardCurrentToHead(analyzer, node);            state.makeReturn();            dontForward = true;            break;        case "ThrowStatement":            forwardCurrentToHead(analyzer, node);            state.makeThrow();            dontForward = true;            break;        case "Identifier":            if (isIdentifierReference(node)) {                state.makeFirstThrowablePathInTryBlock();                dontForward = true;            }            break;        case "CallExpression":        case "ImportExpression":        case "MemberExpression":        case "NewExpression":        case "YieldExpression":            state.makeFirstThrowablePathInTryBlock();            break;        case "WhileStatement":        case "DoWhileStatement":        case "ForStatement":        case "ForInStatement":        case "ForOfStatement":            state.popLoopContext();            break;        case "AssignmentPattern":            state.popForkContext();            break;        case "LabeledStatement":            if (!breakableTypePattern.test(node.body.type)) {                state.popBreakContext();            }            break;        default:            break;    }    // Emits onCodePathSegmentStart events if updated.    if (!dontForward) {        forwardCurrentToHead(analyzer, node);    }    debug.dumpState(node, state, true);}/** * Updates the code path to finalize the current code path. * @param {CodePathAnalyzer} analyzer The instance. * @param {ASTNode} node The current AST node. * @returns {void} */function postprocess(analyzer, node) {    switch (node.type) {        case "Program":        case "FunctionDeclaration":        case "FunctionExpression":        case "ArrowFunctionExpression": {            let codePath = analyzer.codePath;            // Mark the current path as the final node.            CodePath.getState(codePath).makeFinal();            // Emits onCodePathSegmentEnd event of the current segments.            leaveFromCurrentSegment(analyzer, node);            // Emits onCodePathEnd event of this code path.            debug.dump(`onCodePathEnd ${codePath.id}`);            analyzer.emitter.emit("onCodePathEnd", codePath, node);            debug.dumpDot(codePath);            codePath = analyzer.codePath = analyzer.codePath.upper;            if (codePath) {                debug.dumpState(node, CodePath.getState(codePath), true);            }            break;        }        // The `arguments.length >= 1` case is in `preprocess` function.        case "CallExpression":            if (node.optional === true && node.arguments.length === 0) {                CodePath.getState(analyzer.codePath).makeOptionalRight();            }            break;        default:            break;    }}//------------------------------------------------------------------------------// Public Interface//------------------------------------------------------------------------------/** * The class to analyze code paths. * This class implements the EventGenerator interface. */class CodePathAnalyzer {    // eslint-disable-next-line jsdoc/require-description    /**     * @param {EventGenerator} eventGenerator An event generator to wrap.     */    constructor(eventGenerator) {        this.original = eventGenerator;        this.emitter = eventGenerator.emitter;        this.codePath = null;        this.idGenerator = new IdGenerator("s");        this.currentNode = null;        this.onLooped = this.onLooped.bind(this);    }    /**     * Does the process to enter a given AST node.     * This updates state of analysis and calls `enterNode` of the wrapped.     * @param {ASTNode} node A node which is entering.     * @returns {void}     */    enterNode(node) {        this.currentNode = node;        // Updates the code path due to node's position in its parent node.        if (node.parent) {            preprocess(this, node);        }        /*         * Updates the code path.         * And emits onCodePathStart/onCodePathSegmentStart events.         */        processCodePathToEnter(this, node);        // Emits node events.        this.original.enterNode(node);        this.currentNode = null;    }    /**     * Does the process to leave a given AST node.     * This updates state of analysis and calls `leaveNode` of the wrapped.     * @param {ASTNode} node A node which is leaving.     * @returns {void}     */    leaveNode(node) {        this.currentNode = node;        /*         * Updates the code path.         * And emits onCodePathStart/onCodePathSegmentStart events.         */        processCodePathToExit(this, node);        // Emits node events.        this.original.leaveNode(node);        // Emits the last onCodePathStart/onCodePathSegmentStart events.        postprocess(this, node);        this.currentNode = null;    }    /**     * This is called on a code path looped.     * Then this raises a looped event.     * @param {CodePathSegment} fromSegment A segment of prev.     * @param {CodePathSegment} toSegment A segment of next.     * @returns {void}     */    onLooped(fromSegment, toSegment) {        if (fromSegment.reachable && toSegment.reachable) {            debug.dump(`onCodePathSegmentLoop ${fromSegment.id} -> ${toSegment.id}`);            this.emitter.emit(                "onCodePathSegmentLoop",                fromSegment,                toSegment,                this.currentNode            );        }    }}module.exports = CodePathAnalyzer;
 |