Middleware

Middleware

Using middleware

You can add middleware to a client using the client() function directly, or by calling with() on an existing client. This will return a new client with the middleware added:

import jsonmw from '@muze-nl/metro/src/mw/jsonmw'
const client = metro.client( jsonmw() )

With @muze-nl/metro, you get these middleware modules:

In addition there are a few separate middleware libraries:

Creating middleware

A middleware is a function with (request, next) as parameters, returning a response. Both request and response adhere to the Fetch API Request and Response standard.

next is a function that takes a request and returns a Promise<Response>. This function is defined by MetroJS and automatically passed to your middleware function. The idea is that your middleware function can change the request and pass it on to the next middleware or the actual fetch() call, then intercept the response and change that and return it:

async function myMiddleware(req,next) {
  req = req.with('?foo=bar')
  let res = await next(req)
  if (res.ok) {
    res = res.with({headers:{'X-Foo':'bar'}})
  }
  return res
}

/Note/: Both metro.request and metro.response have a with() function. This allows you to create a new request or response, from the existing one, with one or more options added or changed. The original request or response is not changed. See the reference for more information.

Differences to the browser Request and Response objects

The browsers Request and Response classes were never designed with middleware in mind. The consequence is that some things have been altered in metro.request() and metro.response(). Most notably the handling of Request.body and Response.body.

In a normal Request and Response, the moment that you set a body, the body is transformed into a ReadableStream. Streams can only be read asynchronously, and only once. This is clearly not suitable for a middleware function, where you would want to inspect the current body, and possibly alter it.

So in a metro.request or a metro.response, if you set a new body, the original body parameter that you passed to it, is stored as request.data or response.data. metro.request().body and metro.response().body are still ReadableStreams, and work like normal. But now if you pass a FormData or URLSearchParams object as a body, you can still access them in metro.request().data or metro.response().data.

To do this, the metro.request() and metro.response() functions return a Proxy to the Request and Response objects. However, fetch() (the default browser implementation) will not accept Proxies as parameter. So metro.client().fetch()unwraps the Proxies for you.

These Proxies also allow you to do specific metro things, like:

  let req = metro.request('https://example.com', new URLSearchParams('?foo=bar'))
  req = req.with({searchParams: {'foo2':'bar'}})
  console.log(req.data.get('foo2'))

See the issues for a list of issues with the normal Fetch API that metro fixes, or at least makes simpler.