JavaScriptObfuscator.ts 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274
  1. import { inject, injectable, } from 'inversify';
  2. import { ServiceIdentifiers } from './container/ServiceIdentifiers';
  3. import * as esprima from 'esprima';
  4. import * as escodegen from 'escodegen-wallaby';
  5. import * as ESTree from 'estree';
  6. import * as packageJson from 'pjson';
  7. import { ICustomNodeGroup } from './interfaces/custom-nodes/ICustomNodeGroup';
  8. import { IGeneratorOutput } from './interfaces/IGeneratorOutput';
  9. import { IJavaScriptObfuscator } from './interfaces/IJavaScriptObfsucator';
  10. import { ILogger } from './interfaces/logger/ILogger';
  11. import { IObfuscationEventEmitter } from './interfaces/event-emitters/IObfuscationEventEmitter';
  12. import { IObfuscationResult } from './interfaces/IObfuscationResult';
  13. import { IOptions } from './interfaces/options/IOptions';
  14. import { IRandomGenerator } from './interfaces/utils/IRandomGenerator';
  15. import { ISourceMapCorrector } from './interfaces/source-map/ISourceMapCorrector';
  16. import { IStackTraceAnalyzer } from './interfaces/analyzers/stack-trace-analyzer/IStackTraceAnalyzer';
  17. import { IStackTraceData } from './interfaces/analyzers/stack-trace-analyzer/IStackTraceData';
  18. import { IStorage } from './interfaces/storages/IStorage';
  19. import { ITransformersRunner } from './interfaces/node-transformers/ITransformersRunner';
  20. import { LoggingMessage } from './enums/logger/LoggingMessage';
  21. import { NodeTransformer } from './enums/node-transformers/NodeTransformer';
  22. import { ObfuscationEvent } from './enums/event-emitters/ObfuscationEvent';
  23. import { TransformationStage } from './enums/node-transformers/TransformationStage';
  24. import { NodeGuards } from './node/NodeGuards';
  25. @injectable()
  26. export class JavaScriptObfuscator implements IJavaScriptObfuscator {
  27. /**
  28. * @type {GenerateOptions}
  29. */
  30. private static readonly escodegenParams: escodegen.GenerateOptions = {
  31. comment: true,
  32. verbatim: 'x-verbatim-property',
  33. sourceMapWithCode: true
  34. };
  35. /**
  36. * @type {NodeTransformer[]}
  37. */
  38. private static readonly transformersList: NodeTransformer[] = [
  39. NodeTransformer.BlockStatementControlFlowTransformer,
  40. NodeTransformer.ClassDeclarationTransformer,
  41. NodeTransformer.CommentsTransformer,
  42. NodeTransformer.DeadCodeInjectionTransformer,
  43. NodeTransformer.EvalCallExpressionTransformer,
  44. NodeTransformer.FunctionControlFlowTransformer,
  45. NodeTransformer.CatchClauseTransformer,
  46. NodeTransformer.FunctionDeclarationTransformer,
  47. NodeTransformer.FunctionTransformer,
  48. NodeTransformer.LabeledStatementTransformer,
  49. NodeTransformer.LiteralTransformer,
  50. NodeTransformer.MemberExpressionTransformer,
  51. NodeTransformer.MethodDefinitionTransformer,
  52. NodeTransformer.ObfuscatingGuardsTransformer,
  53. NodeTransformer.ObjectExpressionKeysTransformer,
  54. NodeTransformer.ObjectExpressionTransformer,
  55. NodeTransformer.ParentificationTransformer,
  56. NodeTransformer.TemplateLiteralTransformer,
  57. NodeTransformer.VariableDeclarationTransformer
  58. ];
  59. /**
  60. * @type {IStorage<ICustomNodeGroup>}
  61. */
  62. private readonly customNodeGroupStorage: IStorage<ICustomNodeGroup>;
  63. /**
  64. * @type {ILogger}
  65. */
  66. private readonly logger: ILogger;
  67. /**
  68. * @type {IObfuscationEventEmitter}
  69. */
  70. private readonly obfuscationEventEmitter: IObfuscationEventEmitter;
  71. /**
  72. * @type {IOptions}
  73. */
  74. private readonly options: IOptions;
  75. /**
  76. * @type {IRandomGenerator}
  77. */
  78. private readonly randomGenerator: IRandomGenerator;
  79. /**
  80. * @type {ISourceMapCorrector}
  81. */
  82. private readonly sourceMapCorrector: ISourceMapCorrector;
  83. /**
  84. * @type {IStackTraceAnalyzer}
  85. */
  86. private readonly stackTraceAnalyzer: IStackTraceAnalyzer;
  87. /**
  88. * @type {ITransformersRunner}
  89. */
  90. private readonly transformersRunner: ITransformersRunner;
  91. /**
  92. * @param {IStackTraceAnalyzer} stackTraceAnalyzer
  93. * @param {IObfuscationEventEmitter} obfuscationEventEmitter
  94. * @param {IStorage<ICustomNodeGroup>} customNodeGroupStorage
  95. * @param {ITransformersRunner} transformersRunner
  96. * @param {ISourceMapCorrector} sourceMapCorrector
  97. * @param {IRandomGenerator} randomGenerator
  98. * @param {ILogger} logger
  99. * @param {IOptions} options
  100. */
  101. constructor (
  102. @inject(ServiceIdentifiers.IStackTraceAnalyzer) stackTraceAnalyzer: IStackTraceAnalyzer,
  103. @inject(ServiceIdentifiers.IObfuscationEventEmitter) obfuscationEventEmitter: IObfuscationEventEmitter,
  104. @inject(ServiceIdentifiers.TCustomNodeGroupStorage) customNodeGroupStorage: IStorage<ICustomNodeGroup>,
  105. @inject(ServiceIdentifiers.ITransformersRunner) transformersRunner: ITransformersRunner,
  106. @inject(ServiceIdentifiers.ISourceMapCorrector) sourceMapCorrector: ISourceMapCorrector,
  107. @inject(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator,
  108. @inject(ServiceIdentifiers.ILogger) logger: ILogger,
  109. @inject(ServiceIdentifiers.IOptions) options: IOptions
  110. ) {
  111. this.stackTraceAnalyzer = stackTraceAnalyzer;
  112. this.obfuscationEventEmitter = obfuscationEventEmitter;
  113. this.customNodeGroupStorage = customNodeGroupStorage;
  114. this.transformersRunner = transformersRunner;
  115. this.sourceMapCorrector = sourceMapCorrector;
  116. this.randomGenerator = randomGenerator;
  117. this.logger = logger;
  118. this.options = options;
  119. }
  120. /**
  121. * @param {string} sourceCode
  122. * @returns {IObfuscationResult}
  123. */
  124. public obfuscate (sourceCode: string): IObfuscationResult {
  125. const timeStart: number = Date.now();
  126. this.logger.info(LoggingMessage.Version, packageJson.version);
  127. this.logger.info(LoggingMessage.ObfuscationStarted);
  128. this.logger.info(LoggingMessage.RandomGeneratorSeed, this.randomGenerator.getSeed());
  129. // parse AST tree
  130. const astTree: ESTree.Program = this.parseCode(sourceCode);
  131. // obfuscate AST tree
  132. const obfuscatedAstTree: ESTree.Program = this.transformAstTree(astTree);
  133. // generate code
  134. const generatorOutput: IGeneratorOutput = this.generateCode(sourceCode, obfuscatedAstTree);
  135. const obfuscationTime: number = (Date.now() - timeStart) / 1000;
  136. this.logger.success(LoggingMessage.ObfuscationCompleted, obfuscationTime);
  137. return this.getObfuscationResult(generatorOutput);
  138. }
  139. /**
  140. * @param {string} sourceCode
  141. * @returns {Program}
  142. */
  143. private parseCode (sourceCode: string): ESTree.Program {
  144. return esprima.parseScript(sourceCode, {
  145. attachComment: true,
  146. loc: this.options.sourceMap
  147. });
  148. }
  149. /**
  150. * @param {Program} astTree
  151. * @returns {Program}
  152. */
  153. private transformAstTree (astTree: ESTree.Program): ESTree.Program {
  154. const isEmptyAstTree: boolean = NodeGuards.isProgramNode(astTree)
  155. && !astTree.body.length
  156. && !astTree.leadingComments;
  157. if (isEmptyAstTree) {
  158. this.logger.warn(LoggingMessage.EmptySourceCode);
  159. return astTree;
  160. }
  161. astTree = this.runTransformationStage(astTree, TransformationStage.Preparing);
  162. this.logger.info(LoggingMessage.AnalyzingASTTreeStage);
  163. const stackTraceData: IStackTraceData[] = this.stackTraceAnalyzer.analyze(astTree);
  164. this.customNodeGroupStorage
  165. .getStorage()
  166. .forEach((customNodeGroup: ICustomNodeGroup) => {
  167. customNodeGroup.initialize();
  168. this.obfuscationEventEmitter.once(
  169. customNodeGroup.getAppendEvent(),
  170. customNodeGroup.appendCustomNodes.bind(customNodeGroup)
  171. );
  172. });
  173. this.obfuscationEventEmitter.emit(ObfuscationEvent.BeforeObfuscation, astTree, stackTraceData);
  174. if (this.options.deadCodeInjection) {
  175. astTree = this.runTransformationStage(astTree, TransformationStage.DeadCodeInjection);
  176. }
  177. if (this.options.controlFlowFlattening) {
  178. astTree = this.runTransformationStage(astTree, TransformationStage.ControlFlowFlattening);
  179. }
  180. astTree = this.runTransformationStage(astTree, TransformationStage.Converting);
  181. astTree = this.runTransformationStage(astTree, TransformationStage.Obfuscating);
  182. astTree = this.runTransformationStage(astTree, TransformationStage.Finalizing);
  183. this.obfuscationEventEmitter.emit(ObfuscationEvent.AfterObfuscation, astTree, stackTraceData);
  184. return astTree;
  185. }
  186. /**
  187. * @param {string} sourceCode
  188. * @param {Program} astTree
  189. * @returns {IGeneratorOutput}
  190. */
  191. private generateCode (sourceCode: string, astTree: ESTree.Program): IGeneratorOutput {
  192. const escodegenParams: escodegen.GenerateOptions = {
  193. ...JavaScriptObfuscator.escodegenParams
  194. };
  195. if (this.options.sourceMap) {
  196. escodegenParams.sourceMap = 'sourceMap';
  197. escodegenParams.sourceContent = sourceCode;
  198. }
  199. const generatorOutput: IGeneratorOutput = escodegen.generate(astTree, {
  200. ...escodegenParams,
  201. format: {
  202. compact: this.options.compact
  203. }
  204. });
  205. generatorOutput.map = generatorOutput.map ? generatorOutput.map.toString() : '';
  206. return generatorOutput;
  207. }
  208. /**
  209. * @param {IGeneratorOutput} generatorOutput
  210. * @returns {IObfuscationResult}
  211. */
  212. private getObfuscationResult (generatorOutput: IGeneratorOutput): IObfuscationResult {
  213. return this.sourceMapCorrector.correct(
  214. generatorOutput.code,
  215. generatorOutput.map
  216. );
  217. }
  218. /**
  219. * @param {Program} astTree
  220. * @param {TransformationStage} transformationStage
  221. * @returns {Program}
  222. */
  223. private runTransformationStage (astTree: ESTree.Program, transformationStage: TransformationStage): ESTree.Program {
  224. this.logger.info(LoggingMessage.TransformationStage, transformationStage);
  225. return this.transformersRunner.transform(
  226. astTree,
  227. JavaScriptObfuscator.transformersList,
  228. transformationStage
  229. );
  230. }
  231. }