forked from OpenModelica/OpenModelicaLibraryTesting
-
Notifications
You must be signed in to change notification settings - Fork 0
/
all-reports.py
executable file
·268 lines (244 loc) · 13 KB
/
all-reports.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
import urllib.request
import codecs
import sys, argparse, subprocess, os, time
import simplejson as json
import shared
import re
from omcommon import friendlyStr
parser = argparse.ArgumentParser(description='OpenModelica model testing report generation tool')
parser.add_argument('branches', nargs='*')
parser.add_argument('--baseurl', default="http://libraries.openmodelica.org/branches")
parser.add_argument('--historyurl', default="http://libraries.openmodelica.org/branches/history")
parser.add_argument('--githuburl', default="https://github.com/OpenModelica/OpenModelica/commit")
parser.add_argument('--omcgitdir', default="../OpenModelica/OpenModelica")
parser.add_argument('--email', default=False, action='store_true')
args = parser.parse_args()
os.environ['TZ'] = 'Europe/Stockholm'
time.tzset()
branches = [branch.split("/")[-1] for branch in args.branches]
baseurl = args.baseurl
historyurl = args.historyurl
githuburl = args.githuburl
omcgitdir = args.omcgitdir
doemail = args.email
if not os.path.exists(omcgitdir):
raise Exception("Could not find OpenModelica.git directory, set it with --omcgitdir. Tried: %s" % omcgitdir)
dates = {}
dates_str = {}
fields = ["exectime", "frontend", "backend", "simcode", "templates", "compile", "simulate", "verify"]
entryhead = "<tr><th>Branch</th><th>Total</th><th>Frontend</th><th>Backend</th><th>SimCode</th><th>Templates</th><th>Compilation</th><th>Simulation</th><th>Verification</th>\n"
timeMinPhase = 4 # Need to have completed code generation to report performance regressions
timeRel = 1.7 # Minimum 1.7x time is registered as a performance regression
timeAbs = 10 # Ignore performance regressions for times <10s...
libs = {}
import cgi, sqlite3, time, datetime
from omcommon import friendlyStr, multiple_replace
conn = sqlite3.connect('sqlite3.db')
cursor = conn.cursor()
def dateStr(dint):
return str(datetime.datetime.fromtimestamp(dint).strftime('%Y-%m-%d %H:%M:%S'))
def getTagOrVersion(v):
v = v.replace("OpenModelica ","").replace("OMCompiler ","")
m = re.search("[+]g([0-9a-f]{7}[0-9a-f]*)$", v)
if m:
return m.group(1)
return v
def libraryLink(branch, libname):
return '<a href="%s/%s/%s/%s.html">%s</a>' % (baseurl,branch,libname,libname,libname)
def modelLink(libname, modelname, extension, text):
return '<a href="%s/%s/%s/files/%s_%s.%s">%s</a>' % (baseurl,branch,libname,libname,modelname,extension,text)
emails_to_send = {}
for branch in branches:
try:
cursor.execute("SELECT name FROM [sqlite_master] WHERE type='table' AND name=?", (branch,))
v = cursor.fetchone()[0]
except:
raise Exception("No such table '%s'; specify it using --branch=XXX" % branch)
cursor.execute('''CREATE INDEX IF NOT EXISTS [idx_%s_date] ON [%s](date)''' % (branch,branch))
cursor.execute("SELECT date,omcversion FROM [omcversion] WHERE branch LIKE ? COLLATE NOCASE ORDER BY date ASC", (branch,))
entries = cursor.fetchall()
n = len(entries)
urlToOpen = "%s/%s/00_history.html" % (historyurl, branch)
try:
urlContents = urllib.request.urlopen(urlToOpen).read().decode('utf-8')
except:
print(urlToOpen + " failed to open")
raise
generated = False
for i in range(1,n):
d1 = entries[i-1][0]
d2 = entries[i][0]
fname = "history/%s/%s..%s.html" % (branch,dateStr(d1),dateStr(d2))
if fname.replace(" ","%20") in urlContents or fname in urlContents:
continue
print("Generate %s" % fname)
generated = True
v1 = getTagOrVersion(entries[i-1][1])
v2 = getTagOrVersion(entries[i][1])
thirdPartyChanged = ""
with open("history.html.tpl") as fin:
tpl = fin.read()
emails_current = set(["[email protected]"])
if v1 != v2:
try:
gitlog = subprocess.check_output(["git", "log", '--pretty=<tr><td><a href="%s/%%h">%%h</a></td><td>%%an</td><td>%%s</td></tr>' % (githuburl), "%s..%s" % (v1,v2)], cwd=omcgitdir).decode("utf-8")
print("Do git ls-tree for %s %s" % (v1,v2))
try:
t1 = subprocess.check_output(["git", "ls-tree", v1, "OMCompiler/3rdParty"], cwd=omcgitdir).decode("utf-8").strip().split(" ")[2].split("\t")[0]
t2 = subprocess.check_output(["git", "ls-tree", v2, "OMCompiler/3rdParty"], cwd=omcgitdir).decode("utf-8").strip().split(" ")[2].split("\t")[0]
if t1 != t2:
try:
tv2 = len(subprocess.check_output(["git", "rev-list", "%s..%s" % (t2,t1)], cwd=omcgitdir+"/OMCompiler/3rdParty").decode("utf-8").strip().split("\n"))
except subprocess.CalledProcessError as e:
tv2 = 0
if tv2 > 0:
thirdPartyChanged = '<h3>3rdParty changes</h3>Note that the 3rdParty libraries <b>REVERTED TO AN OLD COMMIT</b>: <a href="%s">%s..%s</a>' % (githuburl.replace("OMCompiler/commit", "OMCompiler-3rdParty/compare/%s...%s" % (t2,t1)), t1[:12], t2[:12])
else:
thirdPartyChanged = '<h3>3rdParty changes</h3>Note that the 3rdParty libraries changed: <a href="%s">%s..%s</a>' % (githuburl.replace("OMCompiler/commit", "OMCompiler-3rdParty/compare/%s...%s" % (t1,t2)), t1[:12], t2[:12])
except:
pass
for email in [email.strip() for email in subprocess.check_output(["git", "log", '--pretty=%ae', "%s..%s" % (v1,v2)], cwd=omcgitdir).decode("utf-8").split("\n")]:
if "@" not in email:
continue
emails_current.add(email)
except subprocess.CalledProcessError as e:
print(str(e))
gitlog = "<tr><td>%s..%s</td></tr>" % (v1,v2)
else:
gitlog = ""
tpl = tpl.replace("#OMCGITLOG#",gitlog).replace("#NUMCOMMITS#",str(gitlog.count("<tr>"))).replace("#3rdParty#",thirdPartyChanged)
libnames = [libname for (libname,) in cursor.execute("""SELECT libname FROM [%s] WHERE date=? GROUP BY libname""" % branch, (d2,))]
startdates = {}
# Get previous date of each library run and group them together for fast queries later
for libname in libnames:
ds = cursor.execute("""SELECT date FROM [%s] WHERE date<? AND libname=? ORDER BY date DESC LIMIT 1""" % branch, (d2,libname)).fetchall()
if len(ds)==0:
continue
((d1lib,),) = ds
if d1lib not in startdates:
startdates[d1lib] = []
startdates[d1lib] += [libname]
regressions = []
for d1lib in startdates.keys():
# Order by date so we can select and know which is the older and which is the newer value... for finalphase, and the execution times
# Note: GROUP_CONCAT returns both values as a string... So you need to split it later
query = """SELECT model,libname,GROUP_CONCAT(finalphase),GROUP_CONCAT(frontend),GROUP_CONCAT(backend),GROUP_CONCAT(simcode),GROUP_CONCAT(templates),GROUP_CONCAT(compile),GROUP_CONCAT(simulate) FROM
(SELECT model,libname,finalphase,frontend,backend,simcode,templates,compile,simulate FROM [%s] WHERE date IN (?,?) AND libname IN (%s) ORDER BY date)
GROUP BY model,libname HAVING
(MIN(finalphase) <> MAX(finalphase)) OR
((MIN(finalphase) >= ?) AND
(MAX(frontend) > ?*MIN(frontend) AND MAX(frontend) > ?) OR
(MAX(backend) > ?*MIN(backend) AND MAX(backend) > ?) OR
(MAX(simcode) > ?*MIN(simcode) AND MAX(simcode) > ?) OR
(MAX(templates) > ?*MIN(templates) AND MAX(templates) > ?) OR
(MAX(compile) > ?*MIN(compile) AND MAX(compile) > ?) OR
(MAX(simulate) > ?*MIN(simulate) AND MAX(simulate) > ?)
)
""" % (branch,",".join(["'%s'" % libname for libname in startdates[d1lib]]))
cursor.execute(query, (d1lib,d2,timeMinPhase,timeRel,timeAbs,timeRel,timeAbs,timeRel,timeAbs,timeRel,timeAbs,timeRel,2*timeAbs,timeRel,timeAbs))
regressions += cursor.fetchall()
regressions = sorted(regressions, key = lambda x: (x[1],x[0]))
libs = set()
numImproved = 0
numRegression = 0
numPerformanceImproved = 0
numPerformanceRegression = 0
regstrs = []
for (model,libname,group,frontend,backend,simcode,templates,compile,simulate) in regressions:
libs.add(libname)
(phase1,phase2) = [int(i) for i in group.split(",")]
color = None
if phase2 > phase1:
color = "better"
numImproved += 1
if phase2 < phase1:
color = "warning"
numRegression += 1
if color is not None:
msg = "%s → %s" % (shared.finalphaseName(phase1),shared.finalphaseName(phase2))
elif min(phase1,phase2) >= timeMinPhase:
msgs = []
for (phase,times) in [(1,frontend),(2,backend),(3,simcode),(4,templates),(5,compile),(6,simulate)]:
(t1,t2) = [float(d) for d in times.split(",")]
if t2 > timeRel*t1 and t2 > timeAbs:
color = "warningPerformance"
msgs.append("%s performance %s → %s" % (shared.finalphaseName(phase),friendlyStr(t1),friendlyStr(t2)))
elif t1 > timeRel*t2 and t1 > timeAbs:
if color is None:
color = "betterPerformance"
msgs.append("%s performance %s → %s" % (shared.finalphaseName(phase),friendlyStr(t1),friendlyStr(t2)))
if color is None:
raise Exception("Unknown regression/improvement...")
if color == "betterPerformance":
numPerformanceImproved += 1
else:
numPerformanceRegression += 1
msg = " ".join(msgs)
else:
msg = "" # Happens if we try to generate a report without previous results
regstrs.append('<tr><td>%s</td><td>%s %s</td><td class="%s">%s</td></tr>' % (libraryLink(branch, libname),modelLink(libname, model, "err", model),modelLink(libname, model, "sim", "(sim)"),color,msg))
tpl = tpl.replace("#NUMIMPROVE#",str(numImproved)).replace("#NUMREGRESSION#",str(numRegression)).replace("#NUMPERFIMPROVE#",str(numPerformanceImproved)).replace("#NUMPERFREGRESSION#",str(numPerformanceRegression)).replace("#MODELCHANGES#", "\n".join(regstrs))
tpl = tpl.replace("#BRANCH#",branch).replace("#DATE1#",dateStr(d1)).replace("#DATE2#",dateStr(d2))
libstrs = []
for libname in sorted(list(libs)):
cursor.execute("SELECT libversion,confighash FROM [libversion] WHERE branch LIKE ? COLLATE NOCASE AND date<=? AND libname=? ORDER BY date DESC LIMIT 1", (branch,d1,libname))
(lv1,lh1) = cursor.fetchone()
lv1 = lv1.strip()
cursor.execute("SELECT libversion,confighash FROM [libversion] WHERE branch LIKE ? COLLATE NOCASE AND date<=? AND libname=? ORDER BY date DESC LIMIT 1", (branch,d2,libname))
(lv2,lh2) = cursor.fetchone()
lv2 = lv2.strip()
if lv1 != lv2:
libstrs.append("<tr><td>%s</td><td>From version %s to %s</td></tr>" % (libraryLink(branch, libname),lv1,lv2))
elif lh1 != lh2:
libstrs.append("<tr><td>%s</td><td>Configuration hash (OMC settings or the testing script changed)</td></tr>" % libraryLink(branch, libname))
tpl = tpl.replace("#LIBCHANGES#","\n".join(libstrs)).replace("#NUMLIBS#",str(len(libstrs)))
email_summary_html = '<p><a href="%s/%s/%s">%s %s</a> %d improved, %d regressions; performance %d improved, %d regressions</p>' % (historyurl, branch, os.path.basename(fname).replace(" ","%20"), branch, os.path.basename(fname),numImproved,numRegression,numPerformanceImproved,numPerformanceRegression)
email_summary_plain = '%s/%s/%s: %d improved, %d regressions; performance %d improved, %d regressions</p>' % (historyurl, branch, os.path.basename(fname).replace(" ","%20"), numImproved, numRegression, numPerformanceImproved, numPerformanceRegression)
if sum([numImproved,numRegression,numPerformanceImproved,numPerformanceRegression])>0:
for email in emails_current:
if email not in emails_to_send:
emails_to_send[email] = {"plain":[],"html":[]}
emails_to_send[email]["plain"].append(email_summary_plain)
emails_to_send[email]["html"].append(email_summary_html)
if not os.path.exists(os.path.dirname(fname)):
os.makedirs(os.path.dirname(fname))
with codecs.open(fname, "w", encoding="utf-8") as fout:
fout.write(tpl)
if not os.path.exists(os.path.dirname("history/%s" % branch)):
os.makedirs("history/%s" % branch)
urlContents = urlContents + email_summary_html + "\n"
if generated:
with open("history/%s/00_history.html" % branch, "w") as fout:
fout.write(urlContents)
if not doemail:
# We are done
sys.exit(0)
# OK; send the emails :D
import smtplib
from email.message import EmailMessage
from email.headerregistry import Address
from email.utils import make_msgid
for email in sorted(emails_to_send.keys()):
msg = EmailMessage()
msg['Subject'] = 'OpenModelica Library Testing Regressions'
msg['From'] = Address("OM Hudson", "openmodelicabuilds", "ida.liu.se")
msg['To'] = email
msg.set_content("""\
The following reports contain regressions your account was involved with:
""" + "\n".join(reversed(emails_to_send[email]["plain"])))
msg.add_alternative("""\
<html>
<head></head>
<body>
<p>The following reports contain regressions your account was involved with:</p>
%s
</body>
</html>
""" % "\n".join(reversed(emails_to_send[email]["html"])), subtype='html')
with smtplib.SMTP('smtp.office365.com') as s:
s.starttls()
s.ehlo()
s.login(os.environ["IDA_EMAIL_USR"],os.environ["IDA_EMAIL_PSW"])
s.send_message(msg)