View Javadoc
1   /*
2    * Licensed to the Apache Software Foundation (ASF) under one or more
3    * contributor license agreements.  See the NOTICE file distributed with
4    * this work for additional information regarding copyright ownership.
5    * The ASF licenses this file to You under the Apache License, Version 2.0
6    * (the "License"); you may not use this file except in compliance with
7    * the License.  You may obtain a copy of the License at
8    *
9    *      https://www.apache.org/licenses/LICENSE-2.0
10   *
11   * Unless required by applicable law or agreed to in writing, software
12   * distributed under the License is distributed on an "AS IS" BASIS,
13   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14   * See the License for the specific language governing permissions and
15   * limitations under the License.
16   */
17  
18  package org.apache.commons.codec.digest;
19  
20  import java.io.BufferedInputStream;
21  import java.io.File;
22  import java.io.FileInputStream;
23  import java.io.IOException;
24  import java.io.InputStream;
25  import java.nio.ByteBuffer;
26  import java.security.InvalidKeyException;
27  import java.security.Key;
28  import java.security.NoSuchAlgorithmException;
29  
30  import javax.crypto.Mac;
31  import javax.crypto.spec.SecretKeySpec;
32  
33  import org.apache.commons.codec.binary.Hex;
34  import org.apache.commons.codec.binary.StringUtils;
35  
36  /**
37   * Simplifies common {@link javax.crypto.Mac} tasks. This class is immutable and thread-safe.
38   * However the Mac may not be.
39   * <p>
40   * <strong>Note: Not all JCE implementations support all algorithms. If not supported, an IllegalArgumentException is
41   * thrown.</strong>
42   * </p>
43   * <p>
44   * Sample usage:
45   * </p>
46   * <pre>
47   * import static HmacAlgorithms.*;
48   * byte[] key = {1,2,3,4}; // don't use this actual key!
49   * String valueToDigest = "The quick brown fox jumps over the lazy dog";
50   * byte[] hmac = new HmacUtils(HMAC_SHA_224, key).hmac(valueToDigest);
51   * // Mac re-use
52   * HmacUtils hm1 = new HmacUtils("HmacAlgoName", key); // use a valid name here!
53   * String hexPom = hm1.hmacHex(new File("pom.xml"));
54   * String hexNot = hm1.hmacHex(new File("NOTICE.txt"));
55   * </pre>
56   * @since 1.10
57   */
58  public final class HmacUtils {
59  
60      private static final int STREAM_BUFFER_LENGTH = 1024;
61  
62      /**
63       * Returns an initialized {@code Mac} for the HmacMD5 algorithm.
64       * <p>
65       * Every implementation of the Java platform is required to support this standard Mac algorithm.
66       * </p>
67       *
68       * @param key
69       *            The key for the keyed digest (must not be null)
70       * @return A Mac instance initialized with the given key.
71       * @see Mac#getInstance(String)
72       * @see Mac#init(Key)
73       * @throws IllegalArgumentException
74       *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
75       * @deprecated (1.11) Use {@code getInitializedMac(HmacAlgorithms.HMAC_MD5, byte[])}
76       */
77      @Deprecated
78      public static Mac getHmacMd5(final byte[] key) {
79          return getInitializedMac(HmacAlgorithms.HMAC_MD5, key);
80      }
81  
82      /**
83       * Returns an initialized {@code Mac} for the HmacSHA1 algorithm.
84       * <p>
85       * Every implementation of the Java platform is required to support this standard Mac algorithm.
86       * </p>
87       *
88       * @param key
89       *            The key for the keyed digest (must not be null)
90       * @return A Mac instance initialized with the given key.
91       * @see Mac#getInstance(String)
92       * @see Mac#init(Key)
93       * @throws IllegalArgumentException
94       *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
95       * @deprecated (1.11) Use {@code getInitializedMac(HmacAlgorithms.HMAC_SHA_1, byte[])}
96       */
97      @Deprecated
98      public static Mac getHmacSha1(final byte[] key) {
99          return getInitializedMac(HmacAlgorithms.HMAC_SHA_1, key);
100     }
101 
102     /**
103      * Returns an initialized {@code Mac} for the HmacSHA256 algorithm.
104      * <p>
105      * Every implementation of the Java platform is required to support this standard Mac algorithm.
106      * </p>
107      *
108      * @param key
109      *            The key for the keyed digest (must not be null)
110      * @return A Mac instance initialized with the given key.
111      * @see Mac#getInstance(String)
112      * @see Mac#init(Key)
113      * @throws IllegalArgumentException
114      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
115      * @deprecated (1.11) Use {@code getInitializedMac(HmacAlgorithms.HMAC_SHA_256, byte[])}
116      */
117     @Deprecated
118     public static Mac getHmacSha256(final byte[] key) {
119         return getInitializedMac(HmacAlgorithms.HMAC_SHA_256, key);
120     }
121 
122     /**
123      * Returns an initialized {@code Mac} for the HmacSHA384 algorithm.
124      * <p>
125      * Every implementation of the Java platform is <em>not</em> required to support this Mac algorithm.
126      * </p>
127      *
128      * @param key
129      *            The key for the keyed digest (must not be null)
130      * @return A Mac instance initialized with the given key.
131      * @see Mac#getInstance(String)
132      * @see Mac#init(Key)
133      * @throws IllegalArgumentException
134      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
135      * @deprecated (1.11) Use {@code getInitializedMac(HmacAlgorithms.HMAC_SHA_384, byte[])}
136      */
137     @Deprecated
138     public static Mac getHmacSha384(final byte[] key) {
139         return getInitializedMac(HmacAlgorithms.HMAC_SHA_384, key);
140     }
141 
142     /**
143      * Returns an initialized {@code Mac} for the HmacSHA512 algorithm.
144      * <p>
145      * Every implementation of the Java platform is <em>not</em> required to support this Mac algorithm.
146      * </p>
147      *
148      * @param key
149      *            The key for the keyed digest (must not be null)
150      * @return A Mac instance initialized with the given key.
151      * @see Mac#getInstance(String)
152      * @see Mac#init(Key)
153      * @throws IllegalArgumentException
154      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
155      * @deprecated (1.11) Use {@code getInitializedMac(HmacAlgorithms.HMAC_SHA_512, byte[])}
156      */
157     @Deprecated
158     public static Mac getHmacSha512(final byte[] key) {
159         return getInitializedMac(HmacAlgorithms.HMAC_SHA_512, key);
160     }
161 
162     /**
163      * Returns an initialized {@code Mac} for the given {@code algorithm}.
164      *
165      * @param algorithm
166      *            the name of the algorithm requested. See
167      *            <a href= "https://docs.oracle.com/javase/6/docs/technotes/guides/security/crypto/CryptoSpec.html#AppA"
168      *            >Appendix A in the Java Cryptography Architecture Reference Guide</a> for information about standard
169      *            algorithm names.
170      * @param key
171      *            The key for the keyed digest (must not be null)
172      * @return A Mac instance initialized with the given key.
173      * @see Mac#getInstance(String)
174      * @see Mac#init(Key)
175      * @throws IllegalArgumentException
176      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
177      */
178     public static Mac getInitializedMac(final HmacAlgorithms algorithm, final byte[] key) {
179         return getInitializedMac(algorithm.getName(), key);
180     }
181 
182     /**
183      * Returns an initialized {@code Mac} for the given {@code algorithm}.
184      *
185      * @param algorithm
186      *            the name of the algorithm requested. See
187      *            <a href= "https://docs.oracle.com/javase/6/docs/technotes/guides/security/crypto/CryptoSpec.html#AppA"
188      *            >Appendix A in the Java Cryptography Architecture Reference Guide</a> for information about standard
189      *            algorithm names.
190      * @param key
191      *            The key for the keyed digest (must not be null)
192      * @return A Mac instance initialized with the given key.
193      * @see Mac#getInstance(String)
194      * @see Mac#init(Key)
195      * @throws IllegalArgumentException
196      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
197      */
198     public static Mac getInitializedMac(final String algorithm, final byte[] key) {
199         if (key == null) {
200             throw new IllegalArgumentException("Null key");
201         }
202         try {
203             final SecretKeySpec keySpec = new SecretKeySpec(key, algorithm);
204             final Mac mac = Mac.getInstance(algorithm);
205             mac.init(keySpec);
206             return mac;
207         } catch (final NoSuchAlgorithmException | InvalidKeyException e) {
208             throw new IllegalArgumentException(e);
209         }
210     }
211 
212     /**
213      * Returns a HmacMD5 Message Authentication Code (MAC) for the given key and value.
214      *
215      * @param key
216      *            The key for the keyed digest (must not be null)
217      * @param valueToDigest
218      *            The value (data) which should to digest (maybe empty or null)
219      * @return HmacMD5 MAC for the given key and value
220      * @throws IllegalArgumentException
221      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
222      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_MD5, byte[]).hmac(byte[])}
223      */
224     @Deprecated
225     public static byte[] hmacMd5(final byte[] key, final byte[] valueToDigest) {
226         return new HmacUtils(HmacAlgorithms.HMAC_MD5, key).hmac(valueToDigest);
227     }
228 
229     /**
230      * Returns a HmacMD5 Message Authentication Code (MAC) for the given key and value.
231      *
232      * @param key
233      *            The key for the keyed digest (must not be null)
234      * @param valueToDigest
235      *            The value (data) which should to digest
236      *            <p>
237      *            The InputStream must not be null and will not be closed
238      *            </p>
239      * @return HmacMD5 MAC for the given key and value
240      * @throws IOException
241      *             If an I/O error occurs.
242      * @throws IllegalArgumentException
243      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
244      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_MD5, byte[]).hmac(InputStream)}
245      */
246     @Deprecated
247     public static byte[] hmacMd5(final byte[] key, final InputStream valueToDigest) throws IOException {
248         return new HmacUtils(HmacAlgorithms.HMAC_MD5, key).hmac(valueToDigest);
249     }
250 
251     /**
252      * Returns a HmacMD5 Message Authentication Code (MAC) for the given key and value.
253      *
254      * @param key
255      *            The key for the keyed digest (must not be null)
256      * @param valueToDigest
257      *            The value (data) which should to digest (maybe empty or null)
258      * @return HmacMD5 MAC for the given key and value
259      * @throws IllegalArgumentException
260      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
261      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_MD5, String).hmac(String)}
262      */
263     @Deprecated
264     public static byte[] hmacMd5(final String key, final String valueToDigest) {
265         return new HmacUtils(HmacAlgorithms.HMAC_MD5, key).hmac(valueToDigest);
266     }
267 
268     /**
269      * Returns a HmacMD5 Message Authentication Code (MAC) as a hexadecimal string (lowercase) for the given key and value.
270      *
271      * @param key
272      *            The key for the keyed digest (must not be null)
273      * @param valueToDigest
274      *            The value (data) which should to digest (maybe empty or null)
275      * @return HmacMD5 MAC for the given key and value as a hexadecimal string (lowercase)
276      * @throws IllegalArgumentException
277      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
278      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_MD5, byte[]).hmacHex(byte[])}
279      */
280     @Deprecated
281     public static String hmacMd5Hex(final byte[] key, final byte[] valueToDigest) {
282         return new HmacUtils(HmacAlgorithms.HMAC_MD5, key).hmacHex(valueToDigest);
283     }
284 
285     /**
286      * Returns a HmacMD5 Message Authentication Code (MAC) as a hexadecimal string (lowercase) for the given key and value.
287      *
288      * @param key
289      *            The key for the keyed digest (must not be null)
290      * @param valueToDigest
291      *            The value (data) which should to digest
292      *            <p>
293      *            The InputStream must not be null and will not be closed
294      *            </p>
295      * @return HmacMD5 MAC for the given key and value as a hexadecimal string (lowercase)
296      * @throws IOException
297      *             If an I/O error occurs.
298      * @throws IllegalArgumentException
299      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
300      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_MD5, byte[]).hmacHex(InputStream)}
301      */
302     @Deprecated
303     public static String hmacMd5Hex(final byte[] key, final InputStream valueToDigest) throws IOException {
304         return new HmacUtils(HmacAlgorithms.HMAC_MD5, key).hmacHex(valueToDigest);
305     }
306 
307     /**
308      * Returns a HmacMD5 Message Authentication Code (MAC) as a hexadecimal string (lowercase) for the given key and value.
309      *
310      * @param key
311      *            The key for the keyed digest (must not be null)
312      * @param valueToDigest
313      *            The value (data) which should to digest (maybe empty or null)
314      * @return HmacMD5 MAC for the given key and value as a hexadecimal string (lowercase)
315      * @throws IllegalArgumentException
316      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
317      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_MD5, String).hmacHex(String)}
318      */
319     @Deprecated
320     public static String hmacMd5Hex(final String key, final String valueToDigest) {
321         return new HmacUtils(HmacAlgorithms.HMAC_MD5, key).hmacHex(valueToDigest);
322     }
323 
324     /**
325      * Returns a HmacSHA1 Message Authentication Code (MAC) for the given key and value.
326      *
327      * @param key
328      *            The key for the keyed digest (must not be null)
329      * @param valueToDigest
330      *            The value (data) which should to digest (maybe empty or null)
331      * @return HmacSHA1 MAC for the given key and value
332      * @throws IllegalArgumentException
333      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
334      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_1, byte[]).hmac(byte[])}
335      */
336     @Deprecated
337     public static byte[] hmacSha1(final byte[] key, final byte[] valueToDigest) {
338         return new HmacUtils(HmacAlgorithms.HMAC_SHA_1, key).hmac(valueToDigest);
339     }
340 
341     /**
342      * Returns a HmacSHA1 Message Authentication Code (MAC) for the given key and value.
343      *
344      * @param key
345      *            The key for the keyed digest (must not be null)
346      * @param valueToDigest
347      *            The value (data) which should to digest
348      *            <p>
349      *            The InputStream must not be null and will not be closed
350      *            </p>
351      * @return HmacSHA1 MAC for the given key and value
352      * @throws IOException
353      *             If an I/O error occurs.
354      * @throws IllegalArgumentException
355      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
356      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_1, byte[]).hmac(InputStream)}
357      */
358     @Deprecated
359     public static byte[] hmacSha1(final byte[] key, final InputStream valueToDigest) throws IOException {
360         return new HmacUtils(HmacAlgorithms.HMAC_SHA_1, key).hmac(valueToDigest);
361     }
362 
363     /**
364      * Returns a HmacSHA1 Message Authentication Code (MAC) for the given key and value.
365      *
366      * @param key
367      *            The key for the keyed digest (must not be null)
368      * @param valueToDigest
369      *            The value (data) which should to digest (maybe empty or null)
370      * @return HmacSHA1 MAC for the given key and value
371      * @throws IllegalArgumentException
372      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
373      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_1, String).hmac(String)}
374      */
375     @Deprecated
376     public static byte[] hmacSha1(final String key, final String valueToDigest) {
377         return new HmacUtils(HmacAlgorithms.HMAC_SHA_1, key).hmac(valueToDigest);
378     }
379 
380     // hmacSha1
381 
382     /**
383      * Returns a HmacSHA1 Message Authentication Code (MAC) as hexadecimal string (lowercase) for the given key and value.
384      *
385      * @param key
386      *            The key for the keyed digest (must not be null)
387      * @param valueToDigest
388      *            The value (data) which should to digest (maybe empty or null)
389      * @return HmacSHA1 MAC for the given key and value as hexadecimal string (lowercase)
390      * @throws IllegalArgumentException
391      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
392      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_1, byte[]).hmacHex(byte[])}
393      */
394     @Deprecated
395     public static String hmacSha1Hex(final byte[] key, final byte[] valueToDigest) {
396         return new HmacUtils(HmacAlgorithms.HMAC_SHA_1, key).hmacHex(valueToDigest);
397     }
398 
399     /**
400      * Returns a HmacSHA1 Message Authentication Code (MAC) as hexadecimal string (lowercase) for the given key and value.
401      *
402      * @param key
403      *            The key for the keyed digest (must not be null)
404      * @param valueToDigest
405      *            The value (data) which should to digest
406      *            <p>
407      *            The InputStream must not be null and will not be closed
408      *            </p>
409      * @return HmacSHA1 MAC for the given key and value as hexadecimal string (lowercase)
410      * @throws IOException
411      *             If an I/O error occurs.
412      * @throws IllegalArgumentException
413      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
414      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_1, byte[]).hmacHex(InputStream)}
415      */
416     @Deprecated
417     public static String hmacSha1Hex(final byte[] key, final InputStream valueToDigest) throws IOException {
418         return new HmacUtils(HmacAlgorithms.HMAC_SHA_1, key).hmacHex(valueToDigest);
419     }
420 
421     /**
422      * Returns a HmacSHA1 Message Authentication Code (MAC) as hexadecimal string (lowercase) for the given key and value.
423      *
424      * @param key
425      *            The key for the keyed digest (must not be null)
426      * @param valueToDigest
427      *            The value (data) which should to digest (maybe empty or null)
428      * @return HmacSHA1 MAC for the given key and value as hexadecimal string (lowercase)
429      * @throws IllegalArgumentException
430      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
431      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_1, String).hmacHex(String)}
432      */
433     @Deprecated
434     public static String hmacSha1Hex(final String key, final String valueToDigest) {
435         return new HmacUtils(HmacAlgorithms.HMAC_SHA_1, key).hmacHex(valueToDigest);
436     }
437 
438     /**
439      * Returns a HmacSHA256 Message Authentication Code (MAC) for the given key and value.
440      *
441      * @param key
442      *            The key for the keyed digest (must not be null)
443      * @param valueToDigest
444      *            The value (data) which should to digest (maybe empty or null)
445      * @return HmacSHA256 MAC for the given key and value
446      * @throws IllegalArgumentException
447      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
448      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_256, byte[]).hmac(byte[])}
449      */
450     @Deprecated
451     public static byte[] hmacSha256(final byte[] key, final byte[] valueToDigest) {
452         return new HmacUtils(HmacAlgorithms.HMAC_SHA_256, key).hmac(valueToDigest);
453     }
454 
455     /**
456      * Returns a HmacSHA256 Message Authentication Code (MAC) for the given key and value.
457      *
458      * @param key
459      *            The key for the keyed digest (must not be null)
460      * @param valueToDigest
461      *            The value (data) which should to digest
462      *            <p>
463      *            The InputStream must not be null and will not be closed
464      *            </p>
465      * @return HmacSHA256 MAC for the given key and value
466      * @throws IOException
467      *             If an I/O error occurs.
468      * @throws IllegalArgumentException
469      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
470      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_256, byte[]).hmac(InputStream)}
471      */
472     @Deprecated
473     public static byte[] hmacSha256(final byte[] key, final InputStream valueToDigest) throws IOException {
474         return new HmacUtils(HmacAlgorithms.HMAC_SHA_256, key).hmac(valueToDigest);
475     }
476 
477     /**
478      * Returns a HmacSHA256 Message Authentication Code (MAC) for the given key and value.
479      *
480      * @param key
481      *            The key for the keyed digest (must not be null)
482      * @param valueToDigest
483      *            The value (data) which should to digest (maybe empty or null)
484      * @return HmacSHA256 MAC for the given key and value
485      * @throws IllegalArgumentException
486      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
487      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_256, String).hmac(String)}
488      */
489     @Deprecated
490     public static byte[] hmacSha256(final String key, final String valueToDigest) {
491         return new HmacUtils(HmacAlgorithms.HMAC_SHA_256, key).hmac(valueToDigest);
492     }
493 
494     /**
495      * Returns a HmacSHA256 Message Authentication Code (MAC) as hexadecimal string (lowercase) for the given key and value.
496      *
497      * @param key
498      *            The key for the keyed digest (must not be null)
499      * @param valueToDigest
500      *            The value (data) which should to digest (maybe empty or null)
501      * @return HmacSHA256 MAC for the given key and value as hexadecimal string (lowercase)
502      * @throws IllegalArgumentException
503      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
504      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_256, byte[]).hmacHex(byte[])}
505      */
506     @Deprecated
507     public static String hmacSha256Hex(final byte[] key, final byte[] valueToDigest) {
508         return new HmacUtils(HmacAlgorithms.HMAC_SHA_256, key).hmacHex(valueToDigest);
509     }
510 
511     /**
512      * Returns a HmacSHA256 Message Authentication Code (MAC) as hexadecimal string (lowercase) for the given key and value.
513      *
514      * @param key
515      *            The key for the keyed digest (must not be null)
516      * @param valueToDigest
517      *            The value (data) which should to digest
518      *            <p>
519      *            The InputStream must not be null and will not be closed
520      *            </p>
521      * @return HmacSHA256 MAC for the given key and value as hexadecimal string (lowercase)
522      * @throws IOException
523      *             If an I/O error occurs.
524      * @throws IllegalArgumentException
525      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
526      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_256, byte[]).hmacHex(InputStream)}
527      */
528     @Deprecated
529     public static String hmacSha256Hex(final byte[] key, final InputStream valueToDigest) throws IOException {
530         return new HmacUtils(HmacAlgorithms.HMAC_SHA_256, key).hmacHex(valueToDigest);
531     }
532 
533     /**
534      * Returns a HmacSHA256 Message Authentication Code (MAC) as hexadecimal string (lowercase) for the given key and value.
535      *
536      * @param key
537      *            The key for the keyed digest (must not be null)
538      * @param valueToDigest
539      *            The value (data) which should to digest (maybe empty or null)
540      * @return HmacSHA256 MAC for the given key and value as hexadecimal string (lowercase)
541      * @throws IllegalArgumentException
542      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
543      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_256, String).hmacHex(String)}
544      */
545     @Deprecated
546     public static String hmacSha256Hex(final String key, final String valueToDigest) {
547         return new HmacUtils(HmacAlgorithms.HMAC_SHA_256, key).hmacHex(valueToDigest);
548     }
549 
550     /**
551      * Returns a HmacSHA384 Message Authentication Code (MAC) for the given key and value.
552      *
553      * @param key
554      *            The key for the keyed digest (must not be null)
555      * @param valueToDigest
556      *            The value (data) which should to digest (maybe empty or null)
557      * @return HmacSHA384 MAC for the given key and value
558      * @throws IllegalArgumentException
559      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
560      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_384, byte[]).hmac(byte[])}
561      */
562     @Deprecated
563     public static byte[] hmacSha384(final byte[] key, final byte[] valueToDigest) {
564         return new HmacUtils(HmacAlgorithms.HMAC_SHA_384, key).hmac(valueToDigest);
565     }
566 
567     /**
568      * Returns a HmacSHA384 Message Authentication Code (MAC) for the given key and value.
569      *
570      * @param key
571      *            The key for the keyed digest (must not be null)
572      * @param valueToDigest
573      *            The value (data) which should to digest
574      *            <p>
575      *            The InputStream must not be null and will not be closed
576      *            </p>
577      * @return HmacSHA384 MAC for the given key and value
578      * @throws IOException
579      *             If an I/O error occurs.
580      * @throws IllegalArgumentException
581      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
582      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_384, byte[]).hmac(InputStream)}
583      */
584     @Deprecated
585     public static byte[] hmacSha384(final byte[] key, final InputStream valueToDigest) throws IOException {
586         return new HmacUtils(HmacAlgorithms.HMAC_SHA_384, key).hmac(valueToDigest);
587     }
588 
589     /**
590      * Returns a HmacSHA384 Message Authentication Code (MAC) for the given key and value.
591      *
592      * @param key
593      *            The key for the keyed digest (must not be null)
594      * @param valueToDigest
595      *            The value (data) which should to digest (maybe empty or null)
596      * @return HmacSHA384 MAC for the given key and value
597      * @throws IllegalArgumentException
598      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
599      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_384, String).hmac(String)}
600      */
601     @Deprecated
602     public static byte[] hmacSha384(final String key, final String valueToDigest) {
603         return new HmacUtils(HmacAlgorithms.HMAC_SHA_384, key).hmac(valueToDigest);
604     }
605 
606     // hmacSha384
607 
608     /**
609      * Returns a HmacSHA384 Message Authentication Code (MAC) as hexadecimal string (lowercase) for the given key and value.
610      *
611      * @param key
612      *            The key for the keyed digest (must not be null)
613      * @param valueToDigest
614      *            The value (data) which should to digest (maybe empty or null)
615      * @return HmacSHA384 MAC for the given key and value as hexadecimal string (lowercase)
616      * @throws IllegalArgumentException
617      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
618      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_384, byte[]).hmacHex(byte[])}
619      */
620     @Deprecated
621     public static String hmacSha384Hex(final byte[] key, final byte[] valueToDigest) {
622         return new HmacUtils(HmacAlgorithms.HMAC_SHA_384, key).hmacHex(valueToDigest);
623     }
624 
625     /**
626      * Returns a HmacSHA384 Message Authentication Code (MAC) as hexadecimal string (lowercase) for the given key and value.
627      *
628      * @param key
629      *            The key for the keyed digest (must not be null)
630      * @param valueToDigest
631      *            The value (data) which should to digest
632      *            <p>
633      *            The InputStream must not be null and will not be closed
634      *            </p>
635      * @return HmacSHA384 MAC for the given key and value as hexadecimal string (lowercase)
636      * @throws IOException
637      *             If an I/O error occurs.
638      * @throws IllegalArgumentException
639      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
640      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_384, byte[]).hmacHex(InputStream)}
641      */
642     @Deprecated
643     public static String hmacSha384Hex(final byte[] key, final InputStream valueToDigest) throws IOException {
644         return new HmacUtils(HmacAlgorithms.HMAC_SHA_384, key).hmacHex(valueToDigest);
645     }
646 
647     /**
648      * Returns a HmacSHA384 Message Authentication Code (MAC) as hexadecimal string (lowercase) for the given key and value.
649      *
650      * @param key
651      *            The key for the keyed digest (must not be null)
652      * @param valueToDigest
653      *            The value (data) which should to digest (maybe empty or null)
654      * @return HmacSHA384 MAC for the given key and value as hexadecimal string (lowercase)
655      * @throws IllegalArgumentException
656      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
657      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_384, String).hmacHex(String)}
658      */
659     @Deprecated
660     public static String hmacSha384Hex(final String key, final String valueToDigest) {
661         return new HmacUtils(HmacAlgorithms.HMAC_SHA_384, key).hmacHex(valueToDigest);
662     }
663 
664     /**
665      * Returns a HmacSHA512 Message Authentication Code (MAC) for the given key and value.
666      *
667      * @param key
668      *            The key for the keyed digest (must not be null)
669      * @param valueToDigest
670      *            The value (data) which should to digest (maybe empty or null)
671      * @return HmacSHA512 MAC for the given key and value
672      * @throws IllegalArgumentException
673      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
674      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_512, byte[]).hmac(byte[])}
675      */
676     @Deprecated
677     public static byte[] hmacSha512(final byte[] key, final byte[] valueToDigest) {
678         return new HmacUtils(HmacAlgorithms.HMAC_SHA_512, key).hmac(valueToDigest);
679     }
680 
681     /**
682      * Returns a HmacSHA512 Message Authentication Code (MAC) for the given key and value.
683      *
684      * @param key
685      *            The key for the keyed digest (must not be null)
686      * @param valueToDigest
687      *            The value (data) which should to digest
688      *            <p>
689      *            The InputStream must not be null and will not be closed
690      *            </p>
691      * @return HmacSHA512 MAC for the given key and value
692      * @throws IOException
693      *             If an I/O error occurs.
694      * @throws IllegalArgumentException
695      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
696      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_512, byte[]).hmac(InputStream)}
697      */
698     @Deprecated
699     public static byte[] hmacSha512(final byte[] key, final InputStream valueToDigest) throws IOException {
700         return new HmacUtils(HmacAlgorithms.HMAC_SHA_512, key).hmac(valueToDigest);
701     }
702 
703     /**
704      * Returns a HmacSHA512 Message Authentication Code (MAC) for the given key and value.
705      *
706      * @param key
707      *            The key for the keyed digest (must not be null)
708      * @param valueToDigest
709      *            The value (data) which should to digest (maybe empty or null)
710      * @return HmacSHA512 MAC for the given key and value
711      * @throws IllegalArgumentException
712      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
713      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_512, String).hmac(String)}
714      */
715     @Deprecated
716     public static byte[] hmacSha512(final String key, final String valueToDigest) {
717         return new HmacUtils(HmacAlgorithms.HMAC_SHA_512, key).hmac(valueToDigest);
718     }
719 
720     // hmacSha512
721 
722     /**
723      * Returns a HmacSHA512 Message Authentication Code (MAC) as hexadecimal string (lowercase) for the given key and value.
724      *
725      * @param key
726      *            The key for the keyed digest (must not be null)
727      * @param valueToDigest
728      *            The value (data) which should to digest (maybe empty or null)
729      * @return HmacSHA512 MAC for the given key and value as hexadecimal string (lowercase)
730      * @throws IllegalArgumentException
731      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
732      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_512, byte[]).hmacHex(byte[])}
733      */
734     @Deprecated
735     public static String hmacSha512Hex(final byte[] key, final byte[] valueToDigest) {
736         return new HmacUtils(HmacAlgorithms.HMAC_SHA_512, key).hmacHex(valueToDigest);
737     }
738 
739     /**
740      * Returns a HmacSHA512 Message Authentication Code (MAC) as hexadecimal string (lowercase) for the given key and value.
741      *
742      * @param key
743      *            The key for the keyed digest (must not be null)
744      * @param valueToDigest
745      *            The value (data) which should to digest
746      *            <p>
747      *            The InputStream must not be null and will not be closed
748      *            </p>
749      * @return HmacSHA512 MAC for the given key and value as hexadecimal string (lowercase)
750      * @throws IOException
751      *             If an I/O error occurs.
752      * @throws IllegalArgumentException
753      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
754      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_512, byte[]).hmacHex(InputStream)}
755      */
756     @Deprecated
757     public static String hmacSha512Hex(final byte[] key, final InputStream valueToDigest) throws IOException {
758         return new HmacUtils(HmacAlgorithms.HMAC_SHA_512, key).hmacHex(valueToDigest);
759     }
760 
761     /**
762      * Returns a HmacSHA512 Message Authentication Code (MAC) as hexadecimal string (lowercase) for the given key and value.
763      *
764      * @param key
765      *            The key for the keyed digest (must not be null)
766      * @param valueToDigest
767      *            The value (data) which should to digest (maybe empty or null)
768      * @return HmacSHA512 MAC for the given key and value as hexadecimal string (lowercase)
769      * @throws IllegalArgumentException
770      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
771      * @deprecated (1.11) Use {@code new HmacUtils(HmacAlgorithms.HMAC_SHA_512, String).hmacHex(String)}
772      */
773     @Deprecated
774     public static String hmacSha512Hex(final String key, final String valueToDigest) {
775         return new HmacUtils(HmacAlgorithms.HMAC_SHA_512, key).hmacHex(valueToDigest);
776     }
777 
778     /**
779      * Returns whether this algorithm is available
780      *
781      * @param name the name to check
782      * @return whether this algorithm is available
783      * @since 1.11
784      */
785     public static boolean isAvailable(final HmacAlgorithms name) {
786         try {
787             Mac.getInstance(name.getName());
788             return true;
789         } catch (final NoSuchAlgorithmException e) {
790             return false;
791         }
792     }
793 
794     /**
795      * Returns whether this algorithm is available
796      *
797      * @param name the name to check
798      * @return whether this algorithm is available
799      * @since 1.11
800      */
801     public static boolean isAvailable(final String name) {
802         try {
803             Mac.getInstance(name);
804             return true;
805         } catch (final NoSuchAlgorithmException e) {
806             return false;
807         }
808     }
809 
810     /**
811      * Resets and then updates the given {@link Mac} with the value.
812      *
813      * @param mac
814      *            the initialized {@link Mac} to update
815      * @param valueToDigest
816      *            the value to update the {@link Mac} with (maybe null or empty)
817      * @return the updated {@link Mac}
818      * @throws IllegalStateException
819      *             if the Mac was not initialized
820      */
821     public static Mac updateHmac(final Mac mac, final byte[] valueToDigest) {
822         mac.reset();
823         mac.update(valueToDigest);
824         return mac;
825     }
826 
827     /**
828      * Resets and then updates the given {@link Mac} with the value.
829      *
830      * @param mac
831      *            the initialized {@link Mac} to update
832      * @param valueToDigest
833      *            the value to update the {@link Mac} with
834      *            <p>
835      *            The InputStream must not be null and will not be closed
836      *            </p>
837      * @return the updated {@link Mac}
838      * @throws IOException
839      *             If an I/O error occurs.
840      * @throws IllegalStateException
841      *             If the Mac was not initialized
842      */
843     public static Mac updateHmac(final Mac mac, final InputStream valueToDigest) throws IOException {
844         mac.reset();
845         final byte[] buffer = new byte[STREAM_BUFFER_LENGTH];
846         int read = valueToDigest.read(buffer, 0, STREAM_BUFFER_LENGTH);
847 
848         while (read > -1) {
849             mac.update(buffer, 0, read);
850             read = valueToDigest.read(buffer, 0, STREAM_BUFFER_LENGTH);
851         }
852 
853         return mac;
854     }
855 
856     /**
857      * Resets and then updates the given {@link Mac} with the value.
858      *
859      * @param mac
860      *            the initialized {@link Mac} to update
861      * @param valueToDigest
862      *            the value to update the {@link Mac} with (maybe null or empty)
863      * @return the updated {@link Mac}
864      * @throws IllegalStateException
865      *             if the Mac was not initialized
866      */
867     public static Mac updateHmac(final Mac mac, final String valueToDigest) {
868         mac.reset();
869         mac.update(StringUtils.getBytesUtf8(valueToDigest));
870         return mac;
871     }
872 
873     private final Mac mac;
874 
875     /**
876      * Preserves binary compatibility only.
877      * As for previous versions does not provide useful behavior
878      * @deprecated since 1.11; only useful to preserve binary compatibility
879      */
880     @Deprecated
881     public HmacUtils() {
882         this(null);
883     }
884 
885     /**
886      * Creates an instance using the provided algorithm type.
887      *
888      * @param algorithm to use.
889      * @param key the key to use
890      * @throws IllegalArgumentException
891      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
892      * @since 1.11
893      */
894     public HmacUtils(final HmacAlgorithms algorithm, final byte[] key) {
895         this(algorithm.getName(), key);
896     }
897 
898     /**
899      * Creates an instance using the provided algorithm type.
900      *
901      * @param algorithm to use
902      * @param  key the key to use
903      * @throws IllegalArgumentException
904      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
905      * @since 1.11
906      */
907     public HmacUtils(final HmacAlgorithms algorithm, final String key) {
908         this(algorithm.getName(), StringUtils.getBytesUtf8(key));
909     }
910 
911     private HmacUtils(final Mac mac) {
912         this.mac = mac;
913     }
914 
915     /**
916      * Creates an instance using the provided algorithm type.
917      *
918      * @param algorithm to use
919      * @param  key the key to use
920      * @throws IllegalArgumentException
921      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
922      * @since 1.11
923      */
924     public HmacUtils(final String algorithm, final byte[] key) {
925         this(getInitializedMac(algorithm, key));
926     }
927 
928     /**
929      * Creates an instance using the provided algorithm type.
930      *
931      * @param algorithm to use
932      * @param  key the key to use
933      * @throws IllegalArgumentException
934      *             when a {@link NoSuchAlgorithmException} is caught or key is null or key is invalid.
935      * @since 1.11
936      */
937     public HmacUtils(final String algorithm, final String key) {
938         this(algorithm, StringUtils.getBytesUtf8(key));
939     }
940 
941     /**
942      * Returns the digest for the input data.
943      *
944      * @param valueToDigest the input to use
945      * @return the digest as a byte[]
946      * @since 1.11
947      */
948     public byte[] hmac(final byte[] valueToDigest) {
949         return mac.doFinal(valueToDigest);
950     }
951 
952     /**
953      * Returns the digest for the input data.
954      *
955      * @param valueToDigest the input to use
956      * @return the digest as a byte[]
957      * @since 1.11
958      */
959     public byte[] hmac(final ByteBuffer valueToDigest) {
960         mac.update(valueToDigest);
961         return mac.doFinal();
962     }
963 
964     /**
965      * Returns the digest for the file.
966      *
967      * @param valueToDigest the file to use
968      * @return the digest
969      * @throws IOException
970      *             If an I/O error occurs.
971      * @since 1.11
972      */
973     public byte[] hmac(final File valueToDigest) throws IOException {
974         try (BufferedInputStream stream = new BufferedInputStream(new FileInputStream(valueToDigest))) {
975             return hmac(stream);
976         }
977     }
978 
979     /**
980      * Returns the digest for the stream.
981      *
982      * @param valueToDigest
983      *            the data to use
984      *            <p>
985      *            The InputStream must not be null and will not be closed
986      *            </p>
987      * @return the digest
988      * @throws IOException
989      *             If an I/O error occurs.
990      * @since 1.11
991      */
992     public byte[] hmac(final InputStream valueToDigest) throws IOException {
993         final byte[] buffer = new byte[STREAM_BUFFER_LENGTH];
994         int read;
995 
996         while ((read = valueToDigest.read(buffer, 0, STREAM_BUFFER_LENGTH)) > -1) {
997             mac.update(buffer, 0, read);
998         }
999         return mac.doFinal();
1000     }
1001 
1002     /**
1003      * Returns the digest for the input data.
1004      *
1005      * @param valueToDigest the input to use, treated as UTF-8
1006      * @return the digest as a byte[]
1007      * @since 1.11
1008      */
1009     public byte[] hmac(final String valueToDigest) {
1010         return mac.doFinal(StringUtils.getBytesUtf8(valueToDigest));
1011     }
1012 
1013     /**
1014      * Returns the digest for the input data.
1015      *
1016      * @param valueToDigest the input to use
1017      * @return the digest as a hexadecimal String
1018      * @since 1.11
1019      */
1020     public String hmacHex(final byte[] valueToDigest) {
1021         return Hex.encodeHexString(hmac(valueToDigest));
1022     }
1023 
1024     /**
1025      * Returns the digest for the input data.
1026      *
1027      * @param valueToDigest the input to use
1028      * @return the digest as a hexadecimal String
1029      * @since 1.11
1030      */
1031     public String hmacHex(final ByteBuffer valueToDigest) {
1032         return Hex.encodeHexString(hmac(valueToDigest));
1033     }
1034 
1035     /**
1036      * Returns the digest for the file.
1037      *
1038      * @param valueToDigest the file to use
1039      * @return the digest as a hexadecimal String
1040      * @throws IOException
1041      *             If an I/O error occurs.
1042      * @since 1.11
1043      */
1044     public String hmacHex(final File valueToDigest) throws IOException {
1045         return Hex.encodeHexString(hmac(valueToDigest));
1046     }
1047 
1048     /**
1049      * Returns the digest for the stream.
1050      *
1051      * @param valueToDigest
1052      *            the data to use
1053      *            <p>
1054      *            The InputStream must not be null and will not be closed
1055      *            </p>
1056      * @return the digest as a hexadecimal String
1057      * @throws IOException
1058      *             If an I/O error occurs.
1059      * @since 1.11
1060      */
1061     public String hmacHex(final InputStream valueToDigest) throws IOException {
1062         return Hex.encodeHexString(hmac(valueToDigest));
1063     }
1064 
1065     /**
1066      * Returns the digest for the input data.
1067      *
1068      * @param valueToDigest the input to use, treated as UTF-8
1069      * @return the digest as a hexadecimal String
1070      * @since 1.11
1071      */
1072     public String hmacHex(final String valueToDigest) {
1073         return Hex.encodeHexString(hmac(valueToDigest));
1074     }
1075 
1076 }