Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/radiofrance/rab-q

A tiny (opinionated) wrapper over amqplib for RabbitMQ publish/subscribe pattern
https://github.com/radiofrance/rab-q

amqp broker exchange promise queue rabbitmq

Last synced: 2 months ago
JSON representation

A tiny (opinionated) wrapper over amqplib for RabbitMQ publish/subscribe pattern

Awesome Lists containing this project

README

        

rab-q


:rabbit2: :dash: :dash: :dash: :dash: :turtle:


Bunnies love turtles


A tiny (opinionated) wrapper over amqplib for RabbitMQ publish-subscribe pattern





NPM version



Build Status



Test Coverage



Downloads



RadioFrance

## Table of Contents
* [Features](#features)
* [Our use case at @RadioFrance](#our-use-case-at-radiofrance)
* [Installation](#installation)
* [Usage](#usage)
* [API](#api)
* [Events](#events)
* [See Also](#see-also)

## Features
* __minimal dependencies:__ only [`amqplib`](https://github.com/squaremo/amqp.node) and [`uuid`](https://github.com/kelektiv/node-uuid) are needed
* __promises-based:__ async/await support makes it easy to use
* __small api:__ 4 methods only, there's not much to learn :wink:
* __instantiation-able:__ the library itself is a Class ; get as many independent instance as needed
* __events logging:__ choose your own log transport
* __disconnections-proof:__ auto reconnect/resend when errors happen

## Our use case at @RadioFrance
This wrapper was tailored made for our microservices architecture.

We use RabbitMQ as the bus in the publish-subscribe pattern. A message is never sent directly to a queue, it is always sent to a configured *exchange*.

Every message should be acknowledged by returning a value : `ACK`, `NACK`, `REJECT` (`message.ACK`, `message.NACK` or `message.REJECT` in the *message* object).

A non-acknowledged message will be redelivered one time to the *exchange*. A rejected message will be directly deleted or redirected to the [dead letter exchange](https://www.rabbitmq.com/dlx.html) if configured.

Our queues and exchanges are defined by a external engine. `rab-q` doesn't provide API to create, assert or delete these.

## Installation
```sh
$ npm install rab-q
```

## Usage
```js
const RabQ = require('rab-q');

const rabQ = new RabQ({
exchange: 'your_topic_exchange',
queues: 'queue_bind_to_exchange'
});

rabQ.on('error', err => {
console.error(err);
});
rabQ.on('log', log => {
console[log.level](log.msg, log.err || '');
});

rabQ.subscribesTo(/.*/, message => {
// Do stuff!

// Then acknowledge message by returning a constant
return message.ACK;
});

rabQ.start()
.then(() => {
rabQ.publish('yourRoutingKeyHere', {your: 'message'});
});
```

## API

### new RabQ(options)

#### options

##### username

Type: `string`

Default: `guest`

User to connect on RabbitMQ.

##### password

Type: `string`

Default: `guest`

Password associate to username.

##### protocol

Type: `amqp|amqps`

Default: `amqp`

Connection protocol.

##### hostname

Type: `string`

Default: `localhost`

Hostname to trying to get connection.

##### port

Type: `number`

Default: `5672`

Port of connection.

##### socketOptions

Type: `object`
Default: `null`

socketOptions for amqplib

https://amqp-node.github.io/amqplib/channel_api.html#connect

##### vhost

Type: `string`

Default: `/`

Selected virtual host.

##### exchange

*Required*

Type: `string`

Exchange to publish messages.

##### queues

Type: `Array` `string`

Consumes messages from these queues.

If this option is omitted or has a *falsey* value, server will create random names.

##### maxMessages

Type: `number`

Default: `10`

Defines the number of messages prefetched by channel. Once the max number of messages is reached, RabbitMQ will wait for some messages to be acknowledged before proceeding with new messages.

##### nackDelay

Type: `number`

Default: `0`

Defines a delay in milliseconds before a message is rejected with NACK.

##### reconnectInterval

Type: `number` *(milliseconds)*

Default: `0`

Time in milliseconds before trying to reconnect when connection is lost.

##### autoAck

Type: `boolean`

Default: `false`

Enables auto acknowledgment.

##### autoReconnect

Type: `boolean`

Default: `true`

Enables auto reconnection if an error happens while connecting to the server.

##### validators

Type: `Object of function`

###### validators.consumer

Type: `boolean`

Default: `return true`

Function run before each message treatment. If it return a false value, the message is reject.

##### beforeHook

Type: `function`

Default: () => {}
Parameters: `message` (see below)

Function run before each message treatment, can modify message.

##### afterHook

Type: `function`

Default: () => {}
Parameters: `message` (see below), `subscriberResult` (ACK/NACK/REJECT)

Function run after each message treatment.

##### prePublish

Type: `function`

Default: null
Parameters: `routingKey`, `content`, `properties` (see publish method)
Must return an object with {routingKey, content, properties}

Function run before each publish call

### rabQ.start()

Starts a connection.

Returns a promise resolved with `true` for a successful connection or `false` if a connection already exists.

### rabQ.stop()

Stops and closes a connection.

### rabQ.publish(routingKey, content, properties)

Publishes a message on exchange `rabQ.exchange`.

#### routingKey

*Required*

Type: `string`

A regular expression to match the routing keys of consumed messages.

#### content

*Required*

Type: `Object`

A message object to send to the exchange.

#### properties

Type: `Object`

Default: `{}`

Adds RabbitMQ properties to the message. See http://www.squaremobius.net/amqp.node/channel_api.html#channel_publish (options)

If properties does not have a `headers` key, `properties` is considered to be `headers`.

You can provide a property `x-query-token` in `headers` to trace the lifecyle of a request. If not provided a new UUID will be generated.

### rabQ.subscribesTo(patternMatch, action)

Adds a Subscriber on consumed messages filtered by the routing key

#### patternMatch

*Required*

Type: `RegExp`

A regular expression to match on the routing keys of consumed messages.

#### action(message)

*Required*

Type: `Function`

A function should acknowledge (or not) messages by returning either a value or a resolved promise with value. (`'ACK'`, `'NACK'` or `'REJECT'`)

`message` is a object with following properties:
* __ACK__ `string`: the constant returned for a positive acknowledgment
* __NACK__ `string`: the constant returned for a negative acknowledgment. _NACK will re-queuing message one time._
* __REJECT__ `string`: the constant returned for a rejection without redelivery.
* __content__ `Object`: content of the received message
* __rk__ `string`: routing key of the message
* __queue__ `string`: queue name where the message is consumed
* __token__ `string`: a UUID to identify the message
* __consumeAt__ `number`: timestamp when the message is consumed

## Events

### `'log'`

Emits a log object with the following properties:
* __level__ `string`: can be `info`, `warn`, `error`
* __token__ `string`: the UUID to identify the message
* __msg__ `string`: the message,
* __err__ `Object`: the original error if log level is _error_

### `'error'`

Emits an Error if something goes wrong with the connection. If you don't catch this event the process will sto

## See Also
* [RabbitMQ Website](http://www.rabbitmq.com/)
* [`amqplib`](https://github.com/squaremo/amqp.node)

## License
[CECILL-B](https://spdx.org/licenses/CECILL-B.html)