This TypeScript library provides an Express/Ware-style middleware that:
- is safely typed, and
- supports async / Promises
# run this from your Terminal
npm install @ganbarodigital/ts-lib-middleware
// add this import to your Typescript code
import { MiddlewareStack } from "@ganbarodigital/ts-lib-middleware/lib/v2"
VS Code users: once you've added a single import anywhere in your project, you'll then be able to auto-import anything else that this library exports.
Middleware is simply a list of functions that call each other in a chain:
- the first function calls the second function
- the second function calls the third function
... and so on. This continues until either:
- one of the functions returns a value, or
- one of the functions throws an exception
If none of the functions returns a value, the code falls off the end of the chain, and an exception is thrown (MiddlewareReturnedNoValueError
in our case).
The first thing to note about middleware is that each of these functions is indepedent of each other. They don't have a hard-coded call to the next function in the chain. We decide what the next function is when we put the chain together.
import { OnError, THROW_THE_ERROR } from "@safelytyped/core-types";
import {
DEFAULT_MIDDLEWARE_OPTIONS,
Middleware
} from "@ganbarodigital/ts-lib-middleware/lib/v2";
/**
* an example middleware function
*/
function RejectNegatives(
input: number,
next: Middleware<number, number>,
options: MiddlewareOptions = DEFAULT_MIDDLEWARE_OPTIONS
) {
if (input < 0 ) {
throw new Error("Negative numbers are not allowed");
}
// this function doesn't know what the next function is
// when we write it
//
// it is told what the next function is when we run it
return next(input, next, onError);
}
Middleware functions can be written and published independently. They can be reused in any combination to suit our needs.
Another thing to note about middleware is that each function is free to modify the input value that's being passed along the chain. When we run a middleware chain, we pass an input parameter from our code into the first function in the chain. From that point, it's up to each function to decide whether or not to modify the input data before passing it on to the next function in the chain.
import { OnError, THROW_THE_ERROR } from "@safelytyped/core-types";
import {
DEFAULT_MIDDLEWARE_OPTIONS,
Middleware
} from "@ganbarodigital/ts-lib-middleware/lib/v2";
/**
* an example middleware function
*/
function IntegerNumbersOnly(
input: number,
next: Middleware<number, number>,
options: MiddlewareOptions = DEFAULT_MIDDLEWARE_OPTIONS
) {
// in this example, we (possibly!) change the value of input
// before calling the next function in the chain
input = Math.floor(input);
return next(input, next, onError);
}
The final thing to note about middleware is that each function is free to modify the return value that it gets from the next function in the chain. Not only can each function in the chain do some processing before calling the next function in the chain, it can also do some processing after it has received a return value from the next function.
import { OnError, THROW_THE_ERROR } from "@safelytyped/core-types";
import {
DEFAULT_MIDDLEWARE_OPTIONS,
Middleware
} from "@ganbarodigital/ts-lib-middleware/lib/v2";
/**
* an example middleware function
*/
function IntegerNumbersOnly(
input: number,
next: Middleware<number, number>,
options: MiddlewareOptions = DEFAULT_MIDDLEWARE_OPTIONS
) {
input = Math.floor(input);
// in this example, we make sure that any returned value is also an
// integer
return Math.floor(
next(input, next, onError)
);
}
When we combine these properties, we can do an awful lot with middleware.
Each middleware function ultimately does one (or more!) of the following:
- they validate input data (and reject invalid inputs)
- they filter input data (allowing some parts of the input through, removing others)
- they augment input data (they add additional parts to the input)
- they transform input data (they convert data from one form to another)
- they validate the return value (and reject invalid return values)
- they filter the return value (allowing some parts of the return value through, removing others)
- they augment the return value (they add additional parts to the return value)
- they transform the return value (they convert data from one form to another)
- they process the input data, and produce a return value
As long as each middleware function is doing a sensible unit of work, middleware gives us a highly-reusable toolkit to work with. Think of them like Lego bricks: individual parts that you can assemble to create all sorts of things.
In the JavaScript world, two great examples of middleware are Express.JS and MetalSmith.
Functional programming has this concept called composition, where the output of one function is used as the input to the next function. It's such a fundamental part of functional programming that functional languages include compiler/interpreter support for building new functions using composition.
With middleware, our functions aren't composed in the functional sense:
- we don't use the output of one function as the input to the next
- there's no compiler/interpreter support for this, so we have to emulate the behaviour using a piece of controlling JavaScript / TypeScript that we call the MiddlewareStack
- while the output of one piece of middleware does become the input of the next piece of middleware, all the middleware in the same MiddlewareStack must have the same function signature.
There are some things that our middleware has in common with functional programming:
- your middleware functions are more reusable if they only use the data that's passed in as an input parameter (ie a "pure" function in functional programming terms)
- any app that uses middleware is more reliable if the middleware functions only accept and process immutable data structures
Ware from Segment is the middleware stack that powers Metalsmith. It's available as a standalone package. It's an established and mature package.
There's several important differences between Ware and our Middleware module:
- Safe types! - It should be impossible to build a MiddlewareStack that has incompatible middleware functions in it.
- Immutable MiddlewareStack! - The only way to build a MiddlewareStack is via the constructor.
- Explicit choice of synchronous or asynchronous code! - if a synchronous middleware stack makes more sense for your code, we provide one, and you don't have the added complication of providing a
done()
callback if you're working fully-synchronously. - Our
MiddlewareStack.run()
statement returns whatever the Middleware returns, making it easier to incorporate into larger applications.
import { MiddlewareOptions } from "@ganbarodigital/ts-lib-middleware/v2";
/**
* `AsyncMiddleware` is the function-signature for an individual
* piece of Middleware.
*
* @params input
* this is the data to be passed into this piece of middleware
* @params next
* this is the next piece of middleware in the stack
* @params options.onError
* we call this if a problem occurs
* @returns
* a Promise that will resolve to the final return value of the stack
*
* @template I
* the type (normally an interface) of data that this piece of middleware
* accepts
* @template O
* the type (normally an interface) of data that this piece of middleware
* returns
*/
export type AsyncMiddleware<I, O> = (
input: I,
next: AsyncMiddleware<I, O>,
options?: MiddlewareOptions,
) => Promise<O>;
`AsyncMiddleware` is a _function signature_. Use this to define the type of function that your `AsyncMiddlewareStack` will accept.
For example:
```typescript
export type PrefetchAction = AsyncMiddleware<URL, void>;
// how to import into your own code
import {
AsyncMiddlewareStack,
} from "@ganbarodigital/ts-lib-middleware/lib/v2;
// types used for parameters, return types and errors
import {
AsyncMiddleware,
MiddlewareOptions,
MiddlewareReturnedNoValueError
} from "@ganbarodigital/ts-lib-middleware/lib/v2;
import { OnError, THROW_THE_ERROR } from "@safelytyped/core-types";
/**
* `AsyncMiddlewareStack` is a collection of middleware to be
* executed asynchronously.
*
* @template I
* the input type that the middleware must accept
* @template O
* the return type that the middleware must provide after the Promise
* is resolved
*/
export class AsyncMiddlewareStack<I, O> {
/**
* `constructor()` builds a new AsyncMiddlewareStack.
*
* Use it to add one or more pieces of middleware to this
* AsyncMiddlewareStack. The middleware is executed in the order
* that you add it to the stack.
*
* @param name
* The human-readable name of this AsyncMiddlewareStack. We use this
* in errors thrown by the AsyncMiddlewareStack class.
* @param fns
* The list of middleware to put in the AsyncMiddlewareStack.
*/
public constructor(name: string, ...fns: AsyncMiddleware<I, O>[]);
/**
* `run()` executes the middleware that's on the stack, and returns
* the result.
*
* We execute the middleware in the order that it was added to this
* stack. (IE first item added is the first item we run).
*
* Each piece of middleware either:
*
* - returns a return value of its own, or
* - throws an error, or
* - passes the (probably modified) input on to the next piece of
* middleware in the stack
*
* @param input
* The value to pass into the first function on your MiddlewareStack.
* @param options.onError
* We will call this if something goes wrong.
*/
public async run(
input: I,
{
onError = THROW_THE_ERROR,
}: Partial<MiddlewareOptions> = {},
): Promise<O>;
/**
* `getName()` returns the human-readable name of this
* AsyncMiddlewareStack.
*/
public getName(): string;
/**
* `getStack()` returns the list of middleware items on the stack.
*/
public getStack(): Array<AsyncMiddleware<I, O>>;
}
AsyncMiddlewareStack
is an immutable value type. It holds and asynchronously executes a list of functions, known as middleware.
import { OnError } from "@safelytyped/core-types";
/**
* `Middleware` is a function signature. It describes a single piece of
* Middleware.
*
* @params input
* this is the data to be passed into this piece of middleware
* @params next
* this is the next piece of middleware in the stack
* @params options.onError
* we call this if a problem occurs
* @returns
* the final return value of the stack
*
* @template I
* the type (normally an interface) of data that this piece of middleware
* accepts
* @template O
* the type (normally an interface) of data that this piece of middleware
* returns
*/
export type Middleware<I, O> = (
input: I,
next: Middleware<I, O>,
options?: MiddlewareOptions
) => O;
Middleware
is a function signature. Use this to define the type of function that your MiddlewareStack
will accept.
For example:
export type PrefetchAction = Middleware<URL, void>;
// how to import into your own code
import {
MiddlewareStack,
} from "@ganbarodigital/ts-lib-middleware/lib/v2;
// types used for parameters, return types and errors
import { OnError, THROW_THE_ERROR } from "@safelytyped/core-types";
import {
Middleware,
MiddlewareOptions,
MiddlewareReturnedNoValueError
} from "@ganbarodigital/ts-lib-middleware/lib/v2;
/**
* `MiddlewareStack` is a collection of middleware to be executed
* synchronously.
*
* @template I
* the input type that the middleware must accept
* @template O
* the return type that the middleware must return
*/
export class MiddlewareStack<I, O> {
/**
* `constructor()` builds a new MiddlewareStack.
*
* Use it to add one or more pieces of middleware to this MiddlewareStack.
* The middleware is executed in the order that you add it to the stack.
*
* @param name
* The human-readable name of this MiddlewareStack. We use this in
* errors thrown by the MiddlewareStack class.
* @param fns
* The list of middleware to put in the MiddlewareStack.
*/
public constructor(name: string, ...fns: Middleware<I, O>[]);
/**
* `run()` executes the middleware that's on the stack, and returns
* the result.
*
* We execute the middleware in the order that it was added to this
* stack. (IE first item added is the first item we run).
*
* Each piece of middleware either:
*
* - returns a return value of its own, or
* - throws an error, or
* - passes the (probably modified) input on to the next piece of
* middleware in the stack
*
* @param input
* The value to pass into the first function on your MiddlewareStack.
* @param options.onError
* We will call this if something goes wrong.
*/
public run(input: I, options: MiddlewareOptions = DEFAULT_MIDDLEWARE_OPTIONS): O;
/**
* `getName()` returns the human-readable name of this MiddlewareStack.
*/
public getName(): string;
/**
* `getStack()` returns the list of middleware.
*/
public getStack(): Array<Middleware<I, O>>;
}
MiddlewareStack
is an immutable value type. It holds and executes a list of functions, known as middleware.
import {
AppError,
AppErrorData,
} from "@safelytyped/core-types";
export interface MiddlewareReturnedNoValueData extends ExtraLogsOnlyData {
logsOnly: {
middlewareName: string;
};
}
export class MiddlewareReturnedNoValueError extends AppError<MiddlewareReturnedNoValueData> {
public constructor(params: MiddlewareReturnedNoValueData & AppErrorData);
}
MiddlewareReturnedNoValueError
is a throw
able JavaScript Error
. It is thrown when we've run out of Middleware
to execute in a MiddlewareStack
or AsyncMiddlewareStack
.
The fix? Make sure that the last piece of Middleware
returns a value.
Use npm run clean
to delete all of the compiled code.
Use npm run build
to compile the Typescript into plain Javascript. The compiled code is placed into the lib/
folder.
npm run build
does not compile the unit test code.
Use npm run test
to compile and run the unit tests. The compiled code is placed into the lib/
folder.
Use npm run cover
to compile the unit tests, run them, and see code coverage metrics.
Metrics are written to the terminal, and are also published as HTML into the coverage/
folder.