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

edu.umd.cs.findbugs.classfile.impl.JrtfsCodeBase Maven / Gradle / Ivy

The newest version!
/*
 * FindBugs - Find Bugs in Java programs
 * Copyright (C) 2003-2008 University of Maryland
 *
 * 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.classfile.impl;

import java.io.BufferedInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
import java.net.URL;
import java.net.URLClassLoader;
import java.nio.file.FileSystem;
import java.nio.file.FileSystems;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardOpenOption;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.stream.Stream;

import javax.annotation.CheckForNull;
import javax.annotation.Nonnull;

import edu.umd.cs.findbugs.classfile.ClassDescriptor;
import edu.umd.cs.findbugs.classfile.DescriptorFactory;
import edu.umd.cs.findbugs.classfile.ICodeBaseEntry;
import edu.umd.cs.findbugs.classfile.ICodeBaseIterator;
import edu.umd.cs.findbugs.classfile.ICodeBaseLocator;
import edu.umd.cs.findbugs.classfile.InvalidClassFileFormatException;
import edu.umd.cs.findbugs.classfile.ResourceNotFoundException;
import edu.umd.cs.findbugs.util.ClassName;

/**
 *
 * Code base supporting Java 9 new jimage packed modules
 *
 * @author andrey
 */
public class JrtfsCodeBase extends AbstractScannableCodeBase {
    private static final int PRIME = 31;

    private FileSystem fs;
    private final String fileName;
    private Path root;

    /**
     * Key is package name in bytecode notation (e.g. 'java/lang').
     * 

* Values are either plain Strings for single-module packages, or sets of * Strings for packages spread over multiple modules */ private Map packageToModuleMap; public JrtfsCodeBase(ICodeBaseLocator codeBaseLocator, @Nonnull String fileName) { super(codeBaseLocator); this.fileName = fileName; URL url; try { url = Paths.get(fileName).toUri().toURL(); URLClassLoader loader = new URLClassLoader(new URL[] { url }); fs = FileSystems.newFileSystem(URI.create("jrt:/"), Collections.emptyMap(), loader); root = fs.getPath("modules"); packageToModuleMap = createPackageToModuleMap(fs); } catch (IOException e) { e.printStackTrace(); } } public Map createPackageToModuleMap(FileSystem fs) throws IOException { HashMap packageToModule = new LinkedHashMap<>(); Path path = fs.getPath("packages"); try (Stream packList = Files.list(path)) { packList.forEach(p -> { try (Stream pList = Files.list(p)) { Iterator modIter = pList.iterator(); while (modIter.hasNext()) { Path module = modIter.next(); String packageKey = ClassName.toSlashedClassName(fileName(p)); String modulePath = fileName(module); if (!modIter.hasNext() && !packageToModule.containsKey(packageKey)) { packageToModule.put(packageKey, modulePath); } else { @SuppressWarnings("unchecked") Set modules = (Set) packageToModule.get(packageKey); if (modules == null) { modules = new LinkedHashSet<>(); packageToModule.put(packageKey, modules); } modules.add(modulePath); } } } catch (IOException e) { e.printStackTrace(); } }); } ; return packageToModule; } @Override @CheckForNull public ICodeBaseEntry lookupResource(String resourceName) { resourceName = translateResourceName(resourceName); String packageName = getPackage(resourceName); Object moduleNameOrSet = packageToModuleMap.get(packageName); if (moduleNameOrSet == null) { return null; } if (moduleNameOrSet instanceof String) { return createEntry(resourceName, (String) moduleNameOrSet); } else { @SuppressWarnings("unchecked") Set modules = (Set) moduleNameOrSet; for (String moduleName : modules) { ICodeBaseEntry entry = createEntry(resourceName, moduleName); if (entry != null) { return entry; } } } return null; } @CheckForNull private ICodeBaseEntry createEntry(String resourceName, String moduleName) { Path resolved = root.resolve(moduleName + "/" + resourceName); if (Files.exists(resolved)) { return new JrtfsCodebaseEntry(resolved, root, this); } return null; } private static String getPackage(String resourceName) { int lastSlash = resourceName.lastIndexOf('/'); if (lastSlash > 0) { return resourceName.substring(0, lastSlash); } return resourceName; } @Override public String getPathName() { return fileName; } @Override public int hashCode() { return PRIME + fileName.hashCode(); } @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (!(obj instanceof JrtfsCodeBase)) { return false; } return fileName.equals(((JrtfsCodeBase) obj).fileName); } @Override public String toString() { StringBuilder builder = new StringBuilder(); builder.append("JrtfsCodeBase ["); if (fileName != null) { builder.append("file="); builder.append(fileName); } builder.append("]"); return builder.toString(); } @Override public void close() { if (fs != null) { try { fs.close(); } catch (IOException e) { e.printStackTrace(); } } } @Override public ICodeBaseIterator iterator() throws InterruptedException { return new JrtfsCodeBaseIterator(); } @Nonnull static String fileName(Path p) { Path name = p.getFileName(); return name != null ? name.toString() : ""; } static boolean isClassFile(Path p) { return p.endsWith(".class") && !p.endsWith("module-info.class") && Files.isRegularFile(p); } public class JrtfsCodeBaseIterator implements ICodeBaseIterator { private Iterator iterator; public JrtfsCodeBaseIterator() { try { iterator = Files.walk(root).filter(p -> isClassFile(p)).iterator(); } catch (IOException e) { e.printStackTrace(); } } @Override public boolean hasNext() throws InterruptedException { return iterator.hasNext(); } @Override public ICodeBaseEntry next() throws InterruptedException { if (Thread.interrupted()) { throw new InterruptedException(); } Path next = iterator.next(); return new JrtfsCodebaseEntry(next, root, JrtfsCodeBase.this); } } public static class JrtfsCodebaseEntry extends AbstractScannableCodeBaseEntry { private final Path path; private final Path root; private final JrtfsCodeBase codebase; public JrtfsCodebaseEntry(Path next, Path root, JrtfsCodeBase codebase) { this.path = next; this.root = root; this.codebase = codebase; } @Override public int getNumBytes() { try { return (int) Files.size(path); } catch (IOException e) { e.printStackTrace(); return -1; } } @Override public InputStream openResource() throws IOException { return new BufferedInputStream(Files.newInputStream(path, StandardOpenOption.READ)); } @Override public JrtfsCodeBase getCodeBase() { return codebase; } @Override public ClassDescriptor getClassDescriptor() throws ResourceNotFoundException, InvalidClassFileFormatException { return DescriptorFactory.createClassDescriptorFromResourceName(getResourceName()); } @Override public String getRealResourceName() { return root.relativize(path).toString(); } @Override public int hashCode() { int result = PRIME + codebase.hashCode(); result = PRIME * result + path.hashCode(); return result; } @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (!(obj instanceof JrtfsCodebaseEntry)) { return false; } JrtfsCodebaseEntry other = (JrtfsCodebaseEntry) obj; if (!Objects.equals(codebase, other.codebase)) { return false; } return Objects.equals(path, other.path); } @Override public String toString() { StringBuilder builder = new StringBuilder(); builder.append("JrtfsCodebaseEntry ["); if (path != null) { builder.append("path="); builder.append(path); builder.append(", "); } if (codebase != null) { builder.append("codebase="); builder.append(codebase); } builder.append("]"); return builder.toString(); } } }