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,18 +1,20 @@
1
- import Transform from "./transform";
2
-
1
+ import { NodePath } from "@babel/traverse";
2
+ import { PluginArg, PluginObject } from "./plugin";
3
+ import * as t from "@babel/types";
4
+ import { Order } from "../order";
5
+ import path from "path";
6
+ import {
7
+ NodeSymbol,
8
+ PREDICTABLE,
9
+ UNSAFE,
10
+ variableFunctionName,
11
+ } from "../constants";
12
+ import { ok } from "assert";
3
13
  import {
4
- BlockStatement,
5
- Identifier,
6
- LabeledStatement,
7
- Literal,
8
- Node,
9
- ReturnStatement,
10
- } from "../util/gen";
11
- import { ObfuscateOrder } from "../order";
12
- import { clone, getFunction } from "../util/insert";
13
- import { getIdentifierInfo } from "../util/identifiers";
14
- import { isLoop } from "../util/compare";
15
- import { ExitCallback, walk } from "../traverse";
14
+ getParentFunctionOrProgram,
15
+ getPatternIdentifierNames,
16
+ } from "../utils/ast-utils";
17
+ import { isVariableFunctionIdentifier } from "../utils/function-utils";
16
18
 
17
19
  /**
18
20
  * Preparation arranges the user's code into an AST the obfuscator can easily transform.
@@ -28,211 +30,331 @@ import { ExitCallback, walk } from "../traverse";
28
30
  * - `x => x * 2` -> `x => { return x * 2 }` // Change into Block Statements
29
31
  * - `if(true) return` -> `if (true) { return }`
30
32
  * - `while(a) a--;` -> `while(a) { a-- }`
31
- *
32
- * Label
33
- * - `for(...) { break; }` -> `_1: for(...) { break _1; }`
34
- * - `switch(v) { case 1...break }` -> `_2: switch(v) { case 1...break _2; }`
35
- * - // Control Flow Flattening can safely apply now
36
33
  */
37
- export default class Preparation extends Transform {
38
- constructor(o) {
39
- super(o, ObfuscateOrder.Preparation);
40
- }
41
-
42
- match(object: Node, parents: Node[]) {
43
- return !!object.type;
44
- }
45
-
46
- transform(object: Node, parents: Node[]): void | ExitCallback {
47
- // ExplicitIdentifiers
48
- if (object.type === "Identifier") {
49
- return this.transformExplicitIdentifiers(object, parents);
50
- }
51
-
52
- // ExplicitDeclarations
53
- if (object.type === "VariableDeclaration") {
54
- return this.transformExplicitDeclarations(object, parents);
55
- }
56
-
57
- // Block
58
- switch (object.type) {
59
- /**
60
- * People use shortcuts and its harder to parse.
61
- *
62
- * - `if (a) b()` -> `if (a) { b() }`
63
- * - Ensures all bodies are `BlockStatement`, not individual expression statements
64
- */
65
- case "IfStatement":
66
- if (object.consequent.type != "BlockStatement") {
67
- object.consequent = BlockStatement([clone(object.consequent)]);
68
- }
69
- if (object.alternate && object.alternate.type != "BlockStatement") {
70
- object.alternate = BlockStatement([clone(object.alternate)]);
71
- }
72
- break;
73
-
74
- case "WhileStatement":
75
- case "WithStatement":
76
- case "ForStatement":
77
- case "ForOfStatement":
78
- case "ForInStatement":
79
- if (object.body.type != "BlockStatement") {
80
- object.body = BlockStatement([clone(object.body)]);
81
- }
82
- break;
83
-
84
- case "ArrowFunctionExpression":
85
- if (object.body.type !== "BlockStatement" && object.expression) {
86
- object.body = BlockStatement([ReturnStatement(clone(object.body))]);
87
- object.expression = false;
88
- }
89
- break;
90
- }
91
-
92
- // Label
93
- if (
94
- isLoop(object) ||
95
- (object.type == "BlockStatement" &&
96
- parents[0] &&
97
- parents[0].type == "LabeledStatement" &&
98
- parents[0].body === object)
99
- ) {
100
- return this.transformLabel(object, parents);
101
- }
102
- }
103
-
104
- /**
105
- * Ensures every break; statement has a label to point to.
106
- *
107
- * This is because Control Flow Flattening adds For Loops which label-less break statements point to the nearest,
108
- * when they actually need to point to the original statement.
109
- */
110
- transformLabel(object: Node, parents: Node[]) {
111
- return () => {
112
- var currentLabel =
113
- parents[0].type == "LabeledStatement" && parents[0].label.name;
114
-
115
- var label = currentLabel || this.getPlaceholder();
116
-
117
- walk(object, parents, (o, p) => {
118
- if (o.type == "BreakStatement" || o.type == "ContinueStatement") {
119
- function isContinuableStatement(x) {
120
- return isLoop(x) && x.type !== "SwitchStatement";
34
+ export default ({ Plugin }: PluginArg): PluginObject => {
35
+ const me = Plugin(Order.Preparation);
36
+
37
+ const markFunctionUnsafe = (path: NodePath<t.Node>) => {
38
+ const functionPath = path.findParent(
39
+ (path) => path.isFunction() || path.isProgram()
40
+ );
41
+ if (!functionPath) return;
42
+
43
+ const functionNode = functionPath.node;
44
+
45
+ (functionNode as NodeSymbol)[UNSAFE] = true;
46
+ };
47
+
48
+ return {
49
+ visitor: {
50
+ "ThisExpression|Super": {
51
+ exit(path) {
52
+ markFunctionUnsafe(path);
53
+ },
54
+ },
55
+
56
+ // `Hello ${username}` -> "Hello " + username
57
+ TemplateLiteral: {
58
+ exit(path) {
59
+ // Check if this is a tagged template literal, if yes, skip it
60
+ if (t.isTaggedTemplateExpression(path.parent)) {
61
+ return;
62
+ }
63
+
64
+ const { quasis, expressions } = path.node;
65
+
66
+ // Start with the first quasi (template string part)
67
+ let binaryExpression: t.Expression = t.stringLiteral(
68
+ quasis[0].value.cooked
69
+ );
70
+
71
+ // Loop over the remaining quasis and expressions, concatenating them
72
+ for (let i = 0; i < expressions.length; i++) {
73
+ // Add the expression as part of the binary concatenation
74
+ binaryExpression = t.binaryExpression(
75
+ "+",
76
+ binaryExpression,
77
+ expressions[i] as t.Expression
78
+ );
79
+
80
+ // Add the next quasi (template string part)
81
+ if (quasis[i + 1].value.cooked !== "") {
82
+ binaryExpression = t.binaryExpression(
83
+ "+",
84
+ binaryExpression,
85
+ t.stringLiteral(quasis[i + 1].value.cooked)
86
+ );
87
+ }
121
88
  }
122
- function isBreakableStatement(x) {
123
- return isLoop(x) || (o.label && x.type == "BlockStatement");
89
+
90
+ // Replace the template literal with the constructed binary expression
91
+ path.replaceWith(binaryExpression);
92
+ },
93
+ },
94
+
95
+ // /Hello World/g -> new RegExp("Hello World", "g")
96
+ RegExpLiteral: {
97
+ exit(path) {
98
+ const { pattern, flags } = path.node;
99
+
100
+ // Create a new RegExp() expression using the pattern and flags
101
+ const newRegExpCall = t.newExpression(
102
+ t.identifier("RegExp"), // Identifier for RegExp constructor
103
+ [
104
+ t.stringLiteral(pattern), // First argument: the pattern (no extra escaping needed)
105
+ flags ? t.stringLiteral(flags) : t.stringLiteral(""), // Second argument: the flags (if any)
106
+ ]
107
+ );
108
+
109
+ // Replace the literal regex with the new RegExp() call
110
+ path.replaceWith(newRegExpCall);
111
+ },
112
+ },
113
+
114
+ ReferencedIdentifier: {
115
+ exit(path) {
116
+ const { name } = path.node;
117
+ if (["arguments", "eval"].includes(name)) {
118
+ markFunctionUnsafe(path);
124
119
  }
125
120
 
126
- var fn =
127
- o.type == "ContinueStatement"
128
- ? isContinuableStatement
129
- : isBreakableStatement;
121
+ // When Rename Variables is disabled, __JS_CONFUSER_VAR__ must still be removed
122
+ if (
123
+ !me.obfuscator.hasPlugin(Order.RenameVariables) &&
124
+ isVariableFunctionIdentifier(path)
125
+ ) {
126
+ ok(
127
+ path.parentPath.isCallExpression(),
128
+ variableFunctionName + " must be directly called"
129
+ );
130
+
131
+ var argument = path.parentPath.node.arguments[0];
132
+ t.assertIdentifier(argument);
133
+
134
+ // Remove the variableFunctionName call
135
+ path.parentPath.replaceWith(t.stringLiteral(argument.name));
136
+ }
137
+ },
138
+ },
139
+
140
+ FunctionDeclaration: {
141
+ exit(path) {
142
+ // A function is 'predictable' if the parameter lengths are guaranteed to be known
143
+ // a(true) -> predictable
144
+ // (a || b)(true) -> unpredictable (Must be directly in a Call Expression)
145
+ // a(...args) -> unpredictable (Cannot use SpreadElement)
146
+
147
+ const { name } = path.node.id;
148
+
149
+ var binding = path.scope.getBinding(name);
150
+ var predictable = true;
151
+ var maxArgLength = 0;
152
+
153
+ for (var referencePath of binding.referencePaths) {
154
+ if (!referencePath.parentPath.isCallExpression()) {
155
+ predictable = false;
156
+ break;
157
+ }
158
+
159
+ var argsPath = referencePath.parentPath.get("arguments");
160
+ for (var arg of argsPath) {
161
+ if (arg.isSpreadElement()) {
162
+ predictable = false;
163
+ break;
164
+ }
165
+ }
130
166
 
131
- var loop = p.find(fn);
132
- if (object == loop) {
133
- if (!o.label) {
134
- o.label = Identifier(label);
167
+ if (argsPath.length > maxArgLength) {
168
+ maxArgLength = argsPath.length;
135
169
  }
136
170
  }
137
- }
138
- });
139
-
140
- // Append label statement as this loop has none
141
- if (!currentLabel) {
142
- this.replace(object, LabeledStatement(label, { ...object }));
143
- }
144
- };
145
- }
146
-
147
- /**
148
- * Transforms Identifiers (a.IDENTIFIER, {IDENTIFIER:...}) into string properties
149
- */
150
- transformExplicitIdentifiers(object: Node, parents: Node[]) {
151
- // Mark functions containing 'eval'
152
- // Some transformations avoid functions that have 'eval' to not break them
153
- if (object.name === "eval") {
154
- var fn = getFunction(object, parents);
155
- if (fn) {
156
- fn.$requiresEval = true;
157
- }
158
- }
159
-
160
- var info = getIdentifierInfo(object, parents);
161
- if (info.isPropertyKey || info.isAccessor) {
162
- var propIndex = parents.findIndex(
163
- (x) => x.type == "MethodDefinition" || x.type == "Property"
164
- );
165
-
166
- // Don't change constructor!
167
- if (propIndex !== -1) {
168
- if (
169
- parents[propIndex].type == "MethodDefinition" &&
170
- parents[propIndex].kind == "constructor"
171
- ) {
172
- return;
173
- }
174
- }
175
-
176
- this.replace(object, Literal(object.name));
177
- parents[0].computed = true;
178
- parents[0].shorthand = false;
179
- }
180
- }
181
-
182
- /**
183
- * Transforms VariableDeclaration into single declarations.
184
- */
185
- transformExplicitDeclarations(object: Node, parents: Node[]) {
186
- // for ( var x in ... ) {...}
187
- var forIndex = parents.findIndex(
188
- (x) => x.type == "ForInStatement" || x.type == "ForOfStatement"
189
- );
190
- if (
191
- forIndex != -1 &&
192
- parents[forIndex].left == (parents[forIndex - 1] || object)
193
- ) {
194
- object.declarations.forEach((x) => {
195
- x.init = null;
196
- });
197
- return;
198
- }
199
-
200
- var body = parents[0];
201
- if (isLoop(body) || body.type == "LabeledStatement") {
202
- return;
203
- }
204
-
205
- if (body.type == "ExportNamedDeclaration") {
206
- return;
207
- }
208
-
209
- if (!Array.isArray(body)) {
210
- this.error(new Error("body is " + body.type));
211
- }
212
-
213
- if (object.declarations.length > 1) {
214
- // Make singular
215
-
216
- var index = body.indexOf(object);
217
- if (index == -1) {
218
- this.error(new Error("index is -1"));
219
- }
220
-
221
- var after = object.declarations.slice(1);
222
-
223
- body.splice(
224
- index + 1,
225
- 0,
226
- ...after.map((x) => {
227
- return {
228
- type: "VariableDeclaration",
229
- declarations: [clone(x)],
230
- kind: object.kind,
231
- };
232
- })
233
- );
234
-
235
- object.declarations.length = 1;
236
- }
237
- }
238
- }
171
+
172
+ var definedArgLength = path.get("params").length;
173
+ if (predictable && definedArgLength >= maxArgLength) {
174
+ (path.node as NodeSymbol)[PREDICTABLE] = true;
175
+ }
176
+ },
177
+ },
178
+
179
+ // console.log() -> console["log"]();
180
+ MemberExpression: {
181
+ exit(path) {
182
+ if (!path.node.computed && path.node.property.type === "Identifier") {
183
+ path.node.property = t.stringLiteral(path.node.property.name);
184
+ path.node.computed = true;
185
+ }
186
+ },
187
+ },
188
+
189
+ // { key: true } -> { "key": true }
190
+ "Property|Method": {
191
+ exit(_path) {
192
+ let path = _path as NodePath<t.Property | t.Method>;
193
+
194
+ if (t.isClassPrivateProperty(path.node)) return;
195
+
196
+ if (!path.node.computed && path.node.key.type === "Identifier") {
197
+ // Don't change constructor key
198
+ if (t.isClassMethod(path.node) && path.node.kind === "constructor")
199
+ return;
200
+
201
+ path.node.key = t.stringLiteral(path.node.key.name);
202
+ path.node.computed = true;
203
+ }
204
+ },
205
+ },
206
+
207
+ // var a,b,c -> var a; var b; var c;
208
+ VariableDeclaration: {
209
+ exit(path) {
210
+ if (path.node.declarations.length > 1) {
211
+ // E.g. for (var i = 0, j = 1;;)
212
+ if (path.key === "init" && path.parentPath.isForStatement()) {
213
+ if (
214
+ !path.parentPath.node.test &&
215
+ !path.parentPath.node.update &&
216
+ path.node.kind === "var"
217
+ ) {
218
+ path.parentPath.insertBefore(
219
+ path.node.declarations.map((declaration) =>
220
+ t.variableDeclaration(path.node.kind, [declaration])
221
+ )
222
+ );
223
+ path.remove();
224
+ }
225
+ } else {
226
+ if (path.parentPath.isExportNamedDeclaration()) {
227
+ path.parentPath.replaceWithMultiple(
228
+ path.node.declarations.map((declaration) =>
229
+ t.exportNamedDeclaration(
230
+ t.variableDeclaration(path.node.kind, [declaration])
231
+ )
232
+ )
233
+ );
234
+ } else {
235
+ path
236
+ .replaceWithMultiple(
237
+ path.node.declarations.map((declaration, i) => {
238
+ var names = Array.from(
239
+ getPatternIdentifierNames(path.get("declarations")[i])
240
+ );
241
+ names.forEach((name) => {
242
+ path.scope.removeBinding(name);
243
+ });
244
+
245
+ var newNode = t.variableDeclaration(path.node.kind, [
246
+ declaration,
247
+ ]);
248
+ return newNode;
249
+ })
250
+ )
251
+ .forEach((newPath) => {
252
+ if (newPath.node.kind === "var") {
253
+ var functionOrProgram =
254
+ getParentFunctionOrProgram(newPath);
255
+ functionOrProgram.scope.registerDeclaration(newPath);
256
+ }
257
+ newPath.scope.registerDeclaration(newPath);
258
+ });
259
+ }
260
+ }
261
+ }
262
+ },
263
+ },
264
+
265
+ // () => a() -> () => { return a(); }
266
+ ArrowFunctionExpression: {
267
+ exit(path: NodePath<t.ArrowFunctionExpression>) {
268
+ if (path.node.body.type !== "BlockStatement") {
269
+ path.node.expression = false;
270
+ path.node.body = t.blockStatement([
271
+ t.returnStatement(path.node.body),
272
+ ]);
273
+ }
274
+ },
275
+ },
276
+
277
+ // if (a) b() -> if (a) { b(); }
278
+ // if (a) {b()} else c() -> if (a) { b(); } else { c(); }
279
+ IfStatement: {
280
+ exit(path) {
281
+ if (path.node.consequent.type !== "BlockStatement") {
282
+ path.node.consequent = t.blockStatement([path.node.consequent]);
283
+ }
284
+
285
+ if (
286
+ path.node.alternate &&
287
+ path.node.alternate.type !== "BlockStatement"
288
+ ) {
289
+ path.node.alternate = t.blockStatement([path.node.alternate]);
290
+ }
291
+ },
292
+ },
293
+
294
+ // for() d() -> for() { d(); }
295
+ // while(a) b() -> while(a) { b(); }
296
+ // with(a) b() -> with(a) { b(); }
297
+ "ForStatement|ForInStatement|ForOfStatement|WhileStatement|WithStatement":
298
+ {
299
+ exit(_path) {
300
+ var path = _path as NodePath<
301
+ | t.ForStatement
302
+ | t.ForInStatement
303
+ | t.ForOfStatement
304
+ | t.WhileStatement
305
+ | t.WithStatement
306
+ >;
307
+
308
+ if (path.node.body.type !== "BlockStatement") {
309
+ path.node.body = t.blockStatement([path.node.body]);
310
+ }
311
+ },
312
+ },
313
+
314
+ // function a(param = ()=>b)
315
+ // _getB = ()=> ()=>b
316
+ // function a(param = _getB())
317
+ // Basically Babel scope.rename misses this edge case, so we need to manually handle it
318
+ // Here were essentially making the variables easier to understand
319
+ Function: {
320
+ exit(path) {
321
+ for (var param of path.get("params")) {
322
+ param.traverse({
323
+ "FunctionExpression|ArrowFunctionExpression"(_innerPath) {
324
+ let innerPath = _innerPath as NodePath<
325
+ t.FunctionExpression | t.ArrowFunctionExpression
326
+ >;
327
+ const child = innerPath.find((path) =>
328
+ path.parentPath?.isAssignmentPattern()
329
+ );
330
+
331
+ if (!child) return;
332
+
333
+ if (
334
+ t.isAssignmentPattern(child.parent) &&
335
+ child.parent.right === child.node
336
+ ) {
337
+ var creatorName = me.getPlaceholder();
338
+ var insertPath = path.insertBefore(
339
+ t.variableDeclaration("const", [
340
+ t.variableDeclarator(
341
+ t.identifier(creatorName),
342
+ t.arrowFunctionExpression([], innerPath.node, false)
343
+ ),
344
+ ])
345
+ )[0];
346
+
347
+ path.scope.parent.registerDeclaration(insertPath);
348
+
349
+ innerPath.replaceWith(
350
+ t.callExpression(t.identifier(creatorName), [])
351
+ );
352
+ }
353
+ },
354
+ });
355
+ }
356
+ },
357
+ },
358
+ },
359
+ };
360
+ };