@kodus/cli 0.2.3 → 0.3.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.
- package/README.md +78 -27
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +20 -3
- package/dist/cli.js.map +1 -1
- package/dist/commands/auth/index.d.ts.map +1 -1
- package/dist/commands/auth/index.js +1 -2
- package/dist/commands/auth/index.js.map +1 -1
- package/dist/commands/auth/login.d.ts.map +1 -1
- package/dist/commands/auth/login.js +44 -45
- package/dist/commands/auth/login.js.map +1 -1
- package/dist/commands/auth/logout.d.ts.map +1 -1
- package/dist/commands/auth/logout.js +5 -3
- package/dist/commands/auth/logout.js.map +1 -1
- package/dist/commands/auth/status.d.ts.map +1 -1
- package/dist/commands/auth/status.js +28 -25
- package/dist/commands/auth/status.js.map +1 -1
- package/dist/commands/auth/team-key.d.ts.map +1 -1
- package/dist/commands/auth/team-key.js +49 -29
- package/dist/commands/auth/team-key.js.map +1 -1
- package/dist/commands/auth/token.d.ts.map +1 -1
- package/dist/commands/auth/token.js +12 -10
- package/dist/commands/auth/token.js.map +1 -1
- package/dist/commands/hook/index.d.ts.map +1 -1
- package/dist/commands/hook/index.js +1 -2
- package/dist/commands/hook/index.js.map +1 -1
- package/dist/commands/hook/install.d.ts.map +1 -1
- package/dist/commands/hook/install.js +21 -22
- package/dist/commands/hook/install.js.map +1 -1
- package/dist/commands/hook/status.d.ts.map +1 -1
- package/dist/commands/hook/status.js +10 -8
- package/dist/commands/hook/status.js.map +1 -1
- package/dist/commands/hook/uninstall.d.ts.map +1 -1
- package/dist/commands/hook/uninstall.js +7 -5
- package/dist/commands/hook/uninstall.js.map +1 -1
- package/dist/commands/memory/capture.d.ts.map +1 -1
- package/dist/commands/memory/capture.js +28 -14
- package/dist/commands/memory/capture.js.map +1 -1
- package/dist/commands/memory/disable.d.ts.map +1 -1
- package/dist/commands/memory/disable.js +9 -7
- package/dist/commands/memory/disable.js.map +1 -1
- package/dist/commands/memory/enable.d.ts.map +1 -1
- package/dist/commands/memory/enable.js +18 -14
- package/dist/commands/memory/enable.js.map +1 -1
- package/dist/commands/memory/hooks.d.ts.map +1 -1
- package/dist/commands/memory/hooks.js +37 -17
- package/dist/commands/memory/hooks.js.map +1 -1
- package/dist/commands/memory/index.d.ts.map +1 -1
- package/dist/commands/memory/index.js +1 -2
- package/dist/commands/memory/index.js.map +1 -1
- package/dist/commands/memory/promote.d.ts.map +1 -1
- package/dist/commands/memory/promote.js +14 -10
- package/dist/commands/memory/promote.js.map +1 -1
- package/dist/commands/memory/show.d.ts.map +1 -1
- package/dist/commands/memory/show.js +11 -10
- package/dist/commands/memory/show.js.map +1 -1
- package/dist/commands/memory/status.d.ts.map +1 -1
- package/dist/commands/memory/status.js +22 -20
- package/dist/commands/memory/status.js.map +1 -1
- package/dist/commands/pr.d.ts.map +1 -1
- package/dist/commands/pr.js +131 -13
- package/dist/commands/pr.js.map +1 -1
- package/dist/commands/review.d.ts.map +1 -1
- package/dist/commands/review.js +57 -38
- package/dist/commands/review.js.map +1 -1
- package/dist/commands/status.d.ts +3 -0
- package/dist/commands/status.d.ts.map +1 -0
- package/dist/commands/status.js +119 -0
- package/dist/commands/status.js.map +1 -0
- package/dist/commands/subscribe.d.ts +4 -0
- package/dist/commands/subscribe.d.ts.map +1 -0
- package/dist/commands/subscribe.js +25 -0
- package/dist/commands/subscribe.js.map +1 -0
- package/dist/commands/update.d.ts +9 -0
- package/dist/commands/update.d.ts.map +1 -0
- package/dist/commands/update.js +74 -0
- package/dist/commands/update.js.map +1 -0
- package/dist/formatters/json.d.ts.map +1 -1
- package/dist/formatters/json.js.map +1 -1
- package/dist/formatters/markdown.d.ts.map +1 -1
- package/dist/formatters/markdown.js.map +1 -1
- package/dist/formatters/prompt.d.ts.map +1 -1
- package/dist/formatters/prompt.js +6 -6
- package/dist/formatters/prompt.js.map +1 -1
- package/dist/formatters/terminal.d.ts.map +1 -1
- package/dist/formatters/terminal.js +7 -4
- package/dist/formatters/terminal.js.map +1 -1
- package/dist/index.js +28 -5
- package/dist/index.js.map +1 -1
- package/dist/services/api/api.interface.d.ts +10 -1
- package/dist/services/api/api.interface.d.ts.map +1 -1
- package/dist/services/api/api.real.d.ts +14 -0
- package/dist/services/api/api.real.d.ts.map +1 -1
- package/dist/services/api/api.real.js +163 -58
- package/dist/services/api/api.real.js.map +1 -1
- package/dist/services/api/index.d.ts +2 -2
- package/dist/services/api/index.js +1 -1
- package/dist/services/auth.service.d.ts +1 -0
- package/dist/services/auth.service.d.ts.map +1 -1
- package/dist/services/auth.service.js +21 -2
- package/dist/services/auth.service.js.map +1 -1
- package/dist/services/context.service.d.ts +4 -1
- package/dist/services/context.service.d.ts.map +1 -1
- package/dist/services/context.service.js +30 -19
- package/dist/services/context.service.js.map +1 -1
- package/dist/services/fix.service.d.ts.map +1 -1
- package/dist/services/fix.service.js +3 -2
- package/dist/services/fix.service.js.map +1 -1
- package/dist/services/git.service.d.ts +1 -0
- package/dist/services/git.service.d.ts.map +1 -1
- package/dist/services/git.service.js +123 -48
- package/dist/services/git.service.js.map +1 -1
- package/dist/services/memory.service.d.ts.map +1 -1
- package/dist/services/memory.service.js +93 -35
- package/dist/services/memory.service.js.map +1 -1
- package/dist/services/review.service.d.ts +12 -1
- package/dist/services/review.service.d.ts.map +1 -1
- package/dist/services/review.service.js +88 -65
- package/dist/services/review.service.js.map +1 -1
- package/dist/services/transcript-parser.service.d.ts.map +1 -1
- package/dist/services/transcript-parser.service.js +46 -15
- package/dist/services/transcript-parser.service.js.map +1 -1
- package/dist/types/index.d.ts +12 -1
- package/dist/types/index.d.ts.map +1 -1
- package/dist/types/index.js.map +1 -1
- package/dist/types/memory.d.ts.map +1 -1
- package/dist/ui/interactive.d.ts.map +1 -1
- package/dist/ui/interactive.js +86 -92
- package/dist/ui/interactive.js.map +1 -1
- package/dist/utils/banner.d.ts +2 -0
- package/dist/utils/banner.d.ts.map +1 -0
- package/dist/utils/banner.js +190 -0
- package/dist/utils/banner.js.map +1 -0
- package/dist/utils/cli-exit.d.ts +13 -0
- package/dist/utils/cli-exit.d.ts.map +1 -0
- package/dist/utils/cli-exit.js +29 -0
- package/dist/utils/cli-exit.js.map +1 -0
- package/dist/utils/clipboard.d.ts +2 -0
- package/dist/utils/clipboard.d.ts.map +1 -0
- package/dist/utils/clipboard.js +11 -0
- package/dist/utils/clipboard.js.map +1 -0
- package/dist/utils/config.d.ts +3 -0
- package/dist/utils/config.d.ts.map +1 -1
- package/dist/utils/config.js.map +1 -1
- package/dist/utils/credentials.d.ts.map +1 -1
- package/dist/utils/credentials.js.map +1 -1
- package/dist/utils/device.d.ts.map +1 -1
- package/dist/utils/device.js +17 -6
- package/dist/utils/device.js.map +1 -1
- package/dist/utils/logger.d.ts +13 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +48 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/utils/module-matcher.d.ts.map +1 -1
- package/dist/utils/module-matcher.js +12 -7
- package/dist/utils/module-matcher.js.map +1 -1
- package/dist/utils/rate-limit.d.ts.map +1 -1
- package/dist/utils/rate-limit.js +12 -3
- package/dist/utils/rate-limit.js.map +1 -1
- package/dist/utils/recent-activity.d.ts +9 -0
- package/dist/utils/recent-activity.d.ts.map +1 -0
- package/dist/utils/recent-activity.js +139 -0
- package/dist/utils/recent-activity.js.map +1 -0
- package/dist/utils/skills.d.ts +2 -0
- package/dist/utils/skills.d.ts.map +1 -0
- package/dist/utils/skills.js +40 -0
- package/dist/utils/skills.js.map +1 -0
- package/dist/utils/update-check.d.ts +1 -2
- package/dist/utils/update-check.d.ts.map +1 -1
- package/dist/utils/update-check.js +21 -97
- package/dist/utils/update-check.js.map +1 -1
- package/package.json +83 -59
- package/skills/README.md +49 -0
- package/skills/business-rules-validation/SKILL.md +43 -0
- package/skills/kodus-pr-suggestions-resolver/SKILL.md +61 -0
- package/skills/kodus-review/SKILL.md +40 -0
- package/skills/kodus-review-dev/SKILL.md +41 -0
- package/skills/kodus-review-dev/scripts/run-local-cli.sh +49 -0
- package/dist/commands/__tests__/auth.login-logout.test.d.ts +0 -2
- package/dist/commands/__tests__/auth.login-logout.test.d.ts.map +0 -1
- package/dist/commands/__tests__/auth.login-logout.test.js +0 -125
- package/dist/commands/__tests__/auth.login-logout.test.js.map +0 -1
- package/dist/commands/__tests__/auth.team-key.test.d.ts +0 -2
- package/dist/commands/__tests__/auth.team-key.test.d.ts.map +0 -1
- package/dist/commands/__tests__/auth.team-key.test.js +0 -145
- package/dist/commands/__tests__/auth.team-key.test.js.map +0 -1
- package/dist/commands/__tests__/capture-api.test.d.ts +0 -2
- package/dist/commands/__tests__/capture-api.test.d.ts.map +0 -1
- package/dist/commands/__tests__/capture-api.test.js +0 -88
- package/dist/commands/__tests__/capture-api.test.js.map +0 -1
- package/dist/commands/__tests__/disable.test.d.ts +0 -2
- package/dist/commands/__tests__/disable.test.d.ts.map +0 -1
- package/dist/commands/__tests__/disable.test.js +0 -103
- package/dist/commands/__tests__/disable.test.js.map +0 -1
- package/dist/commands/__tests__/enable.test.d.ts +0 -2
- package/dist/commands/__tests__/enable.test.d.ts.map +0 -1
- package/dist/commands/__tests__/enable.test.js +0 -103
- package/dist/commands/__tests__/enable.test.js.map +0 -1
- package/dist/commands/__tests__/hook.test.d.ts +0 -2
- package/dist/commands/__tests__/hook.test.d.ts.map +0 -1
- package/dist/commands/__tests__/hook.test.js +0 -105
- package/dist/commands/__tests__/hook.test.js.map +0 -1
- package/dist/commands/__tests__/install-merge-hook.test.d.ts +0 -2
- package/dist/commands/__tests__/install-merge-hook.test.d.ts.map +0 -1
- package/dist/commands/__tests__/install-merge-hook.test.js +0 -54
- package/dist/commands/__tests__/install-merge-hook.test.js.map +0 -1
- package/dist/commands/upgrade.d.ts +0 -3
- package/dist/commands/upgrade.d.ts.map +0 -1
- package/dist/commands/upgrade.js +0 -35
- package/dist/commands/upgrade.js.map +0 -1
- package/dist/services/__tests__/auth.service.test.d.ts +0 -2
- package/dist/services/__tests__/auth.service.test.d.ts.map +0 -1
- package/dist/services/__tests__/auth.service.test.js +0 -201
- package/dist/services/__tests__/auth.service.test.js.map +0 -1
- package/dist/services/__tests__/git.service.test.d.ts +0 -2
- package/dist/services/__tests__/git.service.test.d.ts.map +0 -1
- package/dist/services/__tests__/git.service.test.js +0 -61
- package/dist/services/__tests__/git.service.test.js.map +0 -1
- package/dist/services/__tests__/memory.service.test.d.ts +0 -2
- package/dist/services/__tests__/memory.service.test.d.ts.map +0 -1
- package/dist/services/__tests__/memory.service.test.js +0 -279
- package/dist/services/__tests__/memory.service.test.js.map +0 -1
- package/dist/services/__tests__/review.service.auth-fallback.test.d.ts +0 -2
- package/dist/services/__tests__/review.service.auth-fallback.test.d.ts.map +0 -1
- package/dist/services/__tests__/review.service.auth-fallback.test.js +0 -63
- package/dist/services/__tests__/review.service.auth-fallback.test.js.map +0 -1
- package/dist/services/__tests__/review.service.test.d.ts +0 -2
- package/dist/services/__tests__/review.service.test.d.ts.map +0 -1
- package/dist/services/__tests__/review.service.test.js +0 -132
- package/dist/services/__tests__/review.service.test.js.map +0 -1
- package/dist/services/__tests__/review.suggestions-auth.test.d.ts +0 -2
- package/dist/services/__tests__/review.suggestions-auth.test.d.ts.map +0 -1
- package/dist/services/__tests__/review.suggestions-auth.test.js +0 -81
- package/dist/services/__tests__/review.suggestions-auth.test.js.map +0 -1
- package/dist/services/__tests__/transcript-parser.service.test.d.ts +0 -2
- package/dist/services/__tests__/transcript-parser.service.test.d.ts.map +0 -1
- package/dist/services/__tests__/transcript-parser.service.test.js +0 -134
- package/dist/services/__tests__/transcript-parser.service.test.js.map +0 -1
- package/dist/services/api/__tests__/api.real.test.d.ts +0 -2
- package/dist/services/api/__tests__/api.real.test.d.ts.map +0 -1
- package/dist/services/api/__tests__/api.real.test.js +0 -182
- package/dist/services/api/__tests__/api.real.test.js.map +0 -1
- package/dist/utils/__tests__/config.test.d.ts +0 -2
- package/dist/utils/__tests__/config.test.d.ts.map +0 -1
- package/dist/utils/__tests__/config.test.js +0 -79
- package/dist/utils/__tests__/config.test.js.map +0 -1
- package/dist/utils/__tests__/credentials.test.d.ts +0 -2
- package/dist/utils/__tests__/credentials.test.d.ts.map +0 -1
- package/dist/utils/__tests__/credentials.test.js +0 -89
- package/dist/utils/__tests__/credentials.test.js.map +0 -1
- package/dist/utils/__tests__/device.test.d.ts +0 -2
- package/dist/utils/__tests__/device.test.d.ts.map +0 -1
- package/dist/utils/__tests__/device.test.js +0 -108
- package/dist/utils/__tests__/device.test.js.map +0 -1
- package/dist/utils/__tests__/module-matcher.test.d.ts +0 -2
- package/dist/utils/__tests__/module-matcher.test.d.ts.map +0 -1
- package/dist/utils/__tests__/module-matcher.test.js +0 -164
- package/dist/utils/__tests__/module-matcher.test.js.map +0 -1
- package/dist/utils/__tests__/update-check.test.d.ts +0 -2
- package/dist/utils/__tests__/update-check.test.d.ts.map +0 -1
- package/dist/utils/__tests__/update-check.test.js +0 -26
- package/dist/utils/__tests__/update-check.test.js.map +0 -1
|
@@ -1,145 +0,0 @@
|
|
|
1
|
-
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
|
|
2
|
-
vi.mock('../../utils/config.js', () => ({
|
|
3
|
-
loadConfig: vi.fn(),
|
|
4
|
-
saveConfig: vi.fn(),
|
|
5
|
-
clearConfig: vi.fn(),
|
|
6
|
-
}));
|
|
7
|
-
vi.mock('../../utils/credentials.js', () => ({
|
|
8
|
-
clearCredentials: vi.fn(),
|
|
9
|
-
}));
|
|
10
|
-
import { clearConfig, loadConfig, saveConfig } from '../../utils/config.js';
|
|
11
|
-
import { clearCredentials } from '../../utils/credentials.js';
|
|
12
|
-
import { teamKeyAction, teamStatusAction } from '../auth/team-key.js';
|
|
13
|
-
const mockLoadConfig = vi.mocked(loadConfig);
|
|
14
|
-
const mockSaveConfig = vi.mocked(saveConfig);
|
|
15
|
-
const mockClearConfig = vi.mocked(clearConfig);
|
|
16
|
-
const mockClearCredentials = vi.mocked(clearCredentials);
|
|
17
|
-
function mockProcessExit() {
|
|
18
|
-
return vi.spyOn(process, 'exit').mockImplementation((code) => {
|
|
19
|
-
throw new Error(`process.exit:${code ?? 0}`);
|
|
20
|
-
});
|
|
21
|
-
}
|
|
22
|
-
describe('auth team-key command', () => {
|
|
23
|
-
beforeEach(() => {
|
|
24
|
-
vi.clearAllMocks();
|
|
25
|
-
vi.stubGlobal('fetch', vi.fn());
|
|
26
|
-
});
|
|
27
|
-
afterEach(() => {
|
|
28
|
-
vi.unstubAllGlobals();
|
|
29
|
-
vi.restoreAllMocks();
|
|
30
|
-
});
|
|
31
|
-
it('exits when key is missing', async () => {
|
|
32
|
-
const exitSpy = mockProcessExit();
|
|
33
|
-
const errorSpy = vi.spyOn(console, 'error').mockImplementation(() => { });
|
|
34
|
-
await expect(teamKeyAction({})).rejects.toThrow('process.exit:1');
|
|
35
|
-
expect(exitSpy).toHaveBeenCalledWith(1);
|
|
36
|
-
expect(errorSpy).toHaveBeenCalled();
|
|
37
|
-
});
|
|
38
|
-
it('exits when key format is invalid', async () => {
|
|
39
|
-
const exitSpy = mockProcessExit();
|
|
40
|
-
const errorSpy = vi.spyOn(console, 'error').mockImplementation(() => { });
|
|
41
|
-
await expect(teamKeyAction({ key: 'invalid' })).rejects.toThrow('process.exit:1');
|
|
42
|
-
expect(exitSpy).toHaveBeenCalledWith(1);
|
|
43
|
-
expect(errorSpy).toHaveBeenCalled();
|
|
44
|
-
});
|
|
45
|
-
it('saves config and clears credentials when key is valid', async () => {
|
|
46
|
-
const fetchMock = vi.mocked(fetch);
|
|
47
|
-
fetchMock.mockResolvedValue(new Response(JSON.stringify({
|
|
48
|
-
data: {
|
|
49
|
-
team: { name: 'Platform Team' },
|
|
50
|
-
organization: { name: 'Kodus' },
|
|
51
|
-
},
|
|
52
|
-
}), {
|
|
53
|
-
status: 200,
|
|
54
|
-
headers: { 'Content-Type': 'application/json' },
|
|
55
|
-
}));
|
|
56
|
-
mockClearCredentials.mockResolvedValue(undefined);
|
|
57
|
-
await teamKeyAction({ key: 'kodus_abc123' });
|
|
58
|
-
expect(mockSaveConfig).toHaveBeenCalledWith({
|
|
59
|
-
teamKey: 'kodus_abc123',
|
|
60
|
-
teamName: 'Platform Team',
|
|
61
|
-
organizationName: 'Kodus',
|
|
62
|
-
});
|
|
63
|
-
expect(mockClearCredentials).toHaveBeenCalled();
|
|
64
|
-
expect(fetchMock).toHaveBeenCalledWith(expect.stringContaining('/cli/validate-key'), expect.objectContaining({
|
|
65
|
-
headers: expect.objectContaining({ 'X-Team-Key': 'kodus_abc123' }),
|
|
66
|
-
}));
|
|
67
|
-
});
|
|
68
|
-
it('fails and rolls back team config when clearing old credentials throws', async () => {
|
|
69
|
-
const fetchMock = vi.mocked(fetch);
|
|
70
|
-
const exitSpy = mockProcessExit();
|
|
71
|
-
const errorSpy = vi.spyOn(console, 'error').mockImplementation(() => { });
|
|
72
|
-
fetchMock.mockResolvedValue(new Response(JSON.stringify({
|
|
73
|
-
data: {
|
|
74
|
-
teamName: 'Backend Team',
|
|
75
|
-
organizationName: 'Kodus',
|
|
76
|
-
},
|
|
77
|
-
}), {
|
|
78
|
-
status: 200,
|
|
79
|
-
headers: { 'Content-Type': 'application/json' },
|
|
80
|
-
}));
|
|
81
|
-
mockClearCredentials.mockRejectedValue(new Error('fs error'));
|
|
82
|
-
await expect(teamKeyAction({ key: 'kodus_abc123' })).rejects.toThrow('process.exit:1');
|
|
83
|
-
expect(exitSpy).toHaveBeenCalledWith(1);
|
|
84
|
-
expect(errorSpy).toHaveBeenCalled();
|
|
85
|
-
expect(mockClearConfig).toHaveBeenCalledTimes(1);
|
|
86
|
-
expect(mockSaveConfig).toHaveBeenCalled();
|
|
87
|
-
});
|
|
88
|
-
it('exits when API returns invalid key', async () => {
|
|
89
|
-
const fetchMock = vi.mocked(fetch);
|
|
90
|
-
const exitSpy = mockProcessExit();
|
|
91
|
-
const errorSpy = vi.spyOn(console, 'error').mockImplementation(() => { });
|
|
92
|
-
fetchMock.mockResolvedValue(new Response(JSON.stringify({ message: 'Invalid team key' }), {
|
|
93
|
-
status: 401,
|
|
94
|
-
headers: { 'Content-Type': 'application/json' },
|
|
95
|
-
}));
|
|
96
|
-
await expect(teamKeyAction({ key: 'kodus_abc123' })).rejects.toThrow('process.exit:1');
|
|
97
|
-
expect(exitSpy).toHaveBeenCalledWith(1);
|
|
98
|
-
expect(errorSpy).toHaveBeenCalled();
|
|
99
|
-
});
|
|
100
|
-
it('shows device limit message when API returns DEVICE_LIMIT_REACHED with current count', async () => {
|
|
101
|
-
const fetchMock = vi.mocked(fetch);
|
|
102
|
-
const exitSpy = mockProcessExit();
|
|
103
|
-
const errorSpy = vi.spyOn(console, 'error').mockImplementation(() => { });
|
|
104
|
-
fetchMock.mockResolvedValue(new Response(JSON.stringify({
|
|
105
|
-
code: 'DEVICE_LIMIT_REACHED',
|
|
106
|
-
details: { limit: 2, current: 2 },
|
|
107
|
-
}), {
|
|
108
|
-
status: 401,
|
|
109
|
-
headers: { 'Content-Type': 'application/json' },
|
|
110
|
-
}));
|
|
111
|
-
await expect(teamKeyAction({ key: 'kodus_abc123' })).rejects.toThrow('process.exit:1');
|
|
112
|
-
expect(exitSpy).toHaveBeenCalledWith(1);
|
|
113
|
-
const output = errorSpy.mock.calls.map((c) => c.join(' ')).join('\n');
|
|
114
|
-
expect(output).toContain('Device limit reached (2/2). Remove an old device or contact your admin.');
|
|
115
|
-
});
|
|
116
|
-
});
|
|
117
|
-
describe('auth team-status command', () => {
|
|
118
|
-
beforeEach(() => {
|
|
119
|
-
vi.clearAllMocks();
|
|
120
|
-
});
|
|
121
|
-
afterEach(() => {
|
|
122
|
-
vi.restoreAllMocks();
|
|
123
|
-
});
|
|
124
|
-
it('shows not-authenticated message when no team config exists', async () => {
|
|
125
|
-
const logSpy = vi.spyOn(console, 'log').mockImplementation(() => { });
|
|
126
|
-
mockLoadConfig.mockResolvedValue(null);
|
|
127
|
-
await teamStatusAction();
|
|
128
|
-
const output = logSpy.mock.calls.map((c) => c.join(' ')).join('\n');
|
|
129
|
-
expect(output).toContain('Not authenticated with team key');
|
|
130
|
-
});
|
|
131
|
-
it('shows team details when team config exists', async () => {
|
|
132
|
-
const logSpy = vi.spyOn(console, 'log').mockImplementation(() => { });
|
|
133
|
-
mockLoadConfig.mockResolvedValue({
|
|
134
|
-
teamKey: 'kodus_abc123',
|
|
135
|
-
teamName: 'Platform Team',
|
|
136
|
-
organizationName: 'Kodus',
|
|
137
|
-
});
|
|
138
|
-
await teamStatusAction();
|
|
139
|
-
const output = logSpy.mock.calls.map((c) => c.join(' ')).join('\n');
|
|
140
|
-
expect(output).toContain('Authenticated');
|
|
141
|
-
expect(output).toContain('Kodus');
|
|
142
|
-
expect(output).toContain('Platform Team');
|
|
143
|
-
});
|
|
144
|
-
});
|
|
145
|
-
//# sourceMappingURL=auth.team-key.test.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"auth.team-key.test.js","sourceRoot":"","sources":["../../../src/commands/__tests__/auth.team-key.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"capture-api.test.d.ts","sourceRoot":"","sources":["../../../src/commands/__tests__/capture-api.test.ts"],"names":[],"mappings":""}
|
|
@@ -1,88 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
2
|
-
// Mock dependencies before importing the module under test
|
|
3
|
-
vi.mock('../../services/git.service.js', () => ({
|
|
4
|
-
gitService: {
|
|
5
|
-
isGitRepository: vi.fn().mockResolvedValue(true),
|
|
6
|
-
getGitRoot: vi.fn().mockResolvedValue('/fake/repo'),
|
|
7
|
-
getHeadSha: vi.fn().mockResolvedValue('abc1234'),
|
|
8
|
-
getCurrentBranch: vi.fn().mockResolvedValue('feat/test'),
|
|
9
|
-
extractOrgRepo: vi.fn().mockResolvedValue({ org: 'kodustech', repo: 'cli' }),
|
|
10
|
-
},
|
|
11
|
-
}));
|
|
12
|
-
vi.mock('../../services/memory.service.js', () => ({
|
|
13
|
-
memoryService: {
|
|
14
|
-
saveCapture: vi.fn().mockResolvedValue('/fake/path/capture.json'),
|
|
15
|
-
saveBranchCapture: vi.fn().mockResolvedValue('/fake/path/branch-capture.md'),
|
|
16
|
-
},
|
|
17
|
-
}));
|
|
18
|
-
vi.mock('../../services/transcript-parser.service.js', () => ({
|
|
19
|
-
transcriptParserService: {
|
|
20
|
-
parse: vi.fn().mockReturnValue({
|
|
21
|
-
modifiedFiles: ['src/auth/jwt.ts'],
|
|
22
|
-
toolUses: [{ tool: 'Write', filePath: 'src/auth/jwt.ts' }],
|
|
23
|
-
}),
|
|
24
|
-
},
|
|
25
|
-
}));
|
|
26
|
-
vi.mock('../../services/auth.service.js', () => ({
|
|
27
|
-
authService: {
|
|
28
|
-
isAuthenticated: vi.fn().mockResolvedValue(false),
|
|
29
|
-
getValidToken: vi.fn().mockResolvedValue('fake-token'),
|
|
30
|
-
},
|
|
31
|
-
}));
|
|
32
|
-
vi.mock('../../services/api/index.js', () => ({
|
|
33
|
-
api: {
|
|
34
|
-
memory: {
|
|
35
|
-
submitCapture: vi.fn().mockResolvedValue({ id: 'cap-123', accepted: true }),
|
|
36
|
-
},
|
|
37
|
-
},
|
|
38
|
-
}));
|
|
39
|
-
import { captureAction } from '../memory/capture.js';
|
|
40
|
-
import { authService } from '../../services/auth.service.js';
|
|
41
|
-
import { api } from '../../services/api/index.js';
|
|
42
|
-
beforeEach(() => {
|
|
43
|
-
vi.clearAllMocks();
|
|
44
|
-
// Default: stdin is a TTY so no stdin reading
|
|
45
|
-
Object.defineProperty(process.stdin, 'isTTY', { value: true, configurable: true });
|
|
46
|
-
});
|
|
47
|
-
afterEach(() => {
|
|
48
|
-
vi.restoreAllMocks();
|
|
49
|
-
});
|
|
50
|
-
describe('capture API submission', () => {
|
|
51
|
-
it('submits to API on stop event when authenticated', async () => {
|
|
52
|
-
vi.mocked(authService.isAuthenticated).mockResolvedValue(true);
|
|
53
|
-
await captureAction(undefined, { agent: 'claude-code', event: 'stop' });
|
|
54
|
-
// Allow fire-and-forget promise to settle
|
|
55
|
-
await new Promise(resolve => setTimeout(resolve, 50));
|
|
56
|
-
expect(api.memory.submitCapture).toHaveBeenCalledTimes(1);
|
|
57
|
-
const [payload, token] = vi.mocked(api.memory.submitCapture).mock.calls[0];
|
|
58
|
-
expect(payload.branch).toBe('feat/test');
|
|
59
|
-
expect(payload.sha).toBe('abc1234');
|
|
60
|
-
expect(payload.orgRepo).toBe('kodustech/cli');
|
|
61
|
-
expect(payload.agent).toBe('claude-code');
|
|
62
|
-
expect(payload.event).toBe('stop');
|
|
63
|
-
expect(payload.capturedAt).toBeTruthy();
|
|
64
|
-
expect(token).toBe('fake-token');
|
|
65
|
-
});
|
|
66
|
-
it('does NOT submit on non-stop events', async () => {
|
|
67
|
-
vi.mocked(authService.isAuthenticated).mockResolvedValue(true);
|
|
68
|
-
await captureAction(undefined, { agent: 'claude-code', event: 'user-prompt-submit' });
|
|
69
|
-
await new Promise(resolve => setTimeout(resolve, 50));
|
|
70
|
-
expect(api.memory.submitCapture).not.toHaveBeenCalled();
|
|
71
|
-
});
|
|
72
|
-
it('does NOT submit when not authenticated', async () => {
|
|
73
|
-
vi.mocked(authService.isAuthenticated).mockResolvedValue(false);
|
|
74
|
-
await captureAction(undefined, { agent: 'claude-code', event: 'stop' });
|
|
75
|
-
await new Promise(resolve => setTimeout(resolve, 50));
|
|
76
|
-
expect(api.memory.submitCapture).not.toHaveBeenCalled();
|
|
77
|
-
});
|
|
78
|
-
it('does NOT throw if API call fails (fail-open)', async () => {
|
|
79
|
-
vi.mocked(authService.isAuthenticated).mockResolvedValue(true);
|
|
80
|
-
vi.mocked(api.memory.submitCapture).mockRejectedValue(new Error('network error'));
|
|
81
|
-
// Should not throw
|
|
82
|
-
await captureAction(undefined, { agent: 'claude-code', event: 'stop' });
|
|
83
|
-
// Allow fire-and-forget promise to settle
|
|
84
|
-
await new Promise(resolve => setTimeout(resolve, 50));
|
|
85
|
-
expect(api.memory.submitCapture).toHaveBeenCalledTimes(1);
|
|
86
|
-
});
|
|
87
|
-
});
|
|
88
|
-
//# sourceMappingURL=capture-api.test.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"capture-api.test.js","sourceRoot":"","sources":["../../../src/commands/__tests__/capture-api.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAC;AAEzE,2DAA2D;AAC3D,EAAE,CAAC,IAAI,CAAC,+BAA+B,EAAE,GAAG,EAAE,CAAC,CAAC;IAC9C,UAAU,EAAE;QACV,eAAe,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC,IAAI,CAAC;QAChD,UAAU,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC,YAAY,CAAC;QACnD,UAAU,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC,SAAS,CAAC;QAChD,gBAAgB,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC,WAAW,CAAC;QACxD,cAAc,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC,EAAE,GAAG,EAAE,WAAW,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC;KAC7E;CACF,CAAC,CAAC,CAAC;AAEJ,EAAE,CAAC,IAAI,CAAC,kCAAkC,EAAE,GAAG,EAAE,CAAC,CAAC;IACjD,aAAa,EAAE;QACb,WAAW,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC,yBAAyB,CAAC;QACjE,iBAAiB,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC,8BAA8B,CAAC;KAC7E;CACF,CAAC,CAAC,CAAC;AAEJ,EAAE,CAAC,IAAI,CAAC,6CAA6C,EAAE,GAAG,EAAE,CAAC,CAAC;IAC5D,uBAAuB,EAAE;QACvB,KAAK,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,eAAe,CAAC;YAC7B,aAAa,EAAE,CAAC,iBAAiB,CAAC;YAClC,QAAQ,EAAE,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE,QAAQ,EAAE,iBAAiB,EAAE,CAAC;SAC3D,CAAC;KACH;CACF,CAAC,CAAC,CAAC;AAEJ,EAAE,CAAC,IAAI,CAAC,gCAAgC,EAAE,GAAG,EAAE,CAAC,CAAC;IAC/C,WAAW,EAAE;QACX,eAAe,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC,KAAK,CAAC;QACjD,aAAa,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC,YAAY,CAAC;KACvD;CACF,CAAC,CAAC,CAAC;AAEJ,EAAE,CAAC,IAAI,CAAC,6BAA6B,EAAE,GAAG,EAAE,CAAC,CAAC;IAC5C,GAAG,EAAE;QACH,MAAM,EAAE;YACN,aAAa,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC,EAAE,EAAE,EAAE,SAAS,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC;SAC5E;KACF;CACF,CAAC,CAAC,CAAC;AAEJ,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,WAAW,EAAE,MAAM,gCAAgC,CAAC;AAC7D,OAAO,EAAE,GAAG,EAAE,MAAM,6BAA6B,CAAC;AAElD,UAAU,CAAC,GAAG,EAAE;IACd,EAAE,CAAC,aAAa,EAAE,CAAC;IACnB,8CAA8C;IAC9C,MAAM,CAAC,cAAc,CAAC,OAAO,CAAC,KAAK,EAAE,OAAO,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,YAAY,EAAE,IAAI,EAAE,CAAC,CAAC;AACrF,CAAC,CAAC,CAAC;AAEH,SAAS,CAAC,GAAG,EAAE;IACb,EAAE,CAAC,eAAe,EAAE,CAAC;AACvB,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,wBAAwB,EAAE,GAAG,EAAE;IACtC,EAAE,CAAC,iDAAiD,EAAE,KAAK,IAAI,EAAE;QAC/D,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,eAAe,CAAC,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;QAE/D,MAAM,aAAa,CAAC,SAAS,EAAE,EAAE,KAAK,EAAE,aAAa,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;QAExE,0CAA0C;QAC1C,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC;QAEtD,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;QAC1D,MAAM,CAAC,OAAO,EAAE,KAAK,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QAC3E,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACzC,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACpC,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;QAC9C,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QAC1C,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACnC,MAAM,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,UAAU,EAAE,CAAC;QACxC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IACnC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,oCAAoC,EAAE,KAAK,IAAI,EAAE;QAClD,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,eAAe,CAAC,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;QAE/D,MAAM,aAAa,CAAC,SAAS,EAAE,EAAE,KAAK,EAAE,aAAa,EAAE,KAAK,EAAE,oBAAoB,EAAE,CAAC,CAAC;QAEtF,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC;QAEtD,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;IAC1D,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,wCAAwC,EAAE,KAAK,IAAI,EAAE;QACtD,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,eAAe,CAAC,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC;QAEhE,MAAM,aAAa,CAAC,SAAS,EAAE,EAAE,KAAK,EAAE,aAAa,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;QAExE,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC;QAEtD,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;IAC1D,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8CAA8C,EAAE,KAAK,IAAI,EAAE;QAC5D,EAAE,CAAC,MAAM,CAAC,WAAW,CAAC,eAAe,CAAC,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;QAC/D,EAAE,CAAC,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,iBAAiB,CAAC,IAAI,KAAK,CAAC,eAAe,CAAC,CAAC,CAAC;QAElF,mBAAmB;QACnB,MAAM,aAAa,CAAC,SAAS,EAAE,EAAE,KAAK,EAAE,aAAa,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;QAExE,0CAA0C;QAC1C,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC;QAEtD,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;IAC5D,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"disable.test.d.ts","sourceRoot":"","sources":["../../../src/commands/__tests__/disable.test.ts"],"names":[],"mappings":""}
|
|
@@ -1,103 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
2
|
-
import fs from 'fs/promises';
|
|
3
|
-
import path from 'path';
|
|
4
|
-
import os from 'os';
|
|
5
|
-
vi.mock('../../services/git.service.js', () => ({
|
|
6
|
-
gitService: {
|
|
7
|
-
isGitRepository: vi.fn().mockResolvedValue(true),
|
|
8
|
-
getGitRoot: vi.fn(),
|
|
9
|
-
},
|
|
10
|
-
}));
|
|
11
|
-
import { gitService } from '../../services/git.service.js';
|
|
12
|
-
import { disableAction } from '../memory/disable.js';
|
|
13
|
-
import { MERGE_HOOK_MARKER, CODEX_NOTIFY_LINE, CODEX_NOTIFY_LINE_LEGACY } from '../memory/hooks.js';
|
|
14
|
-
let tmpDir;
|
|
15
|
-
let originalHome;
|
|
16
|
-
beforeEach(async () => {
|
|
17
|
-
tmpDir = await fs.mkdtemp(path.join(os.tmpdir(), 'kodus-disable-test-'));
|
|
18
|
-
await fs.mkdir(path.join(tmpDir, '.git', 'hooks'), { recursive: true });
|
|
19
|
-
vi.mocked(gitService.getGitRoot).mockResolvedValue(tmpDir);
|
|
20
|
-
// Override HOME so resolveCodexConfigPath points to tmpDir
|
|
21
|
-
originalHome = os.homedir();
|
|
22
|
-
vi.spyOn(os, 'homedir').mockReturnValue(tmpDir);
|
|
23
|
-
vi.spyOn(console, 'log').mockImplementation(() => { });
|
|
24
|
-
vi.spyOn(console, 'error').mockImplementation(() => { });
|
|
25
|
-
});
|
|
26
|
-
afterEach(async () => {
|
|
27
|
-
await fs.rm(tmpDir, { recursive: true, force: true });
|
|
28
|
-
vi.restoreAllMocks();
|
|
29
|
-
});
|
|
30
|
-
describe('disableAction', () => {
|
|
31
|
-
it('removes Claude hooks from settings.json', async () => {
|
|
32
|
-
const settingsPath = path.join(tmpDir, '.claude', 'settings.json');
|
|
33
|
-
await fs.mkdir(path.dirname(settingsPath), { recursive: true });
|
|
34
|
-
await fs.writeFile(settingsPath, JSON.stringify({
|
|
35
|
-
hooks: {
|
|
36
|
-
UserPromptSubmit: [{
|
|
37
|
-
matcher: '',
|
|
38
|
-
hooks: [{ type: 'command', command: 'kodus decisions capture --agent claude-compatible --event user-prompt-submit' }],
|
|
39
|
-
}],
|
|
40
|
-
Stop: [{
|
|
41
|
-
matcher: '',
|
|
42
|
-
hooks: [{ type: 'command', command: 'kodus decisions capture --agent claude-compatible --event stop' }],
|
|
43
|
-
}],
|
|
44
|
-
},
|
|
45
|
-
}, null, 2));
|
|
46
|
-
await disableAction();
|
|
47
|
-
const settings = JSON.parse(await fs.readFile(settingsPath, 'utf-8'));
|
|
48
|
-
expect(settings.hooks).toBeUndefined();
|
|
49
|
-
});
|
|
50
|
-
it('removes Codex notify line', async () => {
|
|
51
|
-
const codexPath = path.join(tmpDir, '.codex', 'config.toml');
|
|
52
|
-
await fs.mkdir(path.dirname(codexPath), { recursive: true });
|
|
53
|
-
await fs.writeFile(codexPath, `model = "gpt-4"\n${CODEX_NOTIFY_LINE}\n`);
|
|
54
|
-
await disableAction();
|
|
55
|
-
const content = await fs.readFile(codexPath, 'utf-8');
|
|
56
|
-
expect(content).not.toContain(CODEX_NOTIFY_LINE);
|
|
57
|
-
expect(content).toContain('model = "gpt-4"');
|
|
58
|
-
});
|
|
59
|
-
it('removes legacy Codex notify line', async () => {
|
|
60
|
-
const codexPath = path.join(tmpDir, '.codex', 'config.toml');
|
|
61
|
-
await fs.mkdir(path.dirname(codexPath), { recursive: true });
|
|
62
|
-
await fs.writeFile(codexPath, `model = "gpt-4"\n${CODEX_NOTIFY_LINE_LEGACY}\n`);
|
|
63
|
-
await disableAction();
|
|
64
|
-
const content = await fs.readFile(codexPath, 'utf-8');
|
|
65
|
-
expect(content).not.toContain(CODEX_NOTIFY_LINE_LEGACY);
|
|
66
|
-
expect(content).toContain('model = "gpt-4"');
|
|
67
|
-
});
|
|
68
|
-
it('removes kodus section from post-merge hook', async () => {
|
|
69
|
-
const hookPath = path.join(tmpDir, '.git', 'hooks', 'post-merge');
|
|
70
|
-
await fs.writeFile(hookPath, `#!/bin/sh\n${MERGE_HOOK_MARKER}\nif [ -n "$MERGED_BRANCH" ]; then\n kodus decisions promote --branch "$MERGED_BRANCH" &\nfi\n`, { mode: 0o755 });
|
|
71
|
-
await disableAction();
|
|
72
|
-
// Hook file should be deleted (only shebang remained)
|
|
73
|
-
await expect(fs.access(hookPath)).rejects.toThrow();
|
|
74
|
-
});
|
|
75
|
-
it('preserves non-kodus content in post-merge hook', async () => {
|
|
76
|
-
const hookPath = path.join(tmpDir, '.git', 'hooks', 'post-merge');
|
|
77
|
-
await fs.writeFile(hookPath, `#!/bin/sh\necho "custom hook"\n\n${MERGE_HOOK_MARKER}\nif [ -n "$MERGED_BRANCH" ]; then\n kodus decisions promote --branch "$MERGED_BRANCH" &\nfi\n`, { mode: 0o755 });
|
|
78
|
-
await disableAction();
|
|
79
|
-
const content = await fs.readFile(hookPath, 'utf-8');
|
|
80
|
-
expect(content).toContain('echo "custom hook"');
|
|
81
|
-
expect(content).not.toContain(MERGE_HOOK_MARKER);
|
|
82
|
-
});
|
|
83
|
-
it('removes legacy two-if kodus block without leaving orphaned shell lines', async () => {
|
|
84
|
-
const hookPath = path.join(tmpDir, '.git', 'hooks', 'post-merge');
|
|
85
|
-
await fs.writeFile(hookPath, `#!/bin/sh\n${MERGE_HOOK_MARKER}\nMERGED_BRANCH=$(git log -1 --merges --format=%s HEAD 2>/dev/null | sed -n "s/.*Merge branch '\\([^']*\\)'.*/\\1/p")\nif [ -z "$MERGED_BRANCH" ]; then\n MERGED_BRANCH=$(git log -1 --merges --format=%s HEAD 2>/dev/null | sed -n "s/.*Merge pull request .* from [^/]*\\/\\(.*\\)/\\1/p")\nfi\nif [ -n "$MERGED_BRANCH" ]; then\n kodus decisions promote --branch "$MERGED_BRANCH" &\nfi\n`, { mode: 0o755 });
|
|
86
|
-
await disableAction();
|
|
87
|
-
await expect(fs.access(hookPath)).rejects.toThrow();
|
|
88
|
-
});
|
|
89
|
-
it('preserves .kody/ data', async () => {
|
|
90
|
-
const kodyFile = path.join(tmpDir, '.kody', 'pr', 'test.md');
|
|
91
|
-
await fs.mkdir(path.dirname(kodyFile), { recursive: true });
|
|
92
|
-
await fs.writeFile(kodyFile, '# Test memory');
|
|
93
|
-
await disableAction();
|
|
94
|
-
const content = await fs.readFile(kodyFile, 'utf-8');
|
|
95
|
-
expect(content).toBe('# Test memory');
|
|
96
|
-
});
|
|
97
|
-
it('is idempotent (disable when nothing installed reports not found)', async () => {
|
|
98
|
-
await disableAction();
|
|
99
|
-
const calls = vi.mocked(console.log).mock.calls.flat().join('\n');
|
|
100
|
-
expect(calls).toContain('not found');
|
|
101
|
-
});
|
|
102
|
-
});
|
|
103
|
-
//# sourceMappingURL=disable.test.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"disable.test.js","sourceRoot":"","sources":["../../../src/commands/__tests__/disable.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAC;AACzE,OAAO,EAAE,MAAM,aAAa,CAAC;AAC7B,OAAO,IAAI,MAAM,MAAM,CAAC;AACxB,OAAO,EAAE,MAAM,IAAI,CAAC;AAEpB,EAAE,CAAC,IAAI,CAAC,+BAA+B,EAAE,GAAG,EAAE,CAAC,CAAC;IAC9C,UAAU,EAAE;QACV,eAAe,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC,IAAI,CAAC;QAChD,UAAU,EAAE,EAAE,CAAC,EAAE,EAAE;KACpB;CACF,CAAC,CAAC,CAAC;AAEJ,OAAO,EAAE,UAAU,EAAE,MAAM,+BAA+B,CAAC;AAC3D,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,iBAAiB,EAAE,iBAAiB,EAAE,wBAAwB,EAAE,MAAM,oBAAoB,CAAC;AAEpG,IAAI,MAAc,CAAC;AACnB,IAAI,YAAoB,CAAC;AAEzB,UAAU,CAAC,KAAK,IAAI,EAAE;IACpB,MAAM,GAAG,MAAM,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,MAAM,EAAE,EAAE,qBAAqB,CAAC,CAAC,CAAC;IACzE,MAAM,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IACxE,EAAE,CAAC,MAAM,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC;IAE3D,2DAA2D;IAC3D,YAAY,GAAG,EAAE,CAAC,OAAO,EAAE,CAAC;IAC5B,EAAE,CAAC,KAAK,CAAC,EAAE,EAAE,SAAS,CAAC,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC;IAEhD,EAAE,CAAC,KAAK,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC,kBAAkB,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;IACtD,EAAE,CAAC,KAAK,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC,kBAAkB,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;AAC1D,CAAC,CAAC,CAAC;AAEH,SAAS,CAAC,KAAK,IAAI,EAAE;IACnB,MAAM,EAAE,CAAC,EAAE,CAAC,MAAM,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;IACtD,EAAE,CAAC,eAAe,EAAE,CAAC;AACvB,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,eAAe,EAAE,GAAG,EAAE;IAC7B,EAAE,CAAC,yCAAyC,EAAE,KAAK,IAAI,EAAE;QACvD,MAAM,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,SAAS,EAAE,eAAe,CAAC,CAAC;QACnE,MAAM,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAChE,MAAM,EAAE,CAAC,SAAS,CAAC,YAAY,EAAE,IAAI,CAAC,SAAS,CAAC;YAC9C,KAAK,EAAE;gBACL,gBAAgB,EAAE,CAAC;wBACjB,OAAO,EAAE,EAAE;wBACX,KAAK,EAAE,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,8EAA8E,EAAE,CAAC;qBACtH,CAAC;gBACF,IAAI,EAAE,CAAC;wBACL,OAAO,EAAE,EAAE;wBACX,KAAK,EAAE,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,gEAAgE,EAAE,CAAC;qBACxG,CAAC;aACH;SACF,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAEb,MAAM,aAAa,EAAE,CAAC;QAEtB,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC,CAAC;QACtE,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,aAAa,EAAE,CAAC;IACzC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,2BAA2B,EAAE,KAAK,IAAI,EAAE;QACzC,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,QAAQ,EAAE,aAAa,CAAC,CAAC;QAC7D,MAAM,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC7D,MAAM,EAAE,CAAC,SAAS,CAAC,SAAS,EAAE,oBAAoB,iBAAiB,IAAI,CAAC,CAAC;QAEzE,MAAM,aAAa,EAAE,CAAC;QAEtB,MAAM,OAAO,GAAG,MAAM,EAAE,CAAC,QAAQ,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;QACtD,MAAM,CAAC,OAAO,CAAC,CAAC,GAAG,CAAC,SAAS,CAAC,iBAAiB,CAAC,CAAC;QACjD,MAAM,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,iBAAiB,CAAC,CAAC;IAC/C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,kCAAkC,EAAE,KAAK,IAAI,EAAE;QAChD,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,QAAQ,EAAE,aAAa,CAAC,CAAC;QAC7D,MAAM,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC7D,MAAM,EAAE,CAAC,SAAS,CAAC,SAAS,EAAE,oBAAoB,wBAAwB,IAAI,CAAC,CAAC;QAEhF,MAAM,aAAa,EAAE,CAAC;QAEtB,MAAM,OAAO,GAAG,MAAM,EAAE,CAAC,QAAQ,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;QACtD,MAAM,CAAC,OAAO,CAAC,CAAC,GAAG,CAAC,SAAS,CAAC,wBAAwB,CAAC,CAAC;QACxD,MAAM,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,iBAAiB,CAAC,CAAC;IAC/C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,4CAA4C,EAAE,KAAK,IAAI,EAAE;QAC1D,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,YAAY,CAAC,CAAC;QAClE,MAAM,EAAE,CAAC,SAAS,CAAC,QAAQ,EAAE,cAAc,iBAAiB,iGAAiG,EAAE,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC;QAEhL,MAAM,aAAa,EAAE,CAAC;QAEtB,sDAAsD;QACtD,MAAM,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,EAAE,CAAC;IACtD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gDAAgD,EAAE,KAAK,IAAI,EAAE;QAC9D,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,YAAY,CAAC,CAAC;QAClE,MAAM,EAAE,CAAC,SAAS,CAAC,QAAQ,EAAE,oCAAoC,iBAAiB,iGAAiG,EAAE,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC;QAEtM,MAAM,aAAa,EAAE,CAAC;QAEtB,MAAM,OAAO,GAAG,MAAM,EAAE,CAAC,QAAQ,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;QACrD,MAAM,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,oBAAoB,CAAC,CAAC;QAChD,MAAM,CAAC,OAAO,CAAC,CAAC,GAAG,CAAC,SAAS,CAAC,iBAAiB,CAAC,CAAC;IACnD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,wEAAwE,EAAE,KAAK,IAAI,EAAE;QACtF,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,YAAY,CAAC,CAAC;QAClE,MAAM,EAAE,CAAC,SAAS,CAAC,QAAQ,EAAE,cAAc,iBAAiB,kYAAkY,EAAE,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC;QAEjd,MAAM,aAAa,EAAE,CAAC;QAEtB,MAAM,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,EAAE,CAAC;IACtD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,uBAAuB,EAAE,KAAK,IAAI,EAAE;QACrC,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,OAAO,EAAE,IAAI,EAAE,SAAS,CAAC,CAAC;QAC7D,MAAM,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC5D,MAAM,EAAE,CAAC,SAAS,CAAC,QAAQ,EAAE,eAAe,CAAC,CAAC;QAE9C,MAAM,aAAa,EAAE,CAAC;QAEtB,MAAM,OAAO,GAAG,MAAM,EAAE,CAAC,QAAQ,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;QACrD,MAAM,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;IACxC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,kEAAkE,EAAE,KAAK,IAAI,EAAE;QAChF,MAAM,aAAa,EAAE,CAAC;QAEtB,MAAM,KAAK,GAAG,EAAE,CAAC,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAClE,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC;IACvC,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"enable.test.d.ts","sourceRoot":"","sources":["../../../src/commands/__tests__/enable.test.ts"],"names":[],"mappings":""}
|
|
@@ -1,103 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
2
|
-
import fs from 'fs/promises';
|
|
3
|
-
import path from 'path';
|
|
4
|
-
import os from 'os';
|
|
5
|
-
vi.mock('../../services/git.service.js', () => ({
|
|
6
|
-
gitService: {
|
|
7
|
-
isGitRepository: vi.fn().mockResolvedValue(true),
|
|
8
|
-
getGitRoot: vi.fn(),
|
|
9
|
-
},
|
|
10
|
-
}));
|
|
11
|
-
vi.mock('../../utils/module-matcher.js', () => ({
|
|
12
|
-
stringifyYaml: vi.fn((config) => JSON.stringify(config)),
|
|
13
|
-
}));
|
|
14
|
-
import { gitService } from '../../services/git.service.js';
|
|
15
|
-
import { enableAction } from '../memory/enable.js';
|
|
16
|
-
import { MERGE_HOOK_MARKER, CODEX_NOTIFY_LINE, CODEX_NOTIFY_LINE_LEGACY } from '../memory/hooks.js';
|
|
17
|
-
let tmpDir;
|
|
18
|
-
beforeEach(async () => {
|
|
19
|
-
tmpDir = await fs.mkdtemp(path.join(os.tmpdir(), 'kodus-enable-test-'));
|
|
20
|
-
await fs.mkdir(path.join(tmpDir, '.git', 'hooks'), { recursive: true });
|
|
21
|
-
await fs.mkdir(path.join(tmpDir, 'src', 'commands'), { recursive: true });
|
|
22
|
-
await fs.mkdir(path.join(tmpDir, 'src', 'services'), { recursive: true });
|
|
23
|
-
vi.mocked(gitService.getGitRoot).mockResolvedValue(tmpDir);
|
|
24
|
-
vi.spyOn(console, 'log').mockImplementation(() => { });
|
|
25
|
-
vi.spyOn(console, 'error').mockImplementation(() => { });
|
|
26
|
-
});
|
|
27
|
-
afterEach(async () => {
|
|
28
|
-
await fs.rm(tmpDir, { recursive: true, force: true });
|
|
29
|
-
vi.restoreAllMocks();
|
|
30
|
-
});
|
|
31
|
-
describe('enableAction', () => {
|
|
32
|
-
it('installs all hooks + modules.yml when none exist', async () => {
|
|
33
|
-
await enableAction({
|
|
34
|
-
codexConfig: path.join(tmpDir, '.codex', 'config.toml'),
|
|
35
|
-
});
|
|
36
|
-
// Claude settings created
|
|
37
|
-
const settingsPath = path.join(tmpDir, '.claude', 'settings.json');
|
|
38
|
-
const settings = JSON.parse(await fs.readFile(settingsPath, 'utf-8'));
|
|
39
|
-
expect(settings.hooks).toHaveProperty('UserPromptSubmit');
|
|
40
|
-
expect(settings.hooks).toHaveProperty('Stop');
|
|
41
|
-
expect(settings.hooks).toHaveProperty('PostToolUse');
|
|
42
|
-
// Codex config created
|
|
43
|
-
const codexConfig = await fs.readFile(path.join(tmpDir, '.codex', 'config.toml'), 'utf-8');
|
|
44
|
-
expect(codexConfig).toContain(CODEX_NOTIFY_LINE);
|
|
45
|
-
// Post-merge hook created
|
|
46
|
-
const hookContent = await fs.readFile(path.join(tmpDir, '.git', 'hooks', 'post-merge'), 'utf-8');
|
|
47
|
-
expect(hookContent).toContain(MERGE_HOOK_MARKER);
|
|
48
|
-
// modules.yml created
|
|
49
|
-
const modulesContent = await fs.readFile(path.join(tmpDir, '.kody', 'modules.yml'), 'utf-8');
|
|
50
|
-
expect(modulesContent).toContain('commands');
|
|
51
|
-
expect(modulesContent).toContain('services');
|
|
52
|
-
});
|
|
53
|
-
it('is idempotent (second run reports already configured)', async () => {
|
|
54
|
-
const codexConfig = path.join(tmpDir, '.codex', 'config.toml');
|
|
55
|
-
await enableAction({ codexConfig });
|
|
56
|
-
await enableAction({ codexConfig });
|
|
57
|
-
const calls = vi.mocked(console.log).mock.calls.flat().join('\n');
|
|
58
|
-
expect(calls).toContain('already configured');
|
|
59
|
-
});
|
|
60
|
-
it('migrates legacy codex notify line to stop event', async () => {
|
|
61
|
-
const codexConfig = path.join(tmpDir, '.codex', 'config.toml');
|
|
62
|
-
await fs.mkdir(path.dirname(codexConfig), { recursive: true });
|
|
63
|
-
await fs.writeFile(codexConfig, `${CODEX_NOTIFY_LINE_LEGACY}\n`, 'utf-8');
|
|
64
|
-
await enableAction({ codexConfig });
|
|
65
|
-
const content = await fs.readFile(codexConfig, 'utf-8');
|
|
66
|
-
expect(content).toContain(CODEX_NOTIFY_LINE);
|
|
67
|
-
expect(content).not.toContain(CODEX_NOTIFY_LINE_LEGACY);
|
|
68
|
-
});
|
|
69
|
-
it('--agents claude skips codex', async () => {
|
|
70
|
-
await enableAction({
|
|
71
|
-
agents: 'claude',
|
|
72
|
-
codexConfig: path.join(tmpDir, '.codex', 'config.toml'),
|
|
73
|
-
});
|
|
74
|
-
const calls = vi.mocked(console.log).mock.calls.flat().join('\n');
|
|
75
|
-
expect(calls).toContain('Codex notify: skipped');
|
|
76
|
-
// Codex config should not exist
|
|
77
|
-
await expect(fs.access(path.join(tmpDir, '.codex', 'config.toml'))).rejects.toThrow();
|
|
78
|
-
});
|
|
79
|
-
it('--force overwrites modules.yml', async () => {
|
|
80
|
-
const configPath = path.join(tmpDir, '.kody', 'modules.yml');
|
|
81
|
-
await fs.mkdir(path.dirname(configPath), { recursive: true });
|
|
82
|
-
await fs.writeFile(configPath, 'old-content', 'utf-8');
|
|
83
|
-
await enableAction({
|
|
84
|
-
force: true,
|
|
85
|
-
codexConfig: path.join(tmpDir, '.codex', 'config.toml'),
|
|
86
|
-
});
|
|
87
|
-
const content = await fs.readFile(configPath, 'utf-8');
|
|
88
|
-
expect(content).not.toBe('old-content');
|
|
89
|
-
});
|
|
90
|
-
it('without --force preserves existing modules.yml', async () => {
|
|
91
|
-
const configPath = path.join(tmpDir, '.kody', 'modules.yml');
|
|
92
|
-
await fs.mkdir(path.dirname(configPath), { recursive: true });
|
|
93
|
-
await fs.writeFile(configPath, 'old-content', 'utf-8');
|
|
94
|
-
await enableAction({
|
|
95
|
-
codexConfig: path.join(tmpDir, '.codex', 'config.toml'),
|
|
96
|
-
});
|
|
97
|
-
const content = await fs.readFile(configPath, 'utf-8');
|
|
98
|
-
expect(content).toBe('old-content');
|
|
99
|
-
const calls = vi.mocked(console.log).mock.calls.flat().join('\n');
|
|
100
|
-
expect(calls).toContain('already exists');
|
|
101
|
-
});
|
|
102
|
-
});
|
|
103
|
-
//# sourceMappingURL=enable.test.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"enable.test.js","sourceRoot":"","sources":["../../../src/commands/__tests__/enable.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"hook.test.d.ts","sourceRoot":"","sources":["../../../src/commands/__tests__/hook.test.ts"],"names":[],"mappings":""}
|
|
@@ -1,105 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
2
|
-
import fs from 'fs/promises';
|
|
3
|
-
import path from 'path';
|
|
4
|
-
import os from 'os';
|
|
5
|
-
import { KODUS_MARKER, generateHookScript } from '../hook/install.js';
|
|
6
|
-
// Mock gitService
|
|
7
|
-
vi.mock('../../services/git.service.js', () => ({
|
|
8
|
-
gitService: {
|
|
9
|
-
isGitRepository: vi.fn().mockResolvedValue(true),
|
|
10
|
-
getGitRoot: vi.fn(),
|
|
11
|
-
},
|
|
12
|
-
}));
|
|
13
|
-
// Mock inquirer
|
|
14
|
-
vi.mock('inquirer', () => ({
|
|
15
|
-
default: {
|
|
16
|
-
prompt: vi.fn(),
|
|
17
|
-
},
|
|
18
|
-
}));
|
|
19
|
-
import { gitService } from '../../services/git.service.js';
|
|
20
|
-
import inquirer from 'inquirer';
|
|
21
|
-
import { installAction } from '../hook/install.js';
|
|
22
|
-
import { uninstallAction } from '../hook/uninstall.js';
|
|
23
|
-
import { statusAction } from '../hook/status.js';
|
|
24
|
-
let tmpDir;
|
|
25
|
-
beforeEach(async () => {
|
|
26
|
-
tmpDir = await fs.mkdtemp(path.join(os.tmpdir(), 'kodus-hook-test-'));
|
|
27
|
-
const hooksDir = path.join(tmpDir, '.git', 'hooks');
|
|
28
|
-
await fs.mkdir(hooksDir, { recursive: true });
|
|
29
|
-
vi.mocked(gitService.isGitRepository).mockResolvedValue(true);
|
|
30
|
-
vi.mocked(gitService.getGitRoot).mockResolvedValue(tmpDir);
|
|
31
|
-
});
|
|
32
|
-
afterEach(async () => {
|
|
33
|
-
await fs.rm(tmpDir, { recursive: true, force: true });
|
|
34
|
-
vi.restoreAllMocks();
|
|
35
|
-
});
|
|
36
|
-
function hookPath() {
|
|
37
|
-
return path.join(tmpDir, '.git', 'hooks', 'pre-push');
|
|
38
|
-
}
|
|
39
|
-
describe('hook install', () => {
|
|
40
|
-
it('creates hook at the correct path', async () => {
|
|
41
|
-
await installAction({ failOn: 'critical', fast: true });
|
|
42
|
-
const content = await fs.readFile(hookPath(), 'utf-8');
|
|
43
|
-
expect(content).toBeTruthy();
|
|
44
|
-
});
|
|
45
|
-
it('hook contains marker and is executable', async () => {
|
|
46
|
-
await installAction({ failOn: 'critical', fast: true });
|
|
47
|
-
const content = await fs.readFile(hookPath(), 'utf-8');
|
|
48
|
-
expect(content).toContain(KODUS_MARKER);
|
|
49
|
-
const stat = await fs.stat(hookPath());
|
|
50
|
-
// Check executable bit (owner)
|
|
51
|
-
expect(stat.mode & 0o100).toBeTruthy();
|
|
52
|
-
});
|
|
53
|
-
it('respects --fail-on severity option', async () => {
|
|
54
|
-
await installAction({ failOn: 'warning', fast: true });
|
|
55
|
-
const content = await fs.readFile(hookPath(), 'utf-8');
|
|
56
|
-
expect(content).toContain('--fail-on warning');
|
|
57
|
-
});
|
|
58
|
-
it('does not overwrite existing non-kodus hook without --force', async () => {
|
|
59
|
-
// Write a third-party hook
|
|
60
|
-
await fs.writeFile(hookPath(), '#!/bin/sh\necho "third-party hook"\n');
|
|
61
|
-
vi.mocked(inquirer.prompt).mockResolvedValue({ overwrite: false });
|
|
62
|
-
await installAction({});
|
|
63
|
-
const content = await fs.readFile(hookPath(), 'utf-8');
|
|
64
|
-
expect(content).toContain('third-party hook');
|
|
65
|
-
expect(content).not.toContain(KODUS_MARKER);
|
|
66
|
-
});
|
|
67
|
-
it('overwrites existing kodus hook without prompting', async () => {
|
|
68
|
-
// Write an old kodus hook
|
|
69
|
-
await fs.writeFile(hookPath(), `#!/bin/sh\n${KODUS_MARKER}\nkodus review --fail-on error\n`);
|
|
70
|
-
await installAction({ failOn: 'critical', fast: true });
|
|
71
|
-
const content = await fs.readFile(hookPath(), 'utf-8');
|
|
72
|
-
expect(content).toContain('--fail-on critical');
|
|
73
|
-
});
|
|
74
|
-
});
|
|
75
|
-
describe('hook uninstall', () => {
|
|
76
|
-
it('removes hook with kodus marker', async () => {
|
|
77
|
-
await fs.writeFile(hookPath(), `#!/bin/sh\n${KODUS_MARKER}\nkodus review\n`);
|
|
78
|
-
await uninstallAction();
|
|
79
|
-
await expect(fs.access(hookPath())).rejects.toThrow();
|
|
80
|
-
});
|
|
81
|
-
it('does not remove third-party hook', async () => {
|
|
82
|
-
await fs.writeFile(hookPath(), '#!/bin/sh\necho "third-party"\n');
|
|
83
|
-
await uninstallAction();
|
|
84
|
-
const content = await fs.readFile(hookPath(), 'utf-8');
|
|
85
|
-
expect(content).toContain('third-party');
|
|
86
|
-
});
|
|
87
|
-
});
|
|
88
|
-
describe('hook status', () => {
|
|
89
|
-
it('detects installed kodus hook', async () => {
|
|
90
|
-
const script = generateHookScript('critical', true);
|
|
91
|
-
await fs.writeFile(hookPath(), script);
|
|
92
|
-
const consoleSpy = vi.spyOn(console, 'log');
|
|
93
|
-
await statusAction();
|
|
94
|
-
const output = consoleSpy.mock.calls.map((c) => c.join(' ')).join('\n');
|
|
95
|
-
expect(output).toContain('installed');
|
|
96
|
-
expect(output).toContain('critical');
|
|
97
|
-
});
|
|
98
|
-
it('detects when no hook is installed', async () => {
|
|
99
|
-
const consoleSpy = vi.spyOn(console, 'log');
|
|
100
|
-
await statusAction();
|
|
101
|
-
const output = consoleSpy.mock.calls.map((c) => c.join(' ')).join('\n');
|
|
102
|
-
expect(output).toContain('not installed');
|
|
103
|
-
});
|
|
104
|
-
});
|
|
105
|
-
//# sourceMappingURL=hook.test.js.map
|