import "dotenv/config" import express from "express"; import cors from "cors"; import * as fs from "fs"; import rateLimit from "express-rate-limit"; import { shortCommit } from "./modules/sub/currentCommit.js"; 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"; import { Bright, Cyan } from "./modules/sub/consoleText.js"; import stream from "./modules/stream/stream.js"; import loc from "./localization/manager.js"; import { buildFront } from "./modules/build.js"; const commitHash = shortCommit(); const app = express(); app.disable('x-powered-by'); if (fs.existsSync('./.env')) { 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') }); } }) 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') }); } }) await buildFront(); app.use('/api/', apiLimiter); app.use('/api/stream', apiLimiterStream); app.use('/', express.static('./min')); app.use('/', express.static('./src/front')); app.use((req, res, next) => { try { decodeURIComponent(req.path) } catch (e) { return res.redirect(process.env.selfURL); } next(); }); app.get('/api/:type', cors({ origin: process.env.selfURL, optionsSuccessStatus: 200 }), async (req, res) => { 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, format: req.query.format ? req.query.format.slice(0, 5) : "webm", quality: req.query.quality ? req.query.quality.slice(0, 3) : "max", audioFormat: req.query.audioFormat ? req.query.audioFormat.slice(0, 4) : false, isAudioOnly: req.query.audio ? true : false, noWatermark: req.query.nw ? true : false, fullAudio: req.query.ttfull ? true : false, }) res.status(j.status).json(j.body); } else { let j = apiJSON(3, { t: loc(languageCode(req), 'ErrorNoLink', process.env.selfURL) }) res.status(j.status).json(j.body); } 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); } else { let j = apiJSON(0, { t: "no stream id" }) res.status(j.status).json(j.body); } break; default: let j = apiJSON(0, { t: "wrong response type" }) res.status(j.status).json(j.body); break; } } catch (e) { res.status(500).json({ 'status': 'error', 'text': 'something went wrong.' }) } }); app.get("/api", (req, res) => { res.redirect('/api/json') }); app.get("/", (req, res) => { // redirect masochists to a page where they can install a proper browser 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) return } else { res.redirect(internetExplorerRedirect.old) return } } else { res.send(renderPage({ "hash": commitHash, "type": "default", "lang": languageCode(req), "useragent": req.header('user-agent') ? req.header('user-agent') : genericUserAgent })) } }); app.get("/favicon.ico", (req, res) => { res.redirect('/icons/favicon.ico'); }); app.get("/*", (req, res) => { 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('Required config files are missing. Please run "npm run setup" first.') }