MethodDefinitionTransformer.ts 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. import { inject, injectable, } from 'inversify';
  2. import { ServiceIdentifiers } from '../../container/ServiceIdentifiers';
  3. import * as ESTree from 'estree';
  4. import { IOptions } from '../../interfaces/options/IOptions';
  5. import { IRandomGenerator } from '../../interfaces/utils/IRandomGenerator';
  6. import { IVisitor } from '../../interfaces/node-transformers/IVisitor';
  7. import { TransformationStage } from '../../enums/node-transformers/TransformationStage';
  8. import { AbstractNodeTransformer } from '../AbstractNodeTransformer';
  9. import { NodeFactory } from '../../node/NodeFactory';
  10. import { NodeGuards } from '../../node/NodeGuards';
  11. /**
  12. * replaces:
  13. * foo () { //... };
  14. *
  15. * or
  16. * 'foo' () { //... };
  17. *
  18. * on:
  19. * ['foo'] { //... };
  20. *
  21. * Literal node will be obfuscated by LiteralTransformer
  22. */
  23. @injectable()
  24. export class MethodDefinitionTransformer extends AbstractNodeTransformer {
  25. /**
  26. * @type {string[]}
  27. */
  28. private static readonly ignoredNames: string[] = ['constructor'];
  29. /**
  30. * @param {IRandomGenerator} randomGenerator
  31. * @param {IOptions} options
  32. */
  33. public constructor (
  34. @inject(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator,
  35. @inject(ServiceIdentifiers.IOptions) options: IOptions
  36. ) {
  37. super(randomGenerator, options);
  38. }
  39. /**
  40. * @param {TransformationStage} transformationStage
  41. * @returns {IVisitor | null}
  42. */
  43. public getVisitor (transformationStage: TransformationStage): IVisitor | null {
  44. switch (transformationStage) {
  45. case TransformationStage.Converting:
  46. return {
  47. enter: (node: ESTree.Node, parentNode: ESTree.Node | null) => {
  48. if (parentNode && NodeGuards.isMethodDefinitionNode(node)) {
  49. return this.transformNode(node, parentNode);
  50. }
  51. }
  52. };
  53. default:
  54. return null;
  55. }
  56. }
  57. /**
  58. * replaces:
  59. * object.identifier = 1;
  60. *
  61. * on:
  62. * object['identifier'] = 1;
  63. *
  64. * and skip:
  65. * object[identifier] = 1;
  66. * Literal node will be obfuscated by LiteralTransformer
  67. *
  68. * @param {MethodDefinition} methodDefinitionNode
  69. * @param {NodeGuards} parentNode
  70. * @returns {NodeGuards}
  71. */
  72. public transformNode (methodDefinitionNode: ESTree.MethodDefinition, parentNode: ESTree.Node): ESTree.Node {
  73. if (NodeGuards.isIdentifierNode(methodDefinitionNode.key)) {
  74. return this.replaceIdentifierKey(methodDefinitionNode, methodDefinitionNode.key);
  75. }
  76. if (NodeGuards.isLiteralNode(methodDefinitionNode.key)) {
  77. return this.replaceLiteralKey(methodDefinitionNode, methodDefinitionNode.key);
  78. }
  79. return methodDefinitionNode;
  80. }
  81. /**
  82. * @param {MethodDefinition} methodDefinitionNode
  83. * @param {Identifier} keyNode
  84. * @returns {MethodDefinition}
  85. */
  86. private replaceIdentifierKey (
  87. methodDefinitionNode: ESTree.MethodDefinition,
  88. keyNode: ESTree.Identifier
  89. ): ESTree.MethodDefinition {
  90. if (
  91. !MethodDefinitionTransformer.ignoredNames.includes(keyNode.name)
  92. && !methodDefinitionNode.computed
  93. ) {
  94. methodDefinitionNode.computed = true;
  95. methodDefinitionNode.key = NodeFactory.literalNode(keyNode.name);
  96. }
  97. return methodDefinitionNode;
  98. }
  99. /**
  100. * @param {MethodDefinition} methodDefinitionNode
  101. * @param {Literal} keyNode
  102. * @returns {MethodDefinition}
  103. */
  104. private replaceLiteralKey (
  105. methodDefinitionNode: ESTree.MethodDefinition,
  106. keyNode: ESTree.Literal
  107. ): ESTree.MethodDefinition {
  108. if (
  109. typeof keyNode.value === 'string'
  110. && !MethodDefinitionTransformer.ignoredNames.includes(keyNode.value)
  111. && !methodDefinitionNode.computed
  112. ) {
  113. methodDefinitionNode.computed = true;
  114. }
  115. return methodDefinitionNode;
  116. }
  117. }