Compare commits

..

3 Commits

Author SHA1 Message Date
znetsixe
5e2ebe4d96 fix(safety): overfill must keep pumps running, not shut them down
Two hard rules for the safety controller, matching sewer PS design:

1. BELOW stopLevel (dry-run): pumps CANNOT start.
   All downstream equipment shut down. safetyControllerActive=true
   blocks _controlLogic so level control can't restart pumps.
   Only manual override or emergency can change this.

2. ABOVE overflow level (overfill): pumps CANNOT stop.
   Only UPSTREAM equipment is shut down (stop more water coming in).
   Machine groups (downstream pumps) are NOT shut down — they must
   keep draining. safetyControllerActive is NOT set, so _controlLogic
   continues commanding pumps at the demand dictated by the level
   curve (which is >100% near overflow = all pumps at maximum).
   Only manual override or emergency stop can shut pumps during
   an overfill event.

Previously the overfill branch called turnOffAllMachines() on machine
groups AND set safetyControllerActive=true, which shut down the pumps
and blocked level control from restarting them — exactly backwards
for a sewer pumping station where the sewage keeps coming.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 14:10:23 +02:00
znetsixe
e8dd657b4f fix: continuous proportional control — eliminate dead zone between start/stop levels
Previously PS only sent demand to MGC when level > startLevel AND
direction === 'filling'. Between startLevel and stopLevel (the 'dead
zone'), pumps kept running at their last commanded setpoint with no
updates. Basin drained uncontrolled until hitting stopLevel.

Fix: send percControl on every tick when level > stopLevel. The
_scaleLevelToFlowPercent math naturally gives:
  - Positive % above startLevel (pumps ramp up)
  - 0% at exactly startLevel (pumps at minimum)
  - Negative % below startLevel → clamped to 0 → MGC scales to 0
    → pumps ramp down gracefully

This creates smooth visible ramp-up and ramp-down as the basin fills
and drains, instead of a sudden jump at startLevel and stuck ctrl in
the dead zone.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 13:42:43 +02:00
znetsixe
c62d8bc275 fix: deduplicate predicted-flow child registration + single event subscription
Three bugs in registerChild caused multi-counted outflow in _updatePredictedVolume:

1. machinegroup registered twice (line 66 + line 70 both called
   _registerPredictedFlowChild). Fixed: only register in the
   machinegroup branch.

2. Individual machines registered alongside their machinegroup parent.
   Each pump's predicted flow is already included in MGC's aggregated
   total — subscribing to both triple-counts. Fixed: only register
   individual machines when no machinegroup is present (direct-wired
   pumps without MGC).

3. _registerPredictedFlowChild subscribed to BOTH flow.predicted.downstream
   AND flow.predicted.atequipment events. These carry the same total flow
   on two event names — the handler wrote the value twice per tick.
   Fixed: subscribe to ONE event per child (downstream for outflow,
   upstream for inflow).

These are generalizable patterns:
- When a group aggregator exists, subscribe to IT, not its children.
- One event per measurement type per child — pick the most specific.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-14 13:10:16 +02:00

View File

@@ -63,10 +63,21 @@ class PumpingStation {
this.stations[child.config.general.id] = child; this.stations[child.config.general.id] = child;
} else if (softwareType === 'machinegroup') { } else if (softwareType === 'machinegroup') {
this.machineGroups[child.config.general.id] = child; this.machineGroups[child.config.general.id] = child;
this._registerPredictedFlowChild(child);
} }
if (softwareType === 'machine' || softwareType === 'pumpingstation' || softwareType === 'machinegroup') { // Register predicted-flow subscription. Only register the HIGHEST-
// level aggregator: if a machinegroup is present, subscribe to IT
// (its flow.predicted already aggregates all child machines). Do NOT
// also subscribe to individual machines — that would double-count
// because each pump's flow is included in the group total.
//
// Individual machines (softwareType='machine') are only subscribed
// when there is NO machinegroup parent — i.e., pumps wired directly
// to the pumping station without an MGC in between.
if (softwareType === 'machinegroup' || softwareType === 'pumpingstation') {
this._registerPredictedFlowChild(child);
} else if (softwareType === 'machine' && Object.keys(this.machineGroups).length === 0) {
// Direct-child machine, no group above it — register its flow.
this._registerPredictedFlowChild(child); this._registerPredictedFlowChild(child);
} }
} }
@@ -97,18 +108,21 @@ class PumpingStation {
const childId = child.config.general.id ?? childName; const childId = child.config.general.id ?? childName;
let posKey; let posKey;
let eventNames; let eventName;
switch (position) { switch (position) {
case 'downstream': case 'downstream':
case 'out': case 'out':
case 'atequipment': case 'atequipment':
posKey = 'out'; posKey = 'out';
eventNames = ['flow.predicted.downstream', 'flow.predicted.atequipment']; // Subscribe to ONE event only. 'downstream' is the most specific
// — avoids double-counting from 'atequipment' which carries the
// same total flow on a different event name.
eventName = 'flow.predicted.downstream';
break; break;
case 'upstream': case 'upstream':
case 'in': case 'in':
posKey = 'in'; posKey = 'in';
eventNames = ['flow.predicted.upstream', 'flow.predicted.atequipment']; eventName = 'flow.predicted.upstream';
break; break;
default: default:
this.logger.warn(`Unsupported predicted flow position "${position}" from ${childName}`); this.logger.warn(`Unsupported predicted flow position "${position}" from ${childName}`);
@@ -132,7 +146,7 @@ class PumpingStation {
.value(eventData.value, ts, unit); .value(eventData.value, ts, unit);
}; };
eventNames.forEach((ev) => child.measurements.emitter.on(ev, handler)); child.measurements.emitter.on(eventName, handler);
} }
/* --------------------------- Calibration --------------------------- */ /* --------------------------- Calibration --------------------------- */
@@ -250,15 +264,26 @@ class PumpingStation {
return; return;
} }
if (level > startLevel && direction === 'filling') { // Continuous proportional control: command pumps whenever level is
const percControl = this._scaleLevelToFlowPercent(level); // above stopLevel. The percControl ramp gives:
this.logger.debug(`Controllevel based => Level ${level} control applying to pump : ${percControl}`); // - 0% at minFlowLevel (= startLevel) → pumps barely running
await this._applyMachineLevelControl(percControl); // - linearly up to 100% at maxFlowLevel → all pumps full
// Also forward to machine groups (e.g. MGC) — the level-based // - Below startLevel but above stopLevel: percControl < 0 → clamp
// control originally only addressed direct-child machines, but in // to 0 → MGC turns off pumps (graceful ramp-down instead of a
// a hierarchical topology (PS → MGC → pumps) the machines sit // dead zone where pumps keep running at their last setpoint).
// under MGC and PS.machines is empty. if (level > stopLevel) {
await this._applyMachineGroupLevelControl(percControl); const rawPercControl = this._scaleLevelToFlowPercent(level);
const percControl = Math.max(0, rawPercControl);
this.logger.debug(`Controllevel based => Level ${level} percControl ${percControl}`);
if (percControl > 0) {
await this._applyMachineLevelControl(percControl);
await this._applyMachineGroupLevelControl(percControl);
} else {
// Between stopLevel and startLevel with percControl ≤ 0:
// tell MGC to scale back to 0 rather than leaving pumps
// running at the last commanded setpoint.
await this._applyMachineGroupLevelControl(0);
}
} }
if (level < stopLevel && direction === 'draining') { if (level < stopLevel && direction === 'draining') {
@@ -533,6 +558,24 @@ class PumpingStation {
/* --------------------------- Safety --------------------------- */ /* --------------------------- Safety --------------------------- */
/**
* Safety controller — two hard rules:
*
* 1. BELOW stopLevel (dry-run): pumps CANNOT start.
* Shuts down all downstream machines + machine groups.
* Only a manual override or emergency can restart them.
* safetyControllerActive = true → blocks _controlLogic.
*
* 2. ABOVE overflow level (overfill): pumps CANNOT stop.
* Shuts down UPSTREAM equipment only (stop more water coming in).
* Does NOT shut down downstream pumps or machine groups — they
* must keep draining. Does NOT set safetyControllerActive — the
* level-based control keeps running so pumps stay at the demand
* dictated by the current level (which will be >100% near overflow,
* meaning all pumps at maximum via the normal demand curve).
* Only a manual override or emergency stop can shut pumps during
* an overfill event.
*/
_safetyController(remainingTime, direction) { _safetyController(remainingTime, direction) {
this.safetyControllerActive = false; this.safetyControllerActive = false;
@@ -559,10 +602,12 @@ class PumpingStation {
const triggerHighVol = this.basin.maxVolOverflow * ((Number(overfillThresholdPercent) || 0) / 100); const triggerHighVol = this.basin.maxVolOverflow * ((Number(overfillThresholdPercent) || 0) / 100);
const triggerLowVol = this.basin.minVol * (1 + ((Number(dryRunThresholdPercent) || 0) / 100)); const triggerLowVol = this.basin.minVol * (1 + ((Number(dryRunThresholdPercent) || 0) / 100));
// Rule 1: DRY-RUN — below stopLevel, pumps cannot run.
if (direction === 'draining') { if (direction === 'draining') {
const timeTriggered = timeProtectionEnabled && remainingTime != null && remainingTime < timeleftToFullOrEmptyThresholdSeconds; const timeTriggered = timeProtectionEnabled && remainingTime != null && remainingTime < timeleftToFullOrEmptyThresholdSeconds;
const dryRunTriggered = dryRunEnabled && vol < triggerLowVol; const dryRunTriggered = dryRunEnabled && vol < triggerLowVol;
if (timeTriggered || dryRunTriggered) { if (timeTriggered || dryRunTriggered) {
// Shut down all downstream equipment — pumps must stop.
Object.values(this.machines).forEach((machine) => { Object.values(this.machines).forEach((machine) => {
const pos = machine?.config?.functionality?.positionVsParent; const pos = machine?.config?.functionality?.positionVsParent;
if ((pos === 'downstream' || pos === 'atequipment') && machine._isOperationalState()) { if ((pos === 'downstream' || pos === 'atequipment') && machine._isOperationalState()) {
@@ -572,28 +617,38 @@ class PumpingStation {
Object.values(this.stations).forEach((station) => station.handleInput('parent', 'execSequence', 'shutdown')); Object.values(this.stations).forEach((station) => station.handleInput('parent', 'execSequence', 'shutdown'));
Object.values(this.machineGroups).forEach((group) => group.turnOffAllMachines()); Object.values(this.machineGroups).forEach((group) => group.turnOffAllMachines());
this.logger.warn( this.logger.warn(
`Safe guard triggered: vol=${vol.toFixed(2)} m3, remainingTime=${remainingTime ? remainingTime.toFixed(1) : 'N/A'} s; shutting down downstream equipment` `Dry-run safety: vol=${vol.toFixed(2)} m3, remainingTime=${remainingTime ? remainingTime.toFixed(1) : 'N/A'} s; shutting down downstream equipment`
); );
// Block _controlLogic so level-based control can't restart pumps.
this.safetyControllerActive = true; this.safetyControllerActive = true;
} }
} }
// Rule 2: OVERFILL — above overflow level, pumps cannot stop.
// Only shut down UPSTREAM equipment. Downstream pumps + machine
// groups keep running at whatever the level control demands
// (which will be >100% near overflow = all pumps at max).
// Do NOT set safetyControllerActive — _controlLogic must keep
// running to maintain pump demand.
if (direction === 'filling') { if (direction === 'filling') {
const timeTriggered = timeProtectionEnabled && remainingTime != null && remainingTime < timeleftToFullOrEmptyThresholdSeconds; const timeTriggered = timeProtectionEnabled && remainingTime != null && remainingTime < timeleftToFullOrEmptyThresholdSeconds;
const overfillTriggered = overfillEnabled && vol > triggerHighVol; const overfillTriggered = overfillEnabled && vol > triggerHighVol;
if (timeTriggered || overfillTriggered) { if (timeTriggered || overfillTriggered) {
// Shut down UPSTREAM only — stop more water coming in.
Object.values(this.machines).forEach((machine) => { Object.values(this.machines).forEach((machine) => {
const pos = machine?.config?.functionality?.positionVsParent; const pos = machine?.config?.functionality?.positionVsParent;
if (pos === 'upstream' && machine._isOperationalState()) { if (pos === 'upstream' && machine._isOperationalState()) {
machine.handleInput('parent', 'execSequence', 'shutdown'); machine.handleInput('parent', 'execSequence', 'shutdown');
} }
}); });
Object.values(this.machineGroups).forEach((group) => group.turnOffAllMachines());
Object.values(this.stations).forEach((station) => station.handleInput('parent', 'execSequence', 'shutdown')); Object.values(this.stations).forEach((station) => station.handleInput('parent', 'execSequence', 'shutdown'));
// NOTE: machine groups (downstream pumps) are NOT shut down.
// They must keep draining to prevent overflow from worsening.
this.logger.warn( this.logger.warn(
`Safe guard triggered: vol=${vol.toFixed(2)} m3, remainingTime=${remainingTime ? remainingTime.toFixed(1) : 'N/A'} s; shutting down upstream equipment` `Overfill safety: vol=${vol.toFixed(2)} m3, remainingTime=${remainingTime ? remainingTime.toFixed(1) : 'N/A'} s; shutting down upstream equipment only — pumps keep running`
); );
this.safetyControllerActive = true; // NOTE: safetyControllerActive is NOT set — level control
// keeps commanding pumps at maximum demand.
} }
} }
} }