UNPKG

hotweby

Version:

Automatic hot-reloading webserver using websockets

138 lines 19.1 kB
import express from "express"; import * as afs from "fs/promises"; import * as path from "path"; import { WebSocket, WebSocketServer } from "ws"; export function createReloadHtmlCode() { const reloadHtmlScript = async () => { console.log("[HOTWEBY]: connect websocket..."); const connectWs = () => { const wsUrl = (location.protocol === "https:" ? "wss://" : "ws://") + location.host; const ws = new WebSocket(wsUrl); ws.onopen = () => console.log("[HOTWEBY]: websocket connected"); ws.onclose = () => { console.log("[HOTWEBY]: websocket closed, reloading..."); setTimeout(() => location.reload(), 100); }; ws.onerror = err => { console.error("[HOTWEBY]: websocket error, reloading...", err); setTimeout(() => location.reload(), 1000); }; }; connectWs(); }; const reloadHtmlCode = "" + reloadHtmlScript; return "<script>\n(" + reloadHtmlCode + ")()\n</script>"; } export function createExpress(targetDir, reloadHtmlCode, autoExtensionResolution, verbose) { const app = express(); app.use(async (req, res, next) => { if (!req.path.endsWith(".html") && !req.path.endsWith("/")) { return next(); } let reqPath = req.path; if (reqPath.endsWith("/")) { reqPath += "index.html"; } try { const data = await afs.readFile(targetDir + reqPath, "utf8"); res.status(200); res.send(reloadHtmlCode + "\n" + data.toString()); verbose && console.info("Served html-file '" + req.path + "' from '" + targetDir + reqPath + "'"); } catch (err) { console.error("Cant read requested html-file '" + req.path + "'" + "\nfrom '" + targetDir + reqPath + "':\n", err); res.status(503); res.setHeader("Content-Type", "text/plain"); res.setHeader("Retry-After", "5"); res.send("Cant read requested html-file '" + req.path + "'"); } }); autoExtensionResolution && app.use(async (req, res, next) => { const resolvedExtension = await autoResolveExtensions(req.url, targetDir, verbose); if (resolvedExtension) { req.url = resolvedExtension; } next(); }); app.use(express.static(targetDir)); return app; } export async function autoResolveExtensions(reqUrl, targetDir, verbose) { while (reqUrl.startsWith("/")) { reqUrl = reqUrl.slice(1); } if (reqUrl === "") { return undefined; } try { let realPath = targetDir + "/" + reqUrl; if ((await pathType(realPath)) === "none") { const files = await afs.readdir(path.dirname(realPath)); const baseName = path.basename(realPath); for (const file of files) { if (file.startsWith(baseName + ".")) { verbose && console.info("Auto resolved extension for '" + reqUrl + "' to be '" + path.dirname(reqUrl) + "/" + file + "'"); return path.dirname(reqUrl) + "/" + file; } } } } catch (err) { verbose && console.error("Error while auto resolving extension for " + reqUrl + "\n", err); } return undefined; } export async function pathType(path) { try { const stat = await afs.stat(path); return stat.isFile() ? "file" : "dir"; } catch { } return "none"; } export function createWebSocketServer(httpServer, registerTrigger) { const wsServer = new WebSocketServer({ noServer: true }); wsServer.on("connection", (ws, req) => { registerTrigger(() => { if (ws.readyState === ws.OPEN || ws.readyState === ws.CONNECTING) { ws.close(); } }); }); httpServer.on("upgrade", (req, socket, head) => { wsServer.handleUpgrade(req, socket, head, ws => { wsServer.emit("connection", ws, req); }); }); return wsServer; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"server.js","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"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","sourcesContent":["import express from \"express\"\nimport * as afs from \"fs/promises\"\nimport { IncomingMessage, Server, ServerResponse } from \"http\"\nimport * as path from \"path\"\nimport { WebSocket, WebSocketServer } from \"ws\"\nimport { TriggerHandler } from \"./index.js\"\n\nexport function createReloadHtmlCode() {\n    const reloadHtmlScript = async () => {\n        console.log(\"[HOTWEBY]: connect websocket...\")\n\n        const connectWs = () => {\n            const wsUrl =\n                (location.protocol === \"https:\"\n                    ? \"wss://\"\n                    : \"ws://\") + location.host\n\n            const ws = new WebSocket(wsUrl)\n            ws.onopen = () =>\n                console.log(\"[HOTWEBY]: websocket connected\")\n            ws.onclose = () => {\n                console.log(\n                    \"[HOTWEBY]: websocket closed, reloading...\",\n                )\n                setTimeout(() => location.reload(), 100)\n            }\n            ws.onerror = err => {\n                console.error(\n                    \"[HOTWEBY]: websocket error, reloading...\",\n                    err,\n                )\n                setTimeout(() => location.reload(), 1000)\n            }\n        }\n\n        connectWs()\n    }\n\n    const reloadHtmlCode = \"\" + reloadHtmlScript\n\n    return \"<script>\\n(\" + reloadHtmlCode + \")()\\n</script>\"\n}\n\nexport function createExpress(\n    targetDir: string,\n    reloadHtmlCode: string,\n    autoExtensionResolution: boolean,\n    verbose: boolean,\n) {\n    const app = express()\n\n    app.use(async (req, res, next) => {\n        if (\n            !req.path.endsWith(\".html\") &&\n            !req.path.endsWith(\"/\")\n        ) {\n            return next()\n        }\n\n        let reqPath = req.path\n\n        if (reqPath.endsWith(\"/\")) {\n            reqPath += \"index.html\"\n        }\n\n        try {\n            const data = await afs.readFile(\n                targetDir + reqPath,\n                \"utf8\",\n            )\n            res.status(200)\n            res.send(reloadHtmlCode + \"\\n\" + data.toString())\n\n            verbose &&\n                console.info(\n                    \"Served html-file '\" +\n                        req.path +\n                        \"' from '\" +\n                        targetDir +\n                        reqPath +\n                        \"'\",\n                )\n        } catch (err) {\n            console.error(\n                \"Cant read requested html-file '\" +\n                    req.path +\n                    \"'\" +\n                    \"\\nfrom '\" +\n                    targetDir +\n                    reqPath +\n                    \"':\\n\",\n                err,\n            )\n            res.status(503)\n            res.setHeader(\"Content-Type\", \"text/plain\")\n            res.setHeader(\"Retry-After\", \"5\")\n            res.send(\n                \"Cant read requested html-file '\" +\n                    req.path +\n                    \"'\",\n            )\n        }\n    })\n\n    autoExtensionResolution &&\n        app.use(async (req, res, next) => {\n            const resolvedExtension =\n                await autoResolveExtensions(\n                    req.url,\n                    targetDir,\n                    verbose,\n                )\n\n            if (resolvedExtension) {\n                req.url = resolvedExtension\n            }\n\n            next()\n        })\n\n    app.use(express.static(targetDir))\n\n    return app\n}\n\nexport async function autoResolveExtensions(\n    reqUrl: string,\n    targetDir: string,\n    verbose: boolean,\n): Promise<string | undefined> {\n    while (reqUrl.startsWith(\"/\")) {\n        reqUrl = reqUrl.slice(1)\n    }\n\n    if (reqUrl === \"\") {\n        return undefined\n    }\n\n    try {\n        let realPath = targetDir + \"/\" + reqUrl\n        if ((await pathType(realPath)) === \"none\") {\n            const files = await afs.readdir(\n                path.dirname(realPath),\n            )\n\n            const baseName = path.basename(realPath)\n            for (const file of files) {\n                if (file.startsWith(baseName + \".\")) {\n                    verbose &&\n                        console.info(\n                            \"Auto resolved extension for '\" +\n                                reqUrl +\n                                \"' to be '\" +\n                                path.dirname(reqUrl) +\n                                \"/\" +\n                                file +\n                                \"'\",\n                        )\n                    return path.dirname(reqUrl) + \"/\" + file\n                }\n            }\n        }\n    } catch (err) {\n        verbose &&\n            console.error(\n                \"Error while auto resolving extension for \" +\n                    reqUrl +\n                    \"\\n\",\n                err,\n            )\n    }\n\n    return undefined\n}\n\nexport async function pathType(\n    path: string,\n): Promise<\"file\" | \"dir\" | \"none\"> {\n    try {\n        const stat = await afs.stat(path)\n        return stat.isFile() ? \"file\" : \"dir\"\n    } catch {}\n    return \"none\"\n}\n\nexport function createWebSocketServer(\n    httpServer: Server<\n        typeof IncomingMessage,\n        typeof ServerResponse\n    >,\n    registerTrigger: (triggerHandler: TriggerHandler) => void,\n) {\n    const wsServer = new WebSocketServer({ noServer: true })\n    wsServer.on(\"connection\", (ws, req) => {\n        registerTrigger(() => {\n            if (\n                ws.readyState === ws.OPEN ||\n                ws.readyState === ws.CONNECTING\n            ) {\n                ws.close()\n            }\n        })\n    })\n\n    httpServer.on(\"upgrade\", (req, socket, head) => {\n        wsServer.handleUpgrade(req, socket, head, ws => {\n            wsServer.emit(\"connection\", ws, req)\n        })\n    })\n\n    return wsServer\n}\n"]}