Initial code
Signed-off-by: Julien Riou <julien@riou.xyz>
This commit is contained in:
parent
708f9faead
commit
53758081d8
14 changed files with 9071 additions and 0 deletions
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
|
@ -0,0 +1 @@
|
|||
serial2mqtt.ini
|
32
.pre-commit-config.yaml
Normal file
32
.pre-commit-config.yaml
Normal file
|
@ -0,0 +1,32 @@
|
|||
---
|
||||
repos:
|
||||
- repo: https://github.com/pre-commit/pre-commit-hooks
|
||||
rev: master
|
||||
hooks:
|
||||
- id: check-executables-have-shebangs
|
||||
- id: check-merge-conflict
|
||||
- id: double-quote-string-fixer
|
||||
- id: end-of-file-fixer
|
||||
- id: fix-encoding-pragma
|
||||
args: ['--remove']
|
||||
- id: requirements-txt-fixer
|
||||
- id: trailing-whitespace
|
||||
- repo: https://gitlab.com/pycqa/flake8
|
||||
rev: master
|
||||
hooks:
|
||||
- id: flake8
|
||||
args: ['--max-line-length=120']
|
||||
- repo: https://github.com/FalconSocial/pre-commit-python-sorter
|
||||
rev: master
|
||||
hooks:
|
||||
- id: python-import-sorter
|
||||
args: ['--silent-overwrite']
|
||||
- repo: https://github.com/chewse/pre-commit-mirrors-pydocstyle
|
||||
rev: master
|
||||
hooks:
|
||||
- id: pydocstyle
|
||||
args: ['--config=.pydocstyle', '--match="(?!test_).*\.py"']
|
||||
- repo: https://github.com/jumanjihouse/pre-commit-hooks
|
||||
rev: master
|
||||
hooks:
|
||||
- id: shfmt
|
2
.pydocstyle
Normal file
2
.pydocstyle
Normal file
|
@ -0,0 +1,2 @@
|
|||
[pydocstyle]
|
||||
ignore = D100,D104,D400,D203,D204,D101,D213,D202
|
33
README.md
Normal file
33
README.md
Normal file
|
@ -0,0 +1,33 @@
|
|||
# arduino-sensors-toolkit
|
||||
|
||||
This toolkit has been designed to be used with a DHT22 (humidity, temperature) and KY-037 (sound) sensors connected to
|
||||
an Arduino microcontroller. It is part of a self hosting project with specific needs. Feel free to fork this repository
|
||||
and update code for your needs.
|
||||
|
||||
## sensors2serial
|
||||
|
||||
Arduino circuit:
|
||||
|
||||

|
||||
|
||||
## serial2mqtt
|
||||
|
||||
### Requirements
|
||||
|
||||
- serial interface using sensors2serial code (ex: /dev/cuaU0 for FreeBSD, /dev/ttyACM0 for Debian)
|
||||
- MQTT broker (ex: [Mosquitto](https://www.digitalocean.com/community/tutorials/how-to-install-and-secure-the-mosquitto-mqtt-messaging-broker-on-debian-10))
|
||||
|
||||
### Installation
|
||||
|
||||
- Installation on [Debian](debian/README.md)
|
||||
- Installation on [FreeBSD](freebsd/README.md)
|
||||
|
||||
### Configuration
|
||||
|
||||
An optional configuration file can be used with `--config <file>`. See [serial2mqtt.ini.example](serial2mqtt.ini.example).
|
||||
|
||||
### Usage
|
||||
|
||||
```
|
||||
python3 serial2mqtt.py --help
|
||||
```
|
22
debian/README.md
vendored
Normal file
22
debian/README.md
vendored
Normal file
|
@ -0,0 +1,22 @@
|
|||
# Installation on Debian
|
||||
|
||||
Install the following packages:
|
||||
- python3-serial
|
||||
- python3-paho-mqtt
|
||||
|
||||
Then git clone this repository to **/opt/arduino-sensors-toolkit**.
|
||||
|
||||
Create the *serial2mqtt* system user and add it to the *dialout* group to read the serial interface:
|
||||
```
|
||||
# adduser --system --disabled-password --disabled-login --home /var/lib/serial2mqtt \
|
||||
--no-create-home --quiet --force-badname --group serial2mqtt
|
||||
# usermod -a -G dialout serial2mqtt
|
||||
```
|
||||
|
||||
Copy the [serial2mqtt.service](serial2mqtt.service) file to **/etc/systemd/system/serial2mqtt.service**.
|
||||
|
||||
Copy the [serial2mqtt.default](serial2mqtt.default) file to **/etc/default/serial2mqtt**.
|
||||
|
||||
Reload systemd configuration with `systemctl daemon-reload`.
|
||||
|
||||
Enable service with `systemctl enable serial2mqtt.service`.
|
2
debian/serial2mqtt.default
vendored
Normal file
2
debian/serial2mqtt.default
vendored
Normal file
|
@ -0,0 +1,2 @@
|
|||
# Options passed as daemon argument
|
||||
#DAEMON_OPTS="-c /etc/serial2mqtt.ini -v"
|
16
debian/serial2mqtt.service
vendored
Normal file
16
debian/serial2mqtt.service
vendored
Normal file
|
@ -0,0 +1,16 @@
|
|||
[Unit]
|
||||
Description=Read serial port and send sensors measurements to MQTT broker
|
||||
After=syslog.target network.target
|
||||
|
||||
[Service]
|
||||
Type=simple
|
||||
User=serial2mqtt
|
||||
Group=serial2mqtt
|
||||
EnvironmentFile=-/etc/default/serial2mqtt
|
||||
ExecStart=/opt/arduino-sensors-toolkit/serial2mqtt.py $DAEMON_OPTS
|
||||
KillMode=process
|
||||
TimeoutSec=30
|
||||
Restart=no
|
||||
|
||||
[Install]
|
||||
WantedBy=multi-user.target
|
20
freebsd/README.md
Normal file
20
freebsd/README.md
Normal file
|
@ -0,0 +1,20 @@
|
|||
# Installation on FreeBSD
|
||||
|
||||
Install the following packages:
|
||||
- py37-pyserial
|
||||
- py37-paho-mqtt
|
||||
|
||||
Then git clone this repository to **/usr/local/share/arduino-sensors-toolkit**.
|
||||
|
||||
Create the system user:
|
||||
|
||||
```
|
||||
pw user add -n serial2mqtt -s /usr/sbin/nologin -G dialer
|
||||
```
|
||||
|
||||
Copy [serial2mqtt.rc](serial2mqtt.rc) script to **/usr/local/etc/rc.d/serial2mqtt**.
|
||||
|
||||
Enable service:
|
||||
```
|
||||
echo 'serial2mqtt_enable="YES"' >> /etc/rc.conf
|
||||
```
|
65
freebsd/serial2mqtt.rc
Executable file
65
freebsd/serial2mqtt.rc
Executable file
|
@ -0,0 +1,65 @@
|
|||
#!/bin/sh
|
||||
# PROVIDE: serial2mqtt
|
||||
# REQUIRE: DAEMON NETWORKING
|
||||
# BEFORE: LOGIN
|
||||
# KEYWORD: shutdown
|
||||
|
||||
# serial2mqtt_enable (bool): Set to YES to enable serial2mqtt
|
||||
# Default: NO
|
||||
# serial2mqtt_conf (str): serial2mqtt configuration file
|
||||
# Default: ${PREFIX}/etc/serial2mqtt.ini
|
||||
# serial2mqtt_user (str): serial2mqtt daemon user
|
||||
# Default: serial2mqtt
|
||||
# serial2mqtt_group (str): serial2mqtt daemon group
|
||||
# Default: serial2mqtt
|
||||
# serial2mqtt_flags (str): Extra flags passed to serial2mqtt
|
||||
# Default: none
|
||||
|
||||
. /etc/rc.subr
|
||||
|
||||
PATH=${PATH}:/usr/local/sbin:/usr/local/bin
|
||||
|
||||
name="serial2mqtt"
|
||||
rcvar=serial2mqtt_enable
|
||||
load_rc_config $name
|
||||
|
||||
: ${serial2mqtt_enable:="NO"}
|
||||
: ${serial2mqtt_user:="serial2mqtt"}
|
||||
: ${serial2mqtt_group:="serial2mqtt"}
|
||||
: ${serial2mqtt_flags:=""}
|
||||
: ${serial2mqtt_conf:="/usr/local/etc/${name}.ini"}
|
||||
: ${serial2mqtt_options:="${serial2mqtt_flags}"}
|
||||
|
||||
if [ -f ${serial2mqtt_conf} ]; then
|
||||
serial2mqtt_options="${serial2mqtt_options} --config ${serial2mqtt_conf}"
|
||||
fi
|
||||
|
||||
logfile="/var/log/serial2mqtt/${name}.log"
|
||||
pidfile="/var/run/${name}.pid"
|
||||
command=/usr/sbin/daemon
|
||||
start_precmd="serial2mqtt_prestart"
|
||||
start_cmd="serial2mqtt_start"
|
||||
stop_cmd="serial2mqtt_stop"
|
||||
|
||||
serial2mqtt_prestart() {
|
||||
install -d -o ${serial2mqtt_user} -g ${serial2mqtt_group} -m750 /var/log/serial2mqtt
|
||||
}
|
||||
|
||||
serial2mqtt_start() {
|
||||
echo "Starting ${name}"
|
||||
/usr/sbin/daemon -fcr -P ${pidfile} -u ${serial2mqtt_user} -o ${logfile} \
|
||||
/usr/local/bin/python3.7 /usr/local/share/arduino-sensors-toolkit/${name}.py ${serial2mqtt_options}
|
||||
}
|
||||
|
||||
serial2mqtt_stop() {
|
||||
pid=$(check_pidfile $pidfile $command)
|
||||
if [ -n "${pid}" ]; then
|
||||
echo "Stopping ${name} (pid=${pid})"
|
||||
kill -- -${pid}
|
||||
wait_for_pids ${pid}
|
||||
else
|
||||
echo "${name} isn't running"
|
||||
fi
|
||||
}
|
||||
|
||||
run_rc_command "$1"
|
2
requirements.txt
Normal file
2
requirements.txt
Normal file
|
@ -0,0 +1,2 @@
|
|||
paho-mqtt==1.5.0
|
||||
pyserial==3.4
|
37
sensors2serial.ino
Normal file
37
sensors2serial.ino
Normal file
|
@ -0,0 +1,37 @@
|
|||
#include <DHT.h>
|
||||
|
||||
#define KYPIN A0 // analog pin where KY-037 sensor is connected
|
||||
#define DHTPIN 2 // digital pin where DHT22 sensor is connected
|
||||
|
||||
DHT dht(DHTPIN, DHT22); // initialize DHT22 object
|
||||
|
||||
float h; // humidity
|
||||
float t; // temperature
|
||||
int s; // sound
|
||||
|
||||
void setup()
|
||||
{
|
||||
Serial.begin(9600);
|
||||
dht.begin();
|
||||
}
|
||||
|
||||
void loop()
|
||||
{
|
||||
// sensors need some time to produce valid values
|
||||
delay(2000);
|
||||
|
||||
// read values from sensors
|
||||
h = dht.readHumidity();
|
||||
t = dht.readTemperature();
|
||||
s = analogRead(KYPIN);
|
||||
|
||||
// print "<humidity>,<temperature>,<sound>" (CSV-like)
|
||||
// all values are numbers
|
||||
if (!isnan(h) && !isnan(t) && !isnan(s)) {
|
||||
Serial.print(h);
|
||||
Serial.print(",");
|
||||
Serial.print(t);
|
||||
Serial.print(",");
|
||||
Serial.println(s);
|
||||
}
|
||||
}
|
10
serial2mqtt.ini.example
Normal file
10
serial2mqtt.ini.example
Normal file
|
@ -0,0 +1,10 @@
|
|||
[mqtt]
|
||||
host = localhost
|
||||
port = 1883
|
||||
#client_id = serial2mqtt
|
||||
#topic_prefix = sensors
|
||||
#username = ***
|
||||
#password = ***
|
||||
|
||||
[serial]
|
||||
interface = /dev/cuaU0
|
112
serial2mqtt.py
Executable file
112
serial2mqtt.py
Executable file
|
@ -0,0 +1,112 @@
|
|||
#!/usr/bin/env python3
|
||||
import argparse
|
||||
import configparser
|
||||
import logging
|
||||
import os
|
||||
import sys
|
||||
import time
|
||||
|
||||
import paho.mqtt.client as paho
|
||||
import serial
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
|
||||
def read_serial(interface):
|
||||
try:
|
||||
logger.debug('reading serial interface {}'.format(interface))
|
||||
s = serial.Serial(interface)
|
||||
s.flushInput()
|
||||
for i in range(2):
|
||||
data = s.readline()
|
||||
if i > 0:
|
||||
return data
|
||||
except Exception as err:
|
||||
logger.warning(str(err))
|
||||
time.sleep(2)
|
||||
|
||||
|
||||
def parse_sensor_values(data):
|
||||
"""
|
||||
Input line: <humidity>,<temperature>,<sound>
|
||||
Output structure:
|
||||
data = {
|
||||
'humidity': <humidity>,
|
||||
'temperature': <temperature>,
|
||||
'sound': <sound>
|
||||
}
|
||||
"""
|
||||
humidity, temperature, sound = data.decode('utf-8').strip().split(',')
|
||||
return {'humidity': humidity, 'temperature': temperature, 'sound': sound}
|
||||
|
||||
|
||||
def connect_mqtt(config):
|
||||
"""Connect to MQTT broken from MQTT config and return client"""
|
||||
mqtt_host = config.get('host', 'localhost')
|
||||
mqtt_port = int(config.get('port', 1883))
|
||||
mqtt_username = config.get('username')
|
||||
mqtt_password = config.get('password')
|
||||
mqtt_client_id = config.get('client_id', 'serial2mqtt')
|
||||
|
||||
client = paho.Client(client_id=mqtt_client_id)
|
||||
client.on_connect = on_connect
|
||||
if mqtt_username:
|
||||
client.username_pw_set(username=mqtt_username, password=mqtt_password)
|
||||
logging.debug('connecting to MQTT broker at {}:{}'.format(mqtt_host, mqtt_port))
|
||||
client.connect(mqtt_host, mqtt_port)
|
||||
return client
|
||||
|
||||
|
||||
def publish(client, topic, value):
|
||||
logger.debug('publishing to topic "{}" with value "{}"'.format(topic, value))
|
||||
client.publish(topic, value)
|
||||
|
||||
|
||||
def on_connect(client, userdata, flags, rc):
|
||||
if rc > 0:
|
||||
logger.error(paho.connack_string(rc))
|
||||
else:
|
||||
logger.info('connected to MQTT broker')
|
||||
|
||||
|
||||
def main():
|
||||
parser = argparse.ArgumentParser()
|
||||
parser.add_argument('-c', '--config', help='path to configuration file', default='serial2mqtt.ini')
|
||||
parser.add_argument('-v', '--verbose', dest='loglevel', action='store_const', const=logging.INFO,
|
||||
help='print more output')
|
||||
parser.add_argument('-d', '--debug', dest='loglevel', action='store_const', const=logging.DEBUG,
|
||||
default=logging.WARNING, help='print even more output')
|
||||
args = parser.parse_args()
|
||||
|
||||
logging.basicConfig(format='%(levelname)s: %(message)s', level=args.loglevel)
|
||||
|
||||
config = configparser.ConfigParser()
|
||||
if os.path.isfile(args.config):
|
||||
config.read(args.config)
|
||||
|
||||
mqtt_config = config['mqtt'] if 'mqtt' in config else {}
|
||||
mqtt_topic_prefix = mqtt_config.get('topic_prefix', 'sensors')
|
||||
serial_config = config['serial'] if 'serial' in config else {}
|
||||
|
||||
try:
|
||||
client = connect_mqtt(mqtt_config)
|
||||
except Exception as err:
|
||||
logger.error('connection to MQTT broker failed: {}'.format(str(err)))
|
||||
sys.exit(1)
|
||||
|
||||
try:
|
||||
client.loop_start()
|
||||
|
||||
while True:
|
||||
data = read_serial(serial_config.get('interface', '/dev/cuaU0'))
|
||||
if data:
|
||||
data = parse_sensor_values(data)
|
||||
for sensor_name, value in data.items():
|
||||
topic = '/'.join([mqtt_topic_prefix, sensor_name])
|
||||
publish(client, topic, value)
|
||||
except KeyboardInterrupt:
|
||||
pass
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
main()
|
8717
static/circuit.svg
Normal file
8717
static/circuit.svg
Normal file
File diff suppressed because one or more lines are too long
After Width: | Height: | Size: 543 KiB |
Loading…
Add table
Reference in a new issue