mirror of
https://github.com/Hestia-Homes/Model.git
synced 2026-06-08 11:17:27 +00:00
Bill / EnergyBreakdown / BillDerivation / sap_fuel were under domain/property_baseline/ only because Baseline was built first. The Modelling stage now needs them too, so move them (and their tests) to a neutral domain/billing/ — Fuel/FuelRates already live in the shared domain/fuel_rates/. Avoids a modelling -> property_baseline cross-stage import and a package name that wrongly implies ownership (ADR-0011, ADR-0014 amendment). Pure git mv + import rewrite across 10 files; 40 billing/baseline/repo tests pass, pyright strict clean. CONTEXT.md Bill Derivation location updated. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
129 lines
4.6 KiB
Python
129 lines
4.6 KiB
Python
from __future__ import annotations
|
|
|
|
from collections.abc import Mapping, Sequence
|
|
from dataclasses import dataclass
|
|
from enum import Enum
|
|
from typing import Optional, TYPE_CHECKING
|
|
|
|
from domain.fuel_rates.fuel import Fuel
|
|
from domain.billing.sap_fuel import sap_code_to_fuel
|
|
|
|
if TYPE_CHECKING:
|
|
from domain.sap10_calculator.calculator import SapResult
|
|
|
|
|
|
class BillSection(Enum):
|
|
"""A user-meaningful slice of the annual energy bill — the calculator's raw
|
|
end uses folded into the sections the UI shows (ADR-0014)."""
|
|
|
|
HEATING = "HEATING"
|
|
HOT_WATER = "HOT_WATER"
|
|
LIGHTING = "LIGHTING"
|
|
APPLIANCES = "APPLIANCES"
|
|
COOKING = "COOKING"
|
|
PUMPS_FANS = "PUMPS_FANS"
|
|
COOLING = "COOLING"
|
|
|
|
|
|
@dataclass(frozen=True)
|
|
class EnergyLine:
|
|
"""One section's delivered energy on one fuel. A section may have more than
|
|
one line (e.g. gas main heating + electric secondary heating)."""
|
|
|
|
section: BillSection
|
|
fuel: Fuel
|
|
kwh: float
|
|
|
|
|
|
@dataclass(frozen=True)
|
|
class EnergyBreakdown:
|
|
"""A Property's delivered energy per end use, the input to Bill Derivation.
|
|
``exported_kwh`` is PV generation exported to the grid, credited at the SEG
|
|
rate."""
|
|
|
|
lines: Sequence[EnergyLine]
|
|
exported_kwh: float = 0.0
|
|
|
|
@classmethod
|
|
def from_sap_result(cls, result: "SapResult") -> "EnergyBreakdown":
|
|
"""Fold a calculator `SapResult`'s per-end-use delivered kWh into billable
|
|
`EnergyLine`s (ADR-0014). Heating (main / main-2 / secondary) and hot water
|
|
are billed at their resolved fuel (`sap_code_to_fuel`); lighting / pumps-
|
|
fans / appliances / cooking / cooling are electricity by construction. A
|
|
line is emitted only when its kWh is positive; PV export carries to
|
|
`exported_kwh` for the SEG credit. The `from_*` factory mirrors
|
|
`Performance.from_sap_result`; living on the target keeps the calculator
|
|
free of any `property_baseline` dependency."""
|
|
candidates = [
|
|
_fuelled_line(
|
|
BillSection.HEATING,
|
|
result.main_heating_fuel_code,
|
|
result.main_heating_fuel_kwh_per_yr,
|
|
),
|
|
_fuelled_line(
|
|
BillSection.HEATING,
|
|
result.main_2_heating_fuel_code,
|
|
result.main_2_heating_fuel_kwh_per_yr,
|
|
),
|
|
_fuelled_line(
|
|
BillSection.HEATING,
|
|
result.secondary_heating_fuel_code,
|
|
result.secondary_heating_fuel_kwh_per_yr,
|
|
),
|
|
_fuelled_line(
|
|
BillSection.HOT_WATER,
|
|
result.hot_water_fuel_code,
|
|
result.hot_water_kwh_per_yr,
|
|
),
|
|
_electric_line(BillSection.LIGHTING, result.lighting_kwh_per_yr),
|
|
_electric_line(BillSection.PUMPS_FANS, result.pumps_fans_kwh_per_yr),
|
|
_electric_line(BillSection.APPLIANCES, result.appliances_kwh_per_yr),
|
|
_electric_line(BillSection.COOKING, result.cooking_kwh_per_yr),
|
|
_electric_line(BillSection.COOLING, result.space_cooling_fuel_kwh_per_yr),
|
|
]
|
|
return cls(
|
|
lines=[line for line in candidates if line is not None],
|
|
exported_kwh=result.pv_exported_kwh_per_yr,
|
|
)
|
|
|
|
|
|
def _fuelled_line(
|
|
section: BillSection, fuel_code: Optional[int], kwh: float
|
|
) -> Optional[EnergyLine]:
|
|
"""An `EnergyLine` for a fuelled end use, or None when it has no energy. A
|
|
positive kWh with no resolved fuel code is a data gap — raise rather than
|
|
bill it at a default (mirrors the calculator's strict-raise discipline)."""
|
|
if kwh <= 0:
|
|
return None
|
|
if fuel_code is None:
|
|
raise ValueError(
|
|
f"{section.value} has {kwh} kWh but no fuel code on the SapResult; "
|
|
"cannot attribute a billing fuel"
|
|
)
|
|
return EnergyLine(section=section, fuel=sap_code_to_fuel(fuel_code), kwh=kwh)
|
|
|
|
|
|
def _electric_line(section: BillSection, kwh: float) -> Optional[EnergyLine]:
|
|
"""An electricity `EnergyLine` for an electric end use, or None when zero."""
|
|
if kwh <= 0:
|
|
return None
|
|
return EnergyLine(section=section, fuel=Fuel.ELECTRICITY, kwh=kwh)
|
|
|
|
|
|
@dataclass(frozen=True)
|
|
class BillSectionCost:
|
|
"""One section's rolled-up delivered kWh and annual cost (£)."""
|
|
|
|
kwh: float
|
|
cost_gbp: float
|
|
|
|
|
|
@dataclass(frozen=True)
|
|
class Bill:
|
|
"""A Property's annual energy bill, composed per section plus the per-meter
|
|
standing charges and the SEG export credit, and the total (ADR-0014)."""
|
|
|
|
sections: Mapping[BillSection, BillSectionCost]
|
|
standing_charges_gbp: float
|
|
seg_credit_gbp: float
|
|
total_gbp: float
|