NodeUtils.spec.ts 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419
  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('clone <T extends ESTree.Node> (astTree: T): T', () => {
  20. let ifStatementNode1: ESTree.IfStatement,
  21. ifStatementNode2: ESTree.IfStatement,
  22. ifStatementBlockStatementNode1: ESTree.BlockStatement,
  23. ifStatementBlockStatementNode2: ESTree.BlockStatement,
  24. expressionStatementNode1: ESTree.ExpressionStatement,
  25. expressionStatementNode2: ESTree.ExpressionStatement,
  26. expressionStatementNode3: ESTree.ExpressionStatement,
  27. expressionStatementNode4: ESTree.ExpressionStatement,
  28. programNode1: ESTree.Program,
  29. programNode2: ESTree.Program;
  30. beforeEach(() => {
  31. // actual AST tree
  32. expressionStatementNode1 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  33. expressionStatementNode2 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  34. ifStatementBlockStatementNode1 = Nodes.getBlockStatementNode([
  35. expressionStatementNode1,
  36. expressionStatementNode2
  37. ]);
  38. ifStatementNode1 = Nodes.getIfStatementNode(
  39. Nodes.getLiteralNode(true),
  40. ifStatementBlockStatementNode1
  41. );
  42. programNode1 = Nodes.getProgramNode([
  43. ifStatementNode1
  44. ]);
  45. // expected AST tree
  46. expressionStatementNode3 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  47. expressionStatementNode4 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  48. ifStatementBlockStatementNode2 = Nodes.getBlockStatementNode([
  49. expressionStatementNode3,
  50. expressionStatementNode4
  51. ]);
  52. ifStatementNode2 = Nodes.getIfStatementNode(
  53. Nodes.getLiteralNode(true),
  54. ifStatementBlockStatementNode2
  55. );
  56. programNode2 = Nodes.getProgramNode([
  57. ifStatementNode2
  58. ]);
  59. programNode2 = NodeUtils.parentize(programNode2);
  60. });
  61. it('should clone given AST-tree', () => {
  62. assert.deepEqual(NodeUtils.clone(programNode1), programNode2);
  63. });
  64. });
  65. describe('convertCodeToStructure (code: string): ESTree.Node[]', () => {
  66. let code: string,
  67. identifierNode: ESTree.Identifier,
  68. literalNode: ESTree.Literal,
  69. programNode: ESTree.Program,
  70. variableDeclarationNode: ESTree.VariableDeclaration,
  71. variableDeclaratorNode: ESTree.VariableDeclarator;
  72. beforeEach(() => {
  73. code = `
  74. var abc = 'cde';
  75. `;
  76. identifierNode = Nodes.getIdentifierNode('abc');
  77. literalNode = Nodes.getLiteralNode('cde');
  78. variableDeclaratorNode = Nodes.getVariableDeclaratorNode(identifierNode, literalNode);
  79. variableDeclarationNode = Nodes.getVariableDeclarationNode([
  80. variableDeclaratorNode
  81. ]);
  82. programNode = Nodes.getProgramNode([
  83. variableDeclarationNode
  84. ]);
  85. programNode.parentNode = programNode;
  86. variableDeclarationNode.parentNode = programNode;
  87. variableDeclaratorNode.parentNode = variableDeclarationNode;
  88. identifierNode.parentNode = variableDeclaratorNode;
  89. literalNode.parentNode = variableDeclaratorNode;
  90. });
  91. it('should convert code to `ESTree.Node[]` structure array', () => {
  92. assert.deepEqual(NodeUtils.convertCodeToStructure(code), [variableDeclarationNode]);
  93. });
  94. });
  95. describe('convertStructureToCode (structure: ESTree.Node[]): string', () => {
  96. let structure: ESTree.Node[],
  97. expectedCode: string;
  98. beforeEach(() => {
  99. structure = [
  100. Nodes.getProgramNode([
  101. Nodes.getVariableDeclarationNode([
  102. Nodes.getVariableDeclaratorNode(
  103. Nodes.getIdentifierNode('abc'),
  104. Nodes.getLiteralNode('cde')
  105. )
  106. ])
  107. ])
  108. ];
  109. expectedCode = 'var abc = \'cde\';';
  110. });
  111. it('should convert `ESTree.Node[]` structure to source code', () => {
  112. assert.deepEqual(NodeUtils.convertStructureToCode(structure), expectedCode);
  113. });
  114. });
  115. describe('getBlockStatementNodeByIndex (node: ESTree.Node, index: number = 0): ESTree.Node', () => {
  116. let blockStatementNode: ESTree.BlockStatement,
  117. expressionStatementNode1: ESTree.ExpressionStatement,
  118. expressionStatementNode2: ESTree.ExpressionStatement;
  119. beforeEach(() => {
  120. expressionStatementNode1 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  121. expressionStatementNode2 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  122. blockStatementNode = Nodes.getBlockStatementNode([
  123. expressionStatementNode1,
  124. expressionStatementNode2
  125. ]);
  126. });
  127. it('should return block-statement child node of given node if that node has block-statement', () => {
  128. assert.deepEqual(NodeUtils.getBlockStatementNodeByIndex(blockStatementNode), expressionStatementNode1);
  129. assert.deepEqual(NodeUtils.getBlockStatementNodeByIndex(blockStatementNode, 1), expressionStatementNode2);
  130. });
  131. it('should throw a `ReferenceError` if index is out of boundaries', () => {
  132. assert.throws(() => NodeUtils.getBlockStatementNodeByIndex(blockStatementNode, 2), ReferenceError);
  133. });
  134. it('should throw a `TypeError` if node have no a block-statement', () => {
  135. assert.throws(() => NodeUtils.getBlockStatementNodeByIndex(expressionStatementNode1, 1), TypeError);
  136. });
  137. });
  138. describe('getBlockScopesOfNode (node: ESTree.Node, blockScopes: TNodeWithBlockStatement[] = []): TNodeWithBlockStatement[]', () => {
  139. let functionDeclarationBlockStatementNode: ESTree.BlockStatement,
  140. ifStatementBlockStatementNode1: ESTree.BlockStatement,
  141. ifStatementBlockStatementNode2: ESTree.BlockStatement,
  142. ifStatementNode1: ESTree.IfStatement,
  143. ifStatementNode2: ESTree.IfStatement,
  144. expressionStatementNode3: ESTree.ExpressionStatement,
  145. expressionStatementNode2: ESTree.ExpressionStatement,
  146. expressionStatementNode1: ESTree.ExpressionStatement,
  147. functionDeclarationNode: ESTree.FunctionDeclaration,
  148. programNode: ESTree.Program;
  149. beforeEach(() => {
  150. expressionStatementNode1 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  151. expressionStatementNode2 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  152. expressionStatementNode3 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  153. ifStatementBlockStatementNode2 = Nodes.getBlockStatementNode([
  154. expressionStatementNode2,
  155. expressionStatementNode3
  156. ]);
  157. ifStatementNode2 = Nodes.getIfStatementNode(
  158. Nodes.getLiteralNode(true),
  159. ifStatementBlockStatementNode2
  160. );
  161. ifStatementBlockStatementNode1 = Nodes.getBlockStatementNode([
  162. ifStatementNode2
  163. ]);
  164. ifStatementNode1 = Nodes.getIfStatementNode(
  165. Nodes.getLiteralNode(true),
  166. ifStatementBlockStatementNode1
  167. );
  168. functionDeclarationBlockStatementNode = Nodes.getBlockStatementNode([
  169. expressionStatementNode1,
  170. ifStatementNode1
  171. ]);
  172. functionDeclarationNode = Nodes.getFunctionDeclarationNode('test', [], functionDeclarationBlockStatementNode);
  173. programNode = Nodes.getProgramNode([
  174. functionDeclarationNode
  175. ]);
  176. programNode.parentNode = programNode;
  177. functionDeclarationNode.parentNode = programNode;
  178. functionDeclarationBlockStatementNode.parentNode = functionDeclarationNode;
  179. expressionStatementNode1.parentNode = functionDeclarationBlockStatementNode;
  180. ifStatementNode1.parentNode = functionDeclarationBlockStatementNode;
  181. ifStatementBlockStatementNode1.parentNode = ifStatementNode1;
  182. ifStatementNode2.parentNode = ifStatementBlockStatementNode1;
  183. ifStatementBlockStatementNode2.parentNode = ifStatementNode2;
  184. expressionStatementNode3.parentNode = ifStatementBlockStatementNode2;
  185. });
  186. it('should return block-scope node for given node', () => {
  187. assert.deepEqual(NodeUtils.getBlockScopesOfNode(programNode)[0], programNode);
  188. assert.deepEqual(NodeUtils.getBlockScopesOfNode(functionDeclarationNode)[0], programNode);
  189. assert.deepEqual(NodeUtils.getBlockScopesOfNode(functionDeclarationBlockStatementNode)[0], programNode);
  190. assert.deepEqual(NodeUtils.getBlockScopesOfNode(expressionStatementNode1)[0], functionDeclarationBlockStatementNode);
  191. assert.deepEqual(NodeUtils.getBlockScopesOfNode(expressionStatementNode1)[1], programNode);
  192. assert.deepEqual(NodeUtils.getBlockScopesOfNode(ifStatementNode1)[0], functionDeclarationBlockStatementNode);
  193. assert.deepEqual(NodeUtils.getBlockScopesOfNode(ifStatementNode1)[1], programNode);
  194. assert.deepEqual(NodeUtils.getBlockScopesOfNode(ifStatementBlockStatementNode1)[0], functionDeclarationBlockStatementNode);
  195. assert.deepEqual(NodeUtils.getBlockScopesOfNode(ifStatementBlockStatementNode1)[1], programNode);
  196. assert.deepEqual(NodeUtils.getBlockScopesOfNode(ifStatementBlockStatementNode2)[0], functionDeclarationBlockStatementNode);
  197. assert.deepEqual(NodeUtils.getBlockScopesOfNode(ifStatementBlockStatementNode2)[1], programNode);
  198. assert.deepEqual(NodeUtils.getBlockScopesOfNode(expressionStatementNode3)[0], functionDeclarationBlockStatementNode);
  199. assert.deepEqual(NodeUtils.getBlockScopesOfNode(expressionStatementNode3)[1], programNode);
  200. });
  201. it('should throw a `ReferenceError` if node has no `parentNode` property', () => {
  202. assert.throws(() => NodeUtils.getBlockScopesOfNode(expressionStatementNode2)[0], ReferenceError);
  203. });
  204. });
  205. describe('getNodeBlockScopeDepth (node: ESTree.Node, depth: number = 0): number', () => {
  206. let functionDeclarationBlockStatementNode1: ESTree.BlockStatement,
  207. functionDeclarationBlockStatementNode2: ESTree.BlockStatement,
  208. ifStatementBlockStatementNode1: ESTree.BlockStatement,
  209. ifStatementBlockStatementNode2: ESTree.BlockStatement,
  210. ifStatementNode1: ESTree.IfStatement,
  211. ifStatementNode2: ESTree.IfStatement,
  212. expressionStatementNode1: ESTree.ExpressionStatement,
  213. expressionStatementNode2: ESTree.ExpressionStatement,
  214. expressionStatementNode3: ESTree.ExpressionStatement,
  215. functionDeclarationNode1: ESTree.FunctionDeclaration,
  216. functionDeclarationNode2: ESTree.FunctionDeclaration,
  217. programNode: ESTree.Program;
  218. beforeEach(() => {
  219. expressionStatementNode1 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  220. expressionStatementNode2 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  221. expressionStatementNode3 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  222. ifStatementBlockStatementNode2 = Nodes.getBlockStatementNode([
  223. expressionStatementNode3
  224. ]);
  225. ifStatementNode2 = Nodes.getIfStatementNode(
  226. Nodes.getLiteralNode(true),
  227. ifStatementBlockStatementNode2
  228. );
  229. functionDeclarationBlockStatementNode2 = Nodes.getBlockStatementNode([
  230. ifStatementNode2,
  231. expressionStatementNode2
  232. ]);
  233. functionDeclarationNode2 = Nodes.getFunctionDeclarationNode('test', [], functionDeclarationBlockStatementNode2);
  234. ifStatementBlockStatementNode1 = Nodes.getBlockStatementNode([
  235. functionDeclarationNode2
  236. ]);
  237. ifStatementNode1 = Nodes.getIfStatementNode(
  238. Nodes.getLiteralNode(true),
  239. ifStatementBlockStatementNode1
  240. );
  241. functionDeclarationBlockStatementNode1 = Nodes.getBlockStatementNode([
  242. expressionStatementNode1,
  243. ifStatementNode1
  244. ]);
  245. functionDeclarationNode1 = Nodes.getFunctionDeclarationNode('test', [], functionDeclarationBlockStatementNode1);
  246. programNode = Nodes.getProgramNode([
  247. functionDeclarationNode1
  248. ]);
  249. programNode.parentNode = programNode;
  250. functionDeclarationNode1.parentNode = programNode;
  251. functionDeclarationBlockStatementNode1.parentNode = functionDeclarationNode1;
  252. expressionStatementNode1.parentNode = functionDeclarationBlockStatementNode1;
  253. ifStatementNode1.parentNode = functionDeclarationBlockStatementNode1;
  254. ifStatementBlockStatementNode1.parentNode = ifStatementNode1;
  255. functionDeclarationNode2.parentNode = ifStatementBlockStatementNode1;
  256. functionDeclarationBlockStatementNode2.parentNode = functionDeclarationNode2;
  257. expressionStatementNode2.parentNode = functionDeclarationBlockStatementNode2;
  258. ifStatementNode2.parentNode = functionDeclarationBlockStatementNode2;
  259. ifStatementBlockStatementNode2.parentNode = ifStatementNode2;
  260. });
  261. it('should return block-scope depth for given node', () => {
  262. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(programNode), 0);
  263. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(functionDeclarationNode1), 0);
  264. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(functionDeclarationBlockStatementNode1), 1);
  265. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(expressionStatementNode1), 1);
  266. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(ifStatementNode1), 1);
  267. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(ifStatementBlockStatementNode1), 1);
  268. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(functionDeclarationNode2), 1);
  269. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(functionDeclarationBlockStatementNode2), 2);
  270. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(expressionStatementNode2), 2);
  271. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(ifStatementNode2), 2);
  272. assert.deepEqual(NodeUtils.getNodeBlockScopeDepth(ifStatementBlockStatementNode2), 2);
  273. });
  274. it('should throw a `ReferenceError` if node has no `parentNode` property', () => {
  275. assert.throws(() => NodeUtils.getNodeBlockScopeDepth(expressionStatementNode3), ReferenceError);
  276. });
  277. });
  278. describe('getUnaryExpressionArgumentNode (unaryExpressionNode: ESTree.UnaryExpression): ESTree.Node', () => {
  279. let expressionStatementNode: ESTree.ExpressionStatement,
  280. literalNode: ESTree.Literal,
  281. unaryExpressionNode1: ESTree.UnaryExpression,
  282. unaryExpressionNode2: ESTree.UnaryExpression,
  283. programNode: ESTree.Program;
  284. beforeEach(() => {
  285. literalNode = Nodes.getLiteralNode('test');
  286. unaryExpressionNode2 = Nodes.getUnaryExpressionNode('!', literalNode)
  287. unaryExpressionNode1 = Nodes.getUnaryExpressionNode('!', unaryExpressionNode2)
  288. expressionStatementNode = Nodes.getExpressionStatementNode(unaryExpressionNode1);
  289. programNode = Nodes.getProgramNode([
  290. expressionStatementNode
  291. ]);
  292. programNode.parentNode = programNode;
  293. expressionStatementNode.parentNode = programNode;
  294. unaryExpressionNode1.parentNode = expressionStatementNode;
  295. unaryExpressionNode2.parentNode = unaryExpressionNode1;
  296. literalNode.parentNode = unaryExpressionNode2;
  297. });
  298. it('should return unary expression argument node', () => {
  299. assert.deepEqual(NodeUtils.getUnaryExpressionArgumentNode(unaryExpressionNode1), literalNode);
  300. });
  301. });
  302. describe('parentize <T extends ESTree.Node> (astTree: T): T', () => {
  303. let ifStatementNode: ESTree.IfStatement,
  304. ifStatementBlockStatementNode: ESTree.BlockStatement,
  305. expressionStatementNode1: ESTree.ExpressionStatement,
  306. expressionStatementNode2: ESTree.ExpressionStatement,
  307. programNode: ESTree.Program;
  308. beforeEach(() => {
  309. expressionStatementNode1 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  310. expressionStatementNode2 = Nodes.getExpressionStatementNode(Nodes.getIdentifierNode('identifier'));
  311. ifStatementBlockStatementNode = Nodes.getBlockStatementNode([
  312. expressionStatementNode1,
  313. expressionStatementNode2
  314. ]);
  315. ifStatementNode = Nodes.getIfStatementNode(
  316. Nodes.getLiteralNode(true),
  317. ifStatementBlockStatementNode
  318. );
  319. });
  320. it('should parentize given AST-tree with `ProgramNode` as root node', () => {
  321. programNode = Nodes.getProgramNode([
  322. ifStatementNode
  323. ]);
  324. programNode = NodeUtils.parentize(programNode);
  325. assert.deepEqual(programNode.parentNode, programNode);
  326. assert.deepEqual(ifStatementNode.parentNode, programNode);
  327. assert.deepEqual(ifStatementBlockStatementNode.parentNode, ifStatementNode);
  328. assert.deepEqual(expressionStatementNode1.parentNode, ifStatementBlockStatementNode);
  329. assert.deepEqual(expressionStatementNode2.parentNode, ifStatementBlockStatementNode);
  330. });
  331. it('should parentize given AST-tree', () => {
  332. programNode = Nodes.getProgramNode([
  333. ifStatementNode
  334. ]);
  335. programNode.parentNode = programNode;
  336. ifStatementNode = NodeUtils.parentize(ifStatementNode);
  337. assert.deepEqual(ifStatementNode.parentNode, programNode);
  338. assert.deepEqual(ifStatementBlockStatementNode.parentNode, ifStatementNode);
  339. assert.deepEqual(expressionStatementNode1.parentNode, ifStatementBlockStatementNode);
  340. assert.deepEqual(expressionStatementNode2.parentNode, ifStatementBlockStatementNode);
  341. });
  342. });
  343. });