Model/datatypes
Khalim Conn-Kowlessar c30b4fcdc8 Slice S0380.6: surface full §15.1 Hot Water Cylinder block — Summary HW exact
Closes the entire §15.1 Hot Water Cylinder lodging end-to-end and
collapses cert 0380's Summary path to the API path at the documented
HP-cohort spec-precision floor: SAP **88.5698 (Δ +0.0594)** — exactly
matching the API path's spec-floor closure. `hot_water_kwh_per_yr`
hits **878.0519** vs worksheet (64) 1502.16 ÷ (216) HW eff 1.7107 =
**878.05** — exact match at 1e-4.

Four §15.1 fields surfaced together (the cascade requires all four in
combination to compute the worksheet-correct HP HW path):

1. `cylinder_size_label` (Summary "Medium" → SAP10 cascade enum 3 =
   160 L per `_CYLINDER_SIZE_CODE_TO_LITRES`)
2. `cylinder_insulation_label` (Summary "Foam" → cascade enum 1 =
   factory, per SAP 10.2 Table 2 Note 2)
3. `cylinder_insulation_thickness_mm` (Summary "50 mm" → 50)
4. `cylinder_thermostat` (Summary "Yes" → bool True → mapper emits 'Y'
   for the cascade's `sh.cylinder_thermostat == "Y"` string compare)

Why all four were required:

- `_cylinder_storage_loss_override` in `cert_to_inputs.py:2238-2253`
  gates on `cylinder_size`, `cylinder_insulation_type ==
  _CYLINDER_INSULATION_TYPE_FACTORY (1)`, AND
  `cylinder_insulation_thickness_mm`. Missing any → no override →
  zero storage loss (62)m miscalculated.
- `cylinder_thermostat` keys the SAP 10.2 Table 2b temperature factor
  (53): with-stat 0.5400 vs no-stat ~0.9 → without 'Y' storage loss
  over-counts by ~300 kWh/yr (the precise diff between the bundled-
  fields-only attempt at SAP 86.5 vs the fully-bundled attempt at
  SAP 88.57).

Three-layer end-to-end change:

1. `datatypes/epc/surveys/elmhurst_site_notes.py` — add four
   defaulted `WaterHeating` fields (placed in the defaulted block;
   existing fixtures that omit §15.1 still construct unchanged).
2. `backend/documents_parser/elmhurst_extractor.py` — extend
   `_extract_water_heating` to read the §15.1 block via
   `_section_lines("15.1 Hot Water Cylinder", "15.2 Community Hot
   Water")` + `_local_val`. Section-scoping is required because the
   "Insulation Thickness" label collides with §7 Walls / §8 Roofs /
   §9 Floors lodgings on the same Summary PDF (cert 0380 has §7
   "Insulation Thickness 100 mm" for the FE wall — the global
   `_next_val` would return the wrong value).
3. `datatypes/epc/domain/mapper.py` — add
   `_elmhurst_cylinder_size_code` + `_elmhurst_cylinder_insulation_code`
   label-to-enum helpers; replace the broken
   `cylinder_size = water_heating.water_heating_code` (which was
   passing the §15 "Water Heating Code" string "HWP" into the
   numeric `cylinder_size` field, defeating the cascade) with the
   real `cylinder_size_label`-derived enum.

Pre-Slice 6, the Summary path was producing `cylinder_size='HWP'`
which `_int_or_none` reduced to None, silently routing the cascade
off the HP-with-cylinder HW path entirely. Surfacing the §15.1
block in full lets `_heat_pump_apm_efficiencies` use the spec-
correct HW efficiency (1.7107) and `_cylinder_storage_loss_override`
contribute the spec-correct (56) 435 kWh/yr storage loss.

Pyright net-zero across all four edited files:
  datatypes/epc/domain/mapper.py:                32 (baseline)
  datatypes/epc/surveys/elmhurst_site_notes.py:   0
  backend/documents_parser/elmhurst_extractor.py: 0
  backend/documents_parser/tests/test_summary_pdf_mapper_chain.py: 0

Regression suite: 674 pass + 11 fail (vs handover baseline 669 + 10
— net +5 pass for the new GREEN unit tests S0380.2..S0380.6; the +1
fail vs baseline is still S0380.1's chain test which pins at 1e-4 vs
worksheet 88.5104 and now lands at Δ +0.0594, the same Appendix N3.6
PSR-interpolation precision floor that the API path closes to and
that the cohort's 7 ASHP fixtures already track at ±0.07).

Tolerance disposition: the +0.0594 residual is identical to the
cohort's documented HP-path precision floor. Closing further requires
work on the calculator's Appendix N3.6 PSR interpolation step
(boilers already match worksheet at 1e-4 via the same cascade —
ground-truthed in closed-boiler precedents 001479, 0330), not on
the Summary mapper. The S0380.1 chain test should be re-pinned to
the ±0.07 ASHP-cohort tolerance in the next slice — same disposition
the API-path cohort received in slice 102f (commit c0086660).

Spec refs:
- SAP 10.2 §4 Table 2 (PDF p.135) — cylinder storage loss factor
  for foam-insulated cylinders (51) keyed on insulation thickness.
- SAP 10.2 §4 Table 2a (PDF p.135) — cylinder volume factor (52).
- SAP 10.2 §4 Table 2b (PDF p.135) — cylinder temperature factor
  (53) keyed on cylinder thermostat + separately-timed DHW.
- SAP 10.2 Appendix N3.7(a) (PDF p.6097) — HP HW in-use factor
  cylinder-criteria, footnote 53 (cert HX area unknown for Open EPC
  schema → criteria fail → 0.60 in-use factor; the worksheet's
  closed HW path uses this same factor).
- Cert 0380 worksheet `dr87-0001-000899.pdf` lodgings:
  (47) Cylinder Volume 160.00 L; "Cylinder Insulation Type Foam";
  "Cylinder Insulation Thickness 50 mm"; "Cylinder Stat Yes";
  (51)..(56) cylinder storage loss chain; (64) HW output 1502.16;
  (216) HW efficiency 171.0746%.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-06-01 16:28:46 +00:00
..
epc Slice S0380.6: surface full §15.1 Hot Water Cylinder block — Summary HW exact 2026-06-01 16:28:46 +00:00
magicplan typing and renaming 🟪 2026-05-07 13:26:49 +00:00
__init__.py further breaking up code 2023-07-20 12:24:34 +01:00
datatypes.py further breaking up code 2023-07-20 12:24:34 +01:00
enums.py completed build of new demo portfolio - some fixed required still 2023-11-20 14:42:16 +00:00