From ff89d0a7a34b2a4706f2bee7fe8c9ab380983b44 Mon Sep 17 00:00:00 2001 From: Pete Bentley <44170157+prbprbprb@users.noreply.github.com> Date: Thu, 7 Dec 2023 11:10:00 +0000 Subject: [PATCH] Add a public HKDF API, per RFC 5869. (#1163) * Add a public HKDF API, per RFC 5869. Implemented purely from the RFC, no other code reference used. Unit tests test against all available providers to ensure consistency. Creates new Mac instances for each expand/extract operation so probably thread-safe. Probably does too much array copying but we can fix that later if it's an issue. --- common/src/main/java/org/conscrypt/Hkdf.java | 124 ++++++++++++++++++ .../src/test/java/org/conscrypt/HkdfTest.java | 96 ++++++++++++++ common/src/test/resources/crypto/hkdf.txt | 63 +++++++++ .../main/java/org/conscrypt/TestUtils.java | 50 ++++++- .../main/java/org/conscrypt/TestVector.java | 54 ++++++++ 5 files changed, 380 insertions(+), 7 deletions(-) create mode 100644 common/src/main/java/org/conscrypt/Hkdf.java create mode 100644 common/src/test/java/org/conscrypt/HkdfTest.java create mode 100644 common/src/test/resources/crypto/hkdf.txt create mode 100644 testing/src/main/java/org/conscrypt/TestVector.java diff --git a/common/src/main/java/org/conscrypt/Hkdf.java b/common/src/main/java/org/conscrypt/Hkdf.java new file mode 100644 index 000000000..e18b36c0b --- /dev/null +++ b/common/src/main/java/org/conscrypt/Hkdf.java @@ -0,0 +1,124 @@ +/* + * Copyright (C) 2023 The Android Open Source Project + * + * 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 org.conscrypt; + +import java.security.InvalidKeyException; +import java.security.NoSuchAlgorithmException; +import java.util.Objects; + +import javax.crypto.Mac; +import javax.crypto.spec.SecretKeySpec; + +/** + * Hkdf - perform HKDF key derivation operations per RFC 5869. + *
+ * Instances should be instantiated using the standard JCA name for the required HMAC. + *
+ * Each invocation of expand or extract uses a new Mac instance and so instances + * of Hkdf are thread-safe. + */ +public final class Hkdf { + // HMAC algorithm to use. + private final String hmacName; + private final int macLength; + + /** + * Creates an Hkdf instance which will use hmacName as the name for the underlying + * HMAC algorithm, which will be located using normal JCA precedence rules. + *
+ * @param hmacName the name of the HMAC algorithm to use
+ * @throws NoSuchAlgorithmException if hmacName is not a valid HMAC name
+ */
+ public Hkdf(String hmacName) throws NoSuchAlgorithmException {
+ Objects.requireNonNull(hmacName);
+ this.hmacName = hmacName;
+
+ // Stash the MAC length with the bonus that we'll fail fast here if no such algorithm.
+ macLength = Mac.getInstance(hmacName).getMacLength();
+ }
+
+ // Visible for testing.
+ int getMacLength() {
+ return macLength;
+ }
+
+ /**
+ * Performs an HKDF extract operation as specified in RFC 5869.
+ *
+ * @param salt the salt to use
+ * @param ikm initial keying material
+ * @return a pseudorandom key suitable for use in expand operations
+ * @throws InvalidKeyException if the salt is not suitable for use as an HMAC key
+ * @throws NoSuchAlgorithmException if the Mac algorithm is no longer available
+ */
+
+ public byte[] extract(byte[] salt, byte[] ikm)
+ throws InvalidKeyException, NoSuchAlgorithmException {
+ Objects.requireNonNull(salt);
+ Objects.requireNonNull(ikm);
+ Preconditions.checkArgument(ikm.length > 0, "Empty keying material");
+ if (salt.length == 0) {
+ salt = new byte[getMacLength()];
+ }
+ return getMac(salt).doFinal(ikm);
+ }
+
+ /**
+ * Performs an HKDF expand operation as specified in RFC 5869.
+ *
+ * @param prk a pseudorandom key of at least HashLen octets, usually the output from the
+ * extract step. Where HashLen is the key size of the underlying Mac
+ * @param info optional context and application specific information, can be zero length
+ * @param length length of output keying material in bytes (<= 255*HashLen)
+ * @return output of keying material of length bytes
+ * @throws InvalidKeyException if prk is not suitable for use as an HMAC key
+ * @throws IllegalArgumentException if length is out of the allowed range
+ * @throws NoSuchAlgorithmException if the Mac algorithm is no longer available
+ */
+ public byte[] expand(byte[] prk, byte[] info, int length)
+ throws InvalidKeyException, NoSuchAlgorithmException {
+ Objects.requireNonNull(prk);
+ Objects.requireNonNull(info);
+ Preconditions.checkArgument(length >= 0, "Negative length");
+ Preconditions.checkArgument(length < 255 * getMacLength(), "Length too long");
+ Mac mac = getMac(prk);
+ int macLength = getMacLength();
+
+ byte[] t = new byte[0];
+ byte[] output = new byte[length];
+ int outputOffset = 0;
+ byte[] counter = new byte[] { 0x00 };
+ while (outputOffset < length) {
+ counter[0]++;
+ mac.update(t);
+ mac.update(info);
+ t = mac.doFinal(counter);
+ int size = Math.min(macLength, length - outputOffset);
+ System.arraycopy(t, 0, output, outputOffset, size);
+ outputOffset += size;
+ }
+ return output;
+ }
+
+ private Mac getMac(byte[] key) throws InvalidKeyException, NoSuchAlgorithmException {
+ // Can potentially throw NoSuchAlgorithmException if the there has been a change
+ // in installed Providers.
+ Mac mac = Mac.getInstance(hmacName);
+ mac.init(new SecretKeySpec(key, "RAW"));
+ return mac; // https://www.youtube.com/watch?v=uB1D9wWxd2w
+ }
+}
diff --git a/common/src/test/java/org/conscrypt/HkdfTest.java b/common/src/test/java/org/conscrypt/HkdfTest.java
new file mode 100644
index 000000000..3acbdce2e
--- /dev/null
+++ b/common/src/test/java/org/conscrypt/HkdfTest.java
@@ -0,0 +1,96 @@
+/*
+ * Copyright (C) 2023 The Android Open Source Project
+ *
+ * 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 org.conscrypt;
+
+import static org.junit.Assert.assertArrayEquals;
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertThrows;
+
+import org.junit.Test;
+import org.junit.runner.RunWith;
+import org.junit.runners.JUnit4;
+
+import java.security.NoSuchAlgorithmException;
+import java.util.List;
+
+import javax.crypto.Mac;
+
+@RunWith(JUnit4.class)
+public class HkdfTest {
+ private final String SHA256 = "HmacSHA256";
+
+ @Test
+ public void constructor() throws Exception {
+ assertThrows(NullPointerException.class, () -> new Hkdf(null));
+ assertThrows(NoSuchAlgorithmException.class, () -> new Hkdf("No such MAC"));
+
+ Hkdf hkdf = new Hkdf(SHA256);
+ assertEquals(Mac.getInstance(SHA256).getMacLength(), hkdf.getMacLength());
+ }
+
+ @Test
+ public void extract() throws Exception {
+ Hkdf hkdf = new Hkdf(SHA256);
+ assertThrows(NullPointerException.class, () -> hkdf.extract(null, new byte[0]));
+ assertThrows(NullPointerException.class, () -> hkdf.extract(new byte[0], null));
+ assertThrows(NullPointerException.class, () -> hkdf.extract(null, null));
+ assertThrows(IllegalArgumentException.class, () -> hkdf.extract(new byte[0], new byte[0]));
+ }
+
+ @Test
+ public void expand() throws Exception {
+ Hkdf hkdf = new Hkdf(SHA256);
+ int macLen = hkdf.getMacLength();
+ assertThrows(NullPointerException.class, () -> hkdf.expand(null, new byte[0], 1));
+ assertThrows(NullPointerException.class, () -> hkdf.expand(new byte[macLen], null, 1));
+ assertThrows(NullPointerException.class, () -> hkdf.expand(null, null, 1));
+ assertThrows(NullPointerException.class, () -> hkdf.expand(null, null, 1));
+ // Negative length
+ assertThrows(IllegalArgumentException.class,
+ () -> hkdf.expand(new byte[macLen], new byte[0], -1));
+ // PRK too small
+ assertThrows(IllegalArgumentException.class,
+ () -> hkdf.expand(new byte[0], new byte[0], 1));
+ // Length too large
+ assertThrows(IllegalArgumentException.class,
+ () -> hkdf.expand(new byte[macLen], new byte[0], 255 * macLen + 1));
+ }
+
+ @Test
+ public void testVectors() throws Exception {
+ List