Add time extended sensor for Nintendo Switch parental controls (#159227)

This commit is contained in:
Jordan Harvey
2025-12-29 13:00:37 +00:00
committed by GitHub
parent be7b7f3d25
commit a8114b7e4f
5 changed files with 85 additions and 8 deletions

View File

@@ -4,6 +4,7 @@ from __future__ import annotations
from collections.abc import Callable
from dataclasses import dataclass
from datetime import datetime
from enum import StrEnum
from homeassistant.components.sensor import (
@@ -28,13 +29,15 @@ class NintendoParentalControlsSensor(StrEnum):
PLAYING_TIME = "playing_time"
TIME_REMAINING = "time_remaining"
TIME_EXTENDED = "time_extended"
@dataclass(kw_only=True, frozen=True)
class NintendoParentalControlsSensorEntityDescription(SensorEntityDescription):
"""Description for Nintendo parental controls sensor entities."""
value_fn: Callable[[Device], int | float | None]
value_fn: Callable[[Device], datetime | int | float | None]
available_fn: Callable[[Device], bool] = lambda device: True
SENSOR_DESCRIPTIONS: tuple[NintendoParentalControlsSensorEntityDescription, ...] = (
@@ -54,6 +57,15 @@ SENSOR_DESCRIPTIONS: tuple[NintendoParentalControlsSensorEntityDescription, ...]
state_class=SensorStateClass.MEASUREMENT,
value_fn=lambda device: device.today_time_remaining,
),
NintendoParentalControlsSensorEntityDescription(
key=NintendoParentalControlsSensor.TIME_EXTENDED,
translation_key=NintendoParentalControlsSensor.TIME_EXTENDED,
native_unit_of_measurement=UnitOfTime.MINUTES,
device_class=SensorDeviceClass.DURATION,
state_class=SensorStateClass.MEASUREMENT,
value_fn=lambda device: device.extra_playing_time,
available_fn=lambda device: device.extra_playing_time is not None,
),
)
@@ -86,6 +98,11 @@ class NintendoParentalControlsSensorEntity(NintendoDevice, SensorEntity):
self.entity_description = description
@property
def native_value(self) -> int | float | None:
def native_value(self) -> datetime | int | float | None:
"""Return the native value."""
return self.entity_description.value_fn(self._device)
@property
def available(self) -> bool:
"""Return if the sensor is available."""
return super().available and self.entity_description.available_fn(self._device)

View File

@@ -50,6 +50,9 @@
"playing_time": {
"name": "Used screen time"
},
"time_extended": {
"name": "Extended screen time"
},
"time_remaining": {
"name": "Screen time remaining"
}

View File

@@ -41,6 +41,7 @@ def mock_nintendo_device() -> Device:
mock.today_time_remaining = 10
mock.bedtime_alarm = time(hour=19)
mock.timer_mode = DeviceTimerMode.DAILY
mock.extra_playing_time = 30
mock.add_extra_time.return_value = None
mock.set_bedtime_alarm.return_value = None
mock.update_max_daily_playtime.return_value = None

View File

@@ -1,5 +1,61 @@
# serializer version: 1
# name: test_number[sensor.home_assistant_test_screen_time_remaining-entry]
# name: test_sensor[sensor.home_assistant_test_extended_screen_time-entry]
EntityRegistryEntrySnapshot({
'aliases': set({
}),
'area_id': None,
'capabilities': dict({
'state_class': <SensorStateClass.MEASUREMENT: 'measurement'>,
}),
'config_entry_id': <ANY>,
'config_subentry_id': <ANY>,
'device_class': None,
'device_id': <ANY>,
'disabled_by': None,
'domain': 'sensor',
'entity_category': None,
'entity_id': 'sensor.home_assistant_test_extended_screen_time',
'has_entity_name': True,
'hidden_by': None,
'icon': None,
'id': <ANY>,
'labels': set({
}),
'name': None,
'options': dict({
'sensor': dict({
'suggested_display_precision': 2,
}),
}),
'original_device_class': <SensorDeviceClass.DURATION: 'duration'>,
'original_icon': None,
'original_name': 'Extended screen time',
'platform': 'nintendo_parental_controls',
'previous_unique_id': None,
'suggested_object_id': None,
'supported_features': 0,
'translation_key': <NintendoParentalControlsSensor.TIME_EXTENDED: 'time_extended'>,
'unique_id': 'testdevid_time_extended',
'unit_of_measurement': <UnitOfTime.MINUTES: 'min'>,
})
# ---
# name: test_sensor[sensor.home_assistant_test_extended_screen_time-state]
StateSnapshot({
'attributes': ReadOnlyDict({
'device_class': 'duration',
'friendly_name': 'Home Assistant Test Extended screen time',
'state_class': <SensorStateClass.MEASUREMENT: 'measurement'>,
'unit_of_measurement': <UnitOfTime.MINUTES: 'min'>,
}),
'context': <ANY>,
'entity_id': 'sensor.home_assistant_test_extended_screen_time',
'last_changed': <ANY>,
'last_reported': <ANY>,
'last_updated': <ANY>,
'state': '30',
})
# ---
# name: test_sensor[sensor.home_assistant_test_screen_time_remaining-entry]
EntityRegistryEntrySnapshot({
'aliases': set({
}),
@@ -39,7 +95,7 @@
'unit_of_measurement': <UnitOfTime.MINUTES: 'min'>,
})
# ---
# name: test_number[sensor.home_assistant_test_screen_time_remaining-state]
# name: test_sensor[sensor.home_assistant_test_screen_time_remaining-state]
StateSnapshot({
'attributes': ReadOnlyDict({
'device_class': 'duration',
@@ -55,7 +111,7 @@
'state': '10',
})
# ---
# name: test_number[sensor.home_assistant_test_used_screen_time-entry]
# name: test_sensor[sensor.home_assistant_test_used_screen_time-entry]
EntityRegistryEntrySnapshot({
'aliases': set({
}),
@@ -95,7 +151,7 @@
'unit_of_measurement': <UnitOfTime.MINUTES: 'min'>,
})
# ---
# name: test_number[sensor.home_assistant_test_used_screen_time-state]
# name: test_sensor[sensor.home_assistant_test_used_screen_time-state]
StateSnapshot({
'attributes': ReadOnlyDict({
'device_class': 'duration',

View File

@@ -13,14 +13,14 @@ from . import setup_integration
from tests.common import MockConfigEntry, snapshot_platform
async def test_number(
async def test_sensor(
hass: HomeAssistant,
mock_config_entry: MockConfigEntry,
mock_nintendo_client: AsyncMock,
entity_registry: er.EntityRegistry,
snapshot: SnapshotAssertion,
) -> None:
"""Test number platform."""
"""Test sensor platform."""
with patch(
"homeassistant.components.nintendo_parental_controls._PLATFORMS",
[Platform.SENSOR],