gitforest 0.1.0 → 1.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.
- package/LICENSE +21 -0
- package/package.json +24 -4
- package/src/components/onboarding/DirectoriesStep.tsx +19 -19
- package/src/github/auth.ts +3 -3
- package/src/utils/debug.ts +4 -4
- package/.bunignore +0 -7
- package/.github/workflows/ci.yml +0 -73
- package/CLAUDE.md +0 -111
- package/CONTRIBUTING.md +0 -145
- package/bun.lock +0 -267
- package/bunfig.toml +0 -15
- package/cli +0 -0
- package/docs/ai/IMPROVEMENT_PLAN.md +0 -341
- package/docs/ai/VERIFICATION_REPORT.md +0 -87
- package/docs/ai/architecture.md +0 -169
- package/docs/ai/checks/check-2025-12-02-tests.md +0 -40
- package/docs/ai/checks/check-2025-12-02.md +0 -55
- package/docs/ai/checks/test-verification-report.md +0 -85
- package/docs/ai/implementation-guide.md +0 -776
- package/docs/ai/research/gitty-codebase-analysis.md +0 -221
- package/docs/ai/tickets/GENERAL-sitrep.md +0 -30
- package/docs/ai/tickets/TASK-database-tests-sitrep.md +0 -25
- package/docs/ai/tickets/TASK-deprecated-functions-sitrep.md +0 -28
- package/docs/ai/tickets/TASK-detail-modal-sitrep.md +0 -28
- package/docs/ai/tickets/TASK-filter-overlay-sitrep.md +0 -24
- package/docs/ai/tickets/TASK-github-service-sitrep.md +0 -32
- package/docs/ai/tickets/TASK-github-token-sitrep.md +0 -51
- package/docs/ai/tickets/TASK-hascommits-sitrep.md +0 -35
- package/docs/ai/tickets/TASK-keybindings-sitrep.md +0 -26
- package/docs/ai/tickets/TASK-layout-sitrep.md +0 -25
- package/docs/ai/tickets/TASK-markdown-sitrep.md +0 -28
- package/docs/ai/tickets/TASK-project-item-sitrep.md +0 -79
- package/docs/ai/tickets/TASK-sitrep.md +0 -28
- package/docs/ai/tickets/TASK-state-sitrep.md +0 -26
- package/docs/ai/tickets/TASK-types-sitrep.md +0 -25
- package/docs/ai/tickets/TASK-unified-item-fix-sitrep.md +0 -26
- package/docs/ai/tickets/TKT-001-sitrep.md +0 -24
- package/docs/ai/tickets/TKT-002-sitrep.md +0 -25
- package/docs/ai/tickets/TKT-003-git-service-refactoring-complete.md +0 -46
- package/docs/ai/tickets/TKT-003-git-service-refactoring-plan.md +0 -135
- package/docs/ai/tickets/TKT-003-sitrep.md +0 -26
- package/docs/ai/tickets/TKT-004-sitrep.md +0 -27
- package/docs/ai/tickets/TKT-005-sitrep.md +0 -25
- package/docs/ai/tickets/TKT-006-sitrep.md +0 -26
- package/docs/ai/tickets/TKT-007-sitrep.md +0 -30
- package/docs/ai/tickets/TKT-008-sitrep.md +0 -32
- package/docs/ai/tickets/TKT-009-sitrep.md +0 -27
- package/docs/ai/tickets/TKT-010-sitrep.md +0 -27
- package/docs/ai/tickets/TKT-011-sitrep.md +0 -26
- package/docs/ai/tickets/TKT-012-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-actions-sitrep.md +0 -28
- package/docs/ai/tickets/sitreps/TASK-actions-test-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-app-integration-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-background-fetch-sitrep.md +0 -24
- package/docs/ai/tickets/sitreps/TASK-background-fetch-test-sitrep.md +0 -29
- package/docs/ai/tickets/sitreps/TASK-batch-tests-sitrep.md +0 -29
- package/docs/ai/tickets/sitreps/TASK-bun-test-sitrep.md +0 -26
- package/docs/ai/tickets/sitreps/TASK-cache-tests-sitrep.md +0 -30
- package/docs/ai/tickets/sitreps/TASK-cli-tests-sitrep.md +0 -28
- package/docs/ai/tickets/sitreps/TASK-clone-error-handling-sitrep.md +0 -26
- package/docs/ai/tickets/sitreps/TASK-commands-tests-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-component-tests-1-sitrep.md +0 -30
- package/docs/ai/tickets/sitreps/TASK-configloader-tests-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-confirm-dialog-test-sitrep.md +0 -29
- package/docs/ai/tickets/sitreps/TASK-coverage-sitrep.md +0 -95
- package/docs/ai/tickets/sitreps/TASK-database-tests-summary.md +0 -61
- package/docs/ai/tickets/sitreps/TASK-error-boundary-sitrep.md +0 -30
- package/docs/ai/tickets/sitreps/TASK-error-tests-sitrep.md +0 -27
- package/docs/ai/tickets/sitreps/TASK-errors-tests-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-extract-reducer-sitrep.md +0 -27
- package/docs/ai/tickets/sitreps/TASK-filter-overlay-test-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-final-verification-sitrep.md +0 -28
- package/docs/ai/tickets/sitreps/TASK-fix-all-tests-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-fix-hooks-sitrep.md +0 -26
- package/docs/ai/tickets/sitreps/TASK-fix-remaining-tests-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-fix-test-failures-sitrep.md +0 -26
- package/docs/ai/tickets/sitreps/TASK-fix-tests-sitrep.md +0 -24
- package/docs/ai/tickets/sitreps/TASK-formatters-tests-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-git-timeouts-sitrep.md +0 -29
- package/docs/ai/tickets/sitreps/TASK-github-cache-test-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-githubcli-tests-sitrep.md +0 -24
- package/docs/ai/tickets/sitreps/TASK-gitstatus-tests-sitrep.md +0 -24
- package/docs/ai/tickets/sitreps/TASK-hooks-isolation-sitrep.md +0 -27
- package/docs/ai/tickets/sitreps/TASK-keybindings-tests-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-layout-tests-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-mock-factories-sitrep.md +0 -27
- package/docs/ai/tickets/sitreps/TASK-modal-tests-sitrep.md +0 -32
- package/docs/ai/tickets/sitreps/TASK-processbatch-fix-sitrep.md +0 -27
- package/docs/ai/tickets/sitreps/TASK-projectlist-tests-sitrep.md +0 -30
- package/docs/ai/tickets/sitreps/TASK-projectutils-tests-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-scanner-tests-sitrep.md +0 -29
- package/docs/ai/tickets/sitreps/TASK-select-all-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-shell-error-handling-sitrep.md +0 -27
- package/docs/ai/tickets/sitreps/TASK-store-tests-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-test-fixes-sitrep.md +0 -26
- package/docs/ai/tickets/sitreps/TASK-test-summary-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-test-verification-sitrep.md +0 -27
- package/docs/ai/tickets/sitreps/TASK-testsuite-sitrep.md +0 -75
- package/docs/ai/tickets/sitreps/TASK-unified-reducer-tests-sitrep.md +0 -29
- package/docs/ai/tickets/sitreps/TASK-unified-repos-test-sitrep.md +0 -29
- package/docs/ai/tickets/sitreps/TASK-unified-tests-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-useprojects-tests-sitrep.md +0 -25
- package/docs/ai/tickets/sitreps/TASK-utility-tests-sitrep.md +0 -32
- package/docs/ai/tickets/sitreps/TKT-003-git-service-refactoring-sitrep.md +0 -64
- package/docs/ai/tkt-001-fix-database-error.md +0 -217
- package/docs/ai/ui-enhancement-plan.md +0 -562
- package/test/integration/app.isolated.tsx +0 -240
- package/test/integration/cli-commands.test.ts +0 -287
- package/test/integration/cli-validation.test.ts +0 -264
- package/test/integration/git-operations.test.ts +0 -218
- package/test/integration/scanner.test.ts +0 -228
- package/test/preload.ts +0 -18
- package/test/unit/cli/commands.test.ts +0 -13
- package/test/unit/cli/formatters.test.ts +0 -1116
- package/test/unit/cli/github-commands.test.ts +0 -12
- package/test/unit/components/CloneDialog.test.tsx +0 -240
- package/test/unit/components/ColumnHeader.test.tsx +0 -128
- package/test/unit/components/CommandPalette.test.tsx +0 -355
- package/test/unit/components/ConfirmDialog.test.tsx +0 -111
- package/test/unit/components/ErrorBoundary.test.tsx +0 -139
- package/test/unit/components/FilterBar.test.tsx +0 -43
- package/test/unit/components/FilterOptionsOverlay.test.tsx +0 -197
- package/test/unit/components/HelpOverlay.test.tsx +0 -90
- package/test/unit/components/Layout.test.tsx +0 -328
- package/test/unit/components/MarkdownRenderer.test.tsx +0 -45
- package/test/unit/components/ProgressBar.test.tsx +0 -138
- package/test/unit/components/ProjectItem.test.tsx +0 -182
- package/test/unit/components/ProjectList.test.tsx +0 -311
- package/test/unit/components/RepoDetailModal.test.tsx +0 -445
- package/test/unit/components/StatusBar.test.tsx +0 -112
- package/test/unit/components/UnifiedProjectItem.test.tsx +0 -618
- package/test/unit/components/ViewModeIndicator.test.tsx +0 -137
- package/test/unit/components/test-utils.tsx +0 -63
- package/test/unit/config/loader.test.ts +0 -692
- package/test/unit/db/database.test.ts +0 -978
- package/test/unit/db/index.test.ts +0 -314
- package/test/unit/fixtures/setup.ts +0 -186
- package/test/unit/git/commands-untested.test.ts +0 -205
- package/test/unit/git/commands.test.ts +0 -269
- package/test/unit/git/operations.test.ts +0 -322
- package/test/unit/git/status.test.ts +0 -219
- package/test/unit/github/auth.test.ts +0 -317
- package/test/unit/github/cache.test.ts +0 -1028
- package/test/unit/github/cli.test.ts +0 -135
- package/test/unit/github/unified.test.ts +0 -1201
- package/test/unit/graceful-shutdown.test.ts +0 -83
- package/test/unit/hooks/useBackgroundFetch.test.tsx +0 -239
- package/test/unit/hooks/useConfirmDialogActions.test.tsx +0 -81
- package/test/unit/hooks/useKeyBindings.isolated.ts +0 -715
- package/test/unit/hooks/useProjects.test.tsx +0 -186
- package/test/unit/hooks/useUnifiedRepos-simple.test.tsx +0 -115
- package/test/unit/hooks/useUnifiedRepos.test.tsx +0 -177
- package/test/unit/mocks/config.ts +0 -109
- package/test/unit/mocks/git-service.ts +0 -274
- package/test/unit/mocks/github-service.ts +0 -250
- package/test/unit/mocks/index.ts +0 -72
- package/test/unit/mocks/project.ts +0 -148
- package/test/unit/mocks/state-mocks.ts +0 -187
- package/test/unit/mocks/unified.ts +0 -169
- package/test/unit/operations/batch.test.ts +0 -216
- package/test/unit/operations/commands.test.ts +0 -550
- package/test/unit/scanner/errors.test.ts +0 -297
- package/test/unit/scanner/index.test.ts +0 -1011
- package/test/unit/scanner/markers.test.ts +0 -150
- package/test/unit/scanner/submodules.test.ts +0 -99
- package/test/unit/services/git-errors.test.ts +0 -190
- package/test/unit/services/git.test.ts +0 -442
- package/test/unit/services/github-errors.test.ts +0 -293
- package/test/unit/services/github.test.ts +0 -200
- package/test/unit/state/actions.test.ts +0 -217
- package/test/unit/state/reducer.test.ts +0 -745
- package/test/unit/state/store.test.tsx +0 -711
- package/test/unit/types/commands.test.ts +0 -220
- package/test/unit/types/schema.test.ts +0 -179
- package/test/unit/utils/array.test.ts +0 -73
- package/test/unit/utils/debug.test.ts +0 -23
- package/test/unit/utils/errors.test.ts +0 -295
- package/test/unit/utils/markdown.test.ts +0 -163
- package/test/unit/utils/project-utils.test.ts +0 -756
- package/test/unit/utils/rate-limiter.test.ts +0 -256
- package/test/unit/utils/retry.test.ts +0 -165
- package/test/unit/utils/strip-ansi.ts +0 -13
- package/test/unit/utils/timeout.test.ts +0 -93
- package/tsconfig.json +0 -29
|
@@ -1,12 +0,0 @@
|
|
|
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
|
-
});
|
|
@@ -1,240 +0,0 @@
|
|
|
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
|
-
});
|
|
@@ -1,128 +0,0 @@
|
|
|
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
|
-
});
|