2010-07-27 03:15:15 +08:00
|
|
|
""" basic test session implementation.
|
2008-08-16 23:26:59 +08:00
|
|
|
|
2010-07-27 03:15:15 +08:00
|
|
|
* drives collection of tests
|
|
|
|
* triggers executions of tests
|
|
|
|
* produces events used by reporting
|
2008-08-16 23:26:59 +08:00
|
|
|
"""
|
|
|
|
|
|
|
|
import py
|
2010-04-28 03:13:09 +08:00
|
|
|
|
|
|
|
# exitcodes for the command line
|
|
|
|
EXIT_OK = 0
|
|
|
|
EXIT_TESTSFAILED = 1
|
|
|
|
EXIT_INTERRUPTED = 2
|
|
|
|
EXIT_INTERNALERROR = 3
|
|
|
|
EXIT_NOHOSTS = 4
|
2008-08-16 23:26:59 +08:00
|
|
|
|
2009-02-27 18:18:27 +08:00
|
|
|
# imports used for genitems()
|
|
|
|
Item = py.test.collect.Item
|
|
|
|
Collector = py.test.collect.Collector
|
2008-08-16 23:26:59 +08:00
|
|
|
|
2010-07-27 03:15:15 +08:00
|
|
|
class Session(object):
|
2010-01-12 00:09:07 +08:00
|
|
|
nodeid = ""
|
2010-05-25 22:52:09 +08:00
|
|
|
class Interrupted(KeyboardInterrupt):
|
|
|
|
""" signals an interrupted test run. """
|
2010-05-25 23:24:24 +08:00
|
|
|
__module__ = 'builtins' # for py3
|
2010-07-27 03:15:15 +08:00
|
|
|
|
2008-08-16 23:26:59 +08:00
|
|
|
def __init__(self, config):
|
|
|
|
self.config = config
|
2010-07-27 03:15:15 +08:00
|
|
|
self.pluginmanager = config.pluginmanager # shortcut
|
2009-04-09 22:03:09 +08:00
|
|
|
self.pluginmanager.register(self)
|
2010-05-25 22:52:09 +08:00
|
|
|
self._testsfailed = 0
|
2008-08-16 23:26:59 +08:00
|
|
|
self._nomatch = False
|
2009-03-17 15:40:39 +08:00
|
|
|
self.shouldstop = False
|
2008-08-16 23:26:59 +08:00
|
|
|
|
|
|
|
def genitems(self, colitems, keywordexpr=None):
|
|
|
|
""" yield Items from iterating over the given colitems. """
|
2010-01-18 06:23:02 +08:00
|
|
|
if colitems:
|
|
|
|
colitems = list(colitems)
|
2010-07-27 03:15:15 +08:00
|
|
|
while colitems:
|
2008-08-16 23:26:59 +08:00
|
|
|
next = colitems.pop(0)
|
2009-02-27 18:18:27 +08:00
|
|
|
if isinstance(next, (tuple, list)):
|
2010-07-27 03:15:15 +08:00
|
|
|
colitems[:] = list(next) + colitems
|
2009-02-27 18:18:27 +08:00
|
|
|
continue
|
2009-04-09 22:03:09 +08:00
|
|
|
assert self.pluginmanager is next.config.pluginmanager
|
2008-08-16 23:26:59 +08:00
|
|
|
if isinstance(next, Item):
|
|
|
|
remaining = self.filteritems([next])
|
|
|
|
if remaining:
|
2009-05-08 00:01:53 +08:00
|
|
|
self.config.hook.pytest_itemstart(item=next)
|
2010-07-27 03:15:15 +08:00
|
|
|
yield next
|
2008-08-16 23:26:59 +08:00
|
|
|
else:
|
|
|
|
assert isinstance(next, Collector)
|
2009-05-08 00:01:53 +08:00
|
|
|
self.config.hook.pytest_collectstart(collector=next)
|
2009-06-09 00:31:10 +08:00
|
|
|
rep = self.config.hook.pytest_make_collect_report(collector=next)
|
2009-04-06 04:16:27 +08:00
|
|
|
if rep.passed:
|
|
|
|
for x in self.genitems(rep.result, keywordexpr):
|
2010-07-27 03:15:15 +08:00
|
|
|
yield x
|
2009-08-04 18:00:04 +08:00
|
|
|
self.config.hook.pytest_collectreport(report=rep)
|
2009-03-17 15:35:58 +08:00
|
|
|
if self.shouldstop:
|
2010-05-25 22:52:09 +08:00
|
|
|
raise self.Interrupted(self.shouldstop)
|
2008-08-16 23:26:59 +08:00
|
|
|
|
|
|
|
def filteritems(self, colitems):
|
|
|
|
""" return items to process (some may be deselected)"""
|
2010-07-27 03:15:15 +08:00
|
|
|
keywordexpr = self.config.option.keyword
|
2008-08-16 23:26:59 +08:00
|
|
|
if not keywordexpr or self._nomatch:
|
|
|
|
return colitems
|
2010-07-27 03:15:15 +08:00
|
|
|
if keywordexpr[-1] == ":":
|
2008-08-16 23:26:59 +08:00
|
|
|
keywordexpr = keywordexpr[:-1]
|
|
|
|
remaining = []
|
|
|
|
deselected = []
|
|
|
|
for colitem in colitems:
|
|
|
|
if isinstance(colitem, Item):
|
|
|
|
if colitem._skipbykeyword(keywordexpr):
|
|
|
|
deselected.append(colitem)
|
|
|
|
continue
|
|
|
|
remaining.append(colitem)
|
2010-07-27 03:15:15 +08:00
|
|
|
if deselected:
|
2009-05-08 00:01:53 +08:00
|
|
|
self.config.hook.pytest_deselected(items=deselected)
|
2008-08-16 23:26:59 +08:00
|
|
|
if self.config.option.keyword.endswith(":"):
|
|
|
|
self._nomatch = True
|
2010-07-27 03:15:15 +08:00
|
|
|
return remaining
|
2008-08-16 23:26:59 +08:00
|
|
|
|
2010-07-27 03:15:15 +08:00
|
|
|
def collect(self, colitems):
|
2008-08-16 23:26:59 +08:00
|
|
|
keyword = self.config.option.keyword
|
|
|
|
for x in self.genitems(colitems, keyword):
|
|
|
|
yield x
|
|
|
|
|
|
|
|
def sessionstarts(self):
|
|
|
|
""" setup any neccessary resources ahead of the test run. """
|
2009-05-23 03:53:26 +08:00
|
|
|
self.config.hook.pytest_sessionstart(session=self)
|
2010-07-27 03:15:15 +08:00
|
|
|
|
2009-08-04 18:00:04 +08:00
|
|
|
def pytest_runtest_logreport(self, report):
|
|
|
|
if report.failed:
|
2010-05-25 22:52:09 +08:00
|
|
|
self._testsfailed += 1
|
|
|
|
maxfail = self.config.getvalue("maxfail")
|
|
|
|
if maxfail and self._testsfailed >= maxfail:
|
|
|
|
self.shouldstop = "stopping after %d failures" % (
|
|
|
|
self._testsfailed)
|
2009-06-09 00:31:10 +08:00
|
|
|
pytest_collectreport = pytest_runtest_logreport
|
2008-08-16 23:26:59 +08:00
|
|
|
|
2009-07-20 20:01:40 +08:00
|
|
|
def sessionfinishes(self, exitstatus):
|
2010-07-27 03:15:15 +08:00
|
|
|
""" teardown any resources after a test run. """
|
2009-05-23 03:53:26 +08:00
|
|
|
self.config.hook.pytest_sessionfinish(
|
2010-07-27 03:15:15 +08:00
|
|
|
session=self,
|
|
|
|
exitstatus=exitstatus,
|
2009-04-09 07:41:35 +08:00
|
|
|
)
|
2008-08-16 23:26:59 +08:00
|
|
|
|
2010-01-11 04:29:36 +08:00
|
|
|
def main(self, colitems):
|
2008-08-16 23:26:59 +08:00
|
|
|
""" main loop for running tests. """
|
2010-07-27 03:15:15 +08:00
|
|
|
self.shouldstop = False
|
2008-08-16 23:26:59 +08:00
|
|
|
self.sessionstarts()
|
2010-04-28 03:13:09 +08:00
|
|
|
exitstatus = EXIT_OK
|
2008-08-16 23:26:59 +08:00
|
|
|
try:
|
2010-04-26 23:56:39 +08:00
|
|
|
self._mainloop(colitems)
|
|
|
|
if self._testsfailed:
|
2010-04-28 03:13:09 +08:00
|
|
|
exitstatus = EXIT_TESTSFAILED
|
2010-04-26 23:56:39 +08:00
|
|
|
self.sessionfinishes(exitstatus=exitstatus)
|
2008-08-16 23:26:59 +08:00
|
|
|
except KeyboardInterrupt:
|
2009-07-20 20:01:40 +08:00
|
|
|
excinfo = py.code.ExceptionInfo()
|
2009-07-22 20:39:09 +08:00
|
|
|
self.config.hook.pytest_keyboard_interrupt(excinfo=excinfo)
|
2010-04-28 03:13:09 +08:00
|
|
|
exitstatus = EXIT_INTERRUPTED
|
2008-08-16 23:26:59 +08:00
|
|
|
except:
|
2009-07-20 20:01:40 +08:00
|
|
|
excinfo = py.code.ExceptionInfo()
|
2010-04-26 23:56:39 +08:00
|
|
|
self.config.pluginmanager.notify_exception(excinfo)
|
2010-04-28 03:13:09 +08:00
|
|
|
exitstatus = EXIT_INTERNALERROR
|
|
|
|
if exitstatus in (EXIT_INTERNALERROR, EXIT_INTERRUPTED):
|
2010-04-26 23:56:39 +08:00
|
|
|
self.sessionfinishes(exitstatus=exitstatus)
|
2008-08-16 23:26:59 +08:00
|
|
|
return exitstatus
|
2010-04-26 23:56:39 +08:00
|
|
|
|
|
|
|
def _mainloop(self, colitems):
|
2010-07-27 03:15:15 +08:00
|
|
|
for item in self.collect(colitems):
|
|
|
|
if not self.config.option.collectonly:
|
2010-04-26 23:56:39 +08:00
|
|
|
item.config.hook.pytest_runtest_protocol(item=item)
|
2010-05-25 22:52:09 +08:00
|
|
|
if self.shouldstop:
|
|
|
|
raise self.Interrupted(self.shouldstop)
|
|
|
|
|