105 lines
3.3 KiB
TypeScript
105 lines
3.3 KiB
TypeScript
import type { CuratedDispatch, FeedItem, RepoStats } from "./types";
|
|
|
|
const FALLBACK_BASE = "https://api.deepseek.com";
|
|
const FALLBACK_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 base = process.env.DEEPSEEK_BASE_URL ?? FALLBACK_BASE;
|
|
const model = process.env.DEEPSEEK_MODEL ?? FALLBACK_MODEL;
|
|
const res = await fetch(`${base}/v1/chat/completions`, {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
Authorization: `Bearer ${apiKey}`,
|
|
},
|
|
body: JSON.stringify({
|
|
model,
|
|
messages,
|
|
temperature: 0.4,
|
|
max_tokens: 4096,
|
|
reasoning_effort: "high",
|
|
...(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() };
|
|
}
|