Codebase list jd-gui / 0af4a626-0b34-4685-a8a4-c18e8bd66f2d/upstream services / src / main / java / org / jd / gui / service / sourceloader / MavenOrgSourceLoaderProvider.java
0af4a626-0b34-4685-a8a4-c18e8bd66f2d/upstream

Tree @0af4a626-0b34-4685-a8a4-c18e8bd66f2d/upstream (Download .tar.gz)

MavenOrgSourceLoaderProvider.java @0af4a626-0b34-4685-a8a4-c18e8bd66f2d/upstreamraw · history · blame

/*
 * Copyright (c) 2008-2019 Emmanuel Dupuy.
 * This project is distributed under the GPLv3 license.
 * This is a Copyleft license that gives the user the right to use,
 * copy and modify the code freely for non-commercial purposes.
 */

package org.jd.gui.service.sourceloader;

import org.jd.gui.api.API;
import org.jd.gui.api.model.Container;
import org.jd.gui.service.preferencespanel.MavenOrgSourceLoaderPreferencesProvider;
import org.jd.gui.spi.SourceLoader;
import org.jd.gui.util.exception.ExceptionUtil;

import javax.xml.stream.XMLInputFactory;
import javax.xml.stream.XMLStreamConstants;
import javax.xml.stream.XMLStreamReader;
import java.io.*;
import java.net.URL;
import java.security.DigestInputStream;
import java.security.MessageDigest;
import java.util.*;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;

public class MavenOrgSourceLoaderProvider implements SourceLoader {
    protected static final String MAVENORG_SEARCH_URL_PREFIX = "https://search.maven.org/solrsearch/select?q=1:%22";
    protected static final String MAVENORG_SEARCH_URL_SUFFIX = "%22&rows=20&wt=xml";

    protected static final String MAVENORG_LOAD_URL_PREFIX = "https://search.maven.org/classic/remotecontent?filepath=";
    protected static final String MAVENORG_LOAD_URL_SUFFIX = "-sources.jar";

    protected HashSet<Container.Entry> failed = new HashSet<>();
    protected HashMap<Container.Entry, File> cache = new HashMap<>();

    @Override
    public String getSource(API api, Container.Entry entry) {
        if (isActivated(api)) {
            String filters = api.getPreferences().get(MavenOrgSourceLoaderPreferencesProvider.FILTERS);

            if ((filters == null) || filters.isEmpty()) {
                filters = MavenOrgSourceLoaderPreferencesProvider.DEFAULT_FILTERS_VALUE;
            }

            if (accepted(filters, entry.getPath())) {
                return searchSource(entry, cache.get(entry.getContainer().getRoot().getParent()));
            }
        }

        return null;
    }

    @Override
    public String loadSource(API api, Container.Entry entry) {
        if (isActivated(api)) {
            String filters = api.getPreferences().get(MavenOrgSourceLoaderPreferencesProvider.FILTERS);

            if ((filters == null) || filters.isEmpty()) {
                filters = MavenOrgSourceLoaderPreferencesProvider.DEFAULT_FILTERS_VALUE;
            }

            if (accepted(filters, entry.getPath())) {
                return searchSource(entry, downloadSourceJarFile(entry.getContainer().getRoot().getParent()));
            }
        }

        return null;
    }

    @Override
    public File loadSourceFile(API api, Container.Entry entry) {
        return isActivated(api) ? downloadSourceJarFile(entry) : null;
    }

    private static boolean isActivated(API api) {
        return !"false".equals(api.getPreferences().get(MavenOrgSourceLoaderPreferencesProvider.ACTIVATED));
    }

    protected String searchSource(Container.Entry entry, File sourceJarFile) {
        if (sourceJarFile != null) {
            byte[] buffer = new byte[1024 * 2];

            try (ZipInputStream zis = new ZipInputStream(new BufferedInputStream(new FileInputStream(sourceJarFile)))) {
                ZipEntry ze = zis.getNextEntry();
                String name = entry.getPath();

                name = name.substring(0, name.length()-6) + ".java"; // 6 = ".class".length()

                while (ze != null) {
                    if (ze.getName().equals(name)) {
                        ByteArrayOutputStream out = new ByteArrayOutputStream();
                        int read = zis.read(buffer);

                        while (read > 0) {
                            out.write(buffer, 0, read);
                            read = zis.read(buffer);
                        }

                        return new String(out.toByteArray(), "UTF-8");
                    }

                    ze = zis.getNextEntry();
                }

                zis.closeEntry();
            } catch (IOException e) {
                assert ExceptionUtil.printStackTrace(e);
            }
        }

        return null;
    }

    protected File downloadSourceJarFile(Container.Entry entry) {
        if (cache.containsKey(entry)) {
            return cache.get(entry);
        }

        if (!entry.isDirectory() && !failed.contains(entry)) {
            try {
                // SHA-1
                MessageDigest messageDigest = MessageDigest.getInstance("SHA-1");
                byte[] buffer = new byte[1024 * 2];

                try (DigestInputStream is = new DigestInputStream(entry.getInputStream(), messageDigest)) {
                    while (is.read(buffer) > -1);
                }

                byte[] array = messageDigest.digest();
                StringBuilder sb = new StringBuilder();

                for (byte b : array) {
                    sb.append(hexa((b & 255) >> 4));
                    sb.append(hexa(b & 15));
                }

                String sha1 = sb.toString();

                // Search artifact on maven.org
                URL searchUrl = new URL(MAVENORG_SEARCH_URL_PREFIX + sha1 + MAVENORG_SEARCH_URL_SUFFIX);
                boolean sourceAvailable = false;
                String id = null;
                String numFound = null;

                try (InputStream is = searchUrl.openStream()) {
                    XMLStreamReader reader = XMLInputFactory.newInstance().createXMLStreamReader(is);
                    String name = "";

                    while (reader.hasNext()) {
                        switch (reader.next()) {
                            case XMLStreamConstants.START_ELEMENT:
                                if ("str".equals(reader.getLocalName())) {
                                    if ("id".equals(reader.getAttributeValue(null, "name"))) {
                                        name = "id";
                                    } else {
                                        name = "str";
                                    }
                                } else if ("result".equals(reader.getLocalName())) {
                                    numFound = reader.getAttributeValue(null, "numFound");
                                } else {
                                    name = "";
                                }
                                break;
                            case XMLStreamConstants.CHARACTERS:
                                switch (name) {
                                    case "id":
                                        id = reader.getText().trim();
                                        break;
                                    case "str":
                                        sourceAvailable |= "-sources.jar".equals(reader.getText().trim());
                                        break;
                                }
                                break;
                        }
                    }

                    reader.close();
                }

                String groupId=null, artifactId=null, version=null;

                if ("0".equals(numFound)) {
                    // File not indexed by Apache Solr of maven.org -> Try to found groupId, artifactId, version in 'pom.properties'
                    Properties pomProperties = getPomProperties(entry);

                    if (pomProperties != null) {
                        groupId = pomProperties.getProperty("groupId");
                        artifactId = pomProperties.getProperty("artifactId");
                        version = pomProperties.getProperty("version");
                    }
                } else if ("1".equals(numFound) && sourceAvailable) {
                    int index1 = id.indexOf(':');
                    int index2 = id.lastIndexOf(':');

                    groupId = id.substring(0, index1);
                    artifactId = id.substring(index1+1, index2);
                    version = id.substring(index2+1);
                }

                if (artifactId != null) {
                    // Load source
                    String filePath = groupId.replace('.', '/') + '/' + artifactId + '/' + version + '/' + artifactId + '-' + version;
                    URL loadUrl = new URL(MAVENORG_LOAD_URL_PREFIX + filePath + MAVENORG_LOAD_URL_SUFFIX);
                    File tmpFile = File.createTempFile("jd-gui.tmp.", '.' + groupId + '_' + artifactId + '_' + version + "-sources.jar");

                    tmpFile.delete();
                    tmpFile.deleteOnExit();

                    try (InputStream is = new BufferedInputStream(loadUrl.openStream()); OutputStream os = new BufferedOutputStream(new FileOutputStream(tmpFile))) {
                        int read = is.read(buffer);
                        while (read > 0) {
                            os.write(buffer, 0, read);
                            read = is.read(buffer);
                        }
                    }

                    cache.put(entry, tmpFile);
                    return tmpFile;
                }
            } catch (Exception e) {
                assert ExceptionUtil.printStackTrace(e);
            }
        }

        failed.add(entry);
        return null;
    }

    private static Properties getPomProperties(Container.Entry parent) {
        // Search 'META-INF/maven/*/*/pom.properties'
        for (Container.Entry child1 : parent.getChildren()) {
            if (child1.isDirectory() && child1.getPath().equals("META-INF")) {
                for (Container.Entry child2 : child1.getChildren()) {
                    if (child2.isDirectory() && child2.getPath().equals("META-INF/maven")) {
                        if (child2.isDirectory()) {
                            Collection<Container.Entry> children = child2.getChildren();
                            if (children.size() == 1) {
                                Container.Entry entry = children.iterator().next();
                                if (entry.isDirectory()) {
                                    children = entry.getChildren();
                                    if (children.size() == 1) {
                                        entry = children.iterator().next();
                                        for (Container.Entry child3 : entry.getChildren()) {
                                            if (!child3.isDirectory() && child3.getPath().endsWith("/pom.properties")) {
                                                // Load properties
                                                try (InputStream is = child3.getInputStream()) {
                                                    Properties properties = new Properties();
                                                    properties.load(is);
                                                    return properties;
                                                } catch (Exception e) {
                                                    assert ExceptionUtil.printStackTrace(e);
                                                }
                                            }
                                        }
                                    }
                                }
                            }
                        }
                    }
                }
            }
        }

        return null;
    }

    private static char hexa(int i) { return (char)( (i <= 9) ? ('0' + i) : (('a' - 10) + i) ); }

    protected boolean accepted(String filters, String path) {
        // 'filters' example : '+org +com.google +com.ibm +com.jcraft +com.springsource +com.sun -com +java +javax +sun +sunw'
        StringTokenizer tokenizer = new StringTokenizer(filters);

        while (tokenizer.hasMoreTokens()) {
            String filter = tokenizer.nextToken();

            if (filter.length() > 1) {
                String prefix = filter.substring(1).replace('.', '/');

                if (prefix.charAt(prefix.length() - 1) != '/') {
                    prefix += '/';
                }

                if (path.startsWith(prefix)) {
                    return (filter.charAt(0) == '+');
                }
            }
        }

        return false;
    }
}