Porting a EUDAQ Producer#
This how-to goes through the steps necessary to port a EUDAQ Producer to the Constellation framework. In Constellation terminology, a Producer is called a Satellite. As will become clear soon, both the finite state machine that governs the life cycle of a satellite as well as the mechanism for logging information and transmitting data are very similar from an interface perspective.
See also
It is recommend to read through the concepts of a satellite and the satellite implementation tutorial first to get a good understanding of how a satellite works and how they are implemented in C++.
Porting the Finite State Machine Transitions#
Constellation satellites are built around a finite state machine which works similar to the one used in EUDAQs Producers, with the exception that the states have different names and a few more transitions between states are possible. In order to port the functionality of the EUDAQ Producer, the code from its functions can directly be copied into a corresponding Constellation satellite skeleton. The following table helps in finding the corresponding function name:
EUDAQ Producer |
Constellation Satellite |
Notes |
---|---|---|
|
|
Direct equivalent. In EUDAQ, this function only receives the “init” portion of the configuration, in Constellation the entire satellite configuration is provided at this stage. |
|
|
Direct equivalent. Powers up the attached instrument and configures it to a state where it is ready to enter data taking. |
- |
|
Landing is the opposite action as launching and should shut down the attached instrument in a controlled manner to bring it back into the INIT state. EUDAQ does not have a direct equivalent, but |
- |
|
Reconfiguration is a concept specific to Constellation. It is an optional method which allows to alter a subset of configuration values without landing first. |
|
|
Direct equivalent. Starts a new data taking run. |
|
|
Direct equivalent. Stops the run currently in progress. |
|
|
Equivalent with the difference that instead of keeping track of the running via a custom variable (like |
|
- |
In EUDAQ, this command puts the producer back into the initial state. In contrast, resetting a satellite in Constellation means calling its |
|
Destructor |
The EUDAQ equivalent of the |
|
- |
In Constellation no |
Transmitting Data#
Data in EUDAQ is transmitted as RawEvent
objects. There is the possibility of adding multiple so-called data blocks to a single
event, as well as the option to store events as sub-events of others. The creation, outfitting and sending of a message in EUDAQ
follows roughly this pattern:
// Create new event and set its ID
auto event = eudaq::Event::MakeUnique("MyDetectorEvent");
event->SetEventN(m_ev);
// Add data to the event
event->AddBlock(0, data);
// Possibly add a tag
event->SetTag("my_tag", std::to_string(my_value));
// Send the event
SendEvent(std::move(event));
Constellation data messages consist of a header with metadata such as the sending satellite and the continuous message sequence, and any number of frames with binary data. These frames can be likened to the EUDAQ data blocks. A concept akin to sub-events does not exits in Constellation. Knowing this, the above sequence would translate to the following satellite code:
// Create new message, message sequence is handled by Constellation
auto msg = newDataMessage();
// Add data to the message
msg.addFrame(std::move(data));
// Possibly add a tag
msg.addTag("my_tag", my_value)
// Send the message
sendDataMessage(msg);
It should be noted that tag values in EUDAQ are limited to std::string
while Constellation tags can hold any configuration
data type.
See also
Further information on data transmission can be found in here.
Adjusting the Logging Mechanism#
In EUDAQ, log messages are sent via the logging macros EUDAQ_DEBUG
, EUDAQ_INFO
, EUDAQ_WARN
, and EUDAQ_ERROR
which
take a single string as argument. In Constellation, similar log levels exist and logging can
directly be translated to the corresponding macros. Instead of single strings, output streams are used which are more
flexible in converting different variable types to strings automatically. For example the EUDAQ log message:
EUDAQ_DEBUG("Current temperature for channel " + std::to_string(ch) + " is " + std::to_string(t) + "C");
turns into the equivalent Constellation log of the same verbosity level:
LOG(DEBUG) << "Current temperature for channel " << ch << " is " << t << "C";
Error Handling#
In addition to the logging macros, EUDAQ provides the EUDAQ_THROW
macro which can be used to throw exceptions. Instead of
providing macros for this purpose, Constellation encourages the direct use of exceptions. This not only improves clarity but
it also allows for the use of different exceptions classes for different purposes. A EUDAQ error handling code such as
auto power_output_percent = config->Get("power_output_percent", 0);
if(power_output_percent > 100) {
EUDAQ_THROW("Cannot set output power to " + std::to_string(power_output_percent) + "%, 100% is the maximum!");
}
would become a Constellation exception of type InvalidValueError
. This allows the message to be more informative and to
tell the user e.g. in which configuration file and at which position the invalid value can be found:
auto power_output_percent = config.get<int>("power_output_percent");
if(power_output_percent > 100) {
throw InvalidValueError(config, "power_output_percent", "Value too large, 100% is the maximum!");
}
Moreover, the attempt to read configuration keys without default value will automatically emit a MissingKeyError
exception
if the respective key is not present in the configuration.