
org.gradle.cache.internal.DefaultPersistentDirectoryStore Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-api Show documentation
Show all versions of gradle-api Show documentation
Gradle 6.9.1 API redistribution.
/*
* 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.cache.internal;
import org.gradle.cache.CacheBuilder;
import org.gradle.cache.CacheOpenException;
import org.gradle.cache.CleanupAction;
import org.gradle.cache.FileLock;
import org.gradle.cache.FileLockManager;
import org.gradle.cache.LockOptions;
import org.gradle.cache.PersistentIndexedCache;
import org.gradle.cache.PersistentIndexedCacheParameters;
import org.gradle.internal.Factory;
import org.gradle.internal.concurrent.ExecutorFactory;
import org.gradle.internal.logging.progress.ProgressLogger;
import org.gradle.internal.logging.progress.ProgressLoggerFactory;
import org.gradle.internal.serialize.Serializer;
import org.gradle.internal.time.Time;
import org.gradle.internal.time.Timer;
import org.gradle.util.GFileUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.File;
import java.util.Arrays;
import java.util.Collection;
import java.util.concurrent.TimeUnit;
public class DefaultPersistentDirectoryStore implements ReferencablePersistentCache {
private static final Logger LOGGER = LoggerFactory.getLogger(DefaultPersistentDirectoryStore.class);
public static final int CLEANUP_INTERVAL_IN_HOURS = 24;
private final File dir;
private final CacheBuilder.LockTarget lockTarget;
private final LockOptions lockOptions;
private final CleanupAction cleanupAction;
private final FileLockManager lockManager;
private final ExecutorFactory executorFactory;
private final String displayName;
protected final File propertiesFile;
private final File gcFile;
private final ProgressLoggerFactory progressLoggerFactory;
private CacheCoordinator cacheAccess;
public DefaultPersistentDirectoryStore(File dir, String displayName, CacheBuilder.LockTarget lockTarget, LockOptions lockOptions, CleanupAction cleanupAction, FileLockManager fileLockManager, ExecutorFactory executorFactory, ProgressLoggerFactory progressLoggerFactory) {
this.dir = dir;
this.lockTarget = lockTarget;
this.lockOptions = lockOptions;
this.cleanupAction = cleanupAction;
this.lockManager = fileLockManager;
this.executorFactory = executorFactory;
this.propertiesFile = new File(dir, "cache.properties");
this.gcFile = new File(dir, "gc.properties");
this.progressLoggerFactory = progressLoggerFactory;
this.displayName = displayName != null ? (displayName + " (" + dir + ")") : ("cache directory " + dir.getName() + " (" + dir + ")");
}
@Override
public DefaultPersistentDirectoryStore open() {
GFileUtils.mkdirs(dir);
cacheAccess = createCacheAccess();
try {
cacheAccess.open();
} catch (Throwable e) {
throw new CacheOpenException(String.format("Could not open %s.", this), e);
}
return this;
}
private CacheCoordinator createCacheAccess() {
return new DefaultCacheAccess(displayName, getLockTarget(), lockOptions, dir, lockManager, getInitAction(), getCleanupAction(), executorFactory);
}
private File getLockTarget() {
switch (lockTarget) {
case CacheDirectory:
case DefaultTarget:
return dir;
case CachePropertiesFile:
return propertiesFile;
default:
throw new IllegalArgumentException("Unsupported lock target: " + lockTarget);
}
}
protected CacheInitializationAction getInitAction() {
return new CacheInitializationAction() {
public boolean requiresInitialization(FileLock fileLock) {
return false;
}
public void initialize(FileLock fileLock) {
throw new UnsupportedOperationException();
}
};
}
protected CacheCleanupAction getCleanupAction() {
return new Cleanup();
}
@Override
public void close() {
if (cacheAccess != null) {
try {
cacheAccess.close();
} finally {
cacheAccess = null;
}
}
}
@Override
public File getBaseDir() {
return dir;
}
@Override
public Collection getReservedCacheFiles() {
return Arrays.asList(propertiesFile, gcFile, determineLockTargetFile(getLockTarget()));
}
// TODO: Duplicated in DefaultFileLockManager
static File determineLockTargetFile(File target) {
if (target.isDirectory()) {
return new File(target, target.getName() + ".lock");
} else {
return new File(target.getParentFile(), target.getName() + ".lock");
}
}
@Override
public String toString() {
return getDisplayName();
}
@Override
public String getDisplayName() {
return displayName;
}
@Override
public PersistentIndexedCache createCache(PersistentIndexedCacheParameters parameters) {
return cacheAccess.newCache(parameters);
}
@Override
public PersistentIndexedCache createCache(String name, Class keyType, Serializer valueSerializer) {
return cacheAccess.newCache(PersistentIndexedCacheParameters.of(name, keyType, valueSerializer));
}
@Override
public T withFileLock(Factory extends T> action) {
return cacheAccess.withFileLock(action);
}
@Override
public void withFileLock(Runnable action) {
cacheAccess.withFileLock(action);
}
@Override
public T useCache(Factory extends T> action) {
return cacheAccess.useCache(action);
}
@Override
public void useCache(Runnable action) {
cacheAccess.useCache(action);
}
private class Cleanup implements CacheCleanupAction {
@Override
public boolean requiresCleanup() {
if (cleanupAction != null) {
if (!gcFile.exists()) {
GFileUtils.touch(gcFile);
} else {
long duration = System.currentTimeMillis() - gcFile.lastModified();
long timeInHours = TimeUnit.MILLISECONDS.toHours(duration);
LOGGER.debug("{} has last been fully cleaned up {} hours ago", DefaultPersistentDirectoryStore.this, timeInHours);
return timeInHours >= CLEANUP_INTERVAL_IN_HOURS;
}
}
return false;
}
@Override
public void cleanup() {
if (cleanupAction != null) {
String description = "Cleaning " + getDisplayName();
ProgressLogger progressLogger = progressLoggerFactory.newOperation(CacheCleanupAction.class).start(description, description);
Timer timer = Time.startTimer();
try {
cleanupAction.clean(DefaultPersistentDirectoryStore.this, new DefaultCleanupProgressMonitor(progressLogger));
GFileUtils.touch(gcFile);
} finally {
LOGGER.info("{} cleaned up in {}.", DefaultPersistentDirectoryStore.this, timer.getElapsed());
progressLogger.completed();
}
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy