Obfuscator.ts 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  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 { TNodeTransformerFactory } from './types/container/TNodeTransformerFactory';
  6. import { TVisitorDirection } from './types/TVisitorDirection';
  7. import { TVisitorFunction } from './types/TVisitorFunction';
  8. import { ICustomNodeGroup } from './interfaces/custom-nodes/ICustomNodeGroup';
  9. import { IObfuscationEventEmitter } from './interfaces/event-emitters/IObfuscationEventEmitter';
  10. import { IObfuscator } from './interfaces/IObfuscator';
  11. import { IOptions } from './interfaces/options/IOptions';
  12. import { IStackTraceAnalyzer } from './interfaces/stack-trace-analyzer/IStackTraceAnalyzer';
  13. import { IStackTraceData } from './interfaces/stack-trace-analyzer/IStackTraceData';
  14. import { IStorage } from './interfaces/storages/IStorage';
  15. import { IVisitor } from './interfaces/IVisitor';
  16. import { NodeTransformers } from './enums/container/NodeTransformers';
  17. import { ObfuscationEvents } from './enums/ObfuscationEvents';
  18. import { VisitorDirection } from './enums/VisitorDirection';
  19. import { Node } from './node/Node';
  20. import { NodeUtils } from './node/NodeUtils';
  21. @injectable()
  22. export class Obfuscator implements IObfuscator {
  23. /**
  24. * @type {NodeTransformers[]}
  25. */
  26. private static readonly controlFlowTransformersList: NodeTransformers[] = [
  27. NodeTransformers.BlockStatementControlFlowTransformer,
  28. NodeTransformers.FunctionControlFlowTransformer
  29. ];
  30. /**
  31. * @type {NodeTransformers[]}
  32. */
  33. private static readonly convertingTransformersList: NodeTransformers[] = [
  34. NodeTransformers.MemberExpressionTransformer,
  35. NodeTransformers.MethodDefinitionTransformer,
  36. NodeTransformers.TemplateLiteralTransformer
  37. ];
  38. /**
  39. * @type {NodeTransformers[]}
  40. */
  41. private static readonly obfuscatingTransformersList: NodeTransformers[] = [
  42. NodeTransformers.CatchClauseTransformer,
  43. NodeTransformers.FunctionDeclarationTransformer,
  44. NodeTransformers.FunctionTransformer,
  45. NodeTransformers.LabeledStatementTransformer,
  46. NodeTransformers.LiteralTransformer,
  47. NodeTransformers.ObjectExpressionTransformer,
  48. NodeTransformers.VariableDeclarationTransformer
  49. ];
  50. /**
  51. * @type {IStorage<ICustomNodeGroup>}
  52. */
  53. private readonly customNodeGroupStorage: IStorage<ICustomNodeGroup>;
  54. /**
  55. * @type {TNodeTransformerFactory}
  56. */
  57. private readonly nodeTransformerFactory: TNodeTransformerFactory;
  58. /**
  59. * @type {IObfuscationEventEmitter}
  60. */
  61. private readonly obfuscationEventEmitter: IObfuscationEventEmitter;
  62. /**
  63. * @type {IOptions}
  64. */
  65. private readonly options: IOptions;
  66. /**
  67. * @type {IStackTraceAnalyzer}
  68. */
  69. private readonly stackTraceAnalyzer: IStackTraceAnalyzer;
  70. /**
  71. * @param stackTraceAnalyzer
  72. * @param obfuscationEventEmitter
  73. * @param customNodeGroupStorage
  74. * @param nodeTransformerFactory
  75. * @param options
  76. */
  77. constructor (
  78. @inject(ServiceIdentifiers.IStackTraceAnalyzer) stackTraceAnalyzer: IStackTraceAnalyzer,
  79. @inject(ServiceIdentifiers.IObfuscationEventEmitter) obfuscationEventEmitter: IObfuscationEventEmitter,
  80. @inject(ServiceIdentifiers.TCustomNodeGroupStorage) customNodeGroupStorage: IStorage<ICustomNodeGroup>,
  81. @inject(ServiceIdentifiers.Factory__INodeTransformer) nodeTransformerFactory: TNodeTransformerFactory,
  82. @inject(ServiceIdentifiers.IOptions) options: IOptions
  83. ) {
  84. this.stackTraceAnalyzer = stackTraceAnalyzer;
  85. this.obfuscationEventEmitter = obfuscationEventEmitter;
  86. this.customNodeGroupStorage = customNodeGroupStorage;
  87. this.nodeTransformerFactory = nodeTransformerFactory;
  88. this.options = options;
  89. }
  90. /**
  91. * @param astTree
  92. * @returns {ESTree.Program}
  93. */
  94. public obfuscateAstTree (astTree: ESTree.Program): ESTree.Program {
  95. if (Node.isProgramNode(astTree) && !astTree.body.length) {
  96. return astTree;
  97. }
  98. astTree = <ESTree.Program>NodeUtils.parentize(astTree);
  99. const stackTraceData: IStackTraceData[] = this.stackTraceAnalyzer.analyze(astTree.body);
  100. // initialize custom node groups and configure custom nodes
  101. this.customNodeGroupStorage
  102. .getStorage()
  103. .forEach((customNodeGroup: ICustomNodeGroup) => {
  104. customNodeGroup.initialize();
  105. this.obfuscationEventEmitter.once(
  106. customNodeGroup.getAppendEvent(),
  107. customNodeGroup.appendCustomNodes.bind(customNodeGroup)
  108. );
  109. });
  110. this.obfuscationEventEmitter.emit(ObfuscationEvents.BeforeObfuscation, astTree, stackTraceData);
  111. // first pass: control flow flattening
  112. if (this.options.controlFlowFlattening) {
  113. astTree = this.transformAstTree(astTree, Obfuscator.controlFlowTransformersList);
  114. }
  115. // second pass: nodes obfuscation
  116. astTree = this.transformAstTree(astTree, [
  117. ...Obfuscator.convertingTransformersList,
  118. ...Obfuscator.obfuscatingTransformersList
  119. ]);
  120. this.obfuscationEventEmitter.emit(ObfuscationEvents.AfterObfuscation, astTree, stackTraceData);
  121. return astTree;
  122. }
  123. /**
  124. * @param astTree
  125. * @param nodeTransformers
  126. */
  127. private transformAstTree (
  128. astTree: ESTree.Program,
  129. nodeTransformers: NodeTransformers[]
  130. ): ESTree.Program {
  131. const enterVisitors: IVisitor[] = [];
  132. const leaveVisitors: IVisitor[] = [];
  133. const nodeTransformersLength: number = nodeTransformers.length;
  134. let visitor: IVisitor;
  135. for (let i: number = 0; i < nodeTransformersLength; i++) {
  136. visitor = this.nodeTransformerFactory(nodeTransformers[i]).getVisitor();
  137. if (visitor.enter) {
  138. enterVisitors.push(visitor);
  139. }
  140. if (visitor.leave) {
  141. leaveVisitors.push(visitor);
  142. }
  143. }
  144. estraverse.replace(astTree, {
  145. enter: this.mergeVisitorsForDirection(enterVisitors, VisitorDirection.enter),
  146. leave: this.mergeVisitorsForDirection(leaveVisitors, VisitorDirection.leave)
  147. });
  148. return astTree;
  149. }
  150. /**
  151. * @param visitors
  152. * @param direction
  153. * @return {TVisitorDirection}
  154. */
  155. private mergeVisitorsForDirection (visitors: IVisitor[], direction: TVisitorDirection): TVisitorFunction {
  156. if (!visitors.length) {
  157. return (node: ESTree.Node, parentNode: ESTree.Node) => node;
  158. }
  159. const visitorsLength: number = visitors.length;
  160. let visitor: IVisitor;
  161. return (node: ESTree.Node, parentNode: ESTree.Node) => {
  162. for (let i: number = 0; i < visitorsLength; i++) {
  163. visitor = visitors[i];
  164. const visitorFunction: TVisitorFunction | undefined = visitor[direction];
  165. if (!visitorFunction) {
  166. continue;
  167. }
  168. const visitorResult: ESTree.Node | void = visitorFunction(node, parentNode);
  169. if (!visitorResult) {
  170. continue;
  171. }
  172. node = visitorResult;
  173. }
  174. return node;
  175. };
  176. }
  177. }