1
0
mirror of https://github.com/home-assistant/core.git synced 2025-12-20 02:48:57 +00:00

Add option to enable offline mode to lamarzocco (#159094)

Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
This commit is contained in:
Josef Zweck
2025-12-15 19:22:50 +01:00
committed by GitHub
parent 0ced960d1d
commit a3f3586b02
7 changed files with 199 additions and 50 deletions

View File

@@ -33,7 +33,7 @@ from homeassistant.exceptions import ConfigEntryAuthFailed, ConfigEntryNotReady
from homeassistant.helpers import issue_registry as ir
from homeassistant.helpers.aiohttp_client import async_create_clientsession
from .const import CONF_INSTALLATION_KEY, CONF_USE_BLUETOOTH, DOMAIN
from .const import CONF_INSTALLATION_KEY, CONF_OFFLINE_MODE, CONF_USE_BLUETOOTH, DOMAIN
from .coordinator import (
LaMarzoccoBluetoothUpdateCoordinator,
LaMarzoccoConfigEntry,
@@ -118,6 +118,9 @@ async def async_setup_entry(hass: HomeAssistant, entry: LaMarzoccoConfigEntry) -
_LOGGER.info(
"Bluetooth device not found during lamarzocco setup, continuing with cloud only"
)
async def _get_thing_settings() -> None:
"""Get thing settings from cloud to verify details and get BLE token."""
try:
settings = await cloud_client.get_thing_settings(serial)
except AuthFail as ex:
@@ -157,6 +160,9 @@ async def async_setup_entry(hass: HomeAssistant, entry: LaMarzoccoConfigEntry) -
},
)
if not (local_mode := entry.options.get(CONF_OFFLINE_MODE, False)):
await _get_thing_settings()
device = LaMarzoccoMachine(
serial_number=entry.unique_id,
cloud_client=cloud_client,
@@ -170,6 +176,7 @@ async def async_setup_entry(hass: HomeAssistant, entry: LaMarzoccoConfigEntry) -
LaMarzoccoStatisticsUpdateCoordinator(hass, entry, device),
)
if not local_mode:
await asyncio.gather(
coordinators.config_coordinator.async_config_entry_first_refresh(),
coordinators.settings_coordinator.async_config_entry_first_refresh(),
@@ -177,6 +184,11 @@ async def async_setup_entry(hass: HomeAssistant, entry: LaMarzoccoConfigEntry) -
coordinators.statistics_coordinator.async_config_entry_first_refresh(),
)
if local_mode and not bluetooth_client:
raise ConfigEntryNotReady(
translation_domain=DOMAIN, translation_key="bluetooth_required_offline"
)
# bt coordinator only if bluetooth client is available
# and after the initial refresh of the config coordinator
# to fetch only if the others failed

View File

@@ -47,7 +47,7 @@ from homeassistant.helpers.selector import (
from homeassistant.helpers.service_info.dhcp import DhcpServiceInfo
from . import create_client_session
from .const import CONF_INSTALLATION_KEY, CONF_USE_BLUETOOTH, DOMAIN
from .const import CONF_INSTALLATION_KEY, CONF_OFFLINE_MODE, CONF_USE_BLUETOOTH, DOMAIN
from .coordinator import LaMarzoccoConfigEntry
CONF_MACHINE = "machine"
@@ -379,19 +379,30 @@ class LmOptionsFlowHandler(OptionsFlowWithReload):
self, user_input: dict[str, Any] | None = None
) -> ConfigFlowResult:
"""Manage the options for the custom component."""
if user_input:
return self.async_create_entry(title="", data=user_input)
errors: dict[str, str] = {}
if user_input:
if user_input.get(CONF_OFFLINE_MODE) and not user_input.get(
CONF_USE_BLUETOOTH
):
errors[CONF_USE_BLUETOOTH] = "bluetooth_required_offline"
else:
return self.async_create_entry(title="", data=user_input)
options_schema = vol.Schema(
{
vol.Optional(
CONF_USE_BLUETOOTH,
default=self.config_entry.options.get(CONF_USE_BLUETOOTH, True),
): cv.boolean,
vol.Optional(
CONF_OFFLINE_MODE,
default=self.config_entry.options.get(CONF_OFFLINE_MODE, False),
): cv.boolean,
}
)
return self.async_show_form(
step_id="init",
data_schema=options_schema,
errors=errors,
)

View File

@@ -6,3 +6,4 @@ DOMAIN: Final = "lamarzocco"
CONF_USE_BLUETOOTH: Final = "use_bluetooth"
CONF_INSTALLATION_KEY: Final = "installation_key"
CONF_OFFLINE_MODE: Final = "offline_mode"

View File

@@ -23,7 +23,7 @@ from homeassistant.core import HomeAssistant, callback
from homeassistant.exceptions import ConfigEntryAuthFailed
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed
from .const import DOMAIN
from .const import CONF_OFFLINE_MODE, DOMAIN
SCAN_INTERVAL = timedelta(seconds=60)
SETTINGS_UPDATE_INTERVAL = timedelta(hours=8)
@@ -49,7 +49,8 @@ type LaMarzoccoConfigEntry = ConfigEntry[LaMarzoccoRuntimeData]
class LaMarzoccoUpdateCoordinator(DataUpdateCoordinator[None]):
"""Base class for La Marzocco coordinators."""
_default_update_interval = SCAN_INTERVAL
_default_update_interval: timedelta | None = SCAN_INTERVAL
_ignore_offline_mode = False
config_entry: LaMarzoccoConfigEntry
update_success = False
@@ -60,12 +61,17 @@ class LaMarzoccoUpdateCoordinator(DataUpdateCoordinator[None]):
device: LaMarzoccoMachine,
) -> None:
"""Initialize coordinator."""
update_interval = self._default_update_interval
if not self._ignore_offline_mode and entry.options.get(
CONF_OFFLINE_MODE, False
):
update_interval = None
super().__init__(
hass,
_LOGGER,
config_entry=entry,
name=DOMAIN,
update_interval=self._default_update_interval,
update_interval=update_interval,
)
self.device = device
self._websocket_task: Task | None = None
@@ -214,6 +220,8 @@ class LaMarzoccoStatisticsUpdateCoordinator(LaMarzoccoUpdateCoordinator):
class LaMarzoccoBluetoothUpdateCoordinator(LaMarzoccoUpdateCoordinator):
"""Class to handle fetching data from the La Marzocco Bluetooth API centrally."""
_ignore_offline_mode = True
async def _internal_async_setup(self) -> None:
"""Initial setup for Bluetooth coordinator."""
await self.device.get_model_info_from_bluetooth()

View File

@@ -197,6 +197,9 @@
"bluetooth_connection_failed": {
"message": "Error while connecting to machine via Bluetooth"
},
"bluetooth_required_offline": {
"message": "Bluetooth is required when offline mode is enabled, but no Bluetooth device was found"
},
"button_error": {
"message": "Error while executing button {key}"
},
@@ -223,12 +226,17 @@
}
},
"options": {
"error": {
"bluetooth_required_offline": "Bluetooth is required when offline mode is enabled."
},
"step": {
"init": {
"data": {
"offline_mode": "Offline Mode",
"use_bluetooth": "Use Bluetooth"
},
"data_description": {
"offline_mode": "Enable offline mode to operate without internet connectivity through Bluetooth. Only local features will be available. Requires Bluetooth to be enabled.",
"use_bluetooth": "Should the integration try to use Bluetooth to control the machine?"
}
}

View File

@@ -10,7 +10,7 @@ from pylamarzocco.exceptions import BluetoothConnectionFailed, RequestNotSuccess
import pytest
from syrupy.assertion import SnapshotAssertion
from homeassistant.components.lamarzocco.const import DOMAIN
from homeassistant.components.lamarzocco.const import CONF_OFFLINE_MODE, DOMAIN
from homeassistant.config_entries import ConfigEntryState
from homeassistant.const import (
EVENT_HOMEASSISTANT_STOP,
@@ -299,6 +299,71 @@ async def test_setup_through_bluetooth_only(
)
async def test_manual_offline_mode_no_bluetooth_device(
hass: HomeAssistant,
mock_lamarzocco: MagicMock,
mock_config_entry_bluetooth: MockConfigEntry,
freezer: FrozenDateTimeFactory,
) -> None:
"""Test manual offline mode with no Bluetooth device found."""
mock_config_entry_bluetooth.add_to_hass(hass)
hass.config_entries.async_update_entry(
mock_config_entry_bluetooth, options={CONF_OFFLINE_MODE: True}
)
await hass.config_entries.async_setup(mock_config_entry_bluetooth.entry_id)
await hass.async_block_till_done()
assert mock_config_entry_bluetooth.state is ConfigEntryState.SETUP_RETRY
async def test_manual_offline_mode(
hass: HomeAssistant,
mock_lamarzocco: MagicMock,
mock_config_entry_bluetooth: MockConfigEntry,
freezer: FrozenDateTimeFactory,
mock_ble_device_from_address: MagicMock,
) -> None:
"""Test that manual offline mode successfully sets up and updates entities via Bluetooth, and marks non-Bluetooth entities as unavailable."""
mock_config_entry_bluetooth.add_to_hass(hass)
hass.config_entries.async_update_entry(
mock_config_entry_bluetooth, options={CONF_OFFLINE_MODE: True}
)
await hass.config_entries.async_setup(mock_config_entry_bluetooth.entry_id)
await hass.async_block_till_done()
main_switch = f"switch.{mock_lamarzocco.serial_number}"
state = hass.states.get(main_switch)
assert state
assert state.state == STATE_ON
# Simulate Bluetooth update changing machine mode to standby
mock_lamarzocco.dashboard.config[
WidgetType.CM_MACHINE_STATUS
].mode = MachineMode.STANDBY
# Trigger Bluetooth coordinator update
freezer.tick(timedelta(seconds=61))
async_fire_time_changed(hass)
await hass.async_block_till_done()
# Verify entity state was updated
state = hass.states.get(main_switch)
assert state
assert state.state == STATE_OFF
# verify other entities are unavailable
sample_entities = (
f"binary_sensor.{mock_lamarzocco.serial_number}_backflush_active",
f"update.{mock_lamarzocco.serial_number}_gateway_firmware",
)
for entity_id in sample_entities:
state = hass.states.get(entity_id)
assert state
assert state.state == STATE_UNAVAILABLE
@pytest.mark.parametrize(
("mock_ble_device", "has_client"),
[

View File

@@ -11,6 +11,7 @@ import pytest
from homeassistant.components.lamarzocco.config_flow import CONF_MACHINE
from homeassistant.components.lamarzocco.const import (
CONF_INSTALLATION_KEY,
CONF_OFFLINE_MODE,
CONF_USE_BLUETOOTH,
DOMAIN,
)
@@ -522,4 +523,47 @@ async def test_options_flow(
assert result["type"] is FlowResultType.CREATE_ENTRY
assert result["data"] == {
CONF_USE_BLUETOOTH: False,
CONF_OFFLINE_MODE: False,
}
async def test_options_flow_bluetooth_required_for_offline_mode(
hass: HomeAssistant,
mock_config_entry: MockConfigEntry,
) -> None:
"""Test options flow validates that Bluetooth is required when offline mode is enabled."""
await async_init_integration(hass, mock_config_entry)
assert mock_config_entry.state is ConfigEntryState.LOADED
result = await hass.config_entries.options.async_init(mock_config_entry.entry_id)
assert result["type"] is FlowResultType.FORM
assert result["step_id"] == "init"
result = await hass.config_entries.options.async_configure(
result["flow_id"],
user_input={
CONF_USE_BLUETOOTH: False,
CONF_OFFLINE_MODE: True,
},
)
await hass.async_block_till_done()
assert result["type"] is FlowResultType.FORM
assert result["step_id"] == "init"
assert result["errors"] == {CONF_USE_BLUETOOTH: "bluetooth_required_offline"}
# recover
result = await hass.config_entries.options.async_configure(
result["flow_id"],
user_input={
CONF_USE_BLUETOOTH: True,
CONF_OFFLINE_MODE: True,
},
)
await hass.async_block_till_done()
assert result["type"] is FlowResultType.CREATE_ENTRY
assert result["data"] == {
CONF_USE_BLUETOOTH: True,
CONF_OFFLINE_MODE: True,
}