Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add broadcast_param_data config option and default it to False. #715

Merged
merged 1 commit into from
Nov 20, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,9 @@

### User-facing changes

|changed| Single data entries defined in YAML indexed parameters will not be automatically broadcast along indexed dimensions.
To achieve the same functionality as in `<v0.7.dev4`, the user must set the new `init` configuration option `broadcast_param_data` to True (#615).

|changed| Helper functions are now documented on their own page within the "Defining your own math" section of the documentation (#698).

|new| `where(array, condition)` math helper function to apply a where array _inside_ an expression, to enable extending component dimensions on-the-fly, and applying filtering to different components within the expression (#604, #679).
Expand Down
4 changes: 4 additions & 0 deletions docs/creating/config.md
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,10 @@ To test your model pipeline, `config.init.time_subset` is a good way to limit yo
Various capabilities are available to adjust the temporal resolution of a model on-the-fly, both by resampling or using externally-provided clustering.
See our [time adjustment page](../advanced/time.md) for more details.

!!! info "See also"
The full set of available configuration options is documented in the [configuration schema][model-configuration-schema].
This provides you with a description of each configuration option and the default which will be used if you do not provide a value.

## Deep-dive into some key configuration options

### `config.build.backend`
Expand Down
26 changes: 26 additions & 0 deletions docs/creating/parameters.md
Original file line number Diff line number Diff line change
Expand Up @@ -53,3 +53,29 @@ Which will add the new dimension `my_new_dim` to your model: `model.inputs.my_ne

!!! warning
The `parameter` section should not be used for large datasets (e.g., indexing over the time dimension) as it will have a high memory overhead on loading the data.

## Broadcasting data along indexed dimensions

If you want to set the same data for all index items, you can set the `init` [configuration option](config.md) `broadcast_param_data` to True and then use a single value in `data`:

=== "Without broadcasting"

```yaml
my_indexed_param:
data: [1, 1, 1, 1]
index: [my_index_val1, my_index_val2, my_index_val3, my_index_val4]
dims: my_new_dim
```

=== "With broadcasting"

```yaml
my_indexed_param:
data: 1 # All index items will take on this value
index: [my_index_val1, my_index_val2, my_index_val3, my_index_val4]
dims: my_new_dim
```

!!! warning
The danger of broadcasting is that you maybe update `index` as a scenario override without realising that the data will be broadcast over this new index.
E.g., if you start with `!#yaml {data: 1, index: monetary, dims: costs}` and update it with `!#yaml {index: [monetary, emissions]}` then the `data` value of `1` will be set for both `monetary` and `emissions` index values.
55 changes: 38 additions & 17 deletions docs/examples/national_scale/index.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,10 @@
---
costs:
file: "src/calliope/example_models/national_scale/data_tables/costs.csv"
header: [0, 1]
index_col: 0
---

# National Scale Example Model

This example consists of two possible power supply technologies,
Expand Down Expand Up @@ -25,6 +32,29 @@ It does not contain much data, but the scaffolding with which to construct and r

## Model definition

### Referencing tabular data

As of Calliope v0.7.0 it is possible to load tabular data completely separately from the YAML model definition.
To do this we reference data tables under the `data_tables` key:

```yaml
--8<-- "src/calliope/example_models/national_scale/model.yaml:data-tables"
```

In the Calliope national scale example model, we load both timeseries and cost data from file.
As an example, the data in the cost CSV file looks like this:

{{ read_csv(page.meta.costs.file, header=page.meta.costs.header, index_col=page.meta.costs.index_col) }}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If I look at the 'built' markdown file here on GitHub, the links to files and to the header do not work properly, but I assume this is because these files are built into documentation by Calliope in a different way than simple markdown visualisation. So perhaps someone else who knows how to build the docs locally can double-check that all is as expected


You'll notice that in each row there is reference to a technology, and in each column to a cost parameter and a comment on the units being used.
Therefore, we reference `techs` in our data table _rows_, and `parameters` and `comment` in our data table _columns_.
The `comment` information is metadata that we don't need in our Calliope model object, so we _drop_ it on loading the table.
Since all the data refers to the one cost class `monetary`, we don't add that information in the CSV file, but instead add it on as a dimension when loading the file.
Where there is no data for that combination of technology and cost parameter, the value is Not-a-Number (NaN) and this combination will be ignored on loading the table.

!!! info
You can read more about loading data from file in [our dedicated tutorial][loading-tabular-data].

### Indexed parameters

Before we dive into the technologies and nodes in the model, we have defined some parameters that are independent of both of these:
Expand Down Expand Up @@ -110,23 +140,6 @@ The costs are more numerous as well, and include monetary costs for all relevant
* carrier conversion capacity
* variable operational and maintenance costs

### Interlude: inheriting from templates

You will notice that the above technologies _inherit_ `cost_dim_setter`.
Templates allow us to avoid excessive repetition in our model definition.
In this case, `cost_dim_setter` defines the dimension and index of costs, allowing us to keep our definition of technology costs to only defining `data`.
By defining `data`, the technologies override the `null` setting applied by `cost_dim_setter`.
We also use it to set the `interest_rate` for all technologies, which will be used to annualise any investment costs each technology defines.

Technologies and nodes can inherit from anything defined in `templates`.
items in `templates` can also inherit from each other, so you can create inheritance chains.

`cost_dim_setter` looks like this:

```yaml
--8<-- "src/calliope/example_models/national_scale/model_config/techs.yaml:cost-dim-setter"
```

### Storage technologies

The second location allows a limited amount of battery storage to be deployed to better balance the system.
Expand Down Expand Up @@ -184,8 +197,16 @@ Transmission technologies look different to other technologies, as they link the
`free_transmission` allows local power transmission from any of the csp facilities to the nearest location.
As the name suggests, it applies no cost or efficiency losses to this transmission.

### Interlude: inheriting from templates

We can see that those technologies which rely on `free_transmission` inherit a lot of this information from elsewhere in the model definition.
`free_transmission` is defined in `templates`, which makes it inheritable.
[Templates](../../creating/templates.md) allow us to avoid excessive repetition in our model definition.

Technologies and nodes can inherit from anything defined in `templates`.
items in `templates` can also inherit from each other, so you can create inheritance chains.

The `free_transmission` template looks like this:

```yaml
--8<-- "src/calliope/example_models/national_scale/model_config/techs.yaml:free-transmission"
Expand Down
23 changes: 4 additions & 19 deletions docs/examples/urban_scale/index.md
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,8 @@ The import section in our file looks like this:
--8<-- "src/calliope/example_models/urban_scale/model.yaml:import"
```

## Model definition

### Referencing tabular data

As of Calliope v0.7.0 it is possible to load tabular data completely separately from the YAML model definition.
Expand All @@ -56,7 +58,7 @@ To do this we reference data tables under the `data_tables` key:
--8<-- "src/calliope/example_models/urban_scale/model.yaml:data-tables"
```

In the Calliope example models, we only load timeseries data from file, including for [energy demand](#demand-technologies), [electricity export price](#revenue-by-export) and [solar PV resource availability](#supply-technologies).
In the Calliope urban scale example model, we only load timeseries data from file, including for [energy demand](#demand-technologies), [electricity export price](#revenue-by-export) and [solar PV resource availability](#supply-technologies).
These are large tables of data that do not work well in YAML files!
As an example, the data in the energy demand CSV file looks like this:

Expand All @@ -69,8 +71,6 @@ Since all the data refers to the one parameter `sink_use_equals`, we don't add t
!!! info
You can read more about loading data from file in [our dedicated tutorial][loading-tabular-data].

## Model definition

### Indexed parameters

Before we dive into the technologies and nodes in the model, we have defined some parameters that are independent of both of these:
Expand Down Expand Up @@ -127,21 +127,6 @@ The definition of this technology in the example model's configuration looks as
--8<-- "src/calliope/example_models/urban_scale/model_config/techs.yaml:pv"
```

### Interlude: inheriting from templates

You will notice that the above technologies _inherit_ `interest_rate_setter`.
Templates allow us to avoid excessive repetition in our model definition.
In this case, `interest_rate_setter` defines an interest rate that will be used to annualise any investment costs the technology defines.

Technologies / nodes can inherit from anything defined in `templates`.
items in `templates` can also inherit from each other, so you can create inheritance chains.

`interest_rate_setter` looks like this:

```yaml
--8<-- "src/calliope/example_models/urban_scale/model_config/techs.yaml:interest-rate-setter"
```

### Conversion technologies

The example model defines two conversion technologies.
Expand Down Expand Up @@ -241,7 +226,7 @@ Gas is made available in each node without consideration of transmission.
--8<-- "src/calliope/example_models/urban_scale/model_config/techs.yaml:transmission"
```

To avoid excessive duplication in model definition, our transmission technologies inherit most of the their parameters from _templates_:
To avoid excessive duplication in model definition, our transmission technologies inherit most of the their parameters from [templates](../../creating/templates.md):

```yaml
--8<-- "src/calliope/example_models/urban_scale/model_config/techs.yaml:transmission-templates"
Expand Down
2 changes: 1 addition & 1 deletion docs/user_defined_math/syntax.md
Original file line number Diff line number Diff line change
Expand Up @@ -210,7 +210,7 @@ If you define a lookup parameter "lookup_techs" as:
```yaml
parameters:
lookup_techs:
data: True
data: [True, True]
index: [tech_1, tech_2]
dims: [techs]
```
Expand Down
7 changes: 7 additions & 0 deletions src/calliope/config/config_schema.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,13 @@ properties:
Unit of transmission link `distance` (m - metres, km - kilometres).
Automatically derived distances from lat/lon coordinates will be given in this unit.
enum: [m, km]
broadcast_param_data:
type: boolean
default: false
description:
If True, single data entries in YAML indexed parameters will be broadcast across all index items.
If False, the number of data entries in an indexed parameter needs to match the number of index items.
Defaults to False to mitigate unexpected broadcasting when applying overrides.

build:
type: object
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
parameters,cost_flow_cap,cost_storage_cap,cost_area_use,cost_source_cap,cost_flow_in,cost_flow_out
comment,USD per kW,USD per kWh storage capacity,USD per m2,USD per kW,USD per kWh,USD per kWh
ccgt,750,,,,0.02,
csp,1000,50,200,200,,0.002
battery,,200,,,,
region1_to_region2,200,,,,,0.002
16 changes: 15 additions & 1 deletion src/calliope/example_models/national_scale/model.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ config:
# What version of Calliope this model is intended for
calliope_version: 0.7.0
time_subset: ["2005-01-01", "2005-01-05"] # Subset of timesteps
broadcast_param_data: true # allow single indexed parameter data entries to be broadcast across all index items, if there are multiple entries.

build:
ensure_feasibility: true # Switches on the "unmet demand" constraint
Expand All @@ -27,15 +28,28 @@ config:
parameters:
objective_cost_weights:
data: 1
index: [monetary]
index: monetary
dims: costs
# `bigM` sets the scale of unmet demand, which cannot be too high, otherwise the optimisation will not converge
bigM: 1e6
cost_interest_rate:
data: 0.10
index: monetary
dims: costs
# --8<-- [end:parameters]

# --8<-- [start:data-tables]
data_tables:
time_varying_parameters:
data: data_tables/time_varying_params.csv
rows: timesteps
columns: [comment, nodes, techs, parameters]
drop: comment
cost_parameters:
data: data_tables/costs.csv
rows: techs
columns: [parameters, comment]
drop: comment
add_dims:
costs: monetary
# --8<-- [end:data-tables]
Original file line number Diff line number Diff line change
Expand Up @@ -4,40 +4,8 @@

# Note: --8<--start:'' and --8<--end:'' is used in tutorial documentation only

# --8<-- [start:cost-dim-setter]
templates:
cost_dim_setter:
cost_flow_cap:
data: null
index: monetary
dims: costs
cost_flow_in:
data: null
index: monetary
dims: costs
cost_flow_out:
data: null
index: monetary
dims: costs
cost_storage_cap:
data: null
index: monetary
dims: costs
cost_area_use:
data: null
index: monetary
dims: costs
cost_source_cap:
data: null
index: monetary
dims: costs
cost_interest_rate:
data: 0.10
index: monetary
dims: costs
# --8<-- [end:cost-dim-setter]

# --8<-- [start:free-transmission]
templates:
free_transmission:
name: "Local power transmission"
color: "#6783E3"
Expand All @@ -56,24 +24,19 @@ techs:
name: "Combined cycle gas turbine"
color: "#E37A72"
base_tech: supply
template: cost_dim_setter
carrier_out: power
flow_out_eff: 0.5
flow_cap_max: 40000 # kW
flow_cap_max_systemwide: 100000 # kW
flow_ramping: 0.8
lifetime: 25

cost_flow_cap.data: 750 # USD per kW
cost_flow_in.data: 0.02 # USD per kWh
# --8<-- [end:ccgt]

# --8<-- [start:csp]
csp:
name: "Concentrating solar power"
color: "#F9CF22"
base_tech: supply
template: cost_dim_setter
carrier_out: power
source_unit: per_area
include_storage: True
Expand All @@ -85,12 +48,6 @@ techs:
area_use_max: .inf
flow_cap_max: 10000
lifetime: 25

cost_storage_cap.data: 50
cost_area_use.data: 200
cost_source_cap.data: 200
cost_flow_cap.data: 1000
cost_flow_out.data: 0.002
# --8<-- [end:csp]

##
Expand All @@ -101,7 +58,6 @@ techs:
name: "Battery storage"
color: "#3B61E3"
base_tech: storage
template: cost_dim_setter
carrier_in: power
carrier_out: power
flow_cap_max: 1000 # kW
Expand All @@ -112,8 +68,6 @@ techs:
flow_in_eff: 0.95
storage_loss: 0 # No loss over time assumed
lifetime: 25

cost_storage_cap.data: 200 # USD per kWh storage capacity
# --8<-- [end:battery]

##
Expand All @@ -139,13 +93,10 @@ techs:
name: "AC power transmission"
color: "#8465A9"
base_tech: transmission
template: cost_dim_setter
carrier_in: power
carrier_out: power
flow_out_eff: 0.85
lifetime: 25
cost_flow_cap.data: 200
cost_flow_out.data: 0.002
flow_cap_max: 10000

region1_to_region1_1:
Expand Down
5 changes: 5 additions & 0 deletions src/calliope/example_models/urban_scale/model.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ config:
calliope_version: 0.7.0
# Time series data path - can either be a path relative to this file, or an absolute path
time_subset: ["2005-07-01", "2005-07-02"] # Subset of timesteps
broadcast_param_data: true # allow single indexed parameter data entries to be broadcast across all index items, if there are multiple entries.

build:
mode: plan # Choices: plan, operate
Expand All @@ -31,6 +32,10 @@ parameters:
dims: costs
# `bigM` sets the scale of unmet demand, which cannot be too high, otherwise the optimisation will not converge
bigM: 1e6
cost_interest_rate:
data: 0.10
index: monetary
dims: costs
# --8<-- [end:parameters]

# --8<-- [start:data-tables]
Expand Down
Loading