da1d2cd36a
Cherry-picked source changes from PR #889 by @Et9797. Fixes #846. Key changes: - Add ensureMemorySessionIdRegistered() guard in SessionStore.ts - Add ON UPDATE CASCADE migration (schema v21) for observations and session_summaries FK constraints - Change message queue from claim-and-delete to claim-confirm pattern (PendingMessageStore.ts) - Add spawn deduplication and unrecoverable error detection in SessionRoutes.ts and worker-service.ts - Add forceInit flag to SDKAgent for stale session recovery Build artifacts skipped (pre-existing dompurify dep issue). Path fixes (HealthMonitor.ts, worker-utils.ts) already merged via PR #634. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
199 lines
6.0 KiB
TypeScript
199 lines
6.0 KiB
TypeScript
import { describe, it, expect, mock, afterEach } from 'bun:test';
|
|
|
|
// Mock logger BEFORE imports (required pattern)
|
|
mock.module('../../src/utils/logger.js', () => ({
|
|
logger: {
|
|
info: () => {},
|
|
debug: () => {},
|
|
warn: () => {},
|
|
error: () => {},
|
|
formatTool: (toolName: string, toolInput?: any) => toolInput ? `${toolName}(...)` : toolName,
|
|
},
|
|
}));
|
|
|
|
// Import after mocks
|
|
import { extractFirstFile, groupByDate } from '../../src/shared/timeline-formatting.js';
|
|
|
|
afterEach(() => {
|
|
mock.restore();
|
|
});
|
|
|
|
describe('extractFirstFile', () => {
|
|
const cwd = '/Users/test/project';
|
|
|
|
it('should return first modified file as relative path', () => {
|
|
const filesModified = JSON.stringify(['/Users/test/project/src/app.ts', '/Users/test/project/src/utils.ts']);
|
|
|
|
const result = extractFirstFile(filesModified, cwd);
|
|
|
|
expect(result).toBe('src/app.ts');
|
|
});
|
|
|
|
it('should fall back to files_read when modified is empty', () => {
|
|
const filesModified = JSON.stringify([]);
|
|
const filesRead = JSON.stringify(['/Users/test/project/README.md']);
|
|
|
|
const result = extractFirstFile(filesModified, cwd, filesRead);
|
|
|
|
expect(result).toBe('README.md');
|
|
});
|
|
|
|
it('should return General when both are empty arrays', () => {
|
|
const filesModified = JSON.stringify([]);
|
|
const filesRead = JSON.stringify([]);
|
|
|
|
const result = extractFirstFile(filesModified, cwd, filesRead);
|
|
|
|
expect(result).toBe('General');
|
|
});
|
|
|
|
it('should return General when both are null', () => {
|
|
const result = extractFirstFile(null, cwd, null);
|
|
|
|
expect(result).toBe('General');
|
|
});
|
|
|
|
it('should handle invalid JSON in modified and fall back to read', () => {
|
|
const filesModified = 'invalid json {]';
|
|
const filesRead = JSON.stringify(['/Users/test/project/config.json']);
|
|
|
|
const result = extractFirstFile(filesModified, cwd, filesRead);
|
|
|
|
expect(result).toBe('config.json');
|
|
});
|
|
|
|
it('should return relative path (not absolute) for files inside cwd', () => {
|
|
const filesModified = JSON.stringify(['/Users/test/project/deeply/nested/file.ts']);
|
|
|
|
const result = extractFirstFile(filesModified, cwd);
|
|
|
|
expect(result).toBe('deeply/nested/file.ts');
|
|
expect(result).not.toContain('/Users/test/project');
|
|
});
|
|
|
|
it('should handle files that are already relative paths', () => {
|
|
const filesModified = JSON.stringify(['src/component.tsx']);
|
|
|
|
const result = extractFirstFile(filesModified, cwd);
|
|
|
|
expect(result).toBe('src/component.tsx');
|
|
});
|
|
});
|
|
|
|
describe('groupByDate', () => {
|
|
interface TestItem {
|
|
id: number;
|
|
date: string;
|
|
}
|
|
|
|
it('should return empty map for empty array', () => {
|
|
const items: TestItem[] = [];
|
|
|
|
const result = groupByDate(items, (item) => item.date);
|
|
|
|
expect(result.size).toBe(0);
|
|
});
|
|
|
|
it('should group items by formatted date', () => {
|
|
const items: TestItem[] = [
|
|
{ id: 1, date: '2025-01-04T10:00:00Z' },
|
|
{ id: 2, date: '2025-01-04T14:00:00Z' },
|
|
];
|
|
|
|
const result = groupByDate(items, (item) => item.date);
|
|
|
|
expect(result.size).toBe(1);
|
|
const dayItems = Array.from(result.values())[0];
|
|
expect(dayItems).toHaveLength(2);
|
|
expect(dayItems[0].id).toBe(1);
|
|
expect(dayItems[1].id).toBe(2);
|
|
});
|
|
|
|
it('should sort dates chronologically', () => {
|
|
const items: TestItem[] = [
|
|
{ id: 1, date: '2025-01-06T10:00:00Z' },
|
|
{ id: 2, date: '2025-01-04T10:00:00Z' },
|
|
{ id: 3, date: '2025-01-05T10:00:00Z' },
|
|
];
|
|
|
|
const result = groupByDate(items, (item) => item.date);
|
|
|
|
const dates = Array.from(result.keys());
|
|
expect(dates).toHaveLength(3);
|
|
// Dates should be in chronological order (oldest first)
|
|
expect(dates[0]).toContain('Jan 4');
|
|
expect(dates[1]).toContain('Jan 5');
|
|
expect(dates[2]).toContain('Jan 6');
|
|
});
|
|
|
|
it('should group multiple items on same date together', () => {
|
|
const items: TestItem[] = [
|
|
{ id: 1, date: '2025-01-04T08:00:00Z' },
|
|
{ id: 2, date: '2025-01-04T12:00:00Z' },
|
|
{ id: 3, date: '2025-01-04T18:00:00Z' },
|
|
];
|
|
|
|
const result = groupByDate(items, (item) => item.date);
|
|
|
|
expect(result.size).toBe(1);
|
|
const dayItems = Array.from(result.values())[0];
|
|
expect(dayItems).toHaveLength(3);
|
|
expect(dayItems.map(i => i.id)).toEqual([1, 2, 3]);
|
|
});
|
|
|
|
it('should handle items from different days correctly', () => {
|
|
const items: TestItem[] = [
|
|
{ id: 1, date: '2025-01-04T10:00:00Z' },
|
|
{ id: 2, date: '2025-01-05T10:00:00Z' },
|
|
{ id: 3, date: '2025-01-04T15:00:00Z' },
|
|
{ id: 4, date: '2025-01-05T20:00:00Z' },
|
|
];
|
|
|
|
const result = groupByDate(items, (item) => item.date);
|
|
|
|
expect(result.size).toBe(2);
|
|
|
|
const dates = Array.from(result.keys());
|
|
expect(dates[0]).toContain('Jan 4');
|
|
expect(dates[1]).toContain('Jan 5');
|
|
|
|
const jan4Items = result.get(dates[0])!;
|
|
const jan5Items = result.get(dates[1])!;
|
|
|
|
expect(jan4Items).toHaveLength(2);
|
|
expect(jan5Items).toHaveLength(2);
|
|
expect(jan4Items.map(i => i.id)).toEqual([1, 3]);
|
|
expect(jan5Items.map(i => i.id)).toEqual([2, 4]);
|
|
});
|
|
|
|
it('should handle numeric timestamps as date input', () => {
|
|
// Use clearly different dates (24+ hours apart to avoid timezone issues)
|
|
const items = [
|
|
{ id: 1, date: '2025-01-04T00:00:00Z' },
|
|
{ id: 2, date: '2025-01-06T00:00:00Z' }, // 2 days later
|
|
];
|
|
|
|
const result = groupByDate(items, (item) => item.date);
|
|
|
|
expect(result.size).toBe(2);
|
|
const dates = Array.from(result.keys());
|
|
expect(dates).toHaveLength(2);
|
|
expect(dates[0]).toContain('Jan 4');
|
|
expect(dates[1]).toContain('Jan 6');
|
|
});
|
|
|
|
it('should preserve item order within each date group', () => {
|
|
const items: TestItem[] = [
|
|
{ id: 3, date: '2025-01-04T08:00:00Z' },
|
|
{ id: 1, date: '2025-01-04T09:00:00Z' },
|
|
{ id: 2, date: '2025-01-04T10:00:00Z' },
|
|
];
|
|
|
|
const result = groupByDate(items, (item) => item.date);
|
|
|
|
const dayItems = Array.from(result.values())[0];
|
|
// Items should maintain their insertion order
|
|
expect(dayItems.map(i => i.id)).toEqual([3, 1, 2]);
|
|
});
|
|
});
|