qskills 1.0.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 (96) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +512 -0
  3. package/bin/qskills.js +10 -0
  4. package/dist/commands/config.d.ts +3 -0
  5. package/dist/commands/config.d.ts.map +1 -0
  6. package/dist/commands/config.js +142 -0
  7. package/dist/commands/config.js.map +1 -0
  8. package/dist/commands/knowledge/add.d.ts +14 -0
  9. package/dist/commands/knowledge/add.d.ts.map +1 -0
  10. package/dist/commands/knowledge/add.js +89 -0
  11. package/dist/commands/knowledge/add.js.map +1 -0
  12. package/dist/commands/knowledge/index.d.ts +3 -0
  13. package/dist/commands/knowledge/index.d.ts.map +1 -0
  14. package/dist/commands/knowledge/index.js +46 -0
  15. package/dist/commands/knowledge/index.js.map +1 -0
  16. package/dist/commands/knowledge/list.d.ts +10 -0
  17. package/dist/commands/knowledge/list.d.ts.map +1 -0
  18. package/dist/commands/knowledge/list.js +55 -0
  19. package/dist/commands/knowledge/list.js.map +1 -0
  20. package/dist/commands/knowledge/remove.d.ts +6 -0
  21. package/dist/commands/knowledge/remove.d.ts.map +1 -0
  22. package/dist/commands/knowledge/remove.js +33 -0
  23. package/dist/commands/knowledge/remove.js.map +1 -0
  24. package/dist/commands/knowledge/search.d.ts +9 -0
  25. package/dist/commands/knowledge/search.d.ts.map +1 -0
  26. package/dist/commands/knowledge/search.js +59 -0
  27. package/dist/commands/knowledge/search.js.map +1 -0
  28. package/dist/commands/skill/add.d.ts +11 -0
  29. package/dist/commands/skill/add.d.ts.map +1 -0
  30. package/dist/commands/skill/add.js +100 -0
  31. package/dist/commands/skill/add.js.map +1 -0
  32. package/dist/commands/skill/index.d.ts +3 -0
  33. package/dist/commands/skill/index.d.ts.map +1 -0
  34. package/dist/commands/skill/index.js +51 -0
  35. package/dist/commands/skill/index.js.map +1 -0
  36. package/dist/commands/skill/list.d.ts +8 -0
  37. package/dist/commands/skill/list.d.ts.map +1 -0
  38. package/dist/commands/skill/list.js +52 -0
  39. package/dist/commands/skill/list.js.map +1 -0
  40. package/dist/commands/skill/remove.d.ts +7 -0
  41. package/dist/commands/skill/remove.d.ts.map +1 -0
  42. package/dist/commands/skill/remove.js +33 -0
  43. package/dist/commands/skill/remove.js.map +1 -0
  44. package/dist/commands/skill/search.d.ts +7 -0
  45. package/dist/commands/skill/search.d.ts.map +1 -0
  46. package/dist/commands/skill/search.js +57 -0
  47. package/dist/commands/skill/search.js.map +1 -0
  48. package/dist/commands/skill/sync.d.ts +7 -0
  49. package/dist/commands/skill/sync.d.ts.map +1 -0
  50. package/dist/commands/skill/sync.js +83 -0
  51. package/dist/commands/skill/sync.js.map +1 -0
  52. package/dist/commands/sync.d.ts +3 -0
  53. package/dist/commands/sync.d.ts.map +1 -0
  54. package/dist/commands/sync.js +63 -0
  55. package/dist/commands/sync.js.map +1 -0
  56. package/dist/core/config.d.ts +16 -0
  57. package/dist/core/config.d.ts.map +1 -0
  58. package/dist/core/config.js +68 -0
  59. package/dist/core/config.js.map +1 -0
  60. package/dist/core/git-sync.d.ts +31 -0
  61. package/dist/core/git-sync.d.ts.map +1 -0
  62. package/dist/core/git-sync.js +240 -0
  63. package/dist/core/git-sync.js.map +1 -0
  64. package/dist/core/init.d.ts +3 -0
  65. package/dist/core/init.d.ts.map +1 -0
  66. package/dist/core/init.js +134 -0
  67. package/dist/core/init.js.map +1 -0
  68. package/dist/core/scanner.d.ts +27 -0
  69. package/dist/core/scanner.d.ts.map +1 -0
  70. package/dist/core/scanner.js +126 -0
  71. package/dist/core/scanner.js.map +1 -0
  72. package/dist/core/storage.d.ts +41 -0
  73. package/dist/core/storage.d.ts.map +1 -0
  74. package/dist/core/storage.js +293 -0
  75. package/dist/core/storage.js.map +1 -0
  76. package/dist/index.d.ts +4 -0
  77. package/dist/index.d.ts.map +1 -0
  78. package/dist/index.js +19 -0
  79. package/dist/index.js.map +1 -0
  80. package/dist/types/index.d.ts +134 -0
  81. package/dist/types/index.d.ts.map +1 -0
  82. package/dist/types/index.js +26 -0
  83. package/dist/types/index.js.map +1 -0
  84. package/dist/utils/file.d.ts +16 -0
  85. package/dist/utils/file.d.ts.map +1 -0
  86. package/dist/utils/file.js +102 -0
  87. package/dist/utils/file.js.map +1 -0
  88. package/dist/utils/helpers.d.ts +7 -0
  89. package/dist/utils/helpers.d.ts.map +1 -0
  90. package/dist/utils/helpers.js +27 -0
  91. package/dist/utils/helpers.js.map +1 -0
  92. package/dist/utils/logger.d.ts +12 -0
  93. package/dist/utils/logger.d.ts.map +1 -0
  94. package/dist/utils/logger.js +50 -0
  95. package/dist/utils/logger.js.map +1 -0
  96. package/package.json +72 -0
@@ -0,0 +1,240 @@
1
+ import git from 'isomorphic-git';
2
+ import http from 'isomorphic-git/http/node';
3
+ import * as fs from 'fs';
4
+ import { join } from 'path';
5
+ import { expandPath } from '../utils/helpers.js';
6
+ import { logger } from '../utils/logger.js';
7
+ export class GitSync {
8
+ dir;
9
+ url;
10
+ branch;
11
+ token;
12
+ constructor(options) {
13
+ this.dir = expandPath(options.dir);
14
+ this.url = options.url;
15
+ this.branch = options.branch || 'main';
16
+ this.token = options.token;
17
+ }
18
+ getAuth() {
19
+ if (!this.token)
20
+ return undefined;
21
+ return {
22
+ username: this.token,
23
+ password: 'x-oauth-basic'
24
+ };
25
+ }
26
+ async initRepo() {
27
+ try {
28
+ await git.init({
29
+ fs,
30
+ dir: this.dir,
31
+ defaultBranch: this.branch
32
+ });
33
+ logger.debug(`Initialized git repository at ${this.dir}`);
34
+ }
35
+ catch (error) {
36
+ logger.error(`Failed to init repo: ${error}`);
37
+ throw error;
38
+ }
39
+ }
40
+ async clone() {
41
+ try {
42
+ await git.clone({
43
+ fs,
44
+ http,
45
+ dir: this.dir,
46
+ url: this.url,
47
+ ref: this.branch,
48
+ onAuth: () => this.getAuth(),
49
+ singleBranch: true,
50
+ depth: 1
51
+ });
52
+ logger.debug(`Cloned ${this.url} to ${this.dir}`);
53
+ }
54
+ catch (error) {
55
+ logger.error(`Failed to clone repo: ${error}`);
56
+ throw error;
57
+ }
58
+ }
59
+ async pull() {
60
+ const result = {
61
+ success: false,
62
+ pulled: 0,
63
+ pushed: 0,
64
+ conflicts: [],
65
+ errors: []
66
+ };
67
+ try {
68
+ const status = await this.getStatus();
69
+ if (status.modified.length > 0 || status.staged.length > 0) {
70
+ result.conflicts = [...status.modified, ...status.staged];
71
+ result.errors.push('Local changes detected. Please commit or stash first.');
72
+ return result;
73
+ }
74
+ await git.pull({
75
+ fs,
76
+ http,
77
+ dir: this.dir,
78
+ ref: this.branch,
79
+ onAuth: () => this.getAuth(),
80
+ singleBranch: true,
81
+ fastForward: true
82
+ });
83
+ result.pulled = 1;
84
+ result.success = true;
85
+ logger.debug(`Pulled changes from ${this.url}`);
86
+ }
87
+ catch (error) {
88
+ const errorMessage = error instanceof Error ? error.message : String(error);
89
+ result.errors.push(errorMessage);
90
+ logger.error(`Failed to pull: ${errorMessage}`);
91
+ }
92
+ return result;
93
+ }
94
+ async push() {
95
+ const result = {
96
+ success: false,
97
+ pulled: 0,
98
+ pushed: 0,
99
+ conflicts: [],
100
+ errors: []
101
+ };
102
+ try {
103
+ const status = await this.getStatus();
104
+ if (status.ahead === 0) {
105
+ result.success = true;
106
+ return result;
107
+ }
108
+ await git.push({
109
+ fs,
110
+ http,
111
+ dir: this.dir,
112
+ ref: this.branch,
113
+ onAuth: () => this.getAuth()
114
+ });
115
+ result.pushed = status.ahead;
116
+ result.success = true;
117
+ logger.debug(`Pushed changes to ${this.url}`);
118
+ }
119
+ catch (error) {
120
+ const errorMessage = error instanceof Error ? error.message : String(error);
121
+ result.errors.push(errorMessage);
122
+ logger.error(`Failed to push: ${errorMessage}`);
123
+ }
124
+ return result;
125
+ }
126
+ async commit(message) {
127
+ try {
128
+ const status = await this.getStatus();
129
+ for (const file of [...status.modified, ...status.untracked]) {
130
+ await git.add({
131
+ fs,
132
+ dir: this.dir,
133
+ filepath: file
134
+ });
135
+ }
136
+ const sha = await git.commit({
137
+ fs,
138
+ dir: this.dir,
139
+ message,
140
+ author: {
141
+ name: 'qskills',
142
+ email: 'qskills@local'
143
+ }
144
+ });
145
+ logger.debug(`Committed: ${sha}`);
146
+ return sha;
147
+ }
148
+ catch (error) {
149
+ logger.error(`Failed to commit: ${error}`);
150
+ return null;
151
+ }
152
+ }
153
+ async getStatus() {
154
+ const status = {
155
+ ahead: 0,
156
+ behind: 0,
157
+ modified: [],
158
+ untracked: [],
159
+ staged: []
160
+ };
161
+ try {
162
+ const files = await git.statusMatrix({
163
+ fs,
164
+ dir: this.dir
165
+ });
166
+ for (const [filepath, head, workdir, stage] of files) {
167
+ if (head === 1 && workdir === 2 && stage === 2) {
168
+ status.staged.push(filepath);
169
+ }
170
+ else if (head === 1 && workdir === 2) {
171
+ status.modified.push(filepath);
172
+ }
173
+ else if (head === 0 && workdir === 1) {
174
+ status.untracked.push(filepath);
175
+ }
176
+ }
177
+ status.ahead = status.staged.length + status.modified.length > 0 ? 1 : 0;
178
+ }
179
+ catch (error) {
180
+ logger.debug(`Failed to get status: ${error}`);
181
+ }
182
+ return status;
183
+ }
184
+ async hasChanges() {
185
+ const status = await this.getStatus();
186
+ return status.modified.length > 0 || status.untracked.length > 0;
187
+ }
188
+ async addRemotes(remotes) {
189
+ for (const remote of remotes) {
190
+ try {
191
+ await git.addRemote({
192
+ fs,
193
+ dir: this.dir,
194
+ remote: remote.name,
195
+ url: remote.url
196
+ });
197
+ }
198
+ catch {
199
+ logger.debug(`Remote ${remote.name} may already exist`);
200
+ }
201
+ }
202
+ }
203
+ async fetch(remoteName = 'origin') {
204
+ try {
205
+ await git.fetch({
206
+ fs,
207
+ http,
208
+ dir: this.dir,
209
+ remote: remoteName,
210
+ onAuth: () => this.getAuth()
211
+ });
212
+ logger.debug(`Fetched from ${remoteName}`);
213
+ }
214
+ catch (error) {
215
+ logger.error(`Failed to fetch: ${error}`);
216
+ throw error;
217
+ }
218
+ }
219
+ }
220
+ // 辅助函数
221
+ export async function isGitRepo(dir) {
222
+ try {
223
+ const expanded = expandPath(dir);
224
+ await fs.promises.readdir(join(expanded, '.git'));
225
+ return true;
226
+ }
227
+ catch {
228
+ return false;
229
+ }
230
+ }
231
+ export async function getToken() {
232
+ const envToken = process.env.QSKILLS_TOKEN;
233
+ if (envToken)
234
+ return envToken;
235
+ return null;
236
+ }
237
+ export async function setToken(token) {
238
+ process.env.QSKILLS_TOKEN = token;
239
+ }
240
+ //# sourceMappingURL=git-sync.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"git-sync.js","sourceRoot":"","sources":["../../src/core/git-sync.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ export declare function checkFirstRun(): Promise<void>;
2
+ export declare function runFirstTimeSetup(): Promise<void>;
3
+ //# sourceMappingURL=init.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"init.d.ts","sourceRoot":"","sources":["../../src/core/init.ts"],"names":[],"mappings":"AAOA,wBAAsB,aAAa,IAAI,OAAO,CAAC,IAAI,CAAC,CAOnD;AAED,wBAAsB,iBAAiB,IAAI,OAAO,CAAC,IAAI,CAAC,CAwIvD"}
@@ -0,0 +1,134 @@
1
+ import { getConfigManager } from './config.js';
2
+ import { initStorage } from './storage.js';
3
+ import { expandPath } from '../utils/helpers.js';
4
+ import { DEFAULT_CONFIG } from '../types/index.js';
5
+ export async function checkFirstRun() {
6
+ const configManager = getConfigManager();
7
+ const config = await configManager.load();
8
+ if (!config.initialized) {
9
+ await runFirstTimeSetup();
10
+ }
11
+ }
12
+ export async function runFirstTimeSetup() {
13
+ const { select, input, confirm } = await import('@inquirer/prompts');
14
+ console.log('\n欢迎使用 Qcli\n');
15
+ console.log('让我们完成初始配置,只需 3 个步骤\n');
16
+ // 步骤 1: 配置存储路径
17
+ console.log('步骤 1/3:配置存储路径');
18
+ console.log('─'.repeat(40));
19
+ const defaultPath = DEFAULT_CONFIG.storage.baseDir;
20
+ const storagePath = await input({
21
+ message: `存储路径`,
22
+ default: defaultPath,
23
+ validate: (value) => {
24
+ if (!value.trim())
25
+ return '请输入路径';
26
+ return true;
27
+ }
28
+ });
29
+ // 步骤 2: 配置 Git 仓库
30
+ console.log('\n步骤 2/3:配置 Git 仓库(可选)');
31
+ console.log('─'.repeat(40));
32
+ const setupGit = await confirm({
33
+ message: '是否关联远程 Git 仓库进行同步?',
34
+ default: false
35
+ });
36
+ let publicRepoUrl = '';
37
+ let privateRepoUrl = '';
38
+ if (setupGit) {
39
+ const repoType = await select({
40
+ message: '选择仓库配置方式',
41
+ choices: [
42
+ { value: 'public', name: '配置公共仓库' },
43
+ { value: 'private', name: '配置私人仓库' },
44
+ { value: 'both', name: '同时配置两种仓库' }
45
+ ]
46
+ });
47
+ if (repoType === 'public' || repoType === 'both') {
48
+ publicRepoUrl = await input({
49
+ message: '公共仓库 URL',
50
+ validate: (value) => {
51
+ if (!value.trim())
52
+ return '请输入仓库 URL';
53
+ if (!value.startsWith('http') && !value.startsWith('git@')) {
54
+ return '请输入有效的 Git 仓库 URL';
55
+ }
56
+ return true;
57
+ }
58
+ });
59
+ }
60
+ if (repoType === 'private' || repoType === 'both') {
61
+ privateRepoUrl = await input({
62
+ message: '私人仓库 URL',
63
+ validate: (value) => {
64
+ if (!value.trim())
65
+ return '请输入仓库 URL';
66
+ if (!value.startsWith('http') && !value.startsWith('git@')) {
67
+ return '请输入有效的 Git 仓库 URL';
68
+ }
69
+ return true;
70
+ }
71
+ });
72
+ }
73
+ }
74
+ // 步骤 3: 权限设置
75
+ console.log('\n步骤 3/3:权限设置');
76
+ console.log('─'.repeat(40));
77
+ const enableScanner = await confirm({
78
+ message: '启用敏感信息扫描?',
79
+ default: true
80
+ });
81
+ const scanBeforePush = await confirm({
82
+ message: '推送前强制扫描?',
83
+ default: true
84
+ });
85
+ // 构建配置
86
+ const config = {
87
+ ...DEFAULT_CONFIG,
88
+ initialized: true,
89
+ initializedAt: new Date().toISOString(),
90
+ storage: {
91
+ ...DEFAULT_CONFIG.storage,
92
+ baseDir: storagePath
93
+ },
94
+ scanner: {
95
+ ...DEFAULT_CONFIG.scanner,
96
+ enabled: enableScanner
97
+ },
98
+ security: {
99
+ ...DEFAULT_CONFIG.security,
100
+ scanBeforePush
101
+ }
102
+ };
103
+ if (publicRepoUrl) {
104
+ config.remotes.public = {
105
+ url: publicRepoUrl,
106
+ branch: 'main',
107
+ enabled: true
108
+ };
109
+ }
110
+ if (privateRepoUrl) {
111
+ config.remotes.private = {
112
+ url: privateRepoUrl,
113
+ branch: 'main',
114
+ enabled: true
115
+ };
116
+ }
117
+ // 保存配置
118
+ const configManager = getConfigManager();
119
+ await configManager.save(config);
120
+ // 初始化存储
121
+ await initStorage(config);
122
+ // 显示完成信息
123
+ console.log('\n' + '─'.repeat(50));
124
+ console.log('\n配置完成!\n');
125
+ console.log(`存储路径:${expandPath(storagePath)}`);
126
+ if (publicRepoUrl) {
127
+ console.log(`公共仓库:${publicRepoUrl}`);
128
+ }
129
+ if (privateRepoUrl) {
130
+ console.log(`私人仓库:${privateRepoUrl}`);
131
+ }
132
+ console.log('\n运行 `qskills --help` 查看可用命令\n');
133
+ }
134
+ //# sourceMappingURL=init.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"init.js","sourceRoot":"","sources":["../../src/core/init.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,27 @@
1
+ import type { ScanResult, CustomScanRule } from '../types/index.js';
2
+ export declare class Scanner {
3
+ private patterns;
4
+ private skipPatterns;
5
+ private customPatterns;
6
+ constructor(options?: {
7
+ skipPatterns?: string[];
8
+ customPatterns?: CustomScanRule[];
9
+ });
10
+ private globToRegex;
11
+ private shouldSkip;
12
+ scanContent(content: string, filePath: string): Promise<ScanResult>;
13
+ scanFiles(files: Map<string, string>): Promise<ScanResult>;
14
+ scanDirectory(filePaths: string[], readFile: (path: string) => Promise<string>): Promise<ScanResult>;
15
+ private maskSecret;
16
+ getPatternNames(): string[];
17
+ addPattern(pattern: {
18
+ name: string;
19
+ pattern: string;
20
+ severity: 'high' | 'medium' | 'low';
21
+ }): void;
22
+ }
23
+ export declare function getScanner(options?: {
24
+ skipPatterns?: string[];
25
+ customPatterns?: CustomScanRule[];
26
+ }): Scanner;
27
+ //# sourceMappingURL=scanner.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scanner.d.ts","sourceRoot":"","sources":["../../src/core/scanner.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAiB,UAAU,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;AAoBnF,qBAAa,OAAO;IAClB,OAAO,CAAC,QAAQ,CAAgF;IAChG,OAAO,CAAC,YAAY,CAAW;IAC/B,OAAO,CAAC,cAAc,CAAmB;gBAE7B,OAAO,CAAC,EAAE;QAAE,YAAY,CAAC,EAAE,MAAM,EAAE,CAAC;QAAC,cAAc,CAAC,EAAE,cAAc,EAAE,CAAA;KAAE;IAepF,OAAO,CAAC,WAAW;IAQnB,OAAO,CAAC,UAAU;IAIZ,WAAW,CAAC,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,UAAU,CAAC;IAgCnE,SAAS,CAAC,KAAK,EAAE,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,OAAO,CAAC,UAAU,CAAC;IAc1D,aAAa,CAAC,SAAS,EAAE,MAAM,EAAE,EAAE,QAAQ,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,OAAO,CAAC,MAAM,CAAC,GAAG,OAAO,CAAC,UAAU,CAAC;IAqB1G,OAAO,CAAC,UAAU;IAOlB,eAAe,IAAI,MAAM,EAAE;IAI3B,UAAU,CAAC,OAAO,EAAE;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAC;QAAC,QAAQ,EAAE,MAAM,GAAG,QAAQ,GAAG,KAAK,CAAA;KAAE,GAAG,IAAI;CAOlG;AAKD,wBAAgB,UAAU,CAAC,OAAO,CAAC,EAAE;IAAE,YAAY,CAAC,EAAE,MAAM,EAAE,CAAC;IAAC,cAAc,CAAC,EAAE,cAAc,EAAE,CAAA;CAAE,GAAG,OAAO,CAK5G"}
@@ -0,0 +1,126 @@
1
+ // 内置敏感信息检测规则
2
+ const SECRET_PATTERNS = [
3
+ { name: 'aws-access-key', pattern: /AKIA[0-9A-Z]{16}/g, severity: 'high' },
4
+ { name: 'aws-secret-key', pattern: /aws(.{0,20})?['"][0-9a-zA-Z\/+=]{40}['"]/gi, severity: 'high' },
5
+ { name: 'github-token', pattern: /ghp_[a-zA-Z0-9]{36}/g, severity: 'high' },
6
+ { name: 'github-oauth', pattern: /gho_[a-zA-Z0-9]{36}/g, severity: 'high' },
7
+ { name: 'github-app-token', pattern: /(ghu|ghs)_[a-zA-Z0-9]{36}/g, severity: 'high' },
8
+ { name: 'private-key', pattern: /-----BEGIN (?:RSA |EC |DSA |OPENSSH )?PRIVATE KEY-----/g, severity: 'high' },
9
+ { name: 'api-key-generic', pattern: /(api[_-]?key|apikey|secret[_-]?key|access[_-]?key)\s*[:=]\s*['"]?[a-zA-Z0-9_\-]{20,}['"]?/gi, severity: 'medium' },
10
+ { name: 'password-in-code', pattern: /(password|passwd|pwd)\s*[:=]\s*['"][^'"]{8,}['"]/gi, severity: 'medium' },
11
+ { name: 'jwt-token', pattern: /eyJ[a-zA-Z0-9_-]*\.eyJ[a-zA-Z0-9_-]*\.[a-zA-Z0-9_-]*/g, severity: 'medium' },
12
+ { name: 'slack-token', pattern: /xox[baprs]-[0-9]{10,}-[0-9]{10,}-[a-zA-Z0-9]{24}/g, severity: 'high' },
13
+ { name: 'google-api-key', pattern: /AIza[0-9A-Za-z\-_]{35}/g, severity: 'high' },
14
+ { name: 'stripe-key', pattern: /sk_live_[0-9a-zA-Z]{24}/g, severity: 'high' },
15
+ { name: 'database-url', pattern: /(mysql|postgres|mongodb|redis):\/\/[^:]+:[^@]+@[^\s]+/gi, severity: 'high' },
16
+ { name: 'env-file-secrets', pattern: /\.env(\.local|\.production)?/g, severity: 'low' },
17
+ ];
18
+ export class Scanner {
19
+ patterns;
20
+ skipPatterns;
21
+ customPatterns;
22
+ constructor(options) {
23
+ this.patterns = [...SECRET_PATTERNS];
24
+ this.skipPatterns = (options?.skipPatterns || []).map(p => this.globToRegex(p));
25
+ this.customPatterns = options?.customPatterns || [];
26
+ // 添加自定义规则
27
+ for (const rule of this.customPatterns) {
28
+ this.patterns.push({
29
+ name: rule.name,
30
+ pattern: new RegExp(rule.pattern, 'g'),
31
+ severity: rule.severity
32
+ });
33
+ }
34
+ }
35
+ globToRegex(glob) {
36
+ const regex = glob
37
+ .replace(/\*\*/g, '.*')
38
+ .replace(/\*/g, '[^/]*')
39
+ .replace(/\./g, '\\.');
40
+ return new RegExp(regex);
41
+ }
42
+ shouldSkip(filePath) {
43
+ return this.skipPatterns.some(pattern => pattern.test(filePath));
44
+ }
45
+ async scanContent(content, filePath) {
46
+ if (this.shouldSkip(filePath)) {
47
+ return { hasSecrets: false, findings: [] };
48
+ }
49
+ const findings = [];
50
+ for (const { name, pattern, severity } of this.patterns) {
51
+ let match;
52
+ const regex = new RegExp(pattern.source, pattern.flags);
53
+ while ((match = regex.exec(content)) !== null) {
54
+ // 计算行号
55
+ const beforeMatch = content.substring(0, match.index);
56
+ const line = beforeMatch.split('\n').length;
57
+ findings.push({
58
+ file: filePath,
59
+ line,
60
+ type: name,
61
+ match: this.maskSecret(match[0]),
62
+ severity
63
+ });
64
+ }
65
+ }
66
+ return {
67
+ hasSecrets: findings.length > 0,
68
+ findings
69
+ };
70
+ }
71
+ async scanFiles(files) {
72
+ const allFindings = [];
73
+ for (const [filePath, content] of files) {
74
+ const result = await this.scanContent(content, filePath);
75
+ allFindings.push(...result.findings);
76
+ }
77
+ return {
78
+ hasSecrets: allFindings.length > 0,
79
+ findings: allFindings
80
+ };
81
+ }
82
+ async scanDirectory(filePaths, readFile) {
83
+ const allFindings = [];
84
+ for (const filePath of filePaths) {
85
+ if (this.shouldSkip(filePath))
86
+ continue;
87
+ try {
88
+ const content = await readFile(filePath);
89
+ const result = await this.scanContent(content, filePath);
90
+ allFindings.push(...result.findings);
91
+ }
92
+ catch {
93
+ // 忽略无法读取的文件
94
+ }
95
+ }
96
+ return {
97
+ hasSecrets: allFindings.length > 0,
98
+ findings: allFindings
99
+ };
100
+ }
101
+ maskSecret(secret) {
102
+ if (secret.length <= 8) {
103
+ return '*'.repeat(secret.length);
104
+ }
105
+ return secret.substring(0, 4) + '*'.repeat(secret.length - 8) + secret.substring(secret.length - 4);
106
+ }
107
+ getPatternNames() {
108
+ return this.patterns.map(p => p.name);
109
+ }
110
+ addPattern(pattern) {
111
+ this.patterns.push({
112
+ name: pattern.name,
113
+ pattern: new RegExp(pattern.pattern, 'g'),
114
+ severity: pattern.severity
115
+ });
116
+ }
117
+ }
118
+ // 单例
119
+ let scannerInstance = null;
120
+ export function getScanner(options) {
121
+ if (!scannerInstance) {
122
+ scannerInstance = new Scanner(options);
123
+ }
124
+ return scannerInstance;
125
+ }
126
+ //# sourceMappingURL=scanner.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scanner.js","sourceRoot":"","sources":["../../src/core/scanner.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,41 @@
1
+ import type { Config, SkillMeta, KnowledgeItem, LocalIndex, SkillFilter, KnowledgeFilter } from '../types/index.js';
2
+ export declare class Storage {
3
+ private config;
4
+ private index;
5
+ private configPath;
6
+ private baseDir;
7
+ constructor(configPath?: string);
8
+ init(config?: Config): Promise<void>;
9
+ getConfig(): Config | null;
10
+ updateConfig(updates: Partial<Config>): Promise<void>;
11
+ private saveConfig;
12
+ private loadOrCreateIndex;
13
+ private saveIndex;
14
+ addSkill(sourcePath: string, options: {
15
+ name: string;
16
+ type: 'single' | 'folder';
17
+ source: 'public' | 'private';
18
+ description?: string;
19
+ tags?: string[];
20
+ }): Promise<SkillMeta>;
21
+ removeSkill(name: string, source?: 'public' | 'private'): Promise<boolean>;
22
+ getSkill(name: string, source?: 'public' | 'private'): Promise<SkillMeta | null>;
23
+ listSkills(filter?: SkillFilter): Promise<SkillMeta[]>;
24
+ addKnowledge(sourcePath: string, options: {
25
+ title: string;
26
+ type: KnowledgeItem['type'];
27
+ category: string;
28
+ source: 'public' | 'private';
29
+ tags?: string[];
30
+ keywords?: string[];
31
+ summary?: string;
32
+ }): Promise<KnowledgeItem>;
33
+ removeKnowledge(id: string): Promise<boolean>;
34
+ getKnowledge(id: string): Promise<KnowledgeItem | null>;
35
+ listKnowledge(filter?: KnowledgeFilter): Promise<KnowledgeItem[]>;
36
+ getIndex(): LocalIndex | null;
37
+ getBaseDir(): string;
38
+ }
39
+ export declare function getStorage(): Promise<Storage>;
40
+ export declare function initStorage(config: Config): Promise<Storage>;
41
+ //# sourceMappingURL=storage.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"storage.d.ts","sourceRoot":"","sources":["../../src/core/storage.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,MAAM,EAAE,SAAS,EAAE,aAAa,EAAE,UAAU,EAAE,WAAW,EAAE,eAAe,EAAkB,MAAM,mBAAmB,CAAC;AAKpI,qBAAa,OAAO;IAClB,OAAO,CAAC,MAAM,CAAuB;IACrC,OAAO,CAAC,KAAK,CAA2B;IACxC,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,OAAO,CAAc;gBAEjB,UAAU,CAAC,EAAE,MAAM;IAIzB,IAAI,CAAC,MAAM,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAwB1C,SAAS,IAAI,MAAM,GAAG,IAAI;IAIpB,YAAY,CAAC,OAAO,EAAE,OAAO,CAAC,MAAM,CAAC,GAAG,OAAO,CAAC,IAAI,CAAC;YAO7C,UAAU;YAIV,iBAAiB;YAgBjB,SAAS;IAOjB,QAAQ,CACZ,UAAU,EAAE,MAAM,EAClB,OAAO,EAAE;QACP,IAAI,EAAE,MAAM,CAAC;QACb,IAAI,EAAE,QAAQ,GAAG,QAAQ,CAAC;QAC1B,MAAM,EAAE,QAAQ,GAAG,SAAS,CAAC;QAC7B,WAAW,CAAC,EAAE,MAAM,CAAC;QACrB,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC;KACjB,GACA,OAAO,CAAC,SAAS,CAAC;IAoDf,WAAW,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,QAAQ,GAAG,SAAS,GAAG,OAAO,CAAC,OAAO,CAAC;IA0B1E,QAAQ,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,QAAQ,GAAG,SAAS,GAAG,OAAO,CAAC,SAAS,GAAG,IAAI,CAAC;IAchF,UAAU,CAAC,MAAM,CAAC,EAAE,WAAW,GAAG,OAAO,CAAC,SAAS,EAAE,CAAC;IA8BtD,YAAY,CAChB,UAAU,EAAE,MAAM,EAClB,OAAO,EAAE;QACP,KAAK,EAAE,MAAM,CAAC;QACd,IAAI,EAAE,aAAa,CAAC,MAAM,CAAC,CAAC;QAC5B,QAAQ,EAAE,MAAM,CAAC;QACjB,MAAM,EAAE,QAAQ,GAAG,SAAS,CAAC;QAC7B,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC;QAChB,QAAQ,CAAC,EAAE,MAAM,EAAE,CAAC;QACpB,OAAO,CAAC,EAAE,MAAM,CAAC;KAClB,GACA,OAAO,CAAC,aAAa,CAAC;IAoDnB,eAAe,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAyB7C,YAAY,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO,CAAC,aAAa,GAAG,IAAI,CAAC;IAcvD,aAAa,CAAC,MAAM,CAAC,EAAE,eAAe,GAAG,OAAO,CAAC,aAAa,EAAE,CAAC;IA4BvE,QAAQ,IAAI,UAAU,GAAG,IAAI;IAI7B,UAAU,IAAI,MAAM;CAGrB;AAKD,wBAAsB,UAAU,IAAI,OAAO,CAAC,OAAO,CAAC,CAMnD;AAED,wBAAsB,WAAW,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC,CAIlE"}