commissioned

This commit is contained in:
2026-06-23 18:49:47 -04:00
parent 231e03081d
commit 5484bb5fa6
4 changed files with 174 additions and 0 deletions

View File

@@ -12,6 +12,7 @@ Mirrors the `eg4battery/homeassistant/` pattern.
|----------------------------|--------------------------------------------------------------| |----------------------------|--------------------------------------------------------------|
| `mqtt_controls.yaml` | `configuration.yaml``mqtt: !include lvx6048/mqtt_controls.yaml` (or merge by hand) | | `mqtt_controls.yaml` | `configuration.yaml``mqtt: !include lvx6048/mqtt_controls.yaml` (or merge by hand) |
| `template_sensors.yaml` | `configuration.yaml``template: !include lvx6048/template_sensors.yaml` | | `template_sensors.yaml` | `configuration.yaml``template: !include lvx6048/template_sensors.yaml` |
| `daily_energy_package.yaml`| `configuration.yaml``homeassistant: packages: {lvx6048_daily_energy: !include lvx6048/daily_energy_package.yaml}` |
| `lovelace_controls.yaml` | Raw Lovelace card config — paste into a new dashboard view | | `lovelace_controls.yaml` | Raw Lovelace card config — paste into a new dashboard view |
The auto-discovery sensors (battery V, fault code, mode, MPPT power, …) The auto-discovery sensors (battery V, fault code, mode, MPPT power, …)
@@ -103,6 +104,24 @@ mosquitto_sub -h <broker> -u mqtt -P <pass> -v \
…then flip a select in the dashboard. Both inverters should publish …then flip a select in the dashboard. Both inverters should publish
`"Succeeded"` within ~1 s. `"Succeeded"` within ~1 s.
## Daily energy indicators (harvested / used today)
`daily_energy_package.yaml` adds two daily-resetting kWh sensors for a
dashboard tile:
- `sensor.solar_harvested_today` — PV generated today, summed across both
inverters. Sourced from each inverter's lifetime `total_pv_generated_energy`
Wh counter via a daily `utility_meter` (the LVX6048 firmware has no
daily-energy query, so we meter the lifetime counter instead).
- `sensor.solar_used_today` — household load consumed today. No cumulative
load counter exists, so `sensor.lvx6048_stack_ac_output_active_power` (W)
is Riemann-integrated into energy and metered daily.
Requires `template_sensors.yaml` (for the stack load-power rollup). A ready
two-tile Lovelace card is in the file's trailing comment. After enabling,
the two sensors read 0 at local midnight and climb through the day; give the
integration a few minutes to accumulate its first non-zero value.
## Energy / SoC dashboard wiring (optional) ## Energy / SoC dashboard wiring (optional)
Once both inverters' `ac_output_active_power` and the EG4 daemon's Once both inverters' `ac_output_active_power` and the EG4 daemon's

View File

@@ -0,0 +1,88 @@
# Today's solar harvested + household energy used — daily-resetting kWh
# sensors for a dashboard indicator.
#
# This is a Home Assistant *package* (bundles several domains in one file)
# so the whole feature installs as one unit.
#
# Install:
# 1. Copy to <config>/lvx6048/daily_energy_package.yaml
# 2. In configuration.yaml (once), pull it in as a package:
# homeassistant:
# packages:
# lvx6048_daily_energy: !include lvx6048/daily_energy_package.yaml
# 3. Requires template_sensors.yaml already loaded (provides
# sensor.lvx6048_stack_ac_output_active_power).
# 4. Restart HA.
#
# Produces two dashboard-ready entities:
# sensor.solar_harvested_today kWh — PV generated today (both inverters)
# sensor.solar_used_today kWh — household load consumed today
#
# Data sources & why:
# - Harvested: each inverter exposes a *lifetime* Wh counter
# (total_pv_generated_energy); the LVX6048 firmware has no daily-energy
# query (ED is NAKed). A daily utility_meter on the lifetime counter gives
# today's harvest from the inverter's own internal accumulator (accurate).
# - Used: no cumulative load counter exists, so the household load *power*
# (sensor.lvx6048_stack_ac_output_active_power, sum of both inverters'
# AC output from template_sensors.yaml) is Riemann-integrated into energy
# and metered daily. This is an estimate; accuracy tracks powermon's GS
# poll cadence. Counts only load served by the inverters.
#
# NOTE: verify the two source entity_ids match your HA install
# (Developer Tools -> States): sensor.lvx6048_1_total_pv_generated_energy
# and sensor.lvx6048_2_total_pv_generated_energy.
template:
- sensor:
- name: "Solar Harvested Today"
unique_id: solar_harvested_today
unit_of_measurement: "kWh"
device_class: energy
state_class: total_increasing
icon: mdi:solar-power
availability: >
{{ has_value('sensor.solar_pv_today_unit_1')
and has_value('sensor.solar_pv_today_unit_2') }}
state: >
{{ ((states('sensor.solar_pv_today_unit_1') | float(0))
+ (states('sensor.solar_pv_today_unit_2') | float(0))) / 1000 }}
# Riemann-sum integration: household load power (W) -> energy (kWh)
sensor:
- platform: integration
source: sensor.lvx6048_stack_ac_output_active_power
name: "Household Load Energy"
unique_id: household_load_energy_total
unit_prefix: k
round: 3
method: trapezoidal
# Daily-resetting meters (reset at local midnight)
utility_meter:
solar_pv_today_unit_1:
source: sensor.lvx6048_1_total_pv_generated_energy
cycle: daily
solar_pv_today_unit_2:
source: sensor.lvx6048_2_total_pv_generated_energy
cycle: daily
solar_used_today:
source: sensor.household_load_energy
cycle: daily
# ---------------------------------------------------------------------------
# Dashboard card (paste into a Lovelace view, or Raw config editor):
#
# type: horizontal-stack
# cards:
# - type: tile
# entity: sensor.solar_harvested_today
# name: Harvested Today
# icon: mdi:solar-power
# color: amber
# - type: tile
# entity: sensor.solar_used_today
# name: Used Today
# icon: mdi:home-lightning-bolt
# color: blue
# ---------------------------------------------------------------------------

Binary file not shown.

View File

@@ -102,6 +102,7 @@ class AppConfig:
packs: list[PackConfig] packs: list[PackConfig]
cell_count: int = 16 # active mode only cell_count: int = 16 # active mode only
expose_raw_registers: bool = False # publish register_NN entities (modbus_per_pack) expose_raw_registers: bool = False # publish register_NN entities (modbus_per_pack)
soc_estimator: bool = False # publish independent coulomb-count soc_estimated
def load_config(path: Path) -> AppConfig: def load_config(path: Path) -> AppConfig:
@@ -128,6 +129,7 @@ def load_config(path: Path) -> AppConfig:
packs=[PackConfig(**p) for p in raw["packs"]], packs=[PackConfig(**p) for p in raw["packs"]],
cell_count=raw.get("cell_count", 16), cell_count=raw.get("cell_count", 16),
expose_raw_registers=raw.get("expose_raw_registers", False), expose_raw_registers=raw.get("expose_raw_registers", False),
soc_estimator=raw.get("soc_estimator", False),
) )
@@ -733,6 +735,9 @@ _FIELD_META.update({
"model": (None, None, None, "mdi:battery-outline"), "model": (None, None, None, "mdi:battery-outline"),
"firmware_version": (None, None, None, "mdi:chip"), "firmware_version": (None, None, None, "mdi:chip"),
"firmware_date": (None, None, None, "mdi:calendar"), "firmware_date": (None, None, None, "mdi:calendar"),
# independent coulomb-count SoC estimator (opt-in; see estimate_soc)
"soc_estimated": ("%", "battery", "measurement", "mdi:battery-sync"),
"soc_est_anchor": (None, None, None, "mdi:anchor"),
}) })
def field_meta(key: str) -> tuple[str | None, str | None, str | None, str | None]: def field_meta(key: str) -> tuple[str | None, str | None, str | None, str | None]:
if key.startswith("register_"): if key.startswith("register_"):
@@ -759,6 +764,7 @@ _FIELD_PRECISION: dict[str, int] = {
"capacity_ah": 1, "capacity_ah": 1,
"remaining_ah": 1, "remaining_ah": 1,
"error_code": 0, "error_code": 0,
"soc_estimated": 1,
} }
for _i in range(1, 17): for _i in range(1, 17):
_FIELD_PRECISION[f"cell_{_i:02d}_voltage"] = 3 _FIELD_PRECISION[f"cell_{_i:02d}_voltage"] = 3
@@ -863,11 +869,70 @@ class _PackState:
consecutive_errors: int = 0 consecutive_errors: int = 0
response_count: int = 0 response_count: int = 0
first_seen_logged: bool = False first_seen_logged: bool = False
# independent SoC estimator (opt-in; see estimate_soc)
soc_est: float | None = None
last_est_ts: float | None = None
_FAIL_HEARTBEAT_CYCLES = 360 # re-log a stuck failure every ~hour at 10 s cadence _FAIL_HEARTBEAT_CYCLES = 360 # re-log a stuck failure every ~hour at 10 s cadence
# --- independent SoC estimator (coulomb count + voltage anchoring) ----------
# The BMS reg21 SoC tracks current well (validated 2026-06-15: reported dSoC
# matched the integral of pack_current to <1%/hr across all 6 packs) but only
# re-anchors at a full-charge termination the bank rarely reaches while it
# cycles mid-range — so the six counters free-run-drift apart (observed
# 29-60% at an identical 3.33 V/cell). This estimator integrates the same
# trusted current but adds the anchors the BMS lacks: snap to 100% at a
# detected full charge, snap to a low ref at the bottom knee. Published as a
# separate `soc_estimated` entity; it never replaces the raw `soc`. Pure
# function of (readings, prior per-pack state, now) so it can be replayed
# offline against captured current logs.
SOC_EST_CFG = dict(
capacity_ah=100.0, # LP4V2 nameplate (per-pack capacity_ah reg reads 100.0)
coulombic_eff=0.99, # charge-acceptance efficiency, applied to + current
v_full=3.450, # cell Vmax at/above this ...
i_taper=3.0, # ... while charge current has tapered below this (A) => full
soc_full=100.0,
v_empty=3.000, # cell Vmin at/below this (under load) ...
soc_empty=5.0, # ... => bottom anchor
max_gap_s=300.0, # ignore integration across gaps longer than this (restarts)
)
def estimate_soc(readings: dict[str, Any], st: "_PackState", now: float,
cfg: dict = SOC_EST_CFG) -> None:
"""Add `soc_estimated` + `soc_est_anchor` to `readings`, coulomb-counting
pack_current between cycles and re-anchoring at full / empty. No-op for the
cycle if pack_current is missing (can't integrate)."""
cur = readings.get("pack_current")
if cur is None:
return
vmax = readings.get("cell_voltage_max")
vmin = readings.get("cell_voltage_min")
if st.soc_est is None: # seed from the BMS on first sight
bms = readings.get("soc")
st.soc_est = float(bms) if bms is not None else 50.0
st.last_est_ts = now
dt = 0.0 if st.last_est_ts is None else min(now - st.last_est_ts, cfg["max_gap_s"])
st.last_est_ts = now
if dt > 0: # integrate charge (Ah) -> % of pack
eff = cfg["coulombic_eff"] if cur > 0 else 1.0
st.soc_est += (cur * eff * dt / 3600.0) / cfg["capacity_ah"] * 100.0
anchor = "coulomb" # anchors override the free-run integral
if vmax is not None and 0.0 <= cur < cfg["i_taper"] and vmax >= cfg["v_full"]:
st.soc_est, anchor = cfg["soc_full"], "full"
elif vmin is not None and vmin <= cfg["v_empty"]:
st.soc_est, anchor = cfg["soc_empty"], "empty"
st.soc_est = max(0.0, min(100.0, st.soc_est))
readings["soc_estimated"] = round(st.soc_est, 1)
readings["soc_est_anchor"] = anchor
def _resolve_pack_name(addr: int, packs: list[PackConfig]) -> str: def _resolve_pack_name(addr: int, packs: list[PackConfig]) -> str:
for p in packs: for p in packs:
if p.address == addr: if p.address == addr:
@@ -996,6 +1061,8 @@ def run_modbus_per_pack(cfg: AppConfig, publisher: MQTTPublisher,
raise RuntimeError(f"no poller configured for {p.name}") raise RuntimeError(f"no poller configured for {p.name}")
regs = pollers[p.name].poll() regs = pollers[p.name].poll()
readings = decode_eg4_modbus_regs(regs, expose_raw=cfg.expose_raw_registers) readings = decode_eg4_modbus_regs(regs, expose_raw=cfg.expose_raw_registers)
if cfg.soc_estimator:
estimate_soc(readings, st, time.monotonic())
publisher.publish_pack(p.name, readings) publisher.publish_pack(p.name, readings)
st.response_count += 1 st.response_count += 1
if not st.ok and st.consecutive_errors > 0: if not st.ok and st.consecutive_errors > 0: