js-confuser 1.5.9 → 1.7.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (143) hide show
  1. package/.github/workflows/node.js.yml +2 -2
  2. package/CHANGELOG.md +55 -0
  3. package/README.md +346 -165
  4. package/dist/constants.js +6 -2
  5. package/dist/index.js +9 -21
  6. package/dist/obfuscator.js +19 -31
  7. package/dist/options.js +5 -5
  8. package/dist/order.js +1 -3
  9. package/dist/presets.js +6 -7
  10. package/dist/probability.js +2 -4
  11. package/dist/templates/bufferToString.js +13 -0
  12. package/dist/templates/crash.js +3 -3
  13. package/dist/templates/es5.js +18 -0
  14. package/dist/templates/functionLength.js +16 -0
  15. package/dist/transforms/calculator.js +77 -21
  16. package/dist/transforms/controlFlowFlattening/controlFlowFlattening.js +980 -367
  17. package/dist/transforms/controlFlowFlattening/expressionObfuscation.js +4 -1
  18. package/dist/transforms/controlFlowFlattening/switchCaseObfuscation.js +25 -26
  19. package/dist/transforms/deadCode.js +33 -25
  20. package/dist/transforms/dispatcher.js +8 -4
  21. package/dist/transforms/es5/antiDestructuring.js +2 -0
  22. package/dist/transforms/es5/es5.js +31 -34
  23. package/dist/transforms/extraction/duplicateLiteralsRemoval.js +92 -58
  24. package/dist/transforms/finalizer.js +82 -0
  25. package/dist/transforms/flatten.js +229 -148
  26. package/dist/transforms/identifier/globalAnalysis.js +88 -0
  27. package/dist/transforms/identifier/globalConcealing.js +10 -83
  28. package/dist/transforms/identifier/movedDeclarations.js +35 -88
  29. package/dist/transforms/identifier/renameVariables.js +124 -59
  30. package/dist/transforms/identifier/variableAnalysis.js +58 -62
  31. package/dist/transforms/lock/lock.js +0 -37
  32. package/dist/transforms/minify.js +60 -57
  33. package/dist/transforms/opaquePredicates.js +1 -1
  34. package/dist/transforms/preparation/preparation.js +2 -2
  35. package/dist/transforms/preparation.js +231 -0
  36. package/dist/transforms/renameLabels.js +1 -1
  37. package/dist/transforms/rgf.js +139 -247
  38. package/dist/transforms/stack.js +128 -26
  39. package/dist/transforms/string/encoding.js +150 -179
  40. package/dist/transforms/string/stringCompression.js +14 -15
  41. package/dist/transforms/string/stringConcealing.js +25 -8
  42. package/dist/transforms/string/stringEncoding.js +13 -24
  43. package/dist/transforms/transform.js +12 -19
  44. package/dist/traverse.js +24 -10
  45. package/dist/util/gen.js +17 -1
  46. package/dist/util/identifiers.js +37 -3
  47. package/dist/util/insert.js +35 -4
  48. package/dist/util/random.js +15 -0
  49. package/docs/ControlFlowFlattening.md +595 -0
  50. package/{Countermeasures.md → docs/Countermeasures.md} +1 -15
  51. package/{Integrity.md → docs/Integrity.md} +2 -2
  52. package/docs/RGF.md +419 -0
  53. package/package.json +5 -5
  54. package/src/constants.ts +3 -0
  55. package/src/index.ts +2 -2
  56. package/src/obfuscator.ts +19 -31
  57. package/src/options.ts +14 -103
  58. package/src/order.ts +1 -5
  59. package/src/presets.ts +6 -7
  60. package/src/probability.ts +2 -3
  61. package/src/templates/bufferToString.ts +68 -0
  62. package/src/templates/crash.ts +15 -19
  63. package/src/templates/es5.ts +131 -0
  64. package/src/templates/functionLength.ts +14 -0
  65. package/src/transforms/calculator.ts +122 -59
  66. package/src/transforms/controlFlowFlattening/controlFlowFlattening.ts +1583 -571
  67. package/src/transforms/controlFlowFlattening/expressionObfuscation.ts +4 -1
  68. package/src/transforms/deadCode.ts +383 -26
  69. package/src/transforms/dispatcher.ts +9 -4
  70. package/src/transforms/es5/antiDestructuring.ts +2 -0
  71. package/src/transforms/es5/es5.ts +32 -77
  72. package/src/transforms/extraction/duplicateLiteralsRemoval.ts +133 -129
  73. package/src/transforms/{hexadecimalNumbers.ts → finalizer.ts} +29 -13
  74. package/src/transforms/flatten.ts +357 -300
  75. package/src/transforms/identifier/globalAnalysis.ts +85 -0
  76. package/src/transforms/identifier/globalConcealing.ts +14 -103
  77. package/src/transforms/identifier/movedDeclarations.ts +49 -102
  78. package/src/transforms/identifier/renameVariables.ts +149 -78
  79. package/src/transforms/identifier/variableAnalysis.ts +66 -73
  80. package/src/transforms/lock/lock.ts +1 -42
  81. package/src/transforms/minify.ts +91 -75
  82. package/src/transforms/opaquePredicates.ts +2 -2
  83. package/src/transforms/preparation.ts +238 -0
  84. package/src/transforms/renameLabels.ts +2 -2
  85. package/src/transforms/rgf.ts +213 -405
  86. package/src/transforms/stack.ts +156 -36
  87. package/src/transforms/string/encoding.ts +115 -212
  88. package/src/transforms/string/stringCompression.ts +27 -18
  89. package/src/transforms/string/stringConcealing.ts +39 -9
  90. package/src/transforms/string/stringEncoding.ts +18 -18
  91. package/src/transforms/transform.ts +21 -23
  92. package/src/traverse.ts +23 -4
  93. package/src/types.ts +2 -1
  94. package/src/util/gen.ts +28 -3
  95. package/src/util/identifiers.ts +43 -2
  96. package/src/util/insert.ts +38 -3
  97. package/src/util/random.ts +13 -0
  98. package/test/code/Cash.test.ts +1 -1
  99. package/test/code/Dynamic.test.ts +12 -10
  100. package/test/code/ES6.src.js +146 -0
  101. package/test/code/ES6.test.ts +28 -2
  102. package/test/index.test.ts +2 -1
  103. package/test/probability.test.ts +44 -0
  104. package/test/templates/template.test.ts +1 -1
  105. package/test/transforms/antiTooling.test.ts +22 -0
  106. package/test/transforms/calculator.test.ts +40 -0
  107. package/test/transforms/controlFlowFlattening/controlFlowFlattening.test.ts +702 -160
  108. package/test/transforms/controlFlowFlattening/expressionObfuscation.test.ts +173 -0
  109. package/test/transforms/deadCode.test.ts +66 -15
  110. package/test/transforms/dispatcher.test.ts +20 -1
  111. package/test/transforms/es5/antiDestructuring.test.ts +16 -0
  112. package/test/transforms/flatten.test.ts +399 -86
  113. package/test/transforms/identifier/movedDeclarations.test.ts +63 -8
  114. package/test/transforms/identifier/renameVariables.test.ts +119 -0
  115. package/test/transforms/lock/antiDebug.test.ts +2 -2
  116. package/test/transforms/lock/lock.test.ts +1 -48
  117. package/test/transforms/minify.test.ts +104 -0
  118. package/test/transforms/preparation.test.ts +157 -0
  119. package/test/transforms/rgf.test.ts +261 -381
  120. package/test/transforms/stack.test.ts +143 -21
  121. package/test/transforms/string/stringCompression.test.ts +39 -0
  122. package/test/transforms/string/stringConcealing.test.ts +82 -0
  123. package/test/transforms/string/stringEncoding.test.ts +53 -2
  124. package/test/transforms/transform.test.ts +66 -0
  125. package/test/traverse.test.ts +139 -0
  126. package/test/util/identifiers.test.ts +113 -1
  127. package/test/util/insert.test.ts +57 -3
  128. package/src/transforms/controlFlowFlattening/choiceFlowObfuscation.ts +0 -87
  129. package/src/transforms/controlFlowFlattening/controlFlowObfuscation.ts +0 -203
  130. package/src/transforms/controlFlowFlattening/switchCaseObfuscation.ts +0 -130
  131. package/src/transforms/eval.ts +0 -89
  132. package/src/transforms/hideInitializingCode.ts +0 -432
  133. package/src/transforms/identifier/nameRecycling.ts +0 -280
  134. package/src/transforms/label.ts +0 -64
  135. package/src/transforms/preparation/nameConflicts.ts +0 -102
  136. package/src/transforms/preparation/preparation.ts +0 -176
  137. package/test/transforms/controlFlowFlattening/controlFlowObfuscation.test.ts +0 -101
  138. package/test/transforms/controlFlowFlattening/switchCaseObfuscation.test.ts +0 -120
  139. package/test/transforms/eval.test.ts +0 -131
  140. package/test/transforms/hideInitializingCode.test.ts +0 -336
  141. package/test/transforms/identifier/nameRecycling.test.ts +0 -205
  142. package/test/transforms/preparation/nameConflicts.test.ts +0 -52
  143. package/test/transforms/preparation/preparation.test.ts +0 -62
package/docs/RGF.md ADDED
@@ -0,0 +1,419 @@
1
+ ## `RGF`
2
+
3
+ RGF (Runtime-Generated-Functions) uses the [`new Function(code...)`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Function/Function) syntax to construct executable code from strings. (`true/false/0-1`)
4
+
5
+ - **This can break your code.**
6
+ - **Due to the security concerns of arbitrary code execution, you must enable this yourself.**
7
+ - The arbitrary code is also obfuscated.
8
+
9
+ Option name: `rgf`
10
+
11
+ Option values: `true/false/0-1`
12
+
13
+ Note: RGF will only apply to functions that do not rely on any outside-scoped variables. Enable `flatten` along with `rgf` to apply to these functions.
14
+
15
+ Note: Does not apply to arrow, async, or generator functions.
16
+
17
+ Use a number to control the percentage of functions changed.
18
+
19
+ ```js
20
+ // Input
21
+ function printToConsole(message){
22
+ console.log(message);
23
+ }
24
+
25
+ printToConsole("Hello World"); // "Hello World"
26
+
27
+ // Output
28
+ var Ricvq8s = [new Function('function HIGRHaD(ANVivo_){console[\'log\'](ANVivo_)}return HIGRHaD[\'apply\'](this,arguments)')];
29
+ function uhj6obs() {
30
+ return Ricvq8s[0]['apply'](this, arguments);
31
+ }
32
+ uhj6obs('Hello World'); // "Hello World"
33
+ ```
34
+
35
+ ## With `Flatten`
36
+
37
+ Enable `flatten` with `rgf` to apply to functions that rely on outside-scoped variables.
38
+
39
+ Flatten is able to isolate functions from their scope so then RGF can then apply on them.
40
+
41
+ ```js
42
+ {
43
+ target: "node",
44
+ rgf: true,
45
+ flatten: true
46
+ }
47
+ ```
48
+
49
+ ```js
50
+ // Input
51
+ var outsideVariable = 0;
52
+ function incrementOutsideVariable(){
53
+ outsideVariable++;
54
+ }
55
+
56
+ incrementOutsideVariable(); // outsideVariable = 1
57
+ incrementOutsideVariable(); // outsideVariable = 2
58
+ incrementOutsideVariable(); // outsideVariable = 3
59
+
60
+ console.log(outsideVariable); // 3
61
+
62
+ // Output
63
+ var J3NLZFR = [
64
+ new Function(
65
+ "function Q7Rh6l([],reFzsi){reFzsi['XaBIEIZ']++}return Q7Rh6l['apply'](this,arguments)"
66
+ ),
67
+ ];
68
+ function pCG9mH() {
69
+ return J3NLZFR[0]["apply"](this, arguments);
70
+ }
71
+ var outsideVariable = 0;
72
+ function incrementOutsideVariable(...muLxIC) {
73
+ var udg38ch = {
74
+ set ["XaBIEIZ"](H5p1op) {
75
+ outsideVariable = H5p1op;
76
+ },
77
+ get ["XaBIEIZ"]() {
78
+ return outsideVariable;
79
+ },
80
+ };
81
+ return pCG9mH(muLxIC, udg38ch);
82
+ }
83
+ !(incrementOutsideVariable(),
84
+ incrementOutsideVariable(),
85
+ incrementOutsideVariable(),
86
+ console["log"](outsideVariable)); // 3
87
+ ```
88
+
89
+ ## With `String Concealing`
90
+
91
+ Enable `stringConcealing` to encrypt the `new Function(code)` code string.
92
+
93
+ ```js
94
+ // Input
95
+ function add(x, y){
96
+ return x + y;
97
+ }
98
+
99
+ console.log(add(5, 10)); // 15
100
+
101
+ // Output
102
+ var MAKh7o = [],
103
+ BCG3CXC = 0,
104
+ W33d4e = (function () {
105
+ var To7ztdg = [
106
+ "n%v2do>o/Ro<B",
107
+ "ad/z~7_MP#]yCZ(ZlG2Hr@3B3UuHQbXj~7$GZ7w@h#g<J9G",
108
+ ']+=c$.zNPP+/G9BY1G$x3_M+]8EU"[XpkwUdV^QC!Pk.XbK',
109
+ "Hd8=l@<:%59/L]J",
110
+ "u)jdloP1|6k.*Z}iV5UHt#[z<IW31wwn{nnEkj5vyJ",
111
+ "@4P0goQYcRNlUD",
112
+ '"!>fj>~p!It3M^hX',
113
+ ".t)F;!&u)J?+gEFe4zAz;3A",
114
+ ":!><}7qC*2i.2xI",
115
+ "&2rd(iQYsRJ2ID",
116
+ 'H)?/f2lvs4KinCkR]Db.znuuY%"uk,Uorf`yo2:M97n',
117
+ 'KEFK73EeT2m/77Hg,i&Fu#qePMp{Lv(mUzf0v+Jj6U<xhQQn%XM,d^jN%yh*ZFAe>]w;Uvz`Y3G.Z*ClC:?<HwU6g&?E(u+NFEgbL^YI]GJ3nv$Y:"29ZXRipw(wD7=!xS!lPaRi{Z{^EqU2#@lZ3dJuX2@y4@AV{PWlCMszhDm#]eh5G)*[|ja2D:rZ%#L0BJ(u+N"i`Km]2MyTQUO[8Lx7<fY<L|%Te.{+;T@z/z!=86D8fO+*FYg%(:Ef5?L7Y{LntO|LE:?,V[GUY?sQWq>iPeI!Y{DJU3$@Qcf<Kz`m/#F(g!jmFTG&ywX,Zj0T1c>[vP0/oxJk>oCPL?G*aWmUk,(y0CG62t6[)gaz^I3]Z[$yVRra,S7!Hc!i6CAP4m&BRn_URI{c]GlN07361pmfja!i6CAPo{CN/US8U==[${lT,r|]7j}UD=U*7*K0T^4b$q"z^I3]Z[$yp!kbWi|?p>P:_Y2$E`~.8jcB9c!=!Yy#c%x@Moi5e39=TXi#c.~MLOD]Wf=[RvK0U6myWoa=zg>[;NmTd/<P1o:JzI{*bv.!T^a9_o_zU=W}[;FWGvCFph(UE=U*7*|N47Tm_o@|[;c,`pE$nwaxTOU?p>W}FjMR6t{]8d5)VKU,0{8!@+q[wh[i/2s)+{ZR50ixOg#4]I^[5ds!Wz|a?N~<[go@J`+9M6>x=Tgzzgh`(`+9p!kbWijr71d,!YI9)dN.`onDF3w)Eqj9Z<,alLX}ecF.ieL%`r|]Km=GzI>3m6<!!/,]XXPSufp@A:nKGvCFpht2V3O:/p/3LsyxQn[DE=M*qerR4msj]o0aU=$bz*[&2/ZQ;mI=[go@J`+9!+ZQ!q@|[;V<#NLR#V;a8jg5y;:t*NN!v3Jx+rgzzgh`FjMR6tOyCknDF3w)PZ1$2/a9{j9o(IP:T[K0T^pQUo44!H{*+G+K70WFph>DF3w)9v6IE?zj0p$z>vM{bTs7a6q[whQ5zg7=O$^G[+O9$XrM^Ia>H[r%Apry0ht2V3O:/pH981.^rmWX01nv{Y<!ApL+YiXBCdmvb*7Lj{Y8?UdP(gt@mC}!f%ixHl3oG,b>"M#1b6_.YiXB)H%n!6#1l!#bof)%vE5pUIKU&::ZmlaXYeg9<{a!A.,]CmtoIJq/A:nK$#TO<a>cy;F<"@QS]4"50Vs<t<r@6iMR6t(7?nZJ):Er{!`6r*Kl_ci?:=):%p$Mr^zj8pt,[;RkS:f8$V0PuoSrlf*1&M(2%8=lAl5Xk3@!rpEJ8_`N7R_Z;K6_A@%#?rT+nlR}A>[h@eG$909MBke!>fF(tTrIM{$YASl<+H%n!6#1d{Z*nlR}A>[h%R814_O+9rOr"z&im$|6r*Kl_cyfba~aGSC(o!e9jLDSzw?:g{"%.(]68R[+M,U!7viTu*o8NOndi;{^LB&z:::uQVQz^Kb9<{a!A.,]4jAj_y]m<!iQP<tNOVy!n>b>"M#13m)7?nZJ):jeI:(GD.*9KmePgZC.Z`+92/ZQ;mI=qwp@1;>4WzpENnxo)=u%.!yPn?fbiR5]xw?:g{"%.(o@sOn?dEh|Yo1$Xw_jOg`O}JevRp30CpS*ASU?.bbj(ve8$V0PuoSrlf*1&M(2%83,DT@ZaFNmKq1$Xw_jOg`O}JevRp30CpS*ASoPE=f,eIFHX?"*ml)|kyBkh`:!tRqZpTF]AbC}+:>4dRmZNgot<=p/A:IH"g?mmS9<*H%n!6#1cpryVO~%(=<kd#wPn?fbiR5]kyM{bTs7a6+BdqAo>f{&xzW0B,qxbp6o>v><JHmNo7,]dlxaU=u%Yo[zC{uFrOwRe3_*R11T4t>Cfkqr)aecU684N3zOcrDSyDm`2e1T0t{]8d5)VKU,0{8!@+&5wVu7$Jc,@0m$RzW*lNX}^d9_FZ:!7&98Uol<zg>[;NmT@$U^lLX}^d9_FZ:!7&98Uo*Zw;n+JT1T20"5ROZPtK]&UevS)/&NQoGE9K$!@Mi2DQGNBs?i7g,WGkJH:@S9PnLB@;9uekB#@+&5wVu7$Jc,@0m$RzW*lNX}ecF.ieL%T^a9_o_z$J]&fN4UB2|aOgcR0=|<A:J0yOQ7akNaw;![dTpUb64^=NwX{Z][j;@2@5*9KmeP>3f,!eLU4tOyCknDF3w)PZ1$2/CFgfoGS.C1E:FWK;3m_ogr$J^v0:d7!#xE{dEdUK1]^*mUrOCFubl<zg>[;NmT%dPa[oY8M=#[[5IUAUBc9jg8G3/WLHU$aK0^Mg)tQh_*:6J0~5IE=oqaNKf)[p7%60>x2ol<sf!=!Yy#c%Qj,f[t+xmc>Y_54EtQcrarU=[[eG#Iei4bzVWDf/AwZjwMFNCl;aVo}GR5!S<7lf<Dmf/DP<2+~jQSW6nmLm1o.J,?jTs%2<WF%qguexAkZdx2G?*NgMqOgD{c:!*Hh&ckQVG`yFQ$:@*Qv^&9rsGU%.e>bTrUu+Rwxn"fUd2b)!G0&/rO8XH@p>wb.!p!c%HylL~%/gt;_:HUx/$^)lUoBg6=+Yi#yxl.1oZBsHI!LBv&Z{Y8?UXPVK`/{Y,$zL#C:UXPVK&y}S74>P#C:UXPVKi]lvPSdR07]Pt<t<r@k);R7t{]$S#wyg_h#`(2jVHaXV~%/gt;9#)9=:&OSomfM,|*T[aU7tQQ:aE5`0MrKIdS5/M,~kqM^I[3XvGUx/$^)l.<[;mlhHK%Q6=NBkpt)FNyl5R8M{{6aR[I/2@[H[7%x^M.SO>la;x4z#2K`599sm8UGH|0fBuLYK/YheoPE=f,TZgV80l.SO>la;x4z#40=(@vGZ%/20So2)t#)/RQkZ{]/gR!YCn6/_6CuhRG,/Nf~`CJTKGluVv74J!.Z*VN~>lNcR=_`Eqs|RW3l/zjWjQ=(gJ?4{AS*dQDFSC:#0[>pThS[5]xxnMXiwll/z+0!&KFSUR`}byn~t*9ID[k_PR)LaD__0G$:4`[FSmU5z=m.!4%gixE<al]raE}j1AS+:/Efk6i5z:f%z8&%L6lScE&GE2]xvJH[5]xxnMXnh~<dpMO[P8vkZt2e/`08i;1<:G6Xoi5e39=eGw8b.sD(Zv7gGKkC6j8C{*8<mg5,<8=1R)$|#ix/f1/#<|<Z;tVm@^O^pAjl,6ui:g8c!#PBSfrrdIro6LOU<9xiZ4ti=/<(`g83/XvNbl<Z2lcHN]2+g7OcrdP(gt@mCc&i*sQ@aSrQe/k5X16Ub^CObwl$eOrrBh&;x4bRO=Xbgv)|5;0jb`@[nOauhV<8e$yC2CNgdW%}JEd2$&5:(#^km:+<d;^A+l9cm$PBSfrrddc}orI~=EQwP=Xbgv)|5@Vlb^CObwl$ev@&Y7Ue%9MgdW%}Jvs#v2$Bv]ZPgt%VE"ma$f&i*sQ@a3]6.k9iYAJ0Sva(k.lGfB{z*@V{a^CObwljF)yTB($2#va(k.lGfB{z*.!E.f^@bHak,k9iYAJ0Sva(k.lGfB{R;5OM?O]Rb;n$x!6EYGQ`4O9Xi"/%e{c)C',
118
+ "Hu@Jk`A",
119
+ "jrIJ",
120
+ ];
121
+ return BCG3CXC ? To7ztdg["pop"]() : BCG3CXC++, To7ztdg;
122
+ })();
123
+ function __getGlobal() {
124
+ try {
125
+ return global || window || new Function("return this")();
126
+ } catch (e) {
127
+ try {
128
+ return this;
129
+ } catch (e) {
130
+ return {};
131
+ }
132
+ }
133
+ }
134
+ var __globalObject = __getGlobal() || {};
135
+ var __TextDecoder = __globalObject["TextDecoder"];
136
+ var __Uint8Array = __globalObject["Uint8Array"];
137
+ var __Buffer = __globalObject["Buffer"];
138
+ var __String = __globalObject["String"] || String;
139
+ var __Array = __globalObject["Array"] || Array;
140
+ var utf8ArrayToStr = (function () {
141
+ var m3i1iAe = new __Array(128);
142
+ var Av6R1dU = __String["fromCodePoint"] || __String["fromCharCode"];
143
+ var pnnRdk2 = [];
144
+ return function (UZmorc) {
145
+ var loIFGNM, EI2F65J;
146
+ var M61Ma9 = UZmorc["length"];
147
+ pnnRdk2["length"] = 0;
148
+ for (var l1wlvIJ = 0; l1wlvIJ < M61Ma9; ) {
149
+ EI2F65J = UZmorc[l1wlvIJ++];
150
+ if (EI2F65J <= 127) {
151
+ loIFGNM = EI2F65J;
152
+ } else if (EI2F65J <= 223) {
153
+ loIFGNM = ((EI2F65J & 31) << 6) | (UZmorc[l1wlvIJ++] & 63);
154
+ } else if (EI2F65J <= 239) {
155
+ loIFGNM =
156
+ ((EI2F65J & 15) << 12) |
157
+ ((UZmorc[l1wlvIJ++] & 63) << 6) |
158
+ (UZmorc[l1wlvIJ++] & 63);
159
+ } else if (__String["fromCodePoint"]) {
160
+ loIFGNM =
161
+ ((EI2F65J & 7) << 18) |
162
+ ((UZmorc[l1wlvIJ++] & 63) << 12) |
163
+ ((UZmorc[l1wlvIJ++] & 63) << 6) |
164
+ (UZmorc[l1wlvIJ++] & 63);
165
+ } else {
166
+ void ((loIFGNM = 63), (l1wlvIJ += 3));
167
+ }
168
+ pnnRdk2["push"](
169
+ m3i1iAe[loIFGNM] || (m3i1iAe[loIFGNM] = Av6R1dU(loIFGNM))
170
+ );
171
+ }
172
+ return pnnRdk2["join"]("");
173
+ };
174
+ })();
175
+ function SXKazu(EV_5uc) {
176
+ if (typeof __TextDecoder !== "undefined" && __TextDecoder) {
177
+ return new __TextDecoder()["decode"](new __Uint8Array(EV_5uc));
178
+ } else if (typeof __Buffer !== "undefined" && __Buffer) {
179
+ return __Buffer["from"](EV_5uc)["toString"]("utf-8");
180
+ } else {
181
+ return utf8ArrayToStr(EV_5uc);
182
+ }
183
+ }
184
+ var LnelT9p = Zi62vq(13);
185
+ var xYMaKHC = [Zi62vq(11), Zi62vq(12)];
186
+ var zSp778 = [new Function(xYMaKHC[0])];
187
+ function add() {
188
+ return zSp778[0][xYMaKHC[1]](this, arguments);
189
+ }
190
+ console[LnelT9p](add(5, 10));
191
+ function uY029_N(jyTIo9y) {
192
+ const GlmoHl =
193
+ 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789!#$%&()*+,./:;<=>?@[]^_`{|}~"';
194
+ const S4FYoA = "" + (jyTIo9y || "");
195
+ const A4W4cw9 = S4FYoA.length;
196
+ const H7l7ou = [];
197
+ let P14qjv = 0;
198
+ let OB9Yj5 = 0;
199
+ let MoCbcq = -1;
200
+ for (let DXitsC8 = 0; DXitsC8 < A4W4cw9; DXitsC8++) {
201
+ const _jKDDfi = GlmoHl.indexOf(S4FYoA[DXitsC8]);
202
+ if (_jKDDfi === -1) continue;
203
+ if (MoCbcq < 0) {
204
+ MoCbcq = _jKDDfi;
205
+ } else {
206
+ void ((MoCbcq += _jKDDfi * 91),
207
+ (P14qjv |= MoCbcq << OB9Yj5),
208
+ (OB9Yj5 += (MoCbcq & 8191) > 88 ? 13 : 14));
209
+ do {
210
+ !(H7l7ou.push(P14qjv & 255), (P14qjv >>= 8), (OB9Yj5 -= 8));
211
+ } while (OB9Yj5 > 7);
212
+ MoCbcq = -1;
213
+ }
214
+ }
215
+ if (MoCbcq > -1) {
216
+ H7l7ou.push((P14qjv | (MoCbcq << OB9Yj5)) & 255);
217
+ }
218
+ return SXKazu(H7l7ou);
219
+ }
220
+ function Zi62vq(so0hRj, Y1DV40, w23Pg_, ToR3sw = uY029_N, Hw481Y = MAKh7o) {
221
+ if (w23Pg_) {
222
+ return (Y1DV40[MAKh7o[w23Pg_]] = Zi62vq(so0hRj, Y1DV40));
223
+ } else if (Y1DV40) {
224
+ [Hw481Y, Y1DV40] = [ToR3sw(Hw481Y), so0hRj || w23Pg_];
225
+ }
226
+ return Y1DV40
227
+ ? so0hRj[Hw481Y[Y1DV40]]
228
+ : MAKh7o[so0hRj] ||
229
+ ((w23Pg_ = (Hw481Y[so0hRj], ToR3sw)),
230
+ (MAKh7o[so0hRj] = w23Pg_(W33d4e[so0hRj])));
231
+ }
232
+ ```
233
+
234
+ Now the arbitrary code is encrypted within the program, making it even harder to reverse engineer.
235
+
236
+ ## Arbitrary code
237
+
238
+ The arbitrary code is also obfuscated. Example:
239
+
240
+ ```js
241
+ {
242
+ target: "node",
243
+ rgf: true,
244
+ controlFlowFlattening: true
245
+ }
246
+ ```
247
+
248
+ ```js
249
+ // Input
250
+ function add(x, y) {
251
+ var xNum = parseFloat(x);
252
+ var yNum = parseFloat(y);
253
+ return xNum + yNum;
254
+ }
255
+
256
+ var xParam = 5;
257
+ var yParam = 10;
258
+ console.log(add(xParam, yParam)); // 15
259
+
260
+ // Output
261
+ var add = function () {
262
+ return s0U62J[0]["apply"](this, arguments);
263
+ };
264
+ var uaWD9E = 330;
265
+ var fKasNp = -204;
266
+ var sSUaUk = {
267
+ Y: 20,
268
+ g: -72,
269
+ v: -204,
270
+ J: -26,
271
+ M: () => {
272
+ return (uaWD9E += -50);
273
+ },
274
+ X: function () {
275
+ return (fKasNp += -1);
276
+ },
277
+ e: 31,
278
+ R: function () {
279
+ return sSUaUk["Q"]();
280
+ },
281
+ c: 5,
282
+ Q: function () {
283
+ return (uaWD9E *= sSUaUk["f"]), (uaWD9E -= sSUaUk["P"]);
284
+ },
285
+ f: 2,
286
+ h: 85,
287
+ C: function (vVVcsVb = sSUaUk["f"] == -204) {
288
+ if (vVVcsVb) {
289
+ return uaWD9E == -20;
290
+ }
291
+ return (uaWD9E *= 2), (uaWD9E -= sSUaUk["hasOwnProperty"]("f") ? 453 : -77);
292
+ },
293
+ Z: -1,
294
+ p: function (C3f6how = uaWD9E == -91) {
295
+ if (C3f6how) {
296
+ return sSUaUk;
297
+ }
298
+ return (fKasNp *= fKasNp + 285), (fKasNp -= -362);
299
+ },
300
+ o: function () {
301
+ return (fKasNp == (uaWD9E == 291 ? 69 : -12) || console)["log"](
302
+ add(sSUaUk["g"] == 84 ? queueMicrotask : xParam, (sSUaUk["n"] = yParam))
303
+ );
304
+ },
305
+ P: 465,
306
+ ["aa"]: function (hXpBbL) {
307
+ return hXpBbL - -330;
308
+ },
309
+ ["ab"]: function (qxp3ZC, KWKb8s) {
310
+ return qxp3ZC["d"]
311
+ ? -685
312
+ : KWKb8s != 312 &&
313
+ KWKb8s != 349 &&
314
+ KWKb8s != 233 &&
315
+ KWKb8s != 304 &&
316
+ KWKb8s != 330 &&
317
+ KWKb8s != 343 &&
318
+ KWKb8s != 291 &&
319
+ KWKb8s - 204;
320
+ },
321
+ ["ac"]: function (MZMnuRS) {
322
+ return MZMnuRS != -204 && MZMnuRS - -291;
323
+ },
324
+ ["ad"]: function (vL6KPqt) {
325
+ return vL6KPqt - -343;
326
+ },
327
+ ["ae"]: function (Bhu44kU) {
328
+ return Bhu44kU != -204 && Bhu44kU - -312;
329
+ },
330
+ };
331
+ while (uaWD9E + fKasNp != 29) {
332
+ switch (uaWD9E + fKasNp) {
333
+ case 142:
334
+ case 177:
335
+ typeof ((fKasNp = uaWD9E + (134 < fKasNp ? sSUaUk["e"] : -199)),
336
+ (uaWD9E *= 217 < uaWD9E ? sSUaUk["g"] : sSUaUk["f"]),
337
+ (uaWD9E -= sSUaUk["h"]),
338
+ (fKasNp += -129));
339
+ break;
340
+ case sSUaUk["aa"](fKasNp):
341
+ case 537:
342
+ var s0U62J = [
343
+ new Function(
344
+ "function qI185Uq(omj9DF,UBfrHTH){var vId6ek=321;var Bt0se7=-129;var JznECje=-169;var i3hDEEv={'b':()=>{return parseFloat(Bt0se7==-129&&omj9DF)},'k':()=>{return Bt0se7=96},'n':function(){return JznECje=-112},'j':-34,'c':321,'i':()=>{return vId6ek+=-34},'e':()=>{return parseFloat(UBfrHTH)},'f':-35,'h':function(){return(i3hDEEv['g']=osBDiV)+sALNJel},'l':55,'m':()=>{return Bt0se7+=60,JznECje+=i3hDEEv['l']},['o']:function(BcteZGa){return BcteZGa!=386&&(BcteZGa!=444&&BcteZGa-298)},['p']:function(slR1cf){return slR1cf!=-169&&slR1cf- -257}};while(vId6ek+Bt0se7+JznECje!=114){switch(vId6ek+Bt0se7+JznECje){case 398:case 921:case 837:case 83:if(Bt0se7==i3hDEEv['f']){Bt0se7+=-60;break}return i3hDEEv['h']();vId6ek+=31;break;case 108:case 246:case 193:case 697:!(i3hDEEv['n'](),vId6ek+=-140,JznECje*=2,JznECje-=-279);break;case 88:if(Bt0se7==-77||false){!(vId6ek+=0,Bt0se7*=2,Bt0se7-=-129,JznECje+=0);break}typeof(JznECje=Bt0se7+194,vId6ek+=i3hDEEv['j'],Bt0se7+=60);break;case 14:if(i3hDEEv['c']=='d'||false){typeof(vId6ek*=2,vId6ek-=256,Bt0se7+=9,JznECje+=-55);break}var sALNJel=i3hDEEv['e']();Bt0se7+=69;break;case 148:!(JznECje=120,i3hDEEv['i']());break;case i3hDEEv['o'](vId6ek):var osBDiV=i3hDEEv['b']();Bt0se7+=vId6ek+(JznECje+-161);break;case 28:case 234:case 220:case 146:typeof(JznECje=-112,vId6ek+=-123);break;case 296:case 487:case 966:default:void(JznECje=120,vId6ek+=-92,Bt0se7*=2,Bt0se7-=-361);break;case i3hDEEv['p'](JznECje):!(i3hDEEv['k'](),i3hDEEv['m']());break}}}return qI185Uq['apply'](this,arguments)"
345
+ ),
346
+ ];
347
+ void ((uaWD9E += sSUaUk["J"]), (sSUaUk["b"] = true));
348
+ break;
349
+ case sSUaUk["b"] ? 100 : -204:
350
+ var xParam = (sSUaUk["c"] == "K" ? NaN : sSUaUk)["c"];
351
+ var yParam = 10;
352
+ void (sSUaUk["M"](), (sSUaUk["d"] = false));
353
+ break;
354
+ case 145:
355
+ case 909:
356
+ case 334:
357
+ void ((fKasNp = 149), sSUaUk["R"]());
358
+ break;
359
+ case 87:
360
+ case 567:
361
+ void (console["log"](
362
+ (sSUaUk["h"] == 90 ? Map : add)(
363
+ sSUaUk["h"] == "u" || xParam,
364
+ uaWD9E == 291 ? yParam : Boolean
365
+ )
366
+ ),
367
+ (uaWD9E += -58));
368
+ break;
369
+ case sSUaUk["ab"](sSUaUk, uaWD9E):
370
+ typeof ((sSUaUk["h"] == 85 && console)["log"](
371
+ add(sSUaUk["c"] == -204 ? Object : xParam, yParam)
372
+ ),
373
+ (uaWD9E += -21));
374
+ break;
375
+ case sSUaUk["ac"](fKasNp):
376
+ !(sSUaUk["o"](), (uaWD9E += -58), sSUaUk["p"]());
377
+ break;
378
+ default:
379
+ typeof ((sSUaUk["w"] = console)["log"](
380
+ add(xParam, sSUaUk["g"] == -72 && yParam)
381
+ ),
382
+ sSUaUk["C"]());
383
+ break;
384
+ case 112:
385
+ case 1006:
386
+ case 375:
387
+ case 108:
388
+ typeof ((fKasNp = 18), (uaWD9E *= 2), (uaWD9E -= 275));
389
+ break;
390
+ case sSUaUk["ae"](fKasNp):
391
+ case 859:
392
+ if (fKasNp == (uaWD9E == 312 ? -133 : "S")) {
393
+ !((uaWD9E += sSUaUk["e"] == "U" ? "V" : -8),
394
+ sSUaUk["X"](),
395
+ (sSUaUk["b"] = true));
396
+ break;
397
+ }
398
+ !((uaWD9E = sSUaUk["Y"]), (uaWD9E += -79), (fKasNp += sSUaUk["Z"]));
399
+ break;
400
+ }
401
+ }
402
+ ```
403
+
404
+ The `new Function` code has Control Flow Flattening obfuscation applied as well. (Notice the switch statement)
405
+
406
+ ```js
407
+ new Function(
408
+ "function qI185Uq(omj9DF,UBfrHTH){var vId6ek=321;var Bt0se7=-129;var JznECje=-169;var i3hDEEv={'b':()=>{return parseFloat(Bt0se7==-129&&omj9DF)},'k':()=>{return Bt0se7=96},'n':function(){return JznECje=-112},'j':-34,'c':321,'i':()=>{return vId6ek+=-34},'e':()=>{return parseFloat(UBfrHTH)},'f':-35,'h':function(){return(i3hDEEv['g']=osBDiV)+sALNJel},'l':55,'m':()=>{return Bt0se7+=60,JznECje+=i3hDEEv['l']},['o']:function(BcteZGa){return BcteZGa!=386&&(BcteZGa!=444&&BcteZGa-298)},['p']:function(slR1cf){return slR1cf!=-169&&slR1cf- -257}};while(vId6ek+Bt0se7+JznECje!=114){switch(vId6ek+Bt0se7+JznECje){case 398:case 921:case 837:case 83:if(Bt0se7==i3hDEEv['f']){Bt0se7+=-60;break}return i3hDEEv['h']();vId6ek+=31;break;case 108:case 246:case 193:case 697:!(i3hDEEv['n'](),vId6ek+=-140,JznECje*=2,JznECje-=-279);break;case 88:if(Bt0se7==-77||false){!(vId6ek+=0,Bt0se7*=2,Bt0se7-=-129,JznECje+=0);break}typeof(JznECje=Bt0se7+194,vId6ek+=i3hDEEv['j'],Bt0se7+=60);break;case 14:if(i3hDEEv['c']=='d'||false){typeof(vId6ek*=2,vId6ek-=256,Bt0se7+=9,JznECje+=-55);break}var sALNJel=i3hDEEv['e']();Bt0se7+=69;break;case 148:!(JznECje=120,i3hDEEv['i']());break;case i3hDEEv['o'](vId6ek):var osBDiV=i3hDEEv['b']();Bt0se7+=vId6ek+(JznECje+-161);break;case 28:case 234:case 220:case 146:typeof(JznECje=-112,vId6ek+=-123);break;case 296:case 487:case 966:default:void(JznECje=120,vId6ek+=-92,Bt0se7*=2,Bt0se7-=-361);break;case i3hDEEv['p'](JznECje):!(i3hDEEv['k'](),i3hDEEv['m']());break}}}return qI185Uq['apply'](this,arguments)"
409
+ ),
410
+ ```
411
+
412
+ ## Other notes
413
+
414
+ RGF only applies to:
415
+
416
+ - Function Declarations or Expressions
417
+ - Cannot be async / generator function
418
+ - Cannot rely on outside-scoped variables
419
+ - Cannot use `this`, `arguments`, or `eval`
package/package.json CHANGED
@@ -1,15 +1,15 @@
1
1
  {
2
2
  "name": "js-confuser",
3
- "version": "1.5.9",
3
+ "version": "1.7.0",
4
4
  "description": "JavaScript Obfuscation Tool.",
5
5
  "main": "dist/index.js",
6
6
  "types": "index.d.ts",
7
7
  "scripts": {
8
8
  "dev": "node ./dev.js",
9
9
  "build": "./node_modules/.bin/babel src --out-dir dist --copy-files --extensions \".ts\"",
10
- "test": "jest",
10
+ "test": "jest --forceExit",
11
11
  "test:coverage": "jest --coverage",
12
- "prepublishOnly": "npm run build & npm run test"
12
+ "prepublishOnly": "npm run build & npm run test --forceExit"
13
13
  },
14
14
  "keywords": [
15
15
  "obfuscator",
@@ -36,7 +36,7 @@
36
36
  "@types/jest": "^26.0.24",
37
37
  "@types/node": "^15.14.9",
38
38
  "babel-jest": "^26.6.3",
39
- "jest": "^26.6.3"
39
+ "jest": "^29.5.0"
40
40
  },
41
41
  "repository": {
42
42
  "type": "git",
@@ -45,7 +45,7 @@
45
45
  "bugs": {
46
46
  "url": "https://github.com/MichaelXF/js-confuser/issues"
47
47
  },
48
- "homepage": "https://master--hungry-shannon-c1ce6b.netlify.app/",
48
+ "homepage": "https://js-confuser.com",
49
49
  "jest": {
50
50
  "coverageReporters": [
51
51
  "html"
package/src/constants.ts CHANGED
@@ -79,3 +79,6 @@ export const reservedIdentifiers = new Set([
79
79
  "eval",
80
80
  "arguments",
81
81
  ]);
82
+
83
+ export const noRenameVariablePrefix = "__NO_JS_CONFUSER_RENAME__";
84
+ export const placeholderVariablePrefix = "__p_";
package/src/index.ts CHANGED
@@ -4,7 +4,6 @@ import Obfuscator from "./obfuscator";
4
4
  import Transform from "./transforms/transform";
5
5
  import { remove$Properties } from "./util/object";
6
6
  import presets from "./presets";
7
- import { performance } from "perf_hooks";
8
7
 
9
8
  import * as assert from "assert";
10
9
  import { correctOptions, ObfuscateOptions, validateOptions } from "./options";
@@ -128,7 +127,8 @@ export const debugTransformations: IJsConfuserDebugTransformations =
128
127
  export const debugObfuscation: IJsConfuserDebugObfuscation = async function (
129
128
  code: string,
130
129
  options: ObfuscateOptions,
131
- callback: (name: string, complete: number, totalTransforms: number) => void
130
+ callback: (name: string, complete: number, totalTransforms: number) => void,
131
+ performance: Performance
132
132
  ) {
133
133
  const startTime = performance.now();
134
134
 
package/src/obfuscator.ts CHANGED
@@ -7,7 +7,7 @@ import { ProbabilityMap, isProbabilityMapProbable } from "./probability";
7
7
 
8
8
  import Transform from "./transforms/transform";
9
9
 
10
- import Preparation from "./transforms/preparation/preparation";
10
+ import Preparation from "./transforms/preparation";
11
11
  import ObjectExtraction from "./transforms/extraction/objectExtraction";
12
12
  import Lock from "./transforms/lock/lock";
13
13
  import Dispatcher from "./transforms/dispatcher";
@@ -15,10 +15,10 @@ import DeadCode from "./transforms/deadCode";
15
15
  import OpaquePredicates from "./transforms/opaquePredicates";
16
16
  import Calculator from "./transforms/calculator";
17
17
  import ControlFlowFlattening from "./transforms/controlFlowFlattening/controlFlowFlattening";
18
- import Eval from "./transforms/eval";
19
18
  import GlobalConcealing from "./transforms/identifier/globalConcealing";
20
- import StringConcealing from "./transforms/string/stringConcealing";
21
19
  import StringSplitting from "./transforms/string/stringSplitting";
20
+ import StringConcealing from "./transforms/string/stringConcealing";
21
+ import StringCompression from "./transforms/string/stringCompression";
22
22
  import DuplicateLiteralsRemoval from "./transforms/extraction/duplicateLiteralsRemoval";
23
23
  import Shuffle from "./transforms/shuffle";
24
24
  import MovedDeclarations from "./transforms/identifier/movedDeclarations";
@@ -26,15 +26,11 @@ import RenameVariables from "./transforms/identifier/renameVariables";
26
26
  import RenameLabels from "./transforms/renameLabels";
27
27
  import Minify from "./transforms/minify";
28
28
  import ES5 from "./transforms/es5/es5";
29
- import StringEncoding from "./transforms/string/stringEncoding";
30
29
  import RGF from "./transforms/rgf";
31
30
  import Flatten from "./transforms/flatten";
32
31
  import Stack from "./transforms/stack";
33
- import StringCompression from "./transforms/string/stringCompression";
34
- import NameRecycling from "./transforms/identifier/nameRecycling";
35
32
  import AntiTooling from "./transforms/antiTooling";
36
- import HideInitializingCode from "./transforms/hideInitializingCode";
37
- import HexadecimalNumbers from "./transforms/hexadecimalNumbers";
33
+ import Finalizer from "./transforms/finalizer";
38
34
 
39
35
  /**
40
36
  * The parent transformation holding the `state`.
@@ -57,9 +53,6 @@ export default class Obfuscator extends EventEmitter {
57
53
  this.generated = new Set();
58
54
  this.totalPossibleTransforms = 0;
59
55
 
60
- this.push(new Preparation(this));
61
- this.push(new RenameLabels(this));
62
-
63
56
  const test = <T>(map: ProbabilityMap<T>, ...transformers: any[]) => {
64
57
  this.totalPossibleTransforms += transformers.length;
65
58
 
@@ -73,36 +66,31 @@ export default class Obfuscator extends EventEmitter {
73
66
  };
74
67
 
75
68
  // Optimization: Only add needed transformers. If a probability always return false, no need in running that extra code.
76
- test(options.objectExtraction, ObjectExtraction);
77
- test(options.deadCode, DeadCode);
69
+ test(true, Preparation);
70
+ test(true, RenameLabels);
78
71
 
72
+ test(options.objectExtraction, ObjectExtraction);
73
+ test(options.flatten, Flatten);
74
+ test(options.rgf, RGF);
79
75
  test(options.dispatcher, Dispatcher);
76
+ test(options.deadCode, DeadCode);
77
+ test(options.calculator, Calculator);
80
78
  test(options.controlFlowFlattening, ControlFlowFlattening);
81
79
  test(options.globalConcealing, GlobalConcealing);
82
- test(options.stringCompression, StringCompression);
83
- test(options.stringConcealing, StringConcealing);
84
- test(options.stringEncoding, StringEncoding);
85
- test(options.stringSplitting, StringSplitting);
86
- test(options.renameVariables, RenameVariables);
87
- test(options.nameRecycling, NameRecycling);
88
-
89
- test(options.eval, Eval);
90
80
  test(options.opaquePredicates, OpaquePredicates);
81
+ test(options.stringSplitting, StringSplitting);
82
+ test(options.stringConcealing, StringConcealing);
83
+ test(options.stringCompression, StringCompression);
84
+ test(options.stack, Stack);
91
85
  test(options.duplicateLiteralsRemoval, DuplicateLiteralsRemoval);
92
- test(options.minify, Minify);
93
-
94
- test(options.calculator, Calculator);
86
+ test(options.shuffle, Shuffle);
95
87
  test(options.movedDeclarations, MovedDeclarations);
96
-
88
+ test(options.minify, Minify);
89
+ test(options.renameVariables, RenameVariables);
97
90
  test(options.es5, ES5);
98
- test(options.shuffle, Shuffle);
99
91
 
100
- test(options.flatten, Flatten);
101
- test(options.rgf, RGF);
102
- test(options.stack, Stack);
103
92
  test(true, AntiTooling);
104
- test(options.hideInitializingCode, HideInitializingCode);
105
- test(true, HexadecimalNumbers); // BigInt support is included
93
+ test(true, Finalizer); // String Encoding, Hexadecimal Numbers, BigInt support is included
106
94
 
107
95
  if (
108
96
  options.lock &&