All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.maven.shared.jar.classes.JarClassesAnalysis Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */
package org.apache.maven.shared.jar.classes;

import javax.inject.Named;
import javax.inject.Singleton;

import java.io.IOException;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.NavigableMap;
import java.util.Optional;
import java.util.TreeMap;
import java.util.jar.JarEntry;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.stream.Collectors;

import org.apache.bcel.classfile.ClassFormatException;
import org.apache.bcel.classfile.ClassParser;
import org.apache.bcel.classfile.DescendingVisitor;
import org.apache.bcel.classfile.JavaClass;
import org.apache.bcel.classfile.LineNumberTable;
import org.apache.bcel.classfile.Method;
import org.apache.maven.shared.jar.JarAnalyzer;
import org.apache.maven.shared.jar.JarData;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * Analyze the classes in a JAR file. This class is thread safe and immutable as it retains no state.
 *
 * Note that you must first create an instance of {@link org.apache.maven.shared.jar.JarAnalyzer} - see its Javadoc for
 * a typical use.
 *
 * @see #analyze(org.apache.maven.shared.jar.JarAnalyzer)
 */
@Singleton
@Named
@SuppressWarnings("checkstyle:MagicNumber")
public class JarClassesAnalysis {

    private final Logger logger = LoggerFactory.getLogger(getClass());

    /**
     * Constant representing the root content of a Multi-Release JAR file, thus outside of
     * any given META-INF/versions/N/... entry.
     */
    private static final Integer ROOT = 0;

    private static final Pattern ENTRY_FILTER_MULTI_RELEASE = Pattern.compile("^META-INF/versions/([1-9]\\d*)/.*$");

    private static final Map JAVA_CLASS_VERSIONS;

    static {
        HashMap aMap = new HashMap<>();
        aMap.put(65.0, "21");
        aMap.put(64.0, "20");
        aMap.put(63.0, "19");
        aMap.put(62.0, "18");
        aMap.put(61.0, "17");
        aMap.put(60.0, "16");
        aMap.put(59.0, "15");
        aMap.put(58.0, "14");
        aMap.put(57.0, "13");
        aMap.put(56.0, "12");
        aMap.put(55.0, "11");
        aMap.put(54.0, "10");
        aMap.put(53.0, "9");
        aMap.put(52.0, "1.8");
        aMap.put(51.0, "1.7");
        aMap.put(50.0, "1.6");
        aMap.put(49.0, "1.5");
        aMap.put(48.0, "1.4");
        aMap.put(47.0, "1.3");
        aMap.put(46.0, "1.2");
        aMap.put(45.3, "1.1");
        JAVA_CLASS_VERSIONS = Collections.unmodifiableMap(aMap);
    }

    /**
     * Analyze a JAR and find any classes and their details. Note that if the provided JAR analyzer has previously
     * analyzed the JAR, the cached results will be returned. You must obtain a new JAR analyzer to the re-read the
     * contents of the file.
     *
     * @param jarAnalyzer the JAR to analyze. This must not yet have been closed.
     * @return the details of the classes found
     */
    public JarClasses analyze(JarAnalyzer jarAnalyzer) {
        JarData jarData = jarAnalyzer.getJarData();
        JarClasses classes = jarData.getJarClasses();
        if (classes == null) {
            if (jarData.isMultiRelease()) {
                classes = analyzeMultiRelease(jarAnalyzer);
            } else {
                classes = analyzeRoot(jarAnalyzer);
            }
        }
        return classes;
    }

    private Integer jarEntryVersion(JarEntry entry) {
        Matcher matcher = ENTRY_FILTER_MULTI_RELEASE.matcher(entry.getName());
        if (matcher.matches()) {
            return Integer.valueOf(matcher.group(1));
        }
        return ROOT;
    }

    private JarClasses analyzeMultiRelease(JarAnalyzer jarAnalyzer) {
        String jarFilename = jarAnalyzer.getFile().getAbsolutePath();

        Map> mapEntries =
                jarAnalyzer.getEntries().stream().collect(Collectors.groupingBy(this::jarEntryVersion));

        // ordered by increasing Java version
        NavigableMap runtimeVersionsMap = new TreeMap<>();

        for (Map.Entry> mapEntry : mapEntries.entrySet()) {
            Integer runtimeVersion = mapEntry.getKey();
            List runtimeVersionEntryList = mapEntry.getValue();

            List classList = jarAnalyzer.getClassEntries(runtimeVersionEntryList);

            JarClasses classes = analyze(jarFilename, classList);

            runtimeVersionsMap.put(runtimeVersion, new JarVersionedRuntime(runtimeVersionEntryList, classes));
        }

        JarData jarData = jarAnalyzer.getJarData();

        JarVersionedRuntime rootContentVersionedRuntime = runtimeVersionsMap.remove(ROOT);
        jarData.setRootEntries(rootContentVersionedRuntime.getEntries());
        JarClasses rootJarClasses = rootContentVersionedRuntime.getJarClasses();
        jarData.setJarClasses(rootJarClasses);

        jarData.setVersionedRuntimes(new JarVersionedRuntimes(runtimeVersionsMap));

        return rootJarClasses;
    }

    private JarClasses analyzeRoot(JarAnalyzer jarAnalyzer) {
        String jarFilename = jarAnalyzer.getFile().getAbsolutePath();

        List classList = jarAnalyzer.getClassEntries();

        JarClasses classes = analyze(jarFilename, classList);

        jarAnalyzer.getJarData().setJarClasses(classes);
        return classes;
    }

    private JarClasses analyze(String jarFilename, List classList) {
        JarClasses classes = new JarClasses();

        classes.setDebugPresent(false);

        double maxVersion = 0.0;
        double moduleInfoVersion = 0.0;

        for (JarEntry entry : classList) {
            String classname = entry.getName();

            try {
                ClassParser classParser = new ClassParser(jarFilename, classname);

                JavaClass javaClass = classParser.parse();

                String classSignature = javaClass.getClassName();

                if (!classes.isDebugPresent()) {
                    if (hasDebugSymbols(javaClass)) {
                        classes.setDebugPresent(true);
                    }
                }

                double classVersion = javaClass.getMajor();
                if (javaClass.getMinor() > 0) {
                    classVersion = classVersion + javaClass.getMinor() / 10.0;
                }

                if ("module-info".equals(classSignature)) {
                    // ignore the module-info.class for computing the maxVersion, since it will always be >= 9
                    moduleInfoVersion = classVersion;
                } else if (classVersion > maxVersion) {
                    maxVersion = classVersion;
                }

                Method[] methods = javaClass.getMethods();
                for (Method method : methods) {
                    classes.addMethod(classSignature + "." + method.getName() + method.getSignature());
                }

                String classPackageName = javaClass.getPackageName();

                classes.addClassName(classSignature);
                classes.addPackage(classPackageName);

                ImportVisitor importVisitor = new ImportVisitor(javaClass);
                DescendingVisitor descVisitor = new DescendingVisitor(javaClass, importVisitor);
                javaClass.accept(descVisitor);

                classes.addImports(importVisitor.getImports());
            } catch (ClassFormatException e) {
                logger.warn("Unable to process class " + classname + " in JarAnalyzer File " + jarFilename, e);
            } catch (IOException e) {
                logger.warn("Unable to process JarAnalyzer File " + jarFilename, e);
            }
        }

        if (maxVersion == 0.0 && moduleInfoVersion > 0.0) {
            // the one and only class file was module-info.class
            maxVersion = moduleInfoVersion;
        }

        Optional.ofNullable(JAVA_CLASS_VERSIONS.get(maxVersion)).ifPresent(classes::setJdkRevision);

        return classes;
    }

    private boolean hasDebugSymbols(JavaClass javaClass) {
        boolean ret = false;
        Method[] methods = javaClass.getMethods();
        for (Method method : methods) {
            LineNumberTable linenumbers = method.getLineNumberTable();
            if (linenumbers != null && linenumbers.getLength() > 0) {
                ret = true;
                break;
            }
        }
        return ret;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy