@astro-minimax/cli 0.5.0 → 0.7.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +69 -0
- package/dist/commands/ai.d.ts +2 -0
- package/dist/commands/ai.d.ts.map +1 -0
- package/dist/commands/ai.js +99 -0
- package/dist/commands/ai.js.map +1 -0
- package/dist/commands/data.d.ts +2 -0
- package/dist/commands/data.d.ts.map +1 -0
- package/dist/commands/data.js +111 -0
- package/dist/commands/data.js.map +1 -0
- package/dist/commands/hooks.d.ts +2 -0
- package/dist/commands/hooks.d.ts.map +1 -0
- package/dist/commands/hooks.js +378 -0
- package/dist/commands/hooks.js.map +1 -0
- package/dist/commands/init.d.ts +2 -0
- package/dist/commands/init.d.ts.map +1 -0
- package/dist/commands/init.js +50 -0
- package/dist/commands/init.js.map +1 -0
- package/dist/commands/podcast.d.ts +2 -0
- package/dist/commands/podcast.d.ts.map +1 -0
- package/dist/commands/podcast.js +89 -0
- package/dist/commands/podcast.js.map +1 -0
- package/dist/commands/post.d.ts +2 -0
- package/dist/commands/post.d.ts.map +1 -0
- package/dist/commands/post.js +190 -0
- package/dist/commands/post.js.map +1 -0
- package/dist/commands/profile.d.ts +2 -0
- package/dist/commands/profile.d.ts.map +1 -0
- package/dist/commands/profile.js +88 -0
- package/dist/commands/profile.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +81 -0
- package/dist/index.js.map +1 -0
- package/dist/tools/ai-process.d.ts +20 -0
- package/dist/tools/ai-process.d.ts.map +1 -0
- package/dist/tools/ai-process.js +607 -0
- package/dist/tools/ai-process.js.map +1 -0
- package/dist/tools/build-author-context.d.ts +13 -0
- package/dist/tools/build-author-context.d.ts.map +1 -0
- package/dist/tools/build-author-context.js +313 -0
- package/dist/tools/build-author-context.js.map +1 -0
- package/dist/tools/build-voice-profile.d.ts +12 -0
- package/dist/tools/build-voice-profile.d.ts.map +1 -0
- package/dist/tools/build-voice-profile.js +270 -0
- package/dist/tools/build-voice-profile.js.map +1 -0
- package/dist/tools/eval-ai-chat.d.ts +17 -0
- package/dist/tools/eval-ai-chat.d.ts.map +1 -0
- package/dist/tools/eval-ai-chat.js +362 -0
- package/dist/tools/eval-ai-chat.js.map +1 -0
- package/dist/tools/generate-author-profile.d.ts +14 -0
- package/dist/tools/generate-author-profile.d.ts.map +1 -0
- package/dist/tools/generate-author-profile.js +289 -0
- package/dist/tools/generate-author-profile.js.map +1 -0
- package/dist/tools/generate-cover.d.ts +14 -0
- package/dist/tools/generate-cover.d.ts.map +1 -0
- package/dist/tools/generate-cover.js +95 -0
- package/dist/tools/generate-cover.js.map +1 -0
- package/dist/tools/generate-og.d.ts +3 -0
- package/dist/tools/generate-og.d.ts.map +1 -0
- package/dist/tools/generate-og.js +254 -0
- package/dist/tools/generate-og.js.map +1 -0
- package/dist/tools/generate-related.d.ts +11 -0
- package/dist/tools/generate-related.d.ts.map +1 -0
- package/dist/tools/generate-related.js +124 -0
- package/dist/tools/generate-related.js.map +1 -0
- package/dist/tools/generate-tags.d.ts +14 -0
- package/dist/tools/generate-tags.d.ts.map +1 -0
- package/dist/tools/generate-tags.js +182 -0
- package/dist/tools/generate-tags.js.map +1 -0
- package/dist/tools/lib/ai-provider.d.ts +43 -0
- package/dist/tools/lib/ai-provider.d.ts.map +1 -0
- package/dist/tools/lib/ai-provider.js +146 -0
- package/dist/tools/lib/ai-provider.js.map +1 -0
- package/dist/tools/lib/audio-processor.d.ts +46 -0
- package/dist/tools/lib/audio-processor.d.ts.map +1 -0
- package/dist/tools/lib/audio-processor.js +188 -0
- package/dist/tools/lib/audio-processor.js.map +1 -0
- package/dist/tools/lib/frontmatter.d.ts +11 -0
- package/dist/tools/lib/frontmatter.d.ts.map +1 -0
- package/dist/tools/lib/frontmatter.js +80 -0
- package/dist/tools/lib/frontmatter.js.map +1 -0
- package/dist/tools/lib/index.d.ts +7 -0
- package/dist/tools/lib/index.d.ts.map +1 -0
- package/{template/tools/lib/index.ts → dist/tools/lib/index.js} +1 -0
- package/dist/tools/lib/index.js.map +1 -0
- package/dist/tools/lib/markdown.d.ts +6 -0
- package/dist/tools/lib/markdown.d.ts.map +1 -0
- package/dist/tools/lib/markdown.js +34 -0
- package/dist/tools/lib/markdown.js.map +1 -0
- package/dist/tools/lib/posts.d.ts +25 -0
- package/dist/tools/lib/posts.d.ts.map +1 -0
- package/dist/tools/lib/posts.js +63 -0
- package/dist/tools/lib/posts.js.map +1 -0
- package/dist/tools/lib/script-generator.d.ts +61 -0
- package/dist/tools/lib/script-generator.d.ts.map +1 -0
- package/dist/tools/lib/script-generator.js +182 -0
- package/dist/tools/lib/script-generator.js.map +1 -0
- package/dist/tools/lib/tts-provider.d.ts +65 -0
- package/dist/tools/lib/tts-provider.d.ts.map +1 -0
- package/dist/tools/lib/tts-provider.js +116 -0
- package/dist/tools/lib/tts-provider.js.map +1 -0
- package/dist/tools/lib/types.d.ts +129 -0
- package/dist/tools/lib/types.d.ts.map +1 -0
- package/dist/tools/lib/types.js +64 -0
- package/dist/tools/lib/types.js.map +1 -0
- package/dist/tools/lib/utils.d.ts +18 -0
- package/dist/tools/lib/utils.d.ts.map +1 -0
- package/dist/tools/lib/utils.js +121 -0
- package/dist/tools/lib/utils.js.map +1 -0
- package/dist/tools/lib/vectors.d.ts +27 -0
- package/dist/tools/lib/vectors.d.ts.map +1 -0
- package/dist/tools/lib/vectors.js +64 -0
- package/dist/tools/lib/vectors.js.map +1 -0
- package/dist/tools/podcast-feed.d.ts +6 -0
- package/dist/tools/podcast-feed.d.ts.map +1 -0
- package/dist/tools/podcast-feed.js +121 -0
- package/dist/tools/podcast-feed.js.map +1 -0
- package/dist/tools/podcast-generate.d.ts +15 -0
- package/dist/tools/podcast-generate.d.ts.map +1 -0
- package/dist/tools/podcast-generate.js +318 -0
- package/dist/tools/podcast-generate.js.map +1 -0
- package/dist/tools/podcast-list.d.ts +6 -0
- package/dist/tools/podcast-list.d.ts.map +1 -0
- package/dist/tools/podcast-list.js +66 -0
- package/dist/tools/podcast-list.js.map +1 -0
- package/dist/tools/summarize.d.ts +16 -0
- package/dist/tools/summarize.d.ts.map +1 -0
- package/dist/tools/summarize.js +108 -0
- package/dist/tools/summarize.js.map +1 -0
- package/dist/tools/translate.d.ts +13 -0
- package/dist/tools/translate.d.ts.map +1 -0
- package/dist/tools/translate.js +46 -0
- package/dist/tools/translate.js.map +1 -0
- package/dist/tools/vectorize.d.ts +13 -0
- package/dist/tools/vectorize.d.ts.map +1 -0
- package/dist/tools/vectorize.js +87 -0
- package/dist/tools/vectorize.js.map +1 -0
- package/package.json +14 -9
- package/template/astro.config.ts +8 -28
- package/template/datas/ai-seo.json +8 -0
- package/template/datas/ai-skip-list.json +1 -0
- package/template/datas/author-profile-context.json +21 -0
- package/template/datas/author-profile-report.json +21 -0
- package/template/datas/eval/gold-set.json +72 -0
- package/template/functions/README.md +82 -0
- package/template/functions/api/ai-info.ts +2 -2
- package/template/functions/api/chat.ts +4 -1
- package/template/functions/api/notify/comment.ts +140 -68
- package/template/functions/api/notify/debug.ts +41 -0
- package/template/functions/api/notify/status.ts +97 -0
- package/template/functions/api/notify/test-ai-chat.ts +67 -0
- package/template/package.json +22 -25
- package/template/src/config.ts +11 -0
- package/template/src/content.config.ts +29 -16
- package/template/src/env.d.ts +0 -5
- package/index.js +0 -36
- package/template/tools/README.md +0 -169
- package/template/tools/ai-process.ts +0 -816
- package/template/tools/build-author-context.ts +0 -405
- package/template/tools/build-voice-profile.ts +0 -322
- package/template/tools/generate-author-profile.ts +0 -369
- package/template/tools/generate-cover.ts +0 -123
- package/template/tools/generate-og.ts +0 -280
- package/template/tools/generate-related.ts +0 -146
- package/template/tools/generate-tags.ts +0 -251
- package/template/tools/lib/ai-provider.ts +0 -240
- package/template/tools/lib/frontmatter.ts +0 -94
- package/template/tools/lib/markdown.ts +0 -40
- package/template/tools/lib/posts.ts +0 -89
- package/template/tools/lib/utils.ts +0 -138
- package/template/tools/lib/vectors.ts +0 -96
- package/template/tools/summarize.ts +0 -142
- package/template/tools/translate.ts +0 -60
- package/template/tools/vectorize.ts +0 -105
|
@@ -0,0 +1,254 @@
|
|
|
1
|
+
#!/usr/bin/env npx tsx
|
|
2
|
+
/* eslint-disable no-console */
|
|
3
|
+
import { mkdir } from "node:fs/promises";
|
|
4
|
+
import path from "node:path";
|
|
5
|
+
import { spawn, execSync } from "node:child_process";
|
|
6
|
+
import sharp from "sharp";
|
|
7
|
+
const rootDir = process.cwd();
|
|
8
|
+
const inputDir = path.join(rootDir, ".codex-temp", "og");
|
|
9
|
+
const outputJpg = path.join(rootDir, "public", "astro-minimax-og.jpg");
|
|
10
|
+
const outputPng = path.join(rootDir, ".codex-temp", "og-result.png");
|
|
11
|
+
// ─── 画布 ────────────────────────────────────────────────────────────────────
|
|
12
|
+
const outputWidth = 2400;
|
|
13
|
+
const outputHeight = 1260;
|
|
14
|
+
// ─── 旋转角度(负 = 逆时针倾斜) ─────────────────────────────────────────────
|
|
15
|
+
const angleDeg = -32;
|
|
16
|
+
// ─── 卡片内容区尺寸(更大,以便人眼能看清内容)──────────────────────────────
|
|
17
|
+
const cardWidth = 840;
|
|
18
|
+
const cardHeight = 526;
|
|
19
|
+
// ─── 卡片帧留白(内容区四周的空白,让卡片之间有视觉间隙)────────────────────
|
|
20
|
+
// SVG 阴影扩散需要这块空间,同时提供"上下左右间距"
|
|
21
|
+
const cardPad = 36;
|
|
22
|
+
// ─── 网格参数(正交坐标系,旋转前) ─────────────────────────────────────────
|
|
23
|
+
const cols = 4;
|
|
24
|
+
const rows = 3;
|
|
25
|
+
// 列中心点的水平间距 = 卡片宽 + 列间隙
|
|
26
|
+
const colSpacing = cardWidth + cardPad * 2 + 80; // ≈ 1048
|
|
27
|
+
// 行中心点的垂直间距 = 卡片高 + 行间隙
|
|
28
|
+
// !! 关键修复:必须 > cardHeight,否则卡片在旋转前就重叠
|
|
29
|
+
const rowSpacing = cardHeight + cardPad * 2 + 72; // ≈ 670
|
|
30
|
+
// ─── 截图配置 ─────────────────────────────────────────────────────────────────
|
|
31
|
+
const pages = [
|
|
32
|
+
{ url: "/zh/", filename: "home-light.png", theme: "light" },
|
|
33
|
+
{ url: "/zh/", filename: "home-dark.png", theme: "dark" },
|
|
34
|
+
{ url: "/zh/categories/", filename: "categories-light.png", theme: "light" },
|
|
35
|
+
{ url: "/zh/posts/", filename: "posts-light.png", theme: "light" },
|
|
36
|
+
{ url: "/zh/tags/", filename: "tags-light.png", theme: "light" },
|
|
37
|
+
{ url: "/zh/search/", filename: "search-light.png", theme: "light" },
|
|
38
|
+
{ url: "/zh/friends/", filename: "friends-light.png", theme: "light" },
|
|
39
|
+
{ url: "/zh/about/", filename: "about-light.png", theme: "light" },
|
|
40
|
+
{ url: "/zh/posts/dynamic-og-images/", filename: "post-light.png", theme: "light" },
|
|
41
|
+
{ url: "/zh/archives/", filename: "archives-light.png", theme: "light" },
|
|
42
|
+
{ url: "/zh/", filename: "home-light2.png", theme: "light" },
|
|
43
|
+
{ url: "/zh/posts/", filename: "posts-dark.png", theme: "dark" },
|
|
44
|
+
];
|
|
45
|
+
// ─── 布局:gridLayout[行][列] = 文件名 ───────────────────────────────────────
|
|
46
|
+
const gridLayout = [
|
|
47
|
+
["home-dark.png", "posts-light.png", "tags-light.png", "about-light.png"],
|
|
48
|
+
["home-light.png", "categories-light.png", "search-light.png", "post-light.png"],
|
|
49
|
+
["friends-light.png", "archives-light.png", "home-light2.png", "posts-dark.png"],
|
|
50
|
+
];
|
|
51
|
+
// ─── 旋转坐标(将正交网格坐标旋转 angleDeg)─────────────────────────────────
|
|
52
|
+
// !! 关键:必须同时旋转"坐标"和"卡片本身",两者角度保持一致,
|
|
53
|
+
// 才能让同一列的卡片在旋转后依然垂直对齐(沿倾斜方向排成一列)
|
|
54
|
+
function rotatePoint(x, y) {
|
|
55
|
+
const rad = (angleDeg * Math.PI) / 180;
|
|
56
|
+
return {
|
|
57
|
+
x: x * Math.cos(rad) - y * Math.sin(rad),
|
|
58
|
+
y: x * Math.sin(rad) + y * Math.cos(rad),
|
|
59
|
+
};
|
|
60
|
+
}
|
|
61
|
+
// ─── 背景 SVG ─────────────────────────────────────────────────────────────────
|
|
62
|
+
function backgroundSvg(w, h) {
|
|
63
|
+
return `<svg width="${w}" height="${h}" viewBox="0 0 ${w} ${h}" xmlns="http://www.w3.org/2000/svg">
|
|
64
|
+
<defs>
|
|
65
|
+
<linearGradient id="bg" x1="0" y1="0" x2="${w}" y2="${h}" gradientUnits="userSpaceOnUse">
|
|
66
|
+
<stop stop-color="#E8ECF2"/>
|
|
67
|
+
<stop offset="0.5" stop-color="#DEE5EE"/>
|
|
68
|
+
<stop offset="1" stop-color="#D5DDE8"/>
|
|
69
|
+
</linearGradient>
|
|
70
|
+
<filter id="glow" x="-50%" y="-50%" width="200%" height="200%">
|
|
71
|
+
<feGaussianBlur stdDeviation="32"/>
|
|
72
|
+
</filter>
|
|
73
|
+
</defs>
|
|
74
|
+
<rect width="${w}" height="${h}" fill="url(#bg)"/>
|
|
75
|
+
<g opacity="0.78" filter="url(#glow)">
|
|
76
|
+
<rect x="100" y="-900" width="180" height="3200" rx="90" transform="rotate(45 240 660)" fill="#FFFFFF"/>
|
|
77
|
+
<rect x="980" y="-940" width="196" height="3300" rx="98" transform="rotate(45 1060 655)" fill="#F6FAFD"/>
|
|
78
|
+
<rect x="1860" y="-910" width="180" height="3240" rx="90" transform="rotate(45 1880 650)" fill="#FFFFFF"/>
|
|
79
|
+
</g>
|
|
80
|
+
</svg>`;
|
|
81
|
+
}
|
|
82
|
+
// ─── 卡片 SVG(圆角 + 阴影 + 内容图像) ─────────────────────────────────────
|
|
83
|
+
function cardSvg(imageBase64) {
|
|
84
|
+
const p = cardPad;
|
|
85
|
+
const r = 20;
|
|
86
|
+
const fw = cardWidth + p * 2;
|
|
87
|
+
const fh = cardHeight + p * 2;
|
|
88
|
+
return `<svg width="${fw}" height="${fh}" viewBox="0 0 ${fw} ${fh}" xmlns="http://www.w3.org/2000/svg">
|
|
89
|
+
<defs>
|
|
90
|
+
<clipPath id="clip">
|
|
91
|
+
<rect x="${p}" y="${p}" width="${cardWidth}" height="${cardHeight}" rx="${r}"/>
|
|
92
|
+
</clipPath>
|
|
93
|
+
<filter id="shadow" x="-30%" y="-30%" width="160%" height="190%">
|
|
94
|
+
<feDropShadow dx="0" dy="18" stdDeviation="16" flood-color="#1A2840" flood-opacity="0.16"/>
|
|
95
|
+
</filter>
|
|
96
|
+
</defs>
|
|
97
|
+
<g filter="url(#shadow)">
|
|
98
|
+
<rect x="${p}" y="${p}" width="${cardWidth}" height="${cardHeight}" rx="${r}" fill="#FAFAF8"/>
|
|
99
|
+
<image
|
|
100
|
+
href="data:image/png;base64,${imageBase64}"
|
|
101
|
+
x="${p}" y="${p}" width="${cardWidth}" height="${cardHeight}"
|
|
102
|
+
preserveAspectRatio="xMidYMid slice"
|
|
103
|
+
clip-path="url(#clip)"/>
|
|
104
|
+
<rect x="${p}" y="${p}" width="${cardWidth}" height="${cardHeight}"
|
|
105
|
+
rx="${r}" fill="none" stroke="#D4DCE7" stroke-width="1.5"/>
|
|
106
|
+
</g>
|
|
107
|
+
</svg>`;
|
|
108
|
+
}
|
|
109
|
+
// ─── 渲染单张卡片(裁剪 → 帧 → 旋转) ───────────────────────────────────────
|
|
110
|
+
async function renderCard(filename) {
|
|
111
|
+
const isDark = filename.includes("dark");
|
|
112
|
+
const gravity = filename.includes("friends") ? "center" : "north";
|
|
113
|
+
const brightness = isDark ? 1.07 : 1.03;
|
|
114
|
+
const saturation = isDark ? 1.12 : 1.06;
|
|
115
|
+
// 1. 截图裁剪到卡片内容区
|
|
116
|
+
const cropped = await sharp(path.join(inputDir, filename))
|
|
117
|
+
.resize(cardWidth, cardHeight, { fit: "cover", position: gravity })
|
|
118
|
+
.modulate({ brightness, saturation })
|
|
119
|
+
.sharpen()
|
|
120
|
+
.png()
|
|
121
|
+
.toBuffer();
|
|
122
|
+
// 2. 合成卡片帧(圆角 + 阴影 + 边框 + padding)
|
|
123
|
+
const framed = await sharp(Buffer.from(cardSvg(cropped.toString("base64"))))
|
|
124
|
+
.png()
|
|
125
|
+
.toBuffer();
|
|
126
|
+
// 3. 旋转卡片本身(透明背景)
|
|
127
|
+
// !! 与 rotatePoint() 使用同一角度,保证列内对齐
|
|
128
|
+
const rotated = await sharp(framed)
|
|
129
|
+
.rotate(angleDeg, { background: { r: 0, g: 0, b: 0, alpha: 0 } })
|
|
130
|
+
.png()
|
|
131
|
+
.toBuffer();
|
|
132
|
+
const { width = 0, height = 0 } = await sharp(rotated).metadata();
|
|
133
|
+
return { input: rotated, width, height };
|
|
134
|
+
}
|
|
135
|
+
// ─── 启动预览服务器 ────────────────────────────────────────────────────────────
|
|
136
|
+
function startPreviewServer() {
|
|
137
|
+
return new Promise((resolve, reject) => {
|
|
138
|
+
console.log("🚀 Starting preview server...");
|
|
139
|
+
const server = spawn("pnpm", ["run", "preview"], {
|
|
140
|
+
cwd: rootDir, stdio: ["ignore", "pipe", "pipe"], detached: false,
|
|
141
|
+
});
|
|
142
|
+
let url = "";
|
|
143
|
+
server.stdout.on("data", (data) => {
|
|
144
|
+
const out = data.toString();
|
|
145
|
+
console.log(out);
|
|
146
|
+
const m = out.match(/localhost:(\d+)/);
|
|
147
|
+
if (m && !url) {
|
|
148
|
+
url = `http://localhost:${m[1]}`;
|
|
149
|
+
console.log(`✅ Server ready at ${url}`);
|
|
150
|
+
setTimeout(() => resolve({ url, kill: () => server.kill("SIGTERM") }), 2000);
|
|
151
|
+
}
|
|
152
|
+
});
|
|
153
|
+
server.stderr.on("data", (d) => {
|
|
154
|
+
const s = d.toString();
|
|
155
|
+
if (!s.includes("Prebundling") && !s.includes("hmr"))
|
|
156
|
+
console.error(s);
|
|
157
|
+
});
|
|
158
|
+
server.on("error", reject);
|
|
159
|
+
setTimeout(() => { if (!url)
|
|
160
|
+
reject(new Error("Server startup timeout")); }, 30000);
|
|
161
|
+
});
|
|
162
|
+
}
|
|
163
|
+
// ─── 截图 ─────────────────────────────────────────────────────────────────────
|
|
164
|
+
async function captureScreenshots(baseUrl) {
|
|
165
|
+
console.log("\n📸 Starting screenshot capture...");
|
|
166
|
+
await mkdir(inputDir, { recursive: true });
|
|
167
|
+
const puppeteer = await import("puppeteer-core");
|
|
168
|
+
const chromePaths = [
|
|
169
|
+
"/opt/google/chrome/chrome", "/usr/bin/google-chrome",
|
|
170
|
+
"/usr/bin/chromium", "/usr/bin/chromium-browser",
|
|
171
|
+
];
|
|
172
|
+
let chromePath = "";
|
|
173
|
+
for (const p of chromePaths) {
|
|
174
|
+
try {
|
|
175
|
+
execSync(`ls -f ${p}`, { stdio: "ignore" });
|
|
176
|
+
chromePath = p;
|
|
177
|
+
break;
|
|
178
|
+
}
|
|
179
|
+
catch { /* try next */ }
|
|
180
|
+
}
|
|
181
|
+
if (!chromePath) {
|
|
182
|
+
console.error("Chrome not found");
|
|
183
|
+
process.exit(1);
|
|
184
|
+
}
|
|
185
|
+
const browser = await puppeteer.default.launch({
|
|
186
|
+
executablePath: chromePath, headless: true,
|
|
187
|
+
args: ["--no-sandbox", "--disable-setuid-sandbox"],
|
|
188
|
+
});
|
|
189
|
+
try {
|
|
190
|
+
for (const cfg of pages) {
|
|
191
|
+
console.log(` Capturing: ${cfg.filename}`);
|
|
192
|
+
const page = await browser.newPage();
|
|
193
|
+
await page.setViewport({ width: 1920, height: 1080 });
|
|
194
|
+
await page.emulateMediaFeatures([{ name: "prefers-color-scheme", value: cfg.theme }]);
|
|
195
|
+
await page.goto(`${baseUrl}${cfg.url}`, { waitUntil: "networkidle0", timeout: 30000 });
|
|
196
|
+
await new Promise(r => setTimeout(r, 1000));
|
|
197
|
+
await page.screenshot({
|
|
198
|
+
path: path.join(inputDir, cfg.filename), type: "png",
|
|
199
|
+
});
|
|
200
|
+
console.log(` ✅ ${cfg.filename}`);
|
|
201
|
+
await page.close();
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
finally {
|
|
205
|
+
await browser.close();
|
|
206
|
+
}
|
|
207
|
+
console.log("✅ All screenshots captured!");
|
|
208
|
+
}
|
|
209
|
+
// ─── 主流程 ───────────────────────────────────────────────────────────────────
|
|
210
|
+
async function main() {
|
|
211
|
+
await mkdir(path.join(rootDir, ".codex-temp"), { recursive: true });
|
|
212
|
+
const server = await startPreviewServer();
|
|
213
|
+
try {
|
|
214
|
+
await captureScreenshots(server.url);
|
|
215
|
+
console.log("\n🎨 Compositing OG image...");
|
|
216
|
+
const cx = outputWidth / 2; // 1200
|
|
217
|
+
const cy = outputHeight / 2; // 630
|
|
218
|
+
const composites = [];
|
|
219
|
+
for (let col = 0; col < cols; col++) {
|
|
220
|
+
// ── Step 1:确定列中心 X(正交坐标系,旋转前)────────────────────────
|
|
221
|
+
const gridX = (col - (cols - 1) / 2) * colSpacing;
|
|
222
|
+
for (let row = 0; row < rows; row++) {
|
|
223
|
+
const filename = gridLayout[row]?.[col];
|
|
224
|
+
if (!filename)
|
|
225
|
+
continue;
|
|
226
|
+
// ── Step 2:确定行中心 Y(正交坐标系,旋转前)─────────────────────
|
|
227
|
+
// 同一列所有卡片的 gridX 完全相同 → 旋转后沿倾斜方向对齐
|
|
228
|
+
const gridY = (row - (rows - 1) / 2) * rowSpacing;
|
|
229
|
+
// ── Step 3:旋转坐标到最终画布坐标 ───────────────────────────────
|
|
230
|
+
const pos = rotatePoint(gridX, gridY);
|
|
231
|
+
// ── Step 4:渲染(卡片本身也旋转相同角度)─────────────────────────
|
|
232
|
+
const card = await renderCard(filename);
|
|
233
|
+
composites.push({
|
|
234
|
+
input: card.input,
|
|
235
|
+
left: Math.round(cx + pos.x - card.width / 2),
|
|
236
|
+
top: Math.round(cy + pos.y - card.height / 2),
|
|
237
|
+
});
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
const bg = Buffer.from(backgroundSvg(outputWidth, outputHeight));
|
|
241
|
+
const result = sharp(bg).composite(composites);
|
|
242
|
+
await result.clone().png().toFile(outputPng);
|
|
243
|
+
await result.clone().jpeg({ quality: 93, mozjpeg: true }).toFile(outputJpg);
|
|
244
|
+
console.log(`\n✅ Done!`);
|
|
245
|
+
console.log(` PNG → ${outputPng}`);
|
|
246
|
+
console.log(` JPG → ${outputJpg}`);
|
|
247
|
+
}
|
|
248
|
+
finally {
|
|
249
|
+
console.log("\n🛑 Stopping server...");
|
|
250
|
+
server.kill();
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
main().then(() => process.exit(0)).catch(e => { console.error(e); process.exit(1); });
|
|
254
|
+
//# sourceMappingURL=generate-og.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"generate-og.js","sourceRoot":"","sources":["../../src/tools/generate-og.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
#!/usr/bin/env npx tsx
|
|
2
|
+
/**
|
|
3
|
+
* 关联文章推荐工具
|
|
4
|
+
*
|
|
5
|
+
* 用法:
|
|
6
|
+
* pnpm run tools:generate-related # 基于标签/标题的关联推荐
|
|
7
|
+
* pnpm run tools:generate-related --ai # 使用 AI 语义分析(需要向量索引)
|
|
8
|
+
* pnpm run tools:generate-related --verbose # 显示详细评分
|
|
9
|
+
*/
|
|
10
|
+
export {};
|
|
11
|
+
//# sourceMappingURL=generate-related.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"generate-related.d.ts","sourceRoot":"","sources":["../../src/tools/generate-related.ts"],"names":[],"mappings":";AACA;;;;;;;GAOG"}
|
|
@@ -0,0 +1,124 @@
|
|
|
1
|
+
#!/usr/bin/env npx tsx
|
|
2
|
+
/**
|
|
3
|
+
* 关联文章推荐工具
|
|
4
|
+
*
|
|
5
|
+
* 用法:
|
|
6
|
+
* pnpm run tools:generate-related # 基于标签/标题的关联推荐
|
|
7
|
+
* pnpm run tools:generate-related --ai # 使用 AI 语义分析(需要向量索引)
|
|
8
|
+
* pnpm run tools:generate-related --verbose # 显示详细评分
|
|
9
|
+
*/
|
|
10
|
+
import { readFile } from "node:fs/promises";
|
|
11
|
+
import { join } from "node:path";
|
|
12
|
+
import { getAllPosts } from "./lib/posts.js";
|
|
13
|
+
import { cosineSimilarity } from "./lib/vectors.js";
|
|
14
|
+
const VECTOR_INDEX = join(process.cwd(), "src/data/vectors/index.json");
|
|
15
|
+
function tagSimilarity(a, b) {
|
|
16
|
+
let score = 0;
|
|
17
|
+
const aTags = new Set(a.tags.map(t => t.toLowerCase()));
|
|
18
|
+
const bTags = new Set(b.tags.map(t => t.toLowerCase()));
|
|
19
|
+
for (const tag of aTags) {
|
|
20
|
+
if (bTags.has(tag))
|
|
21
|
+
score += 3;
|
|
22
|
+
}
|
|
23
|
+
if (a.category && b.category) {
|
|
24
|
+
if (a.category === b.category)
|
|
25
|
+
score += 5;
|
|
26
|
+
else {
|
|
27
|
+
const ap = a.category.split("/");
|
|
28
|
+
const bp = b.category.split("/");
|
|
29
|
+
if (ap[0] === bp[0])
|
|
30
|
+
score += 2;
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
const aWords = new Set(a.title
|
|
34
|
+
.toLowerCase()
|
|
35
|
+
.split(/\s+/)
|
|
36
|
+
.filter(w => w.length > 2));
|
|
37
|
+
const bWords = new Set(b.title
|
|
38
|
+
.toLowerCase()
|
|
39
|
+
.split(/\s+/)
|
|
40
|
+
.filter(w => w.length > 2));
|
|
41
|
+
for (const w of aWords) {
|
|
42
|
+
if (bWords.has(w))
|
|
43
|
+
score += 1;
|
|
44
|
+
}
|
|
45
|
+
return score;
|
|
46
|
+
}
|
|
47
|
+
async function loadVectorIndex() {
|
|
48
|
+
try {
|
|
49
|
+
const content = await readFile(VECTOR_INDEX, "utf-8");
|
|
50
|
+
return JSON.parse(content);
|
|
51
|
+
}
|
|
52
|
+
catch {
|
|
53
|
+
return null;
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
function vectorSimilarity(postIdA, postIdB, index) {
|
|
57
|
+
const chunksA = index.chunks.filter(c => c.postId === postIdA && c.vector);
|
|
58
|
+
const chunksB = index.chunks.filter(c => c.postId === postIdB && c.vector);
|
|
59
|
+
if (chunksA.length === 0 || chunksB.length === 0)
|
|
60
|
+
return 0;
|
|
61
|
+
let maxSim = 0;
|
|
62
|
+
for (const ca of chunksA) {
|
|
63
|
+
for (const cb of chunksB) {
|
|
64
|
+
const sim = cosineSimilarity(ca.vector, cb.vector);
|
|
65
|
+
if (sim > maxSim)
|
|
66
|
+
maxSim = sim;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
return maxSim;
|
|
70
|
+
}
|
|
71
|
+
async function main() {
|
|
72
|
+
const args = process.argv.slice(2);
|
|
73
|
+
const useAI = args.includes("--ai");
|
|
74
|
+
const verbose = args.includes("--verbose");
|
|
75
|
+
const posts = await getAllPosts();
|
|
76
|
+
console.log(`📚 共 ${posts.length} 篇文章\n`);
|
|
77
|
+
let vectorIndex = null;
|
|
78
|
+
if (useAI) {
|
|
79
|
+
vectorIndex = await loadVectorIndex();
|
|
80
|
+
if (!vectorIndex) {
|
|
81
|
+
console.error("❌ 向量索引未找到。请先运行: pnpm run tools:vectorize");
|
|
82
|
+
process.exit(1);
|
|
83
|
+
}
|
|
84
|
+
console.log(`🧠 使用向量索引(${vectorIndex.method} 模式,${vectorIndex.chunks.length} 个块)\n`);
|
|
85
|
+
}
|
|
86
|
+
const zhPosts = posts.filter(p => p.lang === "zh");
|
|
87
|
+
const enPosts = posts.filter(p => p.lang === "en");
|
|
88
|
+
for (const langPosts of [zhPosts, enPosts]) {
|
|
89
|
+
if (langPosts.length === 0)
|
|
90
|
+
continue;
|
|
91
|
+
const langLabel = langPosts[0].lang === "zh" ? "中文" : "English";
|
|
92
|
+
console.log(`\n=== ${langLabel} (${langPosts.length} 篇) ===\n`);
|
|
93
|
+
for (const post of langPosts) {
|
|
94
|
+
const scores = langPosts
|
|
95
|
+
.filter(p => p.id !== post.id)
|
|
96
|
+
.map(p => {
|
|
97
|
+
let score = tagSimilarity(post, p);
|
|
98
|
+
if (useAI && vectorIndex) {
|
|
99
|
+
const vecSim = vectorSimilarity(post.id, p.id, vectorIndex);
|
|
100
|
+
score += vecSim * 10;
|
|
101
|
+
}
|
|
102
|
+
return { post: p, score };
|
|
103
|
+
})
|
|
104
|
+
.sort((a, b) => b.score - a.score)
|
|
105
|
+
.slice(0, 3);
|
|
106
|
+
console.log(`📝 【${post.title}】`);
|
|
107
|
+
if (scores.length > 0) {
|
|
108
|
+
console.log(" 推荐关联:");
|
|
109
|
+
for (const s of scores) {
|
|
110
|
+
const scoreStr = verbose
|
|
111
|
+
? ` (得分: ${s.score.toFixed(2)})`
|
|
112
|
+
: "";
|
|
113
|
+
console.log(` - ${s.post.title}${scoreStr}`);
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
else {
|
|
117
|
+
console.log(" (无匹配)");
|
|
118
|
+
}
|
|
119
|
+
console.log("");
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
main().catch(console.error);
|
|
124
|
+
//# sourceMappingURL=generate-related.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"generate-related.js","sourceRoot":"","sources":["../../src/tools/generate-related.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
#!/usr/bin/env npx tsx
|
|
2
|
+
/**
|
|
3
|
+
* AI 标签与分类自动生成工具
|
|
4
|
+
*
|
|
5
|
+
* 用法:
|
|
6
|
+
* pnpm run tools:tags <文章路径> # 分析并推荐标签和分类
|
|
7
|
+
* pnpm run tools:tags <文章路径> --write # 推荐并写入 frontmatter
|
|
8
|
+
* pnpm run tools:tags --all # 分析所有文章(dry-run)
|
|
9
|
+
*
|
|
10
|
+
* 环境变量:
|
|
11
|
+
* AI_API_KEY / OPENAI_API_KEY(可选,无 key 时使用关键词匹配)
|
|
12
|
+
*/
|
|
13
|
+
export {};
|
|
14
|
+
//# sourceMappingURL=generate-tags.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"generate-tags.d.ts","sourceRoot":"","sources":["../../src/tools/generate-tags.ts"],"names":[],"mappings":";AACA;;;;;;;;;;GAUG"}
|
|
@@ -0,0 +1,182 @@
|
|
|
1
|
+
#!/usr/bin/env npx tsx
|
|
2
|
+
/**
|
|
3
|
+
* AI 标签与分类自动生成工具
|
|
4
|
+
*
|
|
5
|
+
* 用法:
|
|
6
|
+
* pnpm run tools:tags <文章路径> # 分析并推荐标签和分类
|
|
7
|
+
* pnpm run tools:tags <文章路径> --write # 推荐并写入 frontmatter
|
|
8
|
+
* pnpm run tools:tags --all # 分析所有文章(dry-run)
|
|
9
|
+
*
|
|
10
|
+
* 环境变量:
|
|
11
|
+
* AI_API_KEY / OPENAI_API_KEY(可选,无 key 时使用关键词匹配)
|
|
12
|
+
*/
|
|
13
|
+
import { readFile, writeFile } from "node:fs/promises";
|
|
14
|
+
import { join } from "node:path";
|
|
15
|
+
import { extractFrontmatter } from "./lib/frontmatter.js";
|
|
16
|
+
import { stripMarkdown } from "./lib/markdown.js";
|
|
17
|
+
import { getAllPosts, getExistingTaxonomy, } from "./lib/posts.js";
|
|
18
|
+
import { chatCompletion, hasAPIKey } from "./lib/ai-provider.js";
|
|
19
|
+
const KEYWORD_TAG_MAP = {
|
|
20
|
+
astro: ["astro"],
|
|
21
|
+
react: ["reactjs"],
|
|
22
|
+
vue: ["vue"],
|
|
23
|
+
next: ["nextjs"],
|
|
24
|
+
tailwind: ["tailwindcss"],
|
|
25
|
+
typescript: ["typescript"],
|
|
26
|
+
javascript: ["javascript"],
|
|
27
|
+
css: ["css"],
|
|
28
|
+
html: ["html"],
|
|
29
|
+
markdown: ["markdown"],
|
|
30
|
+
mdx: ["markdown"],
|
|
31
|
+
api: ["api"],
|
|
32
|
+
database: ["database"],
|
|
33
|
+
docker: ["docker"],
|
|
34
|
+
git: ["git"],
|
|
35
|
+
python: ["python"],
|
|
36
|
+
rust: ["rust"],
|
|
37
|
+
node: ["nodejs"],
|
|
38
|
+
deploy: ["deployment"],
|
|
39
|
+
test: ["testing"],
|
|
40
|
+
performance: ["performance"],
|
|
41
|
+
seo: ["seo"],
|
|
42
|
+
a11y: ["accessibility"],
|
|
43
|
+
ai: ["ai"],
|
|
44
|
+
机器学习: ["ai", "machine-learning"],
|
|
45
|
+
深度学习: ["ai", "deep-learning"],
|
|
46
|
+
大模型: ["ai", "llm"],
|
|
47
|
+
};
|
|
48
|
+
function localTagSuggestion(post, existingTags) {
|
|
49
|
+
const text = `${post.title} ${post.description} ${post.body}`.toLowerCase();
|
|
50
|
+
const suggestedTags = new Set();
|
|
51
|
+
for (const [keyword, tags] of Object.entries(KEYWORD_TAG_MAP)) {
|
|
52
|
+
if (text.includes(keyword.toLowerCase())) {
|
|
53
|
+
tags.forEach(t => suggestedTags.add(t));
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
for (const tag of existingTags) {
|
|
57
|
+
if (text.includes(tag.toLowerCase()) && !suggestedTags.has(tag)) {
|
|
58
|
+
suggestedTags.add(tag);
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
const category = post.category || inferCategory(text);
|
|
62
|
+
return {
|
|
63
|
+
tags: Array.from(suggestedTags).slice(0, 6),
|
|
64
|
+
category,
|
|
65
|
+
reasoning: "基于关键词匹配(本地模式,无需 API Key)",
|
|
66
|
+
};
|
|
67
|
+
}
|
|
68
|
+
function inferCategory(text) {
|
|
69
|
+
if (text.includes("教程") ||
|
|
70
|
+
text.includes("tutorial") ||
|
|
71
|
+
text.includes("how to"))
|
|
72
|
+
return "教程";
|
|
73
|
+
if (text.includes("配置") || text.includes("config"))
|
|
74
|
+
return "教程/配置";
|
|
75
|
+
if (text.includes("发布") || text.includes("release"))
|
|
76
|
+
return "发布";
|
|
77
|
+
if (text.includes("示例") ||
|
|
78
|
+
text.includes("example") ||
|
|
79
|
+
text.includes("showcase"))
|
|
80
|
+
return "示例";
|
|
81
|
+
return "技术";
|
|
82
|
+
}
|
|
83
|
+
async function aiTagSuggestion(post, existingTags, existingCategories) {
|
|
84
|
+
const contentPreview = post.body.slice(0, 3000);
|
|
85
|
+
const prompt = `分析以下博客文章,推荐合适的标签和分类。
|
|
86
|
+
|
|
87
|
+
文章标题: ${post.title}
|
|
88
|
+
文章描述: ${post.description}
|
|
89
|
+
正文摘要: ${contentPreview}
|
|
90
|
+
|
|
91
|
+
现有标签体系: ${existingTags.join(", ")}
|
|
92
|
+
现有分类体系: ${existingCategories.join(", ")}
|
|
93
|
+
|
|
94
|
+
请优先使用现有标签和分类以保持一致性。如果确实需要新标签也可以。
|
|
95
|
+
|
|
96
|
+
回复 JSON 格式:
|
|
97
|
+
{
|
|
98
|
+
"tags": ["tag1", "tag2", "tag3"],
|
|
99
|
+
"category": "分类名",
|
|
100
|
+
"reasoning": "推荐理由简述"
|
|
101
|
+
}`;
|
|
102
|
+
const result = await chatCompletion([
|
|
103
|
+
{
|
|
104
|
+
role: "system",
|
|
105
|
+
content: "你是一个技术博客编辑,擅长内容分类。只回复 JSON,不要其他文字。",
|
|
106
|
+
},
|
|
107
|
+
{ role: "user", content: prompt },
|
|
108
|
+
], { maxTokens: 200, responseFormat: "json" });
|
|
109
|
+
return JSON.parse(result);
|
|
110
|
+
}
|
|
111
|
+
async function main() {
|
|
112
|
+
const args = process.argv.slice(2);
|
|
113
|
+
const all = args.includes("--all");
|
|
114
|
+
const write = args.includes("--write");
|
|
115
|
+
const filePath = args.find(a => !a.startsWith("--"));
|
|
116
|
+
const allPosts = await getAllPosts();
|
|
117
|
+
const { tags: existingTags, categories: existingCategories } = getExistingTaxonomy(allPosts);
|
|
118
|
+
console.log(`📊 现有标签 (${existingTags.length}): ${existingTags.slice(0, 15).join(", ")}...`);
|
|
119
|
+
console.log(`📁 现有分类 (${existingCategories.length}): ${existingCategories.join(", ")}\n`);
|
|
120
|
+
const postsToAnalyze = [];
|
|
121
|
+
if (all) {
|
|
122
|
+
postsToAnalyze.push(...allPosts);
|
|
123
|
+
}
|
|
124
|
+
else if (filePath) {
|
|
125
|
+
const fullPath = join(process.cwd(), filePath);
|
|
126
|
+
const content = await readFile(fullPath, "utf-8");
|
|
127
|
+
const fm = extractFrontmatter(content);
|
|
128
|
+
postsToAnalyze.push({
|
|
129
|
+
id: filePath,
|
|
130
|
+
filePath: fullPath,
|
|
131
|
+
lang: "zh",
|
|
132
|
+
title: fm.data.title || "",
|
|
133
|
+
description: fm.data.description || "",
|
|
134
|
+
tags: Array.isArray(fm.data.tags) ? fm.data.tags : [],
|
|
135
|
+
category: fm.data.category || "",
|
|
136
|
+
body: stripMarkdown(content),
|
|
137
|
+
});
|
|
138
|
+
}
|
|
139
|
+
else {
|
|
140
|
+
console.error("用法: pnpm run tools:tags <文章路径> [--write] 或 pnpm run tools:tags --all");
|
|
141
|
+
process.exit(1);
|
|
142
|
+
}
|
|
143
|
+
const mode = hasAPIKey() ? "AI" : "关键词匹配";
|
|
144
|
+
console.log(`🔍 使用 ${mode} 模式分析 ${postsToAnalyze.length} 篇文章...\n`);
|
|
145
|
+
for (const post of postsToAnalyze) {
|
|
146
|
+
let suggestion;
|
|
147
|
+
try {
|
|
148
|
+
if (hasAPIKey()) {
|
|
149
|
+
suggestion = await aiTagSuggestion(post, existingTags, existingCategories);
|
|
150
|
+
}
|
|
151
|
+
else {
|
|
152
|
+
suggestion = localTagSuggestion(post, existingTags);
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
catch (err) {
|
|
156
|
+
console.error(`❌ 分析失败 [${post.title}]:`, err.message);
|
|
157
|
+
continue;
|
|
158
|
+
}
|
|
159
|
+
console.log(`📝 【${post.title}】`);
|
|
160
|
+
console.log(` 当前标签: [${post.tags.join(", ")}]`);
|
|
161
|
+
console.log(` 推荐标签: [${suggestion.tags.join(", ")}]`);
|
|
162
|
+
console.log(` 当前分类: ${post.category || "(无)"}`);
|
|
163
|
+
console.log(` 推荐分类: ${suggestion.category}`);
|
|
164
|
+
console.log(` 理由: ${suggestion.reasoning}\n`);
|
|
165
|
+
if (write && !all) {
|
|
166
|
+
const content = await readFile(post.filePath, "utf-8");
|
|
167
|
+
let newContent = content;
|
|
168
|
+
const tagsYaml = suggestion.tags.map(t => ` - ${t}`).join("\n");
|
|
169
|
+
newContent = newContent.replace(/^tags:\n((?:\s+-\s+.*\n)*)/m, `tags:\n${tagsYaml}\n`);
|
|
170
|
+
if (post.category) {
|
|
171
|
+
newContent = newContent.replace(/^category:.*$/m, `category: ${suggestion.category}`);
|
|
172
|
+
}
|
|
173
|
+
else {
|
|
174
|
+
newContent = newContent.replace(/^(tags:\n(?:\s+-\s+.*\n)*)/, `$1category: ${suggestion.category}\n`);
|
|
175
|
+
}
|
|
176
|
+
await writeFile(post.filePath, newContent, "utf-8");
|
|
177
|
+
console.log(` ✍️ 已更新 frontmatter\n`);
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
main().catch(console.error);
|
|
182
|
+
//# sourceMappingURL=generate-tags.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"generate-tags.js","sourceRoot":"","sources":["../../src/tools/generate-tags.ts"],"names":[],"mappings":"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"}
|