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,153 +1,239 @@
1
- import Transform from "../transform";
2
- import { isBlock } from "../../traverse";
1
+ import { NodePath } from "@babel/core";
2
+ import { Order } from "../../order";
3
+ import { PluginArg, PluginObject } from "../plugin";
4
+ import { NodeSymbol, PREDICTABLE } from "../../constants";
5
+ import * as t from "@babel/types";
6
+ import { isStaticValue } from "../../utils/static-utils";
3
7
  import {
4
- ExpressionStatement,
5
- AssignmentExpression,
6
- Identifier,
7
- Node,
8
- VariableDeclarator,
9
- AssignmentPattern,
10
- } from "../../util/gen";
11
- import {
12
- isForInitialize,
13
- isFunction,
14
- isStrictModeFunction,
8
+ getPatternIdentifierNames,
9
+ isStrictMode,
15
10
  prepend,
16
- } from "../../util/insert";
17
- import { ok } from "assert";
18
- import { ObfuscateOrder } from "../../order";
19
- import { choice } from "../../util/random";
20
- import { predictableFunctionTag } from "../../constants";
21
- import { isIndependent, isMoveable } from "../../util/compare";
22
- import { getFunctionParameters } from "../../util/identifiers";
23
- import { isLexicalScope } from "../../util/scope";
11
+ } from "../../utils/ast-utils";
12
+ import Template from "../../templates/template";
24
13
 
25
14
  /**
26
- * Defines all the names at the top of every lexical block.
15
+ * Moved Declarations moves variables in two ways:
16
+ *
17
+ * 1) Move variables to top of the current block
18
+ * 2) Move variables as unused function parameters
27
19
  */
28
- export default class MovedDeclarations extends Transform {
29
- constructor(o) {
30
- super(o, ObfuscateOrder.MovedDeclarations);
31
- }
20
+ export default ({ Plugin }: PluginArg): PluginObject => {
21
+ const me = Plugin(Order.MovedDeclarations, {
22
+ changeData: {
23
+ variableDeclarations: 0,
24
+ functionParameters: 0,
25
+ },
26
+ });
27
+
28
+ function isFunctionEligibleForParameterPacking(
29
+ functionPath: NodePath<t.Function>,
30
+ proposedParameterName: string
31
+ ) {
32
+ // Getter/setter functions must have zero or one formal parameter
33
+ // We cannot add extra parameters to them
34
+ if (functionPath.isObjectMethod() || functionPath.isClassMethod()) {
35
+ if (functionPath.node.kind !== "method") {
36
+ return false;
37
+ }
38
+ }
39
+
40
+ // Rest params check
41
+ if (functionPath.get("params").find((p) => p.isRestElement())) return false;
42
+
43
+ // Max 1,000 parameters
44
+ if (functionPath.get("params").length > 1_000) return false;
32
45
 
33
- match(object, parents) {
34
- return (
35
- object.type === "VariableDeclaration" &&
36
- object.kind === "var" &&
37
- object.declarations.length === 1 &&
38
- object.declarations[0].id.type === "Identifier"
46
+ // Check for duplicate parameter names
47
+ var bindingIdentifiers = getPatternIdentifierNames(
48
+ functionPath.get("params")
39
49
  );
50
+
51
+ // Duplicate parameter name not allowed
52
+ if (bindingIdentifiers.has(proposedParameterName)) return false;
53
+
54
+ return true;
40
55
  }
41
56
 
42
- transform(object: Node, parents: Node[]) {
43
- return () => {
44
- var forInitializeType = isForInitialize(object, parents);
45
-
46
- // Get the block statement or Program node
47
- var blockIndex = parents.findIndex((x) => isLexicalScope(x));
48
- var block = parents[blockIndex];
49
- var body: Node[] =
50
- block.type === "SwitchCase" ? block.consequent : block.body;
51
- ok(Array.isArray(body), "No body array found.");
52
-
53
- var bodyObject = parents[blockIndex - 2] || object;
54
- var index = body.indexOf(bodyObject);
55
-
56
- var varName = object.declarations[0].id.name;
57
- ok(typeof varName === "string");
58
-
59
- var predictableFunctionIndex = parents.findIndex((x) => isFunction(x));
60
- var predictableFunction = parents[predictableFunctionIndex];
61
-
62
- var deleteStatement = false;
63
-
64
- if (
65
- predictableFunction &&
66
- ((predictableFunction.id &&
67
- predictableFunction.id.name.includes(predictableFunctionTag)) ||
68
- predictableFunction[predictableFunctionTag]) && // Must have predictableFunctionTag in the name, or on object
69
- predictableFunction[predictableFunctionTag] !== false && // If === false, the function is deemed not predictable
70
- predictableFunction.params.length < 1000 && // Max 1,000 parameters
71
- !predictableFunction.params.find((x) => x.type === "RestElement") && // Cannot add parameters after spread operator
72
- !(
73
- ["Property", "MethodDefinition"].includes(
74
- parents[predictableFunctionIndex + 1]?.type
75
- ) && parents[predictableFunctionIndex + 1]?.kind !== "init"
76
- ) && // Preserve getter/setter methods
77
- !getFunctionParameters(
78
- predictableFunction,
79
- parents.slice(predictableFunctionIndex)
80
- ).find((entry) => entry[0].name === varName) // Ensure not duplicate param name
81
- ) {
82
- // Use function f(..., x, y, z) to declare name
83
-
84
- var value = object.declarations[0].init;
85
- var isPredictablyComputed =
86
- predictableFunction.body === block &&
87
- !isStrictModeFunction(predictableFunction) &&
88
- value &&
89
- isIndependent(value, []) &&
90
- isMoveable(value, [object.declarations[0], object, ...parents]);
91
-
92
- var defineWithValue = isPredictablyComputed;
93
-
94
- if (defineWithValue) {
95
- predictableFunction.params.push(
96
- AssignmentPattern(Identifier(varName), value)
57
+ return {
58
+ visitor: {
59
+ FunctionDeclaration: {
60
+ exit(path) {
61
+ var functionPath = path.findParent((path) =>
62
+ path.isFunction()
63
+ ) as NodePath<t.Function>;
64
+
65
+ if (!functionPath || !(functionPath.node as NodeSymbol)[PREDICTABLE])
66
+ return;
67
+
68
+ // Must be direct child of the function
69
+ if (path.parentPath !== functionPath.get("body")) return;
70
+
71
+ const functionName = path.node.id.name;
72
+
73
+ // Must be eligible for parameter packing
74
+ if (
75
+ !isFunctionEligibleForParameterPacking(functionPath, functionName)
76
+ )
77
+ return;
78
+
79
+ var strictMode = isStrictMode(functionPath);
80
+
81
+ // Default parameters are not allowed when 'use strict' is declared
82
+ if (strictMode) return;
83
+
84
+ var functionExpression = path.node as t.Node as t.FunctionExpression;
85
+ functionExpression.type = "FunctionExpression";
86
+ functionExpression.id = null;
87
+
88
+ var identifier = t.identifier(functionName);
89
+ functionPath.node.params.push(identifier);
90
+
91
+ var paramPath = functionPath.get("params").at(-1);
92
+
93
+ // Update binding to point to new path
94
+ const binding = functionPath.scope.getBinding(functionName);
95
+ if (binding) {
96
+ binding.kind = "param";
97
+ binding.path = paramPath;
98
+ binding.identifier = identifier;
99
+ }
100
+
101
+ prepend(
102
+ functionPath,
103
+ new Template(`
104
+ if(!${functionName}) {
105
+ ${functionName} = {functionExpression};
106
+ }
107
+ `).single({ functionExpression: functionExpression })
97
108
  );
98
- object.declarations[0].init = null;
99
- deleteStatement = true;
100
- } else {
101
- predictableFunction.params.push(Identifier(varName));
102
- }
103
- } else {
104
- // Use 'var x, y, z' to declare name
105
-
106
- // Make sure in the block statement, and not already at the top of it
107
- if (index === -1 || index === 0) return;
108
-
109
- var topVariableDeclaration;
110
- if (body[0].type === "VariableDeclaration" && body[0].kind === "var") {
111
- topVariableDeclaration = body[0];
112
- } else {
113
- topVariableDeclaration = {
114
- type: "VariableDeclaration",
115
- declarations: [],
116
- kind: "var",
117
- };
118
-
119
- prepend(block, topVariableDeclaration);
120
- }
121
-
122
- // Add `var x` at the top of the block
123
- topVariableDeclaration.declarations.push(
124
- VariableDeclarator(Identifier(varName))
125
- );
126
- }
127
109
 
128
- var assignmentExpression = AssignmentExpression(
129
- "=",
130
- Identifier(varName),
131
- object.declarations[0].init || Identifier(varName)
132
- );
133
-
134
- if (forInitializeType) {
135
- if (forInitializeType === "initializer") {
136
- // Replace `for (var i = 0...)` to `for (i = 0...)`
137
- this.replace(object, assignmentExpression);
138
- } else if (forInitializeType === "left-hand") {
139
- // Replace `for (var k in...)` to `for (k in ...)`
140
-
141
- this.replace(object, Identifier(varName));
142
- }
143
- } else {
144
- if (deleteStatement && index !== -1) {
145
- body.splice(index, 1);
146
- } else {
147
- // Replace `var x = value` to `x = value`
148
- this.replace(object, ExpressionStatement(assignmentExpression));
149
- }
150
- }
151
- };
152
- }
153
- }
110
+ path.remove();
111
+ me.changeData.functionParameters++;
112
+ },
113
+ },
114
+ VariableDeclaration: {
115
+ exit(path) {
116
+ if (path.node.kind !== "var") return;
117
+ if (path.node.declarations.length !== 1) return;
118
+
119
+ var insertionMethod = "variableDeclaration";
120
+ var functionPath = path.findParent((path) =>
121
+ path.isFunction()
122
+ ) as NodePath<t.Function>;
123
+
124
+ const declaration = path.node.declarations[0];
125
+ if (!t.isIdentifier(declaration.id)) return;
126
+ const varName = declaration.id.name;
127
+
128
+ var allowDefaultParamValue = true;
129
+
130
+ if (functionPath && (functionPath.node as NodeSymbol)[PREDICTABLE]) {
131
+ // Check for "use strict" directive
132
+ // Strict mode disallows non-simple parameters
133
+ // So we can't move the declaration to the function parameters
134
+ var strictMode = isStrictMode(functionPath);
135
+ if (strictMode) {
136
+ allowDefaultParamValue = false;
137
+ }
138
+
139
+ // Cannot add variables after rest element
140
+ // Cannot add over 1,000 parameters
141
+ if (isFunctionEligibleForParameterPacking(functionPath, varName)) {
142
+ insertionMethod = "functionParameter";
143
+ }
144
+ }
145
+
146
+ const { name } = declaration.id;
147
+ const value = declaration.init || t.identifier("undefined");
148
+
149
+ const isStatic = isStaticValue(value);
150
+ let isDefinedAtTop = false;
151
+ const parentPath = path.parentPath;
152
+ if (parentPath.isBlock()) {
153
+ isDefinedAtTop = parentPath.get("body").indexOf(path) === 0;
154
+ }
155
+
156
+ // Already at the top - nothing will change
157
+ if (insertionMethod === "variableDeclaration" && isDefinedAtTop) {
158
+ return;
159
+ }
160
+
161
+ let defaultParamValue: t.Expression;
162
+
163
+ if (
164
+ insertionMethod === "functionParameter" &&
165
+ isStatic &&
166
+ isDefinedAtTop &&
167
+ allowDefaultParamValue
168
+ ) {
169
+ defaultParamValue = value;
170
+ path.remove();
171
+ } else {
172
+ // For-in / For-of can only reference the variable name
173
+ if (
174
+ parentPath.isForInStatement() ||
175
+ parentPath.isForOfStatement()
176
+ ) {
177
+ path.replaceWith(t.identifier(name));
178
+ } else {
179
+ path.replaceWith(
180
+ t.assignmentExpression(
181
+ "=",
182
+ t.identifier(name),
183
+ declaration.init || t.identifier("undefined")
184
+ )
185
+ );
186
+ }
187
+ }
188
+
189
+ switch (insertionMethod) {
190
+ case "functionParameter":
191
+ var identifier = t.identifier(name);
192
+
193
+ var param: t.Pattern | t.Identifier = identifier;
194
+ if (allowDefaultParamValue && defaultParamValue) {
195
+ param = t.assignmentPattern(param, defaultParamValue);
196
+ }
197
+
198
+ functionPath.node.params.push(param);
199
+
200
+ var paramPath = functionPath.get("params").at(-1);
201
+
202
+ // Update binding to point to new path
203
+ const binding = functionPath.scope.getBinding(name);
204
+ if (binding) {
205
+ binding.kind = "param";
206
+ binding.path = paramPath;
207
+ binding.identifier = identifier;
208
+ }
209
+
210
+ me.changeData.functionParameters++;
211
+ break;
212
+ case "variableDeclaration":
213
+ var block = path.findParent((path) =>
214
+ path.isBlock()
215
+ ) as NodePath<t.Block>;
216
+
217
+ var topNode = block.node.body[0];
218
+ const variableDeclarator = t.variableDeclarator(
219
+ t.identifier(name)
220
+ );
221
+
222
+ if (t.isVariableDeclaration(topNode) && topNode.kind === "var") {
223
+ topNode.declarations.push(variableDeclarator);
224
+ break;
225
+ } else {
226
+ block.node.body.unshift(
227
+ t.variableDeclaration("var", [variableDeclarator])
228
+ );
229
+ }
230
+
231
+ me.changeData.variableDeclarations++;
232
+
233
+ break;
234
+ }
235
+ },
236
+ },
237
+ },
238
+ };
239
+ };