gitforest 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 (256) hide show
  1. package/.bunignore +7 -0
  2. package/.github/workflows/ci.yml +73 -0
  3. package/CLAUDE.md +111 -0
  4. package/CONTRIBUTING.md +145 -0
  5. package/README.md +168 -0
  6. package/bun.lock +267 -0
  7. package/bunfig.toml +15 -0
  8. package/cli +0 -0
  9. package/config/gitforest.example.yaml +94 -0
  10. package/docs/ai/IMPROVEMENT_PLAN.md +341 -0
  11. package/docs/ai/VERIFICATION_REPORT.md +87 -0
  12. package/docs/ai/architecture.md +169 -0
  13. package/docs/ai/checks/check-2025-12-02-tests.md +40 -0
  14. package/docs/ai/checks/check-2025-12-02.md +55 -0
  15. package/docs/ai/checks/test-verification-report.md +85 -0
  16. package/docs/ai/implementation-guide.md +776 -0
  17. package/docs/ai/research/gitty-codebase-analysis.md +221 -0
  18. package/docs/ai/tickets/GENERAL-sitrep.md +30 -0
  19. package/docs/ai/tickets/TASK-database-tests-sitrep.md +25 -0
  20. package/docs/ai/tickets/TASK-deprecated-functions-sitrep.md +28 -0
  21. package/docs/ai/tickets/TASK-detail-modal-sitrep.md +28 -0
  22. package/docs/ai/tickets/TASK-filter-overlay-sitrep.md +24 -0
  23. package/docs/ai/tickets/TASK-github-service-sitrep.md +32 -0
  24. package/docs/ai/tickets/TASK-github-token-sitrep.md +51 -0
  25. package/docs/ai/tickets/TASK-hascommits-sitrep.md +35 -0
  26. package/docs/ai/tickets/TASK-keybindings-sitrep.md +26 -0
  27. package/docs/ai/tickets/TASK-layout-sitrep.md +25 -0
  28. package/docs/ai/tickets/TASK-markdown-sitrep.md +28 -0
  29. package/docs/ai/tickets/TASK-project-item-sitrep.md +79 -0
  30. package/docs/ai/tickets/TASK-sitrep.md +28 -0
  31. package/docs/ai/tickets/TASK-state-sitrep.md +26 -0
  32. package/docs/ai/tickets/TASK-types-sitrep.md +25 -0
  33. package/docs/ai/tickets/TASK-unified-item-fix-sitrep.md +26 -0
  34. package/docs/ai/tickets/TKT-001-sitrep.md +24 -0
  35. package/docs/ai/tickets/TKT-002-sitrep.md +25 -0
  36. package/docs/ai/tickets/TKT-003-git-service-refactoring-complete.md +46 -0
  37. package/docs/ai/tickets/TKT-003-git-service-refactoring-plan.md +135 -0
  38. package/docs/ai/tickets/TKT-003-sitrep.md +26 -0
  39. package/docs/ai/tickets/TKT-004-sitrep.md +27 -0
  40. package/docs/ai/tickets/TKT-005-sitrep.md +25 -0
  41. package/docs/ai/tickets/TKT-006-sitrep.md +26 -0
  42. package/docs/ai/tickets/TKT-007-sitrep.md +30 -0
  43. package/docs/ai/tickets/TKT-008-sitrep.md +32 -0
  44. package/docs/ai/tickets/TKT-009-sitrep.md +27 -0
  45. package/docs/ai/tickets/TKT-010-sitrep.md +27 -0
  46. package/docs/ai/tickets/TKT-011-sitrep.md +26 -0
  47. package/docs/ai/tickets/TKT-012-sitrep.md +25 -0
  48. package/docs/ai/tickets/sitreps/TASK-actions-sitrep.md +28 -0
  49. package/docs/ai/tickets/sitreps/TASK-actions-test-sitrep.md +25 -0
  50. package/docs/ai/tickets/sitreps/TASK-app-integration-sitrep.md +25 -0
  51. package/docs/ai/tickets/sitreps/TASK-background-fetch-sitrep.md +24 -0
  52. package/docs/ai/tickets/sitreps/TASK-background-fetch-test-sitrep.md +29 -0
  53. package/docs/ai/tickets/sitreps/TASK-batch-tests-sitrep.md +29 -0
  54. package/docs/ai/tickets/sitreps/TASK-bun-test-sitrep.md +26 -0
  55. package/docs/ai/tickets/sitreps/TASK-cache-tests-sitrep.md +30 -0
  56. package/docs/ai/tickets/sitreps/TASK-cli-tests-sitrep.md +28 -0
  57. package/docs/ai/tickets/sitreps/TASK-clone-error-handling-sitrep.md +26 -0
  58. package/docs/ai/tickets/sitreps/TASK-commands-tests-sitrep.md +25 -0
  59. package/docs/ai/tickets/sitreps/TASK-component-tests-1-sitrep.md +30 -0
  60. package/docs/ai/tickets/sitreps/TASK-configloader-tests-sitrep.md +25 -0
  61. package/docs/ai/tickets/sitreps/TASK-confirm-dialog-test-sitrep.md +29 -0
  62. package/docs/ai/tickets/sitreps/TASK-coverage-sitrep.md +95 -0
  63. package/docs/ai/tickets/sitreps/TASK-database-tests-summary.md +61 -0
  64. package/docs/ai/tickets/sitreps/TASK-error-boundary-sitrep.md +30 -0
  65. package/docs/ai/tickets/sitreps/TASK-error-tests-sitrep.md +27 -0
  66. package/docs/ai/tickets/sitreps/TASK-errors-tests-sitrep.md +25 -0
  67. package/docs/ai/tickets/sitreps/TASK-extract-reducer-sitrep.md +27 -0
  68. package/docs/ai/tickets/sitreps/TASK-filter-overlay-test-sitrep.md +25 -0
  69. package/docs/ai/tickets/sitreps/TASK-final-verification-sitrep.md +28 -0
  70. package/docs/ai/tickets/sitreps/TASK-fix-all-tests-sitrep.md +25 -0
  71. package/docs/ai/tickets/sitreps/TASK-fix-hooks-sitrep.md +26 -0
  72. package/docs/ai/tickets/sitreps/TASK-fix-remaining-tests-sitrep.md +25 -0
  73. package/docs/ai/tickets/sitreps/TASK-fix-test-failures-sitrep.md +26 -0
  74. package/docs/ai/tickets/sitreps/TASK-fix-tests-sitrep.md +24 -0
  75. package/docs/ai/tickets/sitreps/TASK-formatters-tests-sitrep.md +25 -0
  76. package/docs/ai/tickets/sitreps/TASK-git-timeouts-sitrep.md +29 -0
  77. package/docs/ai/tickets/sitreps/TASK-github-cache-test-sitrep.md +25 -0
  78. package/docs/ai/tickets/sitreps/TASK-githubcli-tests-sitrep.md +24 -0
  79. package/docs/ai/tickets/sitreps/TASK-gitstatus-tests-sitrep.md +24 -0
  80. package/docs/ai/tickets/sitreps/TASK-hooks-isolation-sitrep.md +27 -0
  81. package/docs/ai/tickets/sitreps/TASK-keybindings-tests-sitrep.md +25 -0
  82. package/docs/ai/tickets/sitreps/TASK-layout-tests-sitrep.md +25 -0
  83. package/docs/ai/tickets/sitreps/TASK-mock-factories-sitrep.md +27 -0
  84. package/docs/ai/tickets/sitreps/TASK-modal-tests-sitrep.md +32 -0
  85. package/docs/ai/tickets/sitreps/TASK-processbatch-fix-sitrep.md +27 -0
  86. package/docs/ai/tickets/sitreps/TASK-projectlist-tests-sitrep.md +30 -0
  87. package/docs/ai/tickets/sitreps/TASK-projectutils-tests-sitrep.md +25 -0
  88. package/docs/ai/tickets/sitreps/TASK-scanner-tests-sitrep.md +29 -0
  89. package/docs/ai/tickets/sitreps/TASK-select-all-sitrep.md +25 -0
  90. package/docs/ai/tickets/sitreps/TASK-shell-error-handling-sitrep.md +27 -0
  91. package/docs/ai/tickets/sitreps/TASK-store-tests-sitrep.md +25 -0
  92. package/docs/ai/tickets/sitreps/TASK-test-fixes-sitrep.md +26 -0
  93. package/docs/ai/tickets/sitreps/TASK-test-summary-sitrep.md +25 -0
  94. package/docs/ai/tickets/sitreps/TASK-test-verification-sitrep.md +27 -0
  95. package/docs/ai/tickets/sitreps/TASK-testsuite-sitrep.md +75 -0
  96. package/docs/ai/tickets/sitreps/TASK-unified-reducer-tests-sitrep.md +29 -0
  97. package/docs/ai/tickets/sitreps/TASK-unified-repos-test-sitrep.md +29 -0
  98. package/docs/ai/tickets/sitreps/TASK-unified-tests-sitrep.md +25 -0
  99. package/docs/ai/tickets/sitreps/TASK-useprojects-tests-sitrep.md +25 -0
  100. package/docs/ai/tickets/sitreps/TASK-utility-tests-sitrep.md +32 -0
  101. package/docs/ai/tickets/sitreps/TKT-003-git-service-refactoring-sitrep.md +64 -0
  102. package/docs/ai/tkt-001-fix-database-error.md +217 -0
  103. package/docs/ai/ui-enhancement-plan.md +562 -0
  104. package/package.json +50 -0
  105. package/src/app.tsx +43 -0
  106. package/src/cli/config.ts +94 -0
  107. package/src/cli/formatters.ts +632 -0
  108. package/src/cli/index.ts +583 -0
  109. package/src/components/CloneDialog.tsx +137 -0
  110. package/src/components/ColumnHeader.tsx +128 -0
  111. package/src/components/CommandPalette.tsx +120 -0
  112. package/src/components/ConfirmDialog.tsx +105 -0
  113. package/src/components/ErrorBoundary.tsx +128 -0
  114. package/src/components/FilterBar.tsx +71 -0
  115. package/src/components/FilterOptionsOverlay.tsx +131 -0
  116. package/src/components/HelpOverlay.tsx +120 -0
  117. package/src/components/Layout.tsx +379 -0
  118. package/src/components/MarkdownRenderer.tsx +127 -0
  119. package/src/components/ProgressBar.tsx +53 -0
  120. package/src/components/ProjectItem.tsx +143 -0
  121. package/src/components/ProjectList.tsx +90 -0
  122. package/src/components/RepoDetailModal.tsx +367 -0
  123. package/src/components/StatusBar.tsx +188 -0
  124. package/src/components/UnifiedProjectItem.tsx +436 -0
  125. package/src/components/ViewModeIndicator.tsx +37 -0
  126. package/src/components/onboarding/CompleteStep.tsx +82 -0
  127. package/src/components/onboarding/DirectoriesStep.test.tsx +52 -0
  128. package/src/components/onboarding/DirectoriesStep.tsx +847 -0
  129. package/src/components/onboarding/DirectoriesStep.unit.test.ts +345 -0
  130. package/src/components/onboarding/GitHubAuthStep.tsx +268 -0
  131. package/src/components/onboarding/OnboardingWizard.tsx +130 -0
  132. package/src/components/onboarding/WelcomeStep.tsx +69 -0
  133. package/src/config/loader.ts +263 -0
  134. package/src/config/onboarding.ts +67 -0
  135. package/src/constants.ts +96 -0
  136. package/src/db/index.ts +147 -0
  137. package/src/db/schema.ts +70 -0
  138. package/src/git/commands.ts +283 -0
  139. package/src/git/index.ts +2 -0
  140. package/src/git/operations.ts +93 -0
  141. package/src/git/service.ts +539 -0
  142. package/src/git/status.ts +84 -0
  143. package/src/git/types.ts +5 -0
  144. package/src/github/auth.ts +311 -0
  145. package/src/github/cache.ts +231 -0
  146. package/src/github/cli.ts +22 -0
  147. package/src/github/unified.ts +415 -0
  148. package/src/hooks/useBackgroundFetch.ts +76 -0
  149. package/src/hooks/useConfirmDialogActions.ts +120 -0
  150. package/src/hooks/useKeyBindings.ts +656 -0
  151. package/src/hooks/useProjects.ts +47 -0
  152. package/src/hooks/useUnifiedRepos.ts +317 -0
  153. package/src/index.tsx +494 -0
  154. package/src/operations/batch.ts +280 -0
  155. package/src/operations/commands.ts +140 -0
  156. package/src/operations/index.ts +37 -0
  157. package/src/scanner/index.ts +424 -0
  158. package/src/scanner/markers.ts +43 -0
  159. package/src/scanner/submodules.ts +61 -0
  160. package/src/services/git.ts +484 -0
  161. package/src/services/github.ts +676 -0
  162. package/src/services/index.ts +28 -0
  163. package/src/services/types.ts +99 -0
  164. package/src/state/actions.ts +175 -0
  165. package/src/state/reducer.ts +294 -0
  166. package/src/state/store.tsx +216 -0
  167. package/src/state/types.ts +8 -0
  168. package/src/types/index.ts +383 -0
  169. package/src/ui/theme.ts +44 -0
  170. package/src/utils/array.ts +14 -0
  171. package/src/utils/debug.ts +38 -0
  172. package/src/utils/errors.ts +17 -0
  173. package/src/utils/index.ts +8 -0
  174. package/src/utils/markdown.ts +230 -0
  175. package/src/utils/project-utils.ts +129 -0
  176. package/src/utils/rate-limiter.ts +134 -0
  177. package/src/utils/retry.ts +147 -0
  178. package/src/utils/timeout.ts +56 -0
  179. package/test/integration/app.isolated.tsx +240 -0
  180. package/test/integration/cli-commands.test.ts +287 -0
  181. package/test/integration/cli-validation.test.ts +264 -0
  182. package/test/integration/git-operations.test.ts +218 -0
  183. package/test/integration/scanner.test.ts +228 -0
  184. package/test/preload.ts +18 -0
  185. package/test/unit/cli/commands.test.ts +13 -0
  186. package/test/unit/cli/formatters.test.ts +1116 -0
  187. package/test/unit/cli/github-commands.test.ts +12 -0
  188. package/test/unit/components/CloneDialog.test.tsx +240 -0
  189. package/test/unit/components/ColumnHeader.test.tsx +128 -0
  190. package/test/unit/components/CommandPalette.test.tsx +355 -0
  191. package/test/unit/components/ConfirmDialog.test.tsx +111 -0
  192. package/test/unit/components/ErrorBoundary.test.tsx +139 -0
  193. package/test/unit/components/FilterBar.test.tsx +43 -0
  194. package/test/unit/components/FilterOptionsOverlay.test.tsx +197 -0
  195. package/test/unit/components/HelpOverlay.test.tsx +90 -0
  196. package/test/unit/components/Layout.test.tsx +328 -0
  197. package/test/unit/components/MarkdownRenderer.test.tsx +45 -0
  198. package/test/unit/components/ProgressBar.test.tsx +138 -0
  199. package/test/unit/components/ProjectItem.test.tsx +182 -0
  200. package/test/unit/components/ProjectList.test.tsx +311 -0
  201. package/test/unit/components/RepoDetailModal.test.tsx +445 -0
  202. package/test/unit/components/StatusBar.test.tsx +112 -0
  203. package/test/unit/components/UnifiedProjectItem.test.tsx +618 -0
  204. package/test/unit/components/ViewModeIndicator.test.tsx +137 -0
  205. package/test/unit/components/test-utils.tsx +63 -0
  206. package/test/unit/config/loader.test.ts +692 -0
  207. package/test/unit/db/database.test.ts +978 -0
  208. package/test/unit/db/index.test.ts +314 -0
  209. package/test/unit/fixtures/setup.ts +186 -0
  210. package/test/unit/git/commands-untested.test.ts +205 -0
  211. package/test/unit/git/commands.test.ts +269 -0
  212. package/test/unit/git/operations.test.ts +322 -0
  213. package/test/unit/git/status.test.ts +219 -0
  214. package/test/unit/github/auth.test.ts +317 -0
  215. package/test/unit/github/cache.test.ts +1028 -0
  216. package/test/unit/github/cli.test.ts +135 -0
  217. package/test/unit/github/unified.test.ts +1201 -0
  218. package/test/unit/graceful-shutdown.test.ts +83 -0
  219. package/test/unit/hooks/useBackgroundFetch.test.tsx +239 -0
  220. package/test/unit/hooks/useConfirmDialogActions.test.tsx +81 -0
  221. package/test/unit/hooks/useKeyBindings.isolated.ts +715 -0
  222. package/test/unit/hooks/useProjects.test.tsx +186 -0
  223. package/test/unit/hooks/useUnifiedRepos-simple.test.tsx +115 -0
  224. package/test/unit/hooks/useUnifiedRepos.test.tsx +177 -0
  225. package/test/unit/mocks/config.ts +109 -0
  226. package/test/unit/mocks/git-service.ts +274 -0
  227. package/test/unit/mocks/github-service.ts +250 -0
  228. package/test/unit/mocks/index.ts +72 -0
  229. package/test/unit/mocks/project.ts +148 -0
  230. package/test/unit/mocks/state-mocks.ts +187 -0
  231. package/test/unit/mocks/unified.ts +169 -0
  232. package/test/unit/operations/batch.test.ts +216 -0
  233. package/test/unit/operations/commands.test.ts +550 -0
  234. package/test/unit/scanner/errors.test.ts +297 -0
  235. package/test/unit/scanner/index.test.ts +1011 -0
  236. package/test/unit/scanner/markers.test.ts +150 -0
  237. package/test/unit/scanner/submodules.test.ts +99 -0
  238. package/test/unit/services/git-errors.test.ts +190 -0
  239. package/test/unit/services/git.test.ts +442 -0
  240. package/test/unit/services/github-errors.test.ts +293 -0
  241. package/test/unit/services/github.test.ts +200 -0
  242. package/test/unit/state/actions.test.ts +217 -0
  243. package/test/unit/state/reducer.test.ts +745 -0
  244. package/test/unit/state/store.test.tsx +711 -0
  245. package/test/unit/types/commands.test.ts +220 -0
  246. package/test/unit/types/schema.test.ts +179 -0
  247. package/test/unit/utils/array.test.ts +73 -0
  248. package/test/unit/utils/debug.test.ts +23 -0
  249. package/test/unit/utils/errors.test.ts +295 -0
  250. package/test/unit/utils/markdown.test.ts +163 -0
  251. package/test/unit/utils/project-utils.test.ts +756 -0
  252. package/test/unit/utils/rate-limiter.test.ts +256 -0
  253. package/test/unit/utils/retry.test.ts +165 -0
  254. package/test/unit/utils/strip-ansi.ts +13 -0
  255. package/test/unit/utils/timeout.test.ts +93 -0
  256. package/tsconfig.json +29 -0
@@ -0,0 +1,12 @@
1
+ import { describe, test, expect } from "bun:test";
2
+ import { createMockConfig } from "../mocks/config.ts";
3
+
4
+ describe("GitHub CLI Commands", () => {
5
+ describe("listGitHubRepos", () => {
6
+ test("placeholder test", () => {
7
+ const config = createMockConfig();
8
+ expect(config).toBeDefined();
9
+ expect(config.github.defaultVisibility).toBe("private");
10
+ });
11
+ });
12
+ });
@@ -0,0 +1,240 @@
1
+ /**
2
+ * Tests for CloneDialog component
3
+ */
4
+
5
+ import { describe, test, expect, beforeEach, vi } from "bun:test";
6
+ import { render } from "ink-testing-library";
7
+ import { CloneDialog } from "../../../src/components/CloneDialog.tsx";
8
+ import type { UnifiedRepo, DirectoryConfig } from "../../../src/types/index.ts";
9
+
10
+ // Mock data helpers
11
+ function createMockGitHubRepo() {
12
+ return {
13
+ name: "test-repo",
14
+ fullName: "user/test-repo",
15
+ owner: "user",
16
+ description: "Test repository",
17
+ htmlUrl: "https://github.com/user/test-repo",
18
+ sshUrl: "git@github.com:user/test-repo.git",
19
+ cloneUrl: "https://github.com/user/test-repo.git",
20
+ isPrivate: false,
21
+ isArchived: false,
22
+ isFork: false,
23
+ pushedAt: new Date("2024-01-15T09:00:00Z"),
24
+ updatedAt: new Date("2024-01-14T08:00:00Z"),
25
+ defaultBranch: "main",
26
+ language: "TypeScript",
27
+ size: 1024,
28
+ };
29
+ }
30
+
31
+ describe("CloneDialog", () => {
32
+ const mockRepos: UnifiedRepo[] = [
33
+ {
34
+ id: "github-1",
35
+ name: "repo1",
36
+ source: "github",
37
+ local: null,
38
+ github: createMockGitHubRepo(),
39
+ isCloned: false,
40
+ isOnGitHub: true,
41
+ localPath: null,
42
+ },
43
+ {
44
+ id: "github-2",
45
+ name: "repo2",
46
+ source: "github",
47
+ local: null,
48
+ github: createMockGitHubRepo(),
49
+ isCloned: false,
50
+ isOnGitHub: true,
51
+ localPath: null,
52
+ },
53
+ ];
54
+
55
+ const mockDirectories: DirectoryConfig[] = [
56
+ {
57
+ path: "/home/user/projects",
58
+ label: "Projects",
59
+ maxDepth: 2,
60
+ },
61
+ {
62
+ path: "/home/user/work",
63
+ label: "Work",
64
+ maxDepth: 3,
65
+ },
66
+ {
67
+ path: "~/dev",
68
+ label: "Development",
69
+ maxDepth: 2,
70
+ },
71
+ ];
72
+
73
+ let onConfirm: ReturnType<typeof vi.fn>;
74
+ let onCancel: ReturnType<typeof vi.fn>;
75
+ let onSelectDir: ReturnType<typeof vi.fn>;
76
+ let onToggleSSH: ReturnType<typeof vi.fn>;
77
+
78
+ beforeEach(() => {
79
+ onConfirm = vi.fn();
80
+ onCancel = vi.fn();
81
+ onSelectDir = vi.fn();
82
+ onToggleSSH = vi.fn();
83
+ });
84
+
85
+ test("renders repo list", () => {
86
+ const { lastFrame } = render(
87
+ <CloneDialog
88
+ repos={mockRepos}
89
+ directories={mockDirectories}
90
+ selectedDirIndex={0}
91
+ useSSH={true}
92
+ onConfirm={onConfirm}
93
+ onCancel={onCancel}
94
+ onSelectDir={onSelectDir}
95
+ onToggleSSH={onToggleSSH}
96
+ />
97
+ );
98
+
99
+ const output = lastFrame();
100
+ expect(output).toContain("Clone GitHub Repositories (2)");
101
+ expect(output).toContain("Repositories to clone:");
102
+ expect(output).toContain("☁ user/test-repo"); // Uses GitHub full name
103
+ });
104
+
105
+ test("limits display to 5 repos and shows count", () => {
106
+ const manyRepos: UnifiedRepo[] = Array.from({ length: 10 }, (_, i) => ({
107
+ id: `github-${i}`,
108
+ name: `repo${i}`,
109
+ source: "github",
110
+ local: null,
111
+ github: createMockGitHubRepo(),
112
+ isCloned: false,
113
+ isOnGitHub: true,
114
+ localPath: null,
115
+ }));
116
+
117
+ const { lastFrame } = render(
118
+ <CloneDialog
119
+ repos={manyRepos}
120
+ directories={mockDirectories}
121
+ selectedDirIndex={0}
122
+ useSSH={true}
123
+ onConfirm={onConfirm}
124
+ onCancel={onCancel}
125
+ onSelectDir={onSelectDir}
126
+ onToggleSSH={onToggleSSH}
127
+ />
128
+ );
129
+
130
+ const output = lastFrame();
131
+ expect(output).toContain("☁ user/test-repo"); // All show same name due to mock
132
+ expect(output).toContain("...and 5 more");
133
+ });
134
+
135
+ test("renders directory options", () => {
136
+ const { lastFrame } = render(
137
+ <CloneDialog
138
+ repos={mockRepos}
139
+ directories={mockDirectories}
140
+ selectedDirIndex={1}
141
+ useSSH={true}
142
+ onConfirm={onConfirm}
143
+ onCancel={onCancel}
144
+ onSelectDir={onSelectDir}
145
+ onToggleSSH={onToggleSSH}
146
+ />
147
+ );
148
+
149
+ const output = lastFrame();
150
+ expect(output).toContain("Target directory (j/k to select):");
151
+ expect(output).toContain("○ Projects (/home/user/projects)");
152
+ expect(output).toContain("● Work (/home/user/work)"); // Selected
153
+ expect(output).toContain("○ Development (~/dev)");
154
+ });
155
+
156
+ test("renders SSH/HTTPS toggle", () => {
157
+ const { lastFrame } = render(
158
+ <CloneDialog
159
+ repos={mockRepos}
160
+ directories={mockDirectories}
161
+ selectedDirIndex={0}
162
+ useSSH={true}
163
+ onConfirm={onConfirm}
164
+ onCancel={onCancel}
165
+ onSelectDir={onSelectDir}
166
+ onToggleSSH={onToggleSSH}
167
+ />
168
+ );
169
+
170
+ const output = lastFrame();
171
+ expect(output).toContain("Protocol:");
172
+ expect(output).toContain("[●] SSH");
173
+ expect(output).toContain("[ ] HTTPS");
174
+ expect(output).toContain("(p to toggle)");
175
+ });
176
+
177
+ test("toggles to HTTPS", () => {
178
+ const { lastFrame } = render(
179
+ <CloneDialog
180
+ repos={mockRepos}
181
+ directories={mockDirectories}
182
+ selectedDirIndex={0}
183
+ useSSH={false}
184
+ onConfirm={onConfirm}
185
+ onCancel={onCancel}
186
+ onSelectDir={onSelectDir}
187
+ onToggleSSH={onToggleSSH}
188
+ />
189
+ );
190
+
191
+ const output = lastFrame();
192
+ expect(output).toContain("[ ] SSH");
193
+ expect(output).toContain("[●] HTTPS");
194
+ });
195
+
196
+ test("shows action hints", () => {
197
+ const { lastFrame } = render(
198
+ <CloneDialog
199
+ repos={mockRepos}
200
+ directories={mockDirectories}
201
+ selectedDirIndex={0}
202
+ useSSH={true}
203
+ onConfirm={onConfirm}
204
+ onCancel={onCancel}
205
+ onSelectDir={onSelectDir}
206
+ onToggleSSH={onToggleSSH}
207
+ />
208
+ );
209
+
210
+ const output = lastFrame();
211
+ expect(output).toContain("Press");
212
+ expect(output).toContain("Enter/y");
213
+ expect(output).toContain("to clone,");
214
+ expect(output).toContain("Esc/n");
215
+ expect(output).toContain("to cancel");
216
+ });
217
+
218
+ test("handles single repo title", () => {
219
+ const singleRepo = mockRepos[0];
220
+ if (!singleRepo) {
221
+ throw new Error("No mock repo available");
222
+ }
223
+
224
+ const { lastFrame } = render(
225
+ <CloneDialog
226
+ repos={[singleRepo]} // Single repo
227
+ directories={mockDirectories}
228
+ selectedDirIndex={0}
229
+ useSSH={true}
230
+ onConfirm={onConfirm}
231
+ onCancel={onCancel}
232
+ onSelectDir={onSelectDir}
233
+ onToggleSSH={onToggleSSH}
234
+ />
235
+ );
236
+
237
+ const output = lastFrame();
238
+ expect(output).toContain("Clone GitHub Repository"); // Singular title
239
+ });
240
+ });
@@ -0,0 +1,128 @@
1
+ /**
2
+ * Tests for ColumnHeader component
3
+ */
4
+
5
+ import { describe, test, expect } from "bun:test";
6
+ import { render } from "ink-testing-library";
7
+ import { ColumnHeader } from "../../../src/components/ColumnHeader.tsx";
8
+
9
+ describe("ColumnHeader", () => {
10
+ describe("rendering", () => {
11
+ test("renders column headers correctly with default props", () => {
12
+ const { lastFrame } = render(<ColumnHeader />);
13
+
14
+ const output = lastFrame();
15
+ expect(output).toBeDefined();
16
+
17
+ // Check for key headers
18
+ expect(output).toContain("Sel"); // Selection header
19
+ expect(output).toContain("Name"); // Name header
20
+ expect(output).toContain("Branch"); // Branch header
21
+ expect(output).toContain("Sync"); // Sync header
22
+ });
23
+
24
+ test("shows sort indicator for active sort column (status)", () => {
25
+ const { lastFrame } = render(
26
+ <ColumnHeader sortBy="status" sortDirection="desc" />
27
+ );
28
+
29
+ const output = lastFrame();
30
+ // Status column should have down arrow when sorted desc
31
+ expect(output).toContain("↓");
32
+ // Just verify the output is correct - ink-testing-library doesn't show ANSI codes
33
+ expect(output).toContain("Sel ↓");
34
+ });
35
+
36
+ test("shows up arrow for ascending sort", () => {
37
+ const { lastFrame } = render(
38
+ <ColumnHeader sortBy="name" sortDirection="asc" />
39
+ );
40
+
41
+ const output = lastFrame();
42
+ // Name column should have up arrow when sorted asc
43
+ expect(output).toContain("Name ↑");
44
+ });
45
+
46
+ test("handles different column configurations", () => {
47
+ const { lastFrame } = render(
48
+ <ColumnHeader sortBy="stars" sortDirection="desc" />
49
+ );
50
+
51
+ const output = lastFrame();
52
+ // Stars column should be highlighted (check for the star icon and down arrow)
53
+ expect(output).toContain(""); // star icon
54
+ });
55
+
56
+ test("shows non-sorted columns in gray", () => {
57
+ const { lastFrame } = render(
58
+ <ColumnHeader sortBy="name" sortDirection="desc" />
59
+ );
60
+
61
+ const output = lastFrame();
62
+ // Name column should have down arrow
63
+ expect(output).toContain("Name ↓");
64
+ // Verify output structure
65
+ expect(output).toContain("Sel ●"); // Status not sorted
66
+ });
67
+
68
+ test("proper spacing and alignment", () => {
69
+ const { lastFrame } = render(<ColumnHeader />);
70
+
71
+ const output = lastFrame();
72
+ // Should have proper spacing between columns
73
+ expect(output).toContain(" "); // Double spaces between some columns
74
+
75
+ // Check that headers are padded correctly
76
+ // Sel should be padded
77
+ expect(output).toMatch(/Sel\s+/);
78
+ // Name should be padded
79
+ expect(output).toMatch(/Name\s+/);
80
+ });
81
+
82
+ test("renders all column types", () => {
83
+ const { lastFrame } = render(<ColumnHeader />);
84
+
85
+ const output = lastFrame();
86
+
87
+ // Verify all expected columns are present
88
+ const columns = [
89
+ "Sel", // Selection
90
+ "Branch", // Branch
91
+ "Sync", // Sync status
92
+ ];
93
+
94
+ columns.forEach(col => {
95
+ expect(output).toContain(col);
96
+ });
97
+ });
98
+
99
+ test("handles lastActivity sort field", () => {
100
+ const { lastFrame } = render(
101
+ <ColumnHeader sortBy="lastActivity" sortDirection="desc" />
102
+ );
103
+
104
+ const output = lastFrame();
105
+ // Updated column (lastActivity) only shows clock icon, no arrow
106
+ // Just verify the output contains the clock icon
107
+ expect(output).toContain(""); // clock icon
108
+ });
109
+
110
+ test("handles size sort field", () => {
111
+ const { lastFrame } = render(
112
+ <ColumnHeader sortBy="size" sortDirection="asc" />
113
+ );
114
+
115
+ const output = lastFrame();
116
+ // Size column only shows database icon, no arrow
117
+ expect(output).toContain(""); // database icon
118
+ });
119
+
120
+ test("defaults to status column sorting", () => {
121
+ const { lastFrame } = render(<ColumnHeader />);
122
+
123
+ const output = lastFrame();
124
+ // Status column should be highlighted by default
125
+ expect(output).toContain("↓"); // Default sort is status desc
126
+ });
127
+ });
128
+ });