org.gradle.api.internal.changedetection.state.DefaultFileSystemSnapshotter 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 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.api.internal.changedetection.state;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Lists;
import org.gradle.api.file.FileTreeElement;
import org.gradle.api.file.FileVisitDetails;
import org.gradle.api.file.FileVisitor;
import org.gradle.api.file.RelativePath;
import org.gradle.api.internal.cache.StringInterner;
import org.gradle.api.internal.file.FileTreeInternal;
import org.gradle.api.internal.file.collections.DirectoryFileTree;
import org.gradle.api.internal.file.collections.DirectoryFileTreeFactory;
import org.gradle.api.internal.file.collections.SimpleFileCollection;
import org.gradle.cache.internal.ProducerGuard;
import org.gradle.caching.internal.BuildCacheHasher;
import org.gradle.caching.internal.DefaultBuildCacheHasher;
import org.gradle.internal.Factory;
import org.gradle.internal.file.FileMetadataSnapshot;
import org.gradle.internal.hash.FileHasher;
import org.gradle.internal.hash.HashCode;
import org.gradle.internal.nativeintegration.filesystem.FileSystem;
import org.gradle.normalization.internal.InputNormalizationStrategy;
import java.io.File;
import java.util.List;
/**
* Responsible for snapshotting various aspects of the file system.
*
* Currently logic and state are split between this class and {@link FileSystemMirror}, as there are several instances of this class created in different scopes. This introduces some inefficiencies
* that could be improved by shuffling this relationship around.
*
* The implementations attempt to do 2 things: avoid doing the same work in parallel (e.g. scanning the same directory from multiple threads, and avoid doing work where the result is almost certainly
* the same as before (e.g. don't scan the output directory of a task a bunch of times).
*
* The implementations are currently intentionally very, very simple, and so there are a number of ways in which they can be made much more efficient. This can happen over time.
*/
public class DefaultFileSystemSnapshotter implements FileSystemSnapshotter {
private final FileHasher hasher;
private final StringInterner stringInterner;
private final FileSystem fileSystem;
private final DirectoryFileTreeFactory directoryFileTreeFactory;
private final FileSystemMirror fileSystemMirror;
private final ProducerGuard producingSelfSnapshots = ProducerGuard.striped();
private final ProducerGuard producingTrees = ProducerGuard.striped();
private final ProducerGuard producingAllSnapshots = ProducerGuard.striped();
private final DefaultGenericFileCollectionSnapshotter snapshotter;
public DefaultFileSystemSnapshotter(FileHasher hasher, StringInterner stringInterner, FileSystem fileSystem, DirectoryFileTreeFactory directoryFileTreeFactory, FileSystemMirror fileSystemMirror) {
this.hasher = hasher;
this.stringInterner = stringInterner;
this.fileSystem = fileSystem;
this.directoryFileTreeFactory = directoryFileTreeFactory;
this.fileSystemMirror = fileSystemMirror;
snapshotter = new DefaultGenericFileCollectionSnapshotter(stringInterner, directoryFileTreeFactory, this);
}
@Override
public FileSnapshot snapshotSelf(final File file) {
// Could potentially coordinate with a thread that is snapshotting an overlapping directory tree
final String path = file.getAbsolutePath();
return producingSelfSnapshots.guardByKey(path, new Factory() {
@Override
public FileSnapshot create() {
FileSnapshot snapshot = fileSystemMirror.getFile(path);
if (snapshot == null) {
snapshot = calculateDetails(file);
fileSystemMirror.putFile(snapshot);
}
return snapshot;
}
});
}
@Override
public Snapshot snapshotAll(final File file) {
// Could potentially coordinate with a thread that is snapshotting an overlapping directory tree
final String path = file.getAbsolutePath();
return producingAllSnapshots.guardByKey(path, new Factory() {
@Override
public Snapshot create() {
Snapshot snapshot = fileSystemMirror.getContent(path);
if (snapshot == null) {
FileCollectionSnapshot fileCollectionSnapshot = snapshotter.snapshot(new SimpleFileCollection(file), InputPathNormalizationStrategy.ABSOLUTE, InputNormalizationStrategy.NOT_CONFIGURED);
DefaultBuildCacheHasher hasher = new DefaultBuildCacheHasher();
fileCollectionSnapshot.appendToHasher(hasher);
HashCode hashCode = hasher.hash();
snapshot = new HashBackedSnapshot(hashCode);
String internedPath = getPath(file);
fileSystemMirror.putContent(internedPath, snapshot);
}
return snapshot;
}
});
}
@Override
public FileTreeSnapshot snapshotDirectoryTree(final File dir) {
// Could potentially coordinate with a thread that is snapshotting an overlapping directory tree
final String path = dir.getAbsolutePath();
return producingTrees.guardByKey(path, new Factory() {
@Override
public FileTreeSnapshot create() {
FileTreeSnapshot snapshot = fileSystemMirror.getDirectoryTree(path);
if (snapshot == null) {
// Scan the directory
snapshot = doSnapshot(directoryFileTreeFactory.create(dir));
fileSystemMirror.putDirectory(snapshot);
}
return snapshot;
}
});
}
@Override
public FileTreeSnapshot snapshotDirectoryTree(final DirectoryFileTree dirTree) {
// Could potentially coordinate with a thread that is snapshotting an overlapping directory tree
// Currently cache only those trees where we want everything from a directory
if (!dirTree.getPatterns().isEmpty()) {
List elements = Lists.newArrayList();
dirTree.visit(new FileVisitorImpl(elements));
return new DirectoryTreeDetails(dirTree.getDir().getAbsolutePath(), elements);
}
final String path = dirTree.getDir().getAbsolutePath();
return producingTrees.guardByKey(path, new Factory() {
@Override
public FileTreeSnapshot create() {
FileTreeSnapshot snapshot = fileSystemMirror.getDirectoryTree(path);
if (snapshot == null) {
// Scan the directory
snapshot = doSnapshot(dirTree);
fileSystemMirror.putDirectory(snapshot);
}
return snapshot;
}
});
}
@Override
public List snapshotTree(FileTreeInternal tree) {
List elements = Lists.newArrayList();
tree.visitTreeOrBackingFile(new FileVisitorImpl(elements));
return elements;
}
private FileTreeSnapshot doSnapshot(DirectoryFileTree directoryTree) {
String path = getPath(directoryTree.getDir());
List elements = Lists.newArrayList();
directoryTree.visit(new FileVisitorImpl(elements));
return new DirectoryTreeDetails(path, ImmutableList.copyOf(elements));
}
private String getPath(File file) {
return stringInterner.intern(file.getAbsolutePath());
}
private FileSnapshot calculateDetails(File file) {
String path = getPath(file);
FileMetadataSnapshot stat = fileSystem.stat(file);
switch (stat.getType()) {
case Missing:
return new MissingFileSnapshot(path, new RelativePath(true, file.getName()));
case Directory:
return new DirectoryFileSnapshot(path, new RelativePath(false, file.getName()), true);
case RegularFile:
return new RegularFileSnapshot(path, new RelativePath(true, file.getName()), true, fileSnapshot(file, stat));
default:
throw new IllegalArgumentException("Unrecognized file type: " + stat.getType());
}
}
private FileHashSnapshot fileSnapshot(FileTreeElement fileDetails) {
return new FileHashSnapshot(hasher.hash(fileDetails), fileDetails.getLastModified());
}
private FileHashSnapshot fileSnapshot(File file, FileMetadataSnapshot fileDetails) {
return new FileHashSnapshot(hasher.hash(file, fileDetails), fileDetails.getLastModified());
}
private static class HashBackedSnapshot implements Snapshot {
private final HashCode hashCode;
HashBackedSnapshot(HashCode hashCode) {
this.hashCode = hashCode;
}
@Override
public void appendToHasher(BuildCacheHasher hasher) {
hasher.putHash(hashCode);
}
}
private class FileVisitorImpl implements FileVisitor {
private final List fileTreeElements;
FileVisitorImpl(List fileTreeElements) {
this.fileTreeElements = fileTreeElements;
}
@Override
public void visitDir(FileVisitDetails dirDetails) {
fileTreeElements.add(new DirectoryFileSnapshot(getPath(dirDetails.getFile()), dirDetails.getRelativePath(), false));
}
@Override
public void visitFile(FileVisitDetails fileDetails) {
fileTreeElements.add(new RegularFileSnapshot(getPath(fileDetails.getFile()), fileDetails.getRelativePath(), false, fileSnapshot(fileDetails)));
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy