9e45780ba0
First commit of the Next.js community site that powers deepseek-tui.com, deployed via Cloudflare Workers / OpenNext. This commit lands the scaffold and applies the visual + correctness pass requested by community feedback: - Palette: drop the cream/Anthropic-feel paper (#F4F1E8) for a DeepSeek-aligned cool white + soft gray (#FFFFFF / #F4F6FB), with indigo accents kept. Soften default hairlines so a pure-white background reads clean instead of harsh. - Mobile: add a hamburger menu (mobile-menu.tsx) so phones can reach Install / Docs / Activity / Roadmap / Contribute — previously the link list was hidden on phones with no replacement. Tighter hero, flexible button row, viewport-safe code blocks, columnar grids collapse cleanly under 768px, and the printed-almanac center rule is desktop-only now (it sliced through narrow viewports). - "How it works" diagram: replace the hand-rolled ASCII art (which misaligned under CJK monospace because Han characters take 2 columns vs Latin's 1, per dhh's note in WeChat) with a real mermaid diagram rendered client-side via dynamic import. Uses the mermaid.live standard syntax 庄表伟 recommended. - Issue #1104: the docs listed a `deepseek-cn` provider that the v0.8.16 binary doesn't accept (`ProviderArg` in crates/cli only has 9 variants; the 10th lives only in the legacy tui/config.rs). derive-facts.mjs now omits `deepseek-cn` until that variant is wired through the shared ProviderKind, and the install page's China-network recipe uses `base_url` / `DEEPSEEK_BASE_URL` (which actually works on v0.8.16) instead of the unsupported provider. Auto-deploys via .github/workflows/deploy-web.yml on push to main. Co-authored-by: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
59 lines
1.7 KiB
TypeScript
59 lines
1.7 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>();
|
|
|
|
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;
|
|
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,
|
|
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();
|
|
const value = JSON.stringify(d);
|
|
if (env.CURATED_KV) {
|
|
await env.CURATED_KV.put("dispatch:latest", value, { expirationTtl: 60 * 60 * 24 * 7 });
|
|
} else {
|
|
MEM.set("dispatch:latest", value);
|
|
}
|
|
}
|