zengen 0.1.36 → 0.2.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/.github/workflows/pages.yml +1 -1
- package/.zen/meta.json +112 -32
- package/.zen/src/en-US/01d04f7c17b4a541ead9d759d877b30b403e15b849182a49eb1f62bd29ecd18c.md +120 -0
- package/.zen/src/en-US/1b798c44a4f353e47296ca83d5905e37e6aba3e90bbd9bc3b3d34fc12059a2ca.md +75 -0
- package/.zen/src/en-US/1e96be58d76c60056b708eb5bd8b8b81d7b5845d9cfe0b879d85068a5f11df3a.md +189 -0
- package/.zen/src/en-US/5ec990146b35e00de2630559126ee07f7cdcddeb23b0e8cab3d85b4181353e26.md +53 -0
- package/.zen/src/en-US/6124ea88edec5bde737b26b21f71ecfeffe4e73151784856edf813ee231a4baa.md +11 -0
- package/.zen/src/en-US/80ae9bed74fc6348a7c1fe9f33e86b65f5d919169721f77bcf0e1bc29fbdb4f9.md +61 -0
- package/.zen/src/en-US/f0c2799126931ccd113a0c45b1e623870b0d4f4f400becf6dd877da8f1011517.md +40 -0
- package/.zen/src/en-US/fdfca9b960d0eaa8b2b96fe988ead7481d2c0b16f66ebc94fb477139b4178cdc.md +65 -0
- package/.zen/src/zh-Hans/01d04f7c17b4a541ead9d759d877b30b403e15b849182a49eb1f62bd29ecd18c.md +120 -0
- package/.zen/src/zh-Hans/1b798c44a4f353e47296ca83d5905e37e6aba3e90bbd9bc3b3d34fc12059a2ca.md +77 -0
- package/.zen/src/zh-Hans/1e96be58d76c60056b708eb5bd8b8b81d7b5845d9cfe0b879d85068a5f11df3a.md +189 -0
- package/.zen/src/zh-Hans/5ec990146b35e00de2630559126ee07f7cdcddeb23b0e8cab3d85b4181353e26.md +55 -0
- package/.zen/src/zh-Hans/6124ea88edec5bde737b26b21f71ecfeffe4e73151784856edf813ee231a4baa.md +11 -0
- package/.zen/src/zh-Hans/80ae9bed74fc6348a7c1fe9f33e86b65f5d919169721f77bcf0e1bc29fbdb4f9.md +63 -0
- package/.zen/src/zh-Hans/f0c2799126931ccd113a0c45b1e623870b0d4f4f400becf6dd877da8f1011517.md +40 -0
- package/.zen/src/zh-Hans/fdfca9b960d0eaa8b2b96fe988ead7481d2c0b16f66ebc94fb477139b4178cdc.md +65 -0
- package/assets/templates/default/layout.html +274 -0
- package/dist/ai/extractMetadataFromMarkdown.d.ts +8 -0
- package/dist/ai/extractMetadataFromMarkdown.d.ts.map +1 -0
- package/dist/ai/extractMetadataFromMarkdown.js +88 -0
- package/dist/ai/extractMetadataFromMarkdown.js.map +1 -0
- package/dist/ai/translateMarkdown.d.ts +8 -0
- package/dist/ai/translateMarkdown.d.ts.map +1 -0
- package/dist/ai/translateMarkdown.js +29 -0
- package/dist/ai/translateMarkdown.js.map +1 -0
- package/dist/build/pipeline.d.ts +6 -0
- package/dist/build/pipeline.d.ts.map +1 -0
- package/dist/build/pipeline.js +219 -0
- package/dist/build/pipeline.js.map +1 -0
- package/dist/cli.js +10 -118
- package/dist/cli.js.map +1 -1
- package/dist/findEntries.d.ts +10 -0
- package/dist/findEntries.d.ts.map +1 -0
- package/dist/findEntries.js +38 -0
- package/dist/findEntries.js.map +1 -0
- package/dist/index.d.ts +1 -32
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +1 -35
- package/dist/index.js.map +1 -1
- package/dist/metadata.d.ts +14 -0
- package/dist/metadata.d.ts.map +1 -0
- package/dist/metadata.js +78 -0
- package/dist/metadata.js.map +1 -0
- package/dist/paths.d.ts +6 -0
- package/dist/paths.d.ts.map +1 -0
- package/dist/paths.js +10 -0
- package/dist/paths.js.map +1 -0
- package/dist/process/extractMetadataByAI.d.ts +5 -0
- package/dist/process/extractMetadataByAI.d.ts.map +1 -0
- package/dist/process/extractMetadataByAI.js +31 -0
- package/dist/process/extractMetadataByAI.js.map +1 -0
- package/dist/process/template.d.ts +5 -0
- package/dist/process/template.d.ts.map +1 -0
- package/dist/process/template.js +188 -0
- package/dist/process/template.js.map +1 -0
- package/dist/scan/files.d.ts +7 -0
- package/dist/scan/files.d.ts.map +1 -0
- package/dist/scan/files.js +54 -0
- package/dist/scan/files.js.map +1 -0
- package/dist/services/openai.d.ts +41 -0
- package/dist/services/openai.d.ts.map +1 -0
- package/dist/services/openai.js +54 -0
- package/dist/services/openai.js.map +1 -0
- package/dist/types.d.ts +16 -67
- package/dist/types.d.ts.map +1 -1
- package/dist/utils/convertMarkdownToHtml.d.ts +7 -0
- package/dist/utils/convertMarkdownToHtml.d.ts.map +1 -0
- package/dist/utils/convertMarkdownToHtml.js +39 -0
- package/dist/utils/convertMarkdownToHtml.js.map +1 -0
- package/dist/utils/frontmatter.d.ts +6 -0
- package/dist/utils/frontmatter.d.ts.map +1 -0
- package/dist/utils/frontmatter.js +22 -0
- package/dist/utils/frontmatter.js.map +1 -0
- package/docs/deployment/github-pages.md +1 -2
- package/docs/guides/best-practices.md +4 -4
- package/docs/guides/config.md +0 -5
- package/package.json +4 -2
- package/src/ai/extractMetadataFromMarkdown.ts +95 -0
- package/src/ai/translateMarkdown.ts +29 -0
- package/src/build/pipeline.ts +217 -0
- package/src/cli.ts +10 -132
- package/src/findEntries.ts +37 -0
- package/src/index.ts +1 -40
- package/src/metadata.ts +44 -0
- package/src/paths.ts +7 -0
- package/src/process/extractMetadataByAI.ts +31 -0
- package/src/process/template.ts +201 -0
- package/src/scan/files.ts +17 -0
- package/src/services/openai.ts +92 -0
- package/src/types.ts +18 -72
- package/src/utils/convertMarkdownToHtml.ts +32 -0
- package/src/utils/frontmatter.ts +18 -0
- package/.zen/translations.json +0 -51
- package/dist/ai-client.d.ts +0 -34
- package/dist/ai-client.d.ts.map +0 -1
- package/dist/ai-client.js +0 -180
- package/dist/ai-client.js.map +0 -1
- package/dist/ai-processor.d.ts +0 -51
- package/dist/ai-processor.d.ts.map +0 -1
- package/dist/ai-processor.js +0 -215
- package/dist/ai-processor.js.map +0 -1
- package/dist/ai-service.d.ts +0 -79
- package/dist/ai-service.d.ts.map +0 -1
- package/dist/ai-service.js +0 -257
- package/dist/ai-service.js.map +0 -1
- package/dist/builder.d.ts +0 -70
- package/dist/builder.d.ts.map +0 -1
- package/dist/builder.js +0 -854
- package/dist/builder.js.map +0 -1
- package/dist/gitignore.d.ts +0 -41
- package/dist/gitignore.d.ts.map +0 -1
- package/dist/gitignore.js +0 -202
- package/dist/gitignore.js.map +0 -1
- package/dist/gitignore.test.d.ts +0 -2
- package/dist/gitignore.test.d.ts.map +0 -1
- package/dist/gitignore.test.js +0 -309
- package/dist/gitignore.test.js.map +0 -1
- package/dist/markdown.d.ts +0 -35
- package/dist/markdown.d.ts.map +0 -1
- package/dist/markdown.js +0 -221
- package/dist/markdown.js.map +0 -1
- package/dist/navigation.d.ts +0 -46
- package/dist/navigation.d.ts.map +0 -1
- package/dist/navigation.js +0 -196
- package/dist/navigation.js.map +0 -1
- package/dist/scanner.d.ts +0 -26
- package/dist/scanner.d.ts.map +0 -1
- package/dist/scanner.js +0 -190
- package/dist/scanner.js.map +0 -1
- package/dist/template.d.ts +0 -33
- package/dist/template.d.ts.map +0 -1
- package/dist/template.js +0 -434
- package/dist/template.js.map +0 -1
- package/dist/translation-service.d.ts +0 -72
- package/dist/translation-service.d.ts.map +0 -1
- package/dist/translation-service.js +0 -291
- package/dist/translation-service.js.map +0 -1
- package/src/ai-client.ts +0 -227
- package/src/ai-processor.ts +0 -243
- package/src/ai-service.ts +0 -281
- package/src/builder.ts +0 -991
- package/src/gitignore.test.ts +0 -318
- package/src/gitignore.ts +0 -193
- package/src/markdown.ts +0 -212
- package/src/navigation.ts +0 -237
- package/src/scanner.ts +0 -180
- package/src/template.ts +0 -425
- package/src/translation-service.ts +0 -350
package/src/ai-processor.ts
DELETED
|
@@ -1,243 +0,0 @@
|
|
|
1
|
-
import { MarkdownProcessor, FileInfo, ZenConfig } from './types';
|
|
2
|
-
import { AIService } from './ai-service';
|
|
3
|
-
import { AIClient } from './ai-client';
|
|
4
|
-
|
|
5
|
-
/**
|
|
6
|
-
* AI 处理器 - 集成到 Markdown 处理流程中
|
|
7
|
-
*/
|
|
8
|
-
export class AIProcessor implements MarkdownProcessor {
|
|
9
|
-
private aiService: AIService;
|
|
10
|
-
private aiClient: AIClient;
|
|
11
|
-
private enabled: boolean;
|
|
12
|
-
|
|
13
|
-
constructor(config: ZenConfig = {}) {
|
|
14
|
-
// 从配置和环境变量初始化 AI 服务
|
|
15
|
-
const aiConfig = {
|
|
16
|
-
enabled: config.ai?.enabled ?? true,
|
|
17
|
-
model: config.ai?.model,
|
|
18
|
-
temperature: config.ai?.temperature,
|
|
19
|
-
maxTokens: config.ai?.maxTokens,
|
|
20
|
-
};
|
|
21
|
-
|
|
22
|
-
this.aiService = new AIService(aiConfig);
|
|
23
|
-
this.aiClient = new AIClient(this.aiService);
|
|
24
|
-
this.enabled = this.aiService.isEnabled();
|
|
25
|
-
|
|
26
|
-
if (this.enabled) {
|
|
27
|
-
console.log('🤖 AI processor initialized and enabled');
|
|
28
|
-
console.log(` Model: ${this.aiService.getConfig().model}`);
|
|
29
|
-
console.log(` Base URL: ${this.aiService.getConfig().baseUrl}`);
|
|
30
|
-
} else {
|
|
31
|
-
console.log('⚠️ AI processor initialized but disabled (no API key or explicitly disabled)');
|
|
32
|
-
}
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
/**
|
|
36
|
-
* 在解析前处理 - 这里我们提取 AI metadata
|
|
37
|
-
*/
|
|
38
|
-
async beforeParse(content: string, fileInfo: FileInfo): Promise<string> {
|
|
39
|
-
if (!this.enabled) {
|
|
40
|
-
return content;
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
if (!fileInfo.hash) {
|
|
44
|
-
console.warn(`⚠️ Skipping AI processing for ${fileInfo.path}: file hash is missing`);
|
|
45
|
-
return content;
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
try {
|
|
49
|
-
// 检查缓存
|
|
50
|
-
const cachedMetadata = await this.aiService.getCachedMetadata(fileInfo.hash, fileInfo.path);
|
|
51
|
-
if (cachedMetadata) {
|
|
52
|
-
fileInfo.aiMetadata = cachedMetadata;
|
|
53
|
-
this.logMetadata(fileInfo.path, cachedMetadata);
|
|
54
|
-
return content;
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
// 调用 AI 提取 metadata
|
|
58
|
-
const metadata = await this.aiClient.extractMetadata(content, fileInfo.path);
|
|
59
|
-
if (metadata) {
|
|
60
|
-
fileInfo.aiMetadata = metadata;
|
|
61
|
-
|
|
62
|
-
// 缓存结果
|
|
63
|
-
await this.aiService.cacheMetadata(fileInfo.hash, fileInfo.path, metadata);
|
|
64
|
-
|
|
65
|
-
this.logMetadata(fileInfo.path, metadata);
|
|
66
|
-
}
|
|
67
|
-
} catch (error) {
|
|
68
|
-
console.error(`❌ AI processor failed for ${fileInfo.path}:`, error);
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
return content;
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
/**
|
|
75
|
-
* 在解析后处理 - 这里可以添加 AI 增强的 HTML 内容
|
|
76
|
-
*/
|
|
77
|
-
async afterParse(html: string, fileInfo: FileInfo): Promise<string> {
|
|
78
|
-
if (!this.enabled || !fileInfo.aiMetadata) {
|
|
79
|
-
return html;
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
try {
|
|
83
|
-
// 可以在这里添加 AI metadata 到 HTML 中
|
|
84
|
-
// 例如:添加 meta 标签或结构化数据
|
|
85
|
-
return this.enhanceHtmlWithMetadata(html, fileInfo);
|
|
86
|
-
} catch (error) {
|
|
87
|
-
console.error(`❌ Failed to enhance HTML with AI metadata for ${fileInfo.path}:`, error);
|
|
88
|
-
return html;
|
|
89
|
-
}
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
/**
|
|
93
|
-
* 将 AI metadata 添加到 HTML 中
|
|
94
|
-
*/
|
|
95
|
-
private enhanceHtmlWithMetadata(html: string, fileInfo: FileInfo): string {
|
|
96
|
-
const metadata = fileInfo.aiMetadata;
|
|
97
|
-
if (!metadata) {
|
|
98
|
-
return html;
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
// 在 head 部分添加 meta 标签
|
|
102
|
-
const metaTags = `
|
|
103
|
-
<!-- AI Generated Metadata -->
|
|
104
|
-
<meta name="ai-title" content="${this.escapeHtml(metadata.title)}">
|
|
105
|
-
<meta name="ai-summary" content="${this.escapeHtml(metadata.summary)}">
|
|
106
|
-
<meta name="ai-tags" content="${this.escapeHtml(metadata.tags.join(', '))}">
|
|
107
|
-
${metadata.inferred_date ? `<meta name="ai-inferred-date" content="${metadata.inferred_date}">` : ''}
|
|
108
|
-
<meta name="ai-language" content="${metadata.inferred_lang}">
|
|
109
|
-
`;
|
|
110
|
-
|
|
111
|
-
// 在 body 开始处添加结构化数据
|
|
112
|
-
const structuredData = `
|
|
113
|
-
<!-- AI Structured Data -->
|
|
114
|
-
<script type="application/ld+json">
|
|
115
|
-
{
|
|
116
|
-
"@context": "https://schema.org",
|
|
117
|
-
"@type": "Article",
|
|
118
|
-
"headline": "${this.escapeJson(metadata.title)}",
|
|
119
|
-
"description": "${this.escapeJson(metadata.summary)}",
|
|
120
|
-
"keywords": "${this.escapeJson(metadata.tags.join(', '))}",
|
|
121
|
-
"inLanguage": "${metadata.inferred_lang}"
|
|
122
|
-
}
|
|
123
|
-
</script>
|
|
124
|
-
`;
|
|
125
|
-
|
|
126
|
-
// 插入 meta 标签到 head
|
|
127
|
-
const headEndIndex = html.indexOf('</head>');
|
|
128
|
-
if (headEndIndex !== -1) {
|
|
129
|
-
html = html.slice(0, headEndIndex) + metaTags + html.slice(headEndIndex);
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
// 插入结构化数据到 body 开始处
|
|
133
|
-
const bodyStartIndex = html.indexOf('<body');
|
|
134
|
-
if (bodyStartIndex !== -1) {
|
|
135
|
-
const bodyTagEndIndex = html.indexOf('>', bodyStartIndex) + 1;
|
|
136
|
-
html = html.slice(0, bodyTagEndIndex) + structuredData + html.slice(bodyTagEndIndex);
|
|
137
|
-
}
|
|
138
|
-
|
|
139
|
-
return html;
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
/**
|
|
143
|
-
* 转义 HTML 特殊字符
|
|
144
|
-
*/
|
|
145
|
-
private escapeHtml(text: string): string {
|
|
146
|
-
return text
|
|
147
|
-
.replace(/&/g, '&')
|
|
148
|
-
.replace(/</g, '<')
|
|
149
|
-
.replace(/>/g, '>')
|
|
150
|
-
.replace(/"/g, '"')
|
|
151
|
-
.replace(/'/g, ''');
|
|
152
|
-
}
|
|
153
|
-
|
|
154
|
-
/**
|
|
155
|
-
* 转义 JSON 字符串
|
|
156
|
-
*/
|
|
157
|
-
private escapeJson(text: string): string {
|
|
158
|
-
return text
|
|
159
|
-
.replace(/\\/g, '\\\\')
|
|
160
|
-
.replace(/"/g, '\\"')
|
|
161
|
-
.replace(/\n/g, '\\n')
|
|
162
|
-
.replace(/\r/g, '\\r')
|
|
163
|
-
.replace(/\t/g, '\\t');
|
|
164
|
-
}
|
|
165
|
-
|
|
166
|
-
/**
|
|
167
|
-
* 记录 metadata 信息
|
|
168
|
-
*/
|
|
169
|
-
private logMetadata(filePath: string, metadata: any): void {
|
|
170
|
-
console.log(`📊 AI metadata for ${filePath}:`);
|
|
171
|
-
console.log(` Title: ${metadata.title}`);
|
|
172
|
-
console.log(` Summary: ${metadata.summary.substring(0, 50)}...`);
|
|
173
|
-
console.log(` Tags: ${metadata.tags.join(', ')}`);
|
|
174
|
-
if (metadata.inferred_date) {
|
|
175
|
-
console.log(` Inferred Date: ${metadata.inferred_date}`);
|
|
176
|
-
}
|
|
177
|
-
console.log(` Language: ${metadata.inferred_lang}`);
|
|
178
|
-
}
|
|
179
|
-
|
|
180
|
-
/**
|
|
181
|
-
* 批量处理文件
|
|
182
|
-
*/
|
|
183
|
-
async processBatch(files: FileInfo[]): Promise<void> {
|
|
184
|
-
if (!this.enabled) {
|
|
185
|
-
return;
|
|
186
|
-
}
|
|
187
|
-
|
|
188
|
-
console.log(`🤖 Processing ${files.length} files with AI...`);
|
|
189
|
-
|
|
190
|
-
const filesToProcess = files.filter(file => file.hash && !file.aiMetadata);
|
|
191
|
-
if (filesToProcess.length === 0) {
|
|
192
|
-
console.log('📚 All files already have AI metadata or no files to process');
|
|
193
|
-
return;
|
|
194
|
-
}
|
|
195
|
-
|
|
196
|
-
// 准备数据
|
|
197
|
-
const fileData = filesToProcess.map(file => ({
|
|
198
|
-
content: file.content,
|
|
199
|
-
path: file.path,
|
|
200
|
-
hash: file.hash!,
|
|
201
|
-
}));
|
|
202
|
-
|
|
203
|
-
// 批量处理
|
|
204
|
-
const results = await this.aiClient.processFiles(fileData);
|
|
205
|
-
|
|
206
|
-
// 更新文件信息
|
|
207
|
-
for (const file of filesToProcess) {
|
|
208
|
-
const metadata = results.get(file.path);
|
|
209
|
-
if (metadata) {
|
|
210
|
-
file.aiMetadata = metadata;
|
|
211
|
-
}
|
|
212
|
-
}
|
|
213
|
-
|
|
214
|
-
console.log(`✅ AI processing completed for ${results.size} files`);
|
|
215
|
-
}
|
|
216
|
-
|
|
217
|
-
/**
|
|
218
|
-
* 清理缓存
|
|
219
|
-
*/
|
|
220
|
-
async cleanupCache(maxAgeDays: number = 30): Promise<void> {
|
|
221
|
-
await this.aiService.cleanupCache(maxAgeDays);
|
|
222
|
-
}
|
|
223
|
-
|
|
224
|
-
/**
|
|
225
|
-
* 检查是否启用
|
|
226
|
-
*/
|
|
227
|
-
isEnabled(): boolean {
|
|
228
|
-
return this.enabled;
|
|
229
|
-
}
|
|
230
|
-
|
|
231
|
-
/**
|
|
232
|
-
* 获取配置信息
|
|
233
|
-
*/
|
|
234
|
-
getConfigInfo(): string {
|
|
235
|
-
const config = this.aiService.getConfig();
|
|
236
|
-
return `AI Processor Status: ${this.enabled ? 'Enabled' : 'Disabled'}
|
|
237
|
-
API Key: ${config.apiKey ? 'Set' : 'Not set'}
|
|
238
|
-
Base URL: ${config.baseUrl}
|
|
239
|
-
Model: ${config.model}
|
|
240
|
-
Temperature: ${config.temperature}
|
|
241
|
-
Max Tokens: ${config.maxTokens}`;
|
|
242
|
-
}
|
|
243
|
-
}
|
package/src/ai-service.ts
DELETED
|
@@ -1,281 +0,0 @@
|
|
|
1
|
-
import { AIMetadata } from './types';
|
|
2
|
-
import * as fs from 'fs/promises';
|
|
3
|
-
import * as path from 'path';
|
|
4
|
-
import * as crypto from 'crypto';
|
|
5
|
-
|
|
6
|
-
/**
|
|
7
|
-
* AI 服务配置
|
|
8
|
-
*/
|
|
9
|
-
export interface AIConfig {
|
|
10
|
-
enabled: boolean;
|
|
11
|
-
apiKey: string;
|
|
12
|
-
baseUrl: string;
|
|
13
|
-
model: string;
|
|
14
|
-
temperature: number;
|
|
15
|
-
maxTokens: number;
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
/**
|
|
19
|
-
* 单个文件的元数据缓存项
|
|
20
|
-
*/
|
|
21
|
-
export interface FileMetaData {
|
|
22
|
-
hash: string;
|
|
23
|
-
path: string;
|
|
24
|
-
metadata: AIMetadata;
|
|
25
|
-
lastUpdated: string;
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
/**
|
|
29
|
-
* .zen/meta.json 文件结构
|
|
30
|
-
*/
|
|
31
|
-
export interface MetaDataStore {
|
|
32
|
-
version: string;
|
|
33
|
-
timestamp: string;
|
|
34
|
-
files: FileMetaData[];
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
/**
|
|
38
|
-
* AI 服务类
|
|
39
|
-
*/
|
|
40
|
-
export class AIService {
|
|
41
|
-
private config: AIConfig;
|
|
42
|
-
private metaDataPath: string;
|
|
43
|
-
|
|
44
|
-
constructor(config: Partial<AIConfig> = {}) {
|
|
45
|
-
// 从环境变量读取配置
|
|
46
|
-
const apiKey = process.env.OPENAI_API_KEY || '';
|
|
47
|
-
const baseUrl = process.env.OPENAI_BASE_URL || 'https://api.openai.com/v1';
|
|
48
|
-
|
|
49
|
-
// 配置优先级:构造函数参数 > 环境变量 > 默认值
|
|
50
|
-
const model = config.model || process.env.OPENAI_MODEL || 'gpt-3.5-turbo';
|
|
51
|
-
|
|
52
|
-
this.config = {
|
|
53
|
-
enabled: config.enabled ?? apiKey !== '',
|
|
54
|
-
apiKey,
|
|
55
|
-
baseUrl,
|
|
56
|
-
model,
|
|
57
|
-
temperature: 0, // 总是设置为 0,提取内容不需要随机性
|
|
58
|
-
maxTokens: config.maxTokens || 500,
|
|
59
|
-
};
|
|
60
|
-
|
|
61
|
-
this.metaDataPath = path.join(process.cwd(), '.zen', 'meta.json');
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
/**
|
|
65
|
-
* 获取配置
|
|
66
|
-
*/
|
|
67
|
-
getConfig(): AIConfig {
|
|
68
|
-
return this.config;
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
/**
|
|
72
|
-
* 检查是否启用 AI 功能
|
|
73
|
-
*/
|
|
74
|
-
isEnabled(): boolean {
|
|
75
|
-
const enabled = this.config.enabled && this.config.apiKey !== '';
|
|
76
|
-
if (!enabled && this.config.enabled) {
|
|
77
|
-
console.warn(
|
|
78
|
-
'⚠️ AI is enabled but API key is missing. Please set OPENAI_API_KEY environment variable.'
|
|
79
|
-
);
|
|
80
|
-
}
|
|
81
|
-
return enabled;
|
|
82
|
-
}
|
|
83
|
-
|
|
84
|
-
/**
|
|
85
|
-
* 加载 .zen/meta.json 文件
|
|
86
|
-
*/
|
|
87
|
-
async loadMetaData(): Promise<MetaDataStore> {
|
|
88
|
-
try {
|
|
89
|
-
await fs.access(this.metaDataPath);
|
|
90
|
-
const content = await fs.readFile(this.metaDataPath, 'utf-8');
|
|
91
|
-
return JSON.parse(content);
|
|
92
|
-
} catch (error) {
|
|
93
|
-
// 如果文件不存在,返回空结构
|
|
94
|
-
return {
|
|
95
|
-
version: '1.0.0',
|
|
96
|
-
timestamp: new Date().toISOString(),
|
|
97
|
-
files: [],
|
|
98
|
-
};
|
|
99
|
-
}
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
/**
|
|
103
|
-
* 保存 .zen/meta.json 文件
|
|
104
|
-
*/
|
|
105
|
-
async saveMetaData(metaData: MetaDataStore): Promise<void> {
|
|
106
|
-
// 确保 .zen 目录存在
|
|
107
|
-
const zenDir = path.dirname(this.metaDataPath);
|
|
108
|
-
await fs.mkdir(zenDir, { recursive: true });
|
|
109
|
-
|
|
110
|
-
// 更新时间戳
|
|
111
|
-
metaData.timestamp = new Date().toISOString();
|
|
112
|
-
|
|
113
|
-
// 保存文件
|
|
114
|
-
await fs.writeFile(this.metaDataPath, JSON.stringify(metaData, null, 2), 'utf-8');
|
|
115
|
-
}
|
|
116
|
-
|
|
117
|
-
/**
|
|
118
|
-
* 根据文件 hash 获取缓存的 metadata
|
|
119
|
-
*/
|
|
120
|
-
async getCachedMetadata(fileHash: string, filePath: string): Promise<AIMetadata | null> {
|
|
121
|
-
if (!this.isEnabled()) {
|
|
122
|
-
return null;
|
|
123
|
-
}
|
|
124
|
-
|
|
125
|
-
try {
|
|
126
|
-
const metaData = await this.loadMetaData();
|
|
127
|
-
const cachedFile = metaData.files.find(f => f.hash === fileHash);
|
|
128
|
-
|
|
129
|
-
if (cachedFile) {
|
|
130
|
-
if (cachedFile.path === filePath) {
|
|
131
|
-
// 完全匹配:hash 和 path 都相同
|
|
132
|
-
console.log(`📚 Using cached AI metadata for: ${filePath}`);
|
|
133
|
-
return cachedFile.metadata;
|
|
134
|
-
} else {
|
|
135
|
-
// 文件移动情况:hash 相同但 path 不同
|
|
136
|
-
// 更新缓存中的 path 为最新路径
|
|
137
|
-
console.log(`🔄 File moved detected: ${cachedFile.path} -> ${filePath}`);
|
|
138
|
-
await this.cacheMetadata(fileHash, filePath, cachedFile.metadata);
|
|
139
|
-
return cachedFile.metadata;
|
|
140
|
-
}
|
|
141
|
-
}
|
|
142
|
-
} catch (error) {
|
|
143
|
-
console.warn(`⚠️ Failed to load cached metadata:`, error);
|
|
144
|
-
}
|
|
145
|
-
|
|
146
|
-
return null;
|
|
147
|
-
}
|
|
148
|
-
|
|
149
|
-
/**
|
|
150
|
-
* 缓存 metadata 到 .zen/meta.json
|
|
151
|
-
*/
|
|
152
|
-
async cacheMetadata(fileHash: string, filePath: string, metadata: AIMetadata): Promise<void> {
|
|
153
|
-
if (!this.isEnabled()) {
|
|
154
|
-
return;
|
|
155
|
-
}
|
|
156
|
-
|
|
157
|
-
try {
|
|
158
|
-
const metaData = await this.loadMetaData();
|
|
159
|
-
|
|
160
|
-
// 查找是否已存在相同 hash 的缓存(文件移动情况)
|
|
161
|
-
const sameHashIndex = metaData.files.findIndex(f => f.hash === fileHash);
|
|
162
|
-
|
|
163
|
-
// 查找是否已存在相同 path 但不同 hash 的缓存(文件内容更新情况)
|
|
164
|
-
const samePathIndex = metaData.files.findIndex(
|
|
165
|
-
f => f.path === filePath && f.hash !== fileHash
|
|
166
|
-
);
|
|
167
|
-
|
|
168
|
-
if (sameHashIndex >= 0) {
|
|
169
|
-
// 文件移动情况:相同 hash 但 path 可能不同
|
|
170
|
-
// 更新现有缓存项的 path 和 metadata
|
|
171
|
-
metaData.files[sameHashIndex] = {
|
|
172
|
-
hash: fileHash,
|
|
173
|
-
path: filePath,
|
|
174
|
-
metadata,
|
|
175
|
-
lastUpdated: new Date().toISOString(),
|
|
176
|
-
};
|
|
177
|
-
|
|
178
|
-
// 如果存在相同 path 但不同 hash 的旧缓存项,删除它
|
|
179
|
-
if (samePathIndex >= 0 && samePathIndex !== sameHashIndex) {
|
|
180
|
-
metaData.files.splice(samePathIndex, 1);
|
|
181
|
-
}
|
|
182
|
-
} else if (samePathIndex >= 0) {
|
|
183
|
-
// 文件内容更新情况:相同 path 但 hash 不同
|
|
184
|
-
// 删除旧的缓存项,添加新的
|
|
185
|
-
metaData.files.splice(samePathIndex, 1);
|
|
186
|
-
metaData.files.push({
|
|
187
|
-
hash: fileHash,
|
|
188
|
-
path: filePath,
|
|
189
|
-
metadata,
|
|
190
|
-
lastUpdated: new Date().toISOString(),
|
|
191
|
-
});
|
|
192
|
-
} else {
|
|
193
|
-
// 全新的文件,添加新缓存
|
|
194
|
-
metaData.files.push({
|
|
195
|
-
hash: fileHash,
|
|
196
|
-
path: filePath,
|
|
197
|
-
metadata,
|
|
198
|
-
lastUpdated: new Date().toISOString(),
|
|
199
|
-
});
|
|
200
|
-
}
|
|
201
|
-
|
|
202
|
-
await this.saveMetaData(metaData);
|
|
203
|
-
console.log(`💾 Cached AI metadata for: ${filePath}`);
|
|
204
|
-
} catch (error) {
|
|
205
|
-
console.warn(`⚠️ Failed to cache metadata:`, error);
|
|
206
|
-
}
|
|
207
|
-
}
|
|
208
|
-
|
|
209
|
-
/**
|
|
210
|
-
* 清理过期的缓存
|
|
211
|
-
*/
|
|
212
|
-
async cleanupCache(maxAgeDays: number = 30): Promise<void> {
|
|
213
|
-
try {
|
|
214
|
-
const metaData = await this.loadMetaData();
|
|
215
|
-
const cutoffTime = Date.now() - maxAgeDays * 24 * 60 * 60 * 1000;
|
|
216
|
-
const originalCount = metaData.files.length;
|
|
217
|
-
|
|
218
|
-
// 过滤掉过期的缓存
|
|
219
|
-
metaData.files = metaData.files.filter(fileData => {
|
|
220
|
-
const fileTime = new Date(fileData.lastUpdated).getTime();
|
|
221
|
-
return fileTime >= cutoffTime;
|
|
222
|
-
});
|
|
223
|
-
|
|
224
|
-
const cleanedCount = originalCount - metaData.files.length;
|
|
225
|
-
if (cleanedCount > 0) {
|
|
226
|
-
await this.saveMetaData(metaData);
|
|
227
|
-
console.log(`🧹 Cleaned ${cleanedCount} expired AI metadata entries`);
|
|
228
|
-
}
|
|
229
|
-
} catch (error) {
|
|
230
|
-
console.warn(`⚠️ Failed to cleanup cache:`, error);
|
|
231
|
-
}
|
|
232
|
-
}
|
|
233
|
-
|
|
234
|
-
/**
|
|
235
|
-
* 移除孤儿条目(文件已删除但缓存仍存在)
|
|
236
|
-
* @param existingFilePaths 当前存在的文件路径列表
|
|
237
|
-
*/
|
|
238
|
-
async removeOrphanEntries(existingFilePaths: string[]): Promise<void> {
|
|
239
|
-
try {
|
|
240
|
-
const metaData = await this.loadMetaData();
|
|
241
|
-
const originalCount = metaData.files.length;
|
|
242
|
-
|
|
243
|
-
// 创建现有文件路径的 Set 用于快速查找
|
|
244
|
-
const existingPathsSet = new Set(existingFilePaths);
|
|
245
|
-
|
|
246
|
-
// 过滤掉文件已经不存在的缓存条目
|
|
247
|
-
metaData.files = metaData.files.filter(fileData => {
|
|
248
|
-
return existingPathsSet.has(fileData.path);
|
|
249
|
-
});
|
|
250
|
-
|
|
251
|
-
const removedCount = originalCount - metaData.files.length;
|
|
252
|
-
if (removedCount > 0) {
|
|
253
|
-
await this.saveMetaData(metaData);
|
|
254
|
-
console.log(`🗑️ Removed ${removedCount} orphan AI metadata entries`);
|
|
255
|
-
}
|
|
256
|
-
} catch (error) {
|
|
257
|
-
console.warn(`⚠️ Failed to remove orphan entries:`, error);
|
|
258
|
-
}
|
|
259
|
-
}
|
|
260
|
-
|
|
261
|
-
/**
|
|
262
|
-
* 计算文件内容的 hash
|
|
263
|
-
*/
|
|
264
|
-
calculateFileHash(content: string): string {
|
|
265
|
-
return crypto.createHash('sha256').update(content).digest('hex');
|
|
266
|
-
}
|
|
267
|
-
|
|
268
|
-
/**
|
|
269
|
-
* 打印 tokens 使用情况
|
|
270
|
-
*/
|
|
271
|
-
logTokenUsage(filePath: string, tokensUsed: AIMetadata['tokens_used']): void {
|
|
272
|
-
if (!tokensUsed) {
|
|
273
|
-
return;
|
|
274
|
-
}
|
|
275
|
-
|
|
276
|
-
console.log(`🧮 Tokens usage for ${filePath}:`);
|
|
277
|
-
console.log(` Prompt: ${tokensUsed.prompt}`);
|
|
278
|
-
console.log(` Completion: ${tokensUsed.completion}`);
|
|
279
|
-
console.log(` Total: ${tokensUsed.total}`);
|
|
280
|
-
}
|
|
281
|
-
}
|