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>
58 lines
1.6 KiB
TypeScript
58 lines
1.6 KiB
TypeScript
import { NextResponse } from "next/server";
|
|
import { getAgentEnv, safeEqual, createSession } from "@/lib/community-agent";
|
|
|
|
export const dynamic = "force-dynamic";
|
|
|
|
const ALLOWED_LOCALES = new Set(["en", "zh"]);
|
|
|
|
function pickLocale(value: string | null | undefined): string {
|
|
if (!value) return "en";
|
|
return ALLOWED_LOCALES.has(value) ? value : "en";
|
|
}
|
|
|
|
export async function POST(req: Request) {
|
|
const env = await getAgentEnv();
|
|
const url = new URL(req.url);
|
|
const localeFromQuery = pickLocale(url.searchParams.get("locale"));
|
|
|
|
if (!env.MAINTAINER_TOKEN) {
|
|
return new NextResponse("Not configured", {
|
|
status: 503,
|
|
headers: { "Cache-Control": "no-store" },
|
|
});
|
|
}
|
|
|
|
const form = await req.formData();
|
|
const submitted = String(form.get("token") ?? "");
|
|
const locale = pickLocale(String(form.get("locale") ?? localeFromQuery));
|
|
|
|
const valid = await safeEqual(submitted, env.MAINTAINER_TOKEN);
|
|
if (!valid) {
|
|
return NextResponse.redirect(new URL(`/${locale}/admin?err=1`, req.url), {
|
|
status: 303,
|
|
headers: { "Cache-Control": "no-store" },
|
|
});
|
|
}
|
|
|
|
const sid = await createSession(env.CURATED_KV);
|
|
if (!sid) {
|
|
return new NextResponse("Session storage unavailable", {
|
|
status: 503,
|
|
headers: { "Cache-Control": "no-store" },
|
|
});
|
|
}
|
|
|
|
const res = NextResponse.redirect(new URL(`/${locale}/admin`, req.url), {
|
|
status: 303,
|
|
headers: { "Cache-Control": "no-store" },
|
|
});
|
|
res.cookies.set("mt_sid", sid, {
|
|
path: "/",
|
|
httpOnly: true,
|
|
secure: true,
|
|
sameSite: "strict",
|
|
maxAge: 60 * 60 * 24,
|
|
});
|
|
return res;
|
|
}
|