NodeUtils.spec.ts 12 KB

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