NodeUtils.spec.ts 14 KB

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