com.phloc.commons.messagedigest.NonBlockingMessageDigestGenerator Maven / Gradle / Ivy
/**
* Copyright (C) 2006-2015 phloc systems
* http://www.phloc.com
* office[at]phloc[dot]com
*
* 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.phloc.commons.messagedigest;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.Arrays;
import javax.annotation.Nonnegative;
import javax.annotation.Nonnull;
import javax.annotation.concurrent.NotThreadSafe;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.phloc.commons.ValueEnforcer;
import com.phloc.commons.annotations.Nonempty;
import com.phloc.commons.annotations.ReturnsMutableObject;
import com.phloc.commons.collections.ArrayHelper;
import com.phloc.commons.string.ToStringGenerator;
/**
* Base class for creating a cryptographic hash value. Don't mix it up with the
* {@link com.phloc.commons.hash.HashCodeGenerator} which is used to generate
* hash values for Java objects.
*
* @author Philip Helger
*/
@NotThreadSafe
public final class NonBlockingMessageDigestGenerator extends AbstractMessageDigestGenerator
{
private static final Logger s_aLogger = LoggerFactory.getLogger (NonBlockingMessageDigestGenerator.class);
private final MessageDigest m_aMessageDigest;
private byte [] m_aDigest;
/**
* Create a default hash generator with the default algorithm.
*/
public NonBlockingMessageDigestGenerator ()
{
this (DEFAULT_ALGORITHM);
}
/**
* Create a hash generator with a set of possible algorithms to use.
*
* @param aAlgorithms
* The parameters to test. May not be null
.
* @throws NullPointerException
* If the array of algorithms is null
or if one element
* of the array is null
.
* @throws IllegalArgumentException
* If no algorithm was passed or if no applicable algorithm was used.
*/
public NonBlockingMessageDigestGenerator (@Nonnull @Nonempty final EMessageDigestAlgorithm... aAlgorithms)
{
ValueEnforcer.notEmpty (aAlgorithms, "Algortihms");
MessageDigest aMessageDigest = null;
for (final EMessageDigestAlgorithm eMD : aAlgorithms)
try
{
aMessageDigest = MessageDigest.getInstance (eMD.getAlgorithm ());
break;
}
catch (final NoSuchAlgorithmException ex)// NOPMD
{
// Unknown algorithm -> goto next
s_aLogger.warn ("Unsupported message digest algorithm '" + eMD.getAlgorithm () + "' found");
}
if (aMessageDigest == null)
{
// None of the passed algorithms was suitable
throw new IllegalArgumentException ("None of the algorithms in " +
Arrays.toString (aAlgorithms) +
" was applicable!");
}
m_aMessageDigest = aMessageDigest;
}
@Nonnull
public String getAlgorithmName ()
{
return m_aMessageDigest.getAlgorithm ();
}
@Nonnegative
public int getDigestLength ()
{
return m_aMessageDigest.getDigestLength ();
}
@Nonnull
public IMessageDigestGenerator update (final byte aValue)
{
if (m_aDigest != null)
throw new IllegalStateException ("The hash has already been finished. Call reset manually!");
m_aMessageDigest.update (aValue);
return this;
}
@Nonnull
public IMessageDigestGenerator update (@Nonnull final byte [] aValue,
@Nonnegative final int nOfs,
@Nonnegative final int nLen)
{
ValueEnforcer.isArrayOfsLen (aValue, nOfs, nLen);
if (m_aDigest != null)
throw new IllegalStateException ("The hash has already been finished. Call reset manually!");
m_aMessageDigest.update (aValue, nOfs, nLen);
return this;
}
public void reset ()
{
m_aMessageDigest.reset ();
m_aDigest = null;
}
@Nonnull
@ReturnsMutableObject (reason = "design")
private byte [] _getDigest ()
{
if (m_aDigest == null)
m_aDigest = m_aMessageDigest.digest ();
return m_aDigest;
}
@Nonnull
public byte [] getDigest ()
{
return ArrayHelper.getCopy (_getDigest ());
}
@Nonnull
public byte [] getDigest (@Nonnegative final int nLength)
{
return ArrayHelper.getCopy (_getDigest (), 0, nLength);
}
@Override
public String toString ()
{
return new ToStringGenerator (this).append ("messageDigest", m_aMessageDigest)
.appendIfNotNull ("digest", m_aDigest)
.toString ();
}
}