/*
* JBoss, Home of Professional Open Source
* Copyright 2011 Red Hat Inc. and/or its affiliates and other contributors
* as indicated by the @authors tag. All rights reserved.
* See the copyright.txt in the distribution for a
* full listing of individual contributors.
*
* This copyrighted material is made available to anyone wishing to use,
* modify, copy, or redistribute it subject to the terms and conditions
* of the GNU Lesser General Public License, v. 2.1.
* This program is distributed in the hope that it will be useful, but WITHOUT A
* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
* PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details.
* You should have received a copy of the GNU Lesser General Public License,
* v.2.1 along with this distribution; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
* MA 02110-1301, USA.
*/
package org.jboss.as.repository;
import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.SequenceInputStream;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.security.DigestInputStream;
import java.security.MessageDigest;
import java.util.Arrays;
import java.util.Vector;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import org.jboss.as.repository.logging.DeploymentRepositoryLogger;
/**
* Utilities related to deployment content hashes.
*
* @author Brian Stansberry (c) 2011 Red Hat Inc.
*/
class HashUtil {
private HashUtil() {
}
private static char[] table = {
'0', '1', '2', '3', '4', '5', '6', '7',
'8', '9', 'a', 'b', 'c', 'd', 'e', 'f'
};
/**
* Convert a byte array into a hex string.
*
* @param bytes the bytes
* @return the string
*/
public static String bytesToHexString(final byte[] bytes) {
final StringBuilder builder = new StringBuilder(bytes.length * 2);
for (byte b : bytes) {
builder.append(table[b >> 4 & 0x0f]).append(table[b & 0x0f]);
}
return builder.toString();
}
/**
* Convert a hex string into a byte[].
*
* @param s the string
* @return the bytes
*/
public static byte[] hexStringToByteArray(String s) {
int len = s.length();
byte[] data = new byte[len >> 1];
for (int i = 0, j = 0; j < len; i++) {
int x = Character.digit(s.charAt(j), 16) << 4;
j++;
x = x | Character.digit(s.charAt(j), 16);
j++;
data[i] = (byte) (x & 0xFF);
}
return data;
}
public static boolean isEachHexHashInTable(String s) {
// WFLY-6018, check each char is in table, otherwise, there will be StringIndexOutOfBoundsException due to illegal char
char[] array = s.toCharArray();
for (char c : array) {
if (Arrays.binarySearch(table, c) < 0)
return false;
}
return true;
}
public static byte[] hashContent(MessageDigest messageDigest, InputStream stream) throws IOException {
messageDigest.reset();
try (DigestInputStream dis = new DigestInputStream(stream, messageDigest)) {
byte[] bytes = new byte[8192];
while (dis.read(bytes) > -1) {
}
}
return messageDigest.digest();
}
/**
* Hashes a path, if the path points to a directory then hashes the contents recursively.
* @param messageDigest the digest used to hash.
* @param path the file/directory we want to hash.
* @return the resulting hash.
* @throws IOException
*/
public static byte[] hashPath(MessageDigest messageDigest, Path path) throws IOException {
try (InputStream in = getRecursiveContentStream(path)) {
return hashContent(messageDigest, in);
}
}
private static InputStream getRecursiveContentStream(Path path) {
if (Files.isRegularFile(path)) {
try {
return new SequenceInputStream(new ByteArrayInputStream(path.getFileName().toString().getBytes(StandardCharsets.UTF_8)), Files.newInputStream(path));
} catch (IOException ex) {
throw DeploymentRepositoryLogger.ROOT_LOGGER.hashingError(ex, path);
}
} else if (Files.isDirectory(path)) {
try {
Vector<InputStream> v = new Vector<>();
v.add(new ByteArrayInputStream(path.getFileName().toString().getBytes(StandardCharsets.UTF_8)));
try(Stream<Path> paths = Files.list(path)) {
v.addAll(paths.sorted((Path path1, Path path2) -> path1.compareTo(path2)).map(p -> getRecursiveContentStream(p)).collect(Collectors.toList()));
}
return new SequenceInputStream(v.elements());
} catch (IOException ex) {
throw DeploymentRepositoryLogger.ROOT_LOGGER.hashingError(ex, path);
}
}
return emptyStream();
}
/**
* Create an empty non-null) stream.
* @return an empty non-null stream.
*/
public static InputStream emptyStream() {
return new InputStream() {
@Override
public int read() throws IOException {
return -1;
}
};
}
}