NodeUtils.spec.ts 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290
  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('getBlockScopeOfNode (node: ESTree.Node, depth: number = 0): TNodeWithBlockStatement', () => {
  73. let functionDeclarationBlockStatementNode: ESTree.BlockStatement,
  74. ifStatementBlockStatementNode1: ESTree.BlockStatement,
  75. ifStatementBlockStatementNode2: ESTree.BlockStatement,
  76. ifStatementNode1: ESTree.IfStatement,
  77. ifStatementNode2: ESTree.IfStatement,
  78. expressionStatementNode1: ESTree.ExpressionStatement,
  79. expressionStatementNode2: ESTree.ExpressionStatement,
  80. expressionStatementNode3: 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. expressionStatementNode1,
  89. expressionStatementNode2
  90. ]);
  91. ifStatementNode2 = NodeMocks.getIfStatementNode(ifStatementBlockStatementNode2);
  92. ifStatementBlockStatementNode1 = NodeMocks.getBlockStatementNode([
  93. ifStatementNode2
  94. ]);
  95. ifStatementNode1 = NodeMocks.getIfStatementNode(ifStatementBlockStatementNode1);
  96. functionDeclarationBlockStatementNode = NodeMocks.getBlockStatementNode([
  97. expressionStatementNode3,
  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. expressionStatementNode3['parentNode'] = functionDeclarationBlockStatementNode;
  108. ifStatementNode1['parentNode'] = functionDeclarationBlockStatementNode;
  109. ifStatementBlockStatementNode1['parentNode'] = ifStatementNode1;
  110. ifStatementNode2['parentNode'] = ifStatementBlockStatementNode1;
  111. ifStatementBlockStatementNode2['parentNode'] = ifStatementNode2;
  112. expressionStatementNode1['parentNode'] = ifStatementBlockStatementNode2;
  113. });
  114. it('should return block-scope node for given node', () => {
  115. assert.deepEqual(NodeUtils.getBlockScopeOfNode(expressionStatementNode3), functionDeclarationBlockStatementNode);
  116. assert.deepEqual(NodeUtils.getBlockScopeOfNode(expressionStatementNode3, 1), programNode);
  117. assert.deepEqual(NodeUtils.getBlockScopeOfNode(functionDeclarationNode), programNode);
  118. assert.deepEqual(NodeUtils.getBlockScopeOfNode(functionDeclarationBlockStatementNode), programNode);
  119. assert.deepEqual(NodeUtils.getBlockScopeOfNode(programNode), programNode);
  120. assert.deepEqual(NodeUtils.getBlockScopeOfNode(expressionStatementNode1), functionDeclarationBlockStatementNode);
  121. });
  122. it('should throw a `ReferenceError` if node has no `parentNode` property', () => {
  123. assert.throws(() => NodeUtils.getBlockScopeOfNode(expressionStatementNode2), ReferenceError);
  124. });
  125. });
  126. describe('getNodeBlockScopeDepth (node: ESTree.Node, depth: number = 0): number', () => {
  127. let functionDeclarationBlockStatementNode1: ESTree.BlockStatement,
  128. functionDeclarationBlockStatementNode2: ESTree.BlockStatement,
  129. ifStatementBlockStatementNode1: ESTree.BlockStatement,
  130. ifStatementBlockStatementNode2: ESTree.BlockStatement,
  131. ifStatementNode1: ESTree.IfStatement,
  132. ifStatementNode2: ESTree.IfStatement,
  133. expressionStatementNode1: ESTree.ExpressionStatement,
  134. expressionStatementNode2: ESTree.ExpressionStatement,
  135. expressionStatementNode3: ESTree.ExpressionStatement,
  136. functionDeclarationNode1: ESTree.FunctionDeclaration,
  137. functionDeclarationNode2: ESTree.FunctionDeclaration,
  138. programNode: ESTree.Program;
  139. beforeEach(() => {
  140. expressionStatementNode1 = NodeMocks.getExpressionStatementNode();
  141. expressionStatementNode2 = NodeMocks.getExpressionStatementNode();
  142. expressionStatementNode3 = NodeMocks.getExpressionStatementNode();
  143. ifStatementBlockStatementNode2 = NodeMocks.getBlockStatementNode([
  144. expressionStatementNode3
  145. ]);
  146. ifStatementNode2 = NodeMocks.getIfStatementNode(ifStatementBlockStatementNode2);
  147. functionDeclarationBlockStatementNode2 = NodeMocks.getBlockStatementNode([
  148. ifStatementNode2,
  149. expressionStatementNode2
  150. ]);
  151. functionDeclarationNode2 = NodeMocks.getFunctionDeclarationNode('test', functionDeclarationBlockStatementNode2);
  152. ifStatementBlockStatementNode1 = NodeMocks.getBlockStatementNode([
  153. functionDeclarationNode2
  154. ]);
  155. ifStatementNode1 = NodeMocks.getIfStatementNode(ifStatementBlockStatementNode1);
  156. functionDeclarationBlockStatementNode1 = NodeMocks.getBlockStatementNode([
  157. expressionStatementNode1,
  158. ifStatementNode1
  159. ]);
  160. functionDeclarationNode1 = NodeMocks.getFunctionDeclarationNode('test', functionDeclarationBlockStatementNode1);
  161. programNode = NodeMocks.getProgramNode([
  162. functionDeclarationNode1
  163. ]);
  164. programNode['parentNode'] = programNode;
  165. functionDeclarationNode1['parentNode'] = programNode;
  166. functionDeclarationBlockStatementNode1['parentNode'] = functionDeclarationNode1;
  167. expressionStatementNode1['parentNode'] = functionDeclarationBlockStatementNode1;
  168. ifStatementNode1['parentNode'] = functionDeclarationBlockStatementNode1;
  169. ifStatementBlockStatementNode1['parentNode'] = ifStatementNode1;
  170. functionDeclarationNode2['parentNode'] = ifStatementBlockStatementNode1;
  171. functionDeclarationBlockStatementNode2['parentNode'] = functionDeclarationNode2;
  172. expressionStatementNode2['parentNode'] = functionDeclarationBlockStatementNode2;
  173. ifStatementNode2['parentNode'] = functionDeclarationBlockStatementNode2;
  174. ifStatementBlockStatementNode2['parentNode'] = ifStatementNode2;
  175. });
  176. it('should return block-scope depth for given node', () => {
  177. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(programNode), 0);
  178. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(functionDeclarationNode1), 0);
  179. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(functionDeclarationBlockStatementNode1), 1);
  180. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(expressionStatementNode1), 1);
  181. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(ifStatementNode1), 1);
  182. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(ifStatementBlockStatementNode1), 1);
  183. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(functionDeclarationNode2), 1);
  184. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(functionDeclarationBlockStatementNode2), 2);
  185. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(expressionStatementNode2), 2);
  186. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(ifStatementNode2), 2);
  187. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(ifStatementBlockStatementNode2), 2);
  188. });
  189. it('should throw a `ReferenceError` if node has no `parentNode` property', () => {
  190. assert.throws(() => NodeUtils.getNodeBlockScopeDepth(expressionStatementNode3), ReferenceError);
  191. });
  192. });
  193. describe('parentize (node: ESTree.Node): void', () => {
  194. let ifStatementNode: ESTree.IfStatement,
  195. ifStatementBlockStatementNode: ESTree.BlockStatement,
  196. expressionStatementNode1: ESTree.ExpressionStatement,
  197. expressionStatementNode2: ESTree.ExpressionStatement,
  198. programNode: ESTree.Program;
  199. beforeEach(() => {
  200. expressionStatementNode1 = NodeMocks.getExpressionStatementNode();
  201. expressionStatementNode2 = NodeMocks.getExpressionStatementNode();
  202. ifStatementBlockStatementNode = NodeMocks.getBlockStatementNode([
  203. expressionStatementNode1,
  204. expressionStatementNode2
  205. ]);
  206. ifStatementNode = NodeMocks.getIfStatementNode(ifStatementBlockStatementNode);
  207. });
  208. it('should parentize given AST-tree with `ProgramNode` as root node', () => {
  209. programNode = NodeMocks.getProgramNode([
  210. ifStatementNode
  211. ]);
  212. NodeUtils.parentize(programNode);
  213. assert.deepEqual(programNode['parentNode'], programNode);
  214. assert.deepEqual(ifStatementNode['parentNode'], programNode);
  215. assert.deepEqual(ifStatementBlockStatementNode['parentNode'], ifStatementNode);
  216. assert.deepEqual(expressionStatementNode1['parentNode'], ifStatementBlockStatementNode);
  217. assert.deepEqual(expressionStatementNode2['parentNode'], ifStatementBlockStatementNode);
  218. });
  219. it('should parentize given AST-tree', () => {
  220. programNode = NodeMocks.getProgramNode([
  221. ifStatementNode
  222. ]);
  223. programNode['parentNode'] = programNode;
  224. NodeUtils.parentize(ifStatementNode);
  225. assert.deepEqual(ifStatementNode['parentNode'], programNode);
  226. assert.deepEqual(ifStatementBlockStatementNode['parentNode'], ifStatementNode);
  227. assert.deepEqual(expressionStatementNode1['parentNode'], ifStatementBlockStatementNode);
  228. assert.deepEqual(expressionStatementNode2['parentNode'], ifStatementBlockStatementNode);
  229. });
  230. });
  231. });