519423cb0d
- landing/app/mira/wiki: tag=mira-wiki list now shows all wiki paragraphs (depends on KBDB tag filter exposed in matrix/kbdb commit, separate repo) - landing: app/mira hub + feed split + various WIP from prior sessions - registry/components: claude_api / kbdb_create_block / kbdb_get / km_writer / platform_crypto / auth_oauth2 contracts + main.go (accumulated) - .component-builds: pkg-lock updates + index.ts adjustments (WIP) - .agents/specs/arcrun/frontend-redesign: design notes - docs/test_credentials, docs/user_requirements/arcrun-landing-page: WIP docs - cypher-executor: auth-dispatcher / wasi-shim adjustments (WIP) Includes accumulated work from prior sessions plus the wiki UI tag-filter update that surfaces the AI-generated wiki paragraphs at /mira/wiki. 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
83 lines
2.6 KiB
TypeScript
83 lines
2.6 KiB
TypeScript
/**
|
||
* arcrun API component Worker (http_request)
|
||
*
|
||
* POST / → JSON input → WASM (WASI preview1 stdin/stdout) → JSON output
|
||
*
|
||
* host function: http_request(fetch → 回 response body 原文)
|
||
* 此 Worker 不做 credential 注入,只提供通用 HTTP 能力。
|
||
*/
|
||
|
||
import componentWasm from '../component.wasm' assert { type: 'webassembly' };
|
||
import { Hono } from 'hono';
|
||
import { cors } from 'hono/cors';
|
||
import { createWasiShim, type WasiHostFunctions } from '../../../cypher-executor/src/lib/wasi-shim';
|
||
|
||
const app = new Hono();
|
||
app.use('*', cors());
|
||
|
||
app.get('/', (c) => c.json({ ok: true, component: 'http_request' }));
|
||
|
||
app.post('/', async (c) => {
|
||
let input: unknown;
|
||
try {
|
||
input = await c.req.json();
|
||
} catch {
|
||
return c.json({ success: false, error: 'request body must be JSON' }, 400);
|
||
}
|
||
|
||
try {
|
||
const result = await runWasm(input);
|
||
return c.json(result);
|
||
} catch (e) {
|
||
return c.json(
|
||
{ success: false, error: e instanceof Error ? e.message : String(e) },
|
||
500,
|
||
);
|
||
}
|
||
});
|
||
|
||
export default app;
|
||
|
||
// ── WASM runner ──────────────────────────────────────────────────────────────
|
||
|
||
async function runWasm(input: unknown): Promise<unknown> {
|
||
const hostFunctions: WasiHostFunctions = {
|
||
http_request: async (url, method, headersJson, body) => {
|
||
const headers: Record<string, string> = {};
|
||
if (headersJson) {
|
||
try {
|
||
const parsed = JSON.parse(headersJson);
|
||
if (parsed && typeof parsed === 'object') {
|
||
for (const [k, v] of Object.entries(parsed as Record<string, unknown>)) {
|
||
if (typeof v === 'string') headers[k] = v;
|
||
}
|
||
}
|
||
} catch {
|
||
// ignore header parse errors
|
||
}
|
||
}
|
||
const init: RequestInit = { method, headers };
|
||
if (body && method.toUpperCase() !== 'GET' && method.toUpperCase() !== 'HEAD') {
|
||
init.body = body;
|
||
}
|
||
const res = await fetch(url, init);
|
||
return await res.text();
|
||
},
|
||
};
|
||
|
||
const shim = createWasiShim(JSON.stringify(input), hostFunctions);
|
||
|
||
const instance = await WebAssembly.instantiate(
|
||
componentWasm as WebAssembly.Module,
|
||
shim.imports,
|
||
);
|
||
shim.setMemory(instance.exports.memory as WebAssembly.Memory);
|
||
await shim.run(instance);
|
||
|
||
const stdout = shim.getStdout().trim();
|
||
const stderr = shim.getStderr().trim();
|
||
if (stderr) console.error('[http_request wasm stderr]', stderr);
|
||
if (!stdout) throw new Error('WASM component produced no output');
|
||
return JSON.parse(stdout);
|
||
}
|