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>
86 lines
2.8 KiB
TypeScript
86 lines
2.8 KiB
TypeScript
/**
|
|
* arcrun API component Worker (line_notify)
|
|
*
|
|
* POST / → JSON input → WASM (WASI preview1 stdin/stdout) → JSON output
|
|
*
|
|
* 方案 A:import cypher-executor/src/lib/wasi-shim.ts 的 `createWasiShim`
|
|
* 提供單一 `http_request` host function(fetch → 回 response body 原文)。
|
|
* 這個 Worker 不需要 crypto / kv_get(無 credential 解密職責)。
|
|
*
|
|
* Credential 注入路徑:cypher-executor auth-dispatcher 會先 POST 到對應 auth primitive Worker,
|
|
* 把 auth headers 合進本 Worker 的 input(例:`_auth_headers`)。
|
|
*/
|
|
|
|
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: 'line_notify' }));
|
|
|
|
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 {
|
|
// 忽略 header parse 錯誤
|
|
}
|
|
}
|
|
const init: RequestInit = { method, headers };
|
|
if (body && method.toUpperCase() !== 'GET' && method.toUpperCase() !== 'HEAD') {
|
|
init.body = body;
|
|
}
|
|
const res = await fetch(url, init);
|
|
// WASM 端直接 json.Unmarshal 回傳內容,這裡回 response body 原文
|
|
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();
|
|
if (!stdout) throw new Error('WASM component produced no output');
|
|
return JSON.parse(stdout);
|
|
}
|