Project
Loading...
Searching...
No Matches
test_DeviceSpecHelpers.cxx
Go to the documentation of this file.
1// Copyright 2019-2020 CERN and copyright holders of ALICE O2.
2// See https://alice-o2.web.cern.ch/copyright for details of the copyright holders.
3// All rights not expressly granted are reserved.
4//
5// This software is distributed under the terms of the GNU General Public
6// License v3 (GPL Version 3), copied verbatim in the file "COPYING".
7//
8// In applying this license CERN does not waive the privileges and immunities
9// granted to it by virtue of its status as an Intergovernmental Organization
10// or submit itself to any jurisdiction.
11
12#include "Mocking.h"
17#include "../src/DeviceSpecHelpers.h"
18#include <catch_amalgamated.hpp>
19#include <sstream>
20#include <cstring>
21#include <vector>
22#include <map>
23#include <cstring>
24#include "../src/SimpleResourceManager.h"
25#include "../src/ComputingResourceHelpers.h"
26
27namespace o2::framework
28{
29
30using CheckMatrix = std::map<std::string, std::vector<std::pair<std::string, std::string>>>;
31
32// search for an option in the device execution
33bool search(DeviceExecution const& execution, std::string const& option, std::string const& argument)
34{
35 bool foundOption = false;
36 for (auto const& execArg : execution.args) {
37 if (execArg == nullptr) {
38 break;
39 }
40 if (!foundOption) {
41 foundOption = option == execArg;
42 } else if (argument == execArg) {
43 return true;
44 } else {
45 // the required argument to the option is not found
46 foundOption = false;
47 }
48 }
49 return false;
50}
51
52// create the device execution from the device specs and process the command line arguments
53// the check matrix contains a map of options to be founf per processor
54void check(const std::vector<std::string>& arguments,
55 const std::vector<ConfigParamSpec>& workflowOptions,
56 const std::vector<DeviceSpec>& deviceSpecs,
57 CheckMatrix& matrix)
58{
59 std::stringstream output;
60 for (auto const& arg : arguments) {
61 output << " " << arg;
62 }
63
64 std::vector<DeviceExecution> deviceExecutions(deviceSpecs.size());
65 std::vector<DeviceControl> deviceControls(deviceSpecs.size());
66 std::vector<DataProcessorInfo> dataProcessorInfos;
67 for (auto& [name, _] : matrix) {
68 dataProcessorInfos.push_back(DataProcessorInfo{
69 .name = name,
70 .executable = "executable-name",
71 .cmdLineArgs = arguments,
72 .workflowOptions = workflowOptions,
73 });
74 }
75 DriverConfig driverConfig{};
76 DeviceSpecHelpers::prepareArguments(true, true, false, 8080,
77 driverConfig,
78 dataProcessorInfos,
79 deviceSpecs,
80 deviceExecutions,
81 deviceControls, {},
82 "workflow-id");
83
84 for (size_t index = 0; index < deviceSpecs.size(); index++) {
85 const auto& deviceSpec = deviceSpecs[index];
86 const auto& deviceExecution = deviceExecutions[index];
87 std::stringstream execArgs;
88 for (const auto& arg : deviceExecution.args) {
89 if (arg == nullptr) {
90 // the nullptr terminates the argument list
91 break;
92 }
93 execArgs << " " << arg;
94 }
95 for (auto const& testCase : matrix[deviceSpec.name]) {
96 INFO(std::string("can not find option: ") + testCase.first + " " + testCase.second);
97 REQUIRE(search(deviceExecution, testCase.first, testCase.second));
98 }
99 }
100}
101
102TEST_CASE("test_prepareArguments")
103{
104 std::vector<ConfigParamSpec> workflowOptions{
105 {"foo", VariantType::String, "bar", {"the famous foo option"}},
106 {"depth", VariantType::Int, 1, {"number of processors"}},
107 };
108
109 auto algorithm = [](ProcessingContext& ctx) {};
110
111 WorkflowSpec workflow{
112 {"processor0",
113 {},
114 {OutputSpec{{"output"}, "TST", "DUMMYDATA", 0, Lifetime::Timeframe}},
115 AlgorithmSpec(algorithm),
116 Options{
117 {"mode", VariantType::String, "default", {"The Mode"}},
118 }},
119 {"processor1",
120 {InputSpec{"input", "TST", "DUMMYDATA", 0, Lifetime::Timeframe}},
121 {},
122 AlgorithmSpec(algorithm),
123 Options{
124 {"mode", VariantType::String, "default", {"The Mode"}},
125 }},
126 };
127
128 std::vector<DeviceSpec> deviceSpecs;
129
130 std::vector<ComputingResource> resources = {ComputingResourceHelpers::getLocalhostResource()};
131 auto rm = std::make_unique<SimpleResourceManager>(resources);
132
133 auto configContext = makeEmptyConfigContext();
134 auto channelPolicies = ChannelConfigurationPolicy::createDefaultPolicies(*configContext);
135 auto callbacksPolicies = CallbacksPolicy::createDefaultPolicies();
136
138 channelPolicies,
140 callbacksPolicies,
141 deviceSpecs,
142 *rm, "workflow-id", *configContext);
143
144 // Now doing the test cases
145 CheckMatrix matrix;
146
147 // checking with empty command line arguments, all processors must have the options with
148 // default arguments
149 matrix["processor0"] = {{"--depth", "1"}, {"--foo", "bar"}, {"--mode", "default"}};
150 matrix["processor1"] = matrix["processor0"];
151 check({}, workflowOptions, deviceSpecs, matrix);
152
153 // checking with unknown arguments, silently ignored, same test matrix
154 check({"--unknown", "option"}, workflowOptions, deviceSpecs, matrix);
155
156 // configuring mode, both devices must have the option set
157 matrix["processor0"] = {{"--depth", "1"}, {"--foo", "bar"}, {"--mode", "silly"}};
158 matrix["processor1"] = matrix["processor0"];
159 check({"--mode", "silly"}, workflowOptions, deviceSpecs, matrix);
160
161 // configuring option group, only processor0 must have the option set, processor1 default
162 matrix["processor0"] = {{"--depth", "1"}, {"--foo", "bar"}, {"--mode", "silly"}};
163 matrix["processor1"] = {{"--depth", "1"}, {"--foo", "bar"}, {"--mode", "default"}};
164 check({"--processor0", "--mode silly"}, workflowOptions, deviceSpecs, matrix);
165
166 // processor0 must have the mode set to silly via option group, processor1 advanced from the argument
167 matrix["processor0"] = {{"--depth", "1"}, {"--foo", "bar"}, {"--mode", "silly"}};
168 matrix["processor1"] = {{"--depth", "1"}, {"--foo", "bar"}, {"--mode", "advanced"}};
169 check({"--mode", "advanced", "--processor0", "--mode silly"}, workflowOptions, deviceSpecs, matrix);
170
171 // both devices have the workflow option propagated, others defaulted
172 matrix["processor0"] = {{"--depth", "2"}, {"--foo", "bar"}, {"--mode", "default"}};
173 matrix["processor1"] = matrix["processor0"];
174 check({"--depth", "2"}, workflowOptions, deviceSpecs, matrix);
175
176 // both devices have the workflow option propagated, processor0 mode silly via option group
177 matrix["processor0"] = {{"--depth", "2"}, {"--foo", "bar"}, {"--mode", "silly"}};
178 matrix["processor1"] = {{"--depth", "2"}, {"--foo", "bar"}, {"--mode", "default"}};
179 check({"--depth", "2", "--processor0", "--mode silly"}, workflowOptions, deviceSpecs, matrix);
180}
181
182TEST_CASE("CheckOptionReworking")
183{
184 {
185 std::vector<DataProcessorInfo> infos = {
186 {.cmdLineArgs = {"--driver-client-backend", "foo"}},
187 {}};
188 DeviceSpecHelpers::reworkHomogeneousOption(infos, "--driver-client-backend", "stdout://");
189 REQUIRE(infos[0].cmdLineArgs[1] == "foo");
190 REQUIRE(infos[1].cmdLineArgs[1] == "foo");
191 }
192 {
193 std::vector<DataProcessorInfo> infos = {
194 {.cmdLineArgs = {"--driver-client-backend", "foo"}},
195 {.cmdLineArgs = {"--driver-client-backend", "bar"}}};
196 REQUIRE_THROWS_AS(DeviceSpecHelpers::reworkHomogeneousOption(infos, "--driver-client-backend", "stdout://"), o2::framework::RuntimeErrorRef);
197 }
198 {
199 std::vector<DataProcessorInfo> infos = {
200 {.cmdLineArgs = {"--driver-client-backend", "foo"}},
201 {.cmdLineArgs = {"--driver-client-backend", "foo"}}};
202 DeviceSpecHelpers::reworkHomogeneousOption(infos, "--driver-client-backend", "stdout://");
203 REQUIRE(infos[0].cmdLineArgs[1] == "foo");
204 REQUIRE(infos[1].cmdLineArgs[1] == "foo");
205 }
206 {
207 std::vector<DataProcessorInfo> infos = {
208 {.cmdLineArgs = {"foo", "bar"}},
209 {.cmdLineArgs = {"fnjcnak", "foo"}}};
210 DeviceSpecHelpers::reworkHomogeneousOption(infos, "--driver-client-backend", "stdout://");
211 REQUIRE(infos[0].cmdLineArgs[3] == "stdout://");
212 REQUIRE(infos[1].cmdLineArgs[3] == "stdout://");
213 }
214 {
215 std::vector<DataProcessorInfo> infos = {
216 {.cmdLineArgs = {"foo", "bar", "--driver-client-backend", "bar"}},
217 {.cmdLineArgs = {"fnjcnak", "foo"}}};
218 DeviceSpecHelpers::reworkHomogeneousOption(infos, "--driver-client-backend", "stdout://");
219 REQUIRE(infos[0].cmdLineArgs[3] == "bar");
220 REQUIRE(infos[1].cmdLineArgs[3] == "bar");
221 }
222}
223
224TEST_CASE("CheckIntegerReworking")
225{
226 {
227 std::vector<DataProcessorInfo> infos = {
228 {{}, {}, {"--readers", "2"}},
229 {}};
231 infos, "--readers", nullptr, 1, [](long long x, long long y) { return x > y ? x : y; });
232 REQUIRE(infos[0].cmdLineArgs[1] == "2");
233 REQUIRE(infos[1].cmdLineArgs[1] == "2");
234 }
235 {
236 std::vector<DataProcessorInfo> infos = {
237 {},
238 {}};
240 infos, "--readers", nullptr, 1, [](long long x, long long y) { return x > y ? x : y; });
241 REQUIRE(infos[0].cmdLineArgs.size() == 0);
242 REQUIRE(infos[1].cmdLineArgs.size() == 0);
243 }
244 {
245 std::vector<DataProcessorInfo> infos = {
246 {},
247 {}};
249 infos, "--readers", []() { return 1; }, 3, [](long long x, long long y) { return x > y ? x : y; });
250 REQUIRE(infos[0].cmdLineArgs.size() == 2);
251 REQUIRE(infos[1].cmdLineArgs.size() == 2);
252 REQUIRE(infos[0].cmdLineArgs[1] == "1");
253 REQUIRE(infos[1].cmdLineArgs[1] == "1");
254 }
255 {
256 std::vector<DataProcessorInfo> infos = {
257 {{}, {}, {"--readers", "2"}},
258 {{}, {}, {"--readers", "3"}}};
260 infos, "--readers", []() { return 1; }, 1, [](long long x, long long y) { return x > y ? x : y; });
261 REQUIRE(infos[0].cmdLineArgs.size() == 2);
262 REQUIRE(infos[1].cmdLineArgs.size() == 2);
263 REQUIRE(infos[0].cmdLineArgs[1] == "3");
264 REQUIRE(infos[1].cmdLineArgs[1] == "3");
265 }
266 {
267 std::vector<DataProcessorInfo> infos = {
268 {{}, {}, {"--readers", "3"}},
269 {{}, {}, {"--readers", "2"}}};
271 infos, "--readers", []() { return 1; }, 1, [](long long x, long long y) { return x > y ? x : y; });
272 REQUIRE(infos[0].cmdLineArgs.size() == 2);
273 REQUIRE(infos[1].cmdLineArgs.size() == 2);
274 REQUIRE(infos[0].cmdLineArgs[1] == "3");
275 REQUIRE(infos[1].cmdLineArgs[1] == "3");
276 }
277 {
278 std::vector<DataProcessorInfo> infos = {
279 {.cmdLineArgs = {"foo", "bar", "--readers", "3"}},
280 {.cmdLineArgs = {"--readers", "2"}}};
282 infos, "--readers", []() { return 1; }, 1, [](long long x, long long y) { return x > y ? x : y; });
283 REQUIRE(infos[0].cmdLineArgs.size() == 4);
284 REQUIRE(infos[1].cmdLineArgs.size() == 2);
285 REQUIRE(infos[0].cmdLineArgs[3] == "3");
286 REQUIRE(infos[1].cmdLineArgs[1] == "3");
287 }
288}
289
290TEST_CASE("Check validity of the workflow")
291{
292 WorkflowSpec workflow1{
293 {
294 .name = "processor0",
295 .outputs = {OutputSpec{{"output"}, "TST", "DUMMYDATA", 0, Lifetime::Timeframe}},
296 },
297 {
298 .name = "processor1",
299 .inputs = {InputSpec{"input", "TST", "DUMMYDATA", 0, Lifetime::Timeframe}},
300 },
301 };
302 REQUIRE_NOTHROW(DeviceSpecHelpers::validate(workflow1));
303
304 WorkflowSpec workflow2{
305 {
306 .name = "processor0",
307 .outputs = {OutputSpec{{"output"}, "TST", "DUMMYDATA", 0, Lifetime::Timeframe},
308 OutputSpec{{"output2"}, "TST", "DUMMYDATA", 0, Lifetime::Timeframe}},
309 },
310 {
311 .name = "processor1",
312 .inputs = {InputSpec{"input", "TST", "DUMMYDATA", 0, Lifetime::Timeframe}},
313 },
314 };
315 REQUIRE_THROWS(DeviceSpecHelpers::validate(workflow2));
316
317 WorkflowSpec workflow3{
318 {
319 .name = "processor0",
320 .outputs = {OutputSpec{{"output"}, "TST", "DUMMYDATA", 0, Lifetime::Timeframe},
321 OutputSpec{{"output2"}, "TST", "DUMMYDATA", 2, Lifetime::Sporadic}},
322 },
323 {
324 .name = "processor1",
325 .outputs = {OutputSpec{{"output"}, "TST", "DUMMYDATA", 1, Lifetime::Timeframe},
326 OutputSpec{{"output2"}, "TST", "DUMMYDATA", 2, Lifetime::Sporadic}},
327 },
328 {
329 .name = "processor2",
330 .inputs = {InputSpec{"input", "TST", "DUMMYDATA", 2, Lifetime::Timeframe}},
331 },
332 };
333 REQUIRE_NOTHROW(DeviceSpecHelpers::validate(workflow3));
334
335 WorkflowSpec workflow4{
336 {
337 .name = "processor0",
338 .outputs = {OutputSpec{{"output"}, "TST", "DUMMYDATA", 0, Lifetime::Timeframe},
339 OutputSpec{{"output2"}, "TST", "DUMMYDATA", 2, Lifetime::Sporadic}},
340 },
341 {
342 .name = "processor1",
343 .outputs = {OutputSpec{{"output"}, "TST", "DUMMYDATA", 0, Lifetime::Timeframe},
344 OutputSpec{{"output2"}, "TST", "DUMMYDATA", 2, Lifetime::Sporadic}},
345 },
346 {
347 .name = "processor2",
348 .inputs = {InputSpec{"input", "TST", "DUMMYDATA", 2, Lifetime::Timeframe}},
349 },
350 };
351 REQUIRE_THROWS(DeviceSpecHelpers::validate(workflow4));
352}
353
354TEST_CASE("CheckReworkingEnv")
355{
356 DeviceSpec spec{.inputTimesliceId = 1};
357 std::string env = "FOO={timeslice0}";
358 REQUIRE(DeviceSpecHelpers::reworkTimeslicePlaceholder(env, spec) == "FOO=1");
359 env = "FOO={timeslice0} BAR={timeslice4}";
360 REQUIRE(DeviceSpecHelpers::reworkTimeslicePlaceholder(env, spec) == "FOO=1 BAR=5");
361 env = "FOO={timeslice0} BAR={timeslice4} BAZ={timeslice5}";
362 REQUIRE(DeviceSpecHelpers::reworkTimeslicePlaceholder(env, spec) == "FOO=1 BAR=5 BAZ=6");
363 env = "";
364 REQUIRE(DeviceSpecHelpers::reworkTimeslicePlaceholder(env, spec) == "");
365 env = "Plottigat";
366 REQUIRE(DeviceSpecHelpers::reworkTimeslicePlaceholder(env, spec) == "Plottigat");
367 env = "{timeslice}";
368 REQUIRE(DeviceSpecHelpers::reworkTimeslicePlaceholder(env, spec) == "{timeslice}");
369 env = "{timeslicepluto";
370 REQUIRE(DeviceSpecHelpers::reworkTimeslicePlaceholder(env, spec) == "{timeslicepluto");
371}
372
373} // namespace o2::framework
void output(const std::map< std::string, ChannelStat > &channels)
Definition rawdump.cxx:197
std::unique_ptr< ConfigContext > makeEmptyConfigContext()
GLint GLenum GLint x
Definition glcorearb.h:403
GLuint index
Definition glcorearb.h:781
GLuint const GLchar * name
Definition glcorearb.h:781
GLint y
Definition glcorearb.h:270
bpo::variables_map arguments
Defining PrimaryVertex explicitly as messageable.
Definition TFIDInfo.h:20
void check(const std::vector< std::string > &arguments, const std::vector< ConfigParamSpec > &workflowOptions, const std::vector< DeviceSpec > &deviceSpecs, CheckMatrix &matrix)
TEST_CASE("test_prepareArguments")
bool search(DeviceExecution const &execution, std::string const &option, std::string const &argument)
std::vector< DataProcessorSpec > WorkflowSpec
std::vector< ConfigParamSpec > Options
std::map< std::string, std::vector< std::pair< std::string, std::string > > > CheckMatrix
static std::vector< CallbacksPolicy > createDefaultPolicies()
static std::vector< ChannelConfigurationPolicy > createDefaultPolicies(ConfigContext const &configContext)
Default policies to use, based on the contents of the @configContex content.
static std::vector< CompletionPolicy > createDefaultPolicies()
Helper to create the default configuration.
std::string name
Name of the associated DataProcessorSpec.
std::vector< char * > args
The options passed to a given device.
static void validate(WorkflowSpec const &workflow)
static std::string reworkTimeslicePlaceholder(std::string const &str, DeviceSpec const &spec)
static void prepareArguments(bool defaultQuiet, bool defaultStopped, bool intereactive, unsigned short driverPort, DriverConfig const &driverConfig, std::vector< DataProcessorInfo > const &processorInfos, std::vector< DeviceSpec > const &deviceSpecs, std::vector< DeviceExecution > &deviceExecutions, std::vector< DeviceControl > &deviceControls, std::vector< ConfigParamSpec > const &detectedOptions, std::string const &uniqueWorkflowId)
static void reworkHomogeneousOption(std::vector< DataProcessorInfo > &infos, char const *name, char const *defaultValue)
static void reworkIntegerOption(std::vector< DataProcessorInfo > &infos, char const *name, std::function< long long()> defaultValueCallback, long long startValue, std::function< long long(long long, long long)> bestValue)
static void dataProcessorSpecs2DeviceSpecs(const WorkflowSpec &workflow, std::vector< ChannelConfigurationPolicy > const &channelPolicies, std::vector< CompletionPolicy > const &completionPolicies, std::vector< DispatchPolicy > const &dispatchPolicies, std::vector< ResourcePolicy > const &resourcePolicies, std::vector< CallbacksPolicy > const &callbacksPolicies, std::vector< SendingPolicy > const &sendingPolicy, std::vector< ForwardingPolicy > const &forwardingPolicies, std::vector< DeviceSpec > &devices, ResourceManager &resourceManager, std::string const &uniqueWorkflowId, ConfigContext const &configContext, bool optimizeTopology=false, unsigned short resourcesMonitoringInterval=0, std::string const &channelPrefix="", OverrideServiceSpecs const &overrideServices={})
size_t inputTimesliceId
The time pipelining id of this particular device.
Definition DeviceSpec.h:68