tasq/node_modules/@claude-flow/shared/__tests__/hooks/session-hooks.test.ts

358 lines
12 KiB
TypeScript

/**
* V3 Session Hooks Tests
*
* Tests for session-end and session-restore hook functionality.
*
* @module v3/shared/hooks/__tests__/session-hooks.test
*/
import { describe, it, expect, beforeEach, vi } from 'vitest';
import {
createHookRegistry,
createSessionHooksManager,
SessionHooksManager,
HookRegistry,
HookEvent,
InMemorySessionStorage,
} from '../../src/hooks/index.js';
describe('SessionHooksManager', () => {
let registry: HookRegistry;
let sessionManager: SessionHooksManager;
let storage: InMemorySessionStorage;
beforeEach(() => {
registry = createHookRegistry();
storage = new InMemorySessionStorage();
sessionManager = createSessionHooksManager(registry, storage);
});
describe('session lifecycle hooks', () => {
it('should register session hooks on creation', () => {
const startHooks = registry.getHandlers(HookEvent.SessionStart);
const endHooks = registry.getHandlers(HookEvent.SessionEnd);
const resumeHooks = registry.getHandlers(HookEvent.SessionResume);
expect(startHooks.some(h => h.name === 'session-hooks:start')).toBe(true);
expect(endHooks.some(h => h.name === 'session-hooks:end')).toBe(true);
expect(resumeHooks.some(h => h.name === 'session-hooks:resume')).toBe(true);
});
});
describe('session-end hook', () => {
it('should end session and return summary', async () => {
// Simulate session start by triggering tracking
const context = {
event: HookEvent.SessionStart,
timestamp: new Date(),
session: { id: 'test-session', startTime: new Date() },
};
await sessionManager['handleSessionStart'](context);
// Wait a moment to ensure duration > 0
await new Promise(resolve => setTimeout(resolve, 10));
const result = await sessionManager.executeSessionEnd();
expect(result.success).toBe(true);
expect(result.duration).toBeGreaterThanOrEqual(0);
expect(result.summary).toBeDefined();
expect(result.summary!.tasksExecuted).toBe(0);
expect(result.summary!.commandsExecuted).toBe(0);
});
it('should persist session state', async () => {
// Start session
const startContext = {
event: HookEvent.SessionStart,
timestamp: new Date(),
session: { id: 'persist-session', startTime: new Date() },
};
await sessionManager['handleSessionStart'](startContext);
const result = await sessionManager.executeSessionEnd();
expect(result.persistedState).toBeDefined();
expect(result.statePath).toBeDefined();
// Verify state was saved
const sessions = await storage.list();
expect(sessions.length).toBeGreaterThan(0);
});
it('should handle ending session without active session', async () => {
const result = await sessionManager.executeSessionEnd();
expect(result.success).toBe(true);
});
it('should reset activity tracking after session end', async () => {
// Start session
const context = {
event: HookEvent.SessionStart,
timestamp: new Date(),
session: { id: 'reset-session', startTime: new Date() },
};
await sessionManager['handleSessionStart'](context);
await sessionManager.executeSessionEnd();
expect(sessionManager.getCurrentSessionId()).toBeNull();
});
});
describe('session-restore hook', () => {
it('should restore a previous session', async () => {
// Create and end a session first
const startContext = {
event: HookEvent.SessionStart,
timestamp: new Date(),
session: { id: 'restore-test', startTime: new Date() },
};
await sessionManager['handleSessionStart'](startContext);
await sessionManager.executeSessionEnd();
// Restore the session
const result = await sessionManager.executeSessionRestore('restore-test');
expect(result.success).toBe(true);
expect(result.restoredState).toBeDefined();
expect(result.restoredState!.sessionId).toBe('restore-test');
});
it('should restore latest session when no ID specified', async () => {
// Create and end multiple sessions
for (const id of ['session-1', 'session-2', 'session-3']) {
const context = {
event: HookEvent.SessionStart,
timestamp: new Date(),
session: { id, startTime: new Date() },
};
await sessionManager['handleSessionStart'](context);
await sessionManager.executeSessionEnd();
await new Promise(resolve => setTimeout(resolve, 5));
}
const result = await sessionManager.executeSessionRestore();
expect(result.success).toBe(true);
expect(result.restoredState).toBeDefined();
});
it('should fail gracefully when session not found', async () => {
const result = await sessionManager.executeSessionRestore('non-existent');
expect(result.success).toBe(false);
expect(result.error).toBeDefined();
expect(result.warnings).toBeDefined();
});
it('should return warnings for old sessions', async () => {
// Create a session with an old timestamp
const oldSession = {
sessionId: 'old-session',
startTime: new Date(Date.now() - 8 * 24 * 60 * 60 * 1000), // 8 days ago
endTime: new Date(Date.now() - 8 * 24 * 60 * 60 * 1000),
};
await storage.save('old-session', oldSession);
const result = await sessionManager.executeSessionRestore('old-session');
expect(result.success).toBe(true);
expect(result.warnings).toBeDefined();
expect(result.warnings!.some(w => w.includes('days old'))).toBe(true);
});
it('should start a new session after restoration', async () => {
// Create and end a session
const context = {
event: HookEvent.SessionStart,
timestamp: new Date(),
session: { id: 'new-after-restore', startTime: new Date() },
};
await sessionManager['handleSessionStart'](context);
await sessionManager.executeSessionEnd();
await sessionManager.executeSessionRestore('new-after-restore');
expect(sessionManager.getCurrentSessionId()).toBeDefined();
expect(sessionManager.getCurrentSessionId()).toContain('restored');
});
it('should count restored items', async () => {
// Create a session with tasks and agents
const sessionState = {
sessionId: 'count-test',
startTime: new Date(),
endTime: new Date(),
activeTasks: [
{ id: 'task-1', description: 'Task 1', status: 'completed' as const },
{ id: 'task-2', description: 'Task 2', status: 'in_progress' as const },
],
spawnedAgents: [
{ id: 'agent-1', type: 'coder', status: 'active' as const },
],
memoryEntries: [
{ key: 'key-1', namespace: 'default', type: 'string' },
],
};
await storage.save('count-test', sessionState);
const result = await sessionManager.executeSessionRestore('count-test');
expect(result.tasksRestored).toBe(2);
expect(result.agentsRestored).toBe(1);
expect(result.memoryRestored).toBe(1);
});
});
describe('activity tracking', () => {
it('should track task executions', async () => {
// Start session
const context = {
event: HookEvent.SessionStart,
timestamp: new Date(),
session: { id: 'track-tasks', startTime: new Date() },
};
await sessionManager['handleSessionStart'](context);
// Track a successful task
await sessionManager['trackTaskExecution']({
event: HookEvent.PostTaskExecute,
timestamp: new Date(),
metadata: { success: true },
});
// Track a failed task
await sessionManager['trackTaskExecution']({
event: HookEvent.PostTaskExecute,
timestamp: new Date(),
metadata: { success: false },
});
const activity = sessionManager.getCurrentActivity();
expect(activity.tasksExecuted).toBe(2);
expect(activity.tasksSucceeded).toBe(1);
expect(activity.tasksFailed).toBe(1);
});
it('should track command executions', async () => {
// Start session
const context = {
event: HookEvent.SessionStart,
timestamp: new Date(),
session: { id: 'track-commands', startTime: new Date() },
};
await sessionManager['handleSessionStart'](context);
// Track commands
await sessionManager['trackCommandExecution']({
event: HookEvent.PostCommand,
timestamp: new Date(),
});
await sessionManager['trackCommandExecution']({
event: HookEvent.PostCommand,
timestamp: new Date(),
});
const activity = sessionManager.getCurrentActivity();
expect(activity.commandsExecuted).toBe(2);
});
it('should track file modifications', async () => {
// Start session
const context = {
event: HookEvent.SessionStart,
timestamp: new Date(),
session: { id: 'track-files', startTime: new Date() },
};
await sessionManager['handleSessionStart'](context);
// Track file modifications
await sessionManager['trackFileModification']({
event: HookEvent.PostEdit,
timestamp: new Date(),
file: { path: '/src/file1.ts', operation: 'edit' },
});
await sessionManager['trackFileModification']({
event: HookEvent.PostEdit,
timestamp: new Date(),
file: { path: '/src/file2.ts', operation: 'edit' },
});
// Same file again
await sessionManager['trackFileModification']({
event: HookEvent.PostEdit,
timestamp: new Date(),
file: { path: '/src/file1.ts', operation: 'edit' },
});
const activity = sessionManager.getCurrentActivity();
expect(activity.filesModified.size).toBe(2);
});
it('should track agent spawns', async () => {
// Start session
const context = {
event: HookEvent.SessionStart,
timestamp: new Date(),
session: { id: 'track-agents', startTime: new Date() },
};
await sessionManager['handleSessionStart'](context);
// Track agent spawns
await sessionManager['trackAgentSpawn']({
event: HookEvent.PostAgentSpawn,
timestamp: new Date(),
agent: { id: 'agent-1', type: 'coder' },
});
await sessionManager['trackAgentSpawn']({
event: HookEvent.PostAgentSpawn,
timestamp: new Date(),
agent: { id: 'agent-2', type: 'tester' },
});
const activity = sessionManager.getCurrentActivity();
expect(activity.agentsSpawned.size).toBe(2);
});
});
describe('session management', () => {
it('should list available sessions', async () => {
// Create multiple sessions
for (const id of ['list-1', 'list-2', 'list-3']) {
const context = {
event: HookEvent.SessionStart,
timestamp: new Date(),
session: { id, startTime: new Date() },
};
await sessionManager['handleSessionStart'](context);
await sessionManager.executeSessionEnd();
}
const sessions = await sessionManager.listSessions();
expect(sessions.length).toBe(3);
});
it('should delete a session', async () => {
// Create a session
const context = {
event: HookEvent.SessionStart,
timestamp: new Date(),
session: { id: 'delete-me', startTime: new Date() },
};
await sessionManager['handleSessionStart'](context);
await sessionManager.executeSessionEnd();
const deleted = await sessionManager.deleteSession('delete-me');
expect(deleted).toBe(true);
const sessions = await sessionManager.listSessions();
expect(sessions.find(s => s.id === 'delete-me')).toBeUndefined();
});
it('should return false when deleting non-existent session', async () => {
const deleted = await sessionManager.deleteSession('non-existent');
expect(deleted).toBe(false);
});
});
});