blob: 12aee0757bb2e66df65c7139d9ae559f15623bf6 [file] [log] [blame]
// Copyright 2018 The Bazel Authors. All rights reserved.
//
// 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 com.google.devtools.build.lib.vfs;
import com.google.common.collect.ImmutableList;
import com.google.common.hash.HashFunction;
import com.google.common.hash.Hashing;
import com.google.devtools.common.options.Converter;
import com.google.devtools.common.options.OptionsParsingException;
import java.util.HashMap;
import java.util.List;
import java.util.Map.Entry;
/** Type of hash function to use for digesting files. */
// The underlying HashFunctions are immutable and thread safe.
public class DigestHashFunction {
// This map must be declared first to make sure that calls to register() have it ready.
private static final HashMap<String, DigestHashFunction> hashFunctionRegistry = new HashMap<>();
public static final DigestHashFunction MD5 = register(Hashing.md5(), "MD5");
public static final DigestHashFunction SHA1 = register(Hashing.sha1(), "SHA-1", "SHA1");
public static final DigestHashFunction SHA256 = register(Hashing.sha256(), "SHA-256", "SHA256");
private static DigestHashFunction defaultHash;
private static boolean defaultHasBeenSet = false;
private final HashFunction hash;
private final String name;
private DigestHashFunction(HashFunction hash, String name) {
this.hash = hash;
this.name = name;
}
public HashFunction getHash() {
return hash;
}
public boolean isValidDigest(byte[] digest) {
// TODO(b/109764197): Remove this check to accept variable-length hashes.
return digest != null && digest.length * 8 == hash.bits();
}
@Override
public String toString() {
return name;
}
/**
* Creates a new DigestHashFunction that is registered to be recognized by its name in {@link
* DigestFunctionConverter}.
*
* @param hashName the canonical name for this hash function.
* @param altNames alternative names that will be mapped to this function by the converter but
* will not serve as the canonical name for the DigestHashFunction.
* @param hash The {@link HashFunction} to register.
* @throws IllegalArgumentException if the name is already registered.
*/
public static DigestHashFunction register(
HashFunction hash, String hashName, String... altNames) {
DigestHashFunction hashFunction = new DigestHashFunction(hash, hashName);
List<String> names = ImmutableList.<String>builder().add(hashName).add(altNames).build();
synchronized (hashFunctionRegistry) {
for (String name : names) {
if (hashFunctionRegistry.containsKey(name)) {
throw new IllegalArgumentException("Hash function " + name + " is already registered.");
}
hashFunctionRegistry.put(name, hashFunction);
}
}
return hashFunction;
}
/**
* Returns the default DigestHashFunction for this instance of Bazel.
*
* <p>Note: This is a synchronized function, to make sure it does not occur concurrently with
* {@link #setDefault(DigestHashFunction)}. Once this value is set, it's a constant, so to prevent
* blocking calls, users should cache this value if needed.
*
* @throws DefaultNotSetException if the default has not yet been set by a previous call to {@link
* #setDefault}.
*/
public static synchronized DigestHashFunction getDefault() throws DefaultNotSetException {
if (!defaultHasBeenSet) {
throw new DefaultNotSetException("DigestHashFunction default has not been set");
}
return defaultHash;
}
/** Indicates that the default has not been initialized. */
public static final class DefaultNotSetException extends Exception {
DefaultNotSetException(String message) {
super(message);
}
}
/**
* Sets the default DigestHashFunction for this instance of Bazel - can only be set once to
* prevent incongruities.
*
* @throws DefaultAlreadySetException if it was already set.
*/
public static synchronized void setDefault(DigestHashFunction hash)
throws DefaultAlreadySetException {
if (defaultHasBeenSet) {
throw new DefaultAlreadySetException(
String.format(
"setDefault(%s) failed. The default has already been set to %s, you cannot reset it.",
hash.name, defaultHash.name));
}
defaultHash = hash;
defaultHasBeenSet = true;
}
/** Failure to set the default if the default already being set. */
public static final class DefaultAlreadySetException extends Exception {
DefaultAlreadySetException(String message) {
super(message);
}
}
/** Converts a string to its registered {@link DigestHashFunction}. */
public static class DigestFunctionConverter implements Converter<DigestHashFunction> {
@Override
public DigestHashFunction convert(String input) throws OptionsParsingException {
for (Entry<String, DigestHashFunction> possibleFunctions : hashFunctionRegistry.entrySet()) {
if (possibleFunctions.getKey().equalsIgnoreCase(input)) {
return possibleFunctions.getValue();
}
}
throw new OptionsParsingException("Not a valid hash function.");
}
@Override
public String getTypeDescription() {
return "hash function";
}
}
}