backend init
This commit is contained in:
commit
f4fbbbeadd
5
backend/.gitignore
vendored
Normal file
5
backend/.gitignore
vendored
Normal file
@ -0,0 +1,5 @@
|
||||
comment.json
|
||||
node_modules
|
||||
ucs.db
|
||||
ucs.db-shm
|
||||
ucs.db-wal
|
155
backend/main.mjs
Normal file
155
backend/main.mjs
Normal file
@ -0,0 +1,155 @@
|
||||
import express from 'express';
|
||||
import Database from 'better-sqlite3';
|
||||
import { migrate } from '@blackglory/better-sqlite3-migrations';
|
||||
|
||||
// Constants
|
||||
|
||||
const MAX_LENGTHS = {
|
||||
username: 100,
|
||||
user_website: 200,
|
||||
message_body: 1000
|
||||
};
|
||||
|
||||
// functions
|
||||
|
||||
function escapeHtml(unsafe) {
|
||||
return unsafe
|
||||
.replace(/&/g, "&")
|
||||
.replace(/</g, "<")
|
||||
.replace(/>/g, ">")
|
||||
.replace(/"/g, """)
|
||||
.replace(/'/g, "'");
|
||||
}
|
||||
|
||||
// db
|
||||
|
||||
const db = new Database('ucs.db'); // , { verbose: console.log });
|
||||
db.pragma('journal_mode = WAL');
|
||||
|
||||
const migrations = [
|
||||
{ version: 1,
|
||||
up: `
|
||||
CREATE TABLE site (
|
||||
id integer primary key autoincrement,
|
||||
url text not null,
|
||||
comment_token text not null,
|
||||
length_limit integer
|
||||
) STRICT;
|
||||
|
||||
CREATE TABLE comment (
|
||||
id integer not null,
|
||||
site integer not null,
|
||||
path text not null,
|
||||
user text not null,
|
||||
user_website text,
|
||||
message text not null,
|
||||
published text default (datetime('now')),
|
||||
reply_to integer,
|
||||
FOREIGN KEY(site) REFERENCES site(id),
|
||||
PRIMARY KEY (site, path, id)
|
||||
) STRICT;
|
||||
`,
|
||||
down: `
|
||||
DROP TABLE comment;
|
||||
DROP TABLE site;
|
||||
`
|
||||
}
|
||||
];
|
||||
|
||||
migrate(db, migrations, 1);
|
||||
|
||||
// server
|
||||
|
||||
const app = express();
|
||||
const port = 8080;
|
||||
|
||||
app.use(express.static("public"));
|
||||
app.use(express.urlencoded({ extended: true }));
|
||||
|
||||
app.get('/', (req, res) => {
|
||||
res.send(`<p>Hello, <span style="color: blue;">${req.get('User-Agent')}</span>!</p>`);
|
||||
});
|
||||
|
||||
// GET
|
||||
app.get('/url/:site/*', (req, res) => {
|
||||
const site = req.params.site;
|
||||
const path = req.params[0];
|
||||
const stmt = db.prepare(`
|
||||
SELECT
|
||||
c.id as id,
|
||||
user,
|
||||
user_website,
|
||||
message,
|
||||
published,
|
||||
reply_to
|
||||
FROM
|
||||
(SELECT id from site where url = @site) s
|
||||
JOIN (SELECT * FROM comment WHERE path = @path) c
|
||||
ON c.site = s.id
|
||||
ORDER BY c.id
|
||||
;`);
|
||||
const comments = stmt.all({ site, path });
|
||||
res.json(comments);
|
||||
});
|
||||
|
||||
app.use(express.json());
|
||||
|
||||
// POST
|
||||
app.post('/url/:site/*', (req, res) => {
|
||||
const site = req.params.site;
|
||||
const path = req.params[0];
|
||||
let object = {
|
||||
user: escapeHtml(req.body.user),
|
||||
user_website: escapeHtml(req.body.user_website),
|
||||
message: escapeHtml(req.body.message),
|
||||
reply_to: req.body.reply_to || null,
|
||||
site,
|
||||
path,
|
||||
};
|
||||
|
||||
// validation
|
||||
const user_token = req.body.token;
|
||||
const site_info = db.prepare(`SELECT comment_token, length_limit as message_length_limit FROM site WHERE url = ?`).get(site);
|
||||
console.log(site_info.comment_token, site_info.message_length_limit);
|
||||
|
||||
if (user_token !== site_info.comment_token) {
|
||||
res.status(403).json("Wrong token.");
|
||||
} else if (object.user.length > MAX_LENGTHS.username) {
|
||||
res.status(400).json("Username is too long.");
|
||||
} else if (object.user_website > MAX_LENGTHS.user_website) {
|
||||
res.status(400).json("User website is too long.");
|
||||
} else if (object.message > site_info.message_length_limit) {
|
||||
res.status(400).json("Message body is too long.");
|
||||
} else {
|
||||
const stmt = db.prepare(`
|
||||
INSERT INTO comment(id, site, path, user, user_website, message, reply_to)
|
||||
SELECT
|
||||
( SELECT count(*)
|
||||
FROM (SELECT * FROM comment WHERE path = @path) c
|
||||
JOIN (SELECT id FROM site WHERE url = @site) s
|
||||
ON s.id = c.id
|
||||
),
|
||||
( SELECT id FROM site WHERE url = @site ),
|
||||
@path,
|
||||
@user,
|
||||
@user_website,
|
||||
@message,
|
||||
@reply_to
|
||||
RETURNING
|
||||
id as id,
|
||||
user,
|
||||
user_website,
|
||||
message,
|
||||
published,
|
||||
reply_to
|
||||
;
|
||||
`);
|
||||
const comment = stmt.all(object);
|
||||
|
||||
res.json(comment);
|
||||
}
|
||||
});
|
||||
|
||||
app.listen(port, () => {
|
||||
console.log(`Listening on port ${port}`);
|
||||
});
|
1684
backend/package-lock.json
generated
Normal file
1684
backend/package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
24
backend/package.json
Normal file
24
backend/package.json
Normal file
@ -0,0 +1,24 @@
|
||||
{
|
||||
"name": "ucs-backend",
|
||||
"version": "0.1.0",
|
||||
"description": "Backend for the Universal Comment System",
|
||||
"main": "main.mjs",
|
||||
"scripts": {
|
||||
"start": "nodemon main.mjs",
|
||||
"test": "echo \"Error: no test specified\" && exit 1"
|
||||
},
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "https://git.alloca.space/me/ucs.git"
|
||||
},
|
||||
"author": "alloca",
|
||||
"license": "MPL-2.0",
|
||||
"dependencies": {
|
||||
"@blackglory/better-sqlite3-migrations": "^0.1.19",
|
||||
"better-sqlite3": "^11.8.1",
|
||||
"express": "^4.21.2"
|
||||
},
|
||||
"devDependencies": {
|
||||
"nodemon": "^3.1.9"
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user