js-confuser 1.7.3 → 2.0.0-alpha.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 (269) hide show
  1. package/.github/ISSUE_TEMPLATE/bug_report.md +6 -4
  2. package/CHANGELOG.md +70 -0
  3. package/Migration.md +57 -0
  4. package/README.md +23 -929
  5. package/dist/constants.js +65 -14
  6. package/dist/index.js +108 -160
  7. package/dist/obfuscator.js +316 -118
  8. package/dist/options.js +1 -119
  9. package/dist/order.js +30 -30
  10. package/dist/presets.js +47 -45
  11. package/dist/probability.js +25 -32
  12. package/dist/templates/bufferToStringTemplate.js +9 -0
  13. package/dist/templates/deadCodeTemplates.js +9 -0
  14. package/dist/templates/getGlobalTemplate.js +19 -0
  15. package/dist/templates/integrityTemplate.js +30 -0
  16. package/dist/templates/setFunctionLengthTemplate.js +9 -0
  17. package/dist/templates/stringCompressionTemplate.js +10 -0
  18. package/dist/templates/tamperProtectionTemplates.js +21 -0
  19. package/dist/templates/template.js +199 -184
  20. package/dist/transforms/astScrambler.js +100 -0
  21. package/dist/transforms/calculator.js +70 -127
  22. package/dist/transforms/controlFlowFlattening.js +1182 -0
  23. package/dist/transforms/deadCode.js +62 -587
  24. package/dist/transforms/dispatcher.js +300 -313
  25. package/dist/transforms/extraction/duplicateLiteralsRemoval.js +88 -189
  26. package/dist/transforms/extraction/objectExtraction.js +131 -215
  27. package/dist/transforms/finalizer.js +56 -59
  28. package/dist/transforms/flatten.js +275 -276
  29. package/dist/transforms/functionOutlining.js +230 -0
  30. package/dist/transforms/identifier/globalConcealing.js +214 -135
  31. package/dist/transforms/identifier/movedDeclarations.js +167 -91
  32. package/dist/transforms/identifier/renameVariables.js +239 -193
  33. package/dist/transforms/lock/integrity.js +61 -184
  34. package/dist/transforms/lock/lock.js +261 -387
  35. package/dist/transforms/minify.js +431 -436
  36. package/dist/transforms/opaquePredicates.js +65 -118
  37. package/dist/transforms/pack.js +160 -0
  38. package/dist/transforms/plugin.js +179 -0
  39. package/dist/transforms/preparation.js +261 -173
  40. package/dist/transforms/renameLabels.js +132 -56
  41. package/dist/transforms/rgf.js +140 -267
  42. package/dist/transforms/shuffle.js +52 -145
  43. package/dist/transforms/string/encoding.js +44 -175
  44. package/dist/transforms/string/stringCompression.js +79 -155
  45. package/dist/transforms/string/stringConcealing.js +189 -225
  46. package/dist/transforms/string/stringEncoding.js +32 -40
  47. package/dist/transforms/string/stringSplitting.js +54 -55
  48. package/dist/transforms/variableMasking.js +232 -0
  49. package/dist/utils/ControlObject.js +125 -0
  50. package/dist/utils/IntGen.js +46 -0
  51. package/dist/utils/NameGen.js +106 -0
  52. package/dist/utils/ast-utils.js +560 -0
  53. package/dist/utils/function-utils.js +56 -0
  54. package/dist/utils/gen-utils.js +48 -0
  55. package/dist/utils/node.js +77 -0
  56. package/dist/utils/object-utils.js +21 -0
  57. package/dist/utils/random-utils.js +91 -0
  58. package/dist/utils/static-utils.js +64 -0
  59. package/dist/validateOptions.js +122 -0
  60. package/index.d.ts +1 -17
  61. package/package.json +27 -22
  62. package/src/constants.ts +139 -82
  63. package/src/index.ts +70 -165
  64. package/src/obfuscationResult.ts +43 -0
  65. package/src/obfuscator.ts +328 -135
  66. package/src/options.ts +149 -658
  67. package/src/order.ts +14 -14
  68. package/src/presets.ts +39 -34
  69. package/src/probability.ts +21 -36
  70. package/src/templates/bufferToStringTemplate.ts +57 -0
  71. package/src/templates/deadCodeTemplates.ts +1185 -0
  72. package/src/templates/getGlobalTemplate.ts +72 -0
  73. package/src/templates/integrityTemplate.ts +69 -0
  74. package/src/templates/setFunctionLengthTemplate.ts +11 -0
  75. package/src/templates/stringCompressionTemplate.ts +42 -0
  76. package/src/templates/tamperProtectionTemplates.ts +116 -0
  77. package/src/templates/template.ts +149 -157
  78. package/src/transforms/astScrambler.ts +99 -0
  79. package/src/transforms/calculator.ts +96 -226
  80. package/src/transforms/controlFlowFlattening.ts +1594 -0
  81. package/src/transforms/deadCode.ts +85 -676
  82. package/src/transforms/dispatcher.ts +431 -640
  83. package/src/transforms/extraction/duplicateLiteralsRemoval.ts +147 -295
  84. package/src/transforms/extraction/objectExtraction.ts +160 -333
  85. package/src/transforms/finalizer.ts +63 -64
  86. package/src/transforms/flatten.ts +439 -557
  87. package/src/transforms/functionOutlining.ts +225 -0
  88. package/src/transforms/identifier/globalConcealing.ts +255 -266
  89. package/src/transforms/identifier/movedDeclarations.ts +228 -142
  90. package/src/transforms/identifier/renameVariables.ts +250 -271
  91. package/src/transforms/lock/integrity.ts +85 -263
  92. package/src/transforms/lock/lock.ts +338 -579
  93. package/src/transforms/minify.ts +523 -663
  94. package/src/transforms/opaquePredicates.ts +90 -229
  95. package/src/transforms/pack.ts +195 -0
  96. package/src/transforms/plugin.ts +185 -0
  97. package/src/transforms/preparation.ts +337 -231
  98. package/src/transforms/renameLabels.ts +176 -77
  99. package/src/transforms/rgf.ts +293 -424
  100. package/src/transforms/shuffle.ts +80 -254
  101. package/src/transforms/string/encoding.ts +20 -126
  102. package/src/transforms/string/stringCompression.ts +117 -307
  103. package/src/transforms/string/stringConcealing.ts +254 -342
  104. package/src/transforms/string/stringEncoding.ts +28 -47
  105. package/src/transforms/string/stringSplitting.ts +61 -75
  106. package/src/transforms/variableMasking.ts +257 -0
  107. package/src/utils/ControlObject.ts +141 -0
  108. package/src/utils/IntGen.ts +33 -0
  109. package/src/utils/NameGen.ts +106 -0
  110. package/src/utils/ast-utils.ts +667 -0
  111. package/src/utils/function-utils.ts +50 -0
  112. package/src/utils/gen-utils.ts +48 -0
  113. package/src/utils/node.ts +78 -0
  114. package/src/utils/object-utils.ts +21 -0
  115. package/src/utils/random-utils.ts +79 -0
  116. package/src/utils/static-utils.ts +66 -0
  117. package/src/validateOptions.ts +256 -0
  118. package/tsconfig.json +13 -8
  119. package/babel.config.js +0 -12
  120. package/dev.js +0 -8
  121. package/dist/compiler.js +0 -34
  122. package/dist/parser.js +0 -59
  123. package/dist/precedence.js +0 -66
  124. package/dist/templates/bufferToString.js +0 -129
  125. package/dist/templates/core.js +0 -35
  126. package/dist/templates/crash.js +0 -28
  127. package/dist/templates/es5.js +0 -137
  128. package/dist/templates/functionLength.js +0 -34
  129. package/dist/templates/globals.js +0 -9
  130. package/dist/transforms/antiTooling.js +0 -88
  131. package/dist/transforms/controlFlowFlattening/controlFlowFlattening.js +0 -1287
  132. package/dist/transforms/controlFlowFlattening/expressionObfuscation.js +0 -131
  133. package/dist/transforms/es5/antiClass.js +0 -164
  134. package/dist/transforms/es5/antiDestructuring.js +0 -193
  135. package/dist/transforms/es5/antiES6Object.js +0 -185
  136. package/dist/transforms/es5/antiSpreadOperator.js +0 -35
  137. package/dist/transforms/es5/antiTemplate.js +0 -66
  138. package/dist/transforms/es5/es5.js +0 -123
  139. package/dist/transforms/extraction/classExtraction.js +0 -83
  140. package/dist/transforms/identifier/globalAnalysis.js +0 -83
  141. package/dist/transforms/identifier/variableAnalysis.js +0 -104
  142. package/dist/transforms/lock/antiDebug.js +0 -76
  143. package/dist/transforms/stack.js +0 -349
  144. package/dist/transforms/transform.js +0 -372
  145. package/dist/traverse.js +0 -110
  146. package/dist/util/compare.js +0 -145
  147. package/dist/util/gen.js +0 -564
  148. package/dist/util/guard.js +0 -14
  149. package/dist/util/identifiers.js +0 -355
  150. package/dist/util/insert.js +0 -362
  151. package/dist/util/math.js +0 -19
  152. package/dist/util/object.js +0 -40
  153. package/dist/util/random.js +0 -156
  154. package/dist/util/scope.js +0 -20
  155. package/docs/ControlFlowFlattening.md +0 -595
  156. package/docs/Countermeasures.md +0 -70
  157. package/docs/ES5.md +0 -197
  158. package/docs/Integrity.md +0 -82
  159. package/docs/RGF.md +0 -424
  160. package/docs/RenameVariables.md +0 -116
  161. package/docs/TamperProtection.md +0 -100
  162. package/docs/Template.md +0 -117
  163. package/samples/example.js +0 -15
  164. package/samples/high.js +0 -1
  165. package/samples/input.js +0 -3
  166. package/samples/javascriptobfuscator.com.js +0 -8
  167. package/samples/jscrambler_advanced.js +0 -1894
  168. package/samples/jscrambler_light.js +0 -1134
  169. package/samples/low.js +0 -1
  170. package/samples/medium.js +0 -1
  171. package/samples/obfuscator.io.js +0 -1686
  172. package/samples/preemptive.com.js +0 -16
  173. package/src/compiler.ts +0 -35
  174. package/src/parser.ts +0 -49
  175. package/src/precedence.ts +0 -61
  176. package/src/templates/bufferToString.ts +0 -136
  177. package/src/templates/core.ts +0 -29
  178. package/src/templates/crash.ts +0 -23
  179. package/src/templates/es5.ts +0 -131
  180. package/src/templates/functionLength.ts +0 -32
  181. package/src/templates/globals.ts +0 -3
  182. package/src/transforms/antiTooling.ts +0 -102
  183. package/src/transforms/controlFlowFlattening/controlFlowFlattening.ts +0 -2153
  184. package/src/transforms/controlFlowFlattening/expressionObfuscation.ts +0 -179
  185. package/src/transforms/es5/antiClass.ts +0 -276
  186. package/src/transforms/es5/antiDestructuring.ts +0 -294
  187. package/src/transforms/es5/antiES6Object.ts +0 -267
  188. package/src/transforms/es5/antiSpreadOperator.ts +0 -56
  189. package/src/transforms/es5/antiTemplate.ts +0 -98
  190. package/src/transforms/es5/es5.ts +0 -149
  191. package/src/transforms/extraction/classExtraction.ts +0 -168
  192. package/src/transforms/identifier/globalAnalysis.ts +0 -102
  193. package/src/transforms/identifier/variableAnalysis.ts +0 -118
  194. package/src/transforms/lock/antiDebug.ts +0 -112
  195. package/src/transforms/stack.ts +0 -557
  196. package/src/transforms/transform.ts +0 -441
  197. package/src/traverse.ts +0 -120
  198. package/src/types.ts +0 -133
  199. package/src/util/compare.ts +0 -181
  200. package/src/util/gen.ts +0 -651
  201. package/src/util/guard.ts +0 -17
  202. package/src/util/identifiers.ts +0 -494
  203. package/src/util/insert.ts +0 -419
  204. package/src/util/math.ts +0 -15
  205. package/src/util/object.ts +0 -39
  206. package/src/util/random.ts +0 -221
  207. package/src/util/scope.ts +0 -21
  208. package/test/code/Cash.src.js +0 -1011
  209. package/test/code/Cash.test.ts +0 -132
  210. package/test/code/Dynamic.src.js +0 -118
  211. package/test/code/Dynamic.test.ts +0 -49
  212. package/test/code/ES6.src.js +0 -235
  213. package/test/code/ES6.test.ts +0 -42
  214. package/test/code/NewFeatures.test.ts +0 -19
  215. package/test/code/StrictMode.src.js +0 -65
  216. package/test/code/StrictMode.test.js +0 -37
  217. package/test/compare.test.ts +0 -104
  218. package/test/index.test.ts +0 -249
  219. package/test/options.test.ts +0 -150
  220. package/test/presets.test.ts +0 -22
  221. package/test/probability.test.ts +0 -44
  222. package/test/templates/template.test.ts +0 -224
  223. package/test/transforms/antiTooling.test.ts +0 -52
  224. package/test/transforms/calculator.test.ts +0 -78
  225. package/test/transforms/controlFlowFlattening/controlFlowFlattening.test.ts +0 -1274
  226. package/test/transforms/controlFlowFlattening/expressionObfuscation.test.ts +0 -192
  227. package/test/transforms/deadCode.test.ts +0 -85
  228. package/test/transforms/dispatcher.test.ts +0 -457
  229. package/test/transforms/es5/antiClass.test.ts +0 -427
  230. package/test/transforms/es5/antiDestructuring.test.ts +0 -157
  231. package/test/transforms/es5/antiES6Object.test.ts +0 -245
  232. package/test/transforms/es5/antiTemplate.test.ts +0 -116
  233. package/test/transforms/es5/es5.test.ts +0 -110
  234. package/test/transforms/extraction/classExtraction.test.ts +0 -86
  235. package/test/transforms/extraction/duplicateLiteralsRemoval.test.ts +0 -200
  236. package/test/transforms/extraction/objectExtraction.test.ts +0 -491
  237. package/test/transforms/flatten.test.ts +0 -721
  238. package/test/transforms/hexadecimalNumbers.test.ts +0 -62
  239. package/test/transforms/identifier/globalConcealing.test.ts +0 -142
  240. package/test/transforms/identifier/movedDeclarations.test.ts +0 -275
  241. package/test/transforms/identifier/renameVariables.test.ts +0 -695
  242. package/test/transforms/lock/antiDebug.test.ts +0 -66
  243. package/test/transforms/lock/browserLock.test.ts +0 -129
  244. package/test/transforms/lock/countermeasures.test.ts +0 -100
  245. package/test/transforms/lock/integrity.test.ts +0 -161
  246. package/test/transforms/lock/lock.test.ts +0 -204
  247. package/test/transforms/lock/osLock.test.ts +0 -312
  248. package/test/transforms/lock/selfDefending.test.ts +0 -68
  249. package/test/transforms/lock/tamperProtection.test.ts +0 -336
  250. package/test/transforms/minify.test.ts +0 -575
  251. package/test/transforms/opaquePredicates.test.ts +0 -43
  252. package/test/transforms/preparation.test.ts +0 -157
  253. package/test/transforms/renameLabels.test.ts +0 -95
  254. package/test/transforms/rgf.test.ts +0 -378
  255. package/test/transforms/shuffle.test.ts +0 -135
  256. package/test/transforms/stack.test.ts +0 -573
  257. package/test/transforms/string/stringCompression.test.ts +0 -120
  258. package/test/transforms/string/stringConcealing.test.ts +0 -299
  259. package/test/transforms/string/stringEncoding.test.ts +0 -95
  260. package/test/transforms/string/stringSplitting.test.ts +0 -135
  261. package/test/transforms/transform.test.ts +0 -66
  262. package/test/traverse.test.ts +0 -139
  263. package/test/util/compare.test.ts +0 -34
  264. package/test/util/gen.test.ts +0 -121
  265. package/test/util/identifiers.test.ts +0 -253
  266. package/test/util/insert.test.ts +0 -142
  267. package/test/util/math.test.ts +0 -5
  268. package/test/util/random.test.ts +0 -71
  269. /package/dist/{types.js → obfuscationResult.js} +0 -0
@@ -1,424 +1,293 @@
1
- import { compileJsSync } from "../compiler";
2
- import { predictableFunctionTag, reservedIdentifiers } from "../constants";
3
- import Obfuscator from "../obfuscator";
4
- import { ObfuscateOrder } from "../order";
5
- import { ComputeProbabilityMap } from "../probability";
6
- import { FunctionLengthTemplate } from "../templates/functionLength";
7
- import { ObjectDefineProperty } from "../templates/globals";
8
- import Template from "../templates/template";
9
- import { walk } from "../traverse";
10
- import {
11
- ArrayExpression,
12
- BlockStatement,
13
- CallExpression,
14
- ExpressionStatement,
15
- Identifier,
16
- Literal,
17
- LogicalExpression,
18
- MemberExpression,
19
- NewExpression,
20
- Node,
21
- ReturnStatement,
22
- ThisExpression,
23
- VariableDeclaration,
24
- VariableDeclarator,
25
- } from "../util/gen";
26
- import { getIdentifierInfo } from "../util/identifiers";
27
- import {
28
- prepend,
29
- getDefiningContext,
30
- computeFunctionLength,
31
- } from "../util/insert";
32
- import Integrity from "./lock/integrity";
33
- import Transform from "./transform";
34
-
35
- /**
36
- * Converts function to `new Function("..code..")` syntax as an alternative to `eval`. Eval is disabled in many environments.
37
- *
38
- * `new Function("..code..")` runs in an isolated context, meaning all local variables are undefined and throw errors.
39
- *
40
- * Rigorous checks are in place to only include pure functions.
41
- *
42
- * `flatten` can attempt to make function reference-less. Recommended to have flatten enabled with RGF.
43
- */
44
- export default class RGF extends Transform {
45
- // Array of all the `new Function` calls
46
- arrayExpressionElements: Node[];
47
- // The name of the array holding all the `new Function` expressions
48
- arrayExpressionName: string;
49
-
50
- functionLengthName: string;
51
-
52
- getFunctionLengthName(parents: Node[]) {
53
- if (!this.functionLengthName) {
54
- this.functionLengthName = this.getPlaceholder();
55
- }
56
-
57
- return this.functionLengthName;
58
- }
59
-
60
- constructor(o) {
61
- super(o, ObfuscateOrder.RGF);
62
-
63
- this.arrayExpressionName = this.getPlaceholder() + "_rgf";
64
- this.arrayExpressionElements = [];
65
- }
66
-
67
- apply(tree: Node): void {
68
- super.apply(tree);
69
-
70
- // Only add the array if there were converted functions
71
- if (this.arrayExpressionElements.length > 0) {
72
- var variableDeclaration = VariableDeclaration(
73
- VariableDeclarator(
74
- Identifier(this.arrayExpressionName),
75
- ArrayExpression(this.arrayExpressionElements)
76
- )
77
- );
78
-
79
- var nodes: Node[] = [variableDeclaration];
80
-
81
- if (this.options.lock?.tamperProtection) {
82
- // The name of the variable flag if eval is safe to use
83
- var tamperProtectionCheckName = this.getPlaceholder() + "_rgfEvalCheck";
84
-
85
- variableDeclaration.declarations[0].init = LogicalExpression(
86
- "&&",
87
- Identifier(tamperProtectionCheckName),
88
- { ...variableDeclaration.declarations[0].init }
89
- );
90
-
91
- nodes.unshift(
92
- ...new Template(`
93
- var ${tamperProtectionCheckName} = false;
94
- eval(${this.jsConfuserVar(tamperProtectionCheckName)} + "=true");
95
- if(!${tamperProtectionCheckName}) {
96
- {countermeasures}
97
- }
98
- `).compile({
99
- countermeasures: this.lockTransform.getCounterMeasuresCode(
100
- tree,
101
- []
102
- ),
103
- })
104
- );
105
- }
106
-
107
- prepend(tree, ...nodes);
108
- }
109
-
110
- // The function.length helper function must be placed last
111
- if (this.functionLengthName) {
112
- prepend(
113
- tree,
114
- FunctionLengthTemplate.single({
115
- name: this.functionLengthName,
116
- ObjectDefineProperty: this.createInitVariable(ObjectDefineProperty, [
117
- tree,
118
- ]),
119
- })
120
- );
121
- }
122
- }
123
-
124
- match(object, parents) {
125
- return (
126
- (object.type === "FunctionDeclaration" ||
127
- object.type === "FunctionExpression") && // Does not apply to Arrow functions
128
- !object.async && // Does not apply to async/generator functions
129
- !object.generator
130
- );
131
- }
132
-
133
- transform(object: Node, parents: Node[]) {
134
- // Discard getter/setter methods
135
- if (parents[0].type === "Property" && parents[0].value === object) {
136
- if (
137
- parents[0].method ||
138
- parents[0].kind === "get" ||
139
- parents[0].kind === "set"
140
- ) {
141
- return;
142
- }
143
- }
144
-
145
- // Discard class methods
146
- if (parents[0].type === "MethodDefinition" && parents[0].value === object) {
147
- return;
148
- }
149
-
150
- // Avoid applying to the countermeasures function
151
- if (typeof this.options.lock?.countermeasures === "string") {
152
- // function countermeasures(){...}
153
- if (
154
- object.type === "FunctionDeclaration" &&
155
- object.id.type === "Identifier" &&
156
- object.id.name === this.options.lock.countermeasures
157
- ) {
158
- return;
159
- }
160
-
161
- // var countermeasures = function(){...}
162
- if (
163
- parents[0].type === "VariableDeclarator" &&
164
- parents[0].init === object &&
165
- parents[0].id.type === "Identifier" &&
166
- parents[0].id.name === this.options.lock.countermeasures
167
- ) {
168
- return;
169
- }
170
- }
171
-
172
- // Check user option
173
- if (!ComputeProbabilityMap(this.options.rgf, (x) => x, object?.id?.name))
174
- return;
175
-
176
- // Discard functions that use 'eval' function
177
- if (object.$requiresEval) return;
178
-
179
- // Check for 'this', 'arguments' (not allowed!)
180
- var isIllegal = false;
181
- walk(object, parents, (o, p) => {
182
- if (
183
- o.type === "ThisExpression" ||
184
- o.type === "Super" ||
185
- (o.type === "Identifier" && o.name === "arguments")
186
- ) {
187
- isIllegal = true;
188
- return "EXIT";
189
- }
190
- });
191
-
192
- if (isIllegal) return;
193
-
194
- return () => {
195
- // Make sure function is 'reference-less'
196
- var definedMap = new Map<Node, Set<string>>();
197
- var isReferenceLess = true;
198
- var identifierPreventingTransformation: string;
199
-
200
- walk(object, parents, (o, p) => {
201
- if (
202
- o.type === "Identifier" &&
203
- o.name !== this.arrayExpressionName &&
204
- !reservedIdentifiers.has(o.name) &&
205
- !this.options.globalVariables.has(o.name)
206
- ) {
207
- var info = getIdentifierInfo(o, p);
208
- if (!info.spec.isReferenced) {
209
- return;
210
- }
211
-
212
- if (info.spec.isDefined) {
213
- // Add to defined map
214
- var definingContext = getDefiningContext(o, p);
215
-
216
- if (!definedMap.has(definingContext)) {
217
- definedMap.set(definingContext, new Set([o.name]));
218
- } else {
219
- definedMap.get(definingContext).add(o.name);
220
- }
221
- } else {
222
- // This approach is dirty and does not account for hoisted FunctionDeclarations
223
- var isDefinedAbove = false;
224
- for (var pNode of p) {
225
- if (definedMap.has(pNode)) {
226
- if (definedMap.get(pNode).has(o.name)) {
227
- isDefinedAbove = true;
228
- break;
229
- }
230
- }
231
- }
232
-
233
- if (!isDefinedAbove) {
234
- isReferenceLess = false;
235
- identifierPreventingTransformation = o.name;
236
-
237
- return "EXIT";
238
- }
239
- }
240
- }
241
- });
242
-
243
- // This function is not 'reference-less', cannot be RGF'd
244
- if (!isReferenceLess) {
245
- if (object.id) {
246
- this.log(
247
- `${object?.id?.name}() cannot be transformed because of ${identifierPreventingTransformation}`
248
- );
249
- }
250
- return;
251
- }
252
-
253
- // Since `new Function` is completely isolated, create an entire new obfuscator and run remaining transformations.
254
- // RGF runs early and needs completed code before converting to a string.
255
- // (^ the variables haven't been renamed yet)
256
- var obfuscator = new Obfuscator({
257
- ...this.options,
258
- stringEncoding: false,
259
- compact: true,
260
- });
261
-
262
- if (obfuscator.options.lock) {
263
- obfuscator.options.lock = { ...obfuscator.options.lock };
264
- delete obfuscator.options.lock.countermeasures;
265
-
266
- // Integrity will not recursively apply to RGF'd functions. This is intended.
267
- var lockTransform = obfuscator.transforms["Lock"];
268
- if (lockTransform) {
269
- lockTransform.before = lockTransform.before.filter(
270
- (beforeTransform) => !(beforeTransform instanceof Integrity)
271
- );
272
- }
273
- }
274
-
275
- var transforms = obfuscator.array.filter(
276
- (x) => x.priority > this.priority
277
- );
278
-
279
- var embeddedFunctionName = this.getPlaceholder();
280
-
281
- var embeddedFunction = {
282
- type: "FunctionDeclaration",
283
- id: Identifier(embeddedFunctionName),
284
- body: BlockStatement([...object.body.body]),
285
- params: object.params,
286
- async: false,
287
- generator: false,
288
- };
289
-
290
- // The new program will look like this
291
- // new Function(`
292
- // var rgf_array = this[0]
293
- // function greet(message){
294
- // console.log(message)
295
- // }
296
- // return greet.apply(this[1], arguments)
297
- // `)
298
- //
299
- // And called like
300
- // f.apply([ rgf_array, this ], arguments)
301
- var tree = {
302
- type: "Program",
303
- body: [
304
- VariableDeclaration(
305
- VariableDeclarator(
306
- this.arrayExpressionName,
307
- MemberExpression(ThisExpression(), Literal(0))
308
- )
309
- ),
310
- embeddedFunction,
311
- ReturnStatement(
312
- CallExpression(
313
- MemberExpression(
314
- Identifier(embeddedFunctionName),
315
- Literal("apply"),
316
- true
317
- ),
318
- [
319
- MemberExpression(ThisExpression(), Literal(1)),
320
- Identifier("arguments"),
321
- ]
322
- )
323
- ),
324
- ],
325
- };
326
-
327
- transforms.forEach((transform) => {
328
- transform.apply(tree);
329
- });
330
-
331
- var toString = compileJsSync(tree, obfuscator.options);
332
-
333
- // new Function(code)
334
- var newFunctionExpression: Node = NewExpression(Identifier("Function"), [
335
- Literal(toString),
336
- ]);
337
-
338
- if (this.options.lock?.tamperProtection) {
339
- // If tamper protection is enabled, wrap the function in an eval
340
- var randomName = this.getGenerator("randomized").generate();
341
- newFunctionExpression = CallExpression(Identifier("eval"), [
342
- Literal(`function ${randomName}(){ ${toString} } ${randomName}`),
343
- ]);
344
- }
345
-
346
- // The index where this function is placed in the array
347
- var newFunctionExpressionIndex = this.arrayExpressionElements.length;
348
-
349
- // Add it to the array
350
- this.arrayExpressionElements.push(newFunctionExpression);
351
-
352
- // The member expression to retrieve this function
353
- var memberExpression: Node = MemberExpression(
354
- Identifier(this.arrayExpressionName),
355
- Literal(newFunctionExpressionIndex),
356
- true
357
- );
358
-
359
- var originalFunctionLength = computeFunctionLength(object.params);
360
-
361
- // Replace based on type
362
-
363
- // (1) Function Declaration:
364
- // - Replace body with call to new function
365
- if (object.type === "FunctionDeclaration") {
366
- object.body = BlockStatement([
367
- ReturnStatement(
368
- CallExpression(
369
- MemberExpression(memberExpression, Literal("apply"), true),
370
- [
371
- ArrayExpression([
372
- Identifier(this.arrayExpressionName),
373
- ThisExpression(),
374
- ]),
375
- Identifier("arguments"),
376
- ]
377
- )
378
- ),
379
- ]);
380
-
381
- // The parameters are no longer needed ('arguments' is used to capture them)
382
- object.params = [];
383
-
384
- // The function is no longer guaranteed to not have extraneous parameters passed in
385
- object[predictableFunctionTag] = false;
386
-
387
- if (
388
- this.options.preserveFunctionLength &&
389
- originalFunctionLength !== 0
390
- ) {
391
- var body = parents[0] as unknown as Node[];
392
-
393
- body.splice(
394
- body.indexOf(object),
395
- 0,
396
- ExpressionStatement(
397
- CallExpression(Identifier(this.getFunctionLengthName(parents)), [
398
- Identifier(object.id.name),
399
- Literal(originalFunctionLength),
400
- ])
401
- )
402
- );
403
- }
404
- return;
405
- }
406
-
407
- // (2) Function Expression:
408
- // - Replace expression with member expression pointing to new function
409
- if (object.type === "FunctionExpression") {
410
- if (
411
- this.options.preserveFunctionLength &&
412
- originalFunctionLength !== 0
413
- ) {
414
- memberExpression = CallExpression(
415
- Identifier(this.getFunctionLengthName(parents)),
416
- [memberExpression, Literal(originalFunctionLength)]
417
- );
418
- }
419
- this.replace(object, memberExpression);
420
- return;
421
- }
422
- };
423
- }
424
- }
1
+ import { NodePath } from "@babel/core";
2
+ import { PluginArg, PluginObject } from "./plugin";
3
+ import { Order } from "../order";
4
+ import * as t from "@babel/types";
5
+ import Obfuscator from "../obfuscator";
6
+ import { computeProbabilityMap } from "../probability";
7
+ import {
8
+ append,
9
+ getFunctionName,
10
+ isDefiningIdentifier,
11
+ isStrictMode,
12
+ isVariableIdentifier,
13
+ prepend,
14
+ } from "../utils/ast-utils";
15
+ import {
16
+ NodeSymbol,
17
+ PREDICTABLE,
18
+ reservedIdentifiers,
19
+ SKIP,
20
+ UNSAFE,
21
+ } from "../constants";
22
+ import { computeFunctionLength } from "../utils/function-utils";
23
+ import { numericLiteral } from "../utils/node";
24
+ import Template from "../templates/template";
25
+ import { createEvalIntegrityTemplate } from "../templates/tamperProtectionTemplates";
26
+
27
+ /**
28
+ * RGF (Runtime-Generated-Function) uses the `new Function("code")` syntax to create executable code from strings.
29
+ *
30
+ * Limitations:
31
+ *
32
+ * 1. Does not apply to async or generator functions
33
+ * 2. Does not apply to functions that reference outside variables
34
+ */
35
+ export default ({ Plugin }: PluginArg): PluginObject => {
36
+ const me = Plugin(Order.RGF, {
37
+ changeData: {
38
+ functions: 0,
39
+ },
40
+ });
41
+
42
+ const rgfArrayName = me.getPlaceholder() + "_rgf";
43
+ const rgfEvalName = me.getPlaceholder() + "_rgf_eval";
44
+ const rgfArrayExpression = t.arrayExpression([]);
45
+
46
+ let active = true;
47
+
48
+ return {
49
+ visitor: {
50
+ Program: {
51
+ enter(path) {
52
+ path.scope.crawl();
53
+ },
54
+ exit(path) {
55
+ active = false;
56
+ if (rgfArrayExpression.elements.length === 0) return;
57
+
58
+ // Insert the RGF array at the top of the program
59
+ prepend(
60
+ path,
61
+ t.variableDeclaration("var", [
62
+ t.variableDeclarator(
63
+ t.identifier(rgfArrayName),
64
+ rgfArrayExpression
65
+ ),
66
+ ])
67
+ );
68
+
69
+ var rgfEvalIntegrity = me.getPlaceholder() + "_rgf_eval_integrity";
70
+
71
+ prepend(
72
+ path,
73
+ new Template(`
74
+ {EvalIntegrity}
75
+ var ${rgfEvalIntegrity} = {EvalIntegrityName}();
76
+ `).compile({
77
+ EvalIntegrity: createEvalIntegrityTemplate(me, path),
78
+ EvalIntegrityName: me.getPlaceholder(),
79
+ })
80
+ );
81
+
82
+ append(
83
+ path,
84
+ new Template(
85
+ `
86
+ function ${rgfEvalName}(code) {
87
+ if (${rgfEvalIntegrity}) {
88
+ return eval(code);
89
+ }
90
+ }
91
+ `
92
+ )
93
+ .addSymbols(UNSAFE)
94
+ .single()
95
+ );
96
+ },
97
+ },
98
+ "FunctionDeclaration|FunctionExpression": {
99
+ exit(_path) {
100
+ if (!active) return;
101
+ const path = _path as NodePath<
102
+ t.FunctionDeclaration | t.FunctionExpression
103
+ >;
104
+
105
+ if (me.isSkipped(path)) return;
106
+
107
+ // Skip async and generator functions
108
+ if (path.node.async || path.node.generator) return;
109
+
110
+ const name = getFunctionName(path);
111
+ if (name === me.options.lock?.countermeasures) return;
112
+ if (me.obfuscator.isInternalVariable(name)) return;
113
+
114
+ me.log(name);
115
+
116
+ if (
117
+ !computeProbabilityMap(
118
+ me.options.rgf,
119
+ name,
120
+ path.getFunctionParent() === null
121
+ )
122
+ )
123
+ return;
124
+
125
+ // Skip functions with references to outside variables
126
+ // Check the scope to see if this function relies on any variables defined outside the function
127
+ var identifierPreventingTransform: string;
128
+
129
+ path.traverse({
130
+ Identifier(idPath) {
131
+ if (!isVariableIdentifier(idPath)) return;
132
+ if (idPath.isBindingIdentifier() && isDefiningIdentifier(idPath))
133
+ return;
134
+
135
+ const { name } = idPath.node;
136
+ // RGF array name is allowed, it is not considered an outside reference
137
+ if (name === rgfArrayName) return;
138
+ if (reservedIdentifiers.has(name)) return;
139
+ if (me.options.globalVariables.has(name)) return;
140
+
141
+ const binding = idPath.scope.getBinding(name);
142
+ if (!binding) {
143
+ identifierPreventingTransform = name;
144
+ idPath.stop();
145
+ return;
146
+ }
147
+
148
+ // If the binding is not in the current scope, it is an outside reference
149
+ if (binding.scope !== path.scope) {
150
+ identifierPreventingTransform = name;
151
+ idPath.stop();
152
+ }
153
+ },
154
+ });
155
+
156
+ if (identifierPreventingTransform) {
157
+ me.log(
158
+ "Skipping function " +
159
+ name +
160
+ " due to reference to outside variable: " +
161
+ identifierPreventingTransform
162
+ );
163
+ return;
164
+ }
165
+
166
+ const embeddedName = me.getPlaceholder() + "_embedded";
167
+ const replacementName = me.getPlaceholder() + "_replacement";
168
+ const thisName = me.getPlaceholder() + "_this";
169
+
170
+ const lastNode = t.expressionStatement(t.identifier(embeddedName));
171
+ (lastNode as NodeSymbol)[SKIP] = true;
172
+
173
+ // Transform the function
174
+ const evalProgram: t.Program = t.program([
175
+ t.functionDeclaration(
176
+ t.identifier(embeddedName),
177
+ [],
178
+ t.blockStatement([
179
+ t.variableDeclaration("var", [
180
+ t.variableDeclarator(
181
+ t.arrayPattern([
182
+ t.identifier(thisName),
183
+ t.identifier(rgfArrayName),
184
+ ]),
185
+ t.thisExpression()
186
+ ),
187
+ ]),
188
+ t.functionDeclaration(
189
+ t.identifier(replacementName),
190
+ path.node.params as (t.Identifier | t.Pattern)[],
191
+ path.node.body
192
+ ),
193
+ t.returnStatement(
194
+ t.callExpression(
195
+ t.memberExpression(
196
+ t.identifier(replacementName),
197
+ t.identifier("apply")
198
+ ),
199
+ [t.identifier(thisName), t.identifier("arguments")]
200
+ )
201
+ ),
202
+ ])
203
+ ),
204
+ lastNode,
205
+ ]);
206
+
207
+ const strictModeEnforcingBlock = path.find((p) => isStrictMode(p));
208
+ if (strictModeEnforcingBlock) {
209
+ // Preserve 'use strict' directive
210
+ // This is necessary to enure subsequent transforms (Control Flow Flattening) are aware of the strict mode directive
211
+ evalProgram.directives.push(
212
+ t.directive(t.directiveLiteral("use strict"))
213
+ );
214
+ }
215
+
216
+ const evalFile = t.file(evalProgram);
217
+
218
+ var newObfuscator = new Obfuscator(me.options, me.obfuscator);
219
+
220
+ var hasRan = new Set(
221
+ me.obfuscator.plugins
222
+ .filter((plugin, i) => {
223
+ return i <= me.obfuscator.index;
224
+ })
225
+ .map((plugin) => plugin.pluginInstance.order)
226
+ );
227
+
228
+ newObfuscator.plugins = newObfuscator.plugins.filter((plugin) => {
229
+ return (
230
+ plugin.pluginInstance.order == Order.Preparation ||
231
+ !hasRan.has(plugin.pluginInstance.order)
232
+ );
233
+ });
234
+
235
+ newObfuscator.obfuscateAST(evalFile, {
236
+ disablePack: true,
237
+ });
238
+
239
+ const generated = Obfuscator.generateCode(evalFile);
240
+
241
+ var functionExpression = t.callExpression(t.identifier(rgfEvalName), [
242
+ t.stringLiteral(generated),
243
+ ]);
244
+
245
+ var index = rgfArrayExpression.elements.length;
246
+ rgfArrayExpression.elements.push(functionExpression);
247
+
248
+ // Params no longer needed, using 'arguments' instead
249
+ const originalLength = computeFunctionLength(path);
250
+ path.node.params = [];
251
+
252
+ // Function is now unsafe
253
+ (path.node as NodeSymbol)[UNSAFE] = true;
254
+ // Params changed and using 'arguments'
255
+ (path.node as NodeSymbol)[PREDICTABLE] = false;
256
+ me.skip(path);
257
+
258
+ // Update body to point to new function
259
+ path
260
+ .get("body")
261
+ .replaceWith(
262
+ t.blockStatement([
263
+ t.returnStatement(
264
+ t.callExpression(
265
+ t.memberExpression(
266
+ t.memberExpression(
267
+ t.identifier(rgfArrayName),
268
+ numericLiteral(index),
269
+ true
270
+ ),
271
+ t.stringLiteral("apply"),
272
+ true
273
+ ),
274
+ [
275
+ t.arrayExpression([
276
+ t.thisExpression(),
277
+ t.identifier(rgfArrayName),
278
+ ]),
279
+ t.identifier("arguments"),
280
+ ]
281
+ )
282
+ ),
283
+ ])
284
+ );
285
+
286
+ me.setFunctionLength(path, originalLength);
287
+
288
+ me.changeData.functions++;
289
+ },
290
+ },
291
+ },
292
+ };
293
+ };