edu.umd.cs.findbugs.workflow.SetBugDatabaseInfo Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of spotbugs Show documentation
Show all versions of spotbugs Show documentation
SpotBugs: Because it's easy!
/*
* FindBugs - Find bugs in Java programs
* Copyright (C) 2003-2005 William Pugh
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
package edu.umd.cs.findbugs.workflow;
import java.io.File;
import java.io.IOException;
import java.util.Date;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.dom4j.DocumentException;
import edu.umd.cs.findbugs.BugInstance;
import edu.umd.cs.findbugs.DetectorFactoryCollection;
import edu.umd.cs.findbugs.FindBugs;
import edu.umd.cs.findbugs.PackageStats;
import edu.umd.cs.findbugs.Project;
import edu.umd.cs.findbugs.SortedBugCollection;
import edu.umd.cs.findbugs.SourceLineAnnotation;
import edu.umd.cs.findbugs.config.CommandLine;
import edu.umd.cs.findbugs.filter.Filter;
/**
* Java main application to compute update a historical bug collection with
* results from another build/analysis.
*
* @author William Pugh
*/
public class SetBugDatabaseInfo {
/**
*
*/
private static final String USAGE = "Usage: " + " [options] [ []]";
static class SetInfoCommandLine extends CommandLine {
String revisionName;
String projectName;
String exclusionFilterFile;
String lastVersion;
boolean withMessages = false;
boolean purgeStats = false;
boolean purgeClassStats = false;
boolean purgeMissingClasses = false;
boolean resetSource = false;
boolean resetProject = false;
long revisionTimestamp = 0L;
public List sourcePaths = new LinkedList<>();
public List searchSourcePaths = new LinkedList<>();
SetInfoCommandLine() {
addOption("-name", "name", "set name for (last) revision");
addOption("-projectName", "name", "set name for project");
addOption("-timestamp", "when", "set timestamp for (last) revision");
addSwitch("-resetSource", "remove all source search paths");
addSwitch("-resetProject", "remove all source search paths, analysis and auxiliary classpath entries");
addOption("-source", "directory", "Add this directory to the source search path");
addSwitch("-purgeStats", "purge/delete information about sizes of analyzed class files");
addSwitch("-purgeClassStats", "purge/delete information about sizes of analyzed class files, but retain class stats");
addSwitch("-purgeMissingClasses", "purge list of missing classes");
addOption("-findSource", "directory", "Find and add all relevant source directions contained within this directory");
addOption("-suppress", "filter file", "Suppress warnings matched by this file (replaces previous suppressions)");
addOption("-lastVersion", "version", "Trim the history to just include just the specified version");
addSwitch("-withMessages", "Add bug descriptions");
}
@Override
protected void handleOption(String option, String optionExtraPart) throws IOException {
if ("-withMessages".equals(option)) {
withMessages = true;
} else if ("-resetSource".equals(option)) {
resetSource = true;
} else if ("-resetProject".equals(option)) {
resetProject = true;
} else if ("-purgeStats".equals(option)) {
purgeStats = true;
} else if ("-purgeClassStats".equals(option)) {
purgeClassStats = true;
} else if ("-purgeMissingClasses".equals(option)) {
purgeMissingClasses = true;
} else {
throw new IllegalArgumentException("no option " + option);
}
}
@Override
protected void handleOptionWithArgument(String option, String argument) throws IOException {
if ("-name".equals(option)) {
revisionName = argument;
} else if ("-projectName".equals(option)) {
projectName = argument;
} else if ("-suppress".equals(option)) {
exclusionFilterFile = argument;
} else if ("-timestamp".equals(option)) {
revisionTimestamp = Date.parse(argument);
} else if ("-source".equals(option)) {
sourcePaths.add(argument);
} else if ("-lastVersion".equals(option)) {
lastVersion = argument;
} else if ("-findSource".equals(option)) {
searchSourcePaths.add(argument);
} else {
throw new IllegalArgumentException("Can't handle option " + option);
}
}
}
public static void main(String[] args) throws IOException, DocumentException {
FindBugs.setNoAnalysis();
DetectorFactoryCollection.instance();
SetInfoCommandLine commandLine = new SetInfoCommandLine();
int argCount = commandLine.parse(args, 0, 2, USAGE);
SortedBugCollection origCollection = new SortedBugCollection();
if (argCount < args.length) {
origCollection.readXML(args[argCount++]);
} else {
origCollection.readXML(System.in);
}
Project project = origCollection.getProject();
if (commandLine.revisionName != null) {
origCollection.setReleaseName(commandLine.revisionName);
}
if (commandLine.projectName != null) {
origCollection.getProject().setProjectName(commandLine.projectName);
}
if (commandLine.revisionTimestamp != 0) {
origCollection.setTimestamp(commandLine.revisionTimestamp);
}
origCollection.setWithMessages(commandLine.withMessages);
if (commandLine.exclusionFilterFile != null) {
project.setSuppressionFilter(Filter.parseFilter(commandLine.exclusionFilterFile));
}
if (commandLine.resetProject) {
project.getSourceDirList().clear();
project.getFileList().clear();
project.getAuxClasspathEntryList().clear();
}
if (commandLine.resetSource) {
project.getSourceDirList().clear();
}
project.addSourceDirs(commandLine.sourcePaths);
if (commandLine.purgeStats) {
origCollection.getProjectStats().getPackageStats().clear();
}
if (commandLine.purgeClassStats) {
for (PackageStats ps : origCollection.getProjectStats().getPackageStats()) {
ps.getClassStats().clear();
}
}
if (commandLine.purgeMissingClasses) {
origCollection.clearMissingClasses();
}
if (commandLine.lastVersion != null) {
long last = edu.umd.cs.findbugs.workflow.Filter.FilterCommandLine.getVersionNum(origCollection,
commandLine.lastVersion, true);
if (last < origCollection.getSequenceNumber()) {
String name = origCollection.getAppVersionFromSequenceNumber(last).getReleaseName();
long timestamp = origCollection.getAppVersionFromSequenceNumber(last).getTimestamp();
origCollection.setReleaseName(name);
origCollection.setTimestamp(timestamp);
origCollection.trimAppVersions(last);
}
}
Map> missingFiles = new HashMap<>();
if (!commandLine.searchSourcePaths.isEmpty()) {
sourceSearcher = new SourceSearcher(project);
for (BugInstance bug : origCollection.getCollection()) {
SourceLineAnnotation src = bug.getPrimarySourceLineAnnotation();
if (!sourceSearcher.sourceNotFound.contains(src.getClassName()) && !sourceSearcher.findSource(src)) {
Set paths = missingFiles.computeIfAbsent(src.getSourceFile(), k -> new HashSet<>());
String fullPath = fullPath(src);
// System.out.println("Missing " + fullPath);
paths.add(fullPath);
}
}
Set foundPaths = new HashSet<>();
for (String f : commandLine.searchSourcePaths) {
for (File javaFile : RecursiveSearchForJavaFiles.search(new File(f))) {
Set matchingMissingClasses = missingFiles.get(javaFile.getName());
if (matchingMissingClasses == null) {
// System.out.println("Nothing for " + javaFile);
} else {
for (String sourcePath : matchingMissingClasses) {
String path = javaFile.getAbsolutePath();
if (path.endsWith(sourcePath)) {
String dir = path.substring(0, path.length() - sourcePath.length());
foundPaths.add(dir);
}
}
}
}
}
Set toRemove = new HashSet<>();
for (String p1 : foundPaths) {
for (String p2 : foundPaths) {
if (!p1.equals(p2) && p1.startsWith(p2)) {
toRemove.add(p1);
break;
}
}
}
foundPaths.removeAll(toRemove);
project.addSourceDirs(foundPaths);
for (String dir : foundPaths) {
if (argCount < args.length) {
System.out.println("Found " + dir);
}
}
}
if (argCount < args.length) {
origCollection.writeXML(args[argCount++]);
} else {
origCollection.writeXML(System.out);
}
}
static String fullPath(SourceLineAnnotation src) {
return src.getPackageName().replace('.', File.separatorChar) + File.separatorChar + src.getSourceFile();
}
static SourceSearcher sourceSearcher;
}