| Simopticon
    1.1.0
    A framework for the optimization of simulation parameters. | 
Simopticon is a framework which automates the search for optimal parameters for simulated processes. The key strategy is to define parameters that shall be optimized, automatically run simulations with certain parameters, evaluate their performance by calculating a number rating (the lower, the better) and trying to find parameter combinations that minimize the rating.
The described process is distributed over four major components:
Extensions of the framework may introduce new Optimizer, SimulationRunner and Evaluation implementations (see Extension). Currently, there is only one implementation of SimulationRunner and Evaluation, tailored for the optimization of platoon controllers using the Plexe framework. The available Optimizers are explained in Available Optimizers.
The full API documentation may be found on peternaggschga.github.io/Simopticon or in the comprehensive PDF file provided. A more in-depth explanation of Simopticon and its design principles may be found in the german bachelor's thesis that proposed the framework.
The following sections describe the requirements your machine has to fulfill to run Simopticon. They may differ depending on the Optimizer, SimulationRunner and Evaluation implementations you plan to use, therefore, the implementations have their own dependency sections.
The framework itself is developed for Debian-based Unix/Linux machines. Other operating systems might work but are not actively supported. To be able to install the framework, you need the following software:
To enable simulations with Plexe, Version 3.1 of the framework must be installed. Refer to the Plexe install guide for more information. Please mind that you might want to install OMNeT++ Version 6 or higher in order to use the ConstantHeadway Evaluation, even though the installation guide might suggest an older version.
To use the ConstantHeadway Evaluation, OMNeT++ Version 6 or higher is needed. Please refer to the OMNeT++ Install Guide for more information on the requirements.
Check whether Git is installed on your machine and install it if necessary using:
CMake Version 3.25 or higher is needed for building Simopticon. If you don't have CMake installed, follow the guide below. If you have an older version installed, you must first remove it.
First, make sure to install g++ and OpenSSL Development tools.
Then you need to download the latest version of CMake from their download page — search for the source distribution tar package. Unpack the downloaded package using:
Open the newly created directory and run the configuration script with:
When the configuration has completed successfully, you are ready to build and install using:
You may remove the downloaded tar file and extracted directory if needed.
Check whether Python3 development tools are installed on your machine and install them if necessary using:
Go to the directory you want to install Simopticon in, e.g. ~/src. To get the source code, clone the git repository using:
Create a build directory in the downloaded files with:
Build Simopticon by calling:
The resulting executable simopticon may be copied to other locations or referenced via symlinks for more convenient access. The same applies to the config directory in ~/src/simopticon which is used to configure the optimization process (see Usage).
To upgrade to the latest version of Simopticon, the latest release must be pulled and recompiled. Go to the directory, you installed Simopticon in, e.g. ~/src/simopticon. Then pull from master using:
Go to the build directory and rebuild the executable by calling:
The resulting simopticon executable file contains the latest version of Simopticon.
The optimization process and its components are configured using several JSON files. Default examples of such files can be found in the config directory. Be aware, however, that the default files in config must be edited before use, since some file paths must be set which depend on your filesystem.
The options in the JSON files are commented and therefore self-explanatory. The following sections only show options that must be changed to successfully run optimizations.
The main configuration can be found in config/simopticon.json. It contains settings of the Controller and selects the other components. In the controller settings, the key params must be set to reference another JSON file containing an array of ParameterDefinition that are to be optimized.
The main configuration selects which Optimizer, SimulationRunner and Evaluation implementations are to be used. For each of those components, a name of the implementation and a reference to a JSON file configuring it must be given. References are used because different implementations of the same component may vastly differ in their configurable options, and switching the used components gets easier this way.
If you want to use PlexeSimulationRunner, you need to configure config/runners/plexe.json. There you have to set the configDirectory key to match the path to the directory containing your Plexe configuration (omnetpp.ini). For default installations that should be something along the lines of [installation-directory]/plexe/examples/platooning.
If you want to use ConstantHeadway evaluation, you need to configure config/evaluations/constant_headway.json. There you have to set the pythonScript and the omnetppDirectory keys. pythonScript must point to the script constant_headway.py which can be found in src/evaluation/constant_headway. omnetppDirectory must point to the directory where OMNeT++ Version 6 or higher is installed, e.g. ~/src/omnetpp-6.0.1.
Simopticon contains implementations of multiple optimization strategies, which are shortly described here. Which algorithm is used can be selected in the main config (config/simopticon.json).
The DIRECT algorithm is a global optimization algorithm motivated by Lipschitzian optimization. DIRECT is deterministic and tailored to low-dimensional problems. It partitions the search space iteratively into increasingly smaller rectangles which are each sampled on opposing vertices. DIRECT decides which rectangles are explored further, based on the value of those samples and the size of the rectangles. That way, rectangles are sampled either because they yield good values, or because they are large and therefore not yet sampled in detail. This leads to a balance between local refinement and global optimization.
The concrete implementation of DIRECT in Simopticon is a derivative of Adaptive Diagonal Curves and MrDIRECT which are both derivatives of the original DIRECT algorithm. For a more in-depth explanation of the implemented algorithm refer to the german bachelor's thesis that proposed it.
Monte Carlo methods are a simple class of random algorithms and therefore not deterministic. When applied to optimization problems, they show great performance despite their simplicity. Basically, the algorithm iteratively selects random values to be tested and evaluates them.
The RandomNeighbor Optimizer is an implementation of random-restart stochastic hill climbing. It starts at a random point in the search space. In the next step, a new point is randomly chosen from the neighborhood of the current optimum. The next point is chosen completely random (i.e. independent of the optimum) with a predefined probability to ensure global search.
The optimization is invoked on the command line by executing the program built in Setup. The call on the command line has one mandatory and one optional argument. The First argument must be the path to the main config, i.e. config/simopticon.json. A valid call to an optimization could be:
If a second argument is given, instead of running actual simulations with the configured SimulationRunner and evaluating their results with an Evaluation, the StubController is used. StubController can be used to implement and optimize benchmark functions to test Optimizer implementations without relying on actual costly simulations. The second argument holds the name of the function to be optimized, i.e., one of the following:
A valid call to the optimization of a benchmark function could be:
Please note that you need to define the optimized parameters in config/simopticon.json even when you are optimizing a benchmark.
This section goes through the steps you need to undertake to extend the framework with new Optimizer, SimulationRunner or Evaluation implementations.
When developing new implementations of components, please stick to the project structure — Optimizer extensions go into src/optimizer, SimulationRunner extensions go into src/runner and Evaluation extensions go into src/evaluation. If your implementation needs a more sophisticated implementation of the Parameter class than the ones provided in src/parameters, feel free to extend the abstract Parameter class.
Please document your code using Doxygen comments!
The src/Types.h header file defines framework-wide types such as functionValue for values returned by the Evaluation component or coordinate which is used to store Parameter values. The src/ComparisonFunctions.h header file defines comparison functions, which can be used in STL containers that are ordered. E.g. CmpVectorSharedParameter can be used to compare two objects of type vector<shared_ptr<Parameter>>.
To add a new optimization strategy, you have to extend the Optimizer class. You need to override the Optimizer::runOptimization method which should start the optimization process and only return when your strategy is finished or if the Optimizer::abort method is called which you should implement too.
Optimizer extensions can instruct the Controller to start simulations and evaluate them with the Optimizer::requestValues method. Please try to commission as many Parameters as possible in one call of the method so the other components may parallelize calculations.
Please consider overriding the methods provided by the Status interface to give the user a sense of what is happening.
To add a new way of executing simulations, you have to extend the SimulationRunner class. You need to override the SimulationRunner::work function, which is run concurrently for all Parameter vectors provided to SimulationRunner::runSimulations. If you want to prohibit concurrent execution, you may override SimulationRunner::runSimulations instead (in that case, SimulationRunner::work should return an empty pair). See documentation of Multithreaded class for more information on that.
SimulationRunner::work should run a simulation with the given parameters and return a path to the result files and a set of identifiers relating to simulation runs. The interface for the identifiers is very loosely defined — if your Evaluation does not need any identifiers of simulation runs, you may return an empty set. Please be aware that the Controller might try to delete the path you return after some time, so that should not be an empty path! Other than that, it is not further standardized what must be returned as a path and identifiers as long as your Evaluation component can evaluate the simulation based on the returned information.
Please consider overriding the methods provided by the Status interface to give the user a sense of what is happening.
To add a new rating algorithm based on simulation data, you have to extend the Evaluation class. You need to override the Evaluation::processOutput function, which conducts the rating of simulation performance based on the path to the result files and the given identifiers. This process heavily depends on the implemented SimulationRunner, which is responsible for returning result files and run identifiers if necessary. Your Evaluation implementation should rate the given simulation results with a functionValue — the lower, the better.
Please consider overriding the methods provided by the Status interface to give the user a sense of what is happening.
All newly added classes must be registered in CMakeList.txt so the compiler does not ignore them! External dependencies and added libraries should be included there too.
To make your new component available for configuration, you must add it to the constructor of the Controller class. Let's assume you wrote a new Optimizer implementation. First you need to create a JSON configuration file in config/optimizer. There you can define any desired options for your component.
The next step is editing the Controller class to make your Optimizer available. To do that, you find the "Optimizer settings" in the constructor of the Controller. There you add another case to the if-Statement where opt equals the name of your component (this is the name that will be set in the main config later, see Configuration). In the added case you can read the necessary options from the JSON object in optimizerConfig. You have to set Controller::optimizer to an unique_ptr<Optimizer>, owning a new instance of your Optimizer implementation.
When this setup is complete, you may build the framework again and update the main configuration to use your new Optimizer by changing the optimizer.optimizer key to the name of your Optimizer and the optimizer.config key to the path of your created JSON configuration file.