NodeUtils.spec.ts 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321
  1. import * as chai from 'chai';
  2. import { IBlockStatementNode } from "../../src/interfaces/nodes/IBlockStatementNode";
  3. import { IExpressionStatementNode } from "../../src/interfaces/nodes/IExpressionStatementNode";
  4. import { IFunctionDeclarationNode } from "../../src/interfaces/nodes/IFunctionDeclarationNode";
  5. import { IIdentifierNode } from "../../src/interfaces/nodes/IIdentifierNode";
  6. import { IIfStatementNode } from "../../src/interfaces/nodes/IIfStatementNode";
  7. import { ILiteralNode } from "../../src/interfaces/nodes/ILiteralNode";
  8. import { INode } from "../../src/interfaces/nodes/INode";
  9. import { IProgramNode } from "../../src/interfaces/nodes/IProgramNode";
  10. import { IVariableDeclarationNode } from "../../src/interfaces/nodes/IVariableDeclarationNode";
  11. import { IVariableDeclaratorNode } from "../../src/interfaces/nodes/IVariableDeclaratorNode";
  12. import { NodeMocks } from '../mocks/NodeMocks';
  13. import { NodeUtils } from '../../src/NodeUtils';
  14. chai.use(require('chai-members-deep'));
  15. const assert: any = chai.assert;
  16. describe('NodeUtils', () => {
  17. describe('addXVerbatimPropertyToLiterals (node: INode): void', () => {
  18. let literalNode: any,
  19. expectedLiteralNode: any;
  20. beforeEach(() => {
  21. literalNode = NodeMocks.getLiteralNode();
  22. delete literalNode['x-verbatim-property'];
  23. expectedLiteralNode = NodeMocks.getLiteralNode();
  24. NodeUtils.addXVerbatimPropertyToLiterals(literalNode);
  25. });
  26. it('should add `x-verbatim-property` to `Literal` node', () => {
  27. assert.deepEqual(literalNode, expectedLiteralNode);
  28. });
  29. });
  30. describe('appendNode (blockScopeBody: INode[], node: INode): void', () => {
  31. let blockStatementNode: IBlockStatementNode,
  32. expectedBlockStatementNode: IBlockStatementNode,
  33. expressionStatementNode: IExpressionStatementNode;
  34. beforeEach(() => {
  35. expressionStatementNode = NodeMocks.getExpressionStatementNode();
  36. blockStatementNode = NodeMocks.getBlockStatementNode();
  37. expectedBlockStatementNode = NodeMocks.getBlockStatementNode([
  38. expressionStatementNode
  39. ]);
  40. NodeUtils.appendNode(blockStatementNode.body, expressionStatementNode);
  41. });
  42. it('should append given node to a `BlockStatement` node body', () => {
  43. assert.deepEqual(blockStatementNode, expectedBlockStatementNode);
  44. });
  45. it('should does not change `BlockStatement` node body if given node is not a valid Node', () => {
  46. assert.doesNotChange(
  47. () => NodeUtils.appendNode(blockStatementNode.body, <INode>{}),
  48. blockStatementNode,
  49. 'body'
  50. );
  51. });
  52. });
  53. describe('convertCodeToStructure (code: string): INode', () => {
  54. let code: string,
  55. identifierNode: IIdentifierNode,
  56. literalNode: ILiteralNode,
  57. programNode: IProgramNode,
  58. variableDeclarationNode: IVariableDeclarationNode,
  59. variableDeclaratorNode: IVariableDeclaratorNode;
  60. beforeEach(() => {
  61. code = `
  62. var abc = 'cde';
  63. `;
  64. identifierNode = NodeMocks.getIdentifierNode('abc');
  65. literalNode = NodeMocks.getLiteralNode('cde');
  66. variableDeclaratorNode = NodeMocks.getVariableDeclaratorNode(identifierNode, literalNode);
  67. variableDeclarationNode = NodeMocks.getVariableDeclarationNode([
  68. variableDeclaratorNode
  69. ]);
  70. programNode = NodeMocks.getProgramNode([
  71. variableDeclarationNode
  72. ]);
  73. programNode['parentNode'] = programNode;
  74. variableDeclarationNode['parentNode'] = programNode;
  75. variableDeclaratorNode['parentNode'] = variableDeclarationNode;
  76. identifierNode['parentNode'] = variableDeclaratorNode;
  77. literalNode['parentNode'] = variableDeclaratorNode;
  78. });
  79. it('should convert code to `INode` structure', () => {
  80. assert.deepEqual(NodeUtils.convertCodeToStructure(code), variableDeclarationNode);
  81. });
  82. });
  83. describe('getBlockStatementNodeByIndex (node: INode, index: number = 0): INode', () => {
  84. let blockStatementNode: IBlockStatementNode,
  85. expressionStatementNode1: IExpressionStatementNode,
  86. expressionStatementNode2: IExpressionStatementNode;
  87. beforeEach(() => {
  88. expressionStatementNode1 = NodeMocks.getExpressionStatementNode();
  89. expressionStatementNode2 = NodeMocks.getExpressionStatementNode();
  90. blockStatementNode = NodeMocks.getBlockStatementNode([
  91. expressionStatementNode1,
  92. expressionStatementNode2
  93. ]);
  94. });
  95. it('should return block-statement child node of given node if that node has block-statement', () => {
  96. assert.deepEqual(NodeUtils.getBlockStatementNodeByIndex(blockStatementNode), expressionStatementNode1);
  97. assert.deepEqual(NodeUtils.getBlockStatementNodeByIndex(blockStatementNode, 1), expressionStatementNode2);
  98. });
  99. it('should throw a `ReferenceError` if index is out of boundaries', () => {
  100. assert.throws(() => NodeUtils.getBlockStatementNodeByIndex(blockStatementNode, 2), ReferenceError);
  101. });
  102. it('should throw a `TypeError` if node have no a block-statement', () => {
  103. assert.throws(() => NodeUtils.getBlockStatementNodeByIndex(expressionStatementNode1, 1), TypeError);
  104. });
  105. });
  106. describe('getBlockScopeOfNode (node: INode, depth: number = 0): TNodeWithBlockStatement', () => {
  107. let functionDeclarationBlockStatementNode: IBlockStatementNode,
  108. ifStatementBlockStatementNode1: IBlockStatementNode,
  109. ifStatementBlockStatementNode2: IBlockStatementNode,
  110. ifStatementNode1: IIfStatementNode,
  111. ifStatementNode2: IIfStatementNode,
  112. expressionStatementNode1: IExpressionStatementNode,
  113. expressionStatementNode2: IExpressionStatementNode,
  114. expressionStatementNode3: IExpressionStatementNode,
  115. functionDeclarationNode: IFunctionDeclarationNode,
  116. programNode: IProgramNode;
  117. beforeEach(() => {
  118. expressionStatementNode1 = NodeMocks.getExpressionStatementNode();
  119. expressionStatementNode2 = NodeMocks.getExpressionStatementNode();
  120. expressionStatementNode3 = NodeMocks.getExpressionStatementNode();
  121. ifStatementBlockStatementNode2 = NodeMocks.getBlockStatementNode([
  122. expressionStatementNode1,
  123. expressionStatementNode2
  124. ]);
  125. ifStatementNode2 = NodeMocks.getIfStatementNode(ifStatementBlockStatementNode2);
  126. ifStatementBlockStatementNode1 = NodeMocks.getBlockStatementNode([
  127. ifStatementNode2
  128. ]);
  129. ifStatementNode1 = NodeMocks.getIfStatementNode(ifStatementBlockStatementNode1);
  130. functionDeclarationBlockStatementNode = NodeMocks.getBlockStatementNode([
  131. expressionStatementNode3,
  132. ifStatementNode1
  133. ]);
  134. functionDeclarationNode = NodeMocks.getFunctionDeclarationNode('test', functionDeclarationBlockStatementNode);
  135. programNode = NodeMocks.getProgramNode([
  136. functionDeclarationNode
  137. ]);
  138. programNode['parentNode'] = programNode;
  139. functionDeclarationNode['parentNode'] = programNode;
  140. functionDeclarationBlockStatementNode['parentNode'] = functionDeclarationNode;
  141. expressionStatementNode3['parentNode'] = functionDeclarationBlockStatementNode;
  142. ifStatementNode1['parentNode'] = functionDeclarationBlockStatementNode;
  143. ifStatementBlockStatementNode1['parentNode'] = ifStatementNode1;
  144. ifStatementNode2['parentNode'] = ifStatementBlockStatementNode1;
  145. ifStatementBlockStatementNode2['parentNode'] = ifStatementNode2;
  146. expressionStatementNode1['parentNode'] = ifStatementBlockStatementNode2;
  147. });
  148. it('should return block-scope node for given node', () => {
  149. assert.deepEqual(NodeUtils.getBlockScopeOfNode(expressionStatementNode3), functionDeclarationBlockStatementNode);
  150. assert.deepEqual(NodeUtils.getBlockScopeOfNode(expressionStatementNode3, 1), programNode);
  151. assert.deepEqual(NodeUtils.getBlockScopeOfNode(functionDeclarationNode), programNode);
  152. assert.deepEqual(NodeUtils.getBlockScopeOfNode(functionDeclarationBlockStatementNode), programNode);
  153. assert.deepEqual(NodeUtils.getBlockScopeOfNode(programNode), programNode);
  154. assert.deepEqual(NodeUtils.getBlockScopeOfNode(expressionStatementNode1), functionDeclarationBlockStatementNode);
  155. });
  156. it('should throw a `ReferenceError` if node has no `parentNode` property', () => {
  157. assert.throws(() => NodeUtils.getBlockScopeOfNode(expressionStatementNode2), ReferenceError);
  158. });
  159. });
  160. describe('insertNodeAtIndex (blockScopeBody: INode[], node: INode, index: number): void', () => {
  161. let blockStatementNode: IBlockStatementNode,
  162. expectedBlockStatementNode: IBlockStatementNode,
  163. expressionStatementNode1: IExpressionStatementNode,
  164. expressionStatementNode2: IExpressionStatementNode;
  165. beforeEach(() => {
  166. expressionStatementNode1 = NodeMocks.getExpressionStatementNode();
  167. expressionStatementNode2 = NodeMocks.getExpressionStatementNode();
  168. blockStatementNode = NodeMocks.getBlockStatementNode([
  169. expressionStatementNode1
  170. ]);
  171. expectedBlockStatementNode = NodeMocks.getBlockStatementNode([
  172. expressionStatementNode1,
  173. expressionStatementNode1
  174. ]);
  175. NodeUtils.insertNodeAtIndex(blockStatementNode.body, expressionStatementNode2, 1);
  176. });
  177. it('should insert given node in `BlockStatement` node body at index', () => {
  178. assert.deepEqual(blockStatementNode, expectedBlockStatementNode);
  179. });
  180. it('should does not change `BlockStatement` node body if given node is not a valid Node', () => {
  181. assert.doesNotChange(
  182. () => NodeUtils.insertNodeAtIndex(blockStatementNode.body, <INode>{}, 1),
  183. blockStatementNode,
  184. 'body'
  185. );
  186. });
  187. });
  188. describe('parentize (node: INode): void', () => {
  189. let ifStatementNode: IIfStatementNode,
  190. ifStatementBlockStatementNode: IBlockStatementNode,
  191. expressionStatementNode1: IExpressionStatementNode,
  192. expressionStatementNode2: IExpressionStatementNode,
  193. programNode: IProgramNode;
  194. beforeEach(() => {
  195. expressionStatementNode1 = NodeMocks.getExpressionStatementNode();
  196. expressionStatementNode2 = NodeMocks.getExpressionStatementNode();
  197. ifStatementBlockStatementNode = NodeMocks.getBlockStatementNode([
  198. expressionStatementNode1,
  199. expressionStatementNode2
  200. ]);
  201. ifStatementNode = NodeMocks.getIfStatementNode(ifStatementBlockStatementNode);
  202. });
  203. it('should parentize given AST-tree with `ProgramNode` as root node', () => {
  204. programNode = NodeMocks.getProgramNode([
  205. ifStatementNode
  206. ]);
  207. NodeUtils.parentize(programNode);
  208. assert.deepEqual(programNode['parentNode'], programNode);
  209. assert.deepEqual(ifStatementNode['parentNode'], programNode);
  210. assert.deepEqual(ifStatementBlockStatementNode['parentNode'], ifStatementNode);
  211. assert.deepEqual(expressionStatementNode1['parentNode'], ifStatementBlockStatementNode);
  212. assert.deepEqual(expressionStatementNode2['parentNode'], ifStatementBlockStatementNode);
  213. });
  214. it('should parentize given AST-tree', () => {
  215. programNode = NodeMocks.getProgramNode([
  216. ifStatementNode
  217. ]);
  218. programNode['parentNode'] = programNode;
  219. NodeUtils.parentize(ifStatementNode);
  220. assert.deepEqual(ifStatementNode['parentNode'], programNode);
  221. assert.deepEqual(ifStatementBlockStatementNode['parentNode'], ifStatementNode);
  222. assert.deepEqual(expressionStatementNode1['parentNode'], ifStatementBlockStatementNode);
  223. assert.deepEqual(expressionStatementNode2['parentNode'], ifStatementBlockStatementNode);
  224. });
  225. });
  226. describe('prependNode (blockScopeBody: INode[], node: INode): void', () => {
  227. let blockStatementNode: IBlockStatementNode,
  228. expectedBlockStatementNode: IBlockStatementNode,
  229. expressionStatementNode1: IExpressionStatementNode,
  230. expressionStatementNode2: IExpressionStatementNode;
  231. beforeEach(() => {
  232. expressionStatementNode1 = NodeMocks.getExpressionStatementNode();
  233. expressionStatementNode2 = NodeMocks.getExpressionStatementNode();
  234. blockStatementNode = NodeMocks.getBlockStatementNode([
  235. expressionStatementNode1
  236. ]);
  237. expectedBlockStatementNode = NodeMocks.getBlockStatementNode([
  238. expressionStatementNode2,
  239. expressionStatementNode1
  240. ]);
  241. NodeUtils.prependNode(blockStatementNode.body, expressionStatementNode2);
  242. });
  243. it('should prepend given node to a `BlockStatement` node body', () => {
  244. assert.deepEqual(blockStatementNode, expectedBlockStatementNode);
  245. });
  246. it('should does not change `BlockStatement` node body if given node is not a valid Node', () => {
  247. assert.doesNotChange(
  248. () => NodeUtils.prependNode(blockStatementNode.body, <INode>{}),
  249. blockStatementNode,
  250. 'body'
  251. );
  252. });
  253. });
  254. });