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

org.jruby.util.CompoundJarURLStreamHandler Maven / Gradle / Ivy

There is a newer version: 9.4.9.0
Show newest version
package org.jruby.util;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.Closeable;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLConnection;
import java.net.URLStreamHandler;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;
import java.util.jar.JarInputStream;

import static org.jruby.util.URLUtil.getPath;

// From Stas Garifulin's CompoundJarURLStreamHandler. http://jira.codehaus.org/browse/JRUBY-3299
public class CompoundJarURLStreamHandler extends URLStreamHandler {

    public static final String PROTOCOL = "compoundjar";

    private static final CompoundJarURLStreamHandler instance = new CompoundJarURLStreamHandler();

    public static URL createUrl(URL base, List path) throws MalformedURLException {
        return createUrl(base, path.toArray(new String[0]));
    }

    public static URL createUrl(URL base, String... path) throws MalformedURLException {
        StringBuilder pathBuilder = new StringBuilder();

        if (base.getProtocol().equals("jar")) {
            pathBuilder.append(getPath(base));
        } else {
            pathBuilder.append(base.toExternalForm());
        }

        for (String entry : path) {
            pathBuilder.append("!");

            if (!entry.startsWith("/")) {
                pathBuilder.append("/");
            }

            pathBuilder.append(entry);
        }

        return new URL(PROTOCOL, null, -1, pathBuilder.toString(), instance);
    }

    static class CompoundJarURLConnection extends URLConnection {

        private final URL baseJarUrl;

        private final String[] path;

        private static final Map> cache = new ConcurrentHashMap>(16, 0.75f, 4);

        CompoundJarURLConnection(URL url) throws MalformedURLException {
            super(url);
            String spec = getPath(url);
            path = spec.split("\\!\\/");
            baseJarUrl = new URL(path[0]);
        }

        @Override
        public void connect() throws IOException {
            connected = true;
        }

        private InputStream openEntryWithCache(String[] path, InputStream currentStream, int currentDepth) throws IOException {
            final String localPath = path[currentDepth];

            // short-circuit files directly on the filesystem, which JarFile can handle
            if (currentDepth == 1 && path[0].indexOf('!') == -1 && path[0].startsWith("file:")) {
                // it's a top-level jar, just open with JarFile
                JarFile jarFile = new JarFile(path[0].substring(5));
                JarEntry entry = jarFile.getJarEntry(localPath);
                if (entry != null) {
                    return returnOrRecurse(path, jarFile.getInputStream(entry), currentDepth);
                }
                return null;
            }

            // build path for cache lookup
            StringBuilder pathToHereBuffer = new StringBuilder();
            for (int i = 0; i < currentDepth; i++) {
                if (i > 0) pathToHereBuffer.append("!/");
                pathToHereBuffer.append(path[i]);
            }
            String pathToHere = pathToHereBuffer.toString();

            // check for already-read cached jar contents
            Map contents = cache.get(pathToHere);
            if (contents == null) {
                // not found, cache jar contents
                cache.put(pathToHere, contents = new ConcurrentHashMap(16, 0.75f, 2));
                cacheJarFrom(currentStream, contents);
            }

            // now go to cache to find bytes for this elements
            byte[] bytes = contents.get(localPath);
            if (bytes != null) {
                return returnOrRecurse(path, new ByteArrayInputStream(bytes), currentDepth);
            }

            return null;
        }

        private void cacheJarFrom(InputStream currentStream, Map contents) throws IOException {
            JarInputStream currentJar = new JarInputStream(currentStream);
            for (JarEntry entry = currentJar.getNextJarEntry(); entry != null; entry = currentJar.getNextJarEntry()) {
                ByteArrayOutputStream baos = new ByteArrayOutputStream();
                byte[] bytes = new byte[1024];
                int size;
                while ((size = currentJar.read(bytes, 0, 1024)) != -1) {
                    baos.write(bytes, 0, size);
                }
                bytes = baos.toByteArray();
                contents.put(entry.getName(), bytes);
            }
        }

        private InputStream returnOrRecurse(String[] path, InputStream nextStream, int currentDepth) throws IOException {
            if (currentDepth + 1 < path.length) {
                // we're not all the way down, open the inner jar and go deeper
                return openEntryWithCache(path, nextStream, currentDepth + 1);
            } else {
                // we've got it; return an array reading the bytes
                return nextStream;
            }
        }

        private static void close(Closeable resource) {
            if (resource != null) {
                try {
                    resource.close();
                } catch (IOException ignore) {
                }
            }
        }

        @Override
        public InputStream getInputStream() throws IOException {

            InputStream result;

            InputStream baseInputStream = baseJarUrl.openStream();

            if (path.length > 1) {
                try {
                    result = openEntryWithCache(path, baseInputStream, 1);
                } catch (IOException ex) {
                    close(baseInputStream);

                    throw ex;
                } catch (RuntimeException ex) {
                    close(baseInputStream);

                    throw ex;
                }
            } else {
                result = baseInputStream;
            }

            if (result == null) {
                throw new FileNotFoundException(url.toExternalForm());
            }

            return result;
        }
    }

    @Override
    protected URLConnection openConnection(URL url) throws IOException {
        return new CompoundJarURLConnection(url);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy