Moe-Counter/index.js
roffy3051 1b4e98af4a feat: Improve performance
feat: Add `prefix` param
feat: Theme supports `_start` and `_end` as padding image
2024-11-02 21:44:48 +08:00

163 lines
4.1 KiB
JavaScript

"use strict";
require('dotenv').config();
const express = require("express");
const compression = require("compression");
const { z } = require("zod");
const db = require("./db");
const { themeList, getCountImage } = require("./utils/themify");
const { cors, ZodValid } = require("./utils/middleware");
const { randomArray, logger } = require("./utils");
const app = express();
app.use(express.static("assets"));
app.use(compression());
app.use(cors());
app.set("view engine", "pug");
app.get('/', (req, res) => {
const site = process.env.APP_SITE || `${req.protocol}://${req.get('host')}`
const ga_id = process.env.GA_ID || null
res.render('index', {
site,
ga_id,
themeList,
})
});
// get the image
app.get(["/@:name", "/get/@:name"],
ZodValid({
params: z.object({
name: z.string().max(32),
}),
query: z.object({
theme: z.string().default("moebooru"),
padding: z.coerce.number().int().min(0).max(16).default(7),
offset: z.coerce.number().min(-500).max(500).default(0),
align: z.enum(["top", "center", "bottom"]).default("top"),
scale: z.coerce.number().min(0.1).max(2).default(1),
pixelated: z.enum(["0", "1"]).default("1"),
darkmode: z.enum(["0", "1", "auto"]).default("auto"),
// Unusual Options
num: z.coerce.number().int().min(0).max(1e15).default(0), // a carry-safe integer, less than `2^53-1`, and aesthetically pleasing in decimal.
prefix: z.coerce.number().int().min(-1).max(999999).default(-1)
})
}),
async (req, res) => {
const { name } = req.params;
let { theme = "moebooru", num = 0, ...rest } = req.query;
// This helps with GitHub's image cache
res.set({
"content-type": "image/svg+xml",
"cache-control": "max-age=0, no-cache, no-store, must-revalidate",
});
const data = await getCountByName(String(name), Number(num));
if (name === "demo") {
res.set("cache-control", "max-age=31536000");
}
if (theme === "random") {
theme = randomArray(Object.keys(themeList));
}
// Send the generated SVG as the result
const renderSvg = getCountImage({
count: data.num,
theme,
...rest
});
res.send(renderSvg);
logger.debug(
data,
{ theme, ...req.query },
`ip: ${req.headers['x-forwarded-for'] || req.connection.remoteAddress}`,
`ref: ${req.get("Referrer") || null}`,
`ua: ${req.get("User-Agent") || null}`
);
}
);
// JSON record
app.get("/record/@:name", async (req, res) => {
const { name } = req.params;
const data = await getCountByName(name);
res.json(data);
});
app.get("/heart-beat", (req, res) => {
res.set("cache-control", "max-age=0, no-cache, no-store, must-revalidate");
res.send("alive");
logger.debug("heart-beat");
});
const listener = app.listen(process.env.APP_PORT || 3000, () => {
logger.info("Your app is listening on port " + listener.address().port);
});
let __cache_counter = {};
let enablePushDelay = process.env.DB_INTERVAL > 0
let needPush = false;
if (enablePushDelay) {
setInterval(() => {
needPush = true;
}, 1000 * process.env.DB_INTERVAL);
}
async function pushDB() {
if (Object.keys(__cache_counter).length === 0) return;
if (enablePushDelay && !needPush) return;
try {
needPush = false;
logger.info("pushDB", __cache_counter);
const counters = Object.keys(__cache_counter).map((key) => {
return {
name: key,
num: __cache_counter[key],
};
});
await db.setNumMulti(counters);
__cache_counter = {};
} catch (error) {
logger.error("pushDB is error: ", error);
}
}
async function getCountByName(name, num) {
const defaultCount = { name, num: 0 };
if (name === "demo") return { name, num: "0123456789" };
if (num > 0) { return { name, num } };
try {
if (!(name in __cache_counter)) {
const counter = (await db.getNum(name)) || defaultCount;
__cache_counter[name] = counter.num + 1;
} else {
__cache_counter[name]++;
}
pushDB();
return { name, num: __cache_counter[name] };
} catch (error) {
logger.error("get count by name is error: ", error);
return defaultCount;
}
}