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: () => {},
},
}));
// Import after mocks
import { extractFirstFile, groupByDate } from '../../src/common/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]);
});
});