| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326 | import { inject, injectable, } from 'inversify';import { ServiceIdentifiers } from '../../container/ServiceIdentifiers';import * as ESTree from 'estree';import { IOptions } from '../../interfaces/options/IOptions';import { IRandomGenerator } from '../../interfaces/utils/IRandomGenerator';import { IStatementSimplifyData } from '../../interfaces/node-transformers/simplifying-transformers/IStatementSimplifyData';import { IVisitor } from '../../interfaces/node-transformers/IVisitor';import { NodeTransformationStage } from '../../enums/node-transformers/NodeTransformationStage';import { AbstractStatementSimplifyTransformer } from './AbstractStatementSimplifyTransformer';import { NodeGuards } from '../../node/NodeGuards';import { NodeFactory } from '../../node/NodeFactory';import { NodeUtils } from '../../node/NodeUtils';/** * Simplifies `IfStatement` node */@injectable()export class IfStatementSimplifyTransformer extends AbstractStatementSimplifyTransformer {    /**     * @param {IRandomGenerator} randomGenerator     * @param {IOptions} options     */    public constructor (        @inject(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator,        @inject(ServiceIdentifiers.IOptions) options: IOptions    ) {        super(randomGenerator, options);    }    /**     * @param {NodeTransformationStage} nodeTransformationStage     * @returns {IVisitor | null}     */    public getVisitor (nodeTransformationStage: NodeTransformationStage): IVisitor | null {        switch (nodeTransformationStage) {            case NodeTransformationStage.Simplifying:                return {                    leave: (                        node: ESTree.Node,                        parentNode: ESTree.Node | null                    ): ESTree.Node | undefined => {                        if (parentNode && NodeGuards.isIfStatementNode(node)) {                            return this.transformNode(node, parentNode);                        }                    }                };            default:                return null;        }    }    /**     * @param {ESTree.IfStatement} ifStatementNode     * @param {ESTree.Node} parentNode     * @returns {ESTree.IfStatement}     */    public transformNode (        ifStatementNode: ESTree.IfStatement,        parentNode: ESTree.Node    ): ESTree.Node {        const consequentSimplifyData: IStatementSimplifyData | null = this.getStatementSimplifyData(ifStatementNode.consequent);        // Variant #1: no valid consequent expression data        if (!consequentSimplifyData) {            return ifStatementNode;        }        let transformedNode: ESTree.Node;        if (!ifStatementNode.alternate) {            // Variant #2: valid data for consequent expression only            transformedNode = this.getConsequentNode(ifStatementNode, consequentSimplifyData);        } else {            const alternateSimplifyData: IStatementSimplifyData | null = this.getStatementSimplifyData(ifStatementNode.alternate);            if (!alternateSimplifyData) {                return ifStatementNode;            }            // Variant #3: valid data for consequent and alternate expressions            transformedNode = this.getConsequentAndAlternateNode(ifStatementNode, consequentSimplifyData, alternateSimplifyData);        }        return NodeUtils.parentizeNode(transformedNode, parentNode);    }    /**     * @param {ESTree.IfStatement} ifStatementNode     * @param {IStatementSimplifyData} consequentSimplifyData     * @returns {ESTree.Node}     */    protected getConsequentNode (        ifStatementNode: ESTree.IfStatement,        consequentSimplifyData: IStatementSimplifyData    ): ESTree.Node {        /**         * Converts:         * if (true) {         *     const foo = 1;         *     console.log(1);         *     return 1;         * }         *         * to:         * if (true) {         *     const foo = 1;         *     return console.log(1), 1;         * }         */        if (            consequentSimplifyData.leadingStatements.length            || !consequentSimplifyData.trailingStatement        ) {            return NodeFactory.ifStatementNode(                ifStatementNode.test,                this.getPartialStatement(consequentSimplifyData)            );        }        /**         * Converts:         * if (true) {         *     return 1;         * }         *         * to:         * if (true)         *     return 1;         */        if (consequentSimplifyData.hasReturnStatement) {            return NodeFactory.ifStatementNode(                ifStatementNode.test,                consequentSimplifyData.trailingStatement.statement            );        }        /**         * Converts:         * if (true) {         *     console.log(1);         * }         *         * to:         * true && console.log(1);         */        return NodeFactory.expressionStatementNode(            NodeFactory.logicalExpressionNode(                '&&',                ifStatementNode.test,                consequentSimplifyData.trailingStatement.expression            )        );    }    /**     * @param {ESTree.IfStatement} ifStatementNode     * @param {IStatementSimplifyData} consequentSimplifyData     * @param {IStatementSimplifyData} alternateSimplifyData     * @returns {ESTree.Node}     */    protected getConsequentAndAlternateNode (        ifStatementNode: ESTree.IfStatement,        consequentSimplifyData: IStatementSimplifyData,        alternateSimplifyData: IStatementSimplifyData    ): ESTree.Node {        /**         * Converts:         * if (true) {         *     const foo = 1;         *     console.log(1);         *     return 1;         * }         *         * to:         * if (true) {         *     const foo = 1;         *     return console.log(1), 1;         * }         */        if (            consequentSimplifyData.leadingStatements.length            || alternateSimplifyData.leadingStatements.length            || !consequentSimplifyData.trailingStatement            || !alternateSimplifyData.trailingStatement        ) {            return NodeFactory.ifStatementNode(                ifStatementNode.test,                this.getPartialStatement(consequentSimplifyData),                this.getPartialStatement(alternateSimplifyData)            );        }        /**         * Converts:         * if (true) {         *     return 1;         * } else {         *     return 2;         * }         *         * to:         * return true ? 1 : 2;         */        if (consequentSimplifyData.hasReturnStatement && alternateSimplifyData.hasReturnStatement) {            return NodeFactory.returnStatementNode(                NodeFactory.conditionalExpressionNode(                    ifStatementNode.test,                    consequentSimplifyData.trailingStatement.expression,                    alternateSimplifyData.trailingStatement.expression                )            );        }        /**         * Converts:         * if (true) {         *     return 1;         * } else {         *     console.log(2);         * }         *         * to:         * if (true)         *     return 1;         * else         *     console.log(2);         */        if (consequentSimplifyData.hasReturnStatement || alternateSimplifyData.hasReturnStatement) {            return NodeFactory.ifStatementNode(                ifStatementNode.test,                consequentSimplifyData.trailingStatement.statement,                alternateSimplifyData.trailingStatement.statement            );        }        /**         * Converts:         * if (true) {         *     console.log(1);         * } else {         *     console.log(2);         * }         *         * to:         * true ? console.log(1) : console.log(2);         */        return NodeFactory.expressionStatementNode(            NodeFactory.conditionalExpressionNode(                ifStatementNode.test,                consequentSimplifyData.trailingStatement.expression,                alternateSimplifyData.trailingStatement.expression            )        );    }    /**     * @param {IStatementSimplifyData} statementSimplifyData     * @returns {ESTree.Statement}     */    protected getPartialStatement (statementSimplifyData: IStatementSimplifyData): ESTree.Statement {        const partialStatement: ESTree.Statement = super.getPartialStatement(statementSimplifyData);        if (!NodeGuards.isBlockStatementNode(partialStatement)) {            return partialStatement;        }        return partialStatement.body.length === 1        && !this.isProhibitedSingleStatementForIfStatementBranch(partialStatement.body[0])            ? partialStatement.body[0]            : partialStatement;    }    /**     * @param {ESTree.Statement} statement     * @returns {boolean}     */    protected isProhibitedSingleStatementForIfStatementBranch (statement: ESTree.Statement): boolean {        /**         * Function declaration is not allowed outside of block in `strict` mode         */        return NodeGuards.isFunctionDeclarationNode(statement)            /**             * Have to ignore all `IfStatement` nodes             * Also have to ignore any nodes with a single statement as a `body`             * Without ignore it can break following code:             * Input:             * if (condition1) {             *     if (condition2) {             *         var foo = bar();             *     }             * } else {             *     var baz = bark();             * }             *             * Invalid output:             * if (condition1)             *     if (condition2)             *         var foo = bar();             *     else             *         var baz = bark();             *             * See issue: https://github.com/javascript-obfuscator/javascript-obfuscator/issues/860             */            || NodeGuards.isIfStatementNode(statement)            || NodeGuards.isNodeWithSingleStatementBody(statement)            /**             * `let` and `const` variable declarations are not allowed outside of `IfStatement` block statement             * Input:             * if (condition1) {             *     const foo = 1;             * }             *             * Invalid output with runtime error:             * if (condition1)             *     const foo = 1;             */            || (NodeGuards.isVariableDeclarationNode(statement) && statement.kind !== 'var');    }}
 |