js-confuser 1.7.3 → 2.0.0-alpha.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (269) hide show
  1. package/.github/ISSUE_TEMPLATE/bug_report.md +6 -4
  2. package/CHANGELOG.md +70 -0
  3. package/Migration.md +57 -0
  4. package/README.md +23 -929
  5. package/dist/constants.js +65 -14
  6. package/dist/index.js +108 -160
  7. package/dist/obfuscator.js +316 -118
  8. package/dist/options.js +1 -119
  9. package/dist/order.js +30 -30
  10. package/dist/presets.js +47 -45
  11. package/dist/probability.js +25 -32
  12. package/dist/templates/bufferToStringTemplate.js +9 -0
  13. package/dist/templates/deadCodeTemplates.js +9 -0
  14. package/dist/templates/getGlobalTemplate.js +19 -0
  15. package/dist/templates/integrityTemplate.js +30 -0
  16. package/dist/templates/setFunctionLengthTemplate.js +9 -0
  17. package/dist/templates/stringCompressionTemplate.js +10 -0
  18. package/dist/templates/tamperProtectionTemplates.js +21 -0
  19. package/dist/templates/template.js +199 -184
  20. package/dist/transforms/astScrambler.js +100 -0
  21. package/dist/transforms/calculator.js +70 -127
  22. package/dist/transforms/controlFlowFlattening.js +1182 -0
  23. package/dist/transforms/deadCode.js +62 -587
  24. package/dist/transforms/dispatcher.js +300 -313
  25. package/dist/transforms/extraction/duplicateLiteralsRemoval.js +88 -189
  26. package/dist/transforms/extraction/objectExtraction.js +131 -215
  27. package/dist/transforms/finalizer.js +56 -59
  28. package/dist/transforms/flatten.js +275 -276
  29. package/dist/transforms/functionOutlining.js +230 -0
  30. package/dist/transforms/identifier/globalConcealing.js +214 -135
  31. package/dist/transforms/identifier/movedDeclarations.js +167 -91
  32. package/dist/transforms/identifier/renameVariables.js +239 -193
  33. package/dist/transforms/lock/integrity.js +61 -184
  34. package/dist/transforms/lock/lock.js +261 -387
  35. package/dist/transforms/minify.js +431 -436
  36. package/dist/transforms/opaquePredicates.js +65 -118
  37. package/dist/transforms/pack.js +160 -0
  38. package/dist/transforms/plugin.js +179 -0
  39. package/dist/transforms/preparation.js +261 -173
  40. package/dist/transforms/renameLabels.js +132 -56
  41. package/dist/transforms/rgf.js +140 -267
  42. package/dist/transforms/shuffle.js +52 -145
  43. package/dist/transforms/string/encoding.js +44 -175
  44. package/dist/transforms/string/stringCompression.js +79 -155
  45. package/dist/transforms/string/stringConcealing.js +189 -225
  46. package/dist/transforms/string/stringEncoding.js +32 -40
  47. package/dist/transforms/string/stringSplitting.js +54 -55
  48. package/dist/transforms/variableMasking.js +232 -0
  49. package/dist/utils/ControlObject.js +125 -0
  50. package/dist/utils/IntGen.js +46 -0
  51. package/dist/utils/NameGen.js +106 -0
  52. package/dist/utils/ast-utils.js +560 -0
  53. package/dist/utils/function-utils.js +56 -0
  54. package/dist/utils/gen-utils.js +48 -0
  55. package/dist/utils/node.js +77 -0
  56. package/dist/utils/object-utils.js +21 -0
  57. package/dist/utils/random-utils.js +91 -0
  58. package/dist/utils/static-utils.js +64 -0
  59. package/dist/validateOptions.js +122 -0
  60. package/index.d.ts +1 -17
  61. package/package.json +27 -22
  62. package/src/constants.ts +139 -82
  63. package/src/index.ts +70 -165
  64. package/src/obfuscationResult.ts +43 -0
  65. package/src/obfuscator.ts +328 -135
  66. package/src/options.ts +149 -658
  67. package/src/order.ts +14 -14
  68. package/src/presets.ts +39 -34
  69. package/src/probability.ts +21 -36
  70. package/src/templates/bufferToStringTemplate.ts +57 -0
  71. package/src/templates/deadCodeTemplates.ts +1185 -0
  72. package/src/templates/getGlobalTemplate.ts +72 -0
  73. package/src/templates/integrityTemplate.ts +69 -0
  74. package/src/templates/setFunctionLengthTemplate.ts +11 -0
  75. package/src/templates/stringCompressionTemplate.ts +42 -0
  76. package/src/templates/tamperProtectionTemplates.ts +116 -0
  77. package/src/templates/template.ts +149 -157
  78. package/src/transforms/astScrambler.ts +99 -0
  79. package/src/transforms/calculator.ts +96 -226
  80. package/src/transforms/controlFlowFlattening.ts +1594 -0
  81. package/src/transforms/deadCode.ts +85 -676
  82. package/src/transforms/dispatcher.ts +431 -640
  83. package/src/transforms/extraction/duplicateLiteralsRemoval.ts +147 -295
  84. package/src/transforms/extraction/objectExtraction.ts +160 -333
  85. package/src/transforms/finalizer.ts +63 -64
  86. package/src/transforms/flatten.ts +439 -557
  87. package/src/transforms/functionOutlining.ts +225 -0
  88. package/src/transforms/identifier/globalConcealing.ts +255 -266
  89. package/src/transforms/identifier/movedDeclarations.ts +228 -142
  90. package/src/transforms/identifier/renameVariables.ts +250 -271
  91. package/src/transforms/lock/integrity.ts +85 -263
  92. package/src/transforms/lock/lock.ts +338 -579
  93. package/src/transforms/minify.ts +523 -663
  94. package/src/transforms/opaquePredicates.ts +90 -229
  95. package/src/transforms/pack.ts +195 -0
  96. package/src/transforms/plugin.ts +185 -0
  97. package/src/transforms/preparation.ts +337 -231
  98. package/src/transforms/renameLabels.ts +176 -77
  99. package/src/transforms/rgf.ts +293 -424
  100. package/src/transforms/shuffle.ts +80 -254
  101. package/src/transforms/string/encoding.ts +20 -126
  102. package/src/transforms/string/stringCompression.ts +117 -307
  103. package/src/transforms/string/stringConcealing.ts +254 -342
  104. package/src/transforms/string/stringEncoding.ts +28 -47
  105. package/src/transforms/string/stringSplitting.ts +61 -75
  106. package/src/transforms/variableMasking.ts +257 -0
  107. package/src/utils/ControlObject.ts +141 -0
  108. package/src/utils/IntGen.ts +33 -0
  109. package/src/utils/NameGen.ts +106 -0
  110. package/src/utils/ast-utils.ts +667 -0
  111. package/src/utils/function-utils.ts +50 -0
  112. package/src/utils/gen-utils.ts +48 -0
  113. package/src/utils/node.ts +78 -0
  114. package/src/utils/object-utils.ts +21 -0
  115. package/src/utils/random-utils.ts +79 -0
  116. package/src/utils/static-utils.ts +66 -0
  117. package/src/validateOptions.ts +256 -0
  118. package/tsconfig.json +13 -8
  119. package/babel.config.js +0 -12
  120. package/dev.js +0 -8
  121. package/dist/compiler.js +0 -34
  122. package/dist/parser.js +0 -59
  123. package/dist/precedence.js +0 -66
  124. package/dist/templates/bufferToString.js +0 -129
  125. package/dist/templates/core.js +0 -35
  126. package/dist/templates/crash.js +0 -28
  127. package/dist/templates/es5.js +0 -137
  128. package/dist/templates/functionLength.js +0 -34
  129. package/dist/templates/globals.js +0 -9
  130. package/dist/transforms/antiTooling.js +0 -88
  131. package/dist/transforms/controlFlowFlattening/controlFlowFlattening.js +0 -1287
  132. package/dist/transforms/controlFlowFlattening/expressionObfuscation.js +0 -131
  133. package/dist/transforms/es5/antiClass.js +0 -164
  134. package/dist/transforms/es5/antiDestructuring.js +0 -193
  135. package/dist/transforms/es5/antiES6Object.js +0 -185
  136. package/dist/transforms/es5/antiSpreadOperator.js +0 -35
  137. package/dist/transforms/es5/antiTemplate.js +0 -66
  138. package/dist/transforms/es5/es5.js +0 -123
  139. package/dist/transforms/extraction/classExtraction.js +0 -83
  140. package/dist/transforms/identifier/globalAnalysis.js +0 -83
  141. package/dist/transforms/identifier/variableAnalysis.js +0 -104
  142. package/dist/transforms/lock/antiDebug.js +0 -76
  143. package/dist/transforms/stack.js +0 -349
  144. package/dist/transforms/transform.js +0 -372
  145. package/dist/traverse.js +0 -110
  146. package/dist/util/compare.js +0 -145
  147. package/dist/util/gen.js +0 -564
  148. package/dist/util/guard.js +0 -14
  149. package/dist/util/identifiers.js +0 -355
  150. package/dist/util/insert.js +0 -362
  151. package/dist/util/math.js +0 -19
  152. package/dist/util/object.js +0 -40
  153. package/dist/util/random.js +0 -156
  154. package/dist/util/scope.js +0 -20
  155. package/docs/ControlFlowFlattening.md +0 -595
  156. package/docs/Countermeasures.md +0 -70
  157. package/docs/ES5.md +0 -197
  158. package/docs/Integrity.md +0 -82
  159. package/docs/RGF.md +0 -424
  160. package/docs/RenameVariables.md +0 -116
  161. package/docs/TamperProtection.md +0 -100
  162. package/docs/Template.md +0 -117
  163. package/samples/example.js +0 -15
  164. package/samples/high.js +0 -1
  165. package/samples/input.js +0 -3
  166. package/samples/javascriptobfuscator.com.js +0 -8
  167. package/samples/jscrambler_advanced.js +0 -1894
  168. package/samples/jscrambler_light.js +0 -1134
  169. package/samples/low.js +0 -1
  170. package/samples/medium.js +0 -1
  171. package/samples/obfuscator.io.js +0 -1686
  172. package/samples/preemptive.com.js +0 -16
  173. package/src/compiler.ts +0 -35
  174. package/src/parser.ts +0 -49
  175. package/src/precedence.ts +0 -61
  176. package/src/templates/bufferToString.ts +0 -136
  177. package/src/templates/core.ts +0 -29
  178. package/src/templates/crash.ts +0 -23
  179. package/src/templates/es5.ts +0 -131
  180. package/src/templates/functionLength.ts +0 -32
  181. package/src/templates/globals.ts +0 -3
  182. package/src/transforms/antiTooling.ts +0 -102
  183. package/src/transforms/controlFlowFlattening/controlFlowFlattening.ts +0 -2153
  184. package/src/transforms/controlFlowFlattening/expressionObfuscation.ts +0 -179
  185. package/src/transforms/es5/antiClass.ts +0 -276
  186. package/src/transforms/es5/antiDestructuring.ts +0 -294
  187. package/src/transforms/es5/antiES6Object.ts +0 -267
  188. package/src/transforms/es5/antiSpreadOperator.ts +0 -56
  189. package/src/transforms/es5/antiTemplate.ts +0 -98
  190. package/src/transforms/es5/es5.ts +0 -149
  191. package/src/transforms/extraction/classExtraction.ts +0 -168
  192. package/src/transforms/identifier/globalAnalysis.ts +0 -102
  193. package/src/transforms/identifier/variableAnalysis.ts +0 -118
  194. package/src/transforms/lock/antiDebug.ts +0 -112
  195. package/src/transforms/stack.ts +0 -557
  196. package/src/transforms/transform.ts +0 -441
  197. package/src/traverse.ts +0 -120
  198. package/src/types.ts +0 -133
  199. package/src/util/compare.ts +0 -181
  200. package/src/util/gen.ts +0 -651
  201. package/src/util/guard.ts +0 -17
  202. package/src/util/identifiers.ts +0 -494
  203. package/src/util/insert.ts +0 -419
  204. package/src/util/math.ts +0 -15
  205. package/src/util/object.ts +0 -39
  206. package/src/util/random.ts +0 -221
  207. package/src/util/scope.ts +0 -21
  208. package/test/code/Cash.src.js +0 -1011
  209. package/test/code/Cash.test.ts +0 -132
  210. package/test/code/Dynamic.src.js +0 -118
  211. package/test/code/Dynamic.test.ts +0 -49
  212. package/test/code/ES6.src.js +0 -235
  213. package/test/code/ES6.test.ts +0 -42
  214. package/test/code/NewFeatures.test.ts +0 -19
  215. package/test/code/StrictMode.src.js +0 -65
  216. package/test/code/StrictMode.test.js +0 -37
  217. package/test/compare.test.ts +0 -104
  218. package/test/index.test.ts +0 -249
  219. package/test/options.test.ts +0 -150
  220. package/test/presets.test.ts +0 -22
  221. package/test/probability.test.ts +0 -44
  222. package/test/templates/template.test.ts +0 -224
  223. package/test/transforms/antiTooling.test.ts +0 -52
  224. package/test/transforms/calculator.test.ts +0 -78
  225. package/test/transforms/controlFlowFlattening/controlFlowFlattening.test.ts +0 -1274
  226. package/test/transforms/controlFlowFlattening/expressionObfuscation.test.ts +0 -192
  227. package/test/transforms/deadCode.test.ts +0 -85
  228. package/test/transforms/dispatcher.test.ts +0 -457
  229. package/test/transforms/es5/antiClass.test.ts +0 -427
  230. package/test/transforms/es5/antiDestructuring.test.ts +0 -157
  231. package/test/transforms/es5/antiES6Object.test.ts +0 -245
  232. package/test/transforms/es5/antiTemplate.test.ts +0 -116
  233. package/test/transforms/es5/es5.test.ts +0 -110
  234. package/test/transforms/extraction/classExtraction.test.ts +0 -86
  235. package/test/transforms/extraction/duplicateLiteralsRemoval.test.ts +0 -200
  236. package/test/transforms/extraction/objectExtraction.test.ts +0 -491
  237. package/test/transforms/flatten.test.ts +0 -721
  238. package/test/transforms/hexadecimalNumbers.test.ts +0 -62
  239. package/test/transforms/identifier/globalConcealing.test.ts +0 -142
  240. package/test/transforms/identifier/movedDeclarations.test.ts +0 -275
  241. package/test/transforms/identifier/renameVariables.test.ts +0 -695
  242. package/test/transforms/lock/antiDebug.test.ts +0 -66
  243. package/test/transforms/lock/browserLock.test.ts +0 -129
  244. package/test/transforms/lock/countermeasures.test.ts +0 -100
  245. package/test/transforms/lock/integrity.test.ts +0 -161
  246. package/test/transforms/lock/lock.test.ts +0 -204
  247. package/test/transforms/lock/osLock.test.ts +0 -312
  248. package/test/transforms/lock/selfDefending.test.ts +0 -68
  249. package/test/transforms/lock/tamperProtection.test.ts +0 -336
  250. package/test/transforms/minify.test.ts +0 -575
  251. package/test/transforms/opaquePredicates.test.ts +0 -43
  252. package/test/transforms/preparation.test.ts +0 -157
  253. package/test/transforms/renameLabels.test.ts +0 -95
  254. package/test/transforms/rgf.test.ts +0 -378
  255. package/test/transforms/shuffle.test.ts +0 -135
  256. package/test/transforms/stack.test.ts +0 -573
  257. package/test/transforms/string/stringCompression.test.ts +0 -120
  258. package/test/transforms/string/stringConcealing.test.ts +0 -299
  259. package/test/transforms/string/stringEncoding.test.ts +0 -95
  260. package/test/transforms/string/stringSplitting.test.ts +0 -135
  261. package/test/transforms/transform.test.ts +0 -66
  262. package/test/traverse.test.ts +0 -139
  263. package/test/util/compare.test.ts +0 -34
  264. package/test/util/gen.test.ts +0 -121
  265. package/test/util/identifiers.test.ts +0 -253
  266. package/test/util/insert.test.ts +0 -142
  267. package/test/util/math.test.ts +0 -5
  268. package/test/util/random.test.ts +0 -71
  269. /package/dist/{types.js → obfuscationResult.js} +0 -0
@@ -1,8 +1,8 @@
1
- import Transform from "../transform";
2
- import { choice } from "../../util/random";
3
- import { isDirective, isModuleSource } from "../../util/compare";
4
- import { ComputeProbabilityMap } from "../../probability";
5
- import { Identifier } from "../../util/gen";
1
+ import { PluginInstance, PluginObject } from "../plugin";
2
+ import * as t from "@babel/types";
3
+ import { choice } from "../../utils/random-utils";
4
+ import { computeProbabilityMap } from "../../probability";
5
+ import { GEN_NODE, NodeSymbol } from "../../constants";
6
6
 
7
7
  function pad(x: string, len: number): string {
8
8
  while (x.length < len) {
@@ -46,50 +46,31 @@ function toUnicodeRepresentation(str: string) {
46
46
  return escapedString;
47
47
  }
48
48
 
49
- /**
50
- * [String Encoding](https://docs.jscrambler.com/code-integrity/documentation/transformations/string-encoding) transforms a string into an encoded representation.
51
- *
52
- * - Potency Low
53
- * - Resilience Low
54
- * - Cost Low
55
- */
56
- export default class StringEncoding extends Transform {
57
- constructor(o) {
58
- super(o);
59
- }
60
-
61
- match(object, parents) {
62
- return (
63
- object.type == "Literal" &&
64
- typeof object.value === "string" &&
65
- object.value.length > 0 &&
66
- !isModuleSource(object, parents) &&
67
- !isDirective(object, parents)
68
- );
69
- }
49
+ export default (me: PluginInstance): PluginObject => {
50
+ return {
51
+ visitor: {
52
+ StringLiteral: {
53
+ exit(path) {
54
+ const { value } = path.node;
70
55
 
71
- transform(object, parents) {
72
- // Allow percentages
73
- if (
74
- !ComputeProbabilityMap(
75
- this.options.stringEncoding,
76
- (x) => x,
77
- object.value
78
- )
79
- )
80
- return;
56
+ // Allow percentages
57
+ if (!computeProbabilityMap(me.options.stringEncoding, value)) return;
81
58
 
82
- var type = choice(["hexadecimal", "unicode"]);
59
+ var type = choice(["hexadecimal", "unicode"]);
83
60
 
84
- var escapedString = (
85
- type == "hexadecimal" ? toHexRepresentation : toUnicodeRepresentation
86
- )(object.value);
61
+ var escapedString = (
62
+ type == "hexadecimal"
63
+ ? toHexRepresentation
64
+ : toUnicodeRepresentation
65
+ )(value);
87
66
 
88
- return () => {
89
- if (object.type !== "Literal") return;
67
+ var id = t.identifier(`"${escapedString}"`);
90
68
 
91
- // ESCodeGen tries to escape backslashes, here is a work-around
92
- this.replace(object, Identifier(`'${escapedString}'`));
93
- };
94
- }
95
- }
69
+ (id as NodeSymbol)[GEN_NODE] = true;
70
+ path.replaceWith(id);
71
+ path.skip();
72
+ },
73
+ },
74
+ },
75
+ };
76
+ };
@@ -1,86 +1,72 @@
1
- import Transform from "../transform";
2
- import { Node, Literal, BinaryExpression } from "../../util/gen";
3
- import { clone } from "../../util/insert";
4
- import { getRandomInteger, shuffle, splitIntoChunks } from "../../util/random";
5
- import { ObfuscateOrder } from "../../order";
6
- import { isDirective, isModuleSource } from "../../util/compare";
1
+ import { PluginArg, PluginInstance, PluginObject } from "../plugin";
2
+ import { getRandomInteger, splitIntoChunks } from "../../utils/random-utils";
3
+ import { computeProbabilityMap } from "../../probability";
4
+ import { binaryExpression, stringLiteral } from "@babel/types";
7
5
  import { ok } from "assert";
8
- import { ComputeProbabilityMap } from "../../probability";
6
+ import { Order } from "../../order";
7
+ import { ensureComputedExpression } from "../../utils/ast-utils";
9
8
 
10
- export default class StringSplitting extends Transform {
11
- joinPrototype: string;
12
- strings: { [value: string]: string };
9
+ export default ({ Plugin }: PluginArg): PluginObject => {
10
+ const me = Plugin(Order.StringSplitting, {
11
+ changeData: {
12
+ strings: 0,
13
+ },
14
+ });
13
15
 
14
- adders: Node[][];
15
- vars: Node[];
16
+ return {
17
+ visitor: {
18
+ StringLiteral: {
19
+ exit(path) {
20
+ var object = path.node;
16
21
 
17
- constructor(o) {
18
- super(o, ObfuscateOrder.StringSplitting);
19
-
20
- this.joinPrototype = null;
21
- this.strings = Object.create(null);
22
-
23
- this.adders = [];
24
- this.vars = [];
25
- }
22
+ var size = Math.round(
23
+ Math.max(6, object.value.length / getRandomInteger(3, 8))
24
+ );
25
+ if (object.value.length <= size) {
26
+ return;
27
+ }
26
28
 
27
- match(object: Node, parents: Node[]) {
28
- return (
29
- object.type == "Literal" &&
30
- typeof object.value === "string" &&
31
- object.value.length >= 8 &&
32
- !isModuleSource(object, parents) &&
33
- !isDirective(object, parents)
34
- );
35
- }
29
+ var chunks = splitIntoChunks(object.value, size);
30
+ if (!chunks || chunks.length <= 1) {
31
+ return;
32
+ }
36
33
 
37
- transform(object: Node, parents: Node[]) {
38
- return () => {
39
- var size = Math.round(
40
- Math.max(6, object.value.length / getRandomInteger(3, 8))
41
- );
42
- if (object.value.length <= size) {
43
- return;
44
- }
34
+ if (
35
+ !computeProbabilityMap(me.options.stringSplitting, object.value)
36
+ ) {
37
+ return;
38
+ }
45
39
 
46
- var chunks = splitIntoChunks(object.value, size);
47
- if (!chunks || chunks.length <= 1) {
48
- return;
49
- }
40
+ var binExpr;
41
+ var parent;
42
+ var last = chunks.pop();
43
+ chunks.forEach((chunk, i) => {
44
+ if (i == 0) {
45
+ parent = binExpr = binaryExpression(
46
+ "+",
47
+ stringLiteral(chunk),
48
+ stringLiteral("")
49
+ );
50
+ } else {
51
+ binExpr.left = binaryExpression(
52
+ "+",
53
+ { ...binExpr.left },
54
+ stringLiteral(chunk)
55
+ );
56
+ ok(binExpr);
57
+ }
58
+ });
50
59
 
51
- if (
52
- !ComputeProbabilityMap(
53
- this.options.stringSplitting,
54
- (x) => x,
55
- object.value
56
- )
57
- ) {
58
- return;
59
- }
60
+ parent.right = stringLiteral(last);
60
61
 
61
- var binaryExpression;
62
- var parent;
63
- var last = chunks.pop();
64
- chunks.forEach((chunk, i) => {
65
- if (i == 0) {
66
- parent = binaryExpression = BinaryExpression(
67
- "+",
68
- Literal(chunk),
69
- Literal("")
70
- );
71
- } else {
72
- binaryExpression.left = BinaryExpression(
73
- "+",
74
- clone(binaryExpression.left),
75
- Literal(chunk)
76
- );
77
- ok(binaryExpression);
78
- }
79
- });
62
+ me.changeData.strings++;
80
63
 
81
- parent.right = Literal(last);
64
+ ensureComputedExpression(path);
82
65
 
83
- this.replaceIdentifierOrLiteral(object, parent, parents);
84
- };
85
- }
86
- }
66
+ path.replaceWith(parent);
67
+ path.skip();
68
+ },
69
+ },
70
+ },
71
+ };
72
+ };
@@ -0,0 +1,257 @@
1
+ import { Binding, NodePath } from "@babel/traverse";
2
+ import { PluginArg, PluginObject } from "./plugin";
3
+ import * as t from "@babel/types";
4
+ import Template from "../templates/template";
5
+ import { computeProbabilityMap } from "../probability";
6
+ import { Order } from "../order";
7
+ import {
8
+ NodeSymbol,
9
+ PREDICTABLE,
10
+ reservedIdentifiers,
11
+ UNSAFE,
12
+ variableFunctionName,
13
+ } from "../constants";
14
+ import {
15
+ ensureComputedExpression,
16
+ getFunctionName,
17
+ isDefiningIdentifier,
18
+ isStrictMode,
19
+ isVariableIdentifier,
20
+ prepend,
21
+ replaceDefiningIdentifierToMemberExpression,
22
+ } from "../utils/ast-utils";
23
+ import {
24
+ computeFunctionLength,
25
+ isVariableFunctionIdentifier,
26
+ } from "../utils/function-utils";
27
+ import { ok } from "assert";
28
+ import { NameGen } from "../utils/NameGen";
29
+ import { choice, getRandomInteger } from "../utils/random-utils";
30
+ import { createLiteral } from "../utils/node";
31
+
32
+ export default ({ Plugin }: PluginArg): PluginObject => {
33
+ const me = Plugin(Order.VariableMasking, {
34
+ changeData: {
35
+ functions: 0,
36
+ },
37
+ });
38
+
39
+ const transformFunction = (fnPath: NodePath<t.Function>) => {
40
+ // Do not apply to getter/setter methods
41
+ if (fnPath.isObjectMethod() && fnPath.node.kind !== "method") {
42
+ return;
43
+ }
44
+
45
+ // Do not apply to class getters/setters
46
+ if (fnPath.isClassMethod() && fnPath.node.kind !== "method") {
47
+ return;
48
+ }
49
+
50
+ // Do not apply to async or generator functions
51
+ if (fnPath.node.generator || fnPath.node.async) {
52
+ return;
53
+ }
54
+
55
+ // Do not apply to functions with rest parameters or destructuring
56
+ if (fnPath.node.params.some((param) => !t.isIdentifier(param))) {
57
+ return;
58
+ }
59
+
60
+ // Do not apply to 'use strict' functions
61
+ if (isStrictMode(fnPath)) return;
62
+
63
+ // Do not apply to functions marked unsafe
64
+ if ((fnPath.node as NodeSymbol)[UNSAFE]) return;
65
+
66
+ const functionName = getFunctionName(fnPath);
67
+
68
+ if (!computeProbabilityMap(me.options.variableMasking, functionName)) {
69
+ return;
70
+ }
71
+
72
+ const stackName = me.getPlaceholder() + "_varMask";
73
+ const stackMap = new Map<Binding, number | string>();
74
+ const propertyGen = new NameGen("mangled");
75
+ const stackKeys = new Set<string>();
76
+ let needsStack = false;
77
+
78
+ const illegalBindings = new Set<Binding>();
79
+
80
+ function checkBinding(binding: Binding) {
81
+ // Custom illegal check
82
+ // Variable Declarations with more than one declarator are not supported
83
+ // They can be inserted from the user's code even though Preparation phase should prevent it
84
+ // String Compression library includes such code
85
+ // TODO: Support multiple declarators
86
+ var variableDeclaration = binding.path.find((p) =>
87
+ p.isVariableDeclaration()
88
+ ) as NodePath<t.VariableDeclaration>;
89
+ if (
90
+ variableDeclaration &&
91
+ variableDeclaration.node.declarations.length > 1
92
+ ) {
93
+ return false;
94
+ }
95
+
96
+ function checkForUnsafe(valuePath: NodePath) {
97
+ var hasUnsafeNode = false;
98
+
99
+ valuePath.traverse({
100
+ ThisExpression(path) {
101
+ hasUnsafeNode = true;
102
+ path.stop();
103
+ },
104
+ Function(path) {
105
+ if ((path.node as NodeSymbol)[UNSAFE]) {
106
+ hasUnsafeNode = true;
107
+ path.stop();
108
+ }
109
+ },
110
+ });
111
+
112
+ return hasUnsafeNode;
113
+ }
114
+
115
+ // Check function value for 'this'
116
+ // Adding function expression to the stack (member expression)
117
+ // would break the 'this' context
118
+ if (binding.path.isVariableDeclarator()) {
119
+ let init = binding.path.get("init");
120
+ if (init.node) {
121
+ if (checkForUnsafe(init)) return false;
122
+ }
123
+ }
124
+
125
+ // x = function(){ return this }
126
+ // Cannot be transformed to x = stack[0] as 'this' would change
127
+ for (var assignment of binding.constantViolations) {
128
+ if (checkForUnsafe(assignment)) return false;
129
+ }
130
+
131
+ // __JS_CONFUSER_VAR__(identifier) -> __JS_CONFUSER_VAR__(stack.identifier)
132
+ // This cannot be transformed as it would break the user's code
133
+ for (var referencePath of binding.referencePaths) {
134
+ if (isVariableFunctionIdentifier(referencePath)) {
135
+ return false;
136
+ }
137
+ }
138
+
139
+ return true;
140
+ }
141
+
142
+ for (const param of fnPath.get("params")) {
143
+ ok(param.isIdentifier());
144
+
145
+ const paramName = param.node.name;
146
+ const binding = param.scope.getBinding(paramName);
147
+
148
+ if (!binding || !checkBinding(binding)) return;
149
+
150
+ ok(!stackMap.has(binding));
151
+ stackKeys.add(stackMap.size.toString());
152
+ stackMap.set(binding, stackMap.size);
153
+ }
154
+
155
+ fnPath.traverse({
156
+ Identifier(path) {
157
+ if (!isVariableIdentifier(path)) return;
158
+
159
+ if (reservedIdentifiers.has(path.node.name)) return;
160
+ if (me.options.globalVariables.has(path.node.name)) return;
161
+ if (path.node.name === stackName) return;
162
+ if (path.node.name === variableFunctionName) return;
163
+
164
+ const binding = path.scope.getBinding(path.node.name);
165
+ if (!binding || binding.scope !== fnPath.scope) return;
166
+ if (illegalBindings.has(binding)) return;
167
+
168
+ needsStack = true;
169
+
170
+ let index = stackMap.get(binding);
171
+ if (typeof index === "undefined") {
172
+ // Only transform var and let bindings
173
+ // Function declarations could be hoisted and changing them to declarations is breaking
174
+ if (!["var", "let"].includes(binding.kind)) {
175
+ illegalBindings.add(binding);
176
+ return;
177
+ }
178
+
179
+ if (!checkBinding(binding)) {
180
+ illegalBindings.add(binding);
181
+ return;
182
+ }
183
+
184
+ do {
185
+ index = choice([
186
+ stackMap.size,
187
+ propertyGen.generate(),
188
+ getRandomInteger(-250, 250),
189
+ ]);
190
+ } while (!index || stackKeys.has(index.toString()));
191
+
192
+ stackMap.set(binding, index);
193
+ stackKeys.add(index.toString());
194
+ }
195
+
196
+ const memberExpression = t.memberExpression(
197
+ t.identifier(stackName),
198
+ createLiteral(index),
199
+ true
200
+ );
201
+
202
+ if (isDefiningIdentifier(path)) {
203
+ replaceDefiningIdentifierToMemberExpression(path, memberExpression);
204
+
205
+ return;
206
+ }
207
+
208
+ ensureComputedExpression(path);
209
+ path.replaceWith(memberExpression);
210
+ },
211
+ });
212
+
213
+ if (!needsStack) return;
214
+
215
+ const originalParamCount = fnPath.node.params.length;
216
+ const originalLength = computeFunctionLength(fnPath);
217
+
218
+ fnPath.node.params = [t.restElement(t.identifier(stackName))];
219
+
220
+ // Discard extraneous parameters
221
+ // Predictable functions are guaranteed to not have extraneous parameters
222
+ if (!(fnPath.node as NodeSymbol)[PREDICTABLE]) {
223
+ prepend(
224
+ fnPath,
225
+ new Template(`${stackName}["length"] = {originalParamCount};`).single({
226
+ originalParamCount: t.numericLiteral(originalParamCount),
227
+ })
228
+ );
229
+ }
230
+
231
+ // Function is no longer predictable
232
+ (fnPath.node as NodeSymbol)[PREDICTABLE] = false;
233
+
234
+ fnPath.scope.registerBinding("param", fnPath.get("params")[0], fnPath);
235
+
236
+ me.setFunctionLength(fnPath, originalLength);
237
+
238
+ me.changeData.functions++;
239
+ };
240
+
241
+ return {
242
+ visitor: {
243
+ Function: {
244
+ exit(path: NodePath<t.Function>) {
245
+ if (!path.get("body").isBlockStatement()) return;
246
+
247
+ transformFunction(path);
248
+ },
249
+ },
250
+ Program: {
251
+ enter(path) {
252
+ path.scope.crawl();
253
+ },
254
+ },
255
+ },
256
+ };
257
+ };
@@ -0,0 +1,141 @@
1
+ import { NodePath } from "@babel/traverse";
2
+ import * as t from "@babel/types";
3
+ import { PluginInstance } from "../transforms/plugin";
4
+ import { NameGen } from "./NameGen";
5
+ import { prepend } from "./ast-utils";
6
+ import { chance, choice } from "./random-utils";
7
+
8
+ /**
9
+ * A Control Object is an object that is used to store properties that are used in multiple places.
10
+ */
11
+ export default class ControlObject {
12
+ propertyNames = new Set<string>();
13
+ nameGen: NameGen;
14
+ objectName: string | null = null;
15
+ objectPath: NodePath<t.Declaration> | null = null;
16
+ objectExpression: t.ObjectExpression | null = null;
17
+
18
+ constructor(public me: PluginInstance, public blockPath: NodePath<t.Block>) {
19
+ this.nameGen = new NameGen(me.options.identifierGenerator, {
20
+ avoidReserved: true,
21
+ avoidObjectPrototype: true,
22
+ });
23
+ }
24
+
25
+ createMemberExpression(propertyName: string): t.MemberExpression {
26
+ return t.memberExpression(
27
+ t.identifier(this.objectName),
28
+ t.stringLiteral(propertyName),
29
+ true
30
+ );
31
+ }
32
+
33
+ createPredicate() {
34
+ this.ensureCreated();
35
+
36
+ var propertyName = choice(Array.from(this.propertyNames));
37
+ if (!propertyName || chance(50)) {
38
+ propertyName = this.nameGen.generate();
39
+ }
40
+
41
+ return {
42
+ node: t.binaryExpression(
43
+ "in",
44
+ t.stringLiteral(propertyName),
45
+ t.identifier(this.objectName)
46
+ ),
47
+ value: this.propertyNames.has(propertyName),
48
+ };
49
+ }
50
+
51
+ createTruePredicate() {
52
+ var { node, value } = this.createPredicate();
53
+ if (value) {
54
+ return node;
55
+ }
56
+ return t.unaryExpression("!", node);
57
+ }
58
+
59
+ createFalsePredicate() {
60
+ var { node, value } = this.createPredicate();
61
+ if (!value) {
62
+ return node;
63
+ }
64
+ return t.unaryExpression("!", node);
65
+ }
66
+
67
+ private ensureCreated(node?: t.Node) {
68
+ if (!this.objectName) {
69
+ // Object hasn't been created yet
70
+ this.objectName = this.me.getPlaceholder() + "_controlObject";
71
+
72
+ if (node && t.isFunctionExpression(node) && !node.id) {
73
+ // Use function declaration as object
74
+
75
+ let newNode: t.FunctionDeclaration = node as any;
76
+ newNode.type = "FunctionDeclaration";
77
+ newNode.id = t.identifier(this.objectName);
78
+
79
+ let newPath = prepend(
80
+ this.blockPath,
81
+ newNode
82
+ )[0] as NodePath<t.FunctionDeclaration>;
83
+ this.me.skip(newPath);
84
+
85
+ this.objectPath = newPath;
86
+
87
+ return t.identifier(this.objectName);
88
+ } else {
89
+ // Create plain object
90
+ let newPath = prepend(
91
+ this.blockPath,
92
+ t.variableDeclaration("var", [
93
+ t.variableDeclarator(
94
+ t.identifier(this.objectName),
95
+ t.objectExpression([])
96
+ ),
97
+ ])
98
+ )[0] as NodePath<t.VariableDeclaration>;
99
+ this.me.skip(newPath);
100
+
101
+ this.objectPath = newPath;
102
+
103
+ var objectExpression = newPath.node.declarations[0]
104
+ .init as t.ObjectExpression;
105
+
106
+ this.objectExpression = objectExpression;
107
+ this.me.skip(this.objectExpression);
108
+ }
109
+ }
110
+ }
111
+
112
+ addProperty(node: t.Expression) {
113
+ var initialNode = this.ensureCreated(node);
114
+ if (initialNode) return initialNode;
115
+
116
+ const propertyName = this.nameGen.generate();
117
+ this.propertyNames.add(propertyName);
118
+
119
+ // Add an initial property
120
+ if (this.objectExpression) {
121
+ this.objectExpression.properties.push(
122
+ t.objectProperty(t.identifier(propertyName), node)
123
+ );
124
+ } else {
125
+ // Add as assignment expression
126
+
127
+ let assignment = t.assignmentExpression(
128
+ "=",
129
+ this.createMemberExpression(propertyName),
130
+ node
131
+ );
132
+
133
+ var newPath = this.objectPath.insertAfter(
134
+ t.expressionStatement(assignment)
135
+ )[0];
136
+ this.me.skip(newPath);
137
+ }
138
+
139
+ return this.createMemberExpression(propertyName);
140
+ }
141
+ }
@@ -0,0 +1,33 @@
1
+ export class IntGen {
2
+ private min: number;
3
+ private max: number;
4
+ private generatedInts: Set<number>;
5
+
6
+ constructor(min: number = -250, max: number = 250) {
7
+ this.min = min;
8
+ this.max = max;
9
+ this.generatedInts = new Set<number>();
10
+ }
11
+
12
+ private getRandomInt(min: number, max: number): number {
13
+ return Math.floor(Math.random() * (max - min + 1)) + min;
14
+ }
15
+
16
+ generate(): number {
17
+ let randomInt: number;
18
+
19
+ // Keep generating until we find a unique integer
20
+ do {
21
+ randomInt = this.getRandomInt(this.min, this.max);
22
+
23
+ // Expand the range if most integers in the current range are exhausted
24
+ if (this.generatedInts.size >= 0.8 * (this.max - this.min)) {
25
+ this.min -= 100;
26
+ this.max += 100;
27
+ }
28
+ } while (this.generatedInts.has(randomInt));
29
+
30
+ this.generatedInts.add(randomInt);
31
+ return randomInt;
32
+ }
33
+ }