core/tests/components/test_api.py

484 lines
16 KiB
Python
Raw Normal View History

2014-11-23 17:51:16 +00:00
"""
tests.components.test_api
~~~~~~~~~~~~~~~~~~~~~~~~~
2014-11-23 17:51:16 +00:00
2014-11-23 20:57:29 +00:00
Tests Home Assistant HTTP component does what it should do.
2014-11-23 17:51:16 +00:00
"""
# pylint: disable=protected-access,too-many-public-methods
2015-12-15 07:20:43 +00:00
from contextlib import closing
2014-11-23 17:51:16 +00:00
import json
import tempfile
2015-12-15 07:20:43 +00:00
import unittest
from unittest.mock import patch
2014-11-23 17:51:16 +00:00
import requests
from homeassistant import bootstrap, const
2015-08-17 03:44:46 +00:00
import homeassistant.core as ha
2014-11-23 17:51:16 +00:00
import homeassistant.components.http as http
2016-02-14 23:08:23 +00:00
from tests.common import get_test_instance_port, get_test_home_assistant
2014-11-23 20:57:29 +00:00
API_PASSWORD = "test1234"
SERVER_PORT = get_test_instance_port()
2014-11-23 20:57:29 +00:00
HTTP_BASE_URL = "http://127.0.0.1:{}".format(SERVER_PORT)
HA_HEADERS = {const.HTTP_HEADER_HA_AUTH: API_PASSWORD}
2014-11-23 17:51:16 +00:00
2014-11-24 06:18:51 +00:00
hass = None
2014-11-23 17:51:16 +00:00
2014-11-23 20:57:29 +00:00
def _url(path=""):
""" Helper method to generate urls. """
return HTTP_BASE_URL + path
def setUpModule(): # pylint: disable=invalid-name
2016-02-01 10:49:44 +00:00
""" Initializes a Home Assistant server. """
2014-11-23 20:57:29 +00:00
global hass
2016-02-14 23:08:23 +00:00
hass = get_test_home_assistant()
2014-11-23 20:57:29 +00:00
hass.bus.listen('test_event', lambda _: _)
hass.states.set('test.test', 'a_state')
bootstrap.setup_component(
hass, http.DOMAIN,
{http.DOMAIN: {http.CONF_API_PASSWORD: API_PASSWORD,
http.CONF_SERVER_PORT: SERVER_PORT}})
bootstrap.setup_component(hass, 'api')
2014-11-23 20:57:29 +00:00
hass.start()
2014-11-24 06:18:51 +00:00
def tearDownModule(): # pylint: disable=invalid-name
2014-11-23 20:57:29 +00:00
""" Stops the Home Assistant server. """
hass.stop()
class TestAPI(unittest.TestCase):
""" Test the API. """
2016-02-15 07:01:49 +00:00
def tearDown(self):
hass.pool.block_till_done()
# TODO move back to http component and test with use_auth.
def test_access_denied_without_password(self):
req = requests.get(_url(const.URL_API))
2014-11-23 17:51:16 +00:00
self.assertEqual(401, req.status_code)
def test_access_denied_with_wrong_password(self):
2014-11-23 17:51:16 +00:00
req = requests.get(
_url(const.URL_API),
headers={const.HTTP_HEADER_HA_AUTH: 'wrongpassword'})
2014-11-23 17:51:16 +00:00
self.assertEqual(401, req.status_code)
def test_access_with_password_in_url(self):
req = requests.get(
"{}?api_password={}".format(_url(const.URL_API), API_PASSWORD))
self.assertEqual(200, req.status_code)
def test_access_via_session(self):
session = requests.Session()
req = session.get(_url(const.URL_API), headers=HA_HEADERS)
self.assertEqual(200, req.status_code)
req = session.get(_url(const.URL_API))
self.assertEqual(200, req.status_code)
2014-11-23 17:51:16 +00:00
def test_api_list_state_entities(self):
""" Test if the debug interface allows us to list state entities. """
req = requests.get(_url(const.URL_API_STATES),
2014-11-23 17:51:16 +00:00
headers=HA_HEADERS)
remote_data = [ha.State.from_dict(item) for item in req.json()]
2014-11-23 20:57:29 +00:00
self.assertEqual(hass.states.all(), remote_data)
2014-11-23 17:51:16 +00:00
def test_api_get_state(self):
""" Test if the debug interface allows us to get a state. """
req = requests.get(
_url(const.URL_API_STATES_ENTITY.format("test.test")),
2014-11-23 17:51:16 +00:00
headers=HA_HEADERS)
data = ha.State.from_dict(req.json())
2014-11-23 20:57:29 +00:00
state = hass.states.get("test.test")
2014-11-23 17:51:16 +00:00
self.assertEqual(state.state, data.state)
self.assertEqual(state.last_changed, data.last_changed)
self.assertEqual(state.attributes, data.attributes)
def test_api_get_non_existing_state(self):
""" Test if the debug interface allows us to get a state. """
req = requests.get(
_url(const.URL_API_STATES_ENTITY.format("does_not_exist")),
2014-11-23 17:51:16 +00:00
headers=HA_HEADERS)
self.assertEqual(404, req.status_code)
def test_api_state_change(self):
""" Test if we can change the state of an entity that exists. """
2014-11-23 20:57:29 +00:00
hass.states.set("test.test", "not_to_be_set")
2014-11-23 17:51:16 +00:00
requests.post(_url(const.URL_API_STATES_ENTITY.format("test.test")),
data=json.dumps({"state": "debug_state_change2"}),
headers=HA_HEADERS)
2014-11-23 17:51:16 +00:00
self.assertEqual("debug_state_change2",
2014-11-23 20:57:29 +00:00
hass.states.get("test.test").state)
2014-11-23 17:51:16 +00:00
# pylint: disable=invalid-name
def test_api_state_change_of_non_existing_entity(self):
""" Test if the API allows us to change a state of
a non existing entity. """
new_state = "debug_state_change"
req = requests.post(
_url(const.URL_API_STATES_ENTITY.format(
2014-11-23 17:51:16 +00:00
"test_entity.that_does_not_exist")),
data=json.dumps({'state': new_state}),
headers=HA_HEADERS)
2014-11-23 17:51:16 +00:00
2014-11-23 20:57:29 +00:00
cur_state = (hass.states.
2014-11-23 17:51:16 +00:00
get("test_entity.that_does_not_exist").state)
self.assertEqual(201, req.status_code)
self.assertEqual(cur_state, new_state)
# pylint: disable=invalid-name
def test_api_state_change_with_bad_data(self):
""" Test if API sends appropriate error if we omit state. """
req = requests.post(
_url(const.URL_API_STATES_ENTITY.format(
"test_entity.that_does_not_exist")),
data=json.dumps({}),
headers=HA_HEADERS)
self.assertEqual(400, req.status_code)
2014-11-23 17:51:16 +00:00
# pylint: disable=invalid-name
def test_api_fire_event_with_no_data(self):
""" Test if the API allows us to fire an event. """
test_value = []
2015-01-20 07:40:51 +00:00
def listener(event):
2014-11-23 17:51:16 +00:00
""" Helper method that will verify our event got called. """
test_value.append(1)
2014-11-29 07:19:59 +00:00
hass.bus.listen_once("test.event_no_data", listener)
2014-11-23 17:51:16 +00:00
requests.post(
_url(const.URL_API_EVENTS_EVENT.format("test.event_no_data")),
2014-11-23 17:51:16 +00:00
headers=HA_HEADERS)
hass.pool.block_till_done()
2014-11-23 17:51:16 +00:00
self.assertEqual(1, len(test_value))
# pylint: disable=invalid-name
def test_api_fire_event_with_data(self):
""" Test if the API allows us to fire an event. """
test_value = []
2015-01-20 07:40:51 +00:00
def listener(event):
2014-11-23 17:51:16 +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)
2014-11-29 07:19:59 +00:00
hass.bus.listen_once("test_event_with_data", listener)
2014-11-23 17:51:16 +00:00
requests.post(
_url(const.URL_API_EVENTS_EVENT.format("test_event_with_data")),
2014-11-23 17:51:16 +00:00
data=json.dumps({"test": 1}),
headers=HA_HEADERS)
hass.pool.block_till_done()
2014-11-23 17:51:16 +00:00
self.assertEqual(1, len(test_value))
# pylint: disable=invalid-name
def test_api_fire_event_with_invalid_json(self):
""" Test if the API allows us to fire an event. """
test_value = []
2015-01-20 07:40:51 +00:00
def listener(event):
2014-11-23 17:51:16 +00:00
""" Helper method that will verify our event got called. """
test_value.append(1)
2014-11-29 07:19:59 +00:00
hass.bus.listen_once("test_event_bad_data", listener)
2014-11-23 17:51:16 +00:00
req = requests.post(
_url(const.URL_API_EVENTS_EVENT.format("test_event_bad_data")),
2014-11-23 17:51:16 +00:00
data=json.dumps('not an object'),
headers=HA_HEADERS)
hass.pool.block_till_done()
2014-11-23 17:51:16 +00:00
self.assertEqual(422, req.status_code)
self.assertEqual(0, len(test_value))
# Try now with valid but unusable JSON
req = requests.post(
_url(const.URL_API_EVENTS_EVENT.format("test_event_bad_data")),
data=json.dumps([1, 2, 3]),
headers=HA_HEADERS)
hass.pool.block_till_done()
self.assertEqual(422, req.status_code)
self.assertEqual(0, len(test_value))
def test_api_get_config(self):
req = requests.get(_url(const.URL_API_CONFIG),
headers=HA_HEADERS)
self.assertEqual(hass.config.as_dict(), req.json())
def test_api_get_components(self):
req = requests.get(_url(const.URL_API_COMPONENTS),
headers=HA_HEADERS)
self.assertEqual(hass.config.components, req.json())
def test_api_get_error_log(self):
test_content = 'Test String'
with tempfile.NamedTemporaryFile() as log:
log.write(test_content.encode('utf-8'))
log.flush()
with patch.object(hass.config, 'path', return_value=log.name):
req = requests.get(_url(const.URL_API_ERROR_LOG),
headers=HA_HEADERS)
self.assertEqual(test_content, req.text)
self.assertIsNone(req.headers.get('expires'))
2014-11-23 17:51:16 +00:00
def test_api_get_event_listeners(self):
""" Test if we can get the list of events being listened for. """
req = requests.get(_url(const.URL_API_EVENTS),
2014-11-23 17:51:16 +00:00
headers=HA_HEADERS)
2014-11-23 20:57:29 +00:00
local = hass.bus.listeners
2014-11-23 17:51:16 +00:00
for event in req.json():
self.assertEqual(event["listener_count"],
local.pop(event["event"]))
self.assertEqual(0, len(local))
def test_api_get_services(self):
""" Test if we can get a dict describing current services. """
req = requests.get(_url(const.URL_API_SERVICES),
2014-11-23 17:51:16 +00:00
headers=HA_HEADERS)
2014-11-23 20:57:29 +00:00
local_services = hass.services.services
2014-11-23 17:51:16 +00:00
for serv_domain in req.json():
local = local_services.pop(serv_domain["domain"])
self.assertEqual(local, serv_domain["services"])
def test_api_call_service_no_data(self):
""" Test if the API allows us to call a service. """
test_value = []
2015-01-20 07:40:51 +00:00
def listener(service_call):
2014-11-23 17:51:16 +00:00
""" Helper method that will verify that our service got called. """
test_value.append(1)
2014-11-23 20:57:29 +00:00
hass.services.register("test_domain", "test_service", listener)
2014-11-23 17:51:16 +00:00
requests.post(
_url(const.URL_API_SERVICES_SERVICE.format(
2014-11-23 17:51:16 +00:00
"test_domain", "test_service")),
headers=HA_HEADERS)
hass.pool.block_till_done()
2014-11-23 17:51:16 +00:00
self.assertEqual(1, len(test_value))
def test_api_call_service_with_data(self):
""" Test if the API allows us to call a service. """
test_value = []
2015-01-20 07:40:51 +00:00
def listener(service_call):
2014-11-23 17:51:16 +00:00
""" Helper method that will verify that our service got called and
that test if our data came through. """
if "test" in service_call.data:
test_value.append(1)
2014-11-23 20:57:29 +00:00
hass.services.register("test_domain", "test_service", listener)
2014-11-23 17:51:16 +00:00
requests.post(
_url(const.URL_API_SERVICES_SERVICE.format(
2014-11-23 17:51:16 +00:00
"test_domain", "test_service")),
data=json.dumps({"test": 1}),
headers=HA_HEADERS)
hass.pool.block_till_done()
2014-11-23 17:51:16 +00:00
self.assertEqual(1, len(test_value))
2015-12-10 07:56:20 +00:00
def test_api_template(self):
""" Test template API. """
hass.states.set('sensor.temperature', 10)
req = requests.post(
_url(const.URL_API_TEMPLATE),
data=json.dumps({"template":
'{{ states.sensor.temperature.state }}'}),
headers=HA_HEADERS)
self.assertEqual('10', req.text)
2015-12-18 07:33:09 +00:00
def test_api_template_error(self):
""" Test template API. """
hass.states.set('sensor.temperature', 10)
req = requests.post(
_url(const.URL_API_TEMPLATE),
data=json.dumps({"template":
'{{ states.sensor.temperature.state'}),
headers=HA_HEADERS)
self.assertEqual(422, req.status_code)
def test_api_event_forward(self):
""" Test setting up event forwarding. """
req = requests.post(
_url(const.URL_API_EVENT_FORWARD),
headers=HA_HEADERS)
self.assertEqual(400, req.status_code)
req = requests.post(
_url(const.URL_API_EVENT_FORWARD),
data=json.dumps({'host': '127.0.0.1'}),
headers=HA_HEADERS)
self.assertEqual(400, req.status_code)
req = requests.post(
_url(const.URL_API_EVENT_FORWARD),
data=json.dumps({'api_password': 'bla-di-bla'}),
headers=HA_HEADERS)
self.assertEqual(400, req.status_code)
req = requests.post(
_url(const.URL_API_EVENT_FORWARD),
data=json.dumps({
'api_password': 'bla-di-bla',
'host': '127.0.0.1',
'port': 'abcd'
}),
headers=HA_HEADERS)
self.assertEqual(422, req.status_code)
req = requests.post(
_url(const.URL_API_EVENT_FORWARD),
data=json.dumps({
'api_password': 'bla-di-bla',
'host': '127.0.0.1',
'port': get_test_instance_port()
}),
headers=HA_HEADERS)
self.assertEqual(422, req.status_code)
# Setup a real one
req = requests.post(
_url(const.URL_API_EVENT_FORWARD),
data=json.dumps({
'api_password': API_PASSWORD,
'host': '127.0.0.1',
'port': SERVER_PORT
}),
headers=HA_HEADERS)
self.assertEqual(200, req.status_code)
# Delete it again..
req = requests.delete(
_url(const.URL_API_EVENT_FORWARD),
data=json.dumps({}),
headers=HA_HEADERS)
self.assertEqual(400, req.status_code)
req = requests.delete(
_url(const.URL_API_EVENT_FORWARD),
data=json.dumps({
'host': '127.0.0.1',
'port': 'abcd'
}),
headers=HA_HEADERS)
self.assertEqual(422, req.status_code)
req = requests.delete(
_url(const.URL_API_EVENT_FORWARD),
data=json.dumps({
'host': '127.0.0.1',
'port': SERVER_PORT
}),
headers=HA_HEADERS)
self.assertEqual(200, req.status_code)
2015-12-15 07:20:43 +00:00
def test_stream(self):
listen_count = self._listen_count()
with closing(requests.get(_url(const.URL_API_STREAM),
stream=True, headers=HA_HEADERS)) as req:
data = self._stream_next_event(req)
self.assertEqual('ping', data)
2015-12-15 07:27:22 +00:00
self.assertEqual(listen_count + 1, self._listen_count())
2015-12-15 07:20:43 +00:00
hass.bus.fire('test_event')
hass.pool.block_till_done()
data = self._stream_next_event(req)
self.assertEqual('test_event', data['event_type'])
def test_stream_with_restricted(self):
listen_count = self._listen_count()
with closing(requests.get(_url(const.URL_API_STREAM),
data=json.dumps({
'restrict': 'test_event1,test_event3'}),
stream=True, headers=HA_HEADERS)) as req:
data = self._stream_next_event(req)
self.assertEqual('ping', data)
2015-12-15 07:27:22 +00:00
self.assertEqual(listen_count + 2, self._listen_count())
2015-12-15 07:20:43 +00:00
hass.bus.fire('test_event1')
hass.pool.block_till_done()
hass.bus.fire('test_event2')
hass.pool.block_till_done()
hass.bus.fire('test_event3')
hass.pool.block_till_done()
data = self._stream_next_event(req)
self.assertEqual('test_event1', data['event_type'])
data = self._stream_next_event(req)
self.assertEqual('test_event3', data['event_type'])
def _stream_next_event(self, stream):
data = b''
last_new_line = False
for dat in stream.iter_content(1):
if dat == b'\n' and last_new_line:
break
data += dat
last_new_line = dat == b'\n'
conv = data.decode('utf-8').strip()[6:]
return conv if conv == 'ping' else json.loads(conv)
def _listen_count(self):
""" Return number of event listeners. """
return sum(hass.bus.listeners.values())