Skip to content

Commit

Permalink
feat(rest): use @loopback/express-middleware to manage express middle…
Browse files Browse the repository at this point in the history
…ware
  • Loading branch information
raymondfeng committed Mar 14, 2019
1 parent f917549 commit 89c03be
Show file tree
Hide file tree
Showing 3 changed files with 62 additions and 23 deletions.
1 change: 1 addition & 0 deletions packages/rest/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@
"@loopback/http-server": "^1.1.8",
"@loopback/openapi-v3": "^1.3.0",
"@loopback/openapi-v3-types": "^1.0.8",
"@loopback/express-middleware": "^1.0.0-1",
"@types/body-parser": "^1.17.0",
"@types/cors": "^2.8.3",
"@types/express": "^4.11.1",
Expand Down
21 changes: 9 additions & 12 deletions packages/rest/src/keys.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,37 +3,34 @@
// This file is licensed under the MIT License.
// License text available at https://opensource.org/licenses/MIT

import {CoreBindings} from '@loopback/core';
import {BindingKey, Context} from '@loopback/context';

import {CoreBindings} from '@loopback/core';
import {HttpProtocol} from '@loopback/http-server';
import * as https from 'https';
/**
* See https://github.com/Microsoft/TypeScript/issues/26985
*/
// import {OpenApiSpec} from '@loopback/openapi-v3-types';
import {OpenAPIObject as OpenApiSpec} from 'openapi3-ts';

import {ErrorWriterOptions} from 'strong-error-handler';
import {BodyParser, RequestBodyParser} from './body-parsers';
import {HttpHandler} from './http-handler';
import {RestRouter} from './router';
import {SequenceHandler} from './sequence';
import {
BindElement,
FindRoute,
GetFromContext,
InvokeMethod,
LogError,
Request,
Response,
ParseParams,
Reject,
Send,
Request,
RequestBodyParserOptions,
Response,
Send,
} from './types';

import {HttpProtocol} from '@loopback/http-server';
import * as https from 'https';
import {ErrorWriterOptions} from 'strong-error-handler';
import {RestRouter, RestRouterOptions} from './router';
import {RequestBodyParser, BodyParser} from './body-parsers';

/**
* RestServer-specific bindings
*/
Expand Down
63 changes: 52 additions & 11 deletions packages/rest/src/rest.server.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,10 @@ import {
inject,
} from '@loopback/context';
import {Application, CoreBindings, Server} from '@loopback/core';
import {
ExpressBindings,
MiddlewareRegistry,
} from '@loopback/express-middleware';
import {HttpServer, HttpServerOptions} from '@loopback/http-server';
import {getControllerSpec} from '@loopback/openapi-v3';
import {
Expand Down Expand Up @@ -223,6 +227,19 @@ export class RestServer extends Context implements Server, HttpServerLike {
this._applyExpressSettings();
this._requestHandler = this._expressApp;

if (!this.isBound(ExpressBindings.EXPRESS_MIDDLEWARE_REGISTRY)) {
// Set up the default express middleware registry
this.bind(ExpressBindings.EXPRESS_MIDDLEWARE_REGISTRY).toClass(
MiddlewareRegistry,
);
}
const middlewareRegistry = this.getSync(
ExpressBindings.EXPRESS_MIDDLEWARE_REGISTRY,
);
middlewareRegistry.setMiddlewareRegistryOptions({
phasesByOrder: ['cors', 'openapi-spec', 'rest'],
});

// Allow CORS support for all endpoints so that users
// can test with online SwaggerUI instance
const corsOptions = this.config.cors || {
Expand All @@ -233,22 +250,36 @@ export class RestServer extends Context implements Server, HttpServerLike {
maxAge: 86400,
credentials: true,
};
this._expressApp.use(cors(corsOptions));

middlewareRegistry.middleware(cors(corsOptions), {
phase: 'cors',
name: 'cors',
});

// Set up endpoints for OpenAPI spec/ui
this._setupOpenApiSpecEndpoints();
this._setupOpenApiSpecEndpoints(middlewareRegistry);

// Mount our router & request handler
this._expressApp.use(this._basePath, (req, res, next) => {
this._handleHttpRequest(req, res).catch(next);
});
middlewareRegistry.middleware(
(req, res, next) => {
this._handleHttpRequest(req, res).catch(next);
},
{
path: this._basePath,
phase: 'rest',
name: 'rest',
},
);

// Mount our error handler
this._expressApp.use(
middlewareRegistry.errorMiddleware(
(err: Error, req: Request, res: Response, next: Function) => {
this._onUnhandledError(req, res, err);
},
{name: 'error'},
);

this._expressApp.use(middlewareRegistry.requestHandler);
}

/**
Expand All @@ -268,7 +299,7 @@ export class RestServer extends Context implements Server, HttpServerLike {
* Mount /openapi.json, /openapi.yaml for specs and /swagger-ui, /explorer
* to redirect to externally hosted API explorer
*/
protected _setupOpenApiSpecEndpoints() {
protected _setupOpenApiSpecEndpoints(middlewareRegistry: MiddlewareRegistry) {
if (this.config.openApiSpec!.disabled) return;
// NOTE(bajtos) Regular routes are handled through Sequence.
// IMO, this built-in endpoint should not run through a Sequence,
Expand All @@ -280,14 +311,24 @@ export class RestServer extends Context implements Server, HttpServerLike {
const mapping = this.config.openApiSpec!.endpointMapping!;
// Serving OpenAPI spec
for (const p in mapping) {
this._expressApp.get(p, (req, res) =>
this._serveOpenApiSpec(req, res, mapping[p]),
middlewareRegistry.middleware(
(req, res) => this._serveOpenApiSpec(req, res, mapping[p]),
{
path: p,
method: 'get',
phase: 'openapi-spec',
},
);
}

const explorerPaths = ['/swagger-ui', '/explorer'];
this._expressApp.get(explorerPaths, (req, res, next) =>
this._redirectToSwaggerUI(req, res, next),
middlewareRegistry.middleware(
(req, res, next) => this._redirectToSwaggerUI(req, res, next),
{
path: explorerPaths,
method: 'get',
phase: 'openapi-spec',
},
);
}

Expand Down

0 comments on commit 89c03be

Please sign in to comment.