/* * Copyright 2013 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.initialization.buildsrc; import org.gradle.StartParameter; import org.gradle.api.internal.GradleInternal; import org.gradle.api.internal.initialization.ClassLoaderScope; import org.gradle.cache.CacheBuilder; import org.gradle.cache.CacheRepository; import org.gradle.cache.PersistentCache; import org.gradle.cache.internal.FileLockManager; import org.gradle.initialization.GradleLauncher; import org.gradle.initialization.NestedBuildFactory; import org.gradle.internal.classpath.CachedClasspathTransformer; import org.gradle.internal.classpath.ClassPath; import org.gradle.internal.classpath.DefaultClassPath; import org.gradle.internal.operations.BuildOperationContext; import org.gradle.internal.operations.BuildOperationExecutor; import org.gradle.internal.operations.CallableBuildOperation; import org.gradle.internal.progress.BuildOperationDescriptor; import org.gradle.util.GradleVersion; import org.gradle.util.Path; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import java.io.File; import java.util.Collections; import static org.gradle.cache.internal.filelock.LockOptionsBuilder.mode; public class BuildSourceBuilder { private static final Logger LOGGER = LoggerFactory.getLogger(BuildSourceBuilder.class); private final NestedBuildFactory nestedBuildFactory; private final ClassLoaderScope classLoaderScope; private final CacheRepository cacheRepository; private final BuildOperationExecutor buildOperationExecutor; private final CachedClasspathTransformer cachedClasspathTransformer; private final BuildSrcBuildListenerFactory buildSrcBuildListenerFactory; public BuildSourceBuilder(NestedBuildFactory nestedBuildFactory, ClassLoaderScope classLoaderScope, CacheRepository cacheRepository, BuildOperationExecutor buildOperationExecutor, CachedClasspathTransformer cachedClasspathTransformer, BuildSrcBuildListenerFactory buildSrcBuildListenerFactory) { this.nestedBuildFactory = nestedBuildFactory; this.classLoaderScope = classLoaderScope; this.cacheRepository = cacheRepository; this.buildOperationExecutor = buildOperationExecutor; this.cachedClasspathTransformer = cachedClasspathTransformer; this.buildSrcBuildListenerFactory = buildSrcBuildListenerFactory; } public ClassLoaderScope buildAndCreateClassLoader(StartParameter startParameter) { ClassPath classpath = createBuildSourceClasspath(startParameter); ClassLoaderScope childScope = classLoaderScope.createChild(startParameter.getCurrentDir().getAbsolutePath()); childScope.export(cachedClasspathTransformer.transform(classpath)); childScope.lock(); return childScope; } ClassPath createBuildSourceClasspath(final StartParameter startParameter) { assert startParameter.getCurrentDir() != null && startParameter.getBuildFile() == null; LOGGER.debug("Starting to build the build sources."); if (!startParameter.getCurrentDir().isDirectory()) { LOGGER.debug("Gradle source dir does not exist. We leave."); return new DefaultClassPath(); } return buildOperationExecutor.call(new CallableBuildOperation<ClassPath>() { @Override public ClassPath call(BuildOperationContext context) { return buildBuildSrc(startParameter); } @Override public BuildOperationDescriptor.Builder description() { return BuildOperationDescriptor.displayName("Build buildSrc").progressDisplayName("buildSrc"); } }); } private ClassPath buildBuildSrc(StartParameter startParameter) { // If we were not the most recent version of Gradle to build the buildSrc dir, then do a clean build // Otherwise, just to a regular build final PersistentCache buildSrcCache = createCache(startParameter); try { GradleLauncher gradleLauncher = buildGradleLauncher(startParameter); try { return buildSrcCache.useCache(new BuildSrcUpdateFactory(buildSrcCache, gradleLauncher, buildSrcBuildListenerFactory)); } finally { gradleLauncher.stop(); } } finally { // This isn't quite right. We should not unlock the classes until we're finished with them, and the classes may be used across multiple builds buildSrcCache.close(); } } PersistentCache createCache(StartParameter startParameter) { return cacheRepository .cache(new File(startParameter.getCurrentDir(), ".gradle/noVersion/buildSrc")) .withCrossVersionCache(CacheBuilder.LockTarget.CachePropertiesFile) .withDisplayName("buildSrc state cache") .withLockOptions(mode(FileLockManager.LockMode.None).useCrossVersionImplementation()) .withProperties(Collections.singletonMap("gradle.version", GradleVersion.current().getVersion())) .open(); } private GradleLauncher buildGradleLauncher(StartParameter startParameter) { StartParameter startParameterArg = startParameter.newInstance(); startParameterArg.setProjectProperties(startParameter.getProjectProperties()); startParameterArg.setSearchUpwards(false); startParameterArg.setProfile(startParameter.isProfile()); GradleLauncher gradleLauncher = nestedBuildFactory.nestedInstance(startParameterArg); GradleInternal build = gradleLauncher.getGradle(); if (build.getParent().findIdentityPath() == null) { // When nested inside a nested build, we need to synthesize a path for this build, as the root project is not yet known for the parent build // Use the directory structure to do this. This means that the buildSrc build and its containing build may end up with different paths Path path = build.getParent().getParent().getIdentityPath().child(startParameter.getCurrentDir().getParentFile().getName()).child(startParameter.getCurrentDir().getName()); build.setIdentityPath(path); } return gradleLauncher; } }