mirror of
https://github.com/kikootwo/ReadMeABook.git
synced 2026-06-17 19: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.
130 lines
3.9 KiB
TypeScript
130 lines
3.9 KiB
TypeScript
/**
|
|
* Component: Notification Backend API
|
|
* Documentation: documentation/backend/services/notifications.md
|
|
*/
|
|
|
|
import { NextRequest, NextResponse } from 'next/server';
|
|
import { requireAuth, requireAdmin, AuthenticatedRequest } from '@/lib/middleware/auth';
|
|
import { prisma } from '@/lib/db';
|
|
import { getNotificationService, getRegisteredProviderTypes } from '@/lib/services/notification';
|
|
import { RMABLogger } from '@/lib/utils/logger';
|
|
import { z } from 'zod';
|
|
|
|
const logger = RMABLogger.create('API.Admin.Notifications');
|
|
|
|
const CreateBackendSchema = z.object({
|
|
type: z.string().refine((val) => getRegisteredProviderTypes().includes(val), { message: 'Unsupported notification provider type' }),
|
|
name: z.string().min(1),
|
|
config: z.record(z.any()),
|
|
events: z.array(z.enum(['request_pending_approval', 'request_approved', 'request_available', 'request_error'])).min(1),
|
|
enabled: z.boolean().default(true),
|
|
});
|
|
|
|
/**
|
|
* GET /api/admin/notifications
|
|
* List all notification backends (sensitive values masked)
|
|
*/
|
|
export async function GET(request: NextRequest) {
|
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
|
return requireAdmin(req, async () => {
|
|
try {
|
|
const backends = await prisma.notificationBackend.findMany({
|
|
orderBy: { createdAt: 'desc' },
|
|
});
|
|
|
|
const notificationService = getNotificationService();
|
|
|
|
// Mask sensitive config values
|
|
const maskedBackends = backends.map((backend) => ({
|
|
...backend,
|
|
config: notificationService.maskConfig(backend.type, backend.config),
|
|
}));
|
|
|
|
return NextResponse.json({
|
|
success: true,
|
|
backends: maskedBackends,
|
|
});
|
|
} catch (error) {
|
|
logger.error('Failed to fetch notification backends', {
|
|
error: error instanceof Error ? error.message : String(error),
|
|
});
|
|
|
|
return NextResponse.json(
|
|
{
|
|
error: 'FetchError',
|
|
message: 'Failed to fetch notification backends',
|
|
},
|
|
{ status: 500 }
|
|
);
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
/**
|
|
* POST /api/admin/notifications
|
|
* Create new notification backend
|
|
*/
|
|
export async function POST(request: NextRequest) {
|
|
return requireAuth(request, async (req: AuthenticatedRequest) => {
|
|
return requireAdmin(req, async () => {
|
|
try {
|
|
const body = await request.json();
|
|
const { type, name, config, events, enabled } = CreateBackendSchema.parse(body);
|
|
|
|
const notificationService = getNotificationService();
|
|
|
|
// Encrypt sensitive config values
|
|
const encryptedConfig = notificationService.encryptConfig(type, config);
|
|
|
|
// Create backend
|
|
const backend = await prisma.notificationBackend.create({
|
|
data: {
|
|
type,
|
|
name,
|
|
config: encryptedConfig,
|
|
events,
|
|
enabled,
|
|
},
|
|
});
|
|
|
|
logger.info(`Created notification backend: ${name} (${type})`, {
|
|
backendId: backend.id,
|
|
adminId: req.user?.sub,
|
|
});
|
|
|
|
// Return with masked values
|
|
return NextResponse.json({
|
|
success: true,
|
|
backend: {
|
|
...backend,
|
|
config: notificationService.maskConfig(type, backend.config),
|
|
},
|
|
}, { status: 201 });
|
|
} catch (error) {
|
|
logger.error('Failed to create notification backend', {
|
|
error: error instanceof Error ? error.message : String(error),
|
|
});
|
|
|
|
if (error instanceof z.ZodError) {
|
|
return NextResponse.json(
|
|
{
|
|
error: 'ValidationError',
|
|
details: error.errors,
|
|
},
|
|
{ status: 400 }
|
|
);
|
|
}
|
|
|
|
return NextResponse.json(
|
|
{
|
|
error: 'CreateError',
|
|
message: 'Failed to create notification backend',
|
|
},
|
|
{ status: 500 }
|
|
);
|
|
}
|
|
});
|
|
});
|
|
}
|