AMQP 1.0-compliant Node.js client. Since AMQP 1.0 is such a large departure from 0.9.1, I've started a new project rather than fork from node-amqp or amqp.node. Both node-amqp and amqp.node are great 0.9.1 clients and I recommend them, but neither is pursuing a 1.0 implementation. If I can find an easy way to integrate this code back into them, I'll definitely be submitting a PR.
See simple_eventhub_test.js
or simple_activemq_test.js
for examples.
The basic usage is to require the module, new up a client with the appropriate policy for the server you're connecting against, connect, and then send/receive as necessary. So a simple example for a local ActiveMQ server would look like:
var AMQPClient = require('node-amqp-1-0');
var client = new AMQPClient(); // Uses PolicyBase default policy
client.connect('amqp://localhost/myqueue', function(conn_err) {
// ... check for errors ...
client.send(JSON.stringify({ key: "Value" }), function (send_err) {
// ... check for errors ...
});
client.receive(function (rx_err, payload, annotations) {
// ... check for errors ...
console.log('Rx message: ');
console.log(JSON.parse(payload));
});
});
Note that the above JSON.stringify/JSON.parse on send/receive can be moved into the encoder/decoder methods on the policy object - see the Event Hub policy for an example.
NOTE: This is early days - if you have ideas for an alternate API, please feel free to open an issue on GitHub.
I'm trying to manage my remaining work items via Github issues, but they aren't always kept up to date. If you'd like to contribute, feel free to send me an email or pull request. Below is a high-level list of known open issues:
- Disposition frames are not dealt with properly, and thus message lifecycles aren't tracked correctly. Currently we work fine for auto-settled workflows, but not for handshake transfers.
- We work well for sunny-day scenarios and some severed link/connection issues, but not all. I'm building out an AMQPClient-level test suite to allow me to see exactly where we fail here, but that work is ongoing and will likely result in some code restructuring.
- There are some AMQP types we don't process - notably GUID, and the Decimal23/64/128 types. These are unused by the protocol, and no-one seems to be using them to convey information in messages, so ignoring them is likely safe.
Here are my current implementation notes - if you have feedback or critiques on any of these choices, feel free to submit an Issue or even a PR. Trust me, I don't take criticism personally, and I'm new to Nodejs so I could be making "obviously bad" choices to someone who is more familiar with the landscape.
- I've used Node's built-in net/tls classes for communicating with the server.
- Data from the server is written to a circular buffer based on CBarrick's.
- Outgoing data will be encoded using this buffer builder - streaming output won't really work since each outgoing payload needs to be prefixed with its encoded size.
- The connection state will be managed using Stately.js, with state transitions swapping which callback gets invoked on receipt of new data. (e.g. post-connection, we write the AMQP version header and then install a callback to ensure the correct version. Once incoming data is written to the circular buffer, this callback is invoked, and a comparison vs. the expected version triggers another transition).
- Bit-twiddling is done via node-butils.
- Debug output is done via debug with the prefix
amqp10-
. The main client's debug name isamqp10-client
so settingDEBUG=amqp10-client
will get you all top-level debugging output.
Further, detailed implementation nodes are available in the API Readme.
MIT License. If you need a more permissive license, or you want to try your hand at integrating this code into node-amqp or amqp.node and need to match their license, let me know (i.e. open an issue).