Files
codewhale/web/app/api/cron/route.ts
T
Hunter Bown 9e45780ba0 feat(web): community site for deepseek-tui.com (mobile + color refresh) (#1108)
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>
2026-05-07 21:00:06 -05:00

105 lines
3.0 KiB
TypeScript

import { NextResponse } from "next/server";
import { getEnv } from "@/lib/kv";
import {
runCurate,
runTriage,
runPrReview,
runStale,
runDupes,
runDigest,
type AgentEnv,
} from "@/lib/community-agent-tasks";
import { runFactsDrift } from "@/lib/facts-drift";
import { runLinkCheck, runSemanticDrift } from "@/lib/content-watch";
export const dynamic = "force-dynamic";
const TASKS = ["curate", "triage", "pr-review", "stale", "dupes", "digest", "facts-drift", "linkcheck", "semantic-drift"] as const;
type Task = (typeof TASKS)[number];
/**
* Manual trigger surface for community-agent tasks.
*
* Usage:
* GET /api/cron?task=curate
* Header: x-cron-secret: <CRON_SECRET>
*
* Real cron scheduling is handled by worker.ts's scheduled() handler.
*/
export async function GET(req: Request) {
const env = await getEnv();
// Always require auth
if (!env.CRON_SECRET) {
return NextResponse.json(
{ error: "manual trigger disabled in production" },
{ status: 503 }
);
}
const auth = req.headers.get("x-cron-secret");
if (auth !== env.CRON_SECRET) {
return NextResponse.json({ error: "unauthorized" }, { status: 401 });
}
const { searchParams } = new URL(req.url);
const task = searchParams.get("task");
if (!task || !TASKS.includes(task as Task)) {
return NextResponse.json(
{ error: `missing or invalid task. Allowed: ${TASKS.join(", ")}` },
{ status: 400 }
);
}
// Build AgentEnv from the same shape expected by the task functions
const agentEnv: AgentEnv = {
CURATED_KV: env.CURATED_KV,
DEEPSEEK_API_KEY: env.DEEPSEEK_API_KEY,
GITHUB_TOKEN: env.GITHUB_TOKEN,
CRON_SECRET: env.CRON_SECRET,
GITHUB_REPO: env.GITHUB_REPO,
MAINTAINER_TOKEN: undefined,
MAINTAINER_GITHUB_PAT: undefined,
};
try {
let result: Record<string, unknown>;
switch (task) {
case "curate":
result = await runCurate(agentEnv);
break;
case "triage":
result = await runTriage(agentEnv);
break;
case "pr-review":
result = await runPrReview(agentEnv);
break;
case "stale":
result = await runStale(agentEnv);
break;
case "dupes":
result = await runDupes(agentEnv);
break;
case "digest":
result = await runDigest(agentEnv);
break;
case "facts-drift":
result = await runFactsDrift(agentEnv) as unknown as Record<string, unknown>;
break;
case "linkcheck":
result = await runLinkCheck(agentEnv) as unknown as Record<string, unknown>;
break;
case "semantic-drift":
result = await runSemanticDrift(agentEnv) as unknown as Record<string, unknown>;
break;
default:
// unreachable — guarded by TASKS check above
result = { error: "unknown task" };
}
return NextResponse.json({ ok: true, task, result });
} catch (e) {
return NextResponse.json({ ok: false, error: String(e) }, { status: 200 });
}
}