cobalt/src/cobalt.js

154 lines
6.6 KiB
JavaScript
Raw Normal View History

2022-07-08 19:17:56 +01:00
import "dotenv/config"
import express from "express";
import cors from "cors";
2022-07-08 19:17:56 +01:00
import * as fs from "fs";
import rateLimit from "express-rate-limit";
2022-07-17 12:21:51 +01:00
import { shortCommit } from "./modules/sub/currentCommit.js";
2022-07-08 19:17:56 +01:00
import { appName, genericUserAgent, version, internetExplorerRedirect } from "./modules/config.js";
import { getJSON } from "./modules/api.js";
import renderPage from "./modules/pageRender/page.js";
import { apiJSON, languageCode } from "./modules/sub/utils.js";
2022-07-17 12:21:51 +01:00
import { Bright, Cyan } from "./modules/sub/consoleText.js";
2022-07-08 19:17:56 +01:00
import stream from "./modules/stream/stream.js";
import loc from "./localization/manager.js";
import { buildFront } from "./modules/build.js";
2022-09-11 16:04:06 +01:00
import { changelogHistory } from "./modules/pageRender/onDemand.js";
2022-07-08 19:17:56 +01:00
const commitHash = shortCommit();
2022-07-08 19:17:56 +01:00
const app = express();
app.disable('x-powered-by');
if (fs.existsSync('./.env') && process.env.selfURL && process.env.streamSalt && process.env.port) {
2022-07-08 19:17:56 +01:00
const apiLimiter = rateLimit({
windowMs: 20 * 60 * 1000,
max: 100,
standardHeaders: true,
legacyHeaders: false,
handler: (req, res, next, opt) => {
res.status(429).json({ "status": "error", "text": loc(languageCode(req), 'ErrorRateLimit') });
2022-07-08 19:17:56 +01:00
}
})
const apiLimiterStream = rateLimit({
windowMs: 6 * 60 * 1000,
max: 24,
standardHeaders: true,
legacyHeaders: false,
handler: (req, res, next, opt) => {
res.status(429).json({ "status": "error", "text": loc(languageCode(req), 'ErrorRateLimit') });
2022-07-08 19:17:56 +01:00
}
})
await buildFront();
2022-07-08 19:17:56 +01:00
app.use('/api/', apiLimiter);
app.use('/api/stream', apiLimiterStream);
app.use('/', express.static('./min'));
app.use('/', express.static('./src/front'));
2022-07-08 19:17:56 +01:00
app.use((req, res, next) => {
try {
decodeURIComponent(req.path)
}
2022-08-16 08:14:19 +01:00
catch (e) {
return res.redirect(process.env.selfURL);
2022-07-08 19:17:56 +01:00
}
next();
});
app.get('/api/:type', cors({ origin: process.env.selfURL, optionsSuccessStatus: 200 }), async (req, res) => {
2022-07-08 19:17:56 +01:00
try {
switch (req.params.type) {
case 'json':
if (req.query.url && req.query.url.length < 150) {
let j = await getJSON(req.query.url.trim(), languageCode(req), {
ip: req.header('x-forwarded-for') ? req.header('x-forwarded-for') : req.ip,
2022-11-04 08:49:58 +00:00
format: req.query.format ? req.query.format.slice(0, 5) : "mp4",
quality: req.query.quality ? req.query.quality.slice(0, 3) : "mid",
audioFormat: req.query.audioFormat ? req.query.audioFormat.slice(0, 4) : "mp3",
isAudioOnly: !!req.query.audio,
noWatermark: !!req.query.nw,
fullAudio: !!req.query.ttfull,
})
2022-07-08 19:17:56 +01:00
res.status(j.status).json(j.body);
} else {
let j = apiJSON(3, { t: loc(languageCode(req), 'ErrorNoLink', process.env.selfURL) })
2022-09-04 06:32:06 +01:00
if (!typeof j === "undefined" && j.status && j.body) {
res.status(j.status).json(j.body);
} else {
res.status(500).json({ 'status': 'error', 'text': loc(languageCode(req), 'ErrorUnknownStatus') })
}
2022-07-08 19:17:56 +01:00
}
break;
case 'stream':
if (req.query.p) {
res.status(200).json({ "status": "continue" });
} else if (req.query.t) {
let ip = req.header('x-forwarded-for') ? req.header('x-forwarded-for') : req.ip
stream(res, ip, req.query.t, req.query.h, req.query.e);
2022-07-08 19:17:56 +01:00
} else {
2022-09-03 16:32:39 +01:00
let j = apiJSON(0, { t: "no stream id" })
2022-07-08 19:17:56 +01:00
res.status(j.status).json(j.body);
}
break;
2022-09-11 16:04:06 +01:00
case 'onDemand':
if (req.query.blockId) {
let blockId = req.query.blockId.slice(0, 3)
let r, j;
switch(blockId) {
case "0":
r = changelogHistory();
j = r ? apiJSON(3, { t: r }) : apiJSON(0, { t: "couldn't render this block" })
break;
default:
j = apiJSON(0, { t: "couldn't find a block with this id" })
break;
}
res.status(j.status).json(j.body);
} else {
let j = apiJSON(0, { t: "no block id" })
res.status(j.status).json(j.body);
}
break;
2022-07-08 19:17:56 +01:00
default:
2022-09-11 16:04:06 +01:00
let j = apiJSON(0, { t: "unknown response type" })
2022-07-08 19:17:56 +01:00
res.status(j.status).json(j.body);
break;
}
} catch (e) {
res.status(500).json({ 'status': 'error', 'text': 'something went wrong.' })
}
});
2022-09-01 14:51:18 +01:00
app.get("/api", (req, res) => {
2022-07-08 19:17:56 +01:00
res.redirect('/api/json')
});
2022-09-01 14:51:18 +01:00
app.get("/", (req, res) => {
2022-07-08 19:17:56 +01:00
if (req.header("user-agent") && req.header("user-agent").includes("Trident")) {
if (internetExplorerRedirect.newNT.includes(req.header("user-agent").split('NT ')[1].split(';')[0])) {
res.redirect(internetExplorerRedirect.new)
} else {
res.redirect(internetExplorerRedirect.old)
}
} else {
res.send(renderPage({
"hash": commitHash,
"type": "default",
"lang": languageCode(req),
2022-07-08 19:17:56 +01:00
"useragent": req.header('user-agent') ? req.header('user-agent') : genericUserAgent
}))
}
});
2022-09-01 14:51:18 +01:00
app.get("/favicon.ico", (req, res) => {
2022-07-08 19:17:56 +01:00
res.redirect('/icons/favicon.ico');
});
2022-09-01 14:51:18 +01:00
app.get("/*", (req, res) => {
2022-07-08 19:17:56 +01:00
res.redirect('/')
});
app.listen(process.env.port, () => {
console.log(`\n${Bright(`${appName} (${version})`)}\n\nURL: ${Cyan(`${process.env.selfURL}`)}\nPort: ${process.env.port}\nCurrent commit: ${Bright(`${commitHash}`)}\nStart time: ${Bright(Math.floor(new Date().getTime()))}\n`)
});
} else {
console.log(`you can't run the server without generating a .env file. please run the setup script first: npm run setup`)
}