feat: accept any audio/video upload, server converts to 6s compressed MP3
All checks were successful
Deploy / build-and-deploy (push) Successful in 1m44s

This commit is contained in:
2026-06-15 03:25:44 +03:00
parent 6a4be6b939
commit 3f41d7699c
5 changed files with 54 additions and 57 deletions

View File

@@ -6,9 +6,13 @@ import { writeFile, mkdir, unlink } from "fs/promises";
import path from "path"; import path from "path";
import { v4 as uuidv4 } from "uuid"; import { v4 as uuidv4 } from "uuid";
import { eq } from "drizzle-orm"; import { eq } from "drizzle-orm";
import { convertOggToMp3 } from "@/lib/convert"; import { convertToCompressedMp3 } from "@/lib/convert";
export const maxDuration = 120;
const UPLOAD_DIR = path.join(process.cwd(), "public", "uploads", "sounds"); const UPLOAD_DIR = path.join(process.cwd(), "public", "uploads", "sounds");
const ALLOWED_EXTENSIONS = /\.(mp3|ogg|wav|flac|aac|m4a|wma|opus|webm|mp4|avi|mov|mkv|webm|wmv|flv)$/i;
const MAX_SIZE = 50 * 1024 * 1024; // 50 MB
export async function GET() { export async function GET() {
const session = await getServerSession(); const session = await getServerSession();
@@ -43,37 +47,36 @@ export async function POST(req: NextRequest) {
const results: { url: string; originalName: string; duration: number }[] = []; const results: { url: string; originalName: string; duration: number }[] = [];
for (const file of files) { for (const file of files) {
if (!file.name.toLowerCase().endsWith(".ogg")) continue; if (!ALLOWED_EXTENSIONS.test(file.name)) continue;
if (file.size > 2 * 1024 * 1024) continue; if (file.size > MAX_SIZE) continue;
const baseName = uuidv4(); const baseName = uuidv4();
const oggPath = path.join(UPLOAD_DIR, `${baseName}.ogg`); const inputPath = path.join(UPLOAD_DIR, `${baseName}.in`);
const mp3Path = path.join(UPLOAD_DIR, `${baseName}.mp3`); const mp3Path = path.join(UPLOAD_DIR, `${baseName}.mp3`);
const bytes = await file.arrayBuffer(); const bytes = await file.arrayBuffer();
await writeFile(oggPath, Buffer.from(bytes)); await writeFile(inputPath, Buffer.from(bytes));
const duration = Math.round((file.size / (64 * 128)) * 10) / 10;
let storedFilename: string; let storedFilename: string;
if (convertOggToMp3(oggPath, mp3Path)) { if (convertToCompressedMp3(inputPath, mp3Path)) {
await unlink(oggPath).catch(() => {}); await unlink(inputPath).catch(() => {});
storedFilename = `${baseName}.mp3`; storedFilename = `${baseName}.mp3`;
} else { } else {
storedFilename = `${baseName}.ogg`; await unlink(inputPath).catch(() => {});
continue;
} }
const id = uuidv4(); const id = uuidv4();
db.insert(sounds).values({ db.insert(sounds).values({
id, id,
originalName: file.name.replace(/\.ogg$/i, ".mp3"), originalName: file.name,
storedFilename, storedFilename,
duration: Math.min(duration, 6), duration: 6,
uploadedBy: session.id, uploadedBy: session.id,
createdAt: new Date().toISOString(), createdAt: new Date().toISOString(),
}).run(); }).run();
results.push({ url: `/uploads/sounds/${storedFilename}`, originalName: file.name.replace(/\.ogg$/i, ".mp3"), duration: Math.min(duration, 6) }); results.push({ url: `/uploads/sounds/${storedFilename}`, originalName: file.name, duration: 6 });
} }
return NextResponse.json({ sounds: results }); return NextResponse.json({ sounds: results });

View File

@@ -5,10 +5,13 @@ import { sounds } from "@/lib/db/schema";
import { writeFile, mkdir, unlink } from "fs/promises"; import { writeFile, mkdir, unlink } from "fs/promises";
import path from "path"; import path from "path";
import { v4 as uuidv4 } from "uuid"; import { v4 as uuidv4 } from "uuid";
import { convertOggToMp3 } from "@/lib/convert"; import { convertToCompressedMp3 } from "@/lib/convert";
import fs from "fs";
export const maxDuration = 60;
const UPLOAD_DIR = path.join(process.cwd(), "public", "uploads", "sounds"); const UPLOAD_DIR = path.join(process.cwd(), "public", "uploads", "sounds");
const ALLOWED_EXTENSIONS = /\.(mp3|ogg|wav|flac|aac|m4a|wma|opus|webm|mp4|avi|mov|mkv|webm|wmv|flv)$/i;
const MAX_SIZE = 50 * 1024 * 1024; // 50 MB
export async function POST(req: NextRequest) { export async function POST(req: NextRequest) {
const session = await getServerSession(); const session = await getServerSession();
@@ -21,47 +24,43 @@ export async function POST(req: NextRequest) {
const file = formData.get("file") as File | null; const file = formData.get("file") as File | null;
if (!file) return NextResponse.json({ error: "No file" }, { status: 400 }); if (!file) return NextResponse.json({ error: "No file" }, { status: 400 });
if (!file.name.toLowerCase().endsWith(".ogg")) { if (!ALLOWED_EXTENSIONS.test(file.name)) {
return NextResponse.json({ error: "Only OGG files allowed" }, { status: 400 }); return NextResponse.json({ error: "Unsupported file format" }, { status: 400 });
} }
if (file.size > 2 * 1024 * 1024) { if (file.size > MAX_SIZE) {
return NextResponse.json({ error: "File too large (max 2MB)" }, { status: 400 }); return NextResponse.json({ error: "File too large (max 50MB)" }, { status: 400 });
} }
await mkdir(UPLOAD_DIR, { recursive: true }); await mkdir(UPLOAD_DIR, { recursive: true });
const baseName = uuidv4(); const baseName = uuidv4();
const oggPath = path.join(UPLOAD_DIR, `${baseName}.ogg`); const inputPath = path.join(UPLOAD_DIR, `${baseName}.in`);
const mp3Path = path.join(UPLOAD_DIR, `${baseName}.mp3`); const mp3Path = path.join(UPLOAD_DIR, `${baseName}.mp3`);
// save temporary OGG
const bytes = await file.arrayBuffer(); const bytes = await file.arrayBuffer();
await writeFile(oggPath, Buffer.from(bytes)); await writeFile(inputPath, Buffer.from(bytes));
// duration estimate
const duration = Math.round((file.size / (64 * 128)) * 10) / 10;
// convert to MP3
let storedFilename: string; let storedFilename: string;
if (convertOggToMp3(oggPath, mp3Path)) { if (convertToCompressedMp3(inputPath, mp3Path)) {
await unlink(oggPath).catch(() => {}); await unlink(inputPath).catch(() => {});
storedFilename = `${baseName}.mp3`; storedFilename = `${baseName}.mp3`;
} else {
storedFilename = `${baseName}.ogg`;
}
const id = uuidv4(); const id = uuidv4();
db.insert(sounds).values({ db.insert(sounds).values({
id, id,
originalName: file.name.replace(/\.ogg$/i, ".mp3"), originalName: file.name,
storedFilename, storedFilename,
duration: Math.min(duration, 6), duration: 6,
uploadedBy: session.id, uploadedBy: session.id,
createdAt: new Date().toISOString(), createdAt: new Date().toISOString(),
}).run(); }).run();
return NextResponse.json({ url: `/uploads/sounds/${storedFilename}` }); return NextResponse.json({ url: `/uploads/sounds/${storedFilename}` });
}
await unlink(inputPath).catch(() => {});
return NextResponse.json({ error: "Conversion failed" }, { status: 500 });
} catch { } catch {
return NextResponse.json({ error: "Upload failed" }, { status: 500 }); return NextResponse.json({ error: "Upload failed" }, { status: 500 });
} }

View File

@@ -305,7 +305,7 @@ export function ItemEditor({ campaign }: { campaign: Campaign }) {
<input <input
ref={fileInputRef} ref={fileInputRef}
type="file" type="file"
accept=".ogg" accept="audio/*,video/*"
className="hidden" className="hidden"
onChange={e => { const f = e.target.files?.[0]; if (f) uploadSound(f); }} onChange={e => { const f = e.target.files?.[0]; if (f) uploadSound(f); }}
/> />

View File

@@ -79,7 +79,7 @@ function SoundsTab() {
} catch {} } catch {}
setUploading(false); setUploading(false);
}, },
accept: { "audio/ogg": [".ogg"] }, accept: { "audio/*": [".mp3", ".ogg", ".wav", ".flac", ".aac", ".m4a", ".wma", ".opus", ".webm"], "video/*": [".mp4", ".avi", ".mov", ".mkv", ".webm", ".wmv", ".flv"] },
multiple: true, multiple: true,
}); });
@@ -133,7 +133,7 @@ function SoundsTab() {
<div {...getRootProps()} className="border-2 border-dashed border-slate-700/40 rounded-lg p-3 text-center cursor-pointer hover:border-cyan-600/40 transition-colors"> <div {...getRootProps()} className="border-2 border-dashed border-slate-700/40 rounded-lg p-3 text-center cursor-pointer hover:border-cyan-600/40 transition-colors">
<input {...getInputProps()} /> <input {...getInputProps()} />
<p className="text-[10px] font-mono text-slate-500">Drop OGG files here or click to upload</p> <p className="text-[10px] font-mono text-slate-500">Drop audio/video files here or click to upload</p>
</div> </div>
{/* Library */} {/* Library */}

View File

@@ -1,23 +1,18 @@
import { execSync } from "child_process"; import { execSync } from "child_process";
import fs from "fs";
export function convertOggToMp3(inputPath: string, outputPath: string): boolean { /**
* Convert any audio/video file to a 6-second compressed MP3 via ffmpeg.
* Returns true on success, false on failure.
* The output file is always mp3Path (overwritten if exists).
*/
export function convertToCompressedMp3(inputPath: string, mp3Path: string): boolean {
try { try {
execSync( execSync(
`ffmpeg -i "${inputPath}" -codec:a libmp3lame -b:a 128k -y "${outputPath}" 2>/dev/null`, `ffmpeg -i "${inputPath}" -t 6 -vn -codec:a libmp3lame -b:a 64k -y "${mp3Path}" 2>/dev/null`,
{ timeout: 15000 } { timeout: 30000 }
); );
return true; return true;
} catch { } catch {
return false; return false;
} }
} }
export function convertOggToMp3OrCopy(inputPath: string, mp3Path: string): string {
if (convertOggToMp3(inputPath, mp3Path)) {
return mp3Path;
}
// fallback: keep original OGG
fs.copyFileSync(inputPath, mp3Path.replace(/\.mp3$/, ".ogg"));
return mp3Path.replace(/\.mp3$/, ".ogg");
}