redscript-mc 1.2.29 → 2.0.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 (274) hide show
  1. package/.claude/commands/build-test.md +10 -0
  2. package/.claude/commands/deploy-demo.md +12 -0
  3. package/.claude/commands/stage-status.md +13 -0
  4. package/.claude/settings.json +12 -0
  5. package/.github/workflows/ci.yml +1 -0
  6. package/CLAUDE.md +231 -0
  7. package/README.md +29 -28
  8. package/README.zh.md +28 -28
  9. package/demo.gif +0 -0
  10. package/dist/cli.js +2 -554
  11. package/dist/compile.js +2 -266
  12. package/dist/index.js +2 -159
  13. package/dist/lexer/index.js +9 -1
  14. package/dist/lowering/index.js +22 -5
  15. package/dist/src/__tests__/cli.test.d.ts +1 -0
  16. package/dist/src/__tests__/cli.test.js +104 -0
  17. package/dist/src/__tests__/codegen.test.d.ts +1 -0
  18. package/dist/src/__tests__/codegen.test.js +152 -0
  19. package/dist/src/__tests__/compile-all.test.d.ts +10 -0
  20. package/dist/src/__tests__/compile-all.test.js +108 -0
  21. package/dist/src/__tests__/dce.test.d.ts +1 -0
  22. package/dist/src/__tests__/dce.test.js +102 -0
  23. package/dist/src/__tests__/diagnostics.test.d.ts +4 -0
  24. package/dist/src/__tests__/diagnostics.test.js +177 -0
  25. package/dist/src/__tests__/e2e.test.d.ts +6 -0
  26. package/dist/src/__tests__/e2e.test.js +1789 -0
  27. package/dist/src/__tests__/entity-types.test.d.ts +1 -0
  28. package/dist/src/__tests__/entity-types.test.js +203 -0
  29. package/dist/src/__tests__/formatter.test.d.ts +1 -0
  30. package/dist/src/__tests__/formatter.test.js +40 -0
  31. package/dist/src/__tests__/lexer.test.d.ts +1 -0
  32. package/dist/src/__tests__/lexer.test.js +343 -0
  33. package/dist/src/__tests__/lowering.test.d.ts +1 -0
  34. package/dist/src/__tests__/lowering.test.js +1015 -0
  35. package/dist/src/__tests__/macro.test.d.ts +8 -0
  36. package/dist/src/__tests__/macro.test.js +306 -0
  37. package/dist/src/__tests__/mc-integration.test.d.ts +12 -0
  38. package/dist/src/__tests__/mc-integration.test.js +817 -0
  39. package/dist/src/__tests__/mc-syntax.test.d.ts +1 -0
  40. package/dist/src/__tests__/mc-syntax.test.js +124 -0
  41. package/dist/src/__tests__/nbt.test.d.ts +1 -0
  42. package/dist/src/__tests__/nbt.test.js +82 -0
  43. package/dist/src/__tests__/optimizer-advanced.test.d.ts +1 -0
  44. package/dist/src/__tests__/optimizer-advanced.test.js +124 -0
  45. package/dist/src/__tests__/optimizer.test.d.ts +1 -0
  46. package/dist/src/__tests__/optimizer.test.js +149 -0
  47. package/dist/src/__tests__/parser.test.d.ts +1 -0
  48. package/dist/src/__tests__/parser.test.js +807 -0
  49. package/dist/src/__tests__/repl.test.d.ts +1 -0
  50. package/dist/src/__tests__/repl.test.js +27 -0
  51. package/dist/src/__tests__/runtime.test.d.ts +1 -0
  52. package/dist/src/__tests__/runtime.test.js +289 -0
  53. package/dist/src/__tests__/stdlib-advanced.test.d.ts +4 -0
  54. package/dist/src/__tests__/stdlib-advanced.test.js +374 -0
  55. package/dist/src/__tests__/stdlib-bigint.test.d.ts +7 -0
  56. package/dist/src/__tests__/stdlib-bigint.test.js +426 -0
  57. package/dist/src/__tests__/stdlib-math.test.d.ts +7 -0
  58. package/dist/src/__tests__/stdlib-math.test.js +351 -0
  59. package/dist/src/__tests__/stdlib-vec.test.d.ts +4 -0
  60. package/dist/src/__tests__/stdlib-vec.test.js +263 -0
  61. package/dist/src/__tests__/structure-optimizer.test.d.ts +1 -0
  62. package/dist/src/__tests__/structure-optimizer.test.js +33 -0
  63. package/dist/src/__tests__/typechecker.test.d.ts +1 -0
  64. package/dist/src/__tests__/typechecker.test.js +552 -0
  65. package/dist/src/__tests__/var-allocator.test.d.ts +1 -0
  66. package/dist/src/__tests__/var-allocator.test.js +69 -0
  67. package/dist/src/ast/types.d.ts +515 -0
  68. package/dist/src/ast/types.js +9 -0
  69. package/dist/src/builtins/metadata.d.ts +36 -0
  70. package/dist/src/builtins/metadata.js +1014 -0
  71. package/dist/src/cli.d.ts +11 -0
  72. package/dist/src/cli.js +443 -0
  73. package/dist/src/codegen/cmdblock/index.d.ts +26 -0
  74. package/dist/src/codegen/cmdblock/index.js +45 -0
  75. package/dist/src/codegen/mcfunction/index.d.ts +40 -0
  76. package/dist/src/codegen/mcfunction/index.js +606 -0
  77. package/dist/src/codegen/structure/index.d.ts +24 -0
  78. package/dist/src/codegen/structure/index.js +279 -0
  79. package/dist/src/codegen/var-allocator.d.ts +45 -0
  80. package/dist/src/codegen/var-allocator.js +104 -0
  81. package/dist/src/compile.d.ts +37 -0
  82. package/dist/src/compile.js +165 -0
  83. package/dist/src/diagnostics/index.d.ts +44 -0
  84. package/dist/src/diagnostics/index.js +140 -0
  85. package/dist/src/events/types.d.ts +35 -0
  86. package/dist/src/events/types.js +59 -0
  87. package/dist/src/formatter/index.d.ts +1 -0
  88. package/dist/src/formatter/index.js +26 -0
  89. package/dist/src/index.d.ts +22 -0
  90. package/dist/src/index.js +45 -0
  91. package/dist/src/ir/builder.d.ts +33 -0
  92. package/dist/src/ir/builder.js +99 -0
  93. package/dist/src/ir/types.d.ts +132 -0
  94. package/dist/src/ir/types.js +15 -0
  95. package/dist/src/lexer/index.d.ts +37 -0
  96. package/dist/src/lexer/index.js +569 -0
  97. package/dist/src/lowering/index.d.ts +188 -0
  98. package/dist/src/lowering/index.js +3405 -0
  99. package/dist/src/mc-test/client.d.ts +128 -0
  100. package/dist/src/mc-test/client.js +174 -0
  101. package/dist/src/mc-test/runner.d.ts +28 -0
  102. package/dist/src/mc-test/runner.js +151 -0
  103. package/dist/src/mc-test/setup.d.ts +11 -0
  104. package/dist/src/mc-test/setup.js +98 -0
  105. package/dist/src/mc-validator/index.d.ts +17 -0
  106. package/dist/src/mc-validator/index.js +322 -0
  107. package/dist/src/nbt/index.d.ts +86 -0
  108. package/dist/src/nbt/index.js +250 -0
  109. package/dist/src/optimizer/commands.d.ts +38 -0
  110. package/dist/src/optimizer/commands.js +451 -0
  111. package/dist/src/optimizer/dce.d.ts +34 -0
  112. package/dist/src/optimizer/dce.js +639 -0
  113. package/dist/src/optimizer/passes.d.ts +34 -0
  114. package/dist/src/optimizer/passes.js +243 -0
  115. package/dist/src/optimizer/structure.d.ts +9 -0
  116. package/dist/src/optimizer/structure.js +356 -0
  117. package/dist/src/parser/index.d.ts +93 -0
  118. package/dist/src/parser/index.js +1687 -0
  119. package/dist/src/repl.d.ts +16 -0
  120. package/dist/src/repl.js +165 -0
  121. package/dist/src/runtime/index.d.ts +107 -0
  122. package/dist/src/runtime/index.js +1409 -0
  123. package/dist/src/typechecker/index.d.ts +61 -0
  124. package/dist/src/typechecker/index.js +1034 -0
  125. package/dist/src/types/entity-hierarchy.d.ts +29 -0
  126. package/dist/src/types/entity-hierarchy.js +107 -0
  127. package/dist/src2/__tests__/e2e/basic.test.d.ts +8 -0
  128. package/dist/src2/__tests__/e2e/basic.test.js +140 -0
  129. package/dist/src2/__tests__/e2e/macros.test.d.ts +9 -0
  130. package/dist/src2/__tests__/e2e/macros.test.js +182 -0
  131. package/dist/src2/__tests__/e2e/migrate.test.d.ts +13 -0
  132. package/dist/src2/__tests__/e2e/migrate.test.js +2739 -0
  133. package/dist/src2/__tests__/hir/desugar.test.d.ts +1 -0
  134. package/dist/src2/__tests__/hir/desugar.test.js +234 -0
  135. package/dist/src2/__tests__/lir/lower.test.d.ts +1 -0
  136. package/dist/src2/__tests__/lir/lower.test.js +559 -0
  137. package/dist/src2/__tests__/lir/types.test.d.ts +1 -0
  138. package/dist/src2/__tests__/lir/types.test.js +185 -0
  139. package/dist/src2/__tests__/lir/verify.test.d.ts +1 -0
  140. package/dist/src2/__tests__/lir/verify.test.js +221 -0
  141. package/dist/src2/__tests__/mir/arithmetic.test.d.ts +1 -0
  142. package/dist/src2/__tests__/mir/arithmetic.test.js +130 -0
  143. package/dist/src2/__tests__/mir/control-flow.test.d.ts +1 -0
  144. package/dist/src2/__tests__/mir/control-flow.test.js +205 -0
  145. package/dist/src2/__tests__/mir/verify.test.d.ts +1 -0
  146. package/dist/src2/__tests__/mir/verify.test.js +223 -0
  147. package/dist/src2/__tests__/optimizer/block_merge.test.d.ts +1 -0
  148. package/dist/src2/__tests__/optimizer/block_merge.test.js +78 -0
  149. package/dist/src2/__tests__/optimizer/branch_simplify.test.d.ts +1 -0
  150. package/dist/src2/__tests__/optimizer/branch_simplify.test.js +58 -0
  151. package/dist/src2/__tests__/optimizer/constant_fold.test.d.ts +1 -0
  152. package/dist/src2/__tests__/optimizer/constant_fold.test.js +131 -0
  153. package/dist/src2/__tests__/optimizer/copy_prop.test.d.ts +1 -0
  154. package/dist/src2/__tests__/optimizer/copy_prop.test.js +91 -0
  155. package/dist/src2/__tests__/optimizer/dce.test.d.ts +1 -0
  156. package/dist/src2/__tests__/optimizer/dce.test.js +76 -0
  157. package/dist/src2/__tests__/optimizer/pipeline.test.d.ts +1 -0
  158. package/dist/src2/__tests__/optimizer/pipeline.test.js +102 -0
  159. package/dist/src2/emit/compile.d.ts +19 -0
  160. package/dist/src2/emit/compile.js +80 -0
  161. package/dist/src2/emit/index.d.ts +17 -0
  162. package/dist/src2/emit/index.js +172 -0
  163. package/dist/src2/hir/lower.d.ts +15 -0
  164. package/dist/src2/hir/lower.js +378 -0
  165. package/dist/src2/hir/types.d.ts +373 -0
  166. package/dist/src2/hir/types.js +16 -0
  167. package/dist/src2/lir/lower.d.ts +15 -0
  168. package/dist/src2/lir/lower.js +453 -0
  169. package/dist/src2/lir/types.d.ts +136 -0
  170. package/dist/src2/lir/types.js +11 -0
  171. package/dist/src2/lir/verify.d.ts +14 -0
  172. package/dist/src2/lir/verify.js +113 -0
  173. package/dist/src2/mir/lower.d.ts +9 -0
  174. package/dist/src2/mir/lower.js +1030 -0
  175. package/dist/src2/mir/macro.d.ts +22 -0
  176. package/dist/src2/mir/macro.js +168 -0
  177. package/dist/src2/mir/types.d.ts +183 -0
  178. package/dist/src2/mir/types.js +11 -0
  179. package/dist/src2/mir/verify.d.ts +16 -0
  180. package/dist/src2/mir/verify.js +216 -0
  181. package/dist/src2/optimizer/block_merge.d.ts +12 -0
  182. package/dist/src2/optimizer/block_merge.js +84 -0
  183. package/dist/src2/optimizer/branch_simplify.d.ts +9 -0
  184. package/dist/src2/optimizer/branch_simplify.js +28 -0
  185. package/dist/src2/optimizer/constant_fold.d.ts +10 -0
  186. package/dist/src2/optimizer/constant_fold.js +85 -0
  187. package/dist/src2/optimizer/copy_prop.d.ts +9 -0
  188. package/dist/src2/optimizer/copy_prop.js +113 -0
  189. package/dist/src2/optimizer/dce.d.ts +8 -0
  190. package/dist/src2/optimizer/dce.js +155 -0
  191. package/dist/src2/optimizer/pipeline.d.ts +10 -0
  192. package/dist/src2/optimizer/pipeline.js +42 -0
  193. package/dist/tsconfig.tsbuildinfo +1 -0
  194. package/docs/compiler-pipeline-redesign.md +2243 -0
  195. package/docs/optimization-ideas.md +1076 -0
  196. package/editors/vscode/package-lock.json +3 -3
  197. package/editors/vscode/package.json +1 -1
  198. package/examples/readme-demo.mcrs +44 -66
  199. package/jest.config.js +1 -1
  200. package/package.json +6 -5
  201. package/scripts/postbuild.js +15 -0
  202. package/src/__tests__/cli.test.ts +8 -220
  203. package/src/__tests__/dce.test.ts +11 -56
  204. package/src/__tests__/diagnostics.test.ts +59 -38
  205. package/src/__tests__/mc-integration.test.ts +1 -2
  206. package/src/ast/types.ts +6 -1
  207. package/src/cli.ts +29 -156
  208. package/src/compile.ts +6 -162
  209. package/src/index.ts +14 -178
  210. package/src/lexer/index.ts +9 -1
  211. package/src/mc-test/runner.ts +4 -3
  212. package/src/parser/index.ts +1 -1
  213. package/src/repl.ts +1 -1
  214. package/src/runtime/index.ts +1 -1
  215. package/src2/__tests__/e2e/basic.test.ts +154 -0
  216. package/src2/__tests__/e2e/macros.test.ts +199 -0
  217. package/src2/__tests__/e2e/migrate.test.ts +3008 -0
  218. package/src2/__tests__/hir/desugar.test.ts +263 -0
  219. package/src2/__tests__/lir/lower.test.ts +619 -0
  220. package/src2/__tests__/lir/types.test.ts +207 -0
  221. package/src2/__tests__/lir/verify.test.ts +249 -0
  222. package/src2/__tests__/mir/arithmetic.test.ts +156 -0
  223. package/src2/__tests__/mir/control-flow.test.ts +242 -0
  224. package/src2/__tests__/mir/verify.test.ts +254 -0
  225. package/src2/__tests__/optimizer/block_merge.test.ts +84 -0
  226. package/src2/__tests__/optimizer/branch_simplify.test.ts +64 -0
  227. package/src2/__tests__/optimizer/constant_fold.test.ts +145 -0
  228. package/src2/__tests__/optimizer/copy_prop.test.ts +99 -0
  229. package/src2/__tests__/optimizer/dce.test.ts +83 -0
  230. package/src2/__tests__/optimizer/pipeline.test.ts +116 -0
  231. package/src2/emit/compile.ts +99 -0
  232. package/src2/emit/index.ts +222 -0
  233. package/src2/hir/lower.ts +428 -0
  234. package/src2/hir/types.ts +216 -0
  235. package/src2/lir/lower.ts +556 -0
  236. package/src2/lir/types.ts +109 -0
  237. package/src2/lir/verify.ts +129 -0
  238. package/src2/mir/lower.ts +1160 -0
  239. package/src2/mir/macro.ts +167 -0
  240. package/src2/mir/types.ts +106 -0
  241. package/src2/mir/verify.ts +218 -0
  242. package/src2/optimizer/block_merge.ts +93 -0
  243. package/src2/optimizer/branch_simplify.ts +27 -0
  244. package/src2/optimizer/constant_fold.ts +88 -0
  245. package/src2/optimizer/copy_prop.ts +106 -0
  246. package/src2/optimizer/dce.ts +133 -0
  247. package/src2/optimizer/pipeline.ts +44 -0
  248. package/tsconfig.json +2 -2
  249. package/src/__tests__/codegen.test.ts +0 -161
  250. package/src/__tests__/e2e.test.ts +0 -2039
  251. package/src/__tests__/entity-types.test.ts +0 -236
  252. package/src/__tests__/lowering.test.ts +0 -1185
  253. package/src/__tests__/macro.test.ts +0 -343
  254. package/src/__tests__/nbt.test.ts +0 -58
  255. package/src/__tests__/optimizer-advanced.test.ts +0 -144
  256. package/src/__tests__/optimizer.test.ts +0 -162
  257. package/src/__tests__/runtime.test.ts +0 -305
  258. package/src/__tests__/stdlib-advanced.test.ts +0 -379
  259. package/src/__tests__/stdlib-bigint.test.ts +0 -427
  260. package/src/__tests__/stdlib-math.test.ts +0 -374
  261. package/src/__tests__/stdlib-vec.test.ts +0 -259
  262. package/src/__tests__/structure-optimizer.test.ts +0 -38
  263. package/src/__tests__/var-allocator.test.ts +0 -75
  264. package/src/codegen/cmdblock/index.ts +0 -63
  265. package/src/codegen/mcfunction/index.ts +0 -662
  266. package/src/codegen/structure/index.ts +0 -346
  267. package/src/codegen/var-allocator.ts +0 -104
  268. package/src/ir/builder.ts +0 -116
  269. package/src/ir/types.ts +0 -134
  270. package/src/lowering/index.ts +0 -3860
  271. package/src/optimizer/commands.ts +0 -534
  272. package/src/optimizer/dce.ts +0 -679
  273. package/src/optimizer/passes.ts +0 -250
  274. package/src/optimizer/structure.ts +0 -450
@@ -0,0 +1,817 @@
1
+ "use strict";
2
+ /**
3
+ * RedScript MC Integration Tests
4
+ *
5
+ * Tests compiled datapacks against a real Paper 1.21.4 server.
6
+ *
7
+ * Prerequisites:
8
+ * - Paper server running with TestHarnessPlugin on port 25561
9
+ * - MC_SERVER_DIR env var pointing to server directory
10
+ *
11
+ * Run: MC_SERVER_DIR=~/mc-test-server npx jest mc-integration --testTimeout=120000
12
+ */
13
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
14
+ if (k2 === undefined) k2 = k;
15
+ var desc = Object.getOwnPropertyDescriptor(m, k);
16
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
17
+ desc = { enumerable: true, get: function() { return m[k]; } };
18
+ }
19
+ Object.defineProperty(o, k2, desc);
20
+ }) : (function(o, m, k, k2) {
21
+ if (k2 === undefined) k2 = k;
22
+ o[k2] = m[k];
23
+ }));
24
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
25
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
26
+ }) : function(o, v) {
27
+ o["default"] = v;
28
+ });
29
+ var __importStar = (this && this.__importStar) || (function () {
30
+ var ownKeys = function(o) {
31
+ ownKeys = Object.getOwnPropertyNames || function (o) {
32
+ var ar = [];
33
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
34
+ return ar;
35
+ };
36
+ return ownKeys(o);
37
+ };
38
+ return function (mod) {
39
+ if (mod && mod.__esModule) return mod;
40
+ var result = {};
41
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
42
+ __setModuleDefault(result, mod);
43
+ return result;
44
+ };
45
+ })();
46
+ Object.defineProperty(exports, "__esModule", { value: true });
47
+ const fs = __importStar(require("fs"));
48
+ const path = __importStar(require("path"));
49
+ const compile_1 = require("../compile");
50
+ const client_1 = require("../mc-test/client");
51
+ const MC_HOST = process.env.MC_HOST ?? 'localhost';
52
+ const MC_PORT = parseInt(process.env.MC_PORT ?? '25561');
53
+ const MC_SERVER_DIR = process.env.MC_SERVER_DIR ?? path.join(process.env.HOME, 'mc-test-server');
54
+ const DATAPACK_DIR = path.join(MC_SERVER_DIR, 'world', 'datapacks', 'redscript-test');
55
+ const FIXTURE_DIR = path.join(__dirname, 'fixtures');
56
+ let serverOnline = false;
57
+ let mc;
58
+ /** Write compiled RedScript source into the shared test datapack directory.
59
+ * Merges minecraft tag files (tick.json / load.json) instead of overwriting. */
60
+ function writeFixture(source, namespace) {
61
+ fs.mkdirSync(DATAPACK_DIR, { recursive: true });
62
+ // Write pack.mcmeta once
63
+ if (!fs.existsSync(path.join(DATAPACK_DIR, 'pack.mcmeta'))) {
64
+ fs.writeFileSync(path.join(DATAPACK_DIR, 'pack.mcmeta'), JSON.stringify({
65
+ pack: { pack_format: 48, description: 'RedScript integration tests' }
66
+ }));
67
+ }
68
+ const result = (0, compile_1.compile)(source, { namespace });
69
+ for (const file of result.files) {
70
+ if (file.path === 'pack.mcmeta')
71
+ continue;
72
+ const filePath = path.join(DATAPACK_DIR, file.path);
73
+ fs.mkdirSync(path.dirname(filePath), { recursive: true });
74
+ // Merge minecraft tag files (tick.json, load.json) instead of overwriting
75
+ if (file.path.includes('data/minecraft/tags/') && fs.existsSync(filePath)) {
76
+ const existing = JSON.parse(fs.readFileSync(filePath, 'utf-8'));
77
+ const incoming = JSON.parse(file.content);
78
+ const merged = { values: [...new Set([...(existing.values ?? []), ...(incoming.values ?? [])])] };
79
+ fs.writeFileSync(filePath, JSON.stringify(merged, null, 2));
80
+ }
81
+ else {
82
+ fs.writeFileSync(filePath, file.content);
83
+ }
84
+ }
85
+ }
86
+ function writeFixtureFile(fileName, namespace) {
87
+ writeFixture(fs.readFileSync(path.join(FIXTURE_DIR, fileName), 'utf-8'), namespace);
88
+ }
89
+ async function waitForServer(client, timeoutMs = 30000) {
90
+ const deadline = Date.now() + timeoutMs;
91
+ while (Date.now() < deadline) {
92
+ if (await client.isOnline()) {
93
+ return true;
94
+ }
95
+ await new Promise(resolve => setTimeout(resolve, 1000));
96
+ }
97
+ return false;
98
+ }
99
+ beforeAll(async () => {
100
+ mc = new client_1.MCTestClient(MC_HOST, MC_PORT);
101
+ serverOnline = await waitForServer(mc);
102
+ if (!serverOnline) {
103
+ console.warn(`⚠ MC server not running at ${MC_HOST}:${MC_PORT} — skipping integration tests`);
104
+ console.warn(` Run: MC_SERVER_DIR=~/mc-test-server npx ts-node src/mc-test/setup.ts`);
105
+ console.warn(` Then restart the MC server and re-run tests.`);
106
+ return;
107
+ }
108
+ // ── Write fixtures + use safe reloadData (no /reload confirm) ───────
109
+ // counter.mcrs
110
+ if (fs.existsSync(path.join(__dirname, '../examples/counter.mcrs'))) {
111
+ writeFixture(fs.readFileSync(path.join(__dirname, '../examples/counter.mcrs'), 'utf-8'), 'counter');
112
+ }
113
+ if (fs.existsSync(path.join(__dirname, '../examples/world_manager.mcrs'))) {
114
+ writeFixture(fs.readFileSync(path.join(__dirname, '../examples/world_manager.mcrs'), 'utf-8'), 'world_manager');
115
+ }
116
+ writeFixture(`
117
+ @tick
118
+ fn on_tick() {
119
+ scoreboard_set("#tick_counter", #ticks, scoreboard_get("#tick_counter", #ticks) + 1);
120
+ }
121
+ `, 'tick_test');
122
+ writeFixture(`
123
+ fn check_score() {
124
+ let x: int = scoreboard_get("#check_x", #test_score);
125
+ if (x > 5) {
126
+ scoreboard_set("#check_x", #result, 1);
127
+ } else {
128
+ scoreboard_set("#check_x", #result, 0);
129
+ }
130
+ }
131
+ `, 'inline_test');
132
+ // ── E2E scenario fixtures ────────────────────────────────────────────
133
+ // Scenario A: mini game loop (timer countdown + ended flag)
134
+ writeFixture(`
135
+ @tick
136
+ fn game_tick() {
137
+ let time: int = scoreboard_get("#game", #timer);
138
+ if (time > 0) {
139
+ scoreboard_set("#game", #timer, time - 1);
140
+ }
141
+ if (time == 1) {
142
+ scoreboard_set("#game", #ended, 1);
143
+ }
144
+ }
145
+ fn start_game() {
146
+ scoreboard_set("#game", #timer, 5);
147
+ scoreboard_set("#game", #ended, 0);
148
+ }
149
+ `, 'game_loop');
150
+ // Scenario B: two functions, same temp var namespace — verify no collision
151
+ writeFixture(`
152
+ fn calc_sum() {
153
+ let a: int = scoreboard_get("#math", #val_a);
154
+ let b: int = scoreboard_get("#math", #val_b);
155
+ scoreboard_set("#math", #sum, a + b);
156
+ }
157
+ fn calc_product() {
158
+ let x: int = scoreboard_get("#math", #val_x);
159
+ let y: int = scoreboard_get("#math", #val_y);
160
+ scoreboard_set("#math", #product, x * y);
161
+ }
162
+ fn run_both() {
163
+ calc_sum();
164
+ calc_product();
165
+ }
166
+ `, 'math_test');
167
+ // Scenario C: 3-deep call chain, each step modifies shared state
168
+ writeFixture(`
169
+ fn step3() {
170
+ let v: int = scoreboard_get("#chain", #val);
171
+ scoreboard_set("#chain", #val, v * 2);
172
+ }
173
+ fn step2() {
174
+ let v: int = scoreboard_get("#chain", #val);
175
+ scoreboard_set("#chain", #val, v + 5);
176
+ step3();
177
+ }
178
+ fn step1() {
179
+ scoreboard_set("#chain", #val, 10);
180
+ step2();
181
+ }
182
+ `, 'call_chain');
183
+ // Scenario D: setblock batching optimizer — 4 adjacent setblocks → fill
184
+ writeFixture(`
185
+ fn build_row() {
186
+ setblock((0, 70, 0), "minecraft:stone");
187
+ setblock((1, 70, 0), "minecraft:stone");
188
+ setblock((2, 70, 0), "minecraft:stone");
189
+ setblock((3, 70, 0), "minecraft:stone");
190
+ }
191
+ `, 'fill_test');
192
+ // Scenario E: for-range loop — loop counter increments exactly N times
193
+ writeFixture(`
194
+ fn count_to_five() {
195
+ scoreboard_set("#range", #counter, 0);
196
+ for i in 0..5 {
197
+ let c: int = scoreboard_get("#range", #counter);
198
+ scoreboard_set("#range", #counter, c + 1);
199
+ }
200
+ }
201
+ `, 'range_test');
202
+ // Scenario F: function call with return value — verifies $ret propagation
203
+ writeFixture(`
204
+ fn triple(x: int) -> int {
205
+ return x * 3;
206
+ }
207
+ fn run_nested() {
208
+ let a: int = triple(4);
209
+ scoreboard_set("#nested", #result, a);
210
+ }
211
+ `, 'nested_test');
212
+ // Scenario G: match statement dispatches to correct branch
213
+ writeFixture(`
214
+ fn classify(x: int) {
215
+ match (x) {
216
+ 1 => { scoreboard_set("#match", #out, 10); }
217
+ 2 => { scoreboard_set("#match", #out, 20); }
218
+ 3 => { scoreboard_set("#match", #out, 30); }
219
+ _ => { scoreboard_set("#match", #out, -1); }
220
+ }
221
+ }
222
+ `, 'match_test');
223
+ // Scenario H: while loop counts down
224
+ writeFixture(`
225
+ fn countdown() {
226
+ scoreboard_set("#wloop", #i, 10);
227
+ scoreboard_set("#wloop", #steps, 0);
228
+ let i: int = scoreboard_get("#wloop", #i);
229
+ while (i > 0) {
230
+ let s: int = scoreboard_get("#wloop", #steps);
231
+ scoreboard_set("#wloop", #steps, s + 1);
232
+ i = i - 1;
233
+ scoreboard_set("#wloop", #i, i);
234
+ }
235
+ }
236
+ `, 'while_test');
237
+ // Scenario I: multiple if/else branches (boundary test)
238
+ writeFixture(`
239
+ fn classify_score() {
240
+ let x: int = scoreboard_get("#boundary", #input);
241
+ if (x > 100) {
242
+ scoreboard_set("#boundary", #tier, 3);
243
+ } else {
244
+ if (x > 50) {
245
+ scoreboard_set("#boundary", #tier, 2);
246
+ } else {
247
+ if (x > 0) {
248
+ scoreboard_set("#boundary", #tier, 1);
249
+ } else {
250
+ scoreboard_set("#boundary", #tier, 0);
251
+ }
252
+ }
253
+ }
254
+ }
255
+ `, 'boundary_test');
256
+ // Scenario J: entity management — summon via raw commands
257
+ writeFixture(`
258
+ fn tag_entities() {
259
+ raw("summon minecraft:armor_stand 10 65 10");
260
+ raw("summon minecraft:armor_stand 11 65 10");
261
+ raw("summon minecraft:armor_stand 12 65 10");
262
+ }
263
+ `, 'tag_test');
264
+ // Scenario K: mixed arithmetic — order of operations
265
+ writeFixture(`
266
+ fn math_order() {
267
+ let a: int = 2;
268
+ let b: int = 3;
269
+ let c: int = 4;
270
+ scoreboard_set("#order", #r1, a + b * c);
271
+ scoreboard_set("#order", #r2, (a + b) * c);
272
+ let d: int = 100;
273
+ let e: int = d / 3;
274
+ scoreboard_set("#order", #r3, e);
275
+ }
276
+ `, 'order_test');
277
+ // Scenario L: scoreboard read-modify-write chain
278
+ writeFixture(`
279
+ fn chain_rmw() {
280
+ scoreboard_set("#rmw", #v, 1);
281
+ let v: int = scoreboard_get("#rmw", #v);
282
+ scoreboard_set("#rmw", #v, v * 2);
283
+ v = scoreboard_get("#rmw", #v);
284
+ scoreboard_set("#rmw", #v, v * 2);
285
+ v = scoreboard_get("#rmw", #v);
286
+ scoreboard_set("#rmw", #v, v * 2);
287
+ }
288
+ `, 'rmw_test');
289
+ writeFixtureFile('impl-test.mcrs', 'impl_test');
290
+ writeFixtureFile('timeout-test.mcrs', 'timeout_test');
291
+ writeFixtureFile('interval-test.mcrs', 'interval_test');
292
+ writeFixtureFile('is-check-test.mcrs', 'is_check_test');
293
+ writeFixtureFile('event-test.mcrs', 'event_test');
294
+ // ── Full reset + safe data reload ────────────────────────────────────
295
+ await mc.fullReset();
296
+ // Pre-create scoreboards
297
+ for (const obj of ['ticks', 'seconds', 'test_score', 'result', 'calc', 'rs',
298
+ 'timer', 'ended', 'val_a', 'val_b', 'sum', 'val_x', 'val_y', 'product', 'val',
299
+ 'counter', 'out', 'i', 'steps', 'input', 'tier', 'r1', 'r2', 'r3', 'v',
300
+ 'done', 'fired', 'players', 'zombies']) {
301
+ await mc.command(`/scoreboard objectives add ${obj} dummy`).catch(() => { });
302
+ }
303
+ await mc.command('/scoreboard players set counter ticks 0');
304
+ await mc.command('/scoreboard players set #tick_counter ticks 0');
305
+ await mc.command('/scoreboard players set #check_x test_score 10');
306
+ await mc.command('/scoreboard players set #check_x result 99');
307
+ // Safe reload (Bukkit.reloadData — only datapacks, no plugin restart)
308
+ console.log(' Reloading datapacks (safe reloadData)...');
309
+ await mc.reload();
310
+ await new Promise(r => setTimeout(r, 5000)); // wall-clock wait for data reload
311
+ // Initialize __load functions
312
+ await mc.command('/function counter:__load').catch(() => { });
313
+ await mc.command('/function inline_test:__load').catch(() => { });
314
+ await mc.ticks(20);
315
+ console.log(' Setup complete.');
316
+ }, 60000);
317
+ describe('MC Integration Tests', () => {
318
+ // ─── Test 1: Server connectivity ─────────────────────────────────────
319
+ test('server is online and healthy', async () => {
320
+ if (!serverOnline)
321
+ return;
322
+ const status = await mc.status();
323
+ expect(status.online).toBe(true);
324
+ expect(status.tps_1m).toBeGreaterThan(10); // Allow recovery after reload
325
+ console.log(` Server: ${status.version}, TPS: ${status.tps_1m.toFixed(1)}`);
326
+ });
327
+ // ─── Test 2: Counter tick ─────────────────────────────────────────────
328
+ test('counter.mcrs: tick function increments scoreboard over time', async () => {
329
+ if (!serverOnline)
330
+ return;
331
+ await mc.ticks(40); // Wait 2s (counter was already init'd in beforeAll)
332
+ const count = await mc.scoreboard('counter', 'ticks');
333
+ expect(count).toBeGreaterThan(0);
334
+ console.log(` counter/ticks after setup+40 ticks: ${count}`);
335
+ });
336
+ // ─── Test 3: setblock ────────────────────────────────────────────────
337
+ test('world_manager.mcrs: setblock places correct block', async () => {
338
+ if (!serverOnline)
339
+ return;
340
+ // Clear just the lobby area, keep other state
341
+ await mc.fullReset({ x1: -10, y1: 60, z1: -10, x2: 15, y2: 80, z2: 15, resetScoreboards: false });
342
+ await mc.command('/function world_manager:__load');
343
+ await mc.command('/function world_manager:reset_lobby_platform');
344
+ await mc.ticks(10);
345
+ const block = await mc.block(4, 65, 4);
346
+ expect(block.type).toBe('minecraft:gold_block');
347
+ console.log(` Block at (4,65,4): ${block.type}`);
348
+ });
349
+ // ─── Test 4: fill ────────────────────────────────────────────────────
350
+ test('world_manager.mcrs: fill creates smooth_stone floor', async () => {
351
+ if (!serverOnline)
352
+ return;
353
+ // Runs after test 3, floor should still be there
354
+ const block = await mc.block(4, 64, 4);
355
+ expect(block.type).toBe('minecraft:smooth_stone');
356
+ console.log(` Floor at (4,64,4): ${block.type}`);
357
+ });
358
+ // ─── Test 5: Scoreboard arithmetic ───────────────────────────────────
359
+ test('scoreboard arithmetic works via commands', async () => {
360
+ if (!serverOnline)
361
+ return;
362
+ await mc.command('/scoreboard players set TestA calc 10');
363
+ await mc.command('/scoreboard players set TestB calc 25');
364
+ await mc.command('/scoreboard players operation TestA calc += TestB calc');
365
+ await mc.ticks(2);
366
+ const result = await mc.scoreboard('TestA', 'calc');
367
+ expect(result).toBe(35);
368
+ console.log(` 10 + 25 = ${result}`);
369
+ });
370
+ // ─── Test 6: Scoreboard proxy for announce ────────────────────────────
371
+ test('scoreboard proxy test (chat logging not supported for /say)', async () => {
372
+ if (!serverOnline)
373
+ return;
374
+ await mc.command('/scoreboard objectives add announce_test dummy');
375
+ await mc.command('/scoreboard players set announce_marker announce_test 42');
376
+ await mc.ticks(2);
377
+ const marker = await mc.scoreboard('announce_marker', 'announce_test');
378
+ expect(marker).toBe(42);
379
+ console.log(` Marker value: ${marker}`);
380
+ });
381
+ // ─── Test 7: if/else logic via inline script ──────────────────────────
382
+ test('inline rs: if/else (x=10 > 5) sets result=1', async () => {
383
+ if (!serverOnline)
384
+ return;
385
+ // #check_x test_score=10 was set in beforeAll, run check_score
386
+ await mc.command('/function inline_test:check_score');
387
+ await mc.ticks(5);
388
+ const result = await mc.scoreboard('#check_x', 'result');
389
+ expect(result).toBe(1);
390
+ console.log(` if (10 > 5) → result: ${result}`);
391
+ });
392
+ // ─── Test 8: Entity counting ──────────────────────────────────────────
393
+ test('entity query: armor_stands survive peaceful mode', async () => {
394
+ if (!serverOnline)
395
+ return;
396
+ await mc.fullReset({ clearArea: false, killEntities: true, resetScoreboards: false });
397
+ await mc.command('/summon minecraft:armor_stand 0 65 0');
398
+ await mc.command('/summon minecraft:armor_stand 2 65 0');
399
+ await mc.command('/summon minecraft:armor_stand 4 65 0');
400
+ await mc.ticks(5);
401
+ const stands = await mc.entities('@e[type=minecraft:armor_stand]');
402
+ expect(stands.length).toBe(3);
403
+ console.log(` Spawned 3 armor_stands, found: ${stands.length}`);
404
+ await mc.command('/kill @e[type=minecraft:armor_stand]');
405
+ });
406
+ // ─── Test 9: @tick dispatcher runs every tick ─────────────────────────
407
+ test('@tick: tick_test increments #tick_counter every tick', async () => {
408
+ if (!serverOnline)
409
+ return;
410
+ // Reset counter
411
+ await mc.command('/scoreboard players set #tick_counter ticks 0');
412
+ await mc.ticks(40); // 2s
413
+ const ticks = await mc.scoreboard('#tick_counter', 'ticks');
414
+ expect(ticks).toBeGreaterThanOrEqual(10); // At least 10 of 40 ticks fired
415
+ console.log(` #tick_counter after 40 ticks: ${ticks}`);
416
+ });
417
+ // ─── Test 10: fullReset clears blocks ─────────────────────────────────
418
+ test('fullReset clears previously placed blocks', async () => {
419
+ if (!serverOnline)
420
+ return;
421
+ await mc.command('/setblock 5 65 5 minecraft:diamond_block');
422
+ await mc.ticks(2);
423
+ let block = await mc.block(5, 65, 5);
424
+ expect(block.type).toBe('minecraft:diamond_block');
425
+ await mc.fullReset({ x1: 0, y1: 60, z1: 0, x2: 10, y2: 75, z2: 10, resetScoreboards: false });
426
+ block = await mc.block(5, 65, 5);
427
+ expect(block.type).toBe('minecraft:air');
428
+ console.log(` Block after reset: ${block.type} ✓`);
429
+ });
430
+ });
431
+ // ─── E2E Scenario Tests ───────────────────────────────────────────────────────
432
+ describe('E2E Scenario Tests', () => {
433
+ // Scenario A: Mini game loop
434
+ // Verifies: @tick auto-runs, scoreboard read-modify-write, two if conditions
435
+ // in the same function, timer countdown converges to ended=1
436
+ test('A: game_loop timer countdown sets ended=1 after N ticks', async () => {
437
+ if (!serverOnline)
438
+ return;
439
+ // game_tick is @tick - it runs every server tick automatically.
440
+ // start_game sets timer=5, but game_tick may already decrement it by the
441
+ // time we query. Use a large timer and just verify it reaches 0 eventually.
442
+ await mc.command('/scoreboard players set #game timer 0');
443
+ await mc.command('/scoreboard players set #game ended 0');
444
+ await mc.ticks(2);
445
+ await mc.command('/function game_loop:__load');
446
+ await mc.command('/function game_loop:start_game'); // timer=5, ended=0
447
+ // Wait 25 ticks — enough for 5 decrements + margin
448
+ await mc.ticks(25);
449
+ const ended = await mc.scoreboard('#game', 'ended');
450
+ expect(ended).toBe(1);
451
+ const finalTimer = await mc.scoreboard('#game', 'timer');
452
+ expect(finalTimer).toBe(0);
453
+ console.log(` timer hit 0 (final=${finalTimer}), ended=${ended} ✓`);
454
+ });
455
+ // Scenario B: No temp var collision between two functions called in sequence
456
+ // Verifies: each function's temp vars are isolated per-call via globally unique names
457
+ // If there's a bug, calc_product would see sum's leftover $t vars and produce wrong result
458
+ test('B: calc_sum + calc_product called in sequence — no temp var collision', async () => {
459
+ if (!serverOnline)
460
+ return;
461
+ await mc.command('/function math_test:__load');
462
+ await mc.command('/scoreboard players set #math val_a 7');
463
+ await mc.command('/scoreboard players set #math val_b 3');
464
+ await mc.command('/scoreboard players set #math val_x 4');
465
+ await mc.command('/scoreboard players set #math val_y 5');
466
+ await mc.command('/function math_test:run_both'); // calc_sum() then calc_product()
467
+ await mc.ticks(5);
468
+ const sum = await mc.scoreboard('#math', 'sum');
469
+ const product = await mc.scoreboard('#math', 'product');
470
+ expect(sum).toBe(10); // 7 + 3
471
+ expect(product).toBe(20); // 4 × 5
472
+ console.log(` sum=${sum} (expect 10), product=${product} (expect 20) ✓`);
473
+ });
474
+ // Scenario C: 3-deep call chain, shared state threaded through
475
+ // Verifies: function calls preserve scoreboard state across stack frames
476
+ // step1: val=10 → step2: val=10+5=15 → step3: val=15×2=30
477
+ test('C: 3-deep call chain preserves intermediate state (10→15→30)', async () => {
478
+ if (!serverOnline)
479
+ return;
480
+ await mc.command('/function call_chain:__load');
481
+ await mc.command('/scoreboard players set #chain val 0');
482
+ await mc.command('/function call_chain:step1');
483
+ await mc.ticks(5);
484
+ const val = await mc.scoreboard('#chain', 'val');
485
+ expect(val).toBe(30); // (10 + 5) * 2 = 30
486
+ console.log(` call chain result: ${val} (expect 30) ✓`);
487
+ });
488
+ // Scenario D: Setblock batching optimizer — 4 adjacent setblocks compiled to fill
489
+ // Verifies: optimizer's fill-batching pass produces correct MC behavior
490
+ // (not just that the output says "fill", but that ALL 4 blocks are actually stone)
491
+ test('D: fill optimizer — 4 adjacent setblocks all placed correctly', async () => {
492
+ if (!serverOnline)
493
+ return;
494
+ await mc.fullReset({ x1: -5, y1: 65, z1: -5, x2: 10, y2: 75, z2: 10, resetScoreboards: false });
495
+ await mc.command('/function fill_test:__load');
496
+ await mc.command('/function fill_test:build_row');
497
+ await mc.ticks(5);
498
+ // All 4 blocks should be stone (optimizer batched into fill 0 70 0 3 70 0 stone)
499
+ for (let x = 0; x <= 3; x++) {
500
+ const block = await mc.block(x, 70, 0);
501
+ expect(block.type).toBe('minecraft:stone');
502
+ }
503
+ // Neighbors should still be air (fill didn't overshoot)
504
+ const before = await mc.block(-1, 70, 0);
505
+ const after = await mc.block(4, 70, 0);
506
+ expect(before.type).toBe('minecraft:air');
507
+ expect(after.type).toBe('minecraft:air');
508
+ console.log(` fill_test: blocks [0-3,70,0]=stone, [-1]/[4]=air ✓`);
509
+ });
510
+ // Scenario E: for-range loop executes body exactly N times
511
+ // Verifies: for i in 0..5 increments counter 5 times
512
+ test('E: for-range loop increments counter exactly 5 times', async () => {
513
+ if (!serverOnline)
514
+ return;
515
+ await mc.command('/function range_test:__load');
516
+ await mc.command('/function range_test:count_to_five');
517
+ await mc.ticks(10);
518
+ const counter = await mc.scoreboard('#range', 'counter');
519
+ expect(counter).toBe(5);
520
+ console.log(` for-range 0..5 → counter=${counter} (expect 5) ✓`);
521
+ });
522
+ // Scenario F: function return value propagation
523
+ // Verifies: $ret from callee is correctly captured in caller's variable
524
+ test('F: function return value — triple(4) = 12', async () => {
525
+ if (!serverOnline)
526
+ return;
527
+ await mc.command('/function nested_test:__load');
528
+ await mc.command('/function nested_test:run_nested');
529
+ await mc.ticks(10);
530
+ const result = await mc.scoreboard('#nested', 'result');
531
+ expect(result).toBe(12); // triple(4) = 4*3 = 12
532
+ console.log(` triple(4) = ${result} (expect 12) ✓`);
533
+ });
534
+ // Scenario G: match dispatches to correct branch
535
+ // Verifies: match statement selects right arm for values 1, 2, 3, and default
536
+ test('G: match statement dispatches to correct branch', async () => {
537
+ if (!serverOnline)
538
+ return;
539
+ await mc.command('/function match_test:__load');
540
+ // Test match on value 2
541
+ await mc.command('/scoreboard players set $p0 rs 2');
542
+ await mc.command('/function match_test:classify');
543
+ await mc.ticks(5);
544
+ let out = await mc.scoreboard('#match', 'out');
545
+ expect(out).toBe(20);
546
+ console.log(` match(2) → out=${out} (expect 20) ✓`);
547
+ // Test match on value 3
548
+ await mc.command('/scoreboard players set $p0 rs 3');
549
+ await mc.command('/function match_test:classify');
550
+ await mc.ticks(5);
551
+ out = await mc.scoreboard('#match', 'out');
552
+ expect(out).toBe(30);
553
+ console.log(` match(3) → out=${out} (expect 30) ✓`);
554
+ // Test default branch (value 99)
555
+ await mc.command('/scoreboard players set $p0 rs 99');
556
+ await mc.command('/function match_test:classify');
557
+ await mc.ticks(5);
558
+ out = await mc.scoreboard('#match', 'out');
559
+ expect(out).toBe(-1);
560
+ console.log(` match(99) → out=${out} (expect -1, default) ✓`);
561
+ });
562
+ // Scenario H: while loop counts down from 10 to 0
563
+ // Verifies: while loop body executes correct number of iterations
564
+ test('H: while loop counts down 10 steps', async () => {
565
+ if (!serverOnline)
566
+ return;
567
+ await mc.command('/function while_test:__load');
568
+ await mc.command('/function while_test:countdown');
569
+ await mc.ticks(10);
570
+ const i = await mc.scoreboard('#wloop', 'i');
571
+ const steps = await mc.scoreboard('#wloop', 'steps');
572
+ expect(i).toBe(0);
573
+ expect(steps).toBe(10);
574
+ console.log(` while countdown: i=${i} (expect 0), steps=${steps} (expect 10) ✓`);
575
+ });
576
+ // Scenario I: nested if/else boundary classification
577
+ // Verifies: correct branch taken at boundaries (0, 50, 100)
578
+ test('I: nested if/else boundary classification', async () => {
579
+ if (!serverOnline)
580
+ return;
581
+ await mc.command('/function boundary_test:__load');
582
+ // Test x=0 → tier 0
583
+ await mc.command('/scoreboard players set #boundary input 0');
584
+ await mc.command('/function boundary_test:classify_score');
585
+ await mc.ticks(5);
586
+ let tier = await mc.scoreboard('#boundary', 'tier');
587
+ expect(tier).toBe(0);
588
+ console.log(` classify(0) → tier=${tier} (expect 0) ✓`);
589
+ // Test x=50 → tier 1 (> 0 but not > 50)
590
+ await mc.command('/scoreboard players set #boundary input 50');
591
+ await mc.command('/function boundary_test:classify_score');
592
+ await mc.ticks(5);
593
+ tier = await mc.scoreboard('#boundary', 'tier');
594
+ expect(tier).toBe(1);
595
+ console.log(` classify(50) → tier=${tier} (expect 1) ✓`);
596
+ // Test x=51 → tier 2 (> 50 but not > 100)
597
+ await mc.command('/scoreboard players set #boundary input 51');
598
+ await mc.command('/function boundary_test:classify_score');
599
+ await mc.ticks(5);
600
+ tier = await mc.scoreboard('#boundary', 'tier');
601
+ expect(tier).toBe(2);
602
+ console.log(` classify(51) → tier=${tier} (expect 2) ✓`);
603
+ // Test x=101 → tier 3
604
+ await mc.command('/scoreboard players set #boundary input 101');
605
+ await mc.command('/function boundary_test:classify_score');
606
+ await mc.ticks(5);
607
+ tier = await mc.scoreboard('#boundary', 'tier');
608
+ expect(tier).toBe(3);
609
+ console.log(` classify(101) → tier=${tier} (expect 3) ✓`);
610
+ });
611
+ // Scenario J: entity summon and query
612
+ // Verifies: entities spawned via compiled function are queryable
613
+ test('J: summon entities via compiled function', async () => {
614
+ if (!serverOnline)
615
+ return;
616
+ await mc.command('/kill @e[type=minecraft:armor_stand]');
617
+ await mc.ticks(2);
618
+ await mc.command('/function tag_test:__load');
619
+ await mc.command('/function tag_test:tag_entities');
620
+ await mc.ticks(5);
621
+ const stands = await mc.entities('@e[type=minecraft:armor_stand]');
622
+ expect(stands.length).toBe(3);
623
+ console.log(` Summoned 3 armor_stands via tag_test, found: ${stands.length} ✓`);
624
+ await mc.command('/kill @e[type=minecraft:armor_stand]');
625
+ });
626
+ // Scenario K: arithmetic order of operations
627
+ // Verifies: MC scoreboard arithmetic matches expected evaluation order
628
+ test('K: arithmetic order of operations', async () => {
629
+ if (!serverOnline)
630
+ return;
631
+ await mc.command('/function order_test:__load');
632
+ await mc.command('/function order_test:math_order');
633
+ await mc.ticks(10);
634
+ const r1 = await mc.scoreboard('#order', 'r1');
635
+ const r2 = await mc.scoreboard('#order', 'r2');
636
+ const r3 = await mc.scoreboard('#order', 'r3');
637
+ // a + b * c = 2 + 3*4 = 14 (if precedence respected) or (2+3)*4 = 20 (left-to-right)
638
+ // MC scoreboard does left-to-right, so compiler may emit either depending on lowering
639
+ // (a + b) * c = 5 * 4 = 20 (explicit parens)
640
+ expect(r2).toBe(20); // This one is unambiguous
641
+ // 100 / 3 = 33 (integer division)
642
+ expect(r3).toBe(33);
643
+ console.log(` r1=${r1}, r2=${r2} (expect 20), r3=${r3} (expect 33) ✓`);
644
+ });
645
+ // Scenario L: scoreboard read-modify-write chain (1 → 2 → 4 → 8)
646
+ // Verifies: sequential RMW operations don't lose intermediate state
647
+ test('L: scoreboard RMW chain — 1*2*2*2 = 8', async () => {
648
+ if (!serverOnline)
649
+ return;
650
+ await mc.command('/function rmw_test:__load');
651
+ await mc.command('/function rmw_test:chain_rmw');
652
+ await mc.ticks(10);
653
+ const v = await mc.scoreboard('#rmw', 'v');
654
+ expect(v).toBe(8);
655
+ console.log(` RMW chain: 1→2→4→8, got ${v} (expect 8) ✓`);
656
+ });
657
+ });
658
+ describe('MC Integration - New Features', () => {
659
+ test('impl-test.mcrs: Timer::new/start/tick/done works in-game', async () => {
660
+ if (!serverOnline)
661
+ return;
662
+ await mc.command('/scoreboard players set #impl done 0');
663
+ await mc.command('/scoreboard players set timer_ticks rs 0');
664
+ await mc.command('/scoreboard players set timer_active rs 0');
665
+ await mc.command('/function impl_test:__load').catch(() => { });
666
+ await mc.command('/function impl_test:test');
667
+ await mc.ticks(5);
668
+ const done = await mc.scoreboard('#impl', 'done');
669
+ const ticks = await mc.scoreboard('timer_ticks', 'rs');
670
+ expect(done).toBe(1);
671
+ expect(ticks).toBe(3);
672
+ });
673
+ test('timeout-test.mcrs: setTimeout executes after delay', async () => {
674
+ if (!serverOnline)
675
+ return;
676
+ await mc.command('/scoreboard players set #timeout fired 0');
677
+ await mc.command('/function timeout_test:__load').catch(() => { });
678
+ await mc.command('/function timeout_test:start');
679
+ await mc.ticks(10);
680
+ expect(await mc.scoreboard('#timeout', 'fired')).toBe(0);
681
+ await mc.ticks(15);
682
+ expect(await mc.scoreboard('#timeout', 'fired')).toBe(1);
683
+ });
684
+ test('interval-test.mcrs: setInterval repeats on schedule', async () => {
685
+ if (!serverOnline)
686
+ return;
687
+ await mc.command('/scoreboard players set #interval ticks 0');
688
+ await mc.command('/function interval_test:__load').catch(() => { });
689
+ await mc.command('/function interval_test:start');
690
+ await mc.ticks(70);
691
+ const count = await mc.scoreboard('#interval', 'ticks');
692
+ expect(count).toBeGreaterThanOrEqual(3);
693
+ expect(count).toBeLessThanOrEqual(3);
694
+ });
695
+ test('is-check-test.mcrs: foreach is-narrowing correctly matches entity types', async () => {
696
+ if (!serverOnline)
697
+ return;
698
+ await mc.fullReset({ clearArea: false, killEntities: true, resetScoreboards: false });
699
+ await mc.command('/forceload add 0 0').catch(() => { }); // Ensure chunk is loaded
700
+ await mc.command('/scoreboard objectives add armor_stands dummy').catch(() => { });
701
+ await mc.command('/scoreboard objectives add items dummy').catch(() => { });
702
+ await mc.command('/scoreboard players set #is_check armor_stands 0');
703
+ await mc.command('/scoreboard players set #is_check items 0');
704
+ await mc.command('/function is_check_test:__load').catch(() => { });
705
+ // Spawn 2 armor_stands and 1 item (all persist without players)
706
+ await mc.command('/summon minecraft:armor_stand 0 65 0 {Tags:["is_check_target"],NoGravity:1b}');
707
+ await mc.command('/summon minecraft:armor_stand 2 65 0 {Tags:["is_check_target"],NoGravity:1b}');
708
+ await mc.command('/summon minecraft:item 4 65 0 {Tags:["is_check_target"],Item:{id:"minecraft:stone",count:1},Age:-32768}');
709
+ await mc.ticks(5);
710
+ await mc.command('/function is_check_test:check_types');
711
+ await mc.ticks(5);
712
+ const armorStands = await mc.scoreboard('#is_check', 'armor_stands');
713
+ const items = await mc.scoreboard('#is_check', 'items');
714
+ expect(armorStands).toBe(2); // 2 armor_stands matched
715
+ expect(items).toBe(1); // 1 item matched
716
+ await mc.command('/function is_check_test:cleanup').catch(() => { });
717
+ });
718
+ test('event-test.mcrs: @on(PlayerDeath) compiles and loads', async () => {
719
+ if (!serverOnline)
720
+ return;
721
+ // Verify the event system compiles correctly
722
+ await mc.command('/function event_test:__load').catch(() => { });
723
+ await mc.ticks(5);
724
+ // Verify the trigger function exists
725
+ const result = await mc.command('/function event_test:trigger_fake_death');
726
+ expect(result.ok).toBe(true);
727
+ // Verify __tick exists (event dispatcher)
728
+ const tickResult = await mc.command('/function event_test:__tick').catch(() => ({ ok: false }));
729
+ expect(tickResult.ok).toBe(true);
730
+ });
731
+ });
732
+ describe('MC Integration - Extended Coverage', () => {
733
+ test('struct-test.mcrs: struct instantiation and field access', async () => {
734
+ if (!serverOnline)
735
+ return;
736
+ writeFixtureFile('struct-test.mcrs', 'struct_test');
737
+ await mc.reload();
738
+ await mc.command('/function struct_test:__load').catch(() => { });
739
+ await mc.command('/function struct_test:test_struct');
740
+ await mc.ticks(5);
741
+ expect(await mc.scoreboard('#struct_x', 'rs')).toBe(10);
742
+ expect(await mc.scoreboard('#struct_y', 'rs')).toBe(64);
743
+ expect(await mc.scoreboard('#struct_z', 'rs')).toBe(-5);
744
+ expect(await mc.scoreboard('#struct_x2', 'rs')).toBe(15); // 10+5
745
+ expect(await mc.scoreboard('#struct_z2', 'rs')).toBe(-10); // -5*2
746
+ expect(await mc.scoreboard('#struct_alive', 'rs')).toBe(1);
747
+ expect(await mc.scoreboard('#struct_score', 'rs')).toBe(100);
748
+ });
749
+ test('enum-test.mcrs: enum values and match', async () => {
750
+ if (!serverOnline)
751
+ return;
752
+ writeFixtureFile('enum-test.mcrs', 'enum_test');
753
+ await mc.reload();
754
+ await mc.command('/function enum_test:__load').catch(() => { });
755
+ await mc.command('/function enum_test:test_enum');
756
+ await mc.ticks(5);
757
+ expect(await mc.scoreboard('#enum_phase', 'rs')).toBe(2); // Playing=2
758
+ expect(await mc.scoreboard('#enum_match', 'rs')).toBe(2); // matched Playing
759
+ expect(await mc.scoreboard('#enum_rank', 'rs')).toBe(10); // Diamond=10
760
+ expect(await mc.scoreboard('#enum_high', 'rs')).toBe(1); // Diamond > Gold
761
+ });
762
+ test('array-test.mcrs: array operations', async () => {
763
+ if (!serverOnline)
764
+ return;
765
+ writeFixtureFile('array-test.mcrs', 'array_test');
766
+ await mc.reload();
767
+ await mc.command('/function array_test:__load').catch(() => { });
768
+ await mc.command('/function array_test:test_array');
769
+ await mc.ticks(5);
770
+ expect(await mc.scoreboard('#arr_0', 'rs')).toBe(10);
771
+ expect(await mc.scoreboard('#arr_2', 'rs')).toBe(30);
772
+ expect(await mc.scoreboard('#arr_4', 'rs')).toBe(50);
773
+ expect(await mc.scoreboard('#arr_len', 'rs')).toBe(5);
774
+ expect(await mc.scoreboard('#arr_sum', 'rs')).toBe(150); // 10+20+30+40+50
775
+ expect(await mc.scoreboard('#arr_push', 'rs')).toBe(4); // [1,2,3,4].len
776
+ expect(await mc.scoreboard('#arr_pop', 'rs')).toBe(4); // popped value
777
+ });
778
+ test('break-continue-test.mcrs: break and continue statements', async () => {
779
+ if (!serverOnline)
780
+ return;
781
+ writeFixtureFile('break-continue-test.mcrs', 'break_continue_test');
782
+ await mc.reload();
783
+ await mc.command('/function break_continue_test:__load').catch(() => { });
784
+ await mc.command('/function break_continue_test:test_break_continue');
785
+ await mc.ticks(10);
786
+ expect(await mc.scoreboard('#break_at', 'rs')).toBe(5);
787
+ expect(await mc.scoreboard('#sum_evens', 'rs')).toBe(20); // 0+2+4+6+8
788
+ expect(await mc.scoreboard('#while_break', 'rs')).toBe(7);
789
+ expect(await mc.scoreboard('#nested_break', 'rs')).toBe(3); // outer completes 3 times
790
+ });
791
+ test('match-range-test.mcrs: match with range patterns', async () => {
792
+ if (!serverOnline)
793
+ return;
794
+ writeFixtureFile('match-range-test.mcrs', 'match_range_test');
795
+ await mc.reload();
796
+ await mc.command('/function match_range_test:__load').catch(() => { });
797
+ await mc.command('/function match_range_test:test_match_range');
798
+ await mc.ticks(5);
799
+ expect(await mc.scoreboard('#grade', 'rs')).toBe(4); // score=85 → B
800
+ expect(await mc.scoreboard('#boundary_59', 'rs')).toBe(1); // 59 matches 0..59
801
+ expect(await mc.scoreboard('#boundary_60', 'rs')).toBe(2); // 60 matches 60..100
802
+ expect(await mc.scoreboard('#neg_range', 'rs')).toBe(1); // -5 matches ..0
803
+ });
804
+ test('foreach-at-test.mcrs: foreach with at @s context', async () => {
805
+ if (!serverOnline)
806
+ return;
807
+ writeFixtureFile('foreach-at-test.mcrs', 'foreach_at_test');
808
+ await mc.reload();
809
+ await mc.fullReset({ clearArea: false, killEntities: true, resetScoreboards: false });
810
+ await mc.command('/function foreach_at_test:setup').catch(() => { });
811
+ await mc.command('/function foreach_at_test:test_foreach_at');
812
+ await mc.ticks(10);
813
+ expect(await mc.scoreboard('#foreach_count', 'rs')).toBe(3);
814
+ expect(await mc.scoreboard('#foreach_at_count', 'rs')).toBe(3);
815
+ });
816
+ });
817
+ //# sourceMappingURL=mc-integration.test.js.map