99c6b22e83
- Persistent RLM sessions (rlm_open/rlm_eval/rlm_close) with bounded REPL helpers - Fork-aware sub-agent sessions (agent_open/agent_eval/agent_close) with handle_read - Shared handle_read storage with slice/range/count/JSONPath projections - Slash-command routing: /rlm, /agent, /relay (/接力) for handoff prompts - Sidebar renamed to "Work" tab, consistent across Plan/Agent/YOLO modes - Tool papercuts: file_search excludes, grep_files strings, fetch_url JSON, edit_file fuzz, exec_shell merged stdout/stderr, revert_turn no-op reject - CLI reasoning-effort honoured on non-auto exec routes (#1511 @h3c-hexin) - Edit-file replacement boundaries clarified (#1516) - Pandoc output validated before probing (#1523) - Running turns steerable/repaintable (#1533, #1537) - Tasks/Activity Detail calmer under load - npm retry timeout hint (#1538 @reidliu41) - Issue templates improved (#1525 @reidliu41) - Shell: kill process group to prevent UI freeze (#828 @CrepuscularIRIS) - TUI: ignore leaked SGR mouse reports in composer (#1421 @reidliu41) - Footer: keep chips within available width (#1417 @Wenjunyun123) - Session picker: scope Ctrl+R to current workspace (#1395 @LinQ) - Removed stale competitive-analysis doc - Prompts/docs teach only new tool names
67 lines
2.0 KiB
TypeScript
67 lines
2.0 KiB
TypeScript
/**
|
|
* Cloudflare KV access via the OpenNext binding helper.
|
|
* Falls back to in-memory cache for `next dev` outside of `wrangler dev`.
|
|
*/
|
|
import type { CuratedDispatch } from "./types";
|
|
|
|
const MEM = new Map<string, string>();
|
|
|
|
export interface KVNamespace {
|
|
get(key: string): Promise<string | null>;
|
|
put(key: string, value: string, opts?: { expirationTtl?: number }): Promise<void>;
|
|
list(opts?: { prefix?: string; limit?: number }): Promise<{ keys: { name: string }[] }>;
|
|
delete(key: string): Promise<void>;
|
|
}
|
|
|
|
interface CloudflareEnv {
|
|
CURATED_KV?: KVNamespace;
|
|
DEEPSEEK_API_KEY?: string;
|
|
DEEPSEEK_BASE_URL?: string;
|
|
DEEPSEEK_MODEL?: string;
|
|
GITHUB_TOKEN?: string;
|
|
CRON_SECRET?: string;
|
|
GITHUB_REPO?: string;
|
|
}
|
|
|
|
export async function getEnv(): Promise<CloudflareEnv> {
|
|
try {
|
|
const mod = await import("@opennextjs/cloudflare");
|
|
const ctx = await mod.getCloudflareContext({ async: true });
|
|
return ctx.env as CloudflareEnv;
|
|
} catch {
|
|
return {
|
|
DEEPSEEK_API_KEY: process.env.DEEPSEEK_API_KEY,
|
|
DEEPSEEK_BASE_URL: process.env.DEEPSEEK_BASE_URL,
|
|
DEEPSEEK_MODEL: process.env.DEEPSEEK_MODEL,
|
|
GITHUB_TOKEN: process.env.GITHUB_TOKEN,
|
|
CRON_SECRET: process.env.CRON_SECRET,
|
|
GITHUB_REPO: process.env.GITHUB_REPO,
|
|
};
|
|
}
|
|
}
|
|
|
|
export async function getDispatch(): Promise<CuratedDispatch | null> {
|
|
const env = await getEnv();
|
|
const raw = env.CURATED_KV ? await env.CURATED_KV.get("dispatch:latest") : MEM.get("dispatch:latest") ?? null;
|
|
if (!raw) return null;
|
|
try {
|
|
return JSON.parse(raw) as CuratedDispatch;
|
|
} catch {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
export async function putDispatch(d: CuratedDispatch): Promise<void> {
|
|
const env = await getEnv();
|
|
await putDispatchWithKv(env.CURATED_KV, d);
|
|
}
|
|
|
|
export async function putDispatchWithKv(kv: KVNamespace | undefined, d: CuratedDispatch): Promise<void> {
|
|
const value = JSON.stringify(d);
|
|
if (kv) {
|
|
await kv.put("dispatch:latest", value, { expirationTtl: 60 * 60 * 24 * 7 });
|
|
} else {
|
|
MEM.set("dispatch:latest", value);
|
|
}
|
|
}
|