NodeUtils.spec.ts 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334
  1. import * as ESTree from 'estree';
  2. import { assert } from 'chai';
  3. import { Nodes } from '../../../src/node/Nodes';
  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 = Nodes.getLiteralNode('value');
  11. delete literalNode['x-verbatim-property'];
  12. expectedLiteralNode = Nodes.getLiteralNode('value');
  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 = Nodes.getIdentifierNode('abc');
  31. literalNode = Nodes.getLiteralNode('cde');
  32. variableDeclaratorNode = Nodes.getVariableDeclaratorNode(identifierNode, literalNode);
  33. variableDeclarationNode = Nodes.getVariableDeclarationNode([
  34. variableDeclaratorNode
  35. ]);
  36. programNode = Nodes.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('convertStructureToCode (structure: ESTree.Node[]): string', () => {
  50. let structure: ESTree.Node[],
  51. expectedCode: string;
  52. beforeEach(() => {
  53. structure = [
  54. Nodes.getProgramNode([
  55. Nodes.getVariableDeclarationNode([
  56. Nodes.getVariableDeclaratorNode(
  57. Nodes.getIdentifierNode('abc'),
  58. Nodes.getLiteralNode('cde')
  59. )
  60. ])
  61. ])
  62. ];
  63. expectedCode = 'var abc = \'cde\';';
  64. });
  65. it('should convert `ESTree.Node[]` structure to source code', () => {
  66. assert.deepEqual(NodeUtils.convertStructureToCode(structure), expectedCode);
  67. });
  68. });
  69. describe('getBlockStatementNodeByIndex (node: ESTree.Node, index: number = 0): ESTree.Node', () => {
  70. let blockStatementNode: ESTree.BlockStatement,
  71. expressionStatementNode1: ESTree.ExpressionStatement,
  72. expressionStatementNode2: ESTree.ExpressionStatement;
  73. beforeEach(() => {
  74. expressionStatementNode1 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  75. expressionStatementNode2 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  76. blockStatementNode = Nodes.getBlockStatementNode([
  77. expressionStatementNode1,
  78. expressionStatementNode2
  79. ]);
  80. });
  81. it('should return block-statement child node of given node if that node has block-statement', () => {
  82. assert.deepEqual(NodeUtils.getBlockStatementNodeByIndex(blockStatementNode), expressionStatementNode1);
  83. assert.deepEqual(NodeUtils.getBlockStatementNodeByIndex(blockStatementNode, 1), expressionStatementNode2);
  84. });
  85. it('should throw a `ReferenceError` if index is out of boundaries', () => {
  86. assert.throws(() => NodeUtils.getBlockStatementNodeByIndex(blockStatementNode, 2), ReferenceError);
  87. });
  88. it('should throw a `TypeError` if node have no a block-statement', () => {
  89. assert.throws(() => NodeUtils.getBlockStatementNodeByIndex(expressionStatementNode1, 1), TypeError);
  90. });
  91. });
  92. describe('getBlockScopesOfNode (node: ESTree.Node, blockScopes: TNodeWithBlockStatement[] = []): TNodeWithBlockStatement[]', () => {
  93. let functionDeclarationBlockStatementNode: ESTree.BlockStatement,
  94. ifStatementBlockStatementNode1: ESTree.BlockStatement,
  95. ifStatementBlockStatementNode2: ESTree.BlockStatement,
  96. ifStatementNode1: ESTree.IfStatement,
  97. ifStatementNode2: ESTree.IfStatement,
  98. expressionStatementNode3: ESTree.ExpressionStatement,
  99. expressionStatementNode2: ESTree.ExpressionStatement,
  100. expressionStatementNode1: ESTree.ExpressionStatement,
  101. functionDeclarationNode: ESTree.FunctionDeclaration,
  102. programNode: ESTree.Program;
  103. beforeEach(() => {
  104. expressionStatementNode1 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  105. expressionStatementNode2 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  106. expressionStatementNode3 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  107. ifStatementBlockStatementNode2 = Nodes.getBlockStatementNode([
  108. expressionStatementNode2,
  109. expressionStatementNode3
  110. ]);
  111. ifStatementNode2 = Nodes.getIfStatementNode(
  112. Nodes.getLiteralNode(true),
  113. ifStatementBlockStatementNode2
  114. );
  115. ifStatementBlockStatementNode1 = Nodes.getBlockStatementNode([
  116. ifStatementNode2
  117. ]);
  118. ifStatementNode1 = Nodes.getIfStatementNode(
  119. Nodes.getLiteralNode(true),
  120. ifStatementBlockStatementNode1
  121. );
  122. functionDeclarationBlockStatementNode = Nodes.getBlockStatementNode([
  123. expressionStatementNode1,
  124. ifStatementNode1
  125. ]);
  126. functionDeclarationNode = Nodes.getFunctionDeclarationNode('test', [], functionDeclarationBlockStatementNode);
  127. programNode = Nodes.getProgramNode([
  128. functionDeclarationNode
  129. ]);
  130. programNode['parentNode'] = programNode;
  131. functionDeclarationNode['parentNode'] = programNode;
  132. functionDeclarationBlockStatementNode['parentNode'] = functionDeclarationNode;
  133. expressionStatementNode1['parentNode'] = functionDeclarationBlockStatementNode;
  134. ifStatementNode1['parentNode'] = functionDeclarationBlockStatementNode;
  135. ifStatementBlockStatementNode1['parentNode'] = ifStatementNode1;
  136. ifStatementNode2['parentNode'] = ifStatementBlockStatementNode1;
  137. ifStatementBlockStatementNode2['parentNode'] = ifStatementNode2;
  138. expressionStatementNode3['parentNode'] = ifStatementBlockStatementNode2;
  139. });
  140. it('should return block-scope node for given node', () => {
  141. assert.deepEqual(NodeUtils.getBlockScopesOfNode(programNode)[0], programNode);
  142. assert.deepEqual(NodeUtils.getBlockScopesOfNode(functionDeclarationNode)[0], programNode);
  143. assert.deepEqual(NodeUtils.getBlockScopesOfNode(functionDeclarationBlockStatementNode)[0], programNode);
  144. assert.deepEqual(NodeUtils.getBlockScopesOfNode(expressionStatementNode1)[0], functionDeclarationBlockStatementNode);
  145. assert.deepEqual(NodeUtils.getBlockScopesOfNode(expressionStatementNode1)[1], programNode);
  146. assert.deepEqual(NodeUtils.getBlockScopesOfNode(ifStatementNode1)[0], functionDeclarationBlockStatementNode);
  147. assert.deepEqual(NodeUtils.getBlockScopesOfNode(ifStatementNode1)[1], programNode);
  148. assert.deepEqual(NodeUtils.getBlockScopesOfNode(ifStatementBlockStatementNode1)[0], functionDeclarationBlockStatementNode);
  149. assert.deepEqual(NodeUtils.getBlockScopesOfNode(ifStatementBlockStatementNode1)[1], programNode);
  150. assert.deepEqual(NodeUtils.getBlockScopesOfNode(ifStatementBlockStatementNode2)[0], functionDeclarationBlockStatementNode);
  151. assert.deepEqual(NodeUtils.getBlockScopesOfNode(ifStatementBlockStatementNode2)[1], programNode);
  152. assert.deepEqual(NodeUtils.getBlockScopesOfNode(expressionStatementNode3)[0], functionDeclarationBlockStatementNode);
  153. assert.deepEqual(NodeUtils.getBlockScopesOfNode(expressionStatementNode3)[1], programNode);
  154. });
  155. it('should throw a `ReferenceError` if node has no `parentNode` property', () => {
  156. assert.throws(() => NodeUtils.getBlockScopesOfNode(expressionStatementNode2)[0], ReferenceError);
  157. });
  158. });
  159. describe('getNodeBlockScopeDepth (node: ESTree.Node, depth: number = 0): number', () => {
  160. let functionDeclarationBlockStatementNode1: ESTree.BlockStatement,
  161. functionDeclarationBlockStatementNode2: ESTree.BlockStatement,
  162. ifStatementBlockStatementNode1: ESTree.BlockStatement,
  163. ifStatementBlockStatementNode2: ESTree.BlockStatement,
  164. ifStatementNode1: ESTree.IfStatement,
  165. ifStatementNode2: ESTree.IfStatement,
  166. expressionStatementNode1: ESTree.ExpressionStatement,
  167. expressionStatementNode2: ESTree.ExpressionStatement,
  168. expressionStatementNode3: ESTree.ExpressionStatement,
  169. functionDeclarationNode1: ESTree.FunctionDeclaration,
  170. functionDeclarationNode2: ESTree.FunctionDeclaration,
  171. programNode: ESTree.Program;
  172. beforeEach(() => {
  173. expressionStatementNode1 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  174. expressionStatementNode2 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  175. expressionStatementNode3 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  176. ifStatementBlockStatementNode2 = Nodes.getBlockStatementNode([
  177. expressionStatementNode3
  178. ]);
  179. ifStatementNode2 = Nodes.getIfStatementNode(
  180. Nodes.getLiteralNode(true),
  181. ifStatementBlockStatementNode2
  182. );
  183. functionDeclarationBlockStatementNode2 = Nodes.getBlockStatementNode([
  184. ifStatementNode2,
  185. expressionStatementNode2
  186. ]);
  187. functionDeclarationNode2 = Nodes.getFunctionDeclarationNode('test', [], functionDeclarationBlockStatementNode2);
  188. ifStatementBlockStatementNode1 = Nodes.getBlockStatementNode([
  189. functionDeclarationNode2
  190. ]);
  191. ifStatementNode1 = Nodes.getIfStatementNode(
  192. Nodes.getLiteralNode(true),
  193. ifStatementBlockStatementNode1
  194. );
  195. functionDeclarationBlockStatementNode1 = Nodes.getBlockStatementNode([
  196. expressionStatementNode1,
  197. ifStatementNode1
  198. ]);
  199. functionDeclarationNode1 = Nodes.getFunctionDeclarationNode('test', [], functionDeclarationBlockStatementNode1);
  200. programNode = Nodes.getProgramNode([
  201. functionDeclarationNode1
  202. ]);
  203. programNode['parentNode'] = programNode;
  204. functionDeclarationNode1['parentNode'] = programNode;
  205. functionDeclarationBlockStatementNode1['parentNode'] = functionDeclarationNode1;
  206. expressionStatementNode1['parentNode'] = functionDeclarationBlockStatementNode1;
  207. ifStatementNode1['parentNode'] = functionDeclarationBlockStatementNode1;
  208. ifStatementBlockStatementNode1['parentNode'] = ifStatementNode1;
  209. functionDeclarationNode2['parentNode'] = ifStatementBlockStatementNode1;
  210. functionDeclarationBlockStatementNode2['parentNode'] = functionDeclarationNode2;
  211. expressionStatementNode2['parentNode'] = functionDeclarationBlockStatementNode2;
  212. ifStatementNode2['parentNode'] = functionDeclarationBlockStatementNode2;
  213. ifStatementBlockStatementNode2['parentNode'] = ifStatementNode2;
  214. });
  215. it('should return block-scope depth for given node', () => {
  216. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(programNode), 0);
  217. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(functionDeclarationNode1), 0);
  218. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(functionDeclarationBlockStatementNode1), 1);
  219. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(expressionStatementNode1), 1);
  220. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(ifStatementNode1), 1);
  221. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(ifStatementBlockStatementNode1), 1);
  222. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(functionDeclarationNode2), 1);
  223. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(functionDeclarationBlockStatementNode2), 2);
  224. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(expressionStatementNode2), 2);
  225. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(ifStatementNode2), 2);
  226. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(ifStatementBlockStatementNode2), 2);
  227. });
  228. it('should throw a `ReferenceError` if node has no `parentNode` property', () => {
  229. assert.throws(() => NodeUtils.getNodeBlockScopeDepth(expressionStatementNode3), ReferenceError);
  230. });
  231. });
  232. describe('parentize (node: ESTree.Node): void', () => {
  233. let ifStatementNode: ESTree.IfStatement,
  234. ifStatementBlockStatementNode: ESTree.BlockStatement,
  235. expressionStatementNode1: ESTree.ExpressionStatement,
  236. expressionStatementNode2: ESTree.ExpressionStatement,
  237. programNode: ESTree.Program;
  238. beforeEach(() => {
  239. expressionStatementNode1 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  240. expressionStatementNode2 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  241. ifStatementBlockStatementNode = Nodes.getBlockStatementNode([
  242. expressionStatementNode1,
  243. expressionStatementNode2
  244. ]);
  245. ifStatementNode = Nodes.getIfStatementNode(
  246. Nodes.getLiteralNode(true),
  247. ifStatementBlockStatementNode
  248. );
  249. });
  250. it('should parentize given AST-tree with `ProgramNode` as root node', () => {
  251. programNode = Nodes.getProgramNode([
  252. ifStatementNode
  253. ]);
  254. NodeUtils.parentize(programNode);
  255. assert.deepEqual(programNode['parentNode'], programNode);
  256. assert.deepEqual(ifStatementNode['parentNode'], programNode);
  257. assert.deepEqual(ifStatementBlockStatementNode['parentNode'], ifStatementNode);
  258. assert.deepEqual(expressionStatementNode1['parentNode'], ifStatementBlockStatementNode);
  259. assert.deepEqual(expressionStatementNode2['parentNode'], ifStatementBlockStatementNode);
  260. });
  261. it('should parentize given AST-tree', () => {
  262. programNode = Nodes.getProgramNode([
  263. ifStatementNode
  264. ]);
  265. programNode['parentNode'] = programNode;
  266. NodeUtils.parentize(ifStatementNode);
  267. assert.deepEqual(ifStatementNode['parentNode'], programNode);
  268. assert.deepEqual(ifStatementBlockStatementNode['parentNode'], ifStatementNode);
  269. assert.deepEqual(expressionStatementNode1['parentNode'], ifStatementBlockStatementNode);
  270. assert.deepEqual(expressionStatementNode2['parentNode'], ifStatementBlockStatementNode);
  271. });
  272. });
  273. });