Skip to content

dansteingart/nodeforwarder

Repository files navigation

Node Forwarder

A simple serialport to RESTful interface.

Overview

Serial ports, bless their robust and simple yet legacy bound hearts, are one to one. Once a program connects to a serial port, without 1337 skills, only one program can attach to it. This is not inherently good nor bad, it just is.

For debugging hardware, and for having a second input for control loops, making the serial port one-to-many is useful. So let's go.

nodeforwarder connects to your serialport to an http port of your specification, and allows you to read from the serial by points your browser (or better yet cURL like function) to

http://localhost:PORT/read/

and you can write by going to

http://localhost:PORT/write/YOUR STRING HERE

or

http://localhost:PORT/writecf/YOUR STRING HERE

if you want to append a \r\n

Of course, if your computadora is attached to the interwebs, your ports will be accessible to anyone that your firewall allows via http://YOUR_IP_OR_WHATEVERDOTCOM:PORT/

This is also not inherently good or bad, it just is.

Required Libraries

Just npm i, but FYI

  • serialport -> npm install serialport
  • express -> npm install express
  • sleep -> npm install sleep
  • socket.io -> npm install socket.io
  • cors -> npm install cors
  • body-parser -> npm install body-parser

Quick Run

In the directory where you placed the files run

  • on first run npm i
  • then and thereafter node nodeforwader.js iPORT pPORT SERIALSPEED BUFF

where

  • iPORT = the internet port (e.g. localhost:8000 would be 8000).

    • You can make this whatever you want, just make sure there's nothing else trying to run at that port (e.g. pithy, or another forwarder). If you try to start a forwarder where there's a port in use you'll just get an error, so try another port. Generally, ports under 1000 are reserved for system use, you can start those but probably have to sudo your way in. If you don't know what that means don't worry about it
  • pPORT = the location of the serial port.

    • on a linux box this looks like /dev/ttyUSB*, where the * is a number in the order the devices were plugged in currently.
    • on a mac this looks like /dev/tty.usb* is typically a generated string depending on what USB port you plugged into. Note that it may not look like /dev/tty.usb*, but use your Princeton brain and play with the problem.
    • on a Windows box this will be COMX, where X is again the order the device was plugged in on that computer. Note that this is persistent, so if something is COM4 it is always COM4
  • SERIALSPEED = the baud rate for the device on the serial port.

    • If you coded it yourself in arduino, it's the same as Serial.begin(SERIALSPEED), otherwise it is according to the manufacturer's instructions. Good guesses are9600, 57600 and 115200.
  • BUFF = the number of characters to buffer from the serial port

    • 10000 is typically a safe value
    • NB: This buffer lets data persist, but it does not tell you whether data is stale or not, e.g. the system can get into places where the serial port device bonks but the forwarder doesn't crash, and when you read data you'll always see the last message passed. At some point I'll figure out a simple "last message received" check.

phew. not so quick. but all you have to write is something like this

node nodeforwader.js 9000 /dev/ttyUSB0 57600 10000

Using the Nodeforwarder

Let's say you chose an iPORT of 9000. On the computer where you started the forwarder you can browse to that page via

http://localhost:9000.

You should see a basic console.

To write data to the serial port simply, in the browser url, type

http://localhost:9000/write/FOOBAR

and FOOBAR will be sent to the serial device.

If you need to send a \n with string use

http://localhost:9000/writecf/FOOBAR

To read what comes back, type

http://localhost:9000/read and the current buffer contents will display in the browser.

Get URLS

  • /write/[PAYLOAD]
    • Writes PAYLOAD to the serialport. Tries hard to handle what it can, but it's just a url so....
  • /writecf/[PAYLOAD]
    • Writes PAYLOAD with an appended '\r\n' to the serial port
  • /read
    • returns the current entire buffer from the serial port. You'll want to parse it out
  • /lastreading
    • returns the last time the nodeforwarder received data from the serialport, in unixtime (ms).

Post URLS

  • /write
    • You can use whatever you like here and we'll do our best to get the message exactly as formatted to the device. Want to use ArduinoJson? You can!
curl -s -H 'Content-Type: application/json' -X POST -d '{"payload":"love_is_love"}' \ 
http://HOST:PORT/write/

socket.io

If you want to digest the data as it's coming through the serial port without polling per above, you can use socket.io directly in your language of choice. Example below in python

import socketio

sio = socketio.Client()

@sio.event
def connect():
    print('connection established')

@sio.event
def my_message(data):
    print('message received with ', data)
    #sio.emit('my response', {'response': 'my response'})

@sio.on('data')
def data(data):
    print(data,end="")
    #sio.emit('my response', {'response': 'my response'})

@sio.event
def disconnect():
    print('disconnected from server')

sio.connect('http://localhost:9000')
sio.wait()

API, sorta

Now that the forwarder is set up and you know it's working per above, you can use it in whatever language you want so long as that language can deal with opening and reading from URLs (they all can). In python this looks like

from requests import urlopen as uo

import requests as r
from time import time, sleep


out = {}
data = {'res1':22000}
out = {'payload'=data}
r.post("http://localhost:9000/write",data=out) 


out = {}
data = {}
data['mode'] = "manual"
data['dac0'] = 3000
data['dac1'] = 1000
out = {'payload'=data}

r.post("http://localhost:9000/write",data=out) 
## your code here

Updates

  • 2021-12-27

    • POST calls now behaving (I hope)
    • Update README
  • 2021-10-17

    • Improved display in readout.html
    • Update README
  • 2021-10-16

    • Updated to work with Node 16 (some changes to how express and socket.io behave)
    • Console actually works 😅
    • Sorely needed README update

About

serialport to RESTful interface

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published