This repository has been archived on 2025-08-19. You can view files and clone it, but you cannot make any changes to its state, such as pushing and creating new issues, pull requests or comments.
coinstorge/website/src/routes/api/settings/+server.ts

85 lines
2.6 KiB
TypeScript
Raw Normal View History

import { auth } from '$lib/auth';
import { uploadProfilePicture } from '$lib/server/s3';
import { error, json } from '@sveltejs/kit';
import { db } from '$lib/server/db';
import { user } from '$lib/server/db/schema';
import { eq } from 'drizzle-orm';
import { MAX_FILE_SIZE } from '$lib/data/constants';
2025-05-30 14:35:15 +03:00
import { isNameAppropriate } from '$lib/server/moderation';
2025-05-30 14:35:15 +03:00
async function validateInputs(name: string, bio: string, username: string, avatarFile: File | null) {
if (name && name.length < 1) {
throw error(400, 'Name cannot be empty');
}
2025-05-30 14:35:15 +03:00
if (name && !(await isNameAppropriate(name))) {
throw error(400, 'Name contains inappropriate content');
}
if (bio && bio.length > 160) {
throw error(400, 'Bio must be 160 characters or less');
}
if (username && (username.length < 3 || username.length > 30)) {
throw error(400, 'Username must be between 3 and 30 characters');
}
2025-05-30 14:35:15 +03:00
if (username && !(await isNameAppropriate(username))) {
throw error(400, 'Username contains inappropriate content');
}
2025-05-30 16:50:56 +03:00
if (bio && !(await isNameAppropriate(bio))) {
throw error(400, 'Bio contains inappropriate content');
}
if (avatarFile && avatarFile.size > MAX_FILE_SIZE) {
throw error(400, 'Avatar file must be smaller than 1MB');
}
}
export async function POST({ request }) {
const session = await auth.api.getSession({
headers: request.headers
});
if (!session?.user) {
throw error(401, 'Not authenticated');
}
const formData = await request.formData();
const name = formData.get('name') as string;
const bio = formData.get('bio') as string;
const username = formData.get('username') as string;
const avatarFile = formData.get('avatar') as File | null;
2025-05-30 14:35:15 +03:00
await validateInputs(name, bio, username, avatarFile);
const updates: Record<string, any> = {
name,
bio,
username,
updatedAt: new Date()
};
if (avatarFile && avatarFile.size > 0) {
try {
const arrayBuffer = await avatarFile.arrayBuffer();
const key = await uploadProfilePicture(
session.user.id,
new Uint8Array(arrayBuffer),
avatarFile.type,
avatarFile.size
);
updates.image = key;
} catch (e) {
console.error('Avatar upload failed, continuing without update:', e);
}
}
await db.update(user)
.set(updates)
.where(eq(user.id, Number(session.user.id)));
return json({ success: true });
}