CallExpressionControlFlowReplacer.ts 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. import { inject, injectable, } from 'inversify';
  2. import { ServiceIdentifiers } from '../../../container/ServiceIdentifiers';
  3. import * as ESTree from 'estree';
  4. import { TControlFlowCustomNodeFactory } from '../../../types/container/custom-nodes/TControlFlowCustomNodeFactory';
  5. import { TIdentifierNamesGeneratorFactory } from '../../../types/container/generators/TIdentifierNamesGeneratorFactory';
  6. import { TInitialData } from '../../../types/TInitialData';
  7. import { TStatement } from '../../../types/node/TStatement';
  8. import { IControlFlowStorage } from '../../../interfaces/storages/control-flow-transformers/IControlFlowStorage';
  9. import { ICustomNode } from '../../../interfaces/custom-nodes/ICustomNode';
  10. import { IOptions } from '../../../interfaces/options/IOptions';
  11. import { IRandomGenerator } from '../../../interfaces/utils/IRandomGenerator';
  12. import { ControlFlowCustomNode } from '../../../enums/custom-nodes/ControlFlowCustomNode';
  13. import { AbstractControlFlowReplacer } from './AbstractControlFlowReplacer';
  14. import { CallExpressionFunctionNode } from '../../../custom-nodes/control-flow-flattening-nodes/CallExpressionFunctionNode';
  15. import { CallExpressionControlFlowStorageCallNode } from '../../../custom-nodes/control-flow-flattening-nodes/control-flow-storage-nodes/CallExpressionControlFlowStorageCallNode';
  16. import { NodeGuards } from '../../../node/NodeGuards';
  17. @injectable()
  18. export class CallExpressionControlFlowReplacer extends AbstractControlFlowReplacer {
  19. /**
  20. * @type {number}
  21. */
  22. private static readonly usingExistingIdentifierChance: number = 0.5;
  23. /**
  24. * @param {TControlFlowCustomNodeFactory} controlFlowCustomNodeFactory
  25. * @param {TIdentifierNamesGeneratorFactory} identifierNamesGeneratorFactory
  26. * @param {IRandomGenerator} randomGenerator
  27. * @param {IOptions} options
  28. */
  29. public constructor (
  30. @inject(ServiceIdentifiers.Factory__IControlFlowCustomNode)
  31. controlFlowCustomNodeFactory: TControlFlowCustomNodeFactory,
  32. @inject(ServiceIdentifiers.Factory__IIdentifierNamesGenerator)
  33. identifierNamesGeneratorFactory: TIdentifierNamesGeneratorFactory,
  34. @inject(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator,
  35. @inject(ServiceIdentifiers.IOptions) options: IOptions
  36. ) {
  37. super(
  38. controlFlowCustomNodeFactory,
  39. identifierNamesGeneratorFactory,
  40. randomGenerator,
  41. options
  42. );
  43. }
  44. /**
  45. * @param {CallExpression} callExpressionNode
  46. * @param {Node} parentNode
  47. * @param {IControlFlowStorage} controlFlowStorage
  48. * @returns {Node}
  49. */
  50. public replace (
  51. callExpressionNode: ESTree.CallExpression,
  52. parentNode: ESTree.Node,
  53. controlFlowStorage: IControlFlowStorage
  54. ): ESTree.Node {
  55. const callee: ESTree.Expression = <ESTree.Expression>callExpressionNode.callee;
  56. if (!NodeGuards.isIdentifierNode(callee)) {
  57. return callExpressionNode;
  58. }
  59. const replacerId: number = callExpressionNode.arguments.length;
  60. const callExpressionFunctionCustomNode: ICustomNode<TInitialData<CallExpressionFunctionNode>> =
  61. this.controlFlowCustomNodeFactory(ControlFlowCustomNode.CallExpressionFunctionNode);
  62. const expressionArguments: (ESTree.Expression | ESTree.SpreadElement)[] = callExpressionNode.arguments;
  63. callExpressionFunctionCustomNode.initialize(expressionArguments);
  64. const storageKey: string = this.insertCustomNodeToControlFlowStorage(
  65. callExpressionFunctionCustomNode,
  66. controlFlowStorage,
  67. replacerId,
  68. CallExpressionControlFlowReplacer.usingExistingIdentifierChance
  69. );
  70. return this.getControlFlowStorageCallNode(
  71. controlFlowStorage.getStorageId(),
  72. storageKey,
  73. callee,
  74. expressionArguments
  75. );
  76. }
  77. /**
  78. * @param {string} controlFlowStorageId
  79. * @param {string} storageKey
  80. * @param {Expression} callee
  81. * @param {(Expression | SpreadElement)[]} expressionArguments
  82. * @returns {NodeGuards}
  83. */
  84. protected getControlFlowStorageCallNode (
  85. controlFlowStorageId: string,
  86. storageKey: string,
  87. callee: ESTree.Expression,
  88. expressionArguments: (ESTree.Expression | ESTree.SpreadElement)[]
  89. ): ESTree.Node {
  90. const controlFlowStorageCallCustomNode: ICustomNode<TInitialData<CallExpressionControlFlowStorageCallNode>> =
  91. this.controlFlowCustomNodeFactory(ControlFlowCustomNode.CallExpressionControlFlowStorageCallNode);
  92. controlFlowStorageCallCustomNode.initialize(controlFlowStorageId, storageKey, callee, expressionArguments);
  93. const statementNode: TStatement = controlFlowStorageCallCustomNode.getNode()[0];
  94. if (!statementNode || !NodeGuards.isExpressionStatementNode(statementNode)) {
  95. throw new Error('`controlFlowStorageCallCustomNode.getNode()[0]` should returns array with `ExpressionStatement` node');
  96. }
  97. return statementNode.expression;
  98. }
  99. }