215 lines
7.8 KiB
TypeScript
215 lines
7.8 KiB
TypeScript
import React from 'react';
|
|
import { describe, it, expect, beforeEach, vi } from 'vitest';
|
|
import { render, screen, fireEvent, act } from '@testing-library/react';
|
|
import { GitSidebar } from '../../../src/renderer/components/GitSidebar/GitSidebar';
|
|
import { useAppStore } from '../../../src/renderer/store';
|
|
|
|
describe('GitSidebar', () => {
|
|
beforeEach(() => {
|
|
vi.clearAllMocks();
|
|
|
|
useAppStore.setState({
|
|
activeProject: {
|
|
id: 'project-1',
|
|
name: 'Test Project',
|
|
slug: 'test-project',
|
|
isActive: true,
|
|
createdAt: new Date().toISOString(),
|
|
updatedAt: new Date().toISOString(),
|
|
},
|
|
});
|
|
|
|
(window as any).electronAPI = {
|
|
...(window as any).electronAPI,
|
|
app: {
|
|
...(window as any).electronAPI?.app,
|
|
getDefaultProjectPath: vi.fn().mockResolvedValue('/repo/path'),
|
|
},
|
|
git: {
|
|
checkAvailability: vi.fn().mockResolvedValue({ gitFound: true, version: '2.49.0' }),
|
|
getRepoState: vi.fn().mockResolvedValue({ isRepo: false, hasRemote: false }),
|
|
init: vi.fn().mockResolvedValue({ success: true }),
|
|
ensureGitignore: vi.fn().mockResolvedValue({ updated: false, created: false, addedEntries: [] }),
|
|
onInitProgress: vi.fn().mockImplementation(() => () => {}),
|
|
},
|
|
};
|
|
});
|
|
|
|
it('checks gitignore defaults when sidebar loads', async () => {
|
|
render(<GitSidebar />);
|
|
|
|
await screen.findByRole('button', { name: /initialize git/i });
|
|
|
|
expect((window as any).electronAPI.git.ensureGitignore).toHaveBeenCalledWith('/repo/path');
|
|
});
|
|
|
|
it('shows Initialize Git button when active project is not a git repository', async () => {
|
|
render(<GitSidebar />);
|
|
|
|
expect(await screen.findByRole('button', { name: /initialize git/i })).toBeInTheDocument();
|
|
});
|
|
|
|
it('initializes repository and refreshes repo state after clicking Initialize Git', async () => {
|
|
const getRepoStateMock = vi
|
|
.fn()
|
|
.mockResolvedValueOnce({ isRepo: false, hasRemote: false })
|
|
.mockResolvedValueOnce({ isRepo: true, rootPath: '/repo/path', currentBranch: 'main', hasRemote: true });
|
|
|
|
(window as any).electronAPI.git.getRepoState = getRepoStateMock;
|
|
(window as any).electronAPI.git.init = vi.fn().mockResolvedValue({ success: true });
|
|
|
|
render(<GitSidebar />);
|
|
|
|
const initButton = await screen.findByRole('button', { name: /initialize git/i });
|
|
await act(async () => {
|
|
fireEvent.click(initButton);
|
|
});
|
|
|
|
await vi.waitFor(() => {
|
|
expect((window as any).electronAPI.git.init).toHaveBeenCalledWith('/repo/path');
|
|
});
|
|
|
|
await vi.waitFor(() => {
|
|
expect(screen.getByText(/git repository ready/i)).toBeInTheDocument();
|
|
});
|
|
});
|
|
|
|
it('passes remote url to init when user provides one', async () => {
|
|
(window as any).electronAPI.git.init = vi.fn().mockResolvedValue({ success: true });
|
|
(window as any).electronAPI.git.getRepoState = vi
|
|
.fn()
|
|
.mockResolvedValueOnce({ isRepo: false, hasRemote: false })
|
|
.mockResolvedValueOnce({ isRepo: true, rootPath: '/repo/path', currentBranch: 'main', hasRemote: true });
|
|
|
|
render(<GitSidebar />);
|
|
|
|
const remoteInput = await screen.findByPlaceholderText(/optional remote repository url/i);
|
|
await act(async () => {
|
|
fireEvent.change(remoteInput, { target: { value: 'https://github.com/example/repo.git' } });
|
|
});
|
|
|
|
expect((remoteInput as HTMLInputElement).value).toBe('https://github.com/example/repo.git');
|
|
|
|
const initButton = screen.getByRole('button', { name: /initialize git/i });
|
|
await act(async () => {
|
|
fireEvent.click(initButton);
|
|
});
|
|
|
|
await vi.waitFor(() => {
|
|
expect((window as any).electronAPI.git.init).toHaveBeenCalledWith('/repo/path', 'https://github.com/example/repo.git');
|
|
});
|
|
});
|
|
|
|
it('shows detailed progress feedback while initialization is running', async () => {
|
|
let resolveInit: ((value: { success: boolean }) => void) | null = null;
|
|
(window as any).electronAPI.git.init = vi.fn().mockImplementation(
|
|
() =>
|
|
new Promise((resolve) => {
|
|
resolveInit = resolve;
|
|
}),
|
|
);
|
|
|
|
render(<GitSidebar />);
|
|
|
|
const initButton = await screen.findByRole('button', { name: /initialize git/i });
|
|
|
|
await act(async () => {
|
|
fireEvent.click(initButton);
|
|
});
|
|
|
|
expect(screen.getByText(/preparing repository initialization/i)).toBeInTheDocument();
|
|
expect(screen.getByRole('button', { name: /initializing/i })).toBeDisabled();
|
|
|
|
await act(async () => {
|
|
resolveInit?.({ success: true });
|
|
});
|
|
});
|
|
|
|
it('updates progress detail text from init progress events', async () => {
|
|
let resolveInit: ((value: { success: boolean }) => void) | null = null;
|
|
(window as any).electronAPI.git.init = vi.fn().mockImplementation(
|
|
() =>
|
|
new Promise((resolve) => {
|
|
resolveInit = resolve;
|
|
}),
|
|
);
|
|
|
|
render(<GitSidebar />);
|
|
|
|
const onInitProgressMock = (window as any).electronAPI.git.onInitProgress as ReturnType<typeof vi.fn>;
|
|
const subscription = onInitProgressMock.mock.calls[0][0] as (payload: { message: string }) => void;
|
|
|
|
const initButton = await screen.findByRole('button', { name: /initialize git/i });
|
|
await act(async () => {
|
|
fireEvent.click(initButton);
|
|
});
|
|
|
|
await act(async () => {
|
|
subscription({ message: 'Staging project files...', progress: 75 });
|
|
});
|
|
|
|
expect(screen.getByText(/staging project files\.\.\.\s*\(75%\)/i)).toBeInTheDocument();
|
|
|
|
await act(async () => {
|
|
resolveInit?.({ success: true });
|
|
});
|
|
});
|
|
|
|
it('renders a compact transcript of initialization steps', async () => {
|
|
let resolveInit: ((value: { success: boolean }) => void) | null = null;
|
|
(window as any).electronAPI.git.init = vi.fn().mockImplementation(
|
|
() =>
|
|
new Promise((resolve) => {
|
|
resolveInit = resolve;
|
|
}),
|
|
);
|
|
|
|
render(<GitSidebar />);
|
|
|
|
const onInitProgressMock = (window as any).electronAPI.git.onInitProgress as ReturnType<typeof vi.fn>;
|
|
const subscription = onInitProgressMock.mock.calls[0][0] as (payload: { message: string; progress: number }) => void;
|
|
|
|
const initButton = await screen.findByRole('button', { name: /initialize git/i });
|
|
await act(async () => {
|
|
fireEvent.click(initButton);
|
|
});
|
|
|
|
await act(async () => {
|
|
subscription({ message: 'Checking Git availability...', progress: 5 });
|
|
subscription({ message: 'Initializing repository...', progress: 15 });
|
|
});
|
|
|
|
const transcriptToggle = screen.getByRole('button', { name: /initialization transcript/i });
|
|
expect(transcriptToggle).toBeInTheDocument();
|
|
expect(transcriptToggle).toHaveAttribute('aria-expanded', 'false');
|
|
expect(screen.queryByText(/5% — checking git availability/i)).not.toBeInTheDocument();
|
|
|
|
await act(async () => {
|
|
fireEvent.click(transcriptToggle);
|
|
});
|
|
|
|
expect(transcriptToggle).toHaveAttribute('aria-expanded', 'true');
|
|
expect(screen.getByText(/5% — checking git availability/i)).toBeInTheDocument();
|
|
expect(screen.getByText(/15% — initializing repository/i)).toBeInTheDocument();
|
|
|
|
await act(async () => {
|
|
resolveInit?.({ success: true });
|
|
});
|
|
});
|
|
|
|
it('auto-expands transcript when a failed progress event is received', async () => {
|
|
render(<GitSidebar />);
|
|
|
|
const onInitProgressMock = (window as any).electronAPI.git.onInitProgress as ReturnType<typeof vi.fn>;
|
|
const subscription = onInitProgressMock.mock.calls[0][0] as (payload: { phase: string; message: string; progress: number }) => void;
|
|
|
|
await act(async () => {
|
|
subscription({ phase: 'failed', message: 'Failed to configure remote repository.', progress: 100 });
|
|
});
|
|
|
|
const transcriptToggle = screen.getByRole('button', { name: /initialization transcript/i });
|
|
expect(transcriptToggle).toHaveAttribute('aria-expanded', 'true');
|
|
expect(screen.getByText(/100% — failed to configure remote repository/i)).toBeInTheDocument();
|
|
});
|
|
});
|