Add zeroconf discovery to miele (#143259)

* Add zeroconf discovery

* Strip unnecessary code

* Remove one line more

* Remove one more

* Add test for zeroconf flow

* Finish zeroconf flow
This commit is contained in:
Åke Strandberg 2025-04-25 23:18:20 +02:00 committed by GitHub
parent 963f1b1907
commit eec9a28fe8
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
3 changed files with 61 additions and 3 deletions

View File

@ -9,5 +9,6 @@
"loggers": ["pymiele"], "loggers": ["pymiele"],
"quality_scale": "bronze", "quality_scale": "bronze",
"requirements": ["pymiele==0.3.6"], "requirements": ["pymiele==0.3.6"],
"single_config_entry": true "single_config_entry": true,
"zeroconf": ["_mieleathome._tcp.local."]
} }

View File

@ -710,6 +710,11 @@ ZEROCONF = {
"domain": "thread", "domain": "thread",
}, },
], ],
"_mieleathome._tcp.local.": [
{
"domain": "miele",
},
],
"_miio._udp.local.": [ "_miio._udp.local.": [
{ {
"domain": "xiaomi_aqara", "domain": "xiaomi_aqara",

View File

@ -7,7 +7,7 @@ from pymiele import OAUTH2_AUTHORIZE, OAUTH2_TOKEN
import pytest import pytest
from homeassistant.components.miele.const import DOMAIN from homeassistant.components.miele.const import DOMAIN
from homeassistant.config_entries import SOURCE_USER from homeassistant.config_entries import SOURCE_USER, SOURCE_ZEROCONF
from homeassistant.core import HomeAssistant from homeassistant.core import HomeAssistant
from homeassistant.data_entry_flow import FlowResultType from homeassistant.data_entry_flow import FlowResultType
from homeassistant.helpers import config_entry_oauth2_flow from homeassistant.helpers import config_entry_oauth2_flow
@ -154,7 +154,7 @@ async def test_flow_reconfigure_abort(
access_token: str, access_token: str,
expires_at: float, expires_at: float,
) -> None: ) -> None:
"""Test reauth step with correct params and mismatches.""" """Test reconfigure step with correct params."""
CURRENT_TOKEN = { CURRENT_TOKEN = {
"auth_implementation": DOMAIN, "auth_implementation": DOMAIN,
@ -212,3 +212,55 @@ async def test_flow_reconfigure_abort(
assert result.get("reason") == "reconfigure_successful" assert result.get("reason") == "reconfigure_successful"
assert len(hass.config_entries.async_entries(DOMAIN)) == 1 assert len(hass.config_entries.async_entries(DOMAIN)) == 1
@pytest.mark.usefixtures("current_request_with_host")
async def test_zeroconf_flow(
hass: HomeAssistant,
hass_client_no_auth: ClientSessionGenerator,
aioclient_mock: AiohttpClientMocker,
mock_setup_entry: Generator[AsyncMock],
) -> None:
"""Test zeroconf flow."""
result = await hass.config_entries.flow.async_init(
DOMAIN, context={"source": SOURCE_ZEROCONF}
)
state = config_entry_oauth2_flow._encode_jwt(
hass,
{
"flow_id": result["flow_id"],
"redirect_uri": REDIRECT_URL,
},
)
assert result["url"] == (
f"{OAUTH2_AUTHORIZE}?response_type=code&client_id={CLIENT_ID}"
f"&redirect_uri={REDIRECT_URL}"
f"&state={state}"
"&vg=sv-SE"
)
client = await hass_client_no_auth()
resp = await client.get(f"/auth/external/callback?code=abcd&state={state}")
assert resp.status == 200
assert resp.headers["content-type"] == "text/html; charset=utf-8"
aioclient_mock.post(
OAUTH2_TOKEN,
json={
"refresh_token": "mock-refresh-token",
"access_token": "mock-access-token",
"type": "Bearer",
"expires_in": 60,
},
)
assert result.get("type") is FlowResultType.EXTERNAL_STEP
result = await hass.config_entries.flow.async_configure(result["flow_id"])
assert len(hass.config_entries.async_entries(DOMAIN)) == 1
assert len(mock_setup_entry.mock_calls) == 1
assert result["type"] is FlowResultType.CREATE_ENTRY
config_entry = result["result"]
assert config_entry.domain == "miele"