yoda is hosted by Hepforge, IPPP Durham
YODA - Yet more Objects for Data Analysis 2.1.0
ReaderUtils.h
Go to the documentation of this file.
1// -*- C++ -*-
2//
3// This file is part of YODA -- Yet more Objects for Data Analysis
4// Copyright (C) 2008-2025 The YODA collaboration (see AUTHORS for details)
5//
6#ifndef YODA_READERUTILS_H
7#define YODA_READERUTILS_H
8
10#include "YODA/Counter.h"
11#include "YODA/Estimate0D.h"
12#include "YODA/Scatter.h"
13#include "YODA/Histo.h"
14#include "YODA/Profile.h"
15#include "YODA/BinnedEstimate.h"
16
17#ifdef HAVE_HDF5
19#include "YODA/Utils/H5Utils.h"
20#ifdef WITH_HIGHFIVE
21#include <YODA/highfive/H5File.hpp>
22#else
23#include <highfive/H5File.hpp>
24#define YODA_H5 HighFive
25#endif
26#endif
27
28#include <iostream>
29#include <locale>
30#include <cstring>
31#include <regex>
32#include <cmath>
33
34#ifdef WITH_OSX
35#include <xlocale.h>
36#endif
37
38using std::map;
39using std::string;
40using std::vector;
41
42namespace YODA {
43
45 namespace {
46
47 static const std::regex regex_string_pat("([\"\'])(?:(?=(\\\\?))\\2.)*?\\1");
48
49 }
50
51
53
55 class aistringstream {
56 public:
57 // Constructor from char*
58 aistringstream(const char* line=0) {
59 reset(line);
60 _set_locale();
61 }
62 // Constructor from std::string
63 aistringstream(const string& line) {
64 reset(line);
65 _set_locale();
66 }
67 ~aistringstream() {
68 _reset_locale();
69 }
70
71 bool peek(const std::string& s) const {
72 return s == std::string(_next, s.size());
73 }
74
75 // Re-init to new line as char*
76 void reset(const char* line=0) {
77 _next = const_cast<char*>(line);
78 _new_next = _next;
79 _error = false;
80 }
81 // Re-init to new line as std::string
82 void reset(const string& line) { reset(line.c_str()); }
83
84 // Tokenizing stream operator (forwards to specialisations)
85 template<class T>
86 aistringstream& operator >> (T& value) {
87 _get(value);
88 if (_new_next == _next) _error = true; // handy error condition behaviour!
89 _next = _new_next;
90 return *this;
91 }
92
93 // Allow use of operator>> in a while loop
94 operator bool() const { return !_error; }
95
96
97 private:
98
99 // Changes the thread-local locale to interpret numbers in the "C" locale
100 void _set_locale() {
101 _locale_set = newlocale(LC_NUMERIC_MASK, "C", NULL);
102 _locale_prev = uselocale(_locale_set);
103 if (!_locale_prev) {
104 throw ReadError(std::string("Error setting locale: ") + strerror(errno));
105 }
106 }
107 void _reset_locale() {
108 if (!uselocale(_locale_prev)) {
109 throw ReadError(std::string("Error setting locale: ") + strerror(errno));
110 }
111 freelocale(_locale_set);
112 }
113
114 void _get(double& x) { x = std::strtod(_next, &_new_next); }
115 void _get(float& x) { x = std::strtof(_next, &_new_next); }
116 void _get(int& i) { i = std::strtol(_next, &_new_next, 10); } // force base 10!
117 void _get(long& i) { i = std::strtol(_next, &_new_next, 10); } // force base 10!
118 void _get(unsigned int& i) { i = std::strtoul(_next, &_new_next, 10); } // force base 10!
119 void _get(long unsigned int& i) { i = std::strtoul(_next, &_new_next, 10); } // force base 10!
120 void _get(string& x) {
121 while (std::isspace(*_next) && _next[0] != '\0') _next += 1;
122 _new_next = _next;
123 while (!std::isspace(*_new_next) && _new_next[0] != '\0') _new_next += 1;
124 x = string(_next, _new_next-_next);
125 }
126
127 locale_t _locale_set, _locale_prev;
128 char *_next, *_new_next;
129 bool _error;
130 };
131
132
133 public:
134
137
139 virtual ~AOReaderBase() { }
140
141 virtual void parse(const string& line) = 0;
142
143 virtual AnalysisObject* assemble(const string& path = "") = 0;
144
145 #ifdef HAVE_HDF5
146
148
149 virtual void skip(H5FileManager& h5file) {
150 h5file.skipCommon();
151 }
152
153 #endif
154
155 template<typename T>
156 void extractVector(const std::string& line, std::vector<T>& vec) {
157 if constexpr (std::is_same<T, std::string>::value) {
158 string::const_iterator initpos( line.cbegin() );
159 const string::const_iterator finpos( line.cend() );
160 std::smatch m;
161 while ( std::regex_search(initpos, finpos, m, regex_string_pat) ) {
162 string label;
163 std::stringstream ss(m[0].str());
164 ss >> std::quoted(label); // removes outer quotes and de-escapes inner quotes
165 vec.push_back(label);
166 initpos = m.suffix().first;
167 }
168 }
169 else {
170 std::string content = line.substr(line.find(": [")+3);
171 content.pop_back(); // remove the "]" at the end
172 for (const std::string& item : Utils::split(content, ",")) {
173 aiss.reset(item);
174 T tmp;
175 aiss >> tmp;
176 vec.push_back(std::move(tmp));
177 }
178 }
179 }
180
181 protected:
182
183 aistringstream aiss;
184
185 };
186
187
188
189
190 template<class T>
191 class AOReader;
192
193 template<>
194 class AOReader<Counter> : public AOReaderBase {
195
196 Dbn0D dbn;
197
198 public:
199
200 void parse(const string& line) {
201 aiss.reset(line);
202 double sumw(0), sumw2(0), n(0);
203 aiss >> sumw >> sumw2 >> n;
204 dbn = Dbn0D(n, sumw, sumw2);
205 }
206
207 AnalysisObject* assemble(const string& path = "") {
208 auto* ao = new Counter(path);
209 ao->setDbn(dbn);
210 dbn = Dbn0D();
211 return ao;
212 }
213
214 #ifdef HAVE_HDF5
215
217 vector<string> annos = h5file.loadAnnotations();
218 auto* ao = new Counter(h5file.path(), annos.back());
219 annos.pop_back();
220 ao->deserializeMeta(annos);
221 ao->deserializeContent(h5file.loadContent());
222 return ao;
223 };
224
225 #endif
226 };
227
228
229 template<>
231
232 Estimate0D est;
233 vector<string> sources;
234
235 void readErrors(std::map<string,std::pair<double,double>>& errors) {
236 string eDn, eUp;
237 for (size_t i = 0; i < sources.size(); ++i) {
238 aiss >> eDn >> eUp;
239 if (eDn != "---" && eUp != "---") {
240 errors[sources[i]] = { Utils::toDbl(eDn), Utils::toDbl(eUp) };
241 }
242 }
243 }
244
245 public:
246
247 void parse(const string& line) {
248 if (!line.rfind("ErrorLabels: ", 0)) { // parse error labels
249 extractVector<std::string>(line, sources);
250 return;
251 }
252 // parse content
253 aiss.reset(line);
254 double val(0);
255 aiss >> val;
256 std::map<string,std::pair<double,double>> errors;
257 readErrors(errors);
258 est = Estimate0D(val, errors);
259 }
260
261 AnalysisObject* assemble(const string& path = "") {
262
263 auto* ao = new Estimate0D(est, path);
264 est = Estimate0D();
265 sources.clear();
266 return ao;
267 }
268
269 #ifdef HAVE_HDF5
271 vector<string> annos = h5file.loadAnnotations();
272 auto* ao = new Estimate0D(h5file.path(), annos.back());
273 annos.pop_back();
274 ao->deserializeMeta(annos);
275 ao->deserializeContent(h5file.loadContent());
276 ao->deserializeSources(h5file.loadSources());
277 return ao;
278 }
279
280 void skip(H5FileManager& h5file) {
281 h5file.skipCommon();
282 h5file.skipSources();
283 }
284
285 #endif
286 };
287
288
289 template <size_t N>
290 class AOReader<ScatterND<N>> : public AOReaderBase {
291
292 vector<PointND<N>> points;
293
294 template<size_t I>
295 void readCoords(vector<double>& vals, vector<double>& errm, vector<double>& errp) {
296 if constexpr(I < N) {
297 double v(0), em(0), ep(0);
298 aiss >> v >> em >> ep;
299 vals[I] = v;
300 errm[I] = fabs(em);
301 errp[I] = fabs(ep);
302 readCoords<I+1>(vals, errm, errp);
303 }
304 }
305
306 public:
307
308 void parse(const string& line) {
309 aiss.reset(line);
310 vector<double> vals(N), errm(N), errp(N);
311 readCoords<0>(vals, errm, errp);
312 points.push_back(PointND<N>(vals, errm, errp));
313 }
314
315 AnalysisObject* assemble(const string& path = "") {
316 auto* ao = new ScatterND<N>();
317 ao->setPath(path);
318 ao->addPoints(points);
319 points.clear();
320 return ao;
321 }
322
323 #ifdef HAVE_HDF5
324
326 vector<string> annos = h5file.loadAnnotations();
327 auto* ao = new ScatterND<N>(h5file.path(), annos.back());
328 annos.pop_back();
329 ao->deserializeMeta(annos);
330 ao->deserializeContent(h5file.loadContent());
331 return ao;
332 }
333
334 #endif
335 };
336
337
338 template <size_t DbnN, typename... AxisT>
339 class AOReader<BinnedDbn<DbnN, AxisT...>> : public AOReaderBase {
340
341 using BaseT = BinnedDbn<DbnN, AxisT...>;
342
343 template <size_t I>
344 using is_CAxis = typename std::is_floating_point<typename std::tuple_element_t<I, std::tuple<AxisT...>>>;
345
346 std::tuple<vector<AxisT> ...> edges;
347 Dbn<DbnN> yoda1Overflow;
348 vector<Dbn<DbnN>> dbns;
349 vector<size_t> maskedBins;
350 std::array<double,DbnN*(DbnN-1)/2> crossTerms;
351 bool isYODA1 = false;
352 size_t axisCheck = 0;
353
354
355 template<size_t I>
356 void readEdges() { // YODA1 version for backwards compatibility
357 if constexpr(I < sizeof...(AxisT)) {
358 using EdgeT = std::tuple_element_t<I, std::tuple<AxisT...>>;
359 if constexpr (is_CAxis<I>::value) { // continuous case
360 EdgeT lo, hi;
361 aiss >> lo >> hi;
362 auto& curr_edges = std::get<I>(edges);
363 if (!std::isinf(lo)) {
364 if (curr_edges.empty()) curr_edges.push_back(lo);
365 }
366 if (!std::isinf(hi)) {
367 if (curr_edges.size() && curr_edges[ curr_edges.size() - 1 ] != hi) {
368 curr_edges.push_back(hi);
369 }
370 }
371 }
372 else { // discrete case
373 throw BinningError("Discrete axes are not supported in this YODA1-style legacy format.");
374 }
375 readEdges<I+1>();
376 }
377 }
378
379 template<size_t I>
380 void readEdges(const std::string& line) { // YODA2 version
381 if constexpr(I < sizeof...(AxisT)) {
382 if (I == axisCheck) {
383 using EdgeT = std::tuple_element_t<I, std::tuple<AxisT...>>;
384 auto& curr_edges = std::get<I>(edges);
385 extractVector<EdgeT>(line, curr_edges);
386 }
387 readEdges<I+1>(line);
388 }
389 }
390
391 #ifdef HAVE_HDF5
392
393 template <size_t I>
394 void loadEdges(H5FileManager& h5file) {
395 if constexpr(I < sizeof...(AxisT)) {
396 using EdgeT = std::tuple_element_t<I, std::tuple<AxisT...>>;
397 auto& curr_edges = std::get<I>(edges);
398 curr_edges = h5file.loadEdges<EdgeT>();
399 loadEdges<I+1>(h5file);
400 }
401 }
402
403 template <size_t I>
404 void skipEdges(H5FileManager& h5file) {
405 if constexpr(I < sizeof...(AxisT)) {
406 using EdgeT = std::tuple_element_t<I, std::tuple<AxisT...>>;
407 h5file.skipEdges<EdgeT>();
408 skipEdges<I+1>(h5file);
409 }
410 }
411
412 #endif
413
414 template<size_t I>
415 void readDbn(std::array<double,DbnN+1>& sumW, std::array<double,DbnN+1>& sumW2) {
416 if constexpr(I <= DbnN) {
417 double w(0), w2(0);
418 aiss >> w >> w2;
419 sumW[I] = w;
420 sumW2[I] = w2;
421 readDbn<I+1>(sumW, sumW2);
422 }
423 }
424
425 template <class tupleT, size_t... Is>
426 BaseT* make_from_tuple(tupleT&& tuple, std::index_sequence<Is...> ) {
427 BaseT* rtn = new BaseT{std::get<Is>(std::forward<tupleT>(tuple))...};
428 rtn->maskBins(maskedBins);
429 return rtn;
430 }
431
432 template <class tupleT>
433 BaseT* make_from_tuple(tupleT&& tuple) {
434 return make_from_tuple(std::forward<tupleT>(tuple),
435 std::make_index_sequence<sizeof...(AxisT)+1>{});
436 }
437
438 template<size_t I>
439 void clearEdges() {
440 if constexpr(I < sizeof...(AxisT)) {
441 std::get<I>(edges).clear();
442 clearEdges<I+1>();
443 }
444 }
445
446 public:
447
448 void parse(const string& line) {
449 if (line.find("Total") != string::npos) {
450 isYODA1 = true;
451 return; // YODA1 backwards compatibility
452 }
453 if (!line.rfind("Edges(A", 0)) { // parse binning
454 readEdges<0>(line);
455 ++axisCheck;
456 return;
457 }
458 if (!line.rfind("MaskedBins: ", 0)) { // parse indices of masked bins
459 extractVector<size_t>(line, maskedBins);
460 return;
461 }
462 aiss.reset(line);
463 if (line.find("Underflow") != string::npos || line.find("Overflow") != string::npos) {
464 // This must be the YODA1-style format ...
465 if constexpr (sizeof...(AxisT) == 1) {
466 string tmp1, tmp2;
467 aiss >> tmp1 >> tmp2; // not needed
468 }
469 }
470 else if (isYODA1) readEdges<0>();
471 std::array<double,DbnN+1> sumW, sumW2;
472 readDbn<0>(sumW, sumW2);
473 for (size_t i = 0; i < crossTerms.size(); ++i) {
474 double tmp(0.);
475 aiss >> tmp;
476 crossTerms.at(i) = tmp;
477 }
478 double numEntries(0);
479 aiss >> numEntries;
480 if (line.find("Overflow") != string::npos) {
481 if constexpr (sizeof...(AxisT) == 1) {
482 if constexpr (DbnN < 2)
483 yoda1Overflow = Dbn<DbnN>(numEntries, sumW, sumW2);
484 else
485 yoda1Overflow = Dbn<DbnN>(numEntries, sumW, sumW2, crossTerms);
486 }
487 }
488 else {
489 if constexpr (DbnN < 2) {
490 dbns.emplace_back(numEntries, sumW, sumW2);
491 }
492 else {
493 dbns.emplace_back(numEntries, sumW, sumW2, crossTerms);
494 }
495 }
496 }
497
498 AnalysisObject* assemble(const string& path = "") {
499
500 auto args = std::tuple_cat(edges, std::make_tuple(path));
501 BaseT* ao = make_from_tuple(std::move(args));
502
503 size_t global_index = 0;
504 if constexpr (sizeof...(AxisT) == 2) {
505 if (isYODA1) { // 2D objects had no under-/overflows in Y1
506 for (size_t ix = 1; ix < ao->numBinsAt(0)+1; ++ix) { //< visible bins only
507 for (size_t iy = 1; iy < ao->numBinsAt(1)+1; ++iy) { //< visible bins only
508 ao->bin(ix,iy).set(std::move(dbns[global_index++]));
509 }
510 }
511 }
512 }
513 if ( !(isYODA1 && sizeof...(AxisT) == 2) ) { //< still works for Y1-style 1D
514 for (auto&& d : dbns) {
515 ao->bin(global_index++).set(std::move(d));
516 }
517 }
518
519 if constexpr (sizeof...(AxisT) == 1) { // YODA1-style overflows
520 if (isYODA1) ao->bin(global_index).set(yoda1Overflow);
521 yoda1Overflow = Dbn<DbnN>();
522 }
523
524 crossTerms.fill(0);
525 maskedBins.clear();
526 isYODA1 = false;
527 clearEdges<0>();
528 dbns.clear();
529 axisCheck = 0;
530 return ao;
531 }
532
533 #ifdef HAVE_HDF5
534
536 loadEdges<0>(h5file);
537 maskedBins = h5file.loadMasks();
538 vector<string> annos = h5file.loadAnnotations();
539 auto args = std::tuple_cat(edges, std::make_tuple(h5file.path(), annos.back()));
540 BaseT* ao = make_from_tuple(std::move(args));
541 annos.pop_back();
542 ao->deserializeMeta(annos);
543 ao->deserializeContent(h5file.loadContent());
544 maskedBins.clear();
545 clearEdges<0>();
546 return ao;
547 }
548
549 void skip(H5FileManager& h5file) {
550 skipEdges<0>(h5file);
551 h5file.skipMasks();
552 h5file.skipCommon();
553 }
554
555 #endif
556
557 };
558
559
560 template <typename... AxisT>
561 class AOReader<BinnedEstimate<AxisT...>> : public AOReaderBase {
562
563 using BaseT = BinnedEstimate<AxisT...>;
564
565 std::tuple<vector<AxisT> ...> edges;
566 vector<Estimate> estimates;
567 vector<size_t> maskedBins;
568 vector<string> sources;
569 size_t axisCheck = 0;
570
571
572 template<size_t I>
573 void readEdges(const std::string& line) {
574 if constexpr(I < sizeof...(AxisT)) {
575 if (I == axisCheck) {
576 using EdgeT = std::tuple_element_t<I, std::tuple<AxisT...>>;
577 auto& curr_edges = std::get<I>(edges);
578 extractVector<EdgeT>(line, curr_edges);
579 }
580 readEdges<I+1>(line);
581 }
582 }
583
584 void readErrors(std::map<string,std::pair<double,double>>& errors) {
585 string eDn, eUp;
586 for (const std::string& src : sources) {
587 aiss >> eDn >> eUp;
588 if (eDn != "---" && eUp != "---") {
589 errors[src] = { Utils::toDbl(eDn), Utils::toDbl(eUp) };
590 }
591 }
592 }
593
594 #ifdef HAVE_HDF5
595
596 template <size_t I>
597 void loadEdges(H5FileManager& h5file) {
598 if constexpr(I < sizeof...(AxisT)) {
599 using EdgeT = std::tuple_element_t<I, std::tuple<AxisT...>>;
600 auto& curr_edges = std::get<I>(edges);
601 curr_edges = h5file.loadEdges<EdgeT>();
602 loadEdges<I+1>(h5file);
603 }
604 }
605
606 template <size_t I>
607 void skipEdges(H5FileManager& h5file) {
608 if constexpr(I < sizeof...(AxisT)) {
609 using EdgeT = std::tuple_element_t<I, std::tuple<AxisT...>>;
610 h5file.skipEdges<EdgeT>();
611 skipEdges<I+1>(h5file);
612 }
613 }
614
615 #endif
616
617 template <class tupleT, size_t... Is>
618 BaseT* make_from_tuple(tupleT&& tuple, std::index_sequence<Is...> ) {
619 BaseT* rtn = new BaseT{std::get<Is>(std::forward<tupleT>(tuple))...};
620 rtn->maskBins(maskedBins);
621 return rtn;
622 }
623
624 template <class tupleT>
625 BaseT* make_from_tuple(tupleT&& tuple) {
626 return make_from_tuple(std::forward<tupleT>(tuple),
627 std::make_index_sequence<sizeof...(AxisT)+1>{});
628 }
629
630 template<size_t I>
631 void clearEdges() {
632 if constexpr(I < sizeof...(AxisT)) {
633 std::get<I>(edges).clear();
634 clearEdges<I+1>();
635 }
636 }
637
638 public:
639
640 void parse(const string& line) {
641 if (!line.rfind("Edges(A", 0)) { // parse binning
642 readEdges<0>(line);
643 ++axisCheck;
644 return;
645 }
646 if (!line.rfind("MaskedBins: ", 0)) { // parse indices of masked bins
647 extractVector<size_t>(line, maskedBins);
648 return;
649 }
650 if (!line.rfind("ErrorLabels: ", 0)) { // parse error labels
651 extractVector<std::string>(line, sources);
652 return;
653 }
654 // parse bin content
655 aiss.reset(line);
656 double val(0);
657 aiss >> val;
658 std::map<string,std::pair<double,double>> errors;
659 readErrors(errors);
660 estimates.emplace_back(val, errors);
661 }
662
663 AnalysisObject* assemble(const string& path = "") {
664
665 auto args = std::tuple_cat(edges, std::make_tuple(path));
666 BaseT* ao = make_from_tuple(std::move(args));
667
668 size_t global_index = 0;
669 for (auto&& e : estimates) {
670 ao->bin(global_index++) = std::move(e);
671 }
672
673 clearEdges<0>();
674 sources.clear();
675 estimates.clear();
676 maskedBins.clear();
677 axisCheck = 0;
678 return ao;
679 }
680
681 #ifdef HAVE_HDF5
682
684 loadEdges<0>(h5file);
685 maskedBins = h5file.loadMasks();
686
687 vector<string> annos = h5file.loadAnnotations();
688 auto args = std::tuple_cat(edges, std::make_tuple(h5file.path(), annos.back()));
689 BaseT* ao = make_from_tuple(std::move(args));
690 annos.pop_back();
691 ao->deserializeMeta(annos);
692 ao->deserializeContent(h5file.loadContent());
693
694 for (auto& b : ao->bins(true, true)) {
695 b.deserializeSources(h5file.loadSources());
696 }
697
698 maskedBins.clear();
699 clearEdges<0>();
700 return ao;
701 }
702
703 void skip(H5FileManager& h5file) {
704 skipEdges<0>(h5file);
705 h5file.skipMasks();
706 h5file.skipSources();
707 h5file.skipCommon();
708 }
709
710 #endif
711
712 };
713
714
715}
716
717#endif
AOReaderBase()
Default constructor.
virtual void skip(H5FileManager &h5file)
virtual void parse(const string &line)=0
virtual ~AOReaderBase()
Default destructor.
aistringstream aiss
virtual AnalysisObject * assemble(const string &path="")=0
void extractVector(const std::string &line, std::vector< T > &vec)
virtual AnalysisObject * mkFromH5(H5FileManager &)=0
AnalysisObject * mkFromH5(H5FileManager &h5file)
AnalysisObject * assemble(const string &path="")
AnalysisObject * mkFromH5(H5FileManager &h5file)
AnalysisObject * assemble(const string &path="")
void parse(const string &line)
AnalysisObject * mkFromH5(H5FileManager &h5file)
AnalysisObject * assemble(const string &path="")
AnalysisObject * assemble(const string &path="")
void parse(const string &line)
AnalysisObject * mkFromH5(H5FileManager &h5file)
void skip(H5FileManager &h5file)
AnalysisObject * mkFromH5(H5FileManager &h5file)
void parse(const string &line)
AnalysisObject * assemble(const string &path="")
AnalysisObject is the base class for histograms and scatters.
User-facing BinnedDbn class in arbitrary dimension.
Definition BinnedDbn.h:55
Forward declaration.
Error for general binning problems.
Definition Exceptions.h:27
A weighted counter.
Definition Counter.h:26
Partial template specialisation for Dbn0D.
Definition Dbn.h:647
User-facing Dbn class inheriting from DbnBase.
Definition Dbn.h:637
An estimate in 0D.
Definition Estimate0D.h:28
Helper class to extract AO information from a H5 file.
Definition H5Utils.h:272
vector< string > loadSources() noexcept
Labels of error sources of current AO.
Definition H5Utils.h:359
void skipMasks() noexcept
Skips next set of masked indices of current AO.
Definition H5Utils.h:354
vector< size_t > loadMasks() noexcept
Indices of masked bins in current AO.
Definition H5Utils.h:349
const string & path() const
Path of current AO.
Definition H5Utils.h:323
void skipCommon() noexcept
Skips next set of annotations and content of current AO.
Definition H5Utils.h:343
vector< EdgeT > loadEdges() noexcept
Returns next set of edges of type EdgeT.
Definition H5Utils.h:385
vector< string > loadAnnotations() noexcept
Serialized annotations of current AO.
Definition H5Utils.h:333
vector< double > loadContent() noexcept
Serialized content of current AO.
Definition H5Utils.h:338
void skipSources() noexcept
Skips next set of error sources of current AO.
Definition H5Utils.h:379
void skipEdges() noexcept
Skips next set of edges of type EdgeT.
Definition H5Utils.h:399
Error for file reading errors.
Definition Exceptions.h:72
A generic data type which is just a collection of n-dim data points with errors.
Definition Scatter.h:153
Anonymous namespace to limit visibility.
Dbn< 0 > Dbn0D
User-friendly aliases.
Definition Dbn.h:775