1.12.0 • Published 5 years ago

redis-request-broker v1.12.0

Weekly downloads
30
License
MIT
Repository
github
Last release
5 years ago

redis-request-broker

Build Status

A request broker and message broker based on the redis PUB / SUB system. This package can be used for the communication of distributed systems when one service needs to request data from another.

Clients will send requests to only one worker. If no worker is currently idle, the request will be queued and picked up as soon as one becomes available. Workers always handle only one request at a time.

Publishers send messages to all susbcribers that are currently listening.

Goals

  • Simple to use
  • Thread safe
  • Allow multiple parallel workers and clients
  • Limited scope
  • Transparent usage

Example

Client and Worker

const { Worker, Client } = require('redis-request-broker');

async function doWork(data) {
    console.log('Working on', data);
    return data + 1;
}

const w = new Worker('myqueue', doWork);
const c = new Client('myqueue');

async function start() {
    await w.listen();
    await c.connect();

    const result = await c.request(42);
    console.log('result', r);
    
    await w.stop();
    await c.disconnect();
}

start();

Publisher and Subscribers

const { Publisher, Subscriber } = require('redis-request-broker');

function handle(message) {
    console.log('Received message', data);
}

const s1 = new Subscriber('myqueue', handle);
const s2 = new Subscriber('myqueue', handle);
const p = new Publisher('myqueue');

async function start() {
    await s1.listen();
    await s2.listen();
    await p.connect();

    await p.publish(42);
    
    await s1.stop();
    await s2.stop();
    await p.disconnect();
}

start();

Configuration

You can configure each client, worker, publisher and subscriber itself or set process wide defaults:

const { Worker, Client, Defaults } = require('redis-request-broker');
Defaults.setDefaults({ redis: { port: 1234, db: 'separated' } });
const w1 = new Worker('myqueue', someWork, { logger: console.log }); // Options will be merged

Here are all available options:

  • redis: An object that configures the redis connection. It will be passed directly to the createClient method of the underlying redis package. See The redis npm package for more information.

    IMPORTANT: When overriding you should make sure to set either the prefix or the db option or otherwise your keys might get mixed up with other stuff in the database!

    • The default value is { prefix: 'rrb:' }.
    • Example: { port: 1234, db: 'myapp' }

  • timeout: A timeout in ms after which a request fails. For the client that means when it will stop waiting for a response from a worker and rejects the request.

    • The default value is 1000 ms.
    • Example: { timeout: 5000 } // five seconds
  • logger: Allows to inject a custom logger. It has to be a method and is provided with the following arguments:

    • level: The level of the log, as configures using levels. String by default.
    • message: The message of the log.
    • component: The component that issued the log. Can be client, worker, publisher or subscriber.
    • instance: A string id of the instace that issued the log.
    • scope: The current operation that the insance was working on when the log has been issued. Are strings like connect or request

      It is not necessary to use all of the arguments.

    • The default logger is writing error, warning and notice logs to the console.

    • Example: { logger: (level, message) => console.log(message)}
  • levels: Allows to customize what gets passed into the logger method for logging. The package uses five different levels: error, warning, notice, info and debug.

    • The default values are the respective strings ('error' for error, etc.) and are therefore compatible with winston log levels.
    • Example: { levels: { error: 'e', warning: 'w', notice: 'n', info: 'i', debug: 'd' }}
  • minimumRecipients: Sets the minimum amount of recipients that should receive a published message. Only effects the publisher.

    • The default value is 0.
    • Example: { minimumRecipients: 2 }

Transparent Error Handling

If the handle mehod of the worker rejects the request, the error will be transmitted to the client where the reqeust call will be rejected with the same error, if not already timed out.

Here is a simple example of this behavior:

const w = new Worker('throwing', d => { throw d; });
const c = new Client('throwing');
await w.listen();
await c.connect();
try {
  await c.request('My Request');
}
catch (error) {
  console.log(error); // Will log 'My Request'
}

Note that due to the fact that all data that is sent between workers and clients is serialized, some properties of your errors may not be present on the client side, especially methods.

Internally, JSON.stringify and JSON.parse are used for the (de-) serialization.

Inner workings

Here is how it is working:

For every queue there will be a availibilitySet in the reddis db storing ids of workers that are availible. Furthermore, a requestQueue (list) is used for requests that cannot be handled right away. When a client starts a requests, it will pop one id from the availiblitySet and sends the reqeust to a channel spcific for this worker. If no id is in the availiblitySet, the request will be put into the requestQueue instead.

Workers will receive messages and call the provided handler function. Results will be send into a response channel specific for the request. Then the worker will check if there is a request waiting in the requestQueue and handle the oldest request. This will repeat until the request queue is empty. After that the worker puts itself into the availiblitySet again.

Clients choose workers using the SPOP redis command, which means that a random idle worker is chosen when available.

1.12.0

5 years ago

1.11.1

5 years ago

1.9.1

5 years ago

1.9.0

5 years ago

1.9.2

5 years ago

1.11.0

5 years ago

1.10.0

5 years ago

1.8.0

5 years ago

1.7.1

5 years ago

1.7.0

5 years ago

1.6.0

5 years ago

1.5.5

5 years ago

1.5.4

5 years ago

1.5.3

5 years ago

1.5.6

5 years ago

1.5.2

5 years ago

1.5.1

5 years ago

1.5.0

5 years ago

1.4.1

6 years ago

1.4.0

6 years ago

1.3.0

6 years ago

1.2.0

6 years ago

1.1.0

6 years ago

1.0.1

6 years ago