Skip to content

predicer-tools/Predicer

Repository files navigation

documentation

Predicer

‘Predictive decider’ for actors making decisions over multiple stages

If you use Predicer in a public work, please cite the following document

Pursiheimo, E., Sundell, D., Kiviluoma, J., & Hankimaa, H. (2023). Predicer: abstract stochastic optimisation model framework for multi-market operation. Optimization and Engineering, 1-30.

How to install

  • Clone the project to your computer using git.

  • Make sure that the main branch is activated.

  • Open a julia REPL. It is recommended to use a IDE such as VSCode or Atom, as the results from the model are currently not saved in a separate file.

  • Navigate to .\Predicer using the cd command.

      julia> cd("./Predicer)
    
  • type ] to open the package manager in julia, and type activate . to activate the local Julia environment.

      (Predicer) pkg> activate .
    
  • Press backspace to exit the package manager

  • Type using Pkg followed by Pkg.instantiate() to install the required packages. This can take a while.

      julia> using Pkg
      julia> Pkg.instantiate()
    
  • Finally type using Predicer to use the package.

How to use

  • Navigate to the local folder containing the Predicer package.

  • type ] to open the package manager in julia, and type activate . to activate the local Julia environment.

      (Predicer) pkg> activate .
    
  • Press backspace to exit the package manager

  • Type using Predicer to use the package.

      julia> using Predicer
    
  • To generate a model based on a input data file (in the example an Excel file located under Predicer/input_data/) use the Predicer.generate_model(fpath) function, where the parameter 'fpath' is the path to the input data file. The 'generate_model()' function imports the input data from the defined location, and build a model around it. The function returns two values, a "model contents" (mc) dictionary containing the built optimization model, as well as used expressions, indices and constraints for debugging. The other return value is the input data on which the optimization model is built on.

      julia> mc, input_data = Predicer.generate_model(fpath)
    
  • Or if using the example input data file Predicer/input_data/input_data.xlsx

      julia> mc, input_data = Predicer.generate_model("input_data/input_data.xlsx")
    
  • Predicer.solve_model(mc) optimizes the model, and shows the solver output.

      julia> Predicer.solve_model(mc)
    
  • After the model has been successfully optimized, the results of the variables in the model can be obtained using the Predicer.get_result_dataframe() or Predicer.get_all_result_dataframes() functions. Predicer.get_all_result_dataframes returns a dictionary with the variable names as keys, and Dataframes containing the results for those keys as values.

      julia> result_dataframes = Predicer.get_all_result_dataframes(mc, input_data)
      julia> Predicer.get_all_result_dataframes(mc, input_data)
              Dict{Any, Any} with 21 entries:
              "vq_state_dw"      => 24×13 DataFrame…
              "v_set_up"         => 24×1 DataFrame… 
              "v_flow_bal"       => 24×7 DataFrame… 
              "v_bid"            => 24×4 DataFrame… 
              "v_node_delay"     => 24×1 DataFrame… 
              "v_block"          => 0×0 DataFrame   
              "v_res_final"      => 24×10 DataFrame…
              "v_set_down"       => 24×1 DataFrame… 
              "vq_ramp_dw"       => 24×22 DataFrame…
              "v_start"          => 24×4 DataFrame… 
              "vq_state_up"      => 24×13 DataFrame…
              "vq_ramp_up"       => 24×22 DataFrame…
              "v_setpoint"       => 24×1 DataFrame… 
              "v_node_diffusion" => 24×1 DataFrame… 
              "v_online"         => 24×4 DataFrame… 
              "v_stop"           => 24×4 DataFrame… 
              "v_reserve"        => 24×37 DataFrame…
              "v_flow"           => 24×37 DataFrame…
              "v_load"           => 24×10 DataFrame…
              "v_reserve_online" => 24×7 DataFrame… 
              "v_state"          => 24×7 DataFrame… 
    
  • For more specific analysis, Predicer.get_result_dataframe() returns a DataFrame for a specific variable type, with the option to specify the node or process as well as the scenario. Below an example where the values for the v_flow variable for the hp1 process is obtained. The column names show which flow the value is for, with the notation being processname _ from node _ to node _ scenario; hp1_elc_hp1_s1 is for the electricity consumption of the heat pump process, and hp1_hp1_dh_s1 is for the heat production to the district heating node (dh). The types of the available variables can be found in the function documentation or in the example above.

      julia> Predicer.get_result_dataframe(mc, input_data, "v_flow", "hp1",  "s1")
              24×3 DataFrame
              Row │ t                          hp1_elc_hp1_s1  hp1_hp1_dh_s1 
              │ String                     Float64         Float64       
              ─────┼──────────────────────────────────────────────────────────
              1 │ 2022-04-20T00:00:00+00:00       0.5              1.5
              2 │ 2022-04-20T01:00:00+00:00       0.0              0.0
              3 │ 2022-04-20T02:00:00+00:00       0.0              0.0
              ⋮  │             ⋮                    ⋮               ⋮
              22 │ 2022-04-20T21:00:00+00:00       0.970055         3.39519
              23 │ 2022-04-20T22:00:00+00:00       1.42857          5.0
              24 │ 2022-04-20T23:00:00+00:00       1.71429          6.0
    
  • The modelled costs can be retrieved using the get_costs_dataframe() function. This includes realised costs, controlling costs and dummy costs. The realised costs include costs of used fuels and commodities, operational costs, market costs/profits, etc. The controlling costs include the value of the storage at the end of the optimization horizon, deviation from setpoints, etc. Dummy costs include costs for dummy or slack variables, which are used to ensure feasibility during optimization.

      julia> costs_df = Predicer.get_costs_dataframe(mc, input_data)
    
  • The DataFrames obtained from the results can be exported to an xlsx-file using the dfs_to_xlsx() function. The Dataframes can be passed to the function either in a dictionary with the name of the dataframe as the keys, or as a single DataFrame. The other parameters passed to the function are output_path, which is the path to where the file is to be saved, and fname is the name of the exported file. An suffix with the date is automatically added to the end of the filename.

      julia> Predicer.dfs_to_xlsx(df, output_path, fname)
    
  • The resulting bid matrix can be exported to a .xlsx file under Predicer/results by using the Predicer.write_bid_matrix() function

      julia> Predicer.write_bid_matrix(mc, input_data)
    

Example model

A simple example model simple, imaginary energy system is documented here. The input data file of the example model can be found under /input_data/example_model.xlsx. The system contains natural gas, electricity and heat, and electricity and electricity reserve products are sold on an external market m. The example model contains two possible scenarios for input data, s1 and s2, with equal probabilities of occurring (0.5). The data used for prices, heat demand, capacity factors, etc. is randomly generated.

Nodes

The modelled system consists of three main nodes: ng, symbolizing a node containing natural gas, elc symbolizing a node containing electricity, and heat, a node containing heat, such as a district heating system. The ng node is a commodity node, from which natural gas can be bought at a price defined in the price sheet. The heat node has a negative inflow added to it, which can be seen as a heat demand in the system. The elc node is connected to a market node m, npe, from which electricity can be bought or sold.

Processes

Processes are used to convert or transfer energy between nodes in the modelled system. The process windturb symbolizes a wind turbine producing electricity to the elc node. The production of windturb is limited by a capacity factor time series, defined in the cf sheet. The process ngchp converts natural gas from the ng node into electricity to the elc node and heat to the heat node at a fixed ratio, which is defined in the constraint and gen_constraint sheets. The heatpump unit convert electricity from the elc node into heat in the heat node. When the market m is defined, a process for trading between elc and m is automatically generated.

The constraint to define the operation of the ngchp process is setup in the constraints and gen_constraint sheets. The flat efficiency of the ngchp process is set to 0.9, including both heat and power. The maximum capacity of the natural gas input is set to 10, the heat is 6, and the electricity is 3. ngchp should produce electricity and heat at a 1:2 ratio. To achieve this, constraint called ngchp_c1 is defined in the sheet constraints.

The reserve products res_up and res_down are sold in the elc node. The processes ngchp and heatpump are used to offer reserve capacity.

alt_text Basic structure of the example model.

name operator is_setpoint penalty
ngchp_c1 eq 0 0

Further, the factors for the constraint are defined in the gen_constraint sheet. The operator of the created constraint is eq, meaning equal. The sum of the factors (the process branches elc and heat multiplied by given constants) and a given constant should equal 0. With a 1:2 ratio of electricity and heat production, the constants given to the process flows should be -2 and 1, or 2 and -1, for electricity and heat respectively. This can be seen in the table below, where the factors and constraints are defined for s1. The factors have to be defined again for s2.

t ngchp_c1,ngchp,elc,s1 ngchp_c1,ngchp,heat,s1 ngchp_c1,s1
t1 -2 1 0
t2 -2 1 0
tn -2 1 0

Input data description

The basic parameters and usage of the excel-format input data are described here. The input data has to be given to Predicer in a specific form, and the excel-format input data files are not a requirement. Excel has been used during development, since they were considered more convenient than databases or other forms of data structures.

setup

The setup sheet is used to define specific parameters used to define the functionalities of the model. The parameters defined in this sheet often override other information, for example if use_reserves is set to 0 (false), no reserve functionalities will be used in the model even if relevant structures are defined in other parts of the input data. This enables quick testing with or without certain features without needing to change large amounts of data or needing to use separate input data files.

Parameter Type Description
use_market_bids Bool Flag indicating whether market bids are used in the model
use_reserves Bool Flag indicating whether reserves are used in the model. If set to false, no reserve functionalities are used.
use_reserve_realisation Bool Indicates whether reserves can be realised. If set to false, no realisation occurs.
use_node_dummy_variables Bool Indicates if dummy variables should be used in the node balance equations.
use_ramp_dummy_variables Bool Indicates if dummy variables should be used in the ramp balance equations.
common_timesteps Int Indicates the length of a common start, where the parameters and variable values are equal across all scenarios. Default is 0.
common_scenario_name String Name of the common start scenario, if it is used.

nodes

Nodes are fundamental building blocks in Predicer, along with Processes.

Parameter Type Description
node String Name of the node
is_commodity Bool Indicates if the node is a commodity node
is_state Bool Indicates if the node has a state (storage)
is_res Bool Indicates if the node is involved in reserve markets
is_market Bool Indicates if the node is a market node
is_inflow Bool Indicates if the node has an inflow
state_max Float Storage state capacity (if node has a state)
state_min Float Storage state minimum level (if node has a state)
in_max Float Storage state charge capacity
out_max Float Storage state discharge capacity
initial_state Float Initial state of the storage
state_loss_proportional Float Hourly storage loss relative to the state of the storage
scenario_independent_state Bool If true, forces the state variable to be equal in all scenarios
is_temp Bool Flag indicating whether the state of the node models temperature
T_E_conversion Float Conversion coefficient from temperature to energy (kWh/K)
residual_value Float Value of the storage contents at the end of the time range

processes

Processes are fundamental building blocks in Predicer, along with Nodes. They are used to convert or transfer electricity or heat, or etc. between nodes in the modelled system.

Parameter Type Description
process String Name of the process
is_cf Bool Indicates if the process is limited by a capacity factor time series
is_cf_fix Bool Indicates if the process has to match the capacity factor time series
is_online Bool Indicates if the process is an online/offline unit
is_res Bool Indicates if the process participates in reserve markets
conversion Integer Indicates the type of the process. 1 = unit based process, 2 = transfer process, 3 = market process
eff Float Process efficiency (total output / total input)
load_min Float Minimum load of the process as a fraction of total capacity. Only for online processes
load_max Float Maximum load of the process as a fraction of total capacity. Only for online processes
start_cost Float Cost of starting the unit, only for online processes.
min_online Float Minimum time the process has to be online after start up
min_offline Float Minimum time the process has to be offline during shut down
max_online Float Maximum time the process can be online
max_offline Float Maximum time the process can be offline
scenario_independent_online Bool if true, forces the online variable of the process to be equal in all scenarios
initial_state Bool Initial state of the online unit (0 = offline, 1 = online)

groups

The user can define groups of either nodes or processes. Groups are used to define which processes or which nodes can for example participate in the realisation of a reserve. Group membership is defined row by row, with the first column being the type of the group, which is either node or process. The second column is the name of the entit (either a node or a process) which is to be part of the group named in the third column. A group can only contain entities of the specified type, for example adding the node "ng" to a process group will result in an error. Entities can be a part of several groups, and there is no limitation to the number of member sin a group.

Parameter Type Description
type String Type of the group (node/process)
entity String The name of the node or process which is to be a part of a group
group String The name of the group

node_diffusion

Node diffusion is used to model flow of energy between nodes with states, with the size of the flow depending on the level of the node state and the given diffusion coefficient. One possible application of this could be the flow of heat from the inside of a building to the outside during heating season. The flow of energy between nodes N1 and N2 is simply calculated as E = k (T1 - T2), with T1 and T2 being the temperatures of the nodes. If the temperature difference is negative (T2 > T1), the flow of energy goes from N2 to N1, and if (T1 > T2), the energy flows from N1 to N2. The temperatures of the nodes are linked to the level of the node states, either directly if the is_temp flag for the state is 1, or alternatively using the T_E_converison coefficient if the state is modelled as energy. The data for the node diffusion functionality is provided on the node_diffusion sheet in the example excel file in a timeseries format.

Parameter Type Description
node1 String The node from which heat flows if the temperature difference T1 - T2 is positive
node2 String The node to which heat flows if the temperature difference T1 - T2 is positive
diff_coeff Float The diffusion coefficient between the nodes. Must be positive.

node_history

When modelling delay flows between nodes in Predicer, there is a flow from one node n1 at timestep t to another node n2 at timestep t+d, where d is the length of the delay. The node_history sheet is used to define flows into node n2 for the d first timesteps of the optimization horizon, when the model cannot determine the flow from node n1, since it is outside of the optimization horizon. Node history can be seen as an inflow into a node for d timesteps.

The node history data should be provided for all nodes that are on the recieving end of a delay relation. The data for each node is given in two columns, one with the relevant timesteps, and the other with the size of the flow. The notation for the first column is of form nodename,t,scenario and the notation for the second column is of form nodename,scenario. Below is an example from the input_data_delays.xlsx model with a delay flow ending in the dh2 node. The length of the columns don't have to be equal for different nodes.

t dh2,t,s1 d2,s1
1 20.4.2022 0:00 3
2 20.4.2022 1:00 4
3

node_delay

Node delay is used to model a delay in the flow between two nodes, with a river hydropower system being a great example. When water is released from an upstream reservoir it takes a while until the water flow reaches a reservoir downstream. A delay flow is defined between two nodes with node balance, so commodity nodes or market nodes cannot be a part of a delay relation. The delay flow is one-way, with an efficiency of 1.

Parameter Type Description
node1 String Name of the from node
node2 String Name of the to node
delay_ Float Delay between the nodes in hours
min_flow Float Minimum allowed value for the flow
max_flow Float Maximum allowed value for the flow

process_topology

Process topologies are used to define the process flows and capacities in the modelled system. Flows are connections between nodes and processes, and are used to balance the modelled system.

Parameter Type Description
process String Name of the process
source_sink String Determines whether the connection node is a source or a sink for the process
node String Name of the connection node
capacity Float Capacity of the connection
vom_cost Float Variable operational and maintenance cost of using the corresponding process flow
ramp_up Float Determines the hourly upward ramp rate of the corresponding process flow
ramp_down Float Determines the hourly downward ramp rate of the corresponding process flow
initial_load Float Sets the initial value of the process load, from which optimization starts
initial_flow Float Sets the initial value of the process flow, from which optimization starts

efficiencies

Unit-based processes can have a flat efficiency, as defined in the processes sheet, or an efficiency which depends on the load of the process. Load-based efficiency can be defined in the sheet efficiencies. Defining an efficiency in the efficiencies sheet overrides the value given in the processes sheet. The efficiency of a process is defined on two rows; one row for the operating point, op, and one row for the corresponding efficiency, eff. In the example table below, the efficiency of an imaginary gas turbine gas_turb has been defined for four load intervals. The number of given operating points and corresponding efficiencies is chosen by the user, simply by adding or removing columns The operating points are defined on a row, where the first column has the value process,op, and the efficiencies are defined on a row where the value of the first column is process,eff.

process 1 2 3 4
gas_turb,op 0.4 0.6 0.8 1.0
gas_turb,eff 0.27 0.31 0.33 0.34

reserve_type

The sheet reserve_type is used to define the types of reserve used in the model, mainly differing based on reserve activation speed.

Parameter Type Description
type String Name of the reserve type
ramp_factor Float Ramp rate factor of reserve activation speed. (If reserve has to activate in 1 hour, ramp_factor is 1.0. In 15 minutes, ramp_factor is 4)

markets

Markets are a type of node, with which the modelled system can be balanced by buying or selling of a product such as electricity. Markets can either be of the energy type, or of the reserve type.

Parameter Type Description
market String Name of the market
type String type of the market (energy or reserve)
node String Node a market is connected to, or nodegroup a reserve market is ocnnected to.
processgroup String The processgroup the reserve market is connected to. Not used for energy markets.
direction String Direction of the market, only for reserve markets
reserve_type String Determines the type of the reserve
is_bid Bool Determines if bids can be offered to the market
is_limited Bool Determines if reserve markets are limited
min_bid Float Minimum reserve offer if limited
max_bid Float Maximum reserve offer if limited
fee Float Reserve participation fee (per time period) if limited

reserve_realisation

reserve realisation is defined for each defined reserve market separately for each defined scenario. The realisation of the reserve is the expected share of activation for the offered reserve capacity for each timestep. A value of 0.0 means that none of the offered reserve capacity is activated, and 1.0 means that 100% of the offered capacity is activated, and the corresponding energy produced by reserve processes defined by the user. The notation of the reserve_realisation sheet is as such: the reserve markets are given on rows 2:n in the first column, with the scenarios being defined on the first row from column B forward. The realisation probability is given in the intersection of the reserve markets and the scenarios. An example of the layout of the sheet is shown below. The model contains two reserve markets (res_up and res_down), and 3 scenarios (s1, s2 and s3).

The energy imbalance caused by reserve activation is allocated to the nodes that are members in the nodegroup linked to the reserve market, defined in the markets sheet. The necessary actions to produce the reserve are taken by the members of the processgroup defined in the markets sheet.

reserve_products s1 s2 s3
res_up 0.3 0.2 0.3
res_down 0.1 0.2 0.4

Time series data

Time series are used in Predicer to represent parameters that are time-dependent. The notation to define time series data in the excel input files depend on the time series data in question.

The sheet timeseries contains the timesteps used in the model. This sheet contains only one column t, with the given time steps.

Example

| t | | 20.4.2022 1:00 | | 20.4.2022 2:00 | | 20.4.2022 3:00 | | 20.4.2022 4:00 | | 20.4.2022 0:00 | | 20.4.2022 5:00 | | 20.4.2022 6:00 | | 20.4.2022 7:00 | | 20.4.2022 8:00 | | 20.4.2022 9:00 |

Time series notation in the excel-format input data

The first column, t, should contain the time steps for the time series data. The following columns (2-n) should contain the values corresponding to the given time steps. The name of the other columns should start with the name of the linked entity, usually followed by which scenario the value is for. The values can be defined for each scenario in separate columns, or a single column can be used for several scenarios, separated by commas. The notation used for the different time series is given in the table below.

As an example the inflow for the node nn can be given as nn,s1 if the values are given for scenario s1, and nn,s1,s2 if the given values should be for both s1 and s2. If all scenarios should have the same values, they can be defined as nn,ALL.

Sheet Description Notation
cf Capacity factor time series for processes with cf functionality process, scenario(s)
inflow Inflow time series for nodes (inflow positive value, demand negative value) node, scenario(s)
price Price time series for the cost of using commodity nodes node, scenario(s)
node_diffusion Node diffusion relation nodes and diffusion coefficient node1, node2, scenario(s)
market_prices Price time series for the defined markets market, scenario(s)
balance_prices Price time series for balance markets market, direction, scenario(s)
fixed_ts Value time series for setting market volumes to a fixed value market
eff_ts Value time series of the efficiency of processes process, scenariox(s)
cap_ts Value time series limiting a flow of a process Process, connected node, scenario

scenario

The scenarios in Predicer are separate versions of the future, with potentially differing parameter values. Predicer optimizes the optimal course of action, based on the probability of the defined scenarios.

Parameter Type Description
name String Name of the scenario
probability Float Probability of the scenario. The sum of all rows should be 1

risk

The risk sheet in the excel-format input data contains information about the CVaR (conditional value at risk). For details, see [1] and [2]

Risk parameter Description
alfa Risk quantile
beta Share of CVaR in objective function

inflow_blocks

Inflow blocks, or simply blocks, are potential flexibility which can be modelled with Predicer. A block has generally been thought of as "if a demand response action is taken on time t by reducing/increasing inflow to node n by amount x, how must the system compensate on times- t-1, t-2.. or t+1, t+2...", or "if the heating for a building is turned off on time t, what has to be done in the following hours to compensate?". The blocks can thus be seen as a potential for flexibility, and how the system has to be compensated as a consequence of using the potential.

Each block consists of a binary variable, consequent timesteps, and a constant value for each timestep. Each block is linked to a specific node, as well as a specific scenario. Despite being called "Inflow blocks", they can be linked to nodes without any inflow as well. Node inflow is modelled for each timestep and scenario as the given value in the inflow sheet. The product of the block binary variable value and the given constant is added to the inflow for relevant combinations of node, scenario and timestep. Two active blocks cannot overlap in the same node, time and scenario. The user can define any number of blocks for the same time, but only one can be active for a specific node, scenario and timestep.

Inflow blocks are defined in the inflow_blocks sheet. The first column of the sheet is named t, and is not used in the model itself. Each block is defined using one column for the timesteps and scenario columns for scenario based constant values. The name for the timestep column is of the form blockname, nodename and names for the scenario columns are of the form blockname, scenario. It is important, that the columns representing the data for one block have an equal amount of rows. The columns for different blocks can have different amount of rows.

As an example, assume there are two blocks, b1 and b2. The blocks should be defined to the inflow_blocks sheet as following:

t b1, n1 b1, s1 b1, s2 b2, n2 b2, s1 b2, s2
1 20.4.2022 1:00 6 4 20.4.2022 6:00 -3 -2
2 20.4.2022 2:00 -3 -2 20.4.2022 7:00 2 1
3 20.4.2022 3:00 -2 -1 20.4.2022 8:00 1 1
4 20.4.2022 4:00 -1 1

As with the generic constraints described below, the validity of the user input is not checked. The user should thus ensure, that the node linked to the block can handle the change in inflow, especially in nodes with either only consumers or producer. If a block causes a change in the sign of the inflow (- to +, or + to -), the results may be unpredictable. As an example, using a block causing a positive flow of heat into a district heating node without any way to remove the heat would result in high penalty costs, and the model would thus not use the block.

General constraints

General constraints in Predicer can be used to limit or fix process flow variables, online variables, or storage state variables in relation to other variables or a given value. The name and type of the constraint is defined in the sheet constraints, and the factors are defined in the sheet gen_constraint.

constraint

Parameter Type Description
name String Name of the general constraint
operator String The operator used in the general constraint. eq for =, gt for > and lt for <
is_setpoint Bool Indicates whether the constraint is fixed, or if the model can deviate from the value.
penalty Float A user-defined penalty for deviating from the givenn value, if constraint is a setpoint.

gen_constraint

The user can define additional constraints to the Predicer model, making it more flexible. Genral constraints are a powerful tool for customizing the model, but can also cause problems that are difficult to detect and solve. The user can either add "rigid" constraints, essentially defining a value or a range of values from which a variable or a sum of variables cannot deviate. Normal general constraints are applicable for flow, state and online variables.

The user can also define so-called "setpoint" constraints, in which the optimizer can deviate from the values defined by the user. Deviation from the values defined in the setpoint constraints adds and additional cost to the model. The setpoint constraints are only applicable for state and flow variables, not online variables.

The time series data in the sheet gen_constraint has a special notation. As with other time series data, the first column t in the sheet gen_constraint contains the time steps for the constraint. The rest of the columns (2-n) contain information corresponding to specific constraints, defined in the sheet constraint.

"rigid" general constraints contain factors, which add up to a constant. The notation for the first row of columns (2-n) depends on the type of variable the factor refers to. For process flow variables, v_flow, the notation is constraint name, process, process flow, scenario. For process online variables, v_online, the notation is constraint name, process, scenario. For storage state variables, v_state, the notation is constraint name, node, scenario. Each constraint can also have a constant value added, with the notation constraint name, scenario for the column. For "setpoint" constraints, only one column of factors can be defined, as per the previously mentioned notation.

As an example of a normal general constraint, assume a CHP gas turbine gt with a input flow of natural gas ng, as well as outputs of electricity elc and heat in the form of exhaust gases heat. Assume, that the ratio of heat and power should be 3:1, meaning an electrical efficiency of 25%, and the remaining 75% being heat. To ensure this ratio, a general constraint named gt_c1 is defined in the sheet constraint, with the operator set to eq.

name operator is_setpoint penalty
gt_c1 eq 0 0

In the sheet gen_constraint, the names of the columns referencing to the factors should be gt_c1,gt,heat,s1 and gt_c1,gt,elc,s1. This refers to the constraint gt_c1, the process gt, and the process flows heat and elc. The name of the column referencing to the constant should be gt_c1,s1. As the sum of the factors equal the constant, the value of the factors should be 3 or -3 for the factor representing the electricity flow, and -1 or 1, respectively, for the factor representing the heat flow. The value of the constant should be 0. As the sum of the factors equal the constant, it would lead to 3 * elc - 1 * heat = 0 or alternatively -3 * elc + 1 * heat = 0.

t gt_c1,gt,elc,s1 gt_c1,gt,heat,s1 gt_c1,s1
t1 3 -1 0
t2 3 -1 0
tn 3 -1 0

As another example, assume that the operation of two online processes, proc_1 and proc_2, should be limited in regard to eachother. Assume, that the constraint c_online is used to limit the online variables v_online of these processes so, that the processes are not operating at the same time, but that one of the two is always active. The variable v_online is a binary variable with values of 0 or 1. The sum of the online variables of proc_1 and proc_2 should be set to 1, in order to one, but not two, of the processes to be active. As the general constraint is set equal to 0, the addiotional constant should be set to -1. This ensures, that proc_1 + proc_2 - 1 == 0.

t c_online,proc_1,s1 c_online,proc_1,s1 c_online,s1
t1 1 1 -1
t2 1 1 -1
tn 1 1 -1

If both the processes should be either online or offline at the same time, the coefficients for one process should be 1, and the other should be -1, weith the constant set to 0. This would result in the constraint proc_1 - proc_2 + 0 == 0.

As an example of a setpoint general constraint, assume the value of the electricity, elc, production of the process gas_turb has to be between 3 and 8. To do this, two setpoint constraints are defined, c_up and c_dw, for defining an upper and lower boundary for the process flow. As above, the constraints are defined in the gen_constraint sheet. The operator for c_up should be st (= smaller than), and the operator for c_dw should be gt (=greater than). Both constraints are defined as setpoint constraints, with a deviation penalty of 100. The unit of the penalty is simply per unit of variable, meaning a variable deviation of 2 would increase the costs in the model with 200.

name operator is_setpoint penalty
c_up st 1 100
c_dw gt 1 100

In the constraint sheet, the values for the upper and lower boundaries are defined. The constraints c_up is the upper boundary, and a value of 8 is given. For c_dw, a value of 3 is defined. With these constraints, the electricity production of gas_turb has to be between 3 and 8. Any deviation from this range will result in a penalty of 100 for each unit of production exceeding this range.

t c_up,gas_turb,elc,s1 c_dw,gas_turb,elc,s1
t1 8 3
t2 8 3
tn 8 3

References

[1] Krokhmal, P., Uryasev, S., and Palmquist, J., “Portfolio optimization with conditional value-at-risk objective and constraints,” J. Risk, vol. 4, no. 2, pp. 43–68, 2001, doi: 10.21314/jor.2002.057.

[2] Fleten, S. E. and Kristoffersen, T. K., “Stochastic programming for optimizing bidding strategies of a Nordic hydropower producer,” Eur. J. Oper. Res., vol. 181, no. 2, pp. 916–928, 2007, doi: 10.1016/j.ejor.2006.08.023.

Funding

The development of Predicer has been partially funded in the EU Horizon ELEXIA, Business Finland HOPE and Academy of Finland EasyDR projects.

  • European Union ELEXIA: Demonstration of a digitized energy system integration across sectors enhancing flexibility and resilience towards efficient, sustainable, cost-optimised, affordable, secure and stable energy supply
  • Business Finland HOPE: Highly Optimized Energy systems
  • Academy of Finland EasyDR: Enabling demand response through easy to use open source approach

 


EU emblem This work has been partially supported by the EU project ELEXIA (2022-2026), which has received funding from the European Climate, Energy and Mobility programme under the European Union's HORIZON research and Innovation actions under grant N°101075656.

About

'Predictive decider' for actors making decisions over multiple stages

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages