5 # Run compiler unit tests
7 # Copyright (c) 2008/2009 Matthias Kramm <kramm@quiss.org>
9 # This program is free software; you can redistribute it and/or modify
10 # it under the terms of the GNU General Public License as published by
11 # the Free Software Foundation; either version 2 of the License, or
12 # (at your option) any later version.
14 # This program is distributed in the hope that it will be useful,
15 # but WITHOUT ANY WARRANTY; without even the implied warranty of
16 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 # GNU General Public License for more details.
19 # You should have received a copy of the GNU General Public License
20 # along with this program; if not, write to the Free Software
21 # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
29 from optparse import OptionParser
34 for line in s.split("\n"):
35 if line.startswith("[") and line.endswith("]"):
39 if not line.startswith("ok"):
41 if line.startswith("ok "):
45 nr,len = int(line[3:i]),int(line[i+1:])
61 def runcmd(cmd,args,wait):
64 fo = os.fdopen(fo, "wb")
65 p = subprocess.Popen([cmd] + args, executable=cmd, stdout=fo, stderr=fo)
68 for i in range(wait*10):
69 if fi in select.select([fi],[],[], 0.01)[0]:
70 output += os.read(fi, 8192)
71 if "[exit]" in output:
81 os.system("killall -9 %s >/dev/null 2>/dev/null" % cmd)
84 if fi in select.select([fi],[],[], 0.01)[0]:
85 output += os.read(fi, 8192)
91 def __init__(self, filename):
92 self.filename = filename
93 self.filename_milestone = filename+"_milestone"
95 self.filename2status = marshal.load(open(self.filename, "rb"))
97 self.filename2status = {}
99 self.milestone = marshal.load(open(self.filename_milestone, "rb"))
103 def parse_args(self):
104 parser = OptionParser()
105 parser.add_option("-d", "--diff", dest="diff", help="Only run tests that failed the last time",action="store_true")
106 parser.add_option("-a", "--all", dest="all", help="Run all tests (also tests expected fail)",action="store_true")
107 parser.add_option("-t", "--tag", dest="tag", help="Mark the current pass/fail statistic as milestone",action="store_true")
108 (options, args) = parser.parse_args()
110 if args and args[0]=="add":
113 self.milestone[args[1]] = "ok"
114 self.filename2status = self.milestone
118 self.__dict__.update(options.__dict__)
122 self.runtime = 5 # allow more time if we're tagging this state
126 self.checknum = int(args[0])
130 return Cache(filename)
133 fi = open(self.filename, "wb")
134 marshal.dump(self.filename2status, fi)
138 fi = open(self.filename_milestone, "wb")
139 marshal.dump(self.filename2status, fi)
142 def highlight(self, nr, filename):
143 return self.checknum==nr
145 def skip_file(self, nr, filename):
146 if self.checknum>=0 and nr!=self.checknum:
148 if not self.all and self.milestone[filename]!="ok":
150 if self.diff and self.filename2status[filename]=="ok":
154 def file_status(self, filename, status):
155 self.filename2status[filename] = status
158 def __init__(self, cache, nr, file, run):
163 self.flash_output = None
164 self.flash_error = None
165 self.compile_output = None
166 self.compile_error = None
169 try: os.unlink("abc.swf");
171 ret,output = runcmd("./parser",[self.file],wait=cache.runtime)
172 self.compile_error = 0
173 self.compile_output = output
176 self.compile_output += "\nExit status %d" % (-ret)
177 self.exit_status = -ret
178 self.compile_error = 1
180 if not os.path.isfile("abc.swf"):
181 self.compile_error = 1
186 ret,output = runcmd("flashplayer",["abc.swf"],wait=cache.runtime)
187 os.system("killall flashplayer")
188 self.flash_output = output
190 if not check(self.flash_output):
196 print self.r(str(self.nr),3)," ",
197 if self.compile_error:
199 if self.exit_status == 11:
208 if not self.flash_error:
217 def doprintlong(self):
218 print self.nr, self.file
219 print "================================"
220 print "compile:", (self.compile_error and "error" or "ok")
221 print self.compile_output
224 print "================================"
225 print "run:", (self.flash_error and "error" or "ok")
226 print self.flash_output
227 print "================================"
232 return (" "*(l-len(s))) + s
236 return s + (" "*(l-len(s)))
238 class Test(TestBase):
239 def __init__(self, cache, nr, file):
240 TestBase.__init__(self, cache, nr, file, run=1)
241 if self.compile() and self.run():
242 cache.file_status(file, "ok")
244 cache.file_status(file, "error")
246 class ErrTest(TestBase):
247 def __init__(self, cache, nr, file):
248 TestBase.__init__(self, cache, nr, file, run=0)
250 cache.file_status(file, "error")
251 self.compile_error = True
253 cache.file_status(file, "ok")
254 self.compile_error = False
257 def __init__(self, cache, dir):
260 self.errtest = "err" in dir
262 print "-"*40,"tests \""+self.dir+"\"","-"*40
263 for file in sorted(os.listdir(self.dir)):
264 if not file.endswith(".as"):
267 file = os.path.join(self.dir, file)
269 if cache.skip_file(nr, file):
273 test = ErrTest(cache, nr, file)
275 test = Test(cache, nr, file)
277 if not cache.highlight(nr, file):
283 cache = Cache.load(".tests.cache")
287 nr = Suite(cache, "err").run(nr)
288 nr = Suite(cache, "ok").run(nr)