hass.tibber_prices/custom_components/tibber_prices/services/get_price.py
Julian Pawlowski 1d065b11cd fix(services): use injected now in resolve_search_range day offset
_resolve_time_with_day_offset() was calling dt_util.now() internally
instead of using the injected now parameter. This caused incorrect date
calculations in tests and any caller that passes a specific reference time.

Also add missing price_rank_* sensor keys to TIME_SENSITIVE_ENTITY_KEYS
in coordinator/constants.py so quarter-hour refresh is registered for all
11 price rank sensors (current/next/previous interval and hour variants).

Rename dt as dt_utils → dt as dt_util (ICN001) across 11 files to follow
the project-wide import alias convention. Apply ruff auto-fixes for import
ordering and collapsing single-item imports throughout the codebase.

Released-Bug: no
2026-04-14 19:33:24 +00:00

181 lines
5.5 KiB
Python

"""
Service handler for get_price service.
This service fetches raw price interval data for any time range using the
interval pool's intelligent caching. Only intervals not already cached are
fetched from the Tibber API.
Functions:
handle_get_price: Service handler for fetching price data
"""
from __future__ import annotations
import logging
from typing import TYPE_CHECKING
from zoneinfo import ZoneInfo
import voluptuous as vol
from custom_components.tibber_prices.const import DOMAIN
from homeassistant.exceptions import ServiceValidationError
from homeassistant.helpers import config_validation as cv
from homeassistant.util import dt as dt_util
from .helpers import get_entry_and_data
if TYPE_CHECKING:
from datetime import datetime
from homeassistant.core import HomeAssistant, ServiceCall, ServiceResponse
_LOGGER = logging.getLogger(__name__)
GET_PRICE_SERVICE_NAME = "get_price"
GET_PRICE_SERVICE_SCHEMA = vol.Schema(
{
vol.Optional("entry_id", default=""): cv.string,
vol.Required("start_time"): cv.datetime,
vol.Required("end_time"): cv.datetime,
}
)
def _raise_user_data_error() -> None:
"""Raise user data not available error."""
msg = "User data not available"
raise ServiceValidationError(
translation_domain=DOMAIN,
translation_key="user_data_not_available",
) from ValueError(msg)
async def handle_get_price(call: ServiceCall) -> ServiceResponse:
"""
Handle get_price service call.
Fetches price data for a specified time range using the interval pool.
The pool intelligently caches intervals and only fetches missing data from the API.
Args:
call: Service call with entry_id, start_time, and end_time
Returns:
Dict with price data and metadata
Raises:
ServiceValidationError: If arguments invalid or request fails
"""
hass: HomeAssistant = call.hass
entry_id: str = call.data.get("entry_id", "")
start_time: datetime = call.data["start_time"]
end_time: datetime = call.data["end_time"]
# Validate and get entry data
entry, coordinator, _data = get_entry_and_data(hass, entry_id)
# Get home_id from entry
home_id = entry.data.get("home_id")
if not home_id:
raise ServiceValidationError(
translation_domain=DOMAIN,
translation_key="missing_home_id",
)
# Get API client from coordinator
api_client = coordinator.api
# Get user data (needed for timezone) - coordinator doesn't expose this publicly yet
user_data = coordinator._cached_user_data # noqa: SLF001
if not user_data:
_raise_user_data_error()
# Extract home timezone from user_data
home_timezone = None
if user_data and "viewer" in user_data:
for home in user_data["viewer"].get("homes", []):
if home.get("id") == home_id:
home_timezone = home.get("timeZone")
break
if not home_timezone:
raise ServiceValidationError(
translation_domain=DOMAIN,
translation_key="timezone_not_found",
)
# Ensure times are timezone-aware using HOME timezone (not HA server timezone!)
# CRITICAL TWO-STEP PROCESS:
# 1. GUI gives us naive datetime in HA SERVER timezone → localize to HA timezone
# 2. Convert from HA timezone to HOME timezone (Tibber home location)
home_tz = ZoneInfo(home_timezone)
if start_time.tzinfo is None:
# Step 1: Localize to HA server timezone
start_time = dt_util.as_local(start_time)
# Step 2: Convert to home timezone
start_time = start_time.astimezone(home_tz)
if end_time.tzinfo is None:
# Step 1: Localize to HA server timezone
end_time = dt_util.as_local(end_time)
# Step 2: Convert to home timezone
end_time = end_time.astimezone(home_tz)
# Validate: end must be after start
if end_time <= start_time:
raise ServiceValidationError(
translation_domain=DOMAIN,
translation_key="end_before_start",
)
_LOGGER.info(
"get_price service called: entry_id=%s, home_id=%s, range=%s to %s",
entry_id,
home_id,
start_time,
end_time,
)
try:
# Get interval pool from entry runtime_data (one pool per config entry)
pool = entry.runtime_data.interval_pool
# Call the interval pool to get intervals (with intelligent caching)
# Single-home architecture: pool knows its home_id, no parameter needed
price_info, _api_called = await pool.get_intervals(
api_client=api_client,
user_data=user_data,
start_time=start_time,
end_time=end_time,
)
# Note: We ignore api_called flag here - service always returns requested data
# regardless of whether it came from cache or was fetched fresh from API
except Exception as error:
_LOGGER.exception("Error fetching price data")
raise ServiceValidationError(
translation_domain=DOMAIN,
translation_key="price_fetch_failed",
) from error
else:
# Add metadata to response
response = {
"home_id": home_id,
"start_time": start_time.isoformat(),
"end_time": end_time.isoformat(),
"interval_count": len(price_info),
"price_info": price_info,
}
_LOGGER.info(
"get_price service completed: fetched %d intervals",
len(price_info),
)
return response