core/homeassistant/test.py

344 lines
12 KiB
Python
Raw Normal View History

2013-09-28 18:09:36 +00:00
"""
homeassistant.test
~~~~~~~~~~~~~~~~~~
Provides tests to verify that Home Assistant modules do what they should do.
"""
import unittest
import time
import requests
import homeassistant as ha
import homeassistant.remote as remote
2013-10-29 07:22:38 +00:00
import homeassistant.httpinterface as hah
2013-09-28 18:09:36 +00:00
API_PASSWORD = "test1234"
2013-10-29 07:22:38 +00:00
HTTP_BASE_URL = "http://127.0.0.1:{}".format(hah.SERVER_PORT)
2013-09-28 18:09:36 +00:00
2013-11-01 18:34:43 +00:00
def _url(path=""):
""" Helper method to generate urls. """
return HTTP_BASE_URL + path
class HAHelper(object): # pylint: disable=too-few-public-methods
""" Helper class to keep track of current running HA instance. """
core = None
2013-09-28 18:09:36 +00:00
2013-11-01 18:34:43 +00:00
def ensure_homeassistant_started():
""" Ensures home assistant is started. """
2013-09-28 18:09:36 +00:00
2013-11-01 18:34:43 +00:00
if not HAHelper.core:
core = {'eventbus': ha.EventBus()}
core['statemachine'] = ha.StateMachine(core['eventbus'])
2013-10-29 07:22:38 +00:00
core['eventbus'].listen('test_event', len)
2013-11-01 18:34:43 +00:00
core['statemachine'].set_state('test','a_state')
2013-09-28 18:09:36 +00:00
2013-11-01 18:34:43 +00:00
hah.HTTPInterface(core['eventbus'], core['statemachine'],
API_PASSWORD)
2013-09-28 18:09:36 +00:00
2013-11-01 18:34:43 +00:00
core['eventbus'].fire(ha.EVENT_START)
2013-09-28 18:09:36 +00:00
2013-11-01 18:34:43 +00:00
# Give objects time to startup
time.sleep(1)
HAHelper.core = core
return HAHelper.core['eventbus'], HAHelper.core['statemachine']
2013-09-28 18:09:36 +00:00
2013-11-01 18:34:43 +00:00
# pylint: disable=too-many-public-methods
class TestHTTPInterface(unittest.TestCase):
""" Test the HTTP debug interface and API. """
@classmethod
def setUpClass(cls): # pylint: disable=invalid-name
""" things to be run when tests are started. """
2013-11-01 18:34:43 +00:00
cls.eventbus, cls.statemachine = ensure_homeassistant_started()
2013-09-28 18:09:36 +00:00
def test_debug_interface(self):
""" Test if we can login by comparing not logged in screen to
logged in screen. """
2013-10-29 07:22:38 +00:00
with_pw = requests.get(
2013-11-01 18:34:43 +00:00
_url("/?api_password={}".format(API_PASSWORD)))
2013-10-29 07:22:38 +00:00
2013-11-01 18:34:43 +00:00
without_pw = requests.get(_url())
2013-10-29 07:22:38 +00:00
self.assertNotEqual(without_pw.text, with_pw.text)
2013-09-28 18:09:36 +00:00
def test_debug_state_change(self):
""" Test if the debug interface allows us to change a state. """
2013-10-29 07:22:38 +00:00
requests.post(
2013-11-01 18:34:43 +00:00
_url(hah.URL_STATES_CATEGORY.format("test")),
2013-10-29 07:22:38 +00:00
data={"new_state":"debug_state_change",
"api_password":API_PASSWORD})
2013-09-28 18:09:36 +00:00
self.assertEqual(self.statemachine.get_state("test")['state'],
"debug_state_change")
2013-09-28 18:09:36 +00:00
def test_api_password(self):
""" Test if we get access denied if we omit or provide
a wrong api password. """
req = requests.get(
2013-11-01 18:34:43 +00:00
_url(hah.URL_API_STATES_CATEGORY.format("test")))
2013-09-28 18:09:36 +00:00
self.assertEqual(req.status_code, 401)
req = requests.get(
2013-11-01 18:34:43 +00:00
_url(hah.URL_API_STATES_CATEGORY.format("test")),
params={"api_password":"not the password"})
2013-09-28 18:09:36 +00:00
self.assertEqual(req.status_code, 401)
def test_api_list_state_categories(self):
""" Test if the debug interface allows us to list state categories. """
2013-11-01 18:34:43 +00:00
req = requests.get(_url(hah.URL_API_STATES),
data={"api_password":API_PASSWORD})
data = req.json()
self.assertEqual(self.statemachine.categories,
data['categories'])
def test_api_get_state(self):
2013-10-29 07:22:38 +00:00
""" Test if the debug interface allows us to get a state. """
req = requests.get(
2013-11-01 18:34:43 +00:00
_url(hah.URL_API_STATES_CATEGORY.format("test")),
2013-10-29 07:22:38 +00:00
data={"api_password":API_PASSWORD})
data = req.json()
state = self.statemachine.get_state("test")
self.assertEqual(data['category'], "test")
self.assertEqual(data['state'], state['state'])
self.assertEqual(data['last_changed'], state['last_changed'])
self.assertEqual(data['attributes'], state['attributes'])
2013-11-01 18:34:43 +00:00
def test_api_get_non_existing_state(self):
""" Test if the debug interface allows us to get a state. """
req = requests.get(
_url(hah.URL_API_STATES_CATEGORY.format("does_not_exist")),
params={"api_password":API_PASSWORD})
2013-11-01 18:34:43 +00:00
self.assertEqual(req.status_code, 422)
2013-09-28 18:09:36 +00:00
def test_api_state_change(self):
""" Test if we can change the state of a category that exists. """
self.statemachine.set_state("test", "not_to_be_set_state")
2013-11-01 18:34:43 +00:00
requests.post(_url(hah.URL_API_STATES_CATEGORY.format("test")),
2013-10-29 07:22:38 +00:00
data={"new_state":"debug_state_change2",
"api_password":API_PASSWORD})
2013-09-28 18:09:36 +00:00
self.assertEqual(self.statemachine.get_state("test")['state'],
"debug_state_change2")
2013-09-28 18:09:36 +00:00
# pylint: disable=invalid-name
2013-09-28 18:09:36 +00:00
def test_api_state_change_of_non_existing_category(self):
""" Test if the API allows us to change a state of
a non existing category. """
new_state = "debug_state_change"
2013-10-29 07:22:38 +00:00
req = requests.post(
2013-11-01 18:34:43 +00:00
_url(hah.URL_API_STATES_CATEGORY.format(
2013-10-29 07:22:38 +00:00
"test_category_that_does_not_exist")),
data={"new_state": new_state,
"api_password": API_PASSWORD})
cur_state = (self.statemachine.
2013-10-29 07:22:38 +00:00
get_state("test_category_that_does_not_exist")['state'])
2013-09-28 18:09:36 +00:00
2013-10-29 07:22:38 +00:00
self.assertEqual(req.status_code, 201)
self.assertEqual(cur_state, new_state)
2013-09-28 18:09:36 +00:00
# pylint: disable=invalid-name
2013-09-28 18:09:36 +00:00
def test_api_fire_event_with_no_data(self):
""" Test if the API allows us to fire an event. """
test_value = []
def listener(event): # pylint: disable=unused-argument
2013-09-28 18:09:36 +00:00
""" Helper method that will verify our event got called. """
test_value.append(1)
self.eventbus.listen_once("test_event_no_data", listener)
2013-09-28 18:09:36 +00:00
2013-10-29 07:22:38 +00:00
requests.post(
2013-11-01 18:34:43 +00:00
_url(hah.URL_EVENTS_EVENT.format("test_event_no_data")),
2013-10-29 07:22:38 +00:00
data={"api_password":API_PASSWORD})
2013-09-28 18:09:36 +00:00
# Allow the event to take place
time.sleep(1)
self.assertEqual(len(test_value), 1)
# pylint: disable=invalid-name
2013-09-28 18:09:36 +00:00
def test_api_fire_event_with_data(self):
""" Test if the API allows us to fire an event. """
test_value = []
def listener(event): # pylint: disable=unused-argument
2013-09-28 18:09:36 +00:00
""" Helper method that will verify that our event got called and
that test if our data came through. """
if "test" in event.data:
test_value.append(1)
self.eventbus.listen_once("test_event_with_data", listener)
2013-09-28 18:09:36 +00:00
2013-10-29 07:22:38 +00:00
requests.post(
2013-11-01 18:34:43 +00:00
_url(hah.URL_EVENTS_EVENT.format("test_event_with_data")),
2013-10-29 07:22:38 +00:00
data={"event_data":'{"test": 1}',
"api_password":API_PASSWORD})
2013-09-28 18:09:36 +00:00
# Allow the event to take place
time.sleep(1)
self.assertEqual(len(test_value), 1)
# pylint: disable=invalid-name
2013-09-28 18:09:36 +00:00
def test_api_fire_event_with_invalid_json(self):
""" Test if the API allows us to fire an event. """
test_value = []
def listener(event): # pylint: disable=unused-argument
2013-09-28 18:09:36 +00:00
""" Helper method that will verify our event got called. """
test_value.append(1)
self.eventbus.listen_once("test_event_with_bad_data", listener)
2013-09-28 18:09:36 +00:00
2013-10-29 07:22:38 +00:00
req = requests.post(
2013-11-01 18:34:43 +00:00
_url(hah.URL_API_EVENTS_EVENT.format("test_event")),
2013-10-29 07:22:38 +00:00
data={"event_data":'not json',
"api_password":API_PASSWORD})
2013-09-28 18:09:36 +00:00
# It shouldn't but if it fires, allow the event to take place
time.sleep(1)
2013-11-01 18:34:43 +00:00
self.assertEqual(req.status_code, 422)
2013-09-28 18:09:36 +00:00
self.assertEqual(len(test_value), 0)
def test_api_get_event_listeners(self):
""" Test if we can get the list of events being listened for. """
req = requests.get(_url(hah.URL_API_EVENTS),
params={"api_password":API_PASSWORD})
data = req.json()
self.assertEqual(data['listeners'], self.eventbus.listeners)
2013-11-01 18:34:43 +00:00
class TestRemote(unittest.TestCase):
""" Test the homeassistant.remote module. """
@classmethod
def setUpClass(cls): # pylint: disable=invalid-name
""" things to be run when tests are started. """
cls.eventbus, cls.statemachine = ensure_homeassistant_started()
cls.remote_sm = remote.StateMachine("127.0.0.1", API_PASSWORD)
cls.remote_eb = remote.EventBus("127.0.0.1", API_PASSWORD)
cls.sm_with_remote_eb = ha.StateMachine(cls.remote_eb)
cls.sm_with_remote_eb.set_state("test", "a_state")
# pylint: disable=invalid-name
2013-11-01 18:34:43 +00:00
def test_remote_sm_list_state_categories(self):
""" Test if the debug interface allows us to list state categories. """
self.assertEqual(self.statemachine.categories,
self.remote_sm.categories)
def test_remote_sm_get_state(self):
""" Test if the debug interface allows us to list state categories. """
remote_state = self.remote_sm.get_state("test")
state = self.statemachine.get_state("test")
self.assertEqual(remote_state['state'], state['state'])
self.assertEqual(remote_state['last_changed'], state['last_changed'])
self.assertEqual(remote_state['attributes'], state['attributes'])
def test_remote_sm_get_non_existing_state(self):
""" Test if the debug interface allows us to list state categories. """
self.assertEqual(self.remote_sm.get_state("test_does_not_exist"), None)
def test_remote_sm_state_change(self):
""" Test if we can change the state of a category that exists. """
self.remote_sm.set_state("test", "set_remotely", {"test": 1})
state = self.statemachine.get_state("test")
self.assertEqual(state['state'], "set_remotely")
self.assertEqual(state['attributes']['test'], 1)
def test_remote_eb_listening_for_same(self):
""" Test if remote EB correctly reports listener overview. """
self.assertEqual(self.eventbus.listeners, self.remote_eb.listeners)
2013-11-01 18:34:43 +00:00
# pylint: disable=invalid-name
def test_remote_eb_fire_event_with_no_data(self):
""" Test if the remote eventbus allows us to fire an event. """
test_value = []
def listener(event): # pylint: disable=unused-argument
""" Helper method that will verify our event got called. """
test_value.append(1)
self.eventbus.listen_once("test_event_no_data", listener)
self.remote_eb.fire("test_event_no_data")
# Allow the event to take place
time.sleep(1)
self.assertEqual(len(test_value), 1)
# pylint: disable=invalid-name
def test_remote_eb_fire_event_with_data(self):
""" Test if the remote eventbus allows us to fire an event. """
test_value = []
def listener(event): # pylint: disable=unused-argument
""" Helper method that will verify our event got called. """
if event.data["test"] == 1:
test_value.append(1)
self.eventbus.listen_once("test_event_with_data", listener)
self.remote_eb.fire("test_event_with_data", {"test": 1})
# Allow the event to take place
time.sleep(1)
self.assertEqual(len(test_value), 1)
def test_local_sm_with_remote_eb(self):
""" Test if we get the event if we change a state on a
StateMachine connected to a remote eventbus. """
test_value = []
def listener(event): # pylint: disable=unused-argument
""" Helper method that will verify our event got called. """
test_value.append(1)
self.eventbus.listen_once(ha.EVENT_STATE_CHANGED, listener)
self.sm_with_remote_eb.set_state("test", "local sm with remote eb")
# Allow the event to take place
time.sleep(1)
self.assertEqual(len(test_value), 1)