Commit graph

5853 commits

Author SHA1 Message Date
Khalim Conn-Kowlessar
607e52a354 Cohort residual slice 1: 000490 secondary heating cascade closes -£104 cost gap
Lodges `secondary_heating_type=691` (Electricity Electric Panel) on
000490 `build_epc()` to match the U985 worksheet's "Secondary Heating:
Electricity Electric Panel, convector or radiant heaters, SAP Code 691,
Efficiency 100%". Pre-fix the cert lodged no secondary system →
`_secondary_fraction` returned 0.0 → all useful space heat routed to
main 1 → main_fuel +1357 kWh over PDF, secondary -1118 under PDF, cost
-£104 under PDF (-12.9% residual).

Post-fix: Table 11 fraction 0.1000 for gas-combi category cascade fires
→ main 1 = 11491.89 kWh, secondary = 1126.21 kWh. Total cost £807.42
vs PDF £807.54 (Δ -£0.12, -0.015%). SAP integer 58 vs PDF 57 (delta 1,
was 6); continuous 57.57 vs 57.40 (delta 0.18).

E2E test updates:
- New worksheet-level pin `result.secondary_heating_fuel_kwh_per_yr ≈
  U985 (215) = 1118.3275` at abs=10 (loose — absorbs the +0.7% upstream
  useful space heating overshoot which propagates 1:1 to (215). Tightens
  to abs=1e-3 when the useful bias closes).
- Per-fixture constant `LINE_215_SECONDARY_HEATING_FUEL_KWH = 1118.3275`.
- 000490 SAP integer ceiling tightened 3 → 1; continuous 3.0 → 0.5.
- Removed xfail on `test_elmhurst_000490_end_to_end_sap_score_currently_
  within_3_points` and `test_000490_cert_to_inputs_fuel_cost_closes_to_
  within_5pct` — both now pass.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-22 09:53:33 +00:00
Khalim Conn-Kowlessar
fd9df9e502 Appendix L slice 3: docs — SPEC_COVERAGE rows + ADR-0010 amendment + heuristic deprecation note
SPEC_COVERAGE:
- §5 row: note new `annual_lighting_kwh` public leaf + InternalGainsResult
  field + per-fixture U985 (232) abs=1e-4 pin across all 6 Elmhurst fixtures.
- Appendix L row: "Full (cost + gains)" — closes both sides via the same
  L1-L11 cascade; legacy heuristic noted with rip-pending callsites.

ADR-0010 Amendment "Appendix L lighting (2026-05-22)":
- Two engine bugs surfaced + fixed: cosine modulation integral (uniform
  +0.146% bias from continuous-formula vs Σ(L11 monthly)) and cert EPC
  under-lodgement (`build_epc()` skipped bulb counts + windows).
- 000474 hits SAP integer delta=0 (first Elmhurst fixture across the gate).
- 000490 SAP integer + fuel cost xfailed (strict) — Appendix L direction
  correct, other components broken (fuel pricing, Table D1-3 Ecodesign,
  main heating +2.5%). Tracked as next ticket.
- Golden cohort PE tolerance widened 30→35 with rationale.
- Deferred work: cohort SAP-integer residual hunt, heuristic deletion,
  RdSAP→API integration test (end-state e2e harness).

`predicted_lighting_kwh` deprecation note: cite ADR-0010 amendment; name
the two legacy callsites (`domain.ml.ecf`, `domain.ml.transform`) that
block deletion.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-22 09:34:09 +00:00
Khalim Conn-Kowlessar
54cc9bd3ba Appendix L slice 2: cert→cascade lighting kWh + 000474 e2e closes to delta=0
Closes the +9.2% cost residual on 000474 by swapping the legacy
`predicted_lighting_kwh` heuristic (9.3 × TFA × bulb-share) for the
spec-faithful Appendix L L1-L11 cascade that already drove §5 (67)
internal gains. Single source of truth via `InternalGainsResult.
lighting_kwh_per_yr`; the cost side and the gains side now derive
from the same monthly distribution.

Engine bug found during the wire-up: `annual_lighting_kwh` was
returning the L1-L9 continuous formula value (E_L), but the SAP10.2
worksheet lodges line ref (232) as Σ(L11 monthly distribution).
Discrete cosine integral Σ(n_m × factor) / 365 = 0.998539, not 1.0
exactly — caused a uniform +0.146% bias across all 6 Elmhurst
fixtures. Fixed by factoring a private `_lighting_monthly_kwh` and
having `annual_lighting_kwh` sum it directly. Synthetic S1 pin
updated 189.152079 → 188.875713 (post-modulation).

Cert-side updates: lodge `low_energy_fixed_lighting_bulbs_count` +
`sap_windows` on 000474 / 000490 `build_epc()` so the cert→cascade
path receives spec-faithful inputs (was defaulting to L5b/L8c +
C_daylight=1.433 no-bonus). Per-fixture `LINE_232_LIGHTING_KWH_PER_YR`
constants pin each U985 PDF value at 4 d.p.

E2E pin updates (per feedback-e2e-validation-philosophy: components
validate the engine; SAP integer = delta 0 is the integration gate):
- 000474 SAP integer ceiling tightened 3 → 0 (lands at 62 = PDF 62
  exactly); continuous 3.5 → 0.5 (lands at 0.09)
- 000490 SAP integer + fuel-cost tests xfail with rationale —
  Appendix L direction is correct (lighting closes 614→171 = PDF
  171.4217), but cost residual widens past 5% / SAP delta widens
  3→6 due to other broken components (fuel pricing, Table D1-3
  Ecodesign, main heating +2.5%). Re-enable when those close.
- Golden fixtures `_PE_TOLERANCE_KWH_PER_M2` widened 30 → 35 to
  absorb the elec-PEF × lighting-Δ contribution (~4 kWh/m²) on a
  non-Elmhurst cohort whose pre-existing residual already sat near
  -28 kWh/m² from unrelated components.

Component validation: `result.lighting_kwh_per_yr == PDF (232)` to
abs=1e-4 for 000474 (139.9452) + 000490 (171.4217); §5 worksheet-
level pin on `InternalGainsResult.lighting_kwh_per_yr` covers all 6
Elmhurst fixtures at the same tolerance. Existing §5 (67) LINE_67
monthly tuple tests remain green (refactor preserves monthly W
distribution).

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-22 09:15:22 +00:00
Jun-te Kim
0dee917094 unsanistiesed address list instead of raw address lit 2026-05-22 08:27:59 +00:00
Jun-te Kim
91bb4b6571 address list 2026-05-22 08:22:13 +00:00
Jun-te Kim
84098e28ff raw address list repo 2026-05-22 08:17:37 +00:00
Jun-te Kim
5b677dedbe SAL 2026-05-22 08:15:11 +00:00
Jun-te Kim
cf14a4e3aa rename to SAL and AssetList and RawAddresses 2026-05-22 08:14:46 +00:00
Khalim Conn-Kowlessar
f4352587f7 Appendix L slice 1: annual_lighting_kwh extraction
Surfaces the SAP10.2 Appendix L L1-L12 annual lighting kWh as a public
free fn alongside lighting_monthly_w. Refactors lighting_monthly_w to
compose it. One source of truth shared by the §5 gains side and the
forthcoming cost side (inputs.lighting_kwh_per_yr) — slice 2 wires
internal_gains_from_cert + cert_to_inputs.

Synthetic L1-L12 test pins a hand-computed dwelling
(TFA=100, N=2.0, C_L=10000, ε=100, D=1.0) at 189.152079 kWh, abs=1e-3.

6-fixture LINE_67 conformance tests (Elmhurst 000474..000516) act as a
regression check on the monthly cosine + 0.85 internal-fraction
composition — all green.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-22 07:44:24 +00:00
Jun-te Kim
acb306f7b9 asset list from landlord 2026-05-22 07:34:50 +00:00
Khalim Conn-Kowlessar
95086f957e docs: handover for next agent — Appendix L lighting → §11a/§12a/§13a sweep
Rewrites HANDOVER_NEXT.md after the §10a + §4 HW work. Two tickets:

1. **Appendix L lighting predictor swap** (immediate) — replace the
   legacy `domain.ml.demand.predicted_lighting_kwh` heuristic with
   the spec-faithful Appendix L L1-L12 cascade already living in
   `worksheet/internal_gains._lighting_gains_monthly_w`. Single
   slice; closes 000474 cost residual from +9.2% toward ~0%.
2. **§11a SAP rating + §12a CO2 + §13a Primary Energy sweep** —
   per-end-use cascade on top of the §10a `FuelCostResult`. Mirrors
   §10a's pattern (kwargs orchestrator + Result dataclass + cert_to_
   inputs precompute + calculator delegation). ~5 slices.

Carries §A current-state residuals table (000474 + 000490 post-§4
HW), §B/§C tickets with slice plans, §D codebase pointers, §G
deferred-list cross-reference to ADR-0010 amendment + SPEC_COVERAGE
remaining-work sections.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-21 23:03:52 +00:00
Khalim Conn-Kowlessar
c9eb231a9c §4 HW slice 3: docs — SPEC_COVERAGE row + Remaining work + golden note
- SPEC_COVERAGE §4 row: closed (combi-gas single-rate) — PCDB Table
  3b + Eq D1 cascade. 000474 + 000490 HW kWh ≤0.1% of PDF.
  Remaining §4 work list refreshed: storage / FGHRS rows, Table 3c
  two-profile, Electric CPSU Appendix F, instant electric shower,
  Appendix L lighting (separate ticket per memory).
- §4 slice progress table: (61)m row updated with `760e25de` commit
  pointer + dual sourcing (Table 3a default + PCDB Table 3b row 1
  override).
- test_golden_fixtures.py: SAP_TOLERANCE stays ±11 — §4 HW closure
  doesn't shift the oil-heated golden certs because they aren't PCDB
  Table-3b-listed. Comment block updated with the §4 slice 2 note.

No code changes — docs + tolerance comment only.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-21 22:58:16 +00:00
Khalim Conn-Kowlessar
02fc9e4d47 §4 HW slice 2: Equation D1 monthly water-eff cascade
Closes the residual ~1.2% on 000474 HW kWh that slice 1 left (PCDB
Table 3b combi loss landed (61) correctly but the divisor was still
the scalar PCDB summer efficiency 87.0%). Slice 2 promotes that
scalar to the SAP10.2 Appendix D §D2.1 (2) Equation D1 monthly
cascade — η_water,monthly = (Q_space + Q_water) / (Q_space/η_winter
+ Q_water/η_summer) — and folds it into the cert_to_inputs flow:

- worksheet/water_heating.py: water_efficiency_monthly_via_equation_
  d1(...) — pure function over winter/summer efficiencies + (98c)m
  × (204) + (64)m monthly tuples. Implements the spec's two early-
  outs (η_summer ≥ η_winter → all months = η_summer; zero-demand
  months → η_summer).
- rdsap/cert_to_inputs.py: splits _hot_water_fuel_kwh_per_yr (now
  removed) into:
  - _water_heating_worksheet_and_gains: runs §4 (45..65) early so
    §5/§7/§8 can consume (65)m heat gains.
  - _apply_water_efficiency: invoked after §8 produces (98c)m, picks
    monthly cascade for PCDB-tested combis with distinct winter/
    summer effs, falls back to scalar divisor otherwise.
  Pulled secondary_fraction_value computation forward of §4 so the
  post-§8 Q_space = (98c)m × (204) derivation has it in scope.

Outcomes (closes the §10a slice-2 deferred §4 HW debt):
- 000474 HW kWh: 2622 → 2320 (slice 1) → 2292 ✓ matches PDF 2292
  to 0.0%. SAP delta 4 → 3 (ceiling tightened 4 → 3).
- 000490 HW kWh: 3028 → 3028 (slice 1 no-op, no PCDB Table 3b
  data) → 2847 ✓ matches PDF 2851 to 0.1%. SAP delta 2 → 3
  (ceiling loosened 2 → 3 — the closer HW kWh exposes spec-version
  drift on the 000490 cost figure that PDF lodged under cert-
  assessor era prices per ADR-0010 §3).
- 486 tests passing across the domain package; 13 pre-existing
  pyright errors on cert_to_inputs (no net new from this slice).

Remaining 000474 +9% cost residual is Appendix L lighting (528 vs
~169 back-derived) — separate ticket per project memory
`project_section_4_hw_next_ticket` "secondary upstream" note.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-21 22:54:29 +00:00
Khalim Conn-Kowlessar
760e25dea9 §4 HW slice 1: PCDB Table 3b combi-loss override
Closes the dominant ~92% of the 000474 HW kWh +14.4% residual that
the post-§10a Table 32 cost-side fix exposed (pre-§10a wrong prices
had been masking it). 000474 HW fuel kWh tightens 2622 → 2320 (+1.2%
over PDF 2292); remaining +1.2% closes when slice 2 (Eq D1 monthly
cascade) lands. 000490 unaffected — PCDB 10328 lodges separate_dhw_
tests=0 (no Table 3b/3c data), falls through to existing Table 3a
default.

- tables/pcdb/parser.py: GasOilBoilerRecord gains 7 typed fields per
  BRE PCDF Spec v1.0 §7.11 — subsidiary_type (field 16), store_type
  (field 39), separate_dhw_tests (field 48), rejected_energy_
  proportion_r1 (field 51), loss_factor_f1_kwh_per_day (field 52),
  loss_factor_f2_kwh_per_day (field 56), rejected_factor_f3_per_
  litre (field 57). Field positions cross-verified against PDF Σ(61)
  = 337.27 vs 000474 worksheet pin 337.19 (Δ 0.02%).
- worksheet/water_heating.py: combi_loss_monthly_kwh_table_3b_row_1_
  instantaneous(r1, F1, energy_content (45)m, daily HW (44)m) — SAP10.2
  Appendix J Table 3b row 1 formula (61)m = (45)m × r1 × fu + F1 × n_m.
  Other Table 3b rows (storage variants) and Table 3c (two-profile)
  deferred until a fixture exercises.
- rdsap/cert_to_inputs.py: _pcdb_table_3b_combi_loss_override builds
  the (61)m override from the PCDB record when separate_dhw_tests=1
  + subsidiary=0 + store_type=0 (instantaneous non-storage path).
  _hot_water_fuel_kwh_per_yr threaded with pcdb_record kwarg; calls
  water_heating_from_cert with the override when present.
- docs/sap-spec/pcdb_table_105_gas_oil_boilers.jsonl: regenerated via
  the ETL to surface the new typed fields alongside the existing
  efficiency columns.

484 tests passing (was 479). e2e ceilings hold: 000474 SAP delta
4 → 3 (within current ceiling of 4 — will tighten further after
slice 2 Eq D1 cascade lands).

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-21 22:26:41 +00:00
Khalim Conn-Kowlessar
ae8c946179 docs: §10a slice 3 — ADR-0010 amendment + SPEC_COVERAGE row
- ADR-0010 amendment: narrow the SAP10.2 spec target — §10a/§10b
  cost prices source from RdSAP10 Table 32 (per RdSAP10 §19.1),
  not SAP10.2 Table 12. CO2 + PEF stay on Table 12 (RdSAP10 §19.2
  says they're identical). Closes out the 000490 "spec-version
  drift" framing as wrong-table + missing-standing-charges, not
  corpus drift. Names §4 HW + Appendix L as the next-ticket
  upstream debt that pre-§10a wrong-prices had been masking.
- SPEC_COVERAGE: new §10a row (32-field FuelCostResult, three new
  tables/* + worksheet/* modules, per-line-ref status, Remaining
  §10a work list). Updates §12 to "folded into §10a". Updates
  header attribution.

No code changes in this commit — docs only.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-21 20:11:37 +00:00
Khalim Conn-Kowlessar
adfa7f60da §10a slice 2: cert_to_inputs._fuel_cost + calculator delegation
Wires the §10a Fuel costs worksheet block (slice 1's orchestrator)
into the cert → calculator pipeline:

- CalculatorInputs.fuel_cost composite slot (default zero sentinel
  for synthetic-test constructions that don't supply one).
- cert_to_inputs._fuel_cost precompute — resolves Table 32 prices
  per end-use, calls additional_standing_charges_gbp per Table 12
  note (a) for gas/off-peak gating, calls the fuel_cost orchestrator.
  Off-peak certs return a zero FuelCostResult sentinel so the legacy
  scalar fuel-cost-per-kWh fallback fires; Table 12a high-rate
  fraction split + Table12aSystem mapping is deferred to a future
  §10a follow-up slice.
- calculator delegates total_cost / per-end-use cost intermediate
  dict entries to inputs.fuel_cost when the precompute is non-zero;
  falls back to the legacy inline kWh × price math for synthetic
  CalculatorInputs constructions (will be removed when the test
  corpus migrates to fuel_cost=).

Outcomes:
- 000490 SAP rating ceiling tightened 6 → 2 (marquee close-out:
  the cost gap was wrong-table + missing-standing-charges, not the
  spec-version drift the handover suspected).
- 000474 SAP rating ceiling loosened 2 → 4 (post-§10a Table 32 +
  standing-charge fix exposes upstream §4 HW kWh + Appendix L
  lighting overestimates that the wrong pre-§10a prices had been
  masking). §4 HW worksheet tightening is the next ticket.
- Golden corpus SAP tolerance widened 7 → 11 — Table 32 oil price
  rose +55% (4.94 → 7.64 p/kWh) which moves oil-heated certs whose
  lodged actual_sap pre-dates Table 32 (ADR-0010 §3 Validation
  Cohort discipline).
- 2 new cert-round-trip conformance tests on test_fuel_cost.py
  (000474 within existing e2e tolerance; 000490 within 5%).

660 tests passing across the domain package. 0 net new pyright
errors on touched modules.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-21 20:08:41 +00:00
Khalim Conn-Kowlessar
0f255165d5 §10a slice 1: table_32 + table_12a + fuel_cost orchestrator
Establishes the SAP10.2 §10a fuel-cost worksheet block per the
Table 32 (RdSAP10 prices, PDF page 95) + Table 12a (high-rate
fractions, PDF page 191) rewrite scoped in the §10a handover.

- tables/table_32.py: 28 fuel rows pinned verbatim; standing
  charges per fuel; API-enum → Table 32 translation; note (a)
  gating in `additional_standing_charges_gbp` (gas use + off-peak
  electricity rules).
- tables/table_12a.py: `Tariff` enum (incl. TEN_HOUR for spec
  completeness — RdSAP cert flow doesn't route here);
  `Table12aSystem` + `OtherUse` enums; `space_heating_high_rate_
  fraction` / `water_heating_high_rate_fraction` / `other_use_high_
  rate_fraction` lookups; `tariff_from_meter_type` cert resolver
  (Unknown → STANDARD per the spec-faithful policy).
- worksheet/fuel_cost.py: 32-field `FuelCostResult` (line refs
  (240)..(255)) + kwargs `fuel_cost` orchestrator. Off-peak split
  via `_split` helper applied to main 1 / main 2 / secondary /
  water-heating rows; pumps/fans/lighting/cooling/instant-shower
  at single rate (per-row Table 12a split deferred); (252) PV
  credit negative; (255) clamped to >= 0.

130 synthetic unit tests pinned. CalculatorInputs wiring + cert_
to_inputs rewrite + 6-fixture conformance follow in slice 2.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-21 19:40:16 +00:00
Jun-te Kim
94cbf5f516 changed useraddress landlordasset list 2026-05-21 16:59:57 +00:00
Jun-te Kim
8baa4c82aa save correct progress 2026-05-21 16:57:14 +00:00
Jun-te Kim
c833a3c91b feat: implement get_col_to_description_mappings
Collect, per shared landlord_additional_info key, the list of values
across all UserAddress entries. Preserves first-seen key order and
input order of values.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-21 16:33:54 +00:00
Jun-te Kim
b14f98788e added landlord orchestration 2026-05-21 16:32:50 +00:00
Jun-te Kim
4830f82b58 test: add failing tests for get_col_to_description_mappings
Drive the contract for LandlordDescriptionOverridesOrchestrator.
get_col_to_description_mappings: given a list of UserAddress sharing
the same landlord_additional_info keys, return each key mapped to the
list of values found across all addresses.

Tests are red — the method still raises NotImplementedError.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-21 16:32:15 +00:00
Daniel Roth
dc5fcc41bf
Merge pull request #1119 from Hestia-Homes/claude/model-p4
Solar API Client
2026-05-21 17:09:17 +01:00
Daniel Roth
0d4462d131 GoogleSolarApi translates BuildingInsightsNotFoundError to sentinel dict 🟩 2026-05-21 16:06:54 +00:00
Daniel Roth
50e9d88752 GoogleSolarApi delegates get_building_insights to GoogleSolarApiClient 🟩 2026-05-21 16:00:59 +00:00
Daniel Roth
521294ad91 GoogleSolarApi delegates get_building_insights to GoogleSolarApiClient 🟥 2026-05-21 15:59:22 +00:00
Daniel Roth
b1933c07c3 GoogleSolarApiClient propagates exception after retry exhaustion 🟩 2026-05-21 15:53:39 +00:00
Daniel Roth
d1ca9be580 GoogleSolarApiClient raises BuildingInsightsNotFoundError on 404 entity-not-found 🟩 2026-05-21 15:52:52 +00:00
Daniel Roth
573656be64 GoogleSolarApiClient raises BuildingInsightsNotFoundError on 404 entity-not-found 🟥 2026-05-21 15:52:21 +00:00
Daniel Roth
497ef1faed GoogleSolarApiClient retries on transient HTTP errors 🟩 2026-05-21 15:51:01 +00:00
Daniel Roth
44217bf361 GoogleSolarApiClient retries on transient HTTP errors 🟥 2026-05-21 15:49:57 +00:00
Daniel Roth
b0106fa93d GoogleSolarApiClient fetches building insights from the Solar API 🟩 2026-05-21 15:48:45 +00:00
Daniel Roth
629fc34a0f GoogleSolarApiClient fetches building insights from the Solar API 🟥 2026-05-21 15:46:47 +00:00
Daniel Roth
2906a60bbb Merge branch 'main' into feature/solar-api-client 2026-05-21 15:30:57 +00:00
Daniel Roth
dadc202983
Merge pull request #1114 from Hestia-Homes/feature/script-to-rename-sharepoint-files
Script to rename sharepoint files
2026-05-21 16:30:27 +01:00
Daniel Roth
9f7c16ccbd add address list 2026-05-21 15:30:03 +00:00
Jun-te Kim
68809a68c1 renamed to landlord description overrides 2026-05-21 14:26:05 +00:00
Jun-te Kim
aea7251107 added files for landlord_overrides 2026-05-21 14:21:50 +00:00
Khalim Conn-Kowlessar
6d6767ce62 docs: handover for §10a Fuel costs + SPEC_COVERAGE PCDB-followup updates
HANDOVER_NEXT.md rewritten for §10a Fuel costs (xlsx rows ~614-740, spec lines 8044-8084). Covers (240)..(255) line refs — refactor calculator.py inline cost arithmetic into a worksheet-shape `FuelCostResult` orchestrator following the §9a precedent. Three-slice plan: orchestrator + dataclass + synthetic, atomic CalculatorInputs/cert_to_inputs wiring, Table 12a off-peak split.

SPEC_COVERAGE PCDB slice progress table picks up the two fixture-lodgement commits + the mapper-chain regression test; updated narrative confirms no domain-model changes were needed.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-21 13:49:04 +00:00
Khalim Conn-Kowlessar
15d6b78149 pcdb followup: e2e mapper-chain regression test for main_heating_index_number
Pins the API JSON → EpcPropertyDataMapper → CalculatorInputs chain for the 4 corpus PCDB-listed golden certs. Asserts (a) `main_heating_index_number` survives the mapper hop, (b) `cert_to_inputs` resolves Table 105 record by that ID and applies the winter efficiency. Catches future regressions where a mapper change might drop the PCDB pointer silently.

Confirms the API → domain → calculator chain works end-to-end without any new domain object field — `MainHeatingDetail.main_heating_index_number` has existed since schema 17_1 and all mapper paths from 17_1+ pass it through verbatim.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-21 13:45:33 +00:00
Daniel Roth
94e23228cc Merge branch 'main' into feature/script-to-rename-sharepoint-files 2026-05-21 13:40:44 +00:00
Jun-te Kim
617342ef85
Merge pull request #1113 from Hestia-Homes/feature/hyde
some excel files are formatted differently
2026-05-21 12:33:57 +01:00
Khalim Conn-Kowlessar
7d4f3d78dc pcdb followup: 000474 fixture lodges main_heating_index_number=16839; e2e ceiling 7 → 2
PDF "PCDF boiler reference: 16839 Vaillant ecoTEC pro 28 88.70%" → fixture sets `main_heating_index_number=16839` + `main_heating_data_source=1`. cert_to_inputs PCDB precedence resolves Table 105 record 16839 (Vaillant ecoTEC pro 28 VUW GB 286/5-3, 2005-2015, winter 88.7%, summer 87.0%, comparative HW 75.1%).

000474 e2e impact — near-closure:
  - main_heating_efficiency: 0.80 → 0.887
  - hot_water_kwh: 3020 → 2622 (PDF 2292, gap +32% → +14.4%)
  - total_fuel_cost: £778 → £652 (PDF £656, gap +19% → -0.6%)
  - SAP rating: 69 → 63 (PDF 62, +7 → +1)

Ceiling tightened 7 → 2 (SAP integer) and 7.0 → 2.0 (continuous). Residual HW kWh gap (+14.4%) is the Appendix J §3b PCDB combi-loss row that our HW cascade still defaults from Table 3a — closes in a future §4 slice.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-21 11:30:18 +00:00
Khalim Conn-Kowlessar
1b43c95ca6 pcdb followup: 000490 fixture lodges main_heating_index_number=10328 (Vaillant Ecotec Pro)
PDF "PCDF boiler reference: 10328 Vaillant Ecotec Pro 88.20%" lodgement → fixture now sets `main_heating_index_number=10328` + `main_heating_data_source=1` per the API's standard PCDB-lodgement shape. cert_to_inputs PCDB precedence cascade picks up Table 105 record 10328 (winter eff 88.2%, summer 79.6%) and overrides the Table 4a category-2 default.

make_main_heating_detail extended to expose main_heating_index_number / main_heating_data_source / sap_main_heating_code kwargs so fixtures can lodge PCDB pointers without hand-building MainHeatingDetail.

000490 e2e impact:
  - main_heating_fuel: 14334 → 13001.3 kWh (PDF 13003.85 — gap closes to <0.1%, was +10%)
  - HW fuel: 3090.47 → 3028.27 kWh (PDF 2850.57 — gap closes +8.4% → +6.2%)
  - total_fuel_cost: £756.99 → £706.23 (PDF £807.54 — diverges -6.3% → -12.5%, ADR-0010 §3 spec-version artifact)
  - SAP rating: 60 → 63 (PDF 57 — +3 → +6)

The fuel-kWh tightening is the spec-faithful direction. The cost / SAP residuals widen because the cert pre-dates the 14-March-2025 SAP10.2 amendment which lowered gas unit prices ~13%; per ADR-0010 §3 only certs lodged ≥2025-07-01 are spec-comparable on cost-driven outputs. The e2e SAP ceiling is raised 3 → 6 and the cost-rel tolerance 0.10 → 0.15 with a docstring naming the drivers; tightens further when the Validation Cohort filter + Ecodesign/Appendix N adjustments land.

000474 also flagged as Vaillant ecoTEC pro PCDB-lodged; awaiting user's PCDB code lookup for that fixture.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-21 11:22:39 +00:00
Jun-te Kim
dbd03de842 local run changes 2026-05-21 10:37:13 +00:00
Jun-te Kim
856ea6eb93 undo postcodesplitter changes 2026-05-21 10:12:08 +00:00
Khalim Conn-Kowlessar
e63516cb26 docs: SPEC_COVERAGE PCDB integration row + slice progress + gap-list update
Updates the Prioritised gap list item 1 narrative: Table 105 (gas/oil boilers) integration done; remaining = Table 362 heat pumps + Appendix N cascade, equation D1 monthly water heating, Tables 313/353/391/506 ancillaries, condensing-boiler Ecodesign corrections.

Adds a PCDB slice progress table: ETL parser + 8-table JSONL output (`fe04cd3a`), runtime lookup module (`23678228`), cert_to_inputs precedence cascade with widened golden tolerance (`a104dd55`).

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-21 09:51:10 +00:00
Khalim Conn-Kowlessar
a104dd559a pcdb slice 3: cert_to_inputs precedence cascade — Table 105 overrides Table 4a/4b
SAP 10.2 Appendix D2.1: when a cert lodges `main_heating_index_number` that resolves to a Table 105 (Gas/Oil Boilers) PCDB record, the PCDB winter seasonal efficiency overrides `seasonal_efficiency(...)` and the PCDB summer seasonal efficiency overrides the water heating Table 4a default (scalar — equation D1 monthly cascade deferred per Q5 grilling). Heat-network DLF override still wins where applicable.

Cert path: `main is not None and main.main_heating_index_number is not None and gas_oil_boiler_record(...)` is not None → use PCDB; otherwise fall back to the existing Table 4a/4b cascade. None of the 6 Elmhurst fixtures lodge a PCDB pointer, so their existing conformance is untouched.

Synthetic test pins the new precedence: a typical gas-combi cert with `main_heating_index_number=98` (verified Baxi 000098, winter eff 66.0%) produces `inputs.main_heating_efficiency == 0.66` instead of the 0.84 Table 4b code-102 default.

Golden corpus tolerance widened ±5 → ±7 SAP and ±25 → ±30 kWh/m² PE: two of the four PCDB-listed golden certs drift by ~1 SAP point / ~1.5 kWh/m² under the spec-faithful PCDB winter/summer override (the lodged assessor scores predate consistent PCDB use, so the gap widens for those two certs and stays under tolerance for the other two). All 343 tests pass.

Follow-up slices (named in SPEC_COVERAGE remaining work): equation D1 per-month water cascade, Appendix N heat-pump in-use factor + MCS / flow-temp adjustment via Table 362, FGHRS/WWHRS/HIU/storage-heater cert-side cascades via Tables 313/353/506/391.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-21 09:49:58 +00:00
Jun-te Kim
c5ab795f85 redeploy old postcode splitter 2026-05-21 09:46:47 +00:00
Khalim Conn-Kowlessar
236782287e pcdb slice 2: runtime gas_oil_boiler_record lookup via Table 105 NDJSON
Adds the cert-side lookup surface for Table 105: gas_oil_boiler_record(pcdb_id) -> Optional[GasOilBoilerRecord]. NDJSON is loaded once at module import, parsed into a by-pcdb-id dict, and cached by the Python runtime. Lookup is O(1).

Returns None when the cert's main_heating_index_number is not in Table 105 — caller falls back to the existing seasonal_efficiency(...) Table 4a/4b cascade.

Two tests pin the contract: verified Baxi 000098 lookup returns the typed record with brand "Baxi Heating", winter eff 66.0%, summer eff 56.0%; unknown PCDB ID returns None.

Slice 3 wires gas_oil_boiler_record into cert_to_inputs.main_heating_efficiency and water_efficiency precedence cascades per Q5=B (space heating + water heating scalar override).

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-21 09:45:28 +00:00