npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

modbus-serial-ml

v1.2.2

Published

A pure JavaScript implemetation of MODBUS-RTU (and TCP) master for NodeJS !!! this is a forked version !!! in the original version, the parser only check and parse one piece of data, which means an error will occur if the data is emitted more than once.

Downloads

14

Readme

modbus-serial master

A pure JavaScript implemetation of MODBUS-RTU (and TCP) master for NodeJS !!! this is a forked version !!! in the original version, the parser only check and parse one piece of data, which means an error will occur if the data is emitted more than once.

in this version, a frame will be parsed only when the received data bytes is more than expected. timeout is used to avoid data loss;

npm Build Status

This class makes ModbusRTU (and TCP) calls fun and easy.

Modbus is a serial communications protocol, first used in 1979. Modbus is simple and robust, openly published, royalty-free and easy to deploy and maintain.



Install

npm install modbus-serial

For use over serial port (ModbusRTU), also install node-serialport:

npm install serialport

What can I do with this module ?

This class makes it fun and easy to communicate with electronic devices such as irrigation controllers, protocol droids and robots. It talks with devices that use a serial line (e.g. RS485, RS232). Many industrial electronic devices implement modbus. Arduino can also talk modbus and you can control your projects and robots using modbus.

Arduino libraries for modbus slave:

  • https://github.com/yaacov/arduino-modbus-slave
  • https://github.com/smarmengol/Modbus-Master-Slave-for-Arduino

Arduino sketch for irrigation timer with modbus support:

  • https://github.com/yaacov/arduino-irrigation-timer

Node Modbus-WebSocket bridge:

  • https://github.com/yaacov/node-modbus-ws

Compatibility

This class implements:
  • FC1 "Read Coil Status"
  • FC2 "Read Input Status"
  • FC3 "Read Holding Registers"
  • FC4 "Read Input Registers"
  • FC5 "Force Single Coil"
  • FC6 "Preset Single Register"
  • FC15 "Force Multiple Coil"
  • FC16 "Preset Multiple Registers"
Connects types:
  • modbus-RTU (modbus-rtu): Over serial line [require node serialport].
  • modbus-ASCII (modbus-ascii): Over serial line [require node serialport].
  • modbus-TCP (modbus-tcp): Over TCP/IP line.
  • modbus-RTU (telnet): Over Telnet server, TCP/IP serial bridge.
  • modbus-RTU (buffered): Over buffered serial line [require node serialport].
  • modbus-RTU (C701): Over C701 server, commercial UDP to serial bridge.

Examples


Logger
// create an empty modbus client
var ModbusRTU = require("modbus-serial");
var client = new ModbusRTU();

// open connection to a tcp line
client.connectTCP("192.168.1.42", run);

// read the values of 10 registers starting at address 0
// on device number 1. and log the values to the console.
function run() {
    client.setID(1);

    client.readInputRegisters(0, 10)
        .then(console.log)
        .then(run);
}

Read and Write
// create an empty modbus client
var ModbusRTU = require("modbus-serial");
var client = new ModbusRTU();

// open connection to a serial port
client.connectRTU("/dev/ttyUSB0", {baudrate: 9600}, write);

function write() {
    client.setID(1);

    // write the values 0, 0xffff to registers starting at address 5
    // on device number 1.
    client.writeRegisters(5, [0 , 0xffff])
        .then(read);
}

function read() {
    // read the 2 registers starting at address 5
    // on device number 1.
    client.readHoldingRegisters(5, 2)
        .then(console.log);
}

Logger
// create an empty modbus client
var ModbusRTU = require("modbus-serial");
var client = new ModbusRTU();

// open connection to a serial port
client.connectRTU("/dev/ttyUSB0", {baudrate: 9600});
client.setID(1);

// read the values of 10 registers starting at address 0
// on device number 1. and log the values to the console.
setInterval(function() {
    client.readHoldingRegisters(0, 10, function(err, data) {
        console.log(data.data);
    });
}, 1000);

Logger-TCP
// create an empty modbus client
var ModbusRTU = require("modbus-serial");
var client = new ModbusRTU();

// open connection to a tcp line
client.connectTCP("192.168.1.42");
client.setID(1);

// read the values of 10 registers starting at address 0
// on device number 1. and log the values to the console.
setInterval(function() {
    client.readHoldingRegisters(0, 10, function(err, data) {
        console.log(data.data);
    });
}, 1000);

Read raw buffer
// create an empty modbus client
var ModbusRTU = require("modbus-serial");
var client = new ModbusRTU();

// open connection to a serial port
client.connectRTU("/dev/ttyUSB0", {baudrate: 9600}, run);

function run() {
    client.setID(1);

    // read 2 16bit-registers to get one 32bit number
    client.readInputRegisters(5, 2, function(err, data) {
        var int32 = data.buffer.readUInt32BE();
        console.log(int32);
    });
}

Methods


API promises

This communication functions use a pre-set unit-id and can return a promise, Using callbacks is optional.

// set the client's unit id
client.setID(1);

// set a timout for requests default is null (no timeout)
client.setTimeout(1000);

// read 8 discrete inputs starting at input 10
// (function use the unit id 1, we set earlier)
client.readDiscreteInputs(10, 8)
    .then(function(data) {
        console.log(data);
    });

.setID(id)
  Sets the unit id

id {number}: The new client id


.getID()
  Returns the unit id

.setTimeout(duration)
  Sets a timeout for the request

duration {number}: Duration of the timeout


.getTimeout()
  Returns the timeout for the request

.readCoils (address, length)
  Writes "Read Coils" (FC=1) request to serial port.

address {number}: The Data Address of the first register.

length {number}: The total number of registers requested.


.readDiscreteInputs (address, length)
  Writes "Read Discrete Inputs" (FC=2) request to serial port.

address {number}: The Data Address of the first register.

length {number}: The total number of registers requested.


.readHoldingRegisters (address, length)
  Writes "Read Holding Registers" (FC=3) request to serial port.

address {number}: The Data Address of the first register.

length {number}: The total number of registers requested.


.readInputRegisters (address, length)
  Writes "Read Input Registers" (FC=4) request to serial port.

address {number}: The Data Address of the first register.

length {number}: The total number of registers requested.


.writeCoil(address, state)
  Writes "Force Coil Status" (FC=5) request to serial port.

address {number}: The Data Address of the first register.

state {boolean}: The state to force into coil.


.writeCoils(address, array)
  Writes "Force Multiple Coils" (FC=15) request to serial port.

address {number}: The Data Address of the first register.

array {array}: The array of states to force into the coils.


.writeRegisters (address, array)
  Writes "Preset Multiple Registers" (FC=16) request to serial port.

address {number}: The Data Address of the first register.

array {array}: The array of values to set into the registers.



.writeRegister (address, value)
  Writes "Preset Single Register" (FC=6) request to serial port.

address {number}: The Data Address of the first register.

value {number}: The value to set into the register.


API Callbacks

This communication functions use callbacks.


// read 8 holding registers starting at register 10
// (function use the unit id 1)
client.writeFC3(1, 10, 8, function(err, data) {
      if (err) {
            console.log(err);
      } else {
            console.log(data);
      });

.open(callback)
  Opens a modbus connection using the given serial port.

callback {function}: (optional) Called when a connection has been opened.


.close(callback)
  Closes a modbus connection using the given serial port.

callback {function}: (optional) Called when a connection has been closed.


.writeFC1 (unit, address, length, callback)
  Writes "Read coil status" (FC=01) request to serial port.

unit {number}: The slave unit address.

address {number}: The Data Address of the first register.

length {number}: The total number of registers requested.

callback {function}: (optional) Called once the unit returns an answer. The callback should be a function that looks like: function (error, data) { ... }

error - null on success, error string o/w
data - an object with two fildes:
    data.data: array of boolean coils (in multiples of 8 = one byte).
    data.buffer: raw baffer of bytes returned by slave.

.writeFC2 (unit, address, length, callback)
  Writes "Read input status" (FC=02) request to serial port.

unit {number}: The slave unit address.

address {number}: The Data Address of the first register.

length {number}: The total number of registers requested.

callback {function}: (optional) Called once the unit returns an answer. The callback should be a function that looks like: function (error, data) { ... }

error - null on success, error string o/w
data - an object with two fildes:
    data.data: array of boolean digital inputs (in multiples of 8 = one byte).
    data.buffer: raw baffer of bytes returned by slave.

.writeFC3 (unit, address, length, callback)
  Writes "Read Holding Registers" (FC=03) request to serial port.

unit {number}: The slave unit address.

address {number}: The Data Address of the first register.

length {number}: The total number of registers requested.

callback {function}: (optional) Called once the unit returns an answer. The callback should be a function that looks like: function (error, data) { ... }

error - null on success, error string o/w
data - an object with two fildes:
    data.data: array of unsinged 16 bit registers.
    data.buffer: raw baffer of bytes returned by slave.

.writeFC4 (unit, address, length, callback)
  Writes "Read Input Registers" (FC=04) request to serial port.

unit {number}: The slave unit address.

address {number}: The Data Address of the first register.

length {number}: The total number of registers requested.

callback {function}: (optional) Called once the unit returns an answer. The callback should be a function that looks like: function (error, data) { ... }

error - null on success, error string o/w
data - an object with two fildes:
    data.data: array of unsinged 16 bit registers.
    data.buffer: raw baffer of bytes returned by slave.

.writeFC5 (unit, address, state, callback)
  Writes "Force Single Coil" (FC=05) request to serial port.

unit {number}: The slave unit address.

address {number}: The Data Address of the first register.

state {boolean}: The coil state.

callback {function}: (optional) Called once the unit returns an answer. The callback should be a function that looks like: function (error, data) { ... }


.writeFC15 (unit, address, array, callback)
  Writes "Force Multiple Coils" (FC=15) request to serial port.

unit {number}: The slave unit address.

address {number}: The Data Address of the first register.

array {array}: The array of states to send to unit.

callback {function}: (optional) Called once the unit returns an answer. The callback should be a function that looks like: function (error, data) { ... }


.writeFC6 (unit, address, value, callback)
  Writes "Preset Single Register" (FC=6) request to serial port.

unit {number}: The slave unit address.

address {number}: The Data Address of the first register.

value {number}: The value to sent to unit.

callback {function}: (optional) Called once the unit returns an answer. The callback should be a function that looks like: function (error, data) { ... }


.writeFC16 (unit, address, array, callback)
  Writes "Preset Multiple Registers" (FC=16) request to serial port.

unit {number}: The slave unit address.

address {number}: The Data Address of the first register.

array {array}: The array of values to sent to unit.

callback {function}: (optional) Called once the unit returns an answer. The callback should be a function that looks like: function (error, data) { ... }


API connection shorthand

The shorthand connection functions creates a port and open it.

Long way, without shorthand:

// open a serial port
var SerialPort = require("serialport");
var serialPort = new SerialPort("/dev/ttyUSB0", {baudrate: 9600, autoOpen: false});

// create a modbus client using the serial port
var ModbusRTU = require("modbus-serial");
var client = new ModbusRTU(serialPort);

// open connection to a serial port
client.open();

// tell your coffee machine to do something ...

Using shorthand:

// create an empty modbus client
var ModbusRTU = require("modbus-serial");
var client = new ModbusRTU();

// open connection to a serial port
client.connectRTU("/dev/ttyUSB0", {baudrate: 9600});

// tell your robot to do something ...

Using shorthand (TCP):

// create an empty modbus client
var ModbusRTU = require("modbus-serial");
var client = new ModbusRTU();

// open connection to a tcp line
client.connectTCP("192.168.1.42");

// tell your robot to do something ...

.connectRTU (path, options, callback)
  Connect using serial port.

path {string}: The port path (e.g. "/dev/ttyS0")

options {object}: (optional) The options for this connection.

callback {function}: (optional) Called once the client is connected.


.connectRTUBuffered (path, options, callback)
  Connect using buffered serial port.
  Use when serial port has long delays inside packets.

path {string}: The port path (e.g. "/dev/ttyS0")

options {object}: (optional) The options for this connection.

callback {function}: (optional) Called once the client is connected.


.connectTCP (ip, options, callback)
  Connect using tcp/ip.

ip {string}: The port ip (e.g. "24.230.1.42")

options {object}: (optional) The options for this connection.

callback {function}: (optional) Called once the client is connected.


.connectTelnet (ip, options, callback)
  Connect using a telnet server

ip {string}: The port ip (e.g. "24.230.1.42")

options {object}: (optional) The options for this connection.

callback {function}: (optional) Called once the client is connected.


.connectAsciiSerial (path, options, callback)
  Connect using serial port with ASCII encoding.

path {string}: The port path (e.g. "/dev/ttyS0")

options {object}: (optional) The options for this connection.

callback {function}: (optional) Called once the client is connected.