@claudetree/core 0.1.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 (118) hide show
  1. package/LICENSE +21 -0
  2. package/dist/application/SessionManager.d.ts +18 -0
  3. package/dist/application/SessionManager.d.ts.map +1 -0
  4. package/dist/application/SessionManager.js +86 -0
  5. package/dist/application/SessionManager.js.map +1 -0
  6. package/dist/application/WorktreeSyncService.d.ts +17 -0
  7. package/dist/application/WorktreeSyncService.d.ts.map +1 -0
  8. package/dist/application/WorktreeSyncService.js +63 -0
  9. package/dist/application/WorktreeSyncService.js.map +1 -0
  10. package/dist/application/index.d.ts +3 -0
  11. package/dist/application/index.d.ts.map +1 -0
  12. package/dist/application/index.js +3 -0
  13. package/dist/application/index.js.map +1 -0
  14. package/dist/domain/index.d.ts +2 -0
  15. package/dist/domain/index.d.ts.map +1 -0
  16. package/dist/domain/index.js +2 -0
  17. package/dist/domain/index.js.map +1 -0
  18. package/dist/domain/repositories/ICodeReviewRepository.d.ts +7 -0
  19. package/dist/domain/repositories/ICodeReviewRepository.d.ts.map +1 -0
  20. package/dist/domain/repositories/ICodeReviewRepository.js +2 -0
  21. package/dist/domain/repositories/ICodeReviewRepository.js.map +1 -0
  22. package/dist/domain/repositories/IEventRepository.d.ts +8 -0
  23. package/dist/domain/repositories/IEventRepository.d.ts.map +1 -0
  24. package/dist/domain/repositories/IEventRepository.js +2 -0
  25. package/dist/domain/repositories/IEventRepository.js.map +1 -0
  26. package/dist/domain/repositories/ISessionRepository.d.ts +31 -0
  27. package/dist/domain/repositories/ISessionRepository.d.ts.map +1 -0
  28. package/dist/domain/repositories/ISessionRepository.js +2 -0
  29. package/dist/domain/repositories/ISessionRepository.js.map +1 -0
  30. package/dist/domain/repositories/IToolApprovalRepository.d.ts +8 -0
  31. package/dist/domain/repositories/IToolApprovalRepository.d.ts.map +1 -0
  32. package/dist/domain/repositories/IToolApprovalRepository.js +2 -0
  33. package/dist/domain/repositories/IToolApprovalRepository.js.map +1 -0
  34. package/dist/domain/repositories/IWorktreeRepository.d.ts +8 -0
  35. package/dist/domain/repositories/IWorktreeRepository.d.ts.map +1 -0
  36. package/dist/domain/repositories/IWorktreeRepository.js +2 -0
  37. package/dist/domain/repositories/IWorktreeRepository.js.map +1 -0
  38. package/dist/domain/repositories/index.d.ts +6 -0
  39. package/dist/domain/repositories/index.d.ts.map +1 -0
  40. package/dist/domain/repositories/index.js +2 -0
  41. package/dist/domain/repositories/index.js.map +1 -0
  42. package/dist/index.d.ts +4 -0
  43. package/dist/index.d.ts.map +1 -0
  44. package/dist/index.js +4 -0
  45. package/dist/index.js.map +1 -0
  46. package/dist/infra/claude/ClaudeSessionAdapter.d.ts +17 -0
  47. package/dist/infra/claude/ClaudeSessionAdapter.d.ts.map +1 -0
  48. package/dist/infra/claude/ClaudeSessionAdapter.js +162 -0
  49. package/dist/infra/claude/ClaudeSessionAdapter.js.map +1 -0
  50. package/dist/infra/claude/ClaudeSessionAdapter.test.d.ts +2 -0
  51. package/dist/infra/claude/ClaudeSessionAdapter.test.d.ts.map +1 -0
  52. package/dist/infra/claude/ClaudeSessionAdapter.test.js +122 -0
  53. package/dist/infra/claude/ClaudeSessionAdapter.test.js.map +1 -0
  54. package/dist/infra/claude/index.d.ts +2 -0
  55. package/dist/infra/claude/index.d.ts.map +1 -0
  56. package/dist/infra/claude/index.js +2 -0
  57. package/dist/infra/claude/index.js.map +1 -0
  58. package/dist/infra/git/GitWorktreeAdapter.d.ts +14 -0
  59. package/dist/infra/git/GitWorktreeAdapter.d.ts.map +1 -0
  60. package/dist/infra/git/GitWorktreeAdapter.js +74 -0
  61. package/dist/infra/git/GitWorktreeAdapter.js.map +1 -0
  62. package/dist/infra/git/GitWorktreeAdapter.test.d.ts +2 -0
  63. package/dist/infra/git/GitWorktreeAdapter.test.d.ts.map +1 -0
  64. package/dist/infra/git/GitWorktreeAdapter.test.js +104 -0
  65. package/dist/infra/git/GitWorktreeAdapter.test.js.map +1 -0
  66. package/dist/infra/git/index.d.ts +2 -0
  67. package/dist/infra/git/index.d.ts.map +1 -0
  68. package/dist/infra/git/index.js +2 -0
  69. package/dist/infra/git/index.js.map +1 -0
  70. package/dist/infra/github/GitHubAdapter.d.ts +20 -0
  71. package/dist/infra/github/GitHubAdapter.d.ts.map +1 -0
  72. package/dist/infra/github/GitHubAdapter.js +61 -0
  73. package/dist/infra/github/GitHubAdapter.js.map +1 -0
  74. package/dist/infra/github/GitHubAdapter.test.d.ts +2 -0
  75. package/dist/infra/github/GitHubAdapter.test.d.ts.map +1 -0
  76. package/dist/infra/github/GitHubAdapter.test.js +116 -0
  77. package/dist/infra/github/GitHubAdapter.test.js.map +1 -0
  78. package/dist/infra/github/index.d.ts +3 -0
  79. package/dist/infra/github/index.d.ts.map +1 -0
  80. package/dist/infra/github/index.js +2 -0
  81. package/dist/infra/github/index.js.map +1 -0
  82. package/dist/infra/index.d.ts +6 -0
  83. package/dist/infra/index.d.ts.map +1 -0
  84. package/dist/infra/index.js +6 -0
  85. package/dist/infra/index.js.map +1 -0
  86. package/dist/infra/storage/FileCodeReviewRepository.d.ts +14 -0
  87. package/dist/infra/storage/FileCodeReviewRepository.d.ts.map +1 -0
  88. package/dist/infra/storage/FileCodeReviewRepository.js +67 -0
  89. package/dist/infra/storage/FileCodeReviewRepository.js.map +1 -0
  90. package/dist/infra/storage/FileEventRepository.d.ts +16 -0
  91. package/dist/infra/storage/FileEventRepository.d.ts.map +1 -0
  92. package/dist/infra/storage/FileEventRepository.js +58 -0
  93. package/dist/infra/storage/FileEventRepository.js.map +1 -0
  94. package/dist/infra/storage/FileSessionRepository.d.ts +17 -0
  95. package/dist/infra/storage/FileSessionRepository.d.ts.map +1 -0
  96. package/dist/infra/storage/FileSessionRepository.js +68 -0
  97. package/dist/infra/storage/FileSessionRepository.js.map +1 -0
  98. package/dist/infra/storage/FileSessionRepository.test.d.ts +2 -0
  99. package/dist/infra/storage/FileSessionRepository.test.d.ts.map +1 -0
  100. package/dist/infra/storage/FileSessionRepository.test.js +84 -0
  101. package/dist/infra/storage/FileSessionRepository.test.js.map +1 -0
  102. package/dist/infra/storage/FileToolApprovalRepository.d.ts +17 -0
  103. package/dist/infra/storage/FileToolApprovalRepository.d.ts.map +1 -0
  104. package/dist/infra/storage/FileToolApprovalRepository.js +90 -0
  105. package/dist/infra/storage/FileToolApprovalRepository.js.map +1 -0
  106. package/dist/infra/storage/index.d.ts +5 -0
  107. package/dist/infra/storage/index.d.ts.map +1 -0
  108. package/dist/infra/storage/index.js +5 -0
  109. package/dist/infra/storage/index.js.map +1 -0
  110. package/dist/infra/websocket/WebSocketServer.d.ts +15 -0
  111. package/dist/infra/websocket/WebSocketServer.d.ts.map +1 -0
  112. package/dist/infra/websocket/WebSocketServer.js +45 -0
  113. package/dist/infra/websocket/WebSocketServer.js.map +1 -0
  114. package/dist/infra/websocket/index.d.ts +3 -0
  115. package/dist/infra/websocket/index.d.ts.map +1 -0
  116. package/dist/infra/websocket/index.js +2 -0
  117. package/dist/infra/websocket/index.js.map +1 -0
  118. package/package.json +40 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ClaudeSessionAdapter.js","sourceRoot":"","sources":["../../../src/infra/claude/ClaudeSessionAdapter.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=ClaudeSessionAdapter.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ClaudeSessionAdapter.test.d.ts","sourceRoot":"","sources":["../../../src/infra/claude/ClaudeSessionAdapter.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,122 @@
1
+ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
2
+ import { ClaudeSessionAdapter } from './ClaudeSessionAdapter.js';
3
+ // Mock execa
4
+ vi.mock('execa', () => ({
5
+ execa: vi.fn(),
6
+ }));
7
+ describe('ClaudeSessionAdapter', () => {
8
+ let adapter;
9
+ beforeEach(() => {
10
+ adapter = new ClaudeSessionAdapter();
11
+ vi.clearAllMocks();
12
+ });
13
+ afterEach(() => {
14
+ vi.restoreAllMocks();
15
+ });
16
+ describe('buildArgs', () => {
17
+ it('should build basic args with prompt', () => {
18
+ const args = adapter.buildArgs({
19
+ workingDir: '/tmp/test',
20
+ prompt: 'Hello',
21
+ });
22
+ expect(args).toContain('-p');
23
+ expect(args).toContain('Hello');
24
+ expect(args).toContain('--output-format');
25
+ expect(args).toContain('stream-json');
26
+ });
27
+ it('should include allowed tools when provided', () => {
28
+ const args = adapter.buildArgs({
29
+ workingDir: '/tmp/test',
30
+ prompt: 'Hello',
31
+ allowedTools: ['Read', 'Write', 'Bash'],
32
+ });
33
+ expect(args).toContain('--allowedTools');
34
+ expect(args).toContain('Read,Write,Bash');
35
+ });
36
+ it('should include system prompt when provided', () => {
37
+ const args = adapter.buildArgs({
38
+ workingDir: '/tmp/test',
39
+ prompt: 'Hello',
40
+ systemPrompt: 'You are a helpful assistant',
41
+ });
42
+ expect(args).toContain('--append-system-prompt');
43
+ expect(args).toContain('You are a helpful assistant');
44
+ });
45
+ it('should include resume flag when provided', () => {
46
+ const args = adapter.buildArgs({
47
+ workingDir: '/tmp/test',
48
+ prompt: 'Continue',
49
+ resume: 'session-123',
50
+ });
51
+ expect(args).toContain('--resume');
52
+ expect(args).toContain('session-123');
53
+ });
54
+ });
55
+ describe('parseStreamOutput', () => {
56
+ it('should parse assistant message', () => {
57
+ const line = JSON.stringify({
58
+ type: 'assistant',
59
+ message: { content: [{ type: 'text', text: 'Hello!' }] },
60
+ });
61
+ const result = adapter.parseStreamOutput(line);
62
+ expect(result).toEqual({
63
+ type: 'text',
64
+ content: 'Hello!',
65
+ timestamp: expect.any(Date),
66
+ });
67
+ });
68
+ it('should parse tool use', () => {
69
+ const line = JSON.stringify({
70
+ type: 'assistant',
71
+ message: {
72
+ content: [{ type: 'tool_use', name: 'Read', input: { path: '/tmp' } }],
73
+ },
74
+ });
75
+ const result = adapter.parseStreamOutput(line);
76
+ expect(result).toEqual({
77
+ type: 'tool_use',
78
+ content: expect.stringContaining('Read'),
79
+ timestamp: expect.any(Date),
80
+ });
81
+ });
82
+ it('should parse result message with session id', () => {
83
+ const line = JSON.stringify({
84
+ type: 'result',
85
+ session_id: 'abc-123',
86
+ });
87
+ const result = adapter.parseStreamOutput(line);
88
+ expect(result).toEqual({
89
+ type: 'done',
90
+ content: 'abc-123',
91
+ timestamp: expect.any(Date),
92
+ });
93
+ });
94
+ it('should handle invalid JSON gracefully', () => {
95
+ const result = adapter.parseStreamOutput('not json');
96
+ expect(result).toEqual({
97
+ type: 'text',
98
+ content: 'not json',
99
+ timestamp: expect.any(Date),
100
+ });
101
+ });
102
+ });
103
+ describe('isClaudeAvailable', () => {
104
+ it('should return true when claude is installed', async () => {
105
+ const { execa } = await import('execa');
106
+ vi.mocked(execa).mockResolvedValueOnce({
107
+ stdout: '/usr/local/bin/claude',
108
+ stderr: '',
109
+ exitCode: 0,
110
+ });
111
+ const result = await adapter.isClaudeAvailable();
112
+ expect(result).toBe(true);
113
+ });
114
+ it('should return false when claude is not installed', async () => {
115
+ const { execa } = await import('execa');
116
+ vi.mocked(execa).mockRejectedValueOnce(new Error('not found'));
117
+ const result = await adapter.isClaudeAvailable();
118
+ expect(result).toBe(false);
119
+ });
120
+ });
121
+ });
122
+ //# sourceMappingURL=ClaudeSessionAdapter.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ClaudeSessionAdapter.test.js","sourceRoot":"","sources":["../../../src/infra/claude/ClaudeSessionAdapter.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export { ClaudeSessionAdapter, type ClaudeOutputEvent, } from './ClaudeSessionAdapter.js';
2
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/infra/claude/index.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,oBAAoB,EACpB,KAAK,iBAAiB,GACvB,MAAM,2BAA2B,CAAC"}
@@ -0,0 +1,2 @@
1
+ export { ClaudeSessionAdapter, } from './ClaudeSessionAdapter.js';
2
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/infra/claude/index.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,oBAAoB,GAErB,MAAM,2BAA2B,CAAC"}
@@ -0,0 +1,14 @@
1
+ import type { CreateWorktreeInput, Worktree, WorktreeListItem } from '@claudetree/shared';
2
+ import type { IWorktreeRepository } from '../../domain/repositories/IWorktreeRepository.js';
3
+ export declare class GitWorktreeAdapter implements IWorktreeRepository {
4
+ private readonly repoPath;
5
+ private readonly normalizedRepoPath;
6
+ constructor(repoPath: string);
7
+ list(): Promise<WorktreeListItem[]>;
8
+ create(input: CreateWorktreeInput): Promise<Worktree>;
9
+ remove(path: string, force?: boolean): Promise<void>;
10
+ prune(): Promise<void>;
11
+ private parseWorktreeOutput;
12
+ private parseWorktreeBlock;
13
+ }
14
+ //# sourceMappingURL=GitWorktreeAdapter.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"GitWorktreeAdapter.d.ts","sourceRoot":"","sources":["../../../src/infra/git/GitWorktreeAdapter.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EACV,mBAAmB,EACnB,QAAQ,EACR,gBAAgB,EACjB,MAAM,oBAAoB,CAAC;AAC5B,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,kDAAkD,CAAC;AAE5F,qBAAa,kBAAmB,YAAW,mBAAmB;IAGhD,OAAO,CAAC,QAAQ,CAAC,QAAQ;IAFrC,OAAO,CAAC,QAAQ,CAAC,kBAAkB,CAAS;gBAEf,QAAQ,EAAE,MAAM;IAIvC,IAAI,IAAI,OAAO,CAAC,gBAAgB,EAAE,CAAC;IAOnC,MAAM,CAAC,KAAK,EAAE,mBAAmB,GAAG,OAAO,CAAC,QAAQ,CAAC;IAerD,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,UAAQ,GAAG,OAAO,CAAC,IAAI,CAAC;IAQlD,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAI5B,OAAO,CAAC,mBAAmB;IAe3B,OAAO,CAAC,kBAAkB;CAsB3B"}
@@ -0,0 +1,74 @@
1
+ import { execa } from 'execa';
2
+ import { randomUUID } from 'node:crypto';
3
+ import { realpathSync } from 'node:fs';
4
+ export class GitWorktreeAdapter {
5
+ repoPath;
6
+ normalizedRepoPath;
7
+ constructor(repoPath) {
8
+ this.repoPath = repoPath;
9
+ this.normalizedRepoPath = realpathSync(repoPath);
10
+ }
11
+ async list() {
12
+ const { stdout } = await execa('git', ['worktree', 'list', '--porcelain'], {
13
+ cwd: this.repoPath,
14
+ });
15
+ return this.parseWorktreeOutput(stdout);
16
+ }
17
+ async create(input) {
18
+ await execa('git', ['worktree', 'add', '-b', input.branch, input.path], {
19
+ cwd: this.repoPath,
20
+ });
21
+ return {
22
+ id: randomUUID(),
23
+ path: input.path,
24
+ branch: input.branch,
25
+ isMainWorktree: false,
26
+ issueNumber: input.issueNumber ?? null,
27
+ createdAt: new Date(),
28
+ };
29
+ }
30
+ async remove(path, force = false) {
31
+ const args = ['worktree', 'remove', path];
32
+ if (force) {
33
+ args.push('--force');
34
+ }
35
+ await execa('git', args, { cwd: this.repoPath });
36
+ }
37
+ async prune() {
38
+ await execa('git', ['worktree', 'prune'], { cwd: this.repoPath });
39
+ }
40
+ parseWorktreeOutput(output) {
41
+ const worktrees = [];
42
+ const blocks = output.trim().split('\n\n');
43
+ for (const block of blocks) {
44
+ const lines = block.split('\n');
45
+ const worktree = this.parseWorktreeBlock(lines);
46
+ if (worktree) {
47
+ worktrees.push(worktree);
48
+ }
49
+ }
50
+ return worktrees;
51
+ }
52
+ parseWorktreeBlock(lines) {
53
+ let path = '';
54
+ let commit = '';
55
+ let branch = '';
56
+ for (const line of lines) {
57
+ if (line.startsWith('worktree ')) {
58
+ path = line.slice('worktree '.length);
59
+ }
60
+ else if (line.startsWith('HEAD ')) {
61
+ commit = line.slice('HEAD '.length);
62
+ }
63
+ else if (line.startsWith('branch ')) {
64
+ branch = line.slice('branch refs/heads/'.length);
65
+ }
66
+ }
67
+ if (!path)
68
+ return null;
69
+ const normalizedPath = realpathSync(path);
70
+ const isMainWorktree = normalizedPath === this.normalizedRepoPath;
71
+ return { path: normalizedPath, commit, branch, isMainWorktree };
72
+ }
73
+ }
74
+ //# sourceMappingURL=GitWorktreeAdapter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"GitWorktreeAdapter.js","sourceRoot":"","sources":["../../../src/infra/git/GitWorktreeAdapter.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=GitWorktreeAdapter.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"GitWorktreeAdapter.test.d.ts","sourceRoot":"","sources":["../../../src/infra/git/GitWorktreeAdapter.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,104 @@
1
+ import { describe, it, expect, beforeEach, afterEach } from 'vitest';
2
+ import { mkdtemp, rm, realpath } from 'node:fs/promises';
3
+ import { tmpdir } from 'node:os';
4
+ import { join } from 'node:path';
5
+ import { execa } from 'execa';
6
+ import { GitWorktreeAdapter } from './GitWorktreeAdapter.js';
7
+ describe('GitWorktreeAdapter', () => {
8
+ let testRepoPath;
9
+ let adapter;
10
+ beforeEach(async () => {
11
+ // Create temp directory for test repo (resolve symlinks for macOS)
12
+ const tempPath = await mkdtemp(join(tmpdir(), 'claudetree-test-'));
13
+ testRepoPath = await realpath(tempPath);
14
+ // Initialize git repo with initial commit
15
+ await execa('git', ['init'], { cwd: testRepoPath });
16
+ await execa('git', ['config', 'user.email', 'test@test.com'], { cwd: testRepoPath });
17
+ await execa('git', ['config', 'user.name', 'Test'], { cwd: testRepoPath });
18
+ await execa('git', ['commit', '--allow-empty', '-m', 'Initial commit'], { cwd: testRepoPath });
19
+ adapter = new GitWorktreeAdapter(testRepoPath);
20
+ });
21
+ afterEach(async () => {
22
+ // Cleanup temp directory
23
+ await rm(testRepoPath, { recursive: true, force: true });
24
+ });
25
+ describe('list', () => {
26
+ it('should return main worktree when no other worktrees exist', async () => {
27
+ const result = await adapter.list();
28
+ expect(result).toHaveLength(1);
29
+ expect(result[0]).toMatchObject({
30
+ path: testRepoPath,
31
+ isMainWorktree: true,
32
+ });
33
+ expect(result[0]?.branch).toBeDefined();
34
+ expect(result[0]?.commit).toBeDefined();
35
+ });
36
+ it('should return all worktrees', async () => {
37
+ const worktreePath = `${testRepoPath}-worktree-list`;
38
+ await execa('git', ['worktree', 'add', '-b', 'feature-1', worktreePath], {
39
+ cwd: testRepoPath,
40
+ });
41
+ const result = await adapter.list();
42
+ expect(result).toHaveLength(2);
43
+ expect(result.some((w) => w.isMainWorktree)).toBe(true);
44
+ expect(result.some((w) => w.branch === 'feature-1')).toBe(true);
45
+ // Cleanup
46
+ await execa('git', ['worktree', 'remove', worktreePath], { cwd: testRepoPath });
47
+ });
48
+ });
49
+ describe('create', () => {
50
+ it('should create a new worktree with new branch', async () => {
51
+ const worktreePath = `${testRepoPath}-create-new`;
52
+ const result = await adapter.create({
53
+ path: worktreePath,
54
+ branch: 'feature-new',
55
+ });
56
+ expect(result.path).toBe(worktreePath);
57
+ expect(result.branch).toBe('feature-new');
58
+ expect(result.isMainWorktree).toBe(false);
59
+ expect(result.id).toBeDefined();
60
+ // Verify worktree exists
61
+ const list = await adapter.list();
62
+ expect(list.some((w) => w.branch === 'feature-new')).toBe(true);
63
+ // Cleanup
64
+ await adapter.remove(worktreePath);
65
+ });
66
+ it('should store issue number if provided', async () => {
67
+ const worktreePath = `${testRepoPath}-issue`;
68
+ const result = await adapter.create({
69
+ path: worktreePath,
70
+ branch: 'issue-42',
71
+ issueNumber: 42,
72
+ });
73
+ expect(result.issueNumber).toBe(42);
74
+ // Cleanup
75
+ await adapter.remove(worktreePath);
76
+ });
77
+ });
78
+ describe('remove', () => {
79
+ it('should remove an existing worktree', async () => {
80
+ const worktreePath = `${testRepoPath}-to-remove`;
81
+ await adapter.create({ path: worktreePath, branch: 'to-remove' });
82
+ await adapter.remove(worktreePath);
83
+ const list = await adapter.list();
84
+ expect(list.some((w) => w.branch === 'to-remove')).toBe(false);
85
+ });
86
+ it('should force remove worktree with uncommitted changes', async () => {
87
+ const worktreePath = `${testRepoPath}-force-remove`;
88
+ await adapter.create({ path: worktreePath, branch: 'force-remove' });
89
+ // Create uncommitted change
90
+ await execa('touch', ['newfile.txt'], { cwd: worktreePath });
91
+ await execa('git', ['add', 'newfile.txt'], { cwd: worktreePath });
92
+ await adapter.remove(worktreePath, true);
93
+ const list = await adapter.list();
94
+ expect(list.some((w) => w.branch === 'force-remove')).toBe(false);
95
+ });
96
+ });
97
+ describe('prune', () => {
98
+ it('should prune stale worktree entries', async () => {
99
+ // This test verifies prune doesn't throw on valid repo
100
+ await expect(adapter.prune()).resolves.not.toThrow();
101
+ });
102
+ });
103
+ });
104
+ //# sourceMappingURL=GitWorktreeAdapter.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"GitWorktreeAdapter.test.js","sourceRoot":"","sources":["../../../src/infra/git/GitWorktreeAdapter.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export { GitWorktreeAdapter } from './GitWorktreeAdapter.js';
2
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/infra/git/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,kBAAkB,EAAE,MAAM,yBAAyB,CAAC"}
@@ -0,0 +1,2 @@
1
+ export { GitWorktreeAdapter } from './GitWorktreeAdapter.js';
2
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/infra/git/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,kBAAkB,EAAE,MAAM,yBAAyB,CAAC"}
@@ -0,0 +1,20 @@
1
+ import type { Issue, CreatePRInput } from '@claudetree/shared';
2
+ export interface PRResult {
3
+ number: number;
4
+ url: string;
5
+ }
6
+ export interface ParsedIssueUrl {
7
+ owner: string;
8
+ repo: string;
9
+ number: number;
10
+ }
11
+ export declare class GitHubAdapter {
12
+ private octokit;
13
+ constructor(token: string);
14
+ getIssue(owner: string, repo: string, number: number): Promise<Issue>;
15
+ createPR(input: CreatePRInput): Promise<PRResult>;
16
+ getDefaultBranch(owner: string, repo: string): Promise<string>;
17
+ parseIssueUrl(url: string): ParsedIssueUrl | null;
18
+ generateBranchName(issueNumber: number, title: string): string;
19
+ }
20
+ //# sourceMappingURL=GitHubAdapter.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"GitHubAdapter.d.ts","sourceRoot":"","sources":["../../../src/infra/github/GitHubAdapter.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,KAAK,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;AAE/D,MAAM,WAAW,QAAQ;IACvB,MAAM,EAAE,MAAM,CAAC;IACf,GAAG,EAAE,MAAM,CAAC;CACb;AAED,MAAM,WAAW,cAAc;IAC7B,KAAK,EAAE,MAAM,CAAC;IACd,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;CAChB;AAED,qBAAa,aAAa;IACxB,OAAO,CAAC,OAAO,CAAU;gBAEb,KAAK,EAAE,MAAM;IAInB,QAAQ,CAAC,KAAK,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,KAAK,CAAC;IAmBrE,QAAQ,CAAC,KAAK,EAAE,aAAa,GAAG,OAAO,CAAC,QAAQ,CAAC;IAgBjD,gBAAgB,CAAC,KAAK,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IAKpE,aAAa,CAAC,GAAG,EAAE,MAAM,GAAG,cAAc,GAAG,IAAI;IAcjD,kBAAkB,CAAC,WAAW,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,MAAM;CAW/D"}
@@ -0,0 +1,61 @@
1
+ import { Octokit } from 'octokit';
2
+ export class GitHubAdapter {
3
+ octokit;
4
+ constructor(token) {
5
+ this.octokit = new Octokit({ auth: token });
6
+ }
7
+ async getIssue(owner, repo, number) {
8
+ const { data } = await this.octokit.rest.issues.get({
9
+ owner,
10
+ repo,
11
+ issue_number: number,
12
+ });
13
+ return {
14
+ number: data.number,
15
+ title: data.title,
16
+ body: data.body ?? '',
17
+ labels: data.labels.map((l) => typeof l === 'string' ? l : l.name ?? ''),
18
+ state: data.state,
19
+ url: data.html_url,
20
+ };
21
+ }
22
+ async createPR(input) {
23
+ const { data } = await this.octokit.rest.pulls.create({
24
+ owner: input.owner,
25
+ repo: input.repo,
26
+ title: input.title,
27
+ body: input.body,
28
+ head: input.head,
29
+ base: input.base,
30
+ });
31
+ return {
32
+ number: data.number,
33
+ url: data.html_url,
34
+ };
35
+ }
36
+ async getDefaultBranch(owner, repo) {
37
+ const { data } = await this.octokit.rest.repos.get({ owner, repo });
38
+ return data.default_branch;
39
+ }
40
+ parseIssueUrl(url) {
41
+ const match = url.match(/github\.com\/([^/]+)\/([^/]+)\/(issues|pull)\/(\d+)/);
42
+ if (!match)
43
+ return null;
44
+ return {
45
+ owner: match[1],
46
+ repo: match[2],
47
+ number: parseInt(match[4], 10),
48
+ };
49
+ }
50
+ generateBranchName(issueNumber, title) {
51
+ const slug = title
52
+ .toLowerCase()
53
+ .replace(/[^a-z0-9\s-]/g, '')
54
+ .replace(/\s+/g, '-')
55
+ .replace(/-+/g, '-')
56
+ .slice(0, 30)
57
+ .replace(/-$/, '');
58
+ return `issue-${issueNumber}-${slug}`;
59
+ }
60
+ }
61
+ //# sourceMappingURL=GitHubAdapter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"GitHubAdapter.js","sourceRoot":"","sources":["../../../src/infra/github/GitHubAdapter.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAclC,MAAM,OAAO,aAAa;IAChB,OAAO,CAAU;IAEzB,YAAY,KAAa;QACvB,IAAI,CAAC,OAAO,GAAG,IAAI,OAAO,CAAC,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC;IAC9C,CAAC;IAED,KAAK,CAAC,QAAQ,CAAC,KAAa,EAAE,IAAY,EAAE,MAAc;QACxD,MAAM,EAAE,IAAI,EAAE,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC;YAClD,KAAK;YACL,IAAI;YACJ,YAAY,EAAE,MAAM;SACrB,CAAC,CAAC;QAEH,OAAO;YACL,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,IAAI,EAAE,IAAI,CAAC,IAAI,IAAI,EAAE;YACrB,MAAM,EAAE,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAC5B,OAAO,CAAC,KAAK,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,IAAI,EAAE,CACzC;YACD,KAAK,EAAE,IAAI,CAAC,KAA0B;YACtC,GAAG,EAAE,IAAI,CAAC,QAAQ;SACnB,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,QAAQ,CAAC,KAAoB;QACjC,MAAM,EAAE,IAAI,EAAE,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC;YACpD,KAAK,EAAE,KAAK,CAAC,KAAK;YAClB,IAAI,EAAE,KAAK,CAAC,IAAI;YAChB,KAAK,EAAE,KAAK,CAAC,KAAK;YAClB,IAAI,EAAE,KAAK,CAAC,IAAI;YAChB,IAAI,EAAE,KAAK,CAAC,IAAI;YAChB,IAAI,EAAE,KAAK,CAAC,IAAI;SACjB,CAAC,CAAC;QAEH,OAAO;YACL,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,GAAG,EAAE,IAAI,CAAC,QAAQ;SACnB,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,gBAAgB,CAAC,KAAa,EAAE,IAAY;QAChD,MAAM,EAAE,IAAI,EAAE,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;QACpE,OAAO,IAAI,CAAC,cAAc,CAAC;IAC7B,CAAC;IAED,aAAa,CAAC,GAAW;QACvB,MAAM,KAAK,GAAG,GAAG,CAAC,KAAK,CACrB,qDAAqD,CACtD,CAAC;QAEF,IAAI,CAAC,KAAK;YAAE,OAAO,IAAI,CAAC;QAExB,OAAO;YACL,KAAK,EAAE,KAAK,CAAC,CAAC,CAAE;YAChB,IAAI,EAAE,KAAK,CAAC,CAAC,CAAE;YACf,MAAM,EAAE,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAE,EAAE,EAAE,CAAC;SAChC,CAAC;IACJ,CAAC;IAED,kBAAkB,CAAC,WAAmB,EAAE,KAAa;QACnD,MAAM,IAAI,GAAG,KAAK;aACf,WAAW,EAAE;aACb,OAAO,CAAC,eAAe,EAAE,EAAE,CAAC;aAC5B,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC;aACpB,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC;aACnB,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC;aACZ,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;QAErB,OAAO,SAAS,WAAW,IAAI,IAAI,EAAE,CAAC;IACxC,CAAC;CACF"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=GitHubAdapter.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"GitHubAdapter.test.d.ts","sourceRoot":"","sources":["../../../src/infra/github/GitHubAdapter.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,116 @@
1
+ import { describe, it, expect, vi, beforeEach } from 'vitest';
2
+ import { GitHubAdapter } from './GitHubAdapter.js';
3
+ // Mock Octokit
4
+ vi.mock('octokit', () => ({
5
+ Octokit: vi.fn().mockImplementation(() => ({
6
+ rest: {
7
+ issues: {
8
+ get: vi.fn().mockResolvedValue({
9
+ data: {
10
+ number: 42,
11
+ title: 'Add feature X',
12
+ body: 'We need to implement feature X',
13
+ labels: [{ name: 'enhancement' }, { name: 'priority-high' }],
14
+ state: 'open',
15
+ html_url: 'https://github.com/owner/repo/issues/42',
16
+ },
17
+ }),
18
+ },
19
+ pulls: {
20
+ create: vi.fn().mockResolvedValue({
21
+ data: {
22
+ number: 123,
23
+ html_url: 'https://github.com/owner/repo/pull/123',
24
+ },
25
+ }),
26
+ },
27
+ repos: {
28
+ get: vi.fn().mockResolvedValue({
29
+ data: {
30
+ default_branch: 'main',
31
+ },
32
+ }),
33
+ },
34
+ },
35
+ })),
36
+ }));
37
+ describe('GitHubAdapter', () => {
38
+ let adapter;
39
+ beforeEach(() => {
40
+ adapter = new GitHubAdapter('fake-token');
41
+ vi.clearAllMocks();
42
+ });
43
+ describe('getIssue', () => {
44
+ it('should fetch and parse issue data', async () => {
45
+ const issue = await adapter.getIssue('owner', 'repo', 42);
46
+ expect(issue).toEqual({
47
+ number: 42,
48
+ title: 'Add feature X',
49
+ body: 'We need to implement feature X',
50
+ labels: ['enhancement', 'priority-high'],
51
+ state: 'open',
52
+ url: 'https://github.com/owner/repo/issues/42',
53
+ });
54
+ });
55
+ });
56
+ describe('createPR', () => {
57
+ it('should create a pull request', async () => {
58
+ const result = await adapter.createPR({
59
+ owner: 'owner',
60
+ repo: 'repo',
61
+ title: 'feat: Add feature X',
62
+ body: 'Implements #42',
63
+ head: 'issue-42',
64
+ base: 'main',
65
+ });
66
+ expect(result).toEqual({
67
+ number: 123,
68
+ url: 'https://github.com/owner/repo/pull/123',
69
+ });
70
+ });
71
+ });
72
+ describe('getDefaultBranch', () => {
73
+ it('should return default branch name', async () => {
74
+ const branch = await adapter.getDefaultBranch('owner', 'repo');
75
+ expect(branch).toBe('main');
76
+ });
77
+ });
78
+ describe('parseIssueUrl', () => {
79
+ it('should parse GitHub issue URL', () => {
80
+ const result = adapter.parseIssueUrl('https://github.com/owner/repo/issues/42');
81
+ expect(result).toEqual({
82
+ owner: 'owner',
83
+ repo: 'repo',
84
+ number: 42,
85
+ });
86
+ });
87
+ it('should return null for invalid URL', () => {
88
+ const result = adapter.parseIssueUrl('not-a-url');
89
+ expect(result).toBeNull();
90
+ });
91
+ it('should parse PR URL as well', () => {
92
+ const result = adapter.parseIssueUrl('https://github.com/owner/repo/pull/99');
93
+ expect(result).toEqual({
94
+ owner: 'owner',
95
+ repo: 'repo',
96
+ number: 99,
97
+ });
98
+ });
99
+ });
100
+ describe('generateBranchName', () => {
101
+ it('should generate branch name from issue', () => {
102
+ const branch = adapter.generateBranchName(42, 'Add feature X');
103
+ expect(branch).toBe('issue-42-add-feature-x');
104
+ });
105
+ it('should truncate long titles', () => {
106
+ const longTitle = 'This is a very long issue title that should be truncated';
107
+ const branch = adapter.generateBranchName(1, longTitle);
108
+ expect(branch.length).toBeLessThanOrEqual(50);
109
+ });
110
+ it('should handle special characters', () => {
111
+ const branch = adapter.generateBranchName(5, 'Fix: bug [critical] #123');
112
+ expect(branch).toBe('issue-5-fix-bug-critical-123');
113
+ });
114
+ });
115
+ });
116
+ //# sourceMappingURL=GitHubAdapter.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"GitHubAdapter.test.js","sourceRoot":"","sources":["../../../src/infra/github/GitHubAdapter.test.ts"],"names":[],"mappings":"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"}