import { ServiceIdentifiers } from '../../../../src/container/ServiceIdentifiers'; import * as ESTree from 'estree'; import { assert } from 'chai'; import { TStatement } from '../../../../src/types/node/TStatement'; import { IInversifyContainerFacade } from '../../../../src/interfaces/container/IInversifyContainerFacade'; import { IStackTraceAnalyzer } from '../../../../src/interfaces/stack-trace-analyzer/IStackTraceAnalyzer'; import { IStackTraceData } from '../../../../src/interfaces/stack-trace-analyzer/IStackTraceData'; import { readFileAsString } from '../../../helpers/readFileAsString'; import { InversifyContainerFacade } from '../../../../src/container/InversifyContainerFacade'; import { NodeAppender } from '../../../../src/node/NodeAppender'; import { Nodes } from '../../../../src/node/Nodes'; import { NodeUtils } from '../../../../src/node/NodeUtils'; /** * @param fixturePath * @return {TStatement[]} */ const convertCodeToStructure: (fixturePath: string) => TStatement[] = (fixturePath) => { return NodeUtils.convertCodeToStructure( readFileAsString(`${__dirname}${fixturePath}`) ); }; /** * @param fixturePath * @return {ESTree.Program} */ const convertCodeToAst: (fixturePath: string) => ESTree.Program = (fixturePath) => { return Nodes.getProgramNode(convertCodeToStructure(fixturePath)); }; describe('NodeAppender', () => { describe('appendNode (blockScopeNode: TNodeWithBlockStatement[], nodeBodyStatements: TStatement[]): void', () => { let astTree: ESTree.Program, expectedAstTree: ESTree.Program, node: TStatement[]; before(() => { node = convertCodeToStructure('/fixtures/simple-input.js'); astTree = convertCodeToAst('/fixtures/append-node.js'); expectedAstTree = convertCodeToAst('/fixtures/append-node-expected.js'); astTree = NodeUtils.parentize(astTree); expectedAstTree = NodeUtils.parentize(expectedAstTree); NodeAppender.appendNode(astTree, node); }); it('should append given node to a `BlockStatement` node body', () => { assert.deepEqual(astTree, expectedAstTree); }); }); describe('appendNodeToOptimalBlockScope (blockScopeStackTraceData: IStackTraceData[], blockScopeNode: TNodeWithBlockStatement, nodeBodyStatements: TStatement[], index: number = 0): void', () => { let stackTraceAnalyzer: IStackTraceAnalyzer, astTree: ESTree.Program, expectedAstTree: ESTree.Program, node: TStatement[], stackTraceData: IStackTraceData[]; before(() => { const inversifyContainerFacade: IInversifyContainerFacade = new InversifyContainerFacade(); inversifyContainerFacade.load({}); stackTraceAnalyzer = inversifyContainerFacade .get(ServiceIdentifiers.IStackTraceAnalyzer); }); beforeEach(() => { node = convertCodeToStructure('/fixtures/simple-input.js'); }); describe('variant #1: nested function calls', () => { beforeEach(() => { astTree = convertCodeToAst('/fixtures/append-node-to-optimal-block-scope/variant-1.js'); expectedAstTree = convertCodeToAst('/fixtures/append-node-to-optimal-block-scope/variant-1-expected.js'); stackTraceData = stackTraceAnalyzer.analyze(astTree.body); NodeAppender.appendNodeToOptimalBlockScope(stackTraceData, astTree, node); }); it('should append node into first and deepest function call in nested function calls', () => { assert.deepEqual(astTree, expectedAstTree); }); }); describe('variant #2: nested function calls', () => { beforeEach(() => { astTree = convertCodeToAst('/fixtures/append-node-to-optimal-block-scope/variant-2.js'); expectedAstTree = convertCodeToAst('/fixtures/append-node-to-optimal-block-scope/variant-2-expected.js'); stackTraceData = stackTraceAnalyzer.analyze(astTree.body); NodeAppender.appendNodeToOptimalBlockScope(stackTraceData, astTree, node); }); it('should append node into first and deepest function call in nested function calls', () => { assert.deepEqual(astTree, expectedAstTree); }); }); describe('append by specific index', () => { let astTree: ESTree.Program; beforeEach(() => { astTree = convertCodeToAst('/fixtures/append-node-to-optimal-block-scope/by-index.js'); }); describe('variant #1: append by specific index in nested function calls', () => { beforeEach(() => { expectedAstTree = convertCodeToAst('/fixtures/append-node-to-optimal-block-scope/by-index-variant-1-expected.js'); stackTraceData = stackTraceAnalyzer.analyze(astTree.body); NodeAppender.appendNodeToOptimalBlockScope(stackTraceData, astTree, node, 2); }); it('should append node into deepest function call by specified index in nested function calls', () => { assert.deepEqual(astTree, expectedAstTree); }); }); describe('variant #2: append by specific index in nested function calls', () => { beforeEach(() => { expectedAstTree = convertCodeToAst('/fixtures/append-node-to-optimal-block-scope/by-index-variant-2-expected.js'); stackTraceData = stackTraceAnalyzer.analyze(astTree.body); NodeAppender.appendNodeToOptimalBlockScope(stackTraceData, astTree, node, 1); }); it('should append node into deepest function call by specified index in nested function calls', () => { assert.deepEqual(astTree, expectedAstTree); }); }); describe('variant #3: append by specific index in nested function calls', () => { beforeEach(() => { astTree = convertCodeToAst('/fixtures/append-node-to-optimal-block-scope/by-index-variant-3.js'); expectedAstTree = convertCodeToAst('/fixtures/append-node-to-optimal-block-scope/by-index-variant-3-expected.js'); stackTraceData = stackTraceAnalyzer.analyze(astTree.body); NodeAppender.appendNodeToOptimalBlockScope( stackTraceData, astTree, node, stackTraceData.length - 1 ); }); it('should append node into deepest function call by specified index in nested function calls', () => { assert.deepEqual(astTree, expectedAstTree); }); }); }); }); describe('insertNodeAtIndex (blockScopeNode: TNodeWithBlockStatement[], nodeBodyStatements: TStatement[], index: number): void', () => { let astTree: ESTree.Program, expectedAstTree: ESTree.Program, node: TStatement[]; before(() => { node = convertCodeToStructure('/fixtures/simple-input.js'); astTree = convertCodeToAst('/fixtures/insert-node-at-index.js'); expectedAstTree = convertCodeToAst('/fixtures/insert-node-at-index-expected.js'); astTree = NodeUtils.parentize(astTree); expectedAstTree = NodeUtils.parentize(expectedAstTree); NodeAppender.insertNodeAtIndex(astTree, node, 2); }); it('should insert given node in `BlockStatement` node body at index', () => { assert.deepEqual(astTree, expectedAstTree); }); }); describe('prependNode (blockScopeNode: TNodeWithBlockStatement[], nodeBodyStatements: TStatement[]): void', () => { let astTree: ESTree.Program, expectedAstTree: ESTree.Program, node: TStatement[]; before(() => { node = convertCodeToStructure('/fixtures/simple-input.js'); astTree = convertCodeToAst('/fixtures/prepend-node.js'); expectedAstTree = convertCodeToAst('/fixtures/prepend-node-expected.js'); astTree = NodeUtils.parentize(astTree); expectedAstTree = NodeUtils.parentize(expectedAstTree); NodeAppender.prependNode(astTree, node); }); it('should prepend given node to a `BlockStatement` node body', () => { assert.deepEqual(astTree, expectedAstTree); }); }); });