Mosaik is a flexible Smart Grid co-simulation framework.
Mosaik allows you to reuse and combine existing simulation models and simulators to
create large-scale Smart Grid scenarios – and by large-scale we mean thousands of
simulated entities distributed over multiple simulator processes. These scenarios can
then serve as a test bed for various types of control strategies (e.g., multi-agent
systems (MAS) or centralized control).
This guide assumes that you are somewhat proficient with Python and know what
pip and virtualenv is. Else, you should follow the detailed
instructions.
Mosaik runs on Linux, OS X and Windows. It requires Python 3.8 or higher. To install everything, you need the package
manager Pip which is
bundled with Python 3.8 and above.
We also strongly recommend you to install everything into a virtualenv.
You can then install mosaik with pip:
$pipinstallmosaik
This provides you with the mosaik framework. There is also a simple demo
scenario which may help you to get started. Please refer to our detailed
instructions for installation.
For more information about avaiable components and example scenarios visit the
mosaik ecosystem page.
This guide is based on (K)ubuntu 18.04 Bionic Beaver, 64bit.
Mosaik and the demo scenario require Python >= 3.8, which should be fine
for any recent linux distribution. Note that we test mosaik only for the most
(typically three) recent python versions though.
We also need pip, a package manager for Python packages, and
virtualenv, which can create isolated Python environments for different
projects:
Mosaik alone is not very useful (because it needs other simulators to perform
a simulation), so we also provide a small demo scenario and some simple
simulators as well as a mosaik binding for PYPOWER.
PYPOWER requires NumPy and SciPy. We also need to install the revision
control tool git. You can use the packages shipped
with Ubuntu. We use apt-get to install NumPy, SciPy, and h5py as
well as git. By default, venvs are isolated from globally installed
packages. To make them visible, we also have to recreate the venv and set
the --system-site-packages flag:
If no errors occur, the last command will start the demo. The web visualisation
shows the demo in your browser: http://localhost:8000. You can click the nodes of the
topology graph to show a time series of their values. You can also drag them
around to rearrange them.
Mosaik and the demo scenario require Python >= 3.8. OS X only ships with
some outdated versions of Python, so we need to install a recent Python 2
and 3 first. The recommended way of doing this is with the packet manager homebrew.
To install homebrew, we need to open a Terminal and execute the following command:
The homebrew installer asks you to install the command line developer
tools for “xcode-select”. Install them. When you are done, go back to the
terminal and press Enter so that the installer continues.
If this doesn’t work for you, you’ll find more detailed instructions in the
homebrew wiki.
Once the installation is successful, we can install python and
python3:
$brewinstallpythonpython3
This will also install the Python package manager pip.
Next, we need virtualenv which can create isolated Python
environments for different projects:
$pipinstall-Uvirtualenv
Now we need to create a virtual environment for mosaik and its dependencies.
The common location for venvs is under ~/.virtualenvs/:
Mosaik alone is not very useful (because it needs other simulators to perform
a simulation), so we also provide a small demo scenario and some simple
simulators as well as a mosaik binding for PYPOWER.
To clone the demo repository, we need to install git. In order to
compile NumPy, SciPy and h5py (which are required by PYPOWER and the
database adapter) we also need to install gfortran which is included in gcc. You should deactivate
the venv for this:
The file names of the wheels (*.whl-files) may change when version-numbers
change. Please check the output of pip install or the directory ~/wheelhouse/
for the exact file names.
You can now clone the mosaik-demo repository into a folder where you
store all your code and repositories (we’ll use ~/Code/):
If no errors occur, the last command will start the demo. The web visualisation
shows the demo in your browser: http://localhost:8000. You can click the nodes of the
topology graph to show a time series of their values. You can also drag them
around to rearrange them.
Mosaik and the demo scenario require Python >= 3.8. By default, it will
offer you a 32bit installer. You can find the Windows x86-64 MSI installerhere.
When the download finished, double-click the installer.
Select Install for all users and click Next >.
The default installation path is okay. Click Next >.
In the Customize Python page, click on the Python node and select
Entire feature will be installed on local hard drive. Make sure that
Add python.exe to Path is enabled. Click Next >.
When Windows asks you to allow the installation, allow the installation.
Wait. Click Finish.
We also need virtualenv which can create isolated Python environments
for different projects.
Open a terminal window: Press the Windows key (or click on the start
menu) and enter cmd. Press Enter. Your terminal prompt should
look like C:\Users\yourname>. Execute the following command to install
virtualenv:
C:\Users\yourname> pip install -U virtualenv
Note
If your Windows account type is Standard User, you need to open the
terminal with administarator privileges (right-click the Terminal icon,
then open as Administrator). Make then sure that you are in your user
directory:
C:\Windows\system32> cd C:\Users\yourname
C:\Users\yourname>
Now we need to create a virtual environment for mosaik and its dependencies.
The common location for venvs is under Envs/ in your users
directory:
Mosaik alone is not very useful (because it needs other simulators to perform
a simulation), so we also provide a small demo scenario and some simple
simulators as well as a mosaik binding for PYPOWER.
The web visualisation shows the demo in your browser: http://localhost:8000.
You can click the nodes of the topology graph to show a timeline of their values.
You can also drag them around to rearrange them.
You can cancel the simulation by pressing Ctrl-C. More exceptions
may be raised. No problem. :-)
This section describes how mosaik works without going into too much detail.
After reading this, you should have a general understanding of what mosaik does
and how to proceed in order to implement the mosaik API or to create
a simulation scenario.
Mosaik’s main goal is to use existing simulators in
a common context in order to perform a coordinated simulation of a given (Smart
Grid) scenario.
That means that all simulators (or other tools and hardware-in-the-loop)
involved in a simulation usually run in their own process. Mosaik just tries to synchronize these processes and manages the exchange
of data between them.
To allow this, mosaik
provides an API for simulators to communicate with mosaik,
implements handlers for different kinds of simulator processes,
allows the modelling of simulation scenarios involving the different
simulators, and
schedules the step-wise execution of the different simulators and manages
the exchange of data (data-flows) between them.
Although mosaik is written in Python 3, its simulator API completely language
agnostic. It doesn’t matter if your simulator is written in Python 2, Java,
C, matlab or anything else.
We have simulators for households (blue icon) and for photovoltaics (green).
We’re also gonna use a load flow analysis tool (grey), and a monitoring and
analysis tool (yellow).
First, we have to implement the mosaik API for each of these “simulators”. When
we are done with this, we can create a scenario where we connect the households
to nodes in the power grid. Some of the households will also get a PV module.
The monitoring / analysis tool will be connected to the power grid’s
transformer node. When we connect all these entities, we also
tell mosaik about the data-flows between them (e.g., active power feed-in from
the PV modules to a grid node).
When we finally start the simulation, mosaik requests the simulators to perform
simulation steps and exchanges data between them according to the data-flows
described in the scenario. For our simple example, that would roughly look like
this:
The household and PV simulator perform a simulation step for an interval
[0, t[.
Mosaik gets the values for, e.g., P and Q (active and reactive power)
for every household and every PV module.
Mosaik sets the values P and Q for every node of the power grid based on
the data it collected in step 2. The load flow simulator performs
a simulation step for [0, t[ based on these inputs.
Mosaik collects data from the load flow simulator, sends it to the
monitoring tool and lets it also perform a simulation step for [0, t[.
Now the whole process is repeated for [t, t+i[ and so forth until the
simulation ends.
In this example, all simulators had the same step size t, but this is not
necessary. Every simulator can have its one step size (which may even vary
during the simulation). It is also possible that a simulator (e.g., a control
strategy) can set input values (e.g., a schedule) to another simulator (e.g.,
for “intelligent” consumers).
Mosaik consists of four main components that implement the different aspects of
a co-simulation framework:
The mosaik Sim API defines the communication protocol between
simulators and mosaik.
Mosaik uses plain network sockets and JSON encoded messages to communicate
with the simulators. We call this the low-level API. For some programming
languages there also exists a high-level API that implements everything
networking related and offers an abstract base class. You then only have to
write a subclass and implement a few methods.
The Scenario API provides a simple API that allows you to create
your simulation scenarios in pure Python (yes, no
graphical modelling!).
The scenario API allows you to start simulators and instantiate models from
them. This will give you entity sets (sets of entities).
You can then connect the entities with each other in order to establish
data-flows between the simulators.
Mosaik allows you both, connecting one entity at a time as well as
connecting whole entity sets with each other.
The Simulator Manager (or shorter, SimManager) is responsible for
handling the simulator processes and communicating with them.
It is able to a) start new simulator processes, b) connect to already
running process instances, and c) import a simulator module and execute
it in-process if it’s written in Python 3.
The in-process execution has some benefits: it reduces the amount of memory
required (because less processes need to be started) and it avoids the
overhead of (de)serializing and sending messages over the network.
External processes, however, can be executed in parallel which is not
possible with in-process simulators.
Mosaik’s Scheduler uses the event-discrete simulation approach for the coordinated simulation of
a scenario.
Mosaik supports both time-discrete and event-discrete simulations as well
as a combination of both paradigms.
Mosaik is able to handle simulators with different step sizes. A simulator
may even vary its step size during the simulation.
Mosaik tracks the dependencies between the simulators and only
lets them perform a simulation step if necessary (e.g., because its data is
needed by another simulator). It is also able to let multiple simulators
perform their simulation step in parallel if they don’t depend on each
other’s data.
Mosaik as a co-simulation tool organizes the data exchange between simulators
and coordinates the execution of the connected simulaters. This part is called
mosaik-core and contains mosaik itself and APIs for multiple
programming languages.
Mosaik is a co-simulation library. The components and tools
form the mosaik ecosystem.
Mosaik-core without any connected simulators doesn’t do much. This is why we
provide some simple and free simulators so that it is possible to start with
a working Smart-Grid simulation. These simulators belong to a part of mosaik’s ecosystem called
mosaik-components. More detailed documentation for some components can be found in the
component documentation.
To see how these components can be coupled to simulations, also some example scenarios are
provided in mosaik-examples.
Mosaik is developed following the “lean and mean” principle. That means that we
try to keep the software as simple as possible in order to keep it efficient
and easy to maintain. In order to make it easier to set up and run experiments with
mosaik we provide some tools that help building scenarios, connecting
simulators or to visualize and analyze the simulation results. These tools are
located in the mosaik-tools.
For testing simulators or scenarios, mosaik provides some basic simulators,
which allow to specify specific data to be sent.
There are also some implementations done by external users of mosaik. We give an overview
of the external components and external scenarios we know.
mosaik-core
The root folder contains mosaik itself and the high-level API implementations are
provided in the API folder.
This lists the mosaik components that are available on pypi. There are always component in work that are not released yet but are in working condition so if you don’t find what you are searching for here take a look in the repository.
mosaik-104 contains an adapter to receive IEC 60870-5-104 protocol
messages and hands it over to mosaik.
FMI adapter allows to couple Functional Mockup Units (FMU),
which are based on the FMI standard.
component documentation
The components listed above and provided by the mosaik team, have usually a documentation directly in their repository.
For components which need a more detailled documentation to describe how they work, the documentation is integrated here:
The heat pump model comprises four different calculation modes for
simulating the performance of a heat pump - ‘detailed’,
‘fast’, ‘hplib’, and ‘fixed’ modes. The first two are
based on the TESPy library,
the third is based on the hplib
library. In these three
modes, a quasi-steady state modelling approach has been adopted, i.e.,
the conditions of the operation of the heat pump vary with each
time-step and the steady state performance of the heat pump is
calculated at these different conditions for each time-step. The
‘fixed’ mode is the most simplified, operating the heat pump with
a fixed performance irrespective of the different operating conditions.
The model based on hplib, hereafter referred as “hplib” model, is a
parametric fit equation-based model, and thus takes a statistical
approach to predict the performance of the heat pump at different
operating conditions. The model based on TESPy, hereafter referred as
“tespy” model, is more complex and considers the physical states of
the fluids in the different components of the heat pump. Therefore, it
offers greater flexibility than the hplib model in estimating the
performance of the heat pump at different operating conditions.
However, as a result of this increased complexity, the simulation
time for the detailed calculations in tespy model is higher as compared
to that of the simpler calculations in hplib model.
The hplib model is based on hplib (“Heat Pump LIBrary”), an
open-source Python library that simulates the performance heat pumps
using parametric fit equations for the electric power and COP. The fit
parameters are identified by applying a least square regression model on
the publicly available heat pump keymark data of the European Heat Pump
Association (EHPA). It is possible to simulate the performance of both
air and water source heat pumps. The parameters are available for a
generic heat pump of both the types, as well as specific models
available in the market.
The limits on the operation of the heat pump, the supply water and
source air temperature ranges available from the technical datasheets of
the chosen heat pump model, have been added to the model directly
available in the hplib library.
The equations 1 and 2 are the fit equations for the electric power
and COP respectively. The reference values, Pel,ref is the
electrical power consumption at -7°C source temperature and
52°C supply water temperature. In both the equations,
p1-4 are the fit parameters, Tin is the source inlet
temperature, Tout is the supply water temperature, and
Tamb is the ambient temperature.
The evaporator and condenser inlet temperatures are the inputs to the
model. The model checks if they are within the operating range and
ensures that the source air temperature is lower than the incoming water
temperature. The model then calculates the electric power, COP, the
heating capacity, and the condenser mass flow as outputs. The electric power and
COP are estimated as shown in equations 1 and 2
respectively. The heating capacity is calculated from the electric power
and COP. The mass flow in the condenser is calculated assuming a
temperature difference of 5°C.
The user must specify the ‘calc_mode’ parameter as ‘hplib’,
and the ‘heat_source’, either ‘air’ or ‘water’, must
be specified. For the ‘hp_model’ parameter, the user can choose
from the different heat pump models available in the public heatpump keymark
database (the keywords can be obtained from the
‘hplib_database.csv’ file). If the ‘hp_model’ is set to
‘Generic’, the user must additionally specify ‘cons_T’,
‘heat_source_T’, and ‘P_th’.
The limits of operation for the heat pump are not available directly
within the model in the hplib library. If a corresponding equivalent
heat pump model based on TESPy is available, the keyword for that model
can be specified in the ‘equivalent_hp_model’. If not, the
operation limits can be specified via ‘hp_limits’ parameter.
An example of the dictionary with the required parameters can be seen in
the module documentation.
The tespy model is based on TESPy (“Thermal Engineering Systems in
Python”),
an open-source Python library that provides a powerful
simulation package for thermal processes like power plants, district
heating systems, heat pumps etc. An initial version of this model has
been used in a previous work,
and significant changes have been made later for a master’s thesis and for different
research projects. The performance of the heat pump is simulated by
considering the energy and mass balances in the individual
“components” of the heat pump – condenser, evaporator, compressor,
expansion valve, heat exchangers and pumps – and the state of fluids in
the “connections” between these “components.” The
connections and components together form a topological network
that is represented and solved as a system of equations. The schematic
of the heat pump system used in this work is shown in the figure below.
The flexibility offered by the TESPy library in choosing the components
of the network has been implemented through the following features in the model:
Stages of compression
The heat pump model is available in two system configurations, either
with a one-stage compressor or a two-stage compressor.
Additional components
Intercooler between the two stages of compression
Superheater between the evaporator and the compressor
TESPy has two modes of calculation, design and offdesign, to solve
the network. The design mode is used to design the system and forms
the first calculation of the network. While designing the plant, TESPy
offers much greater detail as compared to hplib, in terms of the
parametrization of the individual components, for example, the
isentropic efficiency of the compressor. The offdesign mode is used to
calculate the performance of the system if parameters deviate from the
design point, for example, operation at partial loads or operation at
different temperature/pressure levels. The system calculations from the
design mode form the basis for the offdesign mode. Both of these
calculation modes have been implemented in this model.
Operating temperatures -
Source air: -20°C to 35°C;
Water supply: 15°C to 65°C
Air_60kW_1stage
Stages of compression - 1
Intercooler - No
Superheater - No
Heating capacity range - 11.98 kW - 50.58 kW
Operating temperatures -
Source air: -20°C to 35°C;
Water supply: 15°C to 65°C
Any other heat pump available in the market, with a different heating
capacity and configuration, can be added to the model, following the
procedure shown in the example of the
“Air_30kW” heat pump.
Note
With TESPy, it is possible to simulate the performance of water-water
heat pumps as well. However, this has not yet been integrated into this
model and will be a part of a later release.
An example scenario using the heat pump simulator in the mosaik
environment is available in the ‘run_heatpump.py’ file.
The simulation is configured as shown below. The inputs to the heat pump
model and the outputs from it are handled by ‘mosaik-csv’ .
4SIM_CONFIG={ 5'HeatPumpSim':{ 6'python':'mosaik_components.heatpump.Heat_Pump_mosaik:HeatPumpSimulator', 7}, 8'CSV':{ 9'python':'mosaik_csv:CSV',10},11'CSV_writer':{12'python':'mosaik_csv_writer:CSVWriter'13},14}1516# Create World17world=mosaik.World(SIM_CONFIG)1819START='01.01.2016 00:00'20END=10*15*60# 2.5 Hours or 150 mins
The tespy model is used in the ‘fast’ calculation mode. The ‘Air_8kW’ heat pump is chosen. The
required parameters are set as shown below.
22# Heat pump23params={'calc_mode':'fast',24'hp_model':'Air_8kW',25'heat_source':'air',26}27# configure the simulator28heatpumpsim=world.start('HeatPumpSim',step_size=15*60)29# Instantiate model30heatpump=heatpumpsim.HeatPump(params=params)
The timeseries of heat demand, heat source temperature, and the condenser water inlet
temperature, that are needed as inputs for the model, are available in the ‘heatpump_
data.csv’ file.
32# Input data csv33HEAT_LOAD_DATA=os.path.join(os.path.dirname(os.path.abspath(__file__)),'data','heatpump_data.csv')34# configure the simulator35csv=world.start('CSV',sim_start=START,datafile=HEAT_LOAD_DATA)36# Instantiate model37heat_load=csv.HP()
The output data is saved into ‘hp_trial.csv’ file.
39# Output data storage40# configure the simulator41csv_sim_writer=world.start('CSV_writer',start_date='01.01.2020 00:00',date_format='%d.%m.%Y %H:%M',42output_file='hp_trial.csv')43# Instantiate model44csv_writer=csv_sim_writer.CSVWriter(buff_size=60*60)
The different entities are then connected and the simulation is executed.
46# Connect entities47world.connect(heat_load,heatpump,'Q_Demand','heat_source_T',('heat_source_T','T_amb'),'cond_in_T')48world.connect(heatpump,csv_writer,'Q_Demand','Q_Supplied','heat_source_T','P_Required','COP')4950# Run simulation51world.run(until=END)
calc_mode: The calculation mode that is used by the heat pump model. Currently, ‘detailed’, ‘fast’,
‘hplib’, and ‘fixed’ calculation modes are available. The differences are explained in the documentation.
hp_model: The specific model of the heat pump that must be simulated. The different models available currently
can be found in the documentation. This need not be specified for the ‘fixed’ calculation mode.
heat_source: The fluid that acts as the source of heat for the heat pump, either ‘water’ or ‘air’
If the ‘hplib’ calculation mode is chosen, the following parameter is required in addition to the mandatory
ones:
{'equivalent_hp_model':'Air_30kW',}
equivalent_hp_model: The heat pump model from the saved data file whose limits of operation will be applied
Alternatively, the limits can be directly specified in the following parameter:
The hot water tank model developed for another project, is used in
this work to act as a buffer in between the heating device and the heat
consumer. It is a multinode stratified thermal tank model, where
the tank volume is divided into a specified number of layers (nodes) of
equal volume, each characterized by a specific temperature. A
traditional density distribution approach is adopted where the water
flowing into the tank enters the layer that best matches its density
(i.e., temperature). The model assumes that the fluid streams are fully
mixed before leaving each of the layers and the flows between the layers
follow the law of mass conservation. Heat transfer to the surrounding
environment from the walls of the tank, and the heat transfer between
the layers have been considered.
The schematic of the hot water tank model is shown in the figure below. The
dimensions of the tank are specified in terms of its height, and either
the volume or diameter. The tank can be parametrized with sensors in the
model to record its temperature. The initial temperature of all the
layers must be set at the beginning of the simulation.
The flows into and out of the tank are specified as the connections of
the hot water tank model. The flow going to the heat pump (HP_out),
the space heating demand (SH_out), and the domestic hot water demand
(DHW_out) are connected to the bottom layer, the fourth layer and the
top layer respectively in the example schematic shown below.
As explained earlier, the flows coming into the
tank are not connected to a fixed layer in the tank. They are connected
to the layer with a temperature closest to that of the flow.
Schematic representation of the hot water tank model (example with 6 layers)
The heat transfer coefficient of the walls of the tank (htc_walls) is
assumed to be 0.28 W/m2-K . The heat transfer
coefficient for the heat transfer between the layers of the tank is
assumed to be 1.5 times the thermal conductivity of water.
The value is calculated as 0.897 W/m-K, considering the thermal
conductivity of water to be 0.598 W/m-K. However, these values can
be changed by modifying the parameters dictionary of the hot water tank
model.
The initial temperature profile inside the tank can be specified at the
time of initialization of the model. For flows coming into the tank,
both the temperature and flow rate should be specified. For the flows
going out of the tank, only the flow rate should be specified, as the
temperature is obtained from the corresponding layer of the tank. The
model ensures that the overall flow into and out of the tank is equal.
The model then updates the temperatures of each layer based on the water
flows through the specified connections, the heat transfer between the
layers and the heat transfer to the surrounding environment. The model
has the functionality to flip the layers to ensure a negative
temperature gradient from the top to the bottom of the tank. Finally,
the model updates the connections with respect to the updated layer
temperatures. For the flows going out of the tank, the temperature is
updated. For the flows coming into the tank, the corresponding layer is
updated.
The hot water tank model has one inlet connection (‘cc_in’) and one outlet
connection (‘cc_out’). The required parameters and the initial values are set
as shown below.
22# Hot water tank23params={24'height':2100,25'diameter':1200,26'T_env':20.0,27'htc_walls':0.28,28'htc_layers':0.897,29'n_layers':3,30'n_sensors':3,31'connections':{32'cc_in':{'pos':1500},33'cc_out':{'pos':10},34}35}36init_vals={37'layers':{'T':[30,50,70]}38}39# configure the simulator40hwtsim=world.start('HotWaterTankSim',step_size=15*60,config=params)41# Instantiate model42hwt=hwtsim.HotWaterTank(params=params,init_vals=init_vals)
The mass flow and temperature timeseries for these connections, that are needed as
inputs for the model, are available in the ‘tank_data.csv’ file.
44# Input data csv45HWT_FLOW_DATA=os.path.join(os.path.dirname(os.path.abspath(__file__)),'data','tank_data.csv')46# configure the simulator47csv=world.start('CSV',sim_start=START,datafile=HWT_FLOW_DATA)48# Instantiate model49csv_data=csv.HWT()
The output data is saved into ‘hwt_trial.csv’ file.
51# Output data storage52# configure the simulator53csv_sim_writer=world.start('CSV_writer',start_date='01.01.2020 00:00',date_format='%d.%m.%Y %H:%M',54output_file='hwt_trial.csv')55# Instantiate model56csv_writer=csv_sim_writer.CSVWriter(buff_size=60*60)
The different entities are then connected and the simulation is executed.
58# Connect entities59world.connect(csv_data,hwt,('T_in','cc_in.T'),('F_in','cc_in.F'),('F_out','cc_out.F'))60world.connect(hwt,csv_writer,'cc_in.T','cc_in.F','cc_out.T','cc_out.F','sensor_00.T','sensor_01.T','sensor_02.T')6162# Run the simulation63world.run(until=END)# As fast as possilbe
The hotwatertank module contains classes for the components of the hotwatertank
(Layer, Connection, Sensor, HeatingRod)
and a class for the hotwater tank itself (HotWaterTank).
Stratification is modelled by a number of Layer objects. Heat
producers and consumers can be connected to the hotwatertank via
Connection objects. The temperature at different positions in the
tank can be accessed via Sensor objects.
Hotwater tank parameters are provided at instantiation by the dictionary
params. This is an example, how the dictionary might look like:
volume: alternatively to the diameter the volume of the tank in liter
can be specified
T_env: ambient temperature in °C
htc: heat transfer coefficient of tank walls in W/(m2K)
htc_layers: imaginary heat transfer coefficient between layers in
W/(m2K)
n_layers: number of layers, n layers of the same dimension are
created
n_sensors: number of sensors, the sensors are equidistantly
distributed in the hotwater tank, sensors are named ‘sensor_00’,
‘sensor_01’, …, ‘sensor_n-1’ with ‘sensor_00’ indexing the undermost
sensor
connections: each connection is specified by an dictionary with a
structure analog to the example
heating_rods: each heating rod is specified by an dictionary with a
structure analog to the example
It is also possible to define layers and sensors explicitly:
Initial values for the temperature distribution in the tank or the
initial el. power of the heating rod are provided by the
dictionary init_vals, which might look like this:
{'layers':{'T':50},'hr_1':{'P_el':2000}}
this:
{'layers':{'T':[30,50,70]}'hr_1':{'P_el':2000}}
or this:
{'layers':{'T':[30,70]},'hr_1':{'P_el':2000}}
T: initial temperature of tank in °C, alternatively a
temperature range can be specified, whereby the lower limit defines the
temperature of ther undermost layer and the upper limit the temperature
of the uppermost layer, inbetween a linear temperature gradient is set,
it is also possible to specify the temperature of each layer individually
by passing a list of length n_layers
bottom - bottom of layer relatively to hotwater tank bottom in
mm
top - top of layer relatively to hotwater tank bottom in mm
diameter - diameter of layer in mm
bottom_top - must be True for the bottom or top
layer of the tank. This information is needed to calculate the outer
surface of the layer which in turn is needed to calculate heat losses
to the environment.
params – dictionary containing params which specify a sensor, so
far it contains only one entry pos, which defines the position of
the sensor above hotwater tank bottom
Devices are connected to the hotwater tank via connections.
Each connection is associated with a Layer. For input connections
(F>0) the corresponding layer is determined by temperature comparison. The
layer whose temperature is closest to the connection temperature is the
corresponding one. For output connections (F<0) the corresponding layer
depends on the position of the connection. The corresponding layer of a
connection is not fix, but may change during the simulation, if the flow or
temperatures of the connection or the temperature of the layers changes.
The controller model used in this work utilizes simple Boolean logic to:
Match the heating demands with the supply from the hot water
tank/back up heaters
Control the operation of the heat pump and adjust the mass flows in
the heat pump circuit
Control the in-built heating rod in the hot water tank
In order to perform each of the above functions, the controller must be
initialized with a set of parameters. The controller then analyzes the
information received from the different models and sends the necessary
information back to the models for the progression of the simulation.
The specific parameters and the information exchange for each of the
functions are detailed below.
The controller is initialized with a set point for the domestic hot
water (DHW) supply temperature (T_hr_sp_dhw). For each time step of
the simulation, the controller receives the domestic hot water demand
(dhw_demand), in liters of water, the temperature of the water
available for supply from the hot water tank (dhw_out_T), and the
temperature of the cold inlet water (dhw_in_T). The controller then
calculates the flow rate of water to be supplied from the tank by
dividing the demand with the time step (dhw_out_F). If the water from
the tank is available for supply at a temperature greater than the set
point, flow is adjusted by mixing with the cold inlet water. If the
temperature of water from tank is lower than the set point, the
controller calculates the heat to be supplied by the backup heater to
achieve the set point temperature, which is also the electric power
required (P_hr_dhw) by the heater, assuming 100% efficiency. Finally,
the controller sets the information of the flows to the hot water tank
(dhw_out_F, dhw_in_F).
The controller is initialized with the set points for the space heating
(SH) supply temperature (T_hr_sp_sh) and the temperature
difference in the space heating circuit (sh_dT). For each time step of
the simulation, the controller receives the space heating demand
(sh_demand), in kW, the temperature of the water available for supply
from the hot water tank (sh_out_T). The controller then calculates the
flow rate of water to be supplied from the tank from the demand and the
temperature difference (sh_in_F). If the water from the tank is
available for supply at a temperature greater than the set point, the
return temperature (sh_in_T) is calculated as the difference between
the supply temperature and the temperature difference. If the
temperature of water from tank is lower than the set point, the
controller calculates the heat to be supplied by the backup heater to
achieve the set point temperature, which is also the electric power
required (P_hr_sh) by the heater, assuming 100% efficiency. The
controller also calculates the return temperature as the difference
between the set point temperature and the temperature difference.
Finally, the controller sets the information of the flows to the hot
water tank (sh_out_F, sh_in_F, sh_in_T).
A simple hysteresis control based on the temperature of the bottom layer
of the hot water tank has been implemented for the operation of the heat
pump. The controller is initialized with a higher (T_hp_sp_h) and
lower (T_hp_sp_l) temperature set point for the hot water tank. The
controller then receives the temperature of the bottom layer
(bottom_layer_T) of the hot water tank. The bottom layer of the tank
is controlled to be maintained in between the two temperature limits,
i.e., the heat pump is turned on when the temperature in the bottom
layer of the tank falls below the lower set point. The heat pump is
turned off only when the temperature in the bottom layer of the tank is
greater than the higher set point. The heat pump continues to remain
turned off and turns back on only when the temperature of the bottom
layer falls below the lower set point again. The heat pump is controlled
by setting its status (hp_status) to either ‘on’ or ‘off’
based on the control logic explained above. The heat demand from the
heat pump (hp_demand) is calculated to be sent to the heat pump.
Control strategy 1, for the operation of heat pump
In addition to the control strategy for the heat pump operation
explained earlier, which is based only on the temperature of the bottom
layer of the hot water tank, a control strategy based on the
temperatures of both the bottom and top layers of the tank has been
implemented. The controller is initialized with a higher (T_hp_sp_h)
and lower (T_hp_sp_l) temperature set point for the hot water tank, as
done in the first control strategy. The controller then receives the
temperatures of the top layer (top_layer_T) and the bottom layer
(bottom_layer_T) of the hot water tank. The top layer of the tank is
controlled against the higher set point, i.e., the heat pump is turned
on when the temperature in the top layer of the tank falls below the
higher set point. The bottom layer of the tank is controlled against the
lower set point, i.e., the heat pump is turned off only when the
temperature in the bottom layer of the tank is greater than the lower
set point. In this case, the temperature of the top layer is expected to
be greater than the higher set point due to stratification inside the
tank. The heat pump continues to remain turned off and turns back on
only when the temperature of the top layer falls below the higher set
point again. The heat pump is controlled by setting its status
(hp_status) to either ‘on’ or ‘off’ based on the control
logic explained above. The heat demand from the heat pump (hp_demand)
is calculated to be sent to the heat pump.
Control strategy 2, for the operation of heat pump
Example scenarios of the co-simulation of the heat pump, hot water tank
and controller models are available in the examples folder.
There are cyclic dependencies between the models for each time step, for ex.,
the hot water tank needing the information from the controller regarding
the demands and the water flows, and the controller needing information
from the hot water tank regarding the temperature of the water to
calculate the flows. mosaik offers two different ways to resolve such
cyclic dependencies. The first is the time-shifted resolution, where the
information from one model is passed to the other model in the next time
step. The second is the same-time-loop resolution, where the information
exchange between the models is done in the same time step before
progressing the simulation to the next time step. The mosaik
documentation describes these two ways of dealing with cyclic
dependencies in detail (cyclic-data-flows).
The user can choose between the two types of execution, by specifying
the parameter ‘same_time_loop’, while initializing the simulators for
each of the models. The default execution mode is the time-shifted
resolution. For the same-time-loop resolution, the parameter
‘same_time_loop’ has to be set to ‘True’. Depending on the type of
execution, the way the connections between the different models are
setup varies, and can be seen in the example scenarios below.
Note
All the simulators must be set to the same type of execution
The first example scenario uses the time-based resolution
of the cyclic dependencies offered by mosaik. The different heat pumps, and
calculation modes available in the heat pump model are simulated along with the
hot water tank, with the controller model matching both the space heating and
domestic hot water demand with the heat available in the hot water tank
and controlling the operation of the heat pump.
The simulation is configured as shown below. The inputs/outputs to/from the models
are handled by ‘mosaik-csv’.
5sim_config={ 6'CSV':{ 7'python':'mosaik_csv:CSV', 8}, 9'CSV_writer':{10'python':'mosaik_csv_writer:CSVWriter'11},12'HeatPumpSim':{13'python':'mosaik_components.heatpump.Heat_Pump_mosaik:HeatPumpSimulator',14},15'HotWaterTankSim':{16'python':'mosaik_components.heatpump.hotwatertank.hotwatertank_mosaik:HotWaterTankSimulator',17},18'ControllerSim':{19'python':'mosaik_components.heatpump.controller.controller_mosaik:ControllerSimulator',20},21}2223# The start date, duration, and step size for the simulation24END=10*6025START='01.01.2020 00:00'26STEP_SIZE=60*1
The parameters and/or initial values for the different models are specified.
28#Parameters for mosaik-heatpump29params_hp={'hp_model':'Air_30kW_1stage',30'heat_source':'Air',31'cons_T':35,32'Q_Demand':19780,33'cond_in_T':30,34'heat_source_T':7,35}36#Parameters for hot water tank model37params_hwt={38'height':3600,39'volume':4000,40'T_env':20.0,41'htc_walls':0.28,42'htc_layers':0.897,43'n_layers':6,44'n_sensors':6,45'connections':{46'sh_in':{'pos':10},47'sh_out':{'pos':2150},48'dhw_in':{'pos':10},49'dhw_out':{'pos':3400},50'hp_in':{'pos':10},51'hp_out':{'pos':500},52},53}54init_vals_hwt={55'layers':{'T':[40.0,40.0,40.0,40.0,40.0,40.0]}56}57#Parameters for controller model58params_ctrl={59'T_hp_sp_h':50,60'T_hp_sp_l':40,61'T_hr_sp_dhw':40,62'T_hr_sp_sh':35,63'dhw_in_T':10,64'sh_dT':7,65'operation_mode':'heating',66'control_strategy':'1'67}
The different types of heat pumps and calculation modes that are simulated are
specified.
28# The different types of heat pumps and calculation modes that are simulated29model_list=['Air_30kW_1stage','Air_30kW_1stage','LW 300(L)',None]30calc_mode_list=['detailed','fast','hplib','fixed']31filename_list=['detailed','fast','hplib','fixed']
The mosaik ‘world’, and the simulators of the different models are initialized.
The inputs required for the different models – domestic hot water demand
(DHW Demand); space heating demand (SH Demand); the heat source
temperature, which is the ambient air in this case (T_amb); and the
temperature of the cold water replacing the domestic hot water supplied
from the tank (dhw_in_T) – are available in the ‘scenario_data.csv’ file. The inputs and the outputs are handled by
‘mosaik-csv’ and the output data is saved in csv files.
78# Initialize the world and the simulators.7980world=mosaik.World(sim_config)8182heatpumpsim=world.start('HeatPumpSim',step_size=STEP_SIZE)8384hwtsim=world.start('HotWaterTankSim',step_size=STEP_SIZE,config=params_hwt)8586ctrlsim=world.start('ControllerSim',step_size=STEP_SIZE)8788heat_load_file=os.path.join(os.path.dirname(os.path.abspath(__file__)),'data','scenario_data.csv')89heat_load_sim=world.start('CSV',sim_start=START,datafile=heat_load_file)9091CSV_File='Scenario_'+filename_list[i]+'_time_shifted.csv'92csv_sim_writer=world.start('CSV_writer',start_date='01.01.2020 00:00',date_format='%d.%m.%Y %H:%M',93output_file=CSV_File)
The specific parameters for the different heat pump models and calculation modes are added to the
parameters.
105# Instantiate the models106heatpumps=heatpumpsim.HeatPump.create(1,params=params_hp)107108hwts=hwtsim.HotWaterTank.create(1,params=params_hwt,init_vals=init_vals_hwt)109110ctrls=ctrlsim.Controller.create(1,params=params_ctrl)111112heat_load=heat_load_sim.HEATLOAD.create(1)113114csv_writer=csv_sim_writer.CSVWriter(buff_size=60*60)
The cyclic data flows between the different models are then set up in the time-shifted manner
and the simulation is executed.
116# connections between the different models117world.connect(heat_load[0],ctrls[0],'T_amb',('T_amb','heat_source_T'),('SH Demand [kW]','sh_demand'),118('DHW Demand [L]','dhw_demand'),'dhw_in_T')119120world.connect(hwts[0],ctrls[0],('T_mean','T_mean_hwt'),('mass','hwt_mass'),121('sensor_00.T','bottom_layer_T'),('sensor_04.T','top_layer_T'),122('dhw_out.T','dhw_out_T'),('sh_out.T','sh_out_T'),('hp_out.T','hp_out_T'))123124world.connect(ctrls[0],hwts[0],('sh_in_F','sh_in.F'),('sh_in_T','sh_in.T'),('sh_out_F','sh_out.F'),125('dhw_in_F','dhw_in.F'),('dhw_in_T','dhw_in.T'),('dhw_out_F','dhw_out.F'),('T_amb','T_env'),126time_shifted=True,127initial_data={'sh_in_F':0,'sh_in_T':0,'sh_out_F':0,128'dhw_in_F':0,'dhw_in_T':0,'dhw_out_F':0,129'T_amb':0,130},131)132133world.connect(heatpumps[0],ctrls[0],('Q_Supplied','hp_supply'),('on_fraction','hp_on_fraction'),134('cond_m','hp_cond_m'))135136world.connect(ctrls[0],heatpumps[0],('hp_demand','Q_Demand'),137'T_amb','heat_source_T',time_shifted=True,138initial_data={'hp_demand':0,'T_amb':5,'heat_source_T':5})139140world.connect(hwts[0],heatpumps[0],('hp_out.T','cond_in_T'))141142world.connect(heatpumps[0],hwts[0],('cons_T','hp_in.T'),('cond_m','hp_in.F'),('cond_m_neg','hp_out.F'),143time_shifted=True,initial_data={'cons_T':0,'cond_m':0,'cond_m_neg':0})144145world.connect(heat_load[0],csv_writer,'T_amb','SH Demand [kW]','DHW Demand [L]')146world.connect(heatpumps[0],csv_writer,'Q_Demand','Q_Supplied','T_amb','heat_source_T','cons_T','P_Required',147'COP','cond_m','cond_in_T','on_fraction')148149world.connect(ctrls[0],csv_writer,'heat_demand','heat_supply','hp_demand','sh_supply','sh_demand','hp_supply',150'sh_in_F','sh_in_T','sh_out_F','sh_out_T','dhw_in_F','dhw_in_T','dhw_out_F','dhw_out_T',151'hp_in_F','hp_in_T','hp_out_F','hp_out_T','P_hr_sh','P_hr_dhw','dhw_demand','dhw_supply')152world.connect(hwts[0],csv_writer,'sensor_00.T','sensor_01.T','sensor_02.T','sensor_03.T','sensor_04.T',153'sensor_05.T','sh_out.T','sh_out.F','dhw_out.T','dhw_out.F','hp_in.T','hp_in.F','hp_out.T','hp_out.F',154'T_mean','sh_in.T','sh_in.F','dhw_in.T','dhw_in.F')155156#Run157world.run(until=END)
The second example scenario uses the event-based resolution
of the same-time-loop cycles offered by mosaik. Only the things that need to be changed
when compared to the time-based resolution are shown below.
While initializing the model simulators, the ‘same_time_loop’ parameter has to be set to
‘True’ for all the models.
The cyclic data flows between the different models are then set up in the same-time-loop manner.
116# connections between the different models117world.connect(heat_load[0],ctrls[0],'T_amb',('T_amb','heat_source_T'),('SH Demand [kW]','sh_demand'),118('DHW Demand [L]','dhw_demand'),'dhw_in_T')119120world.connect(hwts[0],ctrls[0],('T_mean','T_mean_hwt'),('mass','hwt_mass'),121('sensor_00.T','bottom_layer_T'),('sensor_04.T','top_layer_T'),122('dhw_out.T','dhw_out_T'),('sh_out.T','sh_out_T'),123('hp_out.T','hp_out_T'))124125world.connect(ctrls[0],hwts[0],('sh_in_F','sh_in.F'),('sh_in_T','sh_in.T'),('sh_out_F','sh_out.F'),126('dhw_in_F','dhw_in.F'),('dhw_in_T','dhw_in.T'),('dhw_out_F','dhw_out.F'),('T_amb_hwt','T_env'),127('hp_in_T','hp_in.T'),('hp_in_F','hp_in.F'),('hp_out_F','hp_out.F'),weak=True)128129world.connect(heatpumps[0],ctrls[0],('Q_Supplied','hp_supply'),('on_fraction','hp_on_fraction'),130('cond_m','hp_in_F'),('cond_m_neg','hp_out_F'),('cons_T','hp_in_T'),weak=True)131132world.connect(ctrls[0],heatpumps[0],('hp_demand','Q_Demand'),('hp_out_T','cond_in_T'),133'T_amb','heat_source_T')134135136world.connect(heat_load[0],csv_writer,'T_amb','SH Demand [kW]','DHW Demand [L]')137world.connect(heatpumps[0],csv_writer,'Q_Demand','Q_Supplied','T_amb','heat_source_T','cons_T','P_Required',138'COP','cond_m','cond_in_T','on_fraction')139140world.connect(ctrls[0],csv_writer,'heat_demand','heat_supply','hp_demand','sh_supply','sh_demand','hp_supply',141'sh_in_F','sh_in_T','sh_out_F','sh_out_T','dhw_in_F','dhw_in_T','dhw_out_F','dhw_out_T',142'hp_in_F','hp_in_T','hp_out_F','hp_out_T','P_hr_sh','P_hr_dhw','dhw_demand','dhw_supply')143world.connect(hwts[0],csv_writer,'sensor_00.T','sensor_01.T','sensor_02.T','sensor_03.T','sensor_04.T','sensor_05.T',144'sh_out.T','sh_out.F','dhw_out.T','dhw_out.F','hp_in.T','hp_in.F','hp_out.T','hp_out.F',145'T_mean','sh_in.T','sh_in.F','dhw_in.T','dhw_in.F')
For the same-time-loop execution, it is important to set the initial event that kick-starts the
simulation, which is the simulation of the hot water tank for this scenario. The simulation is
then executed.
147# To start hwts as first simulator148world.set_initial_event(hwts[0].sid)149150#Run151world.run(until=END)
The tutorial
provided in TESPy’s documentation for simulating heat pumps
has been followed to develop the first design calculation, available in
the ‘Parametrization_NominalData.py’ file. While the tutorial
provides a detailed explanation of the complete parametrization of the model, only
the most relevant parameters are discussed here.
The data obtained from the manufacturer’s datasheet corresponding to the
nominal operating point is shown in table below.
This data has been used to set the parameters of the corresponding components
and connections as described in the tutorial. Since the refrigerant R448A is
not available in TESPy, R404A has been used due to the similarity in their
properties (reference).
TESPy uses the isentropic efficiency of the compressor to calculate the power
consumption as shown in equation below.
Since the isentropic efficiency of the compressor (‘eta_s’) is not available in the
datasheet, the value has been changed on a trial-and-error basis to
match the power consumption calculated by the model to that mentioned in
the datasheet.
Note
For heat pumps having two stages of compression, like the one in this example, in addition
to the isentropic efficiency, the pressure ratio (‘pr’) for the second stage is also a
required parameter.
This has to be adjusted on a trial and error basis as well, so that it works for
the different range of operating conditions of the heat pump. This could be checked, by choosing
data points from the edges of the operation range, and following the same procedure as done for
nominal operating point data.
The design case from the previous step can be used as the basis for
offdesign calculations to predict the system’s performance (in terms
of Electrical power/COP) at different operating conditions, i.e., a
different source or consumer inlet temperature. While the tutorial in
TESPy’s documentation provides a detailed explanation of all the changes
between the two modes of calculations, only the most relevant changes
are discussed here.
The refrigerant and the temperature difference for air in the evaporator
remain unchanged. The source air temperature and the condenser inlet
temperature are available as inputs to the model. The maximum heating
capacity at a particular source air temperature is available from the
datasheet. The condenser outlet temperature is not known in the
off-design case, but must be predicted. In the design mode, the heating
capacity and temperature difference between condenser inlet and outlet
are used to calculate the mass flow in the circuit. This mass flow is
used to calculate the condenser outlet temperature in the off-design
mode.
In order to determine the electrical power consumption, the isentropic
efficiency of the compressor is required. Since this is not available in
the datasheets for the entire range of operation, the default
characteristic curve
available in the TESPy library, shown in figure below, is used.
Default characteristic curve for the isentropic efficiency of the
compressor in TESPy library
The x-axis is the ratio of the mass flow into the
compressor in the design and off-design cases, and the y-axis is the
ratio of the isentropic efficiencies in the design and off-design
conditions.
The default characteristic curve is generic and therefore does not
accurately reflect the performance of the specific model of the heat
pump chosen. Instead of relying on the isentropic efficiency from a
single design point and the default characteristic curve across the
entire range of operation, a series of design points have been developed
based on the data available in the manufacturer’s datasheet for the
operating conditions shown in table 3.3.
The operating range of the heat pump for the source air temperature is
-20°C to 35°C. The actual operation range of the
heat pump on the condenser outlet temperature is 25°C to
60°C. In the model, the range is further increased to
15°C to 60°C, in order to simulate low temperature
lift conditions. The temperature difference in the condenser, constant
at 5°C in the design case, has been used to calculate the
condenser inlet temperature.
Extension of the heating capacity table of the heat pump
The manufacturer’s datasheets contain the heating power curves and the
electrical power curves as shown in figure below.
Heating capacity, electrical power, and COP curves of the chosen
heat pump
In all the plots, the x-axis corresponds to the source (air) temperature.
The data is available for the entire range of source air temperature, but
only for two condenser outlet temperatures, 35°C and 55°C.
The heating capacity increases with an increase in the source air
temperature, but does not change significantly with a change in the
condenser outlet temperature. At a given source air temperature, the
heating capacity for all the other condenser outlet temperatures is
assumed to be the average of the heating capacities at 35°C
and 55°C.
The power consumption changes with both the source air temperature and
the condenser outlet temperature. An approach based on Carnot efficiency
has been used to predict the power consumption at the condenser outlet
temperatures other than 35°C and 55°C. The ideal
COP is calculated for all the operating points, using the equation below
(note that the temperatures have to be in Kelvin scale).
A second order polynomial equation has been fit to the pairs of the
Carnot efficiencies and the corresponding temperature lifts, of
operating points with condenser outlet temperatures 35°C and
55°C, as shown in figure below.
Carnot efficiency/COP vs Temperature Lift plot for the chosen heat
pump
In this figure, the COP of the heat pump is also plotted against the
temperature lift. The Carnot efficiencies of the remaining operating points
are estimated using the fit equation, which in turn are used to estimate
the real COP/power consumption.
For the series of design points identified, the calculated heating
capacity and power consumption data is summarized in the table below.
The tutorial available in the ‘script_etas_gen.ipynb’ is followed to generate the compressor efficiency map. The
model is parametrized for each of the design point in the expanded heating
capacity table from the previous step, as done for the initial parametrization
of the model for the nominal operating point. As the power consumption of the
compressor is dependent on the isentropic efficiency, which is set as a
parameter in the compressor, it is changed for each point in order to match
the power consumption calculated by the model and that in the table. The
isentropic efficiency values are restricted to the range of 0.3 - 0.95.
Note
In the instances when the power values cannot be matched even at the extreme
values for the compressor isentropic efficiency, the extreme values are assumed
despite the difference in power predicted by the model and that in the table.
The compressor isentropic
efficiency map generated as described is summarized in table below.
‘detailed’ calculation mode of the model for the simulations
After establishing the series of design points, the model is then developed
to estimate the performance of the heat pump at any operating
point within its range. The heat pump is used as the primary heating
source for the hot water tank. The temperature of the water flow to the
heat pump from the hot water tank, and the source air temperature are
the required inputs.
The model checks if the conditions are within the operation range of the
heat pump and ensures that the source air temperature is lower than the
incoming water temperature. The source air temperature closest to the
input value is then identified from the expanded heating capacity table.
The model checks that the inlet water temperature is lower than the maximum
possible condenser outlet temperature at the identified design source air
temperature. Assuming a temperature difference of 5°C in the
condenser in the design case, an initial condenser outlet temperature is
estimated. The closest design point to the estimated condenser outlet
temperature is then identified from the extended heating capacity
table.
In case this estimated condenser temperature is greater than the maximum
possible outlet temperature, the maximum value is used as the design point.
The heating capacity and the isentropic efficiency of the compressor
corresponding to the design point source air and condenser outlet water
temperatures are identified from the respective tables, the expanded
heating capacity table and compressor
efficiency table.
The model first performs a network calculation in the design mode at the
identified design point operating conditions. An offdesign mode
calculation of the network is then performed for the actual input
operating conditions, based on the design mode calculation and the
default characteristics of TESPy, to obtain the condenser outlet water
temperature (supply temperature), the mass flow of water in the
condenser and the power consumption of the heat pump.
In addition to the detailed mode of calculation explained above, a fast
calculation mode has also been implemented in the model to improve its
computational speed. The model is discreetly parametrized over the
entire operation range of the heat pump, at a resolution of
1°C for both the inputs, the source air temperature, and the
condenser water inlet temperature. The detailed calculation mode can be
implemented over this range of inputs and the output data from the
model- the coefficient of performance (COP) of the heat pump and the
condenser mass flow rate- can be saved. During the actual simulation, the
saved inputs that are lower/closest to the actual input data are
identified, and the saved output data for these points are used to
calculate the outputs of the model, rather than performing the actual
design and offdesign calculations. Though the granularity of the model
is reduced, there is a significant improvement in the simulation
duration.
Flowchart explaining the fast calculation mode of the TESPy heat pump
model
The data for the ‘fast’ calculation mode can be calculated and saved as
follows:
Adding the new heat pump model
The initial parametrization of the new heat pump model, based on
the nominal operating point, from the ‘Parametrization_NominalData.py’ file, should be added
to the ‘_design_hp’ method of the heat_pump_design.py
file (line 217).
The tutorial in the ‘Fast_Mode_DataProcessing.ipynb’ script should be used to process the fast mode data, to fill
the missing values in the data that result from the errors in the model etc.
While the hplib heat pump model available in the package can simulate
the performance of the different heat pumps from the keymark data, the tespy
model provides fewer options. In order to simulate different heat pumps,
apart from the ones already available in this package, the heat pumps have
to be intergrated to the tespy model. An example of the integration of
the “Air_30kW” heat pump, based on “ait-deutschland LW-300(L)”, shows the procedure in detail. The development of the model to
simulate the performance of this specific heat pump is described in the
steps below:
mosaik-heatpump (v1.0.0) provides models for the simulation of heating systems- consisting of heat pumps, hot water tanks,
and controllers - and adapters for the co-simulation of these models using mosaik.
Installation & Tests
You can install mosaik-heatpump with pip:
pipinstallmosaik-heatpump
You can run the tests with:
pytest
Getting started
A description of the different models available in the package and examples to use individual models can
be found here.
Example scenarios for the co-simulation of all the models can be found here.
Getting help
Please report bugs and ideas for improvement to our issue tracker.
The model was initially used and presented here , but it is not
described in detail. We plan to publish a paper presenting the package, and will update this
section after the publication.
License
The package is completely open source and is covered under the MIT License.
mosaik-examples
The mosaik-demo
contains a simple demo scenario for mosaik.
The DES demo is a simple example
scenario showing the new mosaik 3.0 DES features
maverig mosaik GUI is a visualization component, which is
not maintained anymore.
basic simulators
In order to test custom-made simulators, two basic simulators are provided to use and connect to.
The InputSimulator is a simulator that can
be used to feed either a constant value or the value of a function into a
designated simulator ready to handle the data.
The OutputSimulator writes
data from a custom simulator into a python dictionary.
Users can access this dictionary by calling
get_dict on a
created output simulator entity.
Below is an example code snippet that connects the input simulator with the output simulator and executes ten time steps.
After the simulation is done, the dictionary including the values received by the input simulator is printed.
# The output simulator is initialized.output_dict=world.start("OutputSim")# Two entities of the output simulator model are created.output_model=output_dict.Dict.create(2)# The input simulator is initialized.input=world.start("InputSim",step_size=1)# One function input simulator entity is created.input_model_func=input.Function.create(1,function=sample_function)# One constant input simulator entity is created.input_model_const=input.Constant.create(1,constant=2)# The input entities are connected to separate output entitiesworld.connect(input_model_func[0],output_model[1],"value")world.connect(input_model_const[0],output_model[0],"value")# Run simulation.world.run(until=END)# Dictionary content is printed.pprint(output_dict.get_dict(output_model[0].eid))pprint(output_dict.get_dict(output_model[1].eid))
external components
These components are developed by external users of mosaik and we can not guarantee or support
the flawless integration of these tools with mosaik.
If you also have implemented additional tools for mosaik, simulation models or adapters,
feel free to contact us at mosaik [ A T ] offis.de to be listed here.
pysimmods contains some simulation models,
which can be used in mosaik scenarios.
MIDAS contains a semi-automatic scenario configuration
tool.
mosaik-docker is a package for the deployment
of mosaik with Docker.
ZDIN-ZLE components contains the research and development of digitalized
energy systems in ZLE using mosaik (collection of simulation models).
nestli (Neighborhood Energy System Testing towards Large-scale
Integration) is a co-simulation environment for benchmarking the performance of BACS (building automation and
control systems). Is uses EnergyPlus and FMUs with mosaik.
toolbox_doe_sa is a toolbox with Design of Experiment (DoE) and
Sensitivity Analysis (SA) methods developed in the ERIGrid 2.0 project.
mosaik-demod is a domestic energy demand modeling simulator.
palestrai-mosaik is an adapter to integrate
palaestrAI (an universal framework for multi-agent artificial intelligence)
into mosaik.
QEMS - Quarter Energy Management System contains simulation components,
which are used to simulate an energy management system for neighborhoods for analyzing and optimizing energy flows.
external scenarios
These scenarios are developed by external users of mosaik and we can not guarantee or support
the flawless practicability.
Benchmark Model Multi-Energy Networks STL is based on the
multi-energy networks benchmark and contains a same time loop for improved initialization of the simulators.
ZDIN-ZLE scenarios contains the research and development of digitalized
energy systems in ZLE using mosaik (collection of simulation scenarios).
In the basic tutorial you’ll learn how you can integrate simulators and control
strategy into the mosaik ecosystem as well as how you create simulation
scenarios and execute them.
In the first part, we’ll implement the Sim API for a simple example simulator.
We’ll also create a simulation scenario in which that simulator will send its
data to mosaik-hdf5 which will
store it in an HDF5 database.
In the second part, we’ll also integrate a simple control mechanism into
mosaik. We’ll then create a scenario in which that control mechanism controls
the example simulator from part one.
In the third part, we’ll implement an additional master controller, which communicates
with the other controllers. This communication takes place as same-time loop without
progress in simulation time and illustrated this new mosaik 3.0 feature.
It can be used for negotiation between multiple agents or controllers, like shown
in the tutorial at hand, but also for initialization of simulations consisting of
multiple phsycial systems.
In the next part, we’ll implement a scenario with a new controller, which sets external events.
These external events come from a simple button click-event of a graphical user interface.
Therefore, with this new mosaik 3.0 feature it is possible to do Human-in-the-Loop simulations to support human interactions.
The Odysseus tutorial you’ll learn how to connect the data-stream-management-tool
Odysseus to mosaik. The second part shows some examples on how to use Odysseus.
This tutorial may also be of some use when you want to connect any other component
via ZeroMQ.
The Java API tutorial shows you how to use the Java API. This API is intended to
connect simulators written in Java to mosaik. You can use the Java-API also as a
RCP-Server if you want to run your Java-simulator on a separate machine.
Basic tutorial
Integrating a simulation model into the mosaik ecosystem
In this section we’ll first implement a simple example simulator. We’ll then
implement mosaik’s Sim-API step-by-step.
We want to implement a very simple model with the following behavior:
val0 = init_val
vali = vali − 1 + delta for i ∈ N, i > 0, delta
∈ Z
That simply means our model has a value val to which we add some delta
(which is a positive or negative integer) at every simulation step. Our model
has the attribute delta (with value 1 by default) which can be changed by
control mechanisms to alter the behavior of the model. And it has the (output)
attribute val which is its current value.
Schematic diagram of our example model. You can change the delta and
collect the val as output.
Here is a possible implementation of that simulation model in Python:
# example_model.py"""This module contains a simple example model."""classModel:"""Simple model that increases its value *val* with some *delta* every step. You can optionally set the initial value *init_val*. It defaults to ``0``. """def__init__(self,init_val=0):self.val=init_valself.delta=1defstep(self):"""Perform a simulation step by adding *delta* to *val*."""self.val+=self.delta
So lets start implementing mosaik’s Sim-API for this model. We can use the
Python high-level API for this. This package
eases our workload, because it already implements everything necessary for
communicating with mosaik. It provides an abstract base class which we can
sub-class. So we only need to implement four methods and we are done.
If you already installed mosaik and the demo, you
already have this package installed in your mosaik virtualenv.
We start by creating a new simulator_mosaik.py and import the module
containing the mosaik API as well as our model:
# simulator_mosaik.py"""Mosaik interface for the example simulator."""importmosaik_api_v3importexample_model
Next, we prepare the meta data dictionary that tells mosaik which
time paradigm it follows (time-based, event-based,
or hybrid), which models our simulator implements and which parameters and
attributes it has. Since this data is usually constant, we define this at
module level (which improves readability):
In this case we create a hybrid simulator, because we want to be able to
control it using delta events later. For now, we won’t use delta,
though.
We added our “ExampleModel” model with the parameter init_val and the
attributes delta and val. At this point we don’t care if they are inputs
or outputs. We just list everything we can read or write. The public flag
should usually be True. You can read more about it in the Sim API
docs. From this information, mosaik deduces that our model could
be used in the following way:
# Model name and "params" are used for constructing instances:model=example_model.Model(init_val=42)# "attrs" are normal attributes:print(model.val)print(model.delta)
The package mosaik_api_v3 defines a base class Simulator for which we now
need to write a sub-class:
classExampleSim(mosaik_api_v3.Simulator):def__init__(self):super().__init__(META)self.eid_prefix='Model_'self.entities={}# Maps EIDs to model instances/entitiesself.time=0
In our simulator’s __init__() method (the constructor) we need to call
Simulator.__init__() and pass the meta data dictionary to it.
Simulator.__init__() will add some more information to the meta data and
set it as self.meta to our instance.
We also set a prefix for our entity IDs and prepare a dictionary which will
hold some information about the entities that we gonna create.
We can now start to implement the four API calls init, create, step
and get_data:
This method will be called exactly once while the simulator is being started
via World.start.
It is used for additional initialization tasks (e.g., it can handle parameters
that you pass to a simulator in your scenario definition). It must return the
meta data dictionary self.meta:
definit(self,sid,time_resolution,eid_prefix=None):iffloat(time_resolution)!=1.:raiseValueError('ExampleSim only supports time_resolution=1., but'' %s was set.'%time_resolution)ifeid_prefixisnotNone:self.eid_prefix=eid_prefixreturnself.meta
The first argument is the ID that mosaik gave to that simulator instance. The
second argument is the time resolution of the
scenario. In this example only the default value of 1. (second per integer
time step) is supported. If you set another value in the scenario, the
simulator would throw an error and stop.
In addition to that, you can define further (optional) parameters which you
can later set in your scenario. In this case, we can optionally overwrite the
eid_prefix that we defined in __init__().
create() is called in order to initialize a number of simulation model
instances (entities) within that simulator. It must return a list with some
information about each entity created:
The first two parameters tell mosaik how many instances of which model you
want to create. As in init(), you can specify additional parameters for
your model. They must also appear in the params list in the simulator meta
data or mosaik will reject them. In this case, we allow setting the initial
value init_val for the model instances.
For each entity, we create a new entity ID [1] and a model instance. We also
create a mapping (self.entities) from the entity ID to our model. For each
entity we create we also add a dictionary containing its ID and type to the
entities list which is returned to mosaik. In this example, it has num
entries for the model model, but it may get more complicated if you have,
e.g., hierarchical models.
The step() method tells your simulator to perform a simulation step. It
receives its current simulation time, a dictionary with input values
from other simulators (if there are any), and the time until the simulator can
safely advance its internal time without creating a causality error. For
time-based simulators (as in our example) it can be safely ignored (it is
equal to the end of the simulation then). The method returns to mosaik the
time at which it wants to do its next step. For event-based and hybrid
simulators a next (self-)step is optional. If there is no next self-step, the
return value is None/null.
Note
The max_advance value is not necessarily used and is only for special use
cases where simulators can advance in time without expecting new inputs from
other simulators, e.g. for the integration of a communication simulation.
defstep(self,time,inputs,max_advance):self.time=time# Check for new delta and do step for each model instance:foreid,model_instanceinself.entities.items():ifeidininputs:attrs=inputs[eid]forattr,valuesinattrs.items():new_delta=sum(values.values())model_instance.delta=new_deltamodel_instance.step()returntime+1# Step size is 1 second
In this example, the inputs could be something like this:
The inner dictionaries containing the actual values may contain multiple
entries if multiple source entities provide input for another entity. In
the case above, we have two source entities, ‘Model_0’ providing the
delta value to the destination entity (object of ExampleSim) and
‘Model_1’ providing the delta and val value to the destination entity
(object of ExampleSim). The source entitiy, ‘Model_0’ has the attribute
‘delta’ as the key to another nested dictionary which contains the
simulator id and its corresponding ‘delta’ value. Similarly the source
entity ‘Model_1’ has the attributes ‘delta’ and ‘val’ as the keys to two
other nested dictionaries which contain the simulator id and its
corresponding ‘delta’ and ‘val’ values.
The structure of the inputs dictionary created by mosaik is always the
same as depicted above, only the number of source entities (dependent on
the connections in the scenario (‘Model_0’ and ‘Model_1’ in our case))
and the number of attributes passed by the source entity varies. The
first key of the nested dictionary will be the source entity
(‘Model_1’), the following keys will be the attributes passed by this
source entity to the destination entity (‘delta’: {‘src_id_1’: 42},
‘val’: {‘src_id_1’: 20}).
The simulator receiving these inputs is responsible for aggregating them (e.g., by
taking their sum, minimum or maximum. Since we are not interested in the
source’s IDs, we convert that dict to a list with values.values() before we
calculate the sum of all input values.
After we converted the inputs to something that our simulator can work with,
we let it finally perform its next simulation step.
The return value time+1 tells mosaik that we wish to perform the next
step in one second (in simulation time), as the time_resolution is 1.
(second per integer step). Instead of using a fixed (hardcoded) step size you
can easily implement any other stepping behavior.
The get_data() call allows other simulators to get the values of the
delta and val attributes of our models (the attributes we listed in
the simulator meta data):
defget_data(self,outputs):data={}foreid,attrsinoutputs.items():model=self.entities[eid]data['time']=self.timedata[eid]={}forattrinattrs:ifattrnotinself.meta['models']['ExampleModel']['attrs']:raiseValueError('Unknown output attribute: %s'%attr)# Get model.val or model.delta:data[eid][attr]=getattr(model,attr)returndata
The outputs parameter contains the query and may in our case look like this:
The Outputs dictionary may contain multiple keys if multiple destination
entities ask for the output from the source entity. In this case we have
two destination entities ‘Model_0’ and ‘Model_1’ which are requesting
for the source attributes. ‘Model_0’ is requesting for the two source
attributes ‘delta’ and ‘value’, whereas ‘Model_1’ is requesting for 1
source attribute ‘value’. The structure of the Outputs dictionary
created by mosaik is always the same as depicted above, only the number
of destination entities (dependent on the connections in the scenario
(‘Model_0’ and ‘Model_1’ in our case)) and the number of attributes
requested by the destination entity varies.
In our implementation we loop over each entity ID for which data is requested.
We then loop over all requested attributes and check if they are valid. If so,
we dynamically get the requested value from our model instance via
getattr(obj,'attr'). We store all values in the data dictionary and
return it when we are done.
The last step is adding a main() method to make our simulator executable
(e.g., via python-msimulator_mosaikHOST:PORT). The package
mosaik_api_v3 contains the method start_simulation() which creates
a socket, connects to mosaik and listens for requests from it. You just call it
in your main() and pass an instance of your simulator class to it:
Simulators running on different nodes than the mosaik instance are supported
explicitly with the mosaik Python-API v2.4 upward via the remote flag. A simulator
with the start_simulation() method in its main() can then be called e.g. via
pythonsimulator_mosaik–rHOST:PORT
in the command line. The mosaik scenario, started
independently, can then connect to the simulator via the statement connect: HOST:PORT
in its “sim_config”
( →Configuration).
Note that it may make sense to introduce a short waiting
time into your scenario to give you enough time to start both processes. Alternatively,
the remote connection of simulators supports also a timeout (via the timeout flag,
e.g. –t 60 in the command line call will cause your simulator to wait for 60 seconds
for an initial message from mosaik).
We have now implemented the mosaik Sim-API for our simulator. The following
listing combines all the bits explained above:
# simulator_mosaik.py"""Mosaik interface for the example simulator."""importmosaik_api_v3importexample_modelMETA={'type':'hybrid','models':{'ExampleModel':{'public':True,'params':['init_val'],'attrs':['delta','val'],'trigger':['delta'],},},}classExampleSim(mosaik_api_v3.Simulator):def__init__(self):super().__init__(META)self.eid_prefix='Model_'self.entities={}# Maps EIDs to model instances/entitiesself.time=0definit(self,sid,time_resolution,eid_prefix=None):iffloat(time_resolution)!=1.:raiseValueError('ExampleSim only supports time_resolution=1., but'' %s was set.'%time_resolution)ifeid_prefixisnotNone:self.eid_prefix=eid_prefixreturnself.metadefcreate(self,num,model,init_val):next_eid=len(self.entities)entities=[]foriinrange(next_eid,next_eid+num):model_instance=example_model.Model(init_val)eid='%s%d'%(self.eid_prefix,i)self.entities[eid]=model_instanceentities.append({'eid':eid,'type':model})returnentitiesdefstep(self,time,inputs,max_advance):self.time=time# Check for new delta and do step for each model instance:foreid,model_instanceinself.entities.items():ifeidininputs:attrs=inputs[eid]forattr,valuesinattrs.items():new_delta=sum(values.values())model_instance.delta=new_deltamodel_instance.step()returntime+1# Step size is 1 seconddefget_data(self,outputs):data={}foreid,attrsinoutputs.items():model=self.entities[eid]data['time']=self.timedata[eid]={}forattrinattrs:ifattrnotinself.meta['models']['ExampleModel']['attrs']:raiseValueError('Unknown output attribute: %s'%attr)# Get model.val or model.delta:data[eid][attr]=getattr(model,attr)returndatadefmain():returnmosaik_api_v3.start_simulation(ExampleSim())if__name__=='__main__':main()
We can now start to write our first scenario, which we will do in the next
section.
We will now create a simple scenario with mosaik in which we use
a simple data collector to print some output from our simulation. That
means, we will instantiate a few ExampleModels and a data monitor. We will
then connect the model instances to that monitor and simulate that for some
time.
You should define the most important configuration values for your simulation
as “constants” on top of your scenario file. This makes it easier to
see what’s going on and change the parameter values.
Two of the most important parameters that you need in almost every simulation
are the simulator configuration and the duration of your simulation:
The sim config specifies which simulators are available and how to start
them. In the example above, we list our ExampleSim as well as Collector (the
names are arbitrarily chosen). For each simulator listed, we also specify how
to start it. (If you are using type checking, you can import SimConfig from
mosaik.scenario and change the first line to SIM_CONFIG:SimConfig={,
instead.)
Since our example simulator is, like mosaik, written in Python 3, mosaik can
just import it and execute it in-process. The line 'python':'simulator_mosaik:ExampleSim' tells mosaik to import the package
simulator_mosaik and instantiate the class ExampleSim from it.
The data collector will be started as external process which will communicate
with mosaik via sockets. The line 'cmd':'%(python)scollector.py%(addr)s'
tells mosaik to start the simulator by executing the command pythoncollector.py. Beforehand, mosaik replaces the placeholder %(python)s
with the current python interpreter (the same as used to execute the scenario
script) and %(addr)s with its actual socket address HOSTNAME:PORT so that
the simulator knows where to connect to.
The section about the Sim Manager explains all this in
detail.
Here is the complete file of the data collector. Take care of adding it to your example:
"""A simple data collector that prints all data when the simulation finishes."""importcollectionsimportmosaik_api_v3META={'type':'event-based','models':{'Monitor':{'public':True,'any_inputs':True,'params':[],'attrs':[],},},}classCollector(mosaik_api_v3.Simulator):def__init__(self):super().__init__(META)self.eid=Noneself.data=collections.defaultdict(lambda:collections.defaultdict(dict))definit(self,sid,time_resolution):returnself.metadefcreate(self,num,model):ifnum>1orself.eidisnotNone:raiseRuntimeError('Can only create one instance of Monitor.')self.eid='Monitor'return[{'eid':self.eid,'type':model}]defstep(self,time,inputs,max_advance):data=inputs.get(self.eid,{})forattr,valuesindata.items():forsrc,valueinvalues.items():self.data[src][attr][time]=valuereturnNonedeffinalize(self):print('Collected data:')forsim,sim_datainsorted(self.data.items()):print('- %s:'%sim)forattr,valuesinsorted(sim_data.items()):print(' - %s: %s'%(attr,values))if__name__=='__main__':mosaik_api_v3.start_simulation(Collector())
As its name suggests it collects all data it receives each step in a
dictionary (including the current simulation time) and simply prints
everything at the end of the simulation.
The next thing we do is instantiating a World object. This object will
hold all simulation state. It knows which simulators are available and started,
which entities exist and how they are connected. It also provides most of the
functionality that you need for modelling your scenario:
# Create Worldworld=mosaik.World(SIM_CONFIG)
To get access to mosaik’s world, we need to import mosaik at the beginning
of our scenario script. We also import mosaik.util to get access to some
helper methods later on.
Before we can instantiate any simulation models, we first need to start the
respective simulators. This can be done by calling World.start. It
takes the name of the simulator to start and, optionally, some simulator
parameters which will be passed to the simulators init() method. So lets
start the example simulator and the data collector:
We also set the eid_prefix for our example simulator. What gets returned by
World.start is called a model factory.
We can use this factory object to create model instances within the respective
simulator. In your scenario, such an instance is represented as an
Entity. The model factory presents the available models as if they
were classes within the factory’s namespace. So this is how we can create one
instance of our example model and one ‘Monitor’ instance:
The method World.connect takes one entity pair – the source and the
destination entity, as well as a list of attributes or attribute tuples. If you
only provide single attribute names, mosaik assumes that the source and
destination use the same attribute name. If they differ, you can instead pass
a tuple like ('val_out','val_in').
Quite often, you will neither create single entities nor connect single entity
pairs, but work with large(r) sets of entities. Mosaik allows you to easily
create multiple entities with the same parameters at once. It also provides
some utility functions for connecting sets of entities with each other. So lets
create two more entities and connect them to our monitor:
# Create more entitiesmore_models=examplesim.ExampleModel.create(2,init_val=3)mosaik.util.connect_many_to_one(world,more_models,monitor,'val','delta')
Instead of instantiating the example model directly, we called its static
method create() and passed the number of instances to it. It returns a list
of entities (two in this case). We used the utility function
mosaik.util.connect_many_to_one to connect all of them to the
database. This function has a similar signature as World.connect, but
the first two parameters are a world instance and a set (or list) of entities
that are all connected to the dest_entity.
Mosaik also provides the function mosaik.util.connect_randomly. This
method randomly connects one set of entities to another set. These two methods
should cover most use cases. For more special ones, you can implement custom
functions based on the primitive World.connect.
This section introduced you to the basic of scenario creation in mosaik. For
more details you can check the guide to scenarios.
For your convenience, here is the complete scenario that we created in this
tutorial. You can use this for some more experiments before continuing with
this tutorial:
Now that we integrated our first simulator into mosaik and tested it in
a simple scenario, we should implement a control mechanism and mess around with our
example simulator a little bit.
As you remember, our example models had a value to which they added something
in each step. Eventually, their value will end up being very high. We’ll use
a multi-agent system to keep the values of our models in [-3, 3]. The agents
will monitor the current value of their respective models and when it reaches
-3/3, they will set delta to 1/-1 for their model.
Implementing the Sim API for control strategies is very similar to implementing
it for normal simulators. We start again by importing the mosaik_api_v3
package and defining the simulator meta data:
We set the type of the simulator to ‘event-based’. As we have learned, this
has two main implications:
1. Whenever another simulator provides new input for the simulator, a step is
triggered (at the output time). So we don’t need to take care of the
synchronisation of the models and agents. As our example simulator is of type
time-based, it is only stepped at its self-defined times and will thus not
be triggered by (potential) outputs of the agents. It will receive any output
of the agents in its subsequent step.
2. The provision of output of event-based simulators is optional. So if there’s
nothing to report at a specific step, the attributes can (and should be) omitted
in the get_data’s return dictionary.
Our control mechanism will use agents to control other entities. The agent has
no parameters and two attributes, the input ‘val_in’ and the output ‘delta’.
Again, nothing special is going on here. We pass our meta data dictionary to
our super class and set an empty list for our agents.
Because our agents don’t have an internal concept of time, we don’t need to take
care of the time_resolution of the scenario. And as there aren’t any simulator
parameters either, we don’t need to implement
init. The default implementation will return
the meta data, so there’s nothing we need to do in this case.
Every agent gets an ID like “Agent_*<num>*”. Because there might be multiple
create calls, we need to keep track of how many
agents we already created in order to generate correct entity IDs. We also
create a list of {‘eid’: ‘Agent_<num>’, ‘type’: ‘Agent’} dictionaries for
mosaik.
You may have noticed that we, in contrast to our example simulator, did not
actually instantiate any real simulation models this time. We just pretend to
do it. This okay, since we’ll implement the agent’s “intelligence” directly in
step:
defstep(self,time,inputs,max_advance):self.time=timedata={}foragent_eid,attrsininputs.items():delta_dict=attrs.get('delta',{})iflen(delta_dict)>0:data[agent_eid]={'delta':list(delta_dict.values())[0]}continuevalues_dict=attrs.get('val_in',{})iflen(values_dict)!=1:raiseRuntimeError('Only one ingoing connection allowed per ''agent, but "%s" has %i.'%(agent_eid,len(values_dict)))value=list(values_dict.values())[0]
The inputs arguments is a nested dictionary and will look like this:
For each agent, there’s a dictionary with all input attributes (in this case
only ‘val_in’), containing the source entities (their full_id) with the
corresponding values as key-value pairs.
First we initialize an empty data dict that will contain the set-points
that our control mechanism is creating for the models of the example simulator.
We’ll fill this dict in the following loop. We iterate over all agents and
extract its input ‘val_in’; so values_dict is a dict containing the current
values of all models connected to that agent. In our example we only allow to
connect one model per agent, and fetch its value.
If the value is ≤ -3 or ≥ 3, we have to set a new delta value. Else, we don’t
need to do anything and can continue with a new iteration of the loop.
If we have a new delta, we add it to the data dict:
data[agent_eid]={'delta':delta}
After finishing the loop, the data dict may look like this:
{'Agent_0':{'delta':1},'Agent_2':{'delta':-1},}
Agent_0 sets the new delta = 1, and Agent_2 sets the new delta = -1.
Agent_1 did not set a new delta.
At the end of the step, we put the data dict to the class attribute self.data,
to make it accessible in the get_data method
self.data=data
We return None to mosaik, as we don’t want to step ourself, but only when
the controlled models provide new values.
returnNone
After having called step, mosaik requests the new set-points via the get_data
function. In principle we could just return the self.data dictionary, as we
already constructed that in the adequate format. For illustrative purposes we
do it manually anyhow. Additionally, if we do it like that, we can only send
back the attributes that are actually needed by (connected to) other
simulators:
Here is the complete code for our (very simple) controller / mutli-agent
system:
# controller.py"""A simple demo controller."""importmosaik_api_v3META={'type':'event-based','models':{'Agent':{'public':True,'params':[],'attrs':['val_in','delta'],},},}classController(mosaik_api_v3.Simulator):def__init__(self):super().__init__(META)self.agents=[]self.data={}self.time=0defcreate(self,num,model):n_agents=len(self.agents)entities=[]foriinrange(n_agents,n_agents+num):eid='Agent_%d'%iself.agents.append(eid)entities.append({'eid':eid,'type':model})returnentitiesdefstep(self,time,inputs,max_advance):self.time=timedata={}foragent_eid,attrsininputs.items():delta_dict=attrs.get('delta',{})iflen(delta_dict)>0:data[agent_eid]={'delta':list(delta_dict.values())[0]}continuevalues_dict=attrs.get('val_in',{})iflen(values_dict)!=1:raiseRuntimeError('Only one ingoing connection allowed per ''agent, but "%s" has %i.'%(agent_eid,len(values_dict)))value=list(values_dict.values())[0]ifvalue>=3:delta=-1elifvalue<=-3:delta=1else:continuedata[agent_eid]={'delta':delta}self.data=datareturnNonedefget_data(self,outputs):data={}foragent_eid,attrsinoutputs.items():forattrinattrs:ifattr!='delta':raiseValueError('Unknown output attribute "%s"'%attr)ifagent_eidinself.data:data['time']=self.timedata.setdefault(agent_eid,{})[attr]=self.data[agent_eid][attr]returndatadefmain():returnmosaik_api_v3.start_simulation(Controller())if__name__=='__main__':main()
Next, we’ll create a new scenario to test our controller.
The scenario that we’re going to create in this part of the tutorial will
be similar to the one we created before but incorporate the control mechanism
that we just created.
Again, we start by setting some configuration values and creating a simulation
world:
We use a list comprehension to create three model instances with individual
initial values (-2, 0 and 2). For instantiating the same number of agent
instances we use create() which does the same as a list comprehension but
is a bit shorter.
Finally we establish pairwise bi-directional connections between the models
and the agents:
The important thing here is the weak=True argument that we pass to the
second connection. This tells mosaik how to resolve the cyclic dependency, i.e.
which simulator should be stepped first in case that both simulators have a
scheduled step at the same time. (In our example this will not happen, as the
agents are only stepped by the models’ outputs.)
Finally, we can connect the models and the agents to the monitor and run the simulation:
# Connect to monitormosaik.util.connect_many_to_one(world,models,monitor,'val','delta')mosaik.util.connect_many_to_one(world,agents,monitor,'delta')# Run simulationworld.run(until=END)
In the printed output of the collector, you can see two important things: The
first is that the agents only provide output when the delta of the controlled
model is to be changed. And second, that the new delta is set at the models’
subsequent step after it has been derived by the agents.
Important use cases for same-time loops can be the initialization of simulation and communication between controllers or agents.
As the scenario definition has to provide initialization values for cyclic data-flows and every cyclic data-flow will lead to an incrementing simulation time, it may take some simulation steps until all simulation components are in a stable state, especially, for simulations consisting of multiple physical systems.
The communication between controllers or agents usually takes place at a different time scale than the simulation of the technical systems.
Thus, same-time loops can be helpful to model this behavior in a realistic way.
To give an example of same-time loops in mosaik, the previously shown scenario is extended with a master controller, which takes control over the other controllers.
The communication between these two layers of controllers will take place in the same step without incrementing the simulation time.
The code of the previous scenario is used as a base and extended as shown in the following.
The master controller bases on the code of the controller of the previous scenario.
The first small change for the master controller is in the meta data dictionary, where new attribute names are defined.
The ‘delta_in’ represent the delta values of the controllers, which will be limited by the master controller.
The results of this control function will be returned to the controllers as ‘delta_out’.
The step is changed, so that first the current time is updated in the self.time variable.
Also the control function is changed.
The master controller gets the delta output of the other controllers as ‘delta_in’ and stores the last value of each controller in the self.cache.
This is needed, because the controllers are event-based and the current values are only sent if the values changes.
The control function of the master controller limits the sum of all deltas to be <1 and >-1.
If these limits are exceeded the delta of all controllers will be overwritten by the master controller with 0 and sent to the other controller as ‘delta_out’.
Additionally, two small changes in the get_data method were done.
First, the name was updated to ‘delta_out’ in the check for the correct attribute name.
Second, the current time, which was stored previously in the step, is added to the output cache dictionary.
This informs mosaik that the simulation should start or stay in a same-time loop if also output data for ‘delta_out’ is provided.
The controller has to be extended to handle the ‘delta_out’ from the master controller as input.
If it receives an input value for the attribute ‘delta’, it will not calculate a new delta value, but use the one from the master controller.
The same-time loop in this scenario will always be finished after the second iteration, because the master controller will overwrite the deltas of the controller and will get back zeros as ‘delta_in’.
Thus, it will produce no output in the second iteration and the same-time loop will be finished.
This scenario is based on the previous scenario.
In the following description only the changes are explained, but the full code is shown.
The updated controller and the new master controller are added to the sim config of the scenario.
# demo_3.pyimportmosaikimportmosaik.util# Sim config. and other parametersSIM_CONFIG={'ExampleSim':{'python':'simulator_mosaik:ExampleSim',},'ExampleCtrl':{'python':'controller_demo_3:Controller',},'ExampleMasterCtrl':{'python':'controller_master:Controller',},'Collector':{'cmd':'%(python)s collector.py %(addr)s',},}END=6# 10 seconds# Create Worldworld=mosaik.World(SIM_CONFIG)
The master controller is also started and initialized.
The controllers get different ‘init_val’ values compared to the previous scenario.
Here, it is changed to (-2,0,-2) to have the right timing to get into the same-time loop.
The ‘delta’ outputs of the controllers are connected to the new master controller and the ‘delta_out’ of the master controller is connected to the respective controller.
The weak=True argument defines, that the connection from the controllers to the master controller will be the first to be executed by mosaik.
# Connect entitiesformodel,agentinzip(models,agents):world.connect(model,agent,('val','val_in'))world.connect(agent,model,'delta',weak=True)foragentinagents:world.connect(agent,master_agent[0],('delta','delta_in'))world.connect(master_agent[0],agent,('delta_out','delta'),weak=True)mosaik.util.connect_many_to_one(world,models,monitor,'val','delta')mosaik.util.connect_many_to_one(world,agents,monitor,'delta')world.connect(master_agent[0],monitor,'delta_out')# Run simulationworld.run(until=END)
The printed output of the collector shows the states of the different simulators.
The collector just shows the final result of the same-time loop and not the steps during the loop.
It can be seen that the ‘delta’ of ‘Agent_1’ changes to -1 at time step 2 and at time step 4 all ‘delta’ attributes are set to 0 by the master controller.
A visualization of the execution graph shows the data flows in the simulation.
For the first two time steps, only the controllers are executed, as they do not provide any output for ‘delta’.
Thus, the master controller was not stepped and the simulation was proceeded directly with the next simulation time step.
At simulation time 2, the master controller is stepped, but as the sum of delta values is not exceeding the limits no control action takes place.
At simulation time 4, the master controller is stepped again and this time sends back a value to the controllers to limit their ‘delta’ value.
It can be seen, that the controllers are stepped a second time within the same simulation time and send data again to the master controller.
After this second step of the master controller, it does not send an output again and the simulation proceeds to simulation time 5, where the same-time loop occures again.
This tutorial gives an example on how to set external events for integrating unforeseen
interactions of an external system in soft real-time simulation with rt_factor=1.0.
A typical use case for this feature would be Human-in-the-Loop simulations to support human interactions, e.g., control actions.
In mosaik, such external events can be implemented via the the asynchronous set_event method.
These events will then be scheduled for the next simulation time step.
To give an example of external events in mosaik, a new scenario is created that includes a controller to set external events.
In addition to the controller, a graphical user interface (GUI) is implemented and started in a subprocess for external control actions by the user.
The example code and additional requirements are shown in the following.
First of all, we need to install some additional requirements within the virtual environment
(see installation guide for setting up a virtual environment)
The set-event controller subscribes to external events from the GUI via a zeromq subscriber socket
using the publish-subscribe pattern.
Herefore, a listener thread is created which receives external event messages from the GUI.
More information about the listener thread can be found in the next section.
classController(mosaik_api_v3.Simulator):def__init__(self):super().__init__(META)self.data={}self.time=0self.eid=Noneself.thread=Noneself.initial_timestamp=0self.once=Trueself.context=zmq.Context()# Subscribe to external events from the GUIself.subscriber=self.context.socket(zmq.SUB)self.subscriber.connect("tcp://localhost:5563")self.subscriber.setsockopt(zmq.SUBSCRIBE,b"B")# Listener THREADself.thread=listen_to_external_events(self)defcreate(self,num,model):ifnum>1orself.eidisnotNone:raiseRuntimeError('Can only create one instance of Controller.')self.eid='Controller_set_event'return[{'eid':self.eid,'type':model}]deffinalize(self):self.thread.join(0)sys.exit()
In order to set the event for the next time step, it is necessary to determine the current simulation time in wall clock time.
For this, we need to store the initial timestamp in step once for the first simulation step.
defstep(self,time,inputs,max_advance):# Needed in listener thread to determine the current simulation time in wall clock time.ifself.once:self.initial_timestamp=self.mosaik.world.env.nowself.once=Falseself.time=timeprint(f"In step at time {self.time}")print(f"max_advance {max_advance}")returnNone
The listener thread can be included in the same file as the set-event controller: controller_set_event.py.
The object of the controller class needs to be passed as a parameter to the listen_to_external_events function,
which is called as a thread via the defined decorator @threaded.
The listener thread listens to external event messages from the GUI. Once a message arrives,
the listener thread calls the set_event method to set an external event for the next simulation step in mosaik.
@threadeddeflisten_to_external_events(controller):whileTrue:try:# Receive external event message from GUI[address,contents]=controller.subscriber.recv_multipart(zmq.NOBLOCK)print(f"[{address}] {contents}")current_timestamp=controller.mosaik.world.env.nowreal_time=math.ceil(current_timestamp-controller.initial_timestamp)event_time=real_time+1print(f"Current simulation time: {real_time}")ifcontroller.time<event_time<controller.mosaik.world.until:print(f"Set external Event at time {event_time}")# Set external event in mosaik via asynchronous callcontroller.mosaik.set_event(event_time)exceptzmq.ZMQErrorase:ife.errno==zmq.EAGAIN:# state changed since poll eventpasselse:raise
For the GUI, we create a new python module, e.g., gui_button.py.
The GUI is created with PyQt5 and provides a button to set external events in mosaik every time we click on it.
To enable the set-event controller to perform this control action, a zeromq publisher socket is used
to send a message to the controller’s subscriber that the button has been clicked.
# gui_button.pyimportsysimportzmqfromPyQt5importQtWidgetsfromPyQt5.QtWidgetsimportQApplication,QMainWindowclassPushButtonWindow(QMainWindow):def__init__(self):super(PushButtonWindow,self).__init__()self.button=Noneself.context=zmq.Context()# For external eventsself.publisher=self.context.socket(zmq.PUB)self.publisher.bind("tcp://*:5563")defbutton_clicked(self):self.publisher.send_multipart([b"B",b"Push button was clicked!"])defcreate(self):self.setWindowTitle("MOSAIK 3.0 - External Events")self.button=QtWidgets.QPushButton(self)self.button.setText("Click me to set an external event!")self.button.clicked.connect(self.button_clicked)# Set the central widget of the Window.self.setCentralWidget(self.button)defmain():app=QApplication(sys.argv)window=PushButtonWindow()window.create()window.show()sys.exit(app.exec_())if__name__=="__main__":main()
The set-event controller is started and initialized. Here, an initial event is added to the set-event controller
so that the controller is executed at time=0 to set the initial timestamp. This is needed for the determination of the current simulation time.
The GUI is started in a subprocess and must be manually closed after the simulation is completed.
# Start GUI in a subprocessproc=subprocess.Popen(['python','gui_button.py'])
In order to run the simulation scenario in soft real-time, the rt_factor is set to 1.0.
# Run simulation in real-timeworld.run(until=END,rt_factor=1.0)
Finally, we can run the scenario script as follows:
$pythondemo_4.py
The printed output shows when the external events are triggered (button was clicked) and executed during simulation.
Starting"Controller"as"Controller-0"...WARNING:Controller-0hasnoconnections.Startingsimulation.Instepattime0max_advance60Simulationtooslowforreal-timefactor1.0-9.655498433858156e-05sbehindtime.[b'B']b'Push button was clicked!'Currentsimulationtime:11SetexternalEventattime12Instepattime12max_advance60Simulationtooslowforreal-timefactor1.0-0.000688756990712136sbehindtime.[b'B']b'Push button was clicked!'Currentsimulationtime:16SetexternalEventattime17Instepattime17max_advance60Simulationtooslowforreal-timefactor1.0-0.0013458110042847693sbehindtime.[b'B']b'Push button was clicked!'Currentsimulationtime:26SetexternalEventattime27Instepattime27max_advance60Simulationtooslowforreal-timefactor1.0-0.0013047059765085578sbehindtime.[b'B']b'Push button was clicked!'Currentsimulationtime:29SetexternalEventattime30Instepattime30max_advance60Simulationtooslowforreal-timefactor1.0-0.0019755829707719386sbehindtime.[b'B']b'Push button was clicked!'Currentsimulationtime:33SetexternalEventattime34Instepattime34max_advance60Simulationtooslowforreal-timefactor1.0-0.0011994789820164442sbehindtime.Simulationfinishedsuccessfully.
Odysseus is
a framework for in-memory data stream management that is designed for
online processing of big data. Large volumes of data such as continuously
occurring events or sensor data can be processed in real time. In combination
with mosaik Odysseus can be used to process, visualise and store the results of
mosaik during a simulation.
In this first part of the tutorial we cover the two ways to connect mosaik and
Odysseus, the second part is about how
to use Odysseus to process, visualize and store simulation data.
Note
Connecting mosaik and Odysseus works mosaik >= 3.0
Note
Connecting mosaik and Odysseus works only with mosaik >= 3.0
You can choose between two different solutions to connect mosaik and Odysseus.
Both have their advantages and disadvantages and therefore, the right choice depends on your use case.
We recommend to use the SimAPI version for beginners.
No matter which connection we use, we first have to
download Odysseus Server and Studio Client.
For the first start of Odysseus Studio the default user “System” and password “manager” have to be used,
the tenant can be left empty.
The easiest way to connect to mosaik is to use the mosaik protocol handler in
Odysseus, which is available as installable feature in Odysseus Studio. It uses
the mosaik API through remote procedure calls (RPC) and offers a close coupling
of mosaik and Odysseus. With this, a blocked simulation in mosaik or a blocked
processing in Odysseus will block the other system as well. If this is a problem
in your use case, you should look in the section Connecting via ZeroMQ.
First we have to install the mosaik feature
from the incubcation site in odysseus, which can be found in the Odysseus Wrapper Plugins.
After installing the feature we create a new Odysseus project and in the
project a new Odysseus script file (more information on Odysseus projects and script files can be found in this
tutorial).
To use mosaik as source we can
use the mosaik operator which contains a standard configuration of mandatory parameters.
The script-code in the Odysseus query language PQL
looks like this:
As we can see the protocol ‘mosaik’ is chosen. When the query is started, the mosaik
protocol handler in Odysseus opens a TCP server for receiving data from mosaik.
Before we can receive data, we have to adapt our mosaik scenario.
Here we take the mosaik-demo as an example. The Odysseus simulator is treated
just like any other component in mosaik. It has to be added to the SIM_CONFIG parameter.
For the connection to the simulator the connect command is used and the IP
address and port of Odysseus have to be specified:
After that, we have to initialize the simulator and connect it to all components whose data we want to revceive in Odysseus.
For the mosaik-demo, we have to add the following lines of code to the scenario definition:
Now we have set up everything to receive mosaiks data in Odysseus.
To begin transfering data we have to start first the query in Odysseus and then the simulation in mosaik.
For more information on how to use Odysseus visit part two.
In contrast to the close coupling via mosaik protocol handler the coupling via
ZeroMQ is more loose.
Mosaik sends all data as data stream with ZeroMQ and Odysseus can even be closed
and restarted during the simulation without affecting mosaik.
This behaviour holds the risk of loosing data so it should only be used if this
doesn’t cause problems.
First we have to install the following features
for Odysseus from incubation site:
Odysseus Wrapper Plugins / Zero MQ
Odysseus Wrapper Plugins / mosaik (only if you want to use the mosaik operator)
And from the update site:
Odysseus Odysseus_core Plugins / Json Wrapper
After installing the features we create a new Odysseus project
and in the project a new Odysseus script file.
The messages sent by mosaik are formatted in JSON format and sent via ZeroMQ.
So we have to choose the corresponding ZeroMQ transport handler and JSON protocol handler:
After setting up Odysseus we have to install the mosaik-zmq adapter in our mosaik virtualenv.
It is available on GitLab and PyPI.
To install it we have to activate our mosaik virtualenv and execute (if there are errors during installation have a look in the
readme):
pipinstallmosaik-zmq
The mosaik-zmq adapter is treated in mosaik like any other component of the simulation.
If we use the mosaik demo, we have to add the new simulator to
the SIM_CONFIG parameter:
sim_config={'ZMQ':{'cmd':'mosaik-zmq %(addr)s',},
Also we have to initialize the ZeroMQ simulator and connect it to other components:
For more information on how to use Odysseus visit part two.
Using Odysseus to process, visualize and store simulation data
This tutorial will give some examples on how you can use Odysseus to process,
visualize and store the data from mosaik. More information about connecting
mosaik and Odysseus can be found in the first part of the
tutorial and more about Odysseus in general can be found in its
documentation.
If you have no experience with Odysseus you should first visit the tutorials in
its documentation. Simple query processing
and selection, projection and map
should explain the basics.
Mosaik sends data in JSON format and so the key-value-object has to be used as datatype for receiving in Odysseus.
But most operators in Odysseus are based on relational tuples with a fixed schema,
so it can be useful to transform arriving key-value objects to relational tuples.
For this the totuple operator can be used.
It creates relational tuples with the given attributes and omitts all data, which is not included in the schema:
We can also add computations to the data with a map operator.
The expressions parameter contains first the computation and second the new name for every attribute.
In this example the deviation of voltage to the nominal voltage of 230 V is calculated (more information about the offered functions can be found here):
By using the aggregate
operator we are able to calculate e.g. the average values.
We have to add an timewindow
operator first to have the right timestamps for aggregating.
To visualize data in Odysseus dashboards can be used, which can contain different graphs.
For the data stream shown in the section above an exemplary dashboard could look like the following picture:
More information about dashboards in Odysseus can be found in the documentation.
What do we do if we want to connect a simulator to mosaik which ist written in Java? In this tutorial
we will describe how to create a simple model in Java and integrate it into mosaik using the
mosaik-Java high level API. We will
do this with the help of our simple model from the Python tutorial,
i. e. we will try to replicate the first part of the Python tutorial as close as possible.
If you want to compile the jar yourself, you have to get the sources of the mosaik-java-API for Java
which is provided on Gitlab. Clone it and put it in the development environment of your choice.
Next we create a Java class for our model. Our example model has exact the same behaviour
as our simple model in the Python tutorial. To distinguish it
from the Python-model we call it JModel. The only difference to the Python-model
is that in Java we need two constructors (with and without init value) and getter
and setter methods to access the variables val and delta.
A simulator provides the functionality that is necessary to manages instances of our model and to
execute the models. We need a method addModel to create instances of our model and an ArrayList
models to store them. The step method executes a simulation for each model instance. To access
the values and deltas we need getter and setter methods. In our example the class
implementing these functionalities is called JSimulator.
Finally we need to implement the mosaik-API methods. In our example this is
done in a class called JExampleSim. This class has to extent the abstract class
Simulator from mosaik-java-api which is the Java-equivalent to the
Simulator class in Python. The class Simulator provides
the four mosaik-API-calls init(), create(), step(), and getData()
which we have to implement. For a more detailed explanation of the API-calls see the
API-documentation.
But first we have to put together the meta-data containing information
about models, attributes, and parameters of our simulator. ‘models’ are all models our simulator
provides. In our case this is only JModel. ‘public’: true tells mosaik that it is allowed to
create models of this class. ‘params’ are parameter that are passed during initialisation, in our
case this is init_val. ‘attrs’ is a list of values that can be exchanged.
First method is init() that returns the
meta data. In addition it is possible to pass arguments for initialization. In our
case there is eid_prefix which will be used to name instances of the models:
create() creates new instances of the model JModel by calling the add_model-method of
JSimulator. It also assigns ID (eid) to the models, so that it is able to keep
track of them. It has to return a list with the name (eid) and type of the models. You can find
more details about the return object in the API-documentation.
step() tells the simulator to perform a simulation step. It passes the time, the current
simulation time, and inputs, a JSON data object with input data from preceding simulators. The
structure of inputs is explained in the API-documentation. If there are
new delta-values in inputs they are set in the appropriate model instance. Finally it
calls the simulator’s step()-method which, on its part, calls the step()-methods
of the individual model-instances.
publiclongstep(longtime,Map<String,Object>inputs){//go through entities in inputsfor(Map.Entry<String,Object>entity:inputs.entrySet()){//get attrs from entityMap<String,Object>attrs=(Map<String,Object>)entity.getValue();//go through attrs of the entityfor(Map.Entry<String,Object>attr:attrs.entrySet()){//check if there is a new deltaStringattrName=attr.getKey();if(attrName.equals("delta")){//sum up deltas from different sourcesObject[]values=((Map<String,Object>)attr.getValue()).values().toArray();floatvalue=0;for(inti=0;i<values.length;i++){value+=((Number)values[i]).floatValue();}//set delta Stringeid=entity.getKey();intidx=this.entities.get(eid);this.simulator.set_delta(idx,value);}}}//call step-methodthis.simulator.step();returntime+this.stepSize;}
getData() gets the simulator’s output data from the last simulation step. It
passes outputs, a JSON data object that describes which parameters are requested.
getData() goes through outputs, retrieves the requested values from the appropriate
instances of JModel and puts it in data. The structure of outputs and data is
explained in the API-documentation.
publicMap<String,Object>getData(Map<String,List<String>>outputs){Map<String,Object>data=newHashMap<String,Object>();//*outputs* lists the models and the output values that are requested//go through entities in outputsfor(Map.Entry<String,List<String>>entity:outputs.entrySet()){Stringeid=entity.getKey();List<String>attrs=entity.getValue();HashMap<String,Object>values=newHashMap<String,Object>();intidx=this.entities.get(eid);//go through attrs of the entityfor(Stringattr:attrs){if(attr.equals("val")){values.put(attr,this.simulator.get_val(idx));}elseif(attr.equals("delta")){values.put(attr,this.simulator.get_delta(idx));}}data.put(eid,values);}returndata;}
We use the same scenario as in our Python example demo1.
The only thing we have to change is the way we connect our simulator to mosaik.
There are two ways to do this:
cmd: mosaik calls the Java API by executing the command given in cmd. Mosaik starts the
Java-API in a new process and connects it to mosaik. This works only if your simulator
runs on the same machine as mosaik.
connect: mosaik connects to the Java-API which runs as a TCP server. This works also
if mosaik and the simulator are running on different machines.
For more details about how to connect simulators to mosaik see the section about the
Sim Manager in the mosaik-documentation.
We have to give mosaik the command how to start our Java simulator. This is done in
SIM_CONFIG. The marked lines show the differences to our Python simulator.
# Sim config. and other parametersSIM_CONFIG={'JExampleSim':{'cmd':'java -cp JExampleSim.jar de.offis.mosaik.api.JExampleSim %(addr)s',},'Collector':{'cmd':'python collector.py %(addr)s',},}END=10*60# 10 minutes
The placeholder %(addr)s is later replaced with IP address and port by mosaik. If we now
execute demo_1.py we get the same output as in our Python-example.
Note
The command how to start the Java simulator may differ depending on your operating system.
If the command is complex, e. g. if it contains several libraries, it is usually better
to put it in a script and than call the script in cmd.
In this case the Java API acts as TCP server and listens at the given address
and port. Let’s say the simulator runs on a computer with the IP-address 1.2.3.4.
We can now choose a port that is not assigned by default.
In our example we choose port 5678. Make sure that IP-address and port is accessible from
the computer that hosts mosaik (firewalls etc.).
Note
Of course you can run mosaik and your simulator on the same machine by using
127.0.0.1:5678 (localhost). You may want to do this for testing and experimenting.
Apart from that the connection with cmd (see above) is usually
the better alternative because you don’t have to start the Java part separately.
We have to tell mosaik how to connect to the simulator. This is done in SIM_CONFIG
in our scenario (demo1):
# Sim config. and other parametersSIM_CONFIG={'JExampleSim':{'connect':'1.2.3.4:5678',},'Collector':{'cmd':'python collector.py %(addr)s',},}END=10*60# 10 minutes
The marked lines show the differences to our Python simulator. Our simulator is now called JExampleSim
and we need to give the simulator’s address and port after the connect key word.
Now we start JExampleSim. To tell the mosaik-Java-API to run as TCP-server is done
by starting it with “server” as second argument. The first command line argument is IP-address and port.
The command line in our example looks like this:
If we now execute demo_1.py we get the same output as in
our Python-example.
Note
You can find the source code used in this tutorial in the
mosaik-source-files in the folder
docs/tutorial/code.
Java Generics API tutorial
Integrating a Model in Java with Generics / Annotations API
Additionally to the basic mosaik Java API, another API based on the basic one can be used, called mosaik-java-api-generics,
that contains some quality of life changes, for example automatic meta-model generation based on your model,
automatic model instantiation, input parsing via generics and automatic gathering of data for mosaik.
Note
When to use basic Java API and when to use this API? If you want to use the flexibility from Python, for example using all inputs. If you want to have a more type strict Java-like experience, use this API instead.
This tutorial is also based on the Python tutorial Python tutorial for the simple model and shows how to use the most features of this API.
If you want to compile the jar yourself, you have to get the sources of the mosaik-java-generics-API for Java
which is provided on Gitlab. Clone it and put it in the development environment of your choice.
The example model is again a replication from model.
This model is annotated with @Model, which will tell the parser that this is a model for mosaik.
Normally, the simple class name will be used as model name. This can be customized by setting the value of the annotation.
The model annotation has several different sub-annotations that are useful to customize the behaviour of the of the model parser and
automatic class instantiation.
Some are also needed for some java versions to work, namely where constructor parameter names are not set.
Note
All fields MUST be a non-primitive type. This is needed for internal purposes for the input data parsing.
Else it could be possible that you read input values (for example 0 as demand) that were never set.
Annotated models, that are added to a simulator, will be searched for public fields and getter,
which will then be transformed into attributes.
For the mosaik params, a denoted constructor parameters will be used.
If more than one constructor are available, the constructor used must be annotated with @Model.Constructor.
You can rename constructor parameter by using @Model.Param(“other_name”).
This is especially useful if parameter naming is turned off (default for basic java).
The following example shows how to use the model annotation for the example model:
@Model.Id: If you want to pass the model id into the model, you can annotate a constructor parameter with this annotation. The parameter will not be included into the meta-model.
@Model.Suppress: If you want to use public getter or fields, which should only be used internally, you can annotate it with Suppress. These getter or fields will not be part of the meta-model created.
The simulator implements an interface to the mosaik API.
This util package has two new simulators you can inherit from.
One for simulators with only one model and one for multi model simulation.
The reason to have two simulators is simpler use of generics for single models compared to multi-model simulations.
The simulators also separate entity model and input model.
Separating the input model from the entity model enables you to use only a subset of the entity model as input.
Note
Input models need a default constructor!
This is due to the input parsing where an initially empty model’s fields will be filled iteratively.
The single model simulators fits perfectly for our small example.
We need to inherit from the ModelSimulator class and set the generic types.
The model generics are first the entity model (the created model instances) and second the input model.
The constructor must call the parent constructor with the simulator name, simulation type and entity and input model.
This will be then used to create the meta-model.
The second method that must be implemented is initialize.
It contains the same parameter as the original init method but without does not expect you to return the meta-model,
as this will be done automatically.
The last mandatory method is the modelStep, an abbreviation of the step method.
The parameter contain time, maxAdvance and parsed input from mosaik.
The input will be parsed into the generic input class passed to the simulator and put into a map which contains the model id and
a collection of InputMessages with sender and input class instance method.
The modelStep method for example will first sum the delta of all controller and then call the step method of the different models.
There are other optional methods that can be overridden.
- finishEntityCreation: Can be used to call auxiliary functions for models or modify the entity models and map itself.
- prepareGetData: Method will be called before the data for get_data will be gathered. Can be used if the mosaik model is only a DTO
- setupDone: Can be used to add functionality when all models are created.
- cleanup: Will be called after the simulation is finished, call super or else the entities will not be cleaned up!
The whole code for the simulator can be found here:
publicclassExampleModelSimextendsModelSimulator<AnnotatedExampleModel,AnnotatedExampleModel>{privateintstepSize=60;publicExampleModelSim()throwsException{super("ExampleSim",SimulationType.TimeBased,AnnotatedExampleModel.class,AnnotatedExampleModel.class);}@Overridepublicvoidinitialize(Stringsid,FloattimeResolution,Map<String,Object>simParams){if(simParams.containsKey("step_size")){this.stepSize=((Number)simParams.get("step_size")).intValue();}}@OverridepublicvoidfinishEntityCreation(Map<String,AnnotatedExampleModel>entities){// Change the entity map or call auxiliary methods here}@OverridepublicvoidsetupDone()throwsException{// Call auxiliary methods after entities are created here}@OverridepubliclongmodelStep(longtime,Map<String,Collection<InputMessage<AnnotatedExampleModel>>>inputs,longmaxAdvance){for(Stringid:inputs.keySet()){Collection<InputMessage<AnnotatedExampleModel>>modelCollection=inputs.get(id);doublesum=modelCollection.stream().mapToDouble(modelInputMessage->modelInputMessage.getInputMessage().getDelta()).sum();getEntities().get(id).setDelta(sum);}for(AnnotatedExampleModelinstance:getEntities().values()){instance.step();}returntime+this.stepSize;}}
The second simulator is designed to be able to simulate multiple models.
While this guide only covers an example with one model, the basic concept of this simulator should come clear.
Instead of using generics directly in the class, the simulator contains auxiliary methods to register methods for:
- Model registration
- Model creation finalization
- Input data processing
To show how to use a different input model, the following model is used in this example:
@ModelpublicclassMessageModel{publicDoubledelta;}
We first need to inherit from MultiModelInputSimulator and implement the constructor.
The constructor is the best place to register the models and processing methods.
For our example, we need to register our model and register an input step.
We can additionally register a model finishing method.
You can only register models and methods when no simulation is running. Since there is currently no way to remove registered methods, try to only register them in the constructor.
First call the parent constructor with the simulator name and simulation type.
Additionally, already known entity models can be passed. They could also be registered via registerModel method.
registerStepMethod will take an input model class and a method to process the input data.
Parameters of the method are: time, Map of Map of InputMessage<T>, where T is the passed input model class.
In our method, we will sum all deltas for every model set it for our models.
We then return maxAdvance. This will make the simulationStep the step method which will dictate when we will be called again.
If you have a reason that a registered step method returns a time sooner than the one returned in simulationStep, the soonest time will be returned to mosaik.
registerFinishCreationMethod will take an entity model class and a function which passes a map of entity model instances,
just like the similar single model simulator method, but you need to return the entity map.
Since we have no additionally modifications to make, this method is empty and only there fore display purposes.
privateMap<String,AnnotatedExampleModel>finishModelCreation(Map<String,AnnotatedExampleModel>modelMap){// Put you entity modifications or auxiliary calls herereturnmodelMap;}
Two methods have to be implemented.
First, the initialize method, just like before in the single model simulator example, where the step size of the simulation is set:
Second, the simulationStep method needs to be implemented. Here, we gather all entities of the AnnotatedExampleModel class and call step for every instance.
You can also register methods for entity models before get_data will be called. The procedure is the same as for the other two register methods.
The whole code for the simulator:
publicclassMultiModelExampleSimextendsMultiModelInputSimulator{privateintstepSize=60;publicMultiModelExampleSim(){super("ExampleSim",SimulationType.TimeBased,AnnotatedExampleModel.class);registerStepMethod(MessageModel.class,this::parseInput);registerFinishCreationMethod(AnnotatedExampleModel.class,this::finishModelCreation);}privateMap<String,AnnotatedExampleModel>finishModelCreation(Map<String,AnnotatedExampleModel>modelMap){// Put you entity modifications or auxiliary calls herereturnmodelMap;}@Overridepublicvoidinitialize(Stringsid,FloattimeResolution,Map<String,Object>simParams){if(simParams.containsKey("step_size")){this.stepSize=((Number)simParams.get("step_size")).intValue();}}privatelongparseInput(longtime,Map<String,Map<String,InputMessage<MessageModel>>>inputResult,longmaxAdvance){inputResult.forEach((model,inputMessageMap)->{getEntities(AnnotatedExampleModel.class).get(model).setDelta(inputMessageMap.values().stream().mapToDouble(value->value.getInputMessage().delta).sum());});returnmaxAdvance;}@OverridepubliclongsimulationStep(longtime,Map<String,Object>inputs,longmaxAdvance){getEntities(AnnotatedExampleModel.class).values().forEach(AnnotatedExampleModel::step);returntime+stepSize;}}
We use the same scenario as in our Python example demo1.
The only thing we have to change is the way we connect our simulator to mosaik.
There are two ways to do this:
cmd: mosaik calls the Java API by executing the command given in cmd. Mosaik starts the
Java-API in a new process and connects it to mosaik. This works only if your simulator
runs on the same machine as mosaik.
connect: mosaik connects to the Java-API which runs as a TCP server. This works also
if mosaik and the simulator are running on different machines.
For more details about how to connect simulators to mosaik see the section about the
Sim Manager in the mosaik-documentation.
We have to give mosaik the command how to start our Java simulator. This is done in
SIM_CONFIG. The marked lines show the differences to our Python simulator.
# Sim config. and other parametersSIM_CONFIG={'JExampleSim':{'cmd':'java -cp JExampleSim.jar de.offis.mosaik.api.utils.generics.SimulationStarter %(addr)s',},'Collector':{'cmd':'python collector.py %(addr)s',},}END=10*60# 10 minutes
The placeholder %(addr)s is later replaced with IP address and port by mosaik. If we now
execute demo_1.py we get the same output as in our Python-example.
Note
The command how to start the Java simulator may differ depending on your operating system.
If the command is complex, e. g. if it contains several libraries, it is usually better
to put it in a script and than call the script in cmd.
In this case the Java API acts as TCP server and listens at the given address
and port. Let’s say the simulator runs on a computer with the IP-address 1.2.3.4.
We can now choose a port that is not assigned by default.
In our example we choose port 5678. Make sure that IP-address and port is accessible from
the computer that hosts mosaik (firewalls etc.).
Note
Of course you can run mosaik and your simulator on the same machine by using
127.0.0.1:5678 (localhost). You may want to do this for testing and experimenting.
Apart from that the connection with cmd (see above) is usually
the better alternative because you don’t have to start the Java part separately.
We have to tell mosaik how to connect to the simulator. This is done in SIM_CONFIG
in our scenario (demo1):
# Sim config. and other parametersSIM_CONFIG={'JExampleSim':{'connect':'1.2.3.4:5678',},'Collector':{'cmd':'python collector.py %(addr)s',},}END=10*60# 10 minutes
The marked lines show the differences to our Python simulator. Our simulator is now called JExampleSim
and we need to give the simulator’s address and port after the connect key word.
Now we start JExampleSim. To tell the mosaik-Java-API to run as TCP-server is done
by starting it with “server” as second argument. The first command line argument is IP-address and port.
The command line in our example looks like this:
As last note, the functions used to parse the mosaik data into models and creating the metamodel are available via the ParserHelper class.
Even if you don’t want to use one of the simulators, this class has some QoL methods you may want to use.
Check the inbuilt java-docs of the library to learn more about the methods available and how to use them.
You can use mosaik with Jupyter notebooks to have an interactive experience and to write or use tutorials or to document your steps. We have pre-created tutorials written in Jupyter notebooks which you can use. This page gives a short explanation on how to use those.
Note
Starting from version 3.2, mosaik uses asyncio, which does not support nested event loops natively. This leads to an error when using mosaik in a Jupyter notebook. Luckily, the issue can be resolved by installing the library nest_asyncio and calling
importnest_asyncionest_asyncio.apply()
at the beginning of your notebook (before creating the mosaik World).
Step 1 - Use Jupyter in VS Code: Jupyter notebooks can be used with different UIs. A simple approach is to use Visual Studio Code with its Jupyter extension. You can find detailed information on how to use Jupyter in VS Code in the VS Code documentation.
Step 2 - mosaik Jupyter repository: Checkout our Jupyter notebook examples repository, e.g. via gitclonehttps://gitlab.com/mosaik/examples/mosaik-tutorials-on-binder.git and open the folder in VS Code.
Step 3 - Virtual Environment: Create a virtual environment with the requirements installed. See the following screenshots for an example on how to create a virtual environment.
Step 4 - Run a Jupyter notebook: Choose one of the available notebooks, e.g., _02_simulator_mosaik.ipynb and open it. You can now run the code blocks step by step or all at once with the “Run All” button on the top. Feel free to play with the example code, extend or change it to your needs or to create your own notebooks based on these examples.
Sometimes it is useful to visualize your scenario to understand the behavior of mosaik. You can use the plotting functions in utils for different graphs. The parameters are always the same: the world object and the name of the folder where the figures shall be stored in.
Optional parameters are slice (see below) and show_plot (default: True). With show_plot you can control if a window is opened to show the plot in an interactive window. If set to false, the plot is stored directly. If set to true, you can interact with the plot and the chosen view in stored after you close the window.
The following examples will be done with the following scenario. This code is just to show
how the connections are set up, so that the graphs can be interpreted accordingly. The
important part is the part where the entities are connected.
importmosaik.util# Sim config. and other parametersSIM_CONFIG={'ExampleSim':{'python':'simulator_mosaik:ExampleSim',},'ExampleSim2':{'python':'simulator_mosaik:ExampleSim',},'Collector':{'cmd':'%(python)s collector.py %(addr)s',},}END=10# 10 seconds# Create Worldworld=mosaik.World(SIM_CONFIG,debug=True)# Start simulatorsexamplesim=world.start('ExampleSim',eid_prefix='Model_')examplesim2=world.start('ExampleSim2',eid_prefix='Model2_')collector=world.start('Collector')# Instantiate modelsmodel=examplesim.ExampleModel(init_val=2)model2=examplesim2.ExampleModel(init_val=2)monitor=collector.Monitor()# Connect entitiesworld.connect(model2,model,'val','delta')world.connect(model,model2,'val','delta',initial_data={"val":1,"delta":1},time_shifted=True,weak=True)world.connect(model,monitor,'val','delta')# Create more entitiesmore_models=examplesim.ExampleModel.create(2,init_val=3)mosaik.util.connect_many_to_one(world,more_models,monitor,'val','delta')# Run simulationworld.run(until=END)mosaik.util.plot_dataflow_graph(world,folder='util_figures')#mosaik.util.plot_execution_graph(world, folder='util_figures')#mosaik.util.plot_execution_time(world, folder='util_figures')#mosaik.util.plot_execution_time_per_simulator(world, folder='util_figures')
The dataflow graph shows the direction of the dataflow between the simulators. In the example below,
the ExampleSim simulator sends data to the Collector. The ExampleSim2 sends data to ExampleSim. The
dataflow connection from ExampleSim to ExampleSim2 is both weak (dotted line) and timeshifted (red line),
which can be seen in the red label.
The execution graph shows the order in which the simulators are executed. Differing from the example above,
the connection between ExampleSim and ExampleSim2 is only marked as weak, not as timeshifted.
If we add back the timeshift parameter, we get an additional arrow from ExampleSim to ExampleSim2. That
is because the data from ExampleSim is used in ExampleSim2 in a timeshifted manner, i.e., from the previous
step. This is the Gauss-Seidel scheme.
The execution time graph shows the execution time of the different simulators so that it can be seen
where the simulation takes more or less time. In the example below it can be seen that the Collector
uses comparatively more time than the ExampleSim simulators.
The execution time can also be plotted over the simulation steps per simulator, as can be seen
in the figure below. You can also set the parameter plot_per_simulator=True. In that case,
the plots for the different are separated. This is especially useful if the simulators have
different step sizes.
If you are especially interested in a certain part of the simulation to be shown you can slice the
time steps for the execution graph, the execution time graph, and the execution time per simulator.
You can use the slicing as with Python list slicing. Jumps are not possible. Below you can see
a few examples:
Below is the execution graph sliced as shown in the example code above.
Connecting mosaik with Apache Superset
Using Apache Superset to visualize simulation data
How can data that is generated through a mosaik simulation be explored without too much hassle and programming knowledge? In this tutorial
we will look at Apache Superset, an open source tool for data exploration and visualization.
This tutorial is divided into two parts, the installation of Apache Superset and how to couple it with mosaik and the basic usage of Superset.
For a more in-depth look at Apache Superset and all of its features please consult the tutorials on the Superset website.
This tutorial is based on a superset and timescaledb instance. Both instances were run through docker on ubuntu.
To install one of the databases locally follow corresponding the link.
(This tutorial is written with the Timescale database in mind. However, the other databases, should follow similar steps.)
Make sure to change the port of the database to something different from 5432 if you want to run the database and superset locally, as superset uses the same default port for its databse.
When using docker the command should look something like this:
Afterwards a secret key needs to be set for the production version.
For this the file superset_config.py is needed.
It can be copied into the right place using the command:
To connect superset with the database both superset and the database need to be online.
This connection is done in the superset web application.
The connection between superset and the database is done in the settings->DatabaseConnections menu.
This initiates the add database dialog consisting of three steps:
Step 1: Choosing the correct database(PostgreSQL in this example)
Step 2: Adding the database Credentials. If the database i run locally the IP-Address is 172.18.0.1 by default. If using Windows the IP might be host.docker.internal.
After connecting the database to superset the data can now be visualized. This tutorial shows data that is saved in a Timescale database. This data is saved using the MultiWriter2 of the mosaik Timescale adapter.
To do this first the data needs to be extracted from the databae using SQL. This is done in the SQL Lab:
I the SQL Lab the database the database, schema and table schema of a table in the database can be selected on the left side.
On the right side a sql query can be built.
In this example we use a simple query to get all of the data from the table.
If you are using the single writer from the mosaik timescale component the SQL query will look a bit different with it either being a double cast in case of
the json table_type:
SELECTtime,CAST(CAST(values->'Grid-0.0-LV1.1 Bus 1'ASVARCHAR)ASDOUBLEPRECISION)AS"BUS 1"FROMtesting_jsonWHEREvalue_type='va_degree'
And it being a single cast when it being the table_type string:
Clicking the Save&Explore Button will open up the Chart creation view of superset. This can also be done afterwards by selecting the wanted dataset in the datasets tab.
The default chart view of superset can be divided into two important parts. The left side where you can chose the kind of chart to create as well as input
the data from the dataset into the chart and the right chart where the chart will be displayed.
For this example lets start by selecting a line chart from the left side and then adding data to the relevant fields.
After changing the chart to line chart the relevant fields to fill out are the x-Axis, which in most cases will be the time column, and the metrics, which represent te y values.
Superset can not display simple y value, it is always a sql function. If a simple x/y comparison is needed the avg/min/max of the y values can be used since for only one value this is the value itself.
For selecting the x Axis you can chose from your dataset columns. Most of the time you want the simple time value but a custom sql query can also be used.
When selecting a metric there are many basic sql aggregation functions to choose from.
After selecting the metrics you can render the chart by clickin the CreateChart or UpdateChartbutton
Multiple metrics can be selected but only one x-Axis.
For this example I selected the average, minmum and maximum va_degree of Electric Buses over the timespan of one day in seconds.
If for your chart you cannot see the graph try making the time grain smaller.
There is a number of different charts available to visualize the data. After finishing your chart it needs to be saved inside a dashboard.
This is done by clicking the save button and giving the chart a name and either picking an existing dashboard or selecting the name of a new dashboard to be created.
The mosaik API defines the communication protocol between mosaik and the
simulators it couples. We differentiate between a low-level and
a high-level version of the API.
The low-level API uses plain network sockets to exchange JSON encoded messages.
The high-level API is an implementation of the low-level API in a specific
programming language. It encapsulates all parts related to networking (socket
handling, an event loop, message (de)serialization) and provides an abstract
base class with a few methods that have to be implemented in a subclass.
A high-level API implementation is currently available for Python, Java and Julia. Implementations for other
languages will be added when needed.
The figure below depicts the differences between the two API levels.
This section provides a general overview which API calls exists and when mosaik
calls them. The following sections will go into more detail.
When the connection between a simulator and mosaik is established, mosaik will
first call init(), optionally passing some global parameters to the
simulator. The simulators returns some meta data describing itself.
Following this, mosaik may call create() multiple times in order to
instantiate one of the models that the simulator implements. The return value
contains information describing the entities created.
The end of create phase and the beginning of the step (or simulation) phase
is marked by a call to setup_done(). At this point, all entities are
created and all relations between them are established.
When the simulation has been started, mosaik repeatedly calls step(). This
allows the simulator to step forward in time. It returns the time at which it
wants to perform its next step.
Finally, mosaik sends a stop() message to every simulator to request its
shut-down.
The following figure depicts the sequence of these messages:
After create() or step() have been called, there may be an
arbitrary amount of get_data() calls where mosaik requests the current
values of some entities’ attributes:
These methods are usually sufficient to connect simple simulators to mosaik.
However, control strategies, visualizations or database adapters may need to
actively query mosaik for additional data.
Thus, while a simulator is executing a simulation step, it may make
asynchronous requests to mosaik. It can collect information about the
simulated topology (get_related_entities()), request a new step for
itself (set_event()), get the current simulation progress
(get_progress()), query other entities for data (get_data())
and set data for other entities (set_data()).
The low-level API uses standard TCP sockets. If mosaik starts a simulator, that
simulator needs to connect to mosaik. If mosaik connects to a running instance
of a simulator, that simulator obviously needs to provide a server socket that
mosaik can connect to.
Network messages consists of a four bytes long header and a payload of
arbitrary length. The header is an unsigned integer (uint32)
in network byte order (big-endian) and stores the number of bytes in the
payload. The payload itself is a UTF-8 encoded JSON
list containing the message type, a message ID and the actual content:
Messages send between mosaik and a simulator must follow the request-reply
pattern. That means, that
every request that one party makes must be responded by the other party.
Request use the message type 0, replies uses 1 for success or 2 to
indicate a failure. The message ID is an integer that is unique for every
request that a network socket makes. Replies (no matter if successful or
failed) need to use the message ID of the corresponding request.
The content of a request roughly map to a Python function call:
The content of replies is either the return value of the request, or an error
message or stack trace. Error messages and stack traces should always be
strings. The return value for successful requests depends on the function.
We want to perform the following function call on the remote site:
my_func('hello','world',times=23)-->'thereturnvalue'. This would map
to the following message payload:
[0,1,["my_func",["hello","world"],{"times":23}]]
Our message is a request (message type 0), the message ID is 1 and the
content is a JSON list containing the function name as well as its arguments
and keyword arguments.
The complete message sent via the network will be:
In case of success, the reply’s payload to this request could look like this:
[1,1,"the return value"]
In case of error, this could be the reply’s payload:
[2,1,"Error in your code line 23: ..."]
The actual network messages would be:
\x00\x00\x00\x1a[1,1,"the return value"]
\x00\x00\x00\x29[2,1,"Error in your code line 23: ..."]
All commands that mosaik may send to a simulator are described in-depth in the
next section. All asynchronous requests that a simulator may
make are described in Asynchronous requests.
This section describes the API calls init(), create(), setup_done(),
step(), get_data() and stop(). In addition to these, a simulator
may optionally expose additional functions (referred to as extra methods).
These methods can be called at composition time (when you create your
scenario).
The init call is made once to initialize the simulator. It has one
positional argument, the simulator ID, and time_resolution and an arbitrary
amount of further parameters (sim_params) as keyword arguments.
The return value meta is an object with meta data about the simulator:
The api_version is a string that defines which version of the mosaik API the
simulator implements. Since mosaik API version 2.2, the simulator’s major
version (“x”, in the snippet above) has to be equal to
mosaik’s. Mosaik will cancel the simulation if a version mismatch occurs.
The type defines how the simulator is stepped and for which time the output
is valid. Time-based simulators only decide themselves on which points in
time they want to be stepped (i.e. communicate with the other simulators).
Their output is valid until the next step. Event-based simulators are always
stepped when a predecessor provides new input, but they can also schedule
steps for themselves. A more fine-grained behavior can be set for hybrid
simulators. See the scheduler description for details.
models is an object describing the models provided by this simulator. The
entry public determines whether a model can be instantiated by a user
(true) or if it is a sub-model that cannot be created directly (false).
params is a list of parameter names that can be passed to the model when
creating it. attrs is a list of attribute names that can be accessed (reading
or writing). If the optional any_inputs flag is set to true, any
attributes can be connected to the model, even if they are not attrs. This
may, for example, be useful for databases that don’t know in advance which
attributes of an entity they’ll receive.
extra_methods is an optional list of methods that a simulator provides in
addition to the standard API calls (init(), create() and so on). These
methods can be called while the scenario is being created and can be used for
operations that don’t really belong into init() or create().
Create num instances of model using the provided model_params
num is an integer for the number of model instances to create.
model needs to be a public entry in the simulator’s meta['models'] (see
init).
model_params is an object mapping parameters (from
meta['models'][model]['params'], see init) to their values.
Return a (nested) list of objects describing the created model instances
(entities). The root list must contain exactly num elements. The number of
objects in sub-lists is not constrained:
The entity ID (eid) of an object must be unique within a simulator instance.
For entities in the root list, type must be the same as the model
parameter. The type for objects in sub-lists may be anything that can be found
in meta['models'] (see init). rel is an optional list of
related entities; “related” means that two entities are somehow connect within
the simulator, either logically or via a real data-flow (e.g., grid
nodes are related to their adjacent branches). The children entry is optional
and may contain a sub-list of entities.
Perform the next simulation step at time time using input values from
inputs and return the new simulation time (the time at which step should
be called again) or null if the simulator doesn’t need to step itself.
time, max_advance, and the time_next_step are integers (or null). Their
unit is arbitrary, e.g. seconds (counted from simulation start), but has
to be consistent among all simulators used in a scenario.
inputs is a dict of dicts mapping entity IDs to attributes and dicts of
values (each simulator has to decide on its own how to reduce the values (e.g.,
as its sum, average or maximum):
max_advance tells the simulator how far it can advance its time without
risking any causality error, i.e. it is guaranteed that no external step will
be triggered before max_advance + 1, unless the simulator activates an output
loop earlier than that. For time-based simulators (or hybrid ones without any
triggering input) max_advance is always equal to the end of the simulation
(until). See the description of the scheduler for more
details.
Time-based simulators have set an entry for all requested attributes, whereas
for event-based and hybrid simulators this is optional (e.g. if there’s no new
event).
Event-based and hybrid simulators can optionally set a timing of their
non-persistent output attributes via a time entry, which is valid for all
given (non-persistent) attributes. If not given, it defaults to the current
time of the step. Thus only one output time is possible per step. For further
output times the simulator has to schedule another self-step (via the step’s
return value).
Set data as input data for all affected simulators.
data is an object mapping source entity IDs to objects which in turn map
destination entity IDs to objects of attributes and values
({"src_full_id":{"dest_full_id":{"attr1":"val1","attr2":"val2"}}})
You create a subclass of mosaik_api_v3.Simulator which implements the
four API calls init,
create, step
and get_data. You can optionally override
configure and
finalize. The former allows you to handle
additional command line arguments that your simulator may need. The latter is
called just before the simulator terminates and allows you to perform some
clean-up.
You then call mosaik_api_v3.start_simulation from your main() function
to get everything set-up and running. That function handles the networking as
well as serialization and de-serialization of messages. Commands from the
low-level API are translated to simple function calls. The return value of
these functions is used for the reply.
The api_version is a string that defines which version of the mosaik API
the simulator implements. Since mosaik API version 2.3, the simulator’s
major version (“x”, in the snippet above) has to be
equal to mosaik’s. Mosaik will cancel the simulation if a version mismatch
occurs.
The type defines how the simulator is advanced through time and whether
its attributes are persistent in time or transient.
models is a dictionary describing the models provided by this simulator.
The entry public determines whether a model can be instantiated by a user
(True) or if it is a sub-model that cannot be created directly
(False). params is a list of parameter names that can be passed to
the model when creating it. attrs is a list of attribute names that can
be accessed (reading or writing). If the optional any_inputs flag is set
to true, any attributes can be connected to the model, even if they are
not attrs. This may, for example, be useful for databases that don’t know
in advance which attributes of an entity they’ll receive. trigger is a
list of attribute names that cause the simulator to be stepped when
another simulator provides output which is connected to one of those.
extra_methods is an optional list of methods that a simulator provides in
addition to the standard API calls (init(), create() and so on).
These methods can be called while the scenario is being created and can be
used for operations that don’t really belong into init() or
create().
Create num instances of model using the provided model_params.
num is an integer for the number of model instances to create.
model needs to be a public entry in the simulator’s
meta['models'].
model_params is a dictionary mapping parameters (from
meta['models'][model]['params']) to their values.
Return a (nested) list of dictionaries describing the created model
instances (entities). The root list must contain exactly num
elements. The number of objects in sub-lists is not constrained:
The entity ID (eid) of an object must be unique within a simulator
instance. For entities in the root list, type must be the same as the
model parameter. The type for objects in sub-lists may be anything
that can be found in meta['models']. rel is an optional list of
related entities; “related” means that two entities are somehow connect
within the simulator, either logically or via a real data-flow (e.g.,
grid nodes are related to their adjacent branches). The children
entry is optional and may contain a sub-list of entities.
Perform the next simulation step from time time using input values
from inputs and return the new simulation time (the time at which
step() should be called again).
time and the time returned are integers. Their unit is arbitrary,
e.g. seconds (from simulation start), but has to be consistent among
all simulators used in a simulation.
inputs is a dict of dicts mapping entity IDs to attributes and
dicts of values (each simulator has to decide on its own how to reduce
the values (e.g., as its sum, average or maximum):
max_advance tells the simulator how far it can advance its time
without risking any causality error, i.e. it is guaranteed that no
external step will be triggered before max_advance + 1, unless the
simulator activates an output loop earlier than that. For time-based
simulators (or hybrid ones without any triggering input) max_advance
is always equal to the end of the simulation (until).
Time-based simulators have set an entry for all requested attributes,
whereas for event-based and hybrid simulators this is optional (e.g.
if there’s no new event).
Event-based and hybrid simulators can optionally set a timing of their
non-persistent output attributes via a time entry, which is valid
for all given (non-persistent) attributes. If not given, it defaults
to the current time of the step. Thus only one output time is possible
per step. For further output times the simulator has to schedule
another self-step (via the step’s return value).
The asynchronous requests can be called via the
MosaikRemote proxy self.mosaik from within
step, except for set_data() which has to
be called from another thread/process (see below). They don’t return the
actual results but an event (similar to a future of deferred). The event
will eventually hold the actual result. To wait for that result to arrive, you
simply yield the event, e.g.:
Return the data for the requested attributes attrs.
attrs is a dict of (fully qualified) entity IDs mapping to lists
of attribute names ({'sid/eid':['attr1','attr2']}).
The return value is a dictionary, which maps the input entity IDs to
data dictionaries, which in turn map attribute names to their
respective values:
({'sid/eid':{'attr1':val1,'attr2':val2}}).
Set data as input data for all affected simulators.
data is a dictionary mapping source entity IDs to destination entity
IDs with dictionaries of attributes and values ({'src_full_id':{'dest_full_id':{'attr1':'val1','attr2':'val2'}}}).
simulation is the instance of your API implementation (see
Simulator).
description may override the default description printed with the help on
the command line.
extra_option may be a list of options for docopt
(example: ['-e,--exampleEnableexamplemode']). Commandline
arguments are passed to Simulator.configure so that your API
implementation can handle them.
Here is an example with a bit more context:
importmosaik_api_v3example_meta={'type':'time-based','models'{'A':{'public':True,'params':['init_val'],'attrs':['val_out','dummy_out'],},}}classExampleSim(mosaik_api_v3.Simulator):def__init__(self):super().__init__(example_meta)sim_name='ExampleSimulation'defconfigure(self,args,backend,env):# Here you could handle additional command line argumentsdefinit(self,sid):# Initialize the simulatorreturnself.meta# Implement the remaining methods (create, step, get_data, ...)defmain():importsysdescription='A simple example simulation for mosaik.'extra_options=['--foo Enable foo','--bar BAR The bar parameter',]returnmosaik_api_v3.start_simulation(ExampleSim(),description,extra_options)if__name__=='__main__':sys.exit(main())
Mosaik provides high-level APIs not only for simulators written in Python, but also in various other languages.
Currently, Java and Julia are supported.
Modeling or composing a scenario in mosaik comprises three steps:
starting simulators,
instantiating models within the simulators, and
connecting the model instances of different simulators to establish the data
flow between them.
This page will show you how to create simple scenarios in these three steps.
It will also provide some recipes that allow you to create more complex
scenarios.
The central class for creating scenarios is mosaik.scenario.World (for
your convenience, you can also import World directly from mosaik). This
class stores all data and state that belongs to your scenario and its
simulation. It also provides various methods that allow you to start simulators
and establish the data flows between them.
(You can leave off the type annotation on sim_config if you’re not using type checking.)
As we start simulator instances by using world, it needs to know what
simulators are available and how to start them. This is called the sim config
and is a dict that contains every simulator we want to use together with some
information on how to start it.
In our case, the only simulator is the ExampleSim. It will be started by importing
the module example_sim.mosaik and instantiating the class ExampleSim.
This is only possible with simulators written in Python 3. You can also let
mosaik start simulator as external processes or let it connect to already
running processes. The simulator manager docs explain how
this all works and give you some hints when to use which method of starting
a simulator.
In addition to the sim config you can optionally pass the mosaik_config dictionary to
World in order to overwrite some general parameters for mosaik (e.g.,
the host and port number for its network socket or timeouts). Usually, the
defaults work just well.
Via the time_resolution parameter you can set a global time resolution for
the scenario, which will be passed to each simulator as keyword argument via
the init function (see API init). It tells each simulator how to
translate mosaik’s integer time to simulated time (in seconds from simulation
start). It has to be a float and it defaults to 1..
If you set the debug flag to True an execution graph will be created
during the simulation. This may be useful for debugging and testing. Note
that this increases the memory consumption and simulation time.
>>> world=mosaik.World(sim_config,debug=False)
There are two more technical parameters: You can set the cache flag to False
if the average step size of the simulators is orders of magnitudes larger than
the time resolution, i.e. a time resolution of microseconds where the typical
step size is in the seconds range. This will considerably reduce the
simulation time.
>>> world=mosaik.World(sim_config,cache=True)
Via max_loop_iterations you can limit the maximum iteration count within one
time step for same-time loops. It’s default value is 100.
Now that the basic set-up is done, we can start our simulators:
>>> simulator_0=world.start('ExampleSim',step_size=2)Starting "ExampleSim" as "ExampleSim-0" ...>>> simulator_1=world.start('ExampleSim')Starting "ExampleSim" as "ExampleSim-1" ...
To start a simulator, we call World.start and pass the name of the
simulator. Mosaik looks up that name in its sim config, starts the simulator
for us and returns a ModelFactory. This factory allows us to
instantiate simulation models within that simulator.
In addition to the simulator name, you can pass further parameters for the
simulators. These parameters are passed to the simulator via the init
API call.
Simulators specify a set of public models in their meta data (see init
API call). These models can be accessed with the
ModelFactory that World.start returns as if they were normal
Python classes. So to create one instance of ExampleSim’s model A we just
write:
>>> a=simulator_0.A(init_val=0)
This will create one instance of the A simulation model and pass the model
parameter init_val=0 to it (see create API call).
Lets see what it is that gets returned to us:
A model instance is represented in your scenario as an Entity. The
entity belongs to the simulator ExampleSim-0, has the ID 0.0 and its type
is A. The entity ID is unique within a simulator. To make it globally unique,
we prepend it with the simulator ID. This is called the entity’s full ID (see
Entity.full_id). You can also get a list of its child entities (which
is empty in this case).
In order to instantiate multiple instances of a model, you can either use
a simple list comprehension (or for loop) or call the static method
create of the model:
The list comprehension is more verbose but allows you to pass individual
parameter values to each instance. Using create is more concise but all
three instance will have the same value for init_val. In both cases you’ll
get a list of entities (aka entity sets).
Time-based and hybrid simulators are automatically scheduled for time step 0,
and will organize their scheduling until the simulation’s end themselves
afterward. For event-based simulators this is not the case, as they might only
want to be stepped if an event is created by another simulator for example.
Therefore you might need to set initial events for some event-based ones via
World.set_initial_event, which sets an event for time 0 by default,
or at later times if explicitly stated:
If we would now run our simulation, both, ExampleSim-0 and Example-Sim-1 would run
in parallel and never exchange any data. To change that, we need to connect
the models providing input data to entities requiring this data. In our case,
we will connect the val_out attribute of the A instances with the val_in
attribute of the B instances:
>>> a_set.insert(0,a)# Put our first A instance to the others>>> fora,binzip(a_set,b_set):... world.connect(a,b,('val_out','val_in'))
The method World.connect takes the source entity, the destination
entity and an arbitrary amount of (source attribute, dest. attribute) tuples.
If the name of the source attributes equals that of the destination attribute,
you can alternatively just pass a single string (e.g., connect(a,b,'attr') is the same as connect(a,b,('attr','attr'))).
mosaik deals with two separate types of data exchange between simulators:
First, there are measurements that have a value at each point of time.
Examples include all kinds of physical measurements like the voltage at a
grid node or the power output of a PV system.
Second, there are events (those were introduced in mosaik 3) which happen at
a particular point in time. A typical example is a message between ICT devices
or a set-point message from some controller to the PV system that it controls.
Each attribute of a mosaik simulator can deal either with measurements or with
events. In case of time-based simulators, all attributes work as measurements,
in case of event-based simulators, all attributes work as events. Hybrid-simulators
can work with either on an attribute-by-attribute basis (i.e. each attribute is
either for measurements or for events). For historic reasons, input attributes
are called trigger attributes when they deal with events, and output attributes
are called non-persistent when they deal with events. The opposite terms ‘non-trigger’
and ‘persistent’ are used for measurement attributes (input and output, respectively).
mosaik will complain if you connect a non-persistent output to a non-trigger
input.
This is because the target simulator should be able to rely on always receiving input on its non-trigger attributes, but the source simulator could potentially omit output on its non-persistent attribute.
(mosaik could provide output that the source simulator produced in earlier steps, delaying or even repeating it.
However, this would be semantically unsound as the event is associated with the time at which it was generated by the source simulator, not with the time at which the target simulator happens to run.)
A non-persistent output connected to a non-trigger input leads to a warning.
Usually, the solution to resolve this warning is to change the type of one of the affected attributes: the source attribute from non-persistent to persistent, or the target attribute from non-trigger to trigger.
You can do this without affecting the simulator’s other attributes by changing the simulator’s type to hybrid, where you can then specify which attributes should be trigger and/or non-persistent.
(See here for the format of META.)
Note that the attributes of hybrid simulator behave like measurements by default, so if you are changing an event-based simulator to hybrid, you will have to specify all attributes except for the affected one to be trigger and/or non-persistent if you want to preserve their previous behavior.
We also encourage you to carefully think about the opposite case where you attempt to connect a persistent output to a trigger input, as wanting to do this often indicates ambiguity about what your data represents.
However, because there is the common case of saving data generated in the simulation using some database or writer simulator (regardless of how whether it is an event or a measurement), mosaik will not complain when you set up connections like this.
You are also not allowed to create circular dependencies via standard
connections only (e.g., connect a to b and then connect b to a).
There are several ways to allow a bidirectional or cyclic exchange of data,
which is required for things like control strategies, e.g. via time-shifted
or weak connections. See section How to achieve cyclic data-flows for details.
This will execute the simulation from time 0 until we reach the time until
(in simulated time units). The scheduler section explains in
detail what happens when you call run.
While the simulation is running, the current progress is visualized using a tqdm progress bar. You can turn this off using the print_progress parameter of world.run:
world.run(until=10,print_progress=False)
If you want a more detailed progress report, you can set print_progress='individual' which will produce a separate progress bar for each simulator in your simulation.
We can also set the lazy_stepping flag (default: True).
If True, a simulator can only run ahead one step of its successors.
If False, a simulator always steps as long as all inputs are provided.
This might decrease the simulation time but increase the memory consumption.
>>> world.run(until=END,lazy_stepping=False)
To wrap it all up, this is how our small example scenario finally looks like:
# Setupimportmosaiksim_config={'ExampleSim':{'python':'example_sim.mosaik:ExampleSim'},}world=mosaik.World(sim_config)# Start simulatorssimulator_0=world.start('ExampleSim',step_size=2)simulator_1=world.start('ExampleSim')# Instantiate modelsa_set=[simulator_0.A(init_val=i)foriinrange(3)]b_set=simulator_1.B.create(3,init_val=1)# Connect entitiesfora,binzip(a_set,b_set):world.connect(a,b,('val_out','val_in'))# Run simulationworld.run(until=10)
The guiding principle behind mosaik’s scheduler is that simulator steps that happen at the same (mosaik) time are handled in data-dependency order.
In other words, if simulator A is connected to simulator B (with data flowing from A to B) and both simulators are scheduled to step at time t, simulator A will run first so that simulator B can use the most up-to-date outputs from A in its calculation.
This leads to problems when naively setting up bi-directional or cyclic data flow, like this
# Send battery's active power value to the controllerworld.connect(battery,controller,'P')# Controller sends back a schedule to the batteryworld.connect(controller,battery,'schedule')
The problem is that mosaik cannot step either of the involved simulators as both are waiting for input from the other one.
mosaik will recognize cycles like this and raise an error when you attempt to run a simulation containing them.
To avoid this error, the standard connections in your scenario must form an acyclic (directed) graph (on the level of simulators).
Of course, cyclic data flow is common in co-simulations and mosaik offers two options for this: time-shifted connections and weak connections.
(There are also asynchronous requests, which are deprecated.)
This indicates to mosaik that the output of the source simulator should only be passed to the target simulator at the next time step.
Now the loop is resolved, as the non-time-shifted connections form an acyclic graph again.
(So in the example, battery will run before controller in each step.)
For the first step, the source simulator in a time-shifted connection will not have run, yet.
However, the target simulator might still require input on the connected attribute.
Therefore, you must provide initial_data, which is a dictionary of the attributes of the source simulator and the values they should have for the first step.
In the example above, the result would be a sequential execution of the two simulators.
You can also set the time_shifted flag for both connections, in which case you would get parallel execution of both simulators, with each simulators using the output of the other simulator from the previous step.
Finally, you can set time-shifted to an integer value instead of True.
This will delay data along that connection by that many steps.
Occasionally, you want parts of your simulation to run on a much finer time scale than the rest, without committing to a precise factor between the two.
This is often the case for control strategies that are distributed over two or simulators.
In this case, you can use weak connection.
A weak connection resolves cycles within one time step in much the same way as a time-shifted connection.
However, if the source simulator produces output, this will trigger another step of the target simulator within the same time step.
This might in turn trigger additional of the source simulator again, leading to a so-called same-time loop which runs until one of the simulators interrupts it by not producing output.
(Weak connections therefore only make sense if the target attribute is a trigger attribute.
If it is not, the effect will be identical to a time-shifted connection.)
In most scenarios with same-time loops only some of the simulators will be involved in the loop.
This poses the question when the other simulators are run in relation to them.
In the past, this depended in subtle ways on the order in which connections were made in the scenario script.
Starting with mosaik 3.3, we force you to be more explicit about it.
Weak connections are only allowed between simulators that are part of a common simulator group.
A simulator group can be created using a Python with statement like so:
In this example, simulators A and B are part of a common simulator group, so weak connections between entities of A and B would be admissible.
Simulator C is also part of a group, but its a different group from the one shared by A and B, so weak connections between A (or B) and C would not be allowed.
(You could connect C weakly to itself, though.)
Simulator D is not part of any group and therefore does not support weak connections at all.
Weak connections are implemented by keeping track of a second time component for each simulator within a group.
(This time component is opaque to the simulators.)
When sending data along the weak connection, this sub-time is increased instead of the main time.
Normal (and time-shifted) connections can still be established between simulators within and without the group.
When a connection leaves a group, the corresponding sub-time is forgotten.
For example, suppose that simulator A is connected to simulator D and that it’s the start of the simulation.
As long as A is performing steps within the same-time loop, its main time will stay at 0 and only its sub-time will increase.
Simulator D, being outside of A’s group, does not see A’s sub-time.
It will not consider simulator A’s step 0 to be done, and simulator D therefore will not step.
Once the same-time loop is over, A’s main time will progress at which time D will see the progress and perform its step.
On the other hand, if D were part of A’s group (but still connected non-weakly), it would see A’s sub-time progress right away and therefore perform its step after A’s very first step.
The decision of whether or not to include a simulator in a group therefore depends on how you want the timing to work.
It is in fact possible to nest groups, by nesting the correponding withworld.group() blocks.
In this case, each additional group will introduce yet another finer level of time.
Weak connections between two simulators will increase the sub-time component associated to their closest shared group.
This type of connection is deprecated because it couples the involved simulators too closely.
The final option to resolve the cycle is to use asynchronous requests.
For this you only connect the battery’s P to the controller and let the control strategy set the new schedule via the asynchronous request set_data.
To indicate this in your scenario, you set the async_request flag of World.connect to True:
This way, mosaik will push the value for P from the battery to the controller.
It will then wait until the controller’s step is done before the next step for the battery will be computed.
The advantage of this approach is that the call of set_data is optional, so you don’t need to send a schedule on every step if there’s no new schedule.
(However, much the same effect can be achieved by using some trigger attributes.)
The disadvantage is that you have to implement the set_data call within the simulator with the specific destination, making it less modular.
The step implementation of the controller could roughly look like this:
When you create large-scale scenarios, you often work with large sets of
entities rather than single ones. This section provides some examples how you
can extract a sub-set of entities from a larger entity set based on arbitrary
criteria.
Let’s assume that we have created a power grid with mosaik-pypower:
Since mosaik-pypower’s Grid entity only serves as a container for the buses
and branches of our power grid, we directly bound its children to the name
grid. So grid is now a list containing a RefBus entity and multiple
Transformer, PQBus and Branch entities.
So how do we get a list of all transformers? This way:
transformers=[eforeingridife.type=='Transformer']
How do we get the single RefBus? This way:
refbus=[eforeingridife.type=='RefBus'][0]
Our PQBus entities are named like Busbar_<i> and ConnectionPoint_<i> to
indicate to which buses we can connect consumers and producers and to which we
shouldn’t connect anything. How do we get a list of all ConnectionPoint
buses? We might be tempted to do it this way:
The problem in this particular case is, that mosaik-pypower prepends a “grid
ID” to each entity ID, because it can handle multiple grid instances at once.
So our entity IDs are actually looking like this:
<grid_idx>-ConnectionPoint_<i>. Using regular expressions, we can get our
list:
If we want to connect certain consumers or producers to defined nodes in our
grid (e.g., your boss says: “This PV module needs to be connected to
ConnectionPoint_23!”), creating a dict instead of a list is a good idea:
remove_grididx=lambdae:e.eid.split('-',1)[1]# Little helper functioncps_by_name={remove_grididx(e):eforeingridifregex_conpoint.match(e)}
This will create a mapping where the string 'ConnectionPoint_23' maps to
the corresponding Entity instance.
This was just a small selection of how you can filter entity sets using
list/dict comprehensions. Alternatively, you can also use the filter
function or a normal for loop. You should also take at look at the
itertools and functools modules. You’ll find even more
functionality in specialized packages like PyToolz.
The method World.connect allows you to only connect one pair of
entities with each other. When you work with larger entity sets, you might not
want to connect every entity manually, but use functions that take to sets of
entities and connect them with each other based on some criteria.
The most common case is that you want to randomly connect the entities of one
set to another, for example, when you distribute a number of PV modules over a
power grid.
For this use case, mosaik provides mosaik.util.connect_randomly. It
takes two sets and connects them either evenly or purely randomly:
world=mosaik.World(sim_config)grid=pypower.Grid(gridfile=GRID_FILE).childrenpq_buses=[eforeingridife.type=='PQBus']pvs=pvsim.PV.create(20)# Assuming that len(pvs) < len(pq_buses), this will# connect 0 or 1 PV module to each bus:mosaik.util.connect_randomly(world,pvs,pq_buses,'P')# This will distribute the PV modules purely randomly, but every# bus will have at most 3 modules connected to it.mosaik.util.connect_randomly(world,pvs,pq_buses,'P',evenly=False,max_connects=3)
Another relatively common use case is connecting a set of entities to one other
entity, e.g., when you want to connect a number of controllable energy
producers to a central scheduler. For this use case, mosaik provides
mosaik.util.connect_many_to_one
...pvs=pvsim.PV.create(30)chps=chpsim.CHP.create(20)controller=cs.Scheduler()# Connect all producers to the controller, remember to set the# "async_requests" flag.connect_many_to_one(world,chain(pvs,chps),controller,'P',async_requests=True)
Connection rules are oftentimes highly specific for a project.
connect_randomly and
connect_many_to_one are currently the only functions that
are useful and complicated enough to ship it with mosaik. But writing your own
connection method is not that hard, as you can see in the
connect_many_to_one example:
Sometimes, the entities don’t contain all the information that you need in
order to decide which entity connect to which, but your simulation model could
provide that data. An example for this might be the maximum amount of active
power that a producer is able to produce.
Mosaik allows you to query a simulator for that data during composition time
via World.get_data:
>>> example_simulator=world.start('ExampleSim')Starting "ExampleSim" as "ExampleSim-2" ...>>> entities=example_simulator.A.create(3,init_val=42)>>> data=world.get_data(entities,'val_out')>>> data[entities[0]]{'val_out': 42}
The entities that you pass to this function don’t need to belong to the same
simulator (instance) as long as they all can provide the required attributes.
The World contains two networkx Graphs which hold
information about the data-flows between simulators and the simulation topology
that you created in your scenario. You can use these graphs, for example, to
export the simulation topology that mosaik created into a custom data or file
format.
This is outdated; mosaik now stored information about the data flow
differently.World.df_graph is the directed dataflow graph for your
scenarios. It contains a node for every simulator that you started. The
simulator ID is used to label the nodes. If you established a data-flow between
two simulators (by connecting at least two of their entities), a directed edge
between two nodes is inserted. The edges contain a list of the data-flows as
well as the async_requests, time_shifted, and weak flags
(see How to achieve cyclic data-flows) and the trigger and pred_waiting flags.
The data-flow graph may, for example, look like this:
World.entity_graph is the undirected entity graph. It contains a node
for every entity. The full entity ID ('sim_id.entity_id') is used as node
label. Every node also stores the simulator name and entity type. An edge
between two entities is inserted
if they are somehow related within a simulator (e.g., a PyPower branch is
related to the two PyPower buses to which it is adjacent) (see
create); or
The get_related_entities API call also uses and returns (parts of)
the entity graph. So you can access it in your scenario definition as well as
from with a simulator, control strategy or monitoring tool.
Please consult the networkx documentation for more details about
working with graphs and directed graphs.
It is very easy to do real-time (or “wall-clock time”) simulations in mosaik.
You just pass an rt_factor to World.run to enable it:
world.run(until=10,rt_factor=1)
A real-time factor of 1 means, that 1 simulation time unit (usually
a simulation second) takes 1 second of real time. Thus, if you set the
real-time factor to 0.5, the simulation will run twice as fast as the real
time. If you set it to 1/60, one simulated minute will take one real-time
second.
It may happen that the simulators are too slow for the real-time factor chosen.
That means, they take longer than, e.g., one second to compute a step when
a real-time factor of one second is set. If this happens, mosaik will by
default just print a warning message to stdout. However, you can also let your
simulation crash in this case by setting the parameter rt_strict to True.
Mosaik will then raise a RuntimeError if your simulation is too slow:
A simulator may optionally define additional API methods (see init)
that you can call from your scenario. These methods can implement operations,
like setting some static data to a simulator, which don’t really fit into
init() or create().
These methods are exposed via the model factory that you get when you start
a simulator. In the following example, we’ll call the example_method()
that the example simulator shipped with the mosaik Python API:
>>> world=mosaik.World({'ExampleSim':{... 'python':'example_sim.mosaik:ExampleSim'}})>>> es=world.start('ExampleSim')Starting "ExampleSim" as "ExampleSim-0" ...>>>>>> # Now brace yourself ...>>> es.example_method(23)23>>>>>> world.shutdown()
The simulator manager (or just sim manager) is responsible for starting and
handling the external simulator processes involved in a simulation as well as
for the communication with them.
Usually, these simulators will be started as separate sub-processes which will
then connect to mosaik via network sockets. This has two benefits:
Simulators can be written in any language.
Simulation steps can be performed in parallel, if two processes don’t depend
on each others data.
Simulators written in Python 3 can, for performance reasons, be imported and
executed like normal Python modules. This way, all Sim API calls will be plain
functions calls without the overhead of network communication and message
(de)serialization. However, since Python only runs in one thread at a time,
this will also prevent parallel execution of simulators.
When a (Python 3) simulator is computationally inexpensive, running it
in-process may give you good results. If it performs a lot of expensive
computations, it may be better to start separate processes which can then do
these computations in parallel. In practice, you should try and profile both
ways in order to get the maximum performance out of it.
Sometimes, both ways won’t work because you simply cannot start the simulator
process by yourself. This might be the reason for hardware-in-the-loop or if
a simulator needs to run on a separate machine. In these cases, you can simply
let mosaik connect to a running instance of such a simulator.
Internally, all three kinds of simulator processes (in-process with mosaik,
started by mosaik, connected to by mosaik) are represented by SimProxy
objects so that they all look the same to the other components of mosaik:
The sim manager gets its configuration via the World’s sim_config argument. The sim_config is a dictionary containing simulator
names and description of how to start them:
In the example above, we declare three different simulators. You can freely
choose a name for a simulator. Its configuration should either contain
a python, cmd or connect entry:
Since mosaik 2.3.0 it is possible to pass environment variables to
sub-processes. Using the key env in sim_config allows you to set new
environment variables. That could look like this:
This tells mosaik to run the simulator in process. As a value, you need to
specify the module and class name of the simulator separated by a colon.
In the example, mosaik will importpackage.module and instantiate sim=package.module.SimClass(). This only works for simulators written in
Python 3.
cmd
This tells mosaik to execute the specified command cmd in order to start a
new sub-process for the simulator.
You can use the placeholder %(python)s to use the same Python interpreter
and virtualenv that mosaik currently uses (see sys.executable).
In order to create a socket connection to mosaik the simulator needs to know
the address of mosaik’s server socket. Mosaik will pass this address (in the
form host:port) as a command line argument, so you need to include the
placeholder %(addr)s in your command. Mosaik will replace this with the
actual address.
If the simulator should open a seperate console window, the new_console
keyword can be used. This option is only available on Windows machines
and mosaik version >= 3.2.0.
You can optionally specify a current working directory (cwd). If it is
present, mosaik will change to that directory before executing cmd. Its
default value is '.'.
In our example, mosaik would execute:
$cdsimB/java
$java-jarsimB.jarlocalhost:5555
in order to start SimB.
Note
Please use for the cwd command and for paths in the cmd call only the UNIX/Linux path notation with slashes even if you are using windows. Do not use backslashes or double backslashes.
connect
This tells mosaik to establish a network connection to a running simulator
instance. It will simply connect to host:port – localhost:5678 for
SimC
When you defined your scenario and start the
simulation, mosaik’s scheduler becomes active. It
manages the execution of all involved simulators, keeps them in sync and
handles the data-flows between them.
Mosaik runs the simulation by stepping simulators through time.
Mosaik uses integers for the representation of time (to avoid rounding errors
etc.). Its unit (to how many seconds one integer step corresponds) can be
defined in the scenario, and is passed to every simulation component via the
init function as key-word parameter time_resolution. It’s
a floating point number and defaults to 1..
There are various paradigms for time in simulations: discrete time, continuous
time, and discrete event, to name the probably most common ones. Mosaik supports
discrete-time and discrete-event simulations, including the combination of both.
As these concepts are not always strictly distinguishable, we use a slightly
different notation for the simulator’s types, namely time-based, event-based,
and hybrid. It’s not always obvious which type of simulator is the most
appropriate one for a simulator, and in many cases both would be possible. As
a rough guide we could say:
Time-based simulators are more related to the physical world,
where the state, inputs, and outputs of a system are continuous (e.g. active
power of a PV module). The mapping of those continuous signals to discrete
points in time is then somewhat arbitrary (and depends on the desired precision
and the available computing resources). The lower limit for the temporal
resolution in a mosaik scenario is the unit assigned to the integer time steps.
Event-based simulators are related to the cyber world, where the state(s)
of a system can instantaneously change, and inputs and outputs also occur at
a specific point in time (example: sending/receiving of messages in a
communication simulation). The native way of stepping through time would then
be to just jump between all occurring events. In a mosaik simulation the time
of the events have to be rounded to the mosaik’s integer time steps.
Hybrid simulators can represent any kind of combined systems with both
time-based and event-based components.
Mosaik tracks the current simulation time for every simulator individually. How
simulators step through the time from simulation start to end depends on their
stepping type described above:
When the simulation starts, all time-based (and hybrid) simulators are at time 0. When it asks
a simulator to perform its next step, it passes its current simulation time
tnow to it. After its step, the simulator returns the time at which it
wants to perform its next step (tnext). Thus, a simulator’s step size
doesn’t need to be constant but can vary during the simulation.
All data that a time-based simulator computes during a step is valid for the right-open
interval [tnow, tnext) as shown in the following figure.
Schematic execution of a time-based simulator A. tnow, tnext and
the validity interval for its first step 0 are shown. The figure also shows
that the step size of a simulator may vary during the simulation.
The stepping through time of event-based simulators is rather different.
Event-based simulators are stepped at all times an event is created at. These
events can either be created by other simulators that are connected to this
simulator via providing the connected attribute, or the simulator can also
schedule events for itself via the step function’s return value.
The output provided by event-based simulators is only valid for a specific
point in time, by default for the current time of the step, or for any later
time if explicitly set via the (optional) output time. Providing the output
attributes is optional for event-based simulators. As consequence a simulator
connected to a specific attribute is only triggered/stepped if the output is
actually provided. See the API description for
implementation details.
Event-based simulators do not necessarily start at time 0, but whenever their
first event is scheduled, either by other simulators or via
World.set_initial_event from
the scenario definition.
Schematic execution of an event-based simulation. Depending on A’s
actual output a step of B is triggered (or not), at A’s step time or
later. Simulator A also schedules itself.
Note that it is possible that a simulator is stepped several times at a
specific point in time. See Same-time loops for details.
If there are data-flows between two simulators (because you connected some of
their entities), a simulator can only perform a step if all input data has been
computed.
Let’s assume we created a data-flow from a simulator A to a simulator B and
B wants to perform a step from tnow(B). Mosaik determines which
simulators provide input data for B. This is only A in this example. In
order to provide data for B, A needs to step far enough to produce data for
tnow(B), that means tnext(A) > tnow(B) as the
following figure illustrates.
(a)B cannot yet step because A has not progressed far enough yet
(tnext(A) <= tnow(B)).
(b)B can perform its next step, because A now has progressed far
enough (tnext(A) > tnow(B)).
When this condition is met for all simulators providing input for B, mosaik
collects all input data for B that is valid at tnow(B) (you could
say it takes one snapshot of the global simulation state). It passes
this data to B. Based upon this (and only this) data, B performs its step
[tnow(B), tnext(B)).
This is relatively easy to understand if A and B have the same step size,
as the following figures shows:
In this example, A and B have the same step size. Mosaik steps them
in an alternating order starting with A, because it provides the input
data for B.
If B had a larger step size than A, A would produce new data while B
steps. B would still only use the data that was valid at tnow(B),
because it only “measures” its inputs once at the beginning of its step:
In this example, B has a larger step size. It doesn’t consume all data
that A produces, because it only gets data once at the beginning of its
step.
On the other hand, if A had a larger step size than B, we would reuse the
same data from A multiple times as long as it is valid:
In this example, A has a larger step size. B reuses the same data
multiple times because it is still valid.
The last two examples may look like special cases, but they actually arise from
the approach explained above.
How far is a simulator allowed to advance its time?
As described in the API documentation, mosaik tells the
simulator each step how far it is allowed to advance its internal simulation
time via the max_advance argument. It is guaranteed that no step will be
scheduled until then (inclusively), unless the simulator activates a
triggering dependency loop earlier than that.
Mosaik deduces this from the simulation topology and the progress of the
simulators. Note that the simulator will not necessarily be stepped at
max_advance + 1 as this will only happen if the predecessor actually
provides the connected output attribute(s).
As time-based simulators (or hybrid ones without any triggering input) only
decide themselves when they are stepped, max_advance is always equal to the
end of the simulation for those. But of course they will most likely miss some
updates of the input data if their step size is too large and not synchronized
with their input providers. In order not to miss any input update, you can
change the type of the simulator to hybrid. Then the simulator will be
stepped on each update.
Note
The max_advance value is not necessarily appropriate for real-time simulations as it does not consider eventual steps which are scheduled via the asynchronous set_event() method.
After a simulator is done with its step, mosaik determines, based on the
data-flows that you created in your scenario, which data other simulators need
from it. It makes a get_data() API call to the simulator and stores the data
that this call returns in an internal buffer. It also memorizes for which
time this data is valid.
Before a simulator steps, mosaik determines in a similar fashion what input
data the simulator needs. Mosaik checks if all input-providing simulators have
stepped far enough to (potentially) provide that data and waits otherwise.
After that all input data is collected and then passed to the inputs
parameter of the step() API call.
It is important to understand that simulators don’t talk to each other directly
but that all data flows through mosaik were it can be cached and managed.
Sometimes the simulated system requires cyclic data-flows between components, e.g. a control
mechanism (C) that controls another entity (E) based on its state, e.g. by sending commands
or a schedule.
It is not possible to perform both data-flows (the state from E to C and
the commands/schedule from C to E) at the same time because they depend on
each other (yes, this is similar to the chicken or egg dilemma).
The cycle can be resolved by first stepping E (e.g., from t = 0 to t
= 1). E’s state for that interval can then be used as input for C ’s
step for the same interval. The commands/schedule that C generates for E
will then be used in E’s next step. This results in a serial execution,
also called Gauss-Seidel scheme.
In this example, a controlled entity E provides state data to the
controller C. The commands or schedule from C is used by E in its next
step.
This resolution of the cycle makes sense if you think how this would work in
real life. The controller would measure the data from the controlled unit at
a certain point t. It would then do some calculation which take a certain
amount of time Δt which would be send to the controlled unit at t + Δt.
However, mosaik is not able to automatically resolve that cycle. That’s why you
are not allowed to connect(E,C) and connect(C,E) in a scenario. This can be done
via the time-shifted connection
connect(C,E,(‘c_out’,‘a_in’),time_shifted=True,initial_data={‘c_out’:0}),
which tells mosaik that the output of C is to be used for E’s next time step(s) afterwards.
As for the first step (at time 0) this data cannot be provided yet, you have to set it via the
initial_data argument. In this case, the initial data for ‘a_in’ is 0.
Another way to resolve this cycle is to allow async. requests via the async_requests flag
connect(E,C,async_requests=True) and use the
asynchronous callbackset_data() in C’s step() implementation in order to send the commands or schedule from C
to E. The advantage of this approach is that the call of set_data is optional, i.e. the commands
or schedules don’t need to be sent on every step.
If you set the time_shifted flag for both connections, the simulators can be
executed in parallel (Jacobi scheme). Note that a computationally parallel
execution is only possible for simulators that are not run in-process.
In this example, two entities are running in parallel. The outputs of each
simulator are used by the other one in its next step afterwards.
Sometimes, simulators need to exchange data back and forth at the same time before they
can step to their next time step. Such same-time loop can be defined via a weak
connection. In the connect statement, the connection is marked as weak, e.g. via
world.connect(agent,model,'delta',weak=True).
Loops which are closed by a weak connection can be run multiple times within
the same mosaik time step, as weak connections do not necessarily imply a
temporal progress. This can be used for example to only advance the simulation time
when the state has converged to a stable solution. To activate (and also stay
in) a same-time cycle, a simulator has to provide its ‘cyclic’ attribute(s) via
the get_data function and indicating as output time the current
step time. To escape the cycle, the attribute(s) in the get_data’s return
dictionary have to be omitted or a time later than the step’s time indicated.
An example scenario for this is shown in a tutorial.
A same-time loop with three repetitions between simulator A and B.
To prevent the loop to be run infinite times, mosaik raises a runtime error
when a certain number of iterations within one time step has been reached. The
default maximum iteration count is 100 and can be adjusted via the
max_loop_iterations parameter within the scenario definition if needed (see
mosaik.scenario.World).
By now you should have a general idea of how mosaik handles data-flows between
simulators. You should also have the idea that simulators only perform a step
when all input-providing simulators have stepped far enough. But what if they
don’t have any (connected) inputs? In this section you’ll learn about the
algorithm that mosaik uses to determine whether a simulator can be stepped or
not.
Sim-process running for each simulator in parallel
This is how it works:
Should there be a next step at all? *
Yes: Go to step 2.
No: Stop the simulator.
*We’ll explain how to answer this question below.
2. Is a next step already scheduled, either self-scheduled via step or
by triggering input?
Yes: Go to step 3.
No: Wait until a next step is set. Then go to step 3.
Have all dependent simulators stepped far enough?
Yes: Go to step 4.
No: Wait for all dependencies. Then go step 4.
Collect all required input data.
Send collected input data to simulator, perform the simulation step and
eventually get the time of a next step.
Get all data from this simulator that are connected to other simulators and
store it internally.
Notify other simulators that already wait for this simulator. If there’s
any output which is connected to a triggering input of another simulator,
schedule new steps for it (at output time).
So how do we determine whether a simulator must perform another step or it is
done?
When we start the simulation, we pass a time unto which our simulation should
run (world.run(until=END)). Usually a simulator is done if the time of its
next step is equal or larger than the value of until. This is, however, not
true for all simulators in a simulation. If no one needs the data of a
simulator step, why perform this step?
So the actual algorithm is as follows:
If a simulator has no outgoing data-flows (no other simulator needs its data)
it simulates until the condition tnext > tuntil is met or
none of the simulators which could trigger a step are running anymore.
Else, if a simulator needs to provide data for other simulators, it keeps
running until all of these simulators have stopped.
Mosaik 3 has some new features which required some API changes. For the time
being simulators implementing mosaik-api 3 are still supported, but you will
get a deprecation warning at the beginning. In case that you don’t want to use
any of the new features you could stay with mosaik 2.
But otherwise the upgrade to mosaik 3 is quite easy, as you will see in the
following sections:
Simulation components now have a type (time-based, event-based, or
hybrid) to indicate which simulation time paradigm they implement. See
details here. The type is set in the component’s meta
data, which the component returns to mosaik via the API’s
init function.
A global time resolution is now defined for each
scenario indicating how to translate mosaik’s integer time to simulated time.
It can be set at the instantiation of the simulation’s World in the
setup of the scenario and is set to 1. by
default. It’s value is passed to the components via the init
function.
max_advance tells the simulator how far it can advance its time without
risking any causality error, i.e. it is guaranteed that no external step will
be triggered before max_advance + 1. It is determined for each step and passed
to the component as third positional argument of the API’s step
function.
This is a list of some questions we recently got. If you cannot find an answer
for your questions here, you are welcome to post it on our mailing list.
Maverig, an environment for graphical analysis as well as scenario design has
been developed and is currently in a prototypical state. It may be used for
demonstration purposes, but its maintenance and further development is not our
top priority right now. Please understand, if the installation guide is not
up-to-date with the newest package versions.
However, we argue that graphical tools are not feasible for the design of large
and complex scenarios. For most applications the more flexible scenario design
by code is advantageous.
No, mosaik can be used with any language that provides network sockets and
ways to (de)serialize JSON.
Since implementing network event loops and message (de)serialization is
repetitive work and unnecessary overhead, we provide so called high-level
APIs for certain languages that provide a base class that you can inherit and
just need to implement a few methods representing the API calls.
Currently, high-level API are available for Python, JAVA, and C#,
but an implementation for C++ will follow soon.
Can I use my MATLAB/Simulink models with mosaik? How do I do it?
Yes, you can. We will provide an example soon. In the end, if you manage to
let your model communicate via sockets and are capable of
serializing/deserializing JSON data objects you can use it with mosaik (see
Can I use mosaik only with Python programs? for details).
I can only see active power values in the web visualization. Does mosaik also support reactive power values with PyPower?
Of course, it only depends on your models. The basic models distributed
with mosaik 2 only produce active power outputs (cos phi = 1), so we don’t
display reactive power.
What are the power grid’s parameters? How are the cables’/lines’ parameters formatted?
On this page, we discuss some of the design decisions that we made. This should
explain why some features are (not) present and why they work the way that they
work.
Note
For the sake of readability, some concepts are simplified in the following
sections. For example, the snippet connect(A,B) means we’re connecting
some entities of a simulator A to some entities of simulator B;
simulator and entity are used as if they were the same concept;
A.step() means, that mosaik calls the step() function of
simulator/entity A.
Circular data-flows were one of the harder problems to solve.
When you connect the entities of two simulators with each other, mosaik tracks
the new dependency between these simulators:
connect(A,B)
A would now provide input data for B. When mosaik runs the simulation, the
step for a certain time t would first be computed for A and then for B
with the inputs of A.
In order to connect control strategies (like multi-agent systems) with
to-be-controlled entities, you usually need a circular data-flow. The entity
provides state information for the controller which in turn sends new commands
or schedules to the controlled entity. The naïve way of doing this would be:
connect(A,B,'state')connect(B,A,'schedule')
A would receive schedules via the inputs of A.step(). In A.step(), it
would compute new state information which mosaik would get via A.get_data().
Mosaik would forward this to the inputs of B.step(). B.step() would
calculate some schedules, which mosaik would again get via B.get_data() and
pass to A.step() …
The question that arise here is: Which simulator do we step first – A or B?
Mosaik has no clue. You could say that A needs to step first, because the
data-flow from A to B was established first. However, if you re-arrange
your code and (accidentally) flip both lines, you would get a different
behavior and a very hard to find bug.
What do we learn from that? We need to explicitly tell mosaik how to resolve
these cycles and prohibit normal circular data-flows as in the snippet above.
Mosaik provides two ways for this. The first is via time-shifted connections:
This tells mosaik how to resolve the cycle and throws an error if you
accidentally flip both lines.
Theoretically, we could be done here. But we aren’t. The data-flows in the
example above are passive, meaning that A and B compute data hoping that
someone will use them. This abstraction works reasonably well for normal
simulation models, but control mechanism usually have an active role. They
actively decide whether or not to send commands to the entities they control.
Accordingly, mosaik provides ways for control mechanisms and monitoring tools
to actively collect more data from the simulation and set data to other
entities. These means are implemented as asyncronous requests that a simulator can perform during its step.
Similar to the cyclic data-flows, this requires you to tell mosaik about it to
prevent some scheduling problems:
connect(A,B,async_requests=True)
This prevents A from stepping too far into the future so that B can get
additional data from or set new data to A in B.step().
Since you can set data via an asynchronous request, you can implement cyclic
data-flows with it:
connect(A,B,'state',async_requests=True)
The implementation of A.step() and A.get_data() would be the same. In
B.step() you would still receive the state information from A and compute
the schedules. However, you wouldn’t store them somewhere so that
B.get_data() can return them. Instead, you would just pass them actively to
set_data(). Mosaik stores that data in a special input_buffer of A which
will be added to the input of A’s next step.
So to wrap this up, there are two possibilities to achieve cyclic data-flows:
The first implementation of same-time loops was pretty ad-hoc and therefore showed some difficult-to-explain behavior.
In particular, the order in which simulators in a same-time loop would trigger each other depended in subtle ways on the order in which connections were established.
(The execution order was governed by the rank of a simulator, which was calculated using a topological sort method from networkx.)
As a first solution, we attempted a concept of dense time, where each internal time stamp had two components, called time and micro step.
Only the time component is communicated to the simulators, whereas the micro step component is only used internally.
Progress along weak connections would only increase the micro step component, allowing simulators to perform multiple steps in one time step, while still preserving an order for all steps.
Unfortunately, a very common control structure that usually happened to work under the old scheme was not possible anymore:
The simplest version of the case in question is when there are two simulators that communicate via a same-time loop and which then send the result of their negotiation to a third simulator.
Because users would normally create the connections for the negotiation part earlier in their scenario, the corresponding same-time loop would run first, leading to the expected result.
With the dense-time setup, the third simulator would instead run at micro step 0, and thus only receive the results of the first round of negotiation.
To resolve this problem, we introduce a concept of simulator groups.
A user can create a group in their script using a with statement like
All simulators started in the with block are automatically added to the group.
Users can nest groups by nesting the withworld.group() blocks.
(Using the with statement has the advantage that users cannot nest the groups incorrectly.)
Each group adds an additional tier to the internal times used for simulators contained within, leading to a concept of tiered time.
(The simulators still only get to see the highest level.)
When a connection leaves a group (i.e. leads from a simulator in a group to a simulator not in that group), the corresponding part of the time is cut off.
This results in the simulator outside the group not seeing the steps inside the group.
It therefore only considers the steps of its inputs done when the simulators in the group progress their actual time, i.e. when they conclude their negotiation.
Similarly, a time entering a group will be padded with zeros at the end.
To force users to update their simulations (so they don’t silently start producing completely different results), we decided to outlaw weak connections outside of groups.
A user using weak connections will receive an error message leading them to the documentation that explains how to adapt their scenario.
This is additional mathematical information on tiered time.
Times are used in two different ways:
They can represent specific points in the simulation or they can represent intervals of time.
Previously, both of these concepts were represented by the same data structure internally.
With tiered time, we also introduce tiered interval, so tiered times now only represent points in time.
It is legal to add an interval to a time, and to add two intervals, but it is not legal to add two times.
Speaking of intervals, there are three basic types:
Connections within a group keep the number of tiers fixed an just add something to some tiers (usually nothing at all, or 1 to the first tier for time-shifted connections, or 1 to the last tier for weak connections).
Connections leaving a group cut off some tiers at the end.
Connections entering a group add some tiers at the end.
When calculating the progression within the simulation, it becomes necessary to add these types of intervals together, leading to mixed types.
Therefore, a tiered interval consists of:
A list of addition tiers.
These are added to the corresponding tiers of the tiered time.
All further tiers of the tiered time are cut off.
A list of extension tiers.
There are appended to the result of the addition and cutoff.
A pre-length, which is used as a sanity check.
The pre-length must equal the number of tiers of the time to which the interval is added.
Also, the pre-length serves as an upper bound for the number of addition tiers, so that the time always has enough components to add to.
The addition rules for intervals are set up such that adding two intervals to a time one after the other is the same as adding the sum of the two intervals to the time, and such that the addition of intervals is associative.
(Mathematically, this turns tiered times and intervals into a category with sets of tiered times of the same length as objects and tiered intervals as arrows.)
The simple variant of the logo should be used when the display size for the
logo is so small, that the icons would not be recognizable very well, e.g. in
the headers of letters or papers.
The API reference provides detailed descriptions of mosaik’s classes and
functions. It should be helpful if you plan to extend mosaik with custom
components.
mosaik.scenario — Classes related to the scenario creation
This module provides the interface for users to create simulation scenarios for
mosaik.
The World holds all necessary data for the simulation and allows the
user to start simulators. It provides a ModelFactory (and
a ModelMock) via which the user can instantiate model instances
(entities). The method World.run finally starts the simulation.
cmd (str) – The command to start this simulator. String %(python)s will be replaced by the python command used to start this scenario, %(addr)s will be replaced by the host:port combination to which the simulator should connect.
Optional Keys:
env (Dict[str, str]) – The environment variables to set for this simulator.
cwd (str) – The current working directory for this simulator.
api_version (str) – The API version of the connected simulator. Set this to suppress warnings about this simulator being outdated.
posix (bool) – Whether to split the given shell command using POSIX rules. (Default: False on Windows, True otherwise.)
The world holds all data required to specify and run the scenario.
It provides a method to start a simulator process (start) and
manages the simulator instances.
You have to provide a sim_config which tells the world which simulators
are available and how to start them. See mosaik.simmanager.start
for more details.
mosaik_config can be a dict or list of key-value pairs to set addional
parameters overriding the defaults:
Here, addr is the network address that mosaik will bind its socket to.
start_timeout and stop_timeout specifiy a timeout (in seconds) for
starting/stopping external simulator processes.
If execution_graph is set to True, an execution graph will be created
during the simulation. This may be useful for debugging and testing. Note,
that this increases the memory consumption and simulation time.
The number of seconds that correspond to one mosaik time step in
this situation. The default value is 1.0, meaning that one integer
step corresponds to one second simulated time.
The number of iterations allowed for same-time loops within one
time step. This is checked to prevent accidental infinite loops.
Increase this value if your same-time loops require many iterations
to converge.
Establish a data-flow for each (src_attr,dest_attr) tuple in
attr_pairs. If src_attr and dest_attr have the same name, you
you can optionally only pass one of them as a single string.
Raise a ScenarioError if both entities share
the same simulator instance, if at least one (src. or dest.) attribute
in attr_pairs does not exist, or if the connection would introduce
a cycle in the data-flow (e.g., A → B → C → A).
If the dest simulator may make asynchronous requests to mosaik to
query data from src (or set data to it), async_requests should be
set to True so that the src simulator stays in sync with dest.
An alternative to asynchronous requests are time-shifted connections.
Their data flow is always resolved after normal connections so that
cycles in the data-flow can be realized without introducing deadlocks.
For such a connection time_shifted should be set to True and
initial_data should contain a dict with input data for the first
simulation step of the receiving simulator.
An alternative to using async_requests to realize cyclic data-flow
is given by the time_shifted kwarg. If set to True it marks the
connection as cycle-closing (e.g. C → A). It must always be used with
initial_data specifying a dict with the data sent to the destination
simulator at the first step (e.g. {‘src_attr’: value}).
Start the simulation until the simulation time until is
reached.
Before this method returns, it stops all simulators and closes
mosaik’s server socket. So this method should only be called
once.
Parameters:
until (int) – The end of the simulation in mosaik time steps
(exclusive).
rt_factor (float | None) – The real-time factor. If set to a number > 0,
the simulation will run in real-time mode. A real-time
factor of 1. means that 1 second in simulated time takes
1 second in real time. An real-time factor of 0.5 will let
the simulation run twice as fast as real time. For correct
behavior of the real-time factor, the time resolution of the
scenario has to be set adequately (the default is 1 second).
rt_strict (bool) – If the simulators are too slow for the
real-time factor you chose, mosaik will only print a warning
by default. In order to raise a RuntimeError instead,
you can set rt_strict to True.
Whether progress bars are printed while
the simulation is running. The default is to print one bar
representing the global progress of the simulation. You can
also set the value to 'individual' to get one bar per
simulator in your simulation (in addition to the global
one). A value of False turns off the progress bars
completely.
The progress bars use
tqdm; see their
documentation on how to write to the console without
interfering with the bars.
lazy_stepping (bool) – Whether to prevent simulators from running
ahead of their successors by more than one step. If
False a simulator always steps as long all input is
provided. This might decrease the simulation time but
increase the memory consumption.
Instances of this class are exposed as attributes of
ModelFactory and allow the instantiation of simulator models.
You can call an instance of this class to create exactly one entity:
sim.ModelName(x=23). Alternatively, you can use the create
method to create multiple entities with the same set of parameters at once:
sim.ModelName.create(3,x=23).
Create num entities with the specified model_params and return
a list with the entity dicts.
The returned list of entities is the same as returned by
mosaik_api_v3.Simulator.create, but the simulator is prepended
to every entity ID to make them globally unique.
For every entity, there is an entry in the dict and each entry is itself
a dict with attributes and a list of values. This is, because we may have
inputs from multiple simulators (e.g., different consumers that provide
loads for a node in a power grid) and cannot know how to aggregate that
data (sum, max, …?).
mosaik.simmanager — Management of external processes
The simulation manager is responsible for starting simulation processes and
shutting them down. It also manages the communication between mosaik and the
processes.
It is able to start pure Python simulators in-process (by importing and
instantiating them), to start external simulation processes and to connect to
already running simulators and manage access to them.
Start the simulator sim_name based on the configuration im
world.sim_config, give it the ID sim_id and pass the time_resolution
and the parameters of the dict sim_params to it.
The sim config is a dictionary with one entry for every simulator. The
entry itself tells mosaik how to start the simulator:
ExampleSimA is a pure Python simulator. Mosaik will import the module
example_sim.mosaik and instantiate the class ExampleSim to start
the simulator.
ExampleSimB would be started by executing the command example_sim and
passing the network address of mosaik das command line argument. You can
optionally specify a current working directory. It defaults to ..
ExampleSimC can not be started by mosaik, so mosaik tries to connect to
it.
time_resolution (in seconds) is a global scenario parameter, which tells
the simulators what the integer time step means in seconds. Its default
value is 1., meaning one integer step corresponds to one second simulated
time.
The scheduled next steps this simulator will take, organized as a heap.
Once the immediate next step has been chosen (and the has_next_step event
has been triggered), the step is moved to next_step instead.
An iterable of this sim’s ancestors that can trigger a step of
this simulator. The second component specifies the least amount of
time that output from the ancestor needs to reach us.
This lists those connections that use the timed_input_buffer.
The keys are the entity-attribute pairs of this simulator with
the corresponding list of simulator-time-entity-attribute triples
describing the destinations for that data and the time-shift
occuring along the connection.
Output to pull in whenever this simulator performs a step.
The keys are the source SimRunner and the time shift, the values
are the source and destination entity-attribute pairs.
Schedule a step for this simulator at the given time. This
will trigger a re-evaluation whether this simulator’s next
step is settled, provided that the new step is earlier than the
old one and the simulator is currently awaiting it’s next
settled step.
Return the data for the requested attributes attrs.
attrs is a dict of (fully qualified) entity IDs mapping to lists
of attribute names ({'sid/eid':['attr1','attr2']}).
The return value is a dictionary, which maps the input entity IDs to
data dictionaries, which in turn map attribute names to their
respective values:
({'sid/eid':{'attr1':val1,'attr2':val2}}).
Set data as input data for all affected simulators.
data is a dictionary mapping source entity IDs to destination entity
IDs with dictionaries of attributes and values ({'src_full_id':{'dest_full_id':{'attr1':'val1','attr2':'val2'}}}).
This class provides a singleton instance of a collection of simulation
starters. Default starters are:
- python: start_inproc
- cmd: start_proc
- connect: start_connect
External packages may add additional methods of starting simulations by
adding new elements:
from mosaik.simmanager import StarterCollection
s = StarterCollection()
s[‘my_starter’] = my_starter_func
Randomly connect the entities from
src_set to the entities from dest_set and return a subset of dest_set
containing all entities with a connection.
world is an instance of the World to which the
entities belong.
src_set and dest_set are iterables containing
Entity instances. src_set may be empty,
dest_set must not be empty. Each entity of src_set will be connected to
an entity of dest_set, but not every entity of dest_set will
necessarily have a connection (e.g., if you connect a set of three entities
to a set of four entities). A set of all entities from dest_set, to which
at least one entity from src_set was connected, will be returned.
attrs is a list of attribute names of pairs as in
connect.
If the flag evenly is set to True, entities connections will be
distributed as evenly as possible. That means if you connect a set of three
entities to a set of three entities, there will be three 1:1 connections;
if you connect four entities to three entities, there will be one 2:1 and
two 1:1 connections. If evenly is set to False, connections will be
truly random. That means if you connect three entities to three entities,
you may either have three 1:1 connections, one 2:1 and two 1:1 connections
or just one 3:1 connection.
max_connects lets you set the maximum number of connections that an
entity of dest_set may receive. This argument is only taken into account
if evenly is set to False.
format (Literal['png', 'pdf', 'svg']) – format for created image
show_plot (bool) – whether to open a window to show the plot
slice (Tuple[int, int] | None) – reduce the timeframe that you show in the plot. Usage
as in Python list slicing, i.e., negative values are possible to
start from the end of the list. Jumps are not possible.
slice needs to be a two-element integer list, e.g.
(0,5).
Returns:
None but image file will be written to file system
Creates an image visualizing the data flow graph of a mosaik
scenario. Using the spring layout from Matplotlib (Fruchterman-
Reingold force-directed algorithm) to position the nodes.
format (Literal['png', 'pdf', 'svg']) – format for created image
show_plot (bool) – whether to open a window to show the plot
slice (Tuple[int, int] | None) – reduce the timeframe that you show in the plot.
Usage as in Python list slicing, i.e., negative values are
possible to start from the end of the list. Jumps are not
possible. slice needs to be a two-element integer tuple,
e.g. (0,5).
format (Literal['png', 'pdf', 'svg']) – format for created image
show_plot (bool) – whether to open a window to show the plot
plot_per_simulator (bool) – whether to create a separated plot per
simulator. This is especially useful if the step sizes of the
simulators are very different.
slice (Tuple[int, int] | None) – reduce the timeframe that you show in the plot. Usage
as in Python list slicing, i.e., negative values are possible to
start from the end of the list. Jumps are not possible.
slice needs to be a two-element integer tuple, e.g.
(0,5).
Returns:
None but image file will be written to file system
This simulator gives a steady input to a connected simulator.
This input can either be an constant value or given by a custom
function based on the current time.
When starting the simulator, a custom step size may be provided
using the step_size parameter. The default is 1.
When creating a Constant entity, the constant must be passed as
the parameter constant.
When creating a Function entity, a function should be passed as
the function parameter. This function should take the current
mosaik time and return the desired value. This type of entity only
works when the simulator is started using the “python” method.
In either case, the entity will produce its output on the value
attribute.
This simulator takes the input it is given and writes it into a python dictionary
where the keys are the timestamps of the input and the values are the inputs values.
The dictionary can be retrieved using the get_dict method.
A lot of people were involved in the creation of mosaik and mosaik wouldn’t be
what it is today without any of them:
Martin Tröschel and Astrid Nieße had the original idea for a tool that
lets you integrate existing simulators to perform large-scale Smart Grid
simulations. They also accompanied mosaik’s development many years as group
and project leaders and provided the necessary time and resources for
mosaik’s development.
Steffen Schütte wrote his PhD around mosaik. He and Stefan Scherfke
are the primary authors of mosaik 1.
Ontje Lünsdorf not only contributed code to mosaik 1, but also a lot
of ideas. He held countless discussions with Steffen and Stefan whose results
often greatly improved mosaik.
Stefan Scherfke is the primary author of mosaik 2.0 and 2.1.
Sebastian Rohjans and Sebastian Lehnhoff accompanied mosaik’s development
as group and scientific leaders. They also put lots of effort into making
mosaik open-source software.
Okko Nannen and Florian Schlögl joined the team in May / July 2014.
We’d also like to thank everyone who worked with mosaik and gave us feedback to
make it better.
This is a major upgrade to improve the discrete-event capabilities. Simulators’ steps
can now also be triggered by the output of other simulators.
[NEW] Native support of discrete-event simulations
[NEW] A global time resolution can be set for the scenario.
[NEW] Simulators can request steps asynchronously via set_event() to react to external events.
[NEW] Ability to specify output data as non-persistent (i.e. transient)
[CHANGE] New api 3:
- Simulators have now a type (‘time-based’|’event-based’|’hybrid’).
- time_resolution is passed as argument of the init function.
- max_advance is passed as argument of the step function.
[NEW] When calling the world.start() command for a simulator, users can now set a predefined
value for the posix flag (e.g. True) to prevent automatic detection of the operating system.
This facilitates the creation of some co-simulation cases across OS (e.g. Windows and Linux).
[NEW] Connection option “time_shifted” added as alternative to async_requests. This will
make creating cyclic data dependencies between simulators more usable since usage of
set_data with an API implementation will no longer be needed.
[NEW] API version 2.2: Added an optional “setup_done()” method.
[CHANGE] API version validation: The API version is no longer an integer but
a “major.minor” string. The major part has to math with mosaiks major
version. The minor part may be lower or equal to mosaik’s minor version.
[FIX] Various minor fixes and stability improvements.
[NEW] Mosaik can now perform real-time simulations. Before, this
functionality needed to be implemented by simulators. Now it’s just
World.run(until=x,rt_factor=y), where rt_factor defines the
simulation speed relative to the wall-clock time (issue #24).
[NEW] Simulators can now expose extra methods via their API that can be
called from a mosaik scenario. This allows you to, e.g., store static data in
a data base. These extra API methods need to be specified in the simulator’s
meta data (issue #26).
[NEW] util.connect_many_to_one() helper function.
[NEW] More and better documentation:
Tutorial for integrating simulators, control strategies and for creating
scenarios.
Sim API description
Scenario API description
Sim Manager documentation
Scheduler documentation
Discussion of design decisions
Logo, colors, CI
[NEW] Added util.sync_call() which eases calling proxied methods of
a simulator synchronously.
[CHANGE] The rel attribute in the entity description returned by create()
is now optional.
[CHANGE] Moved proxied methods from SimProxy to SimProxy.proxy in
order to avoid potential name clashes with other attributes.
[CHANGE] Check a simulator’s models and extra API methods for potential name
clashes with the built-in API methods.
[CHANGE] The argument execution_graph of World was renamed to debug.
The execution graph now also stores the time after a simulation step (in
addition to the time before the step).
[FIX] issue #22: The asynchronous requests get_data() and set_data()
now check if the async_requests flag was set in World.connect().
[FIX] issue #23: finalize() is now called for in-process Python
simulators.
[FIX] issue #27: Dramatically improved simulation performance (30 times as
fast in some cases) if simulators use different step sizes (e.g. 1 minute and
1 hour) by improving some internal data structures.
[CHANGE] Separated mosaik’s package and API version. The former stays
a string with a semantic version number; the later is now a simple integer
(issue #17).
[CHANGE] Start/stop timeout for simulators was raised from 2 to 10 seconds.
[CHANGE] Updated the mosaik logo. It now uses the flat colors and has some
improved icon graphics.
[CHANGE] Renamed mosaik.simulator to mosaik.scheduler.
[CHANGE] Entity and the World’s entity graph now store their simulator
name.
[FIX] issue #16: Mosaik now always prints the name of the simulator if it
closes its socket.
[NEW] The model meta data may now contain the any_inputs which, if set
to True, allows any attribute to be connected to that model (useful for
databases and alike).
[CHANGE] The dictionary of input values in the API’s step() call now
also contains the source of a particular value. This is also usefull to for
databases. This may break existing simulators.
[CHANGE] “.” is now used as separator in full entiy IDs instead of “/”
(issue #19).
Mosaik 2 is a complete rewrite of mosaik 1 in order to improve its
maintainability and flexibility. It is still an early alpha version and
neither feature complete nor bug free.
Removed features:
The mosl DSL (including Eclipse xtext and Java) are now gone. Mosaik now
only uses Python.
Mosaik now longer has executables but is now used as a library.
The platform manager is gone.
Mosaik no longer includes a database.
Mosaik no longer includes a web UI.
Mosaik now consists of four core components with the following feature sets:
mosaik API
The API has bean cleaned up and simplified.
Simulators and control strategies share the same API.
There are only four calls from mosaik to a simulator: init, create,
step and get_data.
Simulators / processes can make asynchronous requests to mosaik during a
step: get_progress, get_related_entities, get_data.
ZeroMQ with JSON is replaced by plain network sockets with JSON.
Scenarios:
Pure Python is now used to describe scenarios. This offers you more
flexibility to create complex scenarios.
Scenario creation simplified: Start a simulator to get a model factory.
Use the factory to create model instances (entities). Connect entities.
Run simulation.
Connection rules are are no based on a primitive connect function that
only connects two entities with each other. On top of that, any
connection strategy can be implemented.
Simulation Manager:
Simulators written in Python 3 can be executed in process.
Simulators can be started as external processes.
Mosaik can connect to an already running instance of a simulator. This
can be used as a replacement for the now gone platform manager.
Simulation execution:
The simulation is now event-based. No schedule and no synchronization
points need to be computed.
Simulators can have different and varying step sizes.
Mosaik 1 was nearly a complete rewrite of the previous version and already
incorporated many of the concepts and features described in Steffen Schütte’s
Phd thesis.
It used mosl, a DSL implemented with Eclipse and xtext, to describe
simulators and scenarios. Interprocess communication was done with ZeroMQ and
JSON encoded messages.
This was the first actual version of mosaik that actually worked. However, the
simulators we were using at that time were hard coded into the simulation loop
and we used XML-RPC to communicate with the simulators.
We welcome you to our website. We would like to inform you about the management of your personal data in accordance with Art. 13 General Data Protection Regulation (GDPR).
Controller
The controller responsible for the described data collection and processing is OFFIS e.V., Escherweg 2, 26121 Oldenburg/Germany.
Usage Data
When you visit our website, the data collected from the use of the website is temporarily stored on our web server for statistical purposes in order to improve the quality of our website. This data set contains:
the page, from which the data is requested
the name of the data file,
the date and time of the query,
the amount of data transferred,
the access status (file transmitted, file not found),
a description of the type of browser used,
the IP address of the requesting computer shortened to such an extent that no reidentification of any persona data is possible.
The listed usage data is stored anonymously. The legal basis for the processing of this personal data is provided for in Art. 6 para. 1 lit. f GDPR.
Data Transfer to Third Parties
We do not transfer your personal data to third parties.
Cookies
We use cookies on our website. Cookies are small pieces of data that are stored and read in your end-device. A distinction is made between session cookies, which are deleted when you close your browser, and permanent cookies, which are stored even after your visit has expired. Cookies may contain data that enables the recognition of the device being used. However, in some cases cookies only contain information on certain settings which are not personal data.
We use session cookies and permanent cookies on our website. The data is processed in accordance to Art. 6 para. 1 lit. f GDPR and in the interest of optimizing or enabling user guidance and improving our website presence.
Please be aware that you can set your browser to inform you when cookies are being stored or used on the website you are visiting. Thus, any use of cookies is transparent to you. You have the possibility to delete your browser configuration at any time and prevent any use of new cookies. In the event you refuse the use of cookies, please note that our web sites may not be displayed optimally and some functions are then no longer technically available.
Data Security
To avoid unauthorized access to your data, we have implemented technical and organizational measures. We use encryption technologies on our website. Your data will be transferred to our servers and back again via a connection that is protected by a TLS encryption technology. You can recognize that you are browsing on an encryption secured website by the lock-symbol shown in the address bar of your browser and by the address bar starting with https://.
Your Rights as a User
As a website user, the GDPR grants you certain rights when processing your personal data.
Right of access (Art. 15 GDPR):
You have the right to obtain confirmation as to whether or not personal data concerning you is being processed, and, where that is the case access to the personal data and the information specified in Art. 15 GDPR.
Right to rectification and erasure (Art. 16 and 17 GDPR):
You have the right to obtain without undue delay the rectification of inaccurate personal data concerning you and, if necessary, the right to have incomplete personal data completed.
You also have the right to obtain an erasure of the personal data concerning you without undue delay, if one of the reasons listed in Art. 17 GDPR applies, e.g. if the data is no longer necessary for the intended purpose.
Right to restriction of processing (Art. 18 GDPR):
If one of the conditions set forth in Art. 18 GDPR applies, you shall have the right to restrict the processing of your data to mere storage, e.g. if you revoke consent, to the processing, for the duration of a possible examination.
Right to data portability (Art. 20 GDPR):
In certain situations, listed in Art. 20 GDPR, you have the right to receive the personal data concerning you in a structured, common and machine-readable format or demand a transmission of the data to another third party.
Right to object (Art. 21 GDPR):
If the data is processed pursuant to Art. 6 para. 1 lit. f GDPR (data processing for the purposes of the legitimate interests), you have the right to object to the processing at any time for reasons arising out of your particular situation. We will then no longer process personal data, unless there are demonstrably compelling legitimate grounds for processing, which override the interests, rights and freedoms of the person concerned, or the processing serves the purpose of asserting, exercising or defending legal claims.
Right to lodge a complaint with a supervisory authority:
Pursuant to Art. 77 GDPR, you have the right to lodge a complaint with a supervisory authority if you consider the processing of the data concerning you infringing data protection regulations. The right to lodge a complaint may be invoked in particular in the Member State of your habitual residence, place of work or the place of the alleged infringement.
Contact Details of the Data Protection Officer
Please contact our data protection officer if you have any further questions, suggestions or wishes regarding data protection:
Despite careful control OFFIS assumes no liability for the content of external links. The operators of such a website are solely responsible for its content. At the time of linking the concerned sites were checked for possible violations of law. Illegal contents were not identifiable at that time. A permanent control of the linked pages is not reasonable without specific indications of a violation. Upon notification of violations, OFFIS will remove such links immediately.
Wir nehmen den Schutz Ihrer persönlichen Daten sehr ernst. Ihre Daten werden im Rahmen der gesetzlichen Vorschriften geschützt. Personenbezogene Daten werden auf unseren Internetseiten nur im notwendigen Umfang erhoben. In keinem Fall werden die erhobenen Daten verkauft oder aus anderen Gründen an Dritte weitergegeben.
Verantwortlicher
Verantwortlich für die hier erläuterte Datenverarbeitung ist der OFFIS e.V., Escherweg 2, 26121 Oldenburg.
Erhebung und Verarbeitung von Daten
Jeder Zugriff auf eine unserer Internetseiten und jeder Abruf einer auf den Internetseiten hinterlegten Datei werden von gängigen Webserver-Log-Dateien protokolliert. Die Speicherung dient internen systembezogenen und statistischen Zwecken. Protokolliert wird u.a.:
welche Datei angefordert wurde,
der Name der Datei,
das Datum und die Uhrzeit der Anforderung,
die übertragene Datenmenge,
der Zugriffsstatus (Datei nicht gefunden, Datei übertragen etc.),
der Typ des verwendeten Webbrowsers und
die IP-Adresse des Internetseitenbesuchers
Sämtliche dieser Daten werden ausschließlich anonymisiert gespeichert und ausgewertet. Zu diesem Zweck wird die IP-Adresse des Systems, von dem aus die Internetseite oder Datei angefordert wurde, geeignet anonymisiert. Rechtsgrundlage ist Art. 6 Abs. 1 lit. f DSGVO (Datenschutz-Grundverordnung). Es ist somit weder ein Rückschluss auf eine bestimmte Person möglich, noch erfolgt eine Zusammenführung mit anderen Daten.
Cookies
Darüber hinaus verwenden wir weitere Cookies, um unsere Internetseiten besser auf Ihre Wünsche ausrichten und um statistische Daten über die Nutzung unserer Internetseiten erheben zu können. Durch diese Cookies werden keine Daten erhoben, die einen Rückschluss auf eine bestimmte Person ermöglich. Auch die Installation dieser Cookies wird durch eine entsprechende Browser-Einstellung verhindert. Einmal gesetzte Cookies können Sie jederzeit selbst löschen, indem Sie den entsprechenden Menüpunkt in Ihrem Internet-Browser aufrufen oder die Cookies auf Ihrer Festplatte löschen. Einzelheiten hierzu finden Sie im Hilfemenü Ihres Internet-Browsers.
Weitergehende personenbezogene Daten werden nur erfasst, wenn Sie diese Angaben freiwillig, z.B. im Rahmen einer Anfrage oder Registrierung, machen.
Datensicherheit
Um Ihre Daten vor unerwünschten Zugriffen möglichst umfassend zu schützen, treffen wir technische und organisatorische Maßnahmen. Wir setzen auf unseren Seiten ein Verschlüsselungsverfahren ein. Ihre Angaben werden von Ihrem Rechner zu unserem Server und umgekehrt über das Internet mittels einer TLS-Verschlüsselung übertragen. Sie erkennen dies daran, dass in der Statusleiste Ihres Browsers das Schloss-Symbol geschlossen ist und die Adresszeile mit https:// beginnt.
Bitte beachten Sie, dass außerhalb der vorgenannten Rahmenbedingungen, insbesondere bei der Kommunikation per E-Mail die vollständige Datensicherheit von uns naturgemäß nicht gewährleistet werden kann.
Verwendung der Daten
Wir beachten den Grundsatz der zweckgebundenen Datenverwendung und erheben, verarbeiten und speichern Ihre personenbezogenen Daten nur für die Zwecke, für welche Sie uns diese mitgeteilt haben und für die technische Administration. Eine Weitergabe Ihrer persönlichen Daten an Dritte erfolgt ohne Ihre ausdrückliche Einwilligung nicht, sofern dies nicht zur Erbringung der Dienstleistung oder zur Vertragsdurchführung notwendig ist. Auch die Übermittlung an auskunftsberechtigte staatliche Institution und Behörden erfolgt nur im Rahmen der gesetzlichen Auskunftspflichten oder wenn wir durch eine gerichtliche Entscheidung zur Auskunft verpflichtet werden.
Löschung der Daten
Eine Löschung der gespeicherten personenbezogenen Daten erfolgt, wenn Sie Ihre Einwilligung zur Speicherung widerrufen, wenn deren Kenntnis zur Erfüllung des mit der Speicherung verfolgten Zwecks nicht mehr erforderlich ist oder wenn deren Speicherung aus sonstigen gesetzlichen Gründen unzulässig ist.
Ihre Rechte als Nutzer
Bei Verarbeitung Ihrer personenbezogenen Daten gewährt die DSGVO Ihnen als Webseitennutzer bestimmte Rechte:
Auskunftsrecht (Art. 15 DSGVO):
Sie haben das Recht eine Bestätigung darüber zu verlangen, ob sie betreffende personenbezogene Daten verarbeitet werden; ist dies der Fall, so haben Sie ein Recht auf Auskunft über diese personenbezogenen Daten und auf die in Art. 15 DSGVO im einzelnen aufgeführten Informationen.
Recht auf Berichtigung und Löschung (Art. 16 und 17 DSGVO):
Sie haben das Recht, unverzüglich die Berichtigung sie betreffender unrichtiger personenbezogener Daten und ggf. die Vervollständigung unvollständiger perso-nenbezogener Daten zu verlangen.
Sie haben zudem das Recht, zu verlangen, dass sie betreffende personenbezogene Daten unverzüglich gelöscht werden, sofern einer der in Art. 17 DSGVO im einzelnen aufgeführten Gründe zutrifft, z. B. wenn die Daten für die verfolgten Zwecke nicht mehr benötigt werden.
Recht auf Einschränkung der Verarbeitung (Art. 18 DSGVO):
Sie haben das Recht, die Einschränkung der Verarbeitung zu verlangen, wenn eine der in Art. 18 DSGVO aufgeführten Voraussetzungen gegeben ist, z. B. wenn Sie Widerspruch gegen die Verarbeitung eingelegt haben, für die Dauer einer etwai-gen Prüfung.
Recht auf Datenübertragbarkeit (Art. 20 DSGVO):
In bestimmten Fällen, die in Art. 20 DSGVO im Einzelnen aufgeführt werden, haben Sie das Recht, die sie betreffenden personenbezogenen Daten in einem struktu-rierten, gängigen und maschinenlesbaren Format zu erhalten bzw. die Übermittlung dieser Daten an einen Dritten zu verlangen.
Widerspruchsrecht (Art. 21 DSGVO):
Werden Daten auf Grundlage von Art. 6 Abs. 1 lit. f erhoben (Datenverarbeitung zur Wahrung berechtigter Interessen), steht Ihnen das Recht zu, aus Gründen, die sich aus Ihrer besonderen Situation ergeben, jederzeit gegen die Verarbeitung Wider-spruch einzulegen. Wir verarbeiten die personenbezogenen Daten dann nicht mehr, es sei denn, es liegen nachweisbar zwingende schutzwürdige Gründe für die Verarbeitung vor, die die Interessen, Rechte und Freiheiten der betroffenen Person überwiegen, oder die Verarbeitung dient der Geltendmachung, Ausübung oder Verteidigung von Rechtsansprüchen.
Beschwerderecht bei einer Aufsichtsbehörde
Sie haben gem. Art. 77 DSGVO das Recht auf Beschwerde bei einer Aufsichtsbehör-de, wenn Sie der Ansicht sind, dass die Verarbeitung der Sie betreffenden Daten gegen datenschutzrechtliche Bestimmungen verstößt. Das Beschwerderecht kann insbesondere bei einer Aufsichtsbehörde in dem Mitgliedstaat Ihres Aufenthaltsorts, Ihres Arbeitsplatzes oder des Orts des mutmaßlichen Verstoßes geltend gemacht werden.
A program that is intended to observe and manipulate the state of
objects (simulated or real) of a power system or those that are somehow
connected to the power system; for example a multi-agent system that
controls the feed-in of decentralized producers.
In co-simulation the different subsystems which form a coupled problem
are modeled and simulated in a distributed manner. The modeling is done
on the subsystem level without having the coupled problem in mind. The
coupled simulation is carried out by running the subsystems in a
black-box manner. During the simulation the subsystems will exchange
data. (source: Wikipedia)
Represents an instance of a Model within a mosaik
simulation. Entities can be connected to establish a data-flow
between them. Examples are the nodes and lines of a power grid or single
electric vehicles.
A Model is a simplified representation of a real world object or system.
It reproduces the relevant aspects of that object or system for its
systematic analysis.
Description of the system to be simulated. It includes the used
models and their relations. It includes the state
of the models and their data base. In the mosaik-context it includes also
the simulators.
A program that contains the implementation of one or more
simulation models and is able to execute these
models (that is, to perform a simulation).
Sometimes, the term simulator also refers all kinds of processes that
can talk to mosaik, including actual simulators, control strategies,
visualization servers, database adapters and so on.
An electric power system that utilizes information exchange and control
technologies, distributed computing and associated sensors and actuators,
for purposes such as:
to integrate the behaviour and actions of the network users and other stakeholders,
to efficiently deliver sustainable, economic and secure electricity supplies.
Mosaik executes simulators in discrete time steps. The step size of a
time-based simulator can be an arbitrary integer. It can also vary
during the simulation. Event-based simulators are stepped whenever its
inputs are updated (by other simulators). They can also schedule steps
for themselves.
Mosaik uses integers for the representation of time (to avoid rounding
errors etc.). It’s unit (i.e. to how many seconds one integer step
corresponds) can be defined in the scenario, and is passed to every
simulation component via the init function as key-word
parameter time_resolution. It’s a floating point number and defaults to *1..