9.2. Simulation settings

This section details the TOML input for MUSE. The format for TOML files is described in this previous section. Here, however, we focus on sections and attributes that are specific to MUSE.

The TOML file can be read using read_settings(). The resulting data is used to construct the market clearing algorithm directly in the MCA's factory function.

9.2.1. Main section

This is the topmost section. It contains settings relevant to the simulation as a whole.

time_framework = [2020, 2025, 2030, 2035, 2040, 2045, 2050]
foresight = 5
regions = ["USA"]
interest_rate = 0.1
interpolation_mode = 'Active'
log_level = 'info'

equilibrium_variable = 'demand'
maximum_iterations = 100
tolerance = 0.1
tolerance_unmet_demand = -0.1
time_framework

Required. List of years for which the simulation will run.

foresight

Required. Integer defining the number of years at which prices are updated at every iteration. It needs to equal or be a multiple of the interval of years in the time_framework. Default to 5 years.

region

Subset of regions to consider. If not given, defaults to all regions found in the simulation data.

interpolation_mode

interpolation when reading the initial market. One of linear, nearest, zero, slinear, quadratic, cubic. Defaults to linear.

  • linear interpolates using a linear function.

  • nearest uses nearest-neighbour interpolation. That is, the closest known data point is used as the prediction for the data point to interpolate.

  • zero assumes that the data point to interpolate is zero.

  • slinear uses a spline of order 1. This is a similar method to linear interpolation.

  • quadratic uses a quadratic equation for interpolation. This should be used if you expect your data to take a quadratic form.

  • cubic interpolation is similar to quadratic interpolation, but uses a cubic function for interpolation.

log_level

verbosity of the output. Valid options, from the highest to the lowest level of verbosity, are: “DEBUG”, “INFO”, “WARNING”, “ERROR”, “CRITICAL”.

equilibirum_variable

whether equilibrium of demand or prices should be sought. Defaults to demand.

maximum_iterations

Maximum number of iterations when searching for equilibrium. Defaults to 3.

tolerance

Tolerance criteria when checking for equilibrium. Defaults to 0.1. 0.1 signifies that 10% of a deviation is allowed among the iterative value of either demand or price over a year per region.

tolerance_unmet_demand

Criteria checking whether the demand has been met. Defaults to -0.1.

excluded_commodities

List of commodities excluded from the equilibrium considerations.

plugins

Path or list of paths to extra python plugins, i.e. files with registered functions such as register_output_quantity().

9.2.2. Carbon market

This section contains the settings related to the modelling of the carbon market. If omitted, it defaults to not including the carbon market in the simulation.

Example

[carbon_budget_control]
budget = []
budget

Yearly budget. There should be one item for each year the simulation will run. In other words, if given and not empty, this is a list with the same length as time_framework from the main section. If not given or an empty list, then the carbon market feature is disabled. Defaults to an empty list.

method

Method used to equilibrate the carbon market. Available options are fitting and bisection, however this can be expanded with the @register_carbon_budget_method hook in muse.carbon_budget.

The market-clearing algorithm iterates over the sectors until the market reaches an equilibrium in the foresight period (the period next to the one analysed). This is represented by a stable variation of a commodity demand (or price) between iterations below a defined tolerance. The market-clearing algorithm samples a user-defined set of carbon prices.

When the fitting method is selected, this command builds a regression model of the emissions as a function of the carbon price. It applies to a pool of emissions for all the modelled regions. Therefore, the estimated carbon price applies to all the modelled regions. The regression model, the method calculates iteratively the emissions at pre-defined carbon price sample values. The emissions-carbon price couples are used to used to fit the emission-carbon price relation, is uer-defined (ie. linear or exponential fitter). The new carbon price is estimated as a root of the regression model estimated at the value of the emission equal to the user-defined emission cap in the foresight period. Alongside the selection of the method, the user can define a sample_size, representing the magnitude of the sample for the fitter.

When the bisection method is selected, this command applies a bisection method to solve the carbon market. Similarly to the fitting method, the carbon market includes a pool of all the modelled regions. The obtained carbon price applies to all the regions, as above. This method solves as a typical bisection algorithm. It is coded independently to use the internal signature of the register_carbon_budget_method. The algorithm aims to find a root of the function emissions-carbon price, as for the carbon price at which the carbon budget is met. The algorithm iteratively modifies the carbon price and estimates the corresponding emissions. It stops when the convergence or stop criteria are met. This happens for example either when the carbon budget or the maximum number of iterations are met. Alongside the selection of the method, the user can define a sample_size, representing the number of iterations of the bisection method.

commodities

Commodities that make up the carbon market. Defaults to an empty list.

control_undershoot

Whether to control carbon budget undershoots. This parameter allows for carbon tax credit from one year to be passed to the next in the case of less carbon being emitted than the budget. Defaults to True.

control_overshoot

Whether to control carbon budget overshoots. If the amount of carbon emitted is above the carbon budget, this parameter specifies whether this deficit is carried over to the next year. Defaults to True.

method_options

Additional options for the specific carbon method. In particular, the refine_price activate a sanity check on the adjusted carbon price. The sanity check applies an upper limit on the carbon price obtained from the algorithm (either fitting or bisection), called price_too_high_threshold, a user-defined threshold based on heuristics on the values of the carbon price, reflecting typical historical trends.

fitter

fitter specifies the regression model fit. The regression approximates the model emissions. Predefined options are linear and exponential. Further options can be defined using the @register_carbon_budget_fitter hook in muse.carbon_budget.

9.2.3. Global input files

Defines the paths specific simulation data files. The paths can be formatted as explained in the TOML primer.

[global_input_files]
projections = '{path}/inputs/Projections.csv'
regions = '{path}/inputs/Regions.csv'
global_commodities = '{path}/inputs/MUSEGlobalCommodities.csv'
projections

Path to a csv file giving initial market projection. See Initial Market Projection.

global_commodities

Path to a csv file describing the comodities in the simulation. See Commodity Description.

base_year_import

Optional. Path to a file describing fixed amounts of commodities imported by region. A CSV file containing the consumption in the same format as Initial Market Projection.

base_year_export

Optional. Path to a file describing fixed amounts of commodities imported. A CSV file containing the consumption in the same format as Initial Market Projection.

9.2.4. Timeslices

Time-slices represent a sub-year disaggregation of commodity demand. Generally, timeslices are expected to introduce several levels, e.g. season, day, or hour. The simplest is to show the TOML for the default timeslice:

[timeslices]
winter.weekday.night = 396
winter.weekday.morning = 396
winter.weekday.afternoon = 264
winter.weekday.early-peak = 66
winter.weekday.late-peak = 66
winter.weekday.evening = 396
winter.weekend.night = 156
winter.weekend.morning = 156
winter.weekend.afternoon = 156
winter.weekend.evening = 156
spring-autumn.weekday.night = 792
spring-autumn.weekday.morning = 792
spring-autumn.weekday.afternoon = 528
spring-autumn.weekday.early-peak = 132
spring-autumn.weekday.late-peak = 132
spring-autumn.weekday.evening = 792
spring-autumn.weekend.night = 300
spring-autumn.weekend.morning = 300
spring-autumn.weekend.afternoon = 300
spring-autumn.weekend.evening = 300
summer.weekday.night = 396
summer.weekday.morning  = 396
summer.weekday.afternoon = 264
summer.weekday.early-peak = 66
summer.weekday.late-peak = 66
summer.weekday.evening = 396
summer.weekend.night = 150
summer.weekend.morning = 150
summer.weekend.afternoon = 150
summer.weekend.evening = 150
level_names = ["month", "day", "hour"]

This input introduces three levels, via level_names: month, day, hours. Other simulations may want fewer or more levels. The month level is split into three points of data, winter, spring-autumn, summer. Then day splits out weekdays from weekends, and so on. Each line indicates the number of hours for the relevant slice. It should be noted that the slices are not a cartesian products of each levels. For instance, there no peak periods during weekends. All that matters is that the relative weights (i.e. the number of hours) are consistent and sum up to a year.

The input above defines the finest times slice in the code. In order to define rougher timeslices we can introduce items in each levels that represent aggregates at that level. By default, we have the following:

[timeslices.aggregates]
all-day = ["night", "morning", "afternoon", "early-peak", "late-peak", "evening"]
all-week = ["weekday", "weekend"]
all-year = ["winter", "summer", "spring-autumn"]

Here, all-day aggregates the full day. However, one could potentially create aggregates such as:

   [timeslices.aggregates]
   daylight = ["morning", "afternoon", "early-peak", "late-peak"]
   nightlife = ["evening", "night"]

It is possible to specify a timeslice level for the mca by adding an

mca.timeslice_levels section, using an inline table format. See section on Timeslices_.

outputs_cache

This option behaves exactly like outputs for sectors and accepts the same options but controls the output of cached quantities instead. This option is NOT available for sectors themselves (i.e using [[sector.commercial.outputs_cache]] will have no effect). See muse.outputs.cache for more details.

A single row looks like this:

[[outputs_cache]]
quantity = "production"
sink = "aggregate"
filename = "{cwd}/{default_output_dir}/Cache{Quantity}.csv"
index = false

9.2.5. Standard sectors

A MUSE model requires at least one sector. Sectors are declared in the TOML file by adding a subsection to the sectors section:

[sectors.residential]
type = 'default'
[sectors.power]
type = 'default'

Above, we’ve added two sectors, residential and power. The name of the subsection is only used for identification. In other words, it should be chosen to be meaningful to the user, since it will not affect the model itself.

Sectors are defined in Sector.

A sector accepts these attributes:

type

Defines the kind of sector this is. Standard sectors are those with type “default”. This value corresponds to the name with which a sector class is registered with MUSE, via register_sector(). [INSERT OTHER OPTIONS HERE]

priority

An integer denoting which sectors runs when. Lower values imply the sector will run earlier. Later sectors can depend on earlier sectors for the their input. If two sectors share the same priority, then their order is not defined. Indeed, it should indicate that they can run in parallel. For simplicity, the keyword also accepts standard values:

  • “preset”: 0

  • “demand”: 10

  • “conversion”: 20

  • “supply”: 30

  • “last”: 100

Defaults to “last”.

interpolation

Interpolation method user when filling in missing values. Available interpolation methods depend on the underlying scipy method’s kind attribute.

investment_production

In its simplest form, this is the name of a method to compute the production from a sector, as used when splitting the demand across agents. In other words, this is the computation of the production which affects future investments. In it’s more general form, production can be a subsection of its own, with a “name” attribute. For instance:

[sectors.residential.production]
name = "match"
costing = "prices"

MUSE provides two methods in muse.production:

  • share: the production is the maximum production for the existing capacity and

    the technology’s utilization factor. See muse.production.maximum_production().

  • match: production and demand are matched according to a given cost metric. The

    cost metric defaults to “prices”. It can be modified by using the general form given above, with a “costing” attribute. The latter can be “prices”, “gross_margin”, or “lcoe”. See muse.production.demand_matched_production().

production can also refer to any custom production method registered with MUSE via muse.production.register_production().

Defaults to “share”.

dispatch_production

The name of the production method used to compute the sector’s output, as returned to the muse market clearing algorithm. In other words, this is computation of the production method which will affect other sectors.

It has the same format and options as the production attribute above.

Sectors contain a number of subsections: interactions

Defines interactions between agents. These interactions take place right before new investments are computed. The interactions can be anything. They are expected to modify the agents and their assets. MUSE provides a default set of interactions that have new agents pass on their assets to the corresponding retro agent, and the retro agents pass on the make-up of their assets to the corresponding new agents.

interactions are specified as a TOML array, e.g. with double brackets. Each sector can specify an arbitrary number of interactaction, simply by adding an extra interaction row.

There are two orthogonal concepts to interactions:

In practice, we always consider sequences of nets (i.e. more than one net) that interact using the same interaction function.

Hence, the input looks something like the following:

[[sectors.commercial.interactions]]
net = 'new_to_retro'
interaction = 'transfer'

“new_to_retro” is a function that figures out all “new/retro” pairs of agents. Whereas “transfer” is a function that performs the transfer of assets and information between each pair.

Furthermore, it is possible to pass parameters to either the net of the interaction as follows:

[[sectors.commercial.interactions]]
net = {"name": "some_net", "param": "some value"}
interaction = {"name": "some_interaction", "param": "some other value"}

The parameters will depend on the net and interaction functions. Neither “new_to_retro” nor “transfer” take any arguments at this point. MUSE interaction facilities are defined in muse.interactions.

An error is raised if and empty network is found. This is the case, for example, if a “new_to_retro” type of network has been defined but no retro agents are included in the sector.

technodata

Defines technologies and their features, in terms of costs, efficiencies, and emissions.

technodata are specified as an inline TOML table, e.g. with single brackets. A technodata section would look like:

[sectors.residential.technodata]
  technodata = '{path}/technodata/residential/Technodata.csv'
  commodities_in = '{path}/technodata/residential/CommIn.csv'
  commodities_out = '{path}/technodata/residential/CommOut.csv'

Where: technodata

Path to a csv file containing the characterization of the technologies involved in the sector, e.g. lifetime, capital costs, etc… See Techno-data.

commodities_in

Path to a csv file describing the inputs of each technology involved in the sector. See Commodities.

commodities_out

Path to a csv file describing the outputs of each technology involved in the sector. See Commodities.

Once the finest timeslice and its aggregates are given, it is possible for each sector

to define the timeslice simply by referring to the slices it will use at each level.

timeslice_levels

Optional. These define the timeslices of a sector. If not specified, the finest timeslice levels will be used (See Timeslices). It can be implemented with the following rows:

 [sectors.some_sector.timeslice_levels]
 day = ["daylight", "nightlife"]
 month = ["all-year"]

Above, ``sectors.some_sector.timeslice_levels.week`` defaults its value in the finest
timeslice. Indeed, if the subsection ``sectors.some_sector.timeslice_levels`` is not
given, then the sector will default to using the finest timeslices.

If the MCA uses a rougher
timeslice framework, the market will be expressed within it. Hence information from
sectors with a finer timeslice framework will be lost.

subsectors

Subsectors group together agents into separate groups servicing the demand for different commodities. There should be at least one subsector. And there can be as many as required. For instance, a one-subsector setup would look like:

[sectors.gas.subsectors.all]
agents = '{path}/technodata/Agents.csv'
existing_capacity = '{path}/technodata/gas/Existing.csv'

A two-subsector could look like:

[sectors.gas.subsectors.methane_and_ethanol]
agents = '{path}/technodata/me_agents.csv'
existing_capacity = '{path}/technodata/gas/me_existing.csv'
commodities = ["methane", "ethanol"]

[sectors.gas.subsectors.natural]
agents = '{path}/technodata/nat_agents.csv'
existing_capacity = '{path}/technodata/gas/nat_existing.csv'
commodities = ["refined", "crude"]

In the case of multiple subsectors, it is important to specify disjoint sets of commodities so that each subsector can service a separate demand. The subsectors accept the following keywords:

agents

Path to a csv file describing the agents in the sector. See user_guide/inputs/agents:agents.

existing_capacity

Path to a csv file describing the initial capacity of the sector. See user_guide/inputs/existing_capacity:existing sectoral capacity.

lpsolver

The solver for linear problems to use when figuring out investments. The solvers are registered via register_investment(). At time of writing, three are available:

  • an “adhoc” solver: Simple in-house solver that ranks the technologies according to cost and service the demand incrementally.

  • “scipy” solver: Formulates investment as a true LP problem and solves it using the `scipy solver`_.

  • “cvxopt” solver: Formulates investment as a true LP problem and solves it using the python package `cvxopt`_. `cvxopt`_ is not installed by default. Users can install it with pip install cvxopt or conda install cvxopt.

demand_share

A method used to split the MCA demand into separate parts to be serviced by specific agents. A basic distinction is between new and retrofit agents: the former asked to respond to an increase of commodity demand investing in new assets; the latter asked to invest in new asset to balance the decommissined assets.

There are currently two options:

  • new_and_retro(): the demand is split into a retrofit demand corresponding to demand that used to be serviced by decommisioned assets, and the new demand.

  • market_demand(): simply the consumption for the forecast year.

constraints

The list of constraints to apply to the LP problem solved by the sector. By default all of the following are included:

  • demand(): a lower-bound of the production decision variables specifying the target demand.

  • max_production(): an upper bound limiting how much can be produced for a given capacity.

  • max_capacity_expansion(): an upper bound limiting how much the capacity can grow during each investment event.

  • search_space(): a binary (on-off) constraint specifying which technologies are considered for investment.

output

Outputs are made up of several components. MUSE is designed to allow users to mix-and-match both how and what to save.

output is specified as a TOML array, e.g. with double brackets. Each sector can specify an arbitrary number of outputs, simply by adding an extra output row.

A single row looks like this:

[[sectors.commercial.outputs]]
filename = '{cwd}/Results/{Sector}/{Quantity}/{year}{suffix}'
quantity = "capacity"
sink = 'csv'
overwrite = true

The following attributes are available:

  • quantity: Name of the quantity to save. Currently, capacity exists,

    referring to muse.outputs.capacity(). However, users can customize and create further output quantities by registering with MUSE via muse.outputs.register_output_quantity(). See muse.outputs for more details.

  • sink: the sink is the place (disk, cloud, database, etc…) and format with which

    the computed quantity is saved. Currently only sinks that save to files are implemented. The filename can specified via filename, as given below. The following sinks are available: “csv”, “netcfd”, “excel”. However, more sinks can be added by interested users, and registered with MUSE via muse.outputs.register_output_sink(). See muse.outputs for more details.

  • filename: defines the format of the file where to save the data. There are several

    standard values that are automatically substituted:

    • cwd: current working directory, where MUSE was started

    • path: directory where the TOML file resides

    • sector: name of the current sector (.e.g. “commercial” above)

    • Sector: capitalized name of the current sector

    • quantity: name of the quantity to save (as given by the quantity attribute)

    • Quantity: capitablized name of the quantity to save

    • year: current year

    • suffix: standard suffix/file extension of the sink

    Defaults to {cwd}/{default_output_dir}/{Sector}/{Quantity}/{year}{suffix}.

  • overwrite: If False MUSE will issue an error and abort, instead of

    overwriting an existing file. Defaults to False. This prevents important output files from being overwritten.

There is a special output sink for aggregating over years. It can be invoked as follows:

[[sectors.commercial.outputs]]
quantity = "capacity"
sink.aggregate = 'csv'

Or, if specifying additional output, where … can be any parameter for the final sink:

[[sectors.commercial.outputs]]
quantity = "capacity"
sink.aggregate.name = { ... }

Note that the aggregate sink always overwrites the final file, since it will overwrite itself.

9.2.6. Preset sectors

The commodity production, commodity consumption and product prices of preset sectors are determined exogeneously. They are know from the start of the simulation and are not affected by the simulation.

Preset sectors are defined in PresetSector.

The three components, production, consumption, and prices, can be set independently and not all three need to be set. Production and consumption default to zero, and prices default to leaving things unchanged.

The following defines a standard preset sector where consumption is defined as a function of macro-economic data, i.e. population and gdp.

[sectors.commercial_presets]
type = 'presets'
priority = 'presets'
timeslice_shares_path = '{path}/technodata/TimesliceShareCommercial.csv'
macrodrivers_path = '{path}/technodata/Macrodrivers.csv'
regression_path = '{path}/technodata/regressionparameters.csv'
timeslices_levels = {'day': ['all-day']}
forecast = [0, 5]

The following attributes are accepted:

type

See the attribute in the standard mode, type. Preset sectors are those with type “presets”.

priority

See the attribute in the standard mode, priority.

timeslices_levels

See the attribute in the standard mode, Timeslices.

consumption_path

CSV output files, one per year. This attribute can include wild cards, i.e. ‘*’, which can match anything. For instance: consumption_path = “{cwd}/Consumption*.csv” will match any csv file starting with “Consumption” in the current working directory. The file names must include the year for which it defines the consumption, e.g. Consumption2015.csv.

The CSV format should follow the following format:

Consumption

RegionName

ProcessName

TimeSlice

electricity

diesel

algae

0

USA

fluorescent light

1

1.9

0

0

1

USA

fluorescent light

2

1.8

0

0

The index column as well as “RegionName”, “ProcessName”, and “TimeSlice” must be present. Further columns are reserved for commodities. “TimeSlice” refers to the index of the timeslice. Timeslices should be defined consistently to the sectoral level timeslices. The column “ProcessName” needs to be present and filled in, in order for the data to be read properly but it does not affect the simulation.

supply_path

CSV file, one per year, indicating the amount of a commodities produced. It follows the same format as consumption_path.

supply_path

CSV file, one per year, indicating the amount of a commodities produced. It follows the same format as consumption_path.

prices_path

CSV file indicating the amount of a commodities produced. The format of the CSV files follows that of Initial Market Projection.

demand_path

Incompatible with consumption_path or macrodrivers_path. A CSV file containing the consumption in the same format as Initial Market Projection.

macrodrivers_path

Incompatible with consumption_path or demand_path. Path to a CSV file giving the profile of the macrodrivers. Also requires regression_path.

regression_path

Incompatible with consumption_path. Path to a CSV file giving the regression parameters with respect to the macrodrivers. Also requires macrodrivers_path.

timeslice_shares_path

Incompatible with consumption_path or demand_path. Optional csv file giving shares per timeslice. The timeslice share definition needs to have a consistent number of timeslices as the sectoral level time slices. Requires macrodrivers_path.

filters

Optional dictionary of entries by which to filter the consumption. Requires macrodrivers_path. For instance,

filters.region = ["USA", "ASEA"]
filters.commodity = ["algae", "fluorescent light"]

9.2.7. Legacy Sectors

Legacy sectors wrap sectors developed for a previous version of MUSE to the open-source version.

Preset sectors are defined in PresetSector.

The can be defined in the TOML file as follows:

[global_input_files]
macrodrivers = '{path}/input/Macrodrivers.csv'
regions = '{path}/input/Regions.csv'
global_commodities = '{path}/input/MUSEGlobalCommodities.csv'

[sectors.Industry]
type = 'legacy'
priority = 'demand'
agregation_level = 'month'
excess = 0

userdata_path = '{muse_sectors}/Industry'
technodata_path = '{muse_sectors}/Industry'
timeslices_path = '{muse_sectors}/Industry/TimeslicesIndustry.csv'
output_path = '{path}/output'

For historical reasons, the three global_input_files above are required. The sector itself can use the following attributes.

type

See the attribute in the standard mode, type. Legacy sectors are those with type “legacy”.

priority

See the attribute in the standard mode, priority.

agregation_level

Information relevant to the sector’s timeslice.

excess

Excess factor used to model early obsolescence.

userdata_path

Path to a directory with sector-specific data files.

technodata_path

Path to a technodata CSV file. See. Techno-data.

output_path

Path to a directory where the sector will write output files.