more flexible

This commit is contained in:
leca 2025-06-27 20:31:44 +03:00
parent 044f4094f0
commit 7ab9d70d47
9 changed files with 65 additions and 34 deletions

View File

@ -1,6 +1,7 @@
FROM node:22-bullseye FROM node:22-bullseye
ARG APP_PORT ARG APP_PORT
ENV APP_PORT $APP_PORT ENV APP_PORT $APP_PORT
WORKDIR /app
COPY migrations migrations COPY migrations migrations
COPY public public COPY public public
COPY views views COPY views views

View File

@ -1,2 +1 @@
# captcha_agregator # captcha_agregator

View File

@ -5,18 +5,18 @@ services:
ports: ports:
- ${APP_PORT}:${APP_PORT} - ${APP_PORT}:${APP_PORT}
depends_on: depends_on:
- database database:
condition: service_healthy
restart: on-failure restart: on-failure
volumes: volumes:
- ./data/uploads:${DATA_DIR} - ./data/uploads:${DATA_DIR}
- ./.env:/app/.env:ro
database: database:
hostname: database hostname: database
image: 'postgres:15' image: 'postgres:15'
volumes: volumes:
- ./data/db:/var/lib/postgresql/data - ./data/db:/var/lib/postgresql/data
env_file: .env env_file: .env
ports:
- 5432:5432
environment: environment:
POSTGRES_USER: ${DBUSER} POSTGRES_USER: ${DBUSER}
POSTGRES_PASSWORD: ${DBPASS} POSTGRES_PASSWORD: ${DBPASS}

View File

@ -36,7 +36,7 @@ const blobToBase64 = (blob) => {
reader.onloadend = () => resolve(reader.result); reader.onloadend = () => resolve(reader.result);
reader.readAsDataURL(blob); reader.readAsDataURL(blob);
}); });
} }
const show_stats = async () => { const show_stats = async () => {
const response = await fetch("/api/user/stats", { const response = await fetch("/api/user/stats", {
@ -54,6 +54,19 @@ const show_stats = async () => {
statsText.innerHTML += `You have solved ${stats.my_count} captcha(s)` statsText.innerHTML += `You have solved ${stats.my_count} captcha(s)`
} }
const validate_solution = (settings, solution) => {
if (solution.length != settings.captcha_length) return false;
for (let i = 0; i < solution.length; i++) {
let char = solution[i];
if (!char.match(settings.captcha_regex)) {
console.log("Illegal symbol: " + char);
alert(`Illegal symbol ${char} at position ${i + 1}`);
return false
}
}
return true;
}
window.onload = async () => { window.onload = async () => {
const inputField = document.getElementById("captcha"); const inputField = document.getElementById("captcha");
inputField.focus(); inputField.focus();
@ -62,8 +75,9 @@ window.onload = async () => {
if (!document.cookie.includes('JWT')) { if (!document.cookie.includes('JWT')) {
document.location.href = "/login"; document.location.href = "/login";
} }
const settings = await (await fetch("/api/settings")).json();
const response = await fetch("https://check.ofd.ru/api/captcha/common/img"); const captcha_source_url = settings.captcha_source_url;
const response = await fetch(captcha_source_url);
captcha = await response.blob(); captcha = await response.blob();
const url = URL.createObjectURL(captcha); const url = URL.createObjectURL(captcha);
@ -72,12 +86,16 @@ window.onload = async () => {
form.addEventListener('submit', async (e) => { form.addEventListener('submit', async (e) => {
e.preventDefault(); e.preventDefault();
if (!validate_solution(settings, inputField.value)) {
alert("You must specify valid solution!")
return;
}
// if (!await check_solution(inputField.value)) { // if (!await check_solution(inputField.value)) {
// alert("Капча решена неверно") // alert("Капча решена неверно")
// returnl // returnl
// } // }
const response = await fetch(`/api/captcha/submit`, {method: "POST", headers: {'Content-Type': 'application/json'}, body: JSON.stringify({ "image": await blobToBase64(captcha), "solution": inputField.value})}); const response = await fetch(`/api/captcha/submit`, { method: "POST", headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ "image": await blobToBase64(captcha), "solution": inputField.value }) });
if (response.status == 200) { if (response.status == 200) {
inputField.value = ""; inputField.value = "";
window.location.reload(); window.location.reload();

View File

@ -13,3 +13,4 @@ ADMIN_TOKEN=GENERATE_A_STRONG_TOKEN_HERE
SECRET=GENERATE_A_STRONG_SECRET_HERE SECRET=GENERATE_A_STRONG_SECRET_HERE
#Allowed symbols. Must be a regex #Allowed symbols. Must be a regex
ALPHABET="[123456789ABCDEFGHIJKLMNPQRSTUVWXZY]" ALPHABET="[123456789ABCDEFGHIJKLMNPQRSTUVWXZY]"
CAPTCHA_LENGTH=6

View File

@ -2,7 +2,7 @@ import dotenv from 'dotenv';
dotenv.config({ path: ".env" }); dotenv.config({ path: ".env" });
const getBoolean = value => { return value === 'true'? true : false } const getBoolean = value => { return value === 'true' ? true : false }
const config = { const config = {
dbuser: process.env.DBUSER, dbuser: process.env.DBUSER,
@ -17,7 +17,8 @@ const config = {
data_dir: process.env.DATA_DIR, data_dir: process.env.DATA_DIR,
admin_token: process.env.ADMIN_TOKEN, admin_token: process.env.ADMIN_TOKEN,
secret: process.env.SECRET, secret: process.env.SECRET,
alphabet: process.env.ALPHABET alphabet: process.env.ALPHABET,
captcha_length: Number.parseInt(process.env.CAPTCHA_LENGTH)
} }
export default config; export default config;

View File

@ -4,44 +4,45 @@ import config from '../config.js';
class CaptchaController { class CaptchaController {
async submit(req, res) { async submit(req, res) {
const {image, solution} = req.body; const { image, solution } = req.body;
if (!image) return res.status(400).send({"message":"You must send image blob"}); if (!image) return res.status(400).send({ "message": "You must send image blob" });
if (!solution || solution.length != 6) return res.status(400).send({"message":"You must send a valid solution"});
for (let i = 0; i < solution.length; i ++) { if (!solution || solution.length != config.captcha_length) return res.status(400).send({ "message": "You must send a valid solution" });
for (let i = 0; i < solution.length; i++) {
let char = solution[i]; let char = solution[i];
if (!char.match(config.alphabet)) { if (!char.match(config.alphabet)) {
console.log("Illegal symbol: " + char); console.log("Illegal symbol: " + char);
return res.status(400).send({"message": `Illegal symbol ${char} at position ${i + 1}`}); return res.status(400).send({ "message": `Illegal symbol ${char} at position ${i + 1}` });
} }
} }
try { try {
await CaptchaService.new(image, solution, jwt.decode(req.token).id); await CaptchaService.new(image, solution, jwt.decode(req.token).id);
} catch (e) { } catch (e) {
console.log(`Error upon submitting: ${e}`); console.log(`Error upon submitting: ${e}`);
return res.status(500).send({"message": "Unknown server error. Please, contact the developer."}); return res.status(500).send({ "message": "Unknown server error. Please, contact the developer." });
} }
return res.status(200).send({"message": "Success"}); return res.status(200).send({ "message": "Success" });
} }
async get(req, res) { async get(req, res) {
try { try {
const { id } = req.params; const { id } = req.params;
const captcha = await CaptchaService.get(id); const captcha = await CaptchaService.get(id);
if (captcha == undefined) return res.status(404).send({"message":"no such captcha found"}); if (captcha == undefined) return res.status(404).send({ "message": "no such captcha found" });
return res.status(200).send({"message":"success", "captcha": captcha}) return res.status(200).send({ "message": "success", "captcha": captcha })
} catch (e) { } catch (e) {
console.log(`Error upon requesting one captcha: ${e}`) console.log(`Error upon requesting one captcha: ${e}`)
if (e.code == 'ENOENT') return res.status(404).send({"message": "The ID exists in the DB but I can't find an actual image. Please, contact the developer."}) if (e.code == 'ENOENT') return res.status(404).send({ "message": "The ID exists in the DB but I can't find an actual image. Please, contact the developer." })
return res.status(500).send({"message": "Unknown server error. Please, contact the developer."}) return res.status(500).send({ "message": "Unknown server error. Please, contact the developer." })
} }
} }
async get_all(req, res) { async get_all(req, res) {
try { try {
return res.status(200).send({"message":"success", "captchas": await CaptchaService.get_all()}); return res.status(200).send({ "message": "success", "captchas": await CaptchaService.get_all() });
} catch (e) { } catch (e) {
console.log(`Error upon requesting all captchas: ${e}`) console.log(`Error upon requesting all captchas: ${e}`)
return res.status(500).send({"message": "Unknown server error. Please, contact the developer."}) return res.status(500).send({ "message": "Unknown server error. Please, contact the developer." })
} }
} }
} }

View File

@ -15,6 +15,16 @@ app.use(express.urlencoded({ extended: false }));
app.use(express.json()); app.use(express.json());
app.use(cookieParser()); app.use(cookieParser());
app.get('/api/settings', (req, res) => {
return res.status(200).send(
{
"captcha_source_url": config.captcha_source_url,
"captcha_regex": config.captcha_regex,
"captcha_length": config.captcha_length
}
)
})
app.set('view engine', 'pug'); app.set('view engine', 'pug');
app.use('/api', CaptchaRouter); app.use('/api', CaptchaRouter);

View File

@ -15,7 +15,7 @@ function base64ToArrayBuffer(base64) {
class CaptchaService { class CaptchaService {
async new(image, solution, submitter) { async new(image, solution, submitter) {
const b64 = image.replace(/^data:image\/jpeg;base64,/, ""); const b64 = image.replace(/^.*,/, "");
const arrayBuffer = base64ToArrayBuffer(b64); const arrayBuffer = base64ToArrayBuffer(b64);
const buffer = Buffer.from(arrayBuffer); const buffer = Buffer.from(arrayBuffer);
const hash = createHash('md5').update(buffer).digest('hex'); const hash = createHash('md5').update(buffer).digest('hex');
@ -30,7 +30,7 @@ class CaptchaService {
const path = `${config.data_dir}/${captcha.hash}.jpeg`; const path = `${config.data_dir}/${captcha.hash}.jpeg`;
const image = Buffer.from(await fs.readFile(path)).toString('base64'); const image = Buffer.from(await fs.readFile(path)).toString('base64');
return {"image": image, "solution": captcha.solution, "hash": captcha.hash}; return { "image": image, "solution": captcha.solution, "hash": captcha.hash };
} }
async get_all() { async get_all() {