Skip to content

A web request handling framework that plays nicely with TypeScript

License

Notifications You must be signed in to change notification settings

Deskbot/Quelaag

Repository files navigation

Quelaag

Named after the spider boss from Dark Souls. Typically Pronounced: (/ˈkweɪlæɡ/) (like a combination of "quail" and "lag")

Quelaag is a web request handling library for NodeJS designed to:

  • play well with TypeScript,
  • avoid applying side-effects to the request object,
  • manage dependencies between middleware easily,
  • make code paths explicit for straight-forward debugging.

Design Philosophy:

  • Doing things better is preferred over doing more things.
  • New features must pass trial by YAGNI.
  • Composability is preferred over configurability.
  • It is better to facilitate built-in libraries than to attempt their re-design.

The API is still subject to change.

Install

npm install --save quelaag

Example

import { quelaag, router } from "quelaag";
import { parse } from "cookie"; // a third party cookie header parsing library
import * as http from "http";

const r = router(quelaag({
    cookies(req): Record<string, string> {
        return parse(req.headers.cookie || "");
    },
    userId(req): number {
        return parseInt(this.cookies(req).userId);
    },
    async userIsAdministrator(req): Promise<boolean> {
        const user = await getUserFromDatabase(this.userId(req));
        return user.isAdmin;
    }
}));

r.addEndpoint({
    when: req => req.url === "/admin",
    do: async (req, res, middleware) => {
        if (await middleware.userIsAdministrator()) {
            res.statusCode = 200;
            res.end("Hello User " + middleware.userId());
        } else {
            res.statusCode = 403;
            res.end("403 Forbidden");
        }
    }
});

const server = http.createServer((req, res) => r.route(req, res));
server.listen(8080);

In Quelaag, middleware are functions that are given the request object and return a value (even Promises). Middleware have to be called explicitly. Middleware calls are memoised meaning that for a single request, each middleware function will compute its return value no more than once.

The quelaag function returns a function that generates memoised middleware tied to a specific request object.

router creates an object that chooses how to handle a request. router creates new middleware instances for you, which are passed as the last parameter to all of router's callbacks.

quelaag and router are designed to be flexible in how they can be used. You can use them with NodeJS's built-in libraries or a third-party framework. quelaag can be used without router entirely.

By default, the type of requests and responses are NodeJS's IncomingMessage and ServerResponse. However, these can be overridden with type arguments to quelaag and router. See the examples folder for details.

Middleware in Quelaag

The definition of middleware tends to be quite broad. In-between helper functions in web servers tend to be about either getting information from the request or about modifying the response.

In Quelaag, 'middleware' could be called 'request middleware'. Quelaag is designed to facilitate deriving data from a request. Quelaag is agnostic about how to do 'response middleware'.

Routing

Endpoints

A request will be handled by the first Endpoint with a matching condition. These are created using router.addEndpoint(...). Endpoints are the only place where the response object can be handled. Router and quelaag in no way affect the request or response object.

r.addEndpoint({
    when: req => req.url === "/hello/world",
    do: (req, res) => {
        res.end("hello world");
    }
});
r.addEndpoint({
    when: req => req.url.startsWith("/hello"),
    do: (req, res) => {
        res.end("hello");
    }
});

In this example, a request to "/hello/world" is matched by the condition of both endpoints, however only the first endpoint is called, sending "hello world" to the user.

The Fallback Endpoint

If no endpoint matches, a default can be used, if one has been set with router.setFallbackEndpoint(...);

r.setFallbackEndpoint((req, res) => {
    res.statusCode = 404;
    res.end("404 Not Found");
});

Spies

A request will be handled by all Spies with a matching condition. These are created using router.addSpy(...). The response object is not accessible here.

r.addSpy({
    when: req => req.url === "/hello",
    do: (req) => {
        console.log(req.socket.remoteAddress);
    }
});

Error handling

An error thrown or a promise rejected in a when or do can be caught with an optional catch function on any Endpoint or Spy.

A catch handler can also be given to Quelaag, as a fallback for when a local catch is not defined.

import { quelaag, router } from "quelaag";

const r = router(
    quelaag({}),
    (err) => {
        console.error(err);
    }
);
r.addEndpoint({
    when: req => req.url === "/local/handle",
    do: (req, res, middleware) => {
        throw "will only be caught by the handler given to this Endpoint";
    },
    catch: (err) => {
        console.error(err);
    },
});
r.addEndpoint({
    when: req => req.url === "/quelaag/handle",
    do: (req, res, middleware) => {
        throw "will only be caught by the handler given to Quelaag";
    }
});

Sub-Routing

This feature is designed to allow routing logic and middleware definitions to be split across multiple files.

// ./app.ts

import * as cookie from "cookie";
import { createServer, IncomingMessage } from "http";
import { router, quelaag } from "quelaag";
import { adminRouter } from "./admin";

export const root = router(quelaag({
    cookies(req) {
        return cookie.parse(req.headers.cookie || '');
    },
    userId(req): number {
        return parseInt(this.cookies(req).userId);
    },
    url(req: IncomingMessage): URL {
        return new URL(req.url!, "http://localhost:8080");
    },
}));

root.addSubRouter({
    when: (req, middleware) => middleware.url().pathname.startsWith("/admin"),
    router: () => adminRouter,
});

const server = createServer((req, res) => root.route(req, res));
server.listen(8080);

// ./admin.ts

import { subRouter } from "quelaag";
import { root } from "./app";

export const adminRouter = subRouter<typeof root>()(
    {
        url(req, superMiddleware) {
            return superMiddleware.url();
        },
        async userIsAdministrator(req, superMiddleware): Promise<boolean> {
            const user = await getUserFromDatabase(superMiddleware.userId());
            return user.isAdmin;
        }
    }
);

adminRouter.addEndpoint({
    when: (req, middleware) => middleware.url().pathname.startsWith("/admin"),
    do: async (req, res, middleware) => {
        if (await middleware.userIsAdministrator()) {
            res.end("Greetings planet.");
        } else {
            res.statusCode = 403;
            res.end("403 Forbidden");
        }
    }
});

Quelaag in Isolation

Demonstration of what quelaag does by using it in isolation.

import { quelaag } from "quelaag";

const makeMiddleware = quelaag({
    ip(req): string {
        console.log("ip");
        return req.socket.remoteAddress;
    },
    isMe(req): boolean {
        return this.ip(req) === "127.0.0.1";
    },
});

let req1, req2; // Imagine you got some request objects from somewhere.

const request1 =      makeMiddleware(req1, undefined); // ignore the use of undefined here for now
const request2 =      makeMiddleware(req2, undefined);
const request1Again = makeMiddleware(req1, undefined);

request1.ip();   // prints "ip", returns "127.0.0.1"
request1.ip();   //              returns "127.0.0.1"
request1.isMe(); //              returns true

request2.isMe(); // prints "ip", returns true
request2.ip();   //              returns "127.0.0.1"
request2.ip();   //              returns "127.0.0.1"

request1Again.ip(); // prints "ip", returns "127.0.0.1"

Notice that a method in the middleware specification given to quelaag takes the request object, but the memoised version called later does not.

In order for middleware to call each other, the function can't be defined with arrow syntax, so that this inside the function refers to the middleware object. The noImplicitThis option in your tsconfig needs to be enabled for the type checking on this to be correct.

Sub-Quelaag in Isolation

A demonstration of how Quelaags definitions can be merged.

import { IncomingMessage } from "node:http";
import { quelaag, subquelaag } from "quelaag";

const makeParent = quelaag({
    ip(req: IncomingMessage): string {
        console.log("ip");
        return req.socket.remoteAddress;
    },
});

const makeMiddleware = subquelaag<typeof makeParent>()(
    {
        ip(req, parentMiddleware): string {
            return parentMiddleware.ip();
        },
        isMe(req, parentMiddleware): boolean {
            return parentMiddleware.ip() === "127.0.0.1";
        }
    }
);

let req1: IncomingMessage, req2: IncomingMessage; // Imagine you got some request objects from somewhere.

const request1 =      makeMiddleware(req1, makeParent(req1, undefined));
const request2 =      makeMiddleware(req2, makeParent(req2, undefined));
const request1Again = makeMiddleware(req1, makeParent(req1, undefined));

request1.ip();   // prints "ip", returns "127.0.0.1"
request1.ip();   //              returns "127.0.0.1"
request1.isMe(); //              returns true

request2.isMe(); // prints "ip", returns true
request2.ip();   //              returns "127.0.0.1"
request2.ip();   //              returns "127.0.0.1"

request1Again.ip(); // prints "ip", returns "127.0.0.1"

In addition to the "request", a second parameter called the "context" can be passed to a middleware function. The main purpose of "context" is to allow middleware functions to use Quelaags defined and initialised elsewhere. By default this is undefined and its type will be inferred from the type given in the middleware functions you define. The context can actually be any type allowing you to use multiple parent Quelaags if you wish. The methods on the parent(s) are not inherited by the child Quelaag any methods you want to inherit have to be defined explicitly. This is the easiest way to manage dependencies between sub-quelaags and sub-routers.

When using router and subrouter, the context is handled for you.

Gotchas

No Implicit This

To get the a greater benefit from TypeScript's type inference, you should enable noImplicitThis in your tsconfig.json. It causes this in your middleware specification to be correctly typed instead of treated as any.

Circular Type Inference

When defining middleware, the type of this is defined by the methods in the object, and the type of those methods can be affected by the type of this. [x]

The circular type inference can cause scenarios where code compiles when it shouldn't or doesn't compile when it should.

This is solved by giving all methods a return type.

quelaag({
    number(req) {
        return 100;
    },

    // no return type
    isEven1(req) {
        this.number(req) % 2 == 0;        // compiles
        return this.number(req) % 2 == 0; // error
    },

    // return type is given
    isEven2(req): boolean {
        this.number(req) % 2 == 0;        // compiles
        return this.number(req) % 2 == 0; // compiles
    },
});

Partial Application of Type Arguments

In TypeScript, when a function/type takes type arguments, you can either:

  • specify none of them. The compiler will try to infer the type arguments.
  • specify all of them. The compiler will use what you gave.
  • specify all of them except some ones that have a default (e.g. A extends Animal = Cat). The compiler won't infer the missing type arguments; it will use the defaults.

If you want to specify some type arguments in a call to quelaag or router, but they also require you to give the type of the middleware specification object, which is likely very big and you would rather it be inferred than type it out separately.

To make matters more complicated: These functions have a default type for the middleware specification due to not allowing mandatory types after optional ones, but the default is super general. So the type will be too broad and the compiler will accept it.

The best solution is to assign the complicated object to a variable and use typeof to use its type as a type argument to the function call. I think this is easier than currying the type arguments (i.e. defining nested functions each of which takes a single type argument).

In the future, there may be a way to tell TypeScript which types to infer.

Tip Jar

ko-fi

About

A web request handling framework that plays nicely with TypeScript

Topics

Resources

License

Stars

Watchers

Forks

Packages

No packages published