data:image/s3,"s3://crabby-images/2523c/2523ce4b8b64bade795ffc89574cfc29f35428d3" alt="Deno 2.2 Improves Dependency Management and Expands Node.js Compatibility"
Security News
Deno 2.2 Improves Dependency Management and Expands Node.js Compatibility
Deno 2.2 enhances Node.js compatibility, improves dependency management, adds OpenTelemetry support, and expands linting and task automation for developers.
moleculer
Advanced tools
Moleculer is a fast & powerful microservices framework for NodeJS (>= v6.x).
Please do not use this in production since it's still under heavy development!
$ npm install moleculer --save
or
$ yarn add moleculer
const { ServiceBroker } = require("moleculer");
// Create broker
let broker = new ServiceBroker({
logger: console
});
// Create a service
broker.createService({
name: "math",
actions: {
// You can call it as broker.call("math.add")
add(ctx) {
return Number(ctx.params.a) + Number(ctx.params.b);
},
// You can call it as broker.call("math.sub")
sub(ctx) {
return Number(ctx.params.a) - Number(ctx.params.b);
}
}
});
// Start broker
broker.start();
// Call actions of service
broker.call("math.add", { a: 5, b: 3 })
.then(res => console.log("5 + 3 =", res));
// Call actions with error handling
broker.call("math.sub", { a: 9, b: 2 })
.then(res => console.log("9 - 2 =", res))
.catch(err => console.error(`Error occured! ${err.message}`));
// Chain calls
broker.call("math.add", { a: 3, b: 5})
.then(res => broker.call("math.sub", { a: res, b: 2 }))
.then(res => console.log("3 + 5 - 2 =", res));
Package name | Description | Version | License |
---|---|---|---|
moleculer-web | Official API Gateway service for Moleculer framework |
Package name | Description | Version | License |
---|
We tested some other frameworks and measured the request times.
The ServiceBroker
is the main component of Moleculer. It handles services & events, calls actions and communicates with remote nodes. You need to create an instance of ServiceBroker
on every node.
Create broker with default settings
let { ServiceBroker } = require("moleculer");
let broker = new ServiceBroker();
Create broker with custom settings
let { ServiceBroker } = require("moleculer");
let broker = new ServiceBroker({
logger: console,
logLevel: "info"
});
Create with transporter
let { ServiceBroker, NatsTransporter } = require("moleculer");
let broker = new ServiceBroker({
nodeID: "node-1",
transporter: new NatsTransporter(),
logger: console,
logLevel: "debug",
requestTimeout: 5 * 1000,
requestRetry: 3
});
Create with cacher
let ServiceBroker = require("moleculer").ServiceBroker;
let MemoryCacher = require("moleculer").Cachers.Memory;
let broker = new ServiceBroker({
cacher: new MemoryCacher(),
logger: console,
logLevel: {
"*": "warn", // global log level for every modules
"CACHER": "debug" // custom log level for cacher modules
}
});
All available options:
{
nodeID: null,
logger: null,
logLevel: "info",
transporter: null,
requestTimeout: 0,
requestRetry: 0,
heartbeatInterval: 10,
heartbeatTimeout: 30,
cacher: null,
serializer: null,
validation: true,
metrics: false,
metricsRate: 1,
statistics: false,
internalActions: true
ServiceFactory: null,
ContextFactory: null
}
Name | Type | Default | Description |
---|---|---|---|
nodeID | String | Computer name | This is the ID of node. It identifies a node in the cluster when there are many nodes. |
logger | Object | null | Logger class. During development you can set to console . In production you can set an external logger e.g. winston or pino |
logLevel | String or Object | info | Level of logging (debug, info, warn, error) |
transporter | Transporter | null | Instance of transporter. Required if you have 2 or more nodes. Internal transporters: NatsTransporter |
requestTimeout | Number | 0 | Timeout of request in milliseconds. If the request is timed out, broker will throw a RequestTimeout error. Disable: 0 |
requestRetry | Number | 0 | Count of retry of request. If the request is timed out, broker will try to call again. |
cacher | Cacher | null | Instance of cacher. Built-in cachers: MemoryCacher or RedisCacher |
serializer | Serializer | JSONSerializer | Instance of serializer. Built-in serializers: JSON, Avro or MsgPack |
validation | Boolean | false | Enable action parameters validation. |
metrics | Boolean | false | Enable metrics function. |
metricsRate | Number | 1 | Rate of metrics calls. 1 means 100% |
statistics | Boolean | false | Enable broker statistics. Measure the requests count & latencies |
internalActions | Boolean | true | Register internal actions for metrics & statistics functions |
heartbeatInterval | Number | 10 | Interval (seconds) of sending heartbeat |
heartbeatTimeout | Number | 30 | Timeout (seconds) of heartbeat |
ServiceFactory | Class | null | Custom Service class. Broker will use it when creating a service |
ContextFactory | Class | null | Custom Context class. Broker will use it when creating a context at call |
You can call an action by calling the broker.call
method. Broker will search the service (and the node) that has the given action and it will call it. The function returns with a Promise
.
let promise = broker.call(actionName, params, opts);
The actionName
is a dot-separated string. The first part of it is service name. The seconds part of it is action name. So if you have a posts
service which contains a create
action, you need to use posts.create
string as first parameter.
The params
is an object that will be passed to the action as part of the Context.
The opts
is an object. With this, you can set/override some request parameters, e.g.: timeout
, retryCount
.
Available options:
Name | Type | Default | Description |
---|---|---|---|
timeout | Number | requestTimeout of broker | Timeout of request in milliseconds. If the request is timed out and you don't define fallbackResponse , broker will throw a RequestTimeout error. Disable: 0 or null |
retryCount | Number | requestRetry of broker | Count of retry of request. If the request timed out, broker will try to call again. |
fallbackResponse | Any | null | Return with it, if the request is timed out. More info |
// Call without params
broker.call("user.list").then(res => console.log("User list: ", res));
// Call with params
broker.call("user.get", { id: 3 }).then(res => console.log("User: ", res));
// Call with options
broker.call("user.recommendation", { limit: 5 }, { timeout: 500, fallbackResponse: defaultRecommendation })
.then(res => console.log("Result: ", res));
// Call with error handling
broker.call("posts.update", { id: 2, title: "Modified post title" })
.then(res => console.log("Post updated!"))
.catch(err => console.error("Unable to update Post!", err));
If you call action with timeout
and the request is timed out, broker throws a RequestTimeoutError
error.
But if you set fallbackResponse
in calling options, broker won't throw error, instead returns with this given value. It can be an Object
, Array
...etc.
This can be also a Function
, which returns a Promise
. In this case the broker will pass the current Context
to this function as an argument.
Moleculer uses distributed timeouts.In the chained calls the timeout value will be decremented with the elapsed time. If the timeout value is less or equal than 0, next calls will be skipped because the first call is rejected any way.
Broker has an internal event bus. You can send events locally & globally. The local event will be received only by local services of broker. The global event that will be received by all services on all nodes.
You can send event with emit
and emitLocal
functions. First parameter is the name of event. Second parameter is the payload.
// Emit a local event that will be received only by local services
broker.emitLocal("service.started", { service: service, version: 1 });
// Emit a global event that will be received by all nodes.
// The `user` will be serialized to transportation.
broker.emit("user.created", user);
To subscribe for events use the on
or once
methods. Or in Service use the events
property.
In event names you can use wildcards too.
// Subscribe to `user.created` event
broker.on("user.created", user => console.log("User created:", user));
// Subscribe to `user` events
broker.on("user.*", user => console.log("User event:", user));
// Subscribe to all events
broker.on("**", (payload, sender) => console.log(`Event from ${sender || "local"}:`, payload));
To unsubscribe call the off
method.
Broker supports middlewares. You can add your custom middleware, and it'll be called on every local request. The middleware is a Function
that returns a wrapped action handler.
Example middleware from validators modules:
return function validatorMiddleware(handler, action) {
// Wrap a param validator
if (_.isObject(action.params)) {
return ctx => {
this.validate(action.params, ctx.params);
return handler(ctx);
};
}
return handler;
}.bind(this);
The handler
is the request handler of action, what is defined in Service schema. The action
is the action object from Service schema. The middleware should return with the handler
or a new wrapped handler. In this example above, we check whether the action has a params
props. If yes we return a wrapped handler that calls the validator before calling the original handler
.
If there is no params
property we return the original handler
(skip wrapping).
If you don't call the original handler
it will break the request. You can use it in cachers. If you find the data in cache, don't call the handler, instead return the cached data.
Example code from cacher middleware:
return (handler, action) => {
return function cacherMiddleware(ctx) {
const cacheKey = this.getCacheKey(action.name, ctx.params, action.cache.keys);
const content = this.get(cacheKey);
if (content != null) {
// Found in the cache! Don't call handler, return with the context
ctx.cachedResult = true;
return Promise.resolve(content);
}
// Call the handler
return handler(ctx).then(result => {
// Afterwards save the response to the cache
this.set(cacheKey, result);
return result;
});
}.bind(this);
};
The broker registers some internal actions to check the health of node or get request statistics.
This action lists local services.
broker.call("$node.services").then(res => console.log(res));
This action lists local actions
broker.call("$node.actions").then(res => console.log(res));
This actions lists all connected nodes.
broker.call("$node.list").then(res => console.log(res));
This action returns the health info of process & OS.
broker.call("$node.health").then(res => console.log(res));
Example health info:
{
"cpu": {
"load1": 0,
"load5": 0,
"load15": 0,
"cores": 4,
"utilization": 0
},
"mem": {
"free": 1217519616,
"total": 17161699328,
"percent": 7.094400109979598
},
"os": {
"uptime": 366733.2786046,
"type": "Windows_NT",
"release": "6.1.7601",
"hostname": "Developer-PC",
"arch": "x64",
"platform": "win32",
"user": {
"uid": -1,
"gid": -1,
"username": "Developer",
"homedir": "C:\\Users\\Developer",
"shell": null
}
},
"process": {
"pid": 13096,
"memory": {
"rss": 47173632,
"heapTotal": 31006720,
"heapUsed": 22112024
},
"uptime": 25.447
},
"net": {
"ip": [
"192.168.2.100",
"192.168.232.1",
"192.168.130.1",
"192.168.56.1",
"192.168.99.1"
]
},
"time": {
"now": 1487338958409,
"iso": "2017-02-17T13:42:38.409Z",
"utc": "Fri, 17 Feb 2017 13:42:38 GMT"
}
}
This action returns the request statistics if the statistics
is enabled in options.
broker.call("$node.stats").then(res => console.log(res));
Example statistics:
{
"requests": {
// Total statistics
"total": {
// Count of requests
"count": 45,
// Count of error by code
"errors": {},
// Req/sec values
"rps": {
"current": 0.7999854548099126,
// Last x values
"values": [
0,
6.59868026394721,
2.200440088017604
]
},
// Request latency values (ms)
"latency": {
"mean": 0.8863636363636364,
"median": 0,
"90th": 1,
"95th": 5,
"99th": 12,
"99.5th": 12
}
},
// Action-based statistics
"actions": {
"posts.find": {
"count": 4,
"errors": {},
"rps": {
"current": 0.599970001499925,
"values": [
1.7985611510791368,
0.20004000800160032
]
},
"latency": {
"mean": 7.5,
"median": 5,
"90th": 12,
"95th": 12,
"99th": 12,
"99.5th": 12
}
}
}
}
}
The Service is the other main module in the Moleculer. With the help of this you can define actions.
You need to create a schema to define a service. The schema has some main parts (name
, version
, settings
, actions
, methods
, events
).
{
name: "math",
actions: {
add(ctx) {
return Number(ctx.params.a) + Number(ctx.params.b);
},
sub(ctx) {
return Number(ctx.params.a) - Number(ctx.params.b);
}
}
}
The Service has some base properties in the schema.
{
name: "posts",
version: 1
}
The name
is a mandatory property so it must be defined. It's the first part of actionName when you call it with broker.call
.
The version
is an optional property. If you are running multiple version of the same service this needs to be set. It will be a prefix in the actionName.
{
name: "posts",
version: 2,
actions: {
find() {...}
}
}
You need to call the find
action as
broker.call("v2.posts.find");
You can add custom settings to your service under settings
property in schema. You can reach it in the service via this.settings
.
{
name: "mailer",
settings: {
transport: "mailgun"
},
action: {
send(ctx) {
if (this.settings.transport == "mailgun") {
...
}
}
}
}
The actions are the callable/public methods of the service. They can be called with broker.call
method.
The action could be a function (handler) or an object with some properties and with handler
.
The actions should be placed under actions
key in the service schema.
{
name: "math",
actions: {
// Simple definition, only the handler function
add(ctx) {
return Number(ctx.params.a) + Number(ctx.params.b);
},
// Complex definition, set other properties. In this case
// the `handler` function is required!
mult: {
cache: false,
params: {
a: "required|numeric",
b: "required|numeric"
},
handler(ctx) {
// You can reach action params with `ctx.action.*`
if (ctx.action.cache)
return Number(ctx.params.a) * Number(ctx.params.b);
}
}
}
}
You can call these actions as
broker.call("math.add", { a: 5, b: 7 }).then(res => console.log(res));
broker.call("math.mult", { a: 10, b: 31 }).then(res => console.log(res));
Inside the action you can sub-call other actions in other services with ctx.call
method. It is an alias to broker.call
, just set itself as parent context.
{
name: "posts",
actions: {
get(ctx) => {
// Find a post by ID
let post = posts[ctx.params.id];
// Populate the post.author field through "users" service
// Call the "users.get" action with author ID
return ctx.call("users.get", { id: post.author }).then(user => {
if (user) {
// Replace the author ID with the received user object
post.author = user;
}
return post;
})
}
}
}
You can subscribe to events and can define event handlers in the schema under events
key.
{
name: "users",
actions: {
...
},
events: {
// Subscribe to "user.create" event
// Same as you subscribe with `broker.on("user.create", ...)` in the `created()` method
"user.create": function(payload) {
this.logger.info("Create user...");
// Do something
},
// Subscribe to all "user.*" event
"user.*": function(payload, sender, eventName) {
// Do something with payload. The `eventName` contains the original event name. E.g. `user.modified`.
// The `sender` is the nodeID of sender if the event came from remote node. If the event is local, it'll be `undefined`
}
}
}
You can also create private functions in the Service. They are called as methods
. These functions are private, can't be called with broker.call
. But you can call it inside service actions.
{
name: "mailer",
actions: {
send(ctx) {
// Call the `sendMail` method
return this.sendMail(ctx.params.recipients, ctx.params.subject, ctx.params.body);
}
},
methods: {
// Send an email to recipients
sendMail(recipients, subject, body) {
return new Promise((resolve, reject) => {
...
});
}
}
}
The name of method can't be
name
,version
,settings
,schema
,broker
,actions
,logger
, because these words are reserved.
There are some lifecycle service events, that will be triggered by ServiceBroker.
{
name: "www",
actions: {...},
events: {...},
methods: {...},
created() {
// Fired when the service instance created.
},
started() {
// Fired when `broker.start()` called.
}
stopped() {
// Fired when `broker.stop()` called.
}
}
this
In service functions the this
is always binded to the instance of service. It has some properties & methods that you can use in service functions.
Name | Type | Description |
---|---|---|
this.name | String | Name of service from schema |
this.version | Number | Version of service from schema |
this.settings | Object | Settings of service from schema |
this.schema | Object | Schema definition of service |
this.broker | ServiceBroker | Instance of broker |
this.Promise | Promise | Class of Promise (Bluebird) |
this.logger | Logger | Logger module |
this.actions | Object | Actions of service. Service can call its own actions directly. |
There are several ways to create/load a service.
Call the broker.createService
method with the schema of service as argument. You can use this method when developing or testing.
broker.createService({
name: "math",
actions: {
add(ctx) {
return Number(ctx.params.a) + Number(ctx.params.b);
},
sub(ctx) {
return Number(ctx.params.a) - Number(ctx.params.b);
}
}
});
You can place your service code to a single file and load this file with broker.
math.service.js
// Export the schema of service
module.exports = {
name: "math",
actions: {
add(ctx) {
return Number(ctx.params.a) + Number(ctx.params.b);
},
sub(ctx) {
return Number(ctx.params.a) - Number(ctx.params.b);
}
}
}
index.js
// Create broker
let broker = new ServiceBroker();
// Load service
broker.loadService("./math.service");
// Start broker
broker.start();
In the service file you can also be create the instance of Service. In this case you need to export a function that returns the instance of Service.
// Export a function, that the `loadService` will be call it with the instance of ServiceBroker
module.exports = function(broker) {
return new Service(broker, {
name: "math",
actions: {
add(ctx) {
return Number(ctx.params.a) + Number(ctx.params.b);
},
sub(ctx) {
return Number(ctx.params.a) - Number(ctx.params.b);
}
}
});
}
Or create a function that returns with the schema of service
// Export a function, that the `loadService` will be call with the instance of ServiceBroker
module.exports = function() {
let users = [....];
return {
name: "math",
actions: {
create(ctx) {
users.push(ctx.params);
}
}
};
}
You can load multiple services from a folder.
Syntax
broker.loadServices(folder = "./services", fileMask = "*.service.js");
Example
// Load every *.service.js file from the "./services" folder
broker.loadServices();
// Load every *.service.js file from the current folder
broker.loadServices("./");
// Load every user*.service.js file from the "./svc" folder
broker.loadServices("./svc", "user*.service.js");
If you would like to create local properties/variables in service, we recommend to declare them in the created
handler.
Example for local properties
const http = require("http");
// Simple HTTP server service
module.exports = {
name: "www",
settings: {
port: 3000
},
created() {
// Create HTTP server
this.server = http.createServer(this.httpHandler);
},
started() {
// Listening...
this.server.listen(this.settings.port);
},
stopped() {
// Stop server
this.server.close();
},
methods() {
// HTTP handler
httpHandler(req, res) {
res.end("Hello Moleculer!");
}
}
}
When you call an action, the broker creates a Context
instance which contains all request informations and pass to the action handler as argument.
Available properties & methods of Context
:
Name | Type | Description |
---|---|---|
ctx.id | String | Context ID |
ctx.requestID | String | Request ID. If you make sub-calls in a request, it will be the same ID |
ctx.parentID | String | ID of parent context, if it's a sub-call |
ctx.broker | ServiceBroker | Instance of broker |
ctx.action | Object | Instance of action |
ctx.params | Any | Params of request. Second argument of broker.call |
ctx.meta | Any | Metadata of request. It will be transferred in sub-calls |
ctx.nodeID | String | Node ID |
ctx.logger | Logger | Logger module |
ctx.level | Number | Level of request |
ctx.call() | Function | You can make a sub-call. Same arguments like broker.call |
ctx.emit() | Function | Emit an event, like broker.emit |
In Services every modules have a custom logger instance. It is inherited from the broker logger instance and you can set in options of broker. Every modules add a prefix to the log messages. Using that prefix you can identify the module.
let { ServiceBroker } = require("moleculer");
let broker = new ServiceBroker({
logger: console,
logLevel: "info"
});
broker.createService({
name: "posts",
actions: {
get(ctx) {
ctx.logger.info("Log message via Context logger");
}
},
created() {
this.logger.info("Log message via Service logger");
}
});
broker.call("posts.get").then(() => broker.logger.info("Log message via Broker logger"));
Console messages:
[BROKER] posts service registered!
[POSTS-SVC] Log message via Service logger
[CTX] Log message via Context logger
[BROKER] Log message via Broker logger
If you want to change log level you need to set logLevel
in broker options.
Available log levels: fatal
, error
, warn
, info
, debug
, trace
let broker = new ServiceBroker({
logger: console,
logLevel: "warn" // only print the 'warn' & 'error' log entries
});
You can set custom log levels to every module by prefix.
let broker = new ServiceBroker({
logger: console,
logLevel: {
"*": "warn", // global settings
"BROKER": "info", // Broker logger
"CTX": "debug", // Context logger
"CACHER": "warn", // Cacher logger
"TRANSIT": "trace", // Transit logger
"TX": "info", // Transporter logger
"POSTS-SVC": "error" // Service logger. Generated from name of service
"USERS-SVC": false // No logger
}
});
Moleculer has built-in cache solutions. You have to do two things to enable it:
cache: true
in action definition.let { ServiceBroker } = require("moleculer");
let MemoryCacher = require("moleculer").Cachers.Memory;
let broker = new ServiceBroker({
logger: console,
cacher: new MemoryCacher()
});
broker.createService({
name: "users",
// cache: true, // If you enable here, all actions will be cached!
actions: {
list: {
cache: true, // Cache this action
handler(ctx) {
this.logger.info("Handler called!");
return [
{ id: 1, name: "John" },
{ id: 2, name: "Jane" }
]
}
}
}
});
Promise.resolve()
.then(() => {
// Will be called the handler, because the cache is empty
return broker.call("users.list").then(res => console.log("Users count: " + res.length));
})
.then(() => {
// Return from cache, handler won't be called
return broker.call("users.list").then(res => console.log("Users count from cache: " + res.length));
});
Console messages:
[BROKER] users service registered!
[USERS-SVC] Handler called!
Users count: 2
Users count from cache: 2
The cacher creates keys by service name, action name, and hash of params of context. The key syntax is
<actionName>:<parameters or hash of parameters>
So if you call the posts.list
action with params { limit: 5, offset: 20 }
, the cacher calculate a hash from the params. So next time if you call this action with the same params, it will find in the cache by key.
// Hashed cache key for "posts.find" action
posts.find:0d6bcb785d1ae84c8c20203401460341b84eb8b968cffcf919a9904bb1fbc29a
However the hash calculation is an expensive operation. But you can specify which parameters you want to use for caching. In this case you need to set an object for cache
property that contains the list of parameters.
{
name: "posts",
actions: {
list: {
cache: {
// Only generate cache by from "limit" and "offset" param values
keys: ["limit", "offset"]
},
handler(ctx) {
return this.getList(ctx.params.limit, ctx.params.offset);
}
}
}
}
// If params is { limit: 10, offset: 30 }, the cache will be:
// posts.list:10-30
This solution is faster, so we recommend to use it in production environment.
You can also use the cacher manually. Just call the get
, set
, del
methods of broker.cacher
.
// Save to cache
broker.cacher.set("mykey", { a: 5 });
// Get from cache (Please note! Some cacher maybe returns with Promise)
let obj = broker.cacher.get("mykey", { a: 5 });
// Remove entry from cache
broker.cacher.del("mykey");
// Clean all entries
broker.cacher.clean();
When you create a new model in your service, sometimes you have to clear the old cache entries. For this purpose there are internal events. When an event like this is fired, the cacher will clean the cache.
{
name: "users",
actions: {
create(ctx) {
// Create new user
let user = new User(ctx.params);
// Clear all cache entries
ctx.emit("cache.clean");
// Clear all cache entries which keys start with `users.`
ctx.emit("cache.clean", "users.*");
// Clear multiple cache entries
ctx.emit("cache.clean", [ "users.*", "posts.*" ]);
// Delete only one entry
ctx.emit("cache.del", "users.list");
// Delete multiple entries
ctx.emit("cache.del", [ "users.model:5", "users.model:8" ]);
}
}
}
MemoryCacher
is a built-in memory cache module.
let MemoryCacher = require("moleculer").Cachers.Memory;
let broker = new ServiceBroker({
cacher: new MemoryCacher({
ttl: 30 // Time-to-live is 30sec. Disabled: 0 or null
})
});
RedisCacher
is a built-in Redis based cache module. It uses ioredis
client.
let RedisCacher = require("moleculer").Cachers.Redis;
let broker = new ServiceBroker({
cacher: new RedisCacher({
ttl: 30, // Time-to-live is 30sec. Disabled: 0 or null
prefix: "SERVICER" // Prefix for cache keys
monitor: false // Turn on/off Redis client monitoring. Will be logged (on debug level) every client operations.
// Redis settings, pass to the `new Redis()` constructor
redis: {
host: "redis",
port: 6379,
password: "1234",
db: 0
}
})
});
You can also create your custom cache module. We recommend you to copy the source of MemoryCacher
or RedisCacher
and implement the get
, set
, del
and clean
methods.
Transporter is an important module if you are running services on more nodes. Transporter communicates with every nodes. Send events, call requests...etc.
Moleculer has a built-in transporter for NATS.
NATS Server is a simple, high performance open source messaging system for cloud native applications, IoT messaging, and microservices architectures.
let { ServiceBroker } = require("moleculer");
let NatsTransporter = require("moleculer").Transporters.NATS;
let broker = new ServiceBroker({
nodeID: "server-1",
transporter: new NatsTransporter(),
requestTimeout: 5 * 1000
});
You can pass options to nats.connect()
method.
// Connect to 'nats://localhost:4222'
new NatsTransporter();
// Connect to remote server
new NatsTransporter("nats://nats.server:4222");
// Connect to remote server and change the prefix
new NatsTransporter({
nats: {
url: "nats://nats-server:4222",
},
prefix: "MY-PREFIX" // Use for channel names at subscribe & publish. Default: "MOL"
});
// Connect to remote server with user & pass
new NatsTransporter({
nats: {
url: "nats://nats-server:4222",
user: "admin",
pass: "1234"
}
});
Moleculer has a built-in transporter for Redis.
let { ServiceBroker } = require("moleculer");
let RedisTransporter = require("moleculer").Transporters.Redis;
let broker = new ServiceBroker({
nodeID: "server-1",
transporter: new RedisTransporter(),
requestTimeout: 5 * 1000
});
You can pass options to new Redis()
method.
// Connect to 'redis://localhost:4222'
new RedisTransporter();
// Connect to remote server
new RedisTransporter("redis://redis.server:4222");
// Connect to remote server and change the prefix
new RedisTransporter({
redis: {
url: "redis://redis-server:4222",
},
prefix: "MY-PREFIX" // Use for channel names at subscribe & publish. Default: "MOL"
});
// Connect to remote server with user & pass
new RedisTransporter({
redis: {
url: "redis://redis-server:4222",
user: "admin",
pass: "1234"
}
});
Moleculer has a built-in transporter for MQTT protocol (e.g.: Mosquitto).
let { ServiceBroker } = require("moleculer");
let MqttTransporter = require("moleculer").Transporters.MQTT;
let broker = new ServiceBroker({
nodeID: "server-1",
transporter: new MqttTransporter(),
requestTimeout: 5 * 1000
});
You can pass options to mqtt.connect()
method.
// Connect to 'mqtt://localhost:4222'
new MqttTransporter();
// Connect to remote server
new MqttTransporter("mqtt://mqtt.server:4222");
// Connect to remote server and change the prefix
new MqttTransporter({
mqtt: {
url: "mqtt://mqtt-server:4222",
},
prefix: "MY-PREFIX" // Use for channel names at subscribe & publish. Default: "MOL"
});
// Connect to remote server with user & pass
new MqttTransporter({
mqtt: {
url: "mqtt://mqtt-server:4222",
user: "admin",
pass: "1234"
}
});
You can also create your custom transporter module. We recommend you that copy the source of NatsTransporter
and implement the connect
, disconnect
, subscribe
and publish
methods.
For transportation needs a serializer module which serialize & deserialize the transferred packets. If you don't set serializer, the default is the JSON serializer.
let { ServiceBroker } = require("moleculer");
let NatsTransporter = require("moleculer").Transporters.NATS;
let AvroSerializer = require("moleculer").Serializers.Avro;
let broker = new ServiceBroker({
nodeID: "server-1",
transporter: new NatsTransporter(),
serializer: new AvroSerializer()
});
This is the default serializer. Serialize the packets to JSON string and deserialize the received data to packet.
let broker = new ServiceBroker({
...
// serializer: new JSONSerializer() // don't need to set, because it is the default
});
This is an Avro serializer.
let AvroSerializer = require("moleculer").Serializers.Avro;
let broker = new ServiceBroker({
...
serializer: new AvroSerializer()
});
This is an MsgPack serializer.
let MsgPackSerializer = require("moleculer").Serializers.MsgPack;
let broker = new ServiceBroker({
...
serializer: new MsgPackSerializer()
});
This is a Protocol Buffer serializer.
let ProtoBufSerializer = require("moleculer").Serializers.ProtoBuf;
let broker = new ServiceBroker({
...
serializer: new ProtoBufSerializer()
});
You can also create your custom serializer module. We recommend you that copy the source of JSONSerializer and implement the serialize
and deserialize
methods.
Moleculer has a metrics function. You can turn on it in the broker options with metrics: true
property.
If enabled, the broker emits metrics events in every broker.call
. You can catch this events and transfer to your Tracer system (ZipKin, OpenTracing...etc)
The broker emit an metrics.trace.span.start
event when a new call/request started.
The payload contains the following values:
{
// Context ID
id: '4563b09f-04cf-4891-bc2c-f26f80c3f91e',
// Request ID
requestID: null,
// Level of call
level: 1,
// Start time
startTime: 1493903164726,
// Is it a remote call
remoteCall: false,
// Called action
action: {
name: 'v2.users.get'
}
// Node ID of caller
nodeID: "node-1",
// Target nodeID if it's a remote call
//targetNodeID: "node-2",
// Parent context ID if it is a sub-call
//parent: null
}
The broker emit an metrics.trace.span.finish
when a call/request finished.
The payload contains the following values:
{
// Context ID
id: '4563b09f-04cf-4891-bc2c-f26f80c3f91e',
// Request ID
requestID: null,
// Level of call
level: 1,
// Start time
startTime: 1493903164726,
// End time
endTime: 1493903164731.3684,
// Duration
duration: 5.368304,
// Is it a remote call
remoteCall: false,
// Is it resolved from cache
fromCache: false,
// Called action
action: {
name: 'v2.users.get'
}
// Node ID of caller
nodeID: "node-1",
// Target nodeID if it's a remote call
//targetNodeID: "node-2",
// Parent context ID if it is a sub-call
//parent: null
// Error if the call returned with error
error: {
name: "ValidationError",
message: "Invalid incoming parameters"
}
}
Moleculer has a statistics module that collects and aggregates the count & latency info of the requests.
You can enable it in broker options with statistics: true
property. You can access the statistics data via $node.stats
action.
Moleculer supports several architectures.
In this version you are running every services on one node with one broker. In this case every service can call other services locally. So there is no network latency and no transporter. The local call is the fastest.
This is the well-known microservices architecture when every service running on individual nodes and communicates others via transporter.
In this case we are running coherent services on the same node. It is combine the advantages of monolith and microservices architectures.
For example, if the posts
service calls a lot of times the users
service, we put them together, that we cut down the network latency between services. If this node is overloaded, we will add replicas.
Under development we are measuring every important parts of the framework that we can ensure the best performance.
$ npm test
or in development
$ npm run ci
Please send pull requests improving the usage and fixing bugs, improving documentation and providing better examples, or providing some testing, because these things are important.
Moleculer is available under the MIT license.
Copyright (c) 2017 Ice-Services
FAQs
Fast & powerful microservices framework for Node.JS
The npm package moleculer receives a total of 28,431 weekly downloads. As such, moleculer popularity was classified as popular.
We found that moleculer demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 3 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
Deno 2.2 enhances Node.js compatibility, improves dependency management, adds OpenTelemetry support, and expands linting and task automation for developers.
Security News
React's CRA deprecation announcement sparked community criticism over framework recommendations, leading to quick updates acknowledging build tools like Vite as valid alternatives.
Security News
Ransomware payment rates hit an all-time low in 2024 as law enforcement crackdowns, stronger defenses, and shifting policies make attacks riskier and less profitable.