code-gate 0.1.0 → 1.0.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 (133) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +222 -114
  3. package/README_ZH.md +224 -0
  4. package/bin/code-gate.js +12 -19
  5. package/dist/__tests__/config.test.js +1 -1
  6. package/dist/__tests__/config.test.js.map +1 -1
  7. package/dist/__tests__/render.test.js +24 -7
  8. package/dist/__tests__/render.test.js.map +1 -1
  9. package/dist/cli/commands/hook.d.ts +1 -0
  10. package/dist/cli/commands/hook.js +164 -0
  11. package/dist/cli/commands/hook.js.map +1 -0
  12. package/dist/cli/commands/init.d.ts +1 -0
  13. package/dist/cli/commands/init.js +157 -0
  14. package/dist/cli/commands/init.js.map +1 -0
  15. package/dist/cli/commands/setup.d.ts +1 -0
  16. package/dist/cli/commands/setup.js +19 -0
  17. package/dist/cli/commands/setup.js.map +1 -0
  18. package/dist/cli/index.d.ts +2 -0
  19. package/dist/cli/index.js +26 -0
  20. package/dist/cli/index.js.map +1 -0
  21. package/dist/commands/hook.js +7 -0
  22. package/dist/commands/hook.js.map +1 -1
  23. package/dist/commands/init.js +20 -20
  24. package/dist/config/defaults.d.ts +2 -0
  25. package/dist/config/defaults.js +33 -0
  26. package/dist/config/defaults.js.map +1 -0
  27. package/dist/config/index.d.ts +4 -0
  28. package/dist/config/index.js +77 -0
  29. package/dist/config/index.js.map +1 -0
  30. package/dist/config/types.d.ts +55 -0
  31. package/dist/config/types.js +2 -0
  32. package/dist/config/types.js.map +1 -0
  33. package/dist/config.d.ts +111 -21
  34. package/dist/config.js +83 -21
  35. package/dist/config.js.map +1 -1
  36. package/dist/core/git.d.ts +7 -0
  37. package/dist/core/git.js +67 -0
  38. package/dist/core/git.js.map +1 -0
  39. package/dist/core/review-flow.js +105 -59
  40. package/dist/core/review-flow.js.map +1 -1
  41. package/dist/core/review.d.ts +6 -0
  42. package/dist/core/review.js +167 -0
  43. package/dist/core/review.js.map +1 -0
  44. package/dist/index.d.ts +1 -1
  45. package/dist/index.js +1 -1
  46. package/dist/index.js.map +1 -1
  47. package/dist/llm/base.d.ts +14 -0
  48. package/dist/llm/base.js +10 -0
  49. package/dist/llm/base.js.map +1 -0
  50. package/dist/llm/deepseek.js +3 -3
  51. package/dist/llm/deepseek.js.map +1 -1
  52. package/dist/llm/index.d.ts +4 -0
  53. package/dist/llm/index.js +41 -0
  54. package/dist/llm/index.js.map +1 -0
  55. package/dist/llm/ollama.js +2 -2
  56. package/dist/llm/ollama.js.map +1 -1
  57. package/dist/llm/providers/aliyun.d.ts +4 -0
  58. package/dist/llm/providers/aliyun.js +25 -0
  59. package/dist/llm/providers/aliyun.js.map +1 -0
  60. package/dist/llm/providers/anthropic.d.ts +4 -0
  61. package/dist/llm/providers/anthropic.js +52 -0
  62. package/dist/llm/providers/anthropic.js.map +1 -0
  63. package/dist/llm/providers/azure.d.ts +4 -0
  64. package/dist/llm/providers/azure.js +38 -0
  65. package/dist/llm/providers/azure.js.map +1 -0
  66. package/dist/llm/providers/cohere.d.ts +4 -0
  67. package/dist/llm/providers/cohere.js +39 -0
  68. package/dist/llm/providers/cohere.js.map +1 -0
  69. package/dist/llm/providers/deepseek.d.ts +4 -0
  70. package/dist/llm/providers/deepseek.js +25 -0
  71. package/dist/llm/providers/deepseek.js.map +1 -0
  72. package/dist/llm/providers/gemini.d.ts +4 -0
  73. package/dist/llm/providers/gemini.js +40 -0
  74. package/dist/llm/providers/gemini.js.map +1 -0
  75. package/dist/llm/providers/mistral.d.ts +4 -0
  76. package/dist/llm/providers/mistral.js +42 -0
  77. package/dist/llm/providers/mistral.js.map +1 -0
  78. package/dist/llm/providers/ollama.d.ts +5 -0
  79. package/dist/llm/providers/ollama.js +67 -0
  80. package/dist/llm/providers/ollama.js.map +1 -0
  81. package/dist/llm/providers/openai.d.ts +4 -0
  82. package/dist/llm/providers/openai.js +27 -0
  83. package/dist/llm/providers/openai.js.map +1 -0
  84. package/dist/llm/providers/volcengine.d.ts +4 -0
  85. package/dist/llm/providers/volcengine.js +29 -0
  86. package/dist/llm/providers/volcengine.js.map +1 -0
  87. package/dist/llm/providers/zhipu.d.ts +4 -0
  88. package/dist/llm/providers/zhipu.js +25 -0
  89. package/dist/llm/providers/zhipu.js.map +1 -0
  90. package/dist/locales/de.d.ts +2 -0
  91. package/dist/locales/de.js +33 -0
  92. package/dist/locales/de.js.map +1 -0
  93. package/dist/locales/en.d.ts +2 -0
  94. package/dist/locales/en.js +33 -0
  95. package/dist/locales/en.js.map +1 -0
  96. package/dist/locales/fr.d.ts +2 -0
  97. package/dist/locales/fr.js +33 -0
  98. package/dist/locales/fr.js.map +1 -0
  99. package/dist/locales/index.d.ts +3 -0
  100. package/dist/locales/index.js +41 -0
  101. package/dist/locales/index.js.map +1 -0
  102. package/dist/locales/ja.d.ts +2 -0
  103. package/dist/locales/ja.js +33 -0
  104. package/dist/locales/ja.js.map +1 -0
  105. package/dist/locales/ko.d.ts +2 -0
  106. package/dist/locales/ko.js +33 -0
  107. package/dist/locales/ko.js.map +1 -0
  108. package/dist/locales/types.d.ts +32 -0
  109. package/dist/locales/types.js +2 -0
  110. package/dist/locales/types.js.map +1 -0
  111. package/dist/locales/zh-CN.d.ts +2 -0
  112. package/dist/locales/zh-CN.js +33 -0
  113. package/dist/locales/zh-CN.js.map +1 -0
  114. package/dist/locales/zh-TW.d.ts +2 -0
  115. package/dist/locales/zh-TW.js +33 -0
  116. package/dist/locales/zh-TW.js.map +1 -0
  117. package/dist/ui/render/assets.d.ts +8 -0
  118. package/dist/ui/render/assets.js +89 -0
  119. package/dist/ui/render/assets.js.map +1 -0
  120. package/dist/ui/render/html.d.ts +35 -0
  121. package/dist/ui/render/html.js +294 -0
  122. package/dist/ui/render/html.js.map +1 -0
  123. package/dist/ui/render.d.ts +11 -0
  124. package/dist/ui/render.js +198 -13
  125. package/dist/ui/render.js.map +1 -1
  126. package/dist/ui/server.d.ts +4 -2
  127. package/dist/ui/server.js +30 -8
  128. package/dist/ui/server.js.map +1 -1
  129. package/dist/utils/logger.d.ts +3 -0
  130. package/dist/utils/logger.js +10 -0
  131. package/dist/utils/logger.js.map +1 -0
  132. package/package.json +27 -8
  133. package/scripts/hook.sh +12 -0
@@ -1,4 +1,4 @@
1
- import { renderHTML } from '../ui/render.js';
1
+ import { renderReviewItem } from '../ui/render/html.js';
2
2
  const sampleDiff = `diff --git a/a.txt b/a.txt
3
3
  index e69de29..4b825dc 100644
4
4
  --- a/a.txt
@@ -6,11 +6,28 @@ index e69de29..4b825dc 100644
6
6
  @@ -0,0 +1,1 @@
7
7
  +hello
8
8
  `;
9
- function run() {
10
- const html = renderHTML(sampleDiff, 'review');
11
- if (!html.includes('Code Review'))
12
- throw new Error('render failed');
13
- process.stdout.write('render.test passed\n');
9
+ const sampleReview = 'Looks good';
10
+ function test(name, fn) {
11
+ try {
12
+ fn();
13
+ console.log(`PASS: ${name}`);
14
+ }
15
+ catch (e) {
16
+ console.error(`FAIL: ${name}`, e);
17
+ process.exit(1);
18
+ }
14
19
  }
15
- run();
20
+ function expect(actual) {
21
+ return {
22
+ toContain(expected) {
23
+ if (!actual.includes(expected))
24
+ throw new Error(`Expected "${actual}" to contain "${expected}"`);
25
+ }
26
+ };
27
+ }
28
+ test('renders diff and review', () => {
29
+ const item = renderReviewItem({ file: 'a.txt', review: sampleReview, diff: sampleDiff });
30
+ expect(item.reviewHtml).toContain('Looks good');
31
+ expect(item.diffHtml).toContain('d2h-file-header');
32
+ });
16
33
  //# sourceMappingURL=render.test.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"render.test.js","sourceRoot":"","sources":["../../src/__tests__/render.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAA;AAE5C,MAAM,UAAU,GAAG;;;;;;CAMlB,CAAA;AAED,SAAS,GAAG;IACV,MAAM,IAAI,GAAG,UAAU,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAA;IAC7C,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC;QAAE,MAAM,IAAI,KAAK,CAAC,eAAe,CAAC,CAAA;IACnE,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,sBAAsB,CAAC,CAAA;AAC9C,CAAC;AAED,GAAG,EAAE,CAAA"}
1
+ {"version":3,"file":"render.test.js","sourceRoot":"","sources":["../../src/__tests__/render.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAA;AAEvD,MAAM,UAAU,GAAG;;;;;;CAMlB,CAAA;AAED,MAAM,YAAY,GAAG,YAAY,CAAA;AAEjC,SAAS,IAAI,CAAC,IAAY,EAAE,EAAc;IACxC,IAAI,CAAC;QACH,EAAE,EAAE,CAAA;QACJ,OAAO,CAAC,GAAG,CAAC,SAAS,IAAI,EAAE,CAAC,CAAA;IAC9B,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACX,OAAO,CAAC,KAAK,CAAC,SAAS,IAAI,EAAE,EAAE,CAAC,CAAC,CAAA;QACjC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;IACjB,CAAC;AACH,CAAC;AAED,SAAS,MAAM,CAAC,MAAW;IACzB,OAAO;QACL,SAAS,CAAC,QAAgB;YACxB,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,QAAQ,CAAC;gBAAE,MAAM,IAAI,KAAK,CAAC,aAAa,MAAM,iBAAiB,QAAQ,GAAG,CAAC,CAAA;QAClG,CAAC;KACF,CAAA;AACH,CAAC;AAED,IAAI,CAAC,yBAAyB,EAAE,GAAG,EAAE;IACnC,MAAM,IAAI,GAAG,gBAAgB,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,YAAY,EAAE,IAAI,EAAE,UAAU,EAAE,CAAC,CAAA;IACxF,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,SAAS,CAAC,YAAY,CAAC,CAAA;IAC/C,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,SAAS,CAAC,iBAAiB,CAAC,CAAA;AACpD,CAAC,CAAC,CAAA"}
@@ -0,0 +1 @@
1
+ export declare function runHook(force?: boolean): Promise<void>;
@@ -0,0 +1,164 @@
1
+ import fs from 'node:fs';
2
+ import tty from 'node:tty';
3
+ import readline from 'node:readline';
4
+ import { runReviewFlow } from '../../core/review.js';
5
+ import { loadConfig } from '../../config/index.js';
6
+ import { setLanguage, t } from '../../locales/index.js';
7
+ import { intro, outro, confirm as clackConfirm, spinner, isCancel, cancel } from '@clack/prompts';
8
+ import picocolors from 'picocolors';
9
+ // Hack: Try to restore TTY for git hooks
10
+ // Removed unsafe global stdin hack
11
+ // We will handle TTY explicitly in safeConfirm
12
+ const CANCEL_SYMBOL = Symbol('cancel');
13
+ function isInteractive() {
14
+ return process.stdin.isTTY && process.stdout.isTTY;
15
+ }
16
+ async function safeConfirm(message) {
17
+ if (isInteractive()) {
18
+ return await clackConfirm({ message, initialValue: true, active: 'yes', inactive: 'no' });
19
+ }
20
+ if (!fs.existsSync('/dev/tty')) {
21
+ return false;
22
+ }
23
+ return new Promise((resolve) => {
24
+ const fd = fs.openSync('/dev/tty', 'r');
25
+ const input = new tty.ReadStream(fd);
26
+ const output = process.stdout;
27
+ readline.emitKeypressEvents(input);
28
+ input.setRawMode(true);
29
+ input.resume();
30
+ let value = true; // true = yes, false = no
31
+ let isDone = false;
32
+ const render = (first = false) => {
33
+ const prefix = picocolors.magenta('◆');
34
+ const bar = picocolors.dim('│');
35
+ const yesIcon = value ? picocolors.green('●') : picocolors.dim('○');
36
+ const noIcon = !value ? picocolors.green('●') : picocolors.dim('○');
37
+ const yesText = value ? 'yes' : picocolors.dim('yes');
38
+ const noText = !value ? 'no' : picocolors.dim('no');
39
+ const options = `${yesIcon} ${yesText} / ${noIcon} ${noText}`;
40
+ if (first) {
41
+ output.write(`${prefix} ${message}\n`);
42
+ output.write(`${bar} ${options}`);
43
+ }
44
+ else {
45
+ // Clear current line (options) and rewrite
46
+ // \r: move to start of line
47
+ // \x1b[K: clear line
48
+ output.write(`\r\x1b[K${bar} ${options}`);
49
+ }
50
+ };
51
+ const cleanup = () => {
52
+ isDone = true;
53
+ input.setRawMode(false);
54
+ input.destroy();
55
+ output.write('\n'); // End the line
56
+ };
57
+ const confirm = () => {
58
+ cleanup();
59
+ // Final output style: replace the prompt with a completed state
60
+ // Move up 2 lines (options + prompt)
61
+ // \x1b[1A: up 1 line
62
+ output.write('\x1b[1A\r\x1b[K'); // Clear options line
63
+ output.write('\x1b[1A\r\x1b[K'); // Clear prompt line
64
+ const prefix = picocolors.green('✔');
65
+ const text = picocolors.dim(message);
66
+ output.write(`${prefix} ${text}\n`); // Re-print simplified
67
+ resolve(value);
68
+ };
69
+ const cancelOp = () => {
70
+ cleanup();
71
+ output.write('\x1b[1A\r\x1b[K');
72
+ output.write('\x1b[1A\r\x1b[K');
73
+ const prefix = picocolors.red('✖');
74
+ const text = picocolors.dim(message);
75
+ output.write(`${prefix} ${text}\n`);
76
+ resolve(CANCEL_SYMBOL);
77
+ };
78
+ render(true);
79
+ input.on('keypress', (_, key) => {
80
+ if (isDone)
81
+ return;
82
+ if (key.name === 'return' || key.name === 'enter') {
83
+ confirm();
84
+ return;
85
+ }
86
+ if (key.ctrl && key.name === 'c') {
87
+ cancelOp();
88
+ return;
89
+ }
90
+ if (key.name === 'left' || key.name === 'h') {
91
+ value = true;
92
+ render();
93
+ }
94
+ else if (key.name === 'right' || key.name === 'l') {
95
+ value = false;
96
+ render();
97
+ }
98
+ else if (key.name === 'y') {
99
+ value = true;
100
+ render();
101
+ }
102
+ else if (key.name === 'n') {
103
+ value = false;
104
+ render();
105
+ }
106
+ });
107
+ });
108
+ }
109
+ function printBox(title, body) {
110
+ process.stdout.write('\n');
111
+ process.stdout.write(picocolors.cyan(` ➜ ${title}: `));
112
+ process.stdout.write(picocolors.underline(picocolors.white(body)));
113
+ process.stdout.write('\n\n');
114
+ }
115
+ export async function runHook(force = false) {
116
+ // Load config first to set language
117
+ const cfg = await loadConfig();
118
+ if (cfg.language) {
119
+ setLanguage(cfg.language);
120
+ }
121
+ const canPrompt = isInteractive() || fs.existsSync('/dev/tty');
122
+ if (!canPrompt && !force) {
123
+ process.stdout.write(t('cli.nonInteractive') + '\n');
124
+ process.exit(0);
125
+ return;
126
+ }
127
+ console.clear();
128
+ intro(picocolors.bgBlue(picocolors.white(t('cli.welcome'))));
129
+ const shouldReview = await safeConfirm(t('cli.confirmReview'));
130
+ if (shouldReview === CANCEL_SYMBOL || isCancel(shouldReview)) {
131
+ cancel(t('cli.opCancelled'));
132
+ process.exit(0);
133
+ }
134
+ if (!shouldReview) {
135
+ outro(t('cli.reviewSkipped'));
136
+ process.exit(0);
137
+ }
138
+ const s = spinner();
139
+ s.start(t('cli.initReview'));
140
+ let previewUrl = '';
141
+ const ok = await runReviewFlow({
142
+ onStart: (total) => {
143
+ s.message(t('cli.preparingReview', { total }));
144
+ },
145
+ onProgress: (file, idx, total) => {
146
+ s.message(t('cli.analyzing', { idx, total, file }));
147
+ },
148
+ onServerReady: (url) => {
149
+ previewUrl = url;
150
+ }
151
+ });
152
+ s.stop(t('cli.taskSubmitted'));
153
+ if (previewUrl) {
154
+ printBox(t('cli.previewUrl'), previewUrl);
155
+ }
156
+ const shouldCommit = await safeConfirm(t('cli.confirmCommit'));
157
+ if (shouldCommit === CANCEL_SYMBOL || isCancel(shouldCommit) || !shouldCommit) {
158
+ cancel(t('cli.commitCancelled'));
159
+ process.exit(1);
160
+ }
161
+ outro(picocolors.green(t('cli.commitConfirmed')));
162
+ process.exit(0);
163
+ }
164
+ //# sourceMappingURL=hook.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hook.js","sourceRoot":"","sources":["../../../src/cli/commands/hook.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export declare function runInit(method: string | undefined, genConfig: boolean, force?: boolean): Promise<void>;
@@ -0,0 +1,157 @@
1
+ import fs from 'node:fs';
2
+ import path from 'node:path';
3
+ import { spawnSync } from 'node:child_process';
4
+ import { confirm, intro, outro, log, select, isCancel, cancel } from '@clack/prompts';
5
+ import picocolors from 'picocolors';
6
+ function writeFileSafe(p, content) {
7
+ fs.mkdirSync(path.dirname(p), { recursive: true });
8
+ fs.writeFileSync(p, content, 'utf8');
9
+ }
10
+ function isGitRepo(cwd) {
11
+ const res = spawnSync('git', ['rev-parse', '--is-inside-work-tree'], { cwd });
12
+ return res.status === 0;
13
+ }
14
+ const HOOK_SCRIPT = `./node_modules/.bin/code-gate-hook`;
15
+ function initGitHooks(cwd) {
16
+ const hooksDir = path.join(cwd, '.githooks');
17
+ const preCommit = path.join(hooksDir, 'pre-commit');
18
+ ensurePreCommitContains(preCommit, HOOK_SCRIPT, 'git');
19
+ spawnSync('git', ['config', 'core.hooksPath', '.githooks'], { cwd, stdio: 'inherit' });
20
+ process.stdout.write('code-gate: initialized with native git hooks (.githooks)\n');
21
+ }
22
+ function ensurePreCommitContains(p, line, type) {
23
+ if (fs.existsSync(p)) {
24
+ const content = fs.readFileSync(p, 'utf8');
25
+ if (!content.includes(line)) {
26
+ fs.writeFileSync(p, content.trimEnd() + '\n' + line + '\n', 'utf8');
27
+ }
28
+ return;
29
+ }
30
+ if (type === 'husky') {
31
+ const content = [
32
+ '#!/usr/bin/env sh',
33
+ '. "$(dirname -- "$0")/_/husky.sh"',
34
+ line
35
+ ].join('\n') + '\n';
36
+ writeFileSafe(p, content);
37
+ fs.chmodSync(p, 0o755);
38
+ return;
39
+ }
40
+ const content = ['#!/usr/bin/env sh', line].join('\n') + '\n';
41
+ writeFileSafe(p, content);
42
+ fs.chmodSync(p, 0o755);
43
+ }
44
+ function initHusky(cwd) {
45
+ const huskyDir = path.join(cwd, '.husky');
46
+ if (!fs.existsSync(huskyDir)) {
47
+ spawnSync('npx', ['husky', 'init'], { cwd, stdio: 'inherit' });
48
+ }
49
+ const preCommit = path.join(huskyDir, 'pre-commit');
50
+ ensurePreCommitContains(preCommit, HOOK_SCRIPT, 'husky');
51
+ spawnSync('git', ['config', 'core.hooksPath', '.husky'], { cwd, stdio: 'inherit' });
52
+ process.stdout.write('code-gate: initialized with husky (.husky)\n');
53
+ }
54
+ function generateConfig(cwd, force = false) {
55
+ const configPath = path.join(cwd, '.codegate.js');
56
+ if (fs.existsSync(configPath) && !force)
57
+ return;
58
+ const content = `// code-gate config
59
+ export default {
60
+ provider: 'ollama',
61
+ providerOptions: {
62
+ ollama: {
63
+ baseURL: 'http://localhost:11434',
64
+ model: 'qwen2.5-coder',
65
+ concurrencyFiles: 1
66
+ },
67
+ deepseek: {
68
+ baseURL: 'https://api.deepseek.com',
69
+ apiKeyEnv: 'DEEPSEEK_API_KEY',
70
+ model: 'deepseek-chat',
71
+ concurrencyFiles: 4
72
+ }
73
+ // openai: { baseURL: 'https://api.openai.com/v1', apiKeyEnv: 'OPENAI_API_KEY', model: 'gpt-4o-mini' },
74
+ // anthropic: { baseURL: 'https://api.anthropic.com', apiKeyEnv: 'ANTHROPIC_API_KEY', model: 'claude-3-5-sonnet' },
75
+ // azureOpenAI: { endpoint: 'https://your-endpoint.openai.azure.com', apiKeyEnv: 'AZURE_OPENAI_KEY', deployment: 'gpt-4o-mini', apiVersion: '2024-08-01-preview' }
76
+ // aliyun: { baseURL: 'https://dashscope.aliyuncs.com/compatible-mode/v1', apiKeyEnv: 'DASHSCOPE_API_KEY', model: 'qwen-plus' },
77
+ // volcengine: { baseURL: 'https://ark.cn-beijing.volces.com/api/v3', apiKeyEnv: 'VOLCENGINE_API_KEY', model: 'doubao-pro-32k' },
78
+ // zhipu: { baseURL: 'https://open.bigmodel.cn/api/paas/v4', apiKeyEnv: 'ZHIPU_API_KEY', model: 'glm-4' }
79
+ },
80
+ fileTypes: [],
81
+ exclude: ['**/package-lock.json', '**/yarn.lock', '**/pnpm-lock.yaml'],
82
+ ui: {
83
+ openBrowser: true,
84
+ port: 5175
85
+ },
86
+ limits: {
87
+ maxDiffLines: 10000,
88
+ maxFiles: 100
89
+ },
90
+ reviewMode: 'files',
91
+ language: 'en',
92
+ prompt: 'As a senior code review engineer, review this change from the perspectives of security, performance, code style, and test coverage. Point out issues and suggestions for improvement, and provide necessary example patches.',
93
+ output: {
94
+ dir: '.review-logs'
95
+ }
96
+ }
97
+ `;
98
+ fs.writeFileSync(configPath, content, 'utf8');
99
+ process.stdout.write('code-gate: generated .codegate.js\n');
100
+ }
101
+ async function checkAndAddToGitignore(cwd) {
102
+ const ignoreList = ['.codegate.js', '.review-logs'];
103
+ const gitignorePath = path.join(cwd, '.gitignore');
104
+ // Filter out already ignored
105
+ let content = '';
106
+ if (fs.existsSync(gitignorePath)) {
107
+ content = fs.readFileSync(gitignorePath, 'utf8');
108
+ }
109
+ const toAdd = ignoreList.filter(item => !content.includes(item));
110
+ if (toAdd.length === 0)
111
+ return;
112
+ log.info(picocolors.yellow('Detected the following files suggested to be added to .gitignore:'));
113
+ for (const item of toAdd) {
114
+ console.log(picocolors.dim(` - ${item}`));
115
+ }
116
+ const shouldIgnore = await confirm({
117
+ message: 'Add these files to .gitignore?',
118
+ initialValue: true
119
+ });
120
+ if (shouldIgnore === true) {
121
+ const append = (content.endsWith('\n') || content === '') ? '' : '\n';
122
+ fs.appendFileSync(gitignorePath, `${append}${toAdd.join('\n')}\n`, 'utf8');
123
+ log.success(picocolors.green('Updated .gitignore'));
124
+ }
125
+ }
126
+ export async function runInit(method, genConfig, force = false) {
127
+ const cwd = process.cwd();
128
+ intro(picocolors.bgBlue(picocolors.white(' Code Gate Init ')));
129
+ if (!isGitRepo(cwd)) {
130
+ log.error('not a git repository');
131
+ process.exit(1);
132
+ return;
133
+ }
134
+ if (!method) {
135
+ const selected = await select({
136
+ message: 'Select hook method:',
137
+ options: [
138
+ { value: 'git', label: 'Native Git Hooks', hint: 'recommended' },
139
+ { value: 'husky', label: 'Husky' }
140
+ ]
141
+ });
142
+ if (isCancel(selected)) {
143
+ cancel('Operation cancelled.');
144
+ process.exit(0);
145
+ }
146
+ method = selected;
147
+ }
148
+ if (method === 'husky')
149
+ initHusky(cwd);
150
+ else
151
+ initGitHooks(cwd);
152
+ if (genConfig)
153
+ generateConfig(cwd, force);
154
+ await checkAndAddToGitignore(cwd);
155
+ outro(picocolors.green('Initialization completed!'));
156
+ }
157
+ //# sourceMappingURL=init.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"init.js","sourceRoot":"","sources":["../../../src/cli/commands/init.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export declare function runSetup(): Promise<void>;
@@ -0,0 +1,19 @@
1
+ import fs from 'node:fs';
2
+ import path from 'node:path';
3
+ import { spawnSync } from 'node:child_process';
4
+ function writeFileSafe(p, content) {
5
+ fs.mkdirSync(path.dirname(p), { recursive: true });
6
+ fs.writeFileSync(p, content, 'utf8');
7
+ }
8
+ export async function runSetup() {
9
+ const cwd = process.cwd();
10
+ const hooksDir = path.join(cwd, '.githooks');
11
+ const preCommit = path.join(hooksDir, 'pre-commit');
12
+ const binPath = 'npx code-gate hook';
13
+ const script = `#!/usr/bin/env sh\n${binPath}\n`;
14
+ writeFileSafe(preCommit, script);
15
+ fs.chmodSync(preCommit, 0o755);
16
+ spawnSync('git', ['config', 'core.hooksPath', '.githooks'], { stdio: 'inherit' });
17
+ process.stdout.write('code-gate installed: .githooks/pre-commit\n');
18
+ }
19
+ //# sourceMappingURL=setup.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"setup.js","sourceRoot":"","sources":["../../../src/cli/commands/setup.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,SAAS,CAAA;AACxB,OAAO,IAAI,MAAM,WAAW,CAAA;AAE5B,OAAO,EAAE,SAAS,EAAE,MAAM,oBAAoB,CAAA;AAE9C,SAAS,aAAa,CAAC,CAAS,EAAE,OAAe;IAC/C,EAAE,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAA;IAClD,EAAE,CAAC,aAAa,CAAC,CAAC,EAAE,OAAO,EAAE,MAAM,CAAC,CAAA;AACtC,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,QAAQ;IAC5B,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,EAAE,CAAA;IACzB,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,WAAW,CAAC,CAAA;IAC5C,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,YAAY,CAAC,CAAA;IACnD,MAAM,OAAO,GAAG,oBAAoB,CAAA;IACpC,MAAM,MAAM,GAAG,sBAAsB,OAAO,IAAI,CAAA;IAChD,aAAa,CAAC,SAAS,EAAE,MAAM,CAAC,CAAA;IAChC,EAAE,CAAC,SAAS,CAAC,SAAS,EAAE,KAAK,CAAC,CAAA;IAC9B,SAAS,CAAC,KAAK,EAAE,CAAC,QAAQ,EAAE,gBAAgB,EAAE,WAAW,CAAC,EAAE,EAAE,KAAK,EAAE,SAAS,EAAE,CAAC,CAAA;IACjF,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,6CAA6C,CAAC,CAAA;AACrE,CAAC"}
@@ -0,0 +1,2 @@
1
+ import 'dotenv/config';
2
+ export declare function run(): Promise<void>;
@@ -0,0 +1,26 @@
1
+ import 'dotenv/config';
2
+ import { Command } from 'commander';
3
+ import { runHook } from './commands/hook.js';
4
+ import { runInit } from './commands/init.js';
5
+ export async function run() {
6
+ const program = new Command();
7
+ program.name('code-gate').description('AI commit review tool').version('1.0.0');
8
+ program
9
+ .command('init')
10
+ .description('Initialize integration and generate config')
11
+ .option('-m, --method <method>', 'init method: git|husky')
12
+ .option('-f, --force', 'force overwrite/append')
13
+ .option('--no-config', 'do not generate config file')
14
+ .action(async (opts) => {
15
+ await runInit(opts.method, !!opts.config, !!opts.force);
16
+ });
17
+ program
18
+ .command('hook')
19
+ .description('Run interactive pre-commit review')
20
+ .option('-f, --force', 'force review even when non-interactive', false)
21
+ .action(async (opts) => {
22
+ await runHook(!!opts.force);
23
+ });
24
+ program.parseAsync(process.argv);
25
+ }
26
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/cli/index.ts"],"names":[],"mappings":"AAAA,OAAO,eAAe,CAAA;AACtB,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAA;AACnC,OAAO,EAAE,OAAO,EAAE,MAAM,oBAAoB,CAAA;AAC5C,OAAO,EAAE,OAAO,EAAE,MAAM,oBAAoB,CAAA;AAE5C,MAAM,CAAC,KAAK,UAAU,GAAG;IACvB,MAAM,OAAO,GAAG,IAAI,OAAO,EAAE,CAAA;IAC7B,OAAO,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,WAAW,CAAC,uBAAuB,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,CAAA;IAE/E,OAAO;SACJ,OAAO,CAAC,MAAM,CAAC;SACf,WAAW,CAAC,4CAA4C,CAAC;SACzD,MAAM,CAAC,uBAAuB,EAAE,wBAAwB,CAAC;SACzD,MAAM,CAAC,aAAa,EAAE,wBAAwB,CAAC;SAC/C,MAAM,CAAC,aAAa,EAAE,6BAA6B,CAAC;SACpD,MAAM,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE;QACrB,MAAM,OAAO,CAAC,IAAI,CAAC,MAAgB,EAAE,CAAC,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;IACnE,CAAC,CAAC,CAAA;IAEJ,OAAO;SACJ,OAAO,CAAC,MAAM,CAAC;SACf,WAAW,CAAC,mCAAmC,CAAC;SAChD,MAAM,CAAC,aAAa,EAAE,wCAAwC,EAAE,KAAK,CAAC;SACtE,MAAM,CAAC,KAAK,EAAE,IAAI,EAAE,EAAE;QACrB,MAAM,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;IAC7B,CAAC,CAAC,CAAA;IAEJ,OAAO,CAAC,UAAU,CAAC,OAAO,CAAC,IAAI,CAAC,CAAA;AAClC,CAAC"}
@@ -10,6 +10,13 @@ function askYesNo(question) {
10
10
  const rl = readline.createInterface({ input, output: process.stdout });
11
11
  rl.question(question, (answer) => {
12
12
  rl.close();
13
+ if (input !== process.stdin && typeof input?.close === 'function') {
14
+ try {
15
+ ;
16
+ input.close();
17
+ }
18
+ catch { }
19
+ }
13
20
  resolve(answer.trim().toLowerCase() === 'y');
14
21
  });
15
22
  });
@@ -1 +1 @@
1
- {"version":3,"file":"hook.js","sourceRoot":"","sources":["../../src/commands/hook.ts"],"names":[],"mappings":"AAAA,OAAO,QAAQ,MAAM,eAAe,CAAA;AACpC,OAAO,EAAE,MAAM,SAAS,CAAA;AACxB,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAA;AAEtD,SAAS,aAAa;IACpB,OAAO,OAAO,CAAC,KAAK,CAAC,KAAK,IAAI,OAAO,CAAC,MAAM,CAAC,KAAK,CAAA;AACpD,CAAC;AAED,SAAS,QAAQ,CAAC,QAAgB;IAChC,MAAM,KAAK,GACT,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,gBAAgB,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,CAAA;IACrH,OAAO,IAAI,OAAO,CAAU,CAAC,OAAO,EAAE,EAAE;QACtC,MAAM,EAAE,GAAG,QAAQ,CAAC,eAAe,CAAC,EAAE,KAAK,EAAE,MAAM,EAAE,OAAO,CAAC,MAAM,EAAE,CAAC,CAAA;QACtE,EAAE,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC,MAAM,EAAE,EAAE;YAC/B,EAAE,CAAC,KAAK,EAAE,CAAA;YACV,OAAO,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC,WAAW,EAAE,KAAK,GAAG,CAAC,CAAA;QAC9C,CAAC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;AACJ,CAAC;AAED,SAAS,aAAa,CAAC,KAAa;IAClC,IAAI,CAAC,GAAG,CAAC,CAAA;IACT,MAAM,KAAK,GAAG,CAAC,GAAG,EAAE,IAAI,EAAE,KAAK,EAAE,MAAM,CAAC,CAAA;IACxC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,KAAK,EAAE,CAAC,CAAA;IAChC,MAAM,KAAK,GAAG,WAAW,CAAC,GAAG,EAAE;QAC7B,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,KAAK,CAAC,MAAM,CAAA;QAC1B,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC,CAAA;IACrC,CAAC,EAAE,GAAG,CAAC,CAAA;IACP,OAAO,GAAG,EAAE;QACV,aAAa,CAAC,KAAK,CAAC,CAAA;QACpB,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAA;IAC5B,CAAC,CAAA;AACH,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,OAAO,CAAC,KAAK,GAAG,KAAK;IACzC,MAAM,SAAS,GAAG,aAAa,EAAE,IAAI,EAAE,CAAC,UAAU,CAAC,UAAU,CAAC,CAAA;IAC9D,IAAI,CAAC,SAAS,IAAI,CAAC,KAAK,EAAE,CAAC;QACzB,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,2DAA2D,CAAC,CAAA;QACjF,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;QACf,OAAM;IACR,CAAC;IACD,MAAM,GAAG,GAAG,MAAM,QAAQ,CAAC,wHAAwH,CAAC,CAAA;IACpJ,IAAI,CAAC,GAAG,EAAE,CAAC;QACT,OAAO,CAAC,KAAK,CAAC,KAAK,EAAE,CAAA;QACrB,YAAY,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAA;QACnC,OAAM;IACR,CAAC;IACD,MAAM,IAAI,GAAG,aAAa,CAAC,YAAY,CAAC,CAAA;IACxC,MAAM,EAAE,GAAG,MAAM,aAAa,EAAE,CAAA;IAChC,IAAI,EAAE,CAAA;IACN,IAAI,CAAC,EAAE,EAAE,CAAC;QACR,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,qHAAqH,CAAC,CAAA;QAClJ,IAAI,IAAI,EAAE,CAAC;YACT,OAAO,CAAC,KAAK,CAAC,KAAK,EAAE,CAAA;YACrB,YAAY,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAA;QACrC,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAA;YAC/B,OAAO,CAAC,KAAK,CAAC,KAAK,EAAE,CAAA;YACrB,YAAY,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAA;QACrC,CAAC;IACH,CAAC;SAAM,CAAC;QACN,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,qHAAqH,CAAC,CAAA;QAClJ,IAAI,IAAI,EAAE,CAAC;YACT,OAAO,CAAC,KAAK,CAAC,KAAK,EAAE,CAAA;YACrB,YAAY,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAA;QACrC,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAA;YAC/B,OAAO,CAAC,KAAK,CAAC,KAAK,EAAE,CAAA;YACrB,YAAY,CAAC,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAA;QACrC,CAAC;IACH,CAAC;AACH,CAAC"}
1
+ {"version":3,"file":"hook.js","sourceRoot":"","sources":["../../src/commands/hook.ts"],"names":[],"mappings":"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"}
@@ -68,41 +68,41 @@ function generateConfig(cwd, force = false) {
68
68
  return;
69
69
  const content = `// code-gate 配置文件
70
70
  // provider: 选择使用的 AI 审查引擎,可选值: 'ollama' | 'deepseek'
71
- // review: 审查相关设置
72
- // - enabled: 是否启用审查
73
- // - provider: 覆盖顶层 provider,支持按需切换,可选值: 'ollama' | 'deepseek'
74
- // - mode: 审查展示模式,可选值: 'aggregate'(整合输出)| 'per_file'(按文件 Tab 展示)
75
- // ollama: 本地 Ollama 设置
76
- // - baseURL: Ollama 服务地址,默认 'http://localhost:11434'
77
- // - model: 模型名称,例如 'qwen3:8b'、'deepseek-r1:14b'
78
- // deepseek: DeepSeek 云端设置
79
- // - baseURL: API 地址,默认 'https://api.deepseek.com'
80
- // - apiKeyEnv: 存放密钥的环境变量名,默认 'DEEPSEEK_API_KEY'
81
- // - model: 模型名称,默认 'deepseek-chat'
71
+ // providerOptions: 各 Provider 的配置集合(选填)
72
+ // - deepseek: { baseURL, apiKeyEnv, model }
73
+ // - ollama: { baseURL, model }
74
+ // - openai: { baseURL, apiKeyEnv, model }
75
+ // - anthropic: { baseURL, apiKeyEnv, model }
76
+ // - gemini: { baseURL, apiKeyEnv, model }
77
+ // - cohere: { baseURL, apiKeyEnv, model }
78
+ // - mistral: { baseURL, apiKeyEnv, model }
79
+ // - azureOpenAI: { endpoint, apiKeyEnv, deployment, apiVersion }
82
80
  // fileTypes: 需要审查的文件类型扩展名列表
83
- // scope: 审查范围,可选值: 'staged' | 'allChanged' | { include?: string[], exclude?: string[] }
84
81
  // ui: 页面与交互设置
85
82
  // - openBrowser: 是否自动打开浏览器
86
- // - theme: 主题,目前仅 'github'
87
83
  // - port: 预览服务端口
84
+ // 并发:providerOptions.<provider>.concurrencyFiles 控制按文件的并发审查(默认 DeepSeek=4、Ollama=1,最大 8)
88
85
  // limits: 限制项
89
86
  // - maxDiffLines: 最大 diff 行数
90
87
  // - maxFiles: 最大审查文件数
91
- // rules: 审查关注点,可选值: 'security' | 'performance' | 'style' | 'tests'
92
88
  // prompt: 通用提示词
89
+ // 行为:reviewMode 审查模式:'summary' | 'files' | 'both'(默认 'files';'both' 时汇总最后执行)
93
90
  // output: 输出目录配置
94
91
  // - dir: 本地输出目录
95
92
 
96
93
  export default {
97
94
  provider: 'deepseek',
98
- review: { enabled: true, mode: 'per_file' },
99
- deepseek: { baseURL: 'https://api.deepseek.com', apiKeyEnv: 'DEEPSEEK_API_KEY', model: 'deepseek-chat' },
100
- ollama: { baseURL: 'http://localhost:11434', model: 'qwen3:8b' },
95
+ providerOptions: {
96
+ deepseek: { baseURL: 'https://api.deepseek.com', apiKeyEnv: 'DEEPSEEK_API_KEY', model: 'deepseek-chat', concurrencyFiles: 4, request: { retries: 1, backoffMs: 300 } },
97
+ ollama: { baseURL: 'http://localhost:11434', model: 'qwen3:8b', concurrencyFiles: 1, request: { timeout: 15000, retries: 1, backoffMs: 300 } }
98
+ // openai: { baseURL: 'https://api.openai.com/v1', apiKeyEnv: 'OPENAI_API_KEY', model: 'gpt-4o-mini' },
99
+ // anthropic: { baseURL: 'https://api.anthropic.com', apiKeyEnv: 'ANTHROPIC_API_KEY', model: 'claude-3-5-sonnet' },
100
+ // azureOpenAI: { endpoint: 'https://your-endpoint.openai.azure.com', apiKeyEnv: 'AZURE_OPENAI_KEY', deployment: 'gpt-4o-mini', apiVersion: '2024-08-01-preview' }
101
+ },
101
102
  fileTypes: ['ts', 'tsx', 'js', 'jsx', 'json', 'md', 'py', 'go', 'rs'],
102
- scope: 'staged',
103
- ui: { openBrowser: true, theme: 'github', port: 5175 },
103
+ ui: { openBrowser: true, port: 5175 },
104
104
  limits: { maxDiffLines: 10000, maxFiles: 100 },
105
- rules: { focus: ['security', 'performance', 'style', 'tests'] },
105
+ reviewMode: 'files',
106
106
  prompt: '作为资深代码审查工程师,从安全、性能、代码风格与测试覆盖角度审查本次变更,指出问题与改进建议,并给出必要的示例补丁。',
107
107
  output: { dir: '.code-gate' }
108
108
  }
@@ -0,0 +1,2 @@
1
+ import { Config } from './types.js';
2
+ export declare const defaultConfig: Config;
@@ -0,0 +1,33 @@
1
+ export const defaultConfig = {
2
+ provider: 'ollama',
3
+ providerOptions: {
4
+ deepseek: {
5
+ baseURL: 'https://api.deepseek.com',
6
+ apiKeyEnv: 'DEEPSEEK_API_KEY',
7
+ model: 'deepseek-chat',
8
+ concurrencyFiles: 4
9
+ },
10
+ ollama: {
11
+ baseURL: 'http://localhost:11434',
12
+ model: 'qwen2.5-coder',
13
+ concurrencyFiles: 1
14
+ }
15
+ },
16
+ fileTypes: [],
17
+ exclude: ['**/package-lock.json', '**/yarn.lock', '**/pnpm-lock.yaml'],
18
+ ui: {
19
+ openBrowser: true,
20
+ port: 5175
21
+ },
22
+ limits: {
23
+ maxDiffLines: 10000,
24
+ maxFiles: 100
25
+ },
26
+ reviewMode: 'files',
27
+ language: 'en',
28
+ prompt: 'As a senior code review engineer, review this change from the perspectives of security, performance, code style, and test coverage. Point out issues and suggestions for improvement, and provide necessary example patches.',
29
+ output: {
30
+ dir: '.review-logs'
31
+ }
32
+ };
33
+ //# sourceMappingURL=defaults.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"defaults.js","sourceRoot":"","sources":["../../src/config/defaults.ts"],"names":[],"mappings":"AAEA,MAAM,CAAC,MAAM,aAAa,GAAW;IACnC,QAAQ,EAAE,QAAQ;IAClB,eAAe,EAAE;QACf,QAAQ,EAAE;YACR,OAAO,EAAE,0BAA0B;YACnC,SAAS,EAAE,kBAAkB;YAC7B,KAAK,EAAE,eAAe;YACtB,gBAAgB,EAAE,CAAC;SACpB;QACD,MAAM,EAAE;YACN,OAAO,EAAE,wBAAwB;YACjC,KAAK,EAAE,eAAe;YACtB,gBAAgB,EAAE,CAAC;SACpB;KACF;IACD,SAAS,EAAE,EAAE;IACb,OAAO,EAAE,CAAC,sBAAsB,EAAE,cAAc,EAAE,mBAAmB,CAAC;IACtE,EAAE,EAAE;QACF,WAAW,EAAE,IAAI;QACjB,IAAI,EAAE,IAAI;KACX;IACD,MAAM,EAAE;QACN,YAAY,EAAE,KAAK;QACnB,QAAQ,EAAE,GAAG;KACd;IACD,UAAU,EAAE,OAAO;IACnB,QAAQ,EAAE,IAAI;IACd,MAAM,EACJ,8NAA8N;IAChO,MAAM,EAAE;QACN,GAAG,EAAE,cAAc;KACpB;CACF,CAAA"}
@@ -0,0 +1,4 @@
1
+ import { Config } from './types.js';
2
+ export * from './types.js';
3
+ export * from './defaults.js';
4
+ export declare function loadConfig(cwd?: string): Promise<Config>;