Spaces:
Running
on
CPU Upgrade
Running
on
CPU Upgrade
import { createReadStream, existsSync } from "node:fs" | |
import path from "node:path" | |
import express from "express" | |
import { VideoTask, VideoSequenceRequest } from "./types.mts" | |
import { parseVideoRequest } from "./utils/parseVideoRequest.mts" | |
import { savePendingTask } from "./scheduler/savePendingTask.mts" | |
import { getTask } from "./scheduler/getTask.mts" | |
import { main } from "./main.mts" | |
import { completedFilesDirFilePath } from "./config.mts" | |
import { deleteTask } from "./scheduler/deleteTask.mts" | |
import { getPendingTasks } from "./scheduler/getPendingTasks.mts" | |
main() | |
const app = express() | |
const port = 7860 | |
app.use(express.json()) | |
app.post("/", async (req, res) => { | |
const request = req.body as VideoSequenceRequest | |
const token = `${request.token || ""}` | |
if (token !== process.env.VS_SECRET_ACCESS_TOKEN) { | |
console.log("couldn't find access token in the query") | |
res.status(401) | |
res.write(JSON.stringify({ error: "invalid token" })) | |
res.end() | |
return | |
} | |
let task: VideoTask = null | |
console.log(`creating task from request..`) | |
try { | |
task = await parseVideoRequest(request) | |
} catch (err) { | |
console.error(`failed to create task: ${task}`) | |
res.status(400) | |
res.write(JSON.stringify({ error: "query seems to be malformed" })) | |
res.end() | |
return | |
} | |
console.log(`saving task ${task.id}`) | |
try { | |
await savePendingTask(task) | |
res.status(200) | |
res.write(JSON.stringify(task)) | |
res.end() | |
} catch (err) { | |
console.error(err) | |
res.status(500) | |
res.write(JSON.stringify({ error: "couldn't save the task" })) | |
res.end() | |
} | |
}) | |
// get all pending tasks | |
app.get("/", async (req, res) => { | |
try { | |
const tasks = await getPendingTasks() | |
res.status(200) | |
res.write(JSON.stringify(tasks, null, 2)) | |
res.end() | |
} catch (err) { | |
console.error(err) | |
res.status(500) | |
res.write(JSON.stringify({ error: "couldn't get the tasks" })) | |
res.end() | |
} | |
}) | |
app.get("/:id", async (req, res) => { | |
try { | |
const task = await getTask(req.params.id) | |
res.status(200) | |
res.write(JSON.stringify(task)) | |
res.end() | |
} catch (err) { | |
console.error(err) | |
res.status(404) | |
res.write(JSON.stringify({ error: "couldn't find this task" })) | |
res.end() | |
} | |
}) | |
app.delete("/:id", async (req, res) => { | |
let task: VideoTask = null | |
try { | |
task = await getTask(req.params.id) | |
} catch (err) { | |
console.error(err) | |
res.status(404) | |
res.write(JSON.stringify({ error: "couldn't find this task" })) | |
res.end() | |
} | |
try { | |
await deleteTask(task) | |
res.status(200) | |
res.write(JSON.stringify({ success: true })) | |
res.end() | |
} catch (err) { | |
console.error(err) | |
res.status(500) | |
res.write(JSON.stringify({ success: false, error: "failed to delete the task" })) | |
res.end() | |
} | |
}) | |
app.get("/video/:id\.mp4", async (req, res) => { | |
if (!req.params.id) { | |
res.status(400) | |
res.write(JSON.stringify({ error: "please provide a valid video id" })) | |
res.end() | |
return | |
} | |
let task: VideoTask = null | |
try { | |
task = await getTask(req.params.id) | |
console.log("returning result to user..") | |
} catch (err) { | |
res.status(404) | |
res.write(JSON.stringify({ error: "this video doesn't exist" })) | |
res.end() | |
return | |
} | |
const completedFilePath = path.join(completedFilesDirFilePath, task.fileName) | |
// note: we DON'T want to use the pending file path, as there may be operations on it | |
// (ie. a process might be busy writing stuff to it) | |
const filePath = existsSync(completedFilePath) ? completedFilePath : "" | |
if (!filePath) { | |
res.status(400) | |
res.write(JSON.stringify({ error: "video exists, but cannot be previewed yet" })) | |
res.end() | |
return | |
} | |
// file path exists, let's try to read it | |
try { | |
// do we need this? | |
// res.status(200) | |
// res.setHeader("Content-Type", "media/mp4") | |
console.log(`creating a video read stream from ${filePath}`) | |
const stream = createReadStream(filePath) | |
stream.on('close', () => { | |
console.log(`finished streaming the video`) | |
res.end() | |
}) | |
stream.pipe(res) | |
} catch (err) { | |
console.error(`failed to read the video file at ${filePath}: ${err}`) | |
res.status(500) | |
res.write(JSON.stringify({ error: "failed to read the video file" })) | |
res.end() | |
} | |
}) | |
app.listen(port, () => { console.log(`Open http://localhost:${port}`) }) |