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

liquibase.resource.ClassLoaderResourceAccessor Maven / Gradle / Ivy

There is a newer version: 4.30.0
Show newest version
package liquibase.resource;

import liquibase.Scope;
import liquibase.changelog.DatabaseChangeLog;
import liquibase.util.StreamUtil;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.net.URISyntaxException;
import java.net.URL;
import java.net.URLClassLoader;
import java.net.URLDecoder;
import java.nio.charset.StandardCharsets;
import java.nio.file.*;
import java.nio.file.attribute.BasicFileAttributes;
import java.util.*;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;

/**
 * An implementation of {@link FileSystemResourceAccessor} that builds up the file roots based on the passed {@link ClassLoader}.
 * If you are using a ClassLoader that isn't based on local files, you will need to use a different {@link ResourceAccessor} implementation.
 *
 * @see OSGiResourceAccessor for OSGi-based classloaders
 */
public class ClassLoaderResourceAccessor extends AbstractResourceAccessor implements AutoCloseable {

    private ClassLoader classLoader;
    protected List rootPaths;
    protected SortedSet description;

    public ClassLoaderResourceAccessor() {
        this(Thread.currentThread().getContextClassLoader());
    }

    public ClassLoaderResourceAccessor(ClassLoader classLoader) {
        this.classLoader = classLoader;

    }

    /**
     * Performs the configuration of this resourceAccessor.
     * Not done in the constructor for performance reasons, but can be called at the beginning of every public method.
     */
    protected void init() {
        if (rootPaths == null) {
            this.rootPaths = new ArrayList<>();
            this.description = new TreeSet<>();

            loadRootPaths(classLoader);
        }
    }


    /**
     * The classloader search logic in {@link #list(String, String, boolean, boolean, boolean)} does not handle jar files well.
     * This method is called by that method to populate {@link #rootPaths} with additional paths to search.
     */
    protected void loadRootPaths(ClassLoader classLoader) {
        if (classLoader instanceof URLClassLoader) {
            final URL[] urls = ((URLClassLoader) classLoader).getURLs();
            if (urls != null) {
                for (URL url : urls) {
                    try {
                        addDescription(url);
                        this.rootPaths.add(FileSystems.newFileSystem(Paths.get(url.toURI()), this.getClass().getClassLoader()));
                    } catch (FileSystemAlreadyExistsException e) {
                        //has been defined already, that is OK
                    } catch (ProviderNotFoundException e) {
                        if (url.toExternalForm().startsWith("file:/")) {
                            //that is expected, the classloader itself will handle it
                        } else {
                            Scope.getCurrentScope().getLog(getClass()).info("No filesystem provider for URL " + url.toExternalForm() + ". Will rely on classloader logic for listing files.");
                        }
                    } catch (FileSystemNotFoundException fsnfe) {
                        if (url.toExternalForm().matches(".*!.*!.*")) {
                            //spring sometimes sets up urls with nested urls like jar:file:/path/to/demo-0.0.1-SNAPSHOT.jar!/BOOT-INF/lib/mssql-jdbc-8.2.2.jre8.jar!/ which are not readable.
                            //That is expected, and will be handled by the SpringResourceAccessor
                        } else {
                            Scope.getCurrentScope().getLog(getClass()).info("Configured classpath location " + url.toString() + " does not exist");
                        }
                    } catch (Throwable e) {
                        Scope.getCurrentScope().getLog(getClass()).warning("Cannot create filesystem for url " + url.toExternalForm() + ": " + e.getMessage(), e);
                    }
                }
            }
        }

        final ClassLoader parent = classLoader.getParent();
        if (parent != null) {
            loadRootPaths(parent);
        }

    }

    private void addDescription(URL url) {
        try {
            this.description.add(Paths.get(url.toURI()).toString());
        } catch (Throwable e) {
            this.description.add(url.toExternalForm());
        }
    }

    @Override
    @java.lang.SuppressWarnings("squid:S2095")
    public InputStreamList openStreams(String relativeTo, String streamPath) throws IOException {
        init();

        InputStreamList returnList = new InputStreamList();

        streamPath = getFinalPath(relativeTo, streamPath);

        //sometimes the classloader returns duplicate copies of the same url
        Set seenUrls = new HashSet<>();

        Enumeration resources = classLoader.getResources(streamPath);
        while (resources.hasMoreElements()) {
            URL url = resources.nextElement();

            if (seenUrls.add(url.toExternalForm())) {
                try {
                    returnList.add(url.toURI(), url.openStream());
                } catch (URISyntaxException e) {
                    Scope.getCurrentScope().getLog(getClass()).severe(e.getMessage(), e);
                }
            }
        }

        return returnList;
    }

    /**
     * Generates a final path to streamPath relative to relatoveTo.
     * If the last part of relativeTo contains a dot character (`.`)
     * this part is considered to be a file name, if it does not, it is
     * considered to be a directory.
     * i.e.
*
     * changelog/some.sql   -> some.sql is considered to be a file
     * changelog/some_sql   -> some_sql is considered to be a directory
     * 
* * @param relativeTo starting point of the path resolution (may be null) * @param streamPath a path to a resource relative to relativeTo must not be null * @return a canonicalized absolute path to a resource */ protected String getFinalPath(String relativeTo, String streamPath) { streamPath = streamPath.replace("\\", "/"); streamPath = streamPath.replaceFirst("^classpath\\*?:", ""); if (relativeTo != null) { relativeTo = relativeTo.replace("\\", "/"); relativeTo = relativeTo.replaceFirst("^classpath\\*?:", ""); relativeTo = relativeTo.replaceAll("//+", "/"); // // If this is a simple file name then set the // relativeTo value as if it is a root path // if (!relativeTo.contains("/") && relativeTo.contains(".")) { relativeTo = "/"; } // // If this is not a simple file name and the last component // of the path contains a '.' remove the last component // if (!relativeTo.endsWith("/")) { String lastPortion = relativeTo.replaceFirst(".+/", ""); if (lastPortion.contains(".")) { relativeTo = relativeTo.replaceFirst("/[^/]+?$", ""); } } streamPath = relativeTo + "/" + streamPath; } streamPath = streamPath.replaceAll("//+", "/"); streamPath = streamPath.replaceFirst("^/", ""); return DatabaseChangeLog.normalizePath(streamPath); } @Override public SortedSet list(String relativeTo, String path, boolean recursive, boolean includeFiles, boolean includeDirectories) throws IOException { init(); String finalPath = getFinalPath(relativeTo, path); final SortedSet returnList = listFromClassLoader(finalPath, recursive, includeFiles, includeDirectories); returnList.addAll(listFromRootPaths(finalPath, recursive, includeFiles, includeDirectories)); return returnList; } /** * Called by {@link #list(String, String, boolean, boolean, boolean)} to find files in {@link #rootPaths}. */ protected SortedSet listFromRootPaths(String path, boolean recursive, boolean includeFiles, boolean includeDirectories) { SortedSet returnSet = new TreeSet<>(); SimpleFileVisitor fileVisitor = new SimpleFileVisitor() { @Override public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException { if (includeFiles && attrs.isRegularFile()) { addToReturnList(file); } if (includeDirectories && attrs.isDirectory()) { addToReturnList(file); } return FileVisitResult.CONTINUE; } @Override public FileVisitResult postVisitDirectory(Path dir, IOException exc) throws IOException { if (includeDirectories) { addToReturnList(dir); } return FileVisitResult.CONTINUE; } protected void addToReturnList(Path file) { if (!file.toString().equals(path)) { returnSet.add(file.toString() .replaceFirst("^/", "") .replaceFirst("/$", "") .replaceAll("//+", "/") ); } } }; for (FileSystem fileSystem : rootPaths) { int maxDepth = recursive ? Integer.MAX_VALUE : 1; try { Files.walkFileTree(fileSystem.getPath(path), Collections.singleton(FileVisitOption.FOLLOW_LINKS), maxDepth, fileVisitor); } catch (NoSuchFileException e) { //that is OK } catch (IOException e) { Scope.getCurrentScope().getLog(getClass()).warning("Cannot walk filesystem: " + e.getMessage(), e); } } return returnSet; } /** * Called by {@link #list(String, String, boolean, boolean, boolean)} to find files in {@link #classLoader}. */ protected SortedSet listFromClassLoader(String path, boolean recursive, boolean includeFiles, boolean includeDirectories) { final SortedSet returnSet = new TreeSet<>(); final Enumeration resources; try { resources = classLoader.getResources(path); } catch (IOException e) { Scope.getCurrentScope().getLog(getClass()).severe("Cannot list resources in path " + path + ": " + e.getMessage(), e); return returnSet; } while (resources.hasMoreElements()) { final URL url = resources.nextElement(); final String urlExternalForm = url.toExternalForm(); try { if (urlExternalForm.startsWith("jar:file:") && urlExternalForm.contains("!")) { //We can search the jar directly String jarPath = url.getPath(); jarPath = jarPath.substring(5, jarPath.indexOf("!")); try (JarFile jar = new JarFile(URLDecoder.decode(jarPath, StandardCharsets.UTF_8.name()))) { String comparePath = path; if (comparePath.startsWith("/")) { comparePath = "/" + comparePath; } Enumeration entries = jar.entries(); while (entries.hasMoreElements()) { JarEntry entry = entries.nextElement(); String name = entry.getName(); if (name.startsWith(comparePath) && !comparePath.equals(name)) { if (entry.isDirectory()) { if (!includeDirectories) { continue; } if (recursive || !name.substring(comparePath.length()).contains("/")) { returnSet.add(name); } } else { if (includeFiles) { if (recursive || !name.substring(comparePath.length()).contains("/")) { returnSet.add(name); } } } } } } } else { //fall back to seeing if the stream lists sub-directories final InputStream inputStream = url.openStream(); final String fileList = StreamUtil.readStreamAsString(inputStream); if (!fileList.isEmpty()) { for (String childName : fileList.split("\n")) { String childPath = (path + "/" + childName).replaceAll("//+", "/"); if (isDirectory(childPath)) { if (includeDirectories) { returnSet.add(childPath); } if (recursive) { returnSet.addAll(listFromClassLoader(childPath, recursive, includeFiles, includeDirectories)); } } else { if (includeFiles) { returnSet.add(childPath); } } } } } } catch (IOException e) { Scope.getCurrentScope().getLog(getClass()).severe("Cannot list resources in " + urlExternalForm + ": " + e.getMessage(), e); } } return returnSet; } /** * Used by {@link #listFromClassLoader(String, boolean, boolean, boolean)} to determine if a path is a directory or not. */ protected boolean isDirectory(String path) { try { final Enumeration resources = classLoader.getResources(path); while (resources.hasMoreElements()) { final URL url = resources.nextElement(); final File file = new File(url.toURI()); if (file.exists() && file.isDirectory()) { return true; } } } catch (Exception e) { //not a url we can handle } //fallback logic depends on files having an extension and directories not String lastPortion = path.replaceFirst(".*/", ""); return !lastPortion.contains("."); } @Override public SortedSet describeLocations() { init(); return description; } @Override public void close() throws Exception { if (rootPaths != null) { for (final FileSystem rootPath : rootPaths) { try { rootPath.close(); } catch (final Exception e) { Scope.getCurrentScope().getLog(getClass()).fine("Cannot close path " + e.getMessage(), e); } } } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy