Class Reference for E1039 Core & Analysis Software
TDirectoryHelper.cc
Go to the documentation of this file.
1 #include "TDirectoryHelper.h"
2 
3 #include <TDirectory.h>
4 #include <TFile.h>
5 #include <TH1.h>
6 #include <TObject.h>
7 #include <TROOT.h>
8 
9 #include <cassert>
10 #include <iostream>
11 #include <list>
12 #include <string>
13 #include <vector>
14 
15 using namespace std;
16 
17 //_____________________________________________________________________________
18 void
19 TDirectoryHelper::copyToFile(TDirectory* src, TFile* dest)
20 {
21  TDirectory* save = gDirectory;
22 
23  // We basically have two cases here to consider, depending
24  // on whether (1) or not (2) a TFile was opened prior to calling
25  // the PhotonHistogrammer ctor.
26  // 1. Nothing special to do, all the TDirectory structure
27  // is already attached to the correct file.
28  // 2. We have to "duplicate" the TDirectory structure into
29  // the newly opened file and save the HistogramCollection(s) there.
30 
31  if ( !dest || !src )
32  {
33  return;
34  }
35 
36  if ( !dest->IsWritable() )
37  {
38  cerr << "TDirectoryHelper::copyToFile : destination file is not "
39  " writeable" << endl;
40  return;
41  }
42 
43  duplicateDir(dest,src);
44 
45 // list<TDirectory*> mothers;
46 
47 // TDirectory* mother = fDir;
48 
49 // while ( (mother = dynamic_cast<TDirectory*>(mother->GetMother()) ) )
50 // {
51 // std::string motherName = mother->GetName();
52 // if (motherName != "Rint" )
53 // {
54 // mothers.push_front(mother);
55 // }
56 // }
57 
58 // TDirectory* currentDir = save;
59 
60 // list<TDirectory*>::const_iterator it;
61 
62 // for ( it = mothers.begin(); it != mothers.end() ; it++ )
63 // {
64 
65 // TDirectory* dir;
66 // if ( (dir=(TDirectory*)currentDir->FindObject((*it)->GetName()) ))
67 // {
68 // currentDir = dir;
69 // }
70 // else
71 // {
72 // currentDir = currentDir->mkdir((*it)->GetName(),
73 // (*it)->GetTitle());
74 // }
75 // }
76 
77 // TDirectoryHelper::duplicateDir
78 // ((TDirectory*)save->FindObject(mothers.back()->GetName()),fDir);
79 
80 // }
81 
82  save->cd();
83 
84 }
85 
86 
87 //_____________________________________________________________________________
88 void
89 TDirectoryHelper::duplicateDir(TDirectory* dest, TDirectory* source)
90 {
91  dest->cd();
92 
93  TDirectory* newdir;
94 
95  newdir = static_cast<TDirectory*>(gDirectory->FindObject(source->GetName()));
96 
97  if ( !newdir )
98  {
99  newdir = dest->mkdir(source->GetName(),source->GetTitle());
100  }
101 
102  newdir->cd();
103 
104  TIter next(source->GetList());
105  TObject* obj;
106 
107  while ( (obj=next()) )
108  {
109  TDirectory* dir = dynamic_cast<TDirectory*>(obj);
110  if ( dir )
111  {
112  duplicateDir(newdir,dir);
113  }
114  else
115  {
116  obj->Write();
117  }
118  }
119 }
120 
121 //_____________________________________________________________________________
122 bool
123 TDirectoryHelper::mkpath(TDirectory* dir, const string& path)
124 {
125  static vector<string> paths;
126 
127  splitPath(path,paths);
128 
129  TDirectory* currentdir = dir;
130 
131  for ( size_t i = 0 ; i < paths.size() ; i++) {
132 
133  currentdir->cd();
134 
135  currentdir = dynamic_cast<TDirectory*>(gDirectory->Get(paths[i].c_str()));
136  if (!currentdir) {
137  currentdir = gDirectory->mkdir(paths[i].c_str());
138  assert(currentdir!=0);
139  }
140  }
141 
142  return true;
143 }
144 
145 //_____________________________________________________________________________
146 TDirectory*
147 TDirectoryHelper::mkdir(TDirectory* topDir,
148  const char* path,
149  std::vector<std::string>* titles)
150 {
151  TDirectory* save = gDirectory;
152 
153  TDirectory* dir = topDir;
154  TDirectory* tdir = dir;
155 
156  if ( topDir == 0 )
157  {
158  gROOT->cd();
159  tdir = gDirectory;
160  }
161 
162  dir=tdir;
163 
164  dir->cd();
165  std::vector<std::string> paths;
166 
167  splitPath(path,paths);
168 
169  for (size_t i=0;i<paths.size();i++)
170  {
171 
172  TDirectory* subdir = static_cast<TDirectory*>(dir->FindObject(paths[i].c_str()));
173  if ( subdir==0 )
174  {
175  if ( titles && i<titles->size() )
176  {
177  dir = dir->mkdir(paths[i].c_str(),(*titles)[i].c_str());
178  }
179  else
180  {
181  dir = dir->mkdir(paths[i].c_str());
182  }
183  }
184  else
185  {
186  dir = subdir;
187  }
188  dir->cd();
189  }
190 
191  save->cd();
192 
193  return dir;
194 }
195 
196 //_____________________________________________________________________________
197 bool
198 TDirectoryHelper::pathIsInDir(const string& path, TDirectory* dir)
199 {
200  // This is to avoid annoying ROOT message when a directory does not exist
201  // in Cd(), so we provide this small method to check whereas
202  // a path exists under a directory, but without issuing error message
203  // in case of failure (just returning false in this case).
204 
205  TDirectory* dirsave = gDirectory;
206 
207  static std::vector<string> paths;
208 
209  paths.clear();
210  splitPath(path,paths);
211 
212  bool ok = true;
213 
214  TDirectory* cdir = dir;
215 
216  for ( size_t i = 0 ; i < paths.size() && ok ; i++) {
217 
218  cdir->cd();
219 
220  cdir = dynamic_cast<TDirectory*>(cdir->Get(paths[i].c_str()));
221  if ( !cdir ) {
222  ok = false;
223  }
224  }
225 
226  dirsave->cd();
227 
228  return ok;
229 }
230 
231 //_____________________________________________________________________________
232 TH1*
233 TDirectoryHelper::getHisto(TDirectory* dir, const string& histoname,
234  const string& where)
235 {
236  // Try to find histogram named histoname into directory dir, under
237  // path=where (where e.g. = "/Cut##/OK/C#/V#").
238 
239  TH1* rv = 0;
240 
241  bool ok = pathIsInDir(where,dir);
242 
243  if (ok) {
244 
245  // Path is in dir, we can safely (i.e. without getting ROOT error message
246  // on stdout) cd into it.
247  // dir->cd();
248  ok = dir->cd(where.c_str());
249  assert(ok==true);
250  TObject* obj = gDirectory->Get(histoname.c_str());
251  if ( obj ) {
252  rv = dynamic_cast<TH1*>(obj);
253  if ( !rv ) {
254  cerr << "GetHisto : object " << histoname << " is not a TH1" << endl;
255  }
256  }
257  }
258  return rv;
259 }
260 
261 
262 //_____________________________________________________________________________
263 void
264 TDirectoryHelper::splitPath(const string& path,
265  std::vector<std::string>& paths)
266 {
267  // Given a path e.g. /Cut##/OK/C#/V#, will return
268  // a vector of string with Cut#,
269 
270  paths.clear();
271 
272  std::string str = path;
273 
274  if ( str.empty() )
275  {
276  return;
277  }
278 
279  std::vector<size_t> slashes_pos;
280 
281  if ( str[0] != '/' )
282  {
283  str.insert(str.begin(),'/');
284  }
285 
286  if ( str[str.size()-1] != '/' )
287  {
288  str.push_back('/');
289  }
290 
291  for (size_t i = 0 ; i < str.size() ; i++)
292  {
293  if ( str[i] == '/' )
294  {
295  slashes_pos.push_back(i);
296  }
297  }
298 
299  if ( not slashes_pos.empty() )
300  {
301  for (size_t i = 0 ; i < slashes_pos.size()-1 ; i++)
302  {
303  paths.push_back(str.substr(slashes_pos[i]+1,
304  slashes_pos[i+1]-slashes_pos[i]-1));
305  }
306  }
307 }
308 
static void splitPath(const std::string &path, std::vector< std::string > &paths)
static bool mkpath(TDirectory *dir, const std::string &path)
static TH1 * getHisto(TDirectory *dir, const std::string &histoname, const std::string &where)
static bool pathIsInDir(const std::string &path, TDirectory *dir)
static void duplicateDir(TDirectory *dest, TDirectory *source)
static void copyToFile(TDirectory *src, TFile *dest)
static TDirectory * mkdir(TDirectory *topDir, const char *path, std::vector< std::string > *titles=0)