2018-02-16 22:07:38 +00:00
|
|
|
"""Test config entries API."""
|
|
|
|
|
|
|
|
from collections import OrderedDict
|
2021-01-01 21:31:56 +00:00
|
|
|
from unittest.mock import AsyncMock, patch
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
import pytest
|
|
|
|
import voluptuous as vol
|
|
|
|
|
2019-02-22 16:59:43 +00:00
|
|
|
from homeassistant import config_entries as core_ce, data_entry_flow
|
2019-12-08 16:57:28 +00:00
|
|
|
from homeassistant.components.config import config_entries
|
2018-04-13 14:14:53 +00:00
|
|
|
from homeassistant.config_entries import HANDLERS
|
2019-02-22 16:59:43 +00:00
|
|
|
from homeassistant.core import callback
|
2019-05-13 08:16:55 +00:00
|
|
|
from homeassistant.generated import config_flows
|
2019-12-08 16:57:28 +00:00
|
|
|
from homeassistant.setup import async_setup_component
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-04-15 02:07:05 +00:00
|
|
|
from tests.common import (
|
2019-07-31 19:25:30 +00:00
|
|
|
MockConfigEntry,
|
|
|
|
MockModule,
|
|
|
|
mock_entity_platform,
|
2019-12-08 16:57:28 +00:00
|
|
|
mock_integration,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
|
2018-05-01 18:57:30 +00:00
|
|
|
@pytest.fixture(autouse=True)
|
|
|
|
def mock_test_component(hass):
|
2018-04-17 09:44:32 +00:00
|
|
|
"""Ensure a component called 'test' exists."""
|
2019-07-31 19:25:30 +00:00
|
|
|
mock_integration(hass, MockModule("test"))
|
2018-04-17 09:44:32 +00:00
|
|
|
|
|
|
|
|
2018-02-16 22:07:38 +00:00
|
|
|
@pytest.fixture
|
2018-12-02 15:32:53 +00:00
|
|
|
def client(hass, hass_client):
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Fixture that can interact with the config manager API."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.loop.run_until_complete(async_setup_component(hass, "http", {}))
|
2018-02-16 22:07:38 +00:00
|
|
|
hass.loop.run_until_complete(config_entries.async_setup(hass))
|
2018-12-02 15:32:53 +00:00
|
|
|
yield hass.loop.run_until_complete(hass_client())
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
|
2019-02-22 16:59:43 +00:00
|
|
|
async def test_get_entries(hass, client):
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Test get entries."""
|
2019-08-16 23:19:19 +00:00
|
|
|
with patch.dict(HANDLERS, clear=True):
|
|
|
|
|
|
|
|
@HANDLERS.register("comp1")
|
|
|
|
class Comp1ConfigFlow:
|
|
|
|
"""Config flow with options flow."""
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
@callback
|
|
|
|
def async_get_options_flow(config, options):
|
|
|
|
"""Get options flow."""
|
|
|
|
pass
|
|
|
|
|
|
|
|
hass.helpers.config_entry_flow.register_discovery_flow(
|
|
|
|
"comp2", "Comp 2", lambda: None, core_ce.CONN_CLASS_ASSUMED
|
|
|
|
)
|
|
|
|
|
2020-08-25 22:59:22 +00:00
|
|
|
entry = MockConfigEntry(
|
2019-08-16 23:19:19 +00:00
|
|
|
domain="comp1",
|
|
|
|
title="Test 1",
|
|
|
|
source="bla",
|
|
|
|
connection_class=core_ce.CONN_CLASS_LOCAL_POLL,
|
2020-08-25 22:59:22 +00:00
|
|
|
)
|
|
|
|
entry.supports_unload = True
|
|
|
|
entry.add_to_hass(hass)
|
2019-08-16 23:19:19 +00:00
|
|
|
MockConfigEntry(
|
|
|
|
domain="comp2",
|
|
|
|
title="Test 2",
|
|
|
|
source="bla2",
|
|
|
|
state=core_ce.ENTRY_STATE_LOADED,
|
|
|
|
connection_class=core_ce.CONN_CLASS_ASSUMED,
|
|
|
|
).add_to_hass(hass)
|
2021-02-21 03:21:39 +00:00
|
|
|
MockConfigEntry(
|
|
|
|
domain="comp3",
|
|
|
|
title="Test 3",
|
|
|
|
source="bla3",
|
|
|
|
disabled_by="user",
|
|
|
|
).add_to_hass(hass)
|
2019-08-16 23:19:19 +00:00
|
|
|
|
|
|
|
resp = await client.get("/api/config/config_entries/entry")
|
|
|
|
assert resp.status == 200
|
|
|
|
data = await resp.json()
|
|
|
|
for entry in data:
|
|
|
|
entry.pop("entry_id")
|
|
|
|
assert data == [
|
|
|
|
{
|
|
|
|
"domain": "comp1",
|
|
|
|
"title": "Test 1",
|
|
|
|
"source": "bla",
|
|
|
|
"state": "not_loaded",
|
|
|
|
"connection_class": "local_poll",
|
|
|
|
"supports_options": True,
|
2020-08-25 22:59:22 +00:00
|
|
|
"supports_unload": True,
|
2021-02-21 03:21:39 +00:00
|
|
|
"disabled_by": None,
|
2019-08-16 23:19:19 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"domain": "comp2",
|
|
|
|
"title": "Test 2",
|
|
|
|
"source": "bla2",
|
|
|
|
"state": "loaded",
|
|
|
|
"connection_class": "assumed",
|
|
|
|
"supports_options": False,
|
2020-08-25 22:59:22 +00:00
|
|
|
"supports_unload": False,
|
2021-02-21 03:21:39 +00:00
|
|
|
"disabled_by": None,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"domain": "comp3",
|
|
|
|
"title": "Test 3",
|
|
|
|
"source": "bla3",
|
|
|
|
"state": "not_loaded",
|
|
|
|
"connection_class": "unknown",
|
|
|
|
"supports_options": False,
|
|
|
|
"supports_unload": False,
|
|
|
|
"disabled_by": "user",
|
2019-08-16 23:19:19 +00:00
|
|
|
},
|
|
|
|
]
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
|
2020-01-01 23:16:27 +00:00
|
|
|
async def test_remove_entry(hass, client):
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Test removing an entry via the API."""
|
2019-07-31 19:25:30 +00:00
|
|
|
entry = MockConfigEntry(domain="demo", state=core_ce.ENTRY_STATE_LOADED)
|
2018-02-16 22:07:38 +00:00
|
|
|
entry.add_to_hass(hass)
|
2020-04-04 22:26:08 +00:00
|
|
|
resp = await client.delete(f"/api/config/config_entries/entry/{entry.entry_id}")
|
2018-02-16 22:07:38 +00:00
|
|
|
assert resp.status == 200
|
2020-01-01 23:16:27 +00:00
|
|
|
data = await resp.json()
|
2019-07-31 19:25:30 +00:00
|
|
|
assert data == {"require_restart": True}
|
2018-02-16 22:07:38 +00:00
|
|
|
assert len(hass.config_entries.async_entries()) == 0
|
|
|
|
|
|
|
|
|
2020-08-25 22:59:22 +00:00
|
|
|
async def test_reload_entry(hass, client):
|
|
|
|
"""Test reloading an entry via the API."""
|
|
|
|
entry = MockConfigEntry(domain="demo", state=core_ce.ENTRY_STATE_LOADED)
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
resp = await client.post(
|
|
|
|
f"/api/config/config_entries/entry/{entry.entry_id}/reload"
|
|
|
|
)
|
|
|
|
assert resp.status == 200
|
|
|
|
data = await resp.json()
|
|
|
|
assert data == {"require_restart": True}
|
|
|
|
assert len(hass.config_entries.async_entries()) == 1
|
|
|
|
|
|
|
|
|
|
|
|
async def test_reload_invalid_entry(hass, client):
|
|
|
|
"""Test reloading an invalid entry via the API."""
|
|
|
|
resp = await client.post("/api/config/config_entries/entry/invalid/reload")
|
|
|
|
assert resp.status == 404
|
|
|
|
|
|
|
|
|
2018-12-13 14:30:20 +00:00
|
|
|
async def test_remove_entry_unauth(hass, client, hass_admin_user):
|
|
|
|
"""Test removing an entry via the API."""
|
|
|
|
hass_admin_user.groups = []
|
2019-07-31 19:25:30 +00:00
|
|
|
entry = MockConfigEntry(domain="demo", state=core_ce.ENTRY_STATE_LOADED)
|
2018-12-13 14:30:20 +00:00
|
|
|
entry.add_to_hass(hass)
|
2020-04-04 22:26:08 +00:00
|
|
|
resp = await client.delete(f"/api/config/config_entries/entry/{entry.entry_id}")
|
2018-12-13 14:30:20 +00:00
|
|
|
assert resp.status == 401
|
|
|
|
assert len(hass.config_entries.async_entries()) == 1
|
|
|
|
|
|
|
|
|
2020-08-25 22:59:22 +00:00
|
|
|
async def test_reload_entry_unauth(hass, client, hass_admin_user):
|
|
|
|
"""Test reloading an entry via the API."""
|
|
|
|
hass_admin_user.groups = []
|
|
|
|
entry = MockConfigEntry(domain="demo", state=core_ce.ENTRY_STATE_LOADED)
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
resp = await client.post(
|
|
|
|
f"/api/config/config_entries/entry/{entry.entry_id}/reload"
|
|
|
|
)
|
|
|
|
assert resp.status == 401
|
|
|
|
assert len(hass.config_entries.async_entries()) == 1
|
|
|
|
|
|
|
|
|
|
|
|
async def test_reload_entry_in_failed_state(hass, client, hass_admin_user):
|
|
|
|
"""Test reloading an entry via the API that has already failed to unload."""
|
|
|
|
entry = MockConfigEntry(domain="demo", state=core_ce.ENTRY_STATE_FAILED_UNLOAD)
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
resp = await client.post(
|
|
|
|
f"/api/config/config_entries/entry/{entry.entry_id}/reload"
|
|
|
|
)
|
|
|
|
assert resp.status == 403
|
|
|
|
assert len(hass.config_entries.async_entries()) == 1
|
|
|
|
|
|
|
|
|
2020-01-01 23:16:27 +00:00
|
|
|
async def test_available_flows(hass, client):
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Test querying the available flows."""
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.object(config_flows, "FLOWS", ["hello", "world"]):
|
2020-01-01 23:16:27 +00:00
|
|
|
resp = await client.get("/api/config/config_entries/flow_handlers")
|
2018-02-16 22:07:38 +00:00
|
|
|
assert resp.status == 200
|
2020-01-01 23:16:27 +00:00
|
|
|
data = await resp.json()
|
2020-04-04 22:26:08 +00:00
|
|
|
assert set(data) == {"hello", "world"}
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
############################
|
|
|
|
# FLOW MANAGER API TESTS #
|
|
|
|
############################
|
|
|
|
|
|
|
|
|
2020-01-01 23:16:27 +00:00
|
|
|
async def test_initialize_flow(hass, client):
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Test we can initialize a flow."""
|
2019-07-31 19:25:30 +00:00
|
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
2019-05-13 08:16:55 +00:00
|
|
|
|
2018-09-17 08:12:46 +00:00
|
|
|
class TestFlow(core_ce.ConfigFlow):
|
2020-01-01 23:16:27 +00:00
|
|
|
async def async_step_user(self, user_input=None):
|
2018-02-16 22:07:38 +00:00
|
|
|
schema = OrderedDict()
|
2019-07-31 19:25:30 +00:00
|
|
|
schema[vol.Required("username")] = str
|
|
|
|
schema[vol.Required("password")] = str
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
return self.async_show_form(
|
2019-07-31 19:25:30 +00:00
|
|
|
step_id="user",
|
2018-02-16 22:07:38 +00:00
|
|
|
data_schema=schema,
|
2020-04-24 16:31:56 +00:00
|
|
|
description_placeholders={
|
|
|
|
"url": "https://example.com",
|
|
|
|
"show_advanced_options": self.show_advanced_options,
|
|
|
|
},
|
2019-07-31 19:25:30 +00:00
|
|
|
errors={"username": "Should be unique."},
|
2018-02-16 22:07:38 +00:00
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.dict(HANDLERS, {"test": TestFlow}):
|
2020-01-01 23:16:27 +00:00
|
|
|
resp = await client.post(
|
2020-04-24 16:31:56 +00:00
|
|
|
"/api/config/config_entries/flow",
|
|
|
|
json={"handler": "test", "show_advanced_options": True},
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
assert resp.status == 200
|
2020-01-01 23:16:27 +00:00
|
|
|
data = await resp.json()
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
data.pop("flow_id")
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
assert data == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"type": "form",
|
|
|
|
"handler": "test",
|
|
|
|
"step_id": "user",
|
|
|
|
"data_schema": [
|
|
|
|
{"name": "username", "required": True, "type": "string"},
|
|
|
|
{"name": "password", "required": True, "type": "string"},
|
2018-02-16 22:07:38 +00:00
|
|
|
],
|
2020-04-24 16:31:56 +00:00
|
|
|
"description_placeholders": {
|
|
|
|
"url": "https://example.com",
|
|
|
|
"show_advanced_options": True,
|
|
|
|
},
|
2019-07-31 19:25:30 +00:00
|
|
|
"errors": {"username": "Should be unique."},
|
2018-02-16 22:07:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-12-13 14:30:20 +00:00
|
|
|
async def test_initialize_flow_unauth(hass, client, hass_admin_user):
|
|
|
|
"""Test we can initialize a flow."""
|
|
|
|
hass_admin_user.groups = []
|
|
|
|
|
|
|
|
class TestFlow(core_ce.ConfigFlow):
|
2020-01-01 23:16:27 +00:00
|
|
|
async def async_step_user(self, user_input=None):
|
2018-12-13 14:30:20 +00:00
|
|
|
schema = OrderedDict()
|
2019-07-31 19:25:30 +00:00
|
|
|
schema[vol.Required("username")] = str
|
|
|
|
schema[vol.Required("password")] = str
|
2018-12-13 14:30:20 +00:00
|
|
|
|
|
|
|
return self.async_show_form(
|
2019-07-31 19:25:30 +00:00
|
|
|
step_id="user",
|
2018-12-13 14:30:20 +00:00
|
|
|
data_schema=schema,
|
2019-07-31 19:25:30 +00:00
|
|
|
description_placeholders={"url": "https://example.com"},
|
|
|
|
errors={"username": "Should be unique."},
|
2018-12-13 14:30:20 +00:00
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.dict(HANDLERS, {"test": TestFlow}):
|
|
|
|
resp = await client.post(
|
|
|
|
"/api/config/config_entries/flow", json={"handler": "test"}
|
|
|
|
)
|
2018-12-13 14:30:20 +00:00
|
|
|
|
|
|
|
assert resp.status == 401
|
|
|
|
|
|
|
|
|
2020-01-01 23:16:27 +00:00
|
|
|
async def test_abort(hass, client):
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Test a flow that aborts."""
|
2019-07-31 19:25:30 +00:00
|
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
2019-05-13 08:16:55 +00:00
|
|
|
|
2018-09-17 08:12:46 +00:00
|
|
|
class TestFlow(core_ce.ConfigFlow):
|
2020-01-01 23:16:27 +00:00
|
|
|
async def async_step_user(self, user_input=None):
|
2019-07-31 19:25:30 +00:00
|
|
|
return self.async_abort(reason="bla")
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.dict(HANDLERS, {"test": TestFlow}):
|
2020-01-01 23:16:27 +00:00
|
|
|
resp = await client.post(
|
2019-07-31 19:25:30 +00:00
|
|
|
"/api/config/config_entries/flow", json={"handler": "test"}
|
|
|
|
)
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
assert resp.status == 200
|
2020-01-01 23:16:27 +00:00
|
|
|
data = await resp.json()
|
2019-07-31 19:25:30 +00:00
|
|
|
data.pop("flow_id")
|
2018-02-16 22:07:38 +00:00
|
|
|
assert data == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"description_placeholders": None,
|
|
|
|
"handler": "test",
|
|
|
|
"reason": "bla",
|
|
|
|
"type": "abort",
|
2018-02-16 22:07:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-01-01 23:16:27 +00:00
|
|
|
async def test_create_account(hass, client):
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Test a flow that creates an account."""
|
2019-07-31 19:25:30 +00:00
|
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
2019-05-13 08:16:55 +00:00
|
|
|
|
2020-04-30 23:31:00 +00:00
|
|
|
mock_integration(
|
|
|
|
hass, MockModule("test", async_setup_entry=AsyncMock(return_value=True))
|
|
|
|
)
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2018-09-17 08:12:46 +00:00
|
|
|
class TestFlow(core_ce.ConfigFlow):
|
2018-02-16 22:07:38 +00:00
|
|
|
VERSION = 1
|
|
|
|
|
2020-01-01 23:16:27 +00:00
|
|
|
async def async_step_user(self, user_input=None):
|
2018-02-16 22:07:38 +00:00
|
|
|
return self.async_create_entry(
|
2019-07-31 19:25:30 +00:00
|
|
|
title="Test Entry", data={"secret": "account_token"}
|
2018-02-16 22:07:38 +00:00
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.dict(HANDLERS, {"test": TestFlow}):
|
2020-01-01 23:16:27 +00:00
|
|
|
resp = await client.post(
|
2019-07-31 19:25:30 +00:00
|
|
|
"/api/config/config_entries/flow", json={"handler": "test"}
|
|
|
|
)
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
assert resp.status == 200
|
2019-03-15 14:41:34 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
entries = hass.config_entries.async_entries("test")
|
2019-03-15 14:41:34 +00:00
|
|
|
assert len(entries) == 1
|
|
|
|
|
2020-01-01 23:16:27 +00:00
|
|
|
data = await resp.json()
|
2019-07-31 19:25:30 +00:00
|
|
|
data.pop("flow_id")
|
2018-02-16 22:07:38 +00:00
|
|
|
assert data == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"handler": "test",
|
|
|
|
"title": "Test Entry",
|
|
|
|
"type": "create_entry",
|
|
|
|
"version": 1,
|
|
|
|
"result": entries[0].entry_id,
|
|
|
|
"description": None,
|
|
|
|
"description_placeholders": None,
|
2018-02-16 22:07:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-01-01 23:16:27 +00:00
|
|
|
async def test_two_step_flow(hass, client):
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Test we can finish a two step flow."""
|
2020-04-30 23:31:00 +00:00
|
|
|
mock_integration(
|
|
|
|
hass, MockModule("test", async_setup_entry=AsyncMock(return_value=True))
|
|
|
|
)
|
2019-07-31 19:25:30 +00:00
|
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2018-09-17 08:12:46 +00:00
|
|
|
class TestFlow(core_ce.ConfigFlow):
|
2018-02-16 22:07:38 +00:00
|
|
|
VERSION = 1
|
|
|
|
|
2020-01-01 23:16:27 +00:00
|
|
|
async def async_step_user(self, user_input=None):
|
2018-02-16 22:07:38 +00:00
|
|
|
return self.async_show_form(
|
2019-07-31 19:25:30 +00:00
|
|
|
step_id="account", data_schema=vol.Schema({"user_title": str})
|
|
|
|
)
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2020-01-01 23:16:27 +00:00
|
|
|
async def async_step_account(self, user_input=None):
|
2018-02-16 22:07:38 +00:00
|
|
|
return self.async_create_entry(
|
2019-07-31 19:25:30 +00:00
|
|
|
title=user_input["user_title"], data={"secret": "account_token"}
|
2018-02-16 22:07:38 +00:00
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.dict(HANDLERS, {"test": TestFlow}):
|
2020-01-01 23:16:27 +00:00
|
|
|
resp = await client.post(
|
2019-07-31 19:25:30 +00:00
|
|
|
"/api/config/config_entries/flow", json={"handler": "test"}
|
|
|
|
)
|
2018-02-16 22:07:38 +00:00
|
|
|
assert resp.status == 200
|
2020-01-01 23:16:27 +00:00
|
|
|
data = await resp.json()
|
2019-07-31 19:25:30 +00:00
|
|
|
flow_id = data.pop("flow_id")
|
2018-02-16 22:07:38 +00:00
|
|
|
assert data == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"type": "form",
|
|
|
|
"handler": "test",
|
|
|
|
"step_id": "account",
|
|
|
|
"data_schema": [{"name": "user_title", "type": "string"}],
|
|
|
|
"description_placeholders": None,
|
|
|
|
"errors": None,
|
2018-02-16 22:07:38 +00:00
|
|
|
}
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.dict(HANDLERS, {"test": TestFlow}):
|
2020-01-01 23:16:27 +00:00
|
|
|
resp = await client.post(
|
2020-04-04 22:26:08 +00:00
|
|
|
f"/api/config/config_entries/flow/{flow_id}",
|
2019-07-31 19:25:30 +00:00
|
|
|
json={"user_title": "user-title"},
|
|
|
|
)
|
2018-02-16 22:07:38 +00:00
|
|
|
assert resp.status == 200
|
2019-03-15 14:41:34 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
entries = hass.config_entries.async_entries("test")
|
2019-03-15 14:41:34 +00:00
|
|
|
assert len(entries) == 1
|
|
|
|
|
2020-01-01 23:16:27 +00:00
|
|
|
data = await resp.json()
|
2019-07-31 19:25:30 +00:00
|
|
|
data.pop("flow_id")
|
2018-02-16 22:07:38 +00:00
|
|
|
assert data == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"handler": "test",
|
|
|
|
"type": "create_entry",
|
|
|
|
"title": "user-title",
|
|
|
|
"version": 1,
|
|
|
|
"result": entries[0].entry_id,
|
|
|
|
"description": None,
|
|
|
|
"description_placeholders": None,
|
2018-02-16 22:07:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-12-13 14:30:20 +00:00
|
|
|
async def test_continue_flow_unauth(hass, client, hass_admin_user):
|
|
|
|
"""Test we can't finish a two step flow."""
|
2020-04-30 23:31:00 +00:00
|
|
|
mock_integration(
|
|
|
|
hass, MockModule("test", async_setup_entry=AsyncMock(return_value=True))
|
|
|
|
)
|
2019-07-31 19:25:30 +00:00
|
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
2018-12-13 14:30:20 +00:00
|
|
|
|
|
|
|
class TestFlow(core_ce.ConfigFlow):
|
|
|
|
VERSION = 1
|
|
|
|
|
2020-01-01 23:16:27 +00:00
|
|
|
async def async_step_user(self, user_input=None):
|
2018-12-13 14:30:20 +00:00
|
|
|
return self.async_show_form(
|
2019-07-31 19:25:30 +00:00
|
|
|
step_id="account", data_schema=vol.Schema({"user_title": str})
|
|
|
|
)
|
2018-12-13 14:30:20 +00:00
|
|
|
|
2020-01-01 23:16:27 +00:00
|
|
|
async def async_step_account(self, user_input=None):
|
2018-12-13 14:30:20 +00:00
|
|
|
return self.async_create_entry(
|
2019-07-31 19:25:30 +00:00
|
|
|
title=user_input["user_title"], data={"secret": "account_token"}
|
2018-12-13 14:30:20 +00:00
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.dict(HANDLERS, {"test": TestFlow}):
|
|
|
|
resp = await client.post(
|
|
|
|
"/api/config/config_entries/flow", json={"handler": "test"}
|
|
|
|
)
|
2018-12-13 14:30:20 +00:00
|
|
|
assert resp.status == 200
|
|
|
|
data = await resp.json()
|
2019-07-31 19:25:30 +00:00
|
|
|
flow_id = data.pop("flow_id")
|
2018-12-13 14:30:20 +00:00
|
|
|
assert data == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"type": "form",
|
|
|
|
"handler": "test",
|
|
|
|
"step_id": "account",
|
|
|
|
"data_schema": [{"name": "user_title", "type": "string"}],
|
|
|
|
"description_placeholders": None,
|
|
|
|
"errors": None,
|
2018-12-13 14:30:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
hass_admin_user.groups = []
|
|
|
|
|
|
|
|
resp = await client.post(
|
2020-08-27 11:56:20 +00:00
|
|
|
f"/api/config/config_entries/flow/{flow_id}",
|
|
|
|
json={"user_title": "user-title"},
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-12-13 14:30:20 +00:00
|
|
|
assert resp.status == 401
|
|
|
|
|
|
|
|
|
2019-08-20 07:17:52 +00:00
|
|
|
async def test_get_progress_index(hass, hass_ws_client):
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Test querying for the flows that are in progress."""
|
2019-08-20 07:17:52 +00:00
|
|
|
assert await async_setup_component(hass, "config", {})
|
2019-07-31 19:25:30 +00:00
|
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
2019-08-20 07:17:52 +00:00
|
|
|
ws_client = await hass_ws_client(hass)
|
2019-05-13 08:16:55 +00:00
|
|
|
|
2018-09-17 08:12:46 +00:00
|
|
|
class TestFlow(core_ce.ConfigFlow):
|
2018-02-16 22:07:38 +00:00
|
|
|
VERSION = 5
|
|
|
|
|
2019-08-20 07:17:52 +00:00
|
|
|
async def async_step_hassio(self, info):
|
|
|
|
return await self.async_step_account()
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-08-20 07:17:52 +00:00
|
|
|
async def async_step_account(self, user_input=None):
|
2019-07-31 19:25:30 +00:00
|
|
|
return self.async_show_form(step_id="account")
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.dict(HANDLERS, {"test": TestFlow}):
|
2019-08-20 07:17:52 +00:00
|
|
|
form = await hass.config_entries.flow.async_init(
|
2019-07-31 19:25:30 +00:00
|
|
|
"test", context={"source": "hassio"}
|
|
|
|
)
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-08-20 07:17:52 +00:00
|
|
|
await ws_client.send_json({"id": 5, "type": "config_entries/flow/progress"})
|
|
|
|
response = await ws_client.receive_json()
|
|
|
|
|
|
|
|
assert response["success"]
|
|
|
|
assert response["result"] == [
|
2020-05-06 04:34:51 +00:00
|
|
|
{
|
|
|
|
"flow_id": form["flow_id"],
|
|
|
|
"handler": "test",
|
|
|
|
"step_id": "account",
|
|
|
|
"context": {"source": "hassio"},
|
|
|
|
}
|
2018-02-16 22:07:38 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
|
2019-08-20 07:17:52 +00:00
|
|
|
async def test_get_progress_index_unauth(hass, hass_ws_client, hass_admin_user):
|
2018-12-13 14:30:20 +00:00
|
|
|
"""Test we can't get flows that are in progress."""
|
2019-08-20 07:17:52 +00:00
|
|
|
assert await async_setup_component(hass, "config", {})
|
2018-12-13 14:30:20 +00:00
|
|
|
hass_admin_user.groups = []
|
2019-08-20 07:17:52 +00:00
|
|
|
ws_client = await hass_ws_client(hass)
|
|
|
|
|
|
|
|
await ws_client.send_json({"id": 5, "type": "config_entries/flow/progress"})
|
|
|
|
response = await ws_client.receive_json()
|
|
|
|
|
|
|
|
assert not response["success"]
|
|
|
|
assert response["error"]["code"] == "unauthorized"
|
2018-12-13 14:30:20 +00:00
|
|
|
|
|
|
|
|
2020-01-01 23:16:27 +00:00
|
|
|
async def test_get_progress_flow(hass, client):
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Test we can query the API for same result as we get from init a flow."""
|
2019-07-31 19:25:30 +00:00
|
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
2019-05-13 08:16:55 +00:00
|
|
|
|
2018-09-17 08:12:46 +00:00
|
|
|
class TestFlow(core_ce.ConfigFlow):
|
2020-01-01 23:16:27 +00:00
|
|
|
async def async_step_user(self, user_input=None):
|
2018-02-16 22:07:38 +00:00
|
|
|
schema = OrderedDict()
|
2019-07-31 19:25:30 +00:00
|
|
|
schema[vol.Required("username")] = str
|
|
|
|
schema[vol.Required("password")] = str
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
return self.async_show_form(
|
2019-07-31 19:25:30 +00:00
|
|
|
step_id="user",
|
2018-02-16 22:07:38 +00:00
|
|
|
data_schema=schema,
|
2019-07-31 19:25:30 +00:00
|
|
|
errors={"username": "Should be unique."},
|
2018-02-16 22:07:38 +00:00
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.dict(HANDLERS, {"test": TestFlow}):
|
2020-01-01 23:16:27 +00:00
|
|
|
resp = await client.post(
|
2019-07-31 19:25:30 +00:00
|
|
|
"/api/config/config_entries/flow", json={"handler": "test"}
|
|
|
|
)
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
assert resp.status == 200
|
2020-01-01 23:16:27 +00:00
|
|
|
data = await resp.json()
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2020-01-01 23:16:27 +00:00
|
|
|
resp2 = await client.get(
|
2019-07-31 19:25:30 +00:00
|
|
|
"/api/config/config_entries/flow/{}".format(data["flow_id"])
|
|
|
|
)
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
assert resp2.status == 200
|
2020-01-01 23:16:27 +00:00
|
|
|
data2 = await resp2.json()
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
assert data == data2
|
2018-12-13 14:30:20 +00:00
|
|
|
|
|
|
|
|
2018-12-13 19:08:31 +00:00
|
|
|
async def test_get_progress_flow_unauth(hass, client, hass_admin_user):
|
|
|
|
"""Test we can can't query the API for result of flow."""
|
2019-07-31 19:25:30 +00:00
|
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
2019-05-13 08:16:55 +00:00
|
|
|
|
2018-12-13 14:30:20 +00:00
|
|
|
class TestFlow(core_ce.ConfigFlow):
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
|
|
schema = OrderedDict()
|
2019-07-31 19:25:30 +00:00
|
|
|
schema[vol.Required("username")] = str
|
|
|
|
schema[vol.Required("password")] = str
|
2018-12-13 14:30:20 +00:00
|
|
|
|
|
|
|
return self.async_show_form(
|
2019-07-31 19:25:30 +00:00
|
|
|
step_id="user",
|
2018-12-13 14:30:20 +00:00
|
|
|
data_schema=schema,
|
2019-07-31 19:25:30 +00:00
|
|
|
errors={"username": "Should be unique."},
|
2018-12-13 14:30:20 +00:00
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.dict(HANDLERS, {"test": TestFlow}):
|
|
|
|
resp = await client.post(
|
|
|
|
"/api/config/config_entries/flow", json={"handler": "test"}
|
|
|
|
)
|
2018-12-13 14:30:20 +00:00
|
|
|
|
|
|
|
assert resp.status == 200
|
|
|
|
data = await resp.json()
|
|
|
|
|
|
|
|
hass_admin_user.groups = []
|
|
|
|
|
|
|
|
resp2 = await client.get(
|
2019-07-31 19:25:30 +00:00
|
|
|
"/api/config/config_entries/flow/{}".format(data["flow_id"])
|
|
|
|
)
|
2018-12-13 14:30:20 +00:00
|
|
|
|
|
|
|
assert resp2.status == 401
|
2019-02-22 16:59:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_options_flow(hass, client):
|
|
|
|
"""Test we can change options."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2019-02-22 16:59:43 +00:00
|
|
|
class TestFlow(core_ce.ConfigFlow):
|
|
|
|
@staticmethod
|
|
|
|
@callback
|
2019-08-15 21:11:55 +00:00
|
|
|
def async_get_options_flow(config_entry):
|
2019-02-22 16:59:43 +00:00
|
|
|
class OptionsFlowHandler(data_entry_flow.FlowHandler):
|
|
|
|
async def async_step_init(self, user_input=None):
|
|
|
|
schema = OrderedDict()
|
2019-07-31 19:25:30 +00:00
|
|
|
schema[vol.Required("enabled")] = bool
|
2019-02-22 16:59:43 +00:00
|
|
|
return self.async_show_form(
|
2019-07-31 19:25:30 +00:00
|
|
|
step_id="user",
|
2019-02-22 16:59:43 +00:00
|
|
|
data_schema=schema,
|
2019-07-31 19:25:30 +00:00
|
|
|
description_placeholders={"enabled": "Set to true to be true"},
|
2019-02-22 16:59:43 +00:00
|
|
|
)
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2019-08-15 21:11:55 +00:00
|
|
|
return OptionsFlowHandler()
|
2019-02-22 16:59:43 +00:00
|
|
|
|
|
|
|
MockConfigEntry(
|
2019-07-31 19:25:30 +00:00
|
|
|
domain="test",
|
|
|
|
entry_id="test1",
|
|
|
|
source="bla",
|
2019-02-22 16:59:43 +00:00
|
|
|
connection_class=core_ce.CONN_CLASS_LOCAL_POLL,
|
|
|
|
).add_to_hass(hass)
|
|
|
|
entry = hass.config_entries._entries[0]
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.dict(HANDLERS, {"test": TestFlow}):
|
2019-08-15 21:11:55 +00:00
|
|
|
url = "/api/config/config_entries/options/flow"
|
2019-07-31 19:25:30 +00:00
|
|
|
resp = await client.post(url, json={"handler": entry.entry_id})
|
2019-02-22 16:59:43 +00:00
|
|
|
|
|
|
|
assert resp.status == 200
|
|
|
|
data = await resp.json()
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
data.pop("flow_id")
|
2019-02-22 16:59:43 +00:00
|
|
|
assert data == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"type": "form",
|
|
|
|
"handler": "test1",
|
|
|
|
"step_id": "user",
|
|
|
|
"data_schema": [{"name": "enabled", "required": True, "type": "boolean"}],
|
|
|
|
"description_placeholders": {"enabled": "Set to true to be true"},
|
|
|
|
"errors": None,
|
2019-02-22 16:59:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
async def test_two_step_options_flow(hass, client):
|
|
|
|
"""Test we can finish a two step options flow."""
|
2020-04-30 23:31:00 +00:00
|
|
|
mock_integration(
|
|
|
|
hass, MockModule("test", async_setup_entry=AsyncMock(return_value=True))
|
|
|
|
)
|
2019-02-22 16:59:43 +00:00
|
|
|
|
|
|
|
class TestFlow(core_ce.ConfigFlow):
|
|
|
|
@staticmethod
|
|
|
|
@callback
|
2019-08-15 21:11:55 +00:00
|
|
|
def async_get_options_flow(config_entry):
|
2019-02-22 16:59:43 +00:00
|
|
|
class OptionsFlowHandler(data_entry_flow.FlowHandler):
|
|
|
|
async def async_step_init(self, user_input=None):
|
|
|
|
return self.async_show_form(
|
2019-07-31 19:25:30 +00:00
|
|
|
step_id="finish", data_schema=vol.Schema({"enabled": bool})
|
2019-02-22 16:59:43 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
async def async_step_finish(self, user_input=None):
|
|
|
|
return self.async_create_entry(
|
2019-07-31 19:25:30 +00:00
|
|
|
title="Enable disable", data=user_input
|
2019-02-22 16:59:43 +00:00
|
|
|
)
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2019-08-15 21:11:55 +00:00
|
|
|
return OptionsFlowHandler()
|
2019-02-22 16:59:43 +00:00
|
|
|
|
|
|
|
MockConfigEntry(
|
2019-07-31 19:25:30 +00:00
|
|
|
domain="test",
|
|
|
|
entry_id="test1",
|
|
|
|
source="bla",
|
2019-02-22 16:59:43 +00:00
|
|
|
connection_class=core_ce.CONN_CLASS_LOCAL_POLL,
|
|
|
|
).add_to_hass(hass)
|
|
|
|
entry = hass.config_entries._entries[0]
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.dict(HANDLERS, {"test": TestFlow}):
|
2019-08-15 21:11:55 +00:00
|
|
|
url = "/api/config/config_entries/options/flow"
|
2019-07-31 19:25:30 +00:00
|
|
|
resp = await client.post(url, json={"handler": entry.entry_id})
|
2019-02-22 16:59:43 +00:00
|
|
|
|
|
|
|
assert resp.status == 200
|
|
|
|
data = await resp.json()
|
2019-07-31 19:25:30 +00:00
|
|
|
flow_id = data.pop("flow_id")
|
2019-02-22 16:59:43 +00:00
|
|
|
assert data == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"type": "form",
|
|
|
|
"handler": "test1",
|
|
|
|
"step_id": "finish",
|
|
|
|
"data_schema": [{"name": "enabled", "type": "boolean"}],
|
|
|
|
"description_placeholders": None,
|
|
|
|
"errors": None,
|
2019-02-22 16:59:43 +00:00
|
|
|
}
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch.dict(HANDLERS, {"test": TestFlow}):
|
2019-02-22 16:59:43 +00:00
|
|
|
resp = await client.post(
|
2020-04-04 22:26:08 +00:00
|
|
|
f"/api/config/config_entries/options/flow/{flow_id}",
|
2019-07-31 19:25:30 +00:00
|
|
|
json={"enabled": True},
|
|
|
|
)
|
2019-02-22 16:59:43 +00:00
|
|
|
assert resp.status == 200
|
|
|
|
data = await resp.json()
|
2019-07-31 19:25:30 +00:00
|
|
|
data.pop("flow_id")
|
2019-02-22 16:59:43 +00:00
|
|
|
assert data == {
|
2019-07-31 19:25:30 +00:00
|
|
|
"handler": "test1",
|
|
|
|
"type": "create_entry",
|
|
|
|
"title": "Enable disable",
|
|
|
|
"version": 1,
|
|
|
|
"description": None,
|
|
|
|
"description_placeholders": None,
|
2019-02-22 16:59:43 +00:00
|
|
|
}
|
2019-08-18 04:34:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_list_system_options(hass, hass_ws_client):
|
|
|
|
"""Test that we can list an entries system options."""
|
|
|
|
assert await async_setup_component(hass, "config", {})
|
|
|
|
ws_client = await hass_ws_client(hass)
|
|
|
|
|
|
|
|
entry = MockConfigEntry(domain="demo")
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
|
|
|
|
await ws_client.send_json(
|
|
|
|
{
|
|
|
|
"id": 5,
|
|
|
|
"type": "config_entries/system_options/list",
|
|
|
|
"entry_id": entry.entry_id,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
response = await ws_client.receive_json()
|
|
|
|
|
|
|
|
assert response["success"]
|
|
|
|
assert response["result"] == {"disable_new_entities": False}
|
|
|
|
|
|
|
|
|
|
|
|
async def test_update_system_options(hass, hass_ws_client):
|
|
|
|
"""Test that we can update system options."""
|
|
|
|
assert await async_setup_component(hass, "config", {})
|
|
|
|
ws_client = await hass_ws_client(hass)
|
|
|
|
|
|
|
|
entry = MockConfigEntry(domain="demo")
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
|
|
|
|
await ws_client.send_json(
|
|
|
|
{
|
|
|
|
"id": 5,
|
|
|
|
"type": "config_entries/system_options/update",
|
|
|
|
"entry_id": entry.entry_id,
|
|
|
|
"disable_new_entities": True,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
response = await ws_client.receive_json()
|
|
|
|
|
|
|
|
assert response["success"]
|
|
|
|
assert response["result"]["disable_new_entities"]
|
|
|
|
assert entry.system_options.disable_new_entities
|
2019-12-18 06:41:01 +00:00
|
|
|
|
|
|
|
|
2021-02-21 03:21:39 +00:00
|
|
|
async def test_update_system_options_nonexisting(hass, hass_ws_client):
|
|
|
|
"""Test that we can update entry."""
|
|
|
|
assert await async_setup_component(hass, "config", {})
|
|
|
|
ws_client = await hass_ws_client(hass)
|
|
|
|
|
|
|
|
await ws_client.send_json(
|
|
|
|
{
|
|
|
|
"id": 5,
|
|
|
|
"type": "config_entries/system_options/update",
|
|
|
|
"entry_id": "non_existing",
|
|
|
|
"disable_new_entities": True,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
response = await ws_client.receive_json()
|
|
|
|
|
|
|
|
assert not response["success"]
|
|
|
|
assert response["error"]["code"] == "not_found"
|
|
|
|
|
|
|
|
|
2020-04-14 00:38:39 +00:00
|
|
|
async def test_update_entry(hass, hass_ws_client):
|
|
|
|
"""Test that we can update entry."""
|
|
|
|
assert await async_setup_component(hass, "config", {})
|
|
|
|
ws_client = await hass_ws_client(hass)
|
|
|
|
|
|
|
|
entry = MockConfigEntry(domain="demo", title="Initial Title")
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
|
|
|
|
await ws_client.send_json(
|
|
|
|
{
|
|
|
|
"id": 5,
|
|
|
|
"type": "config_entries/update",
|
|
|
|
"entry_id": entry.entry_id,
|
|
|
|
"title": "Updated Title",
|
|
|
|
}
|
|
|
|
)
|
|
|
|
response = await ws_client.receive_json()
|
|
|
|
|
|
|
|
assert response["success"]
|
|
|
|
assert response["result"]["title"] == "Updated Title"
|
|
|
|
assert entry.title == "Updated Title"
|
|
|
|
|
|
|
|
|
2020-04-14 09:43:22 +00:00
|
|
|
async def test_update_entry_nonexisting(hass, hass_ws_client):
|
|
|
|
"""Test that we can update entry."""
|
|
|
|
assert await async_setup_component(hass, "config", {})
|
|
|
|
ws_client = await hass_ws_client(hass)
|
|
|
|
|
|
|
|
await ws_client.send_json(
|
|
|
|
{
|
|
|
|
"id": 5,
|
|
|
|
"type": "config_entries/update",
|
|
|
|
"entry_id": "non_existing",
|
|
|
|
"title": "Updated Title",
|
|
|
|
}
|
|
|
|
)
|
|
|
|
response = await ws_client.receive_json()
|
|
|
|
|
|
|
|
assert not response["success"]
|
|
|
|
assert response["error"]["code"] == "not_found"
|
|
|
|
|
|
|
|
|
2021-02-21 03:21:39 +00:00
|
|
|
async def test_disable_entry(hass, hass_ws_client):
|
|
|
|
"""Test that we can disable entry."""
|
|
|
|
assert await async_setup_component(hass, "config", {})
|
|
|
|
ws_client = await hass_ws_client(hass)
|
|
|
|
|
|
|
|
entry = MockConfigEntry(domain="demo", state="loaded")
|
|
|
|
entry.add_to_hass(hass)
|
|
|
|
assert entry.disabled_by is None
|
|
|
|
|
|
|
|
# Disable
|
|
|
|
await ws_client.send_json(
|
|
|
|
{
|
|
|
|
"id": 5,
|
|
|
|
"type": "config_entries/disable",
|
|
|
|
"entry_id": entry.entry_id,
|
|
|
|
"disabled_by": "user",
|
|
|
|
}
|
|
|
|
)
|
|
|
|
response = await ws_client.receive_json()
|
|
|
|
|
|
|
|
assert response["success"]
|
|
|
|
assert response["result"] == {"require_restart": True}
|
|
|
|
assert entry.disabled_by == "user"
|
|
|
|
assert entry.state == "failed_unload"
|
|
|
|
|
|
|
|
# Enable
|
|
|
|
await ws_client.send_json(
|
|
|
|
{
|
|
|
|
"id": 6,
|
|
|
|
"type": "config_entries/disable",
|
|
|
|
"entry_id": entry.entry_id,
|
|
|
|
"disabled_by": None,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
response = await ws_client.receive_json()
|
|
|
|
|
|
|
|
assert response["success"]
|
|
|
|
assert response["result"] == {"require_restart": True}
|
|
|
|
assert entry.disabled_by is None
|
|
|
|
assert entry.state == "failed_unload"
|
|
|
|
|
|
|
|
# Enable again -> no op
|
|
|
|
await ws_client.send_json(
|
|
|
|
{
|
|
|
|
"id": 7,
|
|
|
|
"type": "config_entries/disable",
|
|
|
|
"entry_id": entry.entry_id,
|
|
|
|
"disabled_by": None,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
response = await ws_client.receive_json()
|
|
|
|
|
|
|
|
assert response["success"]
|
|
|
|
assert response["result"] == {"require_restart": False}
|
|
|
|
assert entry.disabled_by is None
|
|
|
|
assert entry.state == "failed_unload"
|
|
|
|
|
|
|
|
|
|
|
|
async def test_disable_entry_nonexisting(hass, hass_ws_client):
|
|
|
|
"""Test that we can disable entry."""
|
|
|
|
assert await async_setup_component(hass, "config", {})
|
|
|
|
ws_client = await hass_ws_client(hass)
|
|
|
|
|
|
|
|
await ws_client.send_json(
|
|
|
|
{
|
|
|
|
"id": 5,
|
|
|
|
"type": "config_entries/disable",
|
|
|
|
"entry_id": "non_existing",
|
|
|
|
"disabled_by": "user",
|
|
|
|
}
|
|
|
|
)
|
|
|
|
response = await ws_client.receive_json()
|
|
|
|
|
|
|
|
assert not response["success"]
|
|
|
|
assert response["error"]["code"] == "not_found"
|
|
|
|
|
|
|
|
|
2019-12-18 06:41:01 +00:00
|
|
|
async def test_ignore_flow(hass, hass_ws_client):
|
|
|
|
"""Test we can ignore a flow."""
|
|
|
|
assert await async_setup_component(hass, "config", {})
|
2020-04-30 23:31:00 +00:00
|
|
|
mock_integration(
|
|
|
|
hass, MockModule("test", async_setup_entry=AsyncMock(return_value=True))
|
|
|
|
)
|
2019-12-18 06:41:01 +00:00
|
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
|
|
|
|
|
|
|
class TestFlow(core_ce.ConfigFlow):
|
|
|
|
VERSION = 1
|
|
|
|
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
|
|
await self.async_set_unique_id("mock-unique-id")
|
|
|
|
return self.async_show_form(step_id="account", data_schema=vol.Schema({}))
|
|
|
|
|
|
|
|
ws_client = await hass_ws_client(hass)
|
|
|
|
|
|
|
|
with patch.dict(HANDLERS, {"test": TestFlow}):
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
"test", context={"source": "user"}
|
|
|
|
)
|
|
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_FORM
|
|
|
|
|
|
|
|
await ws_client.send_json(
|
|
|
|
{
|
|
|
|
"id": 5,
|
|
|
|
"type": "config_entries/ignore_flow",
|
|
|
|
"flow_id": result["flow_id"],
|
2021-01-12 08:26:20 +00:00
|
|
|
"title": "Test Integration",
|
2019-12-18 06:41:01 +00:00
|
|
|
}
|
|
|
|
)
|
|
|
|
response = await ws_client.receive_json()
|
|
|
|
|
|
|
|
assert response["success"]
|
|
|
|
|
|
|
|
assert len(hass.config_entries.flow.async_progress()) == 0
|
|
|
|
|
|
|
|
entry = hass.config_entries.async_entries("test")[0]
|
|
|
|
assert entry.source == "ignore"
|
|
|
|
assert entry.unique_id == "mock-unique-id"
|
2021-01-12 08:26:20 +00:00
|
|
|
assert entry.title == "Test Integration"
|
2021-02-21 03:21:39 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_ignore_flow_nonexisting(hass, hass_ws_client):
|
|
|
|
"""Test we can ignore a flow."""
|
|
|
|
assert await async_setup_component(hass, "config", {})
|
|
|
|
ws_client = await hass_ws_client(hass)
|
|
|
|
|
|
|
|
await ws_client.send_json(
|
|
|
|
{
|
|
|
|
"id": 5,
|
|
|
|
"type": "config_entries/ignore_flow",
|
|
|
|
"flow_id": "non_existing",
|
|
|
|
"title": "Test Integration",
|
|
|
|
}
|
|
|
|
)
|
|
|
|
response = await ws_client.receive_json()
|
|
|
|
|
|
|
|
assert not response["success"]
|
|
|
|
assert response["error"]["code"] == "not_found"
|