Huge News!Announcing our $40M Series B led by Abstract Ventures.Learn More
Socket
Sign inDemoInstall
Socket

@priestine/routing

Package Overview
Dependencies
Maintainers
1
Versions
14
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

@priestine/routing

Simple, declarative and dependency-free routing for Node.js

  • 2.0.0
  • Source
  • npm
  • Socket score

Version published
Weekly downloads
3
Maintainers
1
Weekly downloads
 
Created
Source

@priestine/routing

pipeline codecov licence: MIT docs: typedoc code style: prettier versioning: semantics npm npm bundlephobia-min bundlephobia-minzip

@priestine/routing brings simple and eloquent routing to Node.js. It currently only works with Node.js http server yet new releases aim to support other APIs.

TL;DR

import { withHttpRouter, HttpRouter } from '@priestine/routing';
import { createServer } from 'http';

/**
 * Create an empty router
 */
const router = HttpRouter.empty();

/**
* Define a set of functions
*/

// Curried for creating header builders
// `ctx` has { request: IncomingMessage, response: ServerResponse, intermediate: {/* Your data */}, error? } 
const setHeader = (name) => (value) => (ctx) => {
  ctx.response.setHeader(name, value);
};

// Specifically, a Content-Type header builder (we could make it in one go tho)
const setContentTypeHeader = setHeader('Content-Type');

// Anything to be passed to the next middleware should be put to ctx.intermediate
// to keep request and response Node.js-ly pure
const buildHello = (ctx) => {
    ctx.intermediate.helloWorld = {
      id: 'hello-world',
      message: 'Hello World!',
    }
};

// You don't need to actually return the ctx. It's done for you automatically
// In case you run async code, you must create and return a Promise that will resolve when necessary
const sayHello = (ctx) => ctx.response.end(JSON.stringify(ctx.intermediate.helloWorld));

/**
 * Register them in the router
 */
router
  .afterEach([
    buildHello,
    sayHello,
  ])
  .get('/', [
    setContentTypeHeader('application/json'),
  ])
  .get(/^\/(\d+)\/?$/, [
    setContentTypeHeader('text/html'),   
  ])
;

/**
 * Assign the router to serve Node.js HTTP server.
 */
createServer(withHttpRouter(router)).listen(3000);

// Go get http://localhost:3000 or http://localhost:3000/123123123

More

Installation

npm i --save @priestine/routing

or

yarn add @priestine/routing

Routing consists of a few components one should grasp to build routing efficiently:

The Router Thing

  1. HttpRouter is a kind of fluent interface for registering new routes:
  • router.register accepts three required arguments:

    • url - a string or a RegExp describing the URL pathname IncomingMessage url must match
    • methods - an array of HTTP methods that IncomingMessage method must be in
    • middleware - an array of IMiddlewareLike or an HttpPipeline that will be processed if current IncomingMessage matches given url and methods
    const router = HttpRouter.empty();
    router
      .register('/', ['POST', 'PUT'], [
        (ctx) => ctx.response.setHeader('Content-Type', 'application/json'),
        (ctx) => ctx.response.end('{ "success": true }'),
      ])
      .register('/1', ['GET'], MyCustomHttpPipeline)
    ;
    
  • router.get (or one of get, post, put, patch, delete, options, head) registers a route with the same method and accepts only url and an HttpPipeline or array of middleware

    const router = HttpRouter.empty();
    router.get('/', [
      (ctx) => ctx.response.setHeader('Content-Type', 'application/json'),
      (ctx) => ctx.response.end('{ "success": true }'),
    ]);
    
  • router.beforeEach and router.afterEach accept array of middleware or a pipeline that will be executed before/after each registered pipeline

    const runBeforeEach = [
      SetContentTypeHeader('application/json'),
      ExtractQueryParams,
      ExtractRequestParams,
    ];
    
    const runAfterEach = [
      FlushHead,
      CreateJSONResponseBody,
      EndResponse,
    ];
    
    HttpRouter
      .empty()
      .beforeEach(runBeforeEach)
      .afterEach(runAfterEach)
      .get('/', [
        SomeLogic,
      ])
    ;
    
  • router.concat allows concatenating multiple routers into one main router to be used app-wide. This is done to allow building modular routing with separated logic and then merging them to have single route map.

    const MainRouter = HttpRouter.empty();
    const ApiRouter = HttpRouter.empty();
    
    MainRouter
      .get('/', [
        SetContentTypeHeader('text/html'),
        GetTemplate('./templates/index.html'),
        EndResponse,
      ])
    ;
    
    ApiRouter
      .post('/api/v1/user-actions/subscribe', [
        SetContentTypeHeader('application/json'),
        ParseRequestBody,
        JSONParseLens(['intermediate', 'requestBody']),
        UpsertUserSubscriptionStatus,
        EndJSONResponse,
      ])
    ;
    
    MainRouter.concat(ApiRouter);
    

Middleware

Middleware is a reusable piece of business logic that encapsulates one specific step. Middleware in @priestine/routing can be function- or class-based. Each middleware is provided with ctx argument:

ctx = {
  request: IncomingMessage,
  response: ServerResponse,
  intermediate: { route: { url: string | RegExp, method: string } } & TIntermediate,
  error: Error | undefined,
}

If middleware is deterministic (meaning that it returns the context object or Promise resolving into the context object), the context object will be overriden by given value if it successfully validates through isMiddlewareContext guard. This is done for two reasons:

  1. Custom transformations for the context that entirely change it
  2. Easier testing

Having said that, it is entirely optional and you can omit returning the context, thus the argument context will be passed to the next piece of middleware automatically.

For passing any computed data to next middleware, you should assign it to ctx.intermediate that serves a purpose of transferring data along the pipeline.

Function Middleware

Function middleware is a fancy name for a function that accepts ctx.

// Synchronous function middleware
const MyMiddleware = (ctx) => {
  ctx.intermediate.id = 1;
}

// Asynchronous function middleware
const MyAsyncMiddleware = (ctx) => {
  return new Promise((resolve) => setTimeout(() => { resolve(ctx); }, 200));
}

router.get('/', [
  MyAsyncMiddleware,
  MyMiddleware,
]);
Class-based Middleware

Class-based middleware must implement IMiddleware interface (it must have a $process) method that accepts ctx.

NOTE: When registering middleware in the Router, you must provide an instance of a class-based middleware.

class SetContentTypeHeader {
  static applicationJson() {
    return new SetContentTypeHeader('application/json');
  }
  
  constructor(value) {
    this.value = value;
  }
  
  $process(ctx) {
    ctx.response.setHeader('Content-Type', this.value);
  }
}

router.get('/', [
  SetContentTypeHeader.applicationJson(),
])
Parallel vs waterfall (Lazy vs Eager)

Asynchronous middleware in the pipeline can be executed either in parallel or sequentially. Each middleware can dictate which type of execution must be applied to it:

  • parallel execution does not block the pipeline and allows next middleware start processing even if the promise of current middleware has not been resolved. In this case, the Promise containing the result of current middleware computation can be directly assigned to the ctx.intermediate key and awaited later where necessary. This approach allows doing extra checks simultaneously and exit the pipeline if something is not right. This can be referred to as Lazy execution. Example:
/**
 *  `GetCurrentPost` doesn't block the next piece of middleware that can trigger exiting the pipeline
 * if user is not authenticated thus not allowed to see posts in this imaginary scenario. This allows writing middleware
 * in arbitrary order in some cases.
 */
const GetCurrentPost = (ctx) => {
  ctx.intermediate.post = db.collection('posts').findOne({ id: ctx.intermediate.params.id });
};

const GetPostComments = async (ctx) => {
  ctx.intermediate.body = (await ctx.intermediate.post).comments;
};

HttpRouter.empty()
  .get(/^\/posts\/(?<id>(\w+))\/comments/, [
    GetCurrentPost,
    IsAuthenticated,
    GetPostComments,
  ])
;
  • waterfall execution blocks the pipeline until current middleware is done. This is convenient in cases where further execution of the pipeline heavily relies on the result of computation. To inform the pipeline that it needs to wait for current middleware to resolve, you need to return a Promise inside the middleware. This can be referred to as Eager execution. Example:
/**
 * `IsAuthorized` blocks the middleware until the `ctx` is resolved. Thus, the Promise is rejected,
 * the pipeline will be exited and no further computation will be executed, being replaced with emitting a
 * `pipelineError` event.
 */
const IsAuthorized = (ctx) => new Promise((resolve, reject) => {
  db.collection('users').findOne({ _id: ctx.intermediate.userId })
    .then((u) => {
      if (u.roles.includes('admin')) {
        ctx.intermediate.userAuhorized = true;
        resolve(); // To make this middleware deterministic, use resolve(ctx)
        return;
      }
      
      reject(new UnauthorizedError());
    })
  ;
});

HttpRouter.empty()
  .get(/^\/posts\/(?<id>(\w+))\/comments/, [
    IsAuthorized,
    GetCurrentPost,
    GetPostComments,
  ])
;
Generic Context

@priestine/routing is written in TypeScript and provides generic context interfaces for describing types of ctx.intermediate:

import { IHttpContext } from '@priestine/routing';

interface IUserAware {
  user: {
    _id: string;
    name: string;
  };
}

export const GetUser = (ctx: IHttpContext<IUserAware>) => {
  ctx.intermediate.user = {
    _id: '123123123123',
    name: 'Test User',
  };
};

Assigning router to listen for connections

The router itself cannot listen for IncomingMessage's and to make it work you need to wrap it into a helper withHttpRouter and pass it to http.createServer as an argument:

import { createServer } from 'http';
import { HttpRouter, withHttpRouter } from '@priestine/routing';

const router = HttpRouter.empty()
  .get('/', [
    (ctx) => ctx.response.end('hi'),
  ])
;

createServer(withHttpRouter(router)).listen(3000);

Error handling

Error handling in @priestine/routing provides a statically available event emitter that you can use to trigger and/or subscribe to events. Internally, there is only one base event that is emitted by @priestine/routing which is called pipelineError. This event is emitted if incoming request does not match any of registered routes:

HttpRouter.eventEmitter.on('pipelineError', (ctx) => HttpPipeline.of([
  (ctx) => ctx.response.setHeader('Content-Type', 'application/json'),
  (ctx) => ctx.intermediate.body = {
    success: false,
    message: ctx.intermediate.error.message,
  },
  (ctx) => ctx.response.end(JSON.stringify(ctx.intermediate.body)),
]).$process(ctx));

If something goes wrong, the middleware pipeline is aborted and a new pipeline assigned with event listener is processed using the ctx of the pipeline that broke down. You can debug the nature of the error by checking the contents of the ctx.intermediate as it will have all the amendments up to the moment where it crashed.

To force quitting current pipeline, you can either throw (synchronous middleware) or reject() (asynchronous middleware).

Keywords

FAQs

Package last updated on 18 Dec 2018

Did you know?

Socket

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Install

Related posts

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc