2023-11-10 23:41:31 +03:00
|
|
|
import express, { Request, Response } from 'express';
|
2023-10-19 19:46:23 +02:00
|
|
|
import crypto from 'crypto';
|
2023-10-26 21:53:02 +03:00
|
|
|
import multer from 'multer'
|
|
|
|
const upload = multer();
|
2023-10-19 19:46:23 +02:00
|
|
|
|
2023-11-13 00:09:12 +01:00
|
|
|
import { traverseOperations } from "@stirling-pdf/shared-operations/src/workflow/traverseOperations";
|
2023-11-14 23:14:08 +01:00
|
|
|
import { PdfFile, RepresentationType } from '@stirling-pdf/shared-operations/src/wrappers/PdfFile';
|
2023-11-16 02:24:10 +03:00
|
|
|
import { respondWithPdfFiles } from '../../utils/endpoint-utils';
|
2023-10-19 19:46:23 +02:00
|
|
|
|
2023-11-21 00:12:35 +01:00
|
|
|
interface Workflow {
|
|
|
|
eventStream?: express.Response<any, Record<string, any>>,
|
|
|
|
result?: PdfFile[],
|
|
|
|
finished: boolean,
|
|
|
|
createdAt: EpochTimeStamp,
|
|
|
|
finishedAt?: EpochTimeStamp,
|
|
|
|
error?: { type: number, error: string, stack?: string }
|
|
|
|
// TODO: When auth is implemented: owner
|
|
|
|
}
|
|
|
|
|
|
|
|
const activeWorkflows: Record<string, Workflow> = {};
|
2023-10-19 19:46:23 +02:00
|
|
|
|
|
|
|
const router = express.Router();
|
2023-11-13 00:09:12 +01:00
|
|
|
|
2023-10-19 21:23:58 +02:00
|
|
|
router.post("/:workflowUuid?", [
|
2023-11-13 00:09:12 +01:00
|
|
|
upload.array("files"),
|
2023-11-10 23:41:31 +03:00
|
|
|
async (req: Request, res: Response) => {
|
2023-11-13 00:09:12 +01:00
|
|
|
// TODO: Maybe replace with another validator
|
|
|
|
if(req.files?.length == 0) {
|
2023-10-20 00:10:03 +02:00
|
|
|
res.status(400).json({"error": "No files were uploaded."});
|
|
|
|
return;
|
|
|
|
}
|
2023-11-13 00:09:12 +01:00
|
|
|
|
2023-11-20 22:43:09 +01:00
|
|
|
try {
|
|
|
|
var workflow = JSON.parse(req.body.workflow);
|
|
|
|
} catch (err) {
|
|
|
|
if (err instanceof Error) {
|
|
|
|
console.error("malformed workflow-json was provided", err.message);
|
|
|
|
res.status(400).json({error: "Malformed workflow-JSON was provided. See Server-Logs for more info", details: err.message});
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
}
|
2023-10-19 19:46:23 +02:00
|
|
|
|
2023-11-14 23:14:08 +01:00
|
|
|
// TODO: Replace with static multer function of pdffile
|
2023-11-13 00:09:12 +01:00
|
|
|
const inputs = await Promise.all((req.files as Express.Multer.File[]).map(async file => {
|
2023-11-14 23:14:08 +01:00
|
|
|
return new PdfFile(file.originalname.replace(/\.[^/.]+$/, ""), new Uint8Array(await file.buffer), RepresentationType.Uint8Array, file.originalname.replace(/\.[^/.]+$/, ""));
|
2023-10-19 19:46:23 +02:00
|
|
|
}));
|
|
|
|
|
2023-11-14 23:14:08 +01:00
|
|
|
// Allow option to do it synchronously and just make a long request
|
|
|
|
if(req.body.async === "false") {
|
|
|
|
console.log("Don't do async");
|
|
|
|
|
2023-11-20 22:43:09 +01:00
|
|
|
traverseOperations(workflow.operations, inputs, (state) => {
|
2023-11-20 21:45:49 +01:00
|
|
|
console.log("State: ", state);
|
2023-11-20 22:43:09 +01:00
|
|
|
}).then(async (pdfResults) => {
|
|
|
|
console.log("Download");
|
|
|
|
await respondWithPdfFiles(res, pdfResults, "workflow-results");
|
|
|
|
}).catch((err) => {
|
2023-11-21 00:12:35 +01:00
|
|
|
if(err.validationError) {
|
|
|
|
// Bad Request
|
2023-11-20 22:43:09 +01:00
|
|
|
res.status(400).json({error: err});
|
2023-11-21 00:12:35 +01:00
|
|
|
}
|
|
|
|
else if (err instanceof Error) {
|
|
|
|
console.error("Internal Server Error", err);
|
|
|
|
// Internal Server Error
|
|
|
|
res.status(500).json({error: err.message, stack: err.stack});
|
|
|
|
} else {
|
2023-11-20 22:43:09 +01:00
|
|
|
throw err;
|
2023-11-21 00:12:35 +01:00
|
|
|
}
|
2023-11-20 21:45:49 +01:00
|
|
|
})
|
2023-11-14 23:14:08 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
console.log("Start Aync Workflow");
|
|
|
|
// TODO: UUID collision checks
|
|
|
|
let workflowID = req.params.workflowUuid
|
|
|
|
if(!workflowID)
|
|
|
|
workflowID = generateWorkflowID();
|
|
|
|
|
|
|
|
activeWorkflows[workflowID] = {
|
|
|
|
createdAt: Date.now(),
|
2023-11-21 00:12:35 +01:00
|
|
|
finished: false
|
2023-11-14 23:14:08 +01:00
|
|
|
}
|
|
|
|
const activeWorkflow = activeWorkflows[workflowID];
|
|
|
|
|
|
|
|
res.status(200).json({
|
|
|
|
"workflowID": workflowID,
|
|
|
|
"data-recieved": {
|
|
|
|
"fileCount": inputs.length,
|
|
|
|
"workflow": workflow
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2023-11-21 00:12:35 +01:00
|
|
|
traverseOperations(workflow.operations, inputs, (state) => {
|
2023-11-20 21:45:49 +01:00
|
|
|
console.log("State: ", state);
|
2023-11-14 23:14:08 +01:00
|
|
|
if(activeWorkflow.eventStream)
|
2023-11-20 21:45:49 +01:00
|
|
|
activeWorkflow.eventStream.write(`data: ${state}\n\n`);
|
2023-11-21 00:12:35 +01:00
|
|
|
}).then(async (pdfResults) => {
|
|
|
|
if(activeWorkflow.eventStream) {
|
|
|
|
activeWorkflow.eventStream.write(`data: processing done\n\n`);
|
|
|
|
activeWorkflow.eventStream.end();
|
|
|
|
}
|
|
|
|
|
|
|
|
activeWorkflow.result = pdfResults;
|
|
|
|
activeWorkflow.finished = true;
|
|
|
|
activeWorkflow.finishedAt = Date.now();
|
|
|
|
}).catch((err) => {
|
|
|
|
if(err.validationError) {
|
|
|
|
activeWorkflow.error = {type: 500, error: err};
|
|
|
|
activeWorkflow.finished = true;
|
|
|
|
activeWorkflow.finishedAt = Date.now();
|
|
|
|
|
|
|
|
// Bad Request
|
|
|
|
if(activeWorkflow.eventStream) {
|
|
|
|
activeWorkflow.eventStream.write(`data: ${activeWorkflow.error}\n\n`);
|
|
|
|
activeWorkflow.eventStream.end();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (err instanceof Error) {
|
|
|
|
console.error("Internal Server Error", err);
|
|
|
|
activeWorkflow.error = {type: 400, error: err.message, stack: err.stack};
|
|
|
|
activeWorkflow.finished = true;
|
|
|
|
activeWorkflow.finishedAt = Date.now();
|
|
|
|
|
|
|
|
// Internal Server Error
|
|
|
|
if(activeWorkflow.eventStream) {
|
|
|
|
activeWorkflow.eventStream.write(`data: ${activeWorkflow.error}\n\n`);
|
|
|
|
activeWorkflow.eventStream.end();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
});
|
2023-11-14 23:14:08 +01:00
|
|
|
}
|
2023-10-19 19:46:23 +02:00
|
|
|
}
|
|
|
|
]);
|
|
|
|
|
2023-11-10 23:41:31 +03:00
|
|
|
router.get("/progress/:workflowUuid", (req: Request, res: Response) => {
|
2023-10-20 00:10:03 +02:00
|
|
|
if(!req.params.workflowUuid) {
|
|
|
|
res.status(400).json({"error": "No workflowUuid weres provided."});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if(!activeWorkflows.hasOwnProperty(req.params.workflowUuid)) {
|
|
|
|
res.status(400).json({"error": `No workflow with workflowUuid "${req.params.workflowUuid}" was found.`});
|
|
|
|
return;
|
|
|
|
}
|
2023-10-19 19:46:23 +02:00
|
|
|
|
|
|
|
// Return current progress
|
|
|
|
const workflow = activeWorkflows[req.params.workflowUuid];
|
2023-11-21 00:12:35 +01:00
|
|
|
res.status(200).json({ createdAt: workflow.createdAt, finished: workflow.finished, finishedAt: workflow.finishedAt, error: workflow.error });
|
2023-10-19 19:46:23 +02:00
|
|
|
});
|
|
|
|
|
2023-11-10 23:41:31 +03:00
|
|
|
router.get("/progress-stream/:workflowUuid", (req: Request, res: Response) => {
|
2023-10-20 00:10:03 +02:00
|
|
|
if(!req.params.workflowUuid) {
|
|
|
|
res.status(400).json({"error": "No workflowUuid weres provided."});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if(!activeWorkflows.hasOwnProperty(req.params.workflowUuid)) {
|
|
|
|
res.status(400).json({"error": `No workflow with workflowUuid "${req.params.workflowUuid}" was found.`});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Check if already done
|
2023-10-19 21:23:58 +02:00
|
|
|
|
|
|
|
// Send realtime updates
|
|
|
|
res.setHeader('Cache-Control', 'no-cache');
|
|
|
|
res.setHeader('Content-Type', 'text/event-stream');
|
|
|
|
res.setHeader('Access-Control-Allow-Origin', '*');
|
|
|
|
res.setHeader('Connection', 'keep-alive');
|
|
|
|
res.flushHeaders(); // flush the headers to establish SSE with client
|
|
|
|
|
|
|
|
const workflow = activeWorkflows[req.params.workflowUuid];
|
|
|
|
workflow.eventStream = res;
|
|
|
|
|
|
|
|
res.on('close', () => {
|
|
|
|
res.end();
|
|
|
|
// TODO: Abort if not already done?
|
|
|
|
});
|
2023-10-19 19:46:23 +02:00
|
|
|
});
|
|
|
|
|
2023-11-14 23:14:08 +01:00
|
|
|
router.get("/result/:workflowUuid", async (req: Request, res: Response) => {
|
2023-10-20 00:10:03 +02:00
|
|
|
if(!req.params.workflowUuid) {
|
|
|
|
res.status(400).json({"error": "No workflowUuid weres provided."});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if(!activeWorkflows.hasOwnProperty(req.params.workflowUuid)) {
|
|
|
|
res.status(400).json({"error": `No workflow with workflowUuid "${req.params.workflowUuid}" was found.`});
|
|
|
|
return;
|
|
|
|
}
|
2023-10-19 19:46:23 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* If workflow isn't done return error
|
2023-11-21 00:20:19 +01:00
|
|
|
* Send file, if there are multiple outputs return as zip
|
2023-10-19 19:46:23 +02:00
|
|
|
* If download is done, delete results / allow deletion within the next 5-60 mins
|
2023-11-13 00:09:12 +01:00
|
|
|
*/
|
2023-10-19 19:46:23 +02:00
|
|
|
const workflow = activeWorkflows[req.params.workflowUuid];
|
|
|
|
if(!workflow.finished) {
|
|
|
|
res.status(202).json({ message: "Workflow hasn't finished yet. Check progress or connect to progress-steam to get notified when its done." });
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-11-16 02:24:10 +03:00
|
|
|
await respondWithPdfFiles(res, workflow.result, "workflow-results");
|
2023-10-19 19:46:23 +02:00
|
|
|
// Delete workflow / results when done.
|
|
|
|
delete activeWorkflows[req.params.workflowUuid];
|
|
|
|
});
|
|
|
|
|
2023-11-10 23:41:31 +03:00
|
|
|
router.post("/abort/:workflowUuid", (req: Request, res: Response) => {
|
2023-10-20 00:10:03 +02:00
|
|
|
if(!req.params.workflowUuid) {
|
|
|
|
res.status(400).json({"error": "No workflowUuid weres provided."});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if(!activeWorkflows.hasOwnProperty(req.params.workflowUuid)) {
|
|
|
|
res.status(400).json({"error": `No workflow with workflowUuid "${req.params.workflowUuid}" was found.`});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-10-19 19:46:23 +02:00
|
|
|
// TODO: Abort workflow
|
|
|
|
res.status(501).json({"warning": "Abortion has not been implemented yet."});
|
|
|
|
});
|
|
|
|
|
|
|
|
function generateWorkflowID() {
|
|
|
|
return crypto.randomUUID();
|
|
|
|
}
|
|
|
|
|
|
|
|
export default router;
|