mirror of
https://github.com/p-stream/backend.git
synced 2026-04-28 17:33:29 +00:00
196 lines
6.9 KiB
TypeScript
196 lines
6.9 KiB
TypeScript
import { useAuth } from '~/utils/auth';
|
|
import { z } from 'zod';
|
|
import { scopedLogger } from '~/utils/logger';
|
|
|
|
const log = scopedLogger('user-settings');
|
|
|
|
interface UserSettings {
|
|
id: string;
|
|
application_theme: string | null;
|
|
application_language: string;
|
|
default_subtitle_language: string | null;
|
|
proxy_urls: string[];
|
|
trakt_key: string | null;
|
|
febbox_key: string | null;
|
|
enable_thumbnails: boolean;
|
|
enable_autoplay: boolean;
|
|
enable_skip_credits: boolean;
|
|
enable_discover: boolean;
|
|
enable_featured: boolean;
|
|
enable_details_modal: boolean;
|
|
enable_image_logos: boolean;
|
|
enable_carousel_view: boolean;
|
|
source_order: string[];
|
|
enable_source_order: boolean;
|
|
proxy_tmdb: boolean;
|
|
}
|
|
|
|
const userSettingsSchema = z.object({
|
|
applicationTheme: z.string().nullable().optional(),
|
|
applicationLanguage: z.string().optional().default('en'),
|
|
defaultSubtitleLanguage: z.string().nullable().optional(),
|
|
proxyUrls: z.array(z.string()).nullable().optional(),
|
|
traktKey: z.string().nullable().optional(),
|
|
febboxKey: z.string().nullable().optional(),
|
|
enableThumbnails: z.boolean().optional().default(false),
|
|
enableAutoplay: z.boolean().optional().default(true),
|
|
enableSkipCredits: z.boolean().optional().default(true),
|
|
enableDiscover: z.boolean().optional().default(true),
|
|
enableFeatured: z.boolean().optional().default(false),
|
|
enableDetailsModal: z.boolean().optional().default(false),
|
|
enableImageLogos: z.boolean().optional().default(true),
|
|
enableCarouselView: z.boolean().optional().default(false),
|
|
sourceOrder: z.array(z.string()).optional().default([]),
|
|
enableSourceOrder: z.boolean().optional().default(false),
|
|
proxyTmdb: z.boolean().optional().default(false),
|
|
});
|
|
|
|
export default defineEventHandler(async event => {
|
|
const userId = event.context.params?.id;
|
|
|
|
const session = await useAuth().getCurrentSession();
|
|
|
|
if (session.user !== userId) {
|
|
throw createError({
|
|
statusCode: 403,
|
|
message: 'Permission denied',
|
|
});
|
|
}
|
|
|
|
// First check if user exists
|
|
const user = await prisma.users.findUnique({
|
|
where: { id: userId },
|
|
});
|
|
|
|
if (!user) {
|
|
throw createError({
|
|
statusCode: 404,
|
|
message: 'User not found',
|
|
});
|
|
}
|
|
|
|
if (event.method === 'GET') {
|
|
try {
|
|
const settings = await prisma.user_settings.findUnique({
|
|
where: { id: userId },
|
|
}) as unknown as UserSettings | null;
|
|
|
|
return {
|
|
id: userId,
|
|
applicationTheme: settings?.application_theme || null,
|
|
applicationLanguage: settings?.application_language || 'en',
|
|
defaultSubtitleLanguage: settings?.default_subtitle_language || null,
|
|
proxyUrls: settings?.proxy_urls.length === 0 ? null : settings?.proxy_urls || null,
|
|
traktKey: settings?.trakt_key || null,
|
|
febboxKey: settings?.febbox_key || null,
|
|
enableThumbnails: settings?.enable_thumbnails ?? false,
|
|
enableAutoplay: settings?.enable_autoplay ?? true,
|
|
enableSkipCredits: settings?.enable_skip_credits ?? true,
|
|
enableDiscover: settings?.enable_discover ?? true,
|
|
enableFeatured: settings?.enable_featured ?? false,
|
|
enableDetailsModal: settings?.enable_details_modal ?? false,
|
|
enableImageLogos: settings?.enable_image_logos ?? true,
|
|
enableCarouselView: settings?.enable_carousel_view ?? false,
|
|
sourceOrder: settings?.source_order || [],
|
|
enableSourceOrder: settings?.enable_source_order ?? false,
|
|
proxyTmdb: settings?.proxy_tmdb ?? false,
|
|
};
|
|
} catch (error) {
|
|
log.error('Failed to get user settings', {
|
|
userId,
|
|
error: error instanceof Error ? error.message : String(error),
|
|
});
|
|
throw createError({
|
|
statusCode: 500,
|
|
message: 'Failed to get user settings',
|
|
});
|
|
}
|
|
}
|
|
|
|
if (event.method === 'PUT') {
|
|
try {
|
|
const body = await readBody(event);
|
|
log.info('Updating user settings', { userId, body });
|
|
|
|
const validatedBody = userSettingsSchema.parse(body);
|
|
|
|
const data = {
|
|
application_theme: validatedBody.applicationTheme ?? null,
|
|
application_language: validatedBody.applicationLanguage,
|
|
default_subtitle_language: validatedBody.defaultSubtitleLanguage ?? null,
|
|
proxy_urls: validatedBody.proxyUrls === null ? [] : validatedBody.proxyUrls || [],
|
|
trakt_key: validatedBody.traktKey ?? null,
|
|
febbox_key: validatedBody.febboxKey ?? null,
|
|
enable_thumbnails: validatedBody.enableThumbnails,
|
|
enable_autoplay: validatedBody.enableAutoplay,
|
|
enable_skip_credits: validatedBody.enableSkipCredits,
|
|
enable_discover: validatedBody.enableDiscover,
|
|
enable_featured: validatedBody.enableFeatured,
|
|
enable_details_modal: validatedBody.enableDetailsModal,
|
|
enable_image_logos: validatedBody.enableImageLogos,
|
|
enable_carousel_view: validatedBody.enableCarouselView,
|
|
source_order: validatedBody.sourceOrder || [],
|
|
enable_source_order: validatedBody.enableSourceOrder,
|
|
proxy_tmdb: validatedBody.proxyTmdb,
|
|
};
|
|
|
|
log.info('Preparing to upsert settings', { userId, data });
|
|
|
|
const settings = await prisma.user_settings.upsert({
|
|
where: { id: userId },
|
|
update: data,
|
|
create: {
|
|
id: userId,
|
|
...data,
|
|
},
|
|
}) as unknown as UserSettings;
|
|
|
|
log.info('Settings updated successfully', { userId });
|
|
|
|
return {
|
|
id: userId,
|
|
applicationTheme: settings.application_theme,
|
|
applicationLanguage: settings.application_language,
|
|
defaultSubtitleLanguage: settings.default_subtitle_language,
|
|
proxyUrls: settings.proxy_urls.length === 0 ? null : settings.proxy_urls,
|
|
traktKey: settings.trakt_key,
|
|
febboxKey: settings.febbox_key,
|
|
enableThumbnails: settings.enable_thumbnails,
|
|
enableAutoplay: settings.enable_autoplay,
|
|
enableSkipCredits: settings.enable_skip_credits,
|
|
enableDiscover: settings.enable_discover,
|
|
enableFeatured: settings.enable_featured,
|
|
enableDetailsModal: settings.enable_details_modal,
|
|
enableImageLogos: settings.enable_image_logos,
|
|
enableCarouselView: settings.enable_carousel_view,
|
|
sourceOrder: settings.source_order,
|
|
enableSourceOrder: settings.enable_source_order,
|
|
proxyTmdb: settings.proxy_tmdb,
|
|
};
|
|
} catch (error) {
|
|
log.error('Failed to update user settings', {
|
|
userId,
|
|
error: error instanceof Error ? error.message : String(error),
|
|
});
|
|
|
|
if (error instanceof z.ZodError) {
|
|
throw createError({
|
|
statusCode: 400,
|
|
message: 'Invalid settings data',
|
|
cause: error.errors,
|
|
});
|
|
}
|
|
|
|
throw createError({
|
|
statusCode: 500,
|
|
message: 'Failed to update settings',
|
|
cause: error instanceof Error ? error.message : 'Unknown error',
|
|
});
|
|
}
|
|
}
|
|
|
|
throw createError({
|
|
statusCode: 405,
|
|
message: 'Method not allowed',
|
|
});
|
|
});
|