272 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			Python
		
	
	
			
		
		
	
	
			272 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			Python
		
	
	
"""Tests for deCONZ config flow."""
 | 
						|
from unittest.mock import patch
 | 
						|
 | 
						|
import asyncio
 | 
						|
 | 
						|
from homeassistant.components.deconz import config_flow
 | 
						|
from tests.common import MockConfigEntry
 | 
						|
 | 
						|
import pydeconz
 | 
						|
 | 
						|
 | 
						|
async def test_flow_works(hass, aioclient_mock):
 | 
						|
    """Test that config flow works."""
 | 
						|
    aioclient_mock.get(pydeconz.utils.URL_DISCOVER, json=[
 | 
						|
        {'id': 'id', 'internalipaddress': '1.2.3.4', 'internalport': 80}
 | 
						|
    ], headers={'content-type': 'application/json'})
 | 
						|
    aioclient_mock.post('http://1.2.3.4:80/api', json=[
 | 
						|
        {"success": {"username": "1234567890ABCDEF"}}
 | 
						|
    ], headers={'content-type': 'application/json'})
 | 
						|
 | 
						|
    flow = config_flow.DeconzFlowHandler()
 | 
						|
    flow.hass = hass
 | 
						|
 | 
						|
    await flow.async_step_user()
 | 
						|
    result = await flow.async_step_link(user_input={})
 | 
						|
 | 
						|
    assert result['type'] == 'create_entry'
 | 
						|
    assert result['title'] == 'deCONZ-id'
 | 
						|
    assert result['data'] == {
 | 
						|
        'bridgeid': 'id',
 | 
						|
        'host': '1.2.3.4',
 | 
						|
        'port': 80,
 | 
						|
        'api_key': '1234567890ABCDEF'
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
async def test_flow_bridge_discovery_fails(hass, aioclient_mock):
 | 
						|
    """Test config flow works when discovery fails."""
 | 
						|
    flow = config_flow.DeconzFlowHandler()
 | 
						|
    flow.hass = hass
 | 
						|
 | 
						|
    with patch('pydeconz.utils.async_discovery',
 | 
						|
               side_effect=asyncio.TimeoutError):
 | 
						|
        result = await flow.async_step_user()
 | 
						|
 | 
						|
    assert result['type'] == 'form'
 | 
						|
    assert result['step_id'] == 'init'
 | 
						|
 | 
						|
 | 
						|
async def test_flow_no_discovered_bridges(hass, aioclient_mock):
 | 
						|
    """Test config flow discovers no bridges."""
 | 
						|
    aioclient_mock.get(pydeconz.utils.URL_DISCOVER, json=[],
 | 
						|
                       headers={'content-type': 'application/json'})
 | 
						|
 | 
						|
    flow = config_flow.DeconzFlowHandler()
 | 
						|
    flow.hass = hass
 | 
						|
 | 
						|
    result = await flow.async_step_user()
 | 
						|
    assert result['type'] == 'form'
 | 
						|
    assert result['step_id'] == 'init'
 | 
						|
 | 
						|
 | 
						|
async def test_flow_one_bridge_discovered(hass, aioclient_mock):
 | 
						|
    """Test config flow discovers one bridge."""
 | 
						|
    aioclient_mock.get(pydeconz.utils.URL_DISCOVER, json=[
 | 
						|
        {'id': 'id', 'internalipaddress': '1.2.3.4', 'internalport': 80}
 | 
						|
    ], headers={'content-type': 'application/json'})
 | 
						|
 | 
						|
    flow = config_flow.DeconzFlowHandler()
 | 
						|
    flow.hass = hass
 | 
						|
 | 
						|
    result = await flow.async_step_user()
 | 
						|
    assert result['type'] == 'form'
 | 
						|
    assert result['step_id'] == 'link'
 | 
						|
    assert flow.deconz_config['host'] == '1.2.3.4'
 | 
						|
 | 
						|
 | 
						|
async def test_flow_two_bridges_discovered(hass, aioclient_mock):
 | 
						|
    """Test config flow discovers two bridges."""
 | 
						|
    aioclient_mock.get(pydeconz.utils.URL_DISCOVER, json=[
 | 
						|
        {'id': 'id1', 'internalipaddress': '1.2.3.4', 'internalport': 80},
 | 
						|
        {'id': 'id2', 'internalipaddress': '5.6.7.8', 'internalport': 80}
 | 
						|
    ], headers={'content-type': 'application/json'})
 | 
						|
 | 
						|
    flow = config_flow.DeconzFlowHandler()
 | 
						|
    flow.hass = hass
 | 
						|
 | 
						|
    result = await flow.async_step_user()
 | 
						|
    assert result['data_schema']({'host': '1.2.3.4'})
 | 
						|
    assert result['data_schema']({'host': '5.6.7.8'})
 | 
						|
 | 
						|
 | 
						|
async def test_flow_two_bridges_selection(hass, aioclient_mock):
 | 
						|
    """Test config flow selection of one of two bridges."""
 | 
						|
    flow = config_flow.DeconzFlowHandler()
 | 
						|
    flow.hass = hass
 | 
						|
    flow.bridges = [
 | 
						|
        {'bridgeid': 'id1', 'host': '1.2.3.4', 'port': 80},
 | 
						|
        {'bridgeid': 'id2', 'host': '5.6.7.8', 'port': 80}
 | 
						|
    ]
 | 
						|
 | 
						|
    result = await flow.async_step_user(user_input={'host': '1.2.3.4'})
 | 
						|
    assert result['type'] == 'form'
 | 
						|
    assert result['step_id'] == 'link'
 | 
						|
    assert flow.deconz_config['host'] == '1.2.3.4'
 | 
						|
 | 
						|
 | 
						|
async def test_flow_manual_configuration(hass, aioclient_mock):
 | 
						|
    """Test config flow with manual input."""
 | 
						|
    aioclient_mock.get(pydeconz.utils.URL_DISCOVER, json=[])
 | 
						|
 | 
						|
    flow = config_flow.DeconzFlowHandler()
 | 
						|
    flow.hass = hass
 | 
						|
 | 
						|
    user_input = {'host': '1.2.3.4', 'port': 80}
 | 
						|
 | 
						|
    result = await flow.async_step_user(user_input)
 | 
						|
    assert result['type'] == 'form'
 | 
						|
    assert result['step_id'] == 'link'
 | 
						|
    assert flow.deconz_config == user_input
 | 
						|
 | 
						|
 | 
						|
async def test_link_no_api_key(hass):
 | 
						|
    """Test config flow should abort if no API key was possible to retrieve."""
 | 
						|
    flow = config_flow.DeconzFlowHandler()
 | 
						|
    flow.hass = hass
 | 
						|
    flow.deconz_config = {'host': '1.2.3.4', 'port': 80}
 | 
						|
 | 
						|
    with patch('pydeconz.utils.async_get_api_key',
 | 
						|
               side_effect=pydeconz.errors.ResponseError):
 | 
						|
        result = await flow.async_step_link(user_input={})
 | 
						|
 | 
						|
    assert result['type'] == 'form'
 | 
						|
    assert result['step_id'] == 'link'
 | 
						|
    assert result['errors'] == {'base': 'no_key'}
 | 
						|
 | 
						|
 | 
						|
async def test_bridge_discovery(hass):
 | 
						|
    """Test a bridge being discovered."""
 | 
						|
    flow = config_flow.DeconzFlowHandler()
 | 
						|
    flow.hass = hass
 | 
						|
 | 
						|
    result = await flow.async_step_discovery({
 | 
						|
        'host': '1.2.3.4',
 | 
						|
        'port': 80,
 | 
						|
        'serial': 'id'
 | 
						|
    })
 | 
						|
 | 
						|
    assert result['type'] == 'form'
 | 
						|
    assert result['step_id'] == 'link'
 | 
						|
 | 
						|
 | 
						|
async def test_bridge_discovery_already_configured(hass):
 | 
						|
    """Test if a discovered bridge has already been configured."""
 | 
						|
    MockConfigEntry(domain='deconz', data={
 | 
						|
        'host': '1.2.3.4'
 | 
						|
    }).add_to_hass(hass)
 | 
						|
 | 
						|
    flow = config_flow.DeconzFlowHandler()
 | 
						|
    flow.hass = hass
 | 
						|
 | 
						|
    result = await flow.async_step_discovery({
 | 
						|
        'host': '1.2.3.4',
 | 
						|
        'port': 80,
 | 
						|
        'serial': 'id'
 | 
						|
    })
 | 
						|
 | 
						|
    assert result['type'] == 'abort'
 | 
						|
 | 
						|
 | 
						|
async def test_import_without_api_key(hass):
 | 
						|
    """Test importing a host without an API key."""
 | 
						|
    flow = config_flow.DeconzFlowHandler()
 | 
						|
    flow.hass = hass
 | 
						|
 | 
						|
    result = await flow.async_step_import({
 | 
						|
        'host': '1.2.3.4',
 | 
						|
    })
 | 
						|
 | 
						|
    assert result['type'] == 'form'
 | 
						|
    assert result['step_id'] == 'link'
 | 
						|
 | 
						|
 | 
						|
async def test_import_with_api_key(hass):
 | 
						|
    """Test importing a host with an API key."""
 | 
						|
    flow = config_flow.DeconzFlowHandler()
 | 
						|
    flow.hass = hass
 | 
						|
 | 
						|
    result = await flow.async_step_import({
 | 
						|
        'bridgeid': 'id',
 | 
						|
        'host': '1.2.3.4',
 | 
						|
        'port': 80,
 | 
						|
        'api_key': '1234567890ABCDEF'
 | 
						|
    })
 | 
						|
 | 
						|
    assert result['type'] == 'create_entry'
 | 
						|
    assert result['title'] == 'deCONZ-id'
 | 
						|
    assert result['data'] == {
 | 
						|
        'bridgeid': 'id',
 | 
						|
        'host': '1.2.3.4',
 | 
						|
        'port': 80,
 | 
						|
        'api_key': '1234567890ABCDEF'
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
async def test_create_entry(hass, aioclient_mock):
 | 
						|
    """Test that _create_entry work and that bridgeid can be requested."""
 | 
						|
    aioclient_mock.get('http://1.2.3.4:80/api/1234567890ABCDEF/config',
 | 
						|
                       json={"bridgeid": "id"},
 | 
						|
                       headers={'content-type': 'application/json'})
 | 
						|
 | 
						|
    flow = config_flow.DeconzFlowHandler()
 | 
						|
    flow.hass = hass
 | 
						|
    flow.deconz_config = {'host': '1.2.3.4',
 | 
						|
                          'port': 80,
 | 
						|
                          'api_key': '1234567890ABCDEF'}
 | 
						|
 | 
						|
    result = await flow._create_entry()
 | 
						|
 | 
						|
    assert result['type'] == 'create_entry'
 | 
						|
    assert result['title'] == 'deCONZ-id'
 | 
						|
    assert result['data'] == {
 | 
						|
        'bridgeid': 'id',
 | 
						|
        'host': '1.2.3.4',
 | 
						|
        'port': 80,
 | 
						|
        'api_key': '1234567890ABCDEF'
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
async def test_hassio_single_instance(hass):
 | 
						|
    """Test we only allow a single config flow."""
 | 
						|
    MockConfigEntry(domain='deconz', data={
 | 
						|
        'host': '1.2.3.4'
 | 
						|
    }).add_to_hass(hass)
 | 
						|
 | 
						|
    result = await hass.config_entries.flow.async_init(
 | 
						|
        'deconz', context={'source': 'hassio'})
 | 
						|
    assert result['type'] == 'abort'
 | 
						|
    assert result['reason'] == 'one_instance_only'
 | 
						|
 | 
						|
 | 
						|
async def test_hassio_confirm(hass):
 | 
						|
    """Test we can finish a config flow."""
 | 
						|
    result = await hass.config_entries.flow.async_init(
 | 
						|
        'deconz',
 | 
						|
        data={
 | 
						|
            'addon': 'Mock Addon',
 | 
						|
            'host': 'mock-deconz',
 | 
						|
            'port': 80,
 | 
						|
            'serial': 'id',
 | 
						|
            'api_key': '1234567890ABCDEF',
 | 
						|
        },
 | 
						|
        context={'source': 'hassio'}
 | 
						|
    )
 | 
						|
    assert result['type'] == 'form'
 | 
						|
    assert result['step_id'] == 'hassio_confirm'
 | 
						|
    assert result['description_placeholders'] == {
 | 
						|
        'addon': 'Mock Addon',
 | 
						|
    }
 | 
						|
 | 
						|
    result = await hass.config_entries.flow.async_configure(
 | 
						|
        result['flow_id'], user_input={}
 | 
						|
    )
 | 
						|
 | 
						|
    assert result['type'] == 'create_entry'
 | 
						|
    assert result['result'].data == {
 | 
						|
        'host': 'mock-deconz',
 | 
						|
        'port': 80,
 | 
						|
        'bridgeid': 'id',
 | 
						|
        'api_key': '1234567890ABCDEF'
 | 
						|
    }
 |