From d9ecc7098075767fbc8df00bca914be6f4679f81 Mon Sep 17 00:00:00 2001 From: Dusan Vojacek Date: Thu, 21 May 2026 17:52:10 +0200 Subject: [PATCH] dalsi pokus o load first --- backend/services/planning_engine.py | 4 ++++ docs/04-modules/planning.md | 2 +- 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/backend/services/planning_engine.py b/backend/services/planning_engine.py index e88801b..56ebbc8 100644 --- a/backend/services/planning_engine.py +++ b/backend/services/planning_engine.py @@ -1031,6 +1031,9 @@ def solve_dispatch( prob += bc_pv[t] + ge_pv[t] <= pv_sp[t] # Import na deficit po PV→load, nebo na grid-nabíjení (bc_gi). prob += gi[t] <= load_site_expr + bc_gi[t] + # Vybíjení do domu až po pv_ld (Deye load-first); v exportních slotech smí bd→síť. + if t not in discharge_export_slots: + prob += bd[t] <= load_site_expr - pv_ld[t] # Plná bilance (pv_ld+pv_sp rozpad je ortogonální k tokům přebytku). prob += ( pv_a_net + pv_b_effective + gi[t] + bd[t] @@ -1479,6 +1482,7 @@ def solve_dispatch( "planner_daytime_charge_target_enabled": daytime_en, "planner_charge_commitment_penalty_czk_kwh": float(commit_pen), }, + "load_first_enabled": om == "AUTO", "charge_acquisition_buy_czk_kwh": charge_acquisition_czk_kwh, "charge_acquisition_cutoff_at": ( slots[0].charge_acquisition_cutoff_at.isoformat() diff --git a/docs/04-modules/planning.md b/docs/04-modules/planning.md index 905ba36..4b51f64 100644 --- a/docs/04-modules/planning.md +++ b/docs/04-modules/planning.md @@ -14,7 +14,7 @@ - **Grid ze sítě (vrstva B, před FVE):** spot, výchozí **AM/PM 50/50** z `grid_target × charge_slot_buffer` (do `soc_max`); **nevyčerpaný AM Wh přejde do PM** (`R__063`). Výběr: **nejlevnější `buy`** v pásmu (den plánu → před exportním oknem → `buy ASC`). Cap slotů: `ceil(budget/per_slot_wh) × charge_slot_buffer`. **`charge_acquisition`:** vážený `buy` u `allow_grid_charge` před 1. exportem; two-pass v `planning_engine.py`. - **PV vrstva A:** jen pokud `sell ≥ future_sell_opportunity − degradation` (držet FVE na večerní peak, ne „nabíjet z FVE“ při nízkém sell). - **LP (AUTO):** objective explicitně `−ge_pv×sell − ge_bat×sell + ge_bat×acquisition` v exportních slotech; **bez** cross-slot vynucení `ge_pv ≥ surplus`. Guard FVE: `ge_pv=0` jen pokud `sell < charge_acquisition − degrad` (ne `sell < buy` ve slotu). Viz [`planning-arbitrage-accounting.md`](planning-arbitrage-accounting.md). - - **Load-first (Deye, AUTO):** proměnné `pv_ld` (PV → load+EV+TČ), `pv_sp` (přebytek), `bc_pv` / `bc_gi` (nabíjení jen z `pv_sp` resp. `gi`). Bilance `pv_ld + gi + bd = load + ev + hp + bc_pv + bc_gi + ge`; `ge_bat ≥ ge − pv_sp`. Měkká preference `−LOAD_FIRST_INCENTIVE × pv_ld` v objective. Mimo `allow_charge`: `bc_gi=0`, `bc_pv ≤ pv_surplus` (forecast). Implementace: `solve_dispatch()` v `planning_engine.py`; test `LoadFirstDispatchTests`. + - **Load-first (Deye, AUTO):** proměnné `pv_ld` (PV → load+EV+TČ), `pv_sp` (přebytek), `bc_pv` / `bc_gi` (nabíjení jen z `pv_sp` resp. `gi`). Plná bilance `pv_a + pv_b + gi + bd = load + ev + hp + bc + ge`; `bc_pv + ge_pv ≤ pv_sp`; `gi ≤ load + bc_gi`; mimo exportní masku `bd ≤ load − pv_ld`. Snapshot: `solver_params.inputs.load_first_enabled=true`. Mimo `allow_charge`: `bc_gi=0`. Implementace: `planning_engine.py`; test `LoadFirstDispatchTests`. Po nasazení je nutný nový rolling/denní běh — starý plán v DB chování nemění. - **`ref_buy_min` (brána exportu):** `min(buy_price)` horizontu — jen „existuje levný nákup?“, **ne** průměrná cena nabití přes hodiny. Export sloty: `sell > ref_buy_min + degradation` (spot). Viz [`planning-arbitrage-accounting.md`](planning-arbitrage-accounting.md). - Pokud `energy_to_fill <= 0` nebo `charge_slot_buffer = 0`: všechny sloty povoleny. - **LP ekonomické guardy** (`solve_dispatch`, AUTO): `ge_pv=0` pokud `sell < charge_acquisition − degradation` (výjimka: plná baterie, přebytek **pv_b**). Pokud `buy > min(buy)+degradation` mimo charge masku → `gi` jen na load+EV+TČ. Viz `planning_engine.py` po slot pre-selection.