org.gradle.api.internal.changedetection.state.OrderInsensitiveTaskFilePropertyCompareStrategy 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 2016 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.AbstractIterator;
import com.google.common.collect.Iterators;
import com.google.common.collect.ListMultimap;
import com.google.common.collect.Lists;
import com.google.common.collect.MultimapBuilder;
import org.gradle.api.internal.changedetection.rules.ChangeType;
import org.gradle.api.internal.changedetection.rules.FileChange;
import org.gradle.api.internal.changedetection.rules.TaskStateChange;
import org.gradle.caching.internal.BuildCacheHasher;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.Iterator;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
class OrderInsensitiveTaskFilePropertyCompareStrategy implements TaskFilePropertyCompareStrategy.Impl {
private static final Comparator> ENTRY_COMPARATOR = new Comparator>() {
@Override
public int compare(Entry o1, Entry o2) {
return o1.getKey().compareTo(o2.getKey());
}
};
private final boolean includeAdded;
public OrderInsensitiveTaskFilePropertyCompareStrategy(boolean includeAdded) {
this.includeAdded = includeAdded;
}
@Override
public Iterator iterateContentChangesSince(Map current, Map previous, String fileType, boolean pathIsAbsolute) {
if (pathIsAbsolute) {
return iterateChangesForAbsolutePaths(current, previous, fileType);
} else {
return iterateChangesForRelativePaths(current, previous, fileType);
}
}
/**
* A more efficient implementation when absolute paths are used.
*/
private Iterator iterateChangesForAbsolutePaths(final Map current, final Map previous, final String fileType) {
final Set unaccountedForPreviousSnapshots = new LinkedHashSet(previous.keySet());
final Iterator> currentEntries = current.entrySet().iterator();
final List added = new ArrayList();
return new AbstractIterator() {
private Iterator unaccountedForPreviousSnapshotsIterator;
private Iterator addedIterator;
@Override
protected TaskStateChange computeNext() {
while (currentEntries.hasNext()) {
Entry currentEntry = currentEntries.next();
String currentAbsolutePath = currentEntry.getKey();
NormalizedFileSnapshot currentNormalizedSnapshot = currentEntry.getValue();
IncrementalFileSnapshot currentSnapshot = currentNormalizedSnapshot.getSnapshot();
if (unaccountedForPreviousSnapshots.remove(currentAbsolutePath)) {
NormalizedFileSnapshot previousNormalizedSnapshot = previous.get(currentAbsolutePath);
IncrementalFileSnapshot previousSnapshot = previousNormalizedSnapshot.getSnapshot();
if (!currentSnapshot.isContentUpToDate(previousSnapshot)) {
return new FileChange(currentAbsolutePath, ChangeType.MODIFIED, fileType);
}
// else, unchanged; check next file
} else {
added.add(currentAbsolutePath);
}
}
if (unaccountedForPreviousSnapshotsIterator == null) {
unaccountedForPreviousSnapshotsIterator = unaccountedForPreviousSnapshots.iterator();
}
if (unaccountedForPreviousSnapshotsIterator.hasNext()) {
String previousAbsolutePath = unaccountedForPreviousSnapshotsIterator.next();
return new FileChange(previousAbsolutePath, ChangeType.REMOVED, fileType);
}
if (includeAdded) {
if (addedIterator == null) {
addedIterator = added.iterator();
}
if (addedIterator.hasNext()) {
String newAbsolutePath = addedIterator.next();
return new FileChange(newAbsolutePath, ChangeType.ADDED, fileType);
}
}
return endOfData();
}
};
}
private Iterator iterateChangesForRelativePaths(final Map current, Map previous, final String fileType) {
final ListMultimap unaccountedForPreviousSnapshots = MultimapBuilder.hashKeys().linkedListValues().build();
for (Entry entry : previous.entrySet()) {
String absolutePath = entry.getKey();
NormalizedFileSnapshot previousSnapshot = entry.getValue();
unaccountedForPreviousSnapshots.put(previousSnapshot, new IncrementalFileSnapshotWithAbsolutePath(absolutePath, previousSnapshot.getSnapshot()));
}
final Iterator> currentEntries = current.entrySet().iterator();
return new AbstractIterator() {
private Iterator> unaccountedForPreviousSnapshotsIterator;
private final ListMultimap addedFiles = MultimapBuilder.hashKeys().linkedListValues().build();
private Iterator addedFilesIterator;
@Override
protected TaskStateChange computeNext() {
while (currentEntries.hasNext()) {
Entry entry = currentEntries.next();
String currentAbsolutePath = entry.getKey();
NormalizedFileSnapshot currentNormalizedSnapshot = entry.getValue();
IncrementalFileSnapshot currentSnapshot = currentNormalizedSnapshot.getSnapshot();
List previousSnapshotsForNormalizedPath = unaccountedForPreviousSnapshots.get(currentNormalizedSnapshot);
if (previousSnapshotsForNormalizedPath.isEmpty()) {
IncrementalFileSnapshotWithAbsolutePath currentSnapshotWithAbsolutePath = new IncrementalFileSnapshotWithAbsolutePath(currentAbsolutePath, currentSnapshot);
addedFiles.put(currentNormalizedSnapshot.getNormalizedPath(), currentSnapshotWithAbsolutePath);
} else {
IncrementalFileSnapshotWithAbsolutePath previousSnapshotWithAbsolutePath = previousSnapshotsForNormalizedPath.remove(0);
IncrementalFileSnapshot previousSnapshot = previousSnapshotWithAbsolutePath.getSnapshot();
if (!currentSnapshot.isContentUpToDate(previousSnapshot)) {
return new FileChange(currentAbsolutePath, ChangeType.MODIFIED, fileType);
}
}
}
// Create a single iterator to use for all of the still unaccounted files
if (unaccountedForPreviousSnapshotsIterator == null) {
if (unaccountedForPreviousSnapshots.isEmpty()) {
unaccountedForPreviousSnapshotsIterator = Iterators.emptyIterator();
} else {
List> entries = Lists.newArrayList(unaccountedForPreviousSnapshots.entries());
Collections.sort(entries, ENTRY_COMPARATOR);
unaccountedForPreviousSnapshotsIterator = entries.iterator();
}
}
if (unaccountedForPreviousSnapshotsIterator.hasNext()) {
Entry unaccountedForPreviousSnapshotEntry = unaccountedForPreviousSnapshotsIterator.next();
String normalizedPath = unaccountedForPreviousSnapshotEntry.getKey().getNormalizedPath();
List addedFilesForNormalizedPath = addedFiles.get(normalizedPath);
if (!addedFilesForNormalizedPath.isEmpty()) {
// There might be multiple files with the same normalized path, here we choose one of them
IncrementalFileSnapshotWithAbsolutePath modifiedSnapshot = addedFilesForNormalizedPath.remove(0);
return new FileChange(modifiedSnapshot.getAbsolutePath(), ChangeType.MODIFIED, fileType);
} else {
IncrementalFileSnapshotWithAbsolutePath removedSnapshot = unaccountedForPreviousSnapshotEntry.getValue();
return new FileChange(removedSnapshot.getAbsolutePath(), ChangeType.REMOVED, fileType);
}
}
if (includeAdded) {
// Create a single iterator to use for all of the added files
if (addedFilesIterator == null) {
addedFilesIterator = addedFiles.values().iterator();
}
if (addedFilesIterator.hasNext()) {
IncrementalFileSnapshotWithAbsolutePath addedFile = addedFilesIterator.next();
return new FileChange(addedFile.getAbsolutePath(), ChangeType.ADDED, fileType);
}
}
return endOfData();
}
};
}
@Override
public void appendToHasher(BuildCacheHasher hasher, Map snapshots) {
List normalizedSnapshots = Lists.newArrayList(snapshots.values());
Collections.sort(normalizedSnapshots);
for (NormalizedFileSnapshot normalizedSnapshot : normalizedSnapshots) {
normalizedSnapshot.appendToHasher(hasher);
}
}
@Override
public boolean isIncludeAdded() {
return includeAdded;
}
private static class IncrementalFileSnapshotWithAbsolutePath {
private final String absolutePath;
private final IncrementalFileSnapshot snapshot;
public IncrementalFileSnapshotWithAbsolutePath(String absolutePath, IncrementalFileSnapshot snapshot) {
this.absolutePath = absolutePath;
this.snapshot = snapshot;
}
public String getAbsolutePath() {
return absolutePath;
}
public IncrementalFileSnapshot getSnapshot() {
return snapshot;
}
@Override
public String toString() {
return String.format("%s (%s)", getSnapshot(), absolutePath);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy