mirror of
https://github.com/kikootwo/ReadMeABook.git
synced 2026-06-03 12:50:09 +00:00
af0eaceb98
Introduce a provider-based notification system and wire it through the API and admin UI. Added INotificationProvider + notification service implementation and providers (apprise, discord, ntfy, pushover), plus a GET /api/admin/notifications/providers endpoint to expose provider metadata. Refactored code to use provider type strings (removed enum coupling), updated masking/encryption calls, and simplified the test notification endpoint to accept backendId or type+config and call sendToBackend directly. UI: NotificationsTab now fetches provider metadata and renders provider cards and dynamic config forms (fields driven by provider metadata). Added config field rendering, improved backend cards, and edit/delete actions. APIs: New providers route, updated admin notification CRUD routes to validate provider types dynamically, updated test route schema. Added download-client categories POST API to fetch categories from clients and wired postImportCategory handling in download-client routes. Other notable changes: BookDate now fetches Claude models dynamically from Anthropic's Models API; added paginated model fetch helper. Added ALLOW_WEAK_PASSWORD flag exposure to auth providers and password change logic. Doc updates and various tests added/updated. File-organization doc clarifies EPERM fix using stream-based copy.
119 lines
3.4 KiB
TypeScript
119 lines
3.4 KiB
TypeScript
/**
|
|
* Component: Send Notification Processor Tests
|
|
* Documentation: documentation/backend/services/notifications.md
|
|
*/
|
|
|
|
import { beforeEach, describe, expect, it, vi } from 'vitest';
|
|
|
|
const notificationServiceMock = vi.hoisted(() => ({
|
|
sendNotification: vi.fn(),
|
|
}));
|
|
|
|
vi.mock('@/lib/services/notification', () => ({
|
|
getNotificationService: () => notificationServiceMock,
|
|
}));
|
|
|
|
describe('processSendNotification', () => {
|
|
beforeEach(() => {
|
|
vi.clearAllMocks();
|
|
});
|
|
|
|
it('calls notification service with correct payload', async () => {
|
|
const { processSendNotification } = await import('@/lib/processors/send-notification.processor');
|
|
|
|
const payload = {
|
|
event: 'request_approved' as const,
|
|
requestId: 'req-1',
|
|
title: 'Test Book',
|
|
author: 'Test Author',
|
|
userName: 'Test User',
|
|
timestamp: new Date('2024-01-01T00:00:00Z'),
|
|
jobId: 'job-1',
|
|
};
|
|
|
|
await processSendNotification(payload);
|
|
|
|
expect(notificationServiceMock.sendNotification).toHaveBeenCalledWith({
|
|
event: 'request_approved',
|
|
requestId: 'req-1',
|
|
title: 'Test Book',
|
|
author: 'Test Author',
|
|
userName: 'Test User',
|
|
timestamp: expect.any(Date),
|
|
});
|
|
});
|
|
|
|
it('includes error message if provided', async () => {
|
|
const { processSendNotification } = await import('@/lib/processors/send-notification.processor');
|
|
|
|
const payload = {
|
|
event: 'request_error' as const,
|
|
requestId: 'req-1',
|
|
title: 'Test Book',
|
|
author: 'Test Author',
|
|
userName: 'Test User',
|
|
message: 'Download failed',
|
|
timestamp: new Date('2024-01-01T00:00:00Z'),
|
|
jobId: 'job-1',
|
|
};
|
|
|
|
await processSendNotification(payload);
|
|
|
|
expect(notificationServiceMock.sendNotification).toHaveBeenCalledWith({
|
|
event: 'request_error',
|
|
requestId: 'req-1',
|
|
title: 'Test Book',
|
|
author: 'Test Author',
|
|
userName: 'Test User',
|
|
message: 'Download failed',
|
|
timestamp: expect.any(Date),
|
|
});
|
|
});
|
|
|
|
it('does not throw if notification service fails', async () => {
|
|
notificationServiceMock.sendNotification.mockRejectedValue(new Error('Service error'));
|
|
|
|
const { processSendNotification } = await import('@/lib/processors/send-notification.processor');
|
|
|
|
const payload = {
|
|
event: 'request_approved' as const,
|
|
requestId: 'req-1',
|
|
title: 'Test Book',
|
|
author: 'Test Author',
|
|
userName: 'Test User',
|
|
timestamp: new Date('2024-01-01T00:00:00Z'),
|
|
jobId: 'job-1',
|
|
};
|
|
|
|
// Should not throw
|
|
await expect(processSendNotification(payload)).resolves.toBeUndefined();
|
|
});
|
|
|
|
it('processes all event types correctly', async () => {
|
|
const { processSendNotification } = await import('@/lib/processors/send-notification.processor');
|
|
|
|
const events: Array<'request_pending_approval' | 'request_approved' | 'request_available' | 'request_error'> = [
|
|
'request_pending_approval',
|
|
'request_approved',
|
|
'request_available',
|
|
'request_error',
|
|
];
|
|
|
|
for (const event of events) {
|
|
const payload = {
|
|
event,
|
|
requestId: 'req-1',
|
|
title: 'Test Book',
|
|
author: 'Test Author',
|
|
userName: 'Test User',
|
|
timestamp: new Date('2024-01-01T00:00:00Z'),
|
|
jobId: 'job-1',
|
|
};
|
|
|
|
await processSendNotification(payload);
|
|
}
|
|
|
|
expect(notificationServiceMock.sendNotification).toHaveBeenCalledTimes(4);
|
|
});
|
|
});
|