rsmq-ms
Version:
A forked of the really simple message queue based on Redis with millisecond delay support.
466 lines (331 loc) • 16.8 kB
Markdown

# Redis Simple Message Queue
A lightweight message queue for Node.js that requires no dedicated queue server. Just a Redis server.
[](http://travis-ci.org/smrchy/rsmq)
[](https://david-dm.org/smrchy/rsmq)
**tl;dr:** If you run a Redis server and currently use Amazon SQS or a similar message queue you might as well use this fast little replacement. Using a shared Redis server multiple Node.js processes can send / receive messages.
## Features
- Lightweight: **Just Redis** and ~500 lines of javascript.
- Speed: Send/receive 10000+ messages per second on an average machine. It's **just Redis**.
- Guaranteed **delivery of a message to exactly one recipient** within a messages visibility timeout.
- Received messages that are not deleted will reappear after the visibility timeout.
- [Test coverage](http://travis-ci.org/smrchy/rsmq)
- A message is deleted by the message id. The message id is returned by the `sendMessage` and `receiveMessage` method.
- Messages stay in the queue unless deleted.
- Optional RESTful interface via [rest-rsmq](https://github.com/smrchy/rest-rsmq)
- Typescript Typings ❤️
- Optional Promise-based API (only if `Promise` is defined), just suffix your method with `Async`, eg: `sendMessage` -> `sendMessageAsync`, all queue methods are supported
**Note:** RSMQ uses the Redis EVAL command (LUA scripts) so the minimum Redis version is 2.6+.
## Usage
- After creating a queue you can send messages to that queue.
- The messages will be handled in a **FIFO** (first in first out) manner unless specified with a delay.
- Every message has a unique `id` that you can use to delete the message.
- The `sendMessage` method will return the `id` for a sent message.
- The `receiveMessage` method will return an `id` along with the message and some stats.
- Should you not delete the message it will be eligible to be received again after the visibility timeout is reached.
- Please have a look at the `createQueue` and `receiveMessage` methods described below for optional parameters like **visibility timeout** and **delay**.
## Installation
`npm install rsmq`
## Modules for RSMQ
To keep the core of _RSMQ_ small additional functionality is available as modules:
- [**rsmq-worker**](https://github.com/mpneuried/rsmq-worker) Helper to implement a worker with RSMQ.
- [**rest-rsmq**](https://github.com/smrchy/rest-rsmq) A RESTful interface for RSMQ.
- [**rsmq-cli**](https://github.com/mpneuried/rsmq-cli) A command-line interface / Terminal client for RSMQ.
- [**rsmq-promise-native**](https://www.npmjs.com/package/rsmq-promise-native) Native promise interface for RSMQ.
- [**rsmq-promise**](https://www.npmjs.com/package/rsmq-promise) Promise interface for RSMQ (via Bluebird).
## RSMQ in other languages
The simplicity of RSMQ is useful in other languages. Here is a list of implementations in other languages:
- [**Rust**](https://github.com/Couragium/rsmq-async-rs) RSMQ Async for Rust. See [#107](https://github.com/smrchy/rsmq/issues/107)
- [**Java**](https://github.com/igr/jrsmq) RSMQ for Java. See [#48](https://github.com/smrchy/rsmq/issues/48)
- [**PHP**](https://github.com/michsindelar/PhpRSMQ) RSMQ for PHP (work in progress)
- [**CSharp/DotNet**](https://github.com/tontonrally/rsmqCsharp) RSMQ for .NET
Note: Should you plan to port RSQM to another language please make sure to have tests to ensure compatibility with all RSMQ clients. And of course: let me know so i can mention your port here.
## Methods
### Constructor
Creates a new instance of RSMQ.
Parameters:
- `host` (String): _optional (Default: "127.0.0.1")_ The Redis server
- `port` (Number): _optional (Default: 6379)_ The Redis port
- `options` (Object): _optional (Default: {})_ The [Redis options](https://github.com/NodeRedis/node_redis#options-object-properties) object.
- `client` (RedisClient): _optional_ A existing redis client instance. `host` and `server` will be ignored.
- `ns` (String): _optional (Default: "rsmq")_ The namespace prefix used for all keys created by RSMQ
- `realtime` (Boolean): _optional (Default: false)_ Enable realtime PUBLISH of new messages (see the [Realtime section](#realtime))
- `password` (String): _optional (Default: null)_ If your Redis server requires a password supply it here
Example:
```javascript
const RedisSMQ = require("rsmq");
const rsmq = new RedisSMQ({ host: "127.0.0.1", port: 6379, ns: "rsmq" });
```
### Queue
#### createQueue(options, callback)
Create a new queue.
Parameters:
- `qname` (String): The Queue name. Maximum 160 characters; alphanumeric characters, hyphens (-), and underscores (\_) are allowed.
- `vt` (Number): _optional_ _(Default: 30)_ The length of time, in seconds, that a message received from a queue will be invisible to other receiving components when they ask to receive messages. Allowed values: 0-9999999 (around 115 days)
- `delay` (Number): _optional_ _(Default: 0)_ The time in seconds that the delivery of all new messages in the queue will be delayed. Allowed values: 0-9999999 (around 115 days)
- `maxsize` (Number): _optional_ _(Default: 65536)_ The maximum message size in bytes. Allowed values: 1024-65536 and -1 (for unlimited size)
Returns:
- `1` (Number)
Example:
```javascript
rsmq.createQueue({ qname: "myqueue" }, function (err, resp) {
if (err) {
console.error(err);
return;
}
if (resp === 1) {
console.log("queue created");
}
});
```
#### listQueues(options, callback)
List all queues
Returns an array:
- `["qname1", "qname2"]`
Example:
```javascript
rsmq.listQueues(function (err, queues) {
if (err) {
console.error(err);
return;
}
console.log("Active queues: " + queues.join(","));
});
```
#### deleteQueue(options, callback)
Deletes a queue and all messages.
Parameters:
- `qname` (String): The Queue name.
Returns:
- `1` (Number)
Example:
```javascript
rsmq.deleteQueue({ qname: "myqueue" }, function (err, resp) {
if (err) {
console.error(err);
return;
}
if (resp === 1) {
console.log("Queue and all messages deleted.");
} else {
console.log("Queue not found.");
}
});
```
#### getQueueAttributes(options, callback)
Get queue attributes, counter and stats
Parameters:
- `qname` (String): The Queue name.
Returns an object:
- `vt` (Number): The visibility timeout for the queue in seconds
- `delay` (Number): The delay for new messages in seconds
- `maxsize` (Number): The maximum size of a message in bytes
- `totalrecv` (Number): Total number of messages received from the queue
- `totalsent` (Number): Total number of messages sent to the queue
- `created` (Number): Timestamp (epoch in seconds) when the queue was created
- `modified` (Number): Timestamp (epoch in seconds) when the queue was last modified with `setQueueAttributes`
- `msgs` (Number): Current number of messages in the queue
- `hiddenmsgs` (Number): Current number of hidden / not visible messages. A message can be hidden while "in flight" due to a `vt` parameter or when sent with a `delay`
Example:
```javascript
rsmq.getQueueAttributes({ qname: "myqueue" }, function (err, resp) {
if (err) {
console.error(err);
return;
}
console.log("==============================================");
console.log("=================Queue Stats==================");
console.log("==============================================");
console.log("visibility timeout: ", resp.vt);
console.log("delay for new messages: ", resp.delay);
console.log("max size in bytes: ", resp.maxsize);
console.log("total received messages: ", resp.totalrecv);
console.log("total sent messages: ", resp.totalsent);
console.log("created: ", resp.created);
console.log("last modified: ", resp.modified);
console.log("current n of messages: ", resp.msgs);
console.log("hidden messages: ", resp.hiddenmsgs);
});
```
#### setQueueAttributes(options, callback)
Sets queue parameters.
Parameters:
- `qname` (String): The Queue name.
- `vt` (Number): _optional_ \* The length of time, in seconds, that a message received from a queue will be invisible to other receiving components when they ask to receive messages. Allowed values: 0-9999999 (around 115 days)
- `delay` (Number): _optional_ The time in seconds that the delivery of all new messages in the queue will be delayed. Allowed values: 0-9999999 (around 115 days)
- `maxsize` (Number): _optional_ The maximum message size in bytes. Allowed values: 1024-65536 and -1 (for unlimited size)
Note: At least one attribute (vt, delay, maxsize) must be supplied. Only attributes that are supplied will be modified.
Returns an object:
- `vt` (Number): The visibility timeout for the queue in seconds
- `delay` (Number): The delay for new messages in seconds
- `maxsize` (Number): The maximum size of a message in bytes
- `totalrecv` (Number): Total number of messages received from the queue
- `totalsent` (Number): Total number of messages sent to the queue
- `created` (Number): Timestamp (epoch in seconds) when the queue was created
- `modified` (Number): Timestamp (epoch in seconds) when the queue was last modified with `setQueueAttributes`
- `msgs` (Number): Current number of messages in the queue
- `hiddenmsgs` (Number): Current number of hidden / not visible messages. A message can be hidden while "in flight" due to a `vt` parameter or when sent with a `delay`
Example:
```javascript
rsmq.setQueueAttributes({ qname: "myqueue", vt: "30" }, function (err, resp) {
if (err) {
console.error(err);
return;
}
console.log(
"changed the invisibility time of messages that have been received to 30 seconds"
);
console.log(resp);
});
```
### Messages
#### sendMessage
Sends a new message.
Parameters:
- `qname` (String)
- `message` (String)
- `delay` (Number): _optional_ _(Default: queue settings)_ The time in seconds that the delivery of the message will be delayed. Allowed values: 0-9999999 (around 115 days)
- `delayInMillis` (Boolean): If true, delay will be treated as milliseconds instead of seconds
Returns:
- `id` (String): The internal message id.
Example:
```javascript
rsmq.sendMessage(
{ qname: "myqueue", message: "Hello World " },
function (err, resp) {
if (err) {
console.error(err);
return;
}
console.log("Message sent. ID:", resp);
}
);
```
#### receiveMessage(options, callback)
Receive the next message from the queue.
Parameters:
- `qname` (String): The Queue name.
- `vt` (Number): _optional_ _(Default: queue settings)_ The length of time, in seconds, that the received message will be invisible to others. Allowed values: 0-9999999 (around 115 days)
* `vtInMillis` (Boolean): If true, vt will be treated as milliseconds instead of seconds
Returns an object:
- `message` (String): The message's contents.
- `id` (String): The internal message id.
- `sent` (Number): Timestamp of when this message was sent / created.
- `fr` (Number): Timestamp of when this message was first received.
- `rc` (Number): Number of times this message was received.
Note: Will return an empty object if no message is there
Example:
```javascript
rsmq.receiveMessage({ qname: "myqueue" }, function (err, resp) {
if (err) {
console.error(err);
return;
}
if (resp.id) {
console.log("Message received.", resp);
} else {
console.log("No messages for me...");
}
});
```
#### deleteMessage(options, callback)
Parameters:
- `qname` (String): The Queue name.
- `id` (String): message id to delete.
Returns:
- `1` if successful, `0` if the message was not found (Number).
Example:
```javascript
rsmq.deleteMessage(
{ qname: "myqueue", id: "dhoiwpiirm15ce77305a5c3a3b0f230c6e20f09b55" },
function (err, resp) {
if (err) {
console.error(err);
return;
}
if (resp === 1) {
console.log("Message deleted.");
} else {
console.log("Message not found.");
}
}
);
```
#### popMessage(options, callback)
Receive the next message from the queue **and delete it**.
**Important:** This method deletes the message it receives right away. There is no way to receive the message again if something goes wrong while working on the message.
Parameters:
- `qname` (String): The Queue name.
Returns an object:
- `message` (String): The message's contents.
- `id` (String): The internal message id.
- `sent` (Number): Timestamp of when this message was sent / created.
- `fr` (Number): Timestamp of when this message was first received.
- `rc` (Number): Number of times this message was received.
Note: Will return an empty object if no message is there
Example:
```javascript
rsmq.popMessage({ qname: "myqueue" }, function (err, resp) {
if (err) {
console.error(err);
return;
}
if (resp.id) {
console.log("Message received and deleted from queue", resp);
} else {
console.log("No messages for me...");
}
});
```
#### changeMessageVisibility(options, callback)
Change the visibility timer of a single message.
The time when the message will be visible again is calculated from the current time (now) + `vt`.
Parameters:
- `qname` (String): The Queue name.
- `id` (String): The message id.
- `vt` (Number): The length of time, in seconds, that this message will not be visible. Allowed values: 0-9999999 (around 115 days)
* `vtInMillis` (Boolean): If true, vt will be treated as milliseconds instead of seconds
Returns:
- `1` if successful, `0` if the message was not found (Number).
Example:
```javascript
rsmq.changeMessageVisibility(
{
qname: "myqueue",
vt: "60",
id: "dhoiwpiirm15ce77305a5c3a3b0f230c6e20f09b55",
},
function (err, resp) {
if (err) {
console.error(err);
return;
}
if (resp === 1) {
console.log("message hidden for 60 seconds");
}
}
);
```
### quit(callback)
Disconnect the redis client.
This is only useful if you are using rsmq within a script and want node to be able to exit.
## Realtime
When [initializing](#initialize) RSMQ you can enable the realtime PUBLISH for new messages. On every new message that gets sent to RSQM via `sendMessage` a Redis PUBLISH will be issued to `{rsmq.ns}:rt:{qname}`.
Example for RSMQ with default settings:
- The queue `testQueue` already contains 5 messages.
- A new message is being sent to the queue `testQueue`.
- The following Redis command will be issued: `PUBLISH rsmq:rt:testQueue 6`
### How to use the realtime option
Besides the PUBLISH when a new message is sent to RSMQ nothing else will happen. Your app could use the Redis SUBSCRIBE command to be notified of new messages and issue a `receiveMessage` then. However make sure not to listen with multiple workers for new messages with SUBSCRIBE to prevent multiple simultaneous `receiveMessage` calls.
## Changes
see the [CHANGELOG](https://github.com/smrchy/rsmq/blob/master/CHANGELOG.md)
## Other projects
| Name | Description |
| :-------------------------------------------------------------------------------- | :---------------------------------------------------------------------------------------------------------------------------------------------------- |
| [**node-cache**](https://github.com/tcs-de/nodecache) | Simple and fast Node.js internal caching. Node internal in memory cache like memcached. |
| [**redis-tagging**](https://github.com/smrchy/redis-tagging) | A Node.js helper library to make tagging of items in any legacy database (SQL or NoSQL) easy and fast. |
| [**redis-sessions**](https://github.com/smrchy/redis-sessions) | An advanced session store for Node.js and Redis |
| [**rsmq-worker**](https://github.com/mpneuried/rsmq-worker) | Helper to implement a worker based on [RSMQ (Redis Simple Message Queue)](https://github.com/smrchy/rsmq). |
| [**connect-redis-sessions**](https://github.com/mpneuried/connect-redis-sessions) | A connect or express middleware to use [redis sessions](https://github.com/smrchy/redis-sessions) that lets you handle multiple sessions per user_id. |
## The MIT License
Please see the LICENSE.md file.