yoda is hosted by Hepforge, IPPP Durham
YODA - Yet more Objects for Data Analysis 2.1.0
Writer.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_Writer_h
7#define YODA_Writer_h
8
10#include "YODA/Utils/Traits.h"
11
12#ifdef HAVE_HDF5
14#ifdef WITH_HIGHFIVE
15#include <YODA/highfive/H5File.hpp>
16#else
17#include <highfive/H5File.hpp>
18#define YODA_H5 HighFive
19#endif
20#endif
21
22#include <type_traits>
23#include <fstream>
24#include <iostream>
25
26namespace YODA {
27
28 static bool enableH5compression = Utils::getEnvParam("YODA_HDF5_COMPRESSION", true);
29
31 class Writer {
32 public:
33
35 virtual ~Writer() {}
36
37
40
42 void write(const std::string& filename, const AnalysisObject& ao);
43
45 void write(std::ostream& stream, const AnalysisObject& ao) {
46 std::vector<const AnalysisObject*> vec{&ao};
47 write(stream, vec);
48 }
49
51 template <typename T>
52 typename std::enable_if_t<DerefableToAO<T>::value> //< -> void if valid
53 write(std::ostream& stream, const T& ao) {
54 write(stream, *ao);
55 }
56
58 template <typename T>
59 typename std::enable_if_t<DerefableToAO<T>::value> //< -> void if valid
60 write(const std::string& filename, const T& ao) {
61 write(filename, *ao);
62 }
63
65
66
69
76 #ifdef HAVE_HDF5
77 void write(YODA_H5::File& file, const std::vector<const AnalysisObject*>& aos);
78 #endif
79
86 void write(std::ostream& stream, const std::vector<const AnalysisObject*>& aos, int precision = -1);
87
88
93 template <typename RANGE>
94 typename std::enable_if_t<CIterable<RANGE>::value>
95 write(std::ostream& stream, const RANGE& aos) {
96 write(stream, std::begin(aos), std::end(aos));
97 }
98
100 template <typename RANGE>
101 typename std::enable_if_t<CIterable<RANGE>::value>
102 write(const std::string& filename, const RANGE& aos) {
103 write(filename, std::begin(aos), std::end(aos));
104 }
105
107
108
111
116 template <typename AOITER>
117 void write(std::ostream& stream, const AOITER& begin, const AOITER& end, int precision = -1) {
118 std::vector<const AnalysisObject*> vec;
119 // vec.reserve(std::distance(begin, end));
120 for (AOITER ipao = begin; ipao != end; ++ipao) vec.push_back(&(**ipao));
121 write(stream, vec, precision);
122 }
123
124
129 template <typename AOITER>
130 void write(const std::string& filename, const AOITER& begin, const AOITER& end) {
131 std::vector<const AnalysisObject*> vec;
132 // vec.reserve(std::distance(begin, end));
133 for (AOITER ipao = begin; ipao != end; ++ipao) vec.push_back(&(**ipao));
134
135 if (filename != "-") {
136 try {
137 const size_t lastdot = filename.find_last_of(".");
138 std::string fmt = Utils::toLower(lastdot == std::string::npos ? filename : filename.substr(lastdot+1));
139 const bool compress = (fmt == "gz") || (fmt == "h5" && enableH5compression);
140 useCompression(compress);
141 #ifdef HAVE_HDF5
142 // check if the requested format is H5
143 if (Utils::startswith(fmt, "h5")) {
144 try {
145 YODA_H5::File h5(filename, YODA_H5::File::OpenOrCreate | YODA_H5::File::Truncate);
146 write(h5, vec);
147 } catch(...) {
148 throw WriteError("Failed to open HDF5 file " + filename);
149 }
150 return;
151 }
152 #endif
153 // try writing to stream
154 std::ofstream stream;
155 stream.exceptions(std::ofstream::failbit | std::ofstream::badbit);
156 stream.open(filename.c_str());
157 if (stream.fail())
158 throw WriteError("Writing to filename " + filename + " failed");
159 write(stream, vec);
160 } catch (std::ofstream::failure& e) {
161 throw WriteError("Writing to filename " + filename + " failed: " + e.what());
162 }
163 } else {
164 try {
165 write(std::cout, vec);
166 } catch (std::runtime_error& e) {
167 throw WriteError("Writing to stdout failed: " + std::string(e.what()));
168 }
169 }
170
171 }
172
174
176 void setPrecision(int precision) {
177 _precision = precision;
178 }
179
181 void setAOPrecision(const bool needsDP = false) {
182 if (needsDP) _aoprecision = std::numeric_limits<double>::max_digits10;
183 else if (_precision > 0) _aoprecision = _precision;
184 else _aoprecision = 6;
185 }
186
188 void useCompression(const bool compress=true) {
189 _compress = compress;
190 }
191
192
193 protected:
194
197
199 virtual void writeHead(std::ostream&) {}
200
202 virtual void writeBody(std::ostream& stream, const AnalysisObject* ao);
203
205 virtual void writeBody(std::ostream& stream, const AnalysisObject& ao);
206
209 template <typename T>
210 typename std::enable_if_t<DerefableToAO<T>::value> //< -> void if valid
211 writeBody(std::ostream& stream, const T& ao) { writeBody(stream, *ao); }
212
214 virtual void writeFoot(std::ostream& stream) { stream << std::flush; }
215
217
218
221
222 virtual void writeAO(std::ostream& stream, const AnalysisObject& ao) = 0;
223
224 #ifdef HAVE_HDF5
225 virtual void writeAOS(YODA_H5::File& file, const vector<const AnalysisObject*>& aos) = 0;
226 #endif
227
229
230
232 int _precision, _aoprecision;
233
235 bool _compress;
236
237 };
238
239
241 Writer& mkWriter(const std::string& format_name);
242
243
244}
245
246#endif
AnalysisObject is the base class for histograms and scatters.
Error for file writing errors.
Definition Exceptions.h:79
Pure virtual base class for various output writers.
Definition Writer.h:31
virtual void writeHead(std::ostream &)
Write any opening boilerplate required by the format to stream.
Definition Writer.h:199
virtual ~Writer()
Virtual destructor.
Definition Writer.h:35
void write(std::ostream &stream, const AnalysisObject &ao)
Write out object ao to output stream stream.
Definition Writer.h:45
virtual void writeFoot(std::ostream &stream)
Write any closing boilerplate required by the format to stream.
Definition Writer.h:214
std::enable_if_t< DerefableToAO< T >::value > write(const std::string &filename, const T &ao)
Write out pointer-like object ao to file filename.
Definition Writer.h:60
std::enable_if_t< CIterable< RANGE >::value > write(std::ostream &stream, const RANGE &aos)
Definition Writer.h:95
void useCompression(const bool compress=true)
Use libz compression?
Definition Writer.h:188
virtual void writeAO(std::ostream &stream, const AnalysisObject &ao)=0
virtual void writeBody(std::ostream &stream, const AnalysisObject *ao)
Write the body elements corresponding to AnalysisObject ao to stream.
Definition Writer.cc:109
void write(std::ostream &stream, const AOITER &begin, const AOITER &end, int precision=-1)
Definition Writer.h:117
void write(const std::string &filename, const AOITER &begin, const AOITER &end)
Definition Writer.h:130
std::enable_if_t< CIterable< RANGE >::value > write(const std::string &filename, const RANGE &aos)
Write out a collection of objects objs to file filename.
Definition Writer.h:102
void write(const std::string &filename, const AnalysisObject &ao)
Write out object ao to file filename.
Definition Writer.cc:56
virtual void writeAOS(YODA_H5::File &file, const vector< const AnalysisObject * > &aos)=0
std::enable_if_t< DerefableToAO< T >::value > writeBody(std::ostream &stream, const T &ao)
Write the body elements corresponding to AnalysisObject ao to stream.
Definition Writer.h:211
void setAOPrecision(const bool needsDP=false)
Set precision of numerical quantities for current AO in this writer's output.
Definition Writer.h:181
void setPrecision(int precision)
Set precision of numerical quantities in this writer's output.
Definition Writer.h:176
std::enable_if_t< DerefableToAO< T >::value > write(std::ostream &stream, const T &ao)
Write out pointer-like object ao to output stream stream.
Definition Writer.h:53
Anonymous namespace to limit visibility.
static bool enableH5compression
Definition Writer.h:28
Writer & mkWriter(const std::string &format_name)
Factory function to make a writer object by format name or a filename.
Definition Writer.cc:29