hass-core/homeassistant/components/monarch_money/sensor.py
Jeef e4347e5520
Add Monarch Money Integration (#124014)
* Initial commit

* Second commit - with some coverage but errors abount

* Updated testing coverage

* Should be just about ready for PR

* Adding some error handling for wonky acocunts

* Adding USD hardcoded as this is all that is currently supported i believe

* updating snapshots

* updating entity descrition a little

* Addign cashflow in

* adding aggregate sensors

* tweak icons

* refactor some type stuff as well as initialize the pr comment addressing process

* remove empty fields from manifest

* Update homeassistant/components/monarchmoney/sensor.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* move stuff

* get logging out of try block

* get logging out of try block

* using Subscription ID as stored in config entry for unique id soon

* new unique id

* giving cashflow a better unique id

* Moving subscription id stuff into setup of coordinator

* Update homeassistant/components/monarchmoney/config_flow.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* ruff ruff

* ruff ruff

* split ot value and balance sensors... need to go tos leep

* removed icons

* Moved summary into a data class

* efficenty increase

* Update homeassistant/components/monarchmoney/coordinator.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* Update homeassistant/components/monarchmoney/coordinator.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* Update homeassistant/components/monarchmoney/coordinator.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* Update homeassistant/components/monarchmoney/entity.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* refactor continues

* removed a comment

* forgot to add a little bit of info

* updated snapshot

* Updates to monarch money using the new typed/wrapper setup

* backing lib update

* fixing manifest

* fixing manifest

* fixing manifest

* Version 0.2.0

* fixing some types

* more type fixes

* cleanup and bump

* no check

* i think i got it all

* the last thing

* update domain name

* i dont know what is in this commit

* The Great Renaming

* Moving to dict style accounting - as per request

* updating backing deps

* Update homeassistant/components/monarch_money/entity.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* Update tests/components/monarch_money/test_config_flow.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* Update tests/components/monarch_money/test_config_flow.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* Update tests/components/monarch_money/test_config_flow.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* Update homeassistant/components/monarch_money/sensor.py

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>

* some changes

* fixing capitalizaton

* test test test

* Adding dupe test

* addressing pr stuff

* forgot snapshot

* Fix

* Fix

* Update homeassistant/components/monarch_money/sensor.py

---------

Co-authored-by: Joost Lekkerkerker <joostlek@outlook.com>
2024-09-11 17:09:16 +02:00

182 lines
6.1 KiB
Python

"""Sensor config - monarch money."""
from collections.abc import Callable
from dataclasses import dataclass
from datetime import datetime
from typedmonarchmoney.models import MonarchAccount, MonarchCashflowSummary
from homeassistant.components.sensor import (
SensorDeviceClass,
SensorEntity,
SensorEntityDescription,
SensorStateClass,
)
from homeassistant.const import CURRENCY_DOLLAR, PERCENTAGE, EntityCategory
from homeassistant.core import HomeAssistant
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from homeassistant.helpers.typing import StateType
from . import MonarchMoneyConfigEntry
from .entity import MonarchMoneyAccountEntity, MonarchMoneyCashFlowEntity
@dataclass(frozen=True, kw_only=True)
class MonarchMoneyAccountSensorEntityDescription(SensorEntityDescription):
"""Describe an account sensor entity."""
value_fn: Callable[[MonarchAccount], StateType | datetime]
picture_fn: Callable[[MonarchAccount], str | None] | None = None
@dataclass(frozen=True, kw_only=True)
class MonarchMoneyCashflowSensorEntityDescription(SensorEntityDescription):
"""Describe a cashflow sensor entity."""
summary_fn: Callable[[MonarchCashflowSummary], StateType]
# These sensors include assets like a boat that might have value
MONARCH_MONEY_VALUE_SENSORS: tuple[MonarchMoneyAccountSensorEntityDescription, ...] = (
MonarchMoneyAccountSensorEntityDescription(
key="value",
translation_key="value",
state_class=SensorStateClass.TOTAL,
device_class=SensorDeviceClass.MONETARY,
value_fn=lambda account: account.balance,
picture_fn=lambda account: account.logo_url,
native_unit_of_measurement=CURRENCY_DOLLAR,
),
)
# Most accounts are balance sensors
MONARCH_MONEY_SENSORS: tuple[MonarchMoneyAccountSensorEntityDescription, ...] = (
MonarchMoneyAccountSensorEntityDescription(
key="currentBalance",
translation_key="balance",
state_class=SensorStateClass.TOTAL,
device_class=SensorDeviceClass.MONETARY,
value_fn=lambda account: account.balance,
picture_fn=lambda account: account.logo_url,
native_unit_of_measurement=CURRENCY_DOLLAR,
),
)
MONARCH_MONEY_AGE_SENSORS: tuple[MonarchMoneyAccountSensorEntityDescription, ...] = (
MonarchMoneyAccountSensorEntityDescription(
key="age",
translation_key="age",
device_class=SensorDeviceClass.TIMESTAMP,
entity_category=EntityCategory.DIAGNOSTIC,
value_fn=lambda account: account.last_update,
),
)
MONARCH_CASHFLOW_SENSORS: tuple[MonarchMoneyCashflowSensorEntityDescription, ...] = (
MonarchMoneyCashflowSensorEntityDescription(
key="sum_income",
translation_key="sum_income",
summary_fn=lambda summary: summary.income,
state_class=SensorStateClass.TOTAL,
device_class=SensorDeviceClass.MONETARY,
native_unit_of_measurement=CURRENCY_DOLLAR,
),
MonarchMoneyCashflowSensorEntityDescription(
key="sum_expense",
translation_key="sum_expense",
summary_fn=lambda summary: summary.expenses,
state_class=SensorStateClass.TOTAL,
device_class=SensorDeviceClass.MONETARY,
native_unit_of_measurement=CURRENCY_DOLLAR,
),
MonarchMoneyCashflowSensorEntityDescription(
key="savings",
translation_key="savings",
summary_fn=lambda summary: summary.savings,
state_class=SensorStateClass.TOTAL,
device_class=SensorDeviceClass.MONETARY,
native_unit_of_measurement=CURRENCY_DOLLAR,
),
MonarchMoneyCashflowSensorEntityDescription(
key="savings_rate",
translation_key="savings_rate",
summary_fn=lambda summary: summary.savings_rate * 100,
suggested_display_precision=1,
native_unit_of_measurement=PERCENTAGE,
),
)
async def async_setup_entry(
hass: HomeAssistant,
config_entry: MonarchMoneyConfigEntry,
async_add_entities: AddEntitiesCallback,
) -> None:
"""Set up Monarch Money sensors for config entries."""
mm_coordinator = config_entry.runtime_data
entity_list: list[MonarchMoneySensor | MonarchMoneyCashFlowSensor] = [
MonarchMoneyCashFlowSensor(
mm_coordinator,
sensor_description,
)
for sensor_description in MONARCH_CASHFLOW_SENSORS
]
entity_list.extend(
MonarchMoneySensor(
mm_coordinator,
sensor_description,
account,
)
for account in mm_coordinator.balance_accounts
for sensor_description in MONARCH_MONEY_SENSORS
)
entity_list.extend(
MonarchMoneySensor(
mm_coordinator,
sensor_description,
account,
)
for account in mm_coordinator.accounts
for sensor_description in MONARCH_MONEY_AGE_SENSORS
)
entity_list.extend(
MonarchMoneySensor(
mm_coordinator,
sensor_description,
account,
)
for account in mm_coordinator.value_accounts
for sensor_description in MONARCH_MONEY_VALUE_SENSORS
)
async_add_entities(entity_list)
class MonarchMoneyCashFlowSensor(MonarchMoneyCashFlowEntity, SensorEntity):
"""Cashflow summary sensor."""
entity_description: MonarchMoneyCashflowSensorEntityDescription
@property
def native_value(self) -> StateType:
"""Return the state."""
return self.entity_description.summary_fn(self.summary_data)
class MonarchMoneySensor(MonarchMoneyAccountEntity, SensorEntity):
"""Define a monarch money sensor."""
entity_description: MonarchMoneyAccountSensorEntityDescription
@property
def native_value(self) -> StateType | datetime:
"""Return the state."""
return self.entity_description.value_fn(self.account_data)
@property
def entity_picture(self) -> str | None:
"""Return the picture of the account as provided by monarch money if it exists."""
if self.entity_description.picture_fn is not None:
return self.entity_description.picture_fn(self.account_data)
return None