TransformersRunner.ts 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. import { inject, injectable } from 'inversify';
  2. import { ServiceIdentifiers } from '../container/ServiceIdentifiers';
  3. import * as estraverse from 'estraverse';
  4. import * as ESTree from 'estree';
  5. import { TNodeTransformerFactory } from '../types/container/node-transformers/TNodeTransformerFactory';
  6. import { TVisitorDirection } from '../types/node-transformers/TVisitorDirection';
  7. import { TVisitorFunction } from '../types/node-transformers/TVisitorFunction';
  8. import { TVisitorResult } from '../types/node-transformers/TVisitorResult';
  9. import { ITransformersRunner } from '../interfaces/node-transformers/ITransformersRunner';
  10. import { IVisitor } from '../interfaces/node-transformers/IVisitor';
  11. import { NodeTransformer } from '../enums/node-transformers/NodeTransformer';
  12. import { TransformationStage } from '../enums/node-transformers/TransformationStage';
  13. import { VisitorDirection } from '../enums/node-transformers/VisitorDirection';
  14. import { NodeGuards } from '../node/NodeGuards';
  15. @injectable()
  16. export class TransformersRunner implements ITransformersRunner {
  17. /**
  18. * @type {TNodeTransformerFactory}
  19. */
  20. private readonly nodeTransformerFactory: TNodeTransformerFactory;
  21. /**
  22. * @param {TNodeTransformerFactory} nodeTransformerFactory
  23. */
  24. constructor (
  25. @inject(ServiceIdentifiers.Factory__INodeTransformer) nodeTransformerFactory: TNodeTransformerFactory,
  26. ) {
  27. this.nodeTransformerFactory = nodeTransformerFactory;
  28. }
  29. /**
  30. * @param {T} astTree
  31. * @param {NodeTransformer[]} nodeTransformers
  32. * @param {TransformationStage} transformationStage
  33. * @returns {T}
  34. */
  35. public transform <T extends ESTree.Node = ESTree.Program> (
  36. astTree: T,
  37. nodeTransformers: NodeTransformer[],
  38. transformationStage: TransformationStage
  39. ): T {
  40. if (!nodeTransformers.length) {
  41. return astTree;
  42. }
  43. const enterVisitors: IVisitor[] = [];
  44. const leaveVisitors: IVisitor[] = [];
  45. const nodeTransformersLength: number = nodeTransformers.length;
  46. let visitor: IVisitor | null;
  47. for (let i: number = 0; i < nodeTransformersLength; i++) {
  48. visitor = this.nodeTransformerFactory(nodeTransformers[i]).getVisitor(transformationStage);
  49. if (!visitor) {
  50. continue;
  51. }
  52. if (visitor.enter) {
  53. enterVisitors.push({ enter: visitor.enter });
  54. }
  55. if (visitor.leave) {
  56. leaveVisitors.push({ leave: visitor.leave });
  57. }
  58. }
  59. if (!enterVisitors.length && !leaveVisitors.length) {
  60. return astTree;
  61. }
  62. estraverse.replace(astTree, {
  63. enter: this.mergeVisitorsForDirection(enterVisitors, VisitorDirection.Enter),
  64. leave: this.mergeVisitorsForDirection(leaveVisitors, VisitorDirection.Leave)
  65. });
  66. return astTree;
  67. }
  68. /**
  69. * @param {IVisitor[]} visitors
  70. * @param {TVisitorDirection} direction
  71. * @returns {TVisitorFunction}
  72. */
  73. private mergeVisitorsForDirection (visitors: IVisitor[], direction: TVisitorDirection): TVisitorFunction {
  74. const visitorsLength: number = visitors.length;
  75. if (!visitorsLength) {
  76. return (node: ESTree.Node, parentNode: ESTree.Node | null) => node;
  77. }
  78. return (node: ESTree.Node, parentNode: ESTree.Node | null) => {
  79. if (node.ignoredNode) {
  80. return estraverse.VisitorOption.Skip;
  81. }
  82. for (let i: number = 0; i < visitorsLength; i++) {
  83. const visitorFunction: TVisitorFunction | undefined = visitors[i][direction];
  84. if (!visitorFunction) {
  85. continue;
  86. }
  87. const visitorResult: TVisitorResult = visitorFunction(node, parentNode);
  88. if (!visitorResult || !NodeGuards.isNode(visitorResult)) {
  89. continue;
  90. }
  91. node = visitorResult;
  92. }
  93. return node;
  94. };
  95. }
  96. }