/* * 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