JavaScriptObfuscator.ts 8.3 KB

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