13from DQConfMakerBase.DQElements
import DQRegion, DQReference, DQAlgorithm, DQAlgorithmParameter
38def recurse(rdir, dqregion, ignorepath, modelrefs=[], displaystring='Draw=PE', displaystring2D='Draw=COLZ', regex=None, startpath=None, hists=None, manglefunc=None):
39 if manglefunc
is None:
40 manglefunc =
lambda a, b: a
41 for key
in rdir.GetListOfKeys():
42 cl = key.GetClassName(); rcl = ROOT.TClass.GetClass(cl)
43 if ' ' in key.GetName():
44 print(
'WARNING: cannot have spaces in histogram names for han config; not including %s %s' % (cl, key.GetName()))
46 if rcl.InheritsFrom(
'TH1')
or rcl.InheritsFrom(
'TGraph')
or rcl.InheritsFrom(
'TEfficiency'):
47 if '/' in key.GetName():
48 print(
'WARNING: cannot have slashes in histogram names, encountered in directory %s, histogram %s' % (rdir.GetPath(), key.GetName()))
50 if key.GetName() ==
'summary':
51 print(
'WARNING: cannot have histogram named summary, encountered in %s' % rdir.GetPath())
53 fpath = rdir.GetPath().
replace(ignorepath,
'')
54 name = (fpath +
'/' + key.GetName()).lstrip(
'/')
61 if not match:
continue
63 if not regex.match(name):
continue
64 dqpargs = {
'id' : (
'' if fpath
else 'top_level/') + name,
65 'inputdatasource': (startpath +
'/' if startpath
else '') + name,
69 for mref
in modelrefs:
70 newref = DQReference(manglefunc(mref.getReference().
replace(
'same_name', (startpath +
'/' if startpath
else '') + name), mref.id))
71 newref.addAnnotation(
'info', mref.id)
72 lnewrefs.append(newref)
73 dqpargs.update({
'algorithm': repeatalgorithm,
74 'algorithmparameters': algorithmparameters,
76 'references': lnewrefs
79 dqpargs[
'algorithm'] = norefalgorithm
80 dqpar = dqregion.newDQParameter( **dqpargs)
82 if not options.normalize: drawstrs.append(
'NoNorm')
83 if options.logy
and (cl.startswith(
'TH1')
or cl==
'TProfile'): drawstrs.append(
'LogY')
84 if options.logy
and (cl.startswith(
'TH2')
or cl==
'TProfile2D'): drawstrs.append(
'LogZ')
85 if cl.startswith(
'TH1'): drawstrs.append(displaystring)
86 if cl ==
'TProfile': drawstrs.append(displaystring)
87 if cl.startswith(
'TH2')
or cl==
'TProfile2D': drawstrs.append(displaystring2D)
88 if options.scaleref != 1: drawstrs.append(
'ScaleRef=%f' % options.scaleref)
89 if options.ratio: drawstrs.append(
'RatioPad')
91 if options.ratio2D: drawstrs.append(
'Ref2DRatio')
92 if options.ratiorange
is not None:
93 drawstrs.append(
'delta(%f)' % options.ratiorange)
95 drawstrs.append(
'DataName=%s' % options.title)
96 dqpar.addAnnotation(
'display',
','.join(drawstrs))
98 elif rcl.InheritsFrom(
'TDirectory'):
99 newregion = dqregion.newDQRegion( key.GetName(), algorithm=worst )
100 recurse(key.ReadObj(), newregion, ignorepath, modelrefs, displaystring, displaystring2D, regex, startpath, hists, manglefunc)
104 returns True if we should kill this node
105 False if we should not
107 params = dqregion.getDQParameters()
110 subregions = dqregion.getSubRegions()
111 if subregions
is None:
114 subregions = subregions[:]
116 for sr
in subregions:
120 dqregion.delRelation(
'DQRegions', sr)
121 subregions = dqregion.getSubRegions()
122 if subregions
is None:
124 if len(subregions) + len(params) == 0:
139def process(infname, confname, options, refs=None):
141 f = ROOT.TFile.Open(infname,
'READ')
143 print(
'ERROR: cannot open %s' % infname)
146 top_level = DQRegion(id=
'topRegion',algorithm=worst)
147 print(
'Building tree...')
148 refpairs = refs.split(
',')
if refs
else []
150 refdict = dict(_.split(
':', 1)
for _
in refpairs)
151 except Exception
as e:
154 dqrs = [DQReference(reference=
'%s:same_name' % v, id=k)
155 for k, v
in list(refdict.items())]
156 displaystring = options.drawopt
157 if options.refdrawopt:
158 displaystring +=
',' + (
','.join(
'DrawRef=%s' % _
for _
in options.refdrawopt.split(
',')))
159 displaystring2D = options.drawopt2D
160 if options.drawrefopt2D:
161 displaystring2D +=
',' + (
','.join(
'DrawRef2D=%s' % _
for _
in options.drawrefopt2D.split(
',')))
163 if options.startpath:
164 topindir = f.Get(options.startpath)
166 raise ValueError(
"Path %s doesn't exist in input file" % options.startpath)
167 topindirname = f.GetPath() + options.startpath.strip(
'/')
168 startpath = options.startpath.strip(
'/')
171 topindirname = f.GetPath()
175 refstartpaths = options.refstartpath.split(
',')
if options.refstartpath
else []
177 refstartpathdict = dict(_.split(
':')
for _
in refstartpaths)
178 for k, v
in refstartpathdict.items():
179 refstartpathdict[k] = v.strip(
'/')
180 except Exception
as e:
182 def refpath_manglefunc(path, id):
184 pfx = refstartpathdict[id]
186 return path.replace(
':' + (startpath +
'/' if startpath
else ''),
':' + (pfx +
'/' if pfx
else ''), 1)
191 if options.histlistfile:
192 hists = [re.compile(line.rstrip(
'\n'))
for line
in open(options.histlistfile)]
193 if options.pathregex:
print(
"histlistfile given, pathregex is ignored")
194 if options.refmangle:
196 sys.path.append(os.getcwd())
198 manglefunc = importlib.import_module(options.refmangle).mangle
200 manglefunc = refpath_manglefunc
201 recurse(topindir, top_level, topindirname, dqrs, displaystring, displaystring2D,
202 re.compile(options.pathregex), startpath, hists, manglefunc=manglefunc)
203 print(
'Pruning dead branches...')
207 sublevel = top_level.getSubRegions()[:]
209 top_level.delRelation(
'DQRegions', x)
211 print(
'Writing output')
212 writeHanConfiguration( filename = confname , roots = sublevel)
216 import shutil, os, sys, contextlib
218 han_is_found = (ROOT.gSystem.Load(
'libDataQualityInterfaces') != 1)
220 print(
'ERROR: unable to load offline DQMF; unable to proceed')
222 bname = os.path.basename(fname)
229 prebuilt_hcfg =
False
231 @contextlib.contextmanager
234 td = tempfile.mkdtemp()
238 with tmpdir()
as hantmpdir:
240 print(
'====> Processing file %s' % (fname))
241 print(
'====> Generating han configuration file')
242 hantmpinput = os.path.join(hantmpdir, bname)
243 shutil.copyfile(fname, hantmpinput)
244 haninput = hantmpinput
245 hanconfig = os.path.join(hantmpdir,
'han.config')
246 rv =
process(hantmpinput, hanconfig, options, options.reffile)
251 print(
'No histograms to display; exiting with code 0')
254 print(
'====> Compiling han configuration')
255 hanhcfg = os.path.join(hantmpdir,
'han.hcfg')
256 ROOT.dqi.HanConfig().AssembleAndSave( hanconfig, hanhcfg )
257 print(
'====> Executing han')
259 memlimit = resource.getrlimit(resource.RLIMIT_AS)
260 resource.setrlimit(resource.RLIMIT_AS, (memlimit[1], memlimit[1]))
261 hanoutput = haninput.rpartition(
'.')[0] +
'_han.root'
263 rv = ROOT.dqi.HanApp().Analyze( hanhcfg, haninput, hanoutput )
265 raise Exception(
'failure in han')
267 rf = ROOT.TFile.Open(hanoutput,
'UPDATE')
268 HanMetadata.addMetadata(rf,
'AMI', {
'AMI Tag': options.amitag})
270 if not options.hanonly:
271 print(
'====> Dumping web display output')
272 from DataQualityUtils
import handimod
273 handimod.handiWithComparisons( options.title,
277 'https://atlasdqm.web.cern.ch/atlasdqm/js/',
278 3
if options.jsRoot
else 1)
279 if options.hanoutput:
280 from pathlib
import Path
281 print(
'====> Copying han output to', options.hanoutput)
282 target = Path(options.hanoutput)
284 target.parent.mkdir(parents=
True, exist_ok=
True)
285 except Exception
as e:
286 print(
'Unable to create %s for some reason: %s' % (target.parent, e))
287 raise Exception(
'Error during execute')
from e
288 shutil.copy2(hanoutput, options.hanoutput)
289 print(
'====> Cleaning up')
291 except Exception
as e:
294 traceback.print_exc()
295 if 'canonical format' not in str(e):
299 if not prebuilt_hcfg:
300 os.unlink(hantmpinput)
recurse(rdir, dqregion, ignorepath, modelrefs=[], displaystring='Draw=PE', displaystring2D='Draw=COLZ', regex=None, startpath=None, hists=None, manglefunc=None)