n39librarian/middleware/index.ts

441 lines
11 KiB
TypeScript

import express, { Request, Response, Application } from "express";
import dotenv from "dotenv";
import { createPool } from "mariadb";
import { Book, MoveShelfAction } from "./types/Book";
import {
checkoutBook,
findBook,
getBook,
listBooks,
createBook,
deleteBook,
editBook,
returnBook,
listShelves,
} from "./queries";
import { v4 as uuidv4 } from "uuid";
import path from "path";
import session from "express-session";
import { censorBookData } from "./utils/censorBookData";
import * as fs from "fs";
import { checkForThreads } from "./utils/passesSQLInjectionCheck";
import { moveShelf } from "./queries/moveShelf";
dotenv.config();
const app: Application = express();
const port = process.env.PORT || 8000;
let adminKey = process.env.ADMIN_KEY;
const pool = createPool({
host: process.env.DB_HOST ?? "localhost",
port: process.env.DB_PORT ? Number(process.env.DB_PORT) : 3306,
user: process.env.ADMIN_DB_USER ?? "admin",
password: process.env.ADMIN_DB_PW ?? "AdminPassword",
database: "librarian",
connectionLimit: process.env.DB_CONNECTION_LIMIT
? Number(process.env.DB_CONNECTION_LIMIT)
: 10,
});
app.use(express.json());
app.use(
session({
secret: adminKey ?? "superSecretKey",
resave: false,
saveUninitialized: true,
})
);
const isAuthenticated = async (
req: Request<any, any, any, any>,
res: Response,
next: () => Promise<void>
) => {
if (!req.session.authenticated) {
res.status(403).send("unauthorized");
} else {
await next();
}
};
const stringifyResponse = (obj: unknown) =>
JSON.stringify(obj, (_, v) => (typeof v === "bigint" ? v.toString() : v));
app.use(express.static(path.join(__dirname, "frontend")));
app.get("/", function (req, res) {
res.sendFile(path.join(__dirname, "frontend", "index.html"));
});
app.get("/books", function (req, res) {
res.sendFile(path.join(__dirname, "frontend", "index.html"));
});
app.get("/books/*", function (req, res) {
res.sendFile(path.join(__dirname, "frontend", "index.html"));
});
app.post(
"/api/authenticate",
async (
req: Request<undefined, undefined, { password: string }>,
res: Response
) => {
try {
if (req.body.password === adminKey || req.session.authenticated) {
req.session.authenticated = true;
res.status(200).send("ok");
} else {
res.status(403).send("unauthorized");
}
} catch (error) {
res.status(500).send("Internal Server Error: " + error);
}
}
);
app.post(
"/api/updateAdminKey",
async (
req: Request<
undefined,
undefined,
{ oldPassword: string; newPassword: string }
>,
res: Response
) => {
try {
if (req.body.oldPassword === adminKey) {
if (req.body.newPassword.length < 13) {
res
.status(400)
.send("Admin key must contain at least 13 charakters.");
return;
}
fs.readFile("./.env", "utf8", (err, file) => {
if (err) return console.log(err);
const replacedPw = file.replace(
/ADMIN_KEY=.*/g,
`ADMIN_KEY=${req.body.newPassword}`
);
fs.writeFile("./.env", replacedPw, "utf8", function (err) {
if (err) return console.log(err);
});
});
adminKey = req.body.newPassword;
res.status(200).send("ok");
} else {
res.status(403).send("unauthorized");
}
} catch (error) {
res.status(500).send("Internal Server Error: " + error);
}
}
);
app.post(
"/api/books/create",
async (
req: Request<
undefined,
undefined,
Pick<Book, "isbn" | "title" | "shelf" | "published">
>,
res: Response
) => {
isAuthenticated(req, res, async () => {
const { title, isbn, shelf, published } = req.body;
try {
const containsThread = checkForThreads(
[title, isbn, shelf, published],
res
);
if (containsThread) {
return;
}
const conn = await pool.getConnection();
const uuid = uuidv4();
await conn.query(createBook({ ...req.body, uuid }));
const createdBook = await conn.query(getBook({ uuid }));
await conn.end();
res.status(200).send(stringifyResponse(createdBook));
} catch (error) {
res.status(500).send("Internal Server Error: " + error);
}
});
}
);
app.post(
"/api/books/delete",
async (
req: Request<undefined, undefined, Pick<Book, "uuid">>,
res: Response
) => {
await isAuthenticated(req, res, async () => {
try {
const containsThread = checkForThreads([req.body.uuid], res);
if (containsThread) {
return;
}
const conn = await pool.getConnection();
const foundBooks = await conn.query(getBook(req.body));
if (!foundBooks.length) {
await conn.end();
res.status(404).send("Book not found");
} else {
await conn.query(deleteBook(req.body));
await conn.end();
res.status(200).send("Book deleted");
}
} catch (error) {
res.status(500).send("Internal Server Error: " + error);
}
});
}
);
app.post(
"/api/shelf/move",
async (
req: Request<undefined, undefined, MoveShelfAction>,
res: Response
) => {
await isAuthenticated(req, res, async () => {
const { target } = req.body;
try {
const containsThread = checkForThreads([target], res);
if (containsThread) {
return;
}
const conn = await pool.getConnection();
await conn.query(moveShelf(req.body));
await conn.end();
res.status(200).send(`Books moved to shelf ${target}.`);
} catch (error) {
res.status(500).send("Internal Server Error: " + error);
}
});
}
);
app.post(
"/api/books/edit",
async (
req: Request<
undefined,
undefined,
Pick<Book, "uuid" | "isbn" | "title" | "published" | "shelf">
>,
res: Response
) => {
await isAuthenticated(req, res, async () => {
const { uuid, title, isbn, shelf, published } = req.body;
try {
const containsThread = checkForThreads(
[uuid, title, isbn, shelf, published],
res
);
if (containsThread) {
return;
}
const conn = await pool.getConnection();
const foundBooks = await conn.query(getBook(req.body));
if (!foundBooks.length) {
await conn.end();
res.status(404).send("Book not found");
} else {
await conn.query(editBook(req.body));
await conn.end();
res.status(200).send("Book edited.");
}
} catch (error) {
res.status(500).send("Internal Server Error: " + error);
}
});
}
);
app.post(
"/api/books/checkout",
async (
req: Request<
undefined,
undefined,
Pick<Book, "uuid" | "checkoutBy" | "contact">
>,
res: Response<string>
) => {
const { uuid, checkoutBy, contact } = req.body;
try {
const containsThread = checkForThreads([uuid, checkoutBy, contact], res);
if (containsThread) {
return;
}
const conn = await pool.getConnection();
const foundBooks = await conn.query(getBook(req.body));
if (!foundBooks.length) {
await conn.end();
res.status(404).send(foundBooks);
} else {
await conn.query(
checkoutBook({ ...req.body, lastCheckoutDate: Date.now() })
);
await conn.end();
res.status(200).send("Book checked out");
}
} catch (error) {
res.status(500).send("Internal Server Error: " + error);
}
}
);
app.post(
"/api/books/find",
async (
req: Request<
undefined,
undefined,
Pick<Partial<Book>, "isbn" | "title" | "uuid">
>,
res: Response<Book[] | string>
) => {
const { uuid, title, isbn } = req.body;
try {
const containsThread = checkForThreads([uuid, title, isbn], res);
if (containsThread) {
return;
}
if (!isbn && !title && !uuid) {
res.status(400).send("No isbn or title provided");
return;
}
const conn = await pool.getConnection();
const foundBooks = await conn.query(findBook(req.body));
await conn.end();
if (!foundBooks.length) {
res.status(404).send("Book not found");
return;
}
res
.status(200)
.send(
stringifyResponse(
censorBookData(foundBooks, !!req.session.authenticated)
)
);
} catch (error) {
res.status(500).send("Internal Server Error: " + error);
}
}
);
app.get(
"/api/books/list",
async (
req: Request<undefined, undefined, null>,
res: Response<Book[] | string>
) => {
try {
const conn = await pool.getConnection();
const foundBooks = await conn.query(listBooks);
await conn.end();
res
.status(200)
.send(
stringifyResponse(
censorBookData(foundBooks, !!req.session.authenticated)
)
);
} catch (error) {
res.status(500).send("Internal Server Error: " + error);
}
}
);
app.get(
"/api/shelves/list",
async (
req: Request<undefined, undefined, null>,
res: Response<string[] | string>
) => {
try {
const conn = await pool.getConnection();
const books = await conn.query<Book[]>(listShelves);
const shelves = books.map((e) => (!e.shelf ? "AVAILABLE" : e.shelf));
await conn.end();
res.status(200).send(shelves);
} catch (error) {
res.status(500).send("Internal Server Error: " + error);
}
}
);
app.post(
"/api/books/return",
async (
req: Request<
undefined,
undefined,
Pick<Book, "uuid" | "checkoutBy" | "contact">
>,
res: Response
) => {
const { uuid, checkoutBy, contact } = req.body;
try {
const containsThread = checkForThreads([uuid, checkoutBy, contact], res);
if (containsThread) {
return;
}
const conn = await pool.getConnection();
const foundBooks = await conn.query<Book[]>(getBook(req.body));
if (!foundBooks.length) {
await conn.end();
res.status(404).send("Book not found");
return;
}
if (
!req.session.authenticated &&
!(
foundBooks[0].checkoutBy === req.body.checkoutBy &&
foundBooks[0].contact === req.body.contact
)
) {
res.status(403).send("unauthorized");
return;
}
await conn.query(returnBook(req.body));
await conn.end();
res.status(200).send("Book returned");
} catch (error) {
res.status(500).send("Internal Server Error: " + error);
}
}
);
app.listen(port, async () => {
console.log(`Server listening on port ${port}`);
});