js-confuser 1.7.2 → 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 (263) hide show
  1. package/.github/ISSUE_TEMPLATE/bug_report.md +6 -4
  2. package/.github/workflows/node.js.yml +1 -1
  3. package/CHANGELOG.md +105 -0
  4. package/Migration.md +57 -0
  5. package/README.md +23 -913
  6. package/dist/constants.js +69 -13
  7. package/dist/index.js +108 -152
  8. package/dist/obfuscator.js +316 -118
  9. package/dist/options.js +1 -109
  10. package/dist/order.js +30 -30
  11. package/dist/presets.js +47 -45
  12. package/dist/probability.js +25 -32
  13. package/dist/templates/bufferToStringTemplate.js +9 -0
  14. package/dist/templates/deadCodeTemplates.js +9 -0
  15. package/dist/templates/getGlobalTemplate.js +19 -0
  16. package/dist/templates/integrityTemplate.js +30 -0
  17. package/dist/templates/setFunctionLengthTemplate.js +9 -0
  18. package/dist/templates/stringCompressionTemplate.js +10 -0
  19. package/dist/templates/tamperProtectionTemplates.js +21 -0
  20. package/dist/templates/template.js +213 -93
  21. package/dist/transforms/astScrambler.js +100 -0
  22. package/dist/transforms/calculator.js +70 -127
  23. package/dist/transforms/controlFlowFlattening.js +1182 -0
  24. package/dist/transforms/deadCode.js +62 -577
  25. package/dist/transforms/dispatcher.js +300 -309
  26. package/dist/transforms/extraction/duplicateLiteralsRemoval.js +88 -189
  27. package/dist/transforms/extraction/objectExtraction.js +131 -215
  28. package/dist/transforms/finalizer.js +56 -59
  29. package/dist/transforms/flatten.js +275 -276
  30. package/dist/transforms/functionOutlining.js +230 -0
  31. package/dist/transforms/identifier/globalConcealing.js +217 -103
  32. package/dist/transforms/identifier/movedDeclarations.js +167 -91
  33. package/dist/transforms/identifier/renameVariables.js +240 -187
  34. package/dist/transforms/lock/integrity.js +61 -184
  35. package/dist/transforms/lock/lock.js +263 -303
  36. package/dist/transforms/minify.js +431 -436
  37. package/dist/transforms/opaquePredicates.js +65 -118
  38. package/dist/transforms/pack.js +160 -0
  39. package/dist/transforms/plugin.js +179 -0
  40. package/dist/transforms/preparation.js +263 -163
  41. package/dist/transforms/renameLabels.js +132 -56
  42. package/dist/transforms/rgf.js +142 -240
  43. package/dist/transforms/shuffle.js +52 -145
  44. package/dist/transforms/string/encoding.js +45 -173
  45. package/dist/transforms/string/stringCompression.js +81 -126
  46. package/dist/transforms/string/stringConcealing.js +189 -224
  47. package/dist/transforms/string/stringEncoding.js +32 -40
  48. package/dist/transforms/string/stringSplitting.js +54 -55
  49. package/dist/transforms/variableMasking.js +232 -0
  50. package/dist/utils/ControlObject.js +125 -0
  51. package/dist/utils/IntGen.js +46 -0
  52. package/dist/utils/NameGen.js +106 -0
  53. package/dist/utils/ast-utils.js +560 -0
  54. package/dist/utils/function-utils.js +56 -0
  55. package/dist/utils/gen-utils.js +48 -0
  56. package/dist/utils/node.js +77 -0
  57. package/dist/utils/object-utils.js +21 -0
  58. package/dist/utils/random-utils.js +91 -0
  59. package/dist/utils/static-utils.js +64 -0
  60. package/dist/validateOptions.js +122 -0
  61. package/index.d.ts +1 -17
  62. package/package.json +27 -22
  63. package/src/constants.ts +139 -77
  64. package/src/index.ts +70 -163
  65. package/src/obfuscationResult.ts +43 -0
  66. package/src/obfuscator.ts +328 -135
  67. package/src/options.ts +154 -623
  68. package/src/order.ts +14 -14
  69. package/src/presets.ts +39 -34
  70. package/src/probability.ts +21 -36
  71. package/src/templates/{bufferToString.ts → bufferToStringTemplate.ts} +5 -54
  72. package/src/templates/deadCodeTemplates.ts +1185 -0
  73. package/src/templates/getGlobalTemplate.ts +72 -0
  74. package/src/templates/integrityTemplate.ts +69 -0
  75. package/src/templates/setFunctionLengthTemplate.ts +11 -0
  76. package/src/templates/stringCompressionTemplate.ts +42 -0
  77. package/src/templates/tamperProtectionTemplates.ts +116 -0
  78. package/src/templates/template.ts +183 -92
  79. package/src/transforms/astScrambler.ts +99 -0
  80. package/src/transforms/calculator.ts +96 -224
  81. package/src/transforms/controlFlowFlattening.ts +1594 -0
  82. package/src/transforms/deadCode.ts +85 -628
  83. package/src/transforms/dispatcher.ts +431 -636
  84. package/src/transforms/extraction/duplicateLiteralsRemoval.ts +147 -299
  85. package/src/transforms/extraction/objectExtraction.ts +160 -333
  86. package/src/transforms/finalizer.ts +63 -64
  87. package/src/transforms/flatten.ts +439 -557
  88. package/src/transforms/functionOutlining.ts +225 -0
  89. package/src/transforms/identifier/globalConcealing.ts +261 -189
  90. package/src/transforms/identifier/movedDeclarations.ts +228 -142
  91. package/src/transforms/identifier/renameVariables.ts +252 -258
  92. package/src/transforms/lock/integrity.ts +84 -260
  93. package/src/transforms/lock/lock.ts +342 -491
  94. package/src/transforms/minify.ts +523 -663
  95. package/src/transforms/opaquePredicates.ts +90 -229
  96. package/src/transforms/pack.ts +195 -0
  97. package/src/transforms/plugin.ts +185 -0
  98. package/src/transforms/preparation.ts +337 -215
  99. package/src/transforms/renameLabels.ts +176 -77
  100. package/src/transforms/rgf.ts +293 -386
  101. package/src/transforms/shuffle.ts +80 -254
  102. package/src/transforms/string/encoding.ts +26 -129
  103. package/src/transforms/string/stringCompression.ts +118 -236
  104. package/src/transforms/string/stringConcealing.ts +255 -339
  105. package/src/transforms/string/stringEncoding.ts +28 -47
  106. package/src/transforms/string/stringSplitting.ts +61 -75
  107. package/src/transforms/variableMasking.ts +257 -0
  108. package/src/utils/ControlObject.ts +141 -0
  109. package/src/utils/IntGen.ts +33 -0
  110. package/src/utils/NameGen.ts +106 -0
  111. package/src/utils/ast-utils.ts +667 -0
  112. package/src/utils/function-utils.ts +50 -0
  113. package/src/utils/gen-utils.ts +48 -0
  114. package/src/utils/node.ts +78 -0
  115. package/src/utils/object-utils.ts +21 -0
  116. package/src/utils/random-utils.ts +79 -0
  117. package/src/utils/static-utils.ts +66 -0
  118. package/src/validateOptions.ts +256 -0
  119. package/tsconfig.json +13 -8
  120. package/babel.config.js +0 -12
  121. package/dev.js +0 -8
  122. package/dist/compiler.js +0 -34
  123. package/dist/parser.js +0 -59
  124. package/dist/precedence.js +0 -66
  125. package/dist/templates/bufferToString.js +0 -108
  126. package/dist/templates/crash.js +0 -59
  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 -1281
  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 -70
  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 -343
  144. package/dist/transforms/transform.js +0 -350
  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 -9
  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 -130
  154. package/dist/util/scope.js +0 -20
  155. package/docs/ControlFlowFlattening.md +0 -595
  156. package/docs/Countermeasures.md +0 -63
  157. package/docs/ES5.md +0 -197
  158. package/docs/Integrity.md +0 -75
  159. package/docs/RGF.md +0 -419
  160. package/samples/example.js +0 -15
  161. package/samples/high.js +0 -1
  162. package/samples/input.js +0 -3
  163. package/samples/javascriptobfuscator.com.js +0 -8
  164. package/samples/jscrambler_advanced.js +0 -1894
  165. package/samples/jscrambler_light.js +0 -1134
  166. package/samples/low.js +0 -1
  167. package/samples/medium.js +0 -1
  168. package/samples/obfuscator.io.js +0 -1686
  169. package/samples/preemptive.com.js +0 -16
  170. package/src/compiler.ts +0 -35
  171. package/src/parser.ts +0 -49
  172. package/src/precedence.ts +0 -61
  173. package/src/templates/crash.ts +0 -55
  174. package/src/templates/es5.ts +0 -131
  175. package/src/templates/functionLength.ts +0 -32
  176. package/src/templates/globals.ts +0 -3
  177. package/src/transforms/antiTooling.ts +0 -102
  178. package/src/transforms/controlFlowFlattening/controlFlowFlattening.ts +0 -2146
  179. package/src/transforms/controlFlowFlattening/expressionObfuscation.ts +0 -179
  180. package/src/transforms/es5/antiClass.ts +0 -272
  181. package/src/transforms/es5/antiDestructuring.ts +0 -294
  182. package/src/transforms/es5/antiES6Object.ts +0 -267
  183. package/src/transforms/es5/antiSpreadOperator.ts +0 -56
  184. package/src/transforms/es5/antiTemplate.ts +0 -98
  185. package/src/transforms/es5/es5.ts +0 -149
  186. package/src/transforms/extraction/classExtraction.ts +0 -168
  187. package/src/transforms/identifier/globalAnalysis.ts +0 -85
  188. package/src/transforms/identifier/variableAnalysis.ts +0 -118
  189. package/src/transforms/lock/antiDebug.ts +0 -112
  190. package/src/transforms/stack.ts +0 -551
  191. package/src/transforms/transform.ts +0 -453
  192. package/src/traverse.ts +0 -120
  193. package/src/types.ts +0 -131
  194. package/src/util/compare.ts +0 -181
  195. package/src/util/gen.ts +0 -651
  196. package/src/util/guard.ts +0 -7
  197. package/src/util/identifiers.ts +0 -494
  198. package/src/util/insert.ts +0 -419
  199. package/src/util/math.ts +0 -15
  200. package/src/util/object.ts +0 -39
  201. package/src/util/random.ts +0 -141
  202. package/src/util/scope.ts +0 -21
  203. package/test/code/Cash.src.js +0 -1011
  204. package/test/code/Cash.test.ts +0 -49
  205. package/test/code/Dynamic.src.js +0 -118
  206. package/test/code/Dynamic.test.ts +0 -49
  207. package/test/code/ES6.src.js +0 -235
  208. package/test/code/ES6.test.ts +0 -42
  209. package/test/code/NewFeatures.test.ts +0 -19
  210. package/test/code/StrictMode.src.js +0 -65
  211. package/test/code/StrictMode.test.js +0 -37
  212. package/test/compare.test.ts +0 -104
  213. package/test/index.test.ts +0 -249
  214. package/test/options.test.ts +0 -132
  215. package/test/presets.test.ts +0 -22
  216. package/test/probability.test.ts +0 -44
  217. package/test/templates/template.test.ts +0 -14
  218. package/test/transforms/antiTooling.test.ts +0 -52
  219. package/test/transforms/calculator.test.ts +0 -78
  220. package/test/transforms/controlFlowFlattening/controlFlowFlattening.test.ts +0 -1274
  221. package/test/transforms/controlFlowFlattening/expressionObfuscation.test.ts +0 -192
  222. package/test/transforms/deadCode.test.ts +0 -85
  223. package/test/transforms/dispatcher.test.ts +0 -457
  224. package/test/transforms/es5/antiClass.test.ts +0 -427
  225. package/test/transforms/es5/antiDestructuring.test.ts +0 -157
  226. package/test/transforms/es5/antiES6Object.test.ts +0 -245
  227. package/test/transforms/es5/antiTemplate.test.ts +0 -116
  228. package/test/transforms/es5/es5.test.ts +0 -110
  229. package/test/transforms/extraction/classExtraction.test.ts +0 -86
  230. package/test/transforms/extraction/duplicateLiteralsRemoval.test.ts +0 -200
  231. package/test/transforms/extraction/objectExtraction.test.ts +0 -491
  232. package/test/transforms/flatten.test.ts +0 -721
  233. package/test/transforms/hexadecimalNumbers.test.ts +0 -62
  234. package/test/transforms/identifier/globalConcealing.test.ts +0 -72
  235. package/test/transforms/identifier/movedDeclarations.test.ts +0 -275
  236. package/test/transforms/identifier/renameVariables.test.ts +0 -621
  237. package/test/transforms/lock/antiDebug.test.ts +0 -66
  238. package/test/transforms/lock/browserLock.test.ts +0 -129
  239. package/test/transforms/lock/countermeasures.test.ts +0 -100
  240. package/test/transforms/lock/integrity.test.ts +0 -161
  241. package/test/transforms/lock/lock.test.ts +0 -204
  242. package/test/transforms/lock/osLock.test.ts +0 -312
  243. package/test/transforms/lock/selfDefending.test.ts +0 -68
  244. package/test/transforms/minify.test.ts +0 -575
  245. package/test/transforms/opaquePredicates.test.ts +0 -43
  246. package/test/transforms/preparation.test.ts +0 -157
  247. package/test/transforms/renameLabels.test.ts +0 -95
  248. package/test/transforms/rgf.test.ts +0 -378
  249. package/test/transforms/shuffle.test.ts +0 -135
  250. package/test/transforms/stack.test.ts +0 -573
  251. package/test/transforms/string/stringCompression.test.ts +0 -120
  252. package/test/transforms/string/stringConcealing.test.ts +0 -299
  253. package/test/transforms/string/stringEncoding.test.ts +0 -95
  254. package/test/transforms/string/stringSplitting.test.ts +0 -135
  255. package/test/transforms/transform.test.ts +0 -66
  256. package/test/traverse.test.ts +0 -139
  257. package/test/util/compare.test.ts +0 -34
  258. package/test/util/gen.test.ts +0 -121
  259. package/test/util/identifiers.test.ts +0 -253
  260. package/test/util/insert.test.ts +0 -142
  261. package/test/util/math.test.ts +0 -5
  262. package/test/util/random.test.ts +0 -71
  263. /package/dist/{types.js → obfuscationResult.js} +0 -0
@@ -1,636 +1,431 @@
1
- import { walk } from "../traverse";
2
- import {
3
- ArrayExpression,
4
- AssignmentExpression,
5
- BinaryExpression,
6
- CallExpression,
7
- ExpressionStatement,
8
- FunctionDeclaration,
9
- FunctionExpression,
10
- Identifier,
11
- IfStatement,
12
- Literal,
13
- Node,
14
- Location,
15
- MemberExpression,
16
- ObjectExpression,
17
- Property,
18
- ReturnStatement,
19
- VariableDeclaration,
20
- SequenceExpression,
21
- NewExpression,
22
- UnaryExpression,
23
- BlockStatement,
24
- LogicalExpression,
25
- ThisExpression,
26
- VariableDeclarator,
27
- RestElement,
28
- } from "../util/gen";
29
- import { getIdentifierInfo } from "../util/identifiers";
30
- import {
31
- deleteDirect,
32
- getBlockBody,
33
- getVarContext,
34
- isVarContext,
35
- prepend,
36
- append,
37
- computeFunctionLength,
38
- isFunction,
39
- } from "../util/insert";
40
- import Transform from "./transform";
41
- import { isInsideType } from "../util/compare";
42
- import { choice, shuffle } from "../util/random";
43
- import { ComputeProbabilityMap } from "../probability";
44
- import { predictableFunctionTag, reservedIdentifiers } from "../constants";
45
- import { ObfuscateOrder } from "../order";
46
- import Template from "../templates/template";
47
- import { FunctionLengthTemplate } from "../templates/functionLength";
48
- import { ObjectDefineProperty } from "../templates/globals";
49
- import { getLexicalScope } from "../util/scope";
50
-
51
- /**
52
- * A Dispatcher processes function calls. All the function declarations are brought into a dictionary.
53
- *
54
- * ```js
55
- * var param1;
56
- * function dispatcher(key){
57
- * var fns = {
58
- * 'fn1': function(){
59
- * var [arg1] = [param1];
60
- * console.log(arg1);
61
- * }
62
- * }
63
- * return fns[key]();
64
- * };
65
- * param1 = "Hello World";
66
- * dispatcher('fn1'); // > "Hello World"
67
- * ```
68
- *
69
- * Can break code with:
70
- *
71
- * 1. testing function equality,
72
- * 2. using `arguments.callee`,
73
- * 3. using `this`
74
- */
75
- export default class Dispatcher extends Transform {
76
- // Debug mode preserves function names
77
- isDebug = false;
78
- count: number;
79
-
80
- functionLengthName: string;
81
-
82
- constructor(o) {
83
- super(o, ObfuscateOrder.Dispatcher);
84
-
85
- this.count = 0;
86
- }
87
-
88
- apply(tree: Node): void {
89
- super.apply(tree);
90
-
91
- if (this.options.preserveFunctionLength && this.functionLengthName) {
92
- prepend(
93
- tree,
94
- FunctionLengthTemplate.single({
95
- name: this.functionLengthName,
96
- ObjectDefineProperty: this.createInitVariable(ObjectDefineProperty, [
97
- tree,
98
- ]),
99
- })
100
- );
101
- }
102
- }
103
-
104
- match(object: Node, parents: Node[]) {
105
- if (isInsideType("AwaitExpression", object, parents)) {
106
- return false;
107
- }
108
-
109
- return (
110
- isVarContext(object) &&
111
- object.type !== "ArrowFunctionExpression" &&
112
- !object.$dispatcherSkip &&
113
- !parents.find((x) => x.$dispatcherSkip)
114
- );
115
- }
116
-
117
- transform(object: Node, parents: Node[]) {
118
- return () => {
119
- if (ComputeProbabilityMap(this.options.dispatcher, (mode) => mode)) {
120
- if (object.type != "Program" && object.body.type != "BlockStatement") {
121
- return;
122
- }
123
-
124
- // Map of FunctionDeclarations
125
- var functionDeclarations: { [name: string]: Location } =
126
- Object.create(null);
127
-
128
- // Array of Identifier nodes
129
- var identifiers: Location[] = [];
130
- var illegalFnNames: Set<string> = new Set();
131
-
132
- // New Names for Functions
133
- var newFnNames: { [name: string]: string } = Object.create(null); // [old name]: randomized name
134
-
135
- var context = isVarContext(object)
136
- ? object
137
- : getVarContext(object, parents);
138
-
139
- var lexicalScope = isFunction(context) ? context.body : context;
140
-
141
- walk(object, parents, (o: Node, p: Node[]) => {
142
- if (object == o) {
143
- // Fix 1
144
- return;
145
- }
146
-
147
- var c = getVarContext(o, p);
148
- if (o.type == "FunctionDeclaration") {
149
- c = getVarContext(p[0], p.slice(1));
150
- }
151
-
152
- if (context === c) {
153
- if (o.type == "FunctionDeclaration" && o.id.name) {
154
- var name = o.id.name;
155
-
156
- if (
157
- o.$requiresEval ||
158
- o.async ||
159
- o.generator ||
160
- p.find(
161
- (x) => x.$dispatcherSkip || x.type == "MethodDefinition"
162
- ) ||
163
- o.body.type != "BlockStatement"
164
- ) {
165
- illegalFnNames.add(name);
166
- return;
167
- }
168
-
169
- // Must defined in the same block as the current function being scanned
170
- // Solves 'let' and 'class' declaration issue
171
- var ls = getLexicalScope(o, p);
172
- if (ls !== lexicalScope) {
173
- illegalFnNames.add(name);
174
- return;
175
- }
176
-
177
- // If dupe, no routing
178
- if (functionDeclarations[name]) {
179
- illegalFnNames.add(name);
180
- return;
181
- }
182
-
183
- walk(o, p, (oo, pp) => {
184
- if (
185
- (oo.type == "Identifier" && oo.name == "arguments") ||
186
- oo.type == "ThisExpression" ||
187
- oo.type == "Super"
188
- ) {
189
- if (getVarContext(oo, pp) === o) {
190
- illegalFnNames.add(name);
191
- return "EXIT";
192
- }
193
- }
194
-
195
- // Avoid functions with function expressions as they have a different scope
196
- if (
197
- (oo.type === "FunctionExpression" ||
198
- oo.type === "ArrowFunctionExpression") &&
199
- pp.find((x) => x == o.params)
200
- ) {
201
- illegalFnNames.add(name);
202
- return "EXIT";
203
- }
204
- });
205
-
206
- functionDeclarations[name] = [o, p];
207
- }
208
- }
209
-
210
- if (o.type == "Identifier") {
211
- if (reservedIdentifiers.has(o.name)) {
212
- return;
213
- }
214
- var info = getIdentifierInfo(o, p);
215
- if (!info.spec.isReferenced) {
216
- return;
217
- }
218
- if (info.spec.isDefined) {
219
- if (info.isFunctionDeclaration) {
220
- if (
221
- p[0].id &&
222
- (!functionDeclarations[p[0].id.name] ||
223
- functionDeclarations[p[0].id.name][0] !== p[0])
224
- ) {
225
- illegalFnNames.add(o.name);
226
- }
227
- } else {
228
- illegalFnNames.add(o.name);
229
- }
230
- } else if (info.spec.isModified) {
231
- illegalFnNames.add(o.name);
232
- } else {
233
- identifiers.push([o, p]);
234
- }
235
- }
236
- });
237
-
238
- illegalFnNames.forEach((name) => {
239
- delete functionDeclarations[name];
240
- });
241
-
242
- // map original name->new game
243
- var gen = this.getGenerator();
244
- Object.keys(functionDeclarations).forEach((name) => {
245
- newFnNames[name] = this.isDebug
246
- ? "_dispatcher_" + this.count + "_" + name
247
- : gen.generate();
248
- });
249
- // set containing new name
250
- var set = new Set(Object.keys(newFnNames));
251
-
252
- // Only make a dispatcher function if it caught any functions
253
- if (set.size > 0) {
254
- if (!this.functionLengthName) {
255
- this.functionLengthName = this.getPlaceholder();
256
- }
257
-
258
- var payloadArg =
259
- this.getPlaceholder() + "_dispatcher_" + this.count + "_payload";
260
-
261
- var dispatcherFnName =
262
- this.getPlaceholder() +
263
- "_dispatcher_" +
264
- this.count +
265
- predictableFunctionTag;
266
-
267
- this.log(dispatcherFnName, set);
268
- this.count++;
269
-
270
- var expectedGet = gen.generate();
271
- var expectedClearArgs = gen.generate();
272
- var expectedNew = gen.generate();
273
-
274
- var returnProp = gen.generate();
275
- var newReturnMemberName = gen.generate();
276
-
277
- var shuffledKeys = shuffle(Object.keys(functionDeclarations));
278
- var mapName = this.getPlaceholder();
279
-
280
- var cacheName = this.getPlaceholder();
281
-
282
- // creating the dispatcher function
283
- // 1. create function map
284
- var map = VariableDeclaration(
285
- VariableDeclarator(
286
- mapName,
287
- ObjectExpression(
288
- shuffledKeys.map((name) => {
289
- var [def, defParents] = functionDeclarations[name];
290
- var body = getBlockBody(def.body);
291
-
292
- var functionExpression: Node = {
293
- ...def,
294
- expression: false,
295
- type: "FunctionExpression",
296
- id: null,
297
- params: [],
298
- [predictableFunctionTag]: true,
299
- };
300
- this.addComment(functionExpression, name);
301
-
302
- if (def.params.length > 0) {
303
- const fixParam = (param: Node) => {
304
- return param;
305
- };
306
-
307
- var variableDeclaration = VariableDeclaration(
308
- VariableDeclarator(
309
- {
310
- type: "ArrayPattern",
311
- elements: def.params.map(fixParam),
312
- },
313
- Identifier(payloadArg)
314
- )
315
- );
316
-
317
- prepend(def.body, variableDeclaration);
318
- }
319
-
320
- // For logging purposes
321
- var signature =
322
- name +
323
- "(" +
324
- def.params.map((x) => x.name || "<>").join(",") +
325
- ")";
326
- this.log("Added", signature);
327
-
328
- // delete ref in block
329
- if (defParents.length) {
330
- deleteDirect(def, defParents[0]);
331
- }
332
-
333
- this.addComment(functionExpression, signature);
334
- return Property(
335
- Literal(newFnNames[name]),
336
- functionExpression,
337
- false
338
- );
339
- })
340
- )
341
- )
342
- );
343
-
344
- var getterArgName = this.getPlaceholder();
345
-
346
- var x = this.getPlaceholder();
347
- var y = this.getPlaceholder();
348
- var z = this.getPlaceholder();
349
-
350
- function getAccessor() {
351
- return MemberExpression(Identifier(mapName), Identifier(x), true);
352
- }
353
-
354
- // 2. define it
355
- var fn = FunctionDeclaration(
356
- dispatcherFnName,
357
- [Identifier(x), Identifier(y), Identifier(z)],
358
- [
359
- // Define map of callable functions
360
- map,
361
-
362
- // Set returning variable to undefined
363
- VariableDeclaration(VariableDeclarator(returnProp)),
364
-
365
- // Arg to clear the payload
366
- IfStatement(
367
- BinaryExpression(
368
- "==",
369
- Identifier(y),
370
- Literal(expectedClearArgs)
371
- ),
372
- [
373
- ExpressionStatement(
374
- AssignmentExpression(
375
- "=",
376
- Identifier(payloadArg),
377
- ArrayExpression([])
378
- )
379
- ),
380
- ],
381
- null
382
- ),
383
-
384
- VariableDeclaration(
385
- VariableDeclarator(
386
- Identifier("lengths"),
387
- ObjectExpression(
388
- !this.options.preserveFunctionLength
389
- ? []
390
- : shuffledKeys
391
- .map((name) => {
392
- var [def, defParents] = functionDeclarations[name];
393
-
394
- return {
395
- key: newFnNames[name],
396
- value: computeFunctionLength(def.params),
397
- };
398
- })
399
- .filter((item) => item.value !== 0)
400
- .map((item) =>
401
- Property(Literal(item.key), Literal(item.value))
402
- )
403
- )
404
- )
405
- ),
406
-
407
- Template(`
408
- function makeFn${predictableFunctionTag}(){
409
- var fn = function(...args){
410
- ${payloadArg} = args;
411
- return ${mapName}[${x}].call(this)
412
- }, a = lengths[${x}]
413
-
414
- ${
415
- this.options.preserveFunctionLength
416
- ? `if(a){
417
- return ${this.functionLengthName}(fn, a)
418
- }`
419
- : ""
420
- }
421
-
422
- return fn
423
- }
424
- `).single(),
425
-
426
- // Arg to get a function reference
427
- IfStatement(
428
- BinaryExpression("==", Identifier(y), Literal(expectedGet)),
429
- [
430
- // Getter flag: return the function object
431
- ExpressionStatement(
432
- AssignmentExpression(
433
- "=",
434
- Identifier(returnProp),
435
- LogicalExpression(
436
- "||",
437
- MemberExpression(
438
- Identifier(cacheName),
439
- Identifier(x),
440
- true
441
- ),
442
- AssignmentExpression(
443
- "=",
444
- MemberExpression(
445
- Identifier(cacheName),
446
- Identifier(x),
447
- true
448
- ),
449
- CallExpression(
450
- Identifier(`makeFn${predictableFunctionTag}`),
451
- []
452
- )
453
- )
454
- )
455
- )
456
- ),
457
- ],
458
- [
459
- // Call the function, return result
460
- ExpressionStatement(
461
- AssignmentExpression(
462
- "=",
463
- Identifier(returnProp),
464
- CallExpression(getAccessor(), [])
465
- )
466
- ),
467
- ]
468
- ),
469
-
470
- // Check how the function was invoked (new () vs ())
471
- IfStatement(
472
- BinaryExpression("==", Identifier(z), Literal(expectedNew)),
473
- [
474
- // Wrap in object
475
- ReturnStatement(
476
- ObjectExpression([
477
- Property(
478
- Identifier(newReturnMemberName),
479
- Identifier(returnProp),
480
- false
481
- ),
482
- ])
483
- ),
484
- ],
485
- [
486
- // Return raw result
487
- ReturnStatement(Identifier(returnProp)),
488
- ]
489
- ),
490
- ]
491
- );
492
-
493
- append(object, fn);
494
-
495
- if (payloadArg) {
496
- prepend(
497
- object,
498
- VariableDeclaration(
499
- VariableDeclarator(payloadArg, ArrayExpression([]))
500
- )
501
- );
502
- }
503
-
504
- identifiers.forEach(([o, p]) => {
505
- if (o.type != "Identifier") {
506
- return;
507
- }
508
-
509
- var newName = newFnNames[o.name];
510
- if (!newName || typeof newName !== "string") {
511
- return;
512
- }
513
-
514
- if (!functionDeclarations[o.name]) {
515
- this.error(new Error("newName, missing function declaration"));
516
- }
517
-
518
- var info = getIdentifierInfo(o, p);
519
- if (
520
- info.isFunctionCall &&
521
- p[0].type == "CallExpression" &&
522
- p[0].callee === o
523
- ) {
524
- // Invoking call expression: `a();`
525
-
526
- if (o.name == dispatcherFnName) {
527
- return;
528
- }
529
-
530
- this.log(
531
- `${o.name}(${p[0].arguments
532
- .map((_) => "<>")
533
- .join(",")}) -> ${dispatcherFnName}('${newName}')`
534
- );
535
-
536
- var assignmentExpressions: Node[] = [];
537
- var dispatcherArgs: Node[] = [Literal(newName)];
538
-
539
- if (p[0].arguments.length) {
540
- assignmentExpressions = [
541
- AssignmentExpression(
542
- "=",
543
- Identifier(payloadArg),
544
- ArrayExpression(p[0].arguments)
545
- ),
546
- ];
547
- } else {
548
- dispatcherArgs.push(Literal(expectedClearArgs));
549
- }
550
-
551
- var type = choice(["CallExpression", "NewExpression"]);
552
- var callExpression = null;
553
-
554
- switch (type) {
555
- case "CallExpression":
556
- callExpression = CallExpression(
557
- Identifier(dispatcherFnName),
558
- dispatcherArgs
559
- );
560
- break;
561
-
562
- case "NewExpression":
563
- if (dispatcherArgs.length == 1) {
564
- dispatcherArgs.push(Identifier("undefined"));
565
- }
566
- callExpression = MemberExpression(
567
- NewExpression(Identifier(dispatcherFnName), [
568
- ...dispatcherArgs,
569
- Literal(expectedNew),
570
- ]),
571
- Identifier(newReturnMemberName),
572
- false
573
- );
574
- break;
575
- }
576
-
577
- this.addComment(
578
- callExpression,
579
- "Calling " +
580
- o.name +
581
- "(" +
582
- p[0].arguments.map((x) => x.name).join(", ") +
583
- ")"
584
- );
585
-
586
- var expr: Node = assignmentExpressions.length
587
- ? SequenceExpression([...assignmentExpressions, callExpression])
588
- : callExpression;
589
-
590
- // Replace the parent call expression
591
- this.replace(p[0], expr);
592
- } else {
593
- // Non-invoking reference: `a`
594
-
595
- if (info.spec.isDefined) {
596
- if (info.isFunctionDeclaration) {
597
- this.log(
598
- "Skipped getter " + o.name + " (function declaration)"
599
- );
600
- } else {
601
- this.log("Skipped getter " + o.name + " (defined)");
602
- }
603
- return;
604
- }
605
- if (info.spec.isModified) {
606
- this.log("Skipped getter " + o.name + " (modified)");
607
- return;
608
- }
609
-
610
- this.log(
611
- `(getter) ${o.name} -> ${dispatcherFnName}('${newName}')`
612
- );
613
- this.replace(
614
- o,
615
- CallExpression(Identifier(dispatcherFnName), [
616
- Literal(newName),
617
- Literal(expectedGet),
618
- ])
619
- );
620
- }
621
- });
622
-
623
- prepend(
624
- object,
625
- VariableDeclaration(
626
- VariableDeclarator(
627
- Identifier(cacheName),
628
- Template(`Object.create(null)`).single().expression
629
- )
630
- )
631
- );
632
- }
633
- }
634
- };
635
- }
636
- }
1
+ import { PluginArg, PluginObject } from "./plugin";
2
+ import { NodePath } from "@babel/traverse";
3
+ import * as t from "@babel/types";
4
+ import Template from "../templates/template";
5
+ import { ok } from "assert";
6
+ import { chance, getRandomString } from "../utils/random-utils";
7
+ import { computeProbabilityMap } from "../probability";
8
+ import { Order } from "../order";
9
+ import { NodeSymbol, PREDICTABLE, UNSAFE } from "../constants";
10
+ import {
11
+ computeFunctionLength,
12
+ isVariableFunctionIdentifier,
13
+ } from "../utils/function-utils";
14
+ import { SetFunctionLengthTemplate } from "../templates/setFunctionLengthTemplate";
15
+ import { numericLiteral } from "../utils/node";
16
+ import {
17
+ isStrictMode,
18
+ isVariableIdentifier,
19
+ prependProgram,
20
+ } from "../utils/ast-utils";
21
+
22
+ export default ({ Plugin }: PluginArg): PluginObject => {
23
+ const me = Plugin(Order.Dispatcher, {
24
+ changeData: {
25
+ functions: 0,
26
+ },
27
+ });
28
+ let dispatcherCounter = 0;
29
+
30
+ const setFunctionLength = me.getPlaceholder("d_fnLength");
31
+
32
+ // in Debug mode, function names are preserved
33
+ const isDebug = false;
34
+
35
+ return {
36
+ visitor: {
37
+ "Program|Function": {
38
+ exit(_path) {
39
+ const blockPath = _path as NodePath<t.Program | t.Function>;
40
+
41
+ if (blockPath.isProgram()) {
42
+ blockPath.scope.crawl();
43
+
44
+ // Don't insert function length code when disabled
45
+ // Instead insert empty function as the identifier is still referenced
46
+ var insertNode = t.functionDeclaration(
47
+ t.identifier(setFunctionLength),
48
+ [],
49
+ t.blockStatement([])
50
+ );
51
+
52
+ if (me.options.preserveFunctionLength) {
53
+ // Insert function length code
54
+ insertNode = SetFunctionLengthTemplate.single({
55
+ fnName: setFunctionLength,
56
+ });
57
+ }
58
+
59
+ me.skip(prependProgram(_path, insertNode));
60
+ }
61
+
62
+ if ((blockPath.node as NodeSymbol)[UNSAFE]) return;
63
+
64
+ // For testing
65
+ // if (!blockPath.isProgram()) return;
66
+
67
+ var blockStatement: NodePath<t.Block> = blockPath.isProgram()
68
+ ? blockPath
69
+ : (blockPath.get("body") as NodePath<t.BlockStatement>);
70
+
71
+ // Track functions and illegal ones
72
+ // A function is illegal if:
73
+ // - the function is async or generator
74
+ // - the function is redefined
75
+ // - the function uses 'this', 'eval', or 'arguments'
76
+ var functionPaths = new Map<
77
+ string,
78
+ NodePath<t.FunctionDeclaration>
79
+ >();
80
+ var illegalNames = new Set<string>();
81
+
82
+ // Scan for function declarations
83
+ blockPath.traverse({
84
+ // Check for reassigned / redefined functions
85
+ BindingIdentifier: {
86
+ exit(path: NodePath<t.Identifier>) {
87
+ if (!isVariableIdentifier(path)) return;
88
+
89
+ const name = path.node.name;
90
+ if (!path.parentPath?.isFunctionDeclaration()) {
91
+ illegalNames.add(name);
92
+ }
93
+ },
94
+ },
95
+ // Find functions eligible for dispatching
96
+ FunctionDeclaration: {
97
+ exit(path: NodePath<t.FunctionDeclaration>) {
98
+ const name = path.node.id.name;
99
+ // If the function is not named, we can't dispatch it
100
+ if (!name) {
101
+ return;
102
+ }
103
+
104
+ // Do not apply to async or generator functions
105
+ if (path.node.async || path.node.generator) {
106
+ return;
107
+ }
108
+
109
+ // Do not apply to functions in nested scopes
110
+ if (path.parentPath !== blockStatement || me.isSkipped(path)) {
111
+ illegalNames.add(name);
112
+ return;
113
+ }
114
+
115
+ if (isStrictMode(path)) {
116
+ illegalNames.add(name);
117
+ return;
118
+ }
119
+
120
+ // Do not apply to unsafe functions, redefined functions, or internal obfuscator functions
121
+ if (
122
+ (path.node as NodeSymbol)[UNSAFE] ||
123
+ functionPaths.has(name) ||
124
+ me.obfuscator.isInternalVariable(name)
125
+ ) {
126
+ illegalNames.add(name);
127
+ return;
128
+ }
129
+
130
+ // Functions with default parameters are not fully supported
131
+ // (Could be a Function Expression referencing outer scope)
132
+ if (
133
+ path.node.params.find((x) => x.type === "AssignmentPattern")
134
+ ) {
135
+ illegalNames.add(name);
136
+ return;
137
+ }
138
+
139
+ functionPaths.set(name, path);
140
+ },
141
+ },
142
+ });
143
+
144
+ for (let name of illegalNames) {
145
+ functionPaths.delete(name);
146
+ }
147
+
148
+ for (var name of functionPaths.keys()) {
149
+ if (!computeProbabilityMap(me.options.dispatcher, name)) {
150
+ functionPaths.delete(name);
151
+ }
152
+ }
153
+
154
+ // No functions here to change
155
+ if (functionPaths.size === 0) {
156
+ return;
157
+ }
158
+
159
+ me.changeData.functions += functionPaths.size;
160
+
161
+ const dispatcherName =
162
+ me.getPlaceholder() + "_dispatcher_" + dispatcherCounter++;
163
+ const payloadName = me.getPlaceholder() + "_payload";
164
+ const cacheName = me.getPlaceholder() + "_cache";
165
+ const newNameMapping = new Map<string, string>();
166
+
167
+ const keys = {
168
+ placeholderNoMeaning: isDebug ? "noMeaning" : getRandomString(10),
169
+ clearPayload: isDebug ? "clearPayload" : getRandomString(10),
170
+ nonCall: isDebug ? "nonCall" : getRandomString(10),
171
+ returnAsObject: isDebug ? "returnAsObject" : getRandomString(10),
172
+ returnAsObjectProperty: isDebug
173
+ ? "returnAsObjectProperty"
174
+ : getRandomString(10),
175
+ };
176
+
177
+ for (var name of functionPaths.keys()) {
178
+ newNameMapping.set(
179
+ name,
180
+ isDebug ? "_" + name : getRandomString(6) /** "_" + name */
181
+ );
182
+ }
183
+
184
+ // Find identifiers calling/referencing the functions
185
+ blockPath.traverse({
186
+ ReferencedIdentifier: {
187
+ exit(path: NodePath<t.Identifier | t.JSXIdentifier>) {
188
+ if (path.isJSX()) return;
189
+ if (isVariableFunctionIdentifier(path)) return;
190
+ const name = path.node.name;
191
+
192
+ var fnPath = functionPaths.get(name);
193
+ if (!fnPath) return;
194
+
195
+ var newName = newNameMapping.get(name);
196
+
197
+ // Do not replace if not referencing the actual function
198
+ if (path.scope.getBinding(name).path !== fnPath) {
199
+ return;
200
+ }
201
+
202
+ const createDispatcherCall = (name, flagArg?) => {
203
+ var dispatcherArgs = [t.stringLiteral(name)];
204
+ if (flagArg) {
205
+ dispatcherArgs.push(t.stringLiteral(flagArg));
206
+ }
207
+
208
+ var asObject = chance(50);
209
+
210
+ if (asObject) {
211
+ if (dispatcherArgs.length < 2) {
212
+ dispatcherArgs.push(
213
+ t.stringLiteral(keys.placeholderNoMeaning)
214
+ );
215
+ }
216
+ dispatcherArgs.push(t.stringLiteral(keys.returnAsObject));
217
+ }
218
+
219
+ var callExpression: t.CallExpression | t.NewExpression =
220
+ t.callExpression(
221
+ t.identifier(dispatcherName),
222
+ dispatcherArgs
223
+ );
224
+
225
+ if (!asObject) {
226
+ return callExpression;
227
+ }
228
+
229
+ if (chance(50)) {
230
+ (callExpression as t.Node).type = "NewExpression";
231
+ }
232
+
233
+ return t.memberExpression(
234
+ callExpression,
235
+ t.stringLiteral(keys.returnAsObjectProperty),
236
+ true
237
+ );
238
+ };
239
+
240
+ // Replace the identifier with a call to the function
241
+ const parentPath = path.parentPath;
242
+ if (path.key === "callee" && parentPath?.isCallExpression()) {
243
+ var expressions: t.Expression[] = [];
244
+ var callArguments = parentPath.node.arguments;
245
+
246
+ if (callArguments.length === 0) {
247
+ expressions.push(
248
+ // Call the function
249
+ createDispatcherCall(newName, keys.clearPayload)
250
+ );
251
+ } else {
252
+ expressions.push(
253
+ // Prepare the payload arguments
254
+ t.assignmentExpression(
255
+ "=",
256
+ t.identifier(payloadName),
257
+ t.arrayExpression(callArguments as t.Expression[])
258
+ ),
259
+
260
+ // Call the function
261
+ createDispatcherCall(newName)
262
+ );
263
+ }
264
+
265
+ const output =
266
+ expressions.length === 1
267
+ ? expressions[0]
268
+ : t.sequenceExpression(expressions);
269
+
270
+ if (!parentPath.container) return;
271
+ parentPath.replaceWith(output);
272
+ } else {
273
+ if (!path.container) return;
274
+ // Replace non-invocation references with a 'cached' version of the function
275
+ path.replaceWith(createDispatcherCall(newName, keys.nonCall));
276
+ }
277
+ },
278
+ },
279
+ });
280
+
281
+ const fnLengthProperties = [];
282
+
283
+ // Create the dispatcher function
284
+ const objectExpression = t.objectExpression(
285
+ Array.from(newNameMapping).map(([name, newName]) => {
286
+ const originalPath = functionPaths.get(name);
287
+ const originalFn = originalPath.node;
288
+
289
+ if (me.options.preserveFunctionLength) {
290
+ const fnLength = computeFunctionLength(originalPath);
291
+
292
+ if (fnLength >= 1) {
293
+ // 0 is already the default
294
+ fnLengthProperties.push(
295
+ t.objectProperty(
296
+ t.stringLiteral(newName),
297
+ numericLiteral(fnLength)
298
+ )
299
+ );
300
+ }
301
+ }
302
+
303
+ const newBody = [...originalFn.body.body];
304
+ ok(Array.isArray(newBody));
305
+
306
+ // Unpack parameters
307
+ if (originalFn.params.length > 0) {
308
+ newBody.unshift(
309
+ t.variableDeclaration("var", [
310
+ t.variableDeclarator(
311
+ t.arrayPattern([...originalFn.params]),
312
+ t.identifier(payloadName)
313
+ ),
314
+ ])
315
+ );
316
+ }
317
+
318
+ // Add debug label
319
+ if (isDebug) {
320
+ newBody.unshift(
321
+ t.expressionStatement(
322
+ t.stringLiteral(`Dispatcher: ${name} -> ${newName}`)
323
+ )
324
+ );
325
+ }
326
+
327
+ const functionExpression = t.functionExpression(
328
+ null,
329
+ [],
330
+ t.blockStatement(newBody)
331
+ );
332
+
333
+ for (var symbol of Object.getOwnPropertySymbols(originalFn)) {
334
+ (functionExpression as any)[symbol] = (originalFn as any)[
335
+ symbol
336
+ ];
337
+ }
338
+
339
+ (functionExpression as NodeSymbol)[PREDICTABLE] = true;
340
+
341
+ return t.objectProperty(
342
+ t.stringLiteral(newName),
343
+
344
+ functionExpression
345
+ );
346
+ })
347
+ );
348
+
349
+ const fnLengths = t.objectExpression(fnLengthProperties);
350
+
351
+ const dispatcher = new Template(`
352
+ function ${dispatcherName}(name, flagArg, returnTypeArg, fnLengths = {fnLengthsObjectExpression}) {
353
+ var output;
354
+ var fns = {objectExpression};
355
+
356
+ if(flagArg === "${keys.clearPayload}") {
357
+ ${payloadName} = [];
358
+ }
359
+ if(flagArg === "${keys.nonCall}") {
360
+ function createFunction(){
361
+ var fn = function(...args){
362
+ ${payloadName} = args;
363
+ return fns[name].apply(this);
364
+ }
365
+
366
+ var fnLength = fnLengths[name];
367
+ if(fnLength) {
368
+ ${setFunctionLength}(fn, fnLength);
369
+ }
370
+
371
+ return fn;
372
+ }
373
+ output = ${cacheName}[name] || (${cacheName}[name] = createFunction());
374
+ } else {
375
+ output = fns[name]();
376
+ }
377
+
378
+ if(returnTypeArg === "${keys.returnAsObject}") {
379
+ return { "${keys.returnAsObjectProperty}": output };
380
+ } else {
381
+ return output;
382
+ }
383
+ }
384
+ `).single({
385
+ objectExpression,
386
+ fnLengthsObjectExpression: fnLengths,
387
+ });
388
+
389
+ (dispatcher as NodeSymbol)[PREDICTABLE] = true;
390
+
391
+ /**
392
+ * Prepends the node into the block. (And registers the declaration)
393
+ * @param node
394
+ */
395
+ function prepend(node: t.Statement) {
396
+ var newPath = blockStatement.unshiftContainer<any, any, any>(
397
+ "body",
398
+ node
399
+ )[0];
400
+ blockStatement.scope.registerDeclaration(newPath);
401
+ }
402
+
403
+ // Insert the dispatcher function
404
+ prepend(dispatcher);
405
+
406
+ // Insert the payload variable
407
+ prepend(
408
+ t.variableDeclaration("var", [
409
+ t.variableDeclarator(t.identifier(payloadName)),
410
+ ])
411
+ );
412
+
413
+ // Insert the cache variable
414
+ prepend(
415
+ t.variableDeclaration("var", [
416
+ t.variableDeclarator(
417
+ t.identifier(cacheName),
418
+ new Template(`Object["create"](null)`).expression()
419
+ ),
420
+ ])
421
+ );
422
+
423
+ // Remove original functions
424
+ for (let path of functionPaths.values()) {
425
+ path.remove();
426
+ }
427
+ },
428
+ },
429
+ },
430
+ };
431
+ };