forked from OpenModelica/OpenModelicaLibraryTesting
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtest.py
executable file
·861 lines (768 loc) · 35.5 KB
/
test.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
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# TODO: When libraries hash changes, run with the old OMC against the new libs
# Then run with the new OMC against the new libs
import sys
if (sys.version_info < (3, 0)):
raise Exception("Python2 is no longer supported")
import html, shutil, os, re, glob, time, argparse, sqlite3, datetime, math, platform
from joblib import Parallel, delayed
import simplejson as json
import psutil, subprocess, threading, hashlib
from subprocess import call
from monotonic import monotonic
from omcommon import friendlyStr, multiple_replace
from natsort import natsorted
from shared import readConfig, getReferenceFileName, simulationAcceptsFlag
import shared
import signal
def rmtree(f):
try:
shutil.rmtree(f)
except UnicodeDecodeError:
# Yes, we can get UnicodeDecodeError because shutil.rmtree is poorly implemented
subprocess.check_call(["rm", "-rf", f], stderr=subprocess.STDOUT)
def print_linenum(signum, frame):
print("Currently at line", frame.f_lineno)
signal.signal(signal.SIGUSR1, print_linenum)
def runCommand(cmd, prefix, timeout):
process = [None]
def target():
with open(os.devnull, 'w') as FNULL:
process[0] = subprocess.Popen(cmd, shell=True, stdin=FNULL, stdout=FNULL, stderr=FNULL, preexec_fn=os.setpgrp)
while process[0].poll() is None:
process[0].communicate(1)
process[0].wait(1)
thread = threading.Thread(target=target)
thread.start()
thread.join(timeout)
gotTimeout = False
if thread.is_alive():
gotTimeout = True
os.kill(-process[0].pid, signal.SIGTERM)
thread.join(min(10, timeout))
if thread.is_alive():
os.kill(-process[0].pid, signal.SIGKILL)
thread.join()
if clean:
try:
lines = open("%s.tmpfiles" % prefix).readlines()
except:
lines = []
for suffix in [".so",".mos","",".o",".h",".c",".cpp","_info.json",".xml",".tmpfiles",".pipe",".tmpfiles",".libs",".log"]:
for f in glob.glob(prefix+suffix):
lines.append(f)
for f in glob.glob("OM"+prefix+suffix):
lines.append(f)
for line in lines:
f = line.strip()
if os.path.isdir(f):
rmtree(f)
elif os.path.exists(f):
os.unlink(f)
try:
rmtree(prefix)
except OSError:
pass
return 1 if gotTimeout else process[0].returncode
try:
subprocess.check_output(["./testmodel.py", "--help"], stderr=subprocess.STDOUT)
except subprocess.CalledProcessError as e:
print("Sanity check failed (./testmodel.py --help):\n" + e.output.decode())
sys.exit(1)
parser = argparse.ArgumentParser(description='OpenModelica library testing tool')
parser.add_argument('configs', nargs='*')
parser.add_argument('--branch', default='master')
parser.add_argument('--fmi', default=False)
parser.add_argument('--output', default='')
parser.add_argument('--docker', default='')
parser.add_argument('--libraries', default=os.path.expanduser('~/.openmodelica/libraries/'))
parser.add_argument('--extraflags', default='')
parser.add_argument('--extrasimflags', default='')
parser.add_argument('--ompython_omhome', default='')
parser.add_argument('--noclean', action="store_true", default=False)
parser.add_argument('--fmisimulator', default='')
parser.add_argument('--ulimitvmem', help="Virtual memory limit (in kB)", type=int, default=8*1024*1024)
parser.add_argument('--default', action='append', help="Add a default value for some configuration key, such as --default=ulimitExe=60. The equals sign is mandatory.", default=[])
parser.add_argument('-j', '--jobs', default=0)
args = parser.parse_args()
configs = args.configs
branch = args.branch
result_location = args.output
n_jobs = int(args.jobs)
clean = not args.noclean
extraflags = args.extraflags
extrasimflags = args.extrasimflags
ompython_omhome = args.ompython_omhome
fmisimulator = args.fmisimulator or None
allTestsFmi = args.fmi
ulimitMemory = args.ulimitvmem
docker = args.docker
librariespath = args.libraries
overrideDefaults = [arg.split("=", 1) for arg in args.default]
# If -j=0 is specified (or -j is not specified, defaults to 0) then use all available physical CPUS.
if n_jobs == 0:
n_jobs = psutil.cpu_count(logical=False)
# If we are running one test at a time assume that omc is allowed to use multiple
# threads for each individual test.
if n_jobs == 1:
single_thread="" # Alternative: single_thread="-n=0"
else:
single_thread="-n=1"
print("branch: %s, n_jobs: %d" % (branch, n_jobs))
if clean:
print("Removing temporary files, etc to the best ability of the script")
if not librariespath:
print("Error: --libraries is a mandatory argument")
sys.exit(1)
if docker:
print("###")
print("###")
print("Warning: Using docker to run this will fail. It is not fully working and needs to be reworked to use docker exec instead of docker run")
print("###")
print("###")
dir_path = os.path.dirname(os.path.realpath(__file__))
subprocess.check_output(["docker", "pull", docker], stderr=subprocess.STDOUT).strip()
dockerExtraArgs = ["-w", dir_path, "-v", "%s:%s" % (dir_path,dir_path), "--env", "OPENMODELICALIBRARY=%s" % "/usr/lib/omlibrary", "--env", "GC_MARKERS=1", "-v", "%s:/usr/lib/omlibrary" % librariespath]
commands = ["docker", "run", "--user", str(os.getuid())] + dockerExtraArgs + [docker]
omc_cmd = commands + ["omc"]
else:
commands = []
dockerExtraArgs = []
if os.environ.get("OPENMODELICAHOME"):
omc_cmd = ["%s/bin/omc" % os.environ.get("OPENMODELICAHOME")]
else:
omc_cmd = ["omc"]
if result_location != "" and not os.path.exists(result_location):
os.makedirs(result_location)
if configs == []:
print("Error: Expected at least one configuration file to start the library test")
sys.exit(1)
from OMPython import OMCSession, OMCSessionZMQ
version_cmd = "--version"
# Try to make the processes a bit nicer...
os.environ["GC_MARKERS"]="1"
print("Start OMC version")
if ompython_omhome != "":
# Use a different OMC for running OMPython than for running the tests
omhome = os.environ["OPENMODELICAHOME"]
omc_version = subprocess.check_output(omc_cmd + ["--version"], stderr=subprocess.STDOUT).decode("ascii").strip()
os.environ["OPENMODELICAHOME"] = ompython_omhome
omc = OMCSessionZMQ()
ompython_omc_version=omc.sendExpression('getVersion()')
os.environ["OPENMODELICAHOME"] = omhome
else:
omc = OMCSessionZMQ(docker=docker, dockerExtraArgs=dockerExtraArgs)
omhome=omc.sendExpression('getInstallationDirectoryPath()')
omc_version=omc.sendExpression('getVersion()')
ompython_omc_version=omc_version
ompython_omc_version=ompython_omc_version.replace("OMCompiler","").strip()
def timeSeconds(f):
return html.escape("%.2f" % f)
if not docker:
omc.sendExpression('setModelicaPath("%s")' % librariespath)
omc_exe=os.path.join(omhome,"bin","omc")
dygraphs=os.path.join(ompython_omhome or omhome,"share","doc","omc","testmodels","dygraph-combined.js")
print(omc_exe,omc_version,dygraphs)
sys.stdout.flush()
# Do feature checks. Handle things like old RML-style arguments...
subprocess.check_output(omc_cmd + ["-n=1", version_cmd], stderr=subprocess.STDOUT).strip()
sys.stdout.flush()
fmisimulatorversion = None
if fmisimulator:
fmisimulatorversion = subprocess.check_output([fmisimulator, "-v"], stderr=subprocess.STDOUT).strip()
print(fmisimulatorversion)
else:
if allTestsFmi:
raise Exception("No OMSimulator; trying to simulate using FMI")
print("No OMSimulator")
sys.stdout.flush()
try:
os.unlink("HelloWorld")
except OSError:
pass
subprocess.check_output(omc_cmd + ["--simCodeTarget=Cpp", "HelloWorld.mos"], stderr=subprocess.STDOUT)
if os.path.exists("HelloWorld"):
print("Have C++ HelloWorld simulation executable")
haveCppRuntime=simulationAcceptsFlag("")
if haveCppRuntime:
print("Have C++ runtime")
else:
print("C++ HelloWorld simulation executable failed to run")
else:
haveCppRuntime=False
print("No C++ runtime")
sys.stdout.flush()
try:
subprocess.check_output(omc_cmd + ["Architecture.mo"], stderr=subprocess.STDOUT)
except subprocess.CalledProcessError:
print("Patching ModelicaServices for Architecture bug...")
for f in glob.glob(librariespath + "/ModelicaServices*/package.mo") + glob.glob(omhome + "/Modelica */Constants.mo"):
with open(f) as fin:
content = fin.read()
assert(len(content) > 0)
content = content.replace("OpenModelica.Internal.Architecture.integerMax()","2147483647")
assert(len(content) > 0)
with open(f, "w") as fout:
open(f,"w").write(content)
print("Done patching ModelicaServices...")
sys.stdout.flush()
defaultCustomCommands = []
if extraflags:
defaultCustomCommands += [extraflags]
def testHelloWorld(cmd):
with open("HelloWorld.mos") as fin:
helloWorldContents = fin.read()
try:
os.unlink("HelloWorld")
except OSError:
pass
open("HelloWorld.cmd.mos","w").write(cmd + "\n" + helloWorldContents)
try:
out=subprocess.check_output(omc_cmd + ["HelloWorld.cmd.mos"], stderr=subprocess.STDOUT)
if os.path.exists("HelloWorld") and not "Error:" in out.decode():
return True
except subprocess.CalledProcessError as e:
pass
return False
for cmd in [
'setCommandLineOptions("-d=nogen");',
'setCommandLineOptions("-d=initialization");',
'setCommandLineOptions("-d=backenddaeinfo");',
'setCommandLineOptions("-d=discreteinfo");',
'setCommandLineOptions("-d=stateselection");',
'setCommandLineOptions("-d=execstat");',
'setMatchingAlgorithm("PFPlusExt");',
'setIndexReductionMethod("dynamicStateSelection");'
]:
if testHelloWorld(cmd):
defaultCustomCommands.append(cmd)
canChangeOptLevel = False
if testHelloWorld("flags:=getCFlags();setCFlags(flags+\" -O1\");"):
canChangeOptLevel = True
else:
print("Cannot change optimization level")
fmiOK_C = False
fmiOK_Cpp = False
try:
out=subprocess.check_output(omc_cmd + ["--simCodeTarget=C", "FMI.mos"], stderr=subprocess.STDOUT)
if os.path.exists("HelloWorldX.fmu") and not "Error:" in out.decode():
fmiOK_C = True
print("C FMU OK")
else:
print("No C-based FMUs (files not generated in correct location)")
except subprocess.CalledProcessError as e:
pass
try:
out=subprocess.check_output(omc_cmd + ["--simCodeTarget=Cpp", "FMI.mos"], stderr=subprocess.STDOUT)
if os.path.exists("HelloWorldX.fmu") and not "Error:" in out.decode():
fmiOK_Cpp = True
print("C++ FMU OK")
else:
print("No C++-based FMUs (files not generated in correct location)")
except subprocess.CalledProcessError as e:
pass
try:
os.unlink("HelloWorld")
except OSError:
pass
print(subprocess.check_output(omc_cmd + ["HelloWorld.mos"], stderr=subprocess.STDOUT).decode().strip())
assert(os.path.exists("HelloWorld"))
abortSimulationFlag="-abortSlowSimulation" if simulationAcceptsFlag("-abortSlowSimulation") else ""
alarmFlag="-alarm" if simulationAcceptsFlag("-alarm=480") else ""
configs_lst = [readConfig(c, abortSimulationFlag=abortSimulationFlag, alarmFlag=alarmFlag, overrideDefaults=overrideDefaults, defaultCustomCommands=defaultCustomCommands, extrasimflags=extrasimflags) for c in configs]
configs = []
preparedReferenceDirs = {}
for c in configs_lst:
configs = configs + c
for (lib,c) in configs:
if "referenceFiles" in c:
c["referenceFilesURL"] = c["referenceFiles"]
if isinstance(c["referenceFiles"], (str, bytes)):
m = re.search("^[$][A-Z]+", c["referenceFiles"])
if m:
k = m.group(0)[1:]
if k not in os.environ:
raise Exception("Environment variable %s not defined, but used in JSON config for reference files" % k)
c["referenceFiles"] = c["referenceFiles"].replace(m.group(0), os.environ[k])
elif "giturl" in c["referenceFiles"]:
if c["referenceFiles"]["destination"] in preparedReferenceDirs:
(c["referenceFiles"],c["referenceFilesURL"]) = preparedReferenceDirs[destination]
continue
giturl = c["referenceFiles"]["giturl"]
destination = c["referenceFiles"]["destination"]
if not os.path.isdir(destination):
subprocess.check_call(["git", "clone", giturl, destination], stderr=subprocess.STDOUT)
destinationReal = os.path.realpath(destination)
subprocess.check_call(["git", "clean", "-fdx", "--exclude=*.hash"], stderr=subprocess.STDOUT, cwd=destinationReal)
subprocess.check_call(["git", "fetch", "origin"], stderr=subprocess.STDOUT, cwd=destination)
subprocess.check_call(["git", "reset", "--hard", "origin/master"], stderr=subprocess.STDOUT, cwd=destinationReal)
subprocess.check_call(["git", "clean", "-fdx", "--exclude=*.hash"], stderr=subprocess.STDOUT, cwd=destinationReal)
subprocess.check_call(["find", ".", "-name", "*.mat.xz", "-exec", "xz", "--decompress", "--keep", "{}", ";"], stderr=subprocess.STDOUT, cwd=destinationReal)
githash = subprocess.check_output(["git", "rev-parse", "--verify", "HEAD"], stderr=subprocess.STDOUT, cwd=destinationReal)
c["referenceFiles"] = destinationReal
if giturl.startswith("https://github.com"):
c["referenceFilesURL"] = '<a href="%s/tree/%s">%s (%s)</a>' % (giturl,githash.strip(),giturl,githash.strip())
else:
c["referenceFilesURL"] = "%s (%s)" % (giturl,githash.strip())
preparedReferenceDirs[destination] = (c["referenceFiles"],c["referenceFilesURL"])
else:
raise Exception("Unknown referenceFiles in config: %s" % (str(c["referenceFiles"])))
if allTestsFmi:
c["fmi"] = "2.0"
# Create mos-files
conn = sqlite3.connect('sqlite3.db')
cursor = conn.cursor()
user_version = cursor.execute("PRAGMA user_version").fetchone()[0]
if user_version==0:
# BOOLEAN NOT NULL CHECK (verify IN (0,1) AND builds IN (0,1) AND simulates IN (0,1))
# Table to lookup from a run (date, branch) to omcversion used
cursor.execute("CREATE TABLE if not exists [omcversion] (date integer NOT NULL, branch text NOT NULL, omcversion text NOT NULL)")
# Table to lookup from a run (date, branch) which library versions were used
cursor.execute("CREATE TABLE if not exists [libversion] (date integer NOT NULL, branch text NOT NULL, libname text NOT NULL, libversion text NOT NULL, confighash integer NOT NULL)")
elif user_version==1:
cursor.execute("ALTER TABLE [libversion] ADD COLUMN confighash integer NOT NULL DEFAULT(0)")
elif user_version==2:
tables = [tbl for (tbl,) in cursor.execute("SELECT name FROM sqlite_master WHERE type='table'") if tbl not in ["libversion","omcversion"]]
for tbl in tables:
cursor.execute("ALTER TABLE [%s] ADD COLUMN parsing real NOT NULL DEFAULT(0.0)" % tbl)
elif user_version in [3]:
pass
else:
print("Unknown schema user_version=%d" % user_version)
sys.exit(1)
cursor.execute('''CREATE TABLE if not exists [%s]
(date integer NOT NULL, libname text NOT NULL, model text NOT NULL, exectime real NOT NULL,
frontend real NOT NULL, backend real NOT NULL, simcode real NOT NULL, templates real NOT NULL, compile real NOT NULL, simulate real NOT NULL,
verify real NOT NULL, verifyfail integer NOT NULL, verifytotal integer NOT NULL, finalphase integer NOT NULL, parsing real NOT NULL)''' % branch)
cursor.execute('''DROP INDEX IF EXISTS [idx_%s_date]''' % branch)
cursor.execute('''DROP INDEX IF EXISTS idx_omcversion_date''')
cursor.execute('''DROP INDEX IF EXISTS idx_libversion_date''')
# Set user_version to the current schema
cursor.execute("PRAGMA user_version=3")
def strToHashInt(s):
return int(hashlib.sha1((s+"fixCorruptBuilds-2017-03-26").encode("utf-8")).hexdigest()[0:8],16)
def findAllFiles(d):
res = []
for root, dirs, files in os.walk(d):
if "/." in root:
continue
res += [os.path.join(root, f) for f in files if f[0]!="."]
return res
def getmd5(f):
hf = f+".hash"
if not os.path.exists(hf) or (os.path.getmtime(f) > os.path.getmtime(hf)):
with open(hf, "w") as fout:
with open(f, "rb") as fin:
fout.write(hashlib.sha512(fin.read()).hexdigest())
with open(hf) as fin:
return fin.read()
def hashReferenceFiles(s):
if s=="":
return s
files = [f for f in findAllFiles(s) if (not f.endswith(".hash"))]
files = sorted(files)
res = "".join([getmd5(f) for f in files])
return res+"fixCorruptBuilds-2017-06-21"
stats_by_libname = {}
skipped_libs = {}
tests=[]
for (library,conf) in configs:
c=conf.copy()
del(c["configFromFile"])
if "referenceFiles" in c:
del(c["referenceFilesURL"])
confighash = strToHashInt(str(c)+hashReferenceFiles(conf["referenceFiles"]))
else:
confighash = strToHashInt(str(c)+hashReferenceFiles(""))
conf["confighash"] = confighash
conf["omhome"] = omhome
conf["single_thread_cmd"] = single_thread
conf["haveCppRuntime"] = haveCppRuntime
conf["ulimitMemory"] = conf.get("ulimitMemory") or ulimitMemory
if conf.get("fmi"):
conf["haveFMI"] = fmiOK_C
conf["haveFMICpp"] = fmiOK_Cpp
conf["fmisimulator"] = fmisimulator
conf["fmuType"] = conf.get("fmuType", "me")
if (not canChangeOptLevel) and "optlevel" in conf:
print("Deleting optlevel")
del conf["optlevel"]
if not (omc.sendExpression('setCommandLineOptions("-g=Modelica")') or omc.sendExpression('setCommandLineOptions("+g=Modelica")')):
print("Failed to set Modelica grammar")
sys.exit(1)
omc.sendExpression('clear()')
if "loadFileCommands" in conf:
for command in conf["loadFileCommands"]:
if not omc.sendExpression(command):
try:
print("Failed to run command %s: %s" % (command,omc.sendExpression('OpenModelica.Scripting.getErrorString()')))
except:
print("Failed to run command %s OpenModelica.Scripting.getErrorString() failed..." % command)
librariesToLoad = []
else:
librariesToLoad = [[library,conf["libraryVersion"]]] + conf.get("extraLibraries", [])
for (lib,version) in librariesToLoad:
if conf["libraryVersionLatestInPackageManager"]:
availableVersions = omc.sendExpression('getAvailablePackageVersions(%s,"%s")' % (lib,version))
if not availableVersions:
try:
print("Failed to getAvailablePackageVersions %s %s: %s" % (library,version,omc.sendExpression('OpenModelica.Scripting.getErrorString()')))
except:
print("Failed to getAvailablePackageVersions %s %s. OpenModelica.Scripting.getErrorString() failed..." % (library,conf["libraryVersion"]))
versions = "{" + ",".join(['"'+v+'"' for v in availableVersions]) + "}"
else:
versions = '{"%s"}' % version
if not omc.sendExpression('loadModel(%s,%s)' % (lib,versions)):
try:
print("Failed to load library %s %s: %s" % (library,versions,omc.sendExpression('OpenModelica.Scripting.getErrorString()')))
except:
print("Failed to load library %s %s. OpenModelica.Scripting.getErrorString() failed..." % (library,conf["libraryVersion"]))
conf["loadFiles"] = sorted(omc.sendExpression("{getSourceFile(cl) for cl in getClassNames()}"))
if not (omc.sendExpression('setCommandLineOptions("-g=MetaModelica")') or omc.sendExpression('setCommandLineOptions("+g=Modelica")')):
print("Failed to set MetaModelica grammar")
sys.exit(1)
try:
conf["resourceLocation"]=omc.sendExpression('uriToFilename("modelica://%s/Resources")' % library)
except:
conf["resourceLocation"]=""
if "runOnceBeforeTesting" in conf:
for cmd in conf["runOnceBeforeTesting"]:
# replace the resource location in the command if present
cmd = [c.replace("$resourceLocation", conf["resourceLocation"]) for c in cmd]
subprocess.check_call(cmd, stderr=subprocess.STDOUT)
conf["libraryVersionRevision"]=omc.sendExpression('getVersion(%s)' % library)
librarySourceFile=omc.sendExpression('getSourceFile(%s)' % library)
lastChange=(librarySourceFile[:-3]+".last_change") if not librarySourceFile.endswith("package.mo") else (os.path.dirname(librarySourceFile)+".last_change")
if os.path.exists(lastChange):
conf["libraryLastChange"] = " %s (revision %s)" % (conf["libraryVersionRevision"],"\n".join(open(lastChange).readlines()).strip())
else:
conf["libraryLastChange"] = "%s (%s)" % (conf["libraryVersionRevision"], librarySourceFile)
metadataFile = os.path.join(os.path.dirname(librarySourceFile), "openmodelica.metadata.json")
if os.path.exists(metadataFile):
with open(metadataFile) as metadataIn:
metadata = json.load(metadataIn)
conf["libraryLastChange"] = "%s (%s)" % (metadata["version"],metadata.get("sha") or metadata["zipfile"])
conf["metadata"] = json.dumps(metadata, indent=1)
else:
conf["metadata"] = ""
if not conf["libraryVersionRevision"]:
conf["libraryVersionRevision"] = conf["libraryLastChange"]
if conf.get("fmi") and fmisimulatorversion:
conf["libraryVersionRevision"] = conf["libraryVersionRevision"] + " " + fmisimulatorversion.decode("ascii")
conf["libraryLastChange"] = conf["libraryLastChange"] + " " + fmisimulatorversion.decode("ascii")
res=omc.sendExpression('{c for c guard isExperiment(c) and not regexBool(typeNameString(x), "^Modelica_Synchronous\\.WorkInProgress") in getClassNames(%s, recursive=true)}' % library)
if conf.get("ignoreModelPrefix"):
prefix = conf["ignoreModelPrefix"]
res=list(filter(lambda x: not x.startswith(prefix), res))
libName=shared.libname(library, conf)
v = cursor.execute("""SELECT date,libversion,libname,branch,omcversion FROM [libversion] NATURAL JOIN [omcversion]
WHERE libversion=? AND libname=? AND branch=? AND omcversion=? AND confighash=? ORDER BY date DESC LIMIT 1""", (conf["libraryLastChange"],libName,branch,omc_version,confighash)).fetchone()
if libName in stats_by_libname or libName in skipped_libs:
raise Exception("Duplicate libName found: %s" % libName)
if v is None:
stats_by_libname[libName] = {"conf":conf, "stats":[]}
tests = tests + [(r,library,libName,libName+"_"+r,conf) for r in res]
print("Running library %s (%d tests)" % (libName, len(res)))
else:
print("Skipping %s as we already have results for it: %s" % (libName,str(v)))
skipped_libs[libName] = v[0]
try:
del omc
except:
pass
print("Checked which libraries to run")
sys.stdout.flush()
errorOccurred=False
for (modelName,library,libName,name,conf) in tests:
if conf["alarmFlag"]!="":
conf["simFlags"]="%s %s=%d %s" % (conf["abortSlowSimulation"],conf["alarmFlag"],conf["ulimitExe"],conf["extraSimFlags"])
else:
conf["simFlags"]="%s %s" % (conf["abortSlowSimulation"],conf["extraSimFlags"])
replacements = (
(u"#logFile#", "/tmp/OpenModelicaLibraryTesting.log"),
(u"#library#", library),
(u"#modelName#", modelName),
(u"#fileName#", name),
(u"#customCommands#", conf["customCommands"]),
(u"#modelVersion#", conf["libraryVersion"]),
(u"#ulimitOmc#", str(conf["ulimitOmc"])),
(u"#default_tolerance#", str(conf["default_tolerance"])),
(u"#reference_reltol#", str(conf["reference_reltol"])),
(u"#reference_reltolDiffMinMax#", str(conf["reference_reltolDiffMinMax"])),
(u"#reference_rangeDelta#", str(conf["reference_rangeDelta"])),
(u"#simFlags#", conf["simFlags"]),
(u"#referenceFiles#", str(conf.get("referenceFilesURL") or conf.get("referenceFiles") or "")),
(u"#referenceFileNameDelimiter#", conf["referenceFileNameDelimiter"]),
(u"#referenceFileExtension#", conf["referenceFileExtension"]),
)
with open(name + ".conf.json", 'w') as fp:
newconf = dict(conf.items())
newconf["library"] = library
newconf["modelName"] = modelName
newconf["fileName"] = name
try:
newconf["referenceFile"] = getReferenceFileName(newconf)
except Exception as e:
# Find all such errors
print(e)
errorOccurred = True
json.dump(newconf, fp)
if errorOccurred:
sys.exit(1)
print("Created .conf.json files")
sys.stdout.flush()
def runScript(c, timeout, memoryLimit):
j = "files/%s.stat.json" % c
try:
os.remove(j)
except:
pass
start=monotonic()
# runCommand("%s %s %s.mos" % (omc_exe, single_thread, c), prefix=c, timeout=timeout)
if 0 != runCommand("ulimit -v %d; ./testmodel.py --libraries=%s %s --ompython_omhome=%s %s.conf.json > files/%s.cmdout 2>&1" % (memoryLimit, librariespath, ("--docker %s --dockerExtraArgs '%s'" % (docker, " ".join(dockerExtraArgs))) if docker else "", ompython_omhome, c, c), prefix=c, timeout=timeout):
print("files/%s.err" % c)
with open("files/%s.err" % c, "a+") as errfile:
errfile.write("Failed to read output from testmodel.py, exit status != 0:\n")
try:
with open("files/%s.cmdout" % c) as cmdout:
errfile.write(cmdout.read())
except IOError:
pass
except OSError:
pass
if clean:
try:
os.unlink("files/%s.cmdout" % c)
except OSError:
pass
execTime=monotonic()-start
assert(execTime >= 0.0)
try:
data=json.load(open(j))
except:
data = {"phase":0}
data["exectime"] = execTime
json.dump(data, open(j,"w"))
def expectedExec(c):
(model,lib,libName,name,data) = c
if "expectedExec" in data:
return data["expectedExec"]
cursor.execute("SELECT exectime FROM [%s] WHERE libname = ? AND model = ? ORDER BY date DESC LIMIT 1" % branch, (libName,model))
v = cursor.fetchone()
data["expectedExec"] = (v or (0.0,))[0]
return data["expectedExec"]
start=monotonic()
tests=sorted(tests, key=lambda c: expectedExec(c), reverse=True)
stop=monotonic()
print("Querying expected execution time: %s" % friendlyStr(stop-start))
sys.stdout.flush()
# Cleanup old runs
try:
if clean:
rmtree("./files")
print("Cleaned files directory")
except OSError:
pass
try:
os.mkdir("files")
except OSError:
pass
print("Created files directory")
sys.stdout.flush()
if len(tests)==0:
print("Everything already up to date. Not executing any tests.")
sys.exit(0)
print("Starting execution of %d tests. Estimated execution time %s (wrong if there are new or few tests)." % (len(tests), friendlyStr(sum(expectedExec(c) for c in tests)/(1.0*n_jobs))))
sys.stdout.flush()
cmd_res=[0]
start=monotonic()
start_as_time=time.localtime()
testRunStartTimeAsEpoch = int(time.time())
# Need translateModel + make + exe...
cmd_res=Parallel(n_jobs=n_jobs)(delayed(runScript)(name, 2*data["ulimitOmc"]+data["ulimitExe"]+25, data["ulimitMemory"]) for (model,lib,libName,name,data) in tests)
stop=monotonic()
print("Execution time: %s" % friendlyStr(stop-start))
assert(stop-start >= 0.0)
#if max(cmd_res) > 0:
# raise Exception("A command failed with exit status")
def loadJsonOrEmptySet(f):
if os.stat(f).st_size == 0:
return {}
else:
return json.load(open(f))
stats=dict([(name,(name,model,libname,loadJsonOrEmptySet("files/%s.stat.json" % name))) for (model,lib,libname,name,conf) in tests])
#for k in sorted(stats.keys(), key=lambda c: stats[c][3]["exectime"], reverse=True):
# print("%s: exectime %.2f" % (k, stats[k][3]["exectime"]))
for key in stats.keys():
(name,model,libname,data)=stats[key]
stats_by_libname[libname]["stats"].append(stats[key])
values = (testRunStartTimeAsEpoch,
libname,
model,
data.get("exectime") or 0.0,
data.get("frontend") or 0.0,
data.get("backend") or 0.0,
data.get("simcode") or 0.0,
data.get("templates") or 0.0,
data.get("build") or 0.0,
data.get("sim") or 0.0,
(data.get("diff") or {}).get("time") or 0.0,
len((data.get("diff") or {}).get("vars") or []),
(data.get("diff") or {}).get("numCompared") or 0,
data.get("phase") or 0,
data.get("parsing") or 0.0
)
# print values
cursor.execute("INSERT INTO [%s] VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?,?)" % branch, values)
for libname in stats_by_libname.keys():
confighash = stats_by_libname[libname]["conf"]["confighash"]
cursor.execute("INSERT INTO [libversion] VALUES (?,?,?,?,?)", (testRunStartTimeAsEpoch, branch, libname, stats_by_libname[libname]["conf"]["libraryLastChange"], confighash))
cursor.execute("INSERT INTO [omcversion] VALUES (?,?,?)", (testRunStartTimeAsEpoch, branch, omc_version))
"""
# Not really a good thing to do; was just done to make generation of the report simpler
for libname in skipped_libs.keys():
values = (testRunStartTimeAsEpoch, skipped_libs[libname], libname)
cursor.execute("UPDATE [libversion] SET date = ? WHERE date = ? AND libname = ? AND branch = ? AND confighash = ?", (testRunStartTimeAsEpoch, skipped_libs[libname], libname, branch, confighash))
cursor.execute("UPDATE [%s] SET date = ? WHERE date = ? AND libname = ?" % branch, values)
"""
def checkNumSucceeded(numSucceeded, n):
if numSucceeded[n]==numSucceeded[n-1]:
return "#00FF00"
else:
return "#FFCC66"
def checkPhase(phase, n):
if phase >= n:
return "#00FF00"
else:
return "#FFCC66"
def is_non_zero_file(fpath):
return os.path.isfile(fpath) and os.path.getsize(fpath) > 0
def cpu_name():
for line in open("/proc/cpuinfo").readlines():
if "model name" in line.strip():
return (re.sub( ".*model name.*:", "", line,1)).strip()
try:
lsb_release = subprocess.check_output(commands + ["cat","/etc/lsb-release"]).decode().strip()
lsb_release = dict(a.split("=") for a in lsb_release.split("\n"))["DISTRIB_DESCRIPTION"].strip('"')
except:
lsb_release = ""
sysInfo = "%s, %d GB RAM, %s%s" % (cpu_name(), int(math.ceil(psutil.virtual_memory().total / (1024.0**3))), ("Docker " + docker + " ") if docker else "", lsb_release)
htmltpl=open("library.html.tpl").read()
for libname in stats_by_libname.keys():
if libname in skipped_libs:
continue
s = None # Make sure I don't use this
filesList = open(libname + ".files", "w")
filesList.write("/\n")
filesList.write("/%s.html\n" % libname)
filesList.write("/files/\n")
conf = stats_by_libname[libname]["conf"]
stats = stats_by_libname[libname]["stats"]
for s in stats:
filename_prefix = "files/%s_%s" % (s[2],s[1])
filesList.write("/%s*diff*csv\n" % filename_prefix)
filesList.write("/%s*diff*html\n" % filename_prefix)
if is_non_zero_file(filename_prefix+".sim"):
filesList.write("/%s.sim\n" % filename_prefix)
if is_non_zero_file(filename_prefix+".err"):
filesList.write("/%s.err\n" % filename_prefix)
variables = (s[3].get("diff") or {}).get("vars") or []
if len(variables)>0:
filesList.write("/%s.diff.html\n" % filename_prefix)
for v in variables:
filesList.write("/%s.diff.%s.csv\n" % (filename_prefix, v))
filesList.write("/%s.diff.%s.html\n" % (filename_prefix, v))
filesList.close()
testsHTML = "\n".join(['<tr><td>%s%s</td><td bgcolor="%s">%s</td><td bgcolor="%s">%s</td><td bgcolor="%s">%s</td><td>%s</td><td bgcolor="%s">%s</td><td bgcolor="%s">%s</td><td bgcolor="%s">%s</td><td bgcolor="%s">%s</td><td bgcolor="%s">%s</td></tr>\n' %
(lambda filename_prefix, diff:
(
('<a href="%s">%s</a>' % (filename_prefix + ".err", html.escape(s[1]))) if is_non_zero_file(filename_prefix + ".err") else html.escape(s[1]),
(' (<a href="%s">sim</a>)' % (filename_prefix + ".sim")) if is_non_zero_file(filename_prefix + ".sim") else "",
checkPhase(s[3]["phase"], 7) if s[3]["phase"]>=6 else "#FFFFFF",
("%s (%d verified)" % (timeSeconds(diff.get("time")), diff.get("numCompared"))) if s[3]["phase"]>=7 else (" " if diff is None else
('%s (<a href="%s.diff.html">%d/%d failed</a>)' % (timeSeconds(diff.get("time")), filename_prefix, len(diff.get("vars")), diff.get("numCompared")))),
checkPhase(s[3]["phase"], 6),
timeSeconds(s[3].get("sim") or 0),
checkPhase(s[3]["phase"], 5),
timeSeconds(sum(s[3].get(x) or 0.0 for x in ["frontend","backend","simcode","templates","build"])),
timeSeconds(s[3].get("parsing") or 0),
checkPhase(s[3]["phase"], 1),
timeSeconds(s[3].get("frontend") or 0),
checkPhase(s[3]["phase"], 2),
timeSeconds(s[3].get("backend") or 0),
checkPhase(s[3]["phase"], 3),
timeSeconds(s[3].get("simcode") or 0),
checkPhase(s[3]["phase"], 4),
timeSeconds(s[3].get("templates") or 0),
checkPhase(s[3]["phase"], 5),
timeSeconds(s[3].get("build") or 0)
))(filename_prefix="files/%s_%s" % (s[2], s[1]), diff=s[3].get("diff"))
for s in natsorted(stats, key=lambda s: s[1])])
numSucceeded = [len(stats)] + [sum(1 if s[3]["phase"]>=i else 0 for s in stats) for i in range(1,8)]
replacements = (
(u"#sysInfo#", html.escape(sysInfo)),
(u"#omcVersion#", html.escape(omc_version)),
(u"#fmi#", ("<p>"+html.escape("FMI version: %s" % conf.get("fmi"))+"</p>") if conf.get("fmi") else ""),
(u"#optlevel#", html.escape(conf.get("optlevel")) if (canChangeOptLevel and conf.get("optlevel")) else "Tool default"),
(u"#timeStart#", html.escape(time.strftime('%Y-%m-%d %H:%M:%S', start_as_time))),
(u"#fileName#", html.escape(libname)),
(u"#customCommands#", html.escape("\n".join(conf["customCommands"]))),
(u"#libraryVersionRevision#", html.escape(conf["libraryVersionRevision"])),
(u"#metadata#", html.escape(conf["metadata"])),
(u"#ulimitOmc#", html.escape(str(conf["ulimitOmc"]))),
(u"#ulimitExe#", html.escape(str(conf["ulimitExe"]))),
(u"#default_tolerance#", html.escape(str(conf["default_tolerance"]))),
(u"#simFlags#", html.escape(conf.get("simFlags") or "")),
(u"#referenceFiles#", ('<p>Reference Files: %s</p>' % conf["referenceFilesURL"].replace(os.path.dirname(os.path.realpath(__file__)),"")) if ((conf.get("referenceFilesURL") or "") != "") else ""),
(u"#referenceTool#", ('<p>Verified using: %s (diffSimulationResults)</p>' % html.escape(ompython_omc_version)) if ((conf.get("referenceFiles") or "") != "") else ""),
(u"#Total#", html.escape(str(numSucceeded[0]))),
(u"#FrontendColor#", checkNumSucceeded(numSucceeded, 1)),
(u"#BackendColor#", checkNumSucceeded(numSucceeded, 2)),
(u"#SimCodeColor#", checkNumSucceeded(numSucceeded, 3)),
(u"#TemplatesColor#", checkNumSucceeded(numSucceeded, 4)),
(u"#CompilationColor#", checkNumSucceeded(numSucceeded, 5)),
(u"#SimulationColor#", checkNumSucceeded(numSucceeded, 6)),
(u"#VerificationColor#", checkNumSucceeded(numSucceeded, 7)),
(u"#Frontend#", html.escape(str(numSucceeded[1]))),
(u"#Backend#", html.escape(str(numSucceeded[2]))),
(u"#SimCode#", html.escape(str(numSucceeded[3]))),
(u"#Templates#", html.escape(str(numSucceeded[4]))),
(u"#Compilation#", html.escape(str(numSucceeded[5]))),
(u"#Simulation#", html.escape(str(numSucceeded[6]))),
(u"#Verification#", html.escape(str(numSucceeded[7]))),
(u"#totalTime#", html.escape(str(datetime.timedelta(seconds=int(sum(s[3].get("exectime") or 0.0 for s in stats)))))),
(u"#config#", html.escape(json.dumps(conf["configFromFile"], indent=1, sort_keys=True))),
(u"#testsHTML#", testsHTML)
)
open("%s.html" % libname, "w").write(multiple_replace(htmltpl, *replacements))
if result_location != "":
result_location_libname = "%s/%s" % (result_location, libname)
try:
os.mkdir("emptydir")
except:
pass
subprocess.check_output(["rsync", "-aR", "emptydir/", result_location])
subprocess.check_output(["rsync", "-aR", "emptydir/", result_location_libname])
subprocess.check_output(["rsync", "-aR", "emptydir/", result_location_libname+"/files"])
try:
subprocess.check_output(["rsync", "-aR", "--delete-excluded", "--include-from=%s.files" % libname, "--exclude=*", "./", result_location_libname])
except:
subprocess.check_output(["rsync", "-aR", "emptydir/", result_location])
subprocess.check_output(["rsync", "-aR", "emptydir/", result_location_libname])
subprocess.check_output(["rsync", "-aR", "emptydir/", result_location_libname+"/files"])
subprocess.check_output(["rsync", "-aR", "--delete-excluded", "--include-from=%s.files" % libname, "--exclude=*", "./", result_location_libname])
if (conf.get("referenceFiles") or "") != "":
subprocess.check_output(["rsync", "-a", dygraphs, result_location_libname+"/files"])
if clean:
for g in ["*.o","*.so","*.h","*.c","*.cpp","*.simsuccess","*.conf.json","*.tmpfiles","*.log","*.libs","OMCpp*","*.fmu*","temp_*"]:
for f in glob.glob(g):
if os.path.isdir(f):
rmtree(f)
elif os.path.exists(f):
os.unlink(f)
if clean:
rmtree("files/")
# Do not commit until we have generated and uploaded the reports
conn.commit()
conn.close()