/hg/MauveTestCoverage: * src/ClassInfo.java:

ptisnovs at icedtea.classpath.org ptisnovs at icedtea.classpath.org
Fri Mar 23 03:53:28 PDT 2012


changeset eec2474fdaaa in /hg/MauveTestCoverage
details: http://icedtea.classpath.org/hg/MauveTestCoverage?cmd=changeset;node=eec2474fdaaa
author: Pavel Tisnovsky <ptisnovs at redhat.com>
date: Fri Mar 23 11:55:40 2012 +0100

	* src/ClassInfo.java:
		Added new helper class.
		* src/FileUtils.java:
		Added new method for reading contents of text file.
		Refactored.
		* src/ReportGenerator.java:
		New functionality: methods coverage are printed
		in package list (previously only class coverage
		were printed).
		Refactored.
		* templates/all_packages_template.html:
		* templates/index.html:
		Changed width of left column.
		* templates/style.css:
		Tables have black borders (they are more visible).
		* Makefile:
		Updated according to previous changes.


diffstat:

 ChangeLog                            |   20 +++
 Makefile                             |    3 +-
 src/ClassInfo.java                   |  176 +++++++++++++++++++++++++++++++++++
 src/FileUtils.java                   |   25 ++++-
 src/ReportGenerator.java             |  169 ++++++++++++++------------------
 templates/all_packages_template.html |    3 +-
 templates/index.html                 |    2 +-
 templates/style.css                  |    8 +-
 8 files changed, 302 insertions(+), 104 deletions(-)

diffs (truncated from 670 to 500 lines):

diff -r 8c9c61f17fd1 -r eec2474fdaaa ChangeLog
--- a/ChangeLog	Fri Mar 16 11:24:48 2012 +0100
+++ b/ChangeLog	Fri Mar 23 11:55:40 2012 +0100
@@ -1,3 +1,23 @@
+2012-03-23  Pavel Tisnovsky  <ptisnovs at redhat.com>
+
+	* src/ClassInfo.java:
+	Added new helper class.
+	* src/FileUtils.java:
+	Added new method for reading contents of text file.
+	Refactored.
+	* src/ReportGenerator.java:
+	New functionality: methods coverage are printed
+	in package list (previously only class coverage
+	were printed).
+	Refactored.
+	* templates/all_packages_template.html:
+	* templates/index.html:
+	Changed width of left column.
+	* templates/style.css:
+	Tables have black borders (they are more visible).
+	* Makefile:
+	Updated according to previous changes.
+
 2012-03-16  Pavel Tisnovsky  <ptisnovs at redhat.com>
 
 	* src/ReportGenerator.java:
diff -r 8c9c61f17fd1 -r eec2474fdaaa Makefile
--- a/Makefile	Fri Mar 16 11:24:48 2012 +0100
+++ b/Makefile	Fri Mar 23 11:55:40 2012 +0100
@@ -74,7 +74,8 @@
 	$(CLASSDIR)/PrintPublicMethods.class \
 	$(CLASSDIR)/PrintTestCoverage.class \
 	$(CLASSDIR)/ReportGenerator.class \
-	$(CLASSDIR)/FileUtils.class
+	$(CLASSDIR)/FileUtils.class \
+	$(CLASSDIR)/ClassInfo.class
 
 api_class_list:	$(REPORTDIR) $(REPORTDIR)/$(ALL_CLASS_LIST)
 
diff -r 8c9c61f17fd1 -r eec2474fdaaa src/ClassInfo.java
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/ClassInfo.java	Fri Mar 23 11:55:40 2012 +0100
@@ -0,0 +1,176 @@
+/*
+  Test coverage tool.
+
+   Copyright (C) 2012 Red Hat
+
+This tool is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+This tool 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
+General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this tool; see the file COPYING.  If not, write to the
+Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+02110-1301 USA.
+
+Linking this library statically or dynamically with other modules is
+making a combined work based on this library.  Thus, the terms and
+conditions of the GNU General Public License cover the whole
+combination.
+
+As a special exception, the copyright holders of this library give you
+permission to link this library with independent modules to produce an
+executable, regardless of the license terms of these independent
+modules, and to copy and distribute the resulting executable under
+terms of your choice, provided that you also meet, for each linked
+independent module, the terms and conditions of the license of that
+module.  An independent module is a module which is not derived from
+or based on this library.  If you modify this library, you may extend
+this exception to your version of the library, but you are not
+obligated to do so.  If you do not wish to do so, delete this
+exception statement from your version.
+*/
+
+import java.io.File;
+import java.util.List;
+import java.util.Set;
+import java.util.TreeSet;
+
+
+
+/**
+ * Instances of this class contain information about one tested class.
+ * There are three sets of strings stored for each class: set of methods
+ * contained in standard API, set of methods called from tests and the
+ * union of previous two sets. These information are used by test reporter.
+ * 
+ * @author Pavel Tisnovsky <ptisnovs at redhat.com>
+ */
+public class ClassInfo
+{
+    /**
+     * Set of methods contained in standard API.
+     */
+    private Set<String> apiMethods;
+
+    /**
+     * Set of methods called from tests.
+     */
+    private Set<String> testedMethods;
+
+    /**
+     * Union of apiMethods and testedMethods.
+     */
+    private Set<String> allMethods;
+
+    /**
+     * Constructor. It tries to read all required information
+     * for tested class.
+     * 
+     * @param reportDirectory
+     *            directory where all reports are stored
+     * @param className
+     *            name of tested class
+     */
+    public ClassInfo(String reportDirectory, String className)
+    {
+        // read methods described in standard API
+        this.apiMethods = readApiMethods(reportDirectory, className);
+        // read methods called from tests
+        this.testedMethods = readTestedMethods(reportDirectory, className);
+        // compute union of previous two sets
+        computeAllMethodsSet();
+    }
+
+    /**
+     * Compute union of allMethods and testedMethods sets.
+     */
+    private void computeAllMethodsSet()
+    {
+        this.allMethods = new TreeSet<String>();
+        this.allMethods.addAll(this.apiMethods);
+        this.allMethods.addAll(this.testedMethods);
+    }
+
+    /**
+     * Getter for a set apiMethods.
+     *
+     * @return the apiMethods attribute
+     */
+    public Set<String> getApiMethods()
+    {
+        return this.apiMethods;
+    }
+
+    /**
+     * Getter for a set testedMethods.
+     *
+     * @return the testedMethods attribute
+     */
+    public Set<String> getTestedMethods()
+    {
+        return this.testedMethods;
+    }
+
+    /**
+     * Getter for a set allMethods.
+     *
+     * @return the allMethods attribute
+     */
+    public Set<String> getAllMethods()
+    {
+        return this.allMethods;
+    }
+
+    /**
+     * Read all methods for a given class which are covered by a standard API.
+     * 
+     * @param reportDirectory
+     *            directory where all reports are stored
+     * @param testedClass
+     *            name of tested class
+     * @return set of all methods read from a file containing standard API
+     */
+    private static Set<String> readApiMethods(String reportDirectory, String testedClass)
+    {
+        File fileName = new File(reportDirectory, testedClass + "_api.txt");
+        return readMethods(fileName);
+    }
+
+    /**
+     * Read all methods for a given class which are called from tests.
+     * 
+     * @param reportDirectory
+     *            directory where all reports are stored
+     * @param testedClass
+     *            name of tested class
+     * @return set of all methods read from a file containing called methods
+     */
+    private static Set<String> readTestedMethods(String reportDirectory, String testedClass)
+    {
+        File fileName = new File(reportDirectory, testedClass + "_test.txt");
+        return readMethods(fileName);
+    }
+
+    /**
+     * Read set of method names from a text file. No exception is thrown during
+     * reading.
+     * 
+     * @param fileName
+     *            file containing methods list
+     * @return set of method names
+     */
+    private static Set<String> readMethods(File fileName)
+    {
+        Set<String> allMethods = new TreeSet<String>();
+        List<String> methodList = FileUtils.readTextFile(fileName.getAbsolutePath(), false);
+        allMethods.addAll(methodList);
+        return allMethods;
+    }
+
+}
diff -r 8c9c61f17fd1 -r eec2474fdaaa src/FileUtils.java
--- a/src/FileUtils.java	Fri Mar 16 11:24:48 2012 +0100
+++ b/src/FileUtils.java	Fri Mar 23 11:55:40 2012 +0100
@@ -46,6 +46,7 @@
 import java.io.IOException;
 import java.io.InputStreamReader;
 import java.io.OutputStreamWriter;
+import java.util.Collection;
 import java.util.LinkedList;
 import java.util.List;
 
@@ -76,6 +77,21 @@
      */
     static List<String> readTextFile(String fileName)
     {
+        return readTextFile(fileName, true);
+    }
+
+    /**
+     * Read content of given text file and return it as list of strings. No
+     * exception is thrown during reading.
+     * 
+     * @param fileName
+     *            name of file to be read
+     * @param printFileNotFoundException
+     *            whether to print an exception when file is not found
+     * @return list of string containing content of text file
+     */
+    static List<String> readTextFile(String fileName, boolean printFileNotFoundException)
+    {
         BufferedReader reader = null;
         List<String> out = new LinkedList<String>();
         try
@@ -87,7 +103,12 @@
         catch (FileNotFoundException e)
         {
             // might happen - empty list is returned in this case
-            e.printStackTrace();
+            if (printFileNotFoundException)
+            {
+                // in some cases we don't want to see this exception
+                // (because some classes, for example, could not be tested at all)
+                e.printStackTrace();
+            }
         }
         catch (IOException e)
         {
@@ -215,7 +236,7 @@
      * @throws IOException
      *             thrown if an I/O error occurs
      */
-    private static void readAllLinesFromTextFile(BufferedReader bufferedReader, List<String> lines) throws IOException
+    private static void readAllLinesFromTextFile(BufferedReader bufferedReader, Collection<String> lines) throws IOException
     {
         String line;
         // read lines from a text file
diff -r 8c9c61f17fd1 -r eec2474fdaaa src/ReportGenerator.java
--- a/src/ReportGenerator.java	Fri Mar 16 11:24:48 2012 +0100
+++ b/src/ReportGenerator.java	Fri Mar 23 11:55:40 2012 +0100
@@ -36,14 +36,10 @@
 exception statement from your version.
 */
 
-import java.io.BufferedReader;
-import java.io.File;
-import java.io.FileInputStream;
-import java.io.FileNotFoundException;
-import java.io.IOException;
-import java.io.InputStreamReader;
+import java.util.HashMap;
 import java.util.LinkedList;
 import java.util.List;
+import java.util.Map;
 import java.util.Set;
 import java.util.TreeSet;
 
@@ -58,11 +54,6 @@
 public class ReportGenerator
 {
     /**
-     * Default charset used during all input/output operations.
-     */
-    private static final String DEFAULT_CHARSET = "UTF-8";
-
-    /**
      * Location of internal or external URI to standard API JavaDoc
      */
     private static final String DOC_BASE = "http://docs.oracle.com/javase/6/docs/api";
@@ -97,9 +88,10 @@
      *            set of tested classes
      * @param packageNames
      *            set of package names
+     * @param classInfoMap 
      */
     private static void printPackageListToFile(String reportDirectory, Set<String> allClasses,
-                    Set<String> testedClasses, Set<String> packageNames)
+                    Set<String> testedClasses, Set<String> packageNames, Map<String, ClassInfo> classInfoMap)
     {
         List<String> template = FileUtils.readTextFile("templates/all_packages_template.html");
         List<String> out = new LinkedList<String>();
@@ -109,7 +101,7 @@
             // replace text in template where needed
             if ("${PACKAGE_LIST}".equals(templateLine))
             {
-                addPackageList(allClasses, testedClasses, packageNames, out);
+                addPackageList(allClasses, testedClasses, packageNames, out, classInfoMap);
             }
             // normal line
             else
@@ -132,9 +124,10 @@
      *            package for which the report is generated
      * @param testedClasses
      *            set of tested classes
+     * @param classInfoMap 
      */
     private static void printReportForPackageToFile(String reportDirectory, String packageName,
-                    Set<String> testedClasses)
+                    Set<String> testedClasses, Map<String, ClassInfo> classInfoMap)
     {
         // read HTML template
         List<String> template = FileUtils.readTextFile("templates/package_template.html");
@@ -146,7 +139,7 @@
             // replace text in template where needed
             if ("${CLASS_LIST}".equals(templateLine))
             {
-                addClassList(reportDirectory, packageName, testedClasses, out);
+                addClassList(reportDirectory, packageName, testedClasses, out, classInfoMap);
             }
             else if ("${PACKAGE_NAME}".equals(templateLine))
             {
@@ -173,9 +166,10 @@
      *            all checked package names
      * @param testedClasses
      *            set of tested classes
+     * @param classInfoMap 
      */
     private static void printReportForAllClassesInOneFile(String reportDirectory, Set<String> usedPackageNames,
-                    Set<String> testedClasses)
+                    Set<String> testedClasses, Map<String, ClassInfo> classInfoMap)
     {
         List<String> template = FileUtils.readTextFile("templates/all_classes_template.html");
         List<String> out = new LinkedList<String>();
@@ -185,7 +179,7 @@
             // replace text in template where needed
             if ("${PACKAGE_AND_CLASS_LIST}".equals(templateLine))
             {
-                addPackageAndClassList(reportDirectory, usedPackageNames, testedClasses, out);
+                addPackageAndClassList(reportDirectory, usedPackageNames, testedClasses, out, classInfoMap);
             }
             // normal line
             else
@@ -214,25 +208,52 @@
      */
     @SuppressWarnings("boxing")
     private static void addPackageList(Set<String> allClasses, Set<String> testedClasses, Set<String> packageNames,
-                    List<String> out)
+                    List<String> out, Map<String, ClassInfo> classInfoMap)
     {
         // iterate through all package names
         for (String packageName : packageNames)
         {
             // compute number of all classes in a package
             final int allClassesCnt = numberOfClassesInPackage(packageName, allClasses);
-            // compute number of classes coveraged by tests
+            // compute number of classes covered by tests
             final int testedClassesCnt = numberOfClassesInPackage(packageName, testedClasses);
             // -> in percent
-            final float percentage = 100.0f*testedClassesCnt / allClassesCnt;
+            final float classPercentage = allClassesCnt == 0 ? 0.0f : 100.0f * testedClassesCnt / allClassesCnt;
+            // table row background color is based on percentual test coverage ration
+            String backgroundColor1 = generateTableRowBackground(classPercentage);
 
+            int allMethodsCnt=0;
+            int coveragedMethodsCnt=0;
+            for (String className : allClasses)
+            {
+                // count only classes in given package
+                if (className.startsWith(packageName))
+                {
+                    ClassInfo classInfo = classInfoMap.get(className);
+                    allMethodsCnt += classInfo.getAllMethods().size();
+                    coveragedMethodsCnt += classInfo.getTestedMethods().size();
+                }
+            }
+            float methodsPercentage = allMethodsCnt == 0 ? 0.0f : 100.0f * coveragedMethodsCnt / allMethodsCnt;
             // table row background color is based on percentual test coverage ration
-            String backgroundColor = generateTableRowBackground(percentage);
+            String backgroundColor2 = generateTableRowBackground(methodsPercentage);
+
+            // format output string
             String doc = DOC_BASE + "/" + packageName.replace('.', '/') + "/package-summary.html";
-            // format output string
-            String str = String.format("<tr style='background-color:%s'><td><a target='ClassesListFrame' href='%s.html'>%s</a></td><td style='text-align:right'>%d</td><td style='text-align:right'>%d</td><td style='text-align:right'>%5.1f %%</td><td style='text-align:right'><a href='%s' target='_blank'>ext</a></td></tr>",
-                            backgroundColor, packageName, packageName,
-                            allClassesCnt, testedClassesCnt, percentage, doc);
+
+            String str = String.format(
+                            "<tr><td><a target='ClassesListFrame' href='%s.html'>%s</a></td>" +
+                                 "<td style='background-color:%s;text-align:right'>%d</td>" +
+                                 "<td style='background-color:%s;text-align:right'>%d</td>" +
+                                 "<td style='background-color:%s;text-align:right'>%5.1f %%</td>" +
+                                 "<td style='background-color:%s;text-align:right'>%d</td>" +
+                                 "<td style='background-color:%s;text-align:right'>%d</td>" +
+                                 "<td style='background-color:%s;text-align:right'>%5.1f %%</td>" +
+                                 "<td style='text-align:right'><a href='%s' target='_blank'>ext</a></td></tr>",
+                            packageName, packageName,
+                            backgroundColor1, allClassesCnt, backgroundColor1, testedClassesCnt, backgroundColor1, classPercentage,
+                            backgroundColor2, allMethodsCnt, backgroundColor2, coveragedMethodsCnt, backgroundColor2, methodsPercentage,
+                            doc);
             out.add(str);
         }
     }
@@ -270,7 +291,7 @@
         for (String className : classes)
         {
             // count only classes in given package
-            if (className.startsWith(packageName))
+            if (className.substring(0, className.lastIndexOf('.')).equals(packageName))
             {
                 cnt++;
             }
@@ -288,8 +309,9 @@
      *            set of tested classes
      * @param out
      *            list of string which represents generated report
+     * @param classInfoMap 
      */
-    private static void addClassList(String reportDirectory, String packageName, Set<String> testedClasses, List<String> out)
+    private static void addClassList(String reportDirectory, String packageName, Set<String> testedClasses, List<String> out, Map<String, ClassInfo> classInfoMap)
     {
         // iterate through all class names
         for (String className : testedClasses)
@@ -298,7 +320,7 @@
             if (className.startsWith(packageName))
             {
                 //out.add("<a target='ResultsFrame' href='" + className + ".html'>" + className + "</a><br>");
-                out.add(addOneRowToResultsTable(reportDirectory, className));
+                out.add(addOneRowToResultsTable(reportDirectory, className, classInfoMap));
             }
         }
     }
@@ -315,8 +337,9 @@
      *            set of tested classes
      * @param out
      *            list of string which represents generated report
+     * @param classInfoMap 
      */
-    private static void addPackageAndClassList(String reportDirectory, Set<String> usedPackageNames, Set<String> testedClasses, List<String> out)
+    private static void addPackageAndClassList(String reportDirectory, Set<String> usedPackageNames, Set<String> testedClasses, List<String> out, Map<String, ClassInfo> classInfoMap)
     {
         // iterate through all class names
         for (String packageName : usedPackageNames)
@@ -329,7 +352,7 @@
             {
                 if (className.startsWith(packageName))
                 {
-                    out.add(addOneRowToResultsTable(reportDirectory, className));
+                    out.add(addOneRowToResultsTable(reportDirectory, className, classInfoMap));
                 }
             }
             out.add("</table>");
@@ -344,10 +367,10 @@
      * @param className
      *            name of tested class
      */
-    private static String addOneRowToResultsTable(String reportDirectory, String className)
+    private static String addOneRowToResultsTable(String reportDirectory, String className, Map<String, ClassInfo> classInfoMap)
     {
-        Set<String> apiMethods = readApiMethods(reportDirectory, className);
-        Set<String> testedMethods = readTestedMethods(reportDirectory, className);
+        Set<String> apiMethods = classInfoMap.get(className).getApiMethods();
+        Set<String> testedMethods = classInfoMap.get(className).getTestedMethods();
         // compute number of all methods in a class
         final int allMethodsCnt = apiMethods.size();
         // compute number of methods covered by tests
@@ -515,81 +538,31 @@
      *            packages for which the report is generated
      * @param testedClasses
      *            set of tested classes
+     * @param classInfoMap 



More information about the distro-pkg-dev mailing list