49 operator bool()
const {
return m_ptr !=
nullptr &&
m_size>0; }
50 template <
typename T_Other>
52 return m_size == other_range.m_size;
96 template <
typename T_Other>
98 return m_rows == other_range.m_size;
104 template <
class T_Stream,
typename T_Counter>
108 const std::string &label_prefix,
109 const std::size_t column_width,
110 const std::size_t min_label_width,
111 const bool dump_footer,
112 const bool separate_last_row,
113 const unsigned int precision) {
114 if (counter &&
label && counter.equalSize(
label)) {
115 std::size_t max_size =min_label_width;
116 for (
const std::string &name :
label ) {
117 max_size = std::max(max_size, name.size());
119 const std::size_t total_size =max_size+3+2*2+column_width;
120 std::string line(total_size,
'-');
121 std::array<std::size_t,3> vertical_line_pos{0u, max_size+3, line.size()-1};
122 for (std::size_t pos : vertical_line_pos) {
125 out << line << std::endl;
128 auto default_precision = out.precision();
129 for (
const T_Counter &
a : counter) {
130 if (separate_last_row && idx+1 ==
label.size()) {
131 out << line << std::endl;
133 assert( idx <
label.size());
134 out <<
"| " << (label_prefix.empty() ? std::left : std::right)
135 << std::setw(label_prefix.size()) << ( idx==0 ? label_prefix :
empty)
136 << std::setw(max_size-label_prefix.size()) <<
label[idx] << std::right
137 << std::setprecision( precision != std::numeric_limits<unsigned int>::max() ? precision : default_precision)
138 <<
" | " << std::setw(column_width) <<
a <<
" |" << std::endl;
141 out << std::setprecision(default_precision);
143 out << line << std::endl;
152 template <
class T_Stream,
typename T_Counter>
157 const std::string &top_left_label,
158 const std::string &label_prefix,
159 const std::size_t column_width,
160 const std::size_t min_label_width,
161 const bool dump_header,
162 const bool dump_footer,
163 const bool separate_last_row,
164 const std::vector<unsigned int> &precision) {
165 if (counter && row_label && column_label
166 && counter.equalSize(row_label)
167 && counter.nColumns() == column_label.
size()) {
168 std::size_t max_size =std::max(top_left_label.size(),
static_cast<std::size_t
>(min_label_width));
169 for (
const std::string &name : row_label ) {
170 max_size = std::max(max_size, name.size() + label_prefix.size());
172 std::size_t the_width = column_width;
173 for (
const std::string &name : column_label ) {
174 the_width = std::max(the_width, name.size());
176 std::size_t total_size =max_size+2*2;
177 for (std::size_t column_i=0; column_i<column_label.size(); ++column_i) {
178 total_size += the_width + 3;
180 std::string line(total_size,
'-');
184 for (std::size_t column_i=0; column_i<column_label.size(); ++column_i) {
186 pos += the_width + 3;
188 line[line.size()-1]=
'|';
190 out << line << std::endl <<
"| " << std::setw(max_size) << top_left_label <<
" |" << std::left;
191 for (
const std::string &
header : column_label ) {
192 out <<
" " << std::setw(the_width) <<
header <<
" |";
194 out << std::right << std::endl;
196 out << line << std::endl;
199 auto default_precision = out.precision();
201 if (separate_last_row && idx+1 == row_label.size()) {
202 out << line << std::endl;
204 assert( idx < row_label.size());
205 out <<
"| " << (label_prefix.empty() ? std::left : std::right)
206 << std::setw(label_prefix.size()) << ( idx==0 ? label_prefix :
empty)
207 << std::setw(max_size-label_prefix.size()) << row_label[idx] << std::right <<
" |";
208 unsigned int col_i=0;
209 for (
const T_Counter &
a : a_row) {
211 << std::setprecision( (col_i < precision.size()
212 && precision[col_i] != std::numeric_limits<unsigned int>::max())
215 << std::setw(the_width) <<
a <<
" |";
221 out << std::setprecision(default_precision);
230 template <
typename T>
245 unsigned int m_precision = std::numeric_limits<unsigned int>::max();
251 template <
typename T>
285 stat.m_minLabelWidth,
288 stat.m_separateLastRow,
294 template <
typename T_index,
class T_
string>
296 std::initializer_list<std::pair<T_index, T_string> > a_list)
298 std::vector<std::string> labels;
299 labels.resize( n_entries );
300 if (a_list.size() != n_entries) {
301 throw std::logic_error(
"Expected number of entries and elements in the initializer lists do not match.");
303 for (
auto elm : a_list) {
304 labels.at(elm.first) = std::move(elm.second);
309 template<
class T_Collection>
311 std::size_t max_width=0u;
312 for (
const auto &elm : col ) {
313 max_width = std::max( max_width, elm.size());
319 constexpr inline std::size_t
categoryStride([[maybe_unused]]
const std::size_t categories,
320 [[maybe_unused]]
const std::size_t sub_categories,
321 [[maybe_unused]]
const std::size_t n_counter) {
325 [[maybe_unused]]
const std::size_t sub_categories,
326 [[maybe_unused]]
const std::size_t n_counter) {
327 return (categories+1) * n_counter;
329 constexpr inline std::size_t
counterStride([[maybe_unused]]
const std::size_t categories,
330 [[maybe_unused]]
const std::size_t sub_categories,
331 [[maybe_unused]]
const std::size_t n_counter) {
332 return (categories+1);
341 template<
typename T_Output,
typename T_Input, const std::
size_t N>
343 const std::size_t sub_categories,
344 const std::vector< std::array<T_Input, N> > &input_counts) {
345 if (categories*sub_categories!= input_counts.size()) {
346 std::stringstream
msg;
347 msg <<
"Category dimensions (" << categories <<
" * " << sub_categories <<
"="
348 << (categories * sub_categories) <<
") and input counter container size "
349 << input_counts.size() <<
" do not match.";
350 throw std::logic_error(
msg.str());
352 std::vector<T_Output> output_counts;
353 output_counts.resize((categories+1) * (sub_categories+1) * N);
354 const std::size_t sub_category_stride =
subCategoryStride(categories, sub_categories, N);
355 const std::size_t counter_stride =
counterStride(categories, sub_categories, N);
356 const std::size_t category_stride =
categoryStride(categories, sub_categories, N);
358 for (std::size_t sub_category_i=0;
359 sub_category_i < sub_categories;
361 for (std::size_t category_i=0; category_i<categories; ++category_i) {
362 std::size_t src_idx = category_i * sub_categories + sub_category_i;
363 std::size_t dest_idx_base = sub_category_i * sub_category_stride + 0 * counter_stride;
364 std::size_t dest_idx_project_categories_base = dest_idx_base;
365 dest_idx_base += category_i * category_stride;
366 dest_idx_project_categories_base += categories * category_stride;
368 for (std::size_t counter_i=0; counter_i<N; ++counter_i) {
369 std::size_t dest_idx=dest_idx_base + counter_i * counter_stride;
370 assert( src_idx < input_counts.size() && counter_i < input_counts[src_idx].size());
371 assert( dest_idx < output_counts.size());
372 output_counts[dest_idx] = input_counts[src_idx][counter_i];
373 assert( dest_idx_project_categories_base + counter_i * counter_stride < output_counts.size());
374 output_counts[dest_idx_project_categories_base + counter_i * counter_stride] += output_counts[dest_idx];
379 for (std::size_t category_i=0; category_i<=categories; ++category_i) {
380 for (std::size_t counter_i=0; counter_i<N; ++counter_i) {
381 std::size_t dest_idx_base = 0 * sub_category_stride + counter_i * counter_stride + category_i;
382 std::size_t dest_idx_project_sub_categories = sub_categories * sub_category_stride + dest_idx_base;
383 assert( dest_idx_project_sub_categories < output_counts.size() );
384 for (std::size_t sub_category_i=0;
385 sub_category_i<sub_categories;
387 std::size_t sub_category_idx = dest_idx_base + sub_category_i * sub_category_stride;
388 assert( sub_category_idx < output_counts.size() );
389 output_counts[dest_idx_project_sub_categories] += output_counts[sub_category_idx];
393 return output_counts;
398 std::vector< SummandDefinition > >;
402 template <
typename T>
405 return std::make_pair( std::vector< SummandDefinition > { std::make_pair(
static_cast<std::size_t
>(numerator),1)},
406 std::vector< SummandDefinition > { std::make_pair(
static_cast<std::size_t
>(denominator),1)});
410 template <
typename T>
411 inline std::tuple< std::string, RatioDefinition >
413 return std::make_pair( std::move(name),
defineSimpleRatio(numerator, denominator) );
421 template <
typename T>
423 return std::make_pair(
static_cast<std::size_t
>(counter_idx), multiplier) ;
428 std::size_t
computeSum(
const std::vector< SummandDefinition > &sum_def,
429 std::size_t eta_offset,
430 std::size_t row_stride,
432 const std::vector<std::size_t> &stat);
434 inline float computeRatio(std::size_t numerator, std::size_t denominator) {
435 return denominator!=0 ?
static_cast<float>(numerator/
static_cast<double>(denominator)) : 0.f;
445 std::size_t eta_offset,
446 std::size_t row_stride,
448 const std::vector<std::size_t> &stat) {
449 std::size_t numerator=
computeSum(ratio_def.first, eta_offset, row_stride, seed_i, stat);
450 std::size_t denominator=!ratio_def.second.empty()
451 ?
computeSum(ratio_def.second, eta_offset, row_stride, seed_i, stat)
459 std::vector< SummandDefinition >&&numerator,
460 std::vector< SummandDefinition >&&denominator) {
461 return std::make_tuple(std::move(name),
462 std::make_pair(std::move(numerator),
463 std::move(denominator)));
467 inline std::tuple<std::vector<std::string>, std::vector<RatioDefinition> >
470 std::tuple< std::vector<std::string>, std::vector<RatioDefinition> > splitted;
471 std::get<0>(splitted).reserve( a_ratio_list.size() );
472 for (
auto a_ratio : a_ratio_list) {
473 std::get<0>(splitted).emplace_back( std::move(std::get<0>(a_ratio)) );
475 std::get<1>(splitted).reserve( a_ratio_list.size() );
476 for (
auto a_ratio : a_ratio_list) {
477 std::get<1>(splitted).emplace_back( std::move(std::get<1>(a_ratio)) );
483 constexpr inline std::size_t
categoryStride([[maybe_unused]]
const std::size_t categories,
484 [[maybe_unused]]
const std::size_t sub_categories,
485 [[maybe_unused]]
const std::vector<RatioDefinition> &ratio_def) {
489 [[maybe_unused]]
const std::size_t sub_categories,
490 [[maybe_unused]]
const std::vector<RatioDefinition> &ratio_def) {
491 return (categories) * ratio_def.size();
493 constexpr inline std::size_t
ratioStride([[maybe_unused]]
const std::size_t categories,
494 [[maybe_unused]]
const std::size_t sub_categories,
495 [[maybe_unused]]
const std::vector<RatioDefinition> &ratio_def) {
500 std::vector<float>
computeRatios(
const std::vector<RatioDefinition> &ratio_def,
501 const std::size_t categories,
502 const std::size_t sub_categories,
503 const std::vector< std::size_t> &counter);
506 const std::vector<float> &
bins,
508 bool abs_value=
false,
510 std::stringstream range_label;
511 range_label << std::fixed << std::setprecision(precision);
512 if (bin_i==
bins.size()+1) {
513 range_label <<
" All " << variable_name;
517 std::stringstream value_str;
518 value_str << std::fixed << std::setprecision(precision) << 0.;
519 range_label << std::setw(4) << (abs_value ? value_str.str().c_str() :
"-inf") <<
"-";
522 range_label << std::setw(4) <<
bins.at(bin_i-1) <<
"-";
524 if (bin_i>=
bins.size()) {
525 range_label << std::setw(4) <<
"+inf";
528 range_label << std::setw(4) <<
bins.at(bin_i);
531 return range_label.str();
535 std::size_t eta_bin_i,
536 bool abs_eta=
false) {
542template <
typename T, std::
size_t N>
544 const std::array<std::string, N> &
label) {
552template <
typename T, std::
size_t Nrows, std::
size_t Ncolumns>
554 const std::array<std::string, Nrows> &row_label,
555 const std::array<std::string, Ncolumns> &column_label,
556 const std::string &top_left_label=
"") {
559 counter.size(), column_label.size(),
560 !counter.empty() ?
static_cast<std::size_t
>(&counter[1][0] - &counter[0][0]) : 0u,
572 std::size_t start_idx,
573 std::size_t row_stride,
574 const std::vector<std::string> &row_label,
575 const std::vector<std::string> &column_label,
576 const std::string &top_left_label=
"") {
577 if (start_idx + (row_label.size()-1) * row_stride >= counter.size() || row_stride < column_label.size()) {
578 std::stringstream
msg;
579 msg <<
"Counter dimension and label dimensions (" << row_label.size() <<
" * " << column_label.size()
580 <<
") do not match: [" << start_idx <<
", "
581 << start_idx <<
" + " << (row_label.size()-1) <<
" * " << row_stride <<
" = "
582 << (start_idx + (row_label.size()-1) * row_stride)
583 <<
" !< " << counter.size();
584 msg <<
" [row_labels:";
585 for (
const std::string &
label : row_label) {
588 msg <<
"; column_labels:";
589 for (
const std::string &
label : column_label) {
593 throw std::logic_error(
msg.str());
608template <
typename T, std::
size_t N>
610 std::size_t start_row_idx,
611 std::size_t row_stride,
612 std::size_t start_column_idx,
613 std::size_t column_stride,
614 const std::vector<std::string> &row_label,
615 const std::vector<std::string> &column_label,
616 const std::string &top_left_label=
"") {
617 if (start_row_idx + (row_label.size()-1) * row_stride >= counter.size()*N
618 || start_column_idx + (column_label.size()-1) * column_stride >= counter.size()*N
619 || (row_stride*row_label.size()>column_stride && column_stride*column_label.size()>row_stride) ) {
620 std::stringstream
msg;
621 msg <<
"Counter dimension and label dimensions (" << row_label.size() <<
" * " << column_label.size()
622 <<
") do not match: [" << start_row_idx <<
", "
623 << start_row_idx <<
" + " << (row_label.size()-1) <<
" * " << row_stride <<
" = "
624 << (start_row_idx + (row_label.size()-1) * row_stride)
626 << start_column_idx <<
" + " << (column_label.size()-1) <<
" * " << column_stride <<
" = "
627 << (start_column_idx + (column_label.size()-1) * column_stride)
628 <<
" !< " << counter.size()
630 << (start_row_idx + (row_label.size()-1) * row_stride) <<
" >= " << (counter.size()*N)
631 <<
" || " << (start_column_idx + (column_label.size()-1) * column_stride) <<
" >= " << (counter.size()*N)
632 <<
" || ( " << (row_stride*row_label.size()) <<
" > " << column_stride <<
" && " << (column_stride*column_label.size()) <<
" > " << (row_stride)
634 msg <<
" [row_labels:";
635 for (
const std::string &
label : row_label) {
638 msg <<
"; column_labels:";
639 for (
const std::string &
label : column_label) {
643 throw std::logic_error(
msg.str());
661 const std::vector<std::string> &row_label,
662 const std::vector<std::string> &column_label,
663 const std::string &top_left_label=
"") {
664 return makeTable(counter, 0u, column_label.size(), row_label, column_label, top_left_label);
675 return dumpTable(out,
680 stat.m_minLabelWidth,
682 stat.m_separateLastRow,
693 return dumpTable(out,
700 stat.m_minLabelWidth,
703 stat.m_separateLastRow,
714 return dumpTable(out,
719 stat.m_minLabelWidth,
721 stat.m_separateLastRow,
static const std::vector< std::string > bins
MsgStream & operator<<(MsgStream &out, const TableUtils::StatTable< T > &stat)
TableUtils::StatTable< T > makeTable(const std::array< T, N > &counter, const std::array< std::string, N > &label)
static const Attributes_t empty
std::string label(const std::string &format, int i)
std::tuple< std::vector< std::string >, std::vector< RatioDefinition > > splitRatioDefinitionsAndLabels(std::initializer_list< std::tuple< std::string, RatioDefinition > > a_ratio_list)
SummandDefinition defineSummand(T counter_idx, int multiplier)
constexpr std::size_t categoryStride(const std::size_t categories, const std::size_t sub_categories, const std::size_t n_counter)
float computeRatio(std::size_t numerator, std::size_t denominator)
T_Stream & dumpTable(T_Stream &out, Range< T_Counter > counter, const Range< std::string > &label, const std::string &label_prefix, const std::size_t column_width, const std::size_t min_label_width, const bool dump_footer, const bool separate_last_row, const unsigned int precision)
std::size_t maxLabelWidth(const T_Collection &col)
RatioDefinition defineSimpleRatio(T numerator, T denominator)
std::size_t computeSum(const std::vector< SummandDefinition > &sum_def, std::size_t eta_offset, std::size_t row_stride, std::size_t seed_i, const std::vector< std::size_t > &stat)
std::vector< std::string > makeLabelVector(T_index n_entries, std::initializer_list< std::pair< T_index, T_string > > a_list)
constexpr std::size_t subCategoryStride(const std::size_t categories, const std::size_t sub_categories, const std::size_t n_counter)
constexpr std::size_t counterStride(const std::size_t categories, const std::size_t sub_categories, const std::size_t n_counter)
std::string makeEtaBinLabel(const std::vector< float > &eta_bins, std::size_t eta_bin_i, bool abs_eta=false)
std::tuple< std::string, RatioDefinition > makeRatioDefinition(std::string &&name, std::vector< SummandDefinition > &&numerator, std::vector< SummandDefinition > &&denominator)
std::pair< std::vector< SummandDefinition >, std::vector< SummandDefinition > > RatioDefinition
std::string makeBinLabel(const std::string &variable_name, const std::vector< float > &bins, std::size_t bin_i, bool abs_value=false, int precision=1)
std::vector< float > computeRatios(const std::vector< RatioDefinition > &ratio_def, const std::size_t categories, const std::size_t sub_categories, const std::vector< std::size_t > &counter)
constexpr std::size_t ratioStride(const std::size_t categories, const std::size_t sub_categories, const std::vector< RatioDefinition > &ratio_def)
std::vector< T_Output > createCounterArrayWithProjections(const std::size_t categories, const std::size_t sub_categories, const std::vector< std::array< T_Input, N > > &input_counts)
std::pair< std::size_t, int > SummandDefinition
std::string m_labelPrefix
MultiColumnTable & dumpFooter(bool value=true)
Range< std::string > m_columnLabel
friend std::ostream & operator<<(std::ostream &out, const MultiColumnTable &stat)
MultiColumnTable & separateLastRow(bool value=true)
std::size_t m_columnWidth
MultiColumnTable & columnWidth(std::size_t value)
MultiColumnTable & minLabelWidth(std::size_t value)
Range< std::string > m_rowLabel
std::string m_topLeftLable
std::size_t m_minLabelWidth
std::vector< unsigned int > m_precision
MultiColumnTable & precision(std::vector< unsigned int > &&precision)
MultiColumnTable & dumpHeader(bool value=true)
MultiColumnTable & labelPrefix(const std::string &value)
std::size_t m_columnOffset
Range< T > operator*() const
std::size_t m_firstColumnIndex
const_iterator & operator++()
bool operator!=(const const_iterator &other) const
const_iterator begin() const
const_iterator end() const
bool equalSize(const T_Other &other_range)
std::size_t nRows() const
std::size_t m_columnOffset
std::size_t nColumns() const
std::size_t m_firstColumnIndex
Range< T > operator[](std::size_t index) const
const_iterator & operator++()
bool operator!=(const const_iterator &other) const
const T & operator*() const
bool equalSize(const T_Other &other_range)
const_iterator begin() const
const T & operator[](std::size_t index) const
const_iterator end() const
StatTable & precision(unsigned int precision)
Range< std::string > m_label
StatTable & minLabelWidth(std::size_t value)
StatTable & separateLastRow(bool value=true)
StatTable & dumpFooter(bool value=true)
StatTable & columnWidth(std::size_t value)
StatTable & labelPrefix(const std::string &value)
std::size_t m_columnWidth
StatTable & dumpHeader(bool value=true)
std::size_t m_minLabelWidth
std::string m_labelPrefix