mirror of
https://github.com/tahoe-lafs/tahoe-lafs.git
synced 2025-01-04 04:04:10 +00:00
111 lines
4.0 KiB
Python
111 lines
4.0 KiB
Python
|
|
||
|
"""A Trial IReporter plugin that gathers coverage.py code-coverage information.
|
||
|
|
||
|
Once this plugin is installed, trial can be invoked a new --reporter option:
|
||
|
|
||
|
trial --reporter-bwverbose-coverage ARGS
|
||
|
|
||
|
Once such a test run has finished, there will be a .coverage file in the
|
||
|
top-level directory. This file can be turned into a directory of .html files
|
||
|
(with index.html as the starting point) by running:
|
||
|
|
||
|
coverage html -d OUTPUTDIR --omit=PREFIX1,PREFIX2,..
|
||
|
|
||
|
The 'coverage' tool thinks in terms of absolute filenames. 'coverage' doesn't
|
||
|
record data for files that come with Python, but it does record data for all
|
||
|
the various site-package directories. To show only information for Tahoe
|
||
|
source code files, you should provide --omit prefixes for everything else.
|
||
|
This probably means something like:
|
||
|
|
||
|
--omit=/System/,/Library/,support/,src/allmydata/test/
|
||
|
|
||
|
Before using this, you need to install the 'coverage' package, which will
|
||
|
provide an executable tool named 'coverage' (as well as an importable
|
||
|
library). 'coverage report' will produce a basic text summary of the coverage
|
||
|
data. Our 'misc/coverage2text.py' tool produces a slightly more useful
|
||
|
summary, and 'misc/coverage2html.py' will produce a more useful HTML report.
|
||
|
|
||
|
"""
|
||
|
|
||
|
from twisted.trial.reporter import TreeReporter, VerboseTextReporter
|
||
|
|
||
|
# These plugins are registered via twisted/plugins/allmydata_trial.py . See
|
||
|
# the notes there for an explanation of how that works.
|
||
|
|
||
|
# Some notes about how trial Reporters are used:
|
||
|
# * Reporters don't really get told about the suite starting and stopping.
|
||
|
# * The Reporter class is imported before the test classes are.
|
||
|
# * The test classes are imported before the Reporter is created. To get
|
||
|
# control earlier than that requires modifying twisted/scripts/trial.py
|
||
|
# * Then Reporter.__init__ is called.
|
||
|
# * Then tests run, calling things like write() and addSuccess(). Each test is
|
||
|
# framed by a startTest/stopTest call.
|
||
|
# * Then the results are emitted, calling things like printErrors,
|
||
|
# printSummary, and wasSuccessful.
|
||
|
# So for code-coverage (not including import), start in __init__ and finish
|
||
|
# in printSummary. To include import, we have to start in our own import and
|
||
|
# finish in printSummary.
|
||
|
|
||
|
import coverage
|
||
|
cov = coverage.coverage()
|
||
|
cov.start()
|
||
|
|
||
|
|
||
|
class CoverageTextReporter(VerboseTextReporter):
|
||
|
def __init__(self, *args, **kwargs):
|
||
|
VerboseTextReporter.__init__(self, *args, **kwargs)
|
||
|
|
||
|
def stop_coverage(self):
|
||
|
cov.stop()
|
||
|
cov.save()
|
||
|
print "Coverage results written to .coverage"
|
||
|
def printSummary(self):
|
||
|
# for twisted-2.5.x
|
||
|
self.stop_coverage()
|
||
|
return VerboseTextReporter.printSummary(self)
|
||
|
def done(self):
|
||
|
# for twisted-8.x
|
||
|
self.stop_coverage()
|
||
|
return VerboseTextReporter.done(self)
|
||
|
|
||
|
class sample_Reporter(object):
|
||
|
# this class, used as a reporter on a fully-passing test suite, doesn't
|
||
|
# trigger exceptions. So it is a guide to what methods are invoked on a
|
||
|
# Reporter.
|
||
|
def __init__(self, *args, **kwargs):
|
||
|
print "START HERE"
|
||
|
self.r = TreeReporter(*args, **kwargs)
|
||
|
self.shouldStop = self.r.shouldStop
|
||
|
self.separator = self.r.separator
|
||
|
self.testsRun = self.r.testsRun
|
||
|
self._starting2 = False
|
||
|
|
||
|
def write(self, *args):
|
||
|
if not self._starting2:
|
||
|
self._starting2 = True
|
||
|
print "FIRST WRITE"
|
||
|
return self.r.write(*args)
|
||
|
|
||
|
def startTest(self, *args, **kwargs):
|
||
|
return self.r.startTest(*args, **kwargs)
|
||
|
|
||
|
def stopTest(self, *args, **kwargs):
|
||
|
return self.r.stopTest(*args, **kwargs)
|
||
|
|
||
|
def addSuccess(self, *args, **kwargs):
|
||
|
return self.r.addSuccess(*args, **kwargs)
|
||
|
|
||
|
def printErrors(self, *args, **kwargs):
|
||
|
return self.r.printErrors(*args, **kwargs)
|
||
|
|
||
|
def writeln(self, *args, **kwargs):
|
||
|
return self.r.writeln(*args, **kwargs)
|
||
|
|
||
|
def printSummary(self, *args, **kwargs):
|
||
|
print "PRINT SUMMARY"
|
||
|
return self.r.printSummary(*args, **kwargs)
|
||
|
|
||
|
def wasSuccessful(self, *args, **kwargs):
|
||
|
return self.r.wasSuccessful(*args, **kwargs)
|
||
|
|