-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcli.js
82 lines (72 loc) · 2.73 KB
/
cli.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
import express from "express";
import path from "node:path";
import { fileURLToPath } from "node:url";
import { spawn } from "node:child_process";
import multer from "multer";
import fs from "node:fs";
import process from "node:process";
import { Buffer } from "node:buffer";
const __dirname = path.dirname(fileURLToPath(import.meta.url));
const app = express();
const port = process.env.PORT || 3000;
const eventStreamHeaders = {
"Content-Type": "text/event-stream",
"Cache-Control": "no-cache",
Connection: "keep-alive",
};
// Get key frames with defined gap between frames
const keyFrameArgs = (gap) => [
"-vf",
`select='key*(isnan(prev_selected_t)+gte(t-prev_selected_t,${gap}))',setpts='PTS-STARTPTS',showinfo`,
"-vsync",
"vfr",
"-compression_level",
"10",
];
// Convert to single-channel audio with defined bitrate
const audioArgs = (bitrate) => ["-c:a", "libopus", "-b:a", bitrate, "-ar", bitrate];
const writeEvent = (key, data, res) => {
const textData = Buffer.from(data).toString("utf-8");
res.write(`data: ${JSON.stringify({ [key]: textData })}\n\n`);
};
app.use(express.static(path.join(__dirname, "static")));
// Set up multer for handling file uploads
const storage = multer.diskStorage({
destination: (_req, file, callback) => {
const uploadDir = path.join("static", "uploads", file.originalname);
fs.mkdirSync(uploadDir, { recursive: true });
callback(null, uploadDir);
},
filename: (_req, file, callback) => callback(null, file.originalname),
});
const upload = multer({ storage });
const ffmpeg = (args, res) => {
const ffmpeg = spawn("ffmpeg", ["-y", ...args]);
ffmpeg.stdout.on("data", (data) => writeEvent("stdout", data, res));
ffmpeg.stderr.on("data", (data) => writeEvent("stderr", data, res));
ffmpeg.on("close", (code) => {
const output = args
.at(-1)
.replace(/\\/g, "/")
.replace(/^static\//, "");
res.write(`data: ${JSON.stringify({ code, output })}\n\n`);
res.end();
});
};
app.post("/audio", upload.single("file"), (req, res) => {
if (!req.file) return res.status(400).send("No file uploaded.");
const output = path.join(path.dirname(req.file.path), "audio.opus");
const bitrate = req.query.bitrate || "8000";
res.writeHead(200, eventStreamHeaders);
ffmpeg(["-i", req.file.path, ...audioArgs(bitrate), output], res);
});
app.post("/keyframes", upload.single("file"), (req, res) => {
if (!req.file) return res.status(400).send("No file uploaded.");
const gap = req.query.gap || req.body.gap || "1";
const output = path.join(path.dirname(req.file.path), "%04d.jpg");
res.writeHead(200, eventStreamHeaders);
ffmpeg(["-i", req.file.path, ...keyFrameArgs(gap), output], res);
});
app.listen(port, () => {
console.log(`Running at http://localhost:${port}`);
});