org.jreleaser.engine.checksum.Checksum Maven / Gradle / Ivy
The newest version!
/*
* SPDX-License-Identifier: Apache-2.0
*
* Copyright 2020-2024 The JReleaser 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
*
* https://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.jreleaser.engine.checksum;
import org.jreleaser.bundle.RB;
import org.jreleaser.model.JReleaserException;
import org.jreleaser.model.internal.JReleaserContext;
import org.jreleaser.model.internal.common.Artifact;
import org.jreleaser.model.internal.distributions.Distribution;
import org.jreleaser.model.internal.util.Artifacts;
import org.jreleaser.util.Algorithm;
import org.jreleaser.util.ChecksumUtils;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import static java.nio.charset.StandardCharsets.UTF_8;
import static org.jreleaser.model.api.checksum.Checksum.KEY_SKIP_CHECKSUM;
import static org.jreleaser.util.StringUtils.isNotBlank;
/**
* @author Andres Almiray
* @since 0.1.0
*/
public final class Checksum {
private Checksum() {
// noop
}
public static void collectAndWriteChecksums(JReleaserContext context) throws JReleaserException {
context.getLogger().info(RB.$("checksum.header"));
context.getLogger().increaseIndent();
context.getLogger().setPrefix("checksum");
try {
doCollectAndWriteChecksums(context);
} finally {
context.getLogger().restorePrefix();
context.getLogger().decreaseIndent();
}
}
private static void doCollectAndWriteChecksums(JReleaserContext context) throws JReleaserException {
Map> checksums = new LinkedHashMap<>();
if (context.getModel().getChecksum().isFiles()) {
for (Artifact artifact : Artifacts.resolveFiles(context)) {
if (!artifact.isActiveAndSelected() || artifact.extraPropertyIsTrue(KEY_SKIP_CHECKSUM) ||
artifact.isOptional(context) && !artifact.resolvedPathExists()) continue;
for (Algorithm algorithm : context.getModel().getChecksum().getAlgorithms()) {
readHash(context, algorithm, artifact);
List list = checksums.computeIfAbsent(algorithm, k -> new ArrayList<>());
list.add(artifact.getHash(algorithm) + " " + artifact.getEffectivePath(context).getFileName());
}
}
}
if (context.getModel().getChecksum().isArtifacts()) {
for (Distribution distribution : context.getModel().getActiveDistributions()) {
for (Artifact artifact : distribution.getArtifacts()) {
if (!artifact.isActiveAndSelected()) continue;
artifact.getEffectivePath(context, distribution);
if (artifact.isOptional(context) && !artifact.resolvedPathExists()) continue;
for (Algorithm algorithm : context.getModel().getChecksum().getAlgorithms()) {
readHash(context, distribution, algorithm, artifact);
List list = checksums.computeIfAbsent(algorithm, k -> new ArrayList<>());
list.add(artifact.getHash(algorithm) + " " + artifact.getEffectivePath(context, distribution).getFileName());
}
}
}
}
if (checksums.isEmpty()) {
context.getLogger().info(RB.$("checksum.not.enabled"));
context.getLogger().decreaseIndent();
context.getLogger().restorePrefix();
return;
}
for (Map.Entry> entry : checksums.entrySet()) {
Algorithm algorithm = entry.getKey();
List list = entry.getValue();
Path checksumsFilePath = context.getChecksumsDirectory()
.resolve(context.getModel().getChecksum().getResolvedName(context, algorithm));
String newContent = String.join(System.lineSeparator(), list) + System.lineSeparator();
try {
if (Files.exists(checksumsFilePath)) {
String oldContent = new String(Files.readAllBytes(checksumsFilePath), UTF_8);
if (newContent.equals(oldContent)) {
// no need to write down the same content
context.getLogger().info(RB.$("checksum.not.changed"));
context.getLogger().restorePrefix();
context.getLogger().decreaseIndent();
return;
}
}
} catch (IOException ignored) {
// OK
}
try {
if (isNotBlank(newContent)) {
Files.createDirectories(context.getChecksumsDirectory());
Files.write(checksumsFilePath, newContent.getBytes(UTF_8));
} else {
Files.deleteIfExists(checksumsFilePath);
}
} catch (IOException e) {
throw new JReleaserException(RB.$("ERROR_unexpected_error_checksum", checksumsFilePath.toAbsolutePath()), e);
}
}
}
public static void readHash(JReleaserContext context, Distribution distribution, Algorithm algorithm, Artifact artifact) throws JReleaserException {
Path artifactPath = artifact.getEffectivePath(context, distribution);
Path checksumPath = context.getChecksumsDirectory().resolve(distribution.getName())
.resolve(artifactPath.getFileName() + "." + algorithm.formatted());
readHash(context, algorithm, artifact, artifactPath, checksumPath);
}
public static void readHash(JReleaserContext context, Algorithm algorithm, Artifact artifact) throws JReleaserException {
Path artifactPath = artifact.getEffectivePath(context);
Path checksumPath = context.getChecksumsDirectory()
.resolve(artifactPath.getFileName() + "." + algorithm.formatted());
readHash(context, algorithm, artifact, artifactPath, checksumPath);
}
private static void readHash(JReleaserContext context,
Algorithm algorithm,
Artifact artifact,
Path artifactPath,
Path checksumPath) throws JReleaserException {
if (!Files.exists(artifactPath)) {
throw new JReleaserException(RB.$("ERROR_artifact_does_not_exist", context.relativizeToBasedir(artifactPath)));
}
if (!Files.exists(checksumPath)) {
context.getLogger().debug(RB.$("checksum.not.exist"), context.relativizeToBasedir(checksumPath));
calculateHash(context, artifactPath, checksumPath, algorithm);
} else if (artifactPath.toFile().lastModified() > checksumPath.toFile().lastModified()) {
context.getLogger().debug(RB.$("checksum.file.newer"),
context.relativizeToBasedir(artifactPath),
context.relativizeToBasedir(checksumPath));
calculateHash(context, artifactPath, checksumPath, algorithm);
}
try {
context.getLogger().debug(RB.$("checksum.reading"),
context.relativizeToBasedir(checksumPath));
artifact.setHash(algorithm, new String(Files.readAllBytes(checksumPath), UTF_8));
} catch (IOException e) {
throw new JReleaserException(RB.$("ERROR_unexpected_error_hash_read", context.relativizeToBasedir(checksumPath)), e);
}
}
public static String calculateHash(JReleaserContext context, Path input, Path output) throws JReleaserException {
return calculateHash(context, input, output, Algorithm.SHA_256);
}
public static String calculateHash(JReleaserContext context, Path input, Path output, Algorithm algorithm) throws JReleaserException {
try {
context.getLogger().info("{}.{}", context.relativizeToBasedir(input), algorithm.formatted());
String hashcode = ChecksumUtils.checksum(algorithm, Files.readAllBytes(input));
output.toFile().getParentFile().mkdirs();
Files.write(output, hashcode.getBytes(UTF_8));
return hashcode;
} catch (IOException e) {
throw new JReleaserException(RB.$("ERROR_unexpected_error_calculate_checksum", input), e);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy