@ai-hero/sandcastle 0.2.4 → 0.4.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/README.md +322 -42
- package/dist/AgentProvider.d.ts +0 -10
- package/dist/AgentProvider.d.ts.map +1 -1
- package/dist/AgentProvider.js +28 -45
- package/dist/AgentProvider.js.map +1 -1
- package/dist/DockerLifecycle.d.ts +5 -1
- package/dist/DockerLifecycle.d.ts.map +1 -1
- package/dist/DockerLifecycle.js +8 -1
- package/dist/DockerLifecycle.js.map +1 -1
- package/dist/InitService.d.ts.map +1 -1
- package/dist/InitService.js +2 -0
- package/dist/InitService.js.map +1 -1
- package/dist/Orchestrator.d.ts +0 -1
- package/dist/Orchestrator.d.ts.map +1 -1
- package/dist/Orchestrator.js +2 -13
- package/dist/Orchestrator.js.map +1 -1
- package/dist/SandboxFactory.d.ts +22 -11
- package/dist/SandboxFactory.d.ts.map +1 -1
- package/dist/SandboxFactory.js +166 -209
- package/dist/SandboxFactory.js.map +1 -1
- package/dist/SandboxProvider.d.ts +141 -0
- package/dist/SandboxProvider.d.ts.map +1 -0
- package/dist/SandboxProvider.js +27 -0
- package/dist/SandboxProvider.js.map +1 -0
- package/dist/cli.d.ts +7 -5
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +12 -11
- package/dist/cli.js.map +1 -1
- package/dist/createSandbox.d.ts +3 -2
- package/dist/createSandbox.d.ts.map +1 -1
- package/dist/createSandbox.js +19 -32
- package/dist/createSandbox.js.map +1 -1
- package/dist/index.d.ts +3 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +1 -0
- package/dist/index.js.map +1 -1
- package/dist/run.d.ts +4 -25
- package/dist/run.d.ts.map +1 -1
- package/dist/run.js +21 -35
- package/dist/run.js.map +1 -1
- package/dist/sandboxExec.d.ts +12 -0
- package/dist/sandboxExec.d.ts.map +1 -0
- package/dist/sandboxExec.js +26 -0
- package/dist/sandboxExec.js.map +1 -0
- package/dist/sandboxes/docker.d.ts +28 -0
- package/dist/sandboxes/docker.d.ts.map +1 -0
- package/dist/sandboxes/docker.js +134 -0
- package/dist/sandboxes/docker.js.map +1 -0
- package/dist/sandboxes/test-isolated.d.ts +21 -0
- package/dist/sandboxes/test-isolated.d.ts.map +1 -0
- package/dist/sandboxes/test-isolated.js +87 -0
- package/dist/sandboxes/test-isolated.js.map +1 -0
- package/dist/syncIn.d.ts +24 -0
- package/dist/syncIn.d.ts.map +1 -0
- package/dist/syncIn.js +107 -0
- package/dist/syncIn.js.map +1 -0
- package/dist/syncOut.d.ts +27 -0
- package/dist/syncOut.d.ts.map +1 -0
- package/dist/syncOut.js +271 -0
- package/dist/syncOut.js.map +1 -0
- package/dist/templates/blank/.env.example +1 -0
- package/dist/templates/blank/main.mts +2 -0
- package/dist/templates/parallel-planner/.env.example +1 -0
- package/dist/templates/parallel-planner/main.mts +5 -2
- package/dist/templates/sequential-reviewer/.env.example +1 -0
- package/dist/templates/sequential-reviewer/main.mts +3 -1
- package/dist/templates/simple-loop/.env.example +1 -0
- package/dist/templates/simple-loop/main.mts +4 -0
- package/package.json +5 -1
package/dist/syncOut.js
ADDED
|
@@ -0,0 +1,271 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Sync-out: extract changes from an isolated sandbox back to the host.
|
|
3
|
+
*
|
|
4
|
+
* Two-phase approach:
|
|
5
|
+
* 1. Save phase: eagerly save all artifacts (patches, diff, untracked files)
|
|
6
|
+
* to `.sandcastle/patches/<timestamp>/` before attempting to apply.
|
|
7
|
+
* 2. Apply phase: apply from the saved directory.
|
|
8
|
+
* - On success: clean up the patch directory.
|
|
9
|
+
* - On failure: preserve the patch directory and print recovery commands.
|
|
10
|
+
*
|
|
11
|
+
* Three-prong extraction within each phase:
|
|
12
|
+
* 1. Committed changes: `git format-patch` + `git am --3way`
|
|
13
|
+
* 2. Uncommitted changes (staged + unstaged): `git diff HEAD` + `git apply`
|
|
14
|
+
* 3. Untracked files: `git ls-files --others` + `copyOut` each file
|
|
15
|
+
*/
|
|
16
|
+
import { existsSync } from "node:fs";
|
|
17
|
+
import { mkdir, readdir, readFile, rm, stat, writeFile, } from "node:fs/promises";
|
|
18
|
+
import { basename, dirname, join } from "node:path";
|
|
19
|
+
import { Effect } from "effect";
|
|
20
|
+
import { buildRecoveryMessage } from "./RecoveryMessage.js";
|
|
21
|
+
import { SyncError } from "./errors.js";
|
|
22
|
+
/**
|
|
23
|
+
* Execute a command on the host side, returning stdout.
|
|
24
|
+
* Fails with SyncError on non-zero exit.
|
|
25
|
+
*/
|
|
26
|
+
const execHost = (command, cwd) => Effect.tryPromise({
|
|
27
|
+
try: async () => {
|
|
28
|
+
const { exec } = await import("node:child_process");
|
|
29
|
+
const { promisify } = await import("node:util");
|
|
30
|
+
const execAsync = promisify(exec);
|
|
31
|
+
const { stdout } = await execAsync(command, {
|
|
32
|
+
cwd,
|
|
33
|
+
maxBuffer: 10 * 1024 * 1024,
|
|
34
|
+
});
|
|
35
|
+
return stdout;
|
|
36
|
+
},
|
|
37
|
+
catch: (e) => new SyncError({
|
|
38
|
+
message: `Host command failed: ${command}\n${e instanceof Error ? e.message : String(e)}`,
|
|
39
|
+
}),
|
|
40
|
+
});
|
|
41
|
+
/**
|
|
42
|
+
* Execute a command in the sandbox, failing with SyncError if it exits non-zero.
|
|
43
|
+
*/
|
|
44
|
+
const execOk = (handle, command, options) => Effect.tryPromise({
|
|
45
|
+
try: () => handle.exec(command, options),
|
|
46
|
+
catch: (e) => new SyncError({
|
|
47
|
+
message: `Sandbox exec failed: ${command}\n${e instanceof Error ? e.message : String(e)}`,
|
|
48
|
+
}),
|
|
49
|
+
}).pipe(Effect.flatMap((result) => result.exitCode !== 0
|
|
50
|
+
? Effect.fail(new SyncError({
|
|
51
|
+
message: `Sandbox command failed (exit ${result.exitCode}): ${command}\n${result.stderr}`,
|
|
52
|
+
}))
|
|
53
|
+
: Effect.succeed(result)));
|
|
54
|
+
/**
|
|
55
|
+
* Execute a command in the sandbox, returning the result without failing on non-zero exit.
|
|
56
|
+
*/
|
|
57
|
+
const execSandbox = (handle, command, options) => Effect.tryPromise({
|
|
58
|
+
try: () => handle.exec(command, options),
|
|
59
|
+
catch: (e) => new SyncError({
|
|
60
|
+
message: `Sandbox exec failed: ${command}\n${e instanceof Error ? e.message : String(e)}`,
|
|
61
|
+
}),
|
|
62
|
+
});
|
|
63
|
+
/**
|
|
64
|
+
* Check if a patch file is empty or header-only.
|
|
65
|
+
* Merge commits produce patches with headers but no diff content.
|
|
66
|
+
* A patch is considered empty if it has no lines starting with "diff --git".
|
|
67
|
+
*/
|
|
68
|
+
const isEmptyPatch = (patchPath) => Effect.tryPromise({
|
|
69
|
+
try: async () => {
|
|
70
|
+
const info = await stat(patchPath);
|
|
71
|
+
if (info.size === 0)
|
|
72
|
+
return true;
|
|
73
|
+
const content = await readFile(patchPath, "utf-8");
|
|
74
|
+
return !content.includes("diff --git");
|
|
75
|
+
},
|
|
76
|
+
catch: (e) => new SyncError({
|
|
77
|
+
message: `Failed to check patch ${patchPath}: ${e instanceof Error ? e.message : String(e)}`,
|
|
78
|
+
}),
|
|
79
|
+
});
|
|
80
|
+
/**
|
|
81
|
+
* Generate a YYYYMMDD-HHMMSS timestamp directory name.
|
|
82
|
+
* Appends a counter suffix (-1, -2, ...) if the directory already exists.
|
|
83
|
+
*/
|
|
84
|
+
const createPatchDir = (hostRepoDir) => Effect.tryPromise({
|
|
85
|
+
try: async () => {
|
|
86
|
+
const now = new Date();
|
|
87
|
+
const pad = (n) => String(n).padStart(2, "0");
|
|
88
|
+
const base = `${now.getFullYear()}${pad(now.getMonth() + 1)}${pad(now.getDate())}-${pad(now.getHours())}${pad(now.getMinutes())}${pad(now.getSeconds())}`;
|
|
89
|
+
const patchesRoot = join(hostRepoDir, ".sandcastle", "patches");
|
|
90
|
+
await mkdir(patchesRoot, { recursive: true });
|
|
91
|
+
let dirName = base;
|
|
92
|
+
let counter = 0;
|
|
93
|
+
while (existsSync(join(patchesRoot, dirName))) {
|
|
94
|
+
counter++;
|
|
95
|
+
dirName = `${base}-${counter}`;
|
|
96
|
+
}
|
|
97
|
+
const patchDir = join(patchesRoot, dirName);
|
|
98
|
+
await mkdir(patchDir, { recursive: true });
|
|
99
|
+
return patchDir;
|
|
100
|
+
},
|
|
101
|
+
catch: (e) => new SyncError({
|
|
102
|
+
message: `Failed to create patch directory: ${e instanceof Error ? e.message : String(e)}`,
|
|
103
|
+
}),
|
|
104
|
+
});
|
|
105
|
+
/**
|
|
106
|
+
* Sync changes from an isolated sandbox back to the host repo.
|
|
107
|
+
*
|
|
108
|
+
* Two-phase extraction with artifact persistence:
|
|
109
|
+
* 1. Save all artifacts to `.sandcastle/patches/<timestamp>/`
|
|
110
|
+
* 2. Apply from saved directory; on failure, preserve artifacts and print recovery
|
|
111
|
+
*/
|
|
112
|
+
export const syncOut = (hostRepoDir, handle) => Effect.gen(function* () {
|
|
113
|
+
const workspacePath = handle.workspacePath;
|
|
114
|
+
const hostHead = (yield* execHost("git rev-parse HEAD", hostRepoDir)).trim();
|
|
115
|
+
const sandboxHead = (yield* execOk(handle, "git rev-parse HEAD", {
|
|
116
|
+
cwd: workspacePath,
|
|
117
|
+
})).stdout.trim();
|
|
118
|
+
const hasCommits = hostHead !== sandboxHead;
|
|
119
|
+
// Check for uncommitted changes
|
|
120
|
+
const diffResult = yield* execSandbox(handle, "git diff HEAD", {
|
|
121
|
+
cwd: workspacePath,
|
|
122
|
+
});
|
|
123
|
+
const hasDiff = diffResult.exitCode === 0 && diffResult.stdout.trim().length > 0;
|
|
124
|
+
// Check for untracked files
|
|
125
|
+
const lsFilesResult = yield* execSandbox(handle, "git ls-files --others --exclude-standard", { cwd: workspacePath });
|
|
126
|
+
const hasUntracked = lsFilesResult.exitCode === 0 && lsFilesResult.stdout.trim().length > 0;
|
|
127
|
+
const untrackedFiles = hasUntracked
|
|
128
|
+
? lsFilesResult.stdout
|
|
129
|
+
.trim()
|
|
130
|
+
.split("\n")
|
|
131
|
+
.filter((f) => f.length > 0)
|
|
132
|
+
: [];
|
|
133
|
+
// Nothing to sync
|
|
134
|
+
if (!hasCommits && !hasDiff && !hasUntracked) {
|
|
135
|
+
return;
|
|
136
|
+
}
|
|
137
|
+
// --- Phase 1: Save all artifacts ---
|
|
138
|
+
const patchDir = yield* createPatchDir(hostRepoDir);
|
|
139
|
+
const relativePatchDir = join(".sandcastle", "patches", basename(patchDir));
|
|
140
|
+
const nonEmptyPatches = [];
|
|
141
|
+
// Save committed patches
|
|
142
|
+
if (hasCommits) {
|
|
143
|
+
const mkTempResult = yield* execOk(handle, "mktemp -d -t sandcastle-patches-XXXXXX");
|
|
144
|
+
const sandboxPatchDir = mkTempResult.stdout.trim();
|
|
145
|
+
try {
|
|
146
|
+
yield* execOk(handle, `git format-patch "${hostHead}..HEAD" -o "${sandboxPatchDir}"`, { cwd: workspacePath });
|
|
147
|
+
const lsResult = yield* execOk(handle, `ls -1 "${sandboxPatchDir}"`);
|
|
148
|
+
const patchNames = lsResult.stdout
|
|
149
|
+
.trim()
|
|
150
|
+
.split("\n")
|
|
151
|
+
.filter((name) => name.length > 0);
|
|
152
|
+
for (const patchName of patchNames) {
|
|
153
|
+
const sandboxPatchPath = `${sandboxPatchDir}/${patchName}`;
|
|
154
|
+
const hostPatchPath = join(patchDir, patchName);
|
|
155
|
+
yield* Effect.tryPromise({
|
|
156
|
+
try: () => handle.copyOut(sandboxPatchPath, hostPatchPath),
|
|
157
|
+
catch: (e) => new SyncError({
|
|
158
|
+
message: `Failed to copy patch ${patchName}: ${e instanceof Error ? e.message : String(e)}`,
|
|
159
|
+
}),
|
|
160
|
+
});
|
|
161
|
+
if (!(yield* isEmptyPatch(hostPatchPath))) {
|
|
162
|
+
nonEmptyPatches.push(hostPatchPath);
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
finally {
|
|
167
|
+
yield* execSandbox(handle, `rm -rf "${sandboxPatchDir}"`);
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
// Save uncommitted diff
|
|
171
|
+
if (hasDiff) {
|
|
172
|
+
const diffPath = join(patchDir, "changes.patch");
|
|
173
|
+
yield* Effect.tryPromise({
|
|
174
|
+
try: () => writeFile(diffPath, diffResult.stdout),
|
|
175
|
+
catch: (e) => new SyncError({
|
|
176
|
+
message: `Failed to write diff patch: ${e instanceof Error ? e.message : String(e)}`,
|
|
177
|
+
}),
|
|
178
|
+
});
|
|
179
|
+
}
|
|
180
|
+
// Save untracked files
|
|
181
|
+
if (hasUntracked) {
|
|
182
|
+
const untrackedDir = join(patchDir, "untracked");
|
|
183
|
+
for (const relPath of untrackedFiles) {
|
|
184
|
+
const sandboxFilePath = `${workspacePath}/${relPath}`;
|
|
185
|
+
const hostFilePath = join(untrackedDir, relPath);
|
|
186
|
+
yield* Effect.tryPromise({
|
|
187
|
+
try: async () => {
|
|
188
|
+
await mkdir(dirname(hostFilePath), { recursive: true });
|
|
189
|
+
await handle.copyOut(sandboxFilePath, hostFilePath);
|
|
190
|
+
},
|
|
191
|
+
catch: (e) => new SyncError({
|
|
192
|
+
message: `Failed to save untracked file ${relPath}: ${e instanceof Error ? e.message : String(e)}`,
|
|
193
|
+
}),
|
|
194
|
+
});
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
// --- Phase 2: Apply from saved directory ---
|
|
198
|
+
let failedStep;
|
|
199
|
+
// Apply committed patches
|
|
200
|
+
if (nonEmptyPatches.length > 0) {
|
|
201
|
+
const abortResult = yield* Effect.either(execHost("git am --abort", hostRepoDir));
|
|
202
|
+
void abortResult; // ignore abort failures
|
|
203
|
+
const patchArgs = nonEmptyPatches.map((p) => `"${p}"`).join(" ");
|
|
204
|
+
const applyResult = yield* Effect.either(execHost(`git am --3way ${patchArgs}`, hostRepoDir));
|
|
205
|
+
if (applyResult._tag === "Left") {
|
|
206
|
+
failedStep = "commits";
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
// Apply uncommitted diff
|
|
210
|
+
if (!failedStep && hasDiff) {
|
|
211
|
+
const diffPath = join(patchDir, "changes.patch");
|
|
212
|
+
const applyResult = yield* Effect.either(execHost(`git apply "${diffPath}"`, hostRepoDir));
|
|
213
|
+
if (applyResult._tag === "Left") {
|
|
214
|
+
failedStep = "diff";
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
// Copy untracked files
|
|
218
|
+
if (!failedStep && hasUntracked) {
|
|
219
|
+
const copyResult = yield* Effect.either(Effect.tryPromise({
|
|
220
|
+
try: async () => {
|
|
221
|
+
const untrackedDir = join(patchDir, "untracked");
|
|
222
|
+
for (const relPath of untrackedFiles) {
|
|
223
|
+
const srcPath = join(untrackedDir, relPath);
|
|
224
|
+
const destPath = join(hostRepoDir, relPath);
|
|
225
|
+
await mkdir(dirname(destPath), { recursive: true });
|
|
226
|
+
const content = await readFile(srcPath);
|
|
227
|
+
await writeFile(destPath, content);
|
|
228
|
+
}
|
|
229
|
+
},
|
|
230
|
+
catch: (e) => new SyncError({
|
|
231
|
+
message: `Failed to copy untracked files: ${e instanceof Error ? e.message : String(e)}`,
|
|
232
|
+
}),
|
|
233
|
+
}));
|
|
234
|
+
if (copyResult._tag === "Left") {
|
|
235
|
+
failedStep = "untracked";
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
// --- Cleanup or preserve ---
|
|
239
|
+
if (failedStep) {
|
|
240
|
+
const msg = buildRecoveryMessage({
|
|
241
|
+
patchDir: relativePatchDir,
|
|
242
|
+
failedStep,
|
|
243
|
+
hasCommits: nonEmptyPatches.length > 0,
|
|
244
|
+
hasDiff,
|
|
245
|
+
hasUntracked,
|
|
246
|
+
});
|
|
247
|
+
console.error(`\n${msg}`);
|
|
248
|
+
}
|
|
249
|
+
else {
|
|
250
|
+
yield* Effect.tryPromise({
|
|
251
|
+
try: async () => {
|
|
252
|
+
await rm(patchDir, { recursive: true, force: true });
|
|
253
|
+
const patchesRoot = join(hostRepoDir, ".sandcastle", "patches");
|
|
254
|
+
try {
|
|
255
|
+
const remaining = await readdir(patchesRoot);
|
|
256
|
+
if (remaining.length === 0) {
|
|
257
|
+
await rm(join(hostRepoDir, ".sandcastle"), {
|
|
258
|
+
recursive: true,
|
|
259
|
+
force: true,
|
|
260
|
+
});
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
catch {
|
|
264
|
+
// ignore
|
|
265
|
+
}
|
|
266
|
+
},
|
|
267
|
+
catch: () => new SyncError({ message: "Failed to clean up patch directory" }),
|
|
268
|
+
});
|
|
269
|
+
}
|
|
270
|
+
});
|
|
271
|
+
//# sourceMappingURL=syncOut.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"syncOut.js","sourceRoot":"","sources":["../src/syncOut.ts"],"names":[],"mappings":"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"}
|
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
import { run, claudeCode } from "@ai-hero/sandcastle";
|
|
2
|
+
import { docker } from "@ai-hero/sandcastle/sandboxes/docker";
|
|
2
3
|
|
|
3
4
|
// Blank template: customize this to build your own orchestration.
|
|
4
5
|
// Run this with: npx tsx .sandcastle/main.mts
|
|
@@ -6,5 +7,6 @@ import { run, claudeCode } from "@ai-hero/sandcastle";
|
|
|
6
7
|
|
|
7
8
|
await run({
|
|
8
9
|
agent: claudeCode("claude-opus-4-6"),
|
|
10
|
+
sandbox: docker(),
|
|
9
11
|
promptFile: "./.sandcastle/prompt.md",
|
|
10
12
|
});
|
|
@@ -17,6 +17,7 @@
|
|
|
17
17
|
// "scripts": { "sandcastle": "npx tsx .sandcastle/main.mts" }
|
|
18
18
|
|
|
19
19
|
import * as sandcastle from "@ai-hero/sandcastle";
|
|
20
|
+
import { docker } from "@ai-hero/sandcastle/sandboxes/docker";
|
|
20
21
|
|
|
21
22
|
// ---------------------------------------------------------------------------
|
|
22
23
|
// Configuration
|
|
@@ -56,6 +57,7 @@ for (let iteration = 1; iteration <= MAX_ITERATIONS; iteration++) {
|
|
|
56
57
|
const plan = await sandcastle.run({
|
|
57
58
|
hooks,
|
|
58
59
|
copyToSandbox,
|
|
60
|
+
sandbox: docker(),
|
|
59
61
|
name: "planner",
|
|
60
62
|
// One iteration is enough: the planner just needs to read and reason,
|
|
61
63
|
// not write code.
|
|
@@ -105,6 +107,8 @@ for (let iteration = 1; iteration <= MAX_ITERATIONS; iteration++) {
|
|
|
105
107
|
sandcastle.run({
|
|
106
108
|
hooks,
|
|
107
109
|
copyToSandbox,
|
|
110
|
+
// Each agent starts on its own branch via the provider's branchStrategy.
|
|
111
|
+
sandbox: docker({ branchStrategy: { type: "branch", branch: issue.branch } }),
|
|
108
112
|
name: "implementer",
|
|
109
113
|
// Give each agent plenty of room to implement and iterate on tests.
|
|
110
114
|
maxIterations: 100,
|
|
@@ -119,8 +123,6 @@ for (let iteration = 1; iteration <= MAX_ITERATIONS; iteration++) {
|
|
|
119
123
|
ISSUE_TITLE: issue.title,
|
|
120
124
|
BRANCH: issue.branch,
|
|
121
125
|
},
|
|
122
|
-
// Each agent starts on its own branch.
|
|
123
|
-
worktree: { mode: "branch", branch: issue.branch },
|
|
124
126
|
}),
|
|
125
127
|
),
|
|
126
128
|
);
|
|
@@ -179,6 +181,7 @@ for (let iteration = 1; iteration <= MAX_ITERATIONS; iteration++) {
|
|
|
179
181
|
await sandcastle.run({
|
|
180
182
|
hooks,
|
|
181
183
|
copyToSandbox,
|
|
184
|
+
sandbox: docker(),
|
|
182
185
|
name: "merger",
|
|
183
186
|
maxIterations: 10,
|
|
184
187
|
// Sonnet is sufficient for merge conflict resolution.
|
|
@@ -17,6 +17,7 @@
|
|
|
17
17
|
// "scripts": { "sandcastle": "npx tsx .sandcastle/main.mts" }
|
|
18
18
|
|
|
19
19
|
import * as sandcastle from "@ai-hero/sandcastle";
|
|
20
|
+
import { docker } from "@ai-hero/sandcastle/sandboxes/docker";
|
|
20
21
|
|
|
21
22
|
// ---------------------------------------------------------------------------
|
|
22
23
|
// Configuration
|
|
@@ -57,6 +58,7 @@ for (let iteration = 1; iteration <= MAX_ITERATIONS; iteration++) {
|
|
|
57
58
|
const implement = await sandcastle.run({
|
|
58
59
|
hooks,
|
|
59
60
|
copyToSandbox,
|
|
61
|
+
sandbox: docker(),
|
|
60
62
|
name: "implementer",
|
|
61
63
|
maxIterations: 100,
|
|
62
64
|
agent: sandcastle.claudeCode("claude-sonnet-4-6"),
|
|
@@ -84,6 +86,7 @@ for (let iteration = 1; iteration <= MAX_ITERATIONS; iteration++) {
|
|
|
84
86
|
await sandcastle.run({
|
|
85
87
|
hooks,
|
|
86
88
|
copyToSandbox,
|
|
89
|
+
sandbox: docker({ branchStrategy: { type: "branch", branch } }),
|
|
87
90
|
name: "reviewer",
|
|
88
91
|
maxIterations: 10,
|
|
89
92
|
agent: sandcastle.claudeCode("claude-sonnet-4-6"),
|
|
@@ -93,7 +96,6 @@ for (let iteration = 1; iteration <= MAX_ITERATIONS; iteration++) {
|
|
|
93
96
|
promptArgs: {
|
|
94
97
|
BRANCH: branch,
|
|
95
98
|
},
|
|
96
|
-
worktree: { mode: "branch", branch },
|
|
97
99
|
});
|
|
98
100
|
|
|
99
101
|
console.log("\nReview complete.");
|
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
import { run, claudeCode } from "@ai-hero/sandcastle";
|
|
2
|
+
import { docker } from "@ai-hero/sandcastle/sandboxes/docker";
|
|
2
3
|
|
|
3
4
|
// Simple loop: an agent that picks open GitHub issues one by one and closes them.
|
|
4
5
|
// Run this with: npx tsx .sandcastle/main.mts
|
|
@@ -8,6 +9,9 @@ await run({
|
|
|
8
9
|
// A name for this run, shown as a prefix in log output.
|
|
9
10
|
name: "worker",
|
|
10
11
|
|
|
12
|
+
// Sandbox provider — Docker is the default runtime.
|
|
13
|
+
sandbox: docker(),
|
|
14
|
+
|
|
11
15
|
// The agent provider. Pass a model string to claudeCode() — sonnet balances
|
|
12
16
|
// capability and speed for most tasks. Switch to claude-opus-4-6 for harder
|
|
13
17
|
// problems, or claude-haiku-4-5-20251001 for speed.
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@ai-hero/sandcastle",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.4.0",
|
|
4
4
|
"description": "CLI for orchestrating AI agents in isolated sandbox environments",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "./dist/index.js",
|
|
@@ -9,6 +9,10 @@
|
|
|
9
9
|
".": {
|
|
10
10
|
"import": "./dist/index.js",
|
|
11
11
|
"types": "./dist/index.d.ts"
|
|
12
|
+
},
|
|
13
|
+
"./sandboxes/docker": {
|
|
14
|
+
"import": "./dist/sandboxes/docker.js",
|
|
15
|
+
"types": "./dist/sandboxes/docker.d.ts"
|
|
12
16
|
}
|
|
13
17
|
},
|
|
14
18
|
"bin": {
|