diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/home/temphum/__init__.py | 18 | ||||
-rw-r--r-- | src/home/temphum/base.py | 25 | ||||
-rw-r--r-- | src/home/temphum/dht12.py | 22 | ||||
-rw-r--r-- | src/home/temphum/si7021.py | 13 | ||||
-rw-r--r-- | src/temphum.py | 19 | ||||
-rwxr-xr-x | src/temphumd.py (renamed from src/si7021d.py) | 27 |
6 files changed, 110 insertions, 14 deletions
diff --git a/src/home/temphum/__init__.py b/src/home/temphum/__init__.py new file mode 100644 index 0000000..55a7e1f --- /dev/null +++ b/src/home/temphum/__init__.py @@ -0,0 +1,18 @@ +from .base import SensorType, TempHumSensor +from .si7021 import Si7021 +from .dht12 import DHT12 + +__all__ = [ + 'SensorType', + 'TempHumSensor', + 'create_sensor' +] + + +def create_sensor(type: SensorType, bus: int) -> TempHumSensor: + if type == SensorType.Si7021: + return Si7021(bus) + elif type == SensorType.DHT12: + return DHT12(bus) + else: + raise ValueError('unexpected sensor type') diff --git a/src/home/temphum/base.py b/src/home/temphum/base.py new file mode 100644 index 0000000..e774433 --- /dev/null +++ b/src/home/temphum/base.py @@ -0,0 +1,25 @@ +import smbus + +from abc import abstractmethod, ABC +from enum import Enum + + +class TempHumSensor: + @abstractmethod + def humidity(self) -> float: + pass + + @abstractmethod + def temperature(self) -> float: + pass + + +class I2CTempHumSensor(TempHumSensor, ABC): + def __init__(self, bus: int): + super().__init__() + self.bus = smbus.SMBus(bus) + + +class SensorType(Enum): + Si7021 = 'si7021' + DHT12 = 'dht12' diff --git a/src/home/temphum/dht12.py b/src/home/temphum/dht12.py new file mode 100644 index 0000000..d495766 --- /dev/null +++ b/src/home/temphum/dht12.py @@ -0,0 +1,22 @@ +from .base import I2CTempHumSensor + + +class DHT12(I2CTempHumSensor): + i2c_addr = 0x5C + + def _measure(self): + raw = self.bus.read_i2c_block_data(self.i2c_addr, 0, 5) + if (raw[0] + raw[1] + raw[2] + raw[3]) & 0xff != raw[4]: + raise ValueError("checksum error") + return raw + + def temperature(self) -> float: + raw = self._measure() + temp = raw[2] + (raw[3] & 0x7f) * 0.1 + if raw[3] & 0x80: + temp *= -1 + return temp + + def humidity(self) -> float: + raw = self._measure() + return raw[0] + raw[1] * 0.1 diff --git a/src/home/temphum/si7021.py b/src/home/temphum/si7021.py new file mode 100644 index 0000000..6289e15 --- /dev/null +++ b/src/home/temphum/si7021.py @@ -0,0 +1,13 @@ +from .base import I2CTempHumSensor + + +class Si7021(I2CTempHumSensor): + i2c_addr = 0x40 + + def temperature(self) -> float: + raw = self.bus.read_i2c_block_data(self.i2c_addr, 0xE3, 2) + return 175.72 * (raw[0] << 8 | raw[1]) / 65536.0 - 46.85 + + def humidity(self) -> float: + raw = self.bus.read_i2c_block_data(self.i2c_addr, 0xE5, 2) + return 125.0 * (raw[0] << 8 | raw[1]) / 65536.0 - 6.0 diff --git a/src/temphum.py b/src/temphum.py new file mode 100644 index 0000000..dc0b7dd --- /dev/null +++ b/src/temphum.py @@ -0,0 +1,19 @@ +from argparse import ArgumentParser +from home.temphum import SensorType, create_sensor + + +if __name__ == '__main__': + parser = ArgumentParser() + parser.add_argument('-t', '--type', choices=[item.value for item in SensorType], + required=True, + help='Sensor type') + parser.add_argument('-b', '--bus', type=int, default=0, + help='I2C bus number') + arg = parser.parse_args() + + sensor = create_sensor(SensorType(arg.type), arg.bus) + temp = sensor.temperature() + hum = sensor.humidity() + + print(f'temperature: {temp}') + print(f'rel. humidity: {hum}') diff --git a/src/si7021d.py b/src/temphumd.py index fe11787..563d192 100755 --- a/src/si7021d.py +++ b/src/temphumd.py @@ -1,29 +1,26 @@ #!/usr/bin/env python3 -import smbus -import argparse import asyncio import json import logging +from typing import Optional + from home.config import config from home.util import parse_addr +from home.temphum import SensorType, create_sensor, TempHumSensor logger = logging.getLogger(__name__) -bus = None +sensor: Optional[TempHumSensor] = None lock = asyncio.Lock() delay = 0.01 -async def si7021_read(): +async def get_measurements(): async with lock: await asyncio.sleep(delay) - # these are still blocking... meh - raw = bus.read_i2c_block_data(0x40, 0xE3, 2) - temp = 175.72 * (raw[0] << 8 | raw[1]) / 65536.0 - 46.85 - - raw = bus.read_i2c_block_data(0x40, 0xE5, 2) - rh = 125.0 * (raw[0] << 8 | raw[1]) / 65536.0 - 6.0 + temp = sensor.temperature() + rh = sensor.humidity() return rh, temp @@ -41,7 +38,7 @@ async def handle_client(reader, writer): if request == 'read': try: - rh, temp = await asyncio.wait_for(si7021_read(), timeout=3) + rh, temp = await asyncio.wait_for(get_measurements(), timeout=3) data = dict(humidity=rh, temp=temp) except asyncio.TimeoutError as e: logger.exception(e) @@ -68,12 +65,14 @@ async def run_server(host, port): if __name__ == '__main__': config.load() - host, port = parse_addr(config['server']['listen']) + if 'measure_delay' in config['sensor']: + delay = float(config['sensor']['measure_delay']) - delay = float(config['smbus']['delay']) - bus = smbus.SMBus(int(config['smbus']['bus'])) + sensor = create_sensor(SensorType(config['sensor']['type']), + int(config['sensor']['bus'])) try: + host, port = parse_addr(config['server']['listen']) asyncio.run(run_server(host, port)) except KeyboardInterrupt: logging.info('Exiting...') |