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>
101 lines
3.2 KiB
TypeScript
101 lines
3.2 KiB
TypeScript
import type { CuratedDispatch, FeedItem, RepoStats } from "./types";
|
|
|
|
const BASE = process.env.DEEPSEEK_BASE_URL ?? "https://api.deepseek.com";
|
|
const MODEL = process.env.DEEPSEEK_MODEL ?? "deepseek-v4-flash";
|
|
|
|
interface ChatMessage {
|
|
role: "system" | "user" | "assistant";
|
|
content: string;
|
|
}
|
|
|
|
interface ChatResponse {
|
|
choices: { message: { content: string } }[];
|
|
}
|
|
|
|
export async function chat(messages: ChatMessage[], apiKey: string, jsonMode = false): Promise<string> {
|
|
const res = await fetch(`${BASE}/v1/chat/completions`, {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
Authorization: `Bearer ${apiKey}`,
|
|
},
|
|
body: JSON.stringify({
|
|
model: MODEL,
|
|
messages,
|
|
temperature: 0.4,
|
|
...(jsonMode ? { response_format: { type: "json_object" } } : {}),
|
|
}),
|
|
});
|
|
if (!res.ok) {
|
|
const text = await res.text();
|
|
throw new Error(`DeepSeek ${res.status}: ${text}`);
|
|
}
|
|
const data = (await res.json()) as ChatResponse;
|
|
return data.choices[0]?.message?.content ?? "";
|
|
}
|
|
|
|
const SYSTEM_PROMPT = `You are the editor of "今日要闻 / Today's Dispatch", a daily-ish digest for the deepseek-tui open source project.
|
|
|
|
You receive: repo stats and a list of recently updated issues, PRs, and releases.
|
|
Output a single JSON object — no prose around it — matching this exact shape:
|
|
|
|
{
|
|
"headline": "string — one short editorial headline summarising the project's pulse this period (max ~70 chars)",
|
|
"summary": "string — 2-3 sentences in a calm, factual editorial voice. No marketing fluff. No emoji.",
|
|
"highlights": [
|
|
{ "title": "string", "href": "string", "tag": "shipped|merged|opened|discussion|release", "blurb": "one sentence, max ~120 chars" }
|
|
],
|
|
"movers": [
|
|
{ "number": 123, "title": "string", "href": "string", "reason": "one short clause explaining why it matters" }
|
|
]
|
|
}
|
|
|
|
Rules:
|
|
- Pick 3-5 highlights and 3-5 movers from the actual provided items. Never invent.
|
|
- Prefer items with discussion, merged PRs, recent releases, or labelled "good first issue".
|
|
- Tone: like a small-paper editor — measured, specific, never breathless.
|
|
- Never use words like "exciting", "amazing", "powerful", "revolutionary".
|
|
- href must be the html_url provided.`;
|
|
|
|
export async function curate(
|
|
apiKey: string,
|
|
stats: RepoStats,
|
|
feed: FeedItem[]
|
|
): Promise<CuratedDispatch> {
|
|
const trimmedFeed = feed.slice(0, 25).map((f) => ({
|
|
kind: f.kind,
|
|
number: f.number,
|
|
title: f.title,
|
|
state: f.state,
|
|
href: f.url,
|
|
author: f.author,
|
|
updated: f.updatedAt,
|
|
comments: f.comments,
|
|
labels: f.labels.map((l) => l.name),
|
|
}));
|
|
|
|
const userPayload = {
|
|
repo: "Hmbown/deepseek-tui",
|
|
stats: {
|
|
stars: stats.stars,
|
|
forks: stats.forks,
|
|
open_issues: stats.openIssues,
|
|
open_pulls: stats.openPulls,
|
|
latest_release: stats.latestRelease?.tag,
|
|
},
|
|
recent: trimmedFeed,
|
|
};
|
|
|
|
const raw = await chat(
|
|
[
|
|
{ role: "system", content: SYSTEM_PROMPT },
|
|
{ role: "user", content: JSON.stringify(userPayload, null, 2) },
|
|
],
|
|
apiKey,
|
|
true
|
|
);
|
|
|
|
const parsed = JSON.parse(raw) as Omit<CuratedDispatch, "generatedAt">;
|
|
return { ...parsed, generatedAt: new Date().toISOString() };
|
|
}
|