mirror of
https://github.com/home-assistant/core.git
synced 2025-07-23 13:17:32 +00:00
Add additional coverage for ESPHome sensor and number (#95226)
This commit is contained in:
parent
74fb1ba51d
commit
ad17a89531
@ -312,7 +312,6 @@ omit =
|
|||||||
homeassistant/components/esphome/domain_data.py
|
homeassistant/components/esphome/domain_data.py
|
||||||
homeassistant/components/esphome/entry_data.py
|
homeassistant/components/esphome/entry_data.py
|
||||||
homeassistant/components/esphome/light.py
|
homeassistant/components/esphome/light.py
|
||||||
homeassistant/components/esphome/number.py
|
|
||||||
homeassistant/components/esphome/switch.py
|
homeassistant/components/esphome/switch.py
|
||||||
homeassistant/components/etherscan/sensor.py
|
homeassistant/components/etherscan/sensor.py
|
||||||
homeassistant/components/eufy/*
|
homeassistant/components/eufy/*
|
||||||
|
@ -74,9 +74,7 @@ class EsphomeNumber(EsphomeEntity[NumberInfo, NumberState], NumberEntity):
|
|||||||
def native_value(self) -> float | None:
|
def native_value(self) -> float | None:
|
||||||
"""Return the state of the entity."""
|
"""Return the state of the entity."""
|
||||||
state = self._state
|
state = self._state
|
||||||
if math.isnan(state.state):
|
if state.missing_state or math.isnan(state.state):
|
||||||
return None
|
|
||||||
if state.missing_state:
|
|
||||||
return None
|
return None
|
||||||
return state.state
|
return state.state
|
||||||
|
|
||||||
|
91
tests/components/esphome/test_number.py
Normal file
91
tests/components/esphome/test_number.py
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
"""Test ESPHome numbers."""
|
||||||
|
|
||||||
|
import math
|
||||||
|
from unittest.mock import call
|
||||||
|
|
||||||
|
from aioesphomeapi import (
|
||||||
|
APIClient,
|
||||||
|
NumberInfo,
|
||||||
|
NumberMode as ESPHomeNumberMode,
|
||||||
|
NumberState,
|
||||||
|
)
|
||||||
|
|
||||||
|
from homeassistant.components.number import (
|
||||||
|
ATTR_VALUE,
|
||||||
|
DOMAIN as NUMBER_DOMAIN,
|
||||||
|
SERVICE_SET_VALUE,
|
||||||
|
)
|
||||||
|
from homeassistant.const import ATTR_ENTITY_ID, STATE_UNKNOWN
|
||||||
|
from homeassistant.core import HomeAssistant
|
||||||
|
|
||||||
|
|
||||||
|
async def test_generic_number_entity(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
mock_client: APIClient,
|
||||||
|
mock_generic_device_entry,
|
||||||
|
) -> None:
|
||||||
|
"""Test a generic number entity."""
|
||||||
|
entity_info = [
|
||||||
|
NumberInfo(
|
||||||
|
object_id="mynumber",
|
||||||
|
key=1,
|
||||||
|
name="my number",
|
||||||
|
unique_id="my_number",
|
||||||
|
max_value=100,
|
||||||
|
min_value=0,
|
||||||
|
step=1,
|
||||||
|
unit_of_measurement="%",
|
||||||
|
)
|
||||||
|
]
|
||||||
|
states = [NumberState(key=1, state=50)]
|
||||||
|
user_service = []
|
||||||
|
await mock_generic_device_entry(
|
||||||
|
mock_client=mock_client,
|
||||||
|
entity_info=entity_info,
|
||||||
|
user_service=user_service,
|
||||||
|
states=states,
|
||||||
|
)
|
||||||
|
state = hass.states.get("number.test_my_number")
|
||||||
|
assert state is not None
|
||||||
|
assert state.state == "50"
|
||||||
|
|
||||||
|
await hass.services.async_call(
|
||||||
|
NUMBER_DOMAIN,
|
||||||
|
SERVICE_SET_VALUE,
|
||||||
|
{ATTR_ENTITY_ID: "number.test_my_number", ATTR_VALUE: 50},
|
||||||
|
blocking=True,
|
||||||
|
)
|
||||||
|
mock_client.number_command.assert_has_calls([call(1, 50)])
|
||||||
|
mock_client.number_command.reset_mock()
|
||||||
|
|
||||||
|
|
||||||
|
async def test_generic_number_nan(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
mock_client: APIClient,
|
||||||
|
mock_generic_device_entry,
|
||||||
|
) -> None:
|
||||||
|
"""Test a generic number entity with nan state."""
|
||||||
|
entity_info = [
|
||||||
|
NumberInfo(
|
||||||
|
object_id="mynumber",
|
||||||
|
key=1,
|
||||||
|
name="my number",
|
||||||
|
unique_id="my_number",
|
||||||
|
max_value=100,
|
||||||
|
min_value=0,
|
||||||
|
step=1,
|
||||||
|
unit_of_measurement="%",
|
||||||
|
mode=ESPHomeNumberMode.SLIDER,
|
||||||
|
)
|
||||||
|
]
|
||||||
|
states = [NumberState(key=1, state=math.nan)]
|
||||||
|
user_service = []
|
||||||
|
await mock_generic_device_entry(
|
||||||
|
mock_client=mock_client,
|
||||||
|
entity_info=entity_info,
|
||||||
|
user_service=user_service,
|
||||||
|
states=states,
|
||||||
|
)
|
||||||
|
state = hass.states.get("number.test_my_number")
|
||||||
|
assert state is not None
|
||||||
|
assert state.state == STATE_UNKNOWN
|
@ -1,6 +1,9 @@
|
|||||||
"""Test ESPHome sensors."""
|
"""Test ESPHome sensors."""
|
||||||
|
import math
|
||||||
|
|
||||||
from aioesphomeapi import (
|
from aioesphomeapi import (
|
||||||
APIClient,
|
APIClient,
|
||||||
|
EntityCategory as ESPHomeEntityCategory,
|
||||||
LastResetType,
|
LastResetType,
|
||||||
SensorInfo,
|
SensorInfo,
|
||||||
SensorState,
|
SensorState,
|
||||||
@ -10,8 +13,10 @@ from aioesphomeapi import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
from homeassistant.components.sensor import ATTR_STATE_CLASS, SensorStateClass
|
from homeassistant.components.sensor import ATTR_STATE_CLASS, SensorStateClass
|
||||||
from homeassistant.const import STATE_UNKNOWN
|
from homeassistant.const import ATTR_ICON, STATE_UNKNOWN
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
|
from homeassistant.helpers import entity_registry as er
|
||||||
|
from homeassistant.helpers.entity import EntityCategory
|
||||||
|
|
||||||
|
|
||||||
async def test_generic_numeric_sensor(
|
async def test_generic_numeric_sensor(
|
||||||
@ -41,6 +46,41 @@ async def test_generic_numeric_sensor(
|
|||||||
assert state.state == "50"
|
assert state.state == "50"
|
||||||
|
|
||||||
|
|
||||||
|
async def test_generic_numeric_sensor_with_entity_category_and_icon(
|
||||||
|
hass: HomeAssistant,
|
||||||
|
mock_client: APIClient,
|
||||||
|
mock_generic_device_entry,
|
||||||
|
) -> None:
|
||||||
|
"""Test a generic sensor entity."""
|
||||||
|
entity_info = [
|
||||||
|
SensorInfo(
|
||||||
|
object_id="mysensor",
|
||||||
|
key=1,
|
||||||
|
name="my sensor",
|
||||||
|
unique_id="my_sensor",
|
||||||
|
entity_category=ESPHomeEntityCategory.CONFIG,
|
||||||
|
icon="mdi:leaf",
|
||||||
|
)
|
||||||
|
]
|
||||||
|
states = [SensorState(key=1, state=50)]
|
||||||
|
user_service = []
|
||||||
|
await mock_generic_device_entry(
|
||||||
|
mock_client=mock_client,
|
||||||
|
entity_info=entity_info,
|
||||||
|
user_service=user_service,
|
||||||
|
states=states,
|
||||||
|
)
|
||||||
|
state = hass.states.get("sensor.test_my_sensor")
|
||||||
|
assert state is not None
|
||||||
|
assert state.state == "50"
|
||||||
|
assert state.attributes[ATTR_ICON] == "mdi:leaf"
|
||||||
|
entity_reg = er.async_get(hass)
|
||||||
|
entry = entity_reg.async_get("sensor.test_my_sensor")
|
||||||
|
assert entry is not None
|
||||||
|
assert entry.unique_id == "my_sensor"
|
||||||
|
assert entry.entity_category is EntityCategory.CONFIG
|
||||||
|
|
||||||
|
|
||||||
async def test_generic_numeric_sensor_state_class_measurement(
|
async def test_generic_numeric_sensor_state_class_measurement(
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
mock_client: APIClient,
|
mock_client: APIClient,
|
||||||
@ -70,6 +110,11 @@ async def test_generic_numeric_sensor_state_class_measurement(
|
|||||||
assert state is not None
|
assert state is not None
|
||||||
assert state.state == "50"
|
assert state.state == "50"
|
||||||
assert state.attributes[ATTR_STATE_CLASS] == SensorStateClass.MEASUREMENT
|
assert state.attributes[ATTR_STATE_CLASS] == SensorStateClass.MEASUREMENT
|
||||||
|
entity_reg = er.async_get(hass)
|
||||||
|
entry = entity_reg.async_get("sensor.test_my_sensor")
|
||||||
|
assert entry is not None
|
||||||
|
assert entry.unique_id == "my_sensor"
|
||||||
|
assert entry.entity_category is None
|
||||||
|
|
||||||
|
|
||||||
async def test_generic_numeric_sensor_device_class_timestamp(
|
async def test_generic_numeric_sensor_device_class_timestamp(
|
||||||
@ -130,6 +175,56 @@ async def test_generic_numeric_sensor_legacy_last_reset_convert(
|
|||||||
assert state.attributes[ATTR_STATE_CLASS] == SensorStateClass.TOTAL_INCREASING
|
assert state.attributes[ATTR_STATE_CLASS] == SensorStateClass.TOTAL_INCREASING
|
||||||
|
|
||||||
|
|
||||||
|
async def test_generic_numeric_sensor_no_state(
|
||||||
|
hass: HomeAssistant, mock_client: APIClient, mock_generic_device_entry
|
||||||
|
) -> None:
|
||||||
|
"""Test a generic numeric sensor that has no state."""
|
||||||
|
entity_info = [
|
||||||
|
SensorInfo(
|
||||||
|
object_id="mysensor",
|
||||||
|
key=1,
|
||||||
|
name="my sensor",
|
||||||
|
unique_id="my_sensor",
|
||||||
|
)
|
||||||
|
]
|
||||||
|
states = []
|
||||||
|
user_service = []
|
||||||
|
await mock_generic_device_entry(
|
||||||
|
mock_client=mock_client,
|
||||||
|
entity_info=entity_info,
|
||||||
|
user_service=user_service,
|
||||||
|
states=states,
|
||||||
|
)
|
||||||
|
state = hass.states.get("sensor.test_my_sensor")
|
||||||
|
assert state is not None
|
||||||
|
assert state.state == STATE_UNKNOWN
|
||||||
|
|
||||||
|
|
||||||
|
async def test_generic_numeric_sensor_nan_state(
|
||||||
|
hass: HomeAssistant, mock_client: APIClient, mock_generic_device_entry
|
||||||
|
) -> None:
|
||||||
|
"""Test a generic numeric sensor that has nan state."""
|
||||||
|
entity_info = [
|
||||||
|
SensorInfo(
|
||||||
|
object_id="mysensor",
|
||||||
|
key=1,
|
||||||
|
name="my sensor",
|
||||||
|
unique_id="my_sensor",
|
||||||
|
)
|
||||||
|
]
|
||||||
|
states = [SensorState(key=1, state=math.nan, missing_state=False)]
|
||||||
|
user_service = []
|
||||||
|
await mock_generic_device_entry(
|
||||||
|
mock_client=mock_client,
|
||||||
|
entity_info=entity_info,
|
||||||
|
user_service=user_service,
|
||||||
|
states=states,
|
||||||
|
)
|
||||||
|
state = hass.states.get("sensor.test_my_sensor")
|
||||||
|
assert state is not None
|
||||||
|
assert state.state == STATE_UNKNOWN
|
||||||
|
|
||||||
|
|
||||||
async def test_generic_numeric_sensor_missing_state(
|
async def test_generic_numeric_sensor_missing_state(
|
||||||
hass: HomeAssistant, mock_client: APIClient, mock_generic_device_entry
|
hass: HomeAssistant, mock_client: APIClient, mock_generic_device_entry
|
||||||
) -> None:
|
) -> None:
|
||||||
|
Loading…
x
Reference in New Issue
Block a user