Files
vscode/src/vs/platform/agentHost/test/node/sessionDataService.test.ts
Connor Peet da5fb11b3b agentHost: add session-specific metadata
Adds a SQLite DB for session-specific metadata. Stores edits in there.
It can _almost_ restore edits, but I still need to make undoStops be
similarly persisted. That is a project for later this evening.
2026-03-26 15:46:46 -07:00

157 lines
6.2 KiB
TypeScript

/*---------------------------------------------------------------------------------------------
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License. See License.txt in the project root for license information.
*--------------------------------------------------------------------------------------------*/
import assert from 'assert';
import { tmpdir } from 'os';
import { randomUUID } from 'crypto';
import { mkdirSync, rmSync } from 'fs';
import { VSBuffer } from '../../../../base/common/buffer.js';
import { DisposableStore } from '../../../../base/common/lifecycle.js';
import { Schemas } from '../../../../base/common/network.js';
import { URI } from '../../../../base/common/uri.js';
import { ensureNoDisposablesAreLeakedInTestSuite } from '../../../../base/test/common/utils.js';
import { FileService } from '../../../files/common/fileService.js';
import { DiskFileSystemProvider } from '../../../files/node/diskFileSystemProvider.js';
import { InMemoryFileSystemProvider } from '../../../files/common/inMemoryFilesystemProvider.js';
import { NullLogService } from '../../../log/common/log.js';
import { AgentSession } from '../../common/agentService.js';
import { SessionDataService } from '../../node/sessionDataService.js';
import { join } from '../../../../base/common/path.js';
suite('SessionDataService', () => {
const disposables = new DisposableStore();
let fileService: FileService;
let service: SessionDataService;
const basePath = URI.from({ scheme: Schemas.inMemory, path: '/userData' });
setup(() => {
fileService = disposables.add(new FileService(new NullLogService()));
disposables.add(fileService.registerProvider(Schemas.inMemory, disposables.add(new InMemoryFileSystemProvider())));
service = new SessionDataService(basePath, fileService, new NullLogService());
});
teardown(() => disposables.clear());
ensureNoDisposablesAreLeakedInTestSuite();
test('getSessionDataDir returns correct URI', () => {
const session = AgentSession.uri('copilot', 'abc-123');
const dir = service.getSessionDataDir(session);
assert.strictEqual(dir.toString(), URI.joinPath(basePath, 'agentSessionData', 'abc-123').toString());
});
test('getSessionDataDir sanitizes unsafe characters', () => {
const session = AgentSession.uri('copilot', 'foo/bar:baz\\qux');
const dir = service.getSessionDataDir(session);
assert.strictEqual(dir.toString(), URI.joinPath(basePath, 'agentSessionData', 'foo-bar-baz-qux').toString());
});
test('deleteSessionData removes directory', async () => {
const session = AgentSession.uri('copilot', 'session-1');
const dir = service.getSessionDataDir(session);
await fileService.createFolder(dir);
await fileService.writeFile(URI.joinPath(dir, 'snapshot.json'), VSBuffer.fromString('{}'));
assert.ok(await fileService.exists(dir));
await service.deleteSessionData(session);
assert.ok(!(await fileService.exists(dir)));
});
test('deleteSessionData is a no-op when directory does not exist', async () => {
const session = AgentSession.uri('copilot', 'nonexistent');
// Should not throw
await service.deleteSessionData(session);
});
test('cleanupOrphanedData deletes orphans but keeps known sessions', async () => {
const baseDir = URI.joinPath(basePath, 'agentSessionData');
await fileService.createFolder(URI.joinPath(baseDir, 'keep-1'));
await fileService.createFolder(URI.joinPath(baseDir, 'keep-2'));
await fileService.createFolder(URI.joinPath(baseDir, 'orphan-1'));
await fileService.createFolder(URI.joinPath(baseDir, 'orphan-2'));
await service.cleanupOrphanedData(new Set(['keep-1', 'keep-2']));
assert.ok(await fileService.exists(URI.joinPath(baseDir, 'keep-1')));
assert.ok(await fileService.exists(URI.joinPath(baseDir, 'keep-2')));
assert.ok(!(await fileService.exists(URI.joinPath(baseDir, 'orphan-1'))));
assert.ok(!(await fileService.exists(URI.joinPath(baseDir, 'orphan-2'))));
});
test('cleanupOrphanedData is a no-op when base directory does not exist', async () => {
// Should not throw
await service.cleanupOrphanedData(new Set());
});
});
suite('SessionDataService — openDatabase ref-counting', () => {
const disposables = new DisposableStore();
let service: SessionDataService;
let testDir: string;
setup(() => {
testDir = join(tmpdir(), `vscode-session-data-test-${randomUUID()}`);
mkdirSync(testDir, { recursive: true });
const fileService = disposables.add(new FileService(new NullLogService()));
disposables.add(fileService.registerProvider(Schemas.file, disposables.add(new DiskFileSystemProvider(new NullLogService()))));
service = new SessionDataService(URI.file(testDir), fileService, new NullLogService());
});
teardown(() => {
disposables.clear();
rmSync(testDir, { recursive: true, force: true });
});
ensureNoDisposablesAreLeakedInTestSuite();
test('returns a functional database reference', async () => {
const session = AgentSession.uri('copilot', 'ref-test');
const ref = service.openDatabase(session);
disposables.add(ref);
await ref.object.createTurn('turn-1');
const edits = await ref.object.getFileEdits([]);
assert.deepStrictEqual(edits, []);
});
test('multiple references share the same database', async () => {
const session = AgentSession.uri('copilot', 'shared-test');
const ref1 = service.openDatabase(session);
const ref2 = service.openDatabase(session);
assert.strictEqual(ref1.object, ref2.object);
ref1.dispose();
ref2.dispose();
});
test('database remains usable until last reference is disposed', async () => {
const session = AgentSession.uri('copilot', 'refcount-test');
const ref1 = service.openDatabase(session);
const ref2 = service.openDatabase(session);
ref1.dispose();
// ref2 still works
await ref2.object.createTurn('turn-1');
ref2.dispose();
});
test('new reference after all disposed gets a fresh database', async () => {
const session = AgentSession.uri('copilot', 'reopen-test');
const ref1 = service.openDatabase(session);
const db1 = ref1.object;
ref1.dispose();
const ref2 = service.openDatabase(session);
disposables.add(ref2);
// New reference — may or may not be the same object, but must be functional
await ref2.object.createTurn('turn-1');
assert.notStrictEqual(ref2.object, db1);
});
});