ditto/src/controllers/api/media.ts
2023-09-09 21:33:12 -05:00

52 lines
1.5 KiB
TypeScript

import { AppController } from '@/app.ts';
import { Conf } from '@/config.ts';
import { insertUnattachedMedia } from '@/db/unattached-media.ts';
import { z } from '@/deps.ts';
import { fileSchema } from '@/schema.ts';
import { configUploader as uploader } from '@/uploaders/config.ts';
import { parseBody } from '@/utils/web.ts';
import { renderAttachment } from '@/views/attachment.ts';
const uploadSchema = fileSchema
.refine((file) => !!file.type, 'File type is required.')
.refine((file) => file.size <= Conf.maxUploadSize, 'File size is too large.');
const mediaBodySchema = z.object({
file: uploadSchema,
thumbnail: uploadSchema.optional(),
description: z.string().optional(),
focus: z.string().optional(),
});
const mediaController: AppController = async (c) => {
const result = mediaBodySchema.safeParse(await parseBody(c.req.raw));
if (!result.success) {
return c.json({ error: 'Bad request.', schema: result.error }, 422);
}
try {
const { file, description } = result.data;
const { cid } = await uploader.upload(file);
const url = new URL(`/ipfs/${cid}`, Conf.mediaDomain).toString();
const media = await insertUnattachedMedia({
pubkey: c.get('pubkey')!,
url,
data: {
name: file.name,
mime: file.type,
size: file.size,
description,
},
});
return c.json(renderAttachment(media));
} catch (e) {
console.error(e);
return c.json({ error: 'Failed to upload file.' }, 500);
}
};
export { mediaController };