OOPS
compare.py
Go to the documentation of this file.
1 #!/usr/bin/env python3
2 
3 # (C) Copyright 2019 UCAR
4 #
5 # This software is licensed under the terms of the Apache Licence Version 2.0
6 # which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
7 
8 """
9 Comparison of two text files containing Test log
10 from OOPS runs.
11 
12 Comparison of floats uses a maximum relative difference:
13 (float1 - float2)/float1 <= tolerance.
14 
15 Comparison of integers uses a maximum diffrence:
16 integer1 - integer2 <= difference
17 
18 Comparison of strings is exact
19 
20 Failure results in a return code of 1.
21 
22 Call as:
23 compare.py run_file ref_file float_tolerance integer_difference
24 """
25 
26 import re
27 import sys
28 
29 # Method that searches for int and floats in two lines
30 # and compares them one by one
31 def line_diff(line1,line2,lnum,ftol,idif):
32 
33  #Split line by whitespace or '='
34  sline1 = re.split('\s+|=', line1)
35  sline2 = re.split('\s+|=', line2)
36 
37  lineerror = 0
38 
39  for n in range(len(sline1)):
40  if sline1[n] != '':
41 
42  nnm1 = rennm.findall(sline1[n])
43  flt1 = reflt.findall(sline1[n])
44  int1 = reint.findall(sline1[n])
45  dat1 = redat.findall(sline1[n])
46 
47  nnm2 = rennm.findall(sline2[n])
48  flt2 = reflt.findall(sline2[n])
49  int2 = reint.findall(sline2[n])
50  dat2 = redat.findall(sline2[n])
51 
52  found = 0
53 
54  #Compare if non-numeric string
55  if nnm1 and nnm2:
56  found=found+1
57  if (nnm1 != nnm2):
58  lineerror=lineerror+1
59  print("Non numeric string mismatch at line "+str(lnum)+": "\
60  +nnm1[0]+" not equal to "+nnm2[0])
61 
62  #Compare if string/float
63  if flt1 and flt2:
64  found=found+1
65  flt1a = reflte.findall(flt1[0])
66  flt2a = reflte.findall(flt2[0])
67  rdiff = abs(float(flt1a[0])-float(flt2a[0]))/(abs(float(flt1a[0]))+1.0e-6)
68  if (not rdiff <= ftol):
69  lineerror=lineerror+1
70  print("Float mismatch at line "+str(lnum)+": "+\
71  flt1a[0]," not equal to ",flt2a[0]," with max relative difference ", ftol,\
72  " Actual relative difference = ",rdiff)
73 
74  #Compare if integer
75  if int1 and int2:
76  found=found+1
77  int1a = reinte.findall(int1[0])
78  int2a = reinte.findall(int2[0])
79  adiff = abs(int(int1a[0])-int(int2a[0]))
80  if (not adiff <= idif):
81  lineerror=lineerror+1
82  print("Integer mismatch at line "+str(lnum)+": ",\
83  int1a[0]+" not equal to "+int2a[0]," with max difference ", idif,\
84  ". Actual difference = ",adiff)
85 
86  #Compare if date
87  if dat1 and dat2:
88  found=found+1
89  if (dat1 != dat2):
90  lineerror=lineerror+1
91  print("Data string mismatch at line "+str(lnum)+": "+\
92  dat1[0]+" not equal to "+dat2[0])
93 
94 
95  #Compare if one is float and other is integer
96  if flt1 and int2:
97  found=found+1
98  flt1a = reflte.findall(flt1[0])
99  int2a = reinte.findall(int2[0])
100  rdiff = abs(float(flt1a[0])-float(int2a[0]))/(abs(float(flt1a[0]))+1.0e-6)
101  if (not rdiff <= ftol):
102  lineerror=lineerror+1
103  print("Float mismatch at line "+str(lnum)+": "+\
104  flt1a[0]," not equal to ",int2a[0]," with max relative difference ", ftol,\
105  " Actual relative difference = ",rdiff)
106 
107  if int1 and flt2:
108  found=found+1
109  int1a = reinte.findall(int1[0])
110  flt2a = reflte.findall(flt2[0])
111  rdiff = abs(float(int1a[0])-float(flt2a[0]))/(abs(float(int1a[0]))+1.0e-6)
112  if (not rdiff <= ftol):
113  lineerror=lineerror+1
114  print("Float mismatch at line "+str(lnum)+": "+\
115  int1a[0]," not equal to ",flt2a[0]," with max relative difference ", ftol,\
116  " Actual relative difference = ",rdiff)
117 
118  #Exit with error if check has failed
119  if found == 0:
120  if sline1[n] != sline2[n]:
121  print("In looping through line elements did not match either non numeric, float, date or integer. Error at line "\
122  +str(lnum),". Trying to compare \'"+sline1[n]+"\' and \'"+sline2[n]+"\'")
123  exit(1)
124 
125  if found > 1:
126  if sline1[n] != sline2[n]:
127  print("In looping through line elements matched multipe of non numeric, float, date and integer. Error at line "\
128  +str(lnum)+". Trying to compare \'"+sline1[n]+"\' and \'"+sline2[n]+"\'")
129  exit(1)
130 
131  return lineerror
132 
133 
134 # Get file names and tolerance from arguments
135 # -------------------------------------------
136 file_run = open(sys.argv[1], "r")
137 file_ref = open(sys.argv[2], "r")
138 ftol = float(sys.argv[3])
139 idif = int(sys.argv[4])
140 
141 # Write grep results to a new file
142 file_runref = open(str(sys.argv[1])+".ref","w")
143 
144 # Read reference file
145 lines_ref = file_ref.readlines()
146 
147 # Potential regular expressions found in OOPS Test output
148 # -------------------------------------------------------
149 # Regex: ABC[12][ABC] (combination of string and int, e.g. AMSUA-NOAA19
150 rennm = re.compile('(^\D+[\d]*[\D]*$)')
151 
152 # Regex: #[ABC][-]12.34[e[+-]12][,] (combination of string and flaot, e.g. MAX=123.123e-07,
153 reflt = re.compile('(^[\D]*?[-]?\d+\.\d+(?:[e][+-]?[\d]+)?[\,]?$)') #[ABC][-]12.34[e[+-]12][,]
154 
155 # Regex: #[-]12345[,] (combination of integer and comma, e.g. 123,
156 reint = re.compile('(^[-]?\d+[\,]?$)')
157 
158 # Regex: #YYYY-MM-DDTHH:MN:SSZ[:] (date with potential semi-colon)
159 redat = re.compile('(^\d{4}[-]\d{2}[-]\d{2}[T]\d{2}[:]\d{2}[:]\d{2}[Z][\:]?$)')
160 
161 # Sub extractions to convert combination of string and number to just number
162 reflte = re.compile('([+-]?\d+\.\d+(?:[e][+-]?[\d]+)?)') #Float extraction (MAX=123.123e-07, -> 123.123e-07)
163 reinte = re.compile('([+-]?\d+)') #Integer extraction (123, -> 123)
164 
165 
166 # Loop through the run file and search on test string
167 refline = 0
168 error = 0
169 testfound = False
170 for line_run in file_run:
171  if re.search('Test : ', line_run):
172 
173  #Strip line
174  line_strip = line_run[line_run.find('Test : '):]
175 
176  #Check strings and integers
177  lineerror = line_diff(line_strip,lines_ref[refline],refline+1,ftol,idif)
178  error=error+lineerror
179 
180  #Write ref file in case update needed
181  file_runref.write(line_strip)
182 
183  #Tick the reference line
184  refline = refline + 1
185 
186  testfound = True
187 
188 if (len(lines_ref) != refline):
189  print("Test failed. "+str(refline-1)+" matches in run, "+str(len(lines_ref))+" ref")
190  sys.exit(1) #Return failure
191 
192 # Close the new reference file
193 file_runref.close()
194 
195 # Return status
196 if error > 0:
197  sys.exit(1) #Return failure
198 if not testfound:
199  print("Did not find any instances of \'Test : \' in run file")
200  sys.exit(1) #Return failure
201 
202 # Otherwise return success
203 sys.exit(0)
def line_diff(line1, line2, lnum, ftol, idif)
Definition: compare.py:31