mockit.coverage.CodeCoverage Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jmockit-coverage Show documentation
Show all versions of jmockit-coverage Show documentation
JMockit Coverage is a code coverage tool with several metrics (line, path, data) capable of generating HTML
reports. It is designed with ease of use in mind, avoiding the need for complex configuration.
Instead, smart (but overridable) defaults are employed, such as the selection of which classes to consider for
coverage, and where to find sources files for report generation.
/*
* Copyright (c) 2006-2014 Rogério Liesenfeld
* This file is subject to the terms of the MIT license (see LICENSE.txt).
*/
package mockit.coverage;
import java.lang.instrument.*;
import java.security.*;
import org.jetbrains.annotations.*;
import mockit.coverage.data.*;
import mockit.coverage.modification.*;
import mockit.coverage.standalone.*;
public final class CodeCoverage implements ClassFileTransformer
{
private static CodeCoverage instance;
private final ClassModification classModification;
private final OutputFileGenerator outputGenerator;
private boolean inactive;
public static void main(String[] args)
{
OutputFileGenerator generator = createOutputFileGenerator(null);
generator.generateAggregateReportFromInputFiles(args);
}
private static OutputFileGenerator createOutputFileGenerator(@Nullable ClassModification classModification)
{
OutputFileGenerator generator = new OutputFileGenerator(classModification);
CoverageData.instance().setWithCallPoints(generator.isWithCallPoints());
return generator;
}
@SuppressWarnings("unused")
public CodeCoverage() { this(true, true); }
private CodeCoverage(boolean checkIfAlreadyInitialized, final boolean generateOutputOnShutdown)
{
if (checkIfAlreadyInitialized && Startup.isInitialized()) {
throw new IllegalStateException("JMockit: coverage tool already initialized");
}
else if (
generateOutputOnShutdown &&
("none".equals(Configuration.getProperty("output")) ||
"none".equals(Configuration.getProperty("classes")) ||
"none".equals(Configuration.getProperty("metrics")))
) {
throw new IllegalStateException("JMockit: coverage tool disabled");
}
classModification = new ClassModification();
outputGenerator = createOutputFileGenerator(classModification);
instance = this;
Runtime.getRuntime().addShutdownHook(new Thread() {
@Override
public void run()
{
TestRun.terminate();
if (generateOutputOnShutdown) {
if (outputGenerator.isOutputToBeGenerated()) {
outputGenerator.generate(CodeCoverage.this);
}
new CoverageCheck().verifyThresholds();
}
Startup.instrumentation().removeTransformer(CodeCoverage.this);
}
});
}
@NotNull
public static CodeCoverage create(boolean generateOutputOnShutdown)
{
instance = new CodeCoverage(false, generateOutputOnShutdown);
return instance;
}
public static void resetConfiguration()
{
Startup.instrumentation().removeTransformer(instance);
CoverageData.instance().clear();
Startup.instrumentation().addTransformer(create(false));
}
public static void generateOutput(boolean resetState)
{
instance.outputGenerator.generate(null);
if (resetState) {
CoverageData.instance().reset();
}
}
@Override
@Nullable
public byte[] transform(
@Nullable ClassLoader loader, @NotNull String internalClassName, @Nullable Class> classBeingRedefined,
@Nullable ProtectionDomain protectionDomain, @NotNull byte[] originalClassfile)
{
if (loader == null || classBeingRedefined != null || protectionDomain == null || inactive) {
return null;
}
String className = internalClassName.replace('/', '.');
return classModification.modifyClass(className, protectionDomain, originalClassfile);
}
void deactivate() { inactive = true; }
}