/*
* FindBugs - Find bugs in Java programs
* Copyright (C) 2004, Garvin LeClaire <garvin.leclaire@insightbb.com>
*
* 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;
import java.io.IOException;
import java.io.Writer;
import java.util.Iterator;
import javax.annotation.Nonnull;
import org.dom4j.Document;
import org.dom4j.DocumentHelper;
import org.dom4j.Element;
import org.dom4j.io.OutputFormat;
import org.dom4j.io.XMLWriter;
import edu.umd.cs.findbugs.classfile.ClassDescriptor;
import edu.umd.cs.findbugs.core.Priorities;
/**
* BugReporter to output warnings in xdocs format for Maven.
*
* @author Garvin LeClaire
*/
public class XDocsBugReporter extends TextUIBugReporter {
final private SortedBugCollection bugCollection;
final private Project project;
private Document document;
private Element root;
private static final String ROOT_ELEMENT_NAME = "BugCollection";
private static final String PROJECT_ELEMENT_NAME = "Project";
private static final String ERRORS_ELEMENT_NAME = "Errors";
private static final String ANALYSIS_ERROR_ELEMENT_NAME = "AnalysisError";
private static final String MISSING_CLASS_ELEMENT_NAME = "MissingClass";
private static final String SUMMARY_HTML_ELEMENT_NAME = "SummaryHTML";
private static final String ELEMENT_NAME = "BugInstance";
private static final String FILE_ELEMENT_NAME = "file";
public XDocsBugReporter(Project project) {
this.project = project;
this.bugCollection = new SortedBugCollection(project);
this.document = DocumentHelper.createDocument();
this.root = document.addElement(ROOT_ELEMENT_NAME);
}
public void observeClass(ClassDescriptor classDescriptor) {
}
@Override
public void logError(String message) {
bugCollection.addError(message);
super.logError(message);
}
@Override
public void reportMissingClass(ClassNotFoundException ex) {
String missing = AbstractBugReporter.getMissingClassName(ex);
if (!isValidMissingClassMessage(missing)) {
return;
}
bugCollection.addMissingClass(missing);
super.reportMissingClass(ex);
}
@Override
public void doReportBug(BugInstance bugInstance) {
if (bugCollection.add(bugInstance)) {
printBug(bugInstance);
notifyObservers(bugInstance);
}
}
@Override
protected void printBug(BugInstance bugInstance) {
try {
toElement(bugInstance);
} catch (Exception e) {
logError("Couldn't add Element", e);
}
}
public void finish() {
try {
writeXML(outputStream, project);
} catch (Exception e) {
logError("Couldn't write XML output", e);
}
outputStream.flush();
}
private void writeXML(Writer out, Project project) throws IOException {
Document document = endDocument(project);
XMLWriter writer = new XMLWriter(out, OutputFormat.createPrettyPrint());
writer.write(document);
}
private Document endDocument(Project project) {
// Save the error information
Element errorsElement = root.addElement(ERRORS_ELEMENT_NAME);
for (Iterator<AnalysisError> i = bugCollection.errorIterator(); i.hasNext();) {
AnalysisError analysisError = i.next();
errorsElement.addElement(ANALYSIS_ERROR_ELEMENT_NAME).setText(analysisError.getMessage());
}
for (Iterator<String> i = bugCollection.missingClassIterator(); i.hasNext();) {
errorsElement.addElement(MISSING_CLASS_ELEMENT_NAME).setText(i.next());
}
return document;
}
private static String xmlEscape(String theString) {
// Replaces characters '>', '<', '"', '&', ''' with XML equivalents
StringBuilder buf = new StringBuilder();
int len = theString.length();
char theChar;
for (int i = 0; i < len; i++) {
theChar = theString.charAt(i);
switch (theChar) {
case '>':
buf.append(">");
break;
case '<':
buf.append("<");
break;
case '"':
buf.append(""");
break;
case '&':
buf.append("&");
break;
case '\'':
buf.append("'");
break;
default:
buf.append(theChar);
}
}
return buf.toString();
}
public void toElement(BugInstance bugInstance) {
String className = bugInstance.getPrimaryClass().getClassName();
Element element = (Element) root.selectSingleNode(FILE_ELEMENT_NAME + "[@classname='" + className + "']");
if (element == null) {
element = root.addElement(FILE_ELEMENT_NAME);
element.addAttribute("classname", className);
}
element = element.addElement(ELEMENT_NAME);
element.addAttribute("type", bugInstance.getType());
switch (bugInstance.getPriority()) {
case Priorities.EXP_PRIORITY:
element.addAttribute("priority", "Experimental");
break;
case Priorities.LOW_PRIORITY:
element.addAttribute("priority", "Low");
break;
case Priorities.NORMAL_PRIORITY:
element.addAttribute("priority", "Normal");
break;
case Priorities.HIGH_PRIORITY:
element.addAttribute("priority", "High");
break;
}
element.addAttribute("message", xmlEscape(bugInstance.getMessage()));
SourceLineAnnotation line = bugInstance.getPrimarySourceLineAnnotation();
if (line == null) {
element.addAttribute("line", "0");
} else {
element.addAttribute("line", Integer.toString(line.getStartLine()));
}
}
/*
* public static void main(String args[]) { String x =
* "Less than: < Greater than: > Ampersand: & Quotation mark: \" Apostrophe: '"
* ; String y = xmlEscape(x); System.out.println(x); System.out.println(y);
* }
*/
public @Nonnull
BugCollection getBugCollection() {
return bugCollection;
}
}
// vim:ts=3