Skip to content

Effect Tracking & Bounding

Overview

Effects represent quantities that are tracked across the optimization horizon (e.g., cost, CO₂ emissions, primary energy). One effect is designated as the objective to minimize.

Effects are split into two domains based on how they vary over time and how they are weighted in multi-period optimization:

Domain Dims What goes here Multi-period weighting
Temporal (effect, time) Flow costs, running costs, startup costs — anything that varies per timestep Summed over time (× \(\mathrm{w}_t\)), then weighted by period_weights
Lump (effect,) Sizing costs, fixed O&M, one-time CAPEX — anything not per-timestep Weighted by \(\omega_{k,p}\) (defaults to global period_weights)

All domains support cross-effect chains via contribution_from.

In multi-period mode, all variables gain an optional period dimension. See Objective for how the domains are weighted in the objective.

Temporal Domain

Each effect accumulates contributions from all flows at each timestep:

\[ \Phi_{k,t}^{\text{temporal}} = \underbrace{\sum_{f \in \mathcal{F}} \mathrm{c}_{f,k,t} \cdot P_{f,t} \cdot \Delta t_t}_{\text{direct flow contributions}} + \underbrace{\sum_{j \in \mathcal{K}} \alpha_{k,j,t} \cdot \Phi_{j,t}^{\text{temporal}}}_{\text{cross-effect contributions}} \quad \forall \, k, t \]

The coefficient \(\mathrm{c}_{f,k,t}\) specifies how much of effect \(k\) is produced per flow-hour of flow \(f\) (e.g., €/MWh for cost, kg/MWh for emissions).

The cross-effect factor \(\alpha_{k,j,t}\) can be time-varying or constant (both via contribution_from). Because \(\Phi_{k,t}^{\text{temporal}}\) is a variable, the solver resolves multi-level chains (e.g., PE → CO₂ → cost) automatically.

Lump Domain

Sizing costs, fixed costs, and one-time costs (not time-varying) are accumulated per effect:

\[ \Phi_k^{\text{lump}} = \underbrace{\Phi_k^{\text{invest,direct}}}_{\text{direct sizing costs}} + \underbrace{\sum_{j \in \mathcal{K}} \alpha_{k,j} \cdot \Phi_j^{\text{lump}}}_{\text{cross-effect contributions}} \quad \forall \, k \]

where the direct investment term is:

\[ \Phi_k^{\text{invest,direct}} = \sum_{f} \gamma_{f,k} \cdot S_f + \sum_{f} \phi_{f,k} \cdot y_f + \sum_{s} \gamma_{s,k} \cdot S_s + \sum_{s} \phi_{s,k} \cdot y_s \]

Because \(\Phi_k^{\text{lump}}\) is a variable (not an expression), the solver resolves multi-level chains correctly: if PE has sizing costs and CO₂ depends on PE and cost depends on CO₂, the chain propagates through the lump domain just as it does through the temporal domain.

Cross-Effect Contributions

An effect can include a weighted fraction of another effect's value via contribution_from. This enables patterns like carbon pricing (CO₂ → cost) or transitive chains (PE → CO₂ → cost).

The factor \(\alpha_{k,j}\) from contribution_from accepts either a scalar or a Variate:

  • Scalar: applied identically to both temporal and lump domains.
  • Variate (time-varying): applied per-timestep in the temporal domain; for the lump domain the arithmetic mean over time is used (cf_temporal.mean('time') in model.py). A UserWarning is emitted when a time-varying factor is averaged for a non-trivial lump contribution.

If you need different cross-effect factors for the two domains, split into separate effects.

Transitive Chains

Contributions chain transitively. A PE → CO₂ → cost chain is modeled as:

Validation

  • No self-references: an effect cannot reference itself (\(\alpha_{k,k}\)).
  • No cycles: \(k \to j \to \cdots \to k\) is rejected at build time to prevent singular systems.

Total Aggregation

The total effect for each period \(p\) combines both domains:

\[ \Phi_{k,p} = \sum_{t \in \mathcal{T}} \Phi_{k,t,p}^{\text{temporal}} \cdot \mathrm{w}_t + \Phi_{k,p}^{\text{lump}} \quad \forall \, k \in \mathcal{K}, \; p \in \mathcal{P} \]

Weights \(\mathrm{w}_t\) allow scaling timesteps (e.g., a representative week scaled to a year). Single-period models drop the \(p\) index.

Bounds

Three levels of bound granularity, all per-effect:

Per-hour bounds (maximum_per_hour / minimum_per_hour) — see Per-Hour Bounds below.

Per-period bounds (maximum_per_period / minimum_per_period) — each period independently:

\[ \underline{\Phi}_k^{\text{per period}} \leq \Phi_{k,p} \leq \bar{\Phi}_k^{\text{per period}} \quad \forall \, p \]

Aggregate bounds (maximum / minimum) — weighted sum across all periods, where \(\omega_{k,p}\) is Effect.period_weights (falling back to global period_weights, then 1):

\[ \underline{\Phi}_k \leq \sum_{p \in \mathcal{P}} \omega_{k,p} \cdot \Phi_{k,p} \leq \bar{\Phi}_k \]

For physical quantities (e.g., total CO₂ across all years), period_weights should typically encode the period duration so the aggregate is a true physical sum.

This is useful for emission caps or budget constraints:

Per-Hour Bounds

The per-hour bounds are rates (e.g., kg/h, €/h) that scale with the timestep duration \(\Delta t_t\). This ensures the constraint is resolution-independent:

\[ \underline{\Phi}_{k,t}^{\text{per hour}} \cdot \Delta t_t \leq \Phi_{k,t}^{\text{temporal}} \leq \bar{\Phi}_{k,t}^{\text{per hour}} \cdot \Delta t_t \quad \forall \, k \in \mathcal{K}, \; t \in \mathcal{T} \]

For example, maximum_per_hour=100 (kg/h) with a 4-hour timestep allows up to 400 kg of emissions in that timestep:

Parameters

Symbol Description Reference
\(\Phi_{k,t(,p)}^{\text{temporal}}\) Per-timestep effect variable effect--temporal[effect, time(, period)]
\(\Phi_{k(,p)}^{\text{lump}}\) Lump effect variable (sizing + one-time costs) effect--lump[effect(, period)]
\(\Phi_{k(,p)}\) Total effect variable effect--total[effect(, period)]
\(\mathrm{c}_{f,k,t}\) Effect coefficient per flow-hour Flow.effects_per_flow_hour
\(\alpha_{k,j,t}\) Cross-effect contribution factor (time-varying) Effect.contribution_from (Variate)
\(\alpha_{k,j}\) Cross-effect contribution factor (scalar) Effect.contribution_from (scalar)
\(P_{f,t}\) Flow rate variable flow--rate[flow, time]
\(\Delta t_t\) Timestep duration dt
\(\mathrm{w}_t\) Timestep weight weights
\(\bar{\Phi}_k\) Maximum aggregate (weighted sum across periods) Effect.maximum
\(\underline{\Phi}_k\) Minimum aggregate (weighted sum across periods) Effect.minimum
\(\bar{\Phi}_k^{\text{per period}}\) Maximum per period Effect.maximum_per_period
\(\underline{\Phi}_k^{\text{per period}}\) Minimum per period Effect.minimum_per_period
\(\bar{\Phi}_{k,t}^{\text{per hour}}\) Maximum per hour (rate, scaled by \(\Delta t_t\)) Effect.maximum_per_hour
\(\underline{\Phi}_{k,t}^{\text{per hour}}\) Minimum per hour (rate, scaled by \(\Delta t_t\)) Effect.minimum_per_hour
\(\omega_{k,p}\) Period weight (per-effect, falls back to global, then 1) Effect.period_weights / global period_weights

See Notation for the full symbol table.

Examples

Direct effects

A system with two effects — cost (objective) and CO₂ (capped at 1000 kg):

A gas flow with both effect coefficients:

At timestep \(t\) with \(P_{\text{gas},t} = 5\) MW and \(\Delta t = 1\) h:

  • \(\Phi_{\text{cost},t} = 30 \times 5 \times 1 = 150\)
  • \(\Phi_{\text{CO₂},t} = 0.2 \times 5 \times 1 = 1.0\) kg

Carbon pricing via contribution_from

CO₂ priced at 50 €/t into the cost effect:

With \(\alpha_{\text{cost,co2}} = 50\), the per-timestep cost becomes:

\[ \Phi_{\text{cost},t} = \mathrm{c}_{\text{cost}} \cdot P_t \cdot \Delta t + 50 \cdot \Phi_{\text{co2},t} \]

The CO₂ total itself is not affected — contribution_from is one-directional.