org.gradle.caching.local.internal.DirectoryBuildCacheServiceFactory Maven / Gradle / Ivy
/*
* Copyright 2017 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.caching.local.internal;
import org.gradle.api.UncheckedIOException;
import org.gradle.api.internal.file.temp.TemporaryFileProvider;
import org.gradle.cache.CacheBuilder;
import org.gradle.cache.CacheRepository;
import org.gradle.cache.PersistentCache;
import org.gradle.cache.internal.CleanupActionFactory;
import org.gradle.cache.internal.LeastRecentlyUsedCacheCleanup;
import org.gradle.cache.internal.SingleDepthFilesFinder;
import org.gradle.cache.scopes.GlobalScopedCache;
import org.gradle.caching.BuildCacheService;
import org.gradle.caching.BuildCacheServiceFactory;
import org.gradle.caching.local.DirectoryBuildCache;
import org.gradle.internal.file.FileAccessTimeJournal;
import org.gradle.internal.file.FileAccessTracker;
import org.gradle.internal.file.PathToFileResolver;
import org.gradle.internal.file.impl.SingleDepthFileAccessTracker;
import org.gradle.internal.resource.local.PathKeyFileStore;
import javax.inject.Inject;
import java.io.File;
import static org.gradle.cache.FileLockManager.LockMode.OnDemand;
import static org.gradle.cache.internal.filelock.LockOptionsBuilder.mode;
public class DirectoryBuildCacheServiceFactory implements BuildCacheServiceFactory {
public static final String FAILED_READ_SUFFIX = ".failed";
private static final String BUILD_CACHE_VERSION = "1";
private static final String BUILD_CACHE_KEY = "build-cache-" + BUILD_CACHE_VERSION;
private static final String DIRECTORY_BUILD_CACHE_TYPE = "directory";
private static final int FILE_TREE_DEPTH_TO_TRACK_AND_CLEANUP = 1;
private final CacheRepository cacheRepository;
private final GlobalScopedCache globalScopedCache;
private final PathToFileResolver resolver;
private final DirectoryBuildCacheFileStoreFactory fileStoreFactory;
private final CleanupActionFactory cleanupActionFactory;
private final FileAccessTimeJournal fileAccessTimeJournal;
private final TemporaryFileProvider temporaryFileProvider;
@Inject
public DirectoryBuildCacheServiceFactory(CacheRepository cacheRepository, GlobalScopedCache globalScopedCache, PathToFileResolver resolver, DirectoryBuildCacheFileStoreFactory fileStoreFactory,
CleanupActionFactory cleanupActionFactory, FileAccessTimeJournal fileAccessTimeJournal, TemporaryFileProvider temporaryFileProvider) {
this.cacheRepository = cacheRepository;
this.globalScopedCache = globalScopedCache;
this.resolver = resolver;
this.fileStoreFactory = fileStoreFactory;
this.cleanupActionFactory = cleanupActionFactory;
this.fileAccessTimeJournal = fileAccessTimeJournal;
this.temporaryFileProvider = temporaryFileProvider;
}
@Override
public BuildCacheService createBuildCacheService(DirectoryBuildCache configuration, Describer describer) {
Object cacheDirectory = configuration.getDirectory();
File target;
if (cacheDirectory != null) {
target = resolver.resolve(cacheDirectory);
} else {
target = globalScopedCache.baseDirForCrossVersionCache(BUILD_CACHE_KEY);
}
checkDirectory(target);
int removeUnusedEntriesAfterDays = configuration.getRemoveUnusedEntriesAfterDays();
describer.type(DIRECTORY_BUILD_CACHE_TYPE).
config("location", target.getAbsolutePath()).
config("removeUnusedEntriesAfter", String.valueOf(removeUnusedEntriesAfterDays) + " days");
PathKeyFileStore fileStore = fileStoreFactory.createFileStore(target);
PersistentCache persistentCache = cacheRepository
.cache(target)
.withCleanup(cleanupActionFactory.create(new LeastRecentlyUsedCacheCleanup(new SingleDepthFilesFinder(FILE_TREE_DEPTH_TO_TRACK_AND_CLEANUP), fileAccessTimeJournal, removeUnusedEntriesAfterDays)))
.withDisplayName("Build cache")
.withLockOptions(mode(OnDemand))
.withCrossVersionCache(CacheBuilder.LockTarget.DefaultTarget)
.open();
BuildCacheTempFileStore tempFileStore = new DefaultBuildCacheTempFileStore(temporaryFileProvider);
FileAccessTracker fileAccessTracker = new SingleDepthFileAccessTracker(fileAccessTimeJournal, target, FILE_TREE_DEPTH_TO_TRACK_AND_CLEANUP);
return new DirectoryBuildCacheService(fileStore, persistentCache, tempFileStore, fileAccessTracker, FAILED_READ_SUFFIX);
}
private static void checkDirectory(File directory) {
if (directory.exists()) {
if (!directory.isDirectory()) {
throw new IllegalArgumentException(String.format("Cache directory %s must be a directory", directory));
}
if (!directory.canRead()) {
throw new IllegalArgumentException(String.format("Cache directory %s must be readable", directory));
}
if (!directory.canWrite()) {
throw new IllegalArgumentException(String.format("Cache directory %s must be writable", directory));
}
} else {
if (!directory.mkdirs()) {
throw new UncheckedIOException(String.format("Could not create cache directory: %s", directory));
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy