org.gradle.api.internal.changedetection.state.ZipHasher 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 2018 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.ImmutableSet;
import org.apache.commons.compress.utils.Lists;
import org.gradle.api.file.internal.FilePathUtil;
import org.gradle.internal.Factory;
import org.gradle.internal.FileUtils;
import org.gradle.internal.IoActions;
import org.gradle.internal.file.FileType;
import org.gradle.internal.fingerprint.FileSystemLocationFingerprint;
import org.gradle.internal.fingerprint.FingerprintHashingStrategy;
import org.gradle.internal.fingerprint.impl.DefaultFileSystemLocationFingerprint;
import org.gradle.internal.hash.HashCode;
import org.gradle.internal.hash.Hasher;
import org.gradle.internal.hash.Hashing;
import org.gradle.internal.snapshot.RegularFileSnapshot;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.annotation.Nullable;
import java.io.File;
import java.io.IOException;
import java.util.List;
import java.util.Set;
public class ZipHasher implements RegularFileHasher, ConfigurableNormalizer {
private static final Set KNOWN_ZIP_EXTENSIONS = ImmutableSet.of(".zip", ".jar", ".war", ".rar", ".ear", ".apk", ".aar");
private static final Logger LOGGER = LoggerFactory.getLogger(ZipHasher.class);
public static boolean isZipFile(final String name) {
for (String extension : KNOWN_ZIP_EXTENSIONS) {
if (FileUtils.hasExtensionIgnoresCase(name, extension)) {
return true;
}
}
return false;
}
private final ResourceHasher resourceHasher;
private final ResourceFilter resourceFilter;
public ZipHasher(ResourceHasher resourceHasher, ResourceFilter resourceFilter) {
this.resourceHasher = resourceHasher;
this.resourceFilter = resourceFilter;
}
@Nullable
@Override
public HashCode hash(RegularFileSnapshot fileSnapshot) {
return hashZipContents(fileSnapshot);
}
@Override
public void appendConfigurationToHasher(Hasher hasher) {
hasher.putString(getClass().getName());
resourceHasher.appendConfigurationToHasher(hasher);
resourceFilter.appendConfigurationToHasher(hasher);
}
@Nullable
private HashCode hashZipContents(RegularFileSnapshot zipFileSnapshot) {
try {
List fingerprints = fingerprintZipEntries(zipFileSnapshot.getAbsolutePath());
if (fingerprints.isEmpty()) {
return null;
}
Hasher hasher = Hashing.newHasher();
FingerprintHashingStrategy.SORT.appendToHasher(hasher, fingerprints);
return hasher.hash();
} catch (Exception e) {
return hashMalformedZip(zipFileSnapshot, e);
}
}
private List fingerprintZipEntries(String zipFile) throws IOException {
ZipInput input = null;
try {
input = FileZipInput.create(new File(zipFile));
List fingerprints = Lists.newArrayList();
fingerprintZipEntries("", fingerprints, input);
return fingerprints;
} finally {
IoActions.closeQuietly(input);
}
}
private void fingerprintZipEntries(String parentName, List fingerprints, ZipInput input) throws IOException {
fingerprints.add(newZipMarker(parentName));
for (ZipEntry zipEntry : input) {
ZipEntryRelativePath relativePath = new ZipEntryRelativePath(zipEntry);
if (zipEntry.isDirectory() || resourceFilter.shouldBeIgnored(relativePath)) {
continue;
}
String fullName = parentName.isEmpty() ? zipEntry.getName() : parentName + "/" + zipEntry.getName();
if (isZipFile(zipEntry.getName())) {
fingerprintZipEntries(fullName, fingerprints, new StreamZipInput(zipEntry.getInputStream()));
} else {
HashCode hash = resourceHasher.hash(zipEntry);
if (hash != null) {
fingerprints.add(new DefaultFileSystemLocationFingerprint(fullName, FileType.RegularFile, hash));
}
}
}
}
private DefaultFileSystemLocationFingerprint newZipMarker(String relativePath) {
return new DefaultFileSystemLocationFingerprint(relativePath, FileType.RegularFile, HashCode.fromInt(0));
}
private static class ZipEntryRelativePath implements Factory {
private final ZipEntry zipEntry;
private ZipEntryRelativePath(ZipEntry zipEntry) {
this.zipEntry = zipEntry;
}
@Override
public String[] create() {
return FilePathUtil.getPathSegments(zipEntry.getName());
}
}
private HashCode hashMalformedZip(RegularFileSnapshot zipFileSnapshot, Exception e) {
LOGGER.debug("Malformed archive '{}'. Falling back to full content hash instead of entry hashing.", zipFileSnapshot.getName(), e);
return zipFileSnapshot.getHash();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy