Endpoint middleware

Simple use case

Middleware for an endpoint lets you manage request and response directly on a method controller. In most case you will create a middleware to do something on request or response like that:

import {IMiddleware, Middleware, Request, ServerSettingsService} from "@tsed/common";
import {NotAcceptable} from "ts-httpexceptions";

@Middleware()
export class AcceptMimesMiddleware implements IMiddleware {
   
   constructor(private serverSettingsService: ServerSettingsService) {

   }

   use(@Request() request) {

        this.serverSettingsService.acceptMimes
            .forEach((mime) => {
                if (!request.accepts(mime)) {
                    throw new NotAcceptable(mime);
                }
            });
   }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

This middleware is the same as the GlobalAcceptMimesMiddleware.

Then, add your middleware on your endpoint controller's:

import {Controller, Get} from "@tsed/common";

@Controller('/test')
@UseBefore(AcceptMimesMiddleware) // global to the controller
class MyCtrl {
   @Get('/')
   @UseBefore(AcceptMimesMiddleware) // only to this endpoint
   getContent() {}
}     
1
2
3
4
5
6
7
8
9

See middleware call sequence for more information.

Create your own decorator

The previous example work fine in most case, but it doesn't allow us to configure specific parameters for an endpoint. To do that, we need to use the Endpoint API, @EndpointInfo() and create our custom decorator.

1) Create your decorator

To do that we'll proposed a new decorator that take the configuration and wrap the @UseBefore decorator as following:

// decorators/accept-mimes.ts
import {Store, UseBefore} from "@tsed/common";
import AcceptMimesMiddleware from "../middlewares/accept-mimes";

export function AcceptMimes(...mimes: string[]) {
    return <T> (target: any, targetKey: string, descriptor: TypedPropertyDescriptor<T>): TypedPropertyDescriptor<T> => {

        Store
            .from(target, targetKey, descriptor)
            .set(AcceptMimesMiddleware, mimes);

        return UseBefore(AcceptMimesMiddleware)(target, targetKey, descriptor);
    };
}

// OR Shorter

export function AcceptMimes(...mimes: string[]) {
    return Store.decorate((store) => {
        store.set(AcceptMimesMiddleware, mimes);
        return UseBefore(AcceptMimesMiddleware)
    });
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23

2) Create your middleware

// middlewares/accept-mimes.ts
import {IMiddleware, Middleware, EndpointInfo, Endpoint, Request} from "@tsed/common";
import {NotAcceptable} from "ts-httpexceptions";

@Middleware()
export default class AcceptMimesMiddleware implements IMiddleware {
   use(@Request() request, @EndpointInfo() endpoint: Endpoint) {
       
        // get the parameters stored for the current endpoint or on the controller.
        const mimes = endpoint.get(AcceptMimesMiddleware) || [];

        mimes.forEach((mime) => {
            if (!request.accepts(mime)) {
                throw new NotAcceptable(mime);
            }
        });
   }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18

3) Use the decorator on you method Controller

import {Controller, Get} from "@tsed/common";
import {AcceptMimes} from "../decorators/accept-mimes";

@Controller('/test')
class MyCtrl {
   @Get('/')
   @AcceptMimes('application/json')
   getContent() {}
}     
1
2
3
4
5
6
7
8
9

Now our middleware is configurable via decorator !

With this method you can create your own decorators. You can propose your own decorators and middlewares via Pull Request or via an new issue.

How to format the Response with middleware

In next part, we see how we can create a decorator and middleware to check the mime of the request. In this part, we'll see how we can use a decorator and middleware to change the response format. To do that, we take as example the ResponseViewMiddleware implemented in Ts.ED.

Here the code of the middleware:

// middlewares/response-view.ts

import {IMiddleware, Middleware, ResponseData, Response, EndpointInfo, Endpoint} from "@tsed/common";
import {InternalServerError} from "ts-httpexceptions";

@Middleware()
export default class ResponseViewMiddleware implements IMiddleware {

    public use(
        @ResponseData() data: any, // handle the response data sent by the previous middleware
        @EndpointInfo() endpoint: Endpoint,    
        @Response() response: Express.Response
    ) {
        return new Promise((resolve, reject) => {

            const {viewPath, viewOptions} = endpoint.get(ResponseViewMiddleware);

            if (viewPath !== undefined) {

                if (viewOptions !== undefined ) {
                    data = Object.assign({}, data, viewOptions);
                }

                response.render(viewPath, data, (err, html) => {

                    if (err) {
                        reject(new InternalServerError("Error on your template =>" + err));

                    } else {
                        resolve(html);
                    }

                });
            } else {
                resolve();
            }
        });

    }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40

And his decorator:

// middlewares/response-view.ts
import {Endpoint, Store, UseAfter} from "@tsed/common";
import ResponseViewMiddleware from "../middlewares/response-view";

export function ResponseView(viewPath: string, viewOptions?: Object): Function {

    return (target: Function, targetKey: string, descriptor: TypedPropertyDescriptor<any>): TypedPropertyDescriptor<any> => {
        
        Store
           .from(target, targetKey, descriptor)
           .set(ResponseViewMiddleware, {viewPath, viewOptions});

        // Wrap the UserAfter decorator to push the middleware after the endpoint execution
        return UseAfter(ResponseViewMiddleware)(target, targetKey, descriptor);
    };
    
    // or shorter
    return Store.decorate((store) => {
        store.set(ResponseViewMiddleware, {viewPath, viewOptions});
        // Wrap the UserAfter decorator to push the middleware after the endpoint execution
        return UseAfter(ResponseViewMiddleware)
    });
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23

Note: Store is recommended if your decorator can be used in different context.

Finally, we can use this decorator on a controller:

import {Controller, Get} from "@tsed/common";
import {ResponseView} from "../decorators/response-view";

@Controller('/test')
class MyCtrl {
   @Get('/')
   @ResponseView('path/to/page.html')
   getContentHTML() {
      return {data: ... } // will stored to the responseData. Use @ResponseData to retrieve the stored data.
   }
}  
1
2
3
4
5
6
7
8
9
10
11