Mercurial > templog
view py/sensor_ds18b20.py @ 586:87c20b8c5472 default master
port to python3
author | Matt Johnston <matt@ucc.asn.au> |
---|---|
date | Mon, 09 Sep 2019 22:24:10 +0800 |
parents | 97d99eb42d27 |
children |
line wrap: on
line source
#!/usr/bin/env python2.7 import os import re import asyncio import concurrent.futures import config from utils import D,L,W,E,EX class SensorDS18B20(object): THERM_RE = re.compile('.* YES\n.*t=(.*)\n', re.MULTILINE) def __init__(self, server): self.server = server self.readthread = concurrent.futures.ThreadPoolExecutor(max_workers=1) self.master_dir = config.SENSOR_BASE_DIR @asyncio.coroutine def do(self): vals = {} for n in self.sensor_names(): value = yield from self.do_sensor(n) if value is not None: vals[n] = value itemp = self.do_internal() if itemp: vals['internal'] = itemp self.server.add_reading(vals) @asyncio.coroutine def run(self): while True: yield from self.do() yield from asyncio.sleep(config.SENSOR_SLEEP) @asyncio.coroutine def read_wait(self, f): # handles a blocking file read with a threadpool. A # real python thread performs the read while other # asyncio tasks keep running. # the ds18b20 takes ~750ms to read, which is noticable # interactively. loop = asyncio.get_event_loop() return loop.run_in_executor(None, f.read) @asyncio.coroutine def do_sensor(self, s, contents = None): """ contents can be set by the caller for testing """ try: if contents is None: fn = os.path.join(self.master_dir, s, 'w1_slave') with open(fn, 'r') as f: contents = yield from self.read_wait(f) match = self.THERM_RE.match(contents) if match is None: D("no match") return None temp = int(match.groups(1)[0]) / 1000.0 if temp > 80 or temp == 0: E("Problem reading sensor '%s': %f" % (s, temp)) return None return temp except Exception as e: EX("Problem reading sensor '%s': %s" % (s, str(e))) return None def do_internal(self): try: with open(config.INTERNAL_TEMPERATURE, 'r') as f: return int(f.read()) / 1000.0 except Exception as e: EX("Problem reading internal sensor: %s" % str(e)) return None def sensor_names(self): """ Returns a sequence of sensorname """ slaves_path = os.path.join(self.master_dir, "w1_master_slaves") with open(slaves_path, 'r') as f: contents = f.read() if 'not found' in contents: E("No W1 sensors found") return [] names = contents.split() # only ds18b20 names = [n for n in names if n.startswith('28-')] return names def wort_name(self): return config.WORT_NAME def fridge_name(self): return config.FRIDGE_NAME