const Component = require("./_component"); const utils = require("../utils"); const https = require("https"); const http = require("http"); const express = require("express"); const bodyParser = require("body-parser"); const bodyHand = require('body'); const url = require('url'); const path = require('path'); const fs = require('fs'); const fetch = require('node-fetch'); const cookieParser = require('cookie-parser'); const mime = require("mime"); function logger (...p) { //return; console.log("[DEBUG.http]:", ...p); } function fetchProxy (req, res, reverseTo) { const params = { method: req.method.toUpperCase(), headers: Object.assign({}, req.headers), redirect: "follow", follow: 20, compress: false }; delete params.headers.host; delete params.headers.referer; if (!["GET", "HEAD"].includes(params.method)) params.body = req.body; const reqAddress = url.resolve(reverseTo, path.join("/", req.originalUrl)); fetch(reqAddress, params) .then(async result => { const content = await result.arrayBuffer(); //logger("result:", result); //logger("result.arrayBuffer():", content); //logger("result.Buffer:", Buffer.from(content).toString()); const headers = Object.fromEntries(result.headers); //logger("headers Headers", result.headers); //logger("headers Headers.raw()", result.headers.raw()); logger("headers object", headers); Object.entries(headers).forEach(([header, value]) => { res.set(header, value); }); const finalContent = Buffer.from(content); res.status(result.status).send(finalContent); }) .catch(err => { console.error(err); res.status(500).send(req.st500); }); } function authPage (self, req, res, next) { if (self.syntax.auth) { const ACCESS = "Basic " + Buffer.from(`${self.syntax.auth.login}:${self.syntax.auth.password}`).toString('base64'); if (req.headers['authorization'] === ACCESS) { return next(); } else { if (self.syntax.auth.mode === "strict" && req.headers['authorization'] !== undefined) { return res.status(403).send(req.st403); } res.set("WWW-Authenticate", "Basic"); return res.status(401).send(); } } next(); } function redirectPage (redirTo, req, res) { redirTo = url.resolve(redirTo, path.join("/", req.originalUrl)); res.redirect(redirTo); } function toNormalText (httpPath) { const findedUnicodeTexts = httpPath.match(/(\%[A-F0-9]{2}){1,}/g) ?? []; for (let matchedQuery of findedUnicodeTexts) { httpPath = httpPath .split(matchedQuery) .join(Buffer.from( matchedQuery .replace(/\%/g, ""), "hex" ).toString("utf8")) } return httpPath; } function renderDirContent (dirpath, displayedDir) { const files = fs.readdirSync(dirpath).map(f => ({ name: f, path: path.join(displayedDir, f), folder: displayedDir, lstat: fs.lstatSync(path.join(dirpath, f)), sizeReadable: utils.getStrSize(fs.lstatSync(path.join(dirpath, f)).size), hash: "1ac89f5efc", mime: mime.getType(f) })); files.folder = displayedDir; const ht = (i) => parseInt(Buffer.from(i ?? "0").toString("hex"), 16); files.sort((a, b) => a.name - b.name); files.sort((a, b) => ht(a.mime) - ht(b.mime)); files.sort((a, b) => +(b.lstat.isDirectory()) - +(a.lstat.isDirectory())); return `Content of dir: ${displayedDir}

Content of dir: ${displayedDir}


${renderDirContent.genTRs(files)}
Filename Size Checksum
`; } renderDirContent.genTRs = function (files) { const allFiles = files.map(f => `${renderDirContent.getIcon(f)}${f.name + (f.lstat.isDirectory() ? "/" : "")}${f.sizeReadable}${f.hash}`); return [`${renderDirContent.getIcon(null)}../`].concat(allFiles).join("\n\t\t\t"); } renderDirContent.getIcon = function (fileData) { let iconUrl = ""; if (!fileData) iconUrl = ""; else if (fileData.lstat.isDirectory()) { iconUrl = ""; } else { console.log("fileData:", fileData); if (fileData.mime?.includes("image/")) { iconUrl = ""; } else if (fileData.mime?.includes("html")) { iconUrl = ""; } else if (fileData.mime?.includes("video")) { iconUrl = ""; } else if (fileData.mime?.includes("audio/")) { iconUrl = ""; } } return ``; } function simpleHTTP (staticPath, req, res, dirContentFlag = false) { const curpath = toNormalText(req.originalUrl .replace(/\?.{0,}/gmi)); const pathToFile = path.join(staticPath, curpath); console.log("curpath:", curpath); console.log("pathToFile:", pathToFile); let finalPathToFile; let mimetype = "text/html"; let isDir = false; try { const lstat = fs.lstatSync(pathToFile); if (lstat.isFile()) { mimetype = mime.getType(pathToFile); finalPathToFile = pathToFile } else if (lstat.isDirectory()) { finalPathToFile = path.join(pathToFile, "index.html"); isDir = true; } fs.readFile(finalPathToFile, (err, data) => { if (err) { if (isDir && dirContentFlag) { try { return res.send(renderDirContent(pathToFile, curpath)); } catch (e) { return res.status(404).send(req.st404); /*if (e.code === "EACCES") return res.status(404).send(req.st404); return res.status(500).send(req.st500);*/ } } return res.status(404).send(req.st404); } res.set("Content-Type", mimetype); res.send(data); }); } catch (e) { return res.status(404).send(req.st404); } } module.exports = class HTTPServer extends Component { constructor (adr, syntaxTree) { super(adr, syntaxTree); this.routelist = []; } domainAction (context, domain, callCommand, req, res, next) { logger("DOMAIN ACT:", this.syntax); if (!(context instanceof HTTPServer)) throw new SyntaxError("You can't call revHTTP domain from unsupported type. Supported types: revHTTP, simpleHTTP, redirectHTTP"); switch (callCommand) { case "proxy": return fetchProxy(req, res, this.syntax.target); case "auth": const authContext = context.syntax.auth === undefined ? this : context; return authPage(authContext, req, res, next); case "redirect": return redirectPage(this.syntax.target, req, res); case "shttp": return simpleHTTP(this.syntax.target, req, res, !!this.syntax.showDir); } } preinit () { this.app = express(); this.app.use('/*', async (req, res, next) => { // Myself body parser :) bodyHand(req, res, { limit: 9999999999, cache: false, encoding: 'base64' }, (err, body) => { if (!err) { req.body = Buffer.from(body, 'base64'); } else { req.body = undefined; } next(); }); }); this.app.use((req, res, next) => { const currentDomain = req.hostname; const isNotDomained = !this.syntax.domains[currentDomain]; if (!isNotDomained) { return this .callDomain(currentDomain, this.syntax.domains[currentDomain]) .argv("auth", req, res, next); } authPage(this, req, res, next) }); this.app.use((req, res, next) => { req.st403 = "

403: Permission denied


Permission denied.

"; req.st404 = "

404: Page not finded


Please, recheck the URL address and try again.

"; req.st500 = "500: Internal Server Error

Error:


Error catched. Check logs

"; next(); }); this.app.use(cookieParser()); this.routelist.forEach(route => { this.app.use(route); }); } postinit (startHandler) { if (!this.syntax.encryption) { this.server = http.createServer(this.app); } else { this.server = https.createServer({ cert : fs.readFileSync(this.syntax.encryption.public, 'utf-8'), key : fs.readFileSync(this.syntax.encryption.private, 'utf-8') }, this.app); } this.server.listen(this.adr.port, this.adr.address, startHandler); } stop () { this.server.close(() => logger("Server stopped")); } } module.exports.fetchProxy = fetchProxy; module.exports.authPage = authPage; module.exports.redirectPage = redirectPage; module.exports.simpleHTTP = simpleHTTP;