richardzyx / node-microservice

Clean, Direct, Easy to Scale solution to Node Microservice Framework

Geek Repo:Geek Repo

Github PK Tool:Github PK Tool

Node-Microservice Package

Clean, Direct, Easy to Scale solution to Node Microservice Framework.

Inspired by seneca.js, IBM's article on microservice using seneca and MQLight, and RabbitMQ's tutorial on RPC, we aim to create a one stop solution that will allow seneca-style communication between any number of servers and clients, with the ability of load balance offered by message brokers such as RabbitMQ. We decided to do this without any dependency on seneca, thus allowing us the maximum freedom in programming language, and set up RabbitMQ server as our own message broker. Using this package, you only need one function to set up the server, and two functions for the client. You can even separate the connection and send functions for the client, so you only need to connect once for your entire project, and send any number of requests.

Special thanks to my mentor and great boss: dingziran@Wecash! This solution will not exist or even work without his help.

TL;DR: freedom in language, minimum lines of code, over amqp message broker

Advantage/Features:

  • Timeout Mechanism: During our production test, we found that having the timeout mechanism is extremely helpful especially when the internet connection is poor and the number of requests is huge. Both the server and the client take the parameter of timeout, so that when the time is due, the client will automatically timeout the requests, or the tasks will timeout in the queue before being sent to the server.
  • Multiple Requests: While testing all of the available choices, we found that none of them can handle the problem of having multiple requests from one client. By using our own function, we made sure that no matter which requesting client received the response, the request will be resolved and returned properly.
  • Promise: No more call back api! Everything is a Promise.

This package is tested and stable. The performance is great in production for over two months. More extensive testings are welcomed!

What's new

  • Fixed client's timeout error behavior- it now throws an Error object, intead of a string of "Timeout"
  • Added ensureDone to server's option- the server will guarantee to complete the task by asking the MQ to requeue the error/failed message.
  • Fixed client's behavior when you don't care about server's response- it now returns undefined.

Install

npm install node-microservice

Usage

Server:

Just add this one line of code at the end of your service file, pass options as an object, and you have a working server.

exports.server_listen=function(amqp_url,service_name,pro,options)
  • amqp_url is the address of your MQ service. Such as:"amqp://usr:password@128.11.22.230"
  • service_name is the name of the request the server is listening to. The server will only listen to the request that the client sent with the same service name.
  • pro is the function that you want to pass the message/task to. This function MUST be a Promise.

For options:

Must Have: noAck (boolean)

  • if noAck = True: the queue will send tasks to server and then discard regardless of the server's state. Warning: when having two or more servers, setting noAck to True will risk loosing message if one of the server that gets the messages goes offline.
  • if noAck = False: the queue will make sure the server get and finish the task with acknowledgements. Warning: if the server gets the tasks but fails to finish, the message will
      1. Timeout as determined by messageTtl option,
      1. If the server goes offline, the tasks will be requeued and sent to the next available server.

*We recommend setting noAck to False to guarantee message delivery and avoid loosing message when multiple servers are online and working.

Optional:

messageTtl (milliseconds) (must start a new queue if this option was just added, modified, or taken out)

  • Set timeout for messages in the queue. This option has proven to be extremely helpful when the server goes offline and comes back and the messages are requeued. If the messageTtl is set to the same milliseconds as the client's timeout, the queue will make sure the server do not get timed out messages that the client no longer cares about.

*We recommend setting the messageTtl equal to the client's timeout parameter.

prefetch_num (integer bigger than or equal to 1)

  • Set the maximum number of acknowledgements the queue can wait from the server. In other words, it is the maximum number of tasks one server can take each time. That said, this will only be effective when the noAck is set to False. If prefetch_num is not given, the server will simply take as many tasks as possible and this may cause a race condition. In our production experience, we find that a number of 10 or 100 works just fine.

*Only effective when noAck is set to False.

ensureDone (boolean)

  • New: When set to true, if your server return the task with an error, the MQ will take back and requeue the message. Great for production when you need to make huge and consistent queries. It will guarantee the completeness of each task until servers no longer send error and ack the message.

*Only effective when noAck is set to False.

durable (boolean)

  • Make the queue durable as stated on the RabbitMQ website. The default setting is false.

Example:

  • Our Safest/Most Used Options: {noAck:false, prefetch_num:10, messageTtl:60000}
  • Simplest/Minimalist Options(best for just testing): {noAck:true}
  • When you need to make sure all tasks are completed with no error: {noAck:false, prefetch_num:10, messageTtl:60000, ensureDone:true}

Client:

This function will easily set up your client with your message broker over amqp protocol:

exports.connect_amqp=function(amqp_url, [options])
  • amqp_url is the address of your MQ service. Such as:"amqp://usr:password@128.11.22.230"
  • options is the object that contains your graylog server's setting. A typical and minimum setting is included in the testClient.js file in example folder. This is optional if you don't have or don't want to use the graylog system for your project.

This function will send your message to your designated server, and timeout if a response is not received within the given time:

exports.send=function(serviceName,message,timeout)
  • serviceName is the name of the server you want to send your message to. Make sure your server and client have the same serviceName!
  • message is an object that is sent to the message broker. Behind the scene, the message is first transformed to string and then a buffer. After the server received the message, it will first decode and parse it as a JSON object.
  • timeout is the milliseconds you want the client to wait before giving up

We have decided that the client will not have an noAck option and it is set to true since we do not really care about whether the client has received the response. Welcome to fork if you would like to add this feature!

Logging

We had a long discussion with multiple tryouts, and we havedecided to use and only support the graylog system based on graylog2 package for this project. It was not an easy decision for us, and we understand having to install both graylog and elasticsearch would take up to two hours of configurations. So we have made the logging options optional. If you would still like to use the common way console.log, simply uncomment the line in source code and everything will work exactly the same. For detailed description, see client function usage.

Example

You can find one test client and one test server file in the example folder. The amqp_url is fortified! Try with different options.

Message Broker

We recommend and use RabbitMQ for its popularity and wide range of support. For a simple intro to installation and tutorial, please refer to its website. Essentially, since we use amqplib, any message broker based on amqp protocol is just fine.

Version Updates

  • Ver 0.6.0: Production performance is great. Welcome for extensive testings!
  • Ver 0.5.11: Timeout err is thrown with an Error object
  • Ver 0.5.10: minor improve on graylog message; server throw error after disconnection
  • Ver 0.5.9: see New section; added ensureDone for extreme production needs
  • Ver 0.5.8: see New section; fixed client's behavior with empty response and bug on timeout error; the correlationID is used to identify empty response
  • Ver 0.5.7: the logging system is now here with full support to graylog
  • Ver 0.5.6: fixed the problem of not parsing the response from server to JSON in package before passing back to client; no parsing is needed at all now!
  • Ver 0.5.5: fixed problems in server side on error handling including returning messages and shutting down when connection throws an error; updated license from ISC to MIT; minor fixes in readme; no longer prefer global installation

TODO

Definite:

  • Improve Readme with photos/interactions and the underlying mechanism
  • Include a simple RabbitMQ tutorial in README

Maybe:

  • Implement the topic/fanout delivery options
  • Implement the exchange/clustering node options

Any help/fork/issues/contribution is deeply appreciated and welcomed!

July 2015

About

Clean, Direct, Easy to Scale solution to Node Microservice Framework

License:MIT License


Languages

Language:JavaScript 100.0%