✓
Passing This code compiles and runs correctly.
Code
// TEST: Manual disposal at | each |> is NOT allowed
// STATUS: IMPLEMENTED
// MUST_FAIL: KORU032 - Cannot dispose outer-scope resource inside repeating loop body
//
// When an obligation is created OUTSIDE a for-loop, it CANNOT be
// disposed inside | each |> (would run N times).
//
// Here: Open file before loop, try to close inside each - ERROR!
//
// Expected: KORU032 error about disposing in repeating context
~import "$std/control"
~import "$app/fs"
~app.fs:open(path: "outer.txt")
| opened f |>
std.control:for(0..3)
| each _ |>
app.fs:close(file: f.file) // ERROR! Would run 3 times
| closed _ |> _
| done |> _
pub fn main() void {}
Error Verification
Expected Error Pattern
Test expects KORU032 error when trying to dispose outer-scope resource inside loop bodyActual Compiler Output
error[KORU030]: Phantom state mismatch: argument 'file' has no tracked phantom state, but event requires '[!opened]' (consumption). Did you mean to pass a value with state 'app.fs:opened'?
--> phantom_semantic_check:16:0
❌ Compiler coordination error: Phantom semantic validation failed
error: CompilerCoordinationFailed
/Users/larsde/src/koru/tests/regression/300_ADVANCED_FEATURES/330_PHANTOM_TYPES/330_020_for_manual_each_error/backend.zig:9473:17: 0x1005664af in emit (backend)
return error.CompilerCoordinationFailed;
^
/Users/larsde/src/koru/tests/regression/300_ADVANCED_FEATURES/330_PHANTOM_TYPES/330_020_for_manual_each_error/backend.zig:9557:28: 0x1005672b7 in main (backend)
const generated_code = try RuntimeEmitter.emit(compile_allocator, final_ast);
^Imported Files
// Library module: fs
// Defines filesystem operations with cleanup obligations
const std = @import("std");
// File type with phantom states
const File = struct {
handle: i32,
};
// Open a file - returns opened! state (requires cleanup)
~pub event open { path: []const u8 }
| opened *File[opened!]
~proc open {
std.debug.print("Opening file: {s}\n", .{path});
const allocator = std.heap.page_allocator;
const f = allocator.create(File) catch unreachable;
f.* = File{ .handle = 42 };
return .{ .opened = f };
}
// Close a file - CONSUMES opened! state (disposes the resource)
~pub event close { file: *File[!opened] }
| closed
~proc close {
std.debug.print("Closing file\n", .{});
return .{ .closed = .{} };
}
Test Configuration
MUST_FAIL