UNPKG

@jeaks03/overseer

Version:

Just another TypeScript Back-End framework

228 lines 19.8 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); const standard_responses_1 = require("../misc/standard-responses"); const requisites_1 = require("../core/requisites"); const route_utils_1 = require("./route-utils"); const http_error_1 = require("../errors/http-error"); const redirect_1 = require("./redirect"); const core_error_1 = require("../errors/core-error"); const converter_1 = require("../converters/converter"); class AsyncRequestHandler { constructor(serverRequest, serverResponse, router) { this.serverRequest = serverRequest; this.serverResponse = serverResponse; this.router = router; } static async doHandle(serverRequest, serverResponse, router) { const handler = new AsyncRequestHandler(serverRequest, serverResponse, router); await handler.handle(); handler.serverResponse.end(); } async handle() { // Handle cross origin headers and OPTIONS requests this.appendInitialHeaders(); if (route_utils_1.RouteUtils.isMethod(this.serverRequest, 'options')) { return this.handleCORS(); } try { await this.checkAvailableRoute(); } catch (e) { if (e.response) { e.handle(this.serverResponse); } else { http_error_1.HttpError.handleServerError(this.serverResponse, e); } } } /** * Handles possible routing scenarios such as: route exists, resource exists or not found */ async checkAvailableRoute() { const route = this.findRoute(); // Handle existing route if (route) { this.foundRoute = route; await this.routeMatched(); return; } // If no route then handle public resources if (route_utils_1.RouteUtils.isMethod(this.serverRequest, 'get') && this.router.resourceMgr.fileOrIndexExists(route_utils_1.RouteUtils.getBaseUrl(this.serverRequest.url))) { this.router.resourceMgr.handleRequest(route_utils_1.RouteUtils.getBaseUrl(this.serverRequest.url), this.serverResponse); return; } // If no public resource then send 404 this.serverResponse.statusCode = 404; this.serverResponse.setHeader('Content-Type', 'application/json'); } /** * Triggered when a route was found */ async routeMatched() { const rawBody = await this.parseRequest(); const messageConverter = MessageConverterScout. using(this.serverRequest, this.foundRoute) .findReadConverter(rawBody); const body = messageConverter.doRead(rawBody); const pathInfo = this.generatePathInfo(body); // Triggers the controller method associated with this route const output = this.handleRoute(pathInfo); // Handle redirect if (output.body instanceof redirect_1.Redirect) { this.handleRedirect(output.body); return; } // Request is done, write output this.writeResponse(output); } /** * Triggers the controller method associated with the route * @param info param to pass to controller method */ handleRoute(info) { try { return { body: this.foundRoute.handle(info), status: this.foundRoute.details.statusCode }; } catch (e) { return this.handleError(e, info); } } /** * Handles the given error * @param e the error * @param info param to pass to error handler method */ handleError(e, info) { let output = {}; if (e instanceof core_error_1.CoreError) { output.body = e.handle(info); output.status = e.getStatusCode(); } else if (e instanceof http_error_1.HttpError) { output = e.response; } else { console.error(e); output = standard_responses_1.INTERVAL_SERVER_ERROR; } return output; } /** * Generates PathInfo for the controller method to use */ generatePathInfo(body) { return { body, raw: { request: this.serverRequest, response: this.serverResponse }, pathParams: route_utils_1.RouteUtils.parsePathParams(this.serverRequest.url, this.foundRoute.details.path), queryParams: route_utils_1.RouteUtils.parseQueryParams(this.serverRequest.url) }; } /** * Parses the raw http request body to a string promise */ parseRequest() { let requestBody = ''; this.serverRequest.on('data', chunk => requestBody += chunk.toString()); return new Promise((resolve) => this.serverRequest.on('end', () => resolve(requestBody))); } /** * Returns the route associated with the current server request */ findRoute() { const parts = route_utils_1.RouteUtils.getUrlPattern(this.serverRequest.url); let tree = this.router.routeTree; for (const part of parts) { if (!tree[part]) { return null; } tree = tree[part]; } if (!tree || !tree[this.router.routeSym]) { return null; } return tree[this.router.routeSym][this.serverRequest.method.toLowerCase()] || null; } /** * Adds necessary headers to avoid CORS issues */ appendInitialHeaders() { this.serverResponse.setHeader('Access-Control-Allow-Origin', '*'); this.serverResponse.setHeader('Access-Control-Allow-Methods', 'GET, POST, PATCH, PUT, DELETE, OPTIONS'); } /** * Handles redirects */ handleRedirect(body) { this.serverResponse.statusCode = 302; this.serverResponse.setHeader('Location', body.url); } /** * Adds necessary headers to avoid CORS issues in case of OPTIONS requests */ handleCORS() { this.serverResponse.setHeader('Access-Control-Allow-Headers', `Accept, Accept-CH, Accept-Charset, Accept-Datetime, Accept-Encoding, Accept-Ext, Accept-Features, Accept-Language, Accept-Params, Accept-Ranges, Access-Control-Allow-Credentials, Access-Control-Allow-Headers, Access-Control-Allow-Methods, Access-Control-Allow-Origin, Access-Control-Expose-Headers, Access-Control-Max-Age, Access-Control-Request-Headers, Access-Control-Request-Method, Age, Allow, Alternates, Authentication-Info, Authorization, C-Ext, C-Man, C-Opt, C-PEP, C-PEP-Info, CONNECT, Cache-Control, Compliance, Connection, Content-Base, Content-Disposition, Content-Encoding, Content-ID, Content-Language, Content-Length, Content-Location, Content-MD5, Content-Range, Content-Script-Type, Content-Security-Policy, Content-Style-Type, Content-Transfer-Encoding, Content-Type, Content-Version, Cookie, Cost, DAV, DELETE, DNT, DPR, Date, Default-Style, Delta-Base, Depth, Derived-From, Destination, Differential-ID, Digest, ETag, Expect, Expires, Ext, From, GET, GetProfile, HEAD, HTTP-date, Host, IM, If, If-Match, If-Modified-Since, If-None-Match, If-Range, If-Unmodified-Since, Keep-Alive, Label, Last-Event-ID, Last-Modified, Link, Location, Lock-Token, MIME-Version, Man, Max-Forwards, Media-Range, Message-ID, Meter, Negotiate, Non-Compliance, OPTION, OPTIONS, OWS, Opt, Optional, Ordering-Type, Origin, Overwrite, P3P, PEP, PICS-Label, POST, PUT, Pep-Info, Permanent, Position, Pragma, ProfileObject, Protocol, Protocol-Query, Protocol-Request, Proxy-Authenticate, Proxy-Authentication-Info, Proxy-Authorization, Proxy-Features, Proxy-Instruction, Public, RWS, Range, Referer, Refresh, Resolution-Hint, Resolver-Location, Retry-After, Safe, Sec-Websocket-Extensions, Sec-Websocket-Key, Sec-Websocket-Origin, Sec-Websocket-Protocol, Sec-Websocket-Version, Security-Scheme, Server, Set-Cookie, Set-Cookie2, SetProfile, SoapAction, Status, Status-URI, Strict-Transport-Security, SubOK, Subst, Surrogate-Capability, Surrogate-Control, TCN, TE, TRACE, Timeout, Title, Trailer, Transfer-Encoding, UA-Color, UA-Media, UA-Pixels, UA-Resolution, UA-Windowpixels, URI, Upgrade, User-Agent, Variant-Vary, Vary, Version, Via, Viewport-Width, WWW-Authenticate, Want-Digest, Warning, Width, X-Content-Duration, X-Content-Security-Policy, X-Content-Type-Options, X-CustomHeader, X-DNSPrefetch-Control, X-Forwarded-For, X-Forwarded-Port, X-Forwarded-Proto, X-Frame-Options, X-Modified, X-OTHER, X-PING, X-PINGOTHER, X-Powered-By, X-Requested-With`); this.serverResponse.statusCode = 200; } /** * Writes to the http response object * @param output what to write */ writeResponse(output) { const converter = MessageConverterScout .using(this.serverRequest, this.foundRoute) .findWriteConverter(output.body); this.serverResponse.statusCode = output.status; this.serverResponse.setHeader('Content-Type', converter.getContentType()); this.serverResponse.write(converter.doWrite(output.body)); } } exports.AsyncRequestHandler = AsyncRequestHandler; class MessageConverterScout { constructor(serverRequest, foundRoute) { this.serverRequest = serverRequest; this.foundRoute = foundRoute; } static using(serverRequest, foundRoute) { return new MessageConverterScout(serverRequest, foundRoute); } findReadConverter(body) { const type = this.findContentType(this.serverRequest.headers["content-type"], this.foundRoute.details.consumes); const converter = requisites_1.Requisites.findAll(converter_1.Converter).find((x) => x.canRead(body, type)); if (!converter) { throw new http_error_1.HttpError(standard_responses_1.UNSUPPORTED_MEDIA_TYPE); } return converter; } findWriteConverter(body) { const type = this.findContentType(this.serverRequest.headers.accept, this.foundRoute.details.produces); const converter = requisites_1.Requisites.findAll(converter_1.Converter).find((x) => x.canWrite(body, type)); if (!converter) { throw new http_error_1.HttpError(standard_responses_1.UNSUPPORTED_MEDIA_TYPE); } return converter; } findContentType(contentTypeHeader, allowedContentTypes) { const contentTypes = contentTypeHeader; let requiredContentTypes; if (contentTypes) { requiredContentTypes = contentTypes.split(',').map(x => x.trim()).map(x => x.split(';')[0]); } let type; if (!requiredContentTypes || contentTypes.includes('*/*')) { type = allowedContentTypes[0]; } else if (requiredContentTypes) { type = requiredContentTypes.find(ctype => allowedContentTypes.includes(ctype)); } return type; } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"async-request-handler.js","sourceRoot":"","sources":["../../src/routes/async-request-handler.ts"],"names":[],"mappings":"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"}