@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
JavaScript
;
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;