2016-11-25 21:04:06 +00:00
|
|
|
"""The tests for the Home Assistant HTTP component."""
|
2018-05-01 16:20:41 +00:00
|
|
|
import logging
|
2018-08-13 07:26:20 +00:00
|
|
|
import unittest
|
2018-08-14 06:20:17 +00:00
|
|
|
from unittest.mock import patch
|
2018-05-01 16:20:41 +00:00
|
|
|
|
2018-02-15 21:06:14 +00:00
|
|
|
from homeassistant.setup import async_setup_component
|
2016-11-25 21:04:06 +00:00
|
|
|
|
2018-02-15 21:06:14 +00:00
|
|
|
import homeassistant.components.http as http
|
2018-08-14 06:20:17 +00:00
|
|
|
from homeassistant.util.ssl import (
|
|
|
|
server_context_modern, server_context_intermediate)
|
2016-11-27 22:01:12 +00:00
|
|
|
|
|
|
|
|
|
|
|
class TestView(http.HomeAssistantView):
|
2016-12-28 18:04:59 +00:00
|
|
|
"""Test the HTTP views."""
|
2016-11-27 22:01:12 +00:00
|
|
|
|
|
|
|
name = 'test'
|
|
|
|
url = '/hello'
|
|
|
|
|
2018-03-09 01:51:49 +00:00
|
|
|
async def get(self, request):
|
2016-11-27 22:01:12 +00:00
|
|
|
"""Return a get request."""
|
|
|
|
return 'hello'
|
|
|
|
|
|
|
|
|
2018-03-15 20:49:49 +00:00
|
|
|
async def test_registering_view_while_running(hass, aiohttp_client,
|
|
|
|
aiohttp_unused_port):
|
2016-11-27 22:01:12 +00:00
|
|
|
"""Test that we can register a view while the server is running."""
|
2018-03-09 01:51:49 +00:00
|
|
|
await async_setup_component(
|
2016-11-27 22:01:12 +00:00
|
|
|
hass, http.DOMAIN, {
|
|
|
|
http.DOMAIN: {
|
2018-03-15 20:49:49 +00:00
|
|
|
http.CONF_SERVER_PORT: aiohttp_unused_port(),
|
2016-11-27 22:01:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2018-03-09 01:51:49 +00:00
|
|
|
await hass.async_start()
|
2017-11-06 02:42:31 +00:00
|
|
|
# This raises a RuntimeError if app is frozen
|
2016-11-27 22:01:12 +00:00
|
|
|
hass.http.register_view(TestView)
|
|
|
|
|
2016-12-18 20:56:07 +00:00
|
|
|
|
2018-08-13 07:26:20 +00:00
|
|
|
class TestApiConfig(unittest.TestCase):
|
|
|
|
"""Test API configuration methods."""
|
|
|
|
|
|
|
|
def test_api_base_url_with_domain(hass):
|
|
|
|
"""Test setting API URL with domain."""
|
|
|
|
api_config = http.ApiConfig('example.com')
|
|
|
|
assert api_config.base_url == 'http://example.com:8123'
|
|
|
|
|
|
|
|
def test_api_base_url_with_ip(hass):
|
|
|
|
"""Test setting API URL with IP."""
|
|
|
|
api_config = http.ApiConfig('1.1.1.1')
|
|
|
|
assert api_config.base_url == 'http://1.1.1.1:8123'
|
|
|
|
|
|
|
|
def test_api_base_url_with_ip_and_port(hass):
|
|
|
|
"""Test setting API URL with IP and port."""
|
|
|
|
api_config = http.ApiConfig('1.1.1.1', 8124)
|
|
|
|
assert api_config.base_url == 'http://1.1.1.1:8124'
|
|
|
|
|
|
|
|
def test_api_base_url_with_protocol(hass):
|
|
|
|
"""Test setting API URL with protocol."""
|
|
|
|
api_config = http.ApiConfig('https://example.com')
|
|
|
|
assert api_config.base_url == 'https://example.com:8123'
|
|
|
|
|
|
|
|
def test_api_base_url_with_protocol_and_port(hass):
|
|
|
|
"""Test setting API URL with protocol and port."""
|
|
|
|
api_config = http.ApiConfig('https://example.com', 433)
|
|
|
|
assert api_config.base_url == 'https://example.com:433'
|
|
|
|
|
|
|
|
def test_api_base_url_with_ssl_enable(hass):
|
|
|
|
"""Test setting API URL with use_ssl enabled."""
|
|
|
|
api_config = http.ApiConfig('example.com', use_ssl=True)
|
|
|
|
assert api_config.base_url == 'https://example.com:8123'
|
|
|
|
|
|
|
|
def test_api_base_url_with_ssl_enable_and_port(hass):
|
|
|
|
"""Test setting API URL with use_ssl enabled and port."""
|
|
|
|
api_config = http.ApiConfig('1.1.1.1', use_ssl=True, port=8888)
|
|
|
|
assert api_config.base_url == 'https://1.1.1.1:8888'
|
|
|
|
|
|
|
|
def test_api_base_url_with_protocol_and_ssl_enable(hass):
|
|
|
|
"""Test setting API URL with specific protocol and use_ssl enabled."""
|
|
|
|
api_config = http.ApiConfig('http://example.com', use_ssl=True)
|
|
|
|
assert api_config.base_url == 'http://example.com:8123'
|
|
|
|
|
|
|
|
|
2018-03-09 01:51:49 +00:00
|
|
|
async def test_api_base_url_with_domain(hass):
|
2017-11-04 19:04:05 +00:00
|
|
|
"""Test setting API URL."""
|
2018-03-09 01:51:49 +00:00
|
|
|
result = await async_setup_component(hass, 'http', {
|
2017-03-01 04:33:19 +00:00
|
|
|
'http': {
|
|
|
|
'base_url': 'example.com'
|
|
|
|
}
|
|
|
|
})
|
|
|
|
assert result
|
2016-12-18 22:59:45 +00:00
|
|
|
assert hass.config.api.base_url == 'http://example.com'
|
2016-12-18 20:56:07 +00:00
|
|
|
|
|
|
|
|
2018-03-09 01:51:49 +00:00
|
|
|
async def test_api_base_url_with_ip(hass):
|
2017-03-01 04:33:19 +00:00
|
|
|
"""Test setting api url."""
|
2018-03-09 01:51:49 +00:00
|
|
|
result = await async_setup_component(hass, 'http', {
|
2017-03-01 04:33:19 +00:00
|
|
|
'http': {
|
|
|
|
'server_host': '1.1.1.1'
|
|
|
|
}
|
|
|
|
})
|
|
|
|
assert result
|
2016-12-18 20:56:07 +00:00
|
|
|
assert hass.config.api.base_url == 'http://1.1.1.1:8123'
|
|
|
|
|
|
|
|
|
2018-03-09 01:51:49 +00:00
|
|
|
async def test_api_base_url_with_ip_port(hass):
|
2017-03-01 04:33:19 +00:00
|
|
|
"""Test setting api url."""
|
2018-03-09 01:51:49 +00:00
|
|
|
result = await async_setup_component(hass, 'http', {
|
2017-03-01 04:33:19 +00:00
|
|
|
'http': {
|
|
|
|
'base_url': '1.1.1.1:8124'
|
|
|
|
}
|
|
|
|
})
|
|
|
|
assert result
|
|
|
|
assert hass.config.api.base_url == 'http://1.1.1.1:8124'
|
2016-12-18 20:56:07 +00:00
|
|
|
|
|
|
|
|
2018-03-09 01:51:49 +00:00
|
|
|
async def test_api_no_base_url(hass):
|
2017-03-01 04:33:19 +00:00
|
|
|
"""Test setting api url."""
|
2018-03-09 01:51:49 +00:00
|
|
|
result = await async_setup_component(hass, 'http', {
|
2017-03-01 04:33:19 +00:00
|
|
|
'http': {
|
|
|
|
}
|
|
|
|
})
|
|
|
|
assert result
|
2016-12-18 20:56:07 +00:00
|
|
|
assert hass.config.api.base_url == 'http://127.0.0.1:8123'
|
2018-02-15 21:06:14 +00:00
|
|
|
|
|
|
|
|
2018-11-27 09:41:44 +00:00
|
|
|
async def test_not_log_password(hass, aiohttp_client, caplog, legacy_auth):
|
2018-02-15 21:06:14 +00:00
|
|
|
"""Test access with password doesn't get logged."""
|
2018-05-01 16:20:41 +00:00
|
|
|
assert await async_setup_component(hass, 'api', {
|
2018-02-15 21:06:14 +00:00
|
|
|
'http': {
|
|
|
|
http.CONF_API_PASSWORD: 'some-pass'
|
|
|
|
}
|
|
|
|
})
|
2018-03-15 20:49:49 +00:00
|
|
|
client = await aiohttp_client(hass.http.app)
|
2018-05-01 16:20:41 +00:00
|
|
|
logging.getLogger('aiohttp.access').setLevel(logging.INFO)
|
2018-02-15 21:06:14 +00:00
|
|
|
|
2018-03-09 01:51:49 +00:00
|
|
|
resp = await client.get('/api/', params={
|
2018-02-15 21:06:14 +00:00
|
|
|
'api_password': 'some-pass'
|
|
|
|
})
|
|
|
|
|
|
|
|
assert resp.status == 200
|
|
|
|
logs = caplog.text
|
|
|
|
|
|
|
|
# Ensure we don't log API passwords
|
|
|
|
assert '/api/' in logs
|
|
|
|
assert 'some-pass' not in logs
|
2018-08-03 11:52:34 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_proxy_config(hass):
|
|
|
|
"""Test use_x_forwarded_for must config together with trusted_proxies."""
|
|
|
|
assert await async_setup_component(hass, 'http', {
|
|
|
|
'http': {
|
|
|
|
http.CONF_USE_X_FORWARDED_FOR: True,
|
|
|
|
http.CONF_TRUSTED_PROXIES: ['127.0.0.1']
|
|
|
|
}
|
|
|
|
}) is True
|
|
|
|
|
|
|
|
|
|
|
|
async def test_proxy_config_only_use_xff(hass):
|
|
|
|
"""Test use_x_forwarded_for must config together with trusted_proxies."""
|
|
|
|
assert await async_setup_component(hass, 'http', {
|
|
|
|
'http': {
|
|
|
|
http.CONF_USE_X_FORWARDED_FOR: True
|
|
|
|
}
|
|
|
|
}) is not True
|
|
|
|
|
|
|
|
|
|
|
|
async def test_proxy_config_only_trust_proxies(hass):
|
|
|
|
"""Test use_x_forwarded_for must config together with trusted_proxies."""
|
|
|
|
assert await async_setup_component(hass, 'http', {
|
|
|
|
'http': {
|
|
|
|
http.CONF_TRUSTED_PROXIES: ['127.0.0.1']
|
|
|
|
}
|
|
|
|
}) is not True
|
2018-08-14 06:20:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_ssl_profile_defaults_modern(hass):
|
|
|
|
"""Test default ssl profile."""
|
|
|
|
assert await async_setup_component(hass, 'http', {}) is True
|
|
|
|
|
|
|
|
hass.http.ssl_certificate = 'bla'
|
|
|
|
|
|
|
|
with patch('ssl.SSLContext.load_cert_chain'), \
|
|
|
|
patch('homeassistant.util.ssl.server_context_modern',
|
|
|
|
side_effect=server_context_modern) as mock_context:
|
|
|
|
await hass.async_start()
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert len(mock_context.mock_calls) == 1
|
|
|
|
|
|
|
|
|
|
|
|
async def test_ssl_profile_change_intermediate(hass):
|
|
|
|
"""Test setting ssl profile to intermediate."""
|
|
|
|
assert await async_setup_component(hass, 'http', {
|
|
|
|
'http': {
|
|
|
|
'ssl_profile': 'intermediate'
|
|
|
|
}
|
|
|
|
}) is True
|
|
|
|
|
|
|
|
hass.http.ssl_certificate = 'bla'
|
|
|
|
|
|
|
|
with patch('ssl.SSLContext.load_cert_chain'), \
|
|
|
|
patch('homeassistant.util.ssl.server_context_intermediate',
|
|
|
|
side_effect=server_context_intermediate) as mock_context:
|
|
|
|
await hass.async_start()
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert len(mock_context.mock_calls) == 1
|
|
|
|
|
|
|
|
|
|
|
|
async def test_ssl_profile_change_modern(hass):
|
|
|
|
"""Test setting ssl profile to modern."""
|
|
|
|
assert await async_setup_component(hass, 'http', {
|
|
|
|
'http': {
|
|
|
|
'ssl_profile': 'modern'
|
|
|
|
}
|
|
|
|
}) is True
|
|
|
|
|
|
|
|
hass.http.ssl_certificate = 'bla'
|
|
|
|
|
|
|
|
with patch('ssl.SSLContext.load_cert_chain'), \
|
|
|
|
patch('homeassistant.util.ssl.server_context_modern',
|
|
|
|
side_effect=server_context_modern) as mock_context:
|
|
|
|
await hass.async_start()
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
|
|
|
|
assert len(mock_context.mock_calls) == 1
|