var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } return new (P || (P = Promise))(function (resolve, reject) { function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } step((generator = generator.apply(thisArg, _arguments || [])).next()); }); }; import fs from "node:fs"; import path from "node:path"; import Tpl from "./template.js"; export default class Router { constructor() { this.routes = new Map(); this.mimes = new Map(); } importRoutesFromPath(p_1) { return __awaiter(this, arguments, void 0, function* (p, tpl = false) { const dirEntries = yield fs.promises.readdir(path.resolve() + '/' + p, { withFileTypes: true }); for (const tmp of dirEntries) { if (tmp.isFile() && (tmp.name.endsWith(".mjs") || tmp.name.endsWith(".js"))) { const routeModule = (yield import(`${path.resolve()}/${p}/${tmp.name}`)).default; this.use(routeModule(this, tpl)); } else if (tmp.isDirectory()) { yield this.importRoutesFromPath(p + '/' + tmp.name); } } return this; }); } group(basePath, callback) { const self = this; const methods = { get(path, ...handlers) { const fullPath = self.combinePaths(basePath, path); return self.registerRoute(fullPath, "get", handlers); }, post(path, ...handlers) { const fullPath = self.combinePaths(basePath, path); return self.registerRoute(fullPath, "post", handlers); }, put(path, ...handlers) { const fullPath = self.combinePaths(basePath, path); return self.registerRoute(fullPath, "put", handlers); }, delete(path, ...handlers) { const fullPath = self.combinePaths(basePath, path); return self.registerRoute(fullPath, "delete", handlers); }, patch(path, ...handlers) { const fullPath = self.combinePaths(basePath, path); return self.registerRoute(fullPath, "patch", handlers); } }; callback(methods); return this; } combinePaths(basePath, subPath) { if (typeof basePath === "string" && typeof subPath === "string") return `${basePath.replace(/\/$/, "")}/${subPath.replace(/^\//, "")}`; if (basePath instanceof RegExp && typeof subPath === "string") return new RegExp(`${basePath.source}${subPath.replace(/^\//, "")}`); if (typeof basePath === "string" && subPath instanceof RegExp) return new RegExp(`${basePath.replace(/\/$/, "")}${subPath.source}`); if (basePath instanceof RegExp && subPath instanceof RegExp) return new RegExp(`${basePath.source}${subPath.source}`); throw new TypeError("Invalid path types. Both basePath and subPath must be either string or RegExp."); } use(obj) { if (obj instanceof Router) { this.routes = new Map([...this.routes, ...obj.routes]); this.sortRoutes(); } if (obj instanceof Tpl) { this.tpl = obj; } } get(path, cb) { this.registerRoute(path, "get", cb); return this; } post(path, cb) { this.registerRoute(path, "post", cb); return this; } head(path, cb) { this.registerRoute(path, "head", cb); return this; } put(path, cb) { this.registerRoute(path, "put", cb); return this; } delete(path, cb) { this.registerRoute(path, "delete", cb); return this; } patch(path, cb) { this.registerRoute(path, "patch", cb); return this; } registerRoute(path, method, handler) { var _a; const route = (_a = this.routes.get(path)) !== null && _a !== void 0 ? _a : { [method]: handler }; this.routes.set(path, route); console.log("route set:", method.toUpperCase(), path); this.sortRoutes(); return this; } getRoute(path, method) { return [...this.routes.entries()] .find(r => { var _a, _b; return typeof r[0] === "string" ? r[0] === path : ((_b = (_a = r[0]).exec) === null || _b === void 0 ? void 0 : _b.call(_a, path)) && r[1][method.toLowerCase()]; }); } sortRoutes() { this.routes = new Map([...this.routes.entries()].sort().reverse()); return this; } readMimes(file = "/etc/mime.types") { fs.readFileSync(file, "utf-8") .split("\n") .filter(line => !line.startsWith("#") && line) .forEach(line => { const [mimeType, extensions] = line.split(/\s+/); extensions === null || extensions === void 0 ? void 0 : extensions.split(" ").forEach(ext => this.mimes.set(ext, mimeType)); }); } static({ dir = path.resolve() + "/public", route = /^\/public/ }) { if (!this.mimes.size) this.readMimes(); this.get(route, [(req, res) => { try { const filename = req.url.pathname.replace(route, "") || "index.html"; const mime = this.mimes.get(filename.split(".").pop() || ""); const file = path.join(dir, filename); const stat = fs.statSync(file); if (req.headers.range) { const [startStr, endStr] = req.headers.range.replace(/bytes=/, "").split("-"); const start = parseInt(startStr, 10); const end = endStr ? parseInt(endStr, 10) : stat.size - 1; res.writeHead(206, { "Content-Range": `bytes ${start}-${end}/${stat.size}`, "Accept-Ranges": "bytes", "Content-Length": end - start + 1, "Content-Type": mime, }); fs.createReadStream(file, { start, end }).pipe(res); } else { res.writeHead(200, { "Content-Length": stat.size, "Content-Type": mime, }); fs.createReadStream(file).pipe(res); } } catch (err) { console.error(err); res.reply({ code: 404, body: "404 - File not found" }); } }]); return this; } }