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.
Files changed (150) hide show
  1. package/.github/workflows/pages.yml +1 -1
  2. package/.zen/meta.json +112 -32
  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 +40 -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 +11 -0
  16. package/.zen/src/zh-Hans/80ae9bed74fc6348a7c1fe9f33e86b65f5d919169721f77bcf0e1bc29fbdb4f9.md +63 -0
  17. package/.zen/src/zh-Hans/f0c2799126931ccd113a0c45b1e623870b0d4f4f400becf6dd877da8f1011517.md +40 -0
  18. package/.zen/src/zh-Hans/fdfca9b960d0eaa8b2b96fe988ead7481d2c0b16f66ebc94fb477139b4178cdc.md +65 -0
  19. package/assets/templates/default/layout.html +274 -0
  20. package/dist/ai/extractMetadataFromMarkdown.d.ts +8 -0
  21. package/dist/ai/extractMetadataFromMarkdown.d.ts.map +1 -0
  22. package/dist/ai/extractMetadataFromMarkdown.js +88 -0
  23. package/dist/ai/extractMetadataFromMarkdown.js.map +1 -0
  24. package/dist/ai/translateMarkdown.d.ts +8 -0
  25. package/dist/ai/translateMarkdown.d.ts.map +1 -0
  26. package/dist/ai/translateMarkdown.js +29 -0
  27. package/dist/ai/translateMarkdown.js.map +1 -0
  28. package/dist/build/pipeline.d.ts +6 -0
  29. package/dist/build/pipeline.d.ts.map +1 -0
  30. package/dist/build/pipeline.js +219 -0
  31. package/dist/build/pipeline.js.map +1 -0
  32. package/dist/cli.js +10 -118
  33. package/dist/cli.js.map +1 -1
  34. package/dist/findEntries.d.ts +10 -0
  35. package/dist/findEntries.d.ts.map +1 -0
  36. package/dist/findEntries.js +38 -0
  37. package/dist/findEntries.js.map +1 -0
  38. package/dist/index.d.ts +1 -32
  39. package/dist/index.d.ts.map +1 -1
  40. package/dist/index.js +1 -35
  41. package/dist/index.js.map +1 -1
  42. package/dist/metadata.d.ts +14 -0
  43. package/dist/metadata.d.ts.map +1 -0
  44. package/dist/metadata.js +78 -0
  45. package/dist/metadata.js.map +1 -0
  46. package/dist/paths.d.ts +6 -0
  47. package/dist/paths.d.ts.map +1 -0
  48. package/dist/paths.js +10 -0
  49. package/dist/paths.js.map +1 -0
  50. package/dist/process/extractMetadataByAI.d.ts +5 -0
  51. package/dist/process/extractMetadataByAI.d.ts.map +1 -0
  52. package/dist/process/extractMetadataByAI.js +31 -0
  53. package/dist/process/extractMetadataByAI.js.map +1 -0
  54. package/dist/process/template.d.ts +5 -0
  55. package/dist/process/template.d.ts.map +1 -0
  56. package/dist/process/template.js +188 -0
  57. package/dist/process/template.js.map +1 -0
  58. package/dist/scan/files.d.ts +7 -0
  59. package/dist/scan/files.d.ts.map +1 -0
  60. package/dist/scan/files.js +54 -0
  61. package/dist/scan/files.js.map +1 -0
  62. package/dist/services/openai.d.ts +41 -0
  63. package/dist/services/openai.d.ts.map +1 -0
  64. package/dist/services/openai.js +54 -0
  65. package/dist/services/openai.js.map +1 -0
  66. package/dist/types.d.ts +16 -67
  67. package/dist/types.d.ts.map +1 -1
  68. package/dist/utils/convertMarkdownToHtml.d.ts +7 -0
  69. package/dist/utils/convertMarkdownToHtml.d.ts.map +1 -0
  70. package/dist/utils/convertMarkdownToHtml.js +39 -0
  71. package/dist/utils/convertMarkdownToHtml.js.map +1 -0
  72. package/dist/utils/frontmatter.d.ts +6 -0
  73. package/dist/utils/frontmatter.d.ts.map +1 -0
  74. package/dist/utils/frontmatter.js +22 -0
  75. package/dist/utils/frontmatter.js.map +1 -0
  76. package/docs/deployment/github-pages.md +1 -2
  77. package/docs/guides/best-practices.md +4 -4
  78. package/docs/guides/config.md +0 -5
  79. package/package.json +4 -2
  80. package/src/ai/extractMetadataFromMarkdown.ts +95 -0
  81. package/src/ai/translateMarkdown.ts +29 -0
  82. package/src/build/pipeline.ts +217 -0
  83. package/src/cli.ts +10 -132
  84. package/src/findEntries.ts +37 -0
  85. package/src/index.ts +1 -40
  86. package/src/metadata.ts +44 -0
  87. package/src/paths.ts +7 -0
  88. package/src/process/extractMetadataByAI.ts +31 -0
  89. package/src/process/template.ts +201 -0
  90. package/src/scan/files.ts +17 -0
  91. package/src/services/openai.ts +92 -0
  92. package/src/types.ts +18 -72
  93. package/src/utils/convertMarkdownToHtml.ts +32 -0
  94. package/src/utils/frontmatter.ts +18 -0
  95. package/.zen/translations.json +0 -51
  96. package/dist/ai-client.d.ts +0 -34
  97. package/dist/ai-client.d.ts.map +0 -1
  98. package/dist/ai-client.js +0 -180
  99. package/dist/ai-client.js.map +0 -1
  100. package/dist/ai-processor.d.ts +0 -51
  101. package/dist/ai-processor.d.ts.map +0 -1
  102. package/dist/ai-processor.js +0 -215
  103. package/dist/ai-processor.js.map +0 -1
  104. package/dist/ai-service.d.ts +0 -79
  105. package/dist/ai-service.d.ts.map +0 -1
  106. package/dist/ai-service.js +0 -257
  107. package/dist/ai-service.js.map +0 -1
  108. package/dist/builder.d.ts +0 -70
  109. package/dist/builder.d.ts.map +0 -1
  110. package/dist/builder.js +0 -854
  111. package/dist/builder.js.map +0 -1
  112. package/dist/gitignore.d.ts +0 -41
  113. package/dist/gitignore.d.ts.map +0 -1
  114. package/dist/gitignore.js +0 -202
  115. package/dist/gitignore.js.map +0 -1
  116. package/dist/gitignore.test.d.ts +0 -2
  117. package/dist/gitignore.test.d.ts.map +0 -1
  118. package/dist/gitignore.test.js +0 -309
  119. package/dist/gitignore.test.js.map +0 -1
  120. package/dist/markdown.d.ts +0 -35
  121. package/dist/markdown.d.ts.map +0 -1
  122. package/dist/markdown.js +0 -221
  123. package/dist/markdown.js.map +0 -1
  124. package/dist/navigation.d.ts +0 -46
  125. package/dist/navigation.d.ts.map +0 -1
  126. package/dist/navigation.js +0 -196
  127. package/dist/navigation.js.map +0 -1
  128. package/dist/scanner.d.ts +0 -26
  129. package/dist/scanner.d.ts.map +0 -1
  130. package/dist/scanner.js +0 -190
  131. package/dist/scanner.js.map +0 -1
  132. package/dist/template.d.ts +0 -33
  133. package/dist/template.d.ts.map +0 -1
  134. package/dist/template.js +0 -434
  135. package/dist/template.js.map +0 -1
  136. package/dist/translation-service.d.ts +0 -72
  137. package/dist/translation-service.d.ts.map +0 -1
  138. package/dist/translation-service.js +0 -291
  139. package/dist/translation-service.js.map +0 -1
  140. package/src/ai-client.ts +0 -227
  141. package/src/ai-processor.ts +0 -243
  142. package/src/ai-service.ts +0 -281
  143. package/src/builder.ts +0 -991
  144. package/src/gitignore.test.ts +0 -318
  145. package/src/gitignore.ts +0 -193
  146. package/src/markdown.ts +0 -212
  147. package/src/navigation.ts +0 -237
  148. package/src/scanner.ts +0 -180
  149. package/src/template.ts +0 -425
  150. package/src/translation-service.ts +0 -350
@@ -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, '&amp;')
148
- .replace(/</g, '&lt;')
149
- .replace(/>/g, '&gt;')
150
- .replace(/"/g, '&quot;')
151
- .replace(/'/g, '&#039;');
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
- }