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

org.gradle.api.internal.classpath.DefaultModuleRegistry Maven / Gradle / Ivy

There is a newer version: 8.11.1
Show newest version
/*
 * Copyright 2011 the original author or authors.
 *
 * Licensed 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.gradle.api.internal.classpath;

import com.google.common.collect.ImmutableList;
import org.gradle.api.UncheckedIOException;
import org.gradle.api.specs.Spec;
import org.gradle.internal.classpath.CachedJarFileStore;
import org.gradle.internal.classpath.ClassPath;
import org.gradle.internal.classpath.DefaultClassPath;
import org.gradle.internal.installation.GradleInstallation;
import org.gradle.util.GUtil;

import javax.annotation.Nullable;
import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.zip.ZipEntry;
import java.util.zip.ZipFile;

/**
 * Determines the classpath for a module by looking for a '${module}-classpath.properties' resource with 'name' set to the name of the module.
 */
public class DefaultModuleRegistry implements ModuleRegistry, CachedJarFileStore {
    private static final Spec SATISFY_ALL = new Spec() {
        @Override
        public boolean isSatisfiedBy(File element) {
            return true;
        }
    };

    @Nullable
    private final GradleInstallation gradleInstallation;
    private final Map modules = new HashMap();
    private final Map externalModules = new HashMap();
    private final List classpath = new ArrayList();
    private final Map classpathJars = new LinkedHashMap();

    public DefaultModuleRegistry(@Nullable GradleInstallation gradleInstallation) {
        this(ClassPath.EMPTY, gradleInstallation);
    }

    public DefaultModuleRegistry(ClassPath additionalModuleClassPath, @Nullable GradleInstallation gradleInstallation) {
        this(DefaultModuleRegistry.class.getClassLoader(), additionalModuleClassPath, gradleInstallation);
    }

    private DefaultModuleRegistry(ClassLoader classLoader, ClassPath additionalModuleClassPath, @Nullable GradleInstallation gradleInstallation) {
        this.gradleInstallation = gradleInstallation;

        for (File classpathFile : new EffectiveClassPath(classLoader).plus(additionalModuleClassPath).getAsFiles()) {
            classpath.add(classpathFile);
            if (classpathFile.isFile() && !classpathJars.containsKey(classpathFile.getName())) {
                classpathJars.put(classpathFile.getName(), classpathFile);
            }
        }
    }

    @Override
    public List getFileStoreRoots() {
        ImmutableList.Builder builder = ImmutableList.builder();
        if (gradleInstallation != null) {
            builder.addAll(gradleInstallation.getLibDirs());
        }
        builder.addAll(classpath);
        return builder.build();
    }

    @Override
    public ClassPath getAdditionalClassPath() {
        return gradleInstallation == null ? DefaultClassPath.of(classpath) : ClassPath.EMPTY;
    }

    @Override
    public Module getExternalModule(String name) {
        Module module = externalModules.get(name);
        if (module == null) {
            module = loadExternalModule(name);
            externalModules.put(name, module);
        }
        return module;
    }

    private Module loadExternalModule(String name) {
        File externalJar = findJar(name, SATISFY_ALL);
        if (externalJar == null) {
            if (gradleInstallation == null) {
                throw new UnknownModuleException(String.format("Cannot locate JAR for module '%s' in in classpath: %s.", name, classpath));
            }
            throw new UnknownModuleException(String.format("Cannot locate JAR for module '%s' in distribution directory '%s'.", name, gradleInstallation.getGradleHome()));
        }
        return new DefaultModule(name, Collections.singleton(externalJar), Collections.emptySet());
    }

    @Override
    public Module getModule(String name) {
        Module module = modules.get(name);
        if (module == null) {
            module = loadModule(name);
            modules.put(name, module);
        }
        return module;
    }

    @Override
    @Nullable
    public Module findModule(String name) {
        Module module = modules.get(name);
        if (module == null) {
            module = loadOptionalModule(name);
            if (module != null) {
                modules.put(name, module);
            }
        }
        return module;
    }

    private Module loadModule(String moduleName) {
        Module module = loadOptionalModule(moduleName);
        if (module != null) {
            return module;
        }
        if (gradleInstallation == null) {
            throw new UnknownModuleException(String.format("Cannot locate manifest for module '%s' in classpath: %s.", moduleName, classpath));
        }
        throw new UnknownModuleException(String.format("Cannot locate JAR for module '%s' in distribution directory '%s'.", moduleName, gradleInstallation.getGradleHome()));
    }

    private Module loadOptionalModule(final String moduleName) {
        File jarFile = findJar(moduleName, new Spec() {
            @Override
            public boolean isSatisfiedBy(File jarFile) {
                return hasModuleProperties(moduleName, jarFile);
            }
        });
        if (jarFile != null) {
            Set implementationClasspath = new LinkedHashSet();
            implementationClasspath.add(jarFile);
            Properties properties = loadModuleProperties(moduleName, jarFile);
            return module(moduleName, properties, implementationClasspath);
        }

        String resourceName = getClasspathManifestName(moduleName);
        Set implementationClasspath = new LinkedHashSet();
        findImplementationClasspath(moduleName, implementationClasspath);
        for (File file : implementationClasspath) {
            if (file.isDirectory()) {
                File propertiesFile = new File(file, resourceName);
                if (propertiesFile.isFile()) {
                    Properties properties = GUtil.loadProperties(propertiesFile);
                    return module(moduleName, properties, implementationClasspath);
                }
            }
        }
        return null;
    }

    private Module module(String moduleName, Properties properties, Set implementationClasspath) {
        String[] runtimeJarNames = split(properties.getProperty("runtime"));
        Set runtimeClasspath = findDependencyJars(moduleName, runtimeJarNames);

        String[] projects = split(properties.getProperty("projects"));
        String[] optionalProjects = split(properties.getProperty("optional"));
        return new DefaultModule(moduleName, implementationClasspath, runtimeClasspath, projects, optionalProjects);
    }

    private Set findDependencyJars(String moduleName, String[] jarNames) {
        Set runtimeClasspath = new LinkedHashSet();
        for (String jarName : jarNames) {
            runtimeClasspath.add(findDependencyJar(moduleName, jarName));
        }
        return runtimeClasspath;
    }

    private Set getModules(String[] projectNames) {
        Set modules = new LinkedHashSet();
        for (String project : projectNames) {
            modules.add(getModule(project));
        }
        return modules;
    }

    private String[] split(String value) {
        if (value == null) {
            return new String[0];
        }
        value = value.trim();
        if (value.length() == 0) {
            return new String[0];
        }
        return value.split(",");
    }

    private void findImplementationClasspath(String name, Collection implementationClasspath) {
        String projectDirName = name.startsWith("kotlin-compiler-embeddable-")
            ? "kotlin-compiler-embeddable"
            : projectDirNameFrom(name);
        List suffixesForProjectDir = getClasspathSuffixesForProjectDir(projectDirName);
        for (File file : classpath) {
            if (file.isDirectory()) {
                String path = file.getAbsolutePath();
                for (String suffix : suffixesForProjectDir) {
                    if (path.endsWith(suffix)) {
                        implementationClasspath.add(file);
                    }
                }
            }
        }
    }

    private static String projectDirNameFrom(String moduleName) {
        Matcher matcher = Pattern.compile("gradle-(.+)").matcher(moduleName);
        matcher.matches();
        return matcher.group(1);
    }

    /**
     * Provides the locations where the classes and resources of a Gradle module can be found
     * when running in embedded mode from the IDE.
     *
     * 
    *
  • In Eclipse, they are in the bin/ folder.
  • *
  • In IDEA (native import), they are in in the out/production/ folder.
  • *
*
  • In both cases we also include the static and generated resources of the project.
  • */ private List getClasspathSuffixesForProjectDir(String projectDirName) { List suffixes = new ArrayList(); suffixes.add(("/" + projectDirName + "/out/production/classes").replace('/', File.separatorChar)); suffixes.add(("/" + projectDirName + "/out/production/resources").replace('/', File.separatorChar)); suffixes.add(("/" + projectDirName + "/bin").replace('/', File.separatorChar)); suffixes.add(("/" + projectDirName + "/src/main/resources").replace('/', File.separatorChar)); suffixes.add(("/" + projectDirName + "/build/classes/java/main").replace('/', File.separatorChar)); suffixes.add(("/" + projectDirName + "/build/classes/groovy/main").replace('/', File.separatorChar)); suffixes.add(("/" + projectDirName + "/build/resources/main").replace('/', File.separatorChar)); suffixes.add(("/" + projectDirName + "/build/generated-resources/main").replace('/', File.separatorChar)); suffixes.add(("/" + projectDirName + "/build/generated-resources/test").replace('/', File.separatorChar)); return suffixes; } private String toCamelCase(String name) { StringBuffer result = new StringBuffer(); Matcher matcher = Pattern.compile("-([^-])").matcher(name); while (matcher.find()) { matcher.appendReplacement(result, ""); result.append(matcher.group(1).toUpperCase()); } matcher.appendTail(result); return result.toString(); } private Properties loadModuleProperties(String name, File jarFile) { try { ZipFile zipFile = new ZipFile(jarFile); try { String entryName = getClasspathManifestName(name); ZipEntry entry = zipFile.getEntry(entryName); if (entry == null) { throw new IllegalStateException("Did not find " + entryName + " in " + jarFile.getAbsolutePath()); } return GUtil.loadProperties(zipFile.getInputStream(entry)); } finally { zipFile.close(); } } catch (IOException e) { throw new UncheckedIOException(String.format("Could not load properties for module '%s' from %s", name, jarFile), e); } } private boolean hasModuleProperties(String name, File jarFile) { try { ZipFile zipFile = new ZipFile(jarFile); try { String entryName = getClasspathManifestName(name); ZipEntry entry = zipFile.getEntry(entryName); return entry != null; } finally { zipFile.close(); } } catch (IOException e) { throw new UncheckedIOException(String.format("Could not load properties for module '%s' from %s", name, jarFile), e); } } private String getClasspathManifestName(String moduleName) { return moduleName + "-classpath.properties"; } private File findJar(String name, Spec allowedJarFiles) { Pattern pattern = Pattern.compile(Pattern.quote(name) + "-\\d.*\\.jar"); if (gradleInstallation != null) { for (File libDir : gradleInstallation.getLibDirs()) { for (File file : libDir.listFiles()) { if (pattern.matcher(file.getName()).matches()) { return file; } } } } for (File file : classpath) { if (pattern.matcher(file.getName()).matches() && allowedJarFiles.isSatisfiedBy(file)) { return file; } } return null; } private File findDependencyJar(String module, String name) { File jarFile = classpathJars.get(name); if (jarFile != null) { return jarFile; } if (gradleInstallation == null) { throw new IllegalArgumentException(String.format("Cannot find JAR '%s' required by module '%s' using classpath.", name, module)); } for (File libDir : gradleInstallation.getLibDirs()) { jarFile = new File(libDir, name); if (jarFile.isFile()) { return jarFile; } } throw new IllegalArgumentException(String.format("Cannot find JAR '%s' required by module '%s' using classpath or distribution directory '%s'", name, module, gradleInstallation.getGradleHome())); } private class DefaultModule implements Module { private final String name; private final String[] projects; private final String[] optionalProjects; private final ClassPath implementationClasspath; private final ClassPath runtimeClasspath; private final ClassPath classpath; public DefaultModule(String name, Set implementationClasspath, Set runtimeClasspath, String[] projects, String[] optionalProjects) { this.name = name; this.projects = projects; this.optionalProjects = optionalProjects; this.implementationClasspath = DefaultClassPath.of(implementationClasspath); this.runtimeClasspath = DefaultClassPath.of(runtimeClasspath); Set classpath = new LinkedHashSet(); classpath.addAll(implementationClasspath); classpath.addAll(runtimeClasspath); this.classpath = DefaultClassPath.of(classpath); } public DefaultModule(String name, Set singleton, Set files) { this(name, singleton, files, NO_PROJECTS, NO_PROJECTS); } @Override public String toString() { return "module '" + name + "'"; } @Override public Set getRequiredModules() { return getModules(projects); } @Override public ClassPath getImplementationClasspath() { return implementationClasspath; } @Override public ClassPath getRuntimeClasspath() { return runtimeClasspath; } @Override public ClassPath getClasspath() { return classpath; } @Override public Set getAllRequiredModules() { Set modules = new LinkedHashSet(); collectRequiredModules(modules); return modules; } @Override public ClassPath getAllRequiredModulesClasspath() { ClassPath classPath = ClassPath.EMPTY; for (Module module : getAllRequiredModules()) { classPath = classPath.plus(module.getClasspath()); } return classPath; } private void collectRequiredModules(Set modules) { if (!modules.add(this)) { return; } for (Module module : getRequiredModules()) { collectDependenciesOf(module, modules); } for (String optionalProject : optionalProjects) { Module module = findModule(optionalProject); if (module != null) { collectDependenciesOf(module, modules); } } } private void collectDependenciesOf(Module module, Set modules) { ((DefaultModule) module).collectRequiredModules(modules); } private Module findModule(String optionalProject) { try { return getModule(optionalProject); } catch (UnknownModuleException ex) { return null; } } } private static final String[] NO_PROJECTS = new String[0]; }




    © 2015 - 2025 Weber Informatics LLC | Privacy Policy