some docs on backend
This commit is contained in:
parent
f7918b45f1
commit
f0e8b91e78
@ -2,7 +2,7 @@
|
||||
"domain": "localhost",
|
||||
"port": 8080,
|
||||
"db_path": "./ucs.db",
|
||||
"limits_per_second": {
|
||||
"limits_per_minute": {
|
||||
"get": 500,
|
||||
"post": 20
|
||||
},
|
||||
|
@ -1,22 +1,40 @@
|
||||
/**
|
||||
* Comment fetch and submit APIs.
|
||||
*/
|
||||
|
||||
import express from "express";
|
||||
|
||||
import utils from "./utils.mjs";
|
||||
import db from "./db.mjs";
|
||||
|
||||
// server
|
||||
// Router
|
||||
|
||||
const router = express.Router();
|
||||
|
||||
export default router;
|
||||
|
||||
router.use(express.json());
|
||||
|
||||
// GET
|
||||
|
||||
/**
|
||||
* Get comments on site :site and page * as JSON.
|
||||
*/
|
||||
router.get("/:site/*", utils.get_limiter, (req, res) => {
|
||||
const site = req.params.site;
|
||||
const path = req.params[0];
|
||||
const comments = db.pageComments(site, path);
|
||||
res.json(comments);
|
||||
});
|
||||
|
||||
// POST
|
||||
|
||||
/**
|
||||
* Handles the submission of a new comment into :site with page *.
|
||||
*/
|
||||
router.post("/:site/*", utils.post_limiter, (req, res) => {
|
||||
const site_url = req.params.site;
|
||||
const path = req.params[0];
|
||||
|
||||
// Validate token and message are not empty.
|
||||
if (!req.body.token || !req.body.message) {
|
||||
res.status(400).json("הודעה ריקה.");
|
||||
return;
|
||||
@ -29,10 +47,10 @@ router.post("/:site/*", utils.post_limiter, (req, res) => {
|
||||
reply_to: req.body.reply_to || null,
|
||||
};
|
||||
|
||||
// validation
|
||||
const user_token = req.body.token;
|
||||
const site = db.siteInfo(site_url);
|
||||
|
||||
// Other validations.
|
||||
if (user_token !== site.info.comment_token) {
|
||||
res.status(403).json("תשובת סינון שגויה.");
|
||||
} else if (comment.user.length > site.max_lengths.user) {
|
||||
@ -42,14 +60,7 @@ router.post("/:site/*", utils.post_limiter, (req, res) => {
|
||||
} else if (comment.message > site.max_lengths.message) {
|
||||
res.status(400).json("הודעה ארוכה מדי.");
|
||||
} else {
|
||||
// If all validations pass, insert the comment.
|
||||
res.json(db.insertPageComment(site_url, path, comment));
|
||||
}
|
||||
});
|
||||
|
||||
// GET
|
||||
router.get("/:site/*", utils.get_limiter, (req, res) => {
|
||||
const site = req.params.site;
|
||||
const path = req.params[0];
|
||||
const comments = db.pageComments(site, path);
|
||||
res.json(comments);
|
||||
});
|
||||
|
@ -1,11 +1,14 @@
|
||||
/**
|
||||
* Read relevant information from configuration file.
|
||||
*
|
||||
* See the `config/default.json` file for an example config.
|
||||
*/
|
||||
import config from "config";
|
||||
|
||||
const configuration = config.util.toObject();
|
||||
|
||||
// console.log(JSON.stringify(configuration));
|
||||
|
||||
export default {
|
||||
config: configuration,
|
||||
...config,
|
||||
getSite: (site_url) => {
|
||||
return configuration.sites[site_url];
|
||||
},
|
||||
|
@ -1,24 +1,53 @@
|
||||
/**
|
||||
* Database interactions.
|
||||
*/
|
||||
import Database from "better-sqlite3";
|
||||
import { migrate } from "@blackglory/better-sqlite3-migrations";
|
||||
|
||||
import utils from "./utils.mjs";
|
||||
import config from "./config.mjs";
|
||||
|
||||
// class
|
||||
// Interface.
|
||||
|
||||
export class DB {
|
||||
/**
|
||||
* Connect to a database and perform migrations.
|
||||
*/
|
||||
constructor() {
|
||||
this.my_db = createDB();
|
||||
}
|
||||
siteInfo(site) {
|
||||
return getSiteInfo(this.my_db, site);
|
||||
/**
|
||||
* Fetch information about a site.
|
||||
* @param {!string} site url.
|
||||
* @return {!ObjType} site information.
|
||||
*/
|
||||
siteInfo(site_url) {
|
||||
return getSiteInfo(this.my_db, site_url);
|
||||
}
|
||||
/**
|
||||
* Fetch all comments from a specific site.
|
||||
* @param {!string} site url.
|
||||
* @return {!Array<!ObjType>} comment objects.
|
||||
*/
|
||||
siteComments(site_url) {
|
||||
return getSiteComments(this.my_db, site_url);
|
||||
}
|
||||
/**
|
||||
* Fetch comments from a specific page in a specific site.
|
||||
* @param {!string} site url.
|
||||
* @param {!string} page path.
|
||||
* @return {!Array<!ObjType>} comment objects.
|
||||
*/
|
||||
pageComments(site_url, page) {
|
||||
return getPageComments(this.my_db, site_url, page);
|
||||
}
|
||||
/**
|
||||
* Insert a comment into the database.
|
||||
* @param {!string} site url.
|
||||
* @param {!string} page path.
|
||||
* @param {!ObjType} comment.
|
||||
* @return {!ObjType} inserted comment.
|
||||
*/
|
||||
insertPageComment(site_url, path, comment) {
|
||||
return insertPageComment(this.my_db, site_url, path, comment);
|
||||
}
|
||||
@ -27,7 +56,7 @@ export class DB {
|
||||
const db = new DB();
|
||||
export default db;
|
||||
|
||||
// migrations
|
||||
// Migrations
|
||||
|
||||
function migrations() {
|
||||
return [
|
||||
|
@ -1,19 +1,23 @@
|
||||
/**
|
||||
* Serve Atom feeds for comments on sites.
|
||||
*/
|
||||
import express from "express";
|
||||
import { Feed } from "feed";
|
||||
|
||||
import utils from "./utils.mjs";
|
||||
import db from "./db.mjs";
|
||||
|
||||
// Feed
|
||||
// Router
|
||||
|
||||
const router = express.Router();
|
||||
|
||||
export default router;
|
||||
|
||||
router.use(express.json());
|
||||
|
||||
const domain = utils.domain;
|
||||
|
||||
/**
|
||||
* Serve feed for all comments in a specific :site as XML Atom feed.
|
||||
*/
|
||||
router.get("/:site", utils.get_limiter, (req, res) => {
|
||||
const site = req.params.site;
|
||||
|
||||
@ -44,6 +48,9 @@ router.get("/:site", utils.get_limiter, (req, res) => {
|
||||
res.send(xml);
|
||||
});
|
||||
|
||||
/**
|
||||
* Serve feed comments in a specific :site and page as XML Atom feed.
|
||||
*/
|
||||
router.get("/:site/*", utils.get_limiter, (req, res) => {
|
||||
const site = req.params.site;
|
||||
const path = req.params[0];
|
||||
|
@ -2,7 +2,7 @@ import express from "express";
|
||||
import path from "node:path";
|
||||
import { fileURLToPath } from "node:url";
|
||||
|
||||
import { app } from "./server.mjs";
|
||||
import app from "./server.mjs";
|
||||
import api from "./api.mjs";
|
||||
import feed from "./feed.mjs";
|
||||
import utils from "./utils.mjs";
|
||||
|
@ -1,3 +1,6 @@
|
||||
/**
|
||||
* Setup the server app and apply common middlewares.
|
||||
*/
|
||||
import express from "express";
|
||||
import compression from "compression";
|
||||
import helmet from "helmet";
|
||||
@ -8,10 +11,15 @@ import utils from "./utils.mjs";
|
||||
|
||||
// server
|
||||
|
||||
export const app = express();
|
||||
const app = express();
|
||||
|
||||
// get form data as json.
|
||||
app.use(express.urlencoded({ extended: true }));
|
||||
|
||||
// compress responses.
|
||||
app.use(compression());
|
||||
|
||||
// set security policies.
|
||||
app.use(
|
||||
helmet.contentSecurityPolicy({
|
||||
directives: {
|
||||
@ -19,8 +27,10 @@ app.use(
|
||||
},
|
||||
}),
|
||||
);
|
||||
// add logging.
|
||||
app.use(morgan("combined"));
|
||||
|
||||
// set cors options
|
||||
const corsOptions = {
|
||||
origin: utils.cors,
|
||||
optionsSuccessStatus: 200,
|
||||
@ -28,4 +38,7 @@ const corsOptions = {
|
||||
|
||||
app.use(cors(corsOptions));
|
||||
|
||||
// trust reverse proxies.
|
||||
app.set("trust proxy", "127.0.0.1");
|
||||
|
||||
export default app;
|
||||
|
@ -1,26 +1,29 @@
|
||||
/**
|
||||
* Utilities and constants.
|
||||
*/
|
||||
import RateLimit from "express-rate-limit";
|
||||
|
||||
import config from "./config.mjs";
|
||||
|
||||
// Constants
|
||||
|
||||
const domain = process.env.DOMAIN || config.config.domain || "localhost";
|
||||
const domain = process.env.DOMAIN || config.domain || "localhost";
|
||||
|
||||
const port = process.env.PORT || config.config.port || 8080;
|
||||
const port = process.env.PORT || config.port || 8080;
|
||||
|
||||
const db_path = process.env.DB || config.config.db_path || "ucs.db";
|
||||
const db_path = process.env.DB || config.db_path || "ucs.db";
|
||||
|
||||
const cors = (function () {
|
||||
let origins = new Set();
|
||||
for (const site in config.config.sites) {
|
||||
config.config.sites[site].cors.forEach((origin) => {
|
||||
for (const site in config.sites) {
|
||||
config.sites[site].cors.forEach((origin) => {
|
||||
origins.add(origin);
|
||||
});
|
||||
}
|
||||
return Array.from(origins);
|
||||
})();
|
||||
|
||||
// functions
|
||||
// Functions
|
||||
|
||||
function escapeHtml(unsafe) {
|
||||
return unsafe
|
||||
@ -31,7 +34,7 @@ function escapeHtml(unsafe) {
|
||||
.replace(/'/g, "'");
|
||||
}
|
||||
|
||||
// limiters
|
||||
// Limiters
|
||||
|
||||
const limiter = (limit) =>
|
||||
RateLimit({
|
||||
@ -45,6 +48,6 @@ export default {
|
||||
db_path,
|
||||
cors,
|
||||
escapeHtml,
|
||||
get_limiter: limiter(config.config.limits_per_second.get),
|
||||
post_limiter: limiter(config.config.limits_per_second.post),
|
||||
get_limiter: limiter(config.limits_per_minute.get),
|
||||
post_limiter: limiter(config.limits_per_minute.post),
|
||||
};
|
||||
|
Loading…
Reference in New Issue
Block a user