2019-06-05 13:32:10 -04:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
"""
|
|
|
|
A hub for controlling IOT devices.
|
|
|
|
"""
|
|
|
|
import re
|
|
|
|
import json
|
|
|
|
|
|
|
|
import requests
|
2019-06-06 13:16:59 -04:00
|
|
|
from flask import Flask, render_template, request, abort
|
2019-06-05 13:32:10 -04:00
|
|
|
|
|
|
|
|
|
|
|
class RelayDevice:
|
|
|
|
"""
|
|
|
|
Represents a relay receptacle device controlled by an ESP-01. The
|
|
|
|
ESP-01 acts a simple switch to turn each receptable on or off via
|
|
|
|
a relay. Each device only has two outputs, `OUT0` and `OUT2`.
|
|
|
|
"""
|
|
|
|
def __init__(self):
|
|
|
|
self.id = ""
|
|
|
|
self.description = ""
|
|
|
|
self.location = ""
|
|
|
|
self.ip_address = ""
|
|
|
|
self.sub_devices = {}
|
|
|
|
|
|
|
|
self.update()
|
|
|
|
|
|
|
|
def update(self):
|
|
|
|
"""
|
|
|
|
Queries the physical device and updates the internal model as
|
|
|
|
appropriate.
|
|
|
|
"""
|
|
|
|
for name, device in self.sub_devices.items():
|
|
|
|
res = requests.get(self.ip_address)
|
|
|
|
gpio0 = re.search(r"GPIO0: (\bLow|\bHigh)", res.text).groups()[0]
|
|
|
|
device.state = gpio0 == 'High'
|
|
|
|
|
2019-06-06 13:16:59 -04:00
|
|
|
def toggle(self, sub_dev_id):
|
|
|
|
"""
|
|
|
|
Toggles the state of a sub device.
|
|
|
|
"""
|
|
|
|
for sub_dev in sum(self.sub_devices.values(), []):
|
|
|
|
if sub_dev.id == sub_dev_id:
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
|
|
|
params = {sub_dev.gpio: 'toggle'}
|
|
|
|
res = requests.get('http://' + self.ip_address + '/gpio',params=params)
|
|
|
|
res.raise_for_status()
|
|
|
|
|
|
|
|
state = re.search(
|
|
|
|
rf"GPIO{sub_dev.gpio}: (\bLow|\bHigh)",
|
|
|
|
res.text
|
|
|
|
).groups()[0] == 'High'
|
|
|
|
sub_dev.state = state
|
|
|
|
|
|
|
|
return json.dumps({sub_dev_id: state})
|
|
|
|
|
2019-06-05 13:32:10 -04:00
|
|
|
def from_dict(self, data):
|
|
|
|
"""
|
|
|
|
Initializes self with data from JSON dict.
|
|
|
|
"""
|
2019-06-08 03:12:03 -04:00
|
|
|
for key, value in data.items():
|
|
|
|
setattr(self, key, value)
|
2019-06-05 13:32:10 -04:00
|
|
|
for sub_dev_type, sub_devs in data['sub_devices'].items():
|
|
|
|
if sub_dev_type == 'RelayOutlet':
|
|
|
|
self.sub_devices[sub_dev_type] = []
|
|
|
|
for sub_dev in sub_devs:
|
|
|
|
sub_dev = RelayOutlet().from_dict(sub_dev)
|
|
|
|
self.sub_devices[sub_dev_type].append(sub_dev)
|
|
|
|
else:
|
|
|
|
raise ValueError(f"Unknown sub_device type {dev_name}")
|
|
|
|
return self
|
|
|
|
|
|
|
|
|
|
|
|
class RelayOutlet:
|
|
|
|
"""
|
|
|
|
Represents a single outlet on a RelayDevice.
|
|
|
|
"""
|
|
|
|
def __init__(self):
|
|
|
|
self.id = ""
|
|
|
|
self.description = ""
|
2019-06-06 13:16:59 -04:00
|
|
|
self.gpio = ""
|
2019-06-05 13:32:10 -04:00
|
|
|
self.state = False
|
|
|
|
|
|
|
|
def from_dict(self, data):
|
|
|
|
"""
|
|
|
|
Initializes self with data from JSON dict.
|
|
|
|
"""
|
2019-06-08 03:12:03 -04:00
|
|
|
for key, value in data.items():
|
|
|
|
setattr(self, key, value)
|
2019-06-05 13:32:10 -04:00
|
|
|
return self
|
|
|
|
|
|
|
|
|
|
|
|
def init_network():
|
|
|
|
"""
|
|
|
|
Initializes the network of IOT devices.
|
|
|
|
"""
|
|
|
|
with open("devices.json", 'r') as file:
|
|
|
|
data = file.read()
|
|
|
|
data = json.loads(data)
|
|
|
|
network = {}
|
|
|
|
for dev_type, devices in data.items():
|
|
|
|
if dev_type == 'RelayDevice':
|
|
|
|
network[dev_type] = []
|
|
|
|
for device in devices:
|
|
|
|
device = RelayDevice().from_dict(device)
|
|
|
|
network[dev_type].append(device)
|
|
|
|
else:
|
|
|
|
raise ValueError(f"Unknown device type {dev_name}")
|
|
|
|
return network
|
|
|
|
|
|
|
|
|
|
|
|
app = Flask(__name__)
|
|
|
|
network = init_network()
|
|
|
|
|
|
|
|
@app.route('/')
|
|
|
|
def index():
|
|
|
|
"""
|
|
|
|
The index page.
|
|
|
|
"""
|
|
|
|
return render_template('index.html', network=network)
|
|
|
|
|
|
|
|
|
2019-06-06 13:16:59 -04:00
|
|
|
@app.route('/toggle')
|
|
|
|
def toggle():
|
|
|
|
"""
|
|
|
|
Toggles the state of a RelayDevice.
|
|
|
|
"""
|
|
|
|
device_id = request.args.get('device_id')
|
|
|
|
sub_dev_id = request.args.get('sub_dev_id')
|
|
|
|
|
|
|
|
for device in sum(network.values(), []):
|
|
|
|
if device.id == device_id:
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
abort(404)
|
|
|
|
res = device.toggle(sub_dev_id)
|
|
|
|
if not res:
|
|
|
|
abort(404)
|
|
|
|
return res
|
|
|
|
|
|
|
|
|
2019-06-05 13:32:10 -04:00
|
|
|
if __name__ == '__main__':
|
|
|
|
app.run(host='0.0.0.0', port=5300)
|