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.
- package/.claude/commands/build-test.md +10 -0
- package/.claude/commands/deploy-demo.md +12 -0
- package/.claude/commands/stage-status.md +13 -0
- package/.claude/settings.json +12 -0
- package/.github/workflows/ci.yml +1 -0
- package/CLAUDE.md +231 -0
- package/README.md +29 -28
- package/README.zh.md +28 -28
- package/demo.gif +0 -0
- package/dist/cli.js +2 -554
- package/dist/compile.js +2 -266
- package/dist/index.js +2 -159
- package/dist/lexer/index.js +9 -1
- package/dist/lowering/index.js +22 -5
- package/dist/src/__tests__/cli.test.d.ts +1 -0
- package/dist/src/__tests__/cli.test.js +104 -0
- package/dist/src/__tests__/codegen.test.d.ts +1 -0
- package/dist/src/__tests__/codegen.test.js +152 -0
- package/dist/src/__tests__/compile-all.test.d.ts +10 -0
- package/dist/src/__tests__/compile-all.test.js +108 -0
- package/dist/src/__tests__/dce.test.d.ts +1 -0
- package/dist/src/__tests__/dce.test.js +102 -0
- package/dist/src/__tests__/diagnostics.test.d.ts +4 -0
- package/dist/src/__tests__/diagnostics.test.js +177 -0
- package/dist/src/__tests__/e2e.test.d.ts +6 -0
- package/dist/src/__tests__/e2e.test.js +1789 -0
- package/dist/src/__tests__/entity-types.test.d.ts +1 -0
- package/dist/src/__tests__/entity-types.test.js +203 -0
- package/dist/src/__tests__/formatter.test.d.ts +1 -0
- package/dist/src/__tests__/formatter.test.js +40 -0
- package/dist/src/__tests__/lexer.test.d.ts +1 -0
- package/dist/src/__tests__/lexer.test.js +343 -0
- package/dist/src/__tests__/lowering.test.d.ts +1 -0
- package/dist/src/__tests__/lowering.test.js +1015 -0
- package/dist/src/__tests__/macro.test.d.ts +8 -0
- package/dist/src/__tests__/macro.test.js +306 -0
- package/dist/src/__tests__/mc-integration.test.d.ts +12 -0
- package/dist/src/__tests__/mc-integration.test.js +817 -0
- package/dist/src/__tests__/mc-syntax.test.d.ts +1 -0
- package/dist/src/__tests__/mc-syntax.test.js +124 -0
- package/dist/src/__tests__/nbt.test.d.ts +1 -0
- package/dist/src/__tests__/nbt.test.js +82 -0
- package/dist/src/__tests__/optimizer-advanced.test.d.ts +1 -0
- package/dist/src/__tests__/optimizer-advanced.test.js +124 -0
- package/dist/src/__tests__/optimizer.test.d.ts +1 -0
- package/dist/src/__tests__/optimizer.test.js +149 -0
- package/dist/src/__tests__/parser.test.d.ts +1 -0
- package/dist/src/__tests__/parser.test.js +807 -0
- package/dist/src/__tests__/repl.test.d.ts +1 -0
- package/dist/src/__tests__/repl.test.js +27 -0
- package/dist/src/__tests__/runtime.test.d.ts +1 -0
- package/dist/src/__tests__/runtime.test.js +289 -0
- package/dist/src/__tests__/stdlib-advanced.test.d.ts +4 -0
- package/dist/src/__tests__/stdlib-advanced.test.js +374 -0
- package/dist/src/__tests__/stdlib-bigint.test.d.ts +7 -0
- package/dist/src/__tests__/stdlib-bigint.test.js +426 -0
- package/dist/src/__tests__/stdlib-math.test.d.ts +7 -0
- package/dist/src/__tests__/stdlib-math.test.js +351 -0
- package/dist/src/__tests__/stdlib-vec.test.d.ts +4 -0
- package/dist/src/__tests__/stdlib-vec.test.js +263 -0
- package/dist/src/__tests__/structure-optimizer.test.d.ts +1 -0
- package/dist/src/__tests__/structure-optimizer.test.js +33 -0
- package/dist/src/__tests__/typechecker.test.d.ts +1 -0
- package/dist/src/__tests__/typechecker.test.js +552 -0
- package/dist/src/__tests__/var-allocator.test.d.ts +1 -0
- package/dist/src/__tests__/var-allocator.test.js +69 -0
- package/dist/src/ast/types.d.ts +515 -0
- package/dist/src/ast/types.js +9 -0
- package/dist/src/builtins/metadata.d.ts +36 -0
- package/dist/src/builtins/metadata.js +1014 -0
- package/dist/src/cli.d.ts +11 -0
- package/dist/src/cli.js +443 -0
- package/dist/src/codegen/cmdblock/index.d.ts +26 -0
- package/dist/src/codegen/cmdblock/index.js +45 -0
- package/dist/src/codegen/mcfunction/index.d.ts +40 -0
- package/dist/src/codegen/mcfunction/index.js +606 -0
- package/dist/src/codegen/structure/index.d.ts +24 -0
- package/dist/src/codegen/structure/index.js +279 -0
- package/dist/src/codegen/var-allocator.d.ts +45 -0
- package/dist/src/codegen/var-allocator.js +104 -0
- package/dist/src/compile.d.ts +37 -0
- package/dist/src/compile.js +165 -0
- package/dist/src/diagnostics/index.d.ts +44 -0
- package/dist/src/diagnostics/index.js +140 -0
- package/dist/src/events/types.d.ts +35 -0
- package/dist/src/events/types.js +59 -0
- package/dist/src/formatter/index.d.ts +1 -0
- package/dist/src/formatter/index.js +26 -0
- package/dist/src/index.d.ts +22 -0
- package/dist/src/index.js +45 -0
- package/dist/src/ir/builder.d.ts +33 -0
- package/dist/src/ir/builder.js +99 -0
- package/dist/src/ir/types.d.ts +132 -0
- package/dist/src/ir/types.js +15 -0
- package/dist/src/lexer/index.d.ts +37 -0
- package/dist/src/lexer/index.js +569 -0
- package/dist/src/lowering/index.d.ts +188 -0
- package/dist/src/lowering/index.js +3405 -0
- package/dist/src/mc-test/client.d.ts +128 -0
- package/dist/src/mc-test/client.js +174 -0
- package/dist/src/mc-test/runner.d.ts +28 -0
- package/dist/src/mc-test/runner.js +151 -0
- package/dist/src/mc-test/setup.d.ts +11 -0
- package/dist/src/mc-test/setup.js +98 -0
- package/dist/src/mc-validator/index.d.ts +17 -0
- package/dist/src/mc-validator/index.js +322 -0
- package/dist/src/nbt/index.d.ts +86 -0
- package/dist/src/nbt/index.js +250 -0
- package/dist/src/optimizer/commands.d.ts +38 -0
- package/dist/src/optimizer/commands.js +451 -0
- package/dist/src/optimizer/dce.d.ts +34 -0
- package/dist/src/optimizer/dce.js +639 -0
- package/dist/src/optimizer/passes.d.ts +34 -0
- package/dist/src/optimizer/passes.js +243 -0
- package/dist/src/optimizer/structure.d.ts +9 -0
- package/dist/src/optimizer/structure.js +356 -0
- package/dist/src/parser/index.d.ts +93 -0
- package/dist/src/parser/index.js +1687 -0
- package/dist/src/repl.d.ts +16 -0
- package/dist/src/repl.js +165 -0
- package/dist/src/runtime/index.d.ts +107 -0
- package/dist/src/runtime/index.js +1409 -0
- package/dist/src/typechecker/index.d.ts +61 -0
- package/dist/src/typechecker/index.js +1034 -0
- package/dist/src/types/entity-hierarchy.d.ts +29 -0
- package/dist/src/types/entity-hierarchy.js +107 -0
- package/dist/src2/__tests__/e2e/basic.test.d.ts +8 -0
- package/dist/src2/__tests__/e2e/basic.test.js +140 -0
- package/dist/src2/__tests__/e2e/macros.test.d.ts +9 -0
- package/dist/src2/__tests__/e2e/macros.test.js +182 -0
- package/dist/src2/__tests__/e2e/migrate.test.d.ts +13 -0
- package/dist/src2/__tests__/e2e/migrate.test.js +2739 -0
- package/dist/src2/__tests__/hir/desugar.test.d.ts +1 -0
- package/dist/src2/__tests__/hir/desugar.test.js +234 -0
- package/dist/src2/__tests__/lir/lower.test.d.ts +1 -0
- package/dist/src2/__tests__/lir/lower.test.js +559 -0
- package/dist/src2/__tests__/lir/types.test.d.ts +1 -0
- package/dist/src2/__tests__/lir/types.test.js +185 -0
- package/dist/src2/__tests__/lir/verify.test.d.ts +1 -0
- package/dist/src2/__tests__/lir/verify.test.js +221 -0
- package/dist/src2/__tests__/mir/arithmetic.test.d.ts +1 -0
- package/dist/src2/__tests__/mir/arithmetic.test.js +130 -0
- package/dist/src2/__tests__/mir/control-flow.test.d.ts +1 -0
- package/dist/src2/__tests__/mir/control-flow.test.js +205 -0
- package/dist/src2/__tests__/mir/verify.test.d.ts +1 -0
- package/dist/src2/__tests__/mir/verify.test.js +223 -0
- package/dist/src2/__tests__/optimizer/block_merge.test.d.ts +1 -0
- package/dist/src2/__tests__/optimizer/block_merge.test.js +78 -0
- package/dist/src2/__tests__/optimizer/branch_simplify.test.d.ts +1 -0
- package/dist/src2/__tests__/optimizer/branch_simplify.test.js +58 -0
- package/dist/src2/__tests__/optimizer/constant_fold.test.d.ts +1 -0
- package/dist/src2/__tests__/optimizer/constant_fold.test.js +131 -0
- package/dist/src2/__tests__/optimizer/copy_prop.test.d.ts +1 -0
- package/dist/src2/__tests__/optimizer/copy_prop.test.js +91 -0
- package/dist/src2/__tests__/optimizer/dce.test.d.ts +1 -0
- package/dist/src2/__tests__/optimizer/dce.test.js +76 -0
- package/dist/src2/__tests__/optimizer/pipeline.test.d.ts +1 -0
- package/dist/src2/__tests__/optimizer/pipeline.test.js +102 -0
- package/dist/src2/emit/compile.d.ts +19 -0
- package/dist/src2/emit/compile.js +80 -0
- package/dist/src2/emit/index.d.ts +17 -0
- package/dist/src2/emit/index.js +172 -0
- package/dist/src2/hir/lower.d.ts +15 -0
- package/dist/src2/hir/lower.js +378 -0
- package/dist/src2/hir/types.d.ts +373 -0
- package/dist/src2/hir/types.js +16 -0
- package/dist/src2/lir/lower.d.ts +15 -0
- package/dist/src2/lir/lower.js +453 -0
- package/dist/src2/lir/types.d.ts +136 -0
- package/dist/src2/lir/types.js +11 -0
- package/dist/src2/lir/verify.d.ts +14 -0
- package/dist/src2/lir/verify.js +113 -0
- package/dist/src2/mir/lower.d.ts +9 -0
- package/dist/src2/mir/lower.js +1030 -0
- package/dist/src2/mir/macro.d.ts +22 -0
- package/dist/src2/mir/macro.js +168 -0
- package/dist/src2/mir/types.d.ts +183 -0
- package/dist/src2/mir/types.js +11 -0
- package/dist/src2/mir/verify.d.ts +16 -0
- package/dist/src2/mir/verify.js +216 -0
- package/dist/src2/optimizer/block_merge.d.ts +12 -0
- package/dist/src2/optimizer/block_merge.js +84 -0
- package/dist/src2/optimizer/branch_simplify.d.ts +9 -0
- package/dist/src2/optimizer/branch_simplify.js +28 -0
- package/dist/src2/optimizer/constant_fold.d.ts +10 -0
- package/dist/src2/optimizer/constant_fold.js +85 -0
- package/dist/src2/optimizer/copy_prop.d.ts +9 -0
- package/dist/src2/optimizer/copy_prop.js +113 -0
- package/dist/src2/optimizer/dce.d.ts +8 -0
- package/dist/src2/optimizer/dce.js +155 -0
- package/dist/src2/optimizer/pipeline.d.ts +10 -0
- package/dist/src2/optimizer/pipeline.js +42 -0
- package/dist/tsconfig.tsbuildinfo +1 -0
- package/docs/compiler-pipeline-redesign.md +2243 -0
- package/docs/optimization-ideas.md +1076 -0
- package/editors/vscode/package-lock.json +3 -3
- package/editors/vscode/package.json +1 -1
- package/examples/readme-demo.mcrs +44 -66
- package/jest.config.js +1 -1
- package/package.json +6 -5
- package/scripts/postbuild.js +15 -0
- package/src/__tests__/cli.test.ts +8 -220
- package/src/__tests__/dce.test.ts +11 -56
- package/src/__tests__/diagnostics.test.ts +59 -38
- package/src/__tests__/mc-integration.test.ts +1 -2
- package/src/ast/types.ts +6 -1
- package/src/cli.ts +29 -156
- package/src/compile.ts +6 -162
- package/src/index.ts +14 -178
- package/src/lexer/index.ts +9 -1
- package/src/mc-test/runner.ts +4 -3
- package/src/parser/index.ts +1 -1
- package/src/repl.ts +1 -1
- package/src/runtime/index.ts +1 -1
- package/src2/__tests__/e2e/basic.test.ts +154 -0
- package/src2/__tests__/e2e/macros.test.ts +199 -0
- package/src2/__tests__/e2e/migrate.test.ts +3008 -0
- package/src2/__tests__/hir/desugar.test.ts +263 -0
- package/src2/__tests__/lir/lower.test.ts +619 -0
- package/src2/__tests__/lir/types.test.ts +207 -0
- package/src2/__tests__/lir/verify.test.ts +249 -0
- package/src2/__tests__/mir/arithmetic.test.ts +156 -0
- package/src2/__tests__/mir/control-flow.test.ts +242 -0
- package/src2/__tests__/mir/verify.test.ts +254 -0
- package/src2/__tests__/optimizer/block_merge.test.ts +84 -0
- package/src2/__tests__/optimizer/branch_simplify.test.ts +64 -0
- package/src2/__tests__/optimizer/constant_fold.test.ts +145 -0
- package/src2/__tests__/optimizer/copy_prop.test.ts +99 -0
- package/src2/__tests__/optimizer/dce.test.ts +83 -0
- package/src2/__tests__/optimizer/pipeline.test.ts +116 -0
- package/src2/emit/compile.ts +99 -0
- package/src2/emit/index.ts +222 -0
- package/src2/hir/lower.ts +428 -0
- package/src2/hir/types.ts +216 -0
- package/src2/lir/lower.ts +556 -0
- package/src2/lir/types.ts +109 -0
- package/src2/lir/verify.ts +129 -0
- package/src2/mir/lower.ts +1160 -0
- package/src2/mir/macro.ts +167 -0
- package/src2/mir/types.ts +106 -0
- package/src2/mir/verify.ts +218 -0
- package/src2/optimizer/block_merge.ts +93 -0
- package/src2/optimizer/branch_simplify.ts +27 -0
- package/src2/optimizer/constant_fold.ts +88 -0
- package/src2/optimizer/copy_prop.ts +106 -0
- package/src2/optimizer/dce.ts +133 -0
- package/src2/optimizer/pipeline.ts +44 -0
- package/tsconfig.json +2 -2
- package/src/__tests__/codegen.test.ts +0 -161
- package/src/__tests__/e2e.test.ts +0 -2039
- package/src/__tests__/entity-types.test.ts +0 -236
- package/src/__tests__/lowering.test.ts +0 -1185
- package/src/__tests__/macro.test.ts +0 -343
- package/src/__tests__/nbt.test.ts +0 -58
- package/src/__tests__/optimizer-advanced.test.ts +0 -144
- package/src/__tests__/optimizer.test.ts +0 -162
- package/src/__tests__/runtime.test.ts +0 -305
- package/src/__tests__/stdlib-advanced.test.ts +0 -379
- package/src/__tests__/stdlib-bigint.test.ts +0 -427
- package/src/__tests__/stdlib-math.test.ts +0 -374
- package/src/__tests__/stdlib-vec.test.ts +0 -259
- package/src/__tests__/structure-optimizer.test.ts +0 -38
- package/src/__tests__/var-allocator.test.ts +0 -75
- package/src/codegen/cmdblock/index.ts +0 -63
- package/src/codegen/mcfunction/index.ts +0 -662
- package/src/codegen/structure/index.ts +0 -346
- package/src/codegen/var-allocator.ts +0 -104
- package/src/ir/builder.ts +0 -116
- package/src/ir/types.ts +0 -134
- package/src/lowering/index.ts +0 -3860
- package/src/optimizer/commands.ts +0 -534
- package/src/optimizer/dce.ts +0 -679
- package/src/optimizer/passes.ts +0 -250
- 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
|