Move to threads based system.

This commit is contained in:
Gabriel Brown
2026-06-22 10:37:26 -04:00
parent 8ae6c4b533
commit 206b64176b
82 changed files with 6169 additions and 1930 deletions
+142
View File
@@ -0,0 +1,142 @@
import { createServer } from 'node:http';
import type { IncomingMessage, ServerResponse } from 'node:http';
import { env } from './env';
import {
getWorkspaceDiff,
listWorkspaceTree,
openWorkspacePullRequest,
readWorkspaceFile,
runWorkspaceCommand,
sendWorkspaceMessage,
stopWorkspace,
writeWorkspaceFile,
} from './worker';
const sendJson = (response: ServerResponse, status: number, body: unknown) => {
response.writeHead(status, { 'content-type': 'application/json' });
response.end(JSON.stringify(body));
};
const readBody = async (request: IncomingMessage) =>
await new Promise<string>((resolve, reject) => {
let body = '';
request.on('data', (chunk: Buffer) => {
body += chunk.toString('utf8');
});
request.on('end', () => resolve(body));
request.on('error', reject);
});
const parseJson = async <T>(request: IncomingMessage) => {
const body = await readBody(request);
if (!body.trim()) return {} as T;
return JSON.parse(body) as T;
};
const requireAuth = (request: IncomingMessage) => {
const header = request.headers.authorization;
const token = header?.startsWith('Bearer ') ? header.slice(7) : '';
if (!env.internalToken || token !== env.internalToken) {
throw new Error('Unauthorized');
}
};
const jobRoute = (pathname: string) => {
const match = /^\/jobs\/([^/]+)\/([^/]+)$/.exec(pathname);
if (!match?.[1] || !match[2]) return null;
return { jobId: decodeURIComponent(match[1]), action: match[2] };
};
export const startWorkerServer = () => {
const server = createServer((request, response) => {
void (async () => {
try {
requireAuth(request);
const url = new URL(
request.url ?? '/',
`http://localhost:${env.httpPort}`,
);
if (url.pathname === '/health') {
sendJson(response, 200, { ok: true, workerId: env.workerId });
return;
}
const route = jobRoute(url.pathname);
if (!route) {
sendJson(response, 404, { error: 'Not found' });
return;
}
if (request.method === 'GET' && route.action === 'tree') {
sendJson(response, 200, {
tree: await listWorkspaceTree(route.jobId),
});
return;
}
if (request.method === 'GET' && route.action === 'file') {
const filePath = url.searchParams.get('path') ?? '';
sendJson(response, 200, {
path: filePath,
content: await readWorkspaceFile(route.jobId, filePath),
});
return;
}
if (request.method === 'PUT' && route.action === 'file') {
const body = await parseJson<{ path?: string; content?: string }>(
request,
);
sendJson(
response,
200,
await writeWorkspaceFile(
route.jobId,
body.path ?? '',
body.content ?? '',
),
);
return;
}
if (request.method === 'GET' && route.action === 'diff') {
sendJson(response, 200, {
diff: await getWorkspaceDiff(route.jobId),
});
return;
}
if (request.method === 'POST' && route.action === 'message') {
const body = await parseJson<{ content?: string }>(request);
await sendWorkspaceMessage(route.jobId, body.content ?? '');
sendJson(response, 200, { success: true });
return;
}
if (request.method === 'POST' && route.action === 'run-command') {
const body = await parseJson<{ command?: string }>(request);
sendJson(
response,
200,
await runWorkspaceCommand(route.jobId, body.command ?? ''),
);
return;
}
if (request.method === 'POST' && route.action === 'open-pr') {
sendJson(response, 200, await openWorkspacePullRequest(route.jobId));
return;
}
if (request.method === 'POST' && route.action === 'stop') {
sendJson(response, 200, await stopWorkspace(route.jobId));
return;
}
sendJson(response, 404, { error: 'Not found' });
} catch (error) {
const message = error instanceof Error ? error.message : String(error);
sendJson(response, message === 'Unauthorized' ? 401 : 500, {
error: message,
});
}
})();
});
server.listen(env.httpPort, () => {
console.log(
`Spoon agent worker HTTP server listening on port ${env.httpPort}`,
);
});
};