Analyzer.java

  1. /*******************************************************************************
  2.  * Copyright (c) 2009, 2025 Mountainminds GmbH & Co. KG and Contributors
  3.  * This program and the accompanying materials are made available under
  4.  * the terms of the Eclipse Public License 2.0 which is available at
  5.  * http://www.eclipse.org/legal/epl-2.0
  6.  *
  7.  * SPDX-License-Identifier: EPL-2.0
  8.  *
  9.  * Contributors:
  10.  *    Marc R. Hoffmann - initial API and implementation
  11.  *
  12.  *******************************************************************************/
  13. package org.jacoco.core.analysis;

  14. import java.io.File;
  15. import java.io.FileInputStream;
  16. import java.io.IOException;
  17. import java.io.InputStream;
  18. import java.util.StringTokenizer;
  19. import java.util.zip.GZIPInputStream;
  20. import java.util.zip.ZipEntry;
  21. import java.util.zip.ZipInputStream;

  22. import org.jacoco.core.JaCoCo;
  23. import org.jacoco.core.data.ExecutionData;
  24. import org.jacoco.core.data.ExecutionDataStore;
  25. import org.jacoco.core.internal.ContentTypeDetector;
  26. import org.jacoco.core.internal.InputStreams;
  27. import org.jacoco.core.internal.Pack200Streams;
  28. import org.jacoco.core.internal.analysis.ClassAnalyzer;
  29. import org.jacoco.core.internal.analysis.ClassCoverageImpl;
  30. import org.jacoco.core.internal.analysis.StringPool;
  31. import org.jacoco.core.internal.data.CRC64;
  32. import org.jacoco.core.internal.flow.ClassProbesAdapter;
  33. import org.jacoco.core.internal.instr.InstrSupport;
  34. import org.objectweb.asm.ClassReader;
  35. import org.objectweb.asm.ClassVisitor;
  36. import org.objectweb.asm.Opcodes;

  37. /**
  38.  * An {@link Analyzer} instance processes a set of Java class files and
  39.  * calculates coverage data for them. For each class file the result is reported
  40.  * to a given {@link ICoverageVisitor} instance. In addition the
  41.  * {@link Analyzer} requires a {@link ExecutionDataStore} instance that holds
  42.  * the execution data for the classes to analyze. The {@link Analyzer} offers
  43.  * several methods to analyze classes from a variety of sources.
  44.  */
  45. public class Analyzer {

  46.     private final ExecutionDataStore executionData;

  47.     private final ICoverageVisitor coverageVisitor;

  48.     private final StringPool stringPool;

  49.     /**
  50.      * Creates a new analyzer reporting to the given output.
  51.      *
  52.      * @param executionData
  53.      *            execution data
  54.      * @param coverageVisitor
  55.      *            the output instance that will coverage data for every analyzed
  56.      *            class
  57.      */
  58.     public Analyzer(final ExecutionDataStore executionData,
  59.             final ICoverageVisitor coverageVisitor) {
  60.         this.executionData = executionData;
  61.         this.coverageVisitor = coverageVisitor;
  62.         this.stringPool = new StringPool();
  63.     }

  64.     /**
  65.      * Creates an ASM class visitor for analysis.
  66.      *
  67.      * @param classid
  68.      *            id of the class calculated with {@link CRC64}
  69.      * @param className
  70.      *            VM name of the class
  71.      * @return ASM visitor to write class definition to
  72.      */
  73.     private ClassVisitor createAnalyzingVisitor(final long classid,
  74.             final String className) {
  75.         final ExecutionData data = executionData.get(classid);
  76.         final boolean[] probes;
  77.         final boolean noMatch;
  78.         if (data == null) {
  79.             probes = null;
  80.             noMatch = executionData.contains(className);
  81.         } else {
  82.             probes = data.getProbes();
  83.             noMatch = false;
  84.         }
  85.         final ClassCoverageImpl coverage = new ClassCoverageImpl(className,
  86.                 classid, noMatch);
  87.         final ClassAnalyzer analyzer = new ClassAnalyzer(coverage, probes,
  88.                 stringPool) {
  89.             @Override
  90.             public void visitEnd() {
  91.                 super.visitEnd();
  92.                 coverageVisitor.visitCoverage(coverage);
  93.             }
  94.         };
  95.         return new ClassProbesAdapter(analyzer, false);
  96.     }

  97.     private void analyzeClass(final byte[] source) {
  98.         final long classId = CRC64.classId(source);
  99.         final ClassReader reader = InstrSupport.classReaderFor(source);
  100.         if ((reader.getAccess() & Opcodes.ACC_MODULE) != 0) {
  101.             return;
  102.         }
  103.         if ((reader.getAccess() & Opcodes.ACC_SYNTHETIC) != 0) {
  104.             return;
  105.         }
  106.         final ClassVisitor visitor = createAnalyzingVisitor(classId,
  107.                 reader.getClassName());
  108.         reader.accept(visitor, 0);
  109.     }

  110.     /**
  111.      * Analyzes the class definition from a given in-memory buffer.
  112.      *
  113.      * @param buffer
  114.      *            class definitions
  115.      * @param location
  116.      *            a location description used for exception messages
  117.      * @throws IOException
  118.      *             if the class can't be analyzed
  119.      */
  120.     public void analyzeClass(final byte[] buffer, final String location)
  121.             throws IOException {
  122.         try {
  123.             analyzeClass(buffer);
  124.         } catch (final RuntimeException cause) {
  125.             throw analyzerError(location, cause);
  126.         }
  127.     }

  128.     /**
  129.      * Analyzes the class definition from a given input stream. The provided
  130.      * {@link InputStream} is not closed by this method.
  131.      *
  132.      * @param input
  133.      *            stream to read class definition from
  134.      * @param location
  135.      *            a location description used for exception messages
  136.      * @throws IOException
  137.      *             if the stream can't be read or the class can't be analyzed
  138.      */
  139.     public void analyzeClass(final InputStream input, final String location)
  140.             throws IOException {
  141.         final byte[] buffer;
  142.         try {
  143.             buffer = InputStreams.readFully(input);
  144.         } catch (final IOException e) {
  145.             throw analyzerError(location, e);
  146.         }
  147.         analyzeClass(buffer, location);
  148.     }

  149.     private IOException analyzerError(final String location,
  150.             final Exception cause) {
  151.         final IOException ex = new IOException(
  152.                 String.format("Error while analyzing %s with JaCoCo %s/%s.",
  153.                         location, JaCoCo.VERSION, JaCoCo.COMMITID_SHORT));
  154.         ex.initCause(cause);
  155.         return ex;
  156.     }

  157.     /**
  158.      * Analyzes all classes found in the given input stream. The input stream
  159.      * may either represent a single class file, a ZIP archive, a Pack200
  160.      * archive or a gzip stream that is searched recursively for class files.
  161.      * All other content types are ignored. The provided {@link InputStream} is
  162.      * not closed by this method.
  163.      *
  164.      * @param input
  165.      *            input data
  166.      * @param location
  167.      *            a location description used for exception messages
  168.      * @return number of class files found
  169.      * @throws IOException
  170.      *             if the stream can't be read or a class can't be analyzed
  171.      */
  172.     public int analyzeAll(final InputStream input, final String location)
  173.             throws IOException {
  174.         final ContentTypeDetector detector;
  175.         try {
  176.             detector = new ContentTypeDetector(input);
  177.         } catch (final IOException e) {
  178.             throw analyzerError(location, e);
  179.         }
  180.         switch (detector.getType()) {
  181.         case ContentTypeDetector.CLASSFILE:
  182.             analyzeClass(detector.getInputStream(), location);
  183.             return 1;
  184.         case ContentTypeDetector.ZIPFILE:
  185.             return analyzeZip(detector.getInputStream(), location);
  186.         case ContentTypeDetector.GZFILE:
  187.             return analyzeGzip(detector.getInputStream(), location);
  188.         case ContentTypeDetector.PACK200FILE:
  189.             return analyzePack200(detector.getInputStream(), location);
  190.         default:
  191.             return 0;
  192.         }
  193.     }

  194.     /**
  195.      * Analyzes all class files contained in the given file or folder. Class
  196.      * files as well as ZIP files are considered. Folders are searched
  197.      * recursively.
  198.      *
  199.      * @param file
  200.      *            file or folder to look for class files
  201.      * @return number of class files found
  202.      * @throws IOException
  203.      *             if the file can't be read or a class can't be analyzed
  204.      */
  205.     public int analyzeAll(final File file) throws IOException {
  206.         int count = 0;
  207.         if (file.isDirectory()) {
  208.             for (final File f : file.listFiles()) {
  209.                 count += analyzeAll(f);
  210.             }
  211.         } else {
  212.             final InputStream in = new FileInputStream(file);
  213.             try {
  214.                 count += analyzeAll(in, file.getPath());
  215.             } finally {
  216.                 in.close();
  217.             }
  218.         }
  219.         return count;
  220.     }

  221.     /**
  222.      * Analyzes all classes from the given class path. Directories containing
  223.      * class files as well as archive files are considered.
  224.      *
  225.      * @param path
  226.      *            path definition
  227.      * @param basedir
  228.      *            optional base directory, if <code>null</code> the current
  229.      *            working directory is used as the base for relative path
  230.      *            entries
  231.      * @return number of class files found
  232.      * @throws IOException
  233.      *             if a file can't be read or a class can't be analyzed
  234.      */
  235.     public int analyzeAll(final String path, final File basedir)
  236.             throws IOException {
  237.         int count = 0;
  238.         final StringTokenizer st = new StringTokenizer(path,
  239.                 File.pathSeparator);
  240.         while (st.hasMoreTokens()) {
  241.             count += analyzeAll(new File(basedir, st.nextToken()));
  242.         }
  243.         return count;
  244.     }

  245.     private int analyzeZip(final InputStream input, final String location)
  246.             throws IOException {
  247.         final ZipInputStream zip = new ZipInputStream(input);
  248.         ZipEntry entry;
  249.         int count = 0;
  250.         while ((entry = nextEntry(zip, location)) != null) {
  251.             count += analyzeAll(zip, location + "@" + entry.getName());
  252.         }
  253.         return count;
  254.     }

  255.     private ZipEntry nextEntry(final ZipInputStream input,
  256.             final String location) throws IOException {
  257.         try {
  258.             return input.getNextEntry();
  259.         } catch (final IOException e) {
  260.             throw analyzerError(location, e);
  261.         } catch (final IllegalArgumentException e) {
  262.             // might be thrown in JDK versions below 23 - see
  263.             // https://bugs.openjdk.org/browse/JDK-8321156
  264.             // https://github.com/openjdk/jdk/commit/20c71ceacdcb791f5b70cda456bdc47bdd9acf6c
  265.             throw analyzerError(location, e);
  266.         }
  267.     }

  268.     private int analyzeGzip(final InputStream input, final String location)
  269.             throws IOException {
  270.         GZIPInputStream gzipInputStream;
  271.         try {
  272.             gzipInputStream = new GZIPInputStream(input);
  273.         } catch (final IOException e) {
  274.             throw analyzerError(location, e);
  275.         }
  276.         return analyzeAll(gzipInputStream, location);
  277.     }

  278.     private int analyzePack200(final InputStream input, final String location)
  279.             throws IOException {
  280.         InputStream unpackedInput;
  281.         try {
  282.             unpackedInput = Pack200Streams.unpack(input);
  283.         } catch (final IOException e) {
  284.             throw analyzerError(location, e);
  285.         }
  286.         return analyzeAll(unpackedInput, location);
  287.     }

  288. }