CMS 3D CMS Logo

compareDQMOutput.py
Go to the documentation of this file.
1 #!/bin/env python
2 
3 import os
4 import sys
5 import glob
6 import argparse
7 import subprocess
8 from threading import Thread
9 
10 COMPARISON_RESULTS = []
11 
12 def collect_and_compare_files(base_dir, pr_dir, output_dir, num_procs, pr_number, test_number, release_format):
13  files = get_file_pairs(base_dir, pr_dir)
14 
15  threads = []
16  for _ in range(num_procs):
17  thread = Thread(target=compare, args=(base_dir, pr_dir, output_dir, files, pr_number, test_number, release_format))
18  thread.start()
19  threads.append(thread)
20 
21  [thread.join() for thread in threads]
22 
23  COMPARISON_RESULTS.sort(key=lambda k: float(k['workflow']))
24 
25 def compare(base_dir, pr_dir, output_dir, files, pr_number, test_number, release_format):
26  while files:
27  try:
28  file_name = files.pop()
29  command = ['compareHistograms.py', '-b', os.path.join(base_dir, file_name), \
30  '-p', os.path.join(pr_dir, file_name), '-o', output_dir, '-n', pr_number, '-t', test_number, '-r', release_format]
31  print('Running comparison:')
32  print(' '.join(command))
33 
34  output = subprocess.check_output(command)
35 
36  output_elements = output.split('\n')[1:]
37  base_output_filename = output_elements[0]
38  pr_output_filename = output_elements[1]
39  run_nr = base_output_filename.split('_')[2].lstrip('R').lstrip('0')
40  output_numbers = output_elements[2].split(' ')
41 
42  workflow = os.path.basename(os.path.dirname(os.path.join(base_dir, file_name))).split('_')[0]
43  base_dataset = '/' + '/'.join(base_output_filename.rstrip('.root').split('__')[1:])
44  pr_dataset = '/' + '/'.join(pr_output_filename.rstrip('.root').split('__')[1:])
45 
46  COMPARISON_RESULTS.append({'workflow': workflow, 'base_dataset': base_dataset, 'pr_dataset': pr_dataset, 'run_nr': run_nr,\
47  'changed_elements': int(output_numbers[0]), 'removed_elements': int(output_numbers[1]), 'added_elements': int(output_numbers[2])})
48  except Exception as ex:
49  print('Exception comparing two root files: %s' % ex)
50 
51 def get_file_pairs(base_dir, pr_dir):
52  base_files = glob.glob(os.path.join(base_dir, '*.*_*/DQM_*.root'))
53  pr_files = glob.glob(os.path.join(pr_dir, '*.*_*/DQM_*.root'))
54 
55  # Remove base directories and leave
56  # only parts of paths that are same
57  base_files = map(lambda x: os.path.relpath(x, base_dir), base_files)
58  pr_files = map(lambda x: os.path.relpath(x, pr_dir), pr_files)
59 
60  # Find intersection
61  return [value for value in base_files if value in pr_files]
62 
63 def upload_to_gui(output_dir, num_procs):
64  base_files = glob.glob(os.path.join(output_dir, 'base/*.root'))
65  pr_files = glob.glob(os.path.join(output_dir, 'pr/*.root'))
66 
67  files = base_files + pr_files
68 
69  print('Files to be uploaded:')
70  print(files)
71 
72  for _ in range(min(num_procs, len(files))):
73  thread = Thread(target=upload, args=(files,))
74  thread.start()
75 
76 def upload(files):
77  while files:
78  try:
79  file = files.pop()
80  command = ['visDQMUpload.py', 'https://cmsweb.cern.ch/dqm/dev', file]
81  print('Uploading output:')
82  print(' '.join(command))
83 
84  subprocess.call(command)
85  print('')
86  except Exception as ex:
87  # This might throw when another thread pops the last filename immediately after this one
88  # started the loop. In this case this exception can be safely ignored.
89  print('Exception uploading a file: %s' % ex)
90 
91 def generate_summary_html(output_dir, pr_list, summary_dir):
92  template_file_path = os.path.join(os.getenv('CMSSW_BASE'), 'src', 'DQMServices', 'FileIO', 'scripts', 'dqm-histo-comparison-summary-template.html')
93  if not os.path.isfile(template_file_path):
94  template_file_path = os.path.join(os.getenv('CMSSW_RELEASE_BASE'), 'src', 'DQMServices', 'FileIO', 'scripts', 'dqm-histo-comparison-summary-template.html')
95  template_file = open(template_file_path, 'r')
96  result = template_file.read()
97 
98  result = result.replace('$PR_LIST$', pr_list)
99 
100  table_items = ''
101  total_changes = 0
102 
103  for comp in COMPARISON_RESULTS:
104  total_changes += comp['removed_elements'] + comp['added_elements'] + comp['changed_elements']
105  baseline_count = comp['changed_elements'] + comp['removed_elements']
106  pr_count = comp['changed_elements'] + comp['added_elements']
107  overlay_count = baseline_count
108 
109  # Make urls
110  base_url = 'https://cmsweb.cern.ch/dqm/dev/start?runnr=%s;dataset%%3D%s;sampletype%%3Doffline_relval;workspace%%3DEverything;' % (comp['run_nr'], comp['base_dataset'])
111  pr_url = 'https://cmsweb.cern.ch/dqm/dev/start?runnr=%s;dataset%%3D%s;sampletype%%3Doffline_relval;workspace%%3DEverything;' % (comp['run_nr'], comp['pr_dataset'])
112  overlay_url = 'https://cmsweb.cern.ch/dqm/dev/start?runnr=%s;dataset%%3D%s;referenceshow%%3Dall;referencenorm=False;referenceobj1%%3Dother::%s::;sampletype%%3Doffline_relval;workspace%%3DEverything;' \
113  % (comp['run_nr'], comp['pr_dataset'], comp['base_dataset'])
114 
115  table_items += ' <tr>\n'
116  table_items += ' <td><a href="%s" target="_blank">%s baseline GUI</a><span> (%s)</span></td>\n' % (base_url, comp['workflow'], baseline_count)
117  table_items += ' <td><a href="%s" target="_blank">%s pr GUI</a><span> (%s)</span></td>\n' % (pr_url, comp['workflow'], pr_count)
118  table_items += ' <td><a href="%s" target="_blank">%s overlay GUI</a><span> (%s)</span></td>\n' % (overlay_url, comp['workflow'], overlay_count)
119  table_items += ' <td><span class="removed">-%s</span><span class="added">+%s</span><span class="changed">%s</span></td>\n' \
120  % (comp['removed_elements'], comp['added_elements'], comp['changed_elements'])
121  table_items += ' </tr>\n'
122 
123  result = result.replace('$TOTAL_CHANGES$', str(total_changes))
124  result = result.replace('$NUMBER_OF_WORKFLOWS$', str(len(COMPARISON_RESULTS)))
125  result = result.replace('$PER_WORKFLOW_LIST$', table_items)
126  template_file.close()
127 
128  # Write output
129  result_file_path = os.path.join(summary_dir, 'dqm-histo-comparison-summary.html')
130  if os.path.dirname(result_file_path):
131  if not os.path.exists(os.path.dirname(result_file_path)):
132  os.makedirs(os.path.dirname(result_file_path))
133  summary_file = open(result_file_path, 'w')
134  summary_file.write(result)
135  summary_file.close()
136 
137 if __name__ == '__main__':
138  parser = argparse.ArgumentParser(description="This tool compares DQM monitor elements within DQM files found in base-dir with the ones found in in pr-dir. "
139  "All workflow directories are searched for correctly named DQM root files. "
140  "Comparison is done bin by bin and output is written to a root files containing only the changes.")
141  parser.add_argument('-b', '--base-dir', help='Baseline IB directory', default='basedata/')
142  parser.add_argument('-p', '--pr-dir', help='PR directory', default='prdata/')
143  parser.add_argument('-o', '--output-dir', help='Comparison root files output directory', default='dqmHistoComparisonOutput')
144  parser.add_argument('-j', '--nprocs', help='Number of processes', default=1, type=int)
145  parser.add_argument('-n', '--pr-number', help='This is obsolete and should NOT be used.', required=False)
146  parser.add_argument('-t', '--test-number', help='Unique test number to distinguish different comparisons of the same PR.', default='1')
147  parser.add_argument('-r', '--release-format', help='Release format in this format: CMSSW_10_5_X_2019-02-17-0000', required=True)
148  parser.add_argument('-s', '--summary-dir', help='Directory where summary with all links will be saved', default='')
149  parser.add_argument('-l', '--pr-list', help='A list of PRs participating in the comparison', default='')
150  args = parser.parse_args()
151 
152  # Get the number of the PR which triggered the comparison
153  pr_number = 'Unknown'
154  try:
155  pr_number = args.pr_list.split(' ')[0].split('/')[1].replace('#', '_')
156  except:
157  pass
158 
159  collect_and_compare_files(args.base_dir, args.pr_dir, args.output_dir, args.nprocs, pr_number, args.test_number, args.release_format)
160  upload_to_gui(args.output_dir, args.nprocs)
161  generate_summary_html(args.output_dir, args.pr_list, args.summary_dir)
FastTimerService_cff.range
range
Definition: FastTimerService_cff.py:34
dqmMemoryStats.float
float
Definition: dqmMemoryStats.py:127
compareDQMOutput.upload_to_gui
def upload_to_gui(output_dir, num_procs)
Definition: compareDQMOutput.py:63
min
T min(T a, T b)
Definition: MathUtil.h:58
join
static std::string join(char **cmd)
Definition: RemoteFile.cc:17
cms::dd::split
std::vector< std::string_view > split(std::string_view, const char *)
compareDQMOutput.upload
def upload(files)
Definition: compareDQMOutput.py:76
compareDQMOutput.collect_and_compare_files
def collect_and_compare_files(base_dir, pr_dir, output_dir, num_procs, pr_number, test_number, release_format)
Definition: compareDQMOutput.py:12
str
#define str(s)
Definition: TestProcessor.cc:48
compareDQMOutput.compare
def compare(base_dir, pr_dir, output_dir, files, pr_number, test_number, release_format)
Definition: compareDQMOutput.py:25
createfilelist.int
int
Definition: createfilelist.py:10
edm::print
S & print(S &os, JobReport::InputFile const &f)
Definition: JobReport.cc:66
compareDQMOutput.generate_summary_html
def generate_summary_html(output_dir, pr_list, summary_dir)
Definition: compareDQMOutput.py:91
genParticles_cff.map
map
Definition: genParticles_cff.py:11
compareDQMOutput.get_file_pairs
def get_file_pairs(base_dir, pr_dir)
Definition: compareDQMOutput.py:51
python.rootplot.root2matplotlib.replace
def replace(string, replacements)
Definition: root2matplotlib.py:444