added first self-check tests with nose
added version switched to proper logging
This commit is contained in:
@@ -12,4 +12,6 @@
|
|||||||
# GNU General Public License for more details.
|
# GNU General Public License for more details.
|
||||||
#
|
#
|
||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
VERSION = "0.1"
|
@@ -95,7 +95,6 @@ class Controller:
|
|||||||
return "<Controller stripes={} cid={}>".format(len(self.stripes), self.id)
|
return "<Controller stripes={} cid={}>".format(len(self.stripes), self.id)
|
||||||
|
|
||||||
def set_channel(self, channel, val):
|
def set_channel(self, channel, val):
|
||||||
print(type(LED0_OFF_L + 4 * channel), type(val), channel, val, type(self.address))
|
|
||||||
self.bus.write_word_data(int(self.address, 16), LED0_OFF_L + 4 * channel, int(val * 4095))
|
self.bus.write_word_data(int(self.address, 16), LED0_OFF_L + 4 * channel, int(val * 4095))
|
||||||
self.bus.write_word_data(int(self.address, 16), LED0_ON_L + 4 * channel, 0)
|
self.bus.write_word_data(int(self.address, 16), LED0_ON_L + 4 * channel, 0)
|
||||||
|
|
||||||
|
@@ -23,8 +23,12 @@ import os
|
|||||||
import sys
|
import sys
|
||||||
import traceback
|
import traceback
|
||||||
import time
|
import time
|
||||||
|
import logging
|
||||||
|
from multiprocessing import Process
|
||||||
|
|
||||||
from LedD import controller
|
import nose
|
||||||
|
|
||||||
|
from LedD import controller, VERSION
|
||||||
from LedD.decorators import add_action
|
from LedD.decorators import add_action
|
||||||
|
|
||||||
|
|
||||||
@@ -37,13 +41,16 @@ class Daemon:
|
|||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
Daemon.instance = self
|
Daemon.instance = self
|
||||||
|
logging.basicConfig(level=logging.DEBUG,
|
||||||
|
format="[%(asctime)s] %(levelname)s [%(name)s.%(funcName)s:%(lineno)d] %(message)s",
|
||||||
|
datefmt="%H:%M:%S")
|
||||||
try:
|
try:
|
||||||
self.config = configparser.ConfigParser()
|
self.config = configparser.ConfigParser()
|
||||||
try:
|
try:
|
||||||
with open('ledd.config', 'w+') as f:
|
with open('ledd.config', 'w+') as f:
|
||||||
self.config.read_file(f)
|
self.config.read_file(f)
|
||||||
except FileNotFoundError:
|
except FileNotFoundError:
|
||||||
print("no config file found!")
|
logging.info("No config file found!")
|
||||||
|
|
||||||
self.sqldb = sqlite3.connect(self.config.get(self.databaseSection, 'name', fallback='ledd.sqlite'))
|
self.sqldb = sqlite3.connect(self.config.get(self.databaseSection, 'name', fallback='ledd.sqlite'))
|
||||||
self.sqldb.row_factory = sqlite3.Row
|
self.sqldb.row_factory = sqlite3.Row
|
||||||
@@ -54,13 +61,13 @@ class Daemon:
|
|||||||
self.sqldb.commit()
|
self.sqldb.commit()
|
||||||
|
|
||||||
self.controllers = controller.Controller.from_db(self.sqldb)
|
self.controllers = controller.Controller.from_db(self.sqldb)
|
||||||
print(self.controllers)
|
logging.debug(self.controllers)
|
||||||
|
|
||||||
server = self.SocketServer(self.config.get(self.daemonSection, 'host', fallback='0.0.0.0'),
|
server = self.SocketServer(self.config.get(self.daemonSection, 'host', fallback='0.0.0.0'),
|
||||||
self.config.get(self.daemonSection, 'port', fallback=1425))
|
self.config.get(self.daemonSection, 'port', fallback=1425))
|
||||||
asyncore.loop()
|
asyncore.loop()
|
||||||
except (KeyboardInterrupt, SystemExit):
|
except (KeyboardInterrupt, SystemExit):
|
||||||
print("\nShutting down...")
|
logging.info("Exiting")
|
||||||
self.sqldb.close()
|
self.sqldb.close()
|
||||||
sys.exit(0)
|
sys.exit(0)
|
||||||
|
|
||||||
@@ -77,7 +84,7 @@ class Daemon:
|
|||||||
c.close()
|
c.close()
|
||||||
|
|
||||||
if db_version is not None:
|
if db_version is not None:
|
||||||
print("DB connection established; version={}".format(db_version[0]))
|
logging.info("DB connection established; db-version=%s", db_version[0])
|
||||||
return True
|
return True
|
||||||
else:
|
else:
|
||||||
return False
|
return False
|
||||||
@@ -105,7 +112,7 @@ class Daemon:
|
|||||||
:param req_json: dict of request json
|
:param req_json: dict of request json
|
||||||
"""
|
"""
|
||||||
# TODO: add adapter setting stripe with color here
|
# TODO: add adapter setting stripe with color here
|
||||||
print("recieved action: {}".format(req_json['action']))
|
logging.debug("recieved action: %s", req_json['action'])
|
||||||
|
|
||||||
@add_action(action_dict)
|
@add_action(action_dict)
|
||||||
def add_controller(self, req_json):
|
def add_controller(self, req_json):
|
||||||
@@ -115,12 +122,12 @@ class Daemon:
|
|||||||
address: hexdecimal address of controller on i2c bus, e.g. 0x40
|
address: hexdecimal address of controller on i2c bus, e.g. 0x40
|
||||||
:param req_json: dict of request json
|
:param req_json: dict of request json
|
||||||
"""
|
"""
|
||||||
print("recieved action: {}".format(req_json['action']))
|
logging.debug("recieved action: %s", req_json['action'])
|
||||||
try:
|
try:
|
||||||
ncontroller = controller.Controller(Daemon.instance.sqldb, req_json['channels'],
|
ncontroller = controller.Controller(Daemon.instance.sqldb, req_json['channels'],
|
||||||
req_json['i2c_dev'], req_json['address'])
|
req_json['i2c_dev'], req_json['address'])
|
||||||
except OSError as e:
|
except OSError as e:
|
||||||
print("Error opening i2c device!")
|
logging.error("Error opening i2c device: %s", req_json['i2c_dev'])
|
||||||
rjson = {
|
rjson = {
|
||||||
'success': False,
|
'success': False,
|
||||||
'message': "Error while opening i2c device",
|
'message': "Error while opening i2c device",
|
||||||
@@ -146,7 +153,7 @@ class Daemon:
|
|||||||
Required JSON parameters: stripeid: sid
|
Required JSON parameters: stripeid: sid
|
||||||
:param req_json: dict of request json
|
:param req_json: dict of request json
|
||||||
"""
|
"""
|
||||||
print("recieved action: {}".format(req_json['action']))
|
logging.debug("recieved action: %s", req_json['action'])
|
||||||
# TODO: Add get color logic
|
# TODO: Add get color logic
|
||||||
|
|
||||||
@add_action(action_dict)
|
@add_action(action_dict)
|
||||||
@@ -156,11 +163,11 @@ class Daemon:
|
|||||||
Required JSON parameters:
|
Required JSON parameters:
|
||||||
:param req_json: dict of request json
|
:param req_json: dict of request json
|
||||||
"""
|
"""
|
||||||
print("recieved action: {}".format(req_json['action']))
|
logging.debug("recieved action: %s", req_json['action'])
|
||||||
if "stripes" in req_json:
|
if "stripes" in req_json:
|
||||||
for stripe in req_json['stripes']:
|
for stripe in req_json['stripes']:
|
||||||
# TODO: add stripe here
|
# TODO: add stripe here
|
||||||
print(len(req_json['stripes']))
|
logging.debug(len(req_json['stripes']))
|
||||||
|
|
||||||
@add_action(action_dict)
|
@add_action(action_dict)
|
||||||
def get_controllers(self, req_json):
|
def get_controllers(self, req_json):
|
||||||
@@ -169,7 +176,7 @@ class Daemon:
|
|||||||
Required JSON parameters: none
|
Required JSON parameters: none
|
||||||
:param req_json: dict of request json
|
:param req_json: dict of request json
|
||||||
"""
|
"""
|
||||||
print("recieved action: {}".format(req_json['action']))
|
logging.debug("recieved action: %s", req_json['action'])
|
||||||
|
|
||||||
rjson = {
|
rjson = {
|
||||||
'success': True,
|
'success': True,
|
||||||
@@ -187,16 +194,16 @@ class Daemon:
|
|||||||
Required JSON parameters: controller id: cid
|
Required JSON parameters: controller id: cid
|
||||||
:param req_json: dict of request json
|
:param req_json: dict of request json
|
||||||
"""
|
"""
|
||||||
print("recieved action: {}".format(req_json['action']))
|
logging.debug("recieved action: %s", req_json['action'])
|
||||||
|
|
||||||
result = next(filter(lambda x: x.id == req_json['cid'], self.controllers), None)
|
result = next(filter(lambda x: x.id == req_json['cid'], self.controllers), None)
|
||||||
""" :type : Controller """
|
""" :type : Controller """
|
||||||
|
|
||||||
if result is not None:
|
if result is not None:
|
||||||
for i in range(result.channels):
|
for i in range(result.channels):
|
||||||
print("set channel {}={}".format(i, "1"))
|
logging.debug("set channel %d=%s", i, "1")
|
||||||
result.set_channel(i, 1)
|
result.set_channel(i, 1)
|
||||||
time.sleep(2)
|
time.sleep(10)
|
||||||
result.set_channel(i, 0)
|
result.set_channel(i, 0)
|
||||||
|
|
||||||
rjson = {
|
rjson = {
|
||||||
@@ -206,6 +213,23 @@ class Daemon:
|
|||||||
|
|
||||||
return json.dumps(rjson)
|
return json.dumps(rjson)
|
||||||
|
|
||||||
|
@add_action(action_dict)
|
||||||
|
def discover(self, req_json):
|
||||||
|
"""
|
||||||
|
Part of the Color API. Used by mobile applications to find the controller.
|
||||||
|
Required JSON parameters: none
|
||||||
|
:param req_json: dict of request json
|
||||||
|
"""
|
||||||
|
logging.debug("recieved action: %s", req_json['action'])
|
||||||
|
|
||||||
|
rjson = {
|
||||||
|
'success': True,
|
||||||
|
'ref': req_json['ref'],
|
||||||
|
'version': VERSION
|
||||||
|
}
|
||||||
|
|
||||||
|
return json.dumps(rjson)
|
||||||
|
|
||||||
class ConnectionHandler(asyncore.dispatcher_with_send):
|
class ConnectionHandler(asyncore.dispatcher_with_send):
|
||||||
def handle_read(self):
|
def handle_read(self):
|
||||||
data = self.recv(5120)
|
data = self.recv(5120)
|
||||||
@@ -222,7 +246,7 @@ class Daemon:
|
|||||||
if data:
|
if data:
|
||||||
try:
|
try:
|
||||||
json_decoded = json.loads(data.decode())
|
json_decoded = json.loads(data.decode())
|
||||||
print(json.dumps(json_decoded, sort_keys=True, indent=4, separators=(',', ': ')))
|
logging.debug(json.dumps(json_decoded, sort_keys=True))
|
||||||
|
|
||||||
if "action" in json_decoded and "ref" in json_decoded:
|
if "action" in json_decoded and "ref" in json_decoded:
|
||||||
return_data = Daemon.instance.action_dict.get(json_decoded['action'], no_action_found)(
|
return_data = Daemon.instance.action_dict.get(json_decoded['action'], no_action_found)(
|
||||||
@@ -232,13 +256,11 @@ class Daemon:
|
|||||||
if return_data is not None:
|
if return_data is not None:
|
||||||
self.send("{}\n".format(return_data).encode())
|
self.send("{}\n".format(return_data).encode())
|
||||||
else:
|
else:
|
||||||
print("no action or ref value found, ignoring")
|
logging.warning("no action or ref value found in JSON, ignoring")
|
||||||
except TypeError as e:
|
except TypeError:
|
||||||
print("No valid JSON found: {}".format(e))
|
logging.error("No valid JSON found: %s", traceback.format_exc())
|
||||||
traceback.print_exc(file=sys.stdout)
|
|
||||||
except ValueError:
|
except ValueError:
|
||||||
print("No valid JSON detected!")
|
logging.error("No valid JSON detected: %s", traceback.format_exc())
|
||||||
traceback.print_exc(file=sys.stdout)
|
|
||||||
|
|
||||||
class SocketServer(asyncore.dispatcher):
|
class SocketServer(asyncore.dispatcher):
|
||||||
def __init__(self, host, port):
|
def __init__(self, host, port):
|
||||||
@@ -248,9 +270,16 @@ class Daemon:
|
|||||||
self.bind((host, port))
|
self.bind((host, port))
|
||||||
self.listen(5)
|
self.listen(5)
|
||||||
|
|
||||||
|
p = Process(target=self.run_tests)
|
||||||
|
p.start()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def run_tests():
|
||||||
|
nose.run()
|
||||||
|
|
||||||
def handle_accept(self):
|
def handle_accept(self):
|
||||||
pair = self.accept()
|
pair = self.accept()
|
||||||
if pair is not None:
|
if pair is not None:
|
||||||
sock, addr = pair
|
sock, addr = pair
|
||||||
print('Incoming connection from %s' % repr(addr))
|
logging.debug('Incoming connection from %s' % repr(addr))
|
||||||
handler = Daemon.ConnectionHandler(sock)
|
handler = Daemon.ConnectionHandler(sock)
|
||||||
|
61
LedD/tests.py
Normal file
61
LedD/tests.py
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
# LEDD Project
|
||||||
|
# Copyright (C) 2015 LEDD Team
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import configparser
|
||||||
|
import socket
|
||||||
|
import json
|
||||||
|
import uuid
|
||||||
|
|
||||||
|
|
||||||
|
class TestDaemon:
|
||||||
|
s = None
|
||||||
|
""" :type : socket.socket """
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def setup_class(cls):
|
||||||
|
daemon_section = 'daemon'
|
||||||
|
config = configparser.ConfigParser()
|
||||||
|
try:
|
||||||
|
with open('ledd.config', 'w+') as f:
|
||||||
|
config.read_file(f)
|
||||||
|
except FileNotFoundError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
cls.s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
||||||
|
cls.s.settimeout(20)
|
||||||
|
cls.s.connect((config.get(daemon_section, 'host', fallback='0.0.0.0'),
|
||||||
|
config.get(daemon_section, 'port', fallback=1425)))
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def teardown_class(cls):
|
||||||
|
cls.s.close()
|
||||||
|
|
||||||
|
def test_discover(self):
|
||||||
|
ref = uuid.uuid4()
|
||||||
|
sjson = {
|
||||||
|
"action": "discover",
|
||||||
|
"ref": ref.urn[9:]
|
||||||
|
}
|
||||||
|
|
||||||
|
self.s.send(json.dumps(sjson).encode())
|
||||||
|
|
||||||
|
rstr = self.s.recv(1024)
|
||||||
|
|
||||||
|
assert rstr is not None
|
||||||
|
|
||||||
|
rjson = json.loads(self.s.recv(5120))
|
||||||
|
assert rjson['ref'] == ref
|
||||||
|
assert rjson['version'] is not None
|
Reference in New Issue
Block a user