zengen 0.1.36 → 0.2.0

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.
Files changed (153) hide show
  1. package/.github/workflows/pages.yml +1 -1
  2. package/.zen/meta.json +128 -30
  3. package/.zen/src/en-US/01d04f7c17b4a541ead9d759d877b30b403e15b849182a49eb1f62bd29ecd18c.md +120 -0
  4. package/.zen/src/en-US/1b798c44a4f353e47296ca83d5905e37e6aba3e90bbd9bc3b3d34fc12059a2ca.md +75 -0
  5. package/.zen/src/en-US/1e96be58d76c60056b708eb5bd8b8b81d7b5845d9cfe0b879d85068a5f11df3a.md +189 -0
  6. package/.zen/src/en-US/5ec990146b35e00de2630559126ee07f7cdcddeb23b0e8cab3d85b4181353e26.md +53 -0
  7. package/.zen/src/en-US/6124ea88edec5bde737b26b21f71ecfeffe4e73151784856edf813ee231a4baa.md +11 -0
  8. package/.zen/src/en-US/80ae9bed74fc6348a7c1fe9f33e86b65f5d919169721f77bcf0e1bc29fbdb4f9.md +61 -0
  9. package/.zen/src/en-US/f0c2799126931ccd113a0c45b1e623870b0d4f4f400becf6dd877da8f1011517.md +41 -0
  10. package/.zen/src/en-US/fdfca9b960d0eaa8b2b96fe988ead7481d2c0b16f66ebc94fb477139b4178cdc.md +65 -0
  11. package/.zen/src/zh-Hans/01d04f7c17b4a541ead9d759d877b30b403e15b849182a49eb1f62bd29ecd18c.md +120 -0
  12. package/.zen/src/zh-Hans/1b798c44a4f353e47296ca83d5905e37e6aba3e90bbd9bc3b3d34fc12059a2ca.md +77 -0
  13. package/.zen/src/zh-Hans/1e96be58d76c60056b708eb5bd8b8b81d7b5845d9cfe0b879d85068a5f11df3a.md +189 -0
  14. package/.zen/src/zh-Hans/5ec990146b35e00de2630559126ee07f7cdcddeb23b0e8cab3d85b4181353e26.md +55 -0
  15. package/.zen/src/zh-Hans/6124ea88edec5bde737b26b21f71ecfeffe4e73151784856edf813ee231a4baa.md +1 -0
  16. package/.zen/src/zh-Hans/6ad8db715a1b60613fe934fefb29fa981ecad9b63145593accff144d73b44bde.md +175 -0
  17. package/.zen/src/zh-Hans/80ae9bed74fc6348a7c1fe9f33e86b65f5d919169721f77bcf0e1bc29fbdb4f9.md +63 -0
  18. package/.zen/src/zh-Hans/a1580f71c6c6c1ff4a314be72d410a8507af2f087d56360c7f5048d349c21953.md +48 -0
  19. package/.zen/src/zh-Hans/d49012f98c4367b34034063400e2f7826bf0615952210c82396920172d468e2c.md +107 -0
  20. package/.zen/src/zh-Hans/f0c2799126931ccd113a0c45b1e623870b0d4f4f400becf6dd877da8f1011517.md +41 -0
  21. package/.zen/src/zh-Hans/fdfca9b960d0eaa8b2b96fe988ead7481d2c0b16f66ebc94fb477139b4178cdc.md +65 -0
  22. package/assets/templates/default/layout.html +274 -0
  23. package/dist/ai/extractMetadataFromMarkdown.d.ts +8 -0
  24. package/dist/ai/extractMetadataFromMarkdown.d.ts.map +1 -0
  25. package/dist/ai/extractMetadataFromMarkdown.js +88 -0
  26. package/dist/ai/extractMetadataFromMarkdown.js.map +1 -0
  27. package/dist/ai/translateMarkdown.d.ts +8 -0
  28. package/dist/ai/translateMarkdown.d.ts.map +1 -0
  29. package/dist/ai/translateMarkdown.js +29 -0
  30. package/dist/ai/translateMarkdown.js.map +1 -0
  31. package/dist/build/pipeline.d.ts +6 -0
  32. package/dist/build/pipeline.d.ts.map +1 -0
  33. package/dist/build/pipeline.js +218 -0
  34. package/dist/build/pipeline.js.map +1 -0
  35. package/dist/cli.js +10 -118
  36. package/dist/cli.js.map +1 -1
  37. package/dist/findEntries.d.ts +10 -0
  38. package/dist/findEntries.d.ts.map +1 -0
  39. package/dist/findEntries.js +38 -0
  40. package/dist/findEntries.js.map +1 -0
  41. package/dist/index.d.ts +1 -32
  42. package/dist/index.d.ts.map +1 -1
  43. package/dist/index.js +1 -35
  44. package/dist/index.js.map +1 -1
  45. package/dist/metadata.d.ts +14 -0
  46. package/dist/metadata.d.ts.map +1 -0
  47. package/dist/metadata.js +78 -0
  48. package/dist/metadata.js.map +1 -0
  49. package/dist/paths.d.ts +6 -0
  50. package/dist/paths.d.ts.map +1 -0
  51. package/dist/paths.js +10 -0
  52. package/dist/paths.js.map +1 -0
  53. package/dist/process/extractMetadataByAI.d.ts +5 -0
  54. package/dist/process/extractMetadataByAI.d.ts.map +1 -0
  55. package/dist/process/extractMetadataByAI.js +31 -0
  56. package/dist/process/extractMetadataByAI.js.map +1 -0
  57. package/dist/process/template.d.ts +5 -0
  58. package/dist/process/template.d.ts.map +1 -0
  59. package/dist/process/template.js +188 -0
  60. package/dist/process/template.js.map +1 -0
  61. package/dist/scan/files.d.ts +7 -0
  62. package/dist/scan/files.d.ts.map +1 -0
  63. package/dist/scan/files.js +54 -0
  64. package/dist/scan/files.js.map +1 -0
  65. package/dist/services/openai.d.ts +41 -0
  66. package/dist/services/openai.d.ts.map +1 -0
  67. package/dist/services/openai.js +54 -0
  68. package/dist/services/openai.js.map +1 -0
  69. package/dist/types.d.ts +16 -67
  70. package/dist/types.d.ts.map +1 -1
  71. package/dist/utils/convertMarkdownToHtml.d.ts +7 -0
  72. package/dist/utils/convertMarkdownToHtml.d.ts.map +1 -0
  73. package/dist/utils/convertMarkdownToHtml.js +39 -0
  74. package/dist/utils/convertMarkdownToHtml.js.map +1 -0
  75. package/dist/utils/frontmatter.d.ts +6 -0
  76. package/dist/utils/frontmatter.d.ts.map +1 -0
  77. package/dist/utils/frontmatter.js +22 -0
  78. package/dist/utils/frontmatter.js.map +1 -0
  79. package/docs/deployment/github-pages.md +1 -2
  80. package/docs/guides/best-practices.md +4 -4
  81. package/docs/guides/config.md +0 -5
  82. package/package.json +4 -2
  83. package/src/ai/extractMetadataFromMarkdown.ts +95 -0
  84. package/src/ai/translateMarkdown.ts +29 -0
  85. package/src/build/pipeline.ts +211 -0
  86. package/src/cli.ts +10 -132
  87. package/src/findEntries.ts +37 -0
  88. package/src/index.ts +1 -40
  89. package/src/metadata.ts +44 -0
  90. package/src/paths.ts +7 -0
  91. package/src/process/extractMetadataByAI.ts +29 -0
  92. package/src/process/template.ts +201 -0
  93. package/src/scan/files.ts +17 -0
  94. package/src/services/openai.ts +92 -0
  95. package/src/types.ts +18 -72
  96. package/src/utils/convertMarkdownToHtml.ts +32 -0
  97. package/src/utils/frontmatter.ts +18 -0
  98. package/.zen/translations.json +0 -51
  99. package/dist/ai-client.d.ts +0 -34
  100. package/dist/ai-client.d.ts.map +0 -1
  101. package/dist/ai-client.js +0 -180
  102. package/dist/ai-client.js.map +0 -1
  103. package/dist/ai-processor.d.ts +0 -51
  104. package/dist/ai-processor.d.ts.map +0 -1
  105. package/dist/ai-processor.js +0 -215
  106. package/dist/ai-processor.js.map +0 -1
  107. package/dist/ai-service.d.ts +0 -79
  108. package/dist/ai-service.d.ts.map +0 -1
  109. package/dist/ai-service.js +0 -257
  110. package/dist/ai-service.js.map +0 -1
  111. package/dist/builder.d.ts +0 -70
  112. package/dist/builder.d.ts.map +0 -1
  113. package/dist/builder.js +0 -854
  114. package/dist/builder.js.map +0 -1
  115. package/dist/gitignore.d.ts +0 -41
  116. package/dist/gitignore.d.ts.map +0 -1
  117. package/dist/gitignore.js +0 -202
  118. package/dist/gitignore.js.map +0 -1
  119. package/dist/gitignore.test.d.ts +0 -2
  120. package/dist/gitignore.test.d.ts.map +0 -1
  121. package/dist/gitignore.test.js +0 -309
  122. package/dist/gitignore.test.js.map +0 -1
  123. package/dist/markdown.d.ts +0 -35
  124. package/dist/markdown.d.ts.map +0 -1
  125. package/dist/markdown.js +0 -221
  126. package/dist/markdown.js.map +0 -1
  127. package/dist/navigation.d.ts +0 -46
  128. package/dist/navigation.d.ts.map +0 -1
  129. package/dist/navigation.js +0 -196
  130. package/dist/navigation.js.map +0 -1
  131. package/dist/scanner.d.ts +0 -26
  132. package/dist/scanner.d.ts.map +0 -1
  133. package/dist/scanner.js +0 -190
  134. package/dist/scanner.js.map +0 -1
  135. package/dist/template.d.ts +0 -33
  136. package/dist/template.d.ts.map +0 -1
  137. package/dist/template.js +0 -434
  138. package/dist/template.js.map +0 -1
  139. package/dist/translation-service.d.ts +0 -72
  140. package/dist/translation-service.d.ts.map +0 -1
  141. package/dist/translation-service.js +0 -291
  142. package/dist/translation-service.js.map +0 -1
  143. package/src/ai-client.ts +0 -227
  144. package/src/ai-processor.ts +0 -243
  145. package/src/ai-service.ts +0 -281
  146. package/src/builder.ts +0 -991
  147. package/src/gitignore.test.ts +0 -318
  148. package/src/gitignore.ts +0 -193
  149. package/src/markdown.ts +0 -212
  150. package/src/navigation.ts +0 -237
  151. package/src/scanner.ts +0 -180
  152. package/src/template.ts +0 -425
  153. package/src/translation-service.ts +0 -350
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "zengen",
3
- "version": "0.1.36",
3
+ "version": "0.2.0",
4
4
  "description": "ZEN - A minimalist Markdown documentation site builder",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",
@@ -10,6 +10,7 @@
10
10
  "scripts": {
11
11
  "build": "npx rimraf dist && tsc",
12
12
  "dev": "ts-node src/cli.ts",
13
+ "build:doc": "npm run build && node dist/cli.js build --lang zh-Hans --lang en-US --verbose",
13
14
  "test": "npm run build && node --test dist/**/*.test.js",
14
15
  "test:types": "tsc --noEmit",
15
16
  "test:build": "npm run build && test -f dist/index.js && test -f dist/cli.js",
@@ -58,6 +59,7 @@
58
59
  "express": "^4.21.2",
59
60
  "highlight.js": "^11.11.1",
60
61
  "marked": "^17.0.1",
61
- "minimatch": "^10.1.1"
62
+ "minimatch": "^10.1.1",
63
+ "yaml": "^2.8.2"
62
64
  }
63
65
  }
@@ -0,0 +1,95 @@
1
+ import { completeMessages, OpenAIMessage } from '../services/openai';
2
+ import { AIMetadata } from '../types';
3
+
4
+ /**
5
+ * 从 markdown 内容中提取 metadata
6
+ * @param content Markdown 内容
7
+ * @returns Promise<AIMetadata> 提取的元数据,失败时抛出错误
8
+ */
9
+ export async function extractMetadataFromMarkdown(content: string): Promise<AIMetadata> {
10
+ const prompt = buildMetadataPrompt(content);
11
+ const messages: OpenAIMessage[] = [
12
+ {
13
+ role: 'system',
14
+ content:
15
+ '你是一个专业的文档分析助手,擅长从文档中提取结构化信息。请严格按照要求的 JSON 格式返回结果。',
16
+ },
17
+ {
18
+ role: 'user',
19
+ content: prompt,
20
+ },
21
+ ];
22
+
23
+ const response = await completeMessages(messages, {
24
+ response_format: { type: 'json_object' },
25
+ });
26
+
27
+ const metadata = parseMetadataResponse(response.choices[0].message.content);
28
+
29
+ // 添加 tokens 使用情况
30
+ metadata.tokens_used = {
31
+ prompt: response.usage.prompt_tokens,
32
+ completion: response.usage.completion_tokens,
33
+ total: response.usage.total_tokens,
34
+ };
35
+
36
+ return metadata;
37
+ }
38
+
39
+ /**
40
+ * 构建提取 metadata 的 prompt
41
+ */
42
+ function buildMetadataPrompt(content: string): string {
43
+ // 限制内容长度以避免 token 超限
44
+ const maxContentLength = 8000;
45
+ const truncatedContent =
46
+ content.length > maxContentLength
47
+ ? content.substring(0, maxContentLength) + '... [内容已截断]'
48
+ : content;
49
+
50
+ return `请分析以下文档内容,提取以下信息并返回 JSON 格式:
51
+
52
+ 文档内容:
53
+ """
54
+ ${truncatedContent}
55
+ """
56
+
57
+ 请提取:
58
+ 1. title: 文档的标题(简洁明了,不超过 20 个字)
59
+ 2. summary: 文档摘要(控制在 100 字以内,概括主要内容)
60
+ 3. tags: 关键词列表(3-8 个关键词,使用中文或英文)
61
+ 4. inferred_date: 文档中隐含的创建日期(如果有的话,格式:YYYY-MM-DD,没有就留空字符串)
62
+ 5. inferred_lang: 文档使用的语言代码(例如:zh-Hans 表示简体中文,en-US 表示美式英语)
63
+
64
+ 请严格按照以下 JSON 格式返回,不要包含任何其他文本:
65
+ {
66
+ "title": "文档标题",
67
+ "summary": "文档摘要...",
68
+ "tags": ["关键词1", "关键词2", "关键词3"],
69
+ "inferred_date": "2023-01-01",
70
+ "inferred_lang": "zh-Hans"
71
+ }`;
72
+ }
73
+
74
+ /**
75
+ * 解析 AI 返回的 metadata
76
+ */
77
+ function parseMetadataResponse(responseContent: string): AIMetadata {
78
+ try {
79
+ const metadata = JSON.parse(responseContent);
80
+
81
+ // 验证和清理数据
82
+ return {
83
+ title: metadata.title?.trim() || '未命名文档',
84
+ summary: metadata.summary?.trim() || '',
85
+ tags: Array.isArray(metadata.tags)
86
+ ? metadata.tags.map((tag: string) => tag.trim()).filter(Boolean)
87
+ : [],
88
+ inferred_date: metadata.inferred_date?.trim() || undefined,
89
+ inferred_lang: metadata.inferred_lang?.trim() || 'zh-Hans',
90
+ };
91
+ } catch (error) {
92
+ console.error('❌ Failed to parse AI response:', error, 'Response:', responseContent);
93
+ throw error;
94
+ }
95
+ }
@@ -0,0 +1,29 @@
1
+ import { completeMessages, OpenAIMessage } from '../services/openai';
2
+
3
+ /**
4
+ * 将 markdown 翻译为指定的语言
5
+ * @param content Markdown 内容
6
+ * @param targetLang 目标语言代码(例如:zh-Hans, en-US)
7
+ * @returns Promise<string> 翻译后的 Markdown 内容
8
+ */
9
+ export async function translateMarkdown(content: string, targetLang: string): Promise<string> {
10
+ const messages: OpenAIMessage[] = [
11
+ {
12
+ role: 'system',
13
+ content: `你是一个专业的翻译助手,擅长将文档翻译成不同语言,同时保持原有的格式和结构。请将用户输入翻译成 ${targetLang},注意保持 Markdown 格式不变,链接不变,不要翻译代码,但是可以翻译代码中的注释。`,
14
+ },
15
+ {
16
+ role: 'user',
17
+ content: content,
18
+ },
19
+ ];
20
+
21
+ const response = await completeMessages(messages);
22
+ const translatedContent = response.choices[0]?.message?.content?.trim() || '';
23
+
24
+ if (!translatedContent) {
25
+ throw new Error('Empty translation response');
26
+ }
27
+
28
+ return translatedContent;
29
+ }
@@ -0,0 +1,211 @@
1
+ import * as fs from 'fs/promises';
2
+ import * as path from 'path';
3
+ import { translateMarkdown } from '../ai/translateMarkdown';
4
+ import { findMarkdownEntries } from '../findEntries';
5
+ import { loadMetaData, MetaData, saveMetaData } from '../metadata';
6
+ import { INPUT_DIR, ZEN_DIR, ZEN_DIST_DIR, ZEN_SRC_DIR } from '../paths';
7
+ import { extractMetadataByAI } from '../process/extractMetadataByAI';
8
+ import { renderTemplates } from '../process/template';
9
+ import { calculateFileHash } from '../scan/files';
10
+ import { BuildOptions } from '../types';
11
+ import { updateFrontmatter } from '../utils/frontmatter';
12
+
13
+ /**
14
+ * 验证构建配置
15
+ */
16
+ async function validateConfig(options: BuildOptions): Promise<void> {
17
+ const { verbose = false } = options;
18
+
19
+ if (verbose) {
20
+ console.log(`🚀 Starting ZEN build...`);
21
+ console.log(`🔗 Base URL: ${options.baseUrl || '(not set)'}`);
22
+ if (options.langs && options.langs.length > 0) {
23
+ console.log(`🌐 Target languages: ${options.langs.join(', ')}`);
24
+ }
25
+ console.log(`🔍 Verbose mode enabled`);
26
+ }
27
+
28
+ MetaData.options = options;
29
+ }
30
+
31
+ /**
32
+ * 扫描源文件
33
+ */
34
+ async function scanSourceFiles(): Promise<void> {
35
+ console.log(`🔍 Scanning source directory...`);
36
+ const markdownFiles = await findMarkdownEntries(INPUT_DIR);
37
+ const hashes = new Set<string>();
38
+
39
+ for (const relativePath of markdownFiles) {
40
+ const fullPath = path.join(INPUT_DIR, relativePath);
41
+
42
+ try {
43
+ // 检查文件是否存在
44
+ await fs.access(fullPath);
45
+
46
+ const hash = await calculateFileHash(fullPath);
47
+
48
+ hashes.add(hash);
49
+
50
+ const metaWithSameHash = MetaData.files.find(f => f.hash === hash);
51
+ if (metaWithSameHash) {
52
+ metaWithSameHash.path = relativePath;
53
+ } else {
54
+ // 如果没有相同哈希的元数据,则添加一个新的占位符
55
+ MetaData.files.push({
56
+ hash,
57
+ path: relativePath,
58
+ });
59
+ }
60
+ } catch (error) {
61
+ console.warn(`⚠️ File not found or inaccessible: ${fullPath}`, error);
62
+ }
63
+ }
64
+ // 移除不再存在的文件元数据
65
+ MetaData.files = MetaData.files.filter(f => hashes.has(f.hash));
66
+
67
+ console.log(`✅ Found ${MetaData.files.length} Markdown files`);
68
+
69
+ if (MetaData.files.length === 0) {
70
+ console.warn(`⚠️ No Markdown files found in ${INPUT_DIR}`);
71
+ }
72
+ }
73
+
74
+ /**
75
+ * 存储母语文件到 .zen/src
76
+ */
77
+ async function storeNativeFiles(): Promise<void> {
78
+ const {
79
+ options: { verbose },
80
+ files,
81
+ } = MetaData;
82
+ for (const file of MetaData.files) {
83
+ try {
84
+ if (!file.hash) throw new Error(`Missing hash`);
85
+ if (!file.metadata?.inferred_lang) throw new Error(`Missing inferred language`);
86
+ const filePath = path.join(ZEN_SRC_DIR, file.metadata.inferred_lang, file.hash + '.md');
87
+ const originalContent = await fs.readFile(path.join(INPUT_DIR, file.path), 'utf-8');
88
+
89
+ const enhancedContent = updateFrontmatter(originalContent, {
90
+ title: file.metadata.title,
91
+ summary: file.metadata.summary,
92
+ tags: file.metadata.tags,
93
+ inferred_date: file.metadata.inferred_date,
94
+ inferred_lang: file.metadata.inferred_lang,
95
+ });
96
+ await fs.mkdir(path.dirname(filePath), { recursive: true });
97
+
98
+ await fs.writeFile(filePath, enhancedContent, 'utf-8');
99
+ } catch (error) {
100
+ console.warn(`⚠️ Failed to store native file ${file.path}:`, error);
101
+ }
102
+ }
103
+
104
+ if (verbose && files.length > 0) {
105
+ console.log(`💾 Stored ${files.length} native language files to .zen/src`);
106
+ }
107
+ }
108
+
109
+ /**
110
+ * 处理翻译
111
+ */
112
+ async function processTranslations(): Promise<void> {
113
+ const {
114
+ files,
115
+ options: { langs = [], verbose },
116
+ } = MetaData;
117
+
118
+ for (const file of files) {
119
+ if (verbose) console.info(`📄 Processing file for translation: ${file.path}`);
120
+ if (!file.metadata) {
121
+ console.warn(`⚠️ Missing metadata for file: ${file.path}, skipping translation.`);
122
+ continue;
123
+ }
124
+ for (const lang of langs) {
125
+ if (verbose) console.log(`🌐 Translating to ${lang}...`);
126
+ // 存储翻译文件到 .zen/src/{lang}
127
+ const sourcePath = path.join(ZEN_SRC_DIR, file.metadata.inferred_lang, file.hash + '.md'); // 使用已经加强的母语文件路径
128
+ const targetPath = path.join(ZEN_SRC_DIR, lang, file.hash + '.md');
129
+
130
+ try {
131
+ const content = await fs.readFile(sourcePath, 'utf-8');
132
+ if (file.metadata.inferred_lang === lang) {
133
+ if (verbose)
134
+ console.log(`ℹ️ Skipping translation for ${file.path}, already in target language`);
135
+ continue;
136
+ } else {
137
+ // 翻译
138
+ // 先检查是否已经有翻译文件存在
139
+
140
+ const exists = await fs.access(targetPath).then(
141
+ () => true,
142
+ () => false
143
+ );
144
+ if (exists) {
145
+ if (verbose) console.log(`ℹ️ Translation already exists for ${file.path} in ${lang}`);
146
+ continue;
147
+ }
148
+ }
149
+
150
+ const translatedContent = await translateMarkdown(content, lang);
151
+
152
+ await fs.mkdir(path.dirname(targetPath), { recursive: true });
153
+ await fs.writeFile(targetPath, translatedContent, 'utf-8');
154
+
155
+ if (verbose) console.log(`✅ Translated file saved: ${targetPath}`);
156
+ } catch (error) {
157
+ console.error(`❌ Failed to translate to ${lang}:`, error);
158
+ }
159
+ }
160
+ }
161
+ }
162
+
163
+ /**
164
+ * 构建管道(函数组合)
165
+ */
166
+ async function buildPipeline(options: BuildOptions): Promise<void> {
167
+ // 验证配置
168
+ await validateConfig(options);
169
+
170
+ // 清理输出目录
171
+ await fs.rm(ZEN_DIST_DIR, { recursive: true, force: true });
172
+
173
+ // 确保 .zen/.gitignore 文件
174
+ await fs.writeFile(path.join(ZEN_DIR, '.gitignore'), 'dist\n', 'utf-8');
175
+
176
+ // 扫描源文件
177
+ await scanSourceFiles();
178
+
179
+ // 运行 AI 元数据提取
180
+ await extractMetadataByAI();
181
+
182
+ // 存储母语文件
183
+ await storeNativeFiles();
184
+
185
+ // 处理翻译
186
+ await processTranslations();
187
+
188
+ // 渲染模板
189
+ await renderTemplates();
190
+ }
191
+
192
+ /**
193
+ * 主构建函数
194
+ */
195
+ export async function buildSite(options: BuildOptions): Promise<void> {
196
+ const startTime = Date.now();
197
+
198
+ try {
199
+ await loadMetaData();
200
+ await buildPipeline(options);
201
+
202
+ const endTime = Date.now();
203
+ const duration = ((endTime - startTime) / 1000).toFixed(2);
204
+ console.log(`🎉 Build completed in ${duration}s`);
205
+ } catch (error) {
206
+ console.error(`❌ Build failed:`, error);
207
+ throw error;
208
+ } finally {
209
+ await saveMetaData();
210
+ }
211
+ }
package/src/cli.ts CHANGED
@@ -1,65 +1,28 @@
1
1
  #!/usr/bin/env node
2
2
 
3
- import { config } from 'dotenv';
4
3
  import { Cli, Command, Option } from 'clipanion';
5
- import { ZenBuilder } from './builder';
6
- import { ZenConfig } from './types';
4
+ import { config } from 'dotenv';
5
+ import * as fs from 'fs';
7
6
  import * as path from 'path';
8
- import * as fs from 'fs/promises';
9
- import * as fsSync from 'fs';
10
- import * as url from 'url';
7
+ import { buildSite } from './build/pipeline';
11
8
 
12
9
  // 加载 .env 文件中的环境变量
13
10
  config();
14
11
 
15
12
  // 获取版本号 - 从 package.json 读取
16
13
  function getVersion(): string {
17
- try {
18
- const packageJsonPath = path.join(__dirname, '..', 'package.json');
19
- const packageJson = JSON.parse(fsSync.readFileSync(packageJsonPath, 'utf-8'));
20
- return packageJson.version;
21
- } catch {
22
- return '0.1.32';
23
- }
24
- }
25
-
26
- // 基础命令类
27
- abstract class BaseCommand extends Command {
28
- protected async loadConfig(configPath?: string): Promise<ZenConfig> {
29
- if (!configPath) {
30
- return {};
31
- }
32
-
33
- try {
34
- const resolvedPath = path.resolve(configPath);
35
- const configContent = await fs.readFile(resolvedPath, 'utf-8');
36
- return JSON.parse(configContent);
37
- } catch (error) {
38
- this.context.stderr.write(`❌ Failed to load config file: ${error}\n`);
39
- throw error;
40
- }
41
- }
42
-
43
- protected getOutDir(): string {
44
- const currentDir = process.cwd();
45
- return path.join(currentDir, '.zen', 'dist');
46
- }
14
+ const packageJsonPath = path.join(__dirname, '..', 'package.json');
15
+ const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf-8'));
16
+ return packageJson.version;
47
17
  }
48
18
 
49
19
  // Build 命令
50
- class BuildCommand extends BaseCommand {
20
+ class BuildCommand extends Command {
51
21
  static paths = [['build']];
52
22
 
53
23
  template = Option.String('-t,--template');
54
- watch = Option.Boolean('-w,--watch');
55
- serve = Option.Boolean('-s,--serve');
56
- port = Option.String('-p,--port', '3000');
57
- host = Option.String('--host', 'localhost');
58
24
  verbose = Option.Boolean('-v,--verbose');
59
- config = Option.String('-c,--config');
60
25
  baseUrl = Option.String('--base-url');
61
- clean = Option.Boolean('--clean');
62
- ai = Option.Boolean('--ai', { description: 'Enable AI metadata extraction' });
63
26
  lang = Option.Array('--lang', {
64
27
  description: 'Target languages for translation (e.g., en-US, ja-JP)',
65
28
  });
@@ -72,103 +35,18 @@ class BuildCommand extends BaseCommand {
72
35
 
73
36
  Examples:
74
37
  $ zengen build
75
- $ zengen build --watch
76
- $ zengen build --watch --serve
77
- $ zengen build --watch --serve --port 8080
78
- $ zengen build --config zen.config.json
79
- $ zengen build --clean
80
- $ zengen build --ai (requires OPENAI_API_KEY environment variable)
81
38
  $ zengen build --lang en-US --lang ja-JP (translate to English and Japanese)
82
39
  `,
83
40
  });
84
41
 
85
42
  async execute() {
86
43
  try {
87
- // 加载配置文件
88
- const config = await this.loadConfig(this.config);
89
-
90
- // 强制使用当前目录作为 src 目录,输出到 .zen/dist 目录
91
- const currentDir = process.cwd();
92
- const outDir = this.getOutDir();
93
-
94
- // 处理 AI 配置:如果指定了 --ai 参数,启用 AI;否则使用配置中的设置
95
- const aiConfig = {
96
- ...config.ai,
97
- enabled: this.ai ? true : config.ai?.enabled,
98
- };
99
-
100
- // 处理语言配置:命令行参数优先于配置文件
101
- const targetLangs = this.lang && this.lang.length > 0 ? this.lang : config.i18n?.targetLangs;
102
- const i18nConfig =
103
- targetLangs && targetLangs.length > 0
104
- ? {
105
- ...config.i18n,
106
- sourceLang: config.i18n?.sourceLang || 'zh-Hans', // 默认源语言为中文
107
- targetLangs,
108
- }
109
- : undefined;
110
-
111
- // 合并命令行参数和配置
112
- const buildOptions = {
113
- srcDir: currentDir,
114
- outDir: outDir,
44
+ await buildSite({
115
45
  template: this.template ? path.resolve(this.template) : undefined,
116
- watch: this.watch,
117
- serve: this.serve,
118
- port: parseInt(this.port, 10),
119
- host: this.host,
120
46
  verbose: this.verbose,
121
- baseUrl: this.baseUrl || config.baseUrl,
47
+ baseUrl: this.baseUrl,
122
48
  langs: this.lang,
123
- };
124
-
125
- // 创建最终的配置,包含 AI 和 i18n 设置
126
- const finalConfig = {
127
- ...config,
128
- ai: aiConfig.enabled ? aiConfig : undefined,
129
- i18n: i18nConfig,
130
- };
131
-
132
- const builder = new ZenBuilder(finalConfig);
133
-
134
- // 验证配置
135
- const errors = builder.validateConfig(finalConfig);
136
- if (errors.length > 0) {
137
- this.context.stderr.write('❌ Configuration errors:\n');
138
- errors.forEach(error => this.context.stderr.write(` - ${error}\n`));
139
- return 1;
140
- }
141
-
142
- // 警告 --serve 选项需要 --watch 选项
143
- if (this.serve && !this.watch) {
144
- this.context.stdout.write(
145
- '⚠️ Warning: --serve option requires --watch option, ignoring --serve\n'
146
- );
147
- buildOptions.serve = false;
148
- }
149
-
150
- // 清理输出目录
151
- if (this.clean) {
152
- await builder.clean(buildOptions.outDir);
153
- }
154
-
155
- // 构建或监听
156
- if (this.watch) {
157
- await builder.watch(buildOptions);
158
- } else {
159
- // 如果指定了语言参数,使用多语言构建
160
- if (this.lang && this.lang.length > 0) {
161
- const multiLangOptions = {
162
- ...buildOptions,
163
- langs: this.lang,
164
- useMetaData: true,
165
- filterOrphans: true,
166
- };
167
- await builder.buildMultiLang(multiLangOptions);
168
- } else {
169
- await builder.build(buildOptions);
170
- }
171
- }
49
+ });
172
50
 
173
51
  return 0;
174
52
  } catch (error) {
@@ -0,0 +1,37 @@
1
+ import { exec } from 'child_process';
2
+ import { promisify } from 'util';
3
+
4
+ const execAsync = promisify(exec);
5
+
6
+ /**
7
+ * 使用git命令查找所有Markdown文件
8
+ * 使用git ls-files --others --cached --exclude-standard获取所有文件
9
+ * 然后过滤掉.zen目录和只保留.md文件
10
+ *
11
+ * @param dirPath 要扫描的目录路径
12
+ * @returns Promise<string[]> 返回Markdown文件的相对路径数组
13
+ */
14
+ export const findMarkdownEntries = async (dirPath: string): Promise<string[]> => {
15
+ try {
16
+ // 使用git命令获取所有文件(包括已跟踪和未跟踪的文件)
17
+ // 在指定的目录下执行git命令
18
+ const { stdout } = await execAsync('git ls-files --others --cached --exclude-standard', {
19
+ cwd: dirPath,
20
+ });
21
+
22
+ // 按行分割并过滤
23
+ const files = stdout
24
+ .split('\n')
25
+ .filter(line => line.trim() !== '') // 移除空行
26
+ .filter(file => !file.startsWith('.zen')) // 过滤掉.zen目录下的文件
27
+ .filter(file => file.endsWith('.md')); // 只保留.md文件
28
+
29
+ return files;
30
+ } catch (error) {
31
+ console.error('Error finding markdown entries:', error);
32
+
33
+ // 如果git命令失败,返回空数组
34
+ // 这可以处理没有git仓库或git不可用的情况
35
+ return [];
36
+ }
37
+ };
package/src/index.ts CHANGED
@@ -1,40 +1 @@
1
- export { ZenBuilder } from './builder';
2
- export { MarkdownConverter } from './markdown';
3
- export { TemplateEngine } from './template';
4
- export { NavigationGenerator } from './navigation';
5
- export type {
6
- BuildOptions,
7
- FileInfo,
8
- NavigationItem,
9
- TemplateData,
10
- MarkdownProcessor,
11
- ZenConfig,
12
- } from './types';
13
-
14
- /**
15
- * ZEN 文档构建工具
16
- *
17
- * 一个极简主义的 Markdown 文档站点生成器
18
- *
19
- * @example
20
- * ```typescript
21
- * import { ZenBuilder } from 'zengen';
22
- *
23
- * const builder = new ZenBuilder();
24
- * await builder.build({
25
- * srcDir: './docs',
26
- * outDir: './dist'
27
- * });
28
- * ```
29
- */
30
- import { ZenBuilder } from './builder';
31
- import { MarkdownConverter } from './markdown';
32
- import { TemplateEngine } from './template';
33
- import { NavigationGenerator } from './navigation';
34
-
35
- export default {
36
- Builder: ZenBuilder,
37
- MarkdownConverter: MarkdownConverter,
38
- TemplateEngine: TemplateEngine,
39
- NavigationGenerator: NavigationGenerator,
40
- };
1
+ import './cli';
@@ -0,0 +1,44 @@
1
+ import * as fs from 'fs/promises';
2
+ import * as path from 'path';
3
+ import { ZEN_META_PATH } from './paths';
4
+ import { MetaDataStore } from './types';
5
+
6
+ /**
7
+ * 全局 MetaDataStore 单例
8
+ */
9
+ export const MetaData: MetaDataStore = {
10
+ // 稍后覆盖
11
+ version: '1.0.0',
12
+ options: {},
13
+ files: [],
14
+ };
15
+
16
+ /**
17
+ * 从文件中读取数据,覆盖 store,但是要保持它仍然是同一个对象
18
+ */
19
+ export async function loadMetaData(): Promise<void> {
20
+ try {
21
+ await fs.access(ZEN_META_PATH);
22
+ const content = await fs.readFile(ZEN_META_PATH, 'utf-8');
23
+ const newData = JSON.parse(content);
24
+
25
+ // 使用 Object.assign 保持同一个对象引用
26
+ Object.assign(MetaData, newData);
27
+ } catch (error) {
28
+ // 如果文件不存在,初始化默认值
29
+ MetaData.version = '1.0.0';
30
+ MetaData.files = [];
31
+ }
32
+ }
33
+
34
+ /**
35
+ * 将 MetaData 写入 store
36
+ */
37
+ export async function saveMetaData(): Promise<void> {
38
+ // 确保 .zen 目录存在
39
+ const zenDir = path.dirname(ZEN_META_PATH);
40
+ await fs.mkdir(zenDir, { recursive: true });
41
+
42
+ // 保存文件
43
+ await fs.writeFile(ZEN_META_PATH, JSON.stringify(MetaData, null, 2), 'utf-8');
44
+ }
package/src/paths.ts ADDED
@@ -0,0 +1,7 @@
1
+ import { join } from 'path';
2
+
3
+ export const INPUT_DIR = process.cwd();
4
+ export const ZEN_DIR = join(process.cwd(), '.zen');
5
+ export const ZEN_DIST_DIR = join(ZEN_DIR, 'dist');
6
+ export const ZEN_SRC_DIR = join(ZEN_DIR, 'src');
7
+ export const ZEN_META_PATH = join(ZEN_DIR, 'meta.json');