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
@@ -0,0 +1,667 @@
1
+ import * as t from "@babel/types";
2
+ import { NodePath } from "@babel/core";
3
+ import { ok } from "assert";
4
+ import { deepClone } from "./node";
5
+
6
+ export function getPatternIdentifierNames(
7
+ path: NodePath | NodePath[]
8
+ ): Set<string> {
9
+ if (Array.isArray(path)) {
10
+ var allNames = new Set<string>();
11
+ for (var p of path) {
12
+ var names = getPatternIdentifierNames(p);
13
+ for (var name of names) {
14
+ allNames.add(name);
15
+ }
16
+ }
17
+
18
+ return allNames;
19
+ }
20
+ var names = new Set<string>();
21
+
22
+ var functionParent = path.find((parent) => parent.isFunction());
23
+
24
+ path.traverse({
25
+ BindingIdentifier: (bindingPath) => {
26
+ var bindingFunctionParent = bindingPath.find((parent) =>
27
+ parent.isFunction()
28
+ );
29
+ if (functionParent === bindingFunctionParent) {
30
+ names.add(bindingPath.node.name);
31
+ }
32
+ },
33
+ });
34
+
35
+ // Check if the path itself is a binding identifier
36
+ if (path.isBindingIdentifier()) {
37
+ names.add(path.node.name);
38
+ }
39
+
40
+ return names;
41
+ }
42
+
43
+ /**
44
+ * Ensures a `String Literal` is 'computed' before replacing it with a more complex expression.
45
+ *
46
+ * ```js
47
+ * // Input
48
+ * {
49
+ * "myToBeEncodedString": "value"
50
+ * }
51
+ *
52
+ * // Output
53
+ * {
54
+ * ["myToBeEncodedString"]: "value"
55
+ * }
56
+ * ```
57
+ * @param path
58
+ */
59
+ export function ensureComputedExpression(path: NodePath<t.Node>) {
60
+ if (
61
+ (t.isObjectMember(path.parent) ||
62
+ t.isClassMethod(path.parent) ||
63
+ t.isClassProperty(path.parent)) &&
64
+ path.parent.key === path.node &&
65
+ !path.parent.computed
66
+ ) {
67
+ path.parent.computed = true;
68
+ }
69
+ }
70
+
71
+ /**
72
+ * Retrieves a function name from debugging purposes.
73
+ * - Function Declaration / Expression
74
+ * - Variable Declaration
75
+ * - Object property / method
76
+ * - Class property / method
77
+ * - Program returns "[Program]"
78
+ * - Default returns "anonymous"
79
+ * @param path
80
+ * @returns
81
+ */
82
+ export function getFunctionName(path: NodePath<t.Function>): string {
83
+ if (!path) return "null";
84
+ if (path.isProgram()) return "[Program]";
85
+
86
+ // Check function declaration/expression ID
87
+ if (
88
+ (t.isFunctionDeclaration(path.node) || t.isFunctionExpression(path.node)) &&
89
+ path.node.id
90
+ ) {
91
+ return path.node.id.name;
92
+ }
93
+
94
+ // Check for containing variable declaration
95
+ if (
96
+ path.parentPath?.isVariableDeclarator() &&
97
+ t.isIdentifier(path.parentPath.node.id)
98
+ ) {
99
+ return path.parentPath.node.id.name;
100
+ }
101
+
102
+ if (path.isObjectMethod() || path.isClassMethod()) {
103
+ var property = getObjectPropertyAsString(path.node);
104
+ if (property) return property;
105
+ }
106
+
107
+ // Check for containing property in an object
108
+ if (
109
+ path.parentPath?.isObjectProperty() ||
110
+ path.parentPath?.isClassProperty()
111
+ ) {
112
+ var property = getObjectPropertyAsString(path.parentPath.node);
113
+ if (property) return property;
114
+ }
115
+
116
+ var output = "anonymous";
117
+
118
+ if (path.isFunction()) {
119
+ if (path.node.generator) {
120
+ output += "*";
121
+ } else if (path.node.async) {
122
+ output = "async " + output;
123
+ }
124
+ }
125
+
126
+ return output;
127
+ }
128
+
129
+ export function isModuleImport(path: NodePath<t.StringLiteral>) {
130
+ // Import Declaration
131
+ if (path.parentPath.isImportDeclaration()) {
132
+ return true;
133
+ }
134
+
135
+ // Dynamic Import / require() call
136
+ if (
137
+ t.isCallExpression(path.parent) &&
138
+ (t.isIdentifier(path.parent.callee, { name: "require" }) ||
139
+ t.isImport(path.parent.callee)) &&
140
+ path.node === path.parent.arguments[0]
141
+ ) {
142
+ return true;
143
+ }
144
+
145
+ return false;
146
+ }
147
+
148
+ export function getBlock(path: NodePath) {
149
+ return path.find((p) => p.isBlock()) as NodePath<t.Block>;
150
+ }
151
+
152
+ export function getParentFunctionOrProgram(
153
+ path: NodePath
154
+ ): NodePath<t.Function | t.Program> {
155
+ if (path.isProgram()) return path;
156
+
157
+ // Find the nearest function-like parent
158
+ const functionOrProgramPath = path.findParent(
159
+ (parentPath) => parentPath.isFunction() || parentPath.isProgram()
160
+ ) as NodePath<t.Function | t.Program>;
161
+
162
+ ok(functionOrProgramPath);
163
+ return functionOrProgramPath;
164
+ }
165
+
166
+ export function getObjectPropertyAsString(
167
+ property: t.ObjectMember | t.ClassProperty | t.ClassMethod
168
+ ): string {
169
+ ok(
170
+ t.isObjectMember(property) ||
171
+ t.isClassProperty(property) ||
172
+ t.isClassMethod(property)
173
+ );
174
+
175
+ if (!property.computed && t.isIdentifier(property.key)) {
176
+ return property.key.name;
177
+ }
178
+
179
+ if (t.isStringLiteral(property.key)) {
180
+ return property.key.value;
181
+ }
182
+
183
+ if (t.isNumericLiteral(property.key)) {
184
+ return property.key.value.toString();
185
+ }
186
+
187
+ return null;
188
+ }
189
+
190
+ /**
191
+ * Gets the property of a MemberExpression as a string.
192
+ *
193
+ * @param memberPath - The path of the MemberExpression node.
194
+ * @returns The property as a string or null if it cannot be determined.
195
+ */
196
+ export function getMemberExpressionPropertyAsString(
197
+ member: t.MemberExpression
198
+ ): string | null {
199
+ t.assertMemberExpression(member);
200
+
201
+ const property = member.property;
202
+
203
+ if (!member.computed && t.isIdentifier(property)) {
204
+ return property.name;
205
+ }
206
+
207
+ if (t.isStringLiteral(property)) {
208
+ return property.value;
209
+ }
210
+
211
+ if (t.isNumericLiteral(property)) {
212
+ return property.value.toString();
213
+ }
214
+
215
+ return null; // If the property cannot be determined
216
+ }
217
+
218
+ function registerPaths(paths: NodePath[]) {
219
+ for (var path of paths) {
220
+ if (path.isVariableDeclaration() && path.node.kind === "var") {
221
+ getParentFunctionOrProgram(path).scope.registerDeclaration(path);
222
+ }
223
+ path.scope.registerDeclaration(path);
224
+ }
225
+
226
+ return paths;
227
+ }
228
+
229
+ function nodeListToNodes(nodesIn: (t.Statement | t.Statement[])[]) {
230
+ var nodes: t.Statement[] = [];
231
+ if (Array.isArray(nodesIn[0])) {
232
+ ok(nodesIn.length === 1);
233
+ nodes = nodesIn[0];
234
+ } else {
235
+ nodes = nodesIn as t.Statement[];
236
+ }
237
+
238
+ return nodes;
239
+ }
240
+
241
+ /**
242
+ * Appends to the bottom of a block. Preserving last expression for the top level.
243
+ */
244
+ export function append(
245
+ path: NodePath,
246
+ ...nodesIn: (t.Statement | t.Statement[])[]
247
+ ) {
248
+ var nodes = nodeListToNodes(nodesIn);
249
+
250
+ var listParent = path.find(
251
+ (p) => p.isFunction() || p.isBlock() || p.isSwitchCase()
252
+ );
253
+ if (!listParent) {
254
+ throw new Error("Could not find a suitable parent to prepend to");
255
+ }
256
+
257
+ if (listParent.isProgram()) {
258
+ var lastExpression = listParent.get("body").at(-1);
259
+ if (lastExpression.isExpressionStatement()) {
260
+ return registerPaths(lastExpression.insertBefore(nodes));
261
+ }
262
+ }
263
+
264
+ if (listParent.isSwitchCase()) {
265
+ return registerPaths(listParent.pushContainer("consequent", nodes));
266
+ }
267
+
268
+ if (listParent.isFunction()) {
269
+ var body = listParent.get("body");
270
+
271
+ if (listParent.isArrowFunctionExpression() && listParent.node.expression) {
272
+ if (!body.isBlockStatement()) {
273
+ body.replaceWith(
274
+ t.blockStatement([t.returnStatement(body.node as t.Expression)])
275
+ );
276
+ }
277
+ }
278
+
279
+ ok(body.isBlockStatement());
280
+
281
+ return registerPaths(body.pushContainer("body", nodes));
282
+ }
283
+
284
+ ok(listParent.isBlock());
285
+ return registerPaths(listParent.pushContainer("body", nodes));
286
+ }
287
+
288
+ /**
289
+ * Prepends and registers a list of nodes to the beginning of a block.
290
+ *
291
+ * - Preserves import declarations by inserting after the last import declaration.
292
+ * - Handles arrow functions
293
+ * - Handles switch cases
294
+ * @param path
295
+ * @param nodes
296
+ * @returns
297
+ */
298
+ export function prepend(
299
+ path: NodePath,
300
+ ...nodesIn: (t.Statement | t.Statement[])[]
301
+ ): NodePath[] {
302
+ var nodes = nodeListToNodes(nodesIn);
303
+
304
+ var listParent = path.find(
305
+ (p) => p.isFunction() || p.isBlock() || p.isSwitchCase()
306
+ );
307
+ if (!listParent) {
308
+ throw new Error("Could not find a suitable parent to prepend to");
309
+ }
310
+
311
+ if (listParent.isProgram()) {
312
+ // Preserve import declarations
313
+ // Filter out import declarations
314
+ const body = listParent.get("body");
315
+ const lastImportIndex = body.findIndex(
316
+ (path) => !path.isImportDeclaration()
317
+ );
318
+
319
+ if (lastImportIndex === 0 || lastImportIndex === -1) {
320
+ // No non-import declarations, so we can safely unshift everything
321
+ return registerPaths(listParent.unshiftContainer("body", nodes));
322
+ } else {
323
+ // Insert the nodes after the last import declaration
324
+ return registerPaths(body[lastImportIndex - 1].insertAfter(nodes));
325
+ }
326
+ }
327
+
328
+ if (listParent.isFunction()) {
329
+ var body = listParent.get("body");
330
+
331
+ if (listParent.isArrowFunctionExpression() && listParent.node.expression) {
332
+ if (!body.isBlockStatement()) {
333
+ body = body.replaceWith(
334
+ t.blockStatement([t.returnStatement(body.node as t.Expression)])
335
+ )[0];
336
+ }
337
+ }
338
+
339
+ ok(body.isBlockStatement());
340
+
341
+ return registerPaths(body.unshiftContainer("body", nodes));
342
+ }
343
+
344
+ if (listParent.isBlock()) {
345
+ return registerPaths(listParent.unshiftContainer("body", nodes));
346
+ } else if (listParent.isSwitchCase()) {
347
+ return registerPaths(listParent.unshiftContainer("consequent", nodes));
348
+ }
349
+
350
+ ok(false);
351
+ }
352
+
353
+ export function prependProgram(
354
+ path: NodePath,
355
+ ...nodes: (t.Statement | t.Statement[])[]
356
+ ) {
357
+ var program = path.find((p) => p.isProgram());
358
+ ok(program);
359
+ return prepend(program, ...nodes);
360
+ }
361
+
362
+ /**
363
+ * A referenced or binding identifier, only names that reflect variables.
364
+ *
365
+ * - Excludes labels
366
+ *
367
+ * @param path
368
+ * @returns
369
+ */
370
+ export function isVariableIdentifier(path: NodePath<t.Identifier>) {
371
+ if (
372
+ !path.isReferencedIdentifier() &&
373
+ !(path as NodePath).isBindingIdentifier()
374
+ )
375
+ return false;
376
+
377
+ // abc: {} // not a variable identifier
378
+ if (path.key === "label" && path.parentPath?.isLabeledStatement())
379
+ return false;
380
+
381
+ return true;
382
+ }
383
+
384
+ /**
385
+ * Subset of BindingIdentifier, excluding non-defined assignment expressions.
386
+ *
387
+ * @example
388
+ * var a = 1; // true
389
+ * var {c} = {} // true
390
+ * function b() {} // true
391
+ * function d([e] = [], ...f) {} // true
392
+ *
393
+ * f = 0; // false
394
+ * f(); // false
395
+ * @param path
396
+ * @returns
397
+ */
398
+ export function isDefiningIdentifier(path: NodePath<t.Identifier>) {
399
+ if (path.key === "id" && path.parentPath.isFunction()) return true;
400
+ if (path.key === "id" && path.parentPath.isClassDeclaration) return true;
401
+ if (
402
+ path.key === "local" &&
403
+ (path.parentPath.isImportSpecifier() ||
404
+ path.parentPath.isImportDefaultSpecifier() ||
405
+ path.parentPath.isImportNamespaceSpecifier())
406
+ )
407
+ return true;
408
+
409
+ var maxTraversalPath = path.find(
410
+ (p) =>
411
+ (p.key === "id" && p.parentPath?.isVariableDeclarator()) ||
412
+ (p.listKey === "params" && p.parentPath?.isFunction()) ||
413
+ (p.key === "param" && p.parentPath?.isCatchClause())
414
+ );
415
+
416
+ if (!maxTraversalPath) return false;
417
+
418
+ var cursor: NodePath = path;
419
+ while (cursor && cursor !== maxTraversalPath) {
420
+ if (
421
+ cursor.parentPath.isObjectProperty() &&
422
+ cursor.parentPath.parentPath?.isObjectPattern()
423
+ ) {
424
+ if (cursor.key !== "value") {
425
+ return false;
426
+ }
427
+ } else if (cursor.parentPath.isArrayPattern()) {
428
+ if (cursor.listKey !== "elements") {
429
+ return false;
430
+ }
431
+ } else if (cursor.parentPath.isRestElement()) {
432
+ if (cursor.key !== "argument") {
433
+ return false;
434
+ }
435
+ } else if (cursor.parentPath.isAssignmentPattern()) {
436
+ if (cursor.key !== "left") {
437
+ return false;
438
+ }
439
+ } else if (cursor.parentPath.isObjectPattern()) {
440
+ } else return false;
441
+
442
+ cursor = cursor.parentPath;
443
+ }
444
+
445
+ return true;
446
+ }
447
+
448
+ /**
449
+ * @example
450
+ * function id() {} // true
451
+ * class id {} // true
452
+ * var id; // false
453
+ * @param path
454
+ * @returns
455
+ */
456
+ export function isStrictIdentifier(path: NodePath): boolean {
457
+ if (
458
+ path.key === "id" &&
459
+ (path.parentPath.isFunction() || path.parentPath.isClass())
460
+ )
461
+ return true;
462
+
463
+ return false;
464
+ }
465
+
466
+ export function isExportedIdentifier(path: NodePath<t.Identifier>) {
467
+ // Check if the identifier is directly inside an ExportNamedDeclaration
468
+ if (path.parentPath.isExportNamedDeclaration()) {
469
+ return true;
470
+ }
471
+
472
+ // Check if the identifier is in an ExportDefaultDeclaration
473
+ if (path.parentPath.isExportDefaultDeclaration()) {
474
+ return true;
475
+ }
476
+
477
+ // Check if the identifier is within an ExportSpecifier
478
+ if (
479
+ path.parentPath.isExportSpecifier() &&
480
+ path.parentPath.parentPath.isExportNamedDeclaration()
481
+ ) {
482
+ return true;
483
+ }
484
+
485
+ // Check if it's part of an exported variable declaration (e.g., export const a = 1;)
486
+ if (
487
+ path.parentPath.isVariableDeclarator() &&
488
+ path.parentPath.parentPath.parentPath.isExportNamedDeclaration()
489
+ ) {
490
+ return true;
491
+ }
492
+
493
+ // Check if it's part of an exported function declaration (e.g., export function abc() {})
494
+ if (
495
+ (path.parentPath.isFunctionDeclaration() ||
496
+ path.parentPath.isClassDeclaration()) &&
497
+ path.parentPath.parentPath.isExportNamedDeclaration()
498
+ ) {
499
+ return true;
500
+ }
501
+
502
+ return false;
503
+ }
504
+
505
+ /**
506
+ * @example
507
+ * function abc() {
508
+ * "use strict";
509
+ * } // true
510
+ * @param path
511
+ * @returns
512
+ */
513
+ export function isStrictMode(path: NodePath) {
514
+ // Classes are always in strict mode
515
+ if (path.isClass()) return true;
516
+
517
+ if (path.isBlock()) {
518
+ if (path.isTSModuleBlock()) return false;
519
+ return (path.node as t.BlockStatement | t.Program).directives.some(
520
+ (directive) => directive.value.value === "use strict"
521
+ );
522
+ }
523
+
524
+ if (path.isFunction()) {
525
+ const fnBody = path.get("body");
526
+ if (fnBody.isBlock()) {
527
+ return isStrictMode(fnBody);
528
+ }
529
+ }
530
+
531
+ return false;
532
+ }
533
+
534
+ /**
535
+ * A modified identifier is an identifier that is assigned to or updated.
536
+ *
537
+ * - Assignment Expression
538
+ * - Update Expression
539
+ *
540
+ * @param identifierPath
541
+ */
542
+ export function isModifiedIdentifier(identifierPath: NodePath<t.Identifier>) {
543
+ var isModification = false;
544
+ if (identifierPath.parentPath.isUpdateExpression()) {
545
+ isModification = true;
546
+ }
547
+ if (
548
+ identifierPath.find(
549
+ (p) => p.key === "left" && p.parentPath?.isAssignmentExpression()
550
+ )
551
+ ) {
552
+ isModification = true;
553
+ }
554
+
555
+ return isModification;
556
+ }
557
+
558
+ export function replaceDefiningIdentifierToMemberExpression(
559
+ path: NodePath<t.Identifier>,
560
+ memberExpression: t.MemberExpression
561
+ ) {
562
+ // function id(){} -> var id = function() {}
563
+ if (path.key === "id" && path.parentPath.isFunctionDeclaration()) {
564
+ var asFunctionExpression = deepClone(
565
+ path.parentPath.node
566
+ ) as t.Node as t.FunctionExpression;
567
+ asFunctionExpression.type = "FunctionExpression";
568
+
569
+ path.parentPath.replaceWith(
570
+ t.expressionStatement(
571
+ t.assignmentExpression("=", memberExpression, asFunctionExpression)
572
+ )
573
+ );
574
+ return;
575
+ }
576
+
577
+ // class id{} -> var id = class {}
578
+ if (path.key === "id" && path.parentPath.isClassDeclaration()) {
579
+ var asClassExpression = deepClone(
580
+ path.parentPath.node
581
+ ) as t.Node as t.ClassExpression;
582
+ asClassExpression.type = "ClassExpression";
583
+
584
+ path.parentPath.replaceWith(
585
+ t.expressionStatement(
586
+ t.assignmentExpression("=", memberExpression, asClassExpression)
587
+ )
588
+ );
589
+ return;
590
+ }
591
+
592
+ // var id = 1 -> id = 1
593
+ var variableDeclaratorChild = path.find(
594
+ (p) =>
595
+ p.key === "id" &&
596
+ p.parentPath?.isVariableDeclarator() &&
597
+ p.parentPath?.parentPath?.isVariableDeclaration()
598
+ ) as NodePath<t.VariableDeclarator["id"]>;
599
+
600
+ if (variableDeclaratorChild) {
601
+ var variableDeclarator =
602
+ variableDeclaratorChild.parentPath as NodePath<t.VariableDeclarator>;
603
+ var variableDeclaration =
604
+ variableDeclarator.parentPath as NodePath<t.VariableDeclaration>;
605
+
606
+ if (variableDeclaration.type === "VariableDeclaration") {
607
+ ok(
608
+ variableDeclaration.node.declarations.length === 1,
609
+ "Multiple declarations not supported"
610
+ );
611
+ }
612
+
613
+ const id = variableDeclarator.get("id");
614
+ const init = variableDeclarator.get("init");
615
+
616
+ var newExpression: t.Node = id.node;
617
+
618
+ var isForInitializer =
619
+ (variableDeclaration.key === "init" ||
620
+ variableDeclaration.key === "left") &&
621
+ variableDeclaration.parentPath.isFor();
622
+
623
+ if (init.node || !isForInitializer) {
624
+ newExpression = t.assignmentExpression(
625
+ "=",
626
+ id.node,
627
+ init.node || t.identifier("undefined")
628
+ );
629
+ }
630
+
631
+ if (!isForInitializer) {
632
+ newExpression = t.expressionStatement(newExpression as t.Expression);
633
+ }
634
+
635
+ path.replaceWith(memberExpression);
636
+
637
+ if (variableDeclaration.isVariableDeclaration()) {
638
+ variableDeclaration.replaceWith(newExpression);
639
+ }
640
+
641
+ return;
642
+ }
643
+
644
+ // Safely replace the identifier with the member expression
645
+ // ensureComputedExpression(path);
646
+ // path.replaceWith(memberExpression);
647
+ }
648
+
649
+ /**
650
+ * @example
651
+ * undefined // true
652
+ * void 0 // true
653
+ */
654
+ export function isUndefined(path: NodePath) {
655
+ if (path.isIdentifier() && path.node.name === "undefined") {
656
+ return true;
657
+ }
658
+ if (
659
+ path.isUnaryExpression() &&
660
+ path.node.operator === "void" &&
661
+ path.node.argument.type === "NumericLiteral" &&
662
+ path.node.argument.value === 0
663
+ ) {
664
+ return true;
665
+ }
666
+ return false;
667
+ }
@@ -0,0 +1,50 @@
1
+ import { NodePath } from "@babel/traverse";
2
+ import * as t from "@babel/types";
3
+ import { FN_LENGTH, NodeSymbol, variableFunctionName } from "../constants";
4
+
5
+ /**
6
+ * @example __JS_CONFUSER_VAR__(identifier) // true
7
+ * @param path
8
+ * @returns
9
+ */
10
+ export function isVariableFunctionIdentifier(path: NodePath<t.Node>) {
11
+ if (
12
+ path.isIdentifier() &&
13
+ path.listKey === "arguments" &&
14
+ path.key === 0 &&
15
+ path.parentPath?.isCallExpression()
16
+ ) {
17
+ const callee = path.parentPath.get("callee");
18
+ return callee.isIdentifier({ name: variableFunctionName });
19
+ }
20
+
21
+ return false;
22
+ }
23
+
24
+ /**
25
+ * Computes the `function.length` property given the parameter nodes.
26
+ *
27
+ * @example function abc(a, b, c = 1, ...d) {} // abc.length = 2
28
+ */
29
+ export function computeFunctionLength(fnPath: NodePath<t.Function>): number {
30
+ var savedLength = (fnPath.node as NodeSymbol)[FN_LENGTH];
31
+ if (typeof savedLength === "number") {
32
+ return savedLength;
33
+ }
34
+
35
+ var count = 0;
36
+
37
+ for (var parameterNode of fnPath.node.params) {
38
+ if (
39
+ parameterNode.type === "Identifier" ||
40
+ parameterNode.type === "ObjectPattern" ||
41
+ parameterNode.type === "ArrayPattern"
42
+ ) {
43
+ count++;
44
+ } else {
45
+ break;
46
+ }
47
+ }
48
+
49
+ return count;
50
+ }