FunctionDeclarationTransformer.ts 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. import { injectable, inject } from 'inversify';
  2. import { ServiceIdentifiers } from '../../container/ServiceIdentifiers';
  3. import * as estraverse from 'estraverse';
  4. import * as ESTree from 'estree';
  5. import { TIdentifierObfuscatingReplacerFactory } from "../../types/container/node-transformers/TIdentifierObfuscatingReplacerFactory";
  6. import { TNodeWithBlockStatement } from '../../types/node/TNodeWithBlockStatement';
  7. import { IIdentifierObfuscatingReplacer } from '../../interfaces/node-transformers/obfuscating-transformers/IIdentifierObfuscatingReplacer';
  8. import { IOptions } from '../../interfaces/options/IOptions';
  9. import { IVisitor } from '../../interfaces/IVisitor';
  10. import { IdentifierObfuscatingReplacers } from "../../enums/container/node-transformers/IdentifierObfuscatingReplacers";
  11. import { NodeType } from '../../enums/NodeType';
  12. import { AbstractNodeTransformer } from '../AbstractNodeTransformer';
  13. import { Node } from '../../node/Node';
  14. import { NodeUtils } from '../../node/NodeUtils';
  15. /**
  16. * replaces:
  17. * function foo () { //... };
  18. * foo();
  19. *
  20. * on:
  21. * function _0x12d45f () { //... };
  22. * _0x12d45f();
  23. */
  24. @injectable()
  25. export class FunctionDeclarationTransformer extends AbstractNodeTransformer {
  26. /**
  27. * @type {IIdentifierObfuscatingReplacer}
  28. */
  29. private readonly identifierObfuscatingReplacer: IIdentifierObfuscatingReplacer;
  30. /**
  31. * @type {Map<ESTree.Node, ESTree.Identifier[]>}
  32. */
  33. private readonly replaceableIdentifiers: Map <ESTree.Node, ESTree.Identifier[]> = new Map();
  34. /**
  35. * @param identifierObfuscatingReplacerFactory
  36. * @param options
  37. */
  38. constructor (
  39. @inject(ServiceIdentifiers.Factory__IIdentifierObfuscatingReplacer)
  40. identifierObfuscatingReplacerFactory: TIdentifierObfuscatingReplacerFactory,
  41. @inject(ServiceIdentifiers.IOptions) options: IOptions
  42. ) {
  43. super(options);
  44. this.identifierObfuscatingReplacer = identifierObfuscatingReplacerFactory(
  45. IdentifierObfuscatingReplacers.IdentifierObfuscatingReplacer
  46. );
  47. }
  48. /**
  49. * @return {IVisitor}
  50. */
  51. public getVisitor (): IVisitor {
  52. return {
  53. enter: (node: ESTree.Node, parentNode: ESTree.Node) => {
  54. if (Node.isFunctionDeclarationNode(node)) {
  55. return this.transformNode(node, parentNode);
  56. }
  57. }
  58. };
  59. }
  60. /**
  61. * @param functionDeclarationNode
  62. * @param parentNode
  63. * @returns {ESTree.Node}
  64. */
  65. public transformNode (functionDeclarationNode: ESTree.FunctionDeclaration, parentNode: ESTree.Node): ESTree.Node {
  66. const nodeIdentifier: number = this.nodeIdentifier++;
  67. const blockScopeOfFunctionDeclarationNode: TNodeWithBlockStatement = NodeUtils
  68. .getBlockScopesOfNode(functionDeclarationNode)[0];
  69. if (blockScopeOfFunctionDeclarationNode.type === NodeType.Program) {
  70. return functionDeclarationNode;
  71. }
  72. this.storeFunctionName(functionDeclarationNode, nodeIdentifier);
  73. // check for cached identifiers for current scope node. If exist - loop through them.
  74. if (this.replaceableIdentifiers.has(blockScopeOfFunctionDeclarationNode)) {
  75. this.replaceScopeCachedIdentifiers(blockScopeOfFunctionDeclarationNode, nodeIdentifier);
  76. } else {
  77. this.replaceScopeIdentifiers(blockScopeOfFunctionDeclarationNode, nodeIdentifier);
  78. }
  79. return functionDeclarationNode;
  80. }
  81. /**
  82. * @param functionDeclarationNode
  83. * @param nodeIdentifier
  84. */
  85. private storeFunctionName (functionDeclarationNode: ESTree.FunctionDeclaration, nodeIdentifier: number): void {
  86. this.identifierObfuscatingReplacer.storeNames(functionDeclarationNode.id.name, nodeIdentifier);
  87. }
  88. /**
  89. * @param scopeNode
  90. * @param nodeIdentifier
  91. */
  92. private replaceScopeCachedIdentifiers (scopeNode: ESTree.Node, nodeIdentifier: number): void {
  93. const cachedReplaceableIdentifiers: ESTree.Identifier[] = <ESTree.Identifier[]>this.replaceableIdentifiers.get(scopeNode);
  94. cachedReplaceableIdentifiers.forEach((replaceableIdentifier: ESTree.Identifier) => {
  95. const newReplaceableIdentifier: ESTree.Identifier = this.identifierObfuscatingReplacer.replace(replaceableIdentifier.name, nodeIdentifier);
  96. replaceableIdentifier.name = newReplaceableIdentifier.name;
  97. });
  98. }
  99. /**
  100. * @param scopeNode
  101. * @param nodeIdentifier
  102. */
  103. private replaceScopeIdentifiers (scopeNode: ESTree.Node, nodeIdentifier: number): void {
  104. const storedReplaceableIdentifiers: ESTree.Identifier[] = [];
  105. estraverse.replace(scopeNode, {
  106. enter: (node: ESTree.Node, parentNode: ESTree.Node): any => {
  107. if (Node.isReplaceableIdentifierNode(node, parentNode)) {
  108. const newIdentifier: ESTree.Identifier = this.identifierObfuscatingReplacer.replace(node.name, nodeIdentifier);
  109. const newIdentifierName: string = newIdentifier.name;
  110. if (node.name !== newIdentifierName) {
  111. node.name = newIdentifierName;
  112. } else {
  113. storedReplaceableIdentifiers.push(node);
  114. }
  115. }
  116. }
  117. });
  118. this.replaceableIdentifiers.set(scopeNode, storedReplaceableIdentifiers);
  119. }
  120. }