Project
Loading...
Searching...
No Matches
test_GroupSlicer.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 "Framework/ASoA.h"
17#include <arrow/util/config.h>
18#include <iostream>
19
20#include <catch_amalgamated.hpp>
21
22using namespace o2;
23using namespace o2::framework;
24
25namespace o2::aod
26{
27namespace test
28{
31DECLARE_SOA_COLUMN(Bar, bar, float);
32DECLARE_SOA_COLUMN(EventProperty, eventProperty, float);
33} // namespace test
34DECLARE_SOA_TABLE(Events, "AOD", "EVTS",
36 test::ID,
37 test::EventProperty,
38 test::Foo,
39 test::Bar);
40
41using Event = Events::iterator;
42
43namespace test
44{
46DECLARE_SOA_COLUMN(X, x, float);
47DECLARE_SOA_COLUMN(Y, y, float);
48DECLARE_SOA_COLUMN(Z, z, float);
49} // namespace test
50
51namespace unsorted
52{
54}
55
56DECLARE_SOA_TABLE(TrksX, "AOD", "TRKSX",
57 test::EventId,
58 test::X);
59DECLARE_SOA_TABLE(TrksY, "AOD", "TRKSY",
60 test::EventId,
61 test::Y);
62DECLARE_SOA_TABLE(TrksZ, "AOD", "TRKSZ",
63 test::EventId,
64 test::Z);
65DECLARE_SOA_TABLE(TrksU, "AOD", "TRKSU",
66 test::X,
67 test::Y,
68 test::Z);
69
70DECLARE_SOA_TABLE(TrksXU, "AOD", "TRKSXU",
71 unsorted::EventId,
72 test::X);
73DECLARE_SOA_TABLE(TrksYU, "AOD", "TRKSYU",
74 unsorted::EventId,
75 test::Y);
76DECLARE_SOA_TABLE(TrksZU, "AOD", "TRKSZU",
77 unsorted::EventId,
78 test::Z);
79
80namespace test
81{
82DECLARE_SOA_COLUMN(Arr, arr, float[3]);
83DECLARE_SOA_COLUMN(Boo, boo, bool);
84DECLARE_SOA_COLUMN(Lst, lst, std::vector<double>);
85} // namespace test
86
87DECLARE_SOA_TABLE(EventExtra, "AOD", "EVTSXTRA", test::Arr, test::Boo, test::Lst);
88
89} // namespace o2::aod
90TEST_CASE("RelatedByIndex")
91{
93 CHECK(soa::relatedByIndex<aod::Collision, Trks>() == true);
94 CHECK(soa::relatedByIndex<aod::Collision, aod::Tracks>() == true);
95}
96
97TEST_CASE("GroupSlicerOneAssociated")
98{
99 TableBuilder builderE;
100 auto evtsWriter = builderE.cursor<aod::Events>();
101 for (auto i = 0; i < 20; ++i) {
102 evtsWriter(0, i, 0.5f * i, 2.f * i, 3.f * i);
103 }
104 auto evtTable = builderE.finalize();
105
106 TableBuilder builderT;
107 auto trksWriter = builderT.cursor<aod::TrksX>();
108 for (auto i = 0; i < 20; ++i) {
109 for (auto j = 0.f; j < 5; j += 0.5f) {
110 trksWriter(0, i, 0.5f * j);
111 }
112 }
113 auto trkTable = builderT.finalize();
114 aod::Events e{evtTable};
115 aod::TrksX t{trkTable};
116 REQUIRE(e.size() == 20);
117 REQUIRE(t.size() == 10 * 20);
118
119 auto tt = std::make_tuple(t);
120 ArrowTableSlicingCache slices({{soa::getLabelFromType<aod::TrksX>(), "fIndex" + o2::framework::cutString(soa::getLabelFromType<aod::Events>())}});
121 auto s = slices.updateCacheEntry(0, trkTable);
122 o2::framework::GroupSlicer g(e, tt, slices);
123
124 auto count = 0;
125 for (auto& slice : g) {
126 auto as = slice.associatedTables();
127 auto gg = slice.groupingElement();
128 REQUIRE(gg.globalIndex() == count);
129 auto trks = std::get<aod::TrksX>(as);
130 REQUIRE(trks.size() == 10);
131 for (auto& trk : trks) {
132 REQUIRE(trk.eventId() == count);
133 }
134 ++count;
135 }
136}
137
138TEST_CASE("GroupSlicerSeveralAssociated")
139{
140 TableBuilder builderE;
141 auto evtsWriter = builderE.cursor<aod::Events>();
142 for (auto i = 0; i < 20; ++i) {
143 evtsWriter(0, i, 0.5f * i, 2.f * i, 3.f * i);
144 }
145 auto evtTable = builderE.finalize();
146
147 TableBuilder builderTX;
148 auto trksWriterX = builderTX.cursor<aod::TrksX>();
149 TableBuilder builderTY;
150 auto trksWriterY = builderTY.cursor<aod::TrksY>();
151 TableBuilder builderTZ;
152 auto trksWriterZ = builderTZ.cursor<aod::TrksZ>();
153
154 TableBuilder builderTXYZ;
155 auto trksWriterXYZ = builderTXYZ.cursor<aod::TrksU>();
156
157 for (auto i = 0; i < 20; ++i) {
158 for (auto j = 0.f; j < 5; j += 0.5f) {
159 trksWriterX(0, i, 0.5f * j);
160 }
161 for (auto j = 0.f; j < 10; j += 0.5f) {
162 trksWriterY(0, i, 0.5f * j);
163 }
164 for (auto j = 0.f; j < 15; j += 0.5f) {
165 trksWriterZ(0, i, 0.5f * j);
166 }
167
168 for (auto j = 0.f; j < 5; j += 0.5f) {
169 trksWriterXYZ(0, 0.5f * j, 2.f * j, 2.5f * j);
170 }
171 }
172 auto trkTableX = builderTX.finalize();
173 auto trkTableY = builderTY.finalize();
174 auto trkTableZ = builderTZ.finalize();
175
176 auto trkTableXYZ = builderTXYZ.finalize();
177
178 aod::Events e{evtTable};
179 aod::TrksX tx{trkTableX};
180 aod::TrksY ty{trkTableY};
181 aod::TrksZ tz{trkTableZ};
182
183 aod::TrksU tu{trkTableXYZ};
184
185 REQUIRE(e.size() == 20);
186 REQUIRE(tx.size() == 10 * 20);
187 REQUIRE(ty.size() == 20 * 20);
188 REQUIRE(tz.size() == 30 * 20);
189
190 REQUIRE(tu.size() == 10 * 20);
191
192 auto tt = std::make_tuple(tx, ty, tz, tu);
193 auto key = "fIndex" + o2::framework::cutString(soa::getLabelFromType<aod::Events>());
194 ArrowTableSlicingCache slices({{soa::getLabelFromType<aod::TrksX>(), key},
195 {soa::getLabelFromType<aod::TrksY>(), key},
196 {soa::getLabelFromType<aod::TrksZ>(), key}});
197 auto s = slices.updateCacheEntry(0, {trkTableX});
198 s = slices.updateCacheEntry(1, {trkTableY});
199 s = slices.updateCacheEntry(2, {trkTableZ});
200 o2::framework::GroupSlicer g(e, tt, slices);
201
202 auto count = 0;
203 for (auto& slice : g) {
204 auto as = slice.associatedTables();
205 auto gg = slice.groupingElement();
206 REQUIRE(gg.globalIndex() == count);
207 auto trksx = std::get<aod::TrksX>(as);
208 auto trksy = std::get<aod::TrksY>(as);
209 auto trksz = std::get<aod::TrksZ>(as);
210
211 auto trksu = std::get<aod::TrksU>(as);
212
213 REQUIRE(trksx.size() == 10);
214 REQUIRE(trksy.size() == 20);
215 REQUIRE(trksz.size() == 30);
216
217 REQUIRE(trksu.size() == 10 * 20);
218
219 for (auto& trk : trksx) {
220 REQUIRE(trk.eventId() == count);
221 }
222 for (auto& trk : trksy) {
223 REQUIRE(trk.eventId() == count);
224 }
225 for (auto& trk : trksz) {
226 REQUIRE(trk.eventId() == count);
227 }
228
229 ++count;
230 }
231}
232
233TEST_CASE("GroupSlicerMismatchedGroups")
234{
235 TableBuilder builderE;
236 auto evtsWriter = builderE.cursor<aod::Events>();
237 for (auto i = 0; i < 20; ++i) {
238 evtsWriter(0, i, 0.5f * i, 2.f * i, 3.f * i);
239 }
240 auto evtTable = builderE.finalize();
241
242 TableBuilder builderT;
243 auto trksWriter = builderT.cursor<aod::TrksX>();
244 for (auto i = 0; i < 20; ++i) {
245 if (i == 3 || i == 10 || i == 12 || i == 16 || i == 19) {
246 continue;
247 }
248 for (auto j = 0.f; j < 5; j += 0.5f) {
249 trksWriter(0, i, 0.5f * j);
250 }
251 }
252 auto trkTable = builderT.finalize();
253 aod::Events e{evtTable};
254 aod::TrksX t{trkTable};
255 REQUIRE(e.size() == 20);
256 REQUIRE(t.size() == 10 * (20 - 5));
257
258 auto tt = std::make_tuple(t);
259 ArrowTableSlicingCache slices({{soa::getLabelFromType<aod::TrksX>(), "fIndex" + o2::framework::cutString(soa::getLabelFromType<aod::Events>())}});
260 auto s = slices.updateCacheEntry(0, trkTable);
261 o2::framework::GroupSlicer g(e, tt, slices);
262
263 auto count = 0;
264 for (auto& slice : g) {
265 auto as = slice.associatedTables();
266 auto gg = slice.groupingElement();
267 REQUIRE(gg.globalIndex() == count);
268 auto trks = std::get<aod::TrksX>(as);
269 if (count == 3 || count == 10 || count == 12 || count == 16 || count == 19) {
270 REQUIRE(trks.size() == 0);
271 } else {
272 REQUIRE(trks.size() == 10);
273 }
274 for (auto& trk : trks) {
275 REQUIRE(trk.eventId() == count);
276 }
277 ++count;
278 }
279}
280
281TEST_CASE("GroupSlicerMismatchedUnassignedGroups")
282{
283 TableBuilder builderE;
284 auto evtsWriter = builderE.cursor<aod::Events>();
285 for (auto i = 0; i < 20; ++i) {
286 evtsWriter(0, i, 0.5f * i, 2.f * i, 3.f * i);
287 }
288 auto evtTable = builderE.finalize();
289
290 int skip = 0;
291
292 TableBuilder builderT;
293 auto trksWriter = builderT.cursor<aod::TrksX>();
294 for (auto i = 0; i < 20; ++i) {
295 if (i == 3 || i == 10 || i == 12 || i == 16 || i == 19) {
296 for (auto iz = 0; iz < 5; ++iz) {
297 trksWriter(0, -1 - skip, 0.123f * iz + 1.654f);
298 }
299 ++skip;
300 continue;
301 }
302 for (auto j = 0.f; j < 5; j += 0.5f) {
303 trksWriter(0, i, 0.5f * j);
304 }
305 }
306 for (auto i = 0; i < 5; ++i) {
307 trksWriter(0, -6, 0.123f * i + 1.654f);
308 }
309 auto trkTable = builderT.finalize();
310
311 aod::Events e{evtTable};
312 aod::TrksX t{trkTable};
313 REQUIRE(e.size() == 20);
314 REQUIRE(t.size() == (30 + 10 * (20 - 5)));
315
316 auto tt = std::make_tuple(t);
317 ArrowTableSlicingCache slices({{soa::getLabelFromType<aod::TrksX>(), "fIndex" + o2::framework::cutString(soa::getLabelFromType<aod::Events>())}});
318 auto s = slices.updateCacheEntry(0, trkTable);
319 o2::framework::GroupSlicer g(e, tt, slices);
320
321 auto count = 0;
322 for (auto& slice : g) {
323 auto as = slice.associatedTables();
324 auto gg = slice.groupingElement();
325 REQUIRE(gg.globalIndex() == count);
326 auto trks = std::get<aod::TrksX>(as);
327 if (count == 3 || count == 10 || count == 12 || count == 16 || count == 19) {
328 REQUIRE(trks.size() == 0);
329 } else {
330 REQUIRE(trks.size() == 10);
331 }
332 for (auto& trk : trks) {
333 REQUIRE(trk.eventId() == count);
334 }
335 ++count;
336 }
337}
338
339TEST_CASE("GroupSlicerMismatchedFilteredGroups")
340{
341 TableBuilder builderE;
342 auto evtsWriter = builderE.cursor<aod::Events>();
343 for (auto i = 0; i < 20; ++i) {
344 evtsWriter(0, i, 0.5f * i, 2.f * i, 3.f * i);
345 }
346 auto evtTable = builderE.finalize();
347
348 TableBuilder builderT;
349 auto trksWriter = builderT.cursor<aod::TrksX>();
350 for (auto i = 0; i < 20; ++i) {
351 if (i == 3 || i == 10 || i == 12 || i == 16) {
352 continue;
353 }
354 for (auto j = 0.f; j < 5; j += 0.5f) {
355 trksWriter(0, i, 0.5f * j);
356 }
357 }
358 auto trkTable = builderT.finalize();
359 using FilteredEvents = soa::Filtered<aod::Events>;
360 soa::SelectionVector rows{2, 4, 10, 9, 15};
361 FilteredEvents e{{evtTable}, {2, 4, 10, 9, 15}};
362 aod::TrksX t{trkTable};
363 REQUIRE(e.size() == 5);
364 REQUIRE(t.size() == 10 * (20 - 4));
365
366 auto tt = std::make_tuple(t);
367 ArrowTableSlicingCache slices({{soa::getLabelFromType<aod::TrksX>(), "fIndex" + o2::framework::cutString(soa::getLabelFromType<aod::Events>())}});
368 auto s = slices.updateCacheEntry(0, trkTable);
369 o2::framework::GroupSlicer g(e, tt, slices);
370
371 auto count = 0;
372
373 for (auto& slice : g) {
374 auto as = slice.associatedTables();
375 auto gg = slice.groupingElement();
376 REQUIRE(gg.globalIndex() == rows[count]);
377 auto trks = std::get<aod::TrksX>(as);
378 if (rows[count] == 3 || rows[count] == 10 || rows[count] == 12 || rows[count] == 16) {
379 REQUIRE(trks.size() == 0);
380 } else {
381 REQUIRE(trks.size() == 10);
382 }
383 for (auto& trk : trks) {
384 REQUIRE(trk.eventId() == rows[count]);
385 }
386 ++count;
387 }
388}
389
390TEST_CASE("GroupSlicerMismatchedUnsortedFilteredGroups")
391{
392 TableBuilder builderE;
393 auto evtsWriter = builderE.cursor<aod::Events>();
394 for (auto i = 0; i < 20; ++i) {
395 evtsWriter(0, i, 0.5f * i, 2.f * i, 3.f * i);
396 }
397 auto evtTable = builderE.finalize();
398
399 TableBuilder builderT;
400 auto trksWriter = builderT.cursor<aod::TrksXU>();
401 std::vector<int> randomized{10, 2, 1, 0, 15, 3, 6, 4, 14, 5, 7, 9, 8, 19, 11, 13, 17, 12, 18, 19};
402 std::vector<int64_t> sel;
403 sel.resize(10 * (20 - 4));
404 std::iota(sel.begin(), sel.end(), 0);
405 for (auto i : randomized) {
406 if (i == 3 || i == 10 || i == 12 || i == 16) {
407 continue;
408 }
409 for (auto j = 0.f; j < 5; j += 0.5f) {
410 trksWriter(0, i, 0.5f * j);
411 }
412 }
413 auto trkTable = builderT.finalize();
414
415 TableBuilder builderTE;
416 auto trksWriterE = builderTE.cursor<aod::TrksXU>();
417 auto trkTableE = builderTE.finalize();
418
419 using FilteredEvents = soa::Filtered<aod::Events>;
420 soa::SelectionVector rows{2, 4, 10, 9, 15};
421 FilteredEvents e{{evtTable}, {2, 4, 10, 9, 15}};
422 soa::SmallGroups<aod::TrksXU> t{{trkTable}, std::move(sel)};
423
424 REQUIRE(e.size() == 5);
425 REQUIRE(t.size() == 10 * (20 - 4));
426
427 auto tt = std::make_tuple(t);
428 ArrowTableSlicingCache slices({}, {{soa::getLabelFromType<aod::TrksXU>(), "fIndex" + o2::framework::cutString(soa::getLabelFromType<aod::Events>())}});
429 auto s = slices.updateCacheEntryUnsorted(0, trkTable);
430 o2::framework::GroupSlicer g(e, tt, slices);
431
432 unsigned int count = 0;
433
434 for (auto& slice : g) {
435 auto as = slice.associatedTables();
436 auto gg = slice.groupingElement();
437 REQUIRE(gg.globalIndex() == rows[count]);
438 auto trks = std::get<soa::SmallGroups<aod::TrksXU>>(as);
439 if (rows[count] == 3 || rows[count] == 10 || rows[count] == 12 || rows[count] == 16) {
440 REQUIRE(trks.size() == 0);
441 } else {
442 REQUIRE(trks.size() == 10);
443 }
444 for (auto& trk : trks) {
445 REQUIRE(trk.eventId() == rows[count]);
446 }
447 ++count;
448 }
449
450 std::vector<int64_t> sele;
451 soa::SmallGroups<aod::TrksXU> te{{trkTableE}, std::move(sele)};
452 auto tte = std::make_tuple(te);
453 o2::framework::GroupSlicer ge(e, tte, slices);
454
455 count = 0;
456 for (auto& slice : ge) {
457 auto as = slice.associatedTables();
458 auto gg = slice.groupingElement();
459 REQUIRE(gg.globalIndex() == rows[count]);
460 auto trks = std::get<soa::SmallGroups<aod::TrksXU>>(as);
461 REQUIRE(trks.size() == 0);
462 ++count;
463 }
464
465 soa::SmallGroupsUnfiltered<aod::TrksXU> tu{{trkTable}, std::vector<int64_t>{}};
466 auto ttu = std::make_tuple(tu);
467 o2::framework::GroupSlicer gu(e, ttu, slices);
468
469 count = 0;
470 for (auto& slice : gu) {
471 auto as = slice.associatedTables();
472 auto gg = slice.groupingElement();
473 REQUIRE(gg.globalIndex() == rows[count]);
474 auto trks = std::get<soa::SmallGroupsUnfiltered<aod::TrksXU>>(as);
475 if (rows[count] == 3 || rows[count] == 10 || rows[count] == 12 || rows[count] == 16) {
476 REQUIRE(trks.size() == 0);
477 } else {
478 REQUIRE(trks.size() == 10);
479 }
480 ++count;
481 }
482}
483
484namespace o2::aod
485{
486namespace parts
487{
489DECLARE_SOA_COLUMN(Property, property, int);
491} // namespace parts
492DECLARE_SOA_TABLE(Parts, "AOD", "PRTS", soa::Index<>, parts::EventId, parts::Property, parts::RelativesIdSlice);
493
494namespace things
495{
498} // namespace things
499DECLARE_SOA_TABLE(Things, "AOD", "THNGS", soa::Index<>, things::EventId, things::PartId);
500
501} // namespace o2::aod
502
503template <typename... As>
504static void overwriteInternalIndices(std::tuple<As...>& dest, std::tuple<As...> const& src)
505{
506 (std::get<As>(dest).bindInternalIndicesTo(&std::get<As>(src)), ...);
507}
508
509TEST_CASE("GroupSlicerMismatchedUnsortedFilteredGroupsWithSelfIndex")
510{
511 TableBuilder builderE;
512 auto evtsWriter = builderE.cursor<aod::Events>();
513 for (auto i = 0; i < 20; ++i) {
514 evtsWriter(0, i, 0.5f * i, 2.f * i, 3.f * i);
515 }
516 auto evtTable = builderE.finalize();
517
518 TableBuilder builderP;
519 auto partsWriter = builderP.cursor<aod::Parts>();
520 int filler[2];
521 std::random_device rd; // a seed source for the random number engine
522 std::mt19937 gen(rd()); // mersenne_twister_engine seeded with rd()
523 std::uniform_int_distribution<> distrib(0, 99);
524
525 for (auto i = 0; i < 100; ++i) {
526
527 filler[0] = distrib(gen);
528 filler[1] = distrib(gen);
529 if (filler[0] > filler[1]) {
530 std::swap(filler[0], filler[1]);
531 }
532 partsWriter(0, std::floor(i / 10.), i, filler);
533 }
534 auto partsTable = builderP.finalize();
535
536 TableBuilder builderT;
537 auto thingsWriter = builderT.cursor<aod::Things>();
538 for (auto i = 0; i < 10; ++i) {
539 thingsWriter(0, i, distrib(gen));
540 }
541 auto thingsTable = builderT.finalize();
542
543 aod::Events e{evtTable};
544 // aod::Parts p{partsTable};
545 aod::Things t{thingsTable};
546 using FilteredParts = soa::Filtered<aod::Parts>;
547 auto size = distrib(gen);
549 for (auto i = 0; i < size; ++i) {
550 rows.push_back(distrib(gen));
551 }
552 FilteredParts fp{{partsTable}, rows};
553 auto associatedTuple = std::make_tuple(fp, t);
554 ArrowTableSlicingCache slices({{soa::getLabelFromType<aod::Parts>(), "fIndex" + o2::framework::cutString(soa::getLabelFromType<aod::Events>())},
555 {soa::getLabelFromType<aod::Things>(), "fIndex" + o2::framework::cutString(soa::getLabelFromType<aod::Events>())}});
556 auto s0 = slices.updateCacheEntry(0, partsTable);
557 auto s1 = slices.updateCacheEntry(1, thingsTable);
558 o2::framework::GroupSlicer g(e, associatedTuple, slices);
559
560 overwriteInternalIndices(associatedTuple, associatedTuple);
561
562 // For a grouped case, the recursive access of a slice-self index of a filtered table should have consistent types
563 for (auto& slice : g) {
564 auto as = slice.associatedTables();
565 auto gg = slice.groupingElement();
566 overwriteInternalIndices(as, associatedTuple);
567 auto& ts = std::get<1>(as);
568 ts.bindExternalIndices(&e, &std::get<0>(associatedTuple));
569 for (auto& thing : ts) {
570 if (thing.has_part()) {
571 auto part = thing.part_as<FilteredParts>();
572 REQUIRE(std::same_as<std::decay_t<decltype(part)>::parent_t, FilteredParts>);
573 auto rs = part.relatives_as<std::decay_t<decltype(part)::parent_t>>();
574 REQUIRE(std::same_as<std::decay_t<decltype(rs)>, FilteredParts>);
575 for (auto& r : rs) {
576 REQUIRE(std::same_as<std::decay_t<decltype(r)>::parent_t, FilteredParts>);
577 auto rss = r.relatives_as<std::decay_t<decltype(r)>::parent_t>();
578 REQUIRE(std::same_as<std::decay_t<decltype(rss)>, FilteredParts>);
579 for (auto& rr : rss) {
580 REQUIRE(std::same_as<std::decay_t<decltype(rr)>::parent_t, FilteredParts>);
581 auto rsss = rr.relatives_as<std::decay_t<decltype(rr)>::parent_t>();
582 REQUIRE(std::same_as<std::decay_t<decltype(rsss)>, FilteredParts>);
583 for (auto& rrr : rsss) {
584 REQUIRE(std::same_as<std::decay_t<decltype(rrr)>::parent_t, FilteredParts>);
585 auto rssss = rrr.relatives_as<std::decay_t<decltype(rrr)>::parent_t>();
586 REQUIRE(std::same_as<std::decay_t<decltype(rssss)>, FilteredParts>);
587 }
588 }
589 }
590 }
591 }
592 }
593}
594
595TEST_CASE("EmptySliceables")
596{
597 TableBuilder builderE;
598 auto evtsWriter = builderE.cursor<aod::Events>();
599 for (auto i = 0; i < 20; ++i) {
600 evtsWriter(0, i, 0.5f * i, 2.f * i, 3.f * i);
601 }
602 auto evtTable = builderE.finalize();
603
604 TableBuilder builderT;
605 auto trksWriter = builderT.cursor<aod::TrksX>();
606 auto trkTable = builderT.finalize();
607
608 aod::Events e{evtTable};
609 aod::TrksX t{trkTable};
610 REQUIRE(e.size() == 20);
611 REQUIRE(t.size() == 0);
612
613 auto tt = std::make_tuple(t);
614 ArrowTableSlicingCache slices({{soa::getLabelFromType<aod::TrksX>(), "fIndex" + o2::framework::cutString(soa::getLabelFromType<aod::Events>())}});
615 auto s = slices.updateCacheEntry(0, trkTable);
616 o2::framework::GroupSlicer g(e, tt, slices);
617
618 unsigned int count = 0;
619 for (auto& slice : g) {
620 auto as = slice.associatedTables();
621 auto gg = slice.groupingElement();
622 auto trks = std::get<aod::TrksX>(as);
623 REQUIRE(gg.globalIndex() == count);
624 REQUIRE(trks.size() == 0);
625 ++count;
626 }
627}
628
629TEST_CASE("ArrowDirectSlicing")
630{
631 int counts[] = {5, 5, 5, 4, 1};
632 int offsets[] = {0, 5, 10, 15, 19, 20};
633 int ids[] = {0, 1, 2, 3, 4};
634 int sizes[] = {4, 1, 12, 5, 2};
635
637
638 TableBuilder builderE;
639 auto evtsWriter = builderE.cursor<aod::Events>();
640 auto step = 0;
641 for (auto i = 0; i < 20; ++i) {
642 if (i >= offsets[step + 1]) {
643 ++step;
644 }
645 evtsWriter(0, ids[step], 0.5f * i, 2.f * i, 3.f * i);
646 }
647 auto evtTable = builderE.finalize();
648
649 TableBuilder builderEE;
650 auto evtsEWriter = builderEE.cursor<aod::EventExtra>();
651 step = 0;
652
653 for (auto i = 0; i < 20; ++i) {
654 if (i >= offsets[step + 1]) {
655 ++step;
656 }
657 float arr[3] = {0.1f * i, 0.2f * i, 0.3f * i};
658 std::vector<double> d;
659 for (auto z = 0; z < sizes[step]; ++z) {
660 d.push_back((double)z * 0.5);
661 }
662 evtsEWriter(0, arr, i % 2 == 0, d);
663 }
664 auto evtETable = builderEE.finalize();
665
666 aod::Events e{evtTable};
667 aod::EventExtra ee{evtETable};
668 BigE b_e{{evtTable, evtETable}};
669
670 std::vector<std::shared_ptr<arrow::ChunkedArray>> slices_array;
671 std::vector<std::shared_ptr<arrow::ChunkedArray>> slices_bool;
672 std::vector<std::shared_ptr<arrow::ChunkedArray>> slices_vec;
673 auto offset = 0;
674 for (auto i = 0u; i < 5; ++i) {
675 slices_array.emplace_back(evtETable->column(0)->Slice(offset, counts[i]));
676 slices_bool.emplace_back(evtETable->column(1)->Slice(offset, counts[i]));
677 slices_vec.emplace_back(evtETable->column(2)->Slice(offset, counts[i]));
678 offset += counts[i];
679 REQUIRE(slices_array[i]->length() == counts[i]);
680 REQUIRE(slices_bool[i]->length() == counts[i]);
681 REQUIRE(slices_vec[i]->length() == counts[i]);
682 }
683
684 std::vector<arrow::Datum> slices;
685 std::vector<uint64_t> offsts;
686 auto bk = std::make_pair(soa::getLabelFromType<aod::Events>(), "fID");
687 ArrowTableSlicingCache cache({bk});
688 auto s = cache.updateCacheEntry(0, {evtTable});
689 auto lcache = cache.getCacheFor(bk);
690 for (auto i = 0u; i < 5; ++i) {
691 auto [offset, count] = lcache.getSliceFor(i);
692 auto tbl = b_e.asArrowTable()->Slice(offset, count);
693 auto ca = tbl->GetColumnByName("fArr");
694 auto cb = tbl->GetColumnByName("fBoo");
695 auto cv = tbl->GetColumnByName("fLst");
696 REQUIRE(ca->length() == counts[i]);
697 REQUIRE(cb->length() == counts[i]);
698 REQUIRE(cv->length() == counts[i]);
699 REQUIRE(ca->Equals(slices_array[i]));
700 REQUIRE(cb->Equals(slices_bool[i]));
701 REQUIRE(cv->Equals(slices_vec[i]));
702 }
703
704 int j = 0u;
705 for (auto i = 0u; i < 5; ++i) {
706 auto [offset, count] = lcache.getSliceFor(i);
707 auto tbl = BigE{{b_e.asArrowTable()->Slice(offset, count)}, static_cast<uint64_t>(offset)};
708 REQUIRE(tbl.size() == counts[i]);
709 for (auto& row : tbl) {
710 REQUIRE(row.id() == ids[i]);
711 REQUIRE(row.boo() == (j % 2 == 0));
712 auto rid = row.globalIndex();
713 auto arr = row.arr();
714 REQUIRE(arr[0] == 0.1f * (float)rid);
715 REQUIRE(arr[1] == 0.2f * (float)rid);
716 REQUIRE(arr[2] == 0.3f * (float)rid);
717
718 auto d = row.lst();
719 REQUIRE(d.size() == (size_t)sizes[i]);
720 for (auto z = 0u; z < d.size(); ++z) {
721 REQUIRE(d[z] == 0.5 * (double)z);
722 }
723 ++j;
724 }
725 }
726}
727
728TEST_CASE("TestSlicingException")
729{
730 int offsets[] = {0, 5, 10, 15, 19, 20};
731 int ids[] = {0, 1, 2, 4, 3};
732
733 TableBuilder builderE;
734 auto evtsWriter = builderE.cursor<aod::Events>();
735 auto step = 0;
736 for (auto i = 0; i < 20; ++i) {
737 if (i >= offsets[step + 1]) {
738 ++step;
739 }
740 evtsWriter(0, ids[step], 0.5f * i, 2.f * i, 3.f * i);
741 }
742 auto evtTable = builderE.finalize();
743
744 auto bk = std::make_pair(soa::getLabelFromType<aod::Events>(), "fID");
745 ArrowTableSlicingCache cache({bk});
746
747 try {
748 auto s = cache.updateCacheEntry(0, {evtTable});
749 } catch (RuntimeErrorRef re) {
750 REQUIRE(std::string{error_from_ref(re).what} == "Table Events index fID is not sorted: next value 3 < previous value 4!");
751 return;
752 } catch (...) {
753 FAIL("Slicing should have failed due to unsorted index");
754 }
755}
#define DECLARE_SOA_TABLE(_Name_, _Origin_, _Desc_,...)
Definition ASoA.h:3057
#define DECLARE_SOA_COLUMN(_Name_, _Getter_, _Type_)
Definition ASoA.h:2319
#define DECLARE_SOA_SELF_SLICE_INDEX_COLUMN(_Name_, _Getter_)
Definition ASoA.h:2866
#define DECLARE_SOA_INDEX_COLUMN(_Name_, _Getter_)
Definition ASoA.h:2742
default_random_engine gen(dev())
int32_t i
uint32_t j
Definition RawData.h:0
StringRef key
std::shared_ptr< arrow::Table > finalize()
#define CHECK
struct _cl_event * event
Definition glcorearb.h:2982
GLint GLenum GLint x
Definition glcorearb.h:403
GLenum src
Definition glcorearb.h:1767
GLint GLsizei count
Definition glcorearb.h:399
GLsizeiptr size
Definition glcorearb.h:659
GLuint GLsizei const GLuint const GLintptr * offsets
Definition glcorearb.h:2595
GLuint * ids
Definition glcorearb.h:647
GLuint GLsizei const GLuint const GLintptr const GLsizeiptr * sizes
Definition glcorearb.h:2595
GLuint GLfloat GLfloat GLfloat GLfloat GLfloat GLfloat GLfloat GLfloat s1
Definition glcorearb.h:5034
GLintptr offset
Definition glcorearb.h:660
GLuint GLsizei GLsizei * length
Definition glcorearb.h:790
GLboolean GLboolean g
Definition glcorearb.h:1233
GLboolean r
Definition glcorearb.h:1233
GLdouble GLdouble GLdouble z
Definition glcorearb.h:843
GLuint GLfloat GLfloat GLfloat GLfloat GLfloat GLfloat s0
Definition glcorearb.h:5034
Events::iterator Event
Defining PrimaryVertex explicitly as messageable.
Definition TFIDInfo.h:20
TEST_CASE("test_prepareArguments")
RuntimeError & error_from_ref(RuntimeErrorRef)
std::string cutString(std::string &&str)
Definition ASoA.cxx:180
std::vector< int64_t > SelectionVector
Definition ASoA.h:411
a couple of static helper functions to create timestamp values for CCDB queries or override obsolete ...
FIXME: do not use data model tables.
arrow::Status updateCacheEntry(int pos, std::shared_ptr< arrow::Table > const &table)
char what[MAX_RUNTIME_ERROR_SIZE]
HistogramRegistry foo()
std::random_device rd
std::vector< int > row
std::vector< ReadoutWindowData > rows