Skip to content
Snippets Groups Projects
Commit 87c9f081 authored by Birte Kristina Friesel's avatar Birte Kristina Friesel
Browse files

add arduino-nano bme680-max44009-datalogger

parent 99841981
No related branches found
No related tags found
No related merge requests found
# vim:ft=make
#
# Copyright 2020 Daniel Friesel
#
# SPDX-License-Identifier: CC0-1.0
ifdef app
override loop = 1
override arch_drivers += ,i2c
override drivers += ,bme680,max44009
endif
#!/usr/bin/env python3
import json
import paho.mqtt.client as mqtt
import re
import requests
import serial
import serial.threaded
import sys
import time
location = "wohnzimmer"
class SerialReader(serial.threaded.Protocol):
"""
Character- to line-wise data buffer for serial interfaces.
Reads in new data whenever it becomes available and exposes a line-based
interface to applications.
"""
def __init__(self, callback):
"""Create a new SerialReader object."""
self.callback = callback
self.recv_buf = ""
def __call__(self):
return self
def data_received(self, data):
"""Append newly received serial data to the line buffer."""
try:
str_data = data.decode("UTF-8")
self.recv_buf += str_data
# We may get anything between \r\n, \n\r and simple \n newlines.
# We assume that \n is always present and use str.strip to remove leading/trailing \r symbols
# Note: Do not call str.strip on lines[-1]! Otherwise, lines may be mangled
lines = self.recv_buf.split("\n")
if len(lines) > 1:
self.recv_buf = lines[-1]
for line in lines[:-1]:
self.callback(str.strip(line))
except UnicodeDecodeError:
pass
# sys.stderr.write('UART output contains garbage: {data}\n'.format(data = data))
class SerialMonitor:
"""SerialMonitor captures serial output for a specific amount of time."""
def __init__(self, port: str, baud: int, callback):
"""
Create a new SerialMonitor connected to port at the specified baud rate.
Communication uses no parity, no flow control, and one stop bit.
Data collection starts immediately.
"""
self.ser = serial.serial_for_url(port, do_not_open=True)
self.ser.baudrate = baud
self.ser.parity = "N"
self.ser.rtscts = False
self.ser.xonxoff = False
try:
self.ser.open()
except serial.SerialException as e:
sys.stderr.write(
"Could not open serial port {}: {}\n".format(self.ser.name, e)
)
sys.exit(1)
self.reader = SerialReader(callback=callback)
self.worker = serial.threaded.ReaderThread(self.ser, self.reader)
self.worker.start()
def close(self):
"""Close serial connection."""
self.worker.stop()
self.ser.close()
if __name__ == "__main__":
mqtt = mqtt.Client()
mqtt.connect("mqtt.derf0.net")
step = 0
got_data = False
max_accel = 0
max_magnet = 0
vcc = 0
temperature = 0
humidity = 0
pressure = 0
gas = 0
brightness = 0
def parse_line(line):
global got_data
global vcc
global temperature
global humidity
global pressure
global gas
global brightness
match = re.match("BME680 temperature: ([^ ]+)", line)
if match:
temperature = float(match.group(1))
match = re.match("BME680 humidity: ([^ ]+)", line)
if match:
humidity = float(match.group(1))
match = re.match("BME680 pressure: ([^ ]+)", line)
if match:
pressure = float(match.group(1))
match = re.match("BME680 gas resistance: ([^ ]+)", line)
if match:
gas = match.group(1)
match = re.match("VCC: ([^ ]+)", line)
if match:
vcc = int(match.group(1))
match = re.match("MAX44009: ([^ ]+)", line)
if match:
got_data = True
brightness = float(match.group(1))
requests.post(
"http://influxdb.derf0.net:8086/write?db=sensors",
f"bme680,area=hm17,location={location} temperature_celsius={temperature},humidity_relpercent={humidity},pressure_hpa={pressure},air_quality_ohm={gas}",
)
requests.post(
"http://influxdb.derf0.net:8086/write?db=sensors",
f"max44009,area=hm17,location={location} illuminance_lux={brightness}",
)
mqtt.publish(
f"sensor/hm17/{location}/brightness_lux",
brightness
)
mqtt.publish(
f"sensor/hm17/{location}/bme680",
json.dumps(
{
"temperature_celsius": round(temperature, 1),
"humidity_percent": round(humidity, 1),
"pressure_hpa": pressure,
"iaq_ohm": gas,
}
),
)
temperature = None
humidity = None
pressure = None
gas = None
vcc = None
brightness = None
monitor = SerialMonitor("/dev/ttyUSB0", 57600, parse_line)
try:
while True:
time.sleep(5)
step += 1
if step == 4:
if not got_data:
print("Error: received no data for 20 seconds", file=sys.stderr)
sys.exit(1)
got_data = False
step = 0
except KeyboardInterrupt:
monitor.close()
mqtt.disconnect()
/*
* Copyright 2021 Daniel Friesel
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include "arch.h"
#include "driver/gpio.h"
#include "driver/stdout.h"
#if defined(MULTIPASS_ARCH_HAS_I2C) && !defined(DRIVER_SOFTI2C)
#include "driver/i2c.h"
#else
#include "driver/soft_i2c.h"
#endif
#include "driver/bme680.h"
#include "driver/bme680_util.h"
#include "driver/max44009.h"
struct bme680_field_data data;
void loop(void)
{
static unsigned char i = 0;
if ((i == 1) && (ADCSRA & _BV(ADIF))) {
uint8_t adcr_l = ADCL;
uint8_t adcr_h = ADCH;
uint16_t adcr = adcr_l + (adcr_h << 8);
uint16_t vcc = 1100L * 1023 / adcr;
TIFR1 |= _BV(TOV1);
ADCSRA |= _BV(ADIF);
kout << "VCC: " << vcc << endl;
}
if (i == 0) {
bme680.setSensorMode();
}
else if (i == 1) {
if (bme680.getSensorData(&data) == 0) {
kout << "BME680 temperature: " << (float)data.temperature / 100 << " degC" << endl;
kout << "BME680 humidity: " << (float)data.humidity / 1000 << " %" << endl;
kout << "BME680 pressure: " << (float)data.pressure / 100 << " hPa" << endl;
kout << "BME680 gas resistance: " << data.gas_resistance << endl;
}
kout << "MAX44009: ";
kout.printf_float(max44009.getLux());
kout << " lx" << endl;
}
i = (i + 1) % 20;
}
int main(void)
{
unsigned short i = 0;
arch.setup();
gpio.setup();
kout.setup();
// One ADC conversion per four seconds
TCCR0A = 0;
TCCR0B = _BV(CS12) | _BV(CS10);
// Measure internal 1.1V bandgap using VCC as reference on each Timer 0 overflow
ADMUX = _BV(REFS0) | 0x0e;
ADCSRB = _BV(ADTS2);
ADCSRA = _BV(ADEN) | _BV(ADATE) | _BV(ADPS2) | _BV(ADPS1);
if (i2c.setup() != 0) {
kout << "I2C setup failed" << endl;
return 1;
}
kout << "I2C setup OK" << endl;
bme680.intf = BME680_I2C_INTF;
bme680.read = bme680_i2c_read;
bme680.write = bme680_i2c_write;
bme680.delay_ms = bme680_delay_ms;
bme680.amb_temp = 25;
int8_t rslt = BME680_OK;
rslt = bme680.init();
kout << "BME680 init " << rslt << endl;
bme680.power_mode = BME680_FORCED_MODE;
bme680.tph_sett.os_hum = BME680_OS_2X;
bme680.tph_sett.os_pres = BME680_OS_16X;
bme680.tph_sett.os_temp = BME680_OS_2X;
bme680.gas_sett.run_gas = BME680_ENABLE_GAS_MEAS;
bme680.gas_sett.heatr_dur = 100;
bme680.gas_sett.heatr_temp = 300;
bme680.setSensorSettings(BME680_OST_SEL | BME680_OSP_SEL | BME680_OSH_SEL | BME680_GAS_SENSOR_SEL);
arch.delay_ms(200);
arch.idle_loop();
return 0;
}
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment