hermes-agent/apps/desktop/src/lib/desktop-git.test.ts
Brooklyn Nicholson c7542358f2 fix(desktop): remote project picker UX and profile-scoped fs/git routing
Route FS/git REST through the active profile, mount the remote folder picker
at app root, keep the project dialog open while picking, show a first-run
blank state, flip into grouped view on create, and constrain the picker scroll
area so Select stays reachable.
2026-06-28 16:23:39 -05:00

93 lines
3.1 KiB
TypeScript

import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest'
import { $connection } from '@/store/session'
import { desktopGit } from './desktop-git'
const repoStatus = vi.fn(async () => ({ branch: 'main' }))
const worktreeList = vi.fn(async () => [{ branch: 'main', detached: false, isMain: true, locked: false, path: '/r' }])
const localGit = { repoStatus, review: { stage: vi.fn() }, worktreeList }
const api = vi.fn(async ({ path }: { path: string }) => {
if (path.startsWith('/api/git/status')) {
return { branch: 'remote-main' }
}
if (path.startsWith('/api/git/worktrees')) {
return { worktrees: [{ branch: 'main', detached: false, isMain: true, locked: false, path: '/srv/r' }] }
}
if (path.startsWith('/api/git/review/diff')) {
return { diff: 'remote-diff' }
}
return { ok: true }
})
describe('desktop git facade', () => {
beforeEach(() => {
vi.stubGlobal('window', { hermesDesktop: { api, git: localGit } })
$connection.set(null)
})
afterEach(() => {
vi.unstubAllGlobals()
vi.clearAllMocks()
$connection.set(null)
})
it('uses Electron git locally', async () => {
$connection.set({ mode: 'local' } as never)
await expect(desktopGit()?.repoStatus('/work')).resolves.toEqual({ branch: 'main' })
expect(repoStatus).toHaveBeenCalledWith('/work')
expect(api).not.toHaveBeenCalled()
})
it('routes reads through the backend REST mirror on a remote gateway', async () => {
$connection.set({ mode: 'remote' } as never)
await expect(desktopGit()?.repoStatus('/srv/work')).resolves.toEqual({ branch: 'remote-main' })
expect(api).toHaveBeenCalledWith({ path: '/api/git/status?path=%2Fsrv%2Fwork' })
// List endpoints unwrap their envelope to the bare array the bridge returns.
await expect(desktopGit()?.worktreeList('/srv/work')).resolves.toEqual([
{ branch: 'main', detached: false, isMain: true, locked: false, path: '/srv/r' }
])
// review.diff unwraps { diff } to a string.
await expect(desktopGit()?.review.diff('/srv/work', 'a.txt', 'uncommitted', null, false)).resolves.toBe(
'remote-diff'
)
expect(repoStatus).not.toHaveBeenCalled()
})
it('targets the active profile backend so a remote profile never touches the local repo', async () => {
$connection.set({ mode: 'remote', profile: 'remote-docker' } as never)
await desktopGit()?.repoStatus('/srv/work')
await desktopGit()?.review.stage('/srv/work', 'a.txt')
expect(api).toHaveBeenCalledWith({ path: '/api/git/status?path=%2Fsrv%2Fwork', profile: 'remote-docker' })
expect(api).toHaveBeenCalledWith({
body: { file: 'a.txt', path: '/srv/work' },
method: 'POST',
path: '/api/git/review/stage',
profile: 'remote-docker'
})
})
it('sends mutations as POST bodies on a remote gateway', async () => {
$connection.set({ mode: 'remote' } as never)
await desktopGit()?.review.stage('/srv/work', 'a.txt')
expect(api).toHaveBeenCalledWith({
body: { file: 'a.txt', path: '/srv/work' },
method: 'POST',
path: '/api/git/review/stage'
})
expect(localGit.review.stage).not.toHaveBeenCalled()
})
})