2020-03-30 23:13:47 +00:00
|
|
|
"""Tests for the IPP config flow."""
|
|
|
|
import aiohttp
|
|
|
|
from pyipp import IPPConnectionUpgradeRequired
|
|
|
|
|
2020-03-31 23:40:07 +00:00
|
|
|
from homeassistant.components.ipp.const import CONF_BASE_PATH, CONF_UUID, DOMAIN
|
2020-03-30 23:13:47 +00:00
|
|
|
from homeassistant.config_entries import SOURCE_USER, SOURCE_ZEROCONF
|
|
|
|
from homeassistant.const import CONF_HOST, CONF_NAME, CONF_SSL
|
|
|
|
from homeassistant.core import HomeAssistant
|
2020-03-31 23:40:07 +00:00
|
|
|
from homeassistant.data_entry_flow import (
|
|
|
|
RESULT_TYPE_ABORT,
|
|
|
|
RESULT_TYPE_CREATE_ENTRY,
|
|
|
|
RESULT_TYPE_FORM,
|
|
|
|
)
|
2020-03-30 23:13:47 +00:00
|
|
|
|
|
|
|
from . import (
|
|
|
|
MOCK_USER_INPUT,
|
|
|
|
MOCK_ZEROCONF_IPP_SERVICE_INFO,
|
|
|
|
MOCK_ZEROCONF_IPPS_SERVICE_INFO,
|
|
|
|
init_integration,
|
|
|
|
load_fixture_binary,
|
|
|
|
)
|
|
|
|
|
|
|
|
from tests.test_util.aiohttp import AiohttpClientMocker
|
|
|
|
|
|
|
|
|
|
|
|
async def test_show_user_form(hass: HomeAssistant) -> None:
|
|
|
|
"""Test that the user set up form is served."""
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-03-31 23:40:07 +00:00
|
|
|
DOMAIN, context={"source": SOURCE_USER},
|
2020-03-30 23:13:47 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["step_id"] == "user"
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_FORM
|
2020-03-30 23:13:47 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def test_show_zeroconf_form(
|
|
|
|
hass: HomeAssistant, aioclient_mock: AiohttpClientMocker
|
|
|
|
) -> None:
|
|
|
|
"""Test that the zeroconf confirmation form is served."""
|
|
|
|
aioclient_mock.post(
|
|
|
|
"http://EPSON123456.local:631/ipp/print",
|
|
|
|
content=load_fixture_binary("ipp/get-printer-attributes.bin"),
|
|
|
|
headers={"Content-Type": "application/ipp"},
|
|
|
|
)
|
|
|
|
|
|
|
|
discovery_info = MOCK_ZEROCONF_IPP_SERVICE_INFO.copy()
|
2020-03-31 23:40:07 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": SOURCE_ZEROCONF}, data=discovery_info,
|
|
|
|
)
|
2020-03-30 23:13:47 +00:00
|
|
|
|
|
|
|
assert result["step_id"] == "zeroconf_confirm"
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_FORM
|
2020-03-30 23:13:47 +00:00
|
|
|
assert result["description_placeholders"] == {CONF_NAME: "EPSON123456"}
|
|
|
|
|
|
|
|
|
|
|
|
async def test_connection_error(
|
|
|
|
hass: HomeAssistant, aioclient_mock: AiohttpClientMocker
|
|
|
|
) -> None:
|
|
|
|
"""Test we show user form on IPP connection error."""
|
|
|
|
aioclient_mock.post(
|
|
|
|
"http://EPSON123456.local:631/ipp/print", exc=aiohttp.ClientError
|
|
|
|
)
|
|
|
|
|
|
|
|
user_input = MOCK_USER_INPUT.copy()
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-03-31 23:40:07 +00:00
|
|
|
DOMAIN, context={"source": SOURCE_USER}, data=user_input,
|
2020-03-30 23:13:47 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["step_id"] == "user"
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_FORM
|
2020-03-30 23:13:47 +00:00
|
|
|
assert result["errors"] == {"base": "connection_error"}
|
|
|
|
|
|
|
|
|
|
|
|
async def test_zeroconf_connection_error(
|
|
|
|
hass: HomeAssistant, aioclient_mock: AiohttpClientMocker
|
|
|
|
) -> None:
|
|
|
|
"""Test we abort zeroconf flow on IPP connection error."""
|
|
|
|
aioclient_mock.post("http://EPSON123456.local/ipp/print", exc=aiohttp.ClientError)
|
|
|
|
|
|
|
|
discovery_info = MOCK_ZEROCONF_IPP_SERVICE_INFO.copy()
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-03-31 23:40:07 +00:00
|
|
|
DOMAIN, context={"source": SOURCE_ZEROCONF}, data=discovery_info,
|
2020-03-30 23:13:47 +00:00
|
|
|
)
|
|
|
|
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_ABORT
|
2020-03-30 23:13:47 +00:00
|
|
|
assert result["reason"] == "connection_error"
|
|
|
|
|
|
|
|
|
|
|
|
async def test_zeroconf_confirm_connection_error(
|
|
|
|
hass: HomeAssistant, aioclient_mock: AiohttpClientMocker
|
|
|
|
) -> None:
|
|
|
|
"""Test we abort zeroconf flow on IPP connection error."""
|
|
|
|
aioclient_mock.post("http://EPSON123456.local/ipp/print", exc=aiohttp.ClientError)
|
|
|
|
|
|
|
|
discovery_info = MOCK_ZEROCONF_IPP_SERVICE_INFO.copy()
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-03-31 23:40:07 +00:00
|
|
|
DOMAIN,
|
2020-03-30 23:13:47 +00:00
|
|
|
context={
|
|
|
|
"source": SOURCE_ZEROCONF,
|
|
|
|
CONF_HOST: "EPSON123456.local",
|
|
|
|
CONF_NAME: "EPSON123456",
|
|
|
|
},
|
|
|
|
data=discovery_info,
|
|
|
|
)
|
|
|
|
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_ABORT
|
2020-03-30 23:13:47 +00:00
|
|
|
assert result["reason"] == "connection_error"
|
|
|
|
|
|
|
|
|
|
|
|
async def test_user_connection_upgrade_required(
|
|
|
|
hass: HomeAssistant, aioclient_mock: AiohttpClientMocker
|
|
|
|
) -> None:
|
|
|
|
"""Test we show the user form if connection upgrade required by server."""
|
|
|
|
aioclient_mock.post(
|
|
|
|
"http://EPSON123456.local:631/ipp/print", exc=IPPConnectionUpgradeRequired
|
|
|
|
)
|
|
|
|
|
|
|
|
user_input = MOCK_USER_INPUT.copy()
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-03-31 23:40:07 +00:00
|
|
|
DOMAIN, context={"source": SOURCE_USER}, data=user_input,
|
2020-03-30 23:13:47 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["step_id"] == "user"
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_FORM
|
2020-03-30 23:13:47 +00:00
|
|
|
assert result["errors"] == {"base": "connection_upgrade"}
|
|
|
|
|
|
|
|
|
|
|
|
async def test_zeroconf_connection_upgrade_required(
|
|
|
|
hass: HomeAssistant, aioclient_mock: AiohttpClientMocker
|
|
|
|
) -> None:
|
|
|
|
"""Test we abort zeroconf flow on IPP connection error."""
|
|
|
|
aioclient_mock.post(
|
|
|
|
"http://EPSON123456.local/ipp/print", exc=IPPConnectionUpgradeRequired
|
|
|
|
)
|
|
|
|
|
|
|
|
discovery_info = MOCK_ZEROCONF_IPP_SERVICE_INFO.copy()
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-03-31 23:40:07 +00:00
|
|
|
DOMAIN, context={"source": SOURCE_ZEROCONF}, data=discovery_info,
|
2020-03-30 23:13:47 +00:00
|
|
|
)
|
|
|
|
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_ABORT
|
2020-03-30 23:13:47 +00:00
|
|
|
assert result["reason"] == "connection_upgrade"
|
|
|
|
|
|
|
|
|
|
|
|
async def test_user_device_exists_abort(
|
|
|
|
hass: HomeAssistant, aioclient_mock: AiohttpClientMocker
|
|
|
|
) -> None:
|
|
|
|
"""Test we abort user flow if printer already configured."""
|
|
|
|
await init_integration(hass, aioclient_mock)
|
|
|
|
|
|
|
|
user_input = MOCK_USER_INPUT.copy()
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-03-31 23:40:07 +00:00
|
|
|
DOMAIN, context={"source": SOURCE_USER}, data=user_input,
|
2020-03-30 23:13:47 +00:00
|
|
|
)
|
|
|
|
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_ABORT
|
2020-03-30 23:13:47 +00:00
|
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
|
|
|
|
|
|
|
async def test_zeroconf_device_exists_abort(
|
|
|
|
hass: HomeAssistant, aioclient_mock: AiohttpClientMocker
|
|
|
|
) -> None:
|
|
|
|
"""Test we abort zeroconf flow if printer already configured."""
|
|
|
|
await init_integration(hass, aioclient_mock)
|
|
|
|
|
|
|
|
discovery_info = MOCK_ZEROCONF_IPP_SERVICE_INFO.copy()
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-03-31 23:40:07 +00:00
|
|
|
DOMAIN, context={"source": SOURCE_ZEROCONF}, data=discovery_info,
|
2020-03-30 23:13:47 +00:00
|
|
|
)
|
|
|
|
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_ABORT
|
2020-03-30 23:13:47 +00:00
|
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
|
|
|
|
|
|
|
async def test_zeroconf_with_uuid_device_exists_abort(
|
|
|
|
hass: HomeAssistant, aioclient_mock: AiohttpClientMocker
|
|
|
|
) -> None:
|
|
|
|
"""Test we abort zeroconf flow if printer already configured."""
|
|
|
|
await init_integration(hass, aioclient_mock)
|
|
|
|
|
|
|
|
discovery_info = MOCK_ZEROCONF_IPP_SERVICE_INFO.copy()
|
|
|
|
discovery_info["properties"]["UUID"] = "cfe92100-67c4-11d4-a45f-f8d027761251"
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-03-31 23:40:07 +00:00
|
|
|
DOMAIN, context={"source": SOURCE_ZEROCONF}, data=discovery_info,
|
2020-03-30 23:13:47 +00:00
|
|
|
)
|
|
|
|
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_ABORT
|
2020-03-30 23:13:47 +00:00
|
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
|
|
|
|
|
|
|
async def test_full_user_flow_implementation(
|
|
|
|
hass: HomeAssistant, aioclient_mock
|
|
|
|
) -> None:
|
|
|
|
"""Test the full manual user flow from start to finish."""
|
|
|
|
aioclient_mock.post(
|
|
|
|
"http://EPSON123456.local:631/ipp/print",
|
|
|
|
content=load_fixture_binary("ipp/get-printer-attributes.bin"),
|
|
|
|
headers={"Content-Type": "application/ipp"},
|
|
|
|
)
|
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_init(
|
2020-03-31 23:40:07 +00:00
|
|
|
DOMAIN, context={"source": SOURCE_USER},
|
2020-03-30 23:13:47 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert result["step_id"] == "user"
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_FORM
|
2020-03-30 23:13:47 +00:00
|
|
|
|
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"],
|
|
|
|
user_input={CONF_HOST: "EPSON123456.local", CONF_BASE_PATH: "/ipp/print"},
|
|
|
|
)
|
|
|
|
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_CREATE_ENTRY
|
2020-03-30 23:13:47 +00:00
|
|
|
assert result["title"] == "EPSON123456.local"
|
|
|
|
|
|
|
|
assert result["data"]
|
|
|
|
assert result["data"][CONF_HOST] == "EPSON123456.local"
|
|
|
|
assert result["data"][CONF_UUID] == "cfe92100-67c4-11d4-a45f-f8d027761251"
|
|
|
|
|
|
|
|
|
|
|
|
async def test_full_zeroconf_flow_implementation(
|
|
|
|
hass: HomeAssistant, aioclient_mock: AiohttpClientMocker
|
|
|
|
) -> None:
|
|
|
|
"""Test the full manual user flow from start to finish."""
|
|
|
|
aioclient_mock.post(
|
|
|
|
"http://EPSON123456.local:631/ipp/print",
|
|
|
|
content=load_fixture_binary("ipp/get-printer-attributes.bin"),
|
|
|
|
headers={"Content-Type": "application/ipp"},
|
|
|
|
)
|
|
|
|
|
|
|
|
discovery_info = MOCK_ZEROCONF_IPP_SERVICE_INFO.copy()
|
2020-03-31 23:40:07 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": SOURCE_ZEROCONF}, data=discovery_info,
|
|
|
|
)
|
2020-03-30 23:13:47 +00:00
|
|
|
|
|
|
|
assert result["step_id"] == "zeroconf_confirm"
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_FORM
|
2020-03-30 23:13:47 +00:00
|
|
|
|
2020-03-31 23:40:07 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"], user_input={}
|
2020-03-30 23:13:47 +00:00
|
|
|
)
|
|
|
|
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_CREATE_ENTRY
|
2020-03-30 23:13:47 +00:00
|
|
|
assert result["title"] == "EPSON123456"
|
|
|
|
|
|
|
|
assert result["data"]
|
|
|
|
assert result["data"][CONF_HOST] == "EPSON123456.local"
|
|
|
|
assert result["data"][CONF_UUID] == "cfe92100-67c4-11d4-a45f-f8d027761251"
|
|
|
|
assert not result["data"][CONF_SSL]
|
|
|
|
|
|
|
|
|
|
|
|
async def test_full_zeroconf_tls_flow_implementation(
|
|
|
|
hass: HomeAssistant, aioclient_mock: AiohttpClientMocker
|
|
|
|
) -> None:
|
|
|
|
"""Test the full manual user flow from start to finish."""
|
|
|
|
aioclient_mock.post(
|
|
|
|
"https://EPSON123456.local:631/ipp/print",
|
|
|
|
content=load_fixture_binary("ipp/get-printer-attributes.bin"),
|
|
|
|
headers={"Content-Type": "application/ipp"},
|
|
|
|
)
|
|
|
|
|
|
|
|
discovery_info = MOCK_ZEROCONF_IPPS_SERVICE_INFO.copy()
|
2020-03-31 23:40:07 +00:00
|
|
|
result = await hass.config_entries.flow.async_init(
|
|
|
|
DOMAIN, context={"source": SOURCE_ZEROCONF}, data=discovery_info,
|
|
|
|
)
|
2020-03-30 23:13:47 +00:00
|
|
|
|
|
|
|
assert result["step_id"] == "zeroconf_confirm"
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_FORM
|
2020-03-30 23:13:47 +00:00
|
|
|
assert result["description_placeholders"] == {CONF_NAME: "EPSON123456"}
|
|
|
|
|
2020-03-31 23:40:07 +00:00
|
|
|
result = await hass.config_entries.flow.async_configure(
|
|
|
|
result["flow_id"], user_input={}
|
2020-03-30 23:13:47 +00:00
|
|
|
)
|
|
|
|
|
2020-03-31 23:40:07 +00:00
|
|
|
assert result["type"] == RESULT_TYPE_CREATE_ENTRY
|
2020-03-30 23:13:47 +00:00
|
|
|
assert result["title"] == "EPSON123456"
|
|
|
|
|
|
|
|
assert result["data"]
|
|
|
|
assert result["data"][CONF_HOST] == "EPSON123456.local"
|
|
|
|
assert result["data"][CONF_NAME] == "EPSON123456"
|
|
|
|
assert result["data"][CONF_UUID] == "cfe92100-67c4-11d4-a45f-f8d027761251"
|
|
|
|
assert result["data"][CONF_SSL]
|