1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
|
import { mkdir, mkdtemp, readFile, rm, writeFile } from "node:fs/promises";
import { tmpdir } from "node:os";
import { join } from "node:path";
import { localExecBackend } from "@dispatch/exec-backend";
import { createLogger, type ToolExecuteContext } from "@dispatch/kernel";
import { afterEach, beforeEach, describe, expect, it } from "vitest";
import { createWriteFileTool, decideOverwrite, validateArgs } from "./write-file.js";
function stubCtx(overrides?: Partial<ToolExecuteContext>): ToolExecuteContext {
return {
toolCallId: "test-call-1",
onOutput: () => {},
signal: AbortSignal.timeout(5000),
log: createLogger(
{ extensionId: "test" },
{ emit: () => {} },
{ now: () => 0, newId: () => "id" },
),
...overrides,
};
}
/**
* Build a write_file tool wired to the real local ExecBackend (node:fs,
* behavior-identical to today's inline calls). No `@dispatch/*` mocking — the
* real fs edge is exercised, matching the constitution's strict-core rule.
*/
function makeTool(workdir: string) {
return createWriteFileTool({ resolveBackend: () => localExecBackend, workdir });
}
let workdir: string;
beforeEach(async () => {
workdir = await mkdtemp(join(tmpdir(), "tool-write-file-test-"));
});
afterEach(async () => {
await rm(workdir, { recursive: true, force: true });
});
describe("decideOverwrite", () => {
it("returns create when file absent and overwrite is false", () => {
expect(decideOverwrite(false, false)).toBe("create");
});
it("returns create when file absent and overwrite is false (default)", () => {
expect(decideOverwrite(false, false)).toBe("create");
});
it("returns error when file exists and overwrite is false", () => {
const result = decideOverwrite(true, false);
expect(typeof result).toBe("object");
if (typeof result === "object") {
expect(result.error).toContain("already exists");
}
});
it("returns overwrite when file exists and overwrite is true", () => {
expect(decideOverwrite(true, true)).toBe("overwrite");
});
it("returns error when file absent and overwrite is true", () => {
const result = decideOverwrite(false, true);
expect(typeof result).toBe("object");
if (typeof result === "object") {
expect(result.error).toContain("does not exist");
}
});
it("covers all four rows of the truth table", () => {
expect(decideOverwrite(false, false)).toBe("create");
expect(decideOverwrite(true, false)).toEqual(
expect.objectContaining({ error: expect.any(String) }),
);
expect(decideOverwrite(true, true)).toBe("overwrite");
expect(decideOverwrite(false, true)).toEqual(
expect.objectContaining({ error: expect.any(String) }),
);
});
});
describe("validateArgs", () => {
it("returns validated args for valid input", () => {
const result = validateArgs({ path: "foo.txt", content: "hello" });
expect(result).toEqual({ path: "foo.txt", content: "hello", overwrite: false });
});
it("parses overwrite as true", () => {
const result = validateArgs({ path: "foo.txt", content: "x", overwrite: true });
expect(result).toEqual({ path: "foo.txt", content: "x", overwrite: true });
});
it("defaults overwrite to false", () => {
const result = validateArgs({ path: "foo.txt", content: "x" });
expect(result).toEqual({ path: "foo.txt", content: "x", overwrite: false });
});
it("accepts empty string content", () => {
const result = validateArgs({ path: "foo.txt", content: "" });
expect(result).toEqual({ path: "foo.txt", content: "", overwrite: false });
});
it("returns error for null args", () => {
expect(validateArgs(null)).toHaveProperty("error");
});
it("returns error for missing path", () => {
expect(validateArgs({ content: "x" })).toHaveProperty("error");
});
it("returns error for missing content", () => {
expect(validateArgs({ path: "foo.txt" })).toHaveProperty("error");
});
it("returns error for non-string content", () => {
expect(validateArgs({ path: "foo.txt", content: 123 })).toHaveProperty("error");
});
it("returns error for non-boolean overwrite", () => {
expect(validateArgs({ path: "foo.txt", content: "x", overwrite: "yes" })).toHaveProperty(
"error",
);
});
});
describe("createWriteFileTool", () => {
it("creates a new file when overwrite is unset and the file is absent", async () => {
const tool = makeTool(workdir);
const result = await tool.execute({ path: "new-file.txt", content: "hello world" }, stubCtx());
expect(result.isError).toBeUndefined();
expect(result.content).toContain("Created");
const written = await readFile(join(workdir, "new-file.txt"), "utf8");
expect(written).toBe("hello world");
});
it("errors when the file exists and overwrite is unset", async () => {
await writeFile(join(workdir, "existing.txt"), "old content", "utf8");
const tool = makeTool(workdir);
const result = await tool.execute({ path: "existing.txt", content: "new content" }, stubCtx());
expect(result.isError).toBe(true);
expect(result.content).toContain("already exists");
expect(result.content).toContain("overwrite");
const unchanged = await readFile(join(workdir, "existing.txt"), "utf8");
expect(unchanged).toBe("old content");
});
it("overwrites an existing file when overwrite is true", async () => {
await writeFile(join(workdir, "existing.txt"), "old content", "utf8");
const tool = makeTool(workdir);
const result = await tool.execute(
{ path: "existing.txt", content: "new content", overwrite: true },
stubCtx(),
);
expect(result.isError).toBeUndefined();
expect(result.content).toContain("Overwrote");
const written = await readFile(join(workdir, "existing.txt"), "utf8");
expect(written).toBe("new content");
});
it("errors when overwrite is true but the file is absent", async () => {
const tool = makeTool(workdir);
const result = await tool.execute(
{ path: "nonexistent.txt", content: "data", overwrite: true },
stubCtx(),
);
expect(result.isError).toBe(true);
expect(result.content).toContain("does not exist");
});
it("errors when the parent directory does not exist", async () => {
const tool = makeTool(workdir);
const result = await tool.execute({ path: "no/such/dir/file.txt", content: "data" }, stubCtx());
expect(result.isError).toBe(true);
expect(result.content).toContain("Error");
});
it("concurrencySafe is false", () => {
const tool = makeTool(workdir);
expect(tool.concurrencySafe).toBe(false);
});
it("has correct name and parameters shape", () => {
const tool = makeTool(workdir);
expect(tool.name).toBe("write_file");
expect(tool.parameters.type).toBe("object");
expect(tool.parameters.required).toEqual(["path", "content"]);
expect(tool.parameters.properties?.path?.type).toBe("string");
expect(tool.parameters.properties?.content?.type).toBe("string");
expect(tool.parameters.properties?.overwrite?.type).toBe("boolean");
});
it("never throws on bad input (always returns ToolResult)", async () => {
const tool = makeTool(workdir);
const inputs = [null, undefined, 42, "string", {}, { path: "" }, { path: 123 }];
for (const input of inputs) {
const result = await tool.execute(input, stubCtx());
expect(result).toHaveProperty("content");
expect(typeof result.content).toBe("string");
}
});
it("respects ctx.cwd over baked workdir", async () => {
const ctxDir = await mkdtemp(join(tmpdir(), "ctx-cwd-test-"));
try {
const tool = makeTool(workdir);
const result = await tool.execute(
{ path: "ctx-file.txt", content: "from ctx" },
stubCtx({ cwd: ctxDir }),
);
expect(result.isError).toBeUndefined();
const written = await readFile(join(ctxDir, "ctx-file.txt"), "utf8");
expect(written).toBe("from ctx");
} finally {
await rm(ctxDir, { recursive: true, force: true });
}
});
it("writes empty content", async () => {
const tool = makeTool(workdir);
const result = await tool.execute({ path: "empty.txt", content: "" }, stubCtx());
expect(result.isError).toBeUndefined();
const written = await readFile(join(workdir, "empty.txt"), "utf8");
expect(written).toBe("");
});
it("writes content in subdirectory that exists", async () => {
await mkdir(join(workdir, "sub"));
const tool = makeTool(workdir);
const result = await tool.execute({ path: "sub/file.txt", content: "nested" }, stubCtx());
expect(result.isError).toBeUndefined();
const written = await readFile(join(workdir, "sub", "file.txt"), "utf8");
expect(written).toBe("nested");
});
});
|