UNPKG

@darlean/webservice-suite

Version:

Webservice Suite that acts as Web/API Gateway that invokes actors to serve HTTP requests

159 lines (158 loc) 6.98 kB
"use strict"; var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) { var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d; if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc); else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r; return c > 3 && r && Object.defineProperty(target, key, r), r; }; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.WebServiceHostActor = void 0; const base_1 = require("@darlean/base"); const utils_1 = require("@darlean/utils"); const http_1 = require("http"); const url_1 = __importDefault(require("url")); const querystring_1 = __importDefault(require("querystring")); const MAX_BODY_LENGTH = 100 * 1000; class WebServiceHostActor { constructor(config) { this.config = config; const server = (0, http_1.createServer)((req, res) => { setImmediate(async () => { await this.handleRequest(req, res); }); }); this.server = server; } async activate() { const port = this.config.port ?? 80; this.port = port; this.server.listen(port); (0, utils_1.notifier)().info('io.darlean.webservice.Listening', 'Web service [Name] is now listening on port [Port]', () => ({ Name: this.config.name, Port: port })); } async deactivate() { if (this.port) { await new Promise((resolve, error) => { this.server.close((err) => { if (err) { error(err); } resolve(); }); }); } (0, utils_1.notifier)().info('io.darlean.webservice.StoppedListening', 'Web service [Name] stopped listening on port [Port]', () => ({ Name: this.config.name, Port: this.port })); } async touch() { // } async handleRequest(req, res) { try { const urlobj = new url_1.default.URL(req.url ?? '', 'http://' + req.headers.host); // Also decodes special characters like %2f to '/'. The wildcard matching should not depend on percent-encodings. const pathname = decodeURIComponent(urlobj.pathname); for (const handler of this.config.handlers) { if (!!handler.method && handler.method !== req.method) { continue; } const matches = []; if (handler.path) { if (!(0, utils_1.wildcardMatch)(pathname, handler.path, matches)) { continue; } } const buffers = []; let len = 0; for await (const data of req) { const buf = data; len += buf.length; if (len > MAX_BODY_LENGTH) { res.statusCode = 413; res.statusMessage = 'Payload too large'; return; } buffers.push(data); } const finalBuffer = Buffer.concat(buffers); const request = { url: req.url ?? '', hostname: decodeURIComponent(urlobj.hostname), port: parseInt(urlobj.port), protocol: urlobj.protocol, username: decodeURIComponent(urlobj.username), method: req.method, headers: {}, path: pathname, body: finalBuffer }; if (urlobj.search) { const qs = querystring_1.default.parse(urlobj.search.substring(1)); const queryString = {}; for (const [key, value] of Object.entries(qs)) { const v = typeof value === 'string' ? [value] : value ? value : []; queryString[decodeURIComponent(key)] = v.map((x) => decodeURIComponent(x)); } request.searchParams = queryString; } if (request.headers) { for (const [header, value] of Object.entries(req.headers)) { if (typeof value === 'string') { request.headers[header] = value; } } } if (req.headers?.cookies) { request.cookies = []; for (const cookie of req.headers.cookies) { request.cookies.push(cookie); } } if (matches.length > 0) { const placeholders = {}; for (let idx = 0; idx < matches.length; idx++) { const name = handler.placeholders?.[idx] ?? ''.padEnd(idx + 1, '*'); placeholders[name] = matches[idx]; } request.placeholders = placeholders; } const response = await handler.action(request); res.statusCode = response.statusCode; res.statusMessage = response.statusMessage; if (response.headers) { for (const [header, value] of Object.entries(response.headers)) { res.setHeader(header, value); } } if (response.cookies) { for (const cookie of response.cookies) { res.setHeader('set-cookie', cookie); } } res.write(response.body); res.end(); return; } res.statusCode = 404; res.statusMessage = 'File not Dound'; res.end(); } catch (e) { (0, utils_1.notifier)().warning('io.darlean.webservice.ProcessingFailed', 'An error occurred during processing of web service request: [Error]', () => ({ Error: e })); res.statusCode = 500; res.statusMessage = 'Internal server error'; res.end(); } } } __decorate([ (0, base_1.action)() ], WebServiceHostActor.prototype, "touch", null); exports.WebServiceHostActor = WebServiceHostActor;