2019-10-22 23:55:00 +02:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
from flask import Flask, request, json, jsonify, abort, make_response
|
2019-11-03 01:36:23 +01:00
|
|
|
import logging
|
2019-10-22 23:55:00 +02:00
|
|
|
import argparse
|
2019-11-03 13:57:14 +01:00
|
|
|
import time
|
2019-11-03 01:36:23 +01:00
|
|
|
import Client
|
2019-10-22 23:55:00 +02:00
|
|
|
|
2019-11-03 01:36:23 +01:00
|
|
|
CONFIG_FILE = 'config.json'
|
2019-10-22 23:55:00 +02:00
|
|
|
PORT = 5000
|
2019-11-03 13:57:14 +01:00
|
|
|
TIMEOUT_CLIENT = 10
|
2019-11-03 01:36:23 +01:00
|
|
|
core = None
|
|
|
|
|
|
|
|
# sensors
|
|
|
|
NUM_VALUES = 1000
|
2019-10-22 23:55:00 +02:00
|
|
|
|
|
|
|
def setup():
|
|
|
|
# arguments
|
|
|
|
parser = argparse.ArgumentParser(description='homecontrol')
|
2019-11-03 01:36:23 +01:00
|
|
|
parser.add_argument('-p', '--port', dest='port', type=int,
|
|
|
|
help='listening port')
|
2019-10-22 23:55:00 +02:00
|
|
|
parser.add_argument('-c', '--config', dest='config', type=str,
|
2019-11-03 01:36:23 +01:00
|
|
|
help='config file', default=CONFIG_FILE)
|
2019-10-22 23:55:00 +02:00
|
|
|
parser.add_argument('-d', '--debug', dest='debug', action='store_true',
|
2019-11-03 01:36:23 +01:00
|
|
|
help='debug mode')
|
2019-11-03 13:57:14 +01:00
|
|
|
parser.add_argument('-D', '--debugflask', dest='debugflask', action='store_true',
|
|
|
|
help='flask debug mode')
|
2019-10-22 23:55:00 +02:00
|
|
|
|
|
|
|
# parse arguments
|
|
|
|
args = parser.parse_args()
|
|
|
|
|
|
|
|
# initialize config
|
|
|
|
config = {}
|
|
|
|
try:
|
2019-11-03 01:36:23 +01:00
|
|
|
config_file = open(args.config, 'r')
|
2019-10-22 23:55:00 +02:00
|
|
|
config = json.load(config_file)
|
2019-11-03 01:36:23 +01:00
|
|
|
except Exception as ex:
|
|
|
|
logger.error('Exception Type:%s, args:\n%s' % (type(ex).__name__, ex.args))
|
2019-10-22 23:55:00 +02:00
|
|
|
|
|
|
|
# fill new keys with defaults
|
|
|
|
if not config.get('port'):
|
|
|
|
config['port'] = PORT
|
|
|
|
|
|
|
|
# overwrite with arguments
|
|
|
|
if args.port:
|
|
|
|
config['port'] = args.port
|
|
|
|
|
|
|
|
# save to file
|
|
|
|
with open(args.config, 'w') as config_file:
|
|
|
|
json.dump(config, config_file)
|
|
|
|
|
|
|
|
return args, config
|
|
|
|
|
2019-11-03 01:36:23 +01:00
|
|
|
|
2019-10-22 23:55:00 +02:00
|
|
|
class Core:
|
2019-11-03 01:36:23 +01:00
|
|
|
def __init__(self, debug):
|
|
|
|
self.logger = logging.getLogger("Core")
|
|
|
|
if args.debug:
|
|
|
|
self.logger.setLevel(logging.DEBUG)
|
|
|
|
else:
|
|
|
|
self.logger.setLevel(logging.INFO)
|
|
|
|
|
|
|
|
self.actors = {}
|
|
|
|
self.sensors = {}
|
|
|
|
|
|
|
|
def actors_get(self):
|
|
|
|
ret = {}
|
2019-11-03 13:57:14 +01:00
|
|
|
# remove inactive actors
|
|
|
|
for a in list(self.actors):
|
|
|
|
if time.time() - self.actors[a].ts > TIMEOUT_CLIENT:
|
|
|
|
self.logger.info("removing inactive actor: %s" % a)
|
|
|
|
self.actors.pop(a)
|
|
|
|
|
2019-11-03 01:36:23 +01:00
|
|
|
for a in self.actors:
|
|
|
|
ret[a] = self.actors[a].get_info()
|
|
|
|
return ret
|
|
|
|
|
|
|
|
def sensors_get(self):
|
|
|
|
ret = {}
|
2019-11-03 13:57:14 +01:00
|
|
|
# remove inactive sensors
|
|
|
|
for s in list(self.sensors):
|
|
|
|
if time.time() - self.sensors[s].ts > TIMEOUT_CLIENT:
|
|
|
|
self.logger.info("removing inactive sensor: %s" % s)
|
|
|
|
self.sensors.pop(s)
|
|
|
|
|
2019-11-03 01:36:23 +01:00
|
|
|
for s in self.sensors:
|
|
|
|
ret[s] = self.sensors[s].get_info()
|
|
|
|
return ret
|
2019-10-22 23:55:00 +02:00
|
|
|
|
|
|
|
def update(self):
|
2019-11-03 01:36:23 +01:00
|
|
|
# TODO
|
2019-10-22 23:55:00 +02:00
|
|
|
pass
|
|
|
|
|
|
|
|
|
2019-11-03 01:36:23 +01:00
|
|
|
app = Flask(__name__)
|
2019-10-22 23:55:00 +02:00
|
|
|
|
|
|
|
|
2019-11-03 01:36:23 +01:00
|
|
|
@app.route("/actors/get", methods=['GET'])
|
|
|
|
def actors_get():
|
|
|
|
ret = core.actors_get()
|
|
|
|
return make_response(jsonify(ret), 200)
|
2019-10-22 23:55:00 +02:00
|
|
|
|
|
|
|
|
2019-11-03 01:36:23 +01:00
|
|
|
@app.route("/actors/update", methods=['POST'])
|
|
|
|
def actors_update():
|
2019-10-22 23:55:00 +02:00
|
|
|
ret = {}
|
|
|
|
try:
|
|
|
|
content = request.json
|
2019-11-03 13:57:14 +01:00
|
|
|
# logger.debug("/actors/update: %s" % content)
|
|
|
|
id_a = content.get("id")
|
2019-11-03 01:36:23 +01:00
|
|
|
level = content.get("level")
|
|
|
|
if id_a not in core.actors:
|
2019-11-03 13:57:14 +01:00
|
|
|
type_a = content.get("type")
|
|
|
|
levels = content.get("levels")
|
2019-11-03 01:36:23 +01:00
|
|
|
core.actors[id_a] = Client.Actor(id_a, type_a, levels)
|
|
|
|
actor = core.actors[id_a]
|
2019-11-03 13:57:14 +01:00
|
|
|
actor.update()
|
2019-11-03 01:36:23 +01:00
|
|
|
actor.level = level
|
2019-11-03 13:57:14 +01:00
|
|
|
ret = actor.get_queue()
|
2019-10-22 23:55:00 +02:00
|
|
|
except Exception as ex:
|
2019-11-03 01:36:23 +01:00
|
|
|
logger.error('Exception Type:%s, args:\n%s' % (type(ex).__name__, ex.args))
|
|
|
|
abort(400)
|
|
|
|
|
|
|
|
return make_response(jsonify(ret), 200)
|
|
|
|
|
|
|
|
|
2019-11-03 13:57:14 +01:00
|
|
|
@app.route("/actors/command", methods=['POST'])
|
|
|
|
def actors_command():
|
2019-11-03 01:36:23 +01:00
|
|
|
ret = {}
|
|
|
|
try:
|
|
|
|
content = request.json
|
2019-11-03 13:57:14 +01:00
|
|
|
# logger.debug("/actors/command: %s" % content)
|
|
|
|
id_a = content.get("id")
|
|
|
|
command = content.get("command")
|
|
|
|
data = content.get("data")
|
2019-11-03 01:36:23 +01:00
|
|
|
if id_a in core.actors:
|
2019-11-03 13:57:14 +01:00
|
|
|
actor = core.actors[id_a]
|
|
|
|
actor.add_queue(command, data)
|
|
|
|
|
2019-11-03 01:36:23 +01:00
|
|
|
except Exception as ex:
|
|
|
|
logger.error('Exception Type:%s, args:\n%s' % (type(ex).__name__, ex.args))
|
|
|
|
abort(400)
|
2019-11-03 13:57:14 +01:00
|
|
|
|
|
|
|
return make_response(jsonify(ret), 200)
|
|
|
|
|
|
|
|
|
|
|
|
@app.route("/actors/get_level/<id_a>", methods=['GET'])
|
|
|
|
def actors_get_level(id_a):
|
|
|
|
ret = {}
|
|
|
|
# logger.debug("actors/get_level/%s" % id_a)
|
|
|
|
if id_a in core.actors:
|
|
|
|
ret = core.actors[id_a].get_level()
|
|
|
|
else:
|
|
|
|
logger.debug("id_a: %s not in core.actors" % id_a)
|
2019-11-03 01:36:23 +01:00
|
|
|
return make_response(jsonify(ret), 200)
|
|
|
|
|
|
|
|
|
|
|
|
@app.route("/sensors/get", methods=['GET'])
|
|
|
|
def sensors_get():
|
|
|
|
ret = core.sensors_get()
|
|
|
|
return make_response(jsonify(ret), 200)
|
|
|
|
|
|
|
|
|
|
|
|
@app.route("/sensors/get_values/<id_s>", methods=['GET'])
|
|
|
|
def sensors_get_values(id_s):
|
2019-11-03 13:57:14 +01:00
|
|
|
ret = {}
|
2019-11-03 01:36:23 +01:00
|
|
|
logger.debug("sensors/get_values/%s" % id_s)
|
|
|
|
if id_s in core.sensors:
|
|
|
|
ret = core.sensors[id_s].get_values()
|
|
|
|
else:
|
|
|
|
logger.debug("id_s: %s not in core.sensors" % id_s)
|
2019-11-03 13:57:14 +01:00
|
|
|
return make_response(jsonify(ret), 200)
|
2019-11-03 01:36:23 +01:00
|
|
|
|
|
|
|
|
2019-11-03 13:57:14 +01:00
|
|
|
@app.route("/sensors/update", methods=['POST'])
|
2019-11-03 01:36:23 +01:00
|
|
|
def sensors_add_value():
|
|
|
|
ret = {}
|
|
|
|
try:
|
|
|
|
content = request.json
|
2019-11-03 13:57:14 +01:00
|
|
|
# logger.info("sensors_update: %s" % (content))
|
|
|
|
id_s = content.get("id")
|
2019-11-03 01:36:23 +01:00
|
|
|
value = content.get("value")
|
|
|
|
if id_s not in core.sensors:
|
2019-11-03 13:57:14 +01:00
|
|
|
type_s = content.get("type")
|
2019-11-03 01:36:23 +01:00
|
|
|
core.sensors[id_s] = Client.Sensor(id_s, type_s, NUM_VALUES)
|
|
|
|
logger.info("created sensor: %s" % (core.sensors[id_s].get_info()))
|
|
|
|
sensor = core.sensors[id_s]
|
2019-11-03 13:57:14 +01:00
|
|
|
sensor.update()
|
2019-11-03 01:36:23 +01:00
|
|
|
ret = sensor.add_value(value)
|
|
|
|
except Exception as ex:
|
|
|
|
logger.error('Exception Type:%s, args:\n%s' % (type(ex).__name__, ex.args))
|
|
|
|
abort(400)
|
|
|
|
return make_response(jsonify(ret), 200)
|
2019-10-22 23:55:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
args, config = setup()
|
2019-11-03 01:36:23 +01:00
|
|
|
core = Core(args.debug)
|
2019-11-03 13:57:14 +01:00
|
|
|
FORMAT='%(asctime)-8s %(name)12s::%(levelname)-7s %(message)s'
|
|
|
|
DATEFMT='%H:%M:%S'
|
|
|
|
logging.basicConfig(level=logging.WARNING, format=FORMAT, datefmt=DATEFMT)
|
2019-11-03 01:36:23 +01:00
|
|
|
logger = logging.getLogger("homecontrol")
|
|
|
|
if args.debug:
|
|
|
|
logger.setLevel(logging.DEBUG)
|
|
|
|
else:
|
2019-11-03 13:57:14 +01:00
|
|
|
logger.setLevel(logging.WARNING)
|
2019-11-03 01:36:23 +01:00
|
|
|
|
2019-11-03 13:57:14 +01:00
|
|
|
app.run(host='0.0.0.0', port=config['port'], debug=args.debugflask)
|