]> arthur.barton.de Git - bup.git/blob - test/ext/conftest.py
Teach pytest to handle WVSKIP and use it where we can
[bup.git] / test / ext / conftest.py
1
2 from subprocess import CalledProcessError
3 import pytest, subprocess, sys
4
5 from bup.compat import fsdecode
6 from bup.io import byte_stream
7
8 # Handle all test-* files as wvtest protocol subprocesses
9 # cf. https://docs.pytest.org/en/latest/example/nonpython.html
10
11 class BupSubprocFailure(Exception):
12     def __init__(self, msg, cmd, status, failures):
13         super(BupSubprocFailure, self).__init__(msg)
14         self.cmd = cmd
15         self.status = status
16         self.failures = failures
17
18 class BupSubprocTestRunner(pytest.Item):
19
20     def __init__(self, name, parent):
21         super(BupSubprocTestRunner, self).__init__(name, parent)
22
23     def runtest(self):
24         cmd = str(self.fspath)
25         p = subprocess.Popen(cmd,
26                              stdout=subprocess.PIPE,
27                              stderr=subprocess.STDOUT)
28         out = p.communicate()[0]
29         sys.stdout.flush()
30         byte_stream(sys.stdout).write(out)
31         lines = out.splitlines()
32         for line in lines:
33             if line.startswith(b'!') and line.lower().endswith(b' skip ok'):
34                 pytest.skip(line.decode('ascii'))
35                 return
36         failures = [line for line in lines
37                     if (line.startswith(b'!')
38                         and line.lower().endswith(b' failed'))]
39         if failures or p.returncode != 0:
40             raise BupSubprocFailure('%s failed (exit %d, %d failures)'
41                                     % (cmd, p.returncode, len(failures)),
42                                     cmd, p.returncode, failures)
43
44     def repr_failure(self, excinfo):
45         ex = excinfo.value
46         if isinstance(ex, BupSubprocFailure):
47             msg = ['Exit status: %d' % ex.status,
48                    'Failures:']
49             msg.extend(fsdecode(s) for s in ex.failures)
50             return '\n'.join(msg)
51
52     def reportinfo(self):
53         # This does not appear to be documented, but is in the
54         # example, and sets the final report header line (at least)
55         # for failures.
56         test_name = str(self.fspath)
57         linenum = None
58         return self.fspath, linenum, test_name
59
60 class BupSubprocTestFile(pytest.File):
61     def collect(self):
62         name = self.fspath.basename
63         # name='' because there's only one test: running the command.
64         # i.e there are no sub-tests.  Otherwise the status messages
65         # duplicate the test name like this:
66         #   test/ext/test-cat-file.sh::test-cat-file.sh PASSED ...
67         try:
68             yield BupSubprocTestRunner.from_parent(self, name='')
69         except AttributeError:
70             yield BupSubprocTestRunner('', self)
71
72 def pytest_collect_file(parent, path):
73     base = path.basename
74     if base.startswith('test-') and not base.endswith('~'):
75         try:
76             item = BupSubprocTestFile.from_parent(parent, fspath=path)
77         except AttributeError:
78             item = BupSubprocTestFile(path, parent)
79         if base == 'test-release-archive':
80             item.add_marker(pytest.mark.release)
81         return item