2017-09-24 23:28:38 +09:00
|
|
|
from systemd import daemon
|
2018-03-04 21:20:31 +09:00
|
|
|
import errno
|
|
|
|
import logging
|
|
|
|
import signal
|
2017-10-22 18:33:48 +09:00
|
|
|
import socket
|
2018-03-04 21:20:31 +09:00
|
|
|
import sys
|
|
|
|
import time
|
|
|
|
from collections import namedtuple
|
|
|
|
from multiprocessing import BoundedSemaphore, current_process, Process
|
|
|
|
|
|
|
|
from influxdb import InfluxDBClient
|
|
|
|
from miio.airpurifier import AirPurifier
|
|
|
|
|
2017-09-24 23:28:38 +09:00
|
|
|
|
|
|
|
class Message(namedtuple('Message', ('measurement', 'time', 'tags', 'fields'))):
|
|
|
|
__slots__ = ()
|
|
|
|
|
|
|
|
|
2018-03-04 21:20:31 +09:00
|
|
|
class GracefulExit(Exception):
|
|
|
|
pass
|
2017-09-24 23:28:38 +09:00
|
|
|
|
2018-03-04 21:20:31 +09:00
|
|
|
|
|
|
|
class Atmosphere(
|
|
|
|
namedtuple('Atmosphere', ('purifier_activated', 'purifier_fan_rpm', 'temperature', 'aqi', 'humidity'))):
|
|
|
|
__slots__ = ()
|
|
|
|
|
2017-09-24 23:28:38 +09:00
|
|
|
def __new__(cls, status):
|
|
|
|
return super(Atmosphere, cls).__new__(cls,
|
|
|
|
status.power == 'on',
|
|
|
|
int(status.motor_speed),
|
|
|
|
float(status.temperature),
|
|
|
|
int(status.aqi),
|
|
|
|
status.humidity / 100.0
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2018-03-04 21:20:31 +09:00
|
|
|
def signal_handler(signum, frame):
|
|
|
|
raise GracefulExit()
|
|
|
|
|
|
|
|
|
|
|
|
def timerProvider():
|
2017-09-24 23:28:38 +09:00
|
|
|
while True:
|
2018-03-04 21:20:31 +09:00
|
|
|
yield time.time()
|
|
|
|
time.sleep(2.5)
|
|
|
|
|
|
|
|
|
|
|
|
def sendPoints(idx, at, tags, readSem, sendSem, metric, dev):
|
|
|
|
proc = current_process()
|
|
|
|
logger = logging.getLogger('iter %d proc name %s id %s' % (idx, proc.name, proc.pid))
|
|
|
|
logger.info("start")
|
|
|
|
try:
|
|
|
|
readSem.acquire(timeout=15)
|
|
|
|
try:
|
|
|
|
fetch = dev.status()
|
|
|
|
info = Atmosphere(fetch)
|
|
|
|
message = Message('air', int(at * 1000000000), tags, info._asdict())
|
|
|
|
finally:
|
|
|
|
readSem.release()
|
|
|
|
|
|
|
|
sendSem.acquire(timeout=15)
|
|
|
|
|
|
|
|
try:
|
|
|
|
metric.write_points([message._asdict()])
|
|
|
|
finally:
|
|
|
|
sendSem.release()
|
|
|
|
except Exception as e:
|
|
|
|
logger.warning(e)
|
|
|
|
else:
|
|
|
|
logger.info('done')
|
2017-09-24 23:28:38 +09:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
2018-03-04 21:20:31 +09:00
|
|
|
code = 0
|
2017-09-24 23:28:38 +09:00
|
|
|
try:
|
2018-03-04 21:20:31 +09:00
|
|
|
# Use signal handler to throw exception which can be caught to allow
|
|
|
|
# graceful exit.
|
2018-03-04 21:21:03 +09:00
|
|
|
logging.basicConfig(level=logging.WARN)
|
2018-03-04 21:20:31 +09:00
|
|
|
readSem = BoundedSemaphore(3)
|
|
|
|
sendSem = BoundedSemaphore(3)
|
|
|
|
|
|
|
|
metric = InfluxDBClient('db', database='core', use_udp=True, udp_port=8089)
|
|
|
|
air_addr = socket.gethostbyname("air")
|
|
|
|
# dev = create_device(air_addr, AirPurifier)
|
2018-05-15 03:45:55 +09:00
|
|
|
dev = AirPurifier(air_addr, 'b370cd7d111f74949f2cc7023d9c3bea')
|
2018-03-04 21:20:31 +09:00
|
|
|
|
|
|
|
signal.signal(signal.SIGTERM, signal_handler)
|
|
|
|
signal.signal(signal.SIGINT, signal_handler)
|
2017-09-24 23:28:38 +09:00
|
|
|
tags = {
|
|
|
|
'source': 'mi_air_2',
|
|
|
|
'position': 'center_room',
|
|
|
|
}
|
|
|
|
daemon.notify('READY=1')
|
2018-03-04 21:20:31 +09:00
|
|
|
for idx, at in enumerate(timerProvider()):
|
|
|
|
args = idx, at, tags, readSem, sendSem, metric, dev
|
|
|
|
p = Process(target=sendPoints, args=args)
|
|
|
|
p.daemon = True
|
|
|
|
p.start()
|
|
|
|
except GracefulExit as ge:
|
|
|
|
pass
|
|
|
|
except KeyboardInterrupt as ki:
|
|
|
|
pass
|
|
|
|
except Exception as e:
|
|
|
|
logging.error(e)
|
|
|
|
code = errno.EINVAL
|
2017-09-24 23:28:38 +09:00
|
|
|
finally:
|
|
|
|
daemon.notify('STOPPING=1')
|
2018-03-04 21:20:31 +09:00
|
|
|
sys.exit(code)
|